From d78bd1b9df3ccae70375eb7c2c4d715c024a81de Mon Sep 17 00:00:00 2001 From: Rakesh R Date: Mon, 6 Nov 2023 14:25:03 -0500 Subject: [PATCH] Initial dev branch for maya-hydra --- .clang-format | 92 + .clang-format-ignore | 0 .clang-format-include | 8 + .git-blame-ignore-revs | 2 + .github/ISSUE_TEMPLATE/bug_report.md | 34 + .github/ISSUE_TEMPLATE/build-issue-report.md | 24 + .github/ISSUE_TEMPLATE/feature_request.md | 20 + .github/run-clang-format.py | 249 +++ .github/workflows/maya-hydra-new-issues.yml | 16 + .../maya-hydra-preflight-launcher.yml | 93 + .gitignore | 88 + .pre-commit-config.yaml | 10 + CMakeLists.txt | 222 +++ README.md | 80 + README_DOC.md | 4 + build.py | 754 ++++++++ cmake/compiler_config.cmake | 128 ++ cmake/flowViewport_version.info | 3 + cmake/googletest.cmake | 93 + cmake/googletest_download.txt.in | 31 + cmake/googletest_src.txt.in | 28 + cmake/gulrak.cmake | 48 + cmake/jinja.cmake | 56 + cmake/mayahydra_version.info | 3 + cmake/modules/FindMaya.cmake | 495 +++++ cmake/modules/FindUFE.cmake | 122 ++ cmake/modules/FindUSD.cmake | 180 ++ cmake/python.cmake | 224 +++ cmake/test.cmake | 404 +++++ cmake/usd.cmake | 24 + cmake/utils.cmake | 311 ++++ doc/CHANGELOG.md | 81 + .../Hydra for Maya - Corp Contrib Agmt.pdf | Bin 0 -> 501588 bytes doc/CLA/Hydra for Maya - Ind Contrib Agmt.pdf | Bin 0 -> 402932 bytes doc/CONTRIBUTING.md | 32 + doc/LICENSE.md | 201 ++ doc/build.md | 216 +++ doc/codingGuidelines.md | 376 ++++ doc/hydraSelectionDataSource.png | Bin 0 -> 15871 bytes doc/hydraSelectionHighlighting.png | Bin 0 -> 40891 bytes doc/hydraSelectionReprDisplayStyle.png | Bin 0 -> 19766 bytes doc/images/al.png | Bin 0 -> 9119 bytes doc/images/linux.png | Bin 0 -> 5341 bytes doc/images/mac.png | Bin 0 -> 4051 bytes doc/images/pxr.png | Bin 0 -> 4070 bytes doc/images/windows.png | Bin 0 -> 2957 bytes doc/mayaHydraDetails.md | 157 ++ doc/selectionHighlightingArchitecture.md | 369 ++++ lib/CMakeLists.txt | 2 + lib/adskHydraSceneBrowser/CMakeLists.txt | 4 + lib/adskHydraSceneBrowser/lib/CMakeLists.txt | 177 ++ .../lib/adskHydraSceneBrowserApi.h | 39 + lib/adskHydraSceneBrowser/test/CMakeLists.txt | 106 ++ .../test/adskHydraSceneBrowserTestApi.h | 39 + .../test/adskHydraSceneBrowserTestFixture.cpp | 323 ++++ .../test/adskHydraSceneBrowserTestFixture.h | 79 + .../test/adskHydraSceneBrowserTesting.cpp | 54 + .../test/adskHydraSceneBrowserTesting.h | 28 + lib/flowViewport/API/CMakeLists.txt | 29 + .../API/fvpFlowSelectionInterface.h | 59 + lib/flowViewport/API/fvpSelectionClient.h | 50 + lib/flowViewport/API/fvpVersionInterface.h | 49 + .../API/interfacesImp/CMakeLists.txt | 31 + .../fvpSelectionInterfaceImp.cpp | 77 + .../interfacesImp/fvpSelectionInterfaceImp.h | 47 + .../interfacesImp/fvpVersionInterfaceImp.cpp | 48 + .../interfacesImp/fvpVersionInterfaceImp.h | 42 + lib/flowViewport/API/samples/CMakeLists.txt | 29 + .../API/samples/fvpSelectionClientExample.cpp | 33 + .../API/samples/fvpSelectionClientExample.h | 48 + lib/flowViewport/CMakeLists.txt | 114 ++ lib/flowViewport/api.h | 43 + .../colorPreferences/CMakeLists.txt | 34 + .../colorPreferences/fvpColorChanged.cpp | 36 + .../colorPreferences/fvpColorChanged.h | 53 + .../colorPreferences/fvpColorPreferences.cpp | 81 + .../colorPreferences/fvpColorPreferences.h | 102 ++ .../fvpColorPreferencesTokens.cpp | 24 + .../fvpColorPreferencesTokens.h | 43 + .../fvpColorPreferencesTranslator.h | 52 + lib/flowViewport/debugCodes.cpp | 49 + lib/flowViewport/debugCodes.h | 39 + lib/flowViewport/flowViewport.h.src | 63 + lib/flowViewport/global.cpp | 38 + lib/flowViewport/global.h | 36 + lib/flowViewport/sceneIndex/CMakeLists.txt | 43 + .../sceneIndex/fvpMergingSceneIndex.cpp | 62 + .../sceneIndex/fvpMergingSceneIndex.h | 52 + ...assThroughSelectionInterfaceSceneIndex.cpp | 43 + ...pPassThroughSelectionInterfaceSceneIndex.h | 58 + .../sceneIndex/fvpPathInterface.cpp | 22 + .../sceneIndex/fvpPathInterface.h | 61 + .../sceneIndex/fvpPathInterfaceSceneIndex.cpp | 64 + .../sceneIndex/fvpPathInterfaceSceneIndex.h | 66 + .../sceneIndex/fvpRenderIndexProxy.cpp | 129 ++ .../sceneIndex/fvpRenderIndexProxy.h | 96 + .../sceneIndex/fvpSelectionInterface.cpp | 22 + .../sceneIndex/fvpSelectionInterface.h | 63 + .../sceneIndex/fvpSelectionSceneIndex.cpp | 372 ++++ .../sceneIndex/fvpSelectionSceneIndex.h | 153 ++ ...pWireframeSelectionHighlightSceneIndex.cpp | 186 ++ ...fvpWireframeSelectionHighlightSceneIndex.h | 100 + lib/flowViewport/selection/CMakeLists.txt | 31 + .../selection/fvpSelectionTask.cpp | 120 ++ lib/flowViewport/selection/fvpSelectionTask.h | 102 ++ .../selection/fvpSelectionTracker.cpp | 116 ++ .../selection/fvpSelectionTracker.h | 101 ++ lib/flowViewport/tokens.cpp | 53 + lib/flowViewport/tokens.h | 64 + lib/mayaHydra/CMakeLists.txt | 3 + lib/mayaHydra/hydraExtensions/CMakeLists.txt | 214 +++ .../hydraExtensions/adapters/CMakeLists.txt | 59 + .../hydraExtensions/adapters/adapter.cpp | 124 ++ .../hydraExtensions/adapters/adapter.h | 106 ++ .../adapters/adapterDebugCodes.cpp | 72 + .../adapters/adapterDebugCodes.h | 46 + .../adapters/adapterRegistry.cpp | 136 ++ .../adapters/adapterRegistry.h | 95 + .../adapters/aiSkydomeLightAdapter.cpp | 248 +++ .../adapters/areaLightAdapter.cpp | 81 + .../adapters/cameraAdapter.cpp | 303 ++++ .../hydraExtensions/adapters/cameraAdapter.h | 81 + .../adapters/constantShadowMatrix.h | 56 + .../hydraExtensions/adapters/dagAdapter.cpp | 342 ++++ .../hydraExtensions/adapters/dagAdapter.h | 100 + .../adapters/directionalLightAdapter.cpp | 118 ++ .../hydraExtensions/adapters/lightAdapter.cpp | 371 ++++ .../hydraExtensions/adapters/lightAdapter.h | 89 + .../adapters/materialAdapter.cpp | 314 ++++ .../adapters/materialAdapter.h | 78 + .../adapters/materialNetworkConverter.cpp | 1029 +++++++++++ .../adapters/materialNetworkConverter.h | 210 +++ .../hydraExtensions/adapters/mayaAttrs.cpp | 284 +++ .../hydraExtensions/adapters/mayaAttrs.h | 169 ++ .../hydraExtensions/adapters/meshAdapter.cpp | 471 +++++ .../adapters/nurbsCurveAdapter.cpp | 249 +++ .../adapters/pointLightAdapter.cpp | 87 + .../adapters/renderItemAdapter.cpp | 476 +++++ .../adapters/renderItemAdapter.h | 216 +++ .../hydraExtensions/adapters/shapeAdapter.cpp | 156 ++ .../hydraExtensions/adapters/shapeAdapter.h | 95 + .../adapters/spotLightAdapter.cpp | 167 ++ .../hydraExtensions/adapters/tokens.cpp | 28 + .../hydraExtensions/adapters/tokens.h | 92 + lib/mayaHydra/hydraExtensions/api.h | 40 + lib/mayaHydra/hydraExtensions/debugCodes.cpp | 29 + lib/mayaHydra/hydraExtensions/debugCodes.h | 46 + .../hydraExtensions/delegates/CMakeLists.txt | 42 + .../delegates/defaultLightDelegate.cpp | 228 +++ .../delegates/defaultLightDelegate.h | 74 + .../hydraExtensions/delegates/delegate.cpp | 47 + .../hydraExtensions/delegates/delegate.h | 204 +++ .../hydraExtensions/delegates/delegateCtx.cpp | 248 +++ .../hydraExtensions/delegates/delegateCtx.h | 110 ++ .../delegates/delegateDebugCodes.cpp | 125 ++ .../delegates/delegateDebugCodes.h | 58 + .../delegates/delegateRegistry.cpp | 123 ++ .../delegates/delegateRegistry.h | 76 + .../hydraExtensions/delegates/params.h | 47 + .../delegates/sceneDelegate.cpp | 1612 +++++++++++++++++ .../hydraExtensions/delegates/sceneDelegate.h | 311 ++++ .../delegates/testDelegate.cpp | 64 + .../hydraExtensions/delegates/testDelegate.h | 48 + lib/mayaHydra/hydraExtensions/hydraUtils.cpp | 248 +++ lib/mayaHydra/hydraExtensions/hydraUtils.h | 93 + lib/mayaHydra/hydraExtensions/interface.h | 69 + .../hydraExtensions/interfaceImp.cpp | 49 + lib/mayaHydra/hydraExtensions/interfaceImp.h | 46 + lib/mayaHydra/hydraExtensions/mayaHydra.h.src | 50 + .../mayaHydraSceneProducer.cpp | 466 +++++ .../hydraExtensions/mayaHydraSceneProducer.h | 205 +++ lib/mayaHydra/hydraExtensions/mayaUtils.cpp | 73 + lib/mayaHydra/hydraExtensions/mayaUtils.h | 99 + lib/mayaHydra/hydraExtensions/mixedUtils.cpp | 175 ++ lib/mayaHydra/hydraExtensions/mixedUtils.h | 178 ++ lib/mayaHydra/hydraExtensions/plugInfo.json | 137 ++ .../hydraExtensions/sceneIndex/CMakeLists.txt | 43 + .../sceneIndex/mayaHydraCameraDataSource.cpp | 229 +++ .../sceneIndex/mayaHydraCameraDataSource.h | 56 + .../sceneIndex/mayaHydraDataSource.cpp | 424 +++++ .../sceneIndex/mayaHydraDataSource.h | 69 + .../mayaHydraDefaultLightDataSource.cpp | 177 ++ .../mayaHydraDefaultLightDataSource.h | 62 + .../mayaHydraDisplayStyleDataSource.cpp | 184 ++ .../mayaHydraDisplayStyleDataSource.h | 64 + .../sceneIndex/mayaHydraLightDataSource.cpp | 89 + .../sceneIndex/mayaHydraLightDataSource.h | 58 + .../sceneIndex/mayaHydraPrimvarDataSource.cpp | 90 + .../sceneIndex/mayaHydraPrimvarDataSource.h | 92 + .../sceneIndex/mayaHydraSceneIndex.cpp | 1462 +++++++++++++++ .../sceneIndex/mayaHydraSceneIndex.h | 257 +++ .../sceneIndex/registration.cpp | 406 +++++ .../hydraExtensions/sceneIndex/registration.h | 122 ++ lib/mayaHydra/hydraExtensions/shaderDefs.usda | 70 + lib/mayaHydra/mayaPlugin/CMakeLists.txt | 130 ++ .../mayaPlugin/colorNotFoundException.cpp | 47 + .../mayaPlugin/colorNotFoundException.h | 57 + .../mayaColorPreferencesTranslator.cpp | 160 ++ .../mayaColorPreferencesTranslator.h | 111 ++ .../mayaPlugin/plugRegistryHelper.cpp | 312 ++++ lib/mayaHydra/mayaPlugin/plugRegistryHelper.h | 95 + lib/mayaHydra/mayaPlugin/plugin.cpp | 166 ++ lib/mayaHydra/mayaPlugin/pluginDebugCodes.cpp | 45 + lib/mayaHydra/mayaPlugin/pluginDebugCodes.h | 37 + lib/mayaHydra/mayaPlugin/pluginUtils.cpp | 131 ++ lib/mayaHydra/mayaPlugin/pluginUtils.h | 62 + lib/mayaHydra/mayaPlugin/renderGlobals.cpp | 1186 ++++++++++++ lib/mayaHydra/mayaPlugin/renderGlobals.h | 109 ++ lib/mayaHydra/mayaPlugin/renderOverride.cpp | 1308 +++++++++++++ lib/mayaHydra/mayaPlugin/renderOverride.h | 255 +++ .../mayaPlugin/renderOverrideUtils.h | 242 +++ lib/mayaHydra/mayaPlugin/tokens.cpp | 28 + lib/mayaHydra/mayaPlugin/tokens.h | 35 + lib/mayaHydra/mayaPlugin/viewCommand.cpp | 288 +++ lib/mayaHydra/mayaPlugin/viewCommand.h | 38 + lib/mayaHydra/ufeExtensions/CMakeLists.txt | 113 ++ lib/mayaHydra/ufeExtensions/Global.cpp | 142 ++ lib/mayaHydra/ufeExtensions/Global.h | 56 + lib/mayaHydra/ufeExtensions/api.h | 40 + modules/mayaHydra.mod.template | 18 + modules/mayaHydra_Win.mod.template | 22 + plugin/mayaHydraSceneBrowser/CMakeLists.txt | 74 + .../mayaHydraSceneBrowser.cpp | 64 + .../mayaHydraSceneBrowser.h | 30 + .../mayaHydraSceneBrowserTest/CMakeLists.txt | 69 + .../mayaHydraSceneBrowserTestCmd.cpp | 69 + .../mayaHydraSceneBrowserTestCmd.h | 31 + test/CMakeLists.txt | 1 + test/lib/CMakeLists.txt | 105 ++ test/lib/mayaUsd/CMakeLists.txt | 4 + test/lib/mayaUsd/render/CMakeLists.txt | 8 + .../mayaUsd/render/mayaToHydra/CMakeLists.txt | 94 + .../ImageDiffingTest/colored_stripes.png | Bin 0 -> 247 bytes .../colored_stripes_one_pixel_off.png | Bin 0 -> 251 bytes .../colored_stripes_slight_noise.png | Bin 0 -> 10060 bytes .../ImageDiffingTest/cube_scene.png | Bin 0 -> 3181 bytes .../cube_scene_one_pixel_off.png | Bin 0 -> 3238 bytes .../cube_scene_slight_noise.png | Bin 0 -> 282687 bytes .../MtohBasicRenderTest/cube_selected.png | Bin 0 -> 4146 bytes .../MtohBasicRenderTest/cube_unselected.png | Bin 0 -> 3075 bytes .../MtohBasicRenderTest/flat_orange.png | Bin 0 -> 1494 bytes .../MtohBasicRenderTest/flat_orange_bad.png | Bin 0 -> 2015 bytes .../lambertDefaultMaterial/cube_selected.png | Bin 0 -> 4080 bytes .../cube_unselected.png | Bin 0 -> 2821 bytes .../MtohDagChangesTest/instances_0.png | Bin 0 -> 1513 bytes .../MtohDagChangesTest/instances_1.png | Bin 0 -> 1894 bytes .../MtohDagChangesTest/instances_12.png | Bin 0 -> 2273 bytes .../MtohDagChangesTest/instances_123.png | Bin 0 -> 2660 bytes .../MtohDagChangesTest/instances_1234.png | Bin 0 -> 3156 bytes .../MtohDagChangesTest/instances_123456.png | Bin 0 -> 3873 bytes .../MtohDagChangesTest/instances_1235.png | Bin 0 -> 2986 bytes .../MtohDagChangesTest/instances_3.png | Bin 0 -> 1912 bytes .../MtohDagChangesTest/instances_35.png | Bin 0 -> 2266 bytes .../lambertDefaultMaterial/instances_0.png | Bin 0 -> 1513 bytes .../lambertDefaultMaterial/instances_1.png | Bin 0 -> 1825 bytes .../lambertDefaultMaterial/instances_12.png | Bin 0 -> 2136 bytes .../lambertDefaultMaterial/instances_123.png | Bin 0 -> 2485 bytes .../lambertDefaultMaterial/instances_1234.png | Bin 0 -> 2913 bytes .../instances_123456.png | Bin 0 -> 3564 bytes .../lambertDefaultMaterial/instances_1235.png | Bin 0 -> 2780 bytes .../lambertDefaultMaterial/instances_3.png | Bin 0 -> 1871 bytes .../lambertDefaultMaterial/instances_35.png | Bin 0 -> 2188 bytes test/lib/mayaUsd/render/mayaToHydra/README.md | 92 + .../VisibilityTest/cube_unselected.png | Bin 0 -> 2229 bytes .../cube_unselected.png | Bin 0 -> 2821 bytes .../mayaToHydra/VisibilityTest/nothing.png | Bin 0 -> 1511 bytes .../render/mayaToHydra/cpp/CMakeLists.txt | 73 + .../mayaUsd/render/mayaToHydra/cpp/README.md | 33 + .../mayaToHydra/cpp/mayaHydraCppTestsCmd.cpp | 92 + .../mayaToHydra/cpp/mayaHydraCppTestsCmd.h | 31 + .../mayaToHydra/cpp/testColorPreferences.cpp | 319 ++++ .../mayaToHydra/cpp/testColorPreferences.py | 69 + .../mayaToHydra/cpp/testCppFramework.cpp | 25 + .../mayaToHydra/cpp/testCppFramework.py | 39 + .../cpp/testMayaSceneFlattening.cpp | 69 + .../cpp/testMayaSceneFlattening.py | 39 + .../mayaToHydra/cpp/testMayaUsdUfeItems.cpp | 52 + .../mayaToHydra/cpp/testMayaUsdUfeItems.py | 47 + .../mayaToHydra/cpp/testMergingSceneIndex.cpp | 138 ++ .../mayaToHydra/cpp/testMergingSceneIndex.py | 36 + .../cpp/testSelectionSceneIndex.cpp | 202 +++ .../cpp/testSelectionSceneIndex.py | 36 + .../render/mayaToHydra/cpp/testUtils.cpp | 243 +++ .../render/mayaToHydra/cpp/testUtils.h | 179 ++ ...tWireframeSelectionHighlightSceneIndex.cpp | 245 +++ ...stWireframeSelectionHighlightSceneIndex.py | 30 + .../render/mayaToHydra/testImageDiffing.py | 93 + .../mayaUsd/render/mayaToHydra/testMeshes.py | 66 + .../render/mayaToHydra/testMtohBasicRender.py | 79 + .../render/mayaToHydra/testMtohCommand.py | 120 ++ .../render/mayaToHydra/testMtohDagChanges.py | 322 ++++ .../render/mayaToHydra/testNamespaces.py | 53 + .../mayaToHydra/testNewSceneWithStage.py | 51 + .../mayaToHydra/testRendererSwitching.py | 35 + .../render/mayaToHydra/testSceneBrowser.py | 54 + .../render/mayaToHydra/testStageAddPrim.py | 66 + .../render/mayaToHydra/testVisibility.py | 179 ++ .../testSelectionHighlightHierarchy.ma | 284 +++ test/testUtils/__init__.py | 0 test/testUtils/cachingUtils.py | 162 ++ test/testUtils/fixturesUtils.py | 150 ++ test/testUtils/imageUtils.py | 244 +++ test/testUtils/mayaUtils.py | 330 ++++ test/testUtils/mtohUtils.py | 181 ++ test/testUtils/testUtils.py | 104 ++ test/testUtils/ufeScripts/__init__.py | 0 test/testUtils/ufeScripts/ufeSelectCmd.py | 302 +++ test/testUtils/ufeUtils.py | 107 ++ test/testUtils/usdUtils.py | 181 ++ 309 files changed, 36830 insertions(+) create mode 100644 .clang-format create mode 100644 .clang-format-ignore create mode 100644 .clang-format-include create mode 100644 .git-blame-ignore-revs create mode 100644 .github/ISSUE_TEMPLATE/bug_report.md create mode 100644 .github/ISSUE_TEMPLATE/build-issue-report.md create mode 100644 .github/ISSUE_TEMPLATE/feature_request.md create mode 100755 .github/run-clang-format.py create mode 100644 .github/workflows/maya-hydra-new-issues.yml create mode 100644 .github/workflows/maya-hydra-preflight-launcher.yml create mode 100644 .gitignore create mode 100644 .pre-commit-config.yaml create mode 100644 CMakeLists.txt create mode 100644 README.md create mode 100644 README_DOC.md create mode 100755 build.py create mode 100644 cmake/compiler_config.cmake create mode 100644 cmake/flowViewport_version.info create mode 100644 cmake/googletest.cmake create mode 100644 cmake/googletest_download.txt.in create mode 100644 cmake/googletest_src.txt.in create mode 100644 cmake/gulrak.cmake create mode 100644 cmake/jinja.cmake create mode 100644 cmake/mayahydra_version.info create mode 100644 cmake/modules/FindMaya.cmake create mode 100644 cmake/modules/FindUFE.cmake create mode 100644 cmake/modules/FindUSD.cmake create mode 100644 cmake/python.cmake create mode 100644 cmake/test.cmake create mode 100644 cmake/usd.cmake create mode 100644 cmake/utils.cmake create mode 100644 doc/CHANGELOG.md create mode 100644 doc/CLA/Hydra for Maya - Corp Contrib Agmt.pdf create mode 100644 doc/CLA/Hydra for Maya - Ind Contrib Agmt.pdf create mode 100644 doc/CONTRIBUTING.md create mode 100644 doc/LICENSE.md create mode 100644 doc/build.md create mode 100644 doc/codingGuidelines.md create mode 100644 doc/hydraSelectionDataSource.png create mode 100644 doc/hydraSelectionHighlighting.png create mode 100644 doc/hydraSelectionReprDisplayStyle.png create mode 100644 doc/images/al.png create mode 100644 doc/images/linux.png create mode 100644 doc/images/mac.png create mode 100644 doc/images/pxr.png create mode 100644 doc/images/windows.png create mode 100644 doc/mayaHydraDetails.md create mode 100644 doc/selectionHighlightingArchitecture.md create mode 100644 lib/CMakeLists.txt create mode 100644 lib/adskHydraSceneBrowser/CMakeLists.txt create mode 100644 lib/adskHydraSceneBrowser/lib/CMakeLists.txt create mode 100644 lib/adskHydraSceneBrowser/lib/adskHydraSceneBrowserApi.h create mode 100644 lib/adskHydraSceneBrowser/test/CMakeLists.txt create mode 100644 lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestApi.h create mode 100644 lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.cpp create mode 100644 lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.h create mode 100644 lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.cpp create mode 100644 lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.h create mode 100644 lib/flowViewport/API/CMakeLists.txt create mode 100644 lib/flowViewport/API/fvpFlowSelectionInterface.h create mode 100644 lib/flowViewport/API/fvpSelectionClient.h create mode 100644 lib/flowViewport/API/fvpVersionInterface.h create mode 100644 lib/flowViewport/API/interfacesImp/CMakeLists.txt create mode 100644 lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.cpp create mode 100644 lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.h create mode 100644 lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.cpp create mode 100644 lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.h create mode 100644 lib/flowViewport/API/samples/CMakeLists.txt create mode 100644 lib/flowViewport/API/samples/fvpSelectionClientExample.cpp create mode 100644 lib/flowViewport/API/samples/fvpSelectionClientExample.h create mode 100644 lib/flowViewport/CMakeLists.txt create mode 100644 lib/flowViewport/api.h create mode 100644 lib/flowViewport/colorPreferences/CMakeLists.txt create mode 100644 lib/flowViewport/colorPreferences/fvpColorChanged.cpp create mode 100644 lib/flowViewport/colorPreferences/fvpColorChanged.h create mode 100644 lib/flowViewport/colorPreferences/fvpColorPreferences.cpp create mode 100644 lib/flowViewport/colorPreferences/fvpColorPreferences.h create mode 100644 lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.cpp create mode 100644 lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.h create mode 100644 lib/flowViewport/colorPreferences/fvpColorPreferencesTranslator.h create mode 100644 lib/flowViewport/debugCodes.cpp create mode 100644 lib/flowViewport/debugCodes.h create mode 100644 lib/flowViewport/flowViewport.h.src create mode 100644 lib/flowViewport/global.cpp create mode 100644 lib/flowViewport/global.h create mode 100644 lib/flowViewport/sceneIndex/CMakeLists.txt create mode 100644 lib/flowViewport/sceneIndex/fvpMergingSceneIndex.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpMergingSceneIndex.h create mode 100644 lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h create mode 100644 lib/flowViewport/sceneIndex/fvpPathInterface.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpPathInterface.h create mode 100644 lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.h create mode 100644 lib/flowViewport/sceneIndex/fvpRenderIndexProxy.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpRenderIndexProxy.h create mode 100644 lib/flowViewport/sceneIndex/fvpSelectionInterface.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpSelectionInterface.h create mode 100644 lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.h create mode 100644 lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.cpp create mode 100644 lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.h create mode 100644 lib/flowViewport/selection/CMakeLists.txt create mode 100644 lib/flowViewport/selection/fvpSelectionTask.cpp create mode 100644 lib/flowViewport/selection/fvpSelectionTask.h create mode 100644 lib/flowViewport/selection/fvpSelectionTracker.cpp create mode 100644 lib/flowViewport/selection/fvpSelectionTracker.h create mode 100644 lib/flowViewport/tokens.cpp create mode 100644 lib/flowViewport/tokens.h create mode 100644 lib/mayaHydra/CMakeLists.txt create mode 100644 lib/mayaHydra/hydraExtensions/CMakeLists.txt create mode 100644 lib/mayaHydra/hydraExtensions/adapters/CMakeLists.txt create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/aiSkydomeLightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/areaLightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/constantShadowMatrix.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/dagAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/dagAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/directionalLightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/lightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/lightAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/materialAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/materialAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/meshAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/nurbsCurveAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/pointLightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.h create mode 100644 lib/mayaHydra/hydraExtensions/adapters/spotLightAdapter.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/tokens.cpp create mode 100644 lib/mayaHydra/hydraExtensions/adapters/tokens.h create mode 100644 lib/mayaHydra/hydraExtensions/api.h create mode 100644 lib/mayaHydra/hydraExtensions/debugCodes.cpp create mode 100644 lib/mayaHydra/hydraExtensions/debugCodes.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/CMakeLists.txt create mode 100644 lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegate.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegate.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateCtx.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateCtx.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/params.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.h create mode 100644 lib/mayaHydra/hydraExtensions/delegates/testDelegate.cpp create mode 100644 lib/mayaHydra/hydraExtensions/delegates/testDelegate.h create mode 100644 lib/mayaHydra/hydraExtensions/hydraUtils.cpp create mode 100644 lib/mayaHydra/hydraExtensions/hydraUtils.h create mode 100644 lib/mayaHydra/hydraExtensions/interface.h create mode 100644 lib/mayaHydra/hydraExtensions/interfaceImp.cpp create mode 100644 lib/mayaHydra/hydraExtensions/interfaceImp.h create mode 100644 lib/mayaHydra/hydraExtensions/mayaHydra.h.src create mode 100644 lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.cpp create mode 100644 lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.h create mode 100644 lib/mayaHydra/hydraExtensions/mayaUtils.cpp create mode 100644 lib/mayaHydra/hydraExtensions/mayaUtils.h create mode 100644 lib/mayaHydra/hydraExtensions/mixedUtils.cpp create mode 100644 lib/mayaHydra/hydraExtensions/mixedUtils.h create mode 100644 lib/mayaHydra/hydraExtensions/plugInfo.json create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/CMakeLists.txt create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraPrimvarDataSource.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraPrimvarDataSource.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraSceneIndex.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraSceneIndex.h create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/registration.cpp create mode 100644 lib/mayaHydra/hydraExtensions/sceneIndex/registration.h create mode 100644 lib/mayaHydra/hydraExtensions/shaderDefs.usda create mode 100644 lib/mayaHydra/mayaPlugin/CMakeLists.txt create mode 100644 lib/mayaHydra/mayaPlugin/colorNotFoundException.cpp create mode 100644 lib/mayaHydra/mayaPlugin/colorNotFoundException.h create mode 100644 lib/mayaHydra/mayaPlugin/mayaColorPreferencesTranslator.cpp create mode 100644 lib/mayaHydra/mayaPlugin/mayaColorPreferencesTranslator.h create mode 100644 lib/mayaHydra/mayaPlugin/plugRegistryHelper.cpp create mode 100644 lib/mayaHydra/mayaPlugin/plugRegistryHelper.h create mode 100644 lib/mayaHydra/mayaPlugin/plugin.cpp create mode 100644 lib/mayaHydra/mayaPlugin/pluginDebugCodes.cpp create mode 100644 lib/mayaHydra/mayaPlugin/pluginDebugCodes.h create mode 100644 lib/mayaHydra/mayaPlugin/pluginUtils.cpp create mode 100644 lib/mayaHydra/mayaPlugin/pluginUtils.h create mode 100644 lib/mayaHydra/mayaPlugin/renderGlobals.cpp create mode 100644 lib/mayaHydra/mayaPlugin/renderGlobals.h create mode 100644 lib/mayaHydra/mayaPlugin/renderOverride.cpp create mode 100644 lib/mayaHydra/mayaPlugin/renderOverride.h create mode 100644 lib/mayaHydra/mayaPlugin/renderOverrideUtils.h create mode 100644 lib/mayaHydra/mayaPlugin/tokens.cpp create mode 100644 lib/mayaHydra/mayaPlugin/tokens.h create mode 100644 lib/mayaHydra/mayaPlugin/viewCommand.cpp create mode 100644 lib/mayaHydra/mayaPlugin/viewCommand.h create mode 100644 lib/mayaHydra/ufeExtensions/CMakeLists.txt create mode 100644 lib/mayaHydra/ufeExtensions/Global.cpp create mode 100644 lib/mayaHydra/ufeExtensions/Global.h create mode 100644 lib/mayaHydra/ufeExtensions/api.h create mode 100644 modules/mayaHydra.mod.template create mode 100644 modules/mayaHydra_Win.mod.template create mode 100644 plugin/mayaHydraSceneBrowser/CMakeLists.txt create mode 100644 plugin/mayaHydraSceneBrowser/mayaHydraSceneBrowser.cpp create mode 100644 plugin/mayaHydraSceneBrowser/mayaHydraSceneBrowser.h create mode 100644 plugin/mayaHydraSceneBrowserTest/CMakeLists.txt create mode 100644 plugin/mayaHydraSceneBrowserTest/mayaHydraSceneBrowserTestCmd.cpp create mode 100644 plugin/mayaHydraSceneBrowserTest/mayaHydraSceneBrowserTestCmd.h create mode 100644 test/CMakeLists.txt create mode 100644 test/lib/CMakeLists.txt create mode 100644 test/lib/mayaUsd/CMakeLists.txt create mode 100644 test/lib/mayaUsd/render/CMakeLists.txt create mode 100644 test/lib/mayaUsd/render/mayaToHydra/CMakeLists.txt create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/colored_stripes.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/colored_stripes_one_pixel_off.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/colored_stripes_slight_noise.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/cube_scene.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/cube_scene_one_pixel_off.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/ImageDiffingTest/cube_scene_slight_noise.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_selected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_unselected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/flat_orange.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/flat_orange_bad.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_selected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_unselected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_0.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_1.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_12.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_123.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_1234.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_123456.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_1235.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_3.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_35.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_0.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_1.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_12.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_123.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_1234.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_123456.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_1235.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_3.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_35.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/README.md create mode 100644 test/lib/mayaUsd/render/mayaToHydra/VisibilityTest/cube_unselected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/VisibilityTest/lambertDefaultMaterial/cube_unselected.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/VisibilityTest/nothing.png create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/CMakeLists.txt create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/README.md create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.h create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.h create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.cpp create mode 100644 test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testImageDiffing.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testMeshes.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testMtohBasicRender.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testMtohCommand.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testMtohDagChanges.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testNamespaces.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testNewSceneWithStage.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testRendererSwitching.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testSceneBrowser.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testStageAddPrim.py create mode 100644 test/lib/mayaUsd/render/mayaToHydra/testVisibility.py create mode 100644 test/testSamples/testWireframeSelectionHighlight/testSelectionHighlightHierarchy.ma create mode 100644 test/testUtils/__init__.py create mode 100644 test/testUtils/cachingUtils.py create mode 100644 test/testUtils/fixturesUtils.py create mode 100644 test/testUtils/imageUtils.py create mode 100644 test/testUtils/mayaUtils.py create mode 100644 test/testUtils/mtohUtils.py create mode 100644 test/testUtils/testUtils.py create mode 100644 test/testUtils/ufeScripts/__init__.py create mode 100644 test/testUtils/ufeScripts/ufeSelectCmd.py create mode 100644 test/testUtils/ufeUtils.py create mode 100644 test/testUtils/usdUtils.py diff --git a/.clang-format b/.clang-format new file mode 100644 index 0000000000..f4c107d2f6 --- /dev/null +++ b/.clang-format @@ -0,0 +1,92 @@ +--- +Language: Cpp + +BasedOnStyle: WebKit +AlignAfterOpenBracket: AlwaysBreak +AlignConsecutiveMacros: 'true' +AlignConsecutiveDeclarations: 'true' +AlignEscapedNewlines: Left +AlignTrailingComments: 'true' +AllowAllConstructorInitializersOnNextLine: 'false' +AllowAllParametersOfDeclarationOnNextLine: 'false' +AllowShortCaseLabelsOnASingleLine: 'true' +AllowShortIfStatementsOnASingleLine: Never +AllowShortLambdasOnASingleLine: All +AllowShortLoopsOnASingleLine: 'false' +AlwaysBreakTemplateDeclarations: MultiLine +BinPackArguments: 'false' +BinPackParameters: 'false' +BreakBeforeBraces: Custom +BraceWrapping: + AfterClass: 'true' + AfterEnum: 'true' + AfterFunction: 'true' + AfterStruct: 'true' + AfterUnion: 'true' +BreakBeforeTernaryOperators: 'true' +BreakConstructorInitializers: BeforeComma +BreakInheritanceList: BeforeComma +ColumnLimit: '100' +FixNamespaceComments: 'true' +IncludeBlocks: Regroup +IncludeCategories: + + # Desired final ordering: + # 0. Glew must be included before any other GL header + # 1. Related header + # 2. All private headers + # 3. All public headers from this repository (maya-usd) + # 4. Pixar + USD headers + # 5. Autodesk + Maya headers + # 6. Other libraries' headers + # 7. C++ standard library headers + # 8. C system headers + # 9. Conditional includes + + # 0. GL loaders must be included before any other GL header + # Negative priority puts it above the default IncludeIsMainRegex + - Regex: '' + Priority: -1 + + # 1. Related header + # Handled by the default IncludeIsMainRegex regex, and auto-assigned + # Priority 0 + + # 3. All public headers from this repository (maya-usd) + - Regex: '^<(mayaUsd.*|mayaHydraLib|AL|usdMaya)/' + Priority: 3 + + # 4. Pixar + USD headers + - Regex: '^$' + Priority: 7 + + # 8. C system headers + # angle brackets, no directory, end with ".h" + - Regex: '^<[A-Za-z0-9_-]+\.h>$' + Priority: 8 + + # 2. All private headers + - Regex: '^"' + Priority: 2 + + # 6. Other libraries' headers + - Regex: '^<' + Priority: 6 + + # 9. Conditional includes + # Not reordered by clang-format, we need to manually make sure these come last + +MaxEmptyLinesToKeep: '1' +NamespaceIndentation: None +UseTab: Never + +... diff --git a/.clang-format-ignore b/.clang-format-ignore new file mode 100644 index 0000000000..e69de29bb2 diff --git a/.clang-format-include b/.clang-format-include new file mode 100644 index 0000000000..aad3c80ed2 --- /dev/null +++ b/.clang-format-include @@ -0,0 +1,8 @@ +\.c$ +\.cc$ +\.cpp$ +\.cxx$ +\.h$ +\.hh$ +\.hpp$ +\.hxx$ diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs new file mode 100644 index 0000000000..5bc9317bc6 --- /dev/null +++ b/.git-blame-ignore-revs @@ -0,0 +1,2 @@ +# Ignore clang-format commit +7c64b5df1f1cdbf879aaef12ed64b7fdd77a3af1 diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 0000000000..3a9fd6df02 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,34 @@ +--- +name: Bug report +about: Create a report to help us improve +title: '' +labels: bug +assignees: santosg87 + +--- + +**Describe the bug** +A clear and concise description of what the bug is. + +**Steps to reproduce** +Steps to reproduce the behavior: +1. Go to '...' +2. Click on '....' +3. Scroll down to '....' +4. See error + +**Expected behavior** +A clear and concise description of what you expected to happen. + +**Attachments** +If applicable, add screenshots, sample files, etc to help explain your problem. + +**Specs (if applicable):** + - OS & version [e.g. Windows 10] + - Compiler & version [e.g. gcc 6.3.1] + - Maya version [e.g. Maya 2020] + - Maya USD commit SHA [e.g. dev at caa921c1] + - Pixar USD commit SHA [e.g. dev at b85ddac2] + +**Additional context** +Add any other context about the problem here. diff --git a/.github/ISSUE_TEMPLATE/build-issue-report.md b/.github/ISSUE_TEMPLATE/build-issue-report.md new file mode 100644 index 0000000000..69d0d0396f --- /dev/null +++ b/.github/ISSUE_TEMPLATE/build-issue-report.md @@ -0,0 +1,24 @@ +--- +name: Build issue report +about: Before opening a new build issue, please review doc/build.md +title: '' +labels: help wanted +assignees: '' + +--- + +**Describe the issue** +A description of what the issue is. + +**Build log** +Please attach a build_log.txt + +**Specs:** + - OS & version [e.g. Windows 10] + - Compiler & version [e.g. gcc 6.3.1] + - Maya version [e.g. Maya 2020] + - Maya USD commit SHA [e.g. dev at caa921c1] + - Pixar USD commit SHA [e.g. dev at b85ddac2] + +**Additional context** +Add any other context about the problem here. diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 0000000000..11fc491ef1 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,20 @@ +--- +name: Feature request +about: Suggest an idea for this project +title: '' +labels: enhancement +assignees: '' + +--- + +**Is your feature request related to a problem? Please describe.** +A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] + +**Describe the solution you'd like** +A clear and concise description of what you want to happen. + +**Describe alternatives you've considered** +A clear and concise description of any alternative solutions or features you've considered. + +**Additional context** +Add any other context or screenshots about the feature request here. diff --git a/.github/run-clang-format.py b/.github/run-clang-format.py new file mode 100755 index 0000000000..3c92a8ae53 --- /dev/null +++ b/.github/run-clang-format.py @@ -0,0 +1,249 @@ +#!/usr/bin/env python + +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +'''Run clang-format on files in this repository + +By default, runs on all files, but may pass specific files. +''' + +from __future__ import absolute_import, division, print_function, unicode_literals + +import argparse +import inspect +import fnmatch +import io +import os +import re +import stat +import sys +import platform +import time + +from subprocess import check_call, check_output + +if sys.version_info[0] < 3: + # Python-2 check_output doesn't have encoding + def check_output(*args, **kwargs): + import subprocess + kwargs.pop('encoding') + return subprocess.check_output(*args, **kwargs) + + +THIS_FILE = os.path.normpath(os.path.abspath(inspect.getsourcefile(lambda: None))) +THIS_DIR = os.path.dirname(THIS_FILE) +# THIS_DIR = REPO_ROOT/.github +REPO_ROOT = os.path.dirname(THIS_DIR) + +UPDATE_INTERVAL = .2 + + +_last_update_len = 0 +_on_update_line = False + + +def update_status(text): + global _last_update_len + global _on_update_line + sys.stdout.write('\r') + text_len = len(text) + extra_chars = _last_update_len - text_len + if extra_chars > 0: + text += (' ' * extra_chars) + sys.stdout.write(text) + _last_update_len = text_len + sys.stdout.flush() + _on_update_line = True + + +def post_update_print(text): + global _on_update_line + if _on_update_line: + print() + print(text) + _on_update_line = False + + +def regex_from_file(path, glob=False): + with io.open(path, 'r') as f: + patterns = f.read().splitlines() + # ignore comment lines + patterns = [x for x in patterns if x.strip() and not x.lstrip().startswith('#')] + if glob: + patterns = [fnmatch.translate(x) for x in patterns] + regex = '({})'.format('|'.join(patterns)) + return re.compile(regex) + +if platform.system() == "Windows" and sys.version_info >= (3, 6): + import pathlib # Python 3.6 is required for pathlib.Path + def canonicalpath(path): + path = os.path.abspath(os.path.realpath(os.path.normpath(os.path.normcase(path)))) + realpath = str(pathlib.Path(path).resolve()) # To get a properly cased path ie: from r'C:\WiNdOwS\SyStEm32\DeSkToP.iNi' get r'C:\Windows\System32\desktop.ini' + if len(path) == len(realpath): # This is to avoid following symbolic links, there is still the possibility that they could be equal. + path = realpath + if os.path.isabs(path) and path[0].upper() != path[0]: + path = path[0].upper() +path[1:] # path.capitalize() + path = path.replace('\\', '/') + return path +else: + def canonicalpath(path): + path = os.path.abspath(os.path.realpath(os.path.normpath(os.path.normcase(path)))) + return path.replace('\\', '/') + +def run_clang_format(paths=(), verbose=False, commit=None): + """Runs clang-format in-place on repo files + + Returns + ------- + List[str] + Files altered by clang-format + """ + if not paths and not commit: + paths = [REPO_ROOT] + + if commit: + check_call(['git', 'checkout', commit], cwd=REPO_ROOT) + text = check_output( + ['git', 'diff-tree', '--no-commit-id', '--name-only', '-r', + commit], cwd=REPO_ROOT, encoding=sys.stdout.encoding) + commit_paths = text.splitlines() + paths.extend(os.path.join(REPO_ROOT, p) for p in commit_paths) + + files = set() + folders = set() + + include_path = os.path.join(REPO_ROOT, '.clang-format-include') + include_regex = regex_from_file(include_path) + + ignore_path = os.path.join(REPO_ROOT, '.clang-format-ignore') + ignore_regex = regex_from_file(ignore_path, glob=True) + + # tried to parse .gitignore with regex_from_file, but it has + # too much special git syntax. Instead just using `git ls-files` + # as a filter... + git_files = check_output(['git', 'ls-files'], cwd=REPO_ROOT, + encoding=sys.stdout.encoding) + git_files = set(canonicalpath(x.strip()) for x in git_files.splitlines()) + + def print_path(p): + if p.startswith(REPO_ROOT): + p = os.path.relpath(p, REPO_ROOT) + return p + + def passes_filter(path): + rel_path = os.path.relpath(path, REPO_ROOT) + match_path = os.path.join('.', rel_path) + # standardize on '/', because that's what's used in files, + # and output by `git ls-files` + match_path = match_path.replace('\\', '/') + if not include_regex.search(match_path): + return False + if ignore_regex.pattern != '()' and ignore_regex.search(match_path): + return False + return True + + # parse the initial fed-in paths, which may be files or folders + for path in paths: + # Get the stat, so we only do one filesystem call, instead of + # two for os.path.isfile() + os.path.isdir() + try: + st_mode = os.stat(path).st_mode + if stat.S_ISDIR(st_mode): + folders.add(path) + elif stat.S_ISREG(st_mode): + if canonicalpath(path) in git_files: + files.add(path) + except Exception: + print("Given path did not exist: {}".format(path)) + raise + + for folder in folders: + # we already have list of potential files in git_files... + # filter to ones in this folder + folder = canonicalpath(folder) + '/' + files.update(x for x in git_files if x.startswith(folder)) + + # We apply filters even to fed-in paths... this is to aid + # in use of globs on command line + files = sorted(x for x in files if passes_filter(x)) + + clang_format_executable = os.environ.get('CLANG_FORMAT_EXECUTABLE', + 'clang-format') + if verbose: + print("Running clang-format on {} files...".format(len(files))) + last_update = time.time() + + def print_path(p): + if p.startswith(REPO_ROOT): + p = os.path.relpath(p, REPO_ROOT) + return p + + altered = [] + for i, path in enumerate(files): + if verbose: + now = time.time() + if now - last_update > UPDATE_INTERVAL: + last_update = now + update_status('File {}/{} ({:.1f}%) - {}'.format( + i + 1, len(files), (i + 1) / len(files) * 100, + print_path(path))) + # Note - couldn't find a way to get clang-format to return whether + # or not a file was altered with '-i' - so checking ourselves + # Checking mtime is not foolproof, but is faster than reading file + # and comparing, and probably good enough + mtime_orig = os.path.getmtime(path) + check_call([clang_format_executable, '-i', path]) + mtime_new = os.path.getmtime(path) + if mtime_new != mtime_orig: + post_update_print("File altered: {}".format(print_path(path))) + altered.append(path) + post_update_print("Done - altered {} files".format(len(altered))) + return altered + + +def get_parser(): + parser = argparse.ArgumentParser( + description=__doc__, + formatter_class=argparse.ArgumentDefaultsHelpFormatter) + parser.add_argument('paths', nargs='*', + help='Paths to run clang-format on; defaults to all files in repo') + parser.add_argument('-v', '--verbose', action='store_true', + help='Enable more output (ie, progress messages)') + parser.add_argument('-c', '--commit', + help='Git commit / revision / branch; will first check out that commit,' + " then query it for it's list of affected files, to use as the files" + ' to run clang-format on; if PATHS are also manually given, they are' + ' appended') + return parser + + +def main(raw_args=None): + parser = get_parser() + args = parser.parse_args(raw_args) + try: + altered = run_clang_format(paths=args.paths, verbose=args.verbose, + commit=args.commit) + except Exception: + import traceback + traceback.print_exc() + return 1 + if altered: + return 1 + return 0 + + +if __name__ == '__main__': + sys.exit(main()) diff --git a/.github/workflows/maya-hydra-new-issues.yml b/.github/workflows/maya-hydra-new-issues.yml new file mode 100644 index 0000000000..3d9c3d8d9c --- /dev/null +++ b/.github/workflows/maya-hydra-new-issues.yml @@ -0,0 +1,16 @@ +name: Move Issues to Triage +on: + issues: + types: [opened, reopened] + +jobs: + move-triage-card: + runs-on: ubuntu-latest + steps: + - uses: alex-page/github-project-automation-plus@v0.8.3 + # Aug 2023: Update from v0.3.0 to v0.8.3 which uses node16. + # node12 is out of support. + with: + project: Issue Triage + column: Needs triage + repo-token: ${{ secrets.REPO_ACCESS_TOKEN }} \ No newline at end of file diff --git a/.github/workflows/maya-hydra-preflight-launcher.yml b/.github/workflows/maya-hydra-preflight-launcher.yml new file mode 100644 index 0000000000..c19d640d3a --- /dev/null +++ b/.github/workflows/maya-hydra-preflight-launcher.yml @@ -0,0 +1,93 @@ +name: Pre-flight build on pull request + +# Trigger the workflow on pull request (assigned) event for the branch +# https://help.github.com/en/actions/reference/events-that-trigger-workflows +on: + pull_request: + branches: [dev] + types: [assigned] + +jobs: + + clang_format_linter: + timeout-minutes: 30 + runs-on: ubuntu-latest + if: ${{ github.event.assignee.login == 'ecp-maya-devops-adsk' }} + steps: + # Aug 2023: Update from v2 to v3 which uses node16 (node12 is out of support). + - uses: actions/checkout@v3 + - uses: DoozyX/clang-format-lint-action@v0.16.2 + with: + source: '.' + clangFormatVersion: 10 + + # Wait for remote build to start and finish, report results + build_preflight: + timeout-minutes: 400 + runs-on: ubuntu-latest + needs: clang_format_linter + if: ${{ github.event.assignee.login == 'ecp-maya-devops-adsk' }} + steps: + + # Build start info will be committed here when the remote build is launched + - name: Setup transfer repo + uses: actions/checkout@v3 + with: + repository: ecp-maya-devops-adsk/log-transfer + ref: transfer-hydra + path: transfer-hydra + + # Echo the file name - it's hard to find the run_id in the UI + - name: Echo the expected start file name + run: "echo ${{ github.run_id }}_${{ github.run_number }}_start.txt" + + # Wait for remote build to start + - name: Wait until remote build starts + shell: bash + # 180 minutes wait time. There will be overhead for the git pull command, so actual wait time will be slightly more than 180 minutes + run: "cd transfer-hydra ; for (( i=0; i<180; i++ )) ; do if [ -f ${{ github.run_id }}_${{ github.run_number }}_start.txt ] ; then break ; fi ; git pull --quiet ; sleep 60 ; false ; done || exit 1" + + # Show contents of start file + - name: Show build start information + shell: bash + run: "cat transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_start.txt" + + # Grep the start file to show failures + - name: Exit with error if a build failed to start + # Default shell includes "-o pipefail" and "-e". Specify a different shell + shell: bash --noprofile --norc {0} + run: "if grep -i 'Remote build failed' transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_start.txt; then exit 1; else exit 0; fi" + + # Echo the file name - it's hard to find the run_id in the UI + - name: Echo the expected result file name + run: "echo ${{ github.run_id }}_${{ github.run_number }}_result.txt" + + # Wait for remote build to finish and commit results to git + - name: Wait until build results are available + shell: bash + # 180 minutes wait time. There will be overhead for the git pull command, so actual wait time will be slightly more than 180 minutes + run: "cd transfer-hydra ; for (( i=0; i<180; i++ )) ; do if [ -f ${{ github.run_id }}_${{ github.run_number }}_result.txt ] ; then break ; fi ; git pull --quiet ; sleep 60 ; false ; done || exit 1" + + # List files related to this build + - name: List files in transfer-hydra directory + shell: bash + run: "ls -lap transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_*" + + # Upload files related to this build + - name: Upload files in transfer-hydra directory as artifacts + # Aug 2023: Update from v2 to v3 which uses node16 (node12 is out of support). + uses: actions/upload-artifact@v3 + with: + name: build logs + path: "transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_*" + + # Show contents of result file + - name: Show build result information + shell: bash + run: "cat transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_result.txt" + + # Grep the results file to show failures + - name: Exit with error if a build failed + # Default shell includes "-o pipefail" and "-e". Specify a different shell + shell: bash --noprofile --norc {0} + run: "if grep -i failed transfer-hydra/${{ github.run_id }}_${{ github.run_number }}_result.txt; then exit 1; else exit 0; fi" \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000000..f22386a5d2 --- /dev/null +++ b/.gitignore @@ -0,0 +1,88 @@ +##### Editor Specific Ignores ##### + +## Common Editor Resources ## +GTAGS +GRTAGS +GPATH +*.clang_complete +compile_flags.txt +compile_commands.json +# Git Merge conflict files +*.orig + +## Visual Studio / Visual Studio Code ## +*.vs +*.vscode +*.rsuser +*.suo +*.user +*.userosscache +*.sln.docstates + +## Jetbrains (CLion/PyCharm etc) ## +.idea/ + +## Eclipse ## +*.project +*.cproject +*.pydevproject +*.settings/ + +## XCode ## +xcuserdata/ +*.xcuserstate +*.pbxuser + +## Emacs ## +*.dir-locals.el +/.emacs.desktop +/.emacs.desktop.lock +*.elc + +## Vim ## +*.swp +Session.vim +Sessionx.vim + + +##### Build Specific Ignores ##### + +## CMake Artifacts ## +cmake-build-*/ +CMakePresets.json +CMakeUserPresets.json +CMakeSettings.json +CMakeCache.txt +CMakeLists.txt.user +CMakeFiles/cmake.check_cache +*build/ + +## Python ## +__pycache__/ +*.py[cod] +*$py.class +*.egg + +## Compiled Libs ## +*.so +*.dylib +*.dll + +##### Platform Specific Ignores ##### + +## macOS ## +.DS_Store +.AppleDouble +.LSOverride + +## Windows ## +Thumbs.db +Thumbs.db:encryptable +ehthumbs.db +ehthumbs_vista.db + +## Linux ## +.directory +.nfs +# Files ending in ~ are Linux convention backup files +*~ diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 0000000000..10ac6e9df6 --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,10 @@ +repos: +- repo: local + hooks: + + - id: clang-format + name: clang-format + description: Format files with clang-format + entry: .github/run-clang-format.py + language: python + stages: ['commit', 'manual'] diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000000..2006a247b1 --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,222 @@ +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +cmake_minimum_required(VERSION 3.13...3.17) + +project(maya-usd) + +#------------------------------------------------------------------------------ +# options +#------------------------------------------------------------------------------ +option(BUILD_TESTS "Build tests." ON) +option(BUILD_STRICT_MODE "Enforce all warnings as errors." ON) +option(BUILD_SHARED_LIBS "Build libraries as shared or static." ON) +option(BUILD_WITH_PYTHON_3 "Build with python 3." OFF) +if(APPLE) + option(BUILD_UB2 "Build Universal Binary 2 (UB2) Intel64+arm64" OFF) +endif() +set(BUILD_WITH_PYTHON_3_VERSION 3.7 CACHE STRING "The version of Python 3 to build with") +option(CMAKE_WANT_MATERIALX_BUILD "Enable building with MaterialX (experimental)." OFF) + +set(PXR_OVERRIDE_PLUGINPATH_NAME PXR_PLUGINPATH_NAME + CACHE STRING "Name of env var USD searches to find plugins") + +#------------------------------------------------------------------------------ +# internal flags to control build +#------------------------------------------------------------------------------ +# MAYAHYDRA_TO_USD_RELATIVE_PATH : Set this variable to any relative path from install +# folder to USD location. If defined will set relative +# rpaths for USD libraries. + +#------------------------------------------------------------------------------ +# global options +#------------------------------------------------------------------------------ +# Avoid noisy install messages +set(CMAKE_INSTALL_MESSAGE "NEVER") + +if(APPLE) + if(BUILD_UB2) + message(STATUS "Building with Universal Binary 2") + set(CMAKE_OSX_ARCHITECTURES "x86_64;arm64") + set(CMAKE_OSX_DEPLOYMENT_TARGET 11.0) + else() + set(CMAKE_OSX_ARCHITECTURES "x86_64") + set(CMAKE_OSX_DEPLOYMENT_TARGET 10.14) + endif() +endif() + +set(CMAKE_MODULE_PATH + ${CMAKE_MODULE_PATH} + ${CMAKE_CURRENT_SOURCE_DIR}/cmake/modules +) + +# Use RUNPATH instead of RPATH for all shared libs, executables and modules on Linux +if(${CMAKE_SYSTEM_NAME} MATCHES "Linux") # IS_LINUX not yet defined + set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,--enable-new-dtags") + set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,--enable-new-dtags") + set(CMAKE_MODULE_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,--enable-new-dtags") +endif() + +#------------------------------------------------------------------------------ +# modules and definitions +#------------------------------------------------------------------------------ +include(cmake/utils.cmake) +find_package(Maya 2024 REQUIRED) + +if(APPLE AND BUILD_UB2 AND NOT MAYA_MACOSX_BUILT_WITH_UB2) + message(WARNING "Maya was NOT built with Universal Binary 2") +endif() + +include(cmake/mayahydra_version.info) +set(MAYAHYDRA_VERSION "${MAYAHYDRA_MAJOR_VERSION}.${MAYAHYDRA_MINOR_VERSION}.${MAYAHYDRA_PATCH_LEVEL}") +if(DEFINED ENV{PLUGIN_CUT_ID}) + set(MAYAHYDRA_CUT_ID $ENV{PLUGIN_CUT_ID}) +endif() + +include(cmake/flowViewport_version.info) +set(FLOWVIEWPORT_VERSION "${FLOWVIEWPORT_MAJOR_VERSION}.${FLOWVIEWPORT_MINOR_VERSION}.${FLOWVIEWPORT_PATCH_LEVEL}") + +if (DEFINED PYTHON_INCLUDE_DIR AND DEFINED PYTHON_LIBRARIES AND DEFINED Python_EXECUTABLE) + SET(PYTHON_INCLUDE_DIRS "${PYTHON_INCLUDE_DIR}") + SET(PYTHONLIBS_FOUND TRUE) + # Use the Python module to find the python lib. + if(BUILD_WITH_PYTHON_3) + find_package(Python ${BUILD_WITH_PYTHON_3_VERSION} EXACT REQUIRED COMPONENTS Interpreter) + else() + find_package(Python 2.7 EXACT REQUIRED COMPONENTS Interpreter) + endif() + if(NOT Python_Interpreter_FOUND) + set(PYTHONLIBS_FOUND FALSE) + endif() +endif() +if (NOT PYTHONLIBS_FOUND) + include(cmake/python.cmake) +endif() +message(STATUS "Build mayaHydra with Python3 = " ${BUILD_WITH_PYTHON_3}) +message(STATUS " PYTHON_INCLUDE_DIRS = ${PYTHON_INCLUDE_DIRS}") +message(STATUS " PYTHON_LIBRARIES = ${PYTHON_LIBRARIES}") +message(STATUS " Python_EXECUTABLE = ${Python_EXECUTABLE}") + +include(cmake/jinja.cmake) + +find_package(USD 0.22.11 REQUIRED) +if (CMAKE_WANT_MATERIALX_BUILD) + # Requires at least USD 22.11 for hdMtlx module and USD must have been built with MaterialX: + if(NOT TARGET hdMtlx) + set(CMAKE_WANT_MATERIALX_BUILD OFF) + message(WARNING "Disabling MaterialX VP2 rendering: it is not supported by this USD package.") + endif() +endif() +include(cmake/usd.cmake) + +option(BUILD_MAYAHYDRALIB "Build the Maya-To-Hydra plugin and scene delegate." ON) + +find_package(UFE REQUIRED) +if(UFE_FOUND) + message(STATUS "Building with UFE ${UFE_VERSION}.") +else() + message(STATUS "UFE not found.") +endif() + +if(CMAKE_WANT_MATERIALX_BUILD AND MAYA_LIGHTAPI_VERSION LESS 2) + set(CMAKE_WANT_MATERIALX_BUILD OFF) + message(WARNING "Disabling MaterialX VP2 rendering: it is not supported by this version of Maya.") +endif() + +if(TARGET hdMtlx) + # MaterialX was built into USD. We need to know where to find MaterialX targets. + find_package(MaterialX REQUIRED) + if(TARGET MaterialXCore) + message(STATUS "Found MaterialX") + endif() +endif() + +if(MAYA_APP_VERSION VERSION_GREATER 2024) + # Look for Qt6 in the Maya devkit. + # The Qt6 archive in the Maya devkit contains everything needed for the normal cmake find_package. + set(CMAKE_PREFIX_PATH "${MAYA_DEVKIT_LOCATION}/Qt") + set(WANT_QT_VERSION 6.5) + find_package(Qt6 ${WANT_QT_VERSION} COMPONENTS Core Widgets QUIET) + if (Qt6_FOUND) + message(STATUS "Found Qt ${Qt6_VERSION} in Maya devkit. Hydra Scene Browser will be built.") + set(BUILD_HDSB_PLUGIN TRUE) + else() + message(WARNING "Could not find Qt ${WANT_QT_VERSION} in Maya devkit directory: ${MAYA_DEVKIT_LOCATION}. \ + You must extract Qt.tar.gz. Hydra Scene Browser will not be built.") + endif() +else() + message(WARNING "Maya ${MAYA_APP_VERSION} does not contain Qt6 in its devkit. \ + Hydra Scene Browser will not be built.") +endif() + +#------------------------------------------------------------------------------ +# compiler configuration +#------------------------------------------------------------------------------ +include(cmake/compiler_config.cmake) + +#------------------------------------------------------------------------------ +# gulrak filesystem +#------------------------------------------------------------------------------ +include(cmake/gulrak.cmake) + +#------------------------------------------------------------------------------ +# test +#------------------------------------------------------------------------------ +if (BUILD_TESTS) + include(cmake/googletest.cmake) + include(cmake/test.cmake) + fetch_googletest() + enable_testing() + add_subdirectory(test) +endif() + +#------------------------------------------------------------------------------ +# lib +#------------------------------------------------------------------------------ +add_subdirectory(lib) + +if(BUILD_HDSB_PLUGIN) + add_subdirectory(lib/adskHydraSceneBrowser) +endif() + +#------------------------------------------------------------------------------ +# plugin +#------------------------------------------------------------------------------ +if(BUILD_HDSB_PLUGIN) + add_subdirectory(plugin/mayaHydraSceneBrowser) + if (BUILD_TESTS) + add_subdirectory(plugin/mayaHydraSceneBrowserTest) + endif() +endif() + +#------------------------------------------------------------------------------ +# install +#------------------------------------------------------------------------------ +if (DEFINED MAYAHYDRA_TO_USD_RELATIVE_PATH) + set(USD_INSTALL_LOCATION "${CMAKE_INSTALL_PREFIX}/${MAYAHYDRA_TO_USD_RELATIVE_PATH}") +else() + set(USD_INSTALL_LOCATION ${PXR_USD_LOCATION}) +endif() + +#------------------------------------------------------------------------------ +# Maya module files +#------------------------------------------------------------------------------ +if (IS_WINDOWS) + configure_file("modules/mayaHydra_Win.mod.template" ${PROJECT_BINARY_DIR}/mayaHydra.mod) +else() + configure_file("modules/mayaHydra.mod.template" ${PROJECT_BINARY_DIR}/mayaHydra.mod) +endif() + +install(FILES ${PROJECT_BINARY_DIR}/mayaHydra.mod DESTINATION ${CMAKE_INSTALL_PREFIX}) diff --git a/README.md b/README.md new file mode 100644 index 0000000000..a3c1d2b669 --- /dev/null +++ b/README.md @@ -0,0 +1,80 @@ +# Hydra for Maya (Technology Preview) + +The _Hydra for Maya_ project creates a Maya plugin that replaces the main Maya viewport with a Hydra viewer. _Hydra for Maya_ is developed and maintained by Autodesk. The project and this documentation are a work-in-progress and under active development. The contents of this repository are fully open source and open to contributions under the [Apache license](./doc/LICENSE.md)! + +Hydra is the rendering API included with [Pixar's USD](http://openusd.org/). + +## Motivation + +The goal for _Hydra for Maya_ is to introduce Hydra as an open source viewport framework for Maya to extend the viewport render engine through Hydra render delegates. _Hydra for Maya_ uses the previous Maya to Hydra (MtoH) code, which is part of MayaUSD, as a foundation to build on. You can find more details on what changed from MtoH [here](./doc/mayaHydraDetails.md). +With _Hydra for Maya_ we are fully leveraging the new [SceneIndex API (aka Hydra v2)](https://openusd.org/release/api/class_hd_scene_index_base.html#details) for more flexibility and customizability over the Hydra scene graph. Therefore the HdSceneDelegate is *not supported* by this plugin. + +_Hydra for Maya_ is currently a Technology Preview; we are just laying the foundation and there is still work ahead. As the plugin evolves and the Hydra technology matures, you can expect changes to API and to face limitations. + +## What's it good for? + +This project allows you to use Hydra (the pluggable USD render ecosystem) +and Storm (the OpenGL renderer for Hydra, bundled with USD), as an +alternative to Viewport 2.0. + +Using Hydra has big benefits for offline renderers: any renderer that +implements a Hydra render delegate can now have an interactive render viewport +in Maya, along with support for render proxies. + +As an example, when paired with +[arnold-usd](https://github.com/Autodesk/arnold-usd) (Arnold's USD plugin + +render delegate), it provides an Arnold render of the viewport, where both maya +objects and USD objects (through proxies) can be modified interactively. + +This could also be particularly useful for newer renderers, like Radeon +ProRender (which already has a +[render delegate](https://github.com/GPUOpen-LibrariesAndSDKs/RadeonProRenderUSD)), +or in-house renderers, as an easier means of implementing an interactive +render viewport. + +What about HdStorm, Hydra's OpenGL renderer? Why would you want to use HdStorm +instead of "normal" Viewport 2.0, given that there are other methods for displaying +USD proxies in Viewport 2.0? Some potential reasons include: + +- Using HdStorm gives lighting and shading between Hydra-enabled applications: + Maya, Katana, usdview, etc +- HdStorm is open source: you can add core features as you need them +- HdStorm is extensible: you can create plugins for custom objects, which then allows + them to be displayed not just in Maya, but any Hydra-enabled application + +## Known Limitations + +- Only direct texture inputs are supported for Maya materials +- Limited support for Maya shader networks +- Drawing issues with selection and highlighting +- Hydra shading differs from Maya's Viewport 2.0 +- Animation Ghosting has the wrong shading +- Current limitations with USD prims: + - Maya layers don't show effect + - Isolate Select only functions with Maya nodes + - Viewport display modes like Xray, wireframe or default shading do not function + - Selection highlighting not showing + - Gprims currently don't cast shadows in Storm +- No material bindings on GeomSubsets (Hydra v2 limitation) + +## Currently not supported + +- Backface culling +- Screen space effects like depth of field and motion blur +- Arnold lights except the aiSkyDomeLight +- Following Maya node types: + - Bifrost + - nParticles + - Fluid +- Blue Pencil +- Maya's volume and ambient light +- Hardware Fog +- Maya's procedural textures (e.g. noise or pattern) + +## Detailed Documentation + ++ [Contributing](./doc/CONTRIBUTING.md) ++ [Building the mayaHydra.mll plugin](./doc/build.md) ++ [Coding guidelines](./doc/codingGuidelines.md) ++ [License](./doc/LICENSE.md) ++ [Plugin documentation](./README_DOC.md) \ No newline at end of file diff --git a/README_DOC.md b/README_DOC.md new file mode 100644 index 0000000000..4017936ca2 --- /dev/null +++ b/README_DOC.md @@ -0,0 +1,4 @@ +## Plugin Documentation ++ [Technical details of Hydra for Maya](./doc/mayaHydraDetails.md) ++ [Selection in Hydra for Maya](./doc/mayaHydraSelection.md) ++ [Selection Highlighting Architecture](./doc/selectionHighlightingArchitecture.md) \ No newline at end of file diff --git a/build.py b/build.py new file mode 100755 index 0000000000..2181ce9f27 --- /dev/null +++ b/build.py @@ -0,0 +1,754 @@ +#!/usr/bin/env python + +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +from __future__ import print_function + +from distutils.spawn import find_executable +from glob import glob + +import argparse +import contextlib +import codecs +import datetime +import distutils +import distutils.util +import multiprocessing +import os +import platform +import re +import shlex +import shutil +import subprocess +import sys +import tarfile +import time +import zipfile + +############################################################ +# Helpers for printing output +verbosity = 1 + +def Print(msg): + if verbosity > 0: + print(msg) + +def PrintWarning(warning): + if verbosity > 0: + print("WARNING:", warning) + +def PrintStatus(status): + if verbosity >= 1: + print("STATUS:", status) + +def PrintInfo(info): + if verbosity >= 2: + print("INFO:", info) + +def PrintCommandOutput(output): + if verbosity >= 3: + sys.stdout.write(output) + +def PrintError(error): + if verbosity >= 3 and sys.exc_info()[1] is not None: + import traceback + traceback.print_exc() + print("ERROR:", error) + +def Python3(): + return sys.version_info.major == 3 +############################################################ +def Windows(): + return platform.system() == "Windows" + +def Linux(): + return platform.system() == "Linux" + +def MacOS(): + return platform.system() == "Darwin" + +def GetCommandOutput(command): + """Executes the specified command and returns output or None.""" + try: + return subprocess.check_output( + shlex.split(command), stderr=subprocess.STDOUT).strip() + except subprocess.CalledProcessError: + pass + return None + +def GetGitHeadInfo(context): + """Returns HEAD commit id and date.""" + try: + with CurrentWorkingDirectory(context.mayaHydraSrcDir): + commitSha = subprocess.check_output('git rev-parse HEAD', shell = True).decode() + commitDate = subprocess.check_output('git show -s HEAD --format="%ad"', shell = True).decode() + return commitSha, commitDate + except Exception as exp: + PrintError("Failed to run git commands : {exp}".format(exp=exp)) + sys.exit(1) + +def GetXcodeDeveloperDirectory(): + """Returns the active developer directory as reported by 'xcode-select -p'. + Returns None if none is set.""" + if not MacOS(): + return None + + return GetCommandOutput("xcode-select -p") + +def GetVisualStudioCompilerAndVersion(): + """Returns a tuple containing the path to the Visual Studio compiler + and a tuple for its version, e.g. (14, 0). If the compiler is not found + or version number cannot be determined, returns None.""" + if not Windows(): + return None + + msvcCompiler = find_executable('cl') + if msvcCompiler: + # VisualStudioVersion environment variable should be set by the + # Visual Studio Command Prompt. + match = re.search( + r"(\d+)\.(\d+)", + os.environ.get("VisualStudioVersion", "")) + if match: + return (msvcCompiler, tuple(int(v) for v in match.groups())) + return None + +def IsVisualStudioVersionOrGreater(desiredVersion): + if not Windows(): + return False + + msvcCompilerAndVersion = GetVisualStudioCompilerAndVersion() + if msvcCompilerAndVersion: + _, version = msvcCompilerAndVersion + return version >= desiredVersion + return False + +def IsVisualStudio2022OrGreater(): + VISUAL_STUDIO_2022_VERSION = (17, 0) + return IsVisualStudioVersionOrGreater(VISUAL_STUDIO_2022_VERSION) + +def IsVisualStudio2019OrGreater(): + VISUAL_STUDIO_2019_VERSION = (16, 0) + return IsVisualStudioVersionOrGreater(VISUAL_STUDIO_2019_VERSION) + +def IsVisualStudio2017OrGreater(): + VISUAL_STUDIO_2017_VERSION = (15, 0) + return IsVisualStudioVersionOrGreater(VISUAL_STUDIO_2017_VERSION) + +def GetCPUCount(): + try: + return multiprocessing.cpu_count() + except NotImplementedError: + return 1 + +def Run(context, cmd): + """Run the specified command in a subprocess.""" + PrintInfo('Running "{cmd}"'.format(cmd=cmd)) + + with codecs.open(context.logFileLocation, "a", "utf-8") as logfile: + logfile.write("#####################################################################################" + "\n") + logfile.write("log date: " + datetime.datetime.now().strftime("%Y-%m-%d %H:%M") + "\n") + commitID,commitData = GetGitHeadInfo(context) + logfile.write("commit sha: " + commitID) + logfile.write("commit date: " + commitData) + logfile.write("#####################################################################################" + "\n") + logfile.write("\n") + logfile.write(cmd) + logfile.write("\n") + + # Let exceptions escape from subprocess calls -- higher level + # code will handle them. + if context.redirectOutstreamFile: + p = subprocess.Popen(shlex.split(cmd), stdout=subprocess.PIPE, + stderr=subprocess.STDOUT) + encoding = sys.stdout.encoding or "UTF-8" + while True: + l = p.stdout.readline().decode(encoding) + if l != "": + # Avoid "UnicodeEncodeError: 'ascii' codec can't encode + # character" errors by serializing utf8 byte strings. + logfile.write(l) + PrintCommandOutput(l) + elif p.poll() is not None: + break + else: + p = subprocess.Popen(shlex.split(cmd)) + p.wait() + + if p.returncode != 0: + # If verbosity >= 3, we'll have already been printing out command output + # so no reason to print the log file again. + if verbosity < 3: + with open(context.logFileLocation, "r") as logfile: + Print(logfile.read()) + raise RuntimeError("Failed to run '{cmd}'\nSee {log} for more details." + .format(cmd=cmd, log=os.path.abspath(context.logFileLocation))) + +def BuildVariant(context): + if context.buildDebug: + return "Debug" + elif context.buildRelease: + return "Release" + elif context.buildRelWithDebug: + return "RelWithDebInfo" + return "RelWithDebInfo" + +def FormatMultiProcs(numJobs, generator): + tag = "-j" + if generator: + if "Visual Studio" in generator: + tag = "/M:" + elif "Xcode" in generator: + tag = "-j " + + return "{tag}{procs}".format(tag=tag, procs=numJobs) + +def onerror(func, path, exc_info): + """ + If the error is due to an access error (read only file) + add write permission and then retries. + If the error is for another reason it re-raises the error. + """ + import stat + if not os.access(path, os.W_OK): + os.chmod(path, stat.S_IWUSR) + func(path) + else: + raise + +def StartBuild(): + global start_time + start_time = time.time() + +def StopBuild(): + end_time = time.time() + elapsed_seconds = end_time - start_time + hours, remainder = divmod(elapsed_seconds, 3600) + minutes, seconds = divmod(remainder, 60) + print("Elapsed time: {:02}:{:02}:{:02}".format(int(hours), int(minutes), int(seconds))) + +############################################################ +# contextmanager +@contextlib.contextmanager +def CurrentWorkingDirectory(dir): + """Context manager that sets the current working directory to the given + directory and resets it to the original directory when closed.""" + curdir = os.getcwd() + os.chdir(dir) + try: + yield + finally: + os.chdir(curdir) + +############################################################ +# CMAKE +def RunCMake(context, extraArgs=None, stages=None): + """Invoke CMake to configure, build, and install a library whose + source code is located in the current working directory.""" + + srcDir = os.getcwd() + instDir = context.instDir + buildDir = context.buildDir + + if 'clean' in stages and os.path.isdir(buildDir): + shutil.rmtree(buildDir, onerror=onerror) + + if 'clean' in stages and os.path.isdir(instDir): + shutil.rmtree(instDir) + + if not os.path.isdir(buildDir): + os.makedirs(buildDir) + + generator = context.cmakeGenerator + + # On Windows, we need to explicitly specify the generator to ensure we're + # building a 64-bit project. (Surely there is a better way to do this?) + # TODO: figure out exactly what "vcvarsall.bat x64" sets to force x64 + if generator is None and Windows(): + if IsVisualStudio2022OrGreater(): + generator = "Visual Studio 17 2022" + elif IsVisualStudio2019OrGreater(): + generator = "Visual Studio 16 2019" + elif IsVisualStudio2017OrGreater(): + generator = "Visual Studio 15 2017 Win64" + else: + generator = "Visual Studio 14 2015 Win64" + + if generator is not None: + generator = '-G "{gen}"'.format(gen=generator) + + if generator and 'Visual Studio' in generator and IsVisualStudio2019OrGreater(): + generator = generator + " -A x64" + + # get build variant + variant= BuildVariant(context) + + with CurrentWorkingDirectory(buildDir): + # recreate build_log.txt everytime the script runs + if os.path.isfile(context.logFileLocation): + os.remove(context.logFileLocation) + + if 'configure' in stages: + Run(context, + 'cmake ' + '-DCMAKE_INSTALL_PREFIX="{instDir}" ' + '-DCMAKE_BUILD_TYPE={variant} ' + '-DCMAKE_EXPORT_COMPILE_COMMANDS=ON ' + '{generator} ' + '{extraArgs} ' + '"{srcDir}"' + .format(instDir=instDir, + variant=variant, + srcDir=srcDir, + generator=(generator or ""), + extraArgs=(" ".join(extraArgs) if extraArgs else ""))) + + installArg = "" + if 'install' in stages: + installArg = "--target install" + + if 'build' in stages or 'install' in stages: + Run(context, "cmake --build . --config {variant} {installArg} -- {multiproc}" + .format(variant=variant, + installArg=installArg, + multiproc=FormatMultiProcs(context.numJobs, generator))) + +def RunCTest(context, extraArgs=None): + buildDir = context.buildDir + variant = BuildVariant(context) + + with CurrentWorkingDirectory(buildDir): + Run(context, + 'ctest ' + '--output-on-failure ' + '--timeout 500 ' + '-C {variant} ' + '{extraArgs} ' + .format(variant=variant, + extraArgs=(" ".join(extraArgs) if extraArgs else ""))) + +def RunMakeZipArchive(context): + installDir = context.instDir + buildDir = context.buildDir + pkgDir = context.pkgDir + variant = BuildVariant(context) + + # extract version from mayahydra_version.info + mayaHydraVersion = [] + cmakeInfoDir = os.path.join(context.mayaHydraSrcDir, 'cmake') + filename = os.path.join(cmakeInfoDir, 'mayahydra_version.info') + with open(filename, 'r') as filehandle: + content = filehandle.readlines() + for current_line in content: + digitList = re.findall(r'\d+', current_line) + versionStr = ''.join(str(e) for e in digitList) + mayaHydraVersion.append(versionStr) + + majorVersion = mayaHydraVersion[0] + minorVersion = mayaHydraVersion[1] + patchLevel = mayaHydraVersion[2] + + pkgName = 'MayaHydra' + '-' + majorVersion + '.' + minorVersion + '.' + patchLevel + '-' + (platform.system()) + '-' + variant + with CurrentWorkingDirectory(buildDir): + shutil.make_archive(pkgName, 'zip', installDir) + + # copy zip file to package directory + if not os.path.exists(pkgDir): + os.makedirs(pkgDir) + + for file in os.listdir(buildDir): + if file.endswith(".zip"): + zipFile = os.path.join(buildDir, file) + try: + shutil.copy(zipFile, pkgDir) + except Exception as exp: + PrintError("Failed to write to directory {pkgDir} : {exp}".format(pkgDir=pkgDir,exp=exp)) + sys.exit(1) + +def SetupMayaQt(context): + def haveQtHeaders(rootPath): + if os.path.exists(rootPath): + # MayaHydra uses these components from Qt (so at a minimum we must find them). + qtComponentsToFind = ['QtCore', 'QtWidgets'] + # Qt6 includes the entire Qt in a single zip file, which when extracted ends in folder 'Qt'. + startDir = os.path.join(rootPath, 'Qt', 'include') if os.path.exists(os.path.join(rootPath, 'Qt')) else os.path.join(rootPath, 'include') + for root,dirs,files in os.walk(startDir): + if 'qt' not in root.lower() or not files: + continue + if not any(root.endswith(qtComp) for qtComp in qtComponentsToFind): + # Skip any folders that aren't the components we are looking for. + continue + + for qtComp in qtComponentsToFind[:]: # Loop over slice copy as we remove items + if qtComp in root and '{comp}version.h'.format(comp=qtComp.lower()) in files: + qtComponentsToFind.remove(qtComp) + PrintInfo('Found {comp} in {dir}'.format(comp=qtComp, dir=root)) + break # Once we've found (and removed) a component, move to the next os.walk + + if not qtComponentsToFind: # Once we've found them all, we are done. + return True + + def safeTarfileExtract(members): + """Use a function to look for bad paths in the tarfile archive to fix + security/bandit B202: tarfile_unsafe_members.""" + + def isBadPath(path, base): + return not os.path.realpath(os.path.abspath(os.path.join(base, path))).startswith(base) + def isBadLink(info, base): + # Links are interpreted relative to the directory containing the link. + tip = os.path.realpath(os.path.abspath(os.path.join(base, os.path.dirname(info.name)))) + return isBadPath(info.linkname, base=tip) + + base = os.path.realpath(os.path.abspath('.')) + result = [] + for finfo in members: + # If any bad paths for links are found in the tarfile, print an error + # and don't extract anything from tarfile. + if isBadPath(finfo.name, base): + PrintError('Found illegal path {path} in tarfile, blocking tarfile extraction.'.format(path=finfo.name)) + return [] + elif (finfo.issym() or finfo.islnk()) and isBadLink(finfo, base): + PrintError('Found illegal link {link} in tarfile, blocking tarfile extraction.'.format(link=finfo.linkname)) + return [] + else: + result.append(finfo) + return result + + def safeZipfileExtract(zip_file, extract_path='.'): + with zipfile.ZipFile(zip_file, 'r') as zf: + for member in zf.infolist(): + file_path = os.path.realpath(os.path.join(extract_path, member.filename)) + if file_path.startswith(os.path.realpath(extract_path)): + zf.extract(member, extract_path) + + # The list of directories (in order) that we'll search. + dirsToSearch = [context.devkitLocation] + if 'MAYA_DEVKIT_LOCATION' in os.environ: + dirsToSearch.append(os.path.expandvars('$MAYA_DEVKIT_LOCATION')) + dirsToSearch.append(context.mayaLocation) + if 'MAYA_LOCATION' in os.environ: + dirsToSearch.append(os.path.expandvars('$MAYA_LOCATION')) + + # Check if the Qt zip file has been extracted (we need the Qt headers). + for dirToSearch in dirsToSearch: + if haveQtHeaders(dirToSearch): + PrintStatus('Found Maya Qt headers in: {dir}'.format(dir=dirToSearch)) + return + + # Qt6 + # The entire Qt is in a single zip file, which we extract to 'Qt'. + # Then we can simply use find_package(Qt6) on it. + for dirToSearch in dirsToSearch: + # Qt archive was originally named Qt.tar.gz on all platforms. + # Was eventually renamed to Qt.zip (Windows) and Qt.tgz (Linux/Osx). + qtArchiveNames = ['Qt.zip', 'Qt.tar.gz'] if Windows() else ['Qt.tgz', 'Qt.tar.gz'] + for qtArchiveName in qtArchiveNames: + qtArchive = os.path.join(dirToSearch, qtArchiveName) + if os.path.exists(qtArchive): + ext = os.path.splitext(qtArchiveName)[1] + qtZipDirFolder = os.path.dirname(qtArchive) + if os.access(qtZipDirFolder, os.W_OK): + PrintStatus("Could not find Maya Qt6.") + PrintStatus(" Extracting '{zip}' to '{dir}'".format(zip=qtArchive, dir=qtZipDirFolder)) + try: + if ext == '.zip': + safeZipfileExtract(qtArchive, qtZipDirFolder) + else: + archive = tarfile.open(qtArchive, mode='r') + archive.extractall(qtZipDirFolder, members=safeTarfileExtract(archive.getmembers())) + archive.close() + except zipfile.BadZipfile as error: + PrintError(str(error)) + except tarfile.TarError as error: + PrintError(str(error)) + return + +def BuildAndInstall(context, buildArgs, stages): + with CurrentWorkingDirectory(context.mayaHydraSrcDir): + extraArgs = [] + stagesArgs = [] + if context.mayaLocation: + extraArgs.append('-DMAYA_LOCATION="{mayaLocation}"' + .format(mayaLocation=context.mayaLocation)) + + if context.mayaUsdLocation: + extraArgs.append('-DMAYAUSD_LOCATION="{mayaUsdLocation}"' + .format(mayaUsdLocation=context.mayaUsdLocation)) + + if context.pxrUsdLocation: + extraArgs.append('-DPXR_USD_LOCATION="{pxrUsdLocation}"' + .format(pxrUsdLocation=context.pxrUsdLocation)) + + if context.devkitLocation: + extraArgs.append('-DMAYA_DEVKIT_LOCATION="{devkitLocation}"' + .format(devkitLocation=context.devkitLocation)) + + extraArgs += buildArgs + stagesArgs += stages + + RunCMake(context, extraArgs, stagesArgs) + + # Ensure directory structure is created and is writable. + for dir in [context.workspaceDir, context.buildDir, context.instDir]: + try: + if os.path.isdir(dir): + testFile = os.path.join(dir, "canwrite") + open(testFile, "w").close() + os.remove(testFile) + else: + os.makedirs(dir) + except Exception as e: + PrintError("Could not write to directory {dir}. Change permissions " + "or choose a different location to install to." + .format(dir=dir)) + sys.exit(1) + Print("""Success MayaHydra build and install !!!!""") + +def RunTests(context,extraArgs): + RunCTest(context,extraArgs) + Print("""Success running MayaHydra tests !!!!""") + +def Package(context): + RunMakeZipArchive(context) + Print("""Success packaging MayaHydra !!!!""") + Print('Archived package is available in {pkgDir}'.format(pkgDir=context.pkgDir)) + +############################################################ +# ArgumentParser +parser = argparse.ArgumentParser( + formatter_class=argparse.RawDescriptionHelpFormatter) + +parser.add_argument("workspace_location", type=str, + help="Directory where the project use as a workspace to build and install plugin/libraries.") + +parser.add_argument("--generator", type=str, + help=("CMake generator to use when building libraries with " + "cmake")) + +parser.add_argument("-v", "--verbosity", type=int, default=verbosity, + help=("How much output to print while building: 0 = no " + "output; 1 = warnings + status; 2 = info; 3 = " + "command output and tracebacks (default: " + "%(default)s)")) + +parser.add_argument("--build-location", type=str, + help=("Set Build directory " + "(default: /build-location)")) + +parser.add_argument("--install-location", type=str, + help=("Set Install directory " + "(default: /install-location)")) + +parser.add_argument("--maya-location", type=str, + help="Directory where Maya is installed.") + +parser.add_argument("--mayausd-location", type=str, + help="Directory where MayaUsd is installed.") + +parser.add_argument("--pxrusd-location", type=str, + help="Directory where Pixar USD is installed.") + +parser.add_argument("--devkit-location", type=str, + help="Directory where Maya Devkit is installed.") + +varGroup = parser.add_mutually_exclusive_group() +varGroup.add_argument("--build-debug", dest="build_debug", action="store_true", + help="Build in Debug mode (default: %(default)s)") + +varGroup.add_argument("--build-release", dest="build_release", action="store_true", + help="Build in Release mode (default: %(default)s)") + +varGroup.add_argument("--build-relwithdebug", dest="build_relwithdebug", action="store_true", default=True, + help="Build in RelWithDebInfo mode (default: %(default)s)") + +parser.add_argument("--debug-python", dest="debug_python", action="store_true", + help="Define Boost Python Debug if your Python library comes with Debugging symbols (default: %(default)s).") + +# HYDRA-444 build infrastructure for Hydra Scene Browser Library +parser.add_argument("--qt-location", type=str, + help="DEPRECATED: Qt is found automatically in Maya devkit.") + +parser.add_argument("--build-args", type=str, nargs="*", default=[], + help=("Comma-separated list of arguments passed into CMake when building libraries")) + +parser.add_argument("--ctest-args", type=str, nargs="*", default=[], + help=("Comma-separated list of arguments passed into CTest.(e.g -VV, --output-on-failure)")) + +parser.add_argument("--stages", type=str, nargs="*", default=['clean','configure','build','install'], + help=("Comma-separated list of stages to execute.(possible stages: clean, configure, build, install, test, package)")) + +parser.add_argument("-j", "--jobs", type=int, default=GetCPUCount(), + help=("Number of build jobs to run in parallel. " + "(default: # of processors [{0}])" + .format(GetCPUCount()))) + +parser.add_argument("--redirect-outstream-file", type=distutils.util.strtobool, dest="redirect_outstream_file", default=True, + help="Redirect output stream to a file. Set this flag to false to redirect output stream to console instead.") + +args = parser.parse_args() +verbosity = args.verbosity + +############################################################ +# InstallContext +class InstallContext: + def __init__(self, args): + + # Assume the project's top level cmake is in the current source directory + self.mayaHydraSrcDir = os.path.normpath( + os.path.join(os.path.abspath(os.path.dirname(__file__)))) + + # Build type + # Must be done early, so we can call BuildVariant(self) + self.buildDebug = args.build_debug + self.buildRelease = args.build_release + self.buildRelWithDebug = args.build_relwithdebug + + self.debugPython = args.debug_python + + # Workspace directory + self.workspaceDir = os.path.abspath(args.workspace_location) + + # Build directory + self.buildDir = (os.path.abspath(args.build_location) if args.build_location + else os.path.join(self.workspaceDir, "build", BuildVariant(self))) + + # Install directory + self.instDir = (os.path.abspath(args.install_location) if args.install_location + else os.path.join(self.workspaceDir, "install", BuildVariant(self))) + + # Package directory + self.pkgDir = (os.path.join(self.workspaceDir, "package", BuildVariant(self))) + + # CMake generator + self.cmakeGenerator = args.generator + + # Number of jobs + self.numJobs = args.jobs + if self.numJobs <= 0: + raise ValueError("Number of jobs must be greater than 0") + + # Maya Location + self.mayaLocation = (os.path.abspath(args.maya_location) + if args.maya_location else None) + + # MayaUsd Location + self.mayaUsdLocation = (os.path.abspath(args.mayausd_location) + if args.mayausd_location else None) + + # PXR USD Location + self.pxrUsdLocation = (os.path.abspath(args.pxrusd_location) + if args.pxrusd_location else None) + + # Maya Devkit Location + self.devkitLocation = (os.path.abspath(args.devkit_location) + if args.devkit_location else None) + + # DEPRECATED: Qt Location + if args.qt_location: + PrintWarning("--qt-location flag is deprecated as Qt is found automatically in Maya devkit.") + + # Log File Name + logFileName="build_log.txt" + self.logFileLocation=os.path.join(self.buildDir, logFileName) + + # Build arguments + self.buildArgs = list() + for argList in args.build_args: + for arg in argList.split(","): + self.buildArgs.append(arg) + + # Stages arguments + self.stagesArgs = list() + for argList in args.stages: + for arg in argList.split(","): + self.stagesArgs.append(arg) + + # CTest arguments + self.ctestArgs = list() + for argList in args.ctest_args: + for arg in argList.split(","): + self.ctestArgs.append(arg) + + # Redirect output stream to file + self.redirectOutstreamFile = args.redirect_outstream_file +try: + context = InstallContext(args) +except Exception as e: + PrintError(str(e)) + sys.exit(1) + +if __name__ == "__main__": + # Summarize + summaryMsg = """ + Building with settings: + Source directory {mayaHydraSrcDir} + Workspace directory {workspaceDir} + Build directory {buildDir} + Install directory {instDir} + Variant {buildVariant} + Python Debug {debugPython} + CMake generator {cmakeGenerator}""" + + if context.redirectOutstreamFile: + summaryMsg += """ + Build Log {logFileLocation}""" + + if context.buildArgs: + summaryMsg += """ + Build arguments {buildArgs}""" + + if context.stagesArgs: + summaryMsg += """ + Stages arguments {stagesArgs}""" + + if context.ctestArgs: + summaryMsg += """ + CTest arguments {ctestArgs}""" + + summaryMsg = summaryMsg.format( + mayaHydraSrcDir=context.mayaHydraSrcDir, + workspaceDir=context.workspaceDir, + buildDir=context.buildDir, + instDir=context.instDir, + logFileLocation=context.logFileLocation, + buildArgs=context.buildArgs, + stagesArgs=context.stagesArgs, + ctestArgs=context.ctestArgs, + buildVariant=BuildVariant(context), + debugPython=("On" if context.debugPython else "Off"), + cmakeGenerator=("Default" if not context.cmakeGenerator + else context.cmakeGenerator) + ) + + Print(summaryMsg) + + # Make sure Qt from Maya devkit is ready. + if 'configure' in context.stagesArgs: + SetupMayaQt(context) + + # BuildAndInstall + if any(stage in ['clean', 'configure', 'build', 'install'] for stage in context.stagesArgs): + StartBuild() + BuildAndInstall(context, context.buildArgs, context.stagesArgs) + StopBuild() + + # Run Tests + if 'test' in context.stagesArgs: + RunTests(context, context.ctestArgs) + + # Package + if 'package' in context.stagesArgs: + Package(context) diff --git a/cmake/compiler_config.cmake b/cmake/compiler_config.cmake new file mode 100644 index 0000000000..d0612c1248 --- /dev/null +++ b/cmake/compiler_config.cmake @@ -0,0 +1,128 @@ +#------------------------------------------------------------------------------ +# compiler flags/definitions +#------------------------------------------------------------------------------ +set(GNU_CLANG_FLAGS + # we want to be as strict as possible + -Wall + $<$:-Werror> + $<$:-fstack-check> + # optimization + -msse4.2 + # disable warnings + -Wno-deprecated + -Wno-deprecated-declarations + -Wno-unused-local-typedefs +) + +if (CMAKE_CXX_COMPILER_ID MATCHES "Clang") + list(APPEND GNU_CLANG_FLAGS + -Wrange-loop-analysis + ) +endif() + +set(MSVC_FLAGS + # we want to be as strict as possible + /W3 + $<$:/WX> + # enable two-phase name lookup and other strict checks (binding a non-const reference to a temporary, etc..) + $<$>:/permissive-> + # enable pdb generation. + /Zi + # standards compliant. + /Zc:rvalueCast + # The /Zc:inline option strips out the "arch_ctor_" symbols used for + # library initialization by ARCH_CONSTRUCTOR starting in Visual Studio 2019, + # causing release builds to fail. Disable the option for this and later + # versions. + # + # For more details, see: + # https://developercommunity.visualstudio.com/content/problem/914943/zcinline-removes-extern-symbols-inside-anonymous-n.html + $,/Zc:inline-,/Zc:inline> + # enable multiprocessor builds. + /MP + # enable exception handling. + /EHsc + # enable initialization order as a level 3 warning + /w35038 + # disable warnings + /wd4244 + /wd4267 + /wd4273 + /wd4305 + /wd4506 + /wd4996 + /wd4180 + # exporting STL classes + /wd4251 + # Set some warnings as errors (to make it similar to Linux) + /we4101 + /we4189 +) + +set(MSVC_DEFINITIONS + # Make sure WinDef.h does not define min and max macros which + # will conflict with std::min() and std::max(). + NOMINMAX + + _CRT_SECURE_NO_WARNINGS + _SCL_SECURE_NO_WARNINGS + + # Boost + BOOST_ALL_DYN_LINK + BOOST_CONFIG_SUPPRESS_OUTDATED_MESSAGE + + # Needed to prevent Python from adding a define for snprintf + # since it was added in Visual Studio 2015. + HAVE_SNPRINTF +) + +#------------------------------------------------------------------------------ +# compiler configuration +#------------------------------------------------------------------------------ +# Do not use GNU extension +# Use -std=c++11 instead of -std=gnu++11 +set(CMAKE_CXX_EXTENSIONS OFF) + +function(mayaHydra_compile_config TARGET) + # required compiler feature + # Require C++14 if we're either building for Maya 2019 or later, or if we're building against + # USD 20.05 or later. Otherwise require C++11. + if ((MAYA_APP_VERSION VERSION_GREATER_EQUAL 2019) OR (PXR_VERSION VERSION_GREATER_EQUAL 2005)) + target_compile_features(${TARGET} + PRIVATE + cxx_std_14 + ) + else() + target_compile_features(${TARGET} + PRIVATE + cxx_std_11 + ) + endif() + if(IS_GNU OR IS_CLANG) + target_compile_options(${TARGET} + PRIVATE + ${GNU_CLANG_FLAGS} + ) + if(IS_LINUX) + target_compile_definitions(${TARGET} + PRIVATE + _GLIBCXX_USE_CXX11_ABI=$,1,0> + ) + endif() + elseif(IS_MSVC) + target_compile_options(${TARGET} + PRIVATE + ${MSVC_FLAGS} + ) + target_compile_definitions(${TARGET} + PRIVATE + ${MSVC_DEFINITIONS} + ) + endif() + + # Remove annoying TBB warnings. + target_compile_definitions(${TARGET} + PRIVATE + TBB_SUPPRESS_DEPRECATED_MESSAGES + ) +endfunction() diff --git a/cmake/flowViewport_version.info b/cmake/flowViewport_version.info new file mode 100644 index 0000000000..61f68f8a74 --- /dev/null +++ b/cmake/flowViewport_version.info @@ -0,0 +1,3 @@ +set(FLOWVIEWPORT_MAJOR_VERSION 0) +set(FLOWVIEWPORT_MINOR_VERSION 1) +set(FLOWVIEWPORT_PATCH_LEVEL 0) diff --git a/cmake/googletest.cmake b/cmake/googletest.cmake new file mode 100644 index 0000000000..3775389e81 --- /dev/null +++ b/cmake/googletest.cmake @@ -0,0 +1,93 @@ +macro(fetch_googletest) + + if (NOT GTEST_FOUND) + # First see if we can find a gtest that was downloaded and built. + if (NOT GOOGLETEST_BUILD_ROOT) + set(GOOGLETEST_BUILD_ROOT ${CMAKE_CURRENT_BINARY_DIR}) + endif() + if (NOT GTEST_ROOT) + set(GTEST_ROOT "${GOOGLETEST_BUILD_ROOT}/googletest-install") + endif() + find_package(GTest QUIET) + # At this point GTEST_FOUND is set to True in Release but False in Debug. + endif() + + if (NOT GTEST_FOUND) + #====================================================================== + # Download and unpack googletest at configure time. Adapted from + # + # https://github.com/abseil/googletest/blob/master/googletest/README.md + # + # PPT, 22-Nov-2018. + + # Immediately convert CMAKE_MAKE_PROGRAM to forward slashes (if required). + # Attempting to do so in execute_process fails with string invalid escape + # sequence parsing errors. PPT, 22-Nov-2018. + file(TO_CMAKE_PATH ${CMAKE_MAKE_PROGRAM} CMAKE_MAKE_PROGRAM) + + # Set some options used when compiling googletest. + set(CMAKE_CXX_STANDARD 11) + set(CMAKE_CXX_EXTENSIONS OFF) + set(CMAKE_CXX_STANDARD_REQUIRED ON) + if (${CMAKE_SYSTEM_NAME} MATCHES "Linux") + # In gcc 11 there was a new warning (about uninit variable) in googletest. + # Since we have warnings as errors, this causes build error. + # Simply disable all warnings in googletest since we won't fix them anyways. + # We will just update to newer version, if required. + set(disable_all_warnings_flag -w) + + set(glibcxx_abi -D_GLIBCXX_USE_CXX11_ABI=$,1,0>) + endif() + + if (GOOGLETEST_SRC_DIR) + configure_file(cmake/googletest_src.txt.in ${GOOGLETEST_BUILD_ROOT}/googletest-config/CMakeLists.txt) + else() + configure_file(cmake/googletest_download.txt.in ${GOOGLETEST_BUILD_ROOT}/googletest-config/CMakeLists.txt) + endif() + + message(STATUS "========== Installing GoogleTest... ==========") + execute_process(COMMAND "${CMAKE_COMMAND}" -G "${CMAKE_GENERATOR}" -DCMAKE_MAKE_PROGRAM=${CMAKE_MAKE_PROGRAM} . + RESULT_VARIABLE result + WORKING_DIRECTORY ${GOOGLETEST_BUILD_ROOT}/googletest-config ) + if(result) + message(FATAL_ERROR "CMake step for googletest failed: ${result}") + endif() + + execute_process(COMMAND "${CMAKE_COMMAND}" --build . --config ${CMAKE_BUILD_TYPE} + RESULT_VARIABLE result + WORKING_DIRECTORY ${GOOGLETEST_BUILD_ROOT}/googletest-config ) + if(result) + message(FATAL_ERROR "Build step for googletest failed: ${result}") + endif() + message(STATUS "========== ... GoogleTest installed. ==========") + + set(GTEST_ROOT "${GOOGLETEST_BUILD_ROOT}/googletest-install" CACHE PATH "GoogleTest installation root") + endif() + + # FindGTest should get call after GTEST_ROOT is set + find_package(GTest QUIET) + + # https://gitlab.kitware.com/cmake/cmake/issues/17799 + # FindGtest is buggy when dealing with Debug build. + if (CMAKE_BUILD_TYPE MATCHES Debug AND GTEST_FOUND MATCHES FALSE) + # FindGTest.cmake is buggy when looking for only debug config (it expects both). + # So when in debug we set the required gtest vars to the debug libs it would have + # found in the find_package(GTest) above. Then we find again. This will then + # properly set all the vars and import targets for just debug. + set(GTEST_LIBRARY ${GTEST_LIBRARY_DEBUG}) + set(GTEST_MAIN_LIBRARY ${GTEST_MAIN_LIBRARY_DEBUG}) + find_package(GTest QUIET REQUIRED) + endif() + + # On Windows shared libraries are installed in 'bin' instead of 'lib' directory. + if(${CMAKE_SYSTEM_NAME} MATCHES "Windows") + set(GTEST_SHARED_LIB_NAME "gtest.dll") + if(CMAKE_BUILD_TYPE MATCHES Debug) + set(GTEST_SHARED_LIB_NAME "gtestd.dll") + endif() + install(FILES "${GTEST_ROOT}/bin/${GTEST_SHARED_LIB_NAME}" DESTINATION "${CMAKE_INSTALL_PREFIX}/lib/gtest") + else() + install(FILES "${GTEST_LIBRARY}" DESTINATION "${CMAKE_INSTALL_PREFIX}/lib/gtest") + endif() + +endmacro() diff --git a/cmake/googletest_download.txt.in b/cmake/googletest_download.txt.in new file mode 100644 index 0000000000..f0370ec3cc --- /dev/null +++ b/cmake/googletest_download.txt.in @@ -0,0 +1,31 @@ +cmake_minimum_required(VERSION 3.12.0) + +project(googletest-download NONE) + +# Not specifying CONFIGURE_COMMAND, BUILD_COMMAND, or INSTALL_COMMAND means use +# CMake. Specifying these as empty strings means omit the step, which we don't +# want. As per +# https://github.com/abseil/googletest/blob/master/googletest/README.md +# need to force the use of the shared C run-time. + +include(ExternalProject) + +ExternalProject_Add(googletest + GIT_REPOSITORY https://github.com/google/googletest.git + GIT_TAG release-1.10.0 + GIT_CONFIG advice.detachedHead=false + SOURCE_DIR "${GOOGLETEST_BUILD_ROOT}/googletest-src" + BINARY_DIR "${GOOGLETEST_BUILD_ROOT}/googletest-build" + CMAKE_ARGS + "${MAYAUSD_EXTERNAL_PROJECT_GENERAL_SETTINGS}" + "-DCMAKE_INSTALL_PREFIX=${GOOGLETEST_BUILD_ROOT}/googletest-install" + "-Dgtest_force_shared_crt=ON" + "-DBUILD_GMOCK=OFF" + "-DBUILD_SHARED_LIBS=ON" + "-DCMAKE_MACOSX_RPATH=ON" + "-DCMAKE_POSITION_INDEPENDENT_CODE=ON" + "-DCMAKE_CXX_STANDARD=${CMAKE_CXX_STANDARD}" + "-DCMAKE_CXX_EXTENSIONS=${CMAKE_CXX_EXTENSIONS}" + "-DCMAKE_CXX_STANDARD_REQUIRED=${CMAKE_CXX_STANDARD_REQUIRED}" + "-DCMAKE_CXX_FLAGS=${CMAKE_CXX_FLAGS} ${disable_all_warnings_flag} ${glibcxx_abi}" +) diff --git a/cmake/googletest_src.txt.in b/cmake/googletest_src.txt.in new file mode 100644 index 0000000000..aff6d8e55b --- /dev/null +++ b/cmake/googletest_src.txt.in @@ -0,0 +1,28 @@ +cmake_minimum_required(VERSION 3.12.0) + +project(googletest-download NONE) + +# Not specifying CONFIGURE_COMMAND, BUILD_COMMAND, or INSTALL_COMMAND means use +# CMake. Specifying a XXX_COMMAND as empty string means omit the step. +# Need to force the use of the shared C run-time. + +include(ExternalProject) + +ExternalProject_Add(googletest + DOWNLOAD_COMMAND "" + UPDATE_COMMAND "" + SOURCE_DIR "${GOOGLETEST_SRC_DIR}" + BINARY_DIR "${GOOGLETEST_BUILD_ROOT}/googletest-build" + CMAKE_ARGS + "${MAYAUSD_EXTERNAL_PROJECT_GENERAL_SETTINGS}" + "-DCMAKE_INSTALL_PREFIX=${GOOGLETEST_BUILD_ROOT}/googletest-install" + "-Dgtest_force_shared_crt=ON" + "-DBUILD_GMOCK=OFF" + "-DBUILD_SHARED_LIBS=ON" + "-DCMAKE_MACOSX_RPATH=ON" + "-DCMAKE_POSITION_INDEPENDENT_CODE=ON" + "-DCMAKE_CXX_STANDARD=${CMAKE_CXX_STANDARD}" + "-DCMAKE_CXX_EXTENSIONS=${CMAKE_CXX_EXTENSIONS}" + "-DCMAKE_CXX_STANDARD_REQUIRED=${CMAKE_CXX_STANDARD_REQUIRED}" + "-DCMAKE_CXX_FLAGS=${CMAKE_CXX_FLAGS} ${disable_all_warnings_flag} ${glibcxx_abi}" +) diff --git a/cmake/gulrak.cmake b/cmake/gulrak.cmake new file mode 100644 index 0000000000..be9be69239 --- /dev/null +++ b/cmake/gulrak.cmake @@ -0,0 +1,48 @@ +# +# Copyright 2021 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +include(FetchContent) + +set(CONTENT_NAME gulrak) + +set(FETCHCONTENT_QUIET OFF) + +# GULRAK_SOURCE_DIR : Set this to the directory where you have cloned gulrak filesystem repo, +# if you would like to bypass pulling from Github repository via Internet. +if(DEFINED GULRAK_SOURCE_DIR) + file(TO_CMAKE_PATH "${GULRAK_SOURCE_DIR}" GULRAK_SOURCE_DIR) + message(STATUS "**** Building Gulrak From " ${GULRAK_SOURCE_DIR}) + FetchContent_Declare( + ${CONTENT_NAME} + URL ${GULRAK_SOURCE_DIR} + ) + + string(TOUPPER ${CONTENT_NAME} UPPERGULARK) + mark_as_advanced(FETCHCONTENT_SOURCE_DIR_${UPPERGULARK}) + mark_as_advanced(FETCHCONTENT_UPDATES_DISCONNECTED_${UPPERGULARK}) + +else() + message(STATUS "**** Building Gulrak From Github Repository.") + FetchContent_Declare( + ${CONTENT_NAME} + GIT_REPOSITORY https://github.com/gulrak/filesystem.git + GIT_TAG 4e21ab305794f5309a1454b4ae82ab9a0f5e0d25 + USES_TERMINAL_DOWNLOAD TRUE + GIT_CONFIG advice.detachedHead=false + ) +endif() + +FetchContent_MakeAvailable(${CONTENT_NAME}) diff --git a/cmake/jinja.cmake b/cmake/jinja.cmake new file mode 100644 index 0000000000..4aad9d6084 --- /dev/null +++ b/cmake/jinja.cmake @@ -0,0 +1,56 @@ +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +#------------------------------------------------------------------------------ +# +# Gets the Jinja2 and the dependant MarkupSafe python libraries from +# artifactory and set them up. +# +function(init_markupsafe) + + mayaUsd_find_python_module(markupsafe) + + if (NOT MARKUPSAFE_FOUND) + if (NOT MARKUPSAFE_LOCATION) + message(FATAL_ERROR "MARKUPSAFE_LOCATION not set") + endif() + + set(MARKUPSAFE_ROOT "${MARKUPSAFE_LOCATION}/src") + + # Add MarkupSafe to the python path so that Jinja2 can run properly. + mayaUsd_append_path_to_env_var("PYTHONPATH" "${MARKUPSAFE_ROOT}") + endif() + +endfunction() + +function(init_jinja) + + mayaUsd_find_python_module(jinja2) + + if (NOT JINJA2_FOUND) + if (NOT JINJA_LOCATION) + message(FATAL_ERROR "JINJA_LOCATION not set") + endif() + + set(JINJA_ROOT "${JINJA_LOCATION}/src") + + # Add Jinja2 to the python path so that usdGenSchemas can run properly. + mayaUsd_append_path_to_env_var("PYTHONPATH" "${JINJA_ROOT}") + endif() + +endfunction() + +init_markupsafe() +init_jinja() diff --git a/cmake/mayahydra_version.info b/cmake/mayahydra_version.info new file mode 100644 index 0000000000..ce918eb514 --- /dev/null +++ b/cmake/mayahydra_version.info @@ -0,0 +1,3 @@ +set(MAYAHYDRA_MAJOR_VERSION 0) +set(MAYAHYDRA_MINOR_VERSION 5) +set(MAYAHYDRA_PATCH_LEVEL 0) diff --git a/cmake/modules/FindMaya.cmake b/cmake/modules/FindMaya.cmake new file mode 100644 index 0000000000..ffb03d058b --- /dev/null +++ b/cmake/modules/FindMaya.cmake @@ -0,0 +1,495 @@ +# - Maya finder module +# This module searches for a valid Maya instalation. +# It searches for Maya's devkit, libraries, executables +# and related paths (scripts) +# +# Variables that will be defined: +# MAYA_FOUND Defined if a Maya installation has been detected +# MAYA_EXECUTABLE Path to Maya's executable +# MAYA__FOUND Defined if has been found +# MAYA__LIBRARY Path to library +# MAYA_INCLUDE_DIRS Path to the devkit's include directories +# MAYA_API_VERSION Maya version (6-8 digits) +# MAYA_APP_VERSION Maya app version (4 digits) +# MAYA_LIGHTAPI_VERSION Maya light API version (1 or 2 or 3) +# MAYA_PREVIEW_RELEASE_VERSION Preview Release number (3 or more digits) in preview releases, 0 in official releases +# +# Cache variables: +# MAYA_HAS_DEFAULT_MATERIAL_API Presence of a default material API on MRenderItem. +# MAYA_NEW_POINT_SNAPPING_SUPPORT Presence of point new snapping support. +# MAYA_HAS_CRASH_DETECTION Presence of isInCrashHandler API +# MAYA_ENABLE_NEW_PRIM_DELETE Enable new delete behaviour for delete command +# MAYA_HAS_DISPLAY_STYLE_ALL_VIEWPORTS Presence of MFrameContext::getDisplayStyleOfAllViewports. +# MAYA_ARRAY_ITERATOR_DIFFERENCE_TYPE_SUPPORT Presence of maya array iterator difference_type trait +# MAYA_HAS_GET_MEMBER_PATHS Presence of MFnSet::getMemberPaths +# MAYA_HAS_DISPLAY_LAYER_API Presence of MFnDisplayLayer +# MAYA_HAS_NEW_DISPLAY_LAYER_MESSAGING_API Presence of MDisplayLayerMemberChangedFunction +# MAYA_HAS_RENDER_ITEM_HIDE_ON_PLAYBACK_API Presence of MRenderItem has HideOnPlayback API +# MAYA_LINUX_BUILT_WITH_CXX11_ABI Maya Linux was built with new cxx11 ABI. +# MAYA_MACOSX_BUILT_WITH_UB2 Maya OSX was built with Universal Binary 2. + +#============================================================================= +# Copyright 2011-2012 Francisco Requena +# +# Distributed under the OSI-approved BSD License (the "License"); +# see accompanying file Copyright.txt for details. +# +# This software is distributed WITHOUT ANY WARRANTY; without even the +# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. +# See the License for more information. +#============================================================================= +# (To distribute this file outside of CMake, substitute the full +# License text for the above reference.) + +#============================================================================= +# Macro for setting up typical plugin properties. These include: +# - OS-specific plugin suffix (.mll, .so, .bundle) +# - Removal of 'lib' prefix on osx/linux +# - OS-specific defines +# - Post-commnad for correcting Qt library linking on osx +# - Windows link flags for exporting initializePlugin/uninitializePlugin +macro(maya_set_plugin_properties target) + set_target_properties(${target} PROPERTIES + SUFFIX ${MAYA_PLUGIN_SUFFIX}) + + set(_MAYA_DEFINES REQUIRE_IOSTREAM _BOOL) + + if(IS_MACOSX) + set(_MAYA_DEFINES "${_MAYA_DEFINES}" MAC_PLUGIN OSMac_ OSMac_MachO) + set_target_properties(${target} PROPERTIES + PREFIX "") + elseif(WIN32) + set(_MAYA_DEFINES "${_MAYA_DEFINES}" _AFXDLL _MBCS NT_PLUGIN) + set_target_properties( ${target} PROPERTIES + LINK_FLAGS "/export:initializePlugin /export:uninitializePlugin") + else() + set(_MAYA_DEFINES "${_MAYA_DEFINES}" LINUX LINUX_64) + set_target_properties( ${target} PROPERTIES + PREFIX "") + endif() + target_compile_definitions(${target} + PRIVATE + ${_MAYA_DEFINES} + ) +endmacro() +#============================================================================= + +if(IS_MACOSX) + set(MAYA_PLUGIN_SUFFIX ".bundle") +elseif(IS_WINDOWS) + set(MAYA_PLUGIN_SUFFIX ".mll") +else(IS_LINUX) + set(MAYA_PLUGIN_SUFFIX ".so") +endif() + +if(IS_MACOSX) + # On OSX, setting MAYA_LOCATION to either the base installation dir (ie, + # `/Application/Autodesk/maya20xx`), or the Contents folder in the Maya.app dir + # (ie, `/Application/Autodesk/maya20xx/Maya.app/Contents`) are supported. + find_path(MAYA_BASE_DIR + include/maya/MFn.h + HINTS + "${MAYA_LOCATION}/../.." + "$ENV{MAYA_LOCATION}/../.." + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "/Applications/Autodesk/maya2024" + DOC + "Maya installation root directory" + ) + find_path(MAYA_LIBRARY_DIR + libOpenMaya.dylib + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + MacOS/ + Maya.app/Contents/MacOS/ + DOC + "Maya's libraries path" + ) +elseif(IS_LINUX) + find_path(MAYA_BASE_DIR + include/maya/MFn.h + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "/usr/autodesk/maya2024-x64" + DOC + "Maya installation root directory" + ) + find_path(MAYA_LIBRARY_DIR + libOpenMaya.so + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + lib/ + DOC + "Maya's libraries path" + ) +elseif(IS_WINDOWS) + find_path(MAYA_BASE_DIR + include/maya/MFn.h + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "C:/Program Files/Autodesk/Maya2024" + DOC + "Maya installation root directory" + ) + find_path(MAYA_LIBRARY_DIR + OpenMaya.lib + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + lib/ + DOC + "Maya's libraries path" + ) +endif() + +find_path(MAYA_INCLUDE_DIR + maya/MFn.h + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + ../../devkit/include/ + include/ + DOC + "Maya's headers path" +) + +find_path(MAYA_LIBRARY_DIR + OpenMaya + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + ../../devkit/include/ + include/ + DOC + "Maya's libraries path" +) + +list(APPEND MAYA_INCLUDE_DIRS ${MAYA_INCLUDE_DIR}) + +find_path(MAYA_DEVKIT_INC_DIR + GL/glext.h + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + ../../devkit/plug-ins/ + DOC + "Maya's devkit headers path" +) +if(NOT "${MAYA_DEVKIT_INC_DIR}" STREQUAL "MAYA_DEVKIT_INC_DIR-NOTFOUND") + list(APPEND MAYA_INCLUDE_DIRS ${MAYA_DEVKIT_INC_DIR}) +endif() + +set(MAYA_LIBS_TO_FIND + OpenMaya + OpenMayaAnim + OpenMayaFX + OpenMayaRender + OpenMayaUI + Image + Foundation + IMFbase + cg + cgGL + clew +) +if (CMAKE_BUILD_TYPE MATCHES Debug) + list(APPEND MAYA_LIBS_TO_FIND tbb_debug) +else() + list(APPEND MAYA_LIBS_TO_FIND tbb) +endif() + +foreach(MAYA_LIB ${MAYA_LIBS_TO_FIND}) + find_library(MAYA_${MAYA_LIB}_LIBRARY + ${MAYA_LIB} + HINTS + "${MAYA_LIBRARY_DIR}" + DOC + "Maya's ${MAYA_LIB} library path" + # NO_CMAKE_SYSTEM_PATH needed to avoid conflicts between + # Maya's Foundation library and OSX's framework. + NO_CMAKE_SYSTEM_PATH + ) + + if (MAYA_${MAYA_LIB}_LIBRARY) + list(APPEND MAYA_LIBRARIES ${MAYA_${MAYA_LIB}_LIBRARY}) + endif() +endforeach() + +find_program(MAYA_EXECUTABLE + maya + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + Maya.app/Contents/bin/ + bin/ + DOC + "Maya's executable path" +) + +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MTypes.h") + # Tease the MAYA_API_VERSION numbers from the lib headers + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MTypes.h TMP REGEX "#define MAYA_API_VERSION.*$") + string(REGEX MATCHALL "[0-9]+" MAYA_API_VERSION ${TMP}) + + # MAYA_APP_VERSION + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MTypes.h MAYA_APP_VERSION REGEX "#define MAYA_APP_VERSION.*$") + if(MAYA_APP_VERSION) + string(REGEX MATCHALL "[0-9]+" MAYA_APP_VERSION ${MAYA_APP_VERSION}) + else() + string(SUBSTRING ${MAYA_API_VERSION} "0" "4" MAYA_APP_VERSION) + endif() +endif() + +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MDefines.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MDefines.h MAYA_PREVIEW_RELEASE_VERSION REGEX "#define MAYA_PREVIEW_RELEASE_VERSION.*$") + if(MAYA_PREVIEW_RELEASE_VERSION) + string(REGEX MATCHALL "[0-9]+" MAYA_PREVIEW_RELEASE_VERSION ${MAYA_PREVIEW_RELEASE_VERSION}) + else() + set(MAYA_PREVIEW_RELEASE_VERSION 0) + endif() +endif() + +# Determine the Python version and switch between mayapy and mayapy2. +set(MAYAPY_EXE mayapy) +set(MAYA_PY_VERSION 2) +if(${MAYA_APP_VERSION} STRGREATER_EQUAL "2021") + set(MAYA_PY_VERSION 3) + + # check to see if we have a mayapy2 executable + find_program(MAYA_PY_EXECUTABLE2 + mayapy2 + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + Maya.app/Contents/bin/ + bin/ + DOC + "Maya's Python executable path" + ) + if(NOT BUILD_WITH_PYTHON_3 AND MAYA_PY_EXECUTABLE2) + set(MAYAPY_EXE mayapy2) + set(MAYA_PY_VERSION 2) + endif() +endif() + +find_program(MAYA_PY_EXECUTABLE + ${MAYAPY_EXE} + HINTS + "${MAYA_LOCATION}" + "$ENV{MAYA_LOCATION}" + "${MAYA_BASE_DIR}" + PATH_SUFFIXES + Maya.app/Contents/bin/ + bin/ + DOC + "Maya's Python executable path" +) + +set(MAYA_LIGHTAPI_VERSION 1) +if(IS_MACOSX) + set(MAYA_DSO_SUFFIX ".dylib") + set(MAYA_DSO_PREFIX "lib") +elseif(IS_WINDOWS) + set(MAYA_DSO_SUFFIX ".dll") + set(MAYA_DSO_PREFIX "") +else(IS_LINUX) + set(MAYA_DSO_SUFFIX ".so") + set(MAYA_DSO_PREFIX "lib") +endif() +find_file(MAYA_OGSDEVICES_LIBRARY + "${MAYA_DSO_PREFIX}OGSDevices${MAYA_DSO_SUFFIX}" + HINTS + "${MAYA_LIBRARY_DIR}" + "${MAYA_LOCATION}" + PATH_SUFFIXES + lib/ + bin/ + DOC + "Maya's ${MAYA_LIB} library path" + # NO_CMAKE_SYSTEM_PATH needed to avoid conflicts between + # Maya's Foundation library and OSX's framework. + NO_CMAKE_SYSTEM_PATH +) +if (MAYA_OGSDEVICES_LIBRARY) + # Delaying the activation of Light API V2 until the shadow and SSAO issues are fixed. The + # update, to be found in a future PR, will contain this keyword, which is not present in 2022.1: + file(STRINGS ${MAYA_OGSDEVICES_LIBRARY} HAS_LIGHTAPI_2 REGEX "ConnectColorInFragments") + if (HAS_LIGHTAPI_2) + set(MAYA_LIGHTAPI_VERSION 2) + endif() + # In some future Maya updates, there might also be a function to get the ambient light, very + # useful to implement flat shading. + file(STRINGS ${MAYA_OGSDEVICES_LIBRARY} HAS_LIGHTAPI_3 REGEX "AddAmbientLight") + if (HAS_LIGHTAPI_3) + set(MAYA_LIGHTAPI_VERSION 3) + endif() +endif() +message(STATUS "Using Maya Light API Version ${MAYA_LIGHTAPI_VERSION}") + +set(MAYA_HAS_DEFAULT_MATERIAL_API FALSE CACHE INTERNAL "setDefaultMaterialHandling") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MHWGeometry.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MHWGeometry.h MAYA_HAS_API REGEX "setDefaultMaterialHandling") + if(MAYA_HAS_API) + set(MAYA_HAS_DEFAULT_MATERIAL_API TRUE CACHE INTERNAL "setDefaultMaterialHandling") + message(STATUS "Maya has setDefaultMaterialHandling API") + endif() +endif() + +set(MAYA_NEW_POINT_SNAPPING_SUPPORT FALSE CACHE INTERNAL "snapToActive") +if (MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MSelectionContext.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MSelectionContext.h MAYA_HAS_API REGEX "snapToActive") + if(MAYA_HAS_API) + set(MAYA_NEW_POINT_SNAPPING_SUPPORT TRUE CACHE INTERNAL "snapToActive") + message(STATUS "Maya has new point snapping API") + endif() +endif() + +set(MAYA_HAS_CRASH_DETECTION FALSE CACHE INTERNAL "isInCrashHandler") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MGlobal.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MGlobal.h MAYA_HAS_API REGEX "isInCrashHandler") + if(MAYA_HAS_API) + set(MAYA_HAS_CRASH_DETECTION TRUE CACHE INTERNAL "isInCrashHandler") + message(STATUS "Maya has isInCrashHandler API") + endif() +endif() + +set(MAYA_ENABLE_NEW_PRIM_DELETE FALSE CACHE INTERNAL "enableNewPrimDelete") +if (MAYA_API_VERSION VERSION_GREATER_EQUAL 20230000) + set(MAYA_ENABLE_NEW_PRIM_DELETE TRUE CACHE INTERNAL "enableNewPrimDelete") +endif() + +set(MAYA_HAS_DISPLAY_STYLE_ALL_VIEWPORTS FALSE CACHE INTERNAL "DisplayStyleOfAllViewports") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MFrameContext.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MFrameContext.h MAYA_HAS_API REGEX "getDisplayStyleOfAllViewports") + if(MAYA_HAS_API) + set(MAYA_HAS_DISPLAY_STYLE_ALL_VIEWPORTS TRUE CACHE INTERNAL "DisplayStyleOfAllViewports") + message(STATUS "Maya has getDisplayStyleOfAllViewports API") + endif() +endif() + +set(MAYA_ARRAY_ITERATOR_DIFFERENCE_TYPE_SUPPORT FALSE CACHE INTERNAL "hasArrayIteratorDifferenceType") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MArrayIteratorTemplate.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MArrayIteratorTemplate.h MAYA_HAS_API REGEX "difference_type") + if(MAYA_HAS_API) + set(MAYA_ARRAY_ITERATOR_DIFFERENCE_TYPE_SUPPORT TRUE CACHE INTERNAL "hasArrayIteratorDifferenceType") + message(STATUS "Maya array iterator has difference_type trait") + endif() +endif() + +set(MAYA_HAS_GET_MEMBER_PATHS FALSE CACHE INTERNAL "hasGetMemberPaths") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MFnSet.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MFnSet.h MAYA_HAS_API REGEX "getMemberPaths") + if(MAYA_HAS_API) + set(MAYA_HAS_GET_MEMBER_PATHS TRUE CACHE INTERNAL "hasGetMemberPaths") + message(STATUS "MFnSet has getMemberPaths function") + endif() +endif() + +set(MAYA_HAS_DISPLAY_LAYER_API FALSE CACHE INTERNAL "hasDisplayLayerAPI") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MFnDisplayLayer.h") + set(MAYA_HAS_DISPLAY_LAYER_API TRUE CACHE INTERNAL "hasDisplayLayerAPI") + message(STATUS "MFnDisplayLayer exists") +endif() + +set(MAYA_HAS_NEW_DISPLAY_LAYER_MESSAGING_API FALSE CACHE INTERNAL "hasDisplayLayerMemberChangedFunction") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MDisplayLayerMessage.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MDisplayLayerMessage.h MAYA_HAS_API REGEX "MDisplayLayerMemberChangedFunction") + if(MAYA_HAS_API) + set(MAYA_HAS_NEW_DISPLAY_LAYER_MESSAGING_API TRUE CACHE INTERNAL "hasDisplayLayerMemberChangedFunction") + message(STATUS "MDisplayLayerMessage has MDisplayLayerMemberChangedFunction") + endif() +endif() + +set(MAYA_HAS_RENDER_ITEM_HIDE_ON_PLAYBACK_API FALSE CACHE INTERNAL "hasRenderItemHideOnPlaybackFunction") +if(MAYA_INCLUDE_DIRS AND EXISTS "${MAYA_INCLUDE_DIR}/maya/MHWGeometry.h") + file(STRINGS ${MAYA_INCLUDE_DIR}/maya/MHWGeometry.h MAYA_HAS_API REGEX "isHideOnPlayback") + if(MAYA_HAS_API) + set(MAYA_HAS_RENDER_ITEM_HIDE_ON_PLAYBACK_API TRUE CACHE INTERNAL "hasRenderItemHideOnPlaybackFunction") + message(STATUS "MRenderItem has HideOnPlayback API") + endif() +endif() + +set(MAYA_LINUX_BUILT_WITH_CXX11_ABI FALSE CACHE INTERNAL "MayaLinuxBuiltWithCxx11ABI") +if(IS_LINUX AND MAYA_Foundation_LIBRARY) + # Determine if Maya (on Linux) was built using the new CXX11 ABI. + # If yes, then MayaUsd MUST also be built with new ABI. + execute_process( + COMMAND + nm "${MAYA_Foundation_LIBRARY}" + COMMAND + grep findVariableReplacement + COMMAND + grep " T " + WORKING_DIRECTORY + ${MAYA_LIBRARY_DIR} + OUTPUT_VARIABLE + maya_cxx11_abi) + if (NOT ("${maya_cxx11_abi}" STREQUAL "")) + string(FIND ${maya_cxx11_abi} "__cxx1112basic_string" maya_cxx11_abi_index) + if(NOT (${maya_cxx11_abi_index} STREQUAL "-1")) + set(MAYA_LINUX_BUILT_WITH_CXX11_ABI TRUE CACHE INTERNAL "MayaLinuxBuiltWithCxx11ABI") + message(STATUS "Linux: Maya was built with new cxx11 ABI") + endif() + endif() +endif() + +set(MAYA_MACOSX_BUILT_WITH_UB2 FALSE CACHE INTERNAL "MayaMacOSXBuiltWithUB2") +if(IS_MACOSX AND MAYA_Foundation_LIBRARY) + # Determine if Maya (on OSX) was built with Universal Binary 2 (x86_64 & arm64). + # If yes, then MayaUsd can be built with either: Intel, Arm or both. + execute_process( + COMMAND + lipo -archs "${MAYA_Foundation_LIBRARY}" + WORKING_DIRECTORY + ${MAYA_LIBRARY_DIR} + OUTPUT_VARIABLE + maya_lipo_output) + string(REGEX MATCHALL "(x86_64|arm64)" maya_ub2_match ${maya_lipo_output}) + if (maya_ub2_match) + list(FIND maya_ub2_match "x86_64" ub2_index1) + list(FIND maya_ub2_match "arm64" ub2_index2) + if((NOT (${ub2_index1} STREQUAL "-1")) AND (NOT (${ub2_index2} STREQUAL "-1"))) + set(MAYA_MACOSX_BUILT_WITH_UB2 TRUE CACHE INTERNAL "MayaMacOSXBuiltWithUB2") + message(STATUS "MacOSX: Maya was built with Universal Binary 2 (x86_64/arm64)") + endif() + endif() +endif() + +# handle the QUIETLY and REQUIRED arguments and set MAYA_FOUND to TRUE if +# all listed variables are TRUE +include(FindPackageHandleStandardArgs) + +find_package_handle_standard_args(Maya + REQUIRED_VARS + MAYA_EXECUTABLE + MAYA_PY_EXECUTABLE + MAYA_PY_VERSION + MAYA_INCLUDE_DIRS + MAYA_LIBRARIES + MAYA_API_VERSION + MAYA_APP_VERSION + MAYA_LIGHTAPI_VERSION + VERSION_VAR + MAYA_APP_VERSION +) diff --git a/cmake/modules/FindUFE.cmake b/cmake/modules/FindUFE.cmake new file mode 100644 index 0000000000..ddd1a89674 --- /dev/null +++ b/cmake/modules/FindUFE.cmake @@ -0,0 +1,122 @@ +# +# Simple module to find UFE. +# +# This module searches for a valid UFE installation. +# It searches for UFE's libraries and include header files. +# +# Variables that will be defined: +# UFE_FOUND Defined if a UFE installation has been detected +# UFE_LIBRARY Path to UFE library +# UFE_INCLUDE_DIR Path to the UFE include directory +# UFE_VERSION UFE version (major.minor.patch) from ufe.h +# UFE_LIGHTS_SUPPORT Presence of UFE lights support +# UFE_SCENE_SEGMENT_SUPPORT Presence of UFE scene segment support +# + +find_path(UFE_INCLUDE_DIR + ufe/versionInfo.h + HINTS + $ENV{UFE_INCLUDE_ROOT} + ${UFE_INCLUDE_ROOT} + ${MAYA_DEVKIT_LOCATION} + $ENV{MAYA_DEVKIT_LOCATION} + ${MAYA_LOCATION} + $ENV{MAYA_LOCATION} + ${MAYA_BASE_DIR} + PATH_SUFFIXES + devkit/ufe/include + include/ + DOC + "UFE header path" +) + +# Get the UFE_VERSION and features from ufe.h +if(UFE_INCLUDE_DIR AND EXISTS "${UFE_INCLUDE_DIR}/ufe/ufe.h") + # Parse the file and get the three lines that have the version info. + file(STRINGS + "${UFE_INCLUDE_DIR}/ufe/ufe.h" + _ufe_vers + REGEX "#define[ ]+(UFE_MAJOR_VERSION|UFE_MINOR_VERSION|UFE_PATCH_LEVEL)[ ]+[0-9]+$") + + # Then extract the number from each one. + foreach(_ufe_tmp ${_ufe_vers}) + if(_ufe_tmp MATCHES "#define[ ]+(UFE_MAJOR_VERSION|UFE_MINOR_VERSION|UFE_PATCH_LEVEL)[ ]+([0-9]+)$") + set(${CMAKE_MATCH_1} ${CMAKE_MATCH_2}) + endif() + endforeach() + set(UFE_VERSION ${UFE_MAJOR_VERSION}.${UFE_MINOR_VERSION}.${UFE_PATCH_LEVEL}) + + if("${UFE_MAJOR_VERSION}" STREQUAL "0") + math(EXPR UFE_PREVIEW_VERSION_NUM "${UFE_MINOR_VERSION} * 1000 + ${UFE_PATCH_LEVEL}") + endif() + + file(STRINGS + "${UFE_INCLUDE_DIR}/ufe/ufe.h" + _ufe_features + REGEX "#define UFE_V[0-9]+_FEATURES_AVAILABLE$") + foreach(_ufe_tmp ${_ufe_features}) + if(_ufe_tmp MATCHES "#define UFE_V([0-9]+)_FEATURES_AVAILABLE$") + set(CMAKE_UFE_V${CMAKE_MATCH_1}_FEATURES_AVAILABLE ON) + endif() + endforeach() +endif() + +find_library(UFE_LIBRARY + NAMES + ufe_${UFE_MAJOR_VERSION} + HINTS + $ENV{UFE_LIB_ROOT} + ${UFE_LIB_ROOT} + ${MAYA_DEVKIT_LOCATION} + $ENV{MAYA_DEVKIT_LOCATION} + ${MAYA_LOCATION} + $ENV{MAYA_LOCATION} + ${MAYA_BASE_DIR} + PATHS + ${UFE_LIBRARY_DIR} + PATH_SUFFIXES + devkit/ufe/lib + lib/ + DOC + "UFE library" + NO_DEFAULT_PATH +) + +# Handle the QUIETLY and REQUIRED arguments and set UFE_FOUND to TRUE if +# all listed variables are TRUE. +include(FindPackageHandleStandardArgs) + +find_package_handle_standard_args(UFE + REQUIRED_VARS + UFE_INCLUDE_DIR + UFE_LIBRARY + VERSION_VAR + UFE_VERSION +) + +if(UFE_FOUND) + message(STATUS "UFE include dir: ${UFE_INCLUDE_DIR}") + message(STATUS "UFE library: ${UFE_LIBRARY}") + message(STATUS "UFE version: ${UFE_VERSION}") +endif() + +set(UFE_LIGHTS_SUPPORT FALSE CACHE INTERNAL "ufeLights") +if (UFE_INCLUDE_DIR AND EXISTS "${UFE_INCLUDE_DIR}/ufe/lightHandler.h") + set(UFE_LIGHTS_SUPPORT TRUE CACHE INTERNAL "ufeLights") + message(STATUS "Maya has UFE lights API") +endif() + +set(UFE_SCENE_SEGMENT_SUPPORT FALSE CACHE INTERNAL "ufeSceneSegment") +if (UFE_INCLUDE_DIR AND EXISTS "${UFE_INCLUDE_DIR}/ufe/sceneSegmentHandler.h") + set(UFE_SCENE_SEGMENT_SUPPORT TRUE CACHE INTERNAL "ufeSceneSegment") + message(STATUS "Maya has UFE scene segment API") +endif() + +set(UFE_TRIE_NODE_HAS_CHILDREN_COMPONENTS_ACCESSOR FALSE CACHE INTERNAL "ufeTrieNodeHasChildrenComponentsAccessor") +if(UFE_INCLUDE_DIR AND EXISTS "${UFE_INCLUDE_DIR}/ufe/trie.h") + file(STRINGS ${UFE_INCLUDE_DIR}/ufe/trie.h UFE_HAS_API REGEX "childrenComponents") + if(UFE_HAS_API) + set(UFE_TRIE_NODE_HAS_CHILDREN_COMPONENTS_ACCESSOR TRUE CACHE INTERNAL "ufeTrieNodeHasChildrenComponentsAccessor") + message(STATUS "Maya has UFE TrieNode childrenComponents accessor") + endif() +endif() diff --git a/cmake/modules/FindUSD.cmake b/cmake/modules/FindUSD.cmake new file mode 100644 index 0000000000..955cc983c5 --- /dev/null +++ b/cmake/modules/FindUSD.cmake @@ -0,0 +1,180 @@ +# Simple module to find USD. + +# On a system with an existing USD /usr/local installation added to the system +# PATH, use of PATHS in find_path incorrectly causes the existing USD +# installation to be found. As per +# https://cmake.org/cmake/help/v3.4/command/find_path.html +# and +# https://cmake.org/pipermail/cmake/2010-October/040460.html +# HINTS get searched before system paths, which produces the desired result. +find_path(USD_INCLUDE_DIR + NAMES + pxr/pxr.h + HINTS + ${PXR_USD_LOCATION} + $ENV{PXR_USD_LOCATION} + PATH_SUFFIXES + include + DOC + "USD Include directory" +) + +find_file(USD_GENSCHEMA + NAMES + usdGenSchema + PATHS + ${PXR_USD_LOCATION} + $ENV{PXR_USD_LOCATION} + PATH_SUFFIXES + bin + DOC + "USD Gen schema application" +) + +find_file(USD_CONFIG_FILE + NAMES + pxrConfig.cmake + PATHS + ${PXR_USD_LOCATION} + $ENV{PXR_USD_LOCATION} + DOC "USD cmake configuration file" +) + +# PXR_USD_LOCATION might have come in as an environment variable, and +# it could also have been a hint-list, so we'll make sure we set it to +# wherever we found pxrConfig, which is always the correct location. +get_filename_component(PXR_USD_LOCATION "${USD_CONFIG_FILE}" DIRECTORY) + +include(${USD_CONFIG_FILE}) + +if(DEFINED PXR_VERSION) + # Starting in core USD 21.05, pxrConfig.cmake provides the various USD + # version numbers as CMake variables, in which case PXR_VERSION should have + # been defined, along with the major, minor, and patch version numbers, so + # there is no need to extract them from the pxr/pxr.h header file anymore. + # The only thing we need to do is assemble the USD_VERSION version string. + set(USD_VERSION ${PXR_MAJOR_VERSION}.${PXR_MINOR_VERSION}.${PXR_PATCH_VERSION}) +elseif(USD_INCLUDE_DIR AND EXISTS "${USD_INCLUDE_DIR}/pxr/pxr.h") + foreach(_usd_comp MAJOR MINOR PATCH) + file(STRINGS + "${USD_INCLUDE_DIR}/pxr/pxr.h" + _usd_tmp + REGEX "#define PXR_${_usd_comp}_VERSION .*$") + string(REGEX MATCHALL "[0-9]+" USD_${_usd_comp}_VERSION ${_usd_tmp}) + endforeach() + set(USD_VERSION ${USD_MAJOR_VERSION}.${USD_MINOR_VERSION}.${USD_PATCH_VERSION}) + math(EXPR PXR_VERSION "${USD_MAJOR_VERSION} * 10000 + ${USD_MINOR_VERSION} * 100 + ${USD_PATCH_VERSION}") +endif() + +# Note that on Windows with USD <= 0.19.11, USD_LIB_PREFIX should be left at +# default (or set to empty string), even if PXR_LIB_PREFIX was specified when +# building core USD, due to a bug. + +# On all other platforms / versions, it should match the PXR_LIB_PREFIX used +# for building USD (and shouldn't need to be touched if PXR_LIB_PREFIX was not +# used / left at it's default value). Starting with USD 21.11, the default +# value for PXR_LIB_PREFIX was changed to include "usd_". + +if (USD_VERSION VERSION_GREATER_EQUAL "0.21.11") + set(USD_LIB_PREFIX "${CMAKE_SHARED_LIBRARY_PREFIX}usd_" + CACHE STRING "Prefix of USD libraries; generally matches the PXR_LIB_PREFIX used when building core USD") +else() + set(USD_LIB_PREFIX ${CMAKE_SHARED_LIBRARY_PREFIX} + CACHE STRING "Prefix of USD libraries; generally matches the PXR_LIB_PREFIX used when building core USD") +endif() + +if (WIN32) + # ".lib" on Windows + set(USD_LIB_SUFFIX ${CMAKE_STATIC_LIBRARY_SUFFIX} + CACHE STRING "Extension of USD libraries") +else () + # ".so" on Linux, ".dylib" on MacOS + set(USD_LIB_SUFFIX ${CMAKE_SHARED_LIBRARY_SUFFIX} + CACHE STRING "Extension of USD libraries") +endif () + +find_library(USD_LIBRARY + NAMES + ${USD_LIB_PREFIX}usd${USD_LIB_SUFFIX} + HINTS + ${PXR_USD_LOCATION} + $ENV{PXR_USD_LOCATION} + PATH_SUFFIXES + lib + DOC + "Main USD library" +) + +get_filename_component(USD_LIBRARY_DIR ${USD_LIBRARY} DIRECTORY) + +# Get the boost version from the one built with USD +if(USD_INCLUDE_DIR) + file(GLOB _USD_VERSION_HPP_FILE "${USD_INCLUDE_DIR}/boost-*/boost/version.hpp") + list(LENGTH _USD_VERSION_HPP_FILE found_one) + if(${found_one} STREQUAL "1") + list(GET _USD_VERSION_HPP_FILE 0 USD_VERSION_HPP) + file(STRINGS + "${USD_VERSION_HPP}" + _usd_tmp + REGEX "#define BOOST_VERSION .*$") + string(REGEX MATCH "[0-9]+" USD_BOOST_VERSION ${_usd_tmp}) + unset(_usd_tmp) + unset(_USD_VERSION_HPP_FILE) + unset(USD_VERSION_HPP) + endif() +endif() + +# See if MaterialX shaders with color4 inputs exist natively in Sdr: +# Not yet in a tagged USD version: https://github.com/PixarAnimationStudios/USD/pull/1894 +set(USD_HAS_COLOR4_SDR_SUPPORT FALSE CACHE INTERNAL "USD.Sdr.PropertyTypes.Color4") +if (USD_INCLUDE_DIR AND EXISTS "${USD_INCLUDE_DIR}/pxr/usd/sdr/shaderProperty.h") + file(STRINGS ${USD_INCLUDE_DIR}/pxr/usd/sdr/shaderProperty.h USD_HAS_API REGEX "Color4") + if(USD_HAS_API) + set(USD_HAS_COLOR4_SDR_SUPPORT TRUE CACHE INTERNAL "USD.Sdr.PropertyTypes.Color4") + message(STATUS "USD has new Sdr.PropertyTypes.Color4") + endif() +endif() + +# See if MaterialX shaders have full Metadata imported: +# Not yet in a tagged USD version: https://github.com/PixarAnimationStudios/USD/pull/1895 +set(USD_HAS_MX_METADATA_SUPPORT FALSE CACHE INTERNAL "USD.MaterialX.Metadata") +if (USD_LIBRARY_DIR AND EXISTS "${USD_LIBRARY_DIR}/${USD_LIB_PREFIX}usdMtlx${CMAKE_SHARED_LIBRARY_SUFFIX}") + file(STRINGS ${USD_LIBRARY_DIR}/${USD_LIB_PREFIX}usdMtlx${CMAKE_SHARED_LIBRARY_SUFFIX} USD_HAS_API REGEX "uisoftmin") + if(USD_HAS_API) + set(USD_HAS_MX_METADATA_SUPPORT TRUE CACHE INTERNAL "USD.MaterialX.Metadata") + message(STATUS "USD has MaterialX metadata support") + endif() +endif() + +message(STATUS "USD include dir: ${USD_INCLUDE_DIR}") +message(STATUS "USD library dir: ${USD_LIBRARY_DIR}") +message(STATUS "USD version: ${USD_VERSION}") +if(DEFINED USD_BOOST_VERSION) + message(STATUS "USD Boost::boost version: ${USD_BOOST_VERSION}") +endif() + +include(FindPackageHandleStandardArgs) + +find_package_handle_standard_args(USD + REQUIRED_VARS + PXR_USD_LOCATION + USD_INCLUDE_DIR + USD_LIBRARY_DIR + USD_GENSCHEMA + USD_CONFIG_FILE + USD_VERSION + PXR_VERSION + VERSION_VAR + USD_VERSION +) + +find_program(OIIO_idiff_BINARY + idiff + HINTS + ${PXR_USD_LOCATION} + $ENV{PXR_USD_LOCATION} + PATH_SUFFIXES + bin/ + DOC + "OIIO's idiff binary" +) \ No newline at end of file diff --git a/cmake/python.cmake b/cmake/python.cmake new file mode 100644 index 0000000000..31342cde82 --- /dev/null +++ b/cmake/python.cmake @@ -0,0 +1,224 @@ +# - Find python libraries +# This module finds the libraries corresponding to the Python interpreter +# FindPython provides. +# This code sets the following variables: +# +# PYTHONLIBS_FOUND - have the Python libs been found +# PYTHON_PREFIX - path to the Python installation +# PYTHON_LIBRARIES - path to the python library +# PYTHON_INCLUDE_DIRS - path to where Python.h is found +# PYTHON_MODULE_EXTENSION - lib extension, e.g. '.so' or '.pyd' +# PYTHON_MODULE_PREFIX - lib name prefix: usually an empty string +# PYTHON_SITE_PACKAGES - path to installation site-packages +# PYTHON_IS_DEBUG - whether the Python interpreter is a debug build +# +# Thanks to talljimbo for the patch adding the 'LDVERSION' config +# variable usage. + +#============================================================================= +# Copyright 2001-2009 Kitware, Inc. +# Copyright 2012 Continuum Analytics, Inc. +# +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# +# * Redistributions in binary form must reproduce the above copyright +# notice, this list of conditions and the following disclaimer in the +# documentation and/or other materials provided with the distribution. +# +# * Neither the names of Kitware, Inc., the Insight Software Consortium, +# nor the names of their contributors may be used to endorse or promote +# products derived from this software without specific prior written +# permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +#============================================================================= + +# Checking for the extension makes sure that `LibsNew` was found and not just `Libs`. +if(PYTHONLIBS_FOUND AND PYTHON_MODULE_EXTENSION) + return() +endif() + +# On Mac, with Maya 2022+, the Python binaries link against a non-existent path +# maya2022/Maya.app/Contents/Frameworks/Python.framework/Versions/Current/bin/python -> /Library/Frameworks/Python.framework/Versions/3.7/Python +# However, just using the mayapy executable is good enough for the build system +if (${CMAKE_SYSTEM_NAME} MATCHES "Darwin") + MESSAGE(STATUS "Setting Python_EXECUTABLE to MayaPy ${MAYA_PY_EXECUTABLE}") + set(Python_EXECUTABLE ${MAYA_PY_EXECUTABLE}) +endif() + +# Use the Python module to find the python lib. +if(BUILD_WITH_PYTHON_3) + find_package(Python ${BUILD_WITH_PYTHON_3_VERSION} EXACT REQUIRED COMPONENTS Interpreter) +else() + find_package(Python 2.7 EXACT REQUIRED COMPONENTS Interpreter) +endif() + +if(NOT Python_Interpreter_FOUND) + set(PYTHONLIBS_FOUND FALSE) + return() +endif() + +# According to http://stackoverflow.com/questions/646518/python-how-to-detect-debug-interpreter +# testing whether sys has the gettotalrefcount function is a reliable, cross-platform +# way to detect a CPython debug interpreter. +# +# The library suffix is from the config var LDVERSION sometimes, otherwise +# VERSION. VERSION will typically be like "2.7" on unix, and "27" on windows. +execute_process(COMMAND "${Python_EXECUTABLE}" "-c" + "from distutils import sysconfig as s;import sys;import struct; +print('.'.join(str(v) for v in sys.version_info)); +print(sys.prefix); +print(s.get_python_inc(plat_specific=True)); +print(s.get_python_lib(plat_specific=True)); +print(s.get_config_var('SO')); +print(hasattr(sys, 'gettotalrefcount')+0); +print(struct.calcsize('@P')); +print(s.get_config_var('LDVERSION') or s.get_config_var('VERSION')); +print(s.get_config_var('LIBDIR') or ''); +print(s.get_config_var('MULTIARCH') or ''); +" + RESULT_VARIABLE _PYTHON_SUCCESS + OUTPUT_VARIABLE _PYTHON_VALUES + ERROR_VARIABLE _PYTHON_ERROR_VALUE) + +if(NOT _PYTHON_SUCCESS MATCHES 0) + if(PythonLibsNew_FIND_REQUIRED) + message(FATAL_ERROR + "Python config failure:\n${_PYTHON_ERROR_VALUE}") + endif() + set(PYTHONLIBS_FOUND FALSE) + return() +endif() + +# Convert the process output into a list +string(REGEX REPLACE ";" "\\\\;" _PYTHON_VALUES ${_PYTHON_VALUES}) +string(REGEX REPLACE "\n" ";" _PYTHON_VALUES ${_PYTHON_VALUES}) +list(GET _PYTHON_VALUES 0 _PYTHON_VERSION_LIST) +list(GET _PYTHON_VALUES 1 PYTHON_PREFIX) +list(GET _PYTHON_VALUES 2 PYTHON_INCLUDE_DIR) +list(GET _PYTHON_VALUES 3 PYTHON_SITE_PACKAGES) +list(GET _PYTHON_VALUES 4 PYTHON_MODULE_EXTENSION) +list(GET _PYTHON_VALUES 5 PYTHON_IS_DEBUG) +list(GET _PYTHON_VALUES 6 PYTHON_SIZEOF_VOID_P) +list(GET _PYTHON_VALUES 7 PYTHON_LIBRARY_SUFFIX) +list(GET _PYTHON_VALUES 8 PYTHON_LIBDIR) +list(GET _PYTHON_VALUES 9 PYTHON_MULTIARCH) + +message(STATUS "_PYTHON_VERSION_LIST: ${_PYTHON_VERSION_LIST}") +message(STATUS "PYTHON_PREFIX: ${PYTHON_PREFIX}") +message(STATUS "PYTHON_INCLUDE_DIR: ${PYTHON_INCLUDE_DIR}") +message(STATUS "PYTHON_SITE_PACKAGES: ${PYTHON_SITE_PACKAGES}") +message(STATUS "PYTHON_MODULE_EXTENSION: ${PYTHON_MODULE_EXTENSION}") +message(STATUS "PYTHON_IS_DEBUG: ${PYTHON_IS_DEBUG}") +message(STATUS "PYTHON_SIZEOF_VOID_P: ${PYTHON_SIZEOF_VOID_P}") +message(STATUS "PYTHON_LIBRARY_SUFFIX: ${PYTHON_LIBRARY_SUFFIX}") +message(STATUS "PYTHON_MULTIARCH: ${PYTHON_MULTIARCH}") + +# Python, especially on the mac, doesn't always return the right PYTHON_LIBDIR, +# so in the case that libdir is not able to be found, use a relative path from the include_dir +IF(NOT EXISTS ${PYTHON_LIBDIR}) + get_filename_component(_PYTHON_LIBDIR ${PYTHON_INCLUDE_DIR} PATH) + get_filename_component(_PYTHON_LIBDIR ${_PYTHON_LIBDIR} PATH) + set(PYTHON_LIBDIR "${_PYTHON_LIBDIR}/lib") +endif() +message(STATUS "PYTHON_LIBDIR: ${PYTHON_LIBDIR}") + +# Make sure the Python has the same pointer-size as the chosen compiler +# Skip if CMAKE_SIZEOF_VOID_P is not defined +if(CMAKE_SIZEOF_VOID_P AND (NOT "${PYTHON_SIZEOF_VOID_P}" STREQUAL "${CMAKE_SIZEOF_VOID_P}")) + if(PythonLibsNew_FIND_REQUIRED) + math(EXPR _PYTHON_BITS "${PYTHON_SIZEOF_VOID_P} * 8") + math(EXPR _CMAKE_BITS "${CMAKE_SIZEOF_VOID_P} * 8") + message(FATAL_ERROR + "Python config failure: Python is ${_PYTHON_BITS}-bit, " + "chosen compiler is ${_CMAKE_BITS}-bit") + endif() + set(PYTHONLIBS_FOUND FALSE) + return() +endif() + +# The built-in FindPython didn't always give the version numbers +string(REGEX REPLACE "\\." ";" _PYTHON_VERSION_LIST ${_PYTHON_VERSION_LIST}) +list(GET _PYTHON_VERSION_LIST 0 PYTHON_VERSION_MAJOR) +list(GET _PYTHON_VERSION_LIST 1 PYTHON_VERSION_MINOR) +list(GET _PYTHON_VERSION_LIST 2 PYTHON_VERSION_PATCH) + +# Make sure all directory separators are '/' +string(REGEX REPLACE "\\\\" "/" PYTHON_PREFIX ${PYTHON_PREFIX}) +string(REGEX REPLACE "\\\\" "/" PYTHON_INCLUDE_DIR ${PYTHON_INCLUDE_DIR}) +string(REGEX REPLACE "\\\\" "/" PYTHON_SITE_PACKAGES ${PYTHON_SITE_PACKAGES}) + +if(CMAKE_HOST_WIN32) + set(PYTHON_LIBRARY + "${PYTHON_PREFIX}/libs/Python${PYTHON_LIBRARY_SUFFIX}.lib") + + # when run in a venv, PYTHON_PREFIX points to it. But the libraries remain in the + # original python installation. They may be found relative to PYTHON_INCLUDE_DIR. + if(NOT EXISTS "${PYTHON_LIBRARY}") + get_filename_component(_PYTHON_ROOT ${PYTHON_INCLUDE_DIR} DIRECTORY) + set(PYTHON_LIBRARY + "${_PYTHON_ROOT}/libs/Python${PYTHON_LIBRARY_SUFFIX}.lib") + endif() + + # raise an error if the python libs are still not found. + if(NOT EXISTS "${PYTHON_LIBRARY}") + message(FATAL_ERROR "Python libraries not found") + endif() + +else() + if(PYTHON_MULTIARCH) + set(_PYTHON_LIBS_SEARCH "${PYTHON_LIBDIR}/${PYTHON_MULTIARCH}" "${PYTHON_LIBDIR}") + else() + set(_PYTHON_LIBS_SEARCH "${PYTHON_LIBDIR}") + endif() + #message(STATUS "Searching for Python libs in ${_PYTHON_LIBS_SEARCH}") + # Probably this needs to be more involved. It would be nice if the config + # information the python interpreter itself gave us were more complete. + find_library(PYTHON_LIBRARY + NAMES "python${PYTHON_LIBRARY_SUFFIX}" + PATHS ${_PYTHON_LIBS_SEARCH} + NO_DEFAULT_PATH) + + # If all else fails, just set the name/version and let the linker figure out the path. + if(NOT PYTHON_LIBRARY) + set(PYTHON_LIBRARY python${PYTHON_LIBRARY_SUFFIX}) + endif() +endif() + +MARK_AS_ADVANCED( + PYTHON_LIBRARY + PYTHON_INCLUDE_DIR +) + +# We use PYTHON_INCLUDE_DIR, PYTHON_LIBRARY and PYTHON_DEBUG_LIBRARY for the +# cache entries because they are meant to specify the location of a single +# library. We now set the variables listed by the documentation for this +# module. +SET(PYTHON_INCLUDE_DIRS "${PYTHON_INCLUDE_DIR}") +SET(PYTHON_LIBRARIES "${PYTHON_LIBRARY}") + +SET(PYTHON_DEBUG_LIBRARIES "${PYTHON_DEBUG_LIBRARY}") + + +find_package_message(PYTHON + "Found PythonLibs: ${PYTHON_LIBRARY}" + "${Python_EXECUTABLE}${Python_VERSION}") + +set(PYTHONLIBS_FOUND TRUE) diff --git a/cmake/test.cmake b/cmake/test.cmake new file mode 100644 index 0000000000..d9eed2cb85 --- /dev/null +++ b/cmake/test.cmake @@ -0,0 +1,404 @@ +set(MAYA_USD_DIR ${CMAKE_CURRENT_SOURCE_DIR}) + +function(mayaUsd_get_unittest_target unittest_target unittest_basename) + get_filename_component(unittest_name ${unittest_basename} NAME_WE) + set(${unittest_target} "${unittest_name}" PARENT_SCOPE) +endfunction() + +if (OIIO_idiff_BINARY) + set(IMAGE_DIFF_TOOL ${OIIO_idiff_BINARY} CACHE STRING "Use idiff for image comparison") +endif() + +# +# mayaUsd_add_test( +# {PYTHON_MODULE | +# PYTHON_COMMAND | +# PYTHON_SCRIPT | +# COMMAND [ ...] } +# [NO_STANDALONE_INIT] +# [INTERACTIVE] +# [ENV = ...]) +# +# PYTHON_MODULE - Module to import and test with unittest.main. +# PYTHON_COMMAND - Python code to execute; should call sys.exit +# with an appropriate exitcode to indicate success +# or failure. +# PYTHON_SCRIPT - Python script file to execute; should exit with an +# appropriate exitcode to indicate success or failure. +# WORKING_DIRECTORY - Directory from which the test executable will be called. +# COMMAND - Command line to execute as a test +# NO_STANDALONE_INIT - Only allowable with PYTHON_MODULE or +# PYTHON_COMMAND. With those modes, this +# command will generally add some boilerplate code +# to ensure that maya is initialized and exits +# correctly. Use this option to NOT add that code. +# INTERACTIVE - Only allowable with PYTHON_SCRIPT. +# The test is run using an interactive (non-standalone) +# session of Maya, including the UI. +# Tests run in this way should finish by calling Maya's +# quit command and returning an exit code of 0 for +# success or 1 for failure: +# cmds.quit(abort=True, exitCode=exitCode) +# ENV - Set or append the indicated environment variables; +# Since mayaUsd_add_test internally makes changes to +# some environment variables, if a value is given +# for these variables, it is appended; all other +# variables are set exactly as given. The variables +# that mayaUsd_add_test manages (and will append) are: +# PATH +# PYTHONPATH +# MAYA_PLUG_IN_PATH +# MAYA_SCRIPT_PATH +# PXR_PLUGINPATH_NAME +# XBMLANGPATH +# LD_LIBRARY_PATH +# Note that the format of these name/value pairs should +# be the same as that used with +# `set_property(TEST test_name APPEND PROPERTY ENVIRONMENT ...)` +# That means that if the passed in env var is a "list", it +# must already be separated by platform-appropriate +# path-separators, escaped if needed - ie, ":" on +# Linux/MacOS, and "\;" on Windows. Use +# separate_argument_list before passing to this func +# if you start with a cmake-style list. +# +function(mayaUsd_add_test test_name) + # ----------------- + # 1) Arg processing + # ----------------- + + cmake_parse_arguments(PREFIX + "NO_STANDALONE_INIT;INTERACTIVE" # options + "PYTHON_MODULE;PYTHON_COMMAND;PYTHON_SCRIPT;WORKING_DIRECTORY" # one_value keywords + "COMMAND;ENV" # multi_value keywords + ${ARGN} + ) + + # check that they provided one and ONLY 1 of: + # PYTHON_MODULE / PYTHON_COMMAND / PYTHON_SCRIPT / COMMAND + set(NUM_EXCLUSIVE_ITEMS 0) + foreach(option_name PYTHON_MODULE PYTHON_COMMAND PYTHON_SCRIPT COMMAND) + if(PREFIX_${option_name}) + math(EXPR NUM_EXCLUSIVE_ITEMS "${NUM_EXCLUSIVE_ITEMS} + 1") + endif() + endforeach() + if(NOT NUM_EXCLUSIVE_ITEMS EQUAL 1) + message(FATAL_ERROR "mayaUsd_add_test: must be called with exactly " + "one of PYTHON_MODULE, PYTHON_COMMAND, PYTHON_SCRIPT, or COMMAND") + endif() + + if(PREFIX_NO_STANDALONE_INIT AND NOT (PREFIX_PYTHON_MODULE + OR PREFIX_PYTHON_COMMAND)) + message(FATAL_ERROR "mayaUsd_add_test: NO_STANDALONE_INIT may only be " + "used with PYTHON_MODULE or PYTHON_COMMAND") + endif() + + if(PREFIX_INTERACTIVE AND NOT PREFIX_PYTHON_SCRIPT) + message(FATAL_ERROR "mayaUsd_add_test: INTERACTIVE may only be " + "used with PYTHON_SCRIPT") + endif() + + # set the working_dir + if(PREFIX_WORKING_DIRECTORY) + set(WORKING_DIR ${PREFIX_WORKING_DIRECTORY}) + else() + set(WORKING_DIR ${CMAKE_CURRENT_SOURCE_DIR}) + endif() + + # -------------- + # 2) Create test + # -------------- + + set(PYTEST_CODE "") + if(PREFIX_PYTHON_MODULE) + set(MODULE_NAME "${PREFIX_PYTHON_MODULE}") + set(PYTEST_CODE " +import sys +from unittest import main +import ${MODULE_NAME} +main(module=${MODULE_NAME}) +") + elseif(PREFIX_PYTHON_COMMAND) + set(PYTEST_CODE "${PREFIX_PYTHON_COMMAND}") + elseif(PREFIX_PYTHON_SCRIPT) + if (PREFIX_INTERACTIVE) + if(WIN32) + set(QUOTE "'") + else() + set(QUOTE "\\\"") + endif() + set(MEL_PY_EXEC_COMMAND "python(\"\\n\ +import os\\n\ +import sys\\n\ +import time\\n\ +import traceback\\n\ +file = ${QUOTE}${PREFIX_PYTHON_SCRIPT}${QUOTE}\\n\ +if not os.path.isabs(file):\\n\ + file = os.path.join(${QUOTE}${CMAKE_CURRENT_SOURCE_DIR}${QUOTE}, file)\\n\ +openMode = ${QUOTE}rb${QUOTE}\\n\ +compileMode = ${QUOTE}exec${QUOTE}\\n\ +globals = {${QUOTE}__file__${QUOTE}: file, ${QUOTE}__name__${QUOTE}: ${QUOTE}__main__${QUOTE}}\\n\ +try:\\n\ + exec(compile(open(file, openMode).read(), file, compileMode), globals)\\n\ +except Exception:\\n\ + sys.__stderr__.write(traceback.format_exc() + os.linesep)\\n\ + sys.__stderr__.flush()\\n\ + sys.__stdout__.flush()\\n\ + # sleep to give the output streams time to finish flushing - otherwise,\\n\ + # os._exit quits so hard + fast, flush may not happen!\\n\ + time.sleep(.1)\\n\ + os._exit(1)\\n\ +\")") + set(COMMAND_CALL ${MAYA_EXECUTABLE} -c ${MEL_PY_EXEC_COMMAND}) + else() + set(SCRIPT ${CMAKE_BINARY_DIR}/test/Temporary/scripts/runner_${test_name}.py) + FILE(WRITE ${SCRIPT} "${PREFIX_PYTHON_SCRIPT}") + set(COMMAND_CALL ${MAYA_PY_EXECUTABLE} ${SCRIPT}) + endif() + else() + set(COMMAND_CALL ${PREFIX_COMMAND}) + endif() + + if(PYTEST_CODE) + if(NOT PREFIX_NO_STANDALONE_INIT) + # first, indent pycode + mayaUsd_indent(indented_PYTEST_CODE "${PYTEST_CODE}") + # then wrap in try/finally, and call maya.standalone.[un]initialize() + set(PYTEST_CODE " +import maya.standalone +maya.standalone.initialize(name='python') +try: +${indented_PYTEST_CODE} +finally: + maya.standalone.uninitialize() +" + ) + endif() + + set(SCRIPT ${CMAKE_BINARY_DIR}/test/Temporary/scripts/runner_${test_name}.py) + FILE(WRITE ${SCRIPT} "${PYTEST_CODE}") + set(COMMAND_CALL ${MAYA_PY_EXECUTABLE} ${SCRIPT}) + endif() + + add_test( + NAME "${test_name}" + WORKING_DIRECTORY ${WORKING_DIR} + COMMAND ${COMMAND_CALL} + ) + + # ----------------- + # 3) Set up environ + # ----------------- + + set(ALL_PATH_VARS + PYTHONPATH + MAYA_PLUG_IN_PATH + MAYA_SCRIPT_PATH + XBMLANGPATH + ${PXR_OVERRIDE_PLUGINPATH_NAME} + PXR_MTLX_STDLIB_SEARCH_PATHS + ) + + if(IS_WINDOWS) + # Put path at the front of the list of env vars. + list(INSERT ALL_PATH_VARS 0 + PATH + ) + else() + list(APPEND ALL_PATH_VARS + LD_LIBRARY_PATH + ) + endif() + + # Set initial empty values for all path vars + foreach(pathvar ${ALL_PATH_VARS}) + set(MAYAUSD_VARNAME_${pathvar}) + endforeach() + + if(IS_WINDOWS) + list(APPEND MAYAUSD_VARNAME_PATH "${CMAKE_INSTALL_PREFIX}/lib/gtest") + list(APPEND MAYAUSD_VARNAME_PATH "${MAYA_LOCATION}/bin") + endif() + + # NOTE - we prefix varnames with "MAYAUSD_VARNAME_" just to make collision + # with some existing var less likely + + # Emulate what the module files for mayaHydra and mayaUsdPlugin would do. + + # mayaHydra + list(APPEND MAYAUSD_VARNAME_PATH + "${CMAKE_INSTALL_PREFIX}/lib") + list(APPEND MAYAUSD_VARNAME_${PXR_OVERRIDE_PLUGINPATH_NAME} + "${CMAKE_INSTALL_PREFIX}/lib/usd") + list(APPEND MAYAUSD_VARNAME_MAYA_PLUG_IN_PATH + "${CMAKE_INSTALL_PREFIX}/lib/maya") + + # mayaUsdPlugin + if(DEFINED MAYAUSD_LOCATION) + list(APPEND MAYAUSD_VARNAME_PATH + "${MAYAUSD_LOCATION}/lib") + list(APPEND MAYAUSD_VARNAME_PYTHONPATH + "${MAYAUSD_LOCATION}/lib/scripts") + list(APPEND MAYAUSD_VARNAME_MAYA_SCRIPT_PATH + "${MAYAUSD_LOCATION}/lib/scripts") + if (IS_LINUX) + # On Linux the paths in XBMLANGPATH need a /%B at the end. + list(APPEND MAYAUSD_VARNAME_XBMLANGPATH + "${MAYAUSD_LOCATION}/lib/icons/%B") + else() + list(APPEND MAYAUSD_VARNAME_XBMLANGPATH + "${MAYAUSD_LOCATION}/lib/icons") + endif() + list(APPEND MAYAUSD_VARNAME_PYTHONPATH + "${MAYAUSD_LOCATION}/lib/python") + list(APPEND MAYAUSD_VARNAME_${PXR_OVERRIDE_PLUGINPATH_NAME} + "${MAYAUSD_LOCATION}/lib/usd") + list(APPEND MAYAUSD_VARNAME_MAYA_PLUG_IN_PATH + "${MAYAUSD_LOCATION}/plugin/adsk/plugin") + list(APPEND MAYAUSD_VARNAME_PYTHONPATH + "${MAYAUSD_LOCATION}/plugin/adsk/scripts") + list(APPEND MAYAUSD_VARNAME_MAYA_SCRIPT_PATH + "${MAYAUSD_LOCATION}/plugin/adsk/scripts") + list(APPEND MAYAUSD_VARNAME_PXR_MTLX_STDLIB_SEARCH_PATHS + "${PXR_USD_LOCATION}/libraries") + list(APPEND MAYAUSD_VARNAME_PXR_MTLX_STDLIB_SEARCH_PATHS + "${MAYAUSD_LOCATION}/libraries") + endif() + + + if(IS_WINDOWS AND DEFINED ENV{PYTHONHOME}) + # If the environment contains a PYTHONHOME, also set the path to + # that folder so that we can find the python DLLs. + list(APPEND MAYAUSD_VARNAME_PATH $ENV{PYTHONHOME}) + endif() + + # Adjust PYTHONPATH to include the path to our test utilities. + list(APPEND MAYAUSD_VARNAME_PYTHONPATH "${MAYA_USD_DIR}/test/testUtils") + + # Adjust PYTHONPATH to include the path to our test. + list(APPEND MAYAUSD_VARNAME_PYTHONPATH "${CMAKE_CURRENT_SOURCE_DIR}") + + # Adjust PATH and PYTHONPATH to include USD. + # These should come last (esp PYTHONPATH, in case another module is overriding + # with pkgutil) + if (DEFINED MAYAHYDRA_TO_USD_RELATIVE_PATH) + set(USD_INSTALL_LOCATION "${CMAKE_INSTALL_PREFIX}/${MAYAHYDRA_TO_USD_RELATIVE_PATH}") + else() + set(USD_INSTALL_LOCATION ${PXR_USD_LOCATION}) + endif() + # Inherit any existing PYTHONPATH, but keep it at the end. + list(APPEND MAYAUSD_VARNAME_PYTHONPATH + "${USD_INSTALL_LOCATION}/lib/python") + if(IS_WINDOWS) + list(APPEND MAYAUSD_VARNAME_PATH + "${USD_INSTALL_LOCATION}/bin") + list(APPEND MAYAUSD_VARNAME_PATH + "${USD_INSTALL_LOCATION}/lib") + endif() + + # NOTE: this should come after any setting of PATH/PYTHONPATH so + # that our entries will come first. + # Inherit any existing PATH/PYTHONPATH, but keep it at the end. + # This is needed (especially for PATH) because we will overwrite + # both with the values from our list and we need to keep any + # system entries. + list(APPEND MAYAUSD_VARNAME_PATH $ENV{PATH}) + list(APPEND MAYAUSD_VARNAME_PYTHONPATH $ENV{PYTHONPATH}) + + # convert the internally-processed envs from cmake list + foreach(pathvar ${ALL_PATH_VARS}) + separate_argument_list(MAYAUSD_VARNAME_${pathvar}) + endforeach() + + # prepend the passed-in ENV values - assume these are already + # separated + escaped + foreach(name_value_pair ${PREFIX_ENV}) + mayaUsd_split_head_tail("${name_value_pair}" "=" env_name env_value) + if(NOT env_name) + message(FATAL_ERROR "poorly formatted NAME=VALUE pair - name " + "missing: ${name_value_pair}") + endif() + + # now either prepend to existing list, or create new + if("${env_name}" IN_LIST ALL_PATH_VARS) + if(IS_WINDOWS) + set(MAYAUSD_VARNAME_${env_name} + "${env_value}\;${MAYAUSD_VARNAME_${env_name}}") + else() + set(MAYAUSD_VARNAME_${env_name} + "${env_value}:${MAYAUSD_VARNAME_${env_name}}") + endif() + else() + set("MAYAUSD_VARNAME_${env_name}" ${env_value}) + list(APPEND ALL_PATH_VARS "${env_name}") + endif() + endforeach() + + # Unset any MAYA_MODULE_PATH as we set all the individual variables + # so we don't want to conflict with a MayaUsd module. + set_property(TEST ${test_name} APPEND PROPERTY ENVIRONMENT "MAYA_MODULE_PATH=") + + # set all env vars + foreach(pathvar ${ALL_PATH_VARS}) + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "${pathvar}=${MAYAUSD_VARNAME_${pathvar}}") + endforeach() + + # Set a temporary folder path for the TMP,TEMP and MAYA_APP_DIR in which the + # maya profile will be created. + # Note: replace bad chars in test_name with _. + string(REGEX REPLACE "[:<>\|]" "_" SANITIZED_TEST_NAME ${test_name}) + set(MAYA_APP_TEMP_DIR "${CMAKE_BINARY_DIR}/test/Temporary/${SANITIZED_TEST_NAME}") + # Note: ${WORKING_DIR} can point to the source folder, so don't use it + # in any env var that will write files (such as MAYA_APP_DIR). + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "TMP=${MAYA_APP_TEMP_DIR}" + "TEMP=${MAYA_APP_TEMP_DIR}" + "MAYA_APP_DIR=${MAYA_APP_TEMP_DIR}") + file(MAKE_DIRECTORY ${MAYA_APP_TEMP_DIR}) + + # Set the Python major version in MAYA_PYTHON_VERSION. Maya 2020 and + # earlier that are Python 2 only will simply ignore it. + # without "MAYA_NO_STANDALONE_ATEXIT=1", standalone.uninitialize() will + # set exitcode to 0 + # MAYA_DISABLE_CIP=1 Avoid fatal crash on start-up. + # MAYA_DISABLE_CER=1 Customer Error Reporting. + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "MAYA_PYTHON_VERSION=${MAYA_PY_VERSION}" + "MAYA_NO_STANDALONE_ATEXIT=1" + "MAYA_DEBUG_ENABLE_CRASH_REPORTING=1" + "MAYA_DEBUG_NO_SAVE_ON_CRASH=1" + "MAYA_NO_MORE_ASSERT=1" + "MAYA_DISABLE_CIP=1" + "MAYA_DISABLE_CER=1") + + if(IS_MACOSX) + # Necessary for tests like DiffCore to find python + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "DYLD_LIBRARY_PATH=${MAYA_LOCATION}/MacOS:$ENV{DYLD_LIBRARY_PATH}") + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "DYLD_FRAMEWORK_PATH=${MAYA_LOCATION}/Maya.app/Contents/Frameworks") + endif() + + if (PREFIX_INTERACTIVE) + # Add the "interactive" label to all tests that launch the Maya UI. + # This allows bypassing them by using the --label-exclude/-LE option to + # ctest. This is useful when running tests in a headless configuration. + set_property(TEST "${test_name}" APPEND PROPERTY LABELS interactive) + + # When running via remote desktop this env var is needed for Maya + # to function correctly. Has no effect when not running remote. + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "MAYA_ALLOW_OPENGL_REMOTE_SESSION=1") + + # Don't want popup when color management fails. + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "MAYA_CM_DISABLE_ERROR_POPUPS=1") + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "MAYA_COLOR_MGT_NO_LOGGING=1") + + else() + set_property(TEST "${test_name}" APPEND PROPERTY ENVIRONMENT + "MAYA_IGNORE_DIALOGS=1") + endif() +endfunction() diff --git a/cmake/usd.cmake b/cmake/usd.cmake new file mode 100644 index 0000000000..4661081a26 --- /dev/null +++ b/cmake/usd.cmake @@ -0,0 +1,24 @@ +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +function(init_usd) + # Adjust PYTHONPATH, PATH + mayaUsd_append_path_to_env_var("PYTHONPATH" "${PXR_USD_LOCATION}/lib/python") + if(WIN32) + mayaUsd_append_path_to_env_var("PATH" "${PXR_USD_LOCATION}/bin;${PXR_USD_LOCATION}/lib") + endif() +endfunction() + +init_usd() diff --git a/cmake/utils.cmake b/cmake/utils.cmake new file mode 100644 index 0000000000..bb129c41c4 --- /dev/null +++ b/cmake/utils.cmake @@ -0,0 +1,311 @@ +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +include(CMakeParseArguments) + +# The name of the operating system for which CMake is to build +if (${CMAKE_SYSTEM_NAME} MATCHES "Windows") + set(IS_WINDOWS TRUE) +elseif (${CMAKE_SYSTEM_NAME} MATCHES "Linux") + set(IS_LINUX TRUE) +elseif (${CMAKE_SYSTEM_NAME} MATCHES "Darwin") + set(IS_MACOSX TRUE) +endif() + +# compiler type +if (CMAKE_COMPILER_IS_GNUCXX) + set(IS_GNU TRUE) +elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "AppleClang") + set(IS_CLANG TRUE) +elseif(MSVC) + set(IS_MSVC TRUE) +endif() + +# Appends a path to an environment variable. +# Note: if you want to append multiple paths either call this multiple +# times, or send in the paths with the proper platform separator. +# +# envVar The environment variable to modify +# pathToAppend The path to append +# +function(mayaUsd_append_path_to_env_var envVar pathToAppend) + file(TO_NATIVE_PATH "${pathToAppend}" nativePathToAppend) + if(DEFINED ENV{${envVar}}) + if(IS_WINDOWS) + set(NEWPATH "$ENV{${envVar}};${nativePathToAppend}") + else() + set(NEWPATH "$ENV{${envVar}}:${nativePathToAppend}") + endif() + set(ENV{${envVar}} "${NEWPATH}") + else() + set(ENV{${envVar}} "${nativePathToAppend}") + endif() + message("Updated ${envVar}: $ENV{${envVar}}") +endfunction() + +# Finds if a specific Python module is installed in the current Python. +# _FOUND will be set to indicate whether the module was found. +# +# module The python module to find +# +function(mayaUsd_find_python_module module) + string(TOUPPER ${module} module_upper) + set(MODULE_FOUND "${module_upper}_FOUND") + if(NOT ${MODULE_FOUND}) + # Check for Python Executable + if(NOT DEFINED Python_EXECUTABLE OR Python_EXECUTABLE STREQUAL "") + MESSAGE(FATAL_ERROR "Python_EXECUTABLE is not set") + endif() + + if(ARGC GREATER 1 AND ARGV1 STREQUAL "REQUIRED") + set(${module}_FIND_REQUIRED TRUE) + endif() + execute_process(COMMAND "${Python_EXECUTABLE}" "-c" + "from __future__ import print_function; import re, ${module}; print(re.compile('/__init__.py.*').sub('',${module}.__file__))" + RESULT_VARIABLE _${module}_status + OUTPUT_VARIABLE _${module}_location + ERROR_QUIET OUTPUT_STRIP_TRAILING_WHITESPACE) + if(NOT _${module}_status) + set(${MODULE_FOUND} ${_${module}_location} CACHE STRING + "Location of Python module ${module}") + endif(NOT _${module}_status) + endif() +endfunction() + +# Initialize a variable to accumulate an rpath. The origin is the +# RUNTIME DESTINATION of the target. If not absolute it's appended +# to CMAKE_INSTALL_PREFIX. +function(mayaUsd_init_rpath rpathRef origin) + if(NOT IS_ABSOLUTE ${origin}) + if(DEFINED INSTALL_DIR_SUFFIX) + set(origin "${CMAKE_INSTALL_PREFIX}/${INSTALL_DIR_SUFFIX}/${origin}") + else() + set(origin "${CMAKE_INSTALL_PREFIX}/${origin}") + endif() + endif() + # mayaUsd_add_rpath uses REALPATH, so we must make sure we always + # do so here too, to get the right relative path + get_filename_component(origin "${origin}" REALPATH) + set(${rpathRef} "${origin}" PARENT_SCOPE) +endfunction() + +# Add a relative target path to the rpath. If target is absolute compute +# and add a relative path from the origin to the target. +function(mayaUsd_add_rpath rpathRef target) + if(IS_ABSOLUTE "${target}") + # init_rpath calls get_filename_component([...] REALPATH), which does + # symlink resolution, so we must do the same, otherwise relative path + # determination below will fail. + get_filename_component(target "${target}" REALPATH) + # Make target relative to $ORIGIN (which is the first element in + # rpath when initialized with mayaUsd_init_rpath()). + list(GET ${rpathRef} 0 origin) + file(RELATIVE_PATH + target + "${origin}" + "${target}" + ) + if("x${target}" STREQUAL "x") + set(target ".") + endif() + endif() + file(TO_CMAKE_PATH "${target}" target) + set(NEW_RPATH "${${rpathRef}}") + list(APPEND NEW_RPATH "$ORIGIN/${target}") + set(${rpathRef} "${NEW_RPATH}" PARENT_SCOPE) +endfunction() + +function(mayaUsd_install_rpath rpathRef NAME) + # Get and remove the origin. + list(GET ${rpathRef} 0 origin) + set(RPATH ${${rpathRef}}) + list(REMOVE_AT RPATH 0) + + # Canonicalize and uniquify paths. + set(FINAL "") + foreach(path ${RPATH}) + # Replace $ORIGIN with @loader_path + if(IS_MACOSX) + if("${path}/" MATCHES "^[$]ORIGIN/") + # Replace with origin path. + string(REPLACE "$ORIGIN/" "@loader_path/" path "${path}/") + endif() + endif() + + # Strip trailing slashes. + string(REGEX REPLACE "/+$" "" path "${path}") + + # Ignore paths we already have. + if (NOT ";${FINAL};" MATCHES ";${path};") + list(APPEND FINAL "${path}") + endif() + endforeach() + + set_target_properties(${NAME} + PROPERTIES + INSTALL_RPATH_USE_LINK_PATH TRUE + INSTALL_RPATH "${FINAL}" + ) +endfunction() + +# +# mayaUsd_promoteHeaderList( +# [SUBDIR ] +# [FILES ] +# [BASEDIR ]) +# +# SUBDIR - sub-directory in which to promote files. +# FILES - list of files to promote. +# BASESDIR - base dirctory where promoted headers are installed into. +# if not defined, mayaUsd subdirectory is used by default. +# +# +function(mayaUsd_promoteHeaderList) + cmake_parse_arguments(PREFIX + "" + "SUBDIR;BASEDIR" # one_value keywords + "HEADERS" # multi_value keywords + ${ARGN} + ) + + if (PREFIX_HEADERS) + set(HEADERFILES ${PREFIX_HEADERS}) + else() + message(FATAL_ERROR "HEADERS keyword is not specified.") + endif() + + set(BASEDIR ${CMAKE_BINARY_DIR}/include) + if (PREFIX_BASEDIR) + set(BASEDIR ${BASEDIR}/${PREFIX_BASEDIR}) + else() + set(BASEDIR ${BASEDIR}/mayaUsd) + endif() + + if (PREFIX_SUBDIR) + set(BASEDIR ${BASEDIR}/${PREFIX_SUBDIR}) + endif() + + foreach(header ${HEADERFILES}) + set(SRCFILE ${CMAKE_CURRENT_SOURCE_DIR}/${header}) + set(DSTFILE ${BASEDIR}/${header}) + + set(CONTENT "#pragma once\n#include \"${SRCFILE}\"\n") + + if (NOT EXISTS ${DSTFILE}) + message(STATUS "promoting: " ${SRCFILE}) + file(WRITE ${DSTFILE} "${CONTENT}") + else() + file(READ ${DSTFILE} oldContent) + if (NOT "${CONTENT}" STREQUAL "${oldContent}") + message(STATUS "Promoting ${SRCFILE}") + file(WRITE ${DSTFILE} "${CONTENT}") + endif() + endif() + endforeach() +endfunction() + +function(mayaUsd_split_head_tail input_string split_string var_head var_tail) + string(FIND "${input_string}" "${split_string}" head_end) + if("${head_end}" EQUAL -1) + message(FATAL_ERROR "input_string '${input_string}' did not contain " + "split_string '${split_string}'") + endif() + + string(LENGTH "${split_string}" split_string_len) + math(EXPR tail_start "${head_end} + ${split_string_len}") + + string(SUBSTRING "${input_string}" 0 "${head_end}" "${var_head}") + string(SUBSTRING "${input_string}" "${tail_start}" -1 "${var_tail}") + set("${var_head}" "${${var_head}}" PARENT_SCOPE) + set("${var_tail}" "${${var_tail}}" PARENT_SCOPE) +endfunction() + +function(mayaUsd_indent outvar lines) + string(REPLACE "\n" "\n " lines "${lines}") + set("${outvar}" " ${lines}" PARENT_SCOPE) +endfunction() + +# parse list arguments into a new list separated by ";" or ":" +function(separate_argument_list listName) + if(IS_WINDOWS) + string(REPLACE ";" "\;" ${listName} "${${listName}}") + else(IS_LINUX OR IS_MACOSX) + string(REPLACE ";" ":" ${listName} "${${listName}}") + endif() + set(${listName} "${${listName}}" PARENT_SCOPE) +endfunction() + +# python extension module suffix +function(set_python_module_property target) + if(IS_WINDOWS) + set_target_properties(${target} + PROPERTIES + PREFIX "" + SUFFIX ".pyd" + ) + elseif(IS_LINUX OR IS_MACOSX) + set_target_properties(${target} + PROPERTIES + PREFIX "" + SUFFIX ".so" + ) + endif() +endfunction() + +# This fuction will populate "out_var" with the default values external +# projects are expected to need. +# It can take an optional argument that will replace the list separator +# in CMake values. For instance, if an external project uses a different +# list separator, the values in here must be changed to reflect this. +function(get_external_project_default_values out_var) + # Some of these variables might end up not being used by some projects + # Therefore avoid useless warnings in the log. + list(APPEND setting_list --no-warn-unused-cli) + + if(ARGN) + list(GET ARGN 0 custom_sep) + endif() + + # Macro to add the value only if it's present. + macro(external_project_conditional_define option) + if (${option}) + if(custom_sep) + # This will change the list separator to the desired one. + # i.e. -DCMAKE_OSX_ARCHITECTURES=x86_64;arm64 -> -DCMAKE_OSX_ARCHITECTURES=x86_64|arm64 + string(REPLACE ";" ${custom_sep} ${option} "${${option}}") + endif() + list(APPEND setting_list -D${option}=${${option}}) + endif() + endmacro(external_project_conditional_define) + + external_project_conditional_define(CMAKE_INSTALL_MESSAGE) + external_project_conditional_define(CMAKE_BUILD_TYPE) + external_project_conditional_define(CMAKE_MAKE_PROGRAM) + + if(BUILD_UB2) + # UB2 builds require this flag + external_project_conditional_define(CMAKE_OSX_ARCHITECTURES) + external_project_conditional_define(CMAKE_OSX_DEPLOYMENT_TARGET) + endif() + + # Debugging informations for external projects + external_project_conditional_define(CMAKE_VERBOSE_MAKEFILE) + external_project_conditional_define(CMAKE_FIND_DEBUG_MODE) + + set(${out_var} ${setting_list} PARENT_SCOPE) +endfunction(get_external_project_default_values) + +# Create one for all the project using the default list separator +get_external_project_default_values(MAYAUSD_EXTERNAL_PROJECT_GENERAL_SETTINGS "$") diff --git a/doc/CHANGELOG.md b/doc/CHANGELOG.md new file mode 100644 index 0000000000..89c312dcad --- /dev/null +++ b/doc/CHANGELOG.md @@ -0,0 +1,81 @@ +# Changelog + +## [v0.4.0] - 2023-08-30 + +**Build:** +- [HYDRA-453] - Update to USD 23.08 + +**Functionality:** +- [HYDRA-425] - Basic Hydra Scene Browser plugin available with mayaHydra +- [HYDRA-427] - Build up SceneIndex for Maya native scene +- [HYDRA-426] - Add new environment variable as switch for new SceneIndex approach +- [HYDRA-381] - Add support for input color space for DG nodes +- [HYDRA-447] - Default value for Display Purpose in Gloabal Settings +- [HYDRA-448] - Rename Global USD Display Purpose Tag in viewport settings +- [HYDRA-472] - Update Scene Indices chain code in maya-usd related to USD 23.08 integration + +**Fixes:** +- [HYDRA-419] - Domelight texture not dirtying with SceneIndex update +- [HYDRA-180] - aiSkyDomeLight default color does not work +- [HYDRA-182] - UsdStage rprim of sphere type is not rendered +- [HYDRA-289] - Light intensity shows big difference compared to usdview +- [HYDRA-343] - Selecting native Maya nodes in the viewport selects the shape, not the transform +- [HYDRA-350] - Crash when assigning materialX standard surface to prim +- [HYDRA-413] - Referenced maya file with usd stage doesn't render in mayaHydra (Ill-formed SdfPath) +- [HYDRA-421] - Stage Display purpose setting has no effect +- [HYDRA-423] - Hydra Generative Plugin is not loaded in Maya +- [HYDRA-436] - Update issue muting USD layers +- [HYDRA-483] - Fix Scene Browser compilation for USD 23.08 +- [HYDRA-493] - Lighting results differ between SceneIndex and SceneDelegate +- [HYDRA-494] - Shadows are not being cast on Maya native objects +- [HYDRA-320] - MaterialX shading/lighting issue + +## [v0.3.0] - 2023-07-06 + +**Build:** +- [HYDRA-322] - Compilation failures when including terminalsResolvingSceneIndex.h + +**Fixes:** +- [HYDRA-93] - Toggling between vp2 and Hydra stop updating Hydra +- [HYDRA-178] - USD instances are not shown in the mayaHydra viewport +- [HYDRA-184] - Wire frame and wire frame on shaded of USD objects not drawn +- [HYDRA-197] - Prevent scene delegate from being thrown out when we're switching back and forth between VP2 and hydra +- [HYDRA-287] - Hydra : OpenGL states not restored correctly +- [HYDRA-318] - Viewer doesn't update when working in USD stage +- [HYDRA-325] - Purpose defined prims don't render with mayaHydra +- [HYDRA-332] - Artifact with transparent surfaces in the viewport +- [HYDRA-342] - Rotation gizmo is missing its X/Y/Z axis +- [HYDRA-352] - Objects with namespaces will not draw in mayaHydra and switching back to VP2 will crash Maya +- [HYDRA-382] - Crash when going back to VP2 after loading PointMedcity.usd + +## [v0.2.0] - 2023-05-03 + +**Build:** +- [HYDRA-333] - Update mayaHydra to USD 23.02 +- [HYDRA-69] - Restructure directories in repository + +**Performance:** +- [HYDRA-339] - Hydra hit test performance optimization via render a smaller region + +**Miscellaneous:** +- [HYDRA-192] - remove UFE conditional compilation +- [HYDRA-321] - Added required ApiSchema for MaterialX bindings in USD +- [HYDRA-327] - Preserve existence of the previous selection temporarily inside of the Ufe Selection Notification (Clear, Replace) + +**Fixes:** +- [HYDRA-79] - Crash when changing to HydraViewport for a USD referenced object +- [HYDRA-101] - Crash when switching from VP2 to Hydra on MacOS +- [HYDRA-109] - Loading the Arnold plug-in after Hydra causes the viewport to not update properly +- [HYDRA-116] - Meshes are not displayed as expected when switching between viewport modes +- [HYDRA-128] - The latest module template doesn't work +- [HYDRA-179] - MaterialX does not work with usd stage scene index +- [HYDRA-200] - Ghost mesh after deleting everything in the scene +- [HYDRA-211] - Require clang-format by automation +- [HYDRA-239] - Support for Apple Silicon +- [HYDRA-250] - Code path where ria is a nullptr and not tested. +- [HYDRA-266] - Undo, redo does not work with custom scene indices +- [HYDRA-283] - Crash when switching back and forth from VP2 to Hydra GL + + +## [v0.1.0] - 2022-03-27 +- Initial release of Hydra for Maya (Technology Preview) \ No newline at end of file diff --git a/doc/CLA/Hydra for Maya - Corp Contrib Agmt.pdf b/doc/CLA/Hydra for Maya - Corp Contrib Agmt.pdf new file mode 100644 index 0000000000000000000000000000000000000000..7bbcda5f93729f8d7b5aa1b60f5fa35d9ab370c0 GIT binary patch literal 501588 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%0D{x)cJ+0mJ>*h0FbFIWTmRlN!> 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 ypAm8@ikev9xgPCESxf;rJK(i1^f+(9M@W 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 literal 0 HcmV?d00001 diff --git a/doc/CLA/Hydra for Maya - Ind Contrib Agmt.pdf b/doc/CLA/Hydra for Maya - Ind Contrib Agmt.pdf new file mode 100644 index 0000000000000000000000000000000000000000..e75c2e869f75005939d16ec1015a06b82095fd47 GIT binary patch literal 402932 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≫8l)1D^%NS{S9Y^Bomq{v+9{{A%l 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 P9R>^M6c<<0QRe)AT) please sign this one for corporate use ++ [Hydra for Maya - Ind Contrib Agmt.pdf](<./CLA/Hydra for Maya - Ind Contrib Agmt.pdf>) please sign this one if you're an individual contributor + +The documents include instructions on where to send the completed forms to. Once a signed form has been received you will be able to submit pull requests. + + +## Filing Issues + +### Suggestions + +The _Hydra for Maya_ project is meant to evolve with feedback - the project and its users greatly appreciate any thoughts on ways to improve the design or features. Please use the `enhancement` tag to specifically denote issues that are suggestions - this helps us triage and respond appropriately. + +### Bugs + +As with all pieces of software, you may end up running into bugs. Please submit bugs as regular issues on GitHub - Maya developers are regularly monitoring issues and will prioritize and schedule fixes. + +The best bug reports include a detailed way to predictably reproduce the issue, and possibly even a working example that demonstrates the issue. + +## Contributing Code + +The _Hydra for Maya_ project accepts and greatly appreciates contributions. The project follows the [fork & pull](https://help.github.com/articles/using-pull-requests/#fork--pull) model for accepting contributions. + +When contributing code, please also include appropriate tests as part of the pull request, and follow the same comment and coding style as the rest of the project. Take a look through the existing code for examples of the testing and style practices the project follows. + +All development should happen against the "develop" branch of the repository. Please make sure the base branch of your pull request is set to the "develop" branch when filing your pull request. + +It is highly recommended that an issue be logged on GitHub before any work is started. This will allow for early feedback from other developers and avoid multiple parallel efforts. diff --git a/doc/LICENSE.md b/doc/LICENSE.md new file mode 100644 index 0000000000..11069edd79 --- /dev/null +++ b/doc/LICENSE.md @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + +TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + +1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + +2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + +3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + +4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + +5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + +6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + +7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + +8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + +9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + +END OF TERMS AND CONDITIONS + +APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + +Copyright [yyyy] [name of copyright owner] + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/doc/build.md b/doc/build.md new file mode 100644 index 0000000000..ebea19b063 --- /dev/null +++ b/doc/build.md @@ -0,0 +1,216 @@ +# Building + +## Getting and Building the Code + +The simplest way to build the project is by running the supplied **build.py** script. This script builds the project and installs all of the necessary libraries and plug-ins for you. Follow the instructions below to learn how to use the script. + +#### 1. Tools and System Prerequisites + +Before building the project, consult the following table to ensure you use the recommended version of compiler, operating system, cmake, etc. + +| Required | ![](images/windows.png) | ![](images/mac.png) | ![](images/linux.png) | +|:---------------------:|:-------------------------:|:------------------------------------------------------------:|:---------------------------:| +| Operating System | Windows 10
Windows 11 | High Sierra (10.13)
Mojave (10.14)
Catalina (10.15)
Big Sur (11.2.x) | Rocky Linux 8.6 / Linux® Red Hat® Enterprise 8.6 WS | +| Compiler Requirement| Maya 2024 (VS 2022) | Maya 2024 (Xcode 13.4 or higher) | Maya 2024 (gcc 11.2.1) | +| CMake Version (min/max) | 3.13...3.17 | 3.13...3.17 | 3.13...3.17 | +| Python | 3.10.8 | 3.10.8 | 3.10.8 | +| Python Packages | PyYAML, PySide, PyOpenGL, Jinja2 | PyYAML, PySide2, PyOpenGL, Jinja2 | PyYAML, PySide, PyOpenGL, Jinja2 | +| Build generator | Visual Studio, Ninja (Recommended) | XCode, Ninja (Recommended) | Ninja (Recommended) | +| Command processor | Visual Studio X64 2019 command prompt | bash | bash | +| Supported Maya Version| 2024 | 2024 | 2024 | + +| Optional | ![](images/windows.png) | ![](images/mac.png) | ![](images/linux.png) | + +***NOTE:*** Visit the online Maya developer help document under ***Setting up your build environment*** for additional compiler requirements on different platforms. + +#### 2. Download and Build Pixar USD + +See Pixar's official github page for instructions on how to build USD: https://github.com/PixarAnimationStudios/USD. Pixar has recently removed support for building Maya USD libraries/plug-ins in their github repository and ```build_usd.py```. + +| | ![](images/pxr.png) | +|:------------: |:---------------: | +| CommitID/Tags | Recommended : [v23.08](https://github.com/PixarAnimationStudios/OpenUSD/releases/tag/v23.08) | +| CommitID/Tags | For older maya-hydra plugin: [v22.11](https://github.com/PixarAnimationStudios/USD/releases/tag/v22.11) | + +For additional information on building Pixar USD, see the ***Additional Build Instruction*** section below. + +***NOTE:*** Recommended version of USD for building the latest version of MayaHydra plugin is USD23.08 [v23.08](https://github.com/PixarAnimationStudios/OpenUSD/releases/tag/v23.08). If older version of USD needs to be used then maya-hydra v0.1.x is to be used for build and feature compatibility. + +***NOTE:*** Make sure that you don't have an older USD locations in your ```PATH``` and ```PYTHONPATH``` environment settings. ```PATH``` and ```PYTHONPATH``` are automatically adjusted inside the project to point to the correct USD location. See ```cmake/usd.cmake```. + +#### 3. Universal Front End (UFE) + +The Universal Front End (UFE) is a DCC-agnostic component that allows Maya to browse and edit data in multiple data models. This allows Maya to edit pipeline data such as USD. UFE comes installed as a built-in component with Maya 2019 and later. UFE is developed as a separate binary component, and therefore versioned separately from Maya. + +| Ufe Version | Maya Version | Ufe Docs (external) | +|----------------------------|--------------------------------------------------------|:-------------------:| +| v4.0.0 | Maya 2024 | https://help.autodesk.com/view/MAYAUL/2024/ENU/?guid=MAYA_API_REF_ufe_ref_index_html | +| v4.0.1 | Maya 2024.1 | | + +To build the project with UFE support, you will need to use the headers and libraries included in the ***Maya Devkit***: + +https://www.autodesk.com/developer-network/platform-technologies/maya + +#### 4. Download the source code + +Start by cloning the repository: +``` +git clone https://github.com/Autodesk/maya-hydra +cd maya-hydra +``` + +##### Repository Layout + +| Location | Description | +|------------- |--------------------------------------------------------------------------------------------- | +| lib/mayaHydra/mayaPlugin | Contains Maya plugin definition and render override registration | +| lib/mayaHydra/hydraExtensions | Contains extensions to and mechanism needed to interface with hydra classes | +| lib/mayaHydra/ufeExtensions | Contains extensions to translate paths between UFE, USD SdfPath and Maya DAGPath | + +#### 5. How To Use build.py Script + +##### Arguments + +There are four arguments that must be passed to the script: + +| Flags | Description | +|-------------------- |-------------------------------------------------------------------------------------- | +| --maya-location | directory where Maya is installed. | +| --pxrusd-location | directory where Pixar USD Core is installed. | +| --devkit-location | directory where Maya devkit is installed. | +| workspace_location | directory where the project use as a workspace to build and install plugin/libraries | + +``` +Linux: +➜ maya-hydra python build.py --maya-location /usr/autodesk/maya2024 --pxrusd-location /usr/local/USD-Release --devkit-location /usr/local/devkitBase /usr/local/workspace + +MacOSX: +➜ maya-hydra python build.py --maya-location /Applications/Autodesk/maya2024 --pxrusd-location /opt/local/USD-Release --devkit-location /opt/local/devkitBase /opt/local/workspace + +Windows: +c:\maya-hydra> python build.py --maya-location "C:\Program Files\Autodesk\Maya2024" --pxrusd-location C:\USD-Release --devkit-location C:\devkitBase C:\workspace +``` + +##### Build Arguments + +| Flag | Description | +|-------------------- |---------------------------------------------------------------------------------------| +| --build-args | comma-separated list of cmake variables can be also passed to build system. | + +``` +--build-args="-DBUILD_TESTS=OFF" +``` + +##### CMake Options + +Name | Description | Default +--- | --- | --- +BUILD_TESTS | builds all unit tests. | ON +BUILD_STRICT_MODE | enforces all warnings as errors. | ON +BUILD_SHARED_LIBS | build libraries as shared or static. | ON + +##### Stages + +| Flag | Description | +|-------------------- |--------------------------------------------------------------------------------------------------- | +| --stages | comma-separated list of stages can also be passed to the build system. By default "clean, configure, build, install" stages are executed if this argument is not set. | + +| Options | Description | +|----------- |--------------------------------------------------- | +| clean | clean build | +| configure | call this stage every time a cmake file is modified | +| build | builds the project | +| install | installs all the necessary plug-ins and libraries | +| test | runs all unit tests | +| package | bundles up all the installation files as a zip file inside the package directory | + +``` +Examples: +--stages=configure,build,install +--stages=test +``` +***NOTE:*** All the flags can be followed by either ```space``` or ```=``` + +##### CMake Generator + +It is up to the user to select the CMake Generator of choice, but we encourage the use of the Ninja generator. To use the Ninja Generator, you need to first install the Ninja binary from https://ninja-build.org/ + +You then need to set the generator to ```Ninja``` and the ```CMAKE_MAKE_PROGRAM``` variable to the Ninja binary you downloaded. +``` +python build.py --generator Ninja --build-args=-DCMAKE_MAKE_PROGRAM='path to ninja binary' +``` +##### Build and Install locations + +By default, the build and install directories are created inside the **workspace** directory. However, you can change these locations by setting the ```--build-location``` and ```--install-location``` flags. + +##### Build Log + +By default the build log is written into ```build_log.txt``` inside the build directory. If you want to redirect the output stream to the console instead, +you can pass ```--redirect-outstream-file``` and set it to false. + +##### Additional flags and options + +Run the script with the ```--help``` parameter to see all the possible flags and short descriptions. + +#### 6. How To Run Unit Tests + +Unit tests can be run by setting ```--stages=test``` or by simply calling `ctest` directly from the build directory. + +# Additional Build Instruction + +##### Python: + +It is important to use the Python version shipped with Maya and not the system version when building USD on MacOS. Note that this is primarily an issue on MacOS, where Maya's version of Python is likely to conflict with the version provided by the system. + +To build USD and the Maya plug-ins on MacOS for Maya (2024), run: +``` +/Applications/Autodesk/maya2024/Maya.app/Contents/bin/mayapy build_usd.py ~/Desktop/BUILD +``` +By default, ``usdview`` is built which has a dependency on PyOpenGL. Since the Python version of Maya doesn't ship with PyOpenGL you will be prompted with the following error message: +``` +PyOpenGL is not installed. If you have pip installed, run "pip install PyOpenGL" to install it, then re-run this script. +If PyOpenGL is already installed, you may need to update your ```PYTHONPATH``` to indicate where it is located. +``` +The easiest way to bypass this error is by setting ```PYTHONPATH``` to point at your system python or third-party python package manager that has PyOpenGL already installed. +e.g +``` +export PYTHONPATH=$PYTHONPATH:Library/Frameworks/Python.framework/Versions/3.10/lib/python3.10/site-packages +``` +Use `pip list` to see the list of installed packages with your python's package manager. + +e.g +``` +➜ pip list +Package Version +---------- -------- +Jinja2 3.1.2 +MarkupSafe 2.1.1 +pip 22.2.1 +PyOpenGL 3.1.6 +PySide2 5.15.2.1 +PyYAML 6.0 +setuptools 63.2.0 +shiboken2 5.15.2.1 +``` + +##### Dependencies on Linux DSOs when running tests + +Normally either runpath or rpath are used on some DSOs in this library to specify explicit on other libraries (such as USD itself) + +If for some reason you don't want to use either of these options, and switch them off with: +``` +CMAKE_SKIP_RPATH=TRUE +``` +To allow your tests to run, you can inject LD_LIBRARY_PATH into any of the mayaHydra_add_test calls by setting the ADDITIONAL_LD_LIBRARY_PATH cmake variable to $ENV{LD_LIBRARY_PATH} or similar. + +There is a related ADDITIONAL_PXR_PLUGINPATH_NAME cmake var which can be used if schemas are installed in a non-standard location + +# How to Load Plug-ins in Maya + +The provided module files (*.mod) facilitates setting various environment variables for plugins and libraries. After the project is successfully built, ```mayaHydra.mod``` are installed inside the install directory. In order for Maya to discover these mod files, ```MAYA_MODULE_PATH``` environment variable needs to be set to point to the location where the mod files are installed. +Examples: +``` +set MAYA_MODULE_PATH=C:\workspace\install\RelWithDebInfo +export MAYA_MODULE_PATH=/usr/local/workspace/install/RelWithDebInfo +``` +Once MAYA_MODULE_PATH is set, run maya and go to ```Windows -> Setting/Preferences -> Plug-in Manager``` to load the plugins. diff --git a/doc/codingGuidelines.md b/doc/codingGuidelines.md new file mode 100644 index 0000000000..de8a13aa0e --- /dev/null +++ b/doc/codingGuidelines.md @@ -0,0 +1,376 @@ +This document outlines coding guidelines for contributions to the [maya-hydra](https://github.com/autodesk/maya-hydra) project. + +# C++ Coding Guidelines + +Many of the C++ coding guidelines below are validated and enforced through the use of `clang-format` which is provided by the [LLVM project](https://github.com/llvm/llvm-project). Since the adjustments made by `clang-format` can vary from version to version, we standardize this project on a single `clang-format` version to ensure consistent results for all contributions made to maya-hydra. + +| | Version | Source Code | Release | +|:--------------------:|:-------:|:-----------:|:-------:| +| `clang-format`/LLVM | 10.0.0 | [llvmorg-10.0.0 Tag](https://github.com/llvm/llvm-project/tree/llvmorg-10.0.0) | [LLVM 10.0.0](https://github.com/llvm/llvm-project/releases/tag/llvmorg-10.0.0) | + +## Foundation/Critical +### License notice +Every file should start with the Apache 2.0 licensing statement: +```cpp +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +``` + +### Copyright notice +* Every file should contain at least one Copyright line at the top, which can be to Autodesk or the individual/company that contributed the code. +* Multiple copyright lines are allowed, and if a significant new contribution is made to an existing file then an individual or company can append a new line to the copyright section. +* The year the original contribution is made should be included but there is no requirement to update this every year. +* There is no requirement that an Autodesk copyright line should be in all files. If an individual or company contributes new files they do not have to add both their name and Autodesk's name. +* If existing code is being refactored or moved within the repo then the original copyright lines should be maintained. You should only append a new copyright line if significant new changes are added. For example, if some utility code is being moved from a plugin into a common area, and the work involved is only minor name changes or include path updates, then the original copyright should be maintained. In this case, there is no requirement to add a new copyright for the person that handled the refactoring. + +### #pragma once vs include guard +Do not use `#pragma once` to avoid files being `#include’d` multiple times (which can cause duplication of definitions & compilation errors). While it's widely supported, it's a non-standard and non-portable extension. In some cases using `#pragma once` may include two copies of the same file when they are accessible via multiple different paths. + +All code should use include guards instead `#pragma once`. To ensure uniqueness, include guards must include the path to the included header, with path components converted to uppercase, and separated by an underscore. For example, for file lib/mayaHydra/hydraExtensions/sceneIndex/mhMayaSceneIndex.h: +```cpp +#ifndef LIB_MAYAHYDRA_HYDRAEXTENSIONS_SCENEINDEX_MHMAYASCENEINDEX_H +#define LIB_MAYAHYDRA_HYDRAEXTENSIONS_SCENEINDEX_MHMAYASCENEINDEX_H +// … declarations … +#endif // LIB_MAYAHYDRA_HYDRAEXTENSIONS_SCENEINDEX_MHMAYASCENEINDEX_H +``` + +Never use reserved identifiers, like underscore followed immediately by an uppercase letter or double underscore. For example, the following is illegal: +```cpp +// file foobar.h: +#ifndef __LIBRARY_FOOBAR_H +#define __LIBRARY_FOOBAR_H +// … declarations … +#endif // __LIBRARY_FOOBAR_H + +// file foobar2.h: +#ifndef _LIBRARY_FOOBAR2_H +#define _LIBRARY_FOOBAR2_H +// … declarations … +#endif // _LIBRARY_FOOBAR2_H +``` + +### Naming (file, type, variable, constant, function, namespace, macro, template parameters, schema names) +**General Naming Rules** +The _Hydra for Maya_ project strives to use "camel case" naming. That is, each word is capitalized, except possibly the first word: +* UpperCamelCase +* lowerCamelCase + +While underscores in names (_) (e.g., as separator) are not strictly forbidden, they are strongly discouraged. +Optimize for readability by selecting names that are clear to others (e.g., people on different teams.) +Use names that describe the purpose or intent of the object. Do not worry about saving horizontal space. It is more important to make your code easily understandable by others. Minimize the use of abbreviations that would likely be unknown to someone outside your project (especially acronyms and initialisms). +***Project-Related Naming*** +When using Maya Hydra in code, the string MayaHydra or mayaHydra should be used, depending on required capitalization. The string maya-hydra should only be used when referring to the Github repository. + +**Type Names** +All type names (i.e., classes, structs, type aliases, enums, and type template parameters) should use UpperCamelCase, with no underscores. All top-level classes in the MayaHydra namespace should use a short prefix, to categorize them. Nested classes declared inside another class should not have a prefix. The only currently defined prefix is Mh; others will be added as appropriate. For example: +```cpp +class MhSceneIndex; +class MhSceneIndex::Data; +enum Roles; +``` + +**File Names** +Filenames should be lowerCamelCase and should not include underscores (_) or dashes (-). +C++ files should end in .cpp and header files should end in .h. +In general, make your filenames as specific as possible. Where file names declare a single class (which should be the majority of cases), the file name should match the class name, except for the case of the leading character. For example: +``` +mhSceneIndex.cpp +mhSceneIndex.h +``` + +**Variable Names** +Variables names, including function parameters and data members should use lowerCamelCase, with no underscores. For example: +```cpp +const MDagPath& dagPath +const MVector& rayDirection +bool* drawRenderPurpose +``` + +**Class/Struct Data Members** +Non-static data members of classes/structs are named like ordinary non-member variables with leading "_". For example: +```cpp +UsdMayaStageNoticeListener _stageNoticeListener; +std::map _boundingBoxCache; +``` + +For static data members the leading underscore should be omitted. +```cpp +static const MTypeId typeId; +``` + +**Constant Names** +Variables declared constexpr or const, whose value is fixed for the duration of the program, should be named with a leading "k" followed by UpperCamelCase. For example: +```cpp +const int kDaysInAWeek = 7; +const int kMyMagicNumber = 42; +``` + +**Function/Method Names** +All functions should be lowerCamelCase. This avoids inconsistencies with the MayaAPI and virtual methods. For example: +```cpp +MString name() const override; +void registerExitCallback(); +``` + +**Namespace Names** +Namespace names should be UpperCamelCase. Top-level namespace names are based on the project name. +```cpp +namespace MayaAttrs {} +``` +***Use of Pixar Namespace*** +For multiple reasons, some code in the maya-hydra repository places symbols in the Pixar namespace. We intend to move this code to a more appropriate namespace, *e.g.* the MayaHydra namespace. No new code can add symbols to the Pixar namespace. + +When using the Pixar namespace, the PXR_NS macro must be used. This ensures that the maya-hydra code base can be compatible with developers wishing to compile USD code with a Pixar namespace different from the default (pxr). + +**Enumerator Names** +Enumerators (for both scoped and unscoped enums) should be named like constants (i.e., `kEnumName`.) + +The enumeration name, `StringPolicy` is a type and therefore mixed case. +```cpp +enum class StringPolicy +{ + kStringOptional, + kStringMustHaveValue +}; +``` + +**Macro Names** +In general, macros should be avoided (see [Modern C++](https://docs.google.com/document/d/1Jvbpfh2WNzHxGQtjqctZ1K1lnpaAtHOUwm0kmmEcxjY/edit#heading=h.ynbggnv41p3) ). However, if they are absolutely needed, macros should be all capitals, with words separated by underscores. +```cpp +#define ROUND(x) … +#define PI_ROUNDED 3.0 +``` + +**Schema Names** + + +### Documentation (class, method, variable, comments) +* [Doxygen ](http://www.doxygen.nl/index.html) will be used to generate documentation from _Hydra for Maya_ C++ sources. +* Doxygen tags must be used to document classes, function parameters, function return values, and thrown exceptions. +* The _Hydra for Maya_ project does require the use of any Doxygen formatting style ( [Doxygen built-in formatting](http://www.doxygen.nl/manual/commands.html) ) +* Comments for users of classes and functions must be written in headers files. Comments in definition files are meant for contributors and maintainers. + +### Namespaces + +#### In header files (e.g. .h) + +* **Required:** to use fully qualified namespace names. Global scope using directives are not allowed. Inline code can use using directives in implementations, within a scope, when there is no other choice (e.g. when using macros, which are not namespaced). + +```cpp +// In aFile.h +inline PXR_NS::UsdPrim prim() const +{ + PXR_NAMESPACE_USING_DIRECTIVE + TF_AXIOM(fItem != nullptr); + return fItem->prim(); +} +``` + +#### In implementation files (e.g. .cpp) + +* **Recommended:** to use fully qualified namespace names, unless clarity or readability is degraded by use of explicit namespaces, in which case a using directive is acceptable. +* **Recommended:** to use the existing namespace style, and not make gratuitous changes. If the file is using explicit namespaces, new code should follow this style, unless the changes are so significant that clarity or readability is degraded. If the file has one or more using directives, new code should follow this style. + +### Include directive +For source files (.cpp) with an associated header file (.h) that resides in the same directory, it should be `#include`'d with double quotes and no path. This formatting should be followed regardless of with whether the associated header is public or private. For example: +```cpp +// In foobar.cpp +#include "foobar.h" +``` + +All included public header files from outside and inside the project should be `#include`’d using angle brackets. For example: +```cpp +#include +#include +``` + +Private project’s header files should be `#include`'d using double quotes, and a relative path. Private headers may live in the same directory or sub-directories, but they should never be included using "._" or ".._" as part of a relative path. For example: +```cpp +#include "privateUtils.h" +#include "pvt/helperFunctions.h" +``` + +### Include order +Headers should be included in the following order, with each section separated by a blank line and files sorted alphabetically: + +1. Related header +2. All private headers +3. All public headers from this repository (maya-hydra) +4. Pixar + USD headers +5. Autodesk + Maya headers +6. Other libraries' headers +7. C++ standard library headers +8. C system headers +9. Conditional includes + +```cpp +#include "exportTranslator.h" + +#include "private/util.h" + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include +``` + +### Cross-platform development +* Consult the [build.md](./build.md) compatibility table to ensure you use the recommended tool (i.e., compiler, operating system, CMake, etc.) versions. +* Before pull requests (PRs) are considered for integration, they must compile and all tests should pass on all suppported platforms. + +### Conditional compilation (Maya, USD version) +**Maya** + * `MAYA_API_VERSION` is the consistent macro to test Maya version (`MAYA_APP_VERSION` * 10000 + `MAJOR_VERSION` * 100 + `MINOR_VERSION`) + * `MAYA_APP_VERSION` is available only since Maya 2019 and is a simple year number, so it is not allowed. + +**USD** + * `PXR_VERSION` is the macro to test USD version (`PXR_MAJOR_VERSION` * 10000 + `PXR_MINOR_VERSION` * 100 + `PXR_PATCH_VERSION`) + +Respect the minimum supported version for Maya and USD stated in [build.md](https://github.com/Autodesk/maya-hydra/blob/dev/doc/build.md) . + +### std over boost +Recent extensions to the C++ standard introduce many features previously only found in [boost](http://boost.org). To avoid introducing additional dependencies, developers should strive to use functionality in the C++ std over boost. If you encounter usage of boost in the code, consider converting this to the equivalent std mechanism. +Our library currently has the following boost dependencies: +* `boost::python` +* `boost::make_shared` (preferable to replace with `std::shared_ptr`) + +***Update:*** +* `boost::filesystem` and `boost::system` are removed. Until the transition to C++17 std::filesystem, [ghc::filesystem](https://github.com/gulrak/filesystem) must be used as an alternative across the project. + + +## Modern C++ +Our goal is to develop [maya-hydra](https://github.com/autodesk/maya-hydra) following modern C++ practices. We’ll follow the [C++ Core Guidelines](http://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines) and pay attention to: +* `using` (vs `typedef`) keyword +* `virtual`, `override` and `final` keyword +* `default` and `delete` keywords +* `auto` keyword +* initialization - `{}` +* `nullptr` keyword +* … + +## Diagnostic Facilities + +Developers are encouraged to use TF library diagnostic facilities in Maya USD. Please follow below guideline for picking the correct facility: https://graphics.pixar.com/usd/docs/api/page_tf__diagnostic.html + + +# Coding guidelines for Python +We are adopting the [PEP-8](https://www.python.org/dev/peps/pep-0008) style for Python Code with the following modification: +* Mixed-case for variable and function names are allowed + +[Pylint](https://www.pylint.org/) is recommended for automation. + + +# Coding guidelines for CMake +## Modern CMake +1. Target Build and Usage requirements should be very clear. +* build requirements ( everything that is needed to build the target ) +* usage requirements ( everything that is needed to use this target as a dependency of another target) +2. Always use target_xxx() and make sure to add the PUBLIC/PRIVATE/INTERFACE keywords as appropriate. +* target_sources +* target_compile_definitions +* target_compile_options +* target_include_directories +* target_link_libraries +* ... + +Keyword meanings: +* **PRIVATE**: requirement should apply to just this target. +* **PUBLIC**: requirement should apply to this target and anything that links to it. +* **INTERFACE**: requirement should apply just to things that link to it. + +3. Don't use Macros that affect all targets (e.g add_definitions, link_libraries, include_directories). +4. Prefer functions over macros whenever reasonable. +5. Treat warnings as errors. +6. Use cmake_parse_arguments as the recommended way for parsing the arguments given to the macro or function. +7. Don't use file(GLOB). +8. Be explicit by calling set_target_properties when it's appropriate. +9. Links against Boost or GTest using imported targets rather than variables: +e.g Boost::filesystem, Boost::system, GTest::GTest + +## Compiler features/flags/definitions +1. Setting or appending compiler flags/definitions via CMAKE_CXX_FLAGS is NOT allowed. +2. Any front-end flags (e.g. -Wno-xxx, cxx_std_xx) should be added to cmake/compiler_config.cmake +3. All current targets, as well as newly added targets, must use mayaHydra_compile_config function in order to get the project-wide flags/definitions. These flags/definitions are added privately via target_compile_features, target_compile_options, target_compile_definitions. Individual targets are still allowed to call target_compile_definitions, target_compile_features individually for any additional flags/definitions by providing appropriate PUBLIC/INTERFACE/PRIVATE keywords. For example: +```cmake +# ----------------------------------------------------------------------------- +# compiler configuration +# ----------------------------------------------------------------------------- +add_library(${UFE_PYTHON_TARGET_NAME} SHARED) +target_compile_definitions(${UFE_PYTHON_TARGET_NAME} + PRIVATE + MFB_PACKAGE_NAME=${UFE_PYTHON_MODULE_NAME} + MFB_ALT_PACKAGE_NAME=${UFE_PYTHON_MODULE_NAME} + MFB_PACKAGE_MODULE="${PROJECT_NAME}.${UFE_PYTHON_MODULE_NAME}" +) +mayaHydra_compile_config(${UFE_PYTHON_TARGET_NAME}) +``` + +## Dynamic linking and Run-time Search Path +Use provided mayaUsd_xxx_rpath() utility functions to handle run-time search path for both MacOSX and Linux. + +## Naming Conventions + 1. CMake commands are case-insensitive. Use lower_case for CMake functions and macros, Use upper_case for CMake variables + ```cmake +# e.g cmake functions +add_subdirectory(schemas) +target_compile_definitions(....) + +# e.g cmake variables +${CMAKE_CXX_COMPILER_ID} +${CMAKE_SYSTEM_NAME} +${CMAKE_INSTALL_PREFIX} +``` + + 2. Use upper_case for Option names +```cmake +# e.g for options names +option(BUILD_USDMAYA_SCHEMAS "Build optional schemas." ON) +option(BUILD_TESTS "Build tests." ON) +option(BUILD_MAYAHYDRALIB "Build the Maya-To-Hydra plugin and scene delegate." ON) +``` + +3. Use upper_case for Custom variables +```cmake +# e.g for options names +set(Boost_USE_DEBUG_PYTHON ON) + +set(HEADERS + jobArgs.h + modelKindProcessor.h + readJob.h + writeJob.h +) + +set(RESOURCES_INSTALL_PATH ${CMAKE_INSTALL_PREFIX}/lib/usd/${TARGET_NAME}/resources) + +set(USDTRANSACTION_PYTHON_LIBRARY_LOCATION ${AL_INSTALL_PREFIX}/lib/python/AL/usd/transaction) +``` + +4. Respect third-party variables ( don't change them ) +```cmake +# e.g boost +set(BOOST_ROOT ${pxr_usd_location}) +set(Boost_USE_DEBUG_PYTHON ON) +``` + +5. Avoid adding the optional name in endfunction/endmacro([]). +> Although there is no official guideline for Modern CMake practices, the following resources provide ample information on how to adopt these practices: +> [Modern cmake](https://cliutils.gitlab.io/modern-cmake/), [Effective Modern CMake](https://gist.github.com/mbinna/c61dbb39bca0e4fb7d1f73b0d66a4fd1) diff --git a/doc/hydraSelectionDataSource.png b/doc/hydraSelectionDataSource.png new file mode 100644 index 0000000000000000000000000000000000000000..f14ea16b2dc5c9386448a152174d2ff30efb4c07 GIT binary patch literal 15871 zcmdtJc|4SF+XsA6DzqqDi_m6CNQG=eMP)B4$}*UcWe6F>7%Cw#KTBktQb<{{%~)qt zm`W%+gON#N8@sWLG4C}>y6^k>Jn#EF@AAjnXL5bUbzSFqoX7G#R^gY7E^Oo3%L4$w zHr@XJQWGlpjXpa9LgY|90H=e~XM zx)%WKy2Jj@(dQuM3ji8%y4q*1_*&C?3Fxtp(fuAw&7~c6)>*xCMw&sjdK!%!u#cVl zM7H0udAm8Mt8qKe*FB+X{a>S=Vbcy#_rTPgTzhA#6Zg)w1*Y7vZ@F_^~{aHF}w?NpR8`@j{-lp!E*7E4685M(MK1aqA zL-SFHm65r8TG66{5?yg8ebiCY^jc?~>_L2B6(-UpN^*^;8d=Ia-Woz0l(1LmftWQk zS&A*%o3N}mC@5;-_KGTYaY>}3*sSf2CSgS%Dc@%1L1=wij>H=YCl{h< zS3kd75(s43#zn!c@2C+E=P2Z#L7xnee@Hw+qKD?UnJ+@15K$(6=kK#$HakYRxBJoK z9N$>ZgBlS!5!UiLy2=P=R+||#nNvs8J>sMz>g>fc6ss6_HMA(nwOV8nF`($88# zuZZh)$~blj9_MeyihIkCVdQG$=wfw~otf~*qWtrUL-&m4dF#*lmp~qW+y4QPjL=LF z*|DZ{uOE%#UuMdv8%*^JC?@dMTxva~6E3%lUoZqeszj%u>B_h&&UFV)= zz{^RW@)x{ep;O%y!TiH@I?!$T%eAzJoNxF0f0|m;*Yww=d0KG?2hT|b?-=ghjHeT^JAE9~X!p7u1%1bjqs#<;4*6$9Y9) z78GjB;>mBD(0n{r5+1O@ZKnex9?L&}MC}^Q|Jmlp`R~LW+yCMXXqZK?;-EF1xhF1q zB<~ns%him6Hx6qQ@YTR-1~>~U1f%`8|1s&;VWaK}NB)@p>n(p9_5WAc(X=v*SsC66 zVkiJ`9c2F$PuJSWzG@4A7z_YGVVpbJSH8>Qob2mC9&PqTOFMWA`x++2$HBht{I{2D zIj77BOKpod0+Y4~Wz9i7j(TYhH+XmQ)gvyuQfHH-!<60c8PFxmIFZ4k6V#{d?a+@ncWw{}JGV;XX-*#P zt(U~NuTt2*MT$>BEBN#(eL;twY1IH+1AMF0MFstzy;Pi|wOqScfzwQ7&1OwBk(UQ5 zW=NQoJAsUGVdj!M-n~81zMM?*p+X-};pDH_8K!b(Q~MW49{l#!zD6=L{p+~gG;Atj zqHh4Qo!p^@nE0DuBPD}d>{GCinIogv{GraNYc z@|Y%fvQRhq92Py_nSWayk_L?!tsA49wuXw!A(^IdR7$)+7&#zcf(}<$U@9~3N*|sr zy!hxq;wciIHdA5rng+>}#sAb04EfxE*0DI{tz$d0%!uTfdQY8Amw}>|R61V)pidQU zkl}-UznZi#@k(o%PFr3aXB1*MFAL9H%SK-E4>ng7N^gZcwGzk1Iwao8vQv=1UC8Nq zyY^O2zO?wwdueknM8*mu&{88}-F6WDh5=v#IRqyOWs@q$PynZT>T5g4{ z8t}KNOKzqoQMxpC+_{`XbW4y`p}UFAmF*psAJt=wN6FHX3P5MrnT4VOKtDcHIPCf4 z(lTPDT@Kx?TOlCxQ}2bG^MI;i6hZE>O3GL$Jv(fY_nNfOlHDP0B>c=((TVBNJ><<* z6DQG;M;|RCd71S@;>i*|9yh4?-F#NFD$;dsrn!8Wcs9q|;HF&>JY2zNk_LO^L!JX|v9XPh0O`F#nCMEqA!x)QUP zx$9fHn{J9z==C!lVB9uNeaJPC@@YBF`?-`!?KvDC)4CbEMWnca$EZrS&Ea?&{0+t* zS^U+(>A9Dmnh7v(Hjv4U)T_@W35V%JS*_j~O`lJ>i0~&`Qq3;v#SRo37~?B+%TIpm zKz`7MOpxj&rxrJoM-NdP_*n?eI{MVols4%nV`6~N>w+Z~JwZRO!*PYqvhdgCMvF)~ zsU4C!WqaJm%x@Br<5OP(x&IlO(j(>+A4@`x?OQ5R@*&Gd_I|B}?KP8^p9fKolWC+9_|*d@Y0Nrq6II zQ}J%QTxk5Gs8iWS=!{%}3m@>b2bHOpsH4f@0StSIy9SA43B)@~Pt)y}AV&M-=c2u< zS|x{g9|kO2D?Pj$>9qCg?V5QCZMj`QrF`KVNz5OKs4T$V#oVb-Ab8~h3{59C8z?g_97TA6FaZ)>h?W=>sEyU^h3ao&P4zl38u7U#C zqpw-T%Xf=-Y7l2-xeOs)?e*PfpUFy5Hmdfl?3QdUKdn&s>5kevw~mZJ+5_w8sbvW5 zL}F=jx~Hp{aAX1vhH`ybJw$qfhMuskh<1IW(Ptt)LK(v2-}}Od=`s*j$D$_Dnq{-q z54q^_c}W1>^#FGKG(7{dbQ(IOIbS4Be7WBk(lSMTVI`?4vwA+ku;8A)rv}oTKgwnc z_bahn|{=XX<)tI)NMXi;cg2EFe5G0-ZBZ zKuFPCs3QjepyEV@mNqj_o-eDgmJ$e}ElI|9K5167|7zm6-tbPIBi>x&$MgOAKZYm( z0ONdc7za+OvKVY{DciL5&1>YbOw?SeQxe!1ieTC49J zSarxEX9~2IY3?YCl>f?GzBU(M88IWMvVJNP!p#hCAA8QXG$@Z!SupU8n!KLqZveW~ zpbJ-z{qp6+bN{_>|L+5XE64ayD>Il4z}s?@?YmgZnk+E@cyn}6?DA)e3;jZ28G``O z^Et!POWI=ta63|(%g)J0>cHwBeKy}HTw5f|~BpgDRw+6~u>HvbC z?57310q+ZkWq@2W*Q4%fy6nOZM>>OFq`G_OO~LhP`dcGTwZoUq zN0)DYbzZGWY{nt_Z&58$UDUx-1eFY_Q&gM{*|CD!?c$l-MnzF@z3Lm_-XbLuAkTyL zpYC3~M%}vuT+c>UwaGW?20s{PRBsnqr6m+x@2ya(qxyy1aBv%&%~1$^TUi~Of94U5 zC(~ZEYxDBduw6T}5emU4*|RfnhSE42;_a{%!zAdO`b<=%~5 z7ep=0*&4W#!+@R${>?ob1zQyCJkboa|YJrk%Pz=n5l2L2BQNA)g9HyNMk52?c+Z1b zg%sgk;AleG)yW!wEAnQ~4U?``zG+a-6`I7c-9?jSNqvU5L-C zK)>PD_Ul7l*1Gt)E2F)cmo6F_jxPVf18lwwA#b9mBtLqN7Snv~Wq+#$qeg+hvhpZy zT4kCFSGu|+D8`_-Myknr@%Z#);_8Y zuPV((z((aS-OEgMZ^=aulC@1hJ2?L2m->NP{&vF!HS&lhsLp!k5YSFIB<)E36# zV0Ptp;L>JrK~tU@lkoyQCW1CRxV$oaCr~bT1JDENG<)B?@kYP?*D_%l+Q;;0@;sX< z%)#X=H#P&W9)jwhZN{gsaxt!ZqN8PTklC`>!cYUPKgWQ&2VGo*>?lI3-=34r#2I`v zmY<-uc=_!Io{9ca!*?{#Z!?Rt*}9F9^9cz*3wqji#Vf0f%=R62ThwtN(@i__+${q8 zAb~0+Oo-sL^({6~xPa0=L~|w2^(yb^`N0ym#{6^tl0qyWNB@>Pc~f7c`R-f%NT-@g zZN&^wCiUy_lf0kGp^l`Lo=V~WcAxc^4Vp1 zr(I9!nn+p$T2<`zWcluodVb4z-)NznMa%HGxrco=-M`=AoVpR9vJDUk1$l9VYrwni z5w5`7tJ`|-g?KjfCH(?sALM+)28I;hhlG}oG$kNtWZ06@h1W8GZvUrOX+#JC!#&OP zC}&xm-c9VOQ(f>>0#(>N`8WH~oYC@G96|TLmDLp?gFC~ZvVDP5Uxr^kdT$#nRogeI z@4UA_YL*xdMVO9sMC$*|rTus<*X;nRn-kcsL*3hC#iN}O;5~68TT4VJ1PJ=0ZHO5Z zsBc8jP^h9@c1u4^sIU7CK?hMS3PS$BD4ExU4Co2&{x8&2G+nE52|c+1K(Hz4d3R1#}~T0@Fo7QIH!EW{kn0zA)4(Xc#nL>{f44p zS`KrYzdmsXQ;wd0(5>c=wIL;L|jl8QuP)5TBqR{#{+Yq{wpTSS~ z^fokOBg8dXej?;>-yAPLm!dT9=+ePU@mb+oegl~}N76904`zI;Xl(bBx_M|q>c%H@ zmjKuPf>%w*K}`$LBu6Umy+XT~G=gQajQQ^a{_4YC8v!B+_p4%{u*A0yZ5-2Zs~N~; zM&=yb@d;Pq?E4&!HMET-^x_{!ztjz;wMe;5Ga2e84P!gu z@!}pv7DO6+;QP2xCH_~%^KL~)~)#%fI%s(8~3w*Or^v#2P zF^j`@M%>AJoz&t2VJUO8?qu)%v6H>DjFn+rJtB2JNyFNIuCsTC+Dc7F$t~CZZ$nLQ zXG&v!o$MH&etiP>gfYrsGkecZ{$m=Of?!~r!Un8v4ej94T)l$MBO->{Qc#*Avz3Q&rsNnnjS4bg{5s!-^);$7fpb$ z&~vK_baDw_)xyhvFD zbQ|+Pz|IxywdIl7<7vy0IT~;C<<*-A_ID<4k zB6qzkZ#Nf_zuK9tb7fkVX%RUVsmVf7{H(8m+2pvX)A^jfmVfm67+=DXpK%|JvgVg3 z$ciB!8TAo*o+ig7vf(${1#k3TEi)l~&5S!`pQs#P>_k*cZQaepL;LdNUe9`U;jJY} zdOG$}61^Q?1u?YggiF;KD}EfW>-M`ETYnf{CR8XD8WvS8^x)&M#%3FF$X5I;!<{ii zQGr;aQmDlnj;6e9*373`hNGY4*LOzd(txC zzA$^P8yBm4AsJH-Npapt&!)WH2%Xg8Q7DWoXR!AnLvRVNTp!A4$1AlNL5KHMZZf?! zmw$jP#!w&Er$e^Q`_S>0zRVBEl)mlN>L4gNSKS-JB+vRCS08V3{M4ypJm0qDQQ59M;?rQQoN}bWCFENq6_u z7kWLc3$o;ATRDIr4Z?R##CqEhWAVB^>fPbm$5Jr)y-nGpKgzl#6M-k~ip`^Y_kpFL zY>J`NW*fOb_$8g{I9sMG8am_V)wd^wH)|IDkUb?B!4y;rE7?KfAaYpkG+m zTsx=UJ4d5Kv^x&>PFj3@_dxS-o83LvSUa;*ktJ>D+K_@x$n$kxxP2F=-Qs=Vu4<$U zo%rv5rpFQd_f5zJ9$9%sO&<_*Ki*^Kz+#D4`;+dtH0X`(Hb>bs$%2TPxlgTWYm@Bo zNSe=+ufuARX#3f+4!1Y09bJJ+62MsOx_Q+!Py0R8on~IS=y{;vv%BjD*po@SGk!cQ z$IHEYuee(1#Vr=TGl$#Z14x%E8oP+%eJlLdP-TP-6tyU%OY$kj*rwp!&B?~vJUAo6 zqsMrg+S&|IC}Q1iwKP*=a`*OXOHgujbMg52rjrP_qp22W{j2a3$r#jCamMaI+kowo zO8XbRI?9*tm>+LNiHE`K4YmL_93bo4gu)`MP%Bju^XhJWoIwYEA7ww+z~SNc#P4l| zN8D7}H9+($Om429_kP&GAokWJ%qbRyH`Q!LuT+tJ}liN71X_!vK`&?U*D5JZJ{Ujw- zu=Gpvy^H2Qxw#CFrgY}}x^@WB;h)HdD}I`PbZO1^8usmO?zmUQj!-)|J*>wQ2b!rj z^2Hb`)7^%&9bjbn8%0P=LG9Fka!vf97a!RcI>%1#g_5oGs=Zb(fH7#>|ykIA+CSQL*-5<`sit#$+>; zV-z|OLV31yhcw!bW0t@1Wx9+IopKf-8p`=9YJp6Tdb)yzTf6;4f>j4zX})t)2z|WJ zL9KS)HV5tMW5lY@cUmapsVBj2GMAcMZdf~;_FY@S3)kI-^3o01F*0w>V%bNv3xgPz zZQyQJ&pfs5Y{i(zg7TcMraWt4%$Okp9L@aAS^uZ3`=#m!c%O)0xe7khnN)B-G;KLm^B=()t{#IoYNQ;6?ginsiorpxd2nyZbaf(}yy zET7HA3N&3_D)N`se2g5YLfjtz#lCQ8$IBjOx`z z<>GRR13n`@Ok04VX~DkxO;0)y!`Mo1811bO8&YqG$G01rsZMOytOyL;&FE=DidcT$ z09^WmpbES-N(b;5o=++Uh(TqX#{fyP+5A}$%VWeKR+7r(YH3o0zyBcm|kodncF^mR?vpguP!fh}=E_lrNb#nYIZKd4xYdXYseB^gx@3NX1~#~lh**rHz%Ug^p;^pi(r1iJ z#Wve`B22^9V%#6`y~L)29nHm%%E`TY0XUbpY*40ukyCmBHn$=pR zid}DU`R^a?vIzEl!w$SlHUBs&uvF{y%Uzos=G5B1Mq;Qu-;wWV={ZtD23hFFP!>xm zbw%xZUI?iXdx$wPR&09nK$O{B1{APn%p19JTW}F^vo_e0k?pXJa9j}c` zY)Dh+X9-K77reku@SC^>C_gp>TH(KKD2lVdo}X3tgDiAfp4nE=Qj`-*P`1zik^#wz zT{%~4d!0m|O(|)CWhq=Cxg{z;vN)SWHAwQrwVPQ4^BsL(v^X5sM|VI|dF&e>K-8KuG) zgA<_!mmcvaaS`3e=^Q-?AaJPWi$|pvy)U2gqV*>P9irmLEHO+@PhaanOYZ(#YfKgh ztDAl)_V~#Bd4qn#vuKcU$d`sOG4#$(<=vFmoF1-LPMUW}4t`1=Q+3V;kU&%1j(+Q3 zxyG?v%+F>Omrvy?@N*oUDLXRWjHg?8uTZS}`g1l} zHS=QZlL)S{bc$mKMPLUhO2cxWb3)GzV{aUzMfp_E-* z^k*l=)Utra>829;@=)YSL`3!$Qq0t8GtSeZfAny`Fk_=ALHn6*Px(U3P0<8L=yq6a z;Gw|qh@O#?gRIP#pR{g~k1P!AcJRZB>89L_TdED$%O-caxW&;AS+ozidGGa){9aH@ z@MHkv?DVvcr-oE|yRrOs9*tzZ`vT_VX6DTR?__DV%-5bT>-2hA>_TwqbJB9OJ>VPg%S1i5Zw^vjugNJNj*TFu*i1(H0d5z5 zS)*d(_xl8UUcekIY5y)7dz`!LSx~nTW(6X~>SJKNiO-e zQ=X3q@a;xTh2z7^(`oU8gm5Dyn6u7g%_q#5;&R~rcN@Eh3tD!+hT{yd+`tUTmM;I` z(jOwCc>@{9&hda_eLmK(<=!b&fxx+UW3?8S~Z>g2s4!!qUa$!-5eo7|9!W z3&QCy?`~Qv6WZXXEI(c8m5+pYr@}V_U;pAtd2Nl|9na$nvPH_5CndJSXq6D)IEI}| zgUvbbqGH0wdUwhFY8TL{5mnyU{eafG>9`F^@JBqIWsud0WNrP+yCwE_(Vq8dA7Daj zS8yHgv9Ll8oBe+>CH}^Yjy^lpL3Irge&Fl+Dv*@8*+E|JhfwRUeM)y#BaH>RS#M(M zCMBd7y%q=V+uDL67y?BALOXByq(0PyKQPY}WWuF@OvJFWt$82aB_(YR^3k)(9eBHK9XRV| zYtoAYA9aYGvt2jbTcveM_FI~YSj&x7t~Wr;&;MdINiZ~6_Vw&c)=YiH z(KFPn8UEmxI+Y9+K2!e!`f< z!~Le+>{VBf=X0^(ql-K5{(#}f$INhvSeDt_9*F$cm~5TiPD+N^@ehRv=;l$t2KmQM zYBrnU%YAZ9o^u1Z7i^8*0QATIbrLD0S9Z%%TGEoAaTq`M4+uc4B^~W2-k& zANQ=PF&M{3&&U9+>$qXQC$TyQ68;T^ahbK4A0xkyAtWkZ7N z+W$&k@&W^C$J8B-Aex&^aRsx>+t$mA;5Lubs&kaW!(-9e{2>+wml7UNfz1|oi3<}G z0%51$S}ZT;vP;6C?`mtzH@kb+Pnk-kTfSmFg8<_7Yp$-~1cI~3bythgQRou+EGEs~ z>Zxm-M2gd{<`Q0=AB_4%>EvfZqHo|o2!q>4Tc(Y(Vn-Z1KE0HT^ifreeNk%F8_&5D zeHIx-!h};}nK}9u->0vLJQ$o#G^Q+SI?jWlVmE4NP5C2CpGFJ?vyHl((yOb0-F96b z^CaONy@;r`Y{^l5WnIvpEj9hPySeWfsLl$0IR5P;&s*!16fE*R<&>?I!O?PJqtag6 zffmEl1FFLb?HSxp(zb%7)7ZGTDZdDzyu>qek}Wcr7!JmDF_ub&BxtVtTFOphkIn7|YO>`94y?BupQNW5 zSW(qQ!Ac{NHj_PC*UeE4TaS@e*FWwZV^b3QkE50LT_n{N=4zE4X=M}~cHuj+y^45l zk#(jBPGq=&2@yMM0LSa<`eL(3Gu5vZzW;{ZGdhvG&t)@!{!6tIMLJFSTM6E{#H7Ic)K$G;_hVkp zKMM%l?@!27H=3R)V%op&RN6aU=Hr_D0SN?E{k5TuUttq!-*$j)7|wEhV$Yur1Z;oi z@JDU51&$?FzqE+eGuy{e{VOA^S{^w>^#bPBg~|yiWqCzrHL|AyHm|?>Uo>3S!qk83 z&D4DE`RUThVPnM>^EGkLK(3&>>%pZAzr=9^JK$FxL?|o;%JFRbpIvG`f86NMrfaZl z5$c5>bi40Pi$GOG5PY`$pfe&9f1CT=z=j9&=$XAh&R4}Z=C6VV`X?Y zWlkk^R?Jxr%nmP!4}8mRLNdIj1CAnp)@$-I$!}vZA*a$t9S_tA21=TwGdxp+iipRqxD_<&BQV zFJ6YbXB%gyS~vaJgl=4M^+oGz^3t9hlvdwFd6GtEH@8*`RxOXKBuzV1(B}#|trV|N zY7Sn3$Y&_rYo=}nM%LSPIbY^t6RM3w#=N2>1Om@c4Yli5y9ySz8lXL;&KG^x5Y;nV zn7ewg7g0G8&rUm6+p2D38j0nN{yjnun@psX>d$(ZFYBZQ_S9=?0+;0f!~%Bqd6>$L zwgX?ucLgN3_HH~^#FgGbG}4}(Idy*Vnp7{*Me?k_gre`AC9eq${A|%zA@2Ku8NW;k z4Nkc&s(;ro_smO~F0EoN+!@gnd#?4x%~5uKbhv2;(7K*Pgxx#k@sIquk1KIH3>37S z^rV~=eST@H#KE6_N|`=`AK2S~Mytv!n-^8?!6YIQ7t6*_QDZXf1G8h zGU#zghSR<(K+Y(Tkh33`$oUHMCkEWfg%~(q8v_BN^uTIP1J^+?g)Ch^>OmebcFq9@ z3I5Hzeh0&?*V&==e}ug3&M13y?tfd$`7g^pX;lJY-*^0gnvD2VWsrTH86ni=RP(@$nmgqhn7LN}JE%AsbJ^W~GcY+k$Xp(7$$E%?E&QnbZH~v?YxQGK zv`BSwEcA*8I0uw%HSzeQd7nB;C2_?G=||w`S6GjpeoZieJXR>ho;J|wd|Q1Nf>gz% zeW0+TKmoX)t(jYF{0`o_v){A2eL~1yrd#oZ6ihk7Ip=L2sbe~DL-mXQAPm%vx0R>2 zp7k(0I@Bzv7R__Us?*Tv(AWvJ#u)Dm=B6Oxx`zMmQ-j{d;v&vvMRp|4 z)iou1G$sI~*W=&mHT4R9N|s2^L%$nJxPJW(sxbEI{Ws@!l%tm;10Ct=;O3U((0}RP zi)3ppPY>4Xu=_>sLazz2yH|EYP#^dJ_GIP1K1t)T%a#(zptlTZB3NJm$GQX2ASnU2 z)&K3uojrCpax?1TWVY}+lI03YXWiowJab4_A2jL81S#n4u}+JZOxO}ejl$T zBJd#7W@FHQXa)7vFRcI>*1zlDn7hSIp-RwAfh$P=P8zsr5*W3w`pr;I59zDokg61N zh5M;(P?Hupv008F*nhH{{n$h5NR#F5IX>XOSe*-S1?L3c$!2o@&*ipdLa1eOLTrVB z`VL(gT>Vsa5mcK%q7B*7XO93L3b%++0fT$KY4$`@YJSv%wK2e$Qg0J*GD8SNk)Z(Z z!@mo{hbk=$%GmmJWP;})?y1D@UsIxMa${BLlsTf;V*+#Xc`1G#OlheoFfsOLbWRa5 z&D9EW#i=K(zqK=gvB>|}s_b6r1uvc>z4WsLGHY-dcb%)VJ*2NW5THL`-9NoHnpo3` zdunHVHoBFFVM*2B`Ti1$}_#dQ=aN9wpNeooBvv`HPv7-eV!MZ|l<3&>z~~-){RV za1+Ur7`QuUsdMz_Sf}4s(v}KWXL7SSAR7IzOxQ@aTe>JGXtY!|Un;wmB^8oBD`?9! zk9JHjcB058cXzmK?vM60<14E@#Rg~gFbw*zTWV{8F!oY&t>vK#K~r1OTBaB` zDnB1lXrb7uX{2>z7FSlXgc+U@(wEq2y()D7A5}W+;l<^Fb`5U(ib^}kI3Y{`ZNPF$ z-}%ZAOZjPTtbCyYMUZr{LaBQdY0_m@D)Dz!pe7rx1pg%zknp?eQ@l=(a0#5%<^O61 z7j`L|@wvpg_5MH@0`t{^H6H8wBKs)?wV1H_nZL#wQKx)c9J%nZ;qoY07U%3S9>O-~iCoG1AUEYjf}a0P%JxK>z>% literal 0 HcmV?d00001 diff --git a/doc/hydraSelectionHighlighting.png b/doc/hydraSelectionHighlighting.png new file mode 100644 index 0000000000000000000000000000000000000000..1ce4e775edbf976e5d2a2b41e0ac8d59bbed764a GIT binary patch literal 40891 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 CG-0Oz literal 0 HcmV?d00001 diff --git a/doc/images/al.png b/doc/images/al.png new file mode 100644 index 0000000000000000000000000000000000000000..56df5fab2d283bcc3e84b5b22a40129d66e52b8e GIT binary patch literal 9119 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 JDiqLR{{>y3BQ5{{ literal 0 HcmV?d00001 diff --git a/doc/images/linux.png b/doc/images/linux.png new file mode 100644 index 0000000000000000000000000000000000000000..60b130cbb5fe56c914ca2ece18fde6354e73a3d6 GIT binary patch literal 5341 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 literal 0 HcmV?d00001 diff --git a/doc/images/mac.png b/doc/images/mac.png new file mode 100644 index 0000000000000000000000000000000000000000..33fd20967c28d9fd517d3e1fcf4b98ba87d8a88a GIT binary patch literal 4051 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 literal 0 HcmV?d00001 diff --git a/doc/images/pxr.png b/doc/images/pxr.png new file mode 100644 index 0000000000000000000000000000000000000000..ceded6a49dac2de3f7f3197b4df8ad2df003adb9 GIT binary patch literal 4070 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

emuw9t-7shJ0=ohq>E8lg z=G|OhMW~qx#G}_{J(AjJR}Wpgo>ygmX+Kt&cJJ4(Q=fBnwpq!IcAo7eyo;S2J^uE_ zXh6WyY6B9<%ln?mbjg4VD!1B5__pr%@H-WO!)VQ^(73j}O@?^isdpFUp)CY|^Imn& z7IK7g!dy~+lP|Dzxpj}Kfu(IsT7s9T%yG){gtK4M`|~`!8g6wPo6+Z9aKw2{N~hCT zN-*LjD|4q4BF83;P~wV-vD&=1W0RTm$h2Acz7^WS-o7fAt?M$>y2qTSoic0cw-s*M z-uMBf8LJxmimo#r(zMobm)SebbkQd!lXcT2wt@8IyqEugcV5eq_S=)?d3 literal 0 HcmV?d00001 diff --git a/doc/images/windows.png b/doc/images/windows.png new file mode 100644 index 0000000000000000000000000000000000000000..e23f3e43b3b17a277ea040ddac23d95136d7a579 GIT binary patch literal 2957 zcmbVO2~-nz8V?F0$mxk#bPRZ)xh4sL1cHVnV1xi70;0H?Ook+oWI_@MhbZ8J3h~4) zhj@T}$hx9{E7hV1f)y)JbfutpfLav6dQ=M@wL3xJ=|1iHI`7T=uiy9k&j0sjYjB{S zwWXsa4u`WAEaQb>qZjrvCr-e=G2pU#Y_L!-Tdl?6CQUKECb+^P2OQ39Kpq;Qix38| zU{pzhB&Zl6B`Vbz8Y5VesD|KJLuxuBmh(j=s~7*S~kFfu}7 zG+xKXW0t;mL8%_2RcS}UgawS82&u^+iDGnV7$||qaO!xCVpzEZCL;<&iKuj13=587 z)iP9vYGvqWsK<^!Qh+!ANn4c4F%?=JKLK-NBp{zeYeSRN2ss4NqVXCS;U{2b zIvc%Fv$z@r(xIAA6jh8SDtOdnfXg)|h6yYTm#ZYGUhDdS48ntS2pdnOP$@(TNCfGj zAd^J_Su{5y#f?RwyoU-=i99;#Q&12CfgeG!s*ylC=>LKxFe@6>C?U*exe}5hWVK3) z2gV}F;-U&vgDJ+WqkXt9;BtdCXtZ2`9cV-Rd;kHT%K(`S29Zhvhs_lVSpt<-2dQ90 zz+>aF_>kms3Co?v6njvZVj_hJAw-%R-JJ;0s1hR8LkvO)1WOna`lvh)h2xD?Fe)#> z5uoa4WX+XvJN~cN_ila z7DOc^yjmk#T{Y=4DlTz~9P)=?0)g<*FfZh~9OA$*S787}Vn_@TTz1Q>%rZfHdqbt& z3X3guq=D9(DYZY|ZN1Zq=ZVbo_bn)BX1miKu%GB#&(ie0XT{M+UGO+wg>afj#T*r( z?(54Dm1&KjDP@J)#*{a{Ijpi6zXjjgu-2Mh|HG!8yB|-oh_-PpeHxTux@m1=9(kKO z;YZx~@V|?;`>f@+Koe~)z@bqF8h>8bgJ6n@W3F>Pr`pf%+AM0>(A>=`+g|tbZ0R6L z*7Inat~<9wU;9U{wz%afm)F3X#TPjRn`ABZO%rgc@y!zr>9eLUOwri^39csuGt;T84R#OWy8# zapQV9MQl>%o&rMO>kawJ)%HAs{NHP`+8=3J+`e_}+(OB&h!iz9S36TLi8lBe+VdB~ zQ!nuWaI)ixqb*11;c3r2ZLYP~xRdS#!TN*DOk#r^C$KF(nR;~5W4i@=>y~@$F4{Ook~K2Yd$zn}JQj@S7Ex9rnutDfux%nSCJ@9Nr{ z+IT7Y)bxaC|K~kX*@HO~mxlh!HW|2ezWu<<^z9`%r##Pn^Xu{kG{RE)!ly`Mnw6eG&Od(RSu-%dySJ-z8Xi zR1Z{z_ddHsBJukvr<107d^b?&#KRr%eij+t7>)K8I51n@^)Fc&;2(8mfY)p&cv9p1 zTTY$y^=$HU()BskQTHp4r@gicT37RS!qb~n{|)!zUsRA=xRs`Jy-mGpnhULa)ANes zuO`p#Jl9-;bgS3A$^1?gE?-ah{kmECB&WwJ86!VdakP^>dvVqWP8fZf7Sq|p?vB%0 zn~MT-J_D=f#;xZmo44AcPj!!_Ex;9mE^7bv z>cN<@nRt!Ki;Obd_R}2aW3Ia_N-YAlabLY{`gwW2D}k~vNRd-;%*6Mr4FUU97bCg? z9$PqM=No1nESDYHsLsj*UcNZk)36(W?(WHHFIsDu@+NXJ$BlC>H;6r>Lo=1kzIqEOEYV-ZTF^i{05%1{AT;X^GV%a zr3(VAkMm>frwYo3`nxAoMBKhVf6hJS9uEUNFkRb~7S}lakF2iQg*7$e#zDzY!j@Yl zb}yZ5)U8!r76#h2`&LN}>zC3mT%A0w8rXmxet@kt#%Gd3fBAsUUNP6n~BmapVdg1uP@rj4+HZ9z|YDSfn?9~ye zN^QBVbZaN4wj#_Ya;|m%v+D7=KWz+~51340wmtm4xaPrJ&)Qper}(8Lhtov~Hus7O zQc16T!VH;HpZA4jG~1O}sQb-|tly>Xj(nBV+ZBF8>9zQmyFGvF B(MayaHydraPreRender) + A -- Hydra Render --> C(MayaHydraRender) + A -- Maya Render --> D(MayaHydraPostRender) + C --> E[Composited Render] + B --> E + D --> E +``` + +### Color Management(CM) +Maya's Color Managment is applied to all applicable viewports including viewport overrides. MayaHydra plugin override inherits this behavior. So the final frame rendered out from Hydra undergoes Maya's CM. All workflows pertaining to CM within Maya remain the same so user can configure CM with MayaHydra as they would for a VP2 viewport. + +Hydra also support CM internally via Hydra Tasks. In the current version of the plugin, Hydra's CM is not applied. There are a couple of blockers for this to implemented at the moment(5th Oct 2023). USD is on OCIO v1 and Maya uses OCIO v2. There are some incompatibities with these two version and can lead to unpredictable behaviour. + +### Data Access and Interpretation + +Maya contains many different object types and features that affect how they render. To try to replicate that rendering, a plugin viewport would have to access the raw data of each object through node attributes and to reimplement the display logic. This makes supporting a wide range of object types and features too costly. + +Instead of always extracting state from the raw attributes of a Maya node, the MDataServerOperation API provides an abstract view of things to be rendered via the (pre-existing) MRenderItem API. A render item is similar to a Hydra RPrim and roughly corresponds to a single traditional 3D "draw call" in OpenGL or DirectX. It is an association of geometry buffers with an index buffer, a primitive type (triangles, lines, points), a shader, and shader parameters (transform matrices, colors, textures, etc.) The MRenderItem wraps an internal renderable object that VP2 uses internally to draw. + +One effect of the MRenderItem approach is that the plugin viewport no longer needs to care about the details of most Maya node types. For something like a Maya joint node, the original attribute approach involved writing an adapter class that manages all the attributes for the type and its interactions with other DAG objects. In the MRenderItem approach, the plugin doesn't necessarily need any joint-specific code. Instead, it receives an MRenderItem with the line geometry and color information, which is enough to render and support selection. In this way, we leverage the existing Maya viewport code that implements the "business logic" that decides what a joint looks like, while delegating rendering control to the Hydra framework. Here the Maya internal viewport code functions as a data source, serving primitives to Hydra. + +### MRenderItem Method Can be Optional + +The MRenderItem approach might not make sense in all circumstances, and can be optional. The lead case is that mesh geometry buffers might work better using the MFnMesh interface. This is the approach the original MtoH code used. The reason is that the default geometry format that the Maya viewport translation code produces has been triangulated and re-indexed so that all geometry streams share a single index buffer. But Hydra, and especially Storm, prefer untriangulated face data. For the simple default case of drawing a mesh, possibly with its full wireframe displayed (and nothing else), the value of the render item system is unclear. We can bypass the render items and provide raw face data using the MFnMesh extraction method from the original MtoH class HdMayaMeshAdapter, skipping normal buffer generation and re-indexing. This option is likely to be simpler and more efficient in the default case for meshes. When MRenderItems are needed for meshes using extra display modes, we can combine those approaches. We expect animation workflows to fit better with the raw face data approach and modelling workflows to require the MRenderItem approach. This requires more experimentation to validate these assumptions. + +We have the flexibility needed to handle other object types outside the MRenderItem system. This can also be a customization point for other custom sources of data. + +### Adapters for direct translation of Maya data to Hydra + +MayaHydra utilizes USD's PluginRegistry mechanism to translate Maya native data to Hydra. Maya meshes/shapes, camera, lights(spot,area, directional), NURBS Curves, materials and, as a special case, Arnold's SkyDomeLight are handled by these adapaters. MRenderItem described above is also handled by this plugin registry adapter mechanism. + +```mermaid +classDiagram +class Base Adapter { + <> + Shape Adapter + Mesh Adapter + Light Adapter + NURBS Curve Adapter + RenderItem Adapter + Material Adapter +} +``` +### Optional Adapter to Maya meshes + +It was mentioned earlier that the MRenderItem usage is optional. MayaHydra can be told to use native Maya mesh data instead of MRenderItem by setting the environment variable ```MAYA_HYDRA_USE_MESH_ADAPTER``` to 1. + +### Integration with MayaUSD and Other Maya Plugin Nodes Using Scene Indices + +The maya-usd plugin provides access to USD data files inside Maya by injecting that data into Maya, for Viewport 2.0 to render. It extracts that data from USD and tweaks it for compatibility with Maya features via a custom Hydra render delegate. This method doesn't work when the aim is to draw with Hydra Storm or a different render delegate, so we are exploring new methods to integrate MayaHydra with MayaUSD. MayaUSD continues to facilitate the USD data source (through the MayaUsdProxyShape Maya node) and the editing of that data. MayaHydra controls the rendering of that data. + +Experimentally, we have separated the two plugins so that they no longer link together. We use USD's HdSceneIndexPluginRegistry interface to query for a registered HdSceneIndex provider for a Maya node, using a naming convention based on the node type name. Our hope is that this can also be the entry point for third-party plugins to control the viewport rendering of custom node types (MPxLocatorNode or MPxSurfaceShape) purely through the Hydra API instead of through Maya's various viewport APIs. This could potentially replace the Maya APIs MPxSubsceneOverride, MPxDrawOverride, and MPxGeometryOverride. + +### Scene Delegate vs Scene Index +MayaHydra's long term plan is to migrate to using Scene Index completely and remove dependency on SceneDelegates. This aligns with the Pixar's recommendation as Hydra will eventually deprecate Scene Delegate support. During this transition phase, we are supporting both Scene Delegate and Scene Indices. By default, Scene Index is used but it can be overridden to use Scene Delegate using the environment variable ```MAYA_HYDRA_ENABLE_NATIVE_SCENE_INDEX``` to 0. Please note that Scene Delegate mode will not be actively supported and the Scene Index mode is the recommended one. + +### Plugin initialization, render loop and data flow through the plugin. +As described in the section above, the plugin hooks into Maya Viewport through the MRenderOverride API. + +A class diagram with some of the key members looks something like this. For sake of brevity only a few members are listed. Please refer to accompanying code for details. +```mermaid +classDiagram + + class MtohRenderOverride { + -_InitHydraResources() + -_sceneIndexRegistry: std::shared_ptr + -_hgi: HgiUniquePtr + -_hgiDriver: HdDriver + -_engine: HdEngine + -_rendererPlugin: HdRendererPlugin* + -_taskController: HdxTaskController* + -_renderDelegate: HdPluginRenderDelegateUniqueHandle + -_renderIndexProxy: std::unique_ptr + -_renderIndex: HdRenderIndex* + -_fvpSelectionTracker: Fvp::SelectionTrackerSharedPtr + -_selectionSceneIndex: Fvp::SelectionSceneIndexRefPtr + -_mayaHydraSceneProducer: std::unique_ptr + +Render(const MHWRender::MDrawContext& drawContext, const MHWRender::MDataServerOperation::MViewportScene& scene) + } +``` +The class is contains resources pertaining to both Maya and Hydra. + +We will do a high-level walk through of various execution phases, render loop and corresponding data flow. Please note the plugin is undergoing major design changes with newer APIs being introduced. The diagrams below are relevant as of 5th October 2023. + +### Plugin Load +```mermaid +sequenceDiagram + participant MayaHydra as "MayaHydra Plugin" + participant RendererPlugins as "Renderer Plugins" + participant RenderSettingsUI as "Render Settings and UI" + participant MtohRenderOverride as "MtohRenderOverride" + + MayaHydra->>RendererPlugins: Load Plugin + RendererPlugins-->>MayaHydra: Plugins Loaded + MayaHydra->>RendererPlugins: Search for Renderer Plugins + RendererPlugins-->>MayaHydra: Renderer Plugins Found + + loop for each Renderer Plugin + MayaHydra->>RenderSettingsUI: Build Render Settings and UI + RenderSettingsUI-->>MayaHydra: Render Settings and UI Built + end + + MayaHydra->>MtohRenderOverride: Register MtohRenderOverride with Maya + MtohRenderOverride-->>MayaHydra: MtohRenderOverride Registered +``` +### The Render Loop +```mermaid + +stateDiagram + [MayaFrameRefresh] --> MtohRenderOverride + MtohRenderOverride --> InitHydraResources() : Render() + + state InitHydraResources() { + [*] --> HydraResources + HydraResources --> MayaHydraSceneProducer() : RenderDelegate/RenderIndex + MayaHydraSceneProducer() --> Populate() : Creates MayaHydra specific Scene Indices internally + Populate() --> MayaHydraAdapter : Loop over Maya native nodes + MayaHydraAdapter --> SceneIndexRegistration : Flow Viewport API (WIP) to inject various Scene Indices including USD data and SelectionHighlighting + SceneIndexRegistration --> [*] + } + + state MayaHydraAdapter { + [*] --> InsertHydraPrims : Shape/Mesh/Light/Material/ArnoldDomelight + InsertHydraPrims --> [*] + } + + InitHydraResources() --> HandleCompleteViewportScene() : Loops over MRenderItems using DataServer API + state RenderItemAdapter { + [*] --> InsertHydraPrim + InsertHydraPrim --> [*] + } + HandleCompleteViewportScene() --> RenderItemAdapter : Handles VP2 updates + RenderItemAdapter --> UpdateDirtiedPrims + UpdateDirtiedPrims --> SetHydraRenderParams : Global values obtained from VP2 and Maya RenderSettings + SetHydraRenderParams --> HydraExecute() + HydraExecute() --> [MayaFrameRefresh] +``` \ No newline at end of file diff --git a/doc/selectionHighlightingArchitecture.md b/doc/selectionHighlightingArchitecture.md new file mode 100644 index 0000000000..7603f2fa34 --- /dev/null +++ b/doc/selectionHighlightingArchitecture.md @@ -0,0 +1,369 @@ +# Selection Highlighting Architecture + +Selection highlighting changes the in-viewport appearance of selected objects +or object components. + +In the following document we will refer to the software infrastructure that +supports Hydra rendering in this repository as the Flow Viewport library (name +subject to change). + +This document will describe the state of Flow Viewport library selection +highlighting as of 21-Sep-2023. + +## Behavior + +Applications maintain a set of selected objects that the user can add to and +remove from. Selected objects are usually the target of user operations, and +are shown differently in the viewport for ease of understanding. + +An application will provide a way to select an object, or to select components +of an object. For example, for a mesh object, these components may be points, +edges, or faces. At time of writing, only object selection highlighting is +supported, and selection highlighting of components is unimplemented. + +## Selection: Application versus Hydra + +The application maintains an edit-friendly version of the scene. This scene is +translated into a Hydra scene by scene indices. Correspondingly, there are two +versions of the selection, one in the application, with objects and their paths +described with application-specific classes, and a version of the selection in +the Hydra scene, described as prims and their `SdfPath`s + +## Requirements + +Requirements for selection highlighting are: + +- It must be possible to provide selection highlighting in an application that + supports multiple data models (e.g. Maya data and USD data), and plugins to + those data models (e.g. Maya plugin nodes). + +- Selected prims in the Hydra scene index tree must contain a data source + indicating their selection state. This data source is the one used by Pixar + in Hydra code, and thus is used in usdview. + +- Data injecting plugin scene indices must be able to specify their selection + highlight appearance. + +- It must be possible to let a data injecting data model provide prims to Hydra + that already contain selection highlighting. At time of writing + (19-Sep-2023), this is true of Maya native Dag data, where selection + highlighting is done by OGS. + +## Selection Highlighting Styles + +There are at least two approaches to selection highlighting: +- **Added geometry**: adding secondary geometry that indicates the selected +status of objects in the scene, e.g. wireframe or bounding box. +- **Pixed-based modified object appearance**: rendering selected objects in a +special way, e.g. object contour, modified object color, or object overlay. + +The former approach is handled by having a plugin provide a selection +highlighting filtering scene index to the Flow Viewport library, and is the +topic of this document at time of writing (20-Sep-2023). The +latter is handled by having a plugin provide a selection highlighting +task to the Flow Viewport library, and is currently unimplemented. + +## Added Geometry Plugin Software Architecture Requirements + +A selection highlighting plugin that provides added geometry to scene must +provide the following services: + +- A way to translate the application's selection path(s) into Hydra paths: + - So that the appropriate prims in Hydra can be dirtied on selection change. + - So that selected prims in Hydra can have a data source added. + This is embodied in a **Path interface**. + +- A way for the plugin to query the Hydra version of the application + selection: + - So that the plugin can add the appropriate selection highlighting + geometry on those prims that require it, e.g. a different color for the + first selected object, or selection highlighting for a complete hierarchy. + +## Sample Code +### Selection Change + +The following selection change code shows the use of the *Path Interface*, +through the *SceneIndexPath()* method, called on the input scene index. The +path interface allows the selection scene index to translate selected +application paths to selected Hydra scene index paths. +``` +void +SelectionSceneIndex::AddSelection(const Ufe::Path& appPath) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::AddSelection(const Ufe::Path& %s) called.\n", Ufe::PathString::string(appPath).c_str()); + + HdSelectionSchema::Builder selectionBuilder; + selectionBuilder.SetFullySelected( + HdRetainedTypedSampledDataSource::New(true)); + + // Call our input scene index to convert the application path to a scene + // index path. + auto sceneIndexPath = _inputSceneIndexPathInterface->SceneIndexPath(appPath); + + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg(" Adding %s to the Hydra selection.\n", sceneIndexPath.GetText()); + + _selection->pathToState[sceneIndexPath].selectionSources.push_back( + selectionBuilder.Build()); + + _SendPrimsDirtied({{sceneIndexPath, locators}}); +} +``` + +### Wireframe Selection Highlighting + +The following wireframe selection highlighting code shows the use of the +*Selection Interface*, through the *HasFullySelectedAncestorInclusive()* +method, called on the input scene index. The selection interface allows a +selection highlighting filtering scene index to query selected prims. +``` +bool WireframeSelectionHighlightSceneIndex::HasFullySelectedAncestorInclusive(const SdfPath& primPath) const +{ + return _inputSceneIndexSelectionInterface->HasFullySelectedAncestorInclusive(primPath); +} + +HdSceneIndexPrim +WireframeSelectionHighlightSceneIndex::GetPrim(const SdfPath &primPath) const +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::GetPrim(%s) called.\n", primPath.GetText()); + + auto prim = _GetInputSceneIndex()->GetPrim(primPath); + + // If this isn't one of our prims, we're not responsible for providing a + // selection highlight for it. + if (primPath.HasPrefix(_sceneRoot) && + prim.primType == HdPrimTypeTokens->mesh) { + prim.dataSource = HdOverlayContainerDataSource::New( + { prim.dataSource, HasFullySelectedAncestorInclusive(primPath) ? + sSelectedDisplayStyleDataSource : + sUnselectedDisplayStyleDataSource }); + } + return prim; + +} + +void +WireframeSelectionHighlightSceneIndex::_PrimsDirtied( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::DirtiedPrimEntries &entries) +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::_PrimsDirtied() called.\n"); + + HdSceneIndexObserver::DirtiedPrimEntries highlightEntries; + for (const auto& entry : entries) { + // If the dirtied prim isn't one of ours, we're not responsible for + // providing a selection highlight for it. + if (entry.primPath.HasPrefix(_sceneRoot) && + entry.dirtyLocators.Contains( + HdSelectionsSchema::GetDefaultLocator())) { + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg(" %s selections locator dirty.\n", entry.primPath.GetText()); + // All mesh prims recursively under the selection dirty prim have a + // dirty wireframe selection highlight. + dirtySelectionHighlightRecursive(entry.primPath, &highlightEntries); + } + } + + if (!highlightEntries.empty()) { + // Append all incoming dirty entries. + highlightEntries.reserve(highlightEntries.size()+entries.size()); + highlightEntries.insert( + highlightEntries.end(), entries.begin(), entries.end()); + _SendPrimsDirtied(highlightEntries); + } + else { + _SendPrimsDirtied(entries); + } +} +``` + +## Design Option Discussion + +- **Plugins use Hydra selection, not application selection**: selection + highlighting plugins should only deal with the Hydra view of the selection, + not the application view. Selected objects should be Hydra prims, and paths + to them described as `SdfPath`. This keeps plugins independent of any + particular application's representation of selection. + +- **Plugins access the Hydra selection through the scene index tree**: although + the selection is conceptually a singleton, we will provide access to it for + scene index selection highlighting plugins through the scene index tree, by + adding a mixin interface to the plugins. This avoids creating another object + to maintain and encapsulate the Hydra selection, since the selection scene + index is already performing this job. + +## Implementation + +### Hydra Scene and Viewport Selection Result + +The resulting wireframe selection highlighting of USD data is shown here: +![In-viewport wireframe selection highlighting](hydraSelectionHighlighting.png) + +The resulting prim selection data source is shown here: +![Prim selection data source](hydraSelectionDataSource.png) + +The resulting prim wireframe display style data source is shown here: +![Prim wireframe display style data source](hydraSelectionReprDisplayStyle.png) + +### Flow Viewport Library + +The complete implementation of selection highlighting is done in a new library +in the maya-hydra repository. The library is called `flowViewport`, under the +`lib` directory. The library supports [semantic +versioning](https:/semver.org), with classes in a `Fvp` namespace. +The `Fvp` namespace actually contains the flowViewport major version. All +flowViewport files have an `fvp` prefix. + +The `mayaHydraLib` library and the `mayaHydra` plugin both directly depend on +`flowViewport`. + +Having the selection highlighting code in a separate library promotes +reusability and enforces separation of concerns. + +### Added Geometry Selection Highlighting Through Scene Indices + +The Hydra scene index tree was chosen to implement selection highlighting +through additional geometry. This is because a scene index can inject +additional prims into the scene, modify data sources of prims in the scene, +and dirty prims in the scene whose selection status has changed. + +The scene index tree is now the following: +```mermaid +graph BT; + lgcy[Scene delegate legacy]-->hm[Hydra builtin merge]; + ph1[Plugin highlighting 1]-->hm; + subgraph ph[Plugin highlighting] + ph2[Plugin highlighting 2]-->ph1; + ph1-. Selection .->ph2; + end + sn[Selection scene index]-->ph2; + ph2-. Selection .->sn; + fvpm[Flow Viewport merge]-->sn; + subgraph pd[Plugin data] + p1[Plugin 1]; + p2[Plugin 2]; + end + fvpm-. Path .->p1; + p1-->fvpm; + p2-->fvpm; + sn-. Path .->fvpm; +``` +The plugin data and plugin highlighting subtrees are where plugins add their +scene indices. The data scene index is required, and the highlighting scene +index is optional. + +### Object Modeling + +The object modeling is the following: +- **Selection scene index**: builtin provided by the Flow Viewport library. + - Owns and encapsulates the Hydra selection. + - Translates the application selection to Hydra selection. + - Derives from and implements the selection interface. +- **Flow Viewport merging scene index**: builtin provided by the Flow Viewport + library. + - Receives data from data provider plugin scene indices. + - Forward path interface queries to plugin scene indices +- **Plugin data scene index**: provided by plugin. + - Injects plugin data into Hydra +- **Plugin selection highlighting scene index**: provided by plugin. + - Processes dirty selection notifications to dirty the appropriate prim(s) + in plugin data, including hierarchical selection highlighting + - Adds required geometry or data sources to implement selection + highlighting + +### New Scene Index Mixin Interface Base Classes + +The Flow Viewport library has two new mixin interface classes: + +- **Path Interface**: so that the builtin selection scene index can query + plugins to translate selected object application paths to selected Hydra + prim paths. The plugin provides the concrete implementation of this + interface. +- **Selection Interface**: so that the plugin selection highlighting scene + indices can query the selection scene index for selected object status. + Plugins provide a pass-through implementation, and the selection scene index + provides the implementation. + +### Implementation Classes + +- **Wireframe selection highlighting scene index**: + - Uses Hydra HdRepr to add wireframe representation to selected objects + *and their descendants*. + - Requires selected ancestor query from selection interface. + - Dirties descendants on selection dirty. +- **Render index proxy**: + - Provides encapsulated access to the builtin Flow Viewport merging scene + index. + - Other responsibilities to be determined, for future extension, possibly a + [facade design pattern](https://en.wikipedia.org/wiki/Facade_pattern). + +### Class Diagram + +```mermaid +classDiagram +class HdSingleInputFilteringSceneIndexBase +class HdMergingSceneIndex + +class SelectionInterface{ ++IsFullySelected(SdfPath) bool ++HasFullySelectedAncestorInclusive(SdfPath) bool +} + +class PathInterface{ ++SceneIndexPath(Ufe::Path) SdfPath +} + +class SelectionSceneIndex +class MergingSceneIndex +class WireframeSelectionHighlightSceneIndex + +class RenderIndexProxy{ ++MergingSceneIndex mergingSceneIndex ++InsertSceneIndex() ++RemoveSceneIndex() +} + +HdSingleInputFilteringSceneIndexBase <|-- SelectionSceneIndex +SelectionInterface <|-- SelectionSceneIndex + +HdMergingSceneIndex <|-- MergingSceneIndex +PathInterface <|-- MergingSceneIndex + +HdSingleInputFilteringSceneIndexBase <|-- WireframeSelectionHighlightSceneIndex +SelectionInterface <|-- WireframeSelectionHighlightSceneIndex + +RenderIndexProxy *-- MergingSceneIndex : Owns + +WireframeSelectionHighlightSceneIndex ..> SelectionSceneIndex : Selected +SelectionSceneIndex ..> MergingSceneIndex : Path +``` + +## Algorithmic Complexity + +- At time of writing, for an n-element selection, membership lookup is O(log n) + (map of SdfPath). Ancestor membership lookup is O(n), as we loop through + each selected path and inspect the selected path prefix. This could be much + improved (to amortized O(k), for a k-element path) through the use of a + prefix trie, such as `Ufe::Trie`. + +- At time of writing, merging scene index path lookup is O(n), for n input + scene indices. This could be improved by implementing a caching scheme based + on application path, as for a given application path prefix the same + input scene index will always provide the translation to scene index path. + +## Limitations + +- Little investigation of pixel-based selection highlighting capability. + - Needs task-based approach. + - Needs selection tracker object to make selection and data derived from + the selection available to tasks through the task context data + +- No selection highlighting across scene indices: selection state propagates + down app scene hierarchy, so that when an ancestor is selected, a + descendant's appearance may change. This can mean selection state must + propagate across scene index inputs, so that if a Maya Dag ancestor is + selected, a USD descendant's appearance can change. This is the same + situation as global transformation and visibility. diff --git a/lib/CMakeLists.txt b/lib/CMakeLists.txt new file mode 100644 index 0000000000..0688c953f6 --- /dev/null +++ b/lib/CMakeLists.txt @@ -0,0 +1,2 @@ +add_subdirectory(mayaHydra) +add_subdirectory(flowViewport) diff --git a/lib/adskHydraSceneBrowser/CMakeLists.txt b/lib/adskHydraSceneBrowser/CMakeLists.txt new file mode 100644 index 0000000000..d028c133e5 --- /dev/null +++ b/lib/adskHydraSceneBrowser/CMakeLists.txt @@ -0,0 +1,4 @@ +add_subdirectory(lib) +if (BUILD_TESTS) + add_subdirectory(test) +endif() diff --git a/lib/adskHydraSceneBrowser/lib/CMakeLists.txt b/lib/adskHydraSceneBrowser/lib/CMakeLists.txt new file mode 100644 index 0000000000..67ff7c11c8 --- /dev/null +++ b/lib/adskHydraSceneBrowser/lib/CMakeLists.txt @@ -0,0 +1,177 @@ +# ----------------------------------------------------------------------------- +# set target name +# ----------------------------------------------------------------------------- +set(TARGET_NAME adskHydraSceneBrowser) + +# ----------------------------------------------------------------------------- +# Qt safeguard +# ----------------------------------------------------------------------------- +if (NOT Qt6_FOUND) + message(WARNING "No Qt6 package found. Cannot build ${TARGET_NAME}.") + return() +endif() + +# ----------------------------------------------------------------------------- +# add library +# ----------------------------------------------------------------------------- +add_library(${TARGET_NAME} SHARED adskHydraSceneBrowserApi.h) + +# ----------------------------------------------------------------------------- +# setup sources +# ----------------------------------------------------------------------------- +set(HEADERS + dataSourceTreeWidget.h + dataSourceValueTreeView.h + registeredSceneIndexChooser.h + sceneIndexDebuggerWidget.h + sceneIndexObserverLoggingTreeView.h + sceneIndexObserverLoggingWidget.h + sceneIndexTreeWidget.h +) +set(SOURCES + dataSourceTreeWidget.cpp + dataSourceValueTreeView.cpp + registeredSceneIndexChooser.cpp + sceneIndexDebuggerWidget.cpp + sceneIndexObserverLoggingTreeView.cpp + sceneIndexObserverLoggingWidget.cpp + sceneIndexTreeWidget.cpp +) + +# Get the versioning scheme used in the file download URLs : last two digits of the year + digits of the month +# We first strip the major version (e.g. 0.23.8 -> 23.8) +string(REGEX REPLACE "^[0-9]+\\.([0-9]+\\.[0-9]+)$" "\\1" USD_minor_patch_version ${USD_VERSION}) +# We then re-insert leading 0s to single-digit numbers (e.g. 23.8 -> 23.08) +# (note that \\10 works because CMake only captures up to 9 subgroups) +string(REGEX REPLACE "(\\.|^)([1-9])(\\.|$)" "\\10\\2\\3" USD_minor_patch_version ${USD_minor_patch_version}) +set(DOWNLOAD_LOCATION "${CMAKE_CURRENT_BINARY_DIR}/src/download") +set(PATCH_LOCATION "${CMAKE_CURRENT_BINARY_DIR}/src/patch") + +macro(download file) + file(DOWNLOAD + "https://raw.githubusercontent.com/PixarAnimationStudios/OpenUSD/v${USD_minor_patch_version}/extras/imaging/examples/hdui/${file}" + "${DOWNLOAD_LOCATION}/${file}") +endmacro() + +# Insert an include directive at the top a file. +macro(prepend_include FILE_CONTENTS INCLUDE_DIRECTIVE) + set(${FILE_CONTENTS} "${INCLUDE_DIRECTIVE}\n${${FILE_CONTENTS}}") +endmacro() + +# Regex to match a valid variable or class name +set(IDENTIFIER_REGEX [a-zA-Z_][a-zA-Z_0-9]*) + +# Download and patch source files +foreach(SOURCE IN ITEMS ${SOURCES}) + download(${SOURCE}) + + file(READ "${DOWNLOAD_LOCATION}/${SOURCE}" FILE_CONTENTS) + + # We can't construct a QVariant from a C string in Qt6. What we can do however is construct a QVariant + # from a QLatin1StringView constructed with a C string. To patch this in we do the following replacements: + # "QVariant(someVariable.str().c_str())" -> "QVariant(QLatin1StringView(someVariable.str().c_str()))" + # "QVariant(someVariable.str().data())" -> "QVariant(QLatin1StringView(someVariable.str().data()))" + string(REGEX REPLACE "QVariant\\(\(${IDENTIFIER_REGEX}\)\.str\\(\\)\.\(c_str|data\)\\(\\)\\)" "QVariant(QLatin1StringView(\\1.str().\\2()))" FILE_CONTENTS "${FILE_CONTENTS}") + + # Patch in '#include ' to make sure we can use QLatin1StringView + prepend_include(FILE_CONTENTS "#include ") + + # Remove an unused lambda capture to fix a warning-treated-as-error on MacOS + string(REGEX REPLACE "\(\\[this, menu\), menuTreeWidget\(\\]\\(QTreeWidgetItem \\*item, int column\\)\)" "\\1\\2" FILE_CONTENTS "${FILE_CONTENTS}") + + file(WRITE "${PATCH_LOCATION}/${SOURCE}" "${FILE_CONTENTS}") +endforeach() + +# Download and patch header files +foreach(HEADER IN ITEMS ${HEADERS}) + download(${HEADER}) + + file(READ "${DOWNLOAD_LOCATION}/${HEADER}" FILE_CONTENTS) + + # Add DLL import/export support for Windows + # Insert 'HDUI_API' symbols for class declarations + # Note: we assume class declarations are always followed by double colons + string(REGEX REPLACE "class (${IDENTIFIER_REGEX}[^\;]:)" "class HDUI_API \\1" FILE_CONTENTS "${FILE_CONTENTS}") + + # Patch in '#include ' to have HDUI_API defined + prepend_include(FILE_CONTENTS "#include ") + + file(WRITE "${PATCH_LOCATION}/${HEADER}" "${FILE_CONTENTS}") +endforeach() + +# Prepend an absolute path to the downloaded source files, located in +# the CMAKE_CURRENT_BINARY_DIR. +# Note: the sources are downloaded for PixarAnimationStudios/OpenUSD and +# are not located in this repository's folder +list(TRANSFORM SOURCES PREPEND "${PATCH_LOCATION}/") + +# Add CMAKE_CURRENT_BINARY_DIR downloaded sources to the adskHydraSceneBrowser target +target_sources(${TARGET_NAME} + PRIVATE + ${SOURCES} +) + +# ----------------------------------------------------------------------------- +# include directories +# ----------------------------------------------------------------------------- +target_include_directories(${TARGET_NAME} + PUBLIC + ${PXR_INCLUDE_DIRS} + ${CMAKE_CURRENT_SOURCE_DIR} + ${PATCH_LOCATION} +) + +# ----------------------------------------------------------------------------- +# compiler configuration +# ----------------------------------------------------------------------------- +# QT_NO_KEYWORDS prevents Qt from defining the foreach, signals, slots and emit macros. +# this avoids overlap between Qt macros and boost, and enforces using Q_ macros. +set_target_properties(Qt6::Core PROPERTIES INTERFACE_COMPILE_DEFINITIONS QT_NO_KEYWORDS) + +target_compile_definitions(${TARGET_NAME} + PUBLIC + $<$:TBB_USE_DEBUG> + $<$:BOOST_DEBUG_PYTHON> + $<$:BOOST_LINKING_PYTHON> + PRIVATE + HDUI_EXPORT +) + +mayaHydra_compile_config(${TARGET_NAME}) # TODO : This will have to be changed when moving to a standalone repo. + +# ----------------------------------------------------------------------------- +# link libraries +# ----------------------------------------------------------------------------- +target_link_libraries(${TARGET_NAME} + PUBLIC + usd + usdImaging + Qt6::Core + Qt6::Widgets +) + +# ----------------------------------------------------------------------------- +# Qt file processing (moc) +# ----------------------------------------------------------------------------- +set_property(TARGET adskHydraSceneBrowser PROPERTY AUTOMOC ON) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(TARGETS ${TARGET_NAME} + LIBRARY + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib + RUNTIME + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib +) + +list(TRANSFORM HEADERS PREPEND "${PATCH_LOCATION}/") +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/adskHydraSceneBrowser +) + +if(IS_WINDOWS) + install(FILES $ + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib OPTIONAL) +endif() diff --git a/lib/adskHydraSceneBrowser/lib/adskHydraSceneBrowserApi.h b/lib/adskHydraSceneBrowser/lib/adskHydraSceneBrowserApi.h new file mode 100644 index 0000000000..4a066d1b3f --- /dev/null +++ b/lib/adskHydraSceneBrowser/lib/adskHydraSceneBrowserApi.h @@ -0,0 +1,39 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#if defined _WIN32 || defined __CYGWIN__ + +// The main export symbol used for the UI library. +#ifdef HDUI_EXPORT +#ifdef __GNUC__ +#define HDUI_API __attribute__((dllexport)) +#else +#define HDUI_API __declspec(dllexport) +#endif +#else +#ifdef __GNUC__ +#define HDUI_API __attribute__((dllimport)) +#else +#define HDUI_API __declspec(dllimport) +#endif +#endif + +#else +#if __GNUC__ >= 4 +#define HDUI_API __attribute__((visibility("default"))) +#else +#define HDUI_API +#endif +#endif diff --git a/lib/adskHydraSceneBrowser/test/CMakeLists.txt b/lib/adskHydraSceneBrowser/test/CMakeLists.txt new file mode 100644 index 0000000000..1895d92e6b --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/CMakeLists.txt @@ -0,0 +1,106 @@ +# ----------------------------------------------------------------------------- +# set target name +# ----------------------------------------------------------------------------- +set(TARGET_NAME adskHydraSceneBrowserTesting) + +# ----------------------------------------------------------------------------- +# Qt safeguard +# ----------------------------------------------------------------------------- +if (NOT Qt6_FOUND) + message(WARNING "No Qt6 package found. Cannot build ${TARGET_NAME}.") + return() +endif() + +# ----------------------------------------------------------------------------- +# prerequisites +# ----------------------------------------------------------------------------- +find_package(GTest REQUIRED) + +# ----------------------------------------------------------------------------- +# add library +# ----------------------------------------------------------------------------- +add_library(${TARGET_NAME} SHARED) + +# ----------------------------------------------------------------------------- +# file lists +# ----------------------------------------------------------------------------- +set(SHIPPED_HEADERS + adskHydraSceneBrowserTesting.h +) + +set(SOURCES + adskHydraSceneBrowserTestFixture.cpp + adskHydraSceneBrowserTesting.cpp +) + +# ----------------------------------------------------------------------------- +# compiler configuration +# ----------------------------------------------------------------------------- +mayaHydra_compile_config(${TARGET_NAME}) # TODO : This will have to be changed when moving to a standalone repo. + +target_compile_definitions(${TARGET_NAME} + PUBLIC + $<$:LINUX> + $<$:OSMac_> + $<$:GTEST_LINKED_AS_SHARED_LIBRARY> + $<$:TBB_USE_DEBUG> + PRIVATE + HDUITEST_EXPORT +) + +# ----------------------------------------------------------------------------- +# include directories +# ----------------------------------------------------------------------------- +target_include_directories(${TARGET_NAME} + PUBLIC + ${CMAKE_CURRENT_SOURCE_DIR} + ${GTEST_INCLUDE_DIRS} +) + +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + ${SOURCES} +) + +# ----------------------------------------------------------------------------- +# link libraries +# ----------------------------------------------------------------------------- +target_link_libraries(${TARGET_NAME} + PUBLIC + adskHydraSceneBrowser + ${GTEST_LIBRARIES} +) + +# ----------------------------------------------------------------------------- +# runtime search paths +# ----------------------------------------------------------------------------- +# TODO : This will have to be changed when moving to a standalone repo. +if(IS_MACOSX OR IS_LINUX) + mayaUsd_init_rpath(rpath "lib") + mayaUsd_add_rpath(rpath "${CMAKE_INSTALL_PREFIX}/lib/gtest") + mayaUsd_install_rpath(rpath ${TARGET_NAME}) +endif() + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(TARGETS ${TARGET_NAME} + LIBRARY + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib + RUNTIME + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib +) + +list(TRANSFORM SHIPPED_HEADERS PREPEND "${CMAKE_CURRENT_SOURCE_DIR}/") +install(FILES ${SHIPPED_HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/adskHydraSceneBrowserTesting +) + +if(IS_WINDOWS) + install(FILES $ + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib OPTIONAL) +endif() diff --git a/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestApi.h b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestApi.h new file mode 100644 index 0000000000..7b3c407f4f --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestApi.h @@ -0,0 +1,39 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#if defined _WIN32 || defined __CYGWIN__ + +// The main export symbol used for the UI test library. +#ifdef HDUITEST_EXPORT +#ifdef __GNUC__ +#define HDUITEST_API __attribute__((dllexport)) +#else +#define HDUITEST_API __declspec(dllexport) +#endif +#else +#ifdef __GNUC__ +#define HDUITEST_API __attribute__((dllimport)) +#else +#define HDUITEST_API __declspec(dllimport) +#endif +#endif + +#else +#if __GNUC__ >= 4 +#define HDUITEST_API __attribute__((visibility("default"))) +#else +#define HDUITEST_API +#endif +#endif diff --git a/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.cpp b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.cpp new file mode 100644 index 0000000000..aadf7da4e1 --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.cpp @@ -0,0 +1,323 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "adskHydraSceneBrowserTestFixture.h" + +#include "dataSourceTreeWidget.h" +#include "dataSourceValueTreeView.h" +#include "sceneIndexTreeWidget.h" + +#include +#include +#include +#include +#include +#include + +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +template ChildType* FindFirstChild(QObject* qObject) +{ + for (QObject* child : qObject->children()) { + ChildType* castChild = qobject_cast(child); + if (castChild) { + return castChild; + } + } + return nullptr; +} + +QTreeWidgetItemIterator GetIteratorForTree(QTreeWidget* treeWidget) +{ + // Expand all items so the iterator can traverse them + treeWidget->expandAll(); + // Immediately process queued events, otherwise some events might linger and lead to a crash + // trying to access since-deleted items once the Qt event loop resumes and processes the events. + // (e.g. without this there is a crash involving a setExpanded() call) + QApplication::processEvents(QEventLoop::ProcessEventsFlag::EventLoopExec); + return QTreeWidgetItemIterator(treeWidget); +} + +pxr::HdSceneIndexBasePtr AdskHydraSceneBrowserTestFixture::sceneIndex = nullptr; + +void AdskHydraSceneBrowserTestFixture::SetUp() +{ + ASSERT_NE(sceneIndex, nullptr); + + _sceneBrowserWidget->setWindowTitle("Test Hydra Scene Browser"); + _sceneBrowserWidget->SetSceneIndex("", sceneIndex, true); + _sceneBrowserWidget->show(); + + QSplitter* sceneBrowserSplitter = FindFirstChild(_sceneBrowserWidget.get()); + ASSERT_NE(sceneBrowserSplitter, nullptr); + + _primHierarchyWidget = FindFirstChild(sceneBrowserSplitter); + ASSERT_NE(_primHierarchyWidget, nullptr); + _dataSourceHierarchyWidget + = FindFirstChild(sceneBrowserSplitter); + ASSERT_NE(_dataSourceHierarchyWidget, nullptr); + _dataSourceValueView = FindFirstChild(sceneBrowserSplitter); + ASSERT_NE(_dataSourceValueView, nullptr); +} + +void AdskHydraSceneBrowserTestFixture::TearDown() { _sceneBrowserWidget->close(); } + +void AdskHydraSceneBrowserTestFixture::SetReferenceSceneIndex( + pxr::HdSceneIndexBasePtr referenceSceneIndex) +{ + sceneIndex = referenceSceneIndex; +} + +void AdskHydraSceneBrowserTestFixture::ComparePrimHierarchy( + bool compareDataSourceHierarchy, + bool compareDataSourceValues) +{ + // Setup traversal data structures (depth-first search) + QTreeWidgetItemIterator itPrimsTreeWidget = GetIteratorForTree(_primHierarchyWidget); + std::stack primPathsStack({ pxr::SdfPath::AbsoluteRootPath() }); + + // Traverse hierarchy and compare (depth-first search) + while (*itPrimsTreeWidget && !primPathsStack.empty()) { + // Get the objects for the current step + QTreeWidgetItem* primQtItem = *itPrimsTreeWidget; + pxr::SdfPath primPath = primPathsStack.top(); + + // Compare prim name + std::string actualPrimName = primQtItem->text(0).toStdString(); + // SdfPath::GetElementString returns an empty string if the path is the the absolute root + // (/), as it is not considered to be an element. However, the browser does displays it as + // "/". + std::string expectedPrimName + = primPath.IsAbsoluteRootPath() ? "/" : primPath.GetElementString(); + EXPECT_EQ(actualPrimName, expectedPrimName); + + // Compare prim type + pxr::HdSceneIndexPrim prim = sceneIndex->GetPrim(primPath); + if (primQtItem->columnCount() > 1) { + std::string actualPrimType = primQtItem->text(1).toStdString(); + std::string expectedPrimType = prim.primType; + EXPECT_EQ(actualPrimType, expectedPrimType); + } else { + // In this case, the Qt prim item only has a column for its name, + // so we at least make sure the prim type is empty. + // So far it seems this case only happens for the root path. + EXPECT_EQ(prim.primType, pxr::TfToken()) + << "Prim had a non-empty type but its Qt item had no column for it."; + } + + // Compare data source + if (compareDataSourceHierarchy) { + _primHierarchyWidget->setCurrentItem(primQtItem); + CompareDataSourceHierarchy( + { primPath.GetElementToken(), prim.dataSource }, compareDataSourceValues); + } + + // Prepare next step (need to pop the stack before pushing the next elements) + itPrimsTreeWidget++; + primPathsStack.pop(); + + // Push child paths on the stack + pxr::SdfPathVector childPaths = sceneIndex->GetChildPrimPaths(primPath); + for (auto itChildPaths = childPaths.rbegin(); itChildPaths != childPaths.rend(); + itChildPaths++) { + primPathsStack.push(*itChildPaths); + } + } +} + +void AdskHydraSceneBrowserTestFixture::CompareDataSourceHierarchy( + DataSourceEntry rootDataSourceEntry, + bool compareValues) +{ + // Setup traversal data structures (depth-first search) + QTreeWidgetItemIterator itDataSourceTreeWidget = GetIteratorForTree(_dataSourceHierarchyWidget); + std::stack dataSourceStack({ rootDataSourceEntry }); + + // Traverse hierarchy and compare (depth-first search) + while (*itDataSourceTreeWidget && !dataSourceStack.empty()) { + // Get the objects for the current step + QTreeWidgetItem* dataSourceQtItem = *itDataSourceTreeWidget; + DataSourceEntry dataSourceEntry = dataSourceStack.top(); + + // Compare data source name + std::string actualDataSourceName = dataSourceQtItem->text(0).toStdString(); + std::string expectedDataSourceName = dataSourceEntry.name; + EXPECT_EQ(actualDataSourceName, expectedDataSourceName); + + // Compare data source value + if (compareValues) { + _dataSourceHierarchyWidget->setCurrentItem(dataSourceQtItem); + if (auto sampledDataSource + = pxr::HdSampledDataSource::Cast(dataSourceEntry.dataSource)) { + CompareDataSourceValue(sampledDataSource); + } + } + + // Prepare next step (need to pop the stack before pushing the next elements) + itDataSourceTreeWidget++; + dataSourceStack.pop(); + + // Push child data sources on the stack + if (auto containerDataSource + = pxr::HdContainerDataSource::Cast(dataSourceEntry.dataSource)) { + pxr::TfTokenVector childNames = containerDataSource->GetNames(); + for (auto itChildNames = childNames.rbegin(); itChildNames != childNames.rend(); + itChildNames++) { + pxr::TfToken dataSourceName = *itChildNames; + pxr::HdDataSourceBaseHandle dataSource = containerDataSource->Get(dataSourceName); + if (dataSource) { + dataSourceStack.push({ dataSourceName, dataSource }); + } + } + } else if ( + auto vectorDataSource = pxr::HdVectorDataSource::Cast(dataSourceEntry.dataSource)) { + for (size_t iElement = 0; iElement < vectorDataSource->GetNumElements(); iElement++) { + size_t reversedElementIndex = vectorDataSource->GetNumElements() - 1 - iElement; + pxr::TfToken dataSourceName = pxr::TfToken(std::to_string(reversedElementIndex)); + pxr::HdDataSourceBaseHandle dataSource + = vectorDataSource->GetElement(reversedElementIndex); + if (dataSource) { + dataSourceStack.push({ dataSourceName, dataSource }); + } + } + } + } +} + +void AdskHydraSceneBrowserTestFixture::CompareDataSourceValue( + pxr::HdSampledDataSourceHandle sampledDataSource) +{ + _dataSourceValueView->expandAll(); + + pxr::VtValue value = sampledDataSource->GetValue(0.0f); + + // The supported value types can be found in dataSourceValueTreeView.cpp, in the + // Hdui_GetModelFromValue function. + if (!value.IsArrayValued()) { + CompareValueContent(value); + } else { + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + CompareIfArray(value); + } +} + +bool AdskHydraSceneBrowserTestFixture::MatchesFallbackTextOutput(const std::string& text) { + // Regex for matching the fallback text output used for types that don't provide a custom one. + // Identifies a literal <', followed by a valid C++ type name (possibly templated), then a literal ', + // then a space, an @ symbol and another space, a hexadecimal 32 to 64 bit address (case-insensitive, + // potentially prefixed with 0x), and finally a literal >. Example matches : + // <'ArResolverContext' @ 0x251ffa80> // Linux + // <'ArResolverContext' @ 000001D3A4296670> // Windows + // <'vector >' @ 0x261b8c20> // Linux + // <'vector >' @ 000001D49F3390B0> // Windows + + // The space in the second [] group is intentional and must be matched, + // see the above templated examples + std::regex fallbackTextOutputRegex("<'[a-zA-Z_][ a-zA-Z_0-9<>,&*()]*' @ (0x)?[0-9a-fA-F]{8,16}>"); + return std::regex_match(text, fallbackTextOutputRegex); +} + +void AdskHydraSceneBrowserTestFixture::CompareValueContent(const pxr::VtValue& value) +{ + QAbstractItemModel* dataSourceItemModel = _dataSourceValueView->model(); + EXPECT_EQ(dataSourceItemModel->rowCount(), 1); + + QModelIndex valueIndex = dataSourceItemModel->index(0, 0); + QVariant valueData = dataSourceItemModel->data(valueIndex, Qt::DisplayRole); + QString valueText = valueData.toString(); + std::string actualValue = valueText.toStdString(); + + std::ostringstream valueStream; + valueStream << value; + std::string expectedValue = valueStream.str(); + + if (!MatchesFallbackTextOutput(expectedValue)) { + // Happy path : the concrete type of the VtValue supports text output. + // (this is an assumption and not a truly reliable check; see the not-so-happy path + // for more details) + EXPECT_EQ(actualValue, expectedValue); + } else { + // Not-so-happy path : the concrete type of the VtValue does not support text output. + // + // If a type does not provide custom text output, this will cause Vt_StreamOutGeneric to be + // called by Vt_StreamOutImpl (see streamOut.h/.cpp). This function outputs the name of the + // concrete type followed by the address of the held object. Example outputs : + // <'ArResolverContext' @ 0x251ffa80> + // <'ArResolverContext' @ 000001D3A4296670> + // + // Since it is possible that some data sources return a copy of their underlying object when + // calling GetValue(), the object held by the VtValue passed in as the parameter to this + // function may differ from the one held by the VtValue used by the scene browser. In such + // cases, the printed addresses won't match and the test will fail. + // + // This workaround instead compares the values only up to their type name in these cases. + // The regex check could technically prevent fully comparing values if their custom text + // output perfectly matches the regex, but this seems very unlikely. + std::string valueTypeName = value.GetTypeName(); + size_t indexOfTypeName = expectedValue.find(valueTypeName); + size_t substringComparisonLength = indexOfTypeName + valueTypeName.size(); + EXPECT_EQ( + actualValue.substr(0, substringComparisonLength), + expectedValue.substr(0, substringComparisonLength)); + } +} + +template +void AdskHydraSceneBrowserTestFixture::CompareIfArray(const pxr::VtValue& value) +{ + if (value.IsHolding>()) { + CompareArrayContents(value.UncheckedGet>()); + } +} + +template +void AdskHydraSceneBrowserTestFixture::CompareArrayContents( + const pxr::VtArray& vtArray) +{ + QAbstractItemModel* dataSourceItemModel = _dataSourceValueView->model(); + EXPECT_EQ(static_cast(dataSourceItemModel->rowCount()), vtArray.size()); + + size_t nbRowsToTraverse + = std::min(static_cast(dataSourceItemModel->rowCount()), vtArray.size()); + for (size_t iRow = 0; iRow < nbRowsToTraverse; iRow++) { + QModelIndex valueIndex = dataSourceItemModel->index(iRow, 0); + QVariant valueData = dataSourceItemModel->data(valueIndex, Qt::DisplayRole); + QString valueText = valueData.toString(); + std::string actualValue = valueText.toStdString(); + + std::ostringstream valueStream; + valueStream << vtArray.cdata()[iRow]; + std::string expectedValue = valueStream.str(); + + EXPECT_EQ(actualValue, expectedValue); + } +} diff --git a/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.h b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.h new file mode 100644 index 0000000000..df0b285521 --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTestFixture.h @@ -0,0 +1,79 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef ADSK_HYDRA_SCENE_BROWSER_TEST_FIXTURE_H +#define ADSK_HYDRA_SCENE_BROWSER_TEST_FIXTURE_H + +#include +#include +#include +#include +#include +#include +#include + +#include + +#include +#include +#include +#include + +struct DataSourceEntry +{ + pxr::TfToken name; + pxr::HdDataSourceBaseHandle dataSource; +}; + +class AdskHydraSceneBrowserTestFixture : public ::testing::Test +{ +public: + AdskHydraSceneBrowserTestFixture() = default; + ~AdskHydraSceneBrowserTestFixture() override = default; + + void SetUp() override; + void TearDown() override; + + static void SetReferenceSceneIndex(pxr::HdSceneIndexBasePtr referenceSceneIndex); + +protected: + void ComparePrimHierarchy( + bool compareDataSourceHierarchy = false, + bool compareDataSourceValues = false); + + void + CompareDataSourceHierarchy(DataSourceEntry rootDataSourceEntry, bool compareValues = false); + + void CompareDataSourceValue(pxr::HdSampledDataSourceHandle sampledDataSource); + + bool MatchesFallbackTextOutput(const std::string& text); + + void CompareValueContent(const pxr::VtValue& value); + + template void CompareIfArray(const pxr::VtValue& value); + + template + void CompareArrayContents(const pxr::VtArray& vtArray); + + std::unique_ptr _sceneBrowserWidget + = std::make_unique(); + pxr::HduiSceneIndexTreeWidget* _primHierarchyWidget = nullptr; + pxr::HduiDataSourceTreeWidget* _dataSourceHierarchyWidget = nullptr; + pxr::HduiDataSourceValueTreeView* _dataSourceValueView = nullptr; + + static pxr::HdSceneIndexBasePtr sceneIndex; +}; + +#endif // ADSK_HYDRA_SCENE_BROWSER_TEST_FIXTURE_H diff --git a/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.cpp b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.cpp new file mode 100644 index 0000000000..93ff0cf277 --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.cpp @@ -0,0 +1,54 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "adskHydraSceneBrowserTesting.h" + +#include "adskHydraSceneBrowserTestFixture.h" + +#include + +bool RunTestsWithFilter(std::string filter) +{ + // Create GoogleTest arguments + int argc = 1; + std::vector argv(argc); + argv[0] = const_cast("AdskHydraSceneBrowserTesting"); + + // Setup GoogleTest + ::testing::GTEST_FLAG(filter) = filter; + ::testing::InitGoogleTest(&argc, argv.data()); + + // Run the tests + int testsResult = RUN_ALL_TESTS(); + + // Return pass/fail status (testsResult == 0 if all tests passed, 1 otherwise) + return testsResult == 0; +} + +namespace AdskHydraSceneBrowserTesting { +bool RunFullSceneIndexComparisonTest(pxr::HdSceneIndexBasePtr referenceSceneIndex) +{ + AdskHydraSceneBrowserTestFixture::SetReferenceSceneIndex(referenceSceneIndex); + return RunTestsWithFilter("AdskHydraSceneBrowserTestFixture.FullSceneIndexComparison"); +} +} // namespace AdskHydraSceneBrowserTesting + +TEST_F(AdskHydraSceneBrowserTestFixture, FullSceneIndexComparison) +{ + // We want to do a full comparison, so also compare both data sources hierarchy and values + bool compareDataSourceHierarchy = true; + bool compareDataSourceValues = true; + ComparePrimHierarchy(compareDataSourceHierarchy, compareDataSourceValues); +} diff --git a/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.h b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.h new file mode 100644 index 0000000000..41047bcc3d --- /dev/null +++ b/lib/adskHydraSceneBrowser/test/adskHydraSceneBrowserTesting.h @@ -0,0 +1,28 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef ADSK_HYDRA_SCENE_BROWSER_TESTING_H +#define ADSK_HYDRA_SCENE_BROWSER_TESTING_H + +#include "adskHydraSceneBrowserTestApi.h" + +#include + +namespace AdskHydraSceneBrowserTesting { +HDUITEST_API +bool RunFullSceneIndexComparisonTest(pxr::HdSceneIndexBasePtr referenceSceneIndex); +} // namespace AdskHydraSceneBrowserTesting + +#endif // ADSK_HYDRA_SCENE_BROWSER_TESTING_H diff --git a/lib/flowViewport/API/CMakeLists.txt b/lib/flowViewport/API/CMakeLists.txt new file mode 100644 index 0000000000..2df5b785c5 --- /dev/null +++ b/lib/flowViewport/API/CMakeLists.txt @@ -0,0 +1,29 @@ +set(HEADERS + fvpSelectionClient.h + fvpFlowSelectionInterface.h + fvpVersionInterface.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/API +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/API +) + +# ----------------------------------------------------------------------------- +# subdirectories +# ----------------------------------------------------------------------------- +add_subdirectory(interfacesImp) +add_subdirectory(samples) diff --git a/lib/flowViewport/API/fvpFlowSelectionInterface.h b/lib/flowViewport/API/fvpFlowSelectionInterface.h new file mode 100644 index 0000000000..8140deaa22 --- /dev/null +++ b/lib/flowViewport/API/fvpFlowSelectionInterface.h @@ -0,0 +1,59 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef FLOW_VIEWPORT_API_SELECTION_INTERFACE_H +#define FLOW_VIEWPORT_API_SELECTION_INTERFACE_H + +#include "flowViewport/api.h" + +namespace FVP_NS_DEF +{ + class SelectionClient;//Predeclaration + + /** + * FlowSelectionInterface is used to register/unregister selection callbacks SelectionClient for an Hydra viewport. + * To get an instance of the FlowSelectionInterface class, please use : + * Fvp::FlowSelectionInterface& flowSelectionInterface = Fvp::FlowSelectionInterface::Get(); + */ + class FlowSelectionInterface + { + public: + + /** + * @brief Access to this interface + * + * @return A reference on the FlowSelectionInterface + */ + static FVP_API FlowSelectionInterface& Get(); + + /** + * @brief Register a callbacks SelectionClient instance + * + * @param[in] client is the SelectionClient instance you are registering. + */ + virtual void RegisterSelectionClient(SelectionClient& client) = 0; + + /** + * @brief Unregister a callbacks SelectionClient instance + * + * @param[in] client is an SelectionClient instance you have previously registered and want to unregister. + */ + virtual void UnregisterSelectionClient(SelectionClient& client)= 0; + }; + +}//end of namespace + +#endif //FLOW_VIEWPORT_API_SELECTION_INTERFACE_H diff --git a/lib/flowViewport/API/fvpSelectionClient.h b/lib/flowViewport/API/fvpSelectionClient.h new file mode 100644 index 0000000000..6166578a4e --- /dev/null +++ b/lib/flowViewport/API/fvpSelectionClient.h @@ -0,0 +1,50 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef FLOW_VIEWPORT_API_SELECTIONCLIENT_H +#define FLOW_VIEWPORT_API_SELECTIONCLIENT_H + +#include "flowViewport/api.h" + +#include +#include + +namespace FVP_NS_DEF +{ + /** + * SelectionClient is the definition of selection callbacks for an Hydra viewport. + * Subclass this class to register an instance of SelectionClient with Fvp::FlowSelectionInterface::RegisterSelectionClient + */ + class FVP_API SelectionClient + { + public: + + /** + * @brief Is a dummy selection callback as a placeholder + * + */ + virtual void DummySelectionCallback() = 0; + + /// Destructor + virtual ~SelectionClient() = default; + }; + + ///Array of SelectionClient + using SelectionClientSet = std::set; + +}//end of namespace FVP_NS_DEF + +#endif //FLOW_VIEWPORT_API_SELECTIONCLIENT_H diff --git a/lib/flowViewport/API/fvpVersionInterface.h b/lib/flowViewport/API/fvpVersionInterface.h new file mode 100644 index 0000000000..2ccdaaf067 --- /dev/null +++ b/lib/flowViewport/API/fvpVersionInterface.h @@ -0,0 +1,49 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef FLOW_VIEWPORT_API_VERSION_INTERFACE_H +#define FLOW_VIEWPORT_API_VERSION_INTERFACE_H + +#include "flowViewport/api.h" + +namespace FVP_NS_DEF +{ + /** + * VersionInterface is used to get the version of the Flow Viewport API. + * + * To get an instance of the VersionInterface class, please use : + * Fvp::VersionInterface& versionInterface = Fvp::VersionInterface::Get(); + */ + class VersionInterface + { + public: + static FVP_API VersionInterface& Get(); + + /** + * @brief Get the version of the flow viewport API. + * + * Get the version of the flow viewport API so you can handle different versions in your plugin if it changes. + * + * @param[out] majorVersion is the major version number. + * @param[out] minorVersion is the minor version number. + * @param[out] patchLevel is the patch level number. + */ + virtual void GetVersion(int& majorVersion, int& minorVersion, int& patchLevel) = 0; + }; + +}//end of namespace + +#endif //FLOW_VIEWPORT_API_VERSION_INTERFACE_H diff --git a/lib/flowViewport/API/interfacesImp/CMakeLists.txt b/lib/flowViewport/API/interfacesImp/CMakeLists.txt new file mode 100644 index 0000000000..5bd63fedbc --- /dev/null +++ b/lib/flowViewport/API/interfacesImp/CMakeLists.txt @@ -0,0 +1,31 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + fvpSelectionInterfaceImp.cpp + fvpVersionInterfaceImp.cpp +) + +set(HEADERS + fvpSelectionInterfaceImp.h + fvpVersionInterfaceImp.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/API/interfacesImp +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/API/interfacesImp +) diff --git a/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.cpp b/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.cpp new file mode 100644 index 0000000000..6e73ee50f2 --- /dev/null +++ b/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.cpp @@ -0,0 +1,77 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +//Local headers +#include "fvpSelectionInterfaceImp.h" + +//STL +#include + +namespace{ + static std::mutex _viewportSelectClient_mutex; +} + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +static SelectionInterfaceImp theInterface; + +SelectionClientSet _viewportSelectionClients; + +FlowSelectionInterface& FlowSelectionInterface::Get() +{ + return theInterface; +} + +SelectionInterfaceImp& SelectionInterfaceImp::Get() +{ + return theInterface; +} + +void SelectionInterfaceImp::RegisterSelectionClient(SelectionClient& client) +{ + std::lock_guard lock(_viewportSelectClient_mutex); + + auto foundResult = _viewportSelectionClients.find(&client); + if (foundResult == _viewportSelectionClients.cend()){ + _viewportSelectionClients.insert(&client); + } +} + +void SelectionInterfaceImp::UnregisterSelectionClient(SelectionClient& client) +{ + std::lock_guard lock(_viewportSelectClient_mutex); + + auto foundResult = _viewportSelectionClients.find(&client); + if (foundResult != _viewportSelectionClients.end()){ + _viewportSelectionClients.erase(foundResult); + } +} + +void SelectionInterfaceImp::DummySelectionCallback() +{ + std::lock_guard lock(_viewportSelectClient_mutex); + + for (auto _selectionClient : _viewportSelectionClients) { + if (_selectionClient){ + _selectionClient->DummySelectionCallback(); + } + } +} + +} //End of namespace FVP_NS_DEF + diff --git a/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.h b/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.h new file mode 100644 index 0000000000..2bba24c526 --- /dev/null +++ b/lib/flowViewport/API/interfacesImp/fvpSelectionInterfaceImp.h @@ -0,0 +1,47 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef FLOW_VIEWPORT_API_INTERFACESIMP_SELECTIONINTERFACEIMP_H +#define FLOW_VIEWPORT_API_INTERFACESIMP_SELECTIONINTERFACEIMP_H + +//Local headers +#include +#include +#include + +namespace FVP_NS_DEF { + +///Is a singleton, use SelectionInterfaceImp& selInterfaceImp = SelectionInterfaceImp::Get() to get an instance of that interface +class SelectionInterfaceImp : public FVP_NS_DEF::FlowSelectionInterface +{ +public: + SelectionInterfaceImp() = default; + virtual ~SelectionInterfaceImp() = default; + + ///Interface accessor + static SelectionInterfaceImp& Get(); + + //From FVP_NS_DEF::SelectionInterface + void RegisterSelectionClient(SelectionClient& client)override; + void UnregisterSelectionClient(SelectionClient& client)override; + + //To be called by maya-hydra + void DummySelectionCallback(); +}; + +} //End of namespace FVP_NS_DEF + +#endif // FLOW_VIEWPORT_API_INTERFACESIMP_SELECTIONINTERFACEIMP_H \ No newline at end of file diff --git a/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.cpp b/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.cpp new file mode 100644 index 0000000000..398fbecbf3 --- /dev/null +++ b/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.cpp @@ -0,0 +1,48 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +//Local headers +#include "fvpVersionInterfaceImp.h" + +namespace FVP_NS_DEF { + +//Interface version +#define FLOW_VIEWPORT_API_MAJOR_VERSION 0 +#define FLOW_VIEWPORT_API_MINOR_VERSION 1 +#define FLOW_VIEWPORT_API_PATCH_LEVEL 0 + +static VersionInterfaceImp theInterface; + +VersionInterface& VersionInterface::Get() +{ + return theInterface; +} + +VersionInterfaceImp& VersionInterfaceImp::Get() +{ + return theInterface; +} + +///////////////////////////////// SelectionInterfaceImp //////////////////// +void VersionInterfaceImp::GetVersion(int& majorVersion, int& minorVersion, int&patchLevel) +{ + majorVersion = FLOW_VIEWPORT_API_MAJOR_VERSION; + minorVersion = FLOW_VIEWPORT_API_MINOR_VERSION; + patchLevel = FLOW_VIEWPORT_API_PATCH_LEVEL; +} + +} //End of namespace FVP_NS_DEF + diff --git a/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.h b/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.h new file mode 100644 index 0000000000..7f508bb06c --- /dev/null +++ b/lib/flowViewport/API/interfacesImp/fvpVersionInterfaceImp.h @@ -0,0 +1,42 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef FLOW_VIEWPORT_API_INTERFACESIMP_VERSIONINTERFACE_IMP_H +#define FLOW_VIEWPORT_API_INTERFACESIMP_VERSIONINTERFACE_IMP_H + +//Local headers +#include +#include + +namespace FVP_NS_DEF { + +///Is a singleton, use VersionInterfaceImp& versionInterface = VersionInterfaceImp::Get() to get an instance of that interface +class VersionInterfaceImp : public FVP_NS_DEF::VersionInterface +{ +public: + VersionInterfaceImp() = default; + virtual ~VersionInterfaceImp() = default; + + ///Interface accessor + static VersionInterfaceImp& Get(); + + //From FVP_NS_DEF::VersionInterface + void GetVersion(int& majorVersion, int& minorVersion, int& patchLevel)override; +}; + +} //End of namespace FVP_NS_DEF + +#endif // FLOW_VIEWPORT_API_INTERFACESIMP_VERSIONINTERFACE_IMP_H \ No newline at end of file diff --git a/lib/flowViewport/API/samples/CMakeLists.txt b/lib/flowViewport/API/samples/CMakeLists.txt new file mode 100644 index 0000000000..3b50d4ee66 --- /dev/null +++ b/lib/flowViewport/API/samples/CMakeLists.txt @@ -0,0 +1,29 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + fvpSelectionClientExample.cpp +) + +set(HEADERS + fvpSelectionClientExample.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/API/samples +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/API/samples +) diff --git a/lib/flowViewport/API/samples/fvpSelectionClientExample.cpp b/lib/flowViewport/API/samples/fvpSelectionClientExample.cpp new file mode 100644 index 0000000000..9545a8ddbe --- /dev/null +++ b/lib/flowViewport/API/samples/fvpSelectionClientExample.cpp @@ -0,0 +1,33 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +//Local headers +#include "fvpSelectionClientExample.h" + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +SelectionClientExample::~SelectionClientExample() +{ +} + +void SelectionClientExample::DummySelectionCallback() +{ + +} + +} //end of namespace FVP_NS_DEF \ No newline at end of file diff --git a/lib/flowViewport/API/samples/fvpSelectionClientExample.h b/lib/flowViewport/API/samples/fvpSelectionClientExample.h new file mode 100644 index 0000000000..5a3ad4df10 --- /dev/null +++ b/lib/flowViewport/API/samples/fvpSelectionClientExample.h @@ -0,0 +1,48 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// +#ifndef FLOW_VIEWPORT_API_SAMPLES_SELECTIONCLIENTEXAMPLE_H +#define FLOW_VIEWPORT_API_SAMPLES_SELECTIONCLIENTEXAMPLE_H + +//Local headers +#include "flowViewport/api.h" +#include "flowViewport/API/fvpSelectionClient.h" +#include "flowViewport/API/fvpFlowSelectionInterface.h" + +namespace FVP_NS_DEF { + +///Implementation of an SelectionClient which is the way to communicate with our Hydra viewport plugin to deal with selection +class FVP_API SelectionClientExample : public SelectionClient +{ +public: + SelectionClientExample() = default; + ~SelectionClientExample() override; + + ///Set the hydra interface + void SetHydraInterface(FlowSelectionInterface* si) {_hydraViewportSelectionInterface = si;} + + ///From FlowViewport::SelectionClient + ///This is a dummy callback function to be replaced + void DummySelectionCallback()override; + +protected: + ///Store the Hydra interface pointer + FlowSelectionInterface* _hydraViewportSelectionInterface = nullptr; +}; + +} //end of namespace FVP_NS_DEF + +#endif //FLOW_VIEWPORT_API_SAMPLES_SELECTIONCLIENTEXAMPLE_H diff --git a/lib/flowViewport/CMakeLists.txt b/lib/flowViewport/CMakeLists.txt new file mode 100644 index 0000000000..a331172a8a --- /dev/null +++ b/lib/flowViewport/CMakeLists.txt @@ -0,0 +1,114 @@ +set(TARGET_NAME flowViewport) + +add_library(${TARGET_NAME} SHARED) + +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + debugCodes.cpp + global.cpp + tokens.cpp +) + +set(HEADERS + api.h + debugCodes.h + global.h + tokens.h +) + +# ----------------------------------------------------------------------------- +# include directories +# ----------------------------------------------------------------------------- +target_include_directories(${TARGET_NAME} + PUBLIC + ${PXR_INCLUDE_DIRS} + ${CMAKE_BINARY_DIR}/include + ${UFE_INCLUDE_DIR} +) + +# ----------------------------------------------------------------------------- +# compiler configuration +# ----------------------------------------------------------------------------- + +# Without the TBB_USE_DEBUG, BOOST_DEBUG_PYTHON and BOOST_LINKING_PYTHON +# the Debug variant link fails, even though there is no Python or Boost +# at time of writing in the Flow Viewport library. The assumption is that +# Hydra USD headers impose this requirement. PPT, 26-Sep-2023. + +target_compile_definitions(${TARGET_NAME} + PUBLIC + $<$:TBB_USE_DEBUG> + $<$:BOOST_DEBUG_PYTHON> + $<$:BOOST_LINKING_PYTHON> + PRIVATE + FVP_EXPORT + $<$:OSMac_> + # Copy-pasted from lib/mayaUsd/CMakeLists.txt +) + +mayaHydra_compile_config(${TARGET_NAME}) + +# ----------------------------------------------------------------------------- +# link libraries +# ----------------------------------------------------------------------------- +target_link_libraries(${TARGET_NAME} + PUBLIC + hd + hdx + sdf + ${UFE_LIBRARY} + PRIVATE + ar + gf + tf +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +set(SRCFILE ${CMAKE_CURRENT_SOURCE_DIR}/flowViewport.h.src) +set(DSTFILE ${CMAKE_BINARY_DIR}/include/flowViewport/flowViewport.h) +configure_file(${SRCFILE} ${DSTFILE}) + +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME} +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- + +install(TARGETS ${TARGET_NAME} + LIBRARY + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib + RUNTIME + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib +) + +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport +) + +install(FILES ${CMAKE_BINARY_DIR}/include/flowViewport/flowViewport.h + DESTINATION ${CMAKE_INSTALL_PREFIX}/include/flowViewport +) + +if(IS_WINDOWS) + install(FILES $ + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib OPTIONAL) +endif() + +# ----------------------------------------------------------------------------- +# subdirectories +# ----------------------------------------------------------------------------- +add_subdirectory(colorPreferences) +add_subdirectory(sceneIndex) +add_subdirectory(API) +add_subdirectory(selection) diff --git a/lib/flowViewport/api.h b/lib/flowViewport/api.h new file mode 100644 index 0000000000..f6ed103df1 --- /dev/null +++ b/lib/flowViewport/api.h @@ -0,0 +1,43 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_API_H +#define FVP_API_H + +#ifdef __GNUC__ +#define FVP_API_EXPORT __attribute__((visibility("default"))) +#define FVP_API_IMPORT +#elif defined(_WIN32) || defined(_WIN64) +#define FVP_API_EXPORT __declspec(dllexport) +#define FVP_API_IMPORT __declspec(dllimport) +#else +#define FVP_API_EXPORT +#define FVP_API_IMPORT +#endif + +#if defined(FVP_STATIC) +#define FVP_API +#else +#if defined(FVP_EXPORT) +#define FVP_API FVP_API_EXPORT +#else +#define FVP_API FVP_API_IMPORT +#endif +#endif + +// Convenience symbol versioning include: because api.h is widely +// included, this reduces the need to explicitly include flowViewport.h. +#include + +#endif // FVP_API_H diff --git a/lib/flowViewport/colorPreferences/CMakeLists.txt b/lib/flowViewport/colorPreferences/CMakeLists.txt new file mode 100644 index 0000000000..a220aba4b0 --- /dev/null +++ b/lib/flowViewport/colorPreferences/CMakeLists.txt @@ -0,0 +1,34 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + fvpColorChanged.cpp + fvpColorPreferences.cpp + fvpColorPreferencesTokens.cpp +) + +set(HEADERS + fvpColorChanged.h + fvpColorPreferences.h + fvpColorPreferencesTokens.h + fvpColorPreferencesTranslator.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/colorPreferences +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/colorPreferences +) diff --git a/lib/flowViewport/colorPreferences/fvpColorChanged.cpp b/lib/flowViewport/colorPreferences/fvpColorChanged.cpp new file mode 100644 index 0000000000..d268ae7edf --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorChanged.cpp @@ -0,0 +1,36 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "fvpColorChanged.h" + +namespace FVP_NS_DEF { + +ColorChanged::ColorChanged( + const PXR_NS::TfToken& token, + const PXR_NS::GfVec4f& oldColor, + const PXR_NS::GfVec4f& newColor) + : _token(token) + , _oldColor(oldColor) + , _newColor(newColor) +{ +} + +const PXR_NS::TfToken& ColorChanged::token() const { return _token; } + +const PXR_NS::GfVec4f& ColorChanged::oldColor() const { return _oldColor; } + +const PXR_NS::GfVec4f& ColorChanged::newColor() const { return _newColor; } + +} // namespace FVP_NS_DEF diff --git a/lib/flowViewport/colorPreferences/fvpColorChanged.h b/lib/flowViewport/colorPreferences/fvpColorChanged.h new file mode 100644 index 0000000000..8644eb6b4d --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorChanged.h @@ -0,0 +1,53 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_COLOR_CHANGED_H +#define FVP_COLOR_CHANGED_H + +#include "flowViewport/api.h" + +#include +#include + +#include + +namespace FVP_NS_DEF { + +/// \class ColorChanged +/// +/// \brief \p Notification class representing a color change and containing +/// the information associated with it. Objects of this class are immutable. +/// +class FVP_API ColorChanged : public Notification +{ +public: + ColorChanged( + const PXR_NS::TfToken& token, + const PXR_NS::GfVec4f& oldColor, + const PXR_NS::GfVec4f& newColor); + ~ColorChanged() override = default; + + const PXR_NS::TfToken& token() const; + const PXR_NS::GfVec4f& oldColor() const; + const PXR_NS::GfVec4f& newColor() const; + +private: + PXR_NS::TfToken _token; + PXR_NS::GfVec4f _oldColor; + PXR_NS::GfVec4f _newColor; +}; + +} // namespace FVP_NS_DEF + +#endif // FVP_COLOR_CHANGED_H diff --git a/lib/flowViewport/colorPreferences/fvpColorPreferences.cpp b/lib/flowViewport/colorPreferences/fvpColorPreferences.cpp new file mode 100644 index 0000000000..a9a6a1113e --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorPreferences.cpp @@ -0,0 +1,81 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "fvpColorPreferences.h" + +#include "fvpColorChanged.h" +#include "fvpColorPreferencesTranslator.h" + +#include + +namespace { +std::shared_ptr instance; +} // namespace + +namespace FVP_NS_DEF { + +ColorPreferences& ColorPreferences::getInstance() +{ + if (!instance) { + instance = std::shared_ptr(new ColorPreferences()); + } + return *instance; +} + +void ColorPreferences::deleteInstance() +{ + if (instance) { + instance.reset(); + } +} + +void ColorPreferences::operator()(const Notification& notification) +{ + PXR_NAMESPACE_USING_DIRECTIVE + if (TF_VERIFY(dynamic_cast(¬ification) != nullptr)) { + notify(notification); + } +} + +bool ColorPreferences::getColor(const PXR_NS::TfToken& preference, PXR_NS::GfVec4f& outColor) const +{ + PXR_NAMESPACE_USING_DIRECTIVE + + if (!TF_VERIFY(_translator != nullptr)) { + return false; + } + + return _translator->getColor(preference, outColor); +} + +void ColorPreferences::setTranslator( + const std::shared_ptr& newTranslator) +{ + if (static_cast(newTranslator) != static_cast(_translator)) { + // Happy path : we're either doing null ptr -> valid ptr, or valid ptr -> null ptr + _translator = newTranslator; + return; + } + PXR_NAMESPACE_USING_DIRECTIVE + if (newTranslator != nullptr && _translator != nullptr) { + TF_CODING_ERROR("ColorPreferences::setTranslator was called with a non-null translator " + "while already having an active one. The second call will be ignored."); + } else if (newTranslator == nullptr && _translator == nullptr) { + TF_CODING_WARNING("ColorPreferences::setTranslator was called with a null translator while " + "already having none."); + } +} + +} // namespace FVP_NS_DEF diff --git a/lib/flowViewport/colorPreferences/fvpColorPreferences.h b/lib/flowViewport/colorPreferences/fvpColorPreferences.h new file mode 100644 index 0000000000..7c6ef15a7f --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorPreferences.h @@ -0,0 +1,102 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_COLOR_PREFERENCES_H +#define FVP_COLOR_PREFERENCES_H + +#include "flowViewport/api.h" + +#include +#include + +#include +#include +#include + +#include +#include + +namespace FVP_NS_DEF { + +class ColorPreferencesTranslator; + +/// \class ColorPreferences +/// +/// \brief Singleton used to retrieve color preferences and +/// subscribe to \p ColorChanged notifications. +/// +/// The \p ColorPreferences class acts as the entry point for Flow Viewport +/// users to get informed about the color preferences of the host. +/// It is a singleton that provides two services : +/// - It rebroadcasts the notifications it receives from the host. It will only rebroadcast +/// notifications of type ColorChanged. +/// - It forwards the \p getColor() calls it receives to its \p ColorPreferencesTranslator. +/// The \p ColorPreferencesTranslator must be supplied by the host to +/// provide the translation between the host and the Flow Viewport. +/// +class FVP_API ColorPreferences final + : public Observer + , public Subject + , public std::enable_shared_from_this +{ +public: + ~ColorPreferences() override = default; + + /// @brief Returns the singleton instance of this class. The referenced object is managed by a + /// \p shared_ptr, enabling the use of \p shared_from_this(). Creates a new instance if none + /// currently exists. + /// @return The singleton instance of this class, backed by a \p shared_ptr. + static ColorPreferences& getInstance(); + + /// @brief Deletes the current singleton instance of this class, if one exists. + static void deleteInstance(); + + /// @brief Operator overload to receive notifications about color changes. + /// This will in turn rebroadcast the notification to all its observers, + /// but only if the notification is of type ColorChanged. + /// If the notification is of any other type, it will not be rebroadcast. + /// This will get called automatically by calling notify on this \p Subject, + /// you do not need to call this manually. + void operator()(const Notification& notification) override; + + /// @brief Retrieve the color value for a given color preference. + /// + /// @param[in] preference The color preference we want to know the color of. + /// @param[out] outColor Output parameter to store the color resulting from + /// the query. + /// + /// @return True if the color was found and \p outColor was populated, + /// false otherwise. + bool getColor(const PXR_NS::TfToken& preference, PXR_NS::GfVec4f& outColor) const; + + /// @brief Set the translator for which to forward \p getColor() calls to. + /// + /// @param[in] translator is a pointer to an implementation of the + /// \p ColorPreferencesTranslator interface. It is the object to which + /// the \p getColor() calls will be forwarded to. + void setTranslator(const std::shared_ptr& newTranslator); + +private: + ColorPreferences() = default; + ColorPreferences(const ColorPreferences&) = delete; + ColorPreferences(ColorPreferences&&) = delete; + ColorPreferences& operator=(const ColorPreferences&) = delete; + ColorPreferences& operator=(ColorPreferences&&) = delete; + + std::shared_ptr _translator; +}; + +} // namespace FVP_NS_DEF + +#endif // FVP_COLOR_PREFERENCES_H diff --git a/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.cpp b/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.cpp new file mode 100644 index 0000000000..eb06c17659 --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.cpp @@ -0,0 +1,24 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "fvpColorPreferencesTokens.h" + +// *** TODO / FIXME *** Figure out how to put tokens into non-Pixar namespace. + +PXR_NAMESPACE_OPEN_SCOPE + +TF_DEFINE_PUBLIC_TOKENS(FvpColorPreferencesTokens, FVP_COLOR_PREFERENCES_TOKENS); + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.h b/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.h new file mode 100644 index 0000000000..95f27308ac --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorPreferencesTokens.h @@ -0,0 +1,43 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_COLOR_PREFERENCES_TOKENS_H +#define FVP_COLOR_PREFERENCES_TOKENS_H + +#include "flowViewport/api.h" + +#include + +// *** TODO / FIXME *** Figure out how to put tokens into non-Pixar namespace. + +PXR_NAMESPACE_OPEN_SCOPE + +// clang-format off +#define FVP_COLOR_PREFERENCES_TOKENS \ + /* The "unspecified" token should only be used by hosts who are incapable of sending precise notifications. */ \ + /* In such cases, it should be used only when sending notifications, and the colors contained in the */ \ + /* corresponding ColorChanged notification should be ignored by the notification recipient. */ \ + (unspecified) \ + (wireframeSelection) \ + (wireframeSelectionSecondary) \ + (vertexSelection) \ + (edgeSelection) \ + (faceSelection) +// clang-format on + +TF_DECLARE_PUBLIC_TOKENS(FvpColorPreferencesTokens, FVP_API, FVP_COLOR_PREFERENCES_TOKENS); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // FVP_COLOR_PREFERENCES_TOKENS_H diff --git a/lib/flowViewport/colorPreferences/fvpColorPreferencesTranslator.h b/lib/flowViewport/colorPreferences/fvpColorPreferencesTranslator.h new file mode 100644 index 0000000000..383e553567 --- /dev/null +++ b/lib/flowViewport/colorPreferences/fvpColorPreferencesTranslator.h @@ -0,0 +1,52 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_COLOR_PREFERENCES_TRANSLATOR_H +#define FVP_COLOR_PREFERENCES_TRANSLATOR_H + +#include "flowViewport/api.h" + +#include +#include + +#include + +namespace FVP_NS_DEF { + +/// \class ColorPreferencesTranslator +/// +/// A \p ColorPreferencesTranslator acts as a translation layer between +/// the host and the Flow Viewport interface. It is an interface which +/// the host implements. It only needs to implement the virtual method +/// \p getColor() to return a requested color preference. +/// +class FVP_API ColorPreferencesTranslator +{ +public: + /// @brief Retrieve the color value for a given color preference. + /// + /// @param[in] preference The color preference we want to know the color of. + /// @param[out] outColor Output parameter to store the color resulting from + /// the query. + /// + /// @return True if the color was found and \p outColor was populated, + /// false otherwise. + virtual bool getColor(const PXR_NS::TfToken& preference, PXR_NS::GfVec4f& outColor) const = 0; + + virtual ~ColorPreferencesTranslator() = default; +}; + +} // namespace FVP_NS_DEF + +#endif // FVP_COLOR_PREFERENCES_TRANSLATOR_H diff --git a/lib/flowViewport/debugCodes.cpp b/lib/flowViewport/debugCodes.cpp new file mode 100644 index 0000000000..4df57df760 --- /dev/null +++ b/lib/flowViewport/debugCodes.cpp @@ -0,0 +1,49 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "debugCodes.h" + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// Some variables to enable debug printing information +TF_REGISTRY_FUNCTION(TfDebug) +{ + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_SELECTION_SCENE_INDEX, + "Print information about the Flow Viewport selection scene index."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_SELECTION_TASK, + "Print information about the Flow Viewport selection task."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_SELECTION_TRACKER, + "Print information about the Flow Viewport selection tracker."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_APP_SELECTION_CHANGE, + "Print information about application selection changes sent to the Flow Viewport."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_MERGING_SCENE_INDEX, + "Print information about the Flow Viewport merging scene index."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + PXR_NS::FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX, + "Print information about the Flow Viewport wireframe selection highlight scene index."); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/flowViewport/debugCodes.h b/lib/flowViewport/debugCodes.h new file mode 100644 index 0000000000..29e5d6b04b --- /dev/null +++ b/lib/flowViewport/debugCodes.h @@ -0,0 +1,39 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_DEBUG_CODES_H +#define FVP_DEBUG_CODES_H + +#include +#include + +// No success in defining TF_DEBUG codes in anything but the pxr namespace. +// PPT, 29-Jun-2023. + +PXR_NAMESPACE_OPEN_SCOPE + +// clang-format off +TF_DEBUG_CODES( + FVP_SELECTION_SCENE_INDEX + , FVP_SELECTION_TASK + , FVP_SELECTION_TRACKER + , FVP_APP_SELECTION_CHANGE + , FVP_MERGING_SCENE_INDEX + , FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX +); +// clang-format on + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // FVP_DEBUG_CODES_H diff --git a/lib/flowViewport/flowViewport.h.src b/lib/flowViewport/flowViewport.h.src new file mode 100644 index 0000000000..f5aac2c9f4 --- /dev/null +++ b/lib/flowViewport/flowViewport.h.src @@ -0,0 +1,63 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#pragma once + +#include + +#define FLOWVIEWPORT_MAJOR_VERSION ${FLOWVIEWPORT_MAJOR_VERSION} +#define FLOWVIEWPORT_MINOR_VERSION ${FLOWVIEWPORT_MINOR_VERSION} +#define FLOWVIEWPORT_PATCH_LEVEL ${FLOWVIEWPORT_PATCH_LEVEL} +#define FLOWVIEWPORT_API_VERSION (FLOWVIEWPORT_MAJOR_VERSION * 10000 + FLOWVIEWPORT_MINOR_VERSION * 100 + FLOWVIEWPORT_PATCH_LEVEL) + +// Flow viewport public namespace string will never change. +#define FVP_NS Fvp +// C preprocessor trickery to expand arguments. +#define FLOWVIEWPORT_CONCAT(A, B) FLOWVIEWPORT_CONCAT_IMPL(A, B) +#define FLOWVIEWPORT_CONCAT_IMPL(A, B) A##B +// Versioned namespace includes the major version number. +#define FVP_VERSIONED_NS FLOWVIEWPORT_CONCAT(FVP_NS, _v${FLOWVIEWPORT_MAJOR_VERSION}) + +// 2023-10-12 : To avoid recreating redundant classes to implement the Observer pattern, +// we reuse the UFE classes for it (plus they have already been tested in the wild). +// Bringing the UFE classes into the FVP namespace allows us to simply copy over the UFE code +// without having to change the dependent code, should we want to remove the UFE dependency. +namespace UFE_VERSIONED_NS { + class Notification; + class Observer; + class Subject; +} // namespace UFE_VERSIONED_NS + +namespace FVP_VERSIONED_NS { + using UFE_VERSIONED_NS::Notification; + using UFE_VERSIONED_NS::Observer; + using UFE_VERSIONED_NS::Subject; +} // namespace FVP_VERSIONED_NS + +// With a using namespace declaration, pull in the versioned namespace into the +// Flow viewport public namespace, to allow client code to use the plain Flow +// viewport namespace, e.g. Fvp::Class. +namespace FVP_NS { + using namespace FVP_VERSIONED_NS; +} + +// Macro to place the Flow viewport symbols in the versioned namespace, which +// is how they will appear in the shared library, e.g. Fvp_v1::Class. +#ifdef DOXYGEN +#define FVP_NS_DEF FVP_NS +#else +#define FVP_NS_DEF FVP_VERSIONED_NS +#endif diff --git a/lib/flowViewport/global.cpp b/lib/flowViewport/global.cpp new file mode 100644 index 0000000000..23a16de714 --- /dev/null +++ b/lib/flowViewport/global.cpp @@ -0,0 +1,38 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "global.h" + +#include + +namespace FVP_NS_DEF { + +void initialize(const InitializationParams& params) +{ + params.colorPreferencesNotificationProvider->addObserver( + Fvp::ColorPreferences::getInstance().shared_from_this()); + Fvp::ColorPreferences::getInstance().setTranslator(params.colorPreferencesTranslator); +} + +void finalize(const InitializationParams& params) +{ + params.colorPreferencesNotificationProvider->removeObserver( + Fvp::ColorPreferences::getInstance().shared_from_this()); + Fvp::ColorPreferences::getInstance().setTranslator(nullptr); + + Fvp::ColorPreferences::deleteInstance(); +} + +} // namespace FVP_NS_DEF diff --git a/lib/flowViewport/global.h b/lib/flowViewport/global.h new file mode 100644 index 0000000000..0cf2737a4f --- /dev/null +++ b/lib/flowViewport/global.h @@ -0,0 +1,36 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_GLOBAL_H +#define FVP_GLOBAL_H + +#include +#include + +#include + +namespace FVP_NS_DEF { + +struct FVP_API InitializationParams +{ + std::shared_ptr colorPreferencesNotificationProvider; + std::shared_ptr colorPreferencesTranslator; +}; + +void FVP_API initialize(const InitializationParams& params); +void FVP_API finalize(const InitializationParams& params); + +} // namespace FVP_NS_DEF + +#endif // FVP_GLOBAL_H diff --git a/lib/flowViewport/sceneIndex/CMakeLists.txt b/lib/flowViewport/sceneIndex/CMakeLists.txt new file mode 100644 index 0000000000..04dd9b47c9 --- /dev/null +++ b/lib/flowViewport/sceneIndex/CMakeLists.txt @@ -0,0 +1,43 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + fvpMergingSceneIndex.cpp + fvpPassThroughSelectionInterfaceSceneIndex.cpp + fvpPathInterface.cpp + fvpPathInterfaceSceneIndex.cpp + fvpRenderIndexProxy.cpp + fvpSelectionInterface.cpp + fvpSelectionSceneIndex.cpp + fvpWireframeSelectionHighlightSceneIndex.cpp +) + +set(HEADERS + fvpMergingSceneIndex.h + fvpPassThroughSelectionInterfaceSceneIndex.h + fvpPathInterface.h + fvpPathInterfaceSceneIndex.h + fvpRenderIndexProxy.h + fvpSelectionInterface.h + fvpSelectionSceneIndex.h + fvpWireframeSelectionHighlightSceneIndex.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/sceneIndex +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/sceneIndex +) diff --git a/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.cpp b/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.cpp new file mode 100644 index 0000000000..6a46bfbe80 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.cpp @@ -0,0 +1,62 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpMergingSceneIndex.h" + +#include "flowViewport/debugCodes.h" + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +/* static */ +MergingSceneIndexRefPtr MergingSceneIndex::New() { + TF_DEBUG(FVP_MERGING_SCENE_INDEX) + .Msg("MergingSceneIndex::New() called.\n"); + return TfCreateRefPtr(new MergingSceneIndex); +} + +MergingSceneIndex::MergingSceneIndex() : HdMergingSceneIndex() +{ + TF_DEBUG(FVP_MERGING_SCENE_INDEX) + .Msg("MergingSceneIndex::MergingSceneIndex() called.\n"); +} + +SdfPath MergingSceneIndex::SceneIndexPath(const Ufe::Path& appPath) const +{ + // FLOW_VIEWPORT_TODO May be able to use a caching scheme for app path to + // scene index path conversion using the run-time ID of the UFE path, as it + // is likely that the input scene index that provided a previous answer + // will do so again. To be determined if the following direct approach has + // a measurable performance impact. PPT, 18-Sep-2023. + + // Iterate over input scene indices and ask them to convert the path if + // they support the path interface. + auto inputScenes = GetInputScenes(); + for (const auto& inputScene : inputScenes) { + // Unfortunate that we have to dynamic cast, as soon as we add an input + // scene we know whether it supports the PathInterface or not. + auto pathInterface = dynamic_cast(&*inputScene); + if (pathInterface) { + auto sceneIndexPath = pathInterface->SceneIndexPath(appPath); + if (!sceneIndexPath.IsEmpty()) { + return sceneIndexPath; + } + } + } + return SdfPath(); +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.h b/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.h new file mode 100644 index 0000000000..d4b98b26cb --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpMergingSceneIndex.h @@ -0,0 +1,52 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_MERGING_SCENE_INDEX_H +#define FVP_MERGING_SCENE_INDEX_H + +#include "flowViewport/api.h" +#include "flowViewport/sceneIndex/fvpPathInterface.h" + +#include + +namespace FVP_NS_DEF { + +// Pixar declarePtrs.h TF_DECLARE_REF_PTRS macro unusable, places resulting +// type in PXR_NS. +class MergingSceneIndex; +typedef PXR_NS::TfRefPtr MergingSceneIndexRefPtr; +typedef PXR_NS::TfRefPtr MergingSceneIndexConstRefPtr; + +/// \class MergingSceneIndex +/// +/// A merging scene index that delegates conversion of application paths to +/// scene index paths to its inputs. +/// +class MergingSceneIndex + : public PXR_NS::HdMergingSceneIndex, public PathInterface +{ +public: + FVP_API + static MergingSceneIndexRefPtr New(); + + FVP_API + PXR_NS::SdfPath SceneIndexPath(const Ufe::Path& appPath) const override; + +private: + MergingSceneIndex(); +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.cpp b/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.cpp new file mode 100644 index 0000000000..7cd82362aa --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.cpp @@ -0,0 +1,43 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h" + +#include "flowViewport/debugCodes.h" + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +PassThroughSelectionInterfaceSceneIndexBase:: +PassThroughSelectionInterfaceSceneIndexBase(HdSceneIndexBaseRefPtr const &inputSceneIndex) + : HdSingleInputFilteringSceneIndexBase(inputSceneIndex) + , _inputSceneIndexSelectionInterface(dynamic_cast(&*_GetInputSceneIndex())) +{ + TF_AXIOM(_inputSceneIndexSelectionInterface); +} + +bool PassThroughSelectionInterfaceSceneIndexBase::IsFullySelected(const SdfPath& primPath) const +{ + return _inputSceneIndexSelectionInterface->IsFullySelected(primPath); +} + +bool PassThroughSelectionInterfaceSceneIndexBase::HasFullySelectedAncestorInclusive(const SdfPath& primPath) const +{ + return _inputSceneIndexSelectionInterface->HasFullySelectedAncestorInclusive(primPath); +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h b/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h new file mode 100644 index 0000000000..641c16e3f6 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h @@ -0,0 +1,58 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_PASS_THROUGH_SELECTION_INTERFACE_SCENE_INDEX_H +#define FVP_PASS_THROUGH_SELECTION_INTERFACE_SCENE_INDEX_H + +#include "flowViewport/api.h" +#include "flowViewport/sceneIndex/fvpSelectionInterface.h" + +#include + +namespace FVP_NS_DEF { + +/// \class PassThroughSelectionInterfaceSceneIndexBase +/// +/// Convenience base class that passes through the SelectionInterface queries +/// to its input filtering scene index. +/// +class PassThroughSelectionInterfaceSceneIndexBase + : public PXR_NS::HdSingleInputFilteringSceneIndexBase, + public SelectionInterface +{ +public: + + //! Selection interface overrides. + //@{ + FVP_API + bool IsFullySelected(const PXR_NS::SdfPath& primPath) const override; + + FVP_API + bool HasFullySelectedAncestorInclusive(const PXR_NS::SdfPath& primPath) const override; + //@} + +protected: + + FVP_API + PassThroughSelectionInterfaceSceneIndexBase( + const PXR_NS::HdSceneIndexBaseRefPtr &inputSceneIndex); + +private: + + const SelectionInterface* const _inputSceneIndexSelectionInterface; +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpPathInterface.cpp b/lib/flowViewport/sceneIndex/fvpPathInterface.cpp new file mode 100644 index 0000000000..28445061d3 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPathInterface.cpp @@ -0,0 +1,22 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpPathInterface.h" + +namespace FVP_NS_DEF { + +PathInterface::~PathInterface() {} + +} diff --git a/lib/flowViewport/sceneIndex/fvpPathInterface.h b/lib/flowViewport/sceneIndex/fvpPathInterface.h new file mode 100644 index 0000000000..55d16c944e --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPathInterface.h @@ -0,0 +1,61 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_PATH_INTERFACE_H +#define FVP_PATH_INTERFACE_H + +#include "flowViewport/api.h" + +#include + +#include + +UFE_NS_DEF { +class Path; +} + +PXR_NAMESPACE_OPEN_SCOPE +class SdfPath; +PXR_NAMESPACE_CLOSE_SCOPE + +namespace FVP_NS_DEF { + +/// \class PathInterface +/// +/// A pure interface class to allow for conversion between an application's +/// path, expressed as a Ufe::Path, into an SdfPath valid for a scene index. +/// To be used as a mix-in class for scene indices. +/// +class PathInterface +{ +public: + + //! Return the prim path corresponding to the argument application path. + //! If no such path exists, an empty SdfPath should be returned. + //! \return scene index path. + FVP_API + virtual PXR_NS::SdfPath SceneIndexPath(const Ufe::Path& appPath) const = 0; + +protected: + + FVP_API + PathInterface() = default; + + FVP_API + virtual ~PathInterface(); +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.cpp b/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.cpp new file mode 100644 index 0000000000..d0807f782b --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.cpp @@ -0,0 +1,64 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.h" + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +PathInterfaceSceneIndexBase:: +PathInterfaceSceneIndexBase(HdSceneIndexBaseRefPtr const &inputSceneIndex) + : HdSingleInputFilteringSceneIndexBase(inputSceneIndex) +{} + +HdSceneIndexPrim +PathInterfaceSceneIndexBase::GetPrim(const SdfPath &primPath) const +{ + return _GetInputSceneIndex()->GetPrim(primPath); +} + +SdfPathVector +PathInterfaceSceneIndexBase::GetChildPrimPaths(const SdfPath &primPath) const +{ + return _GetInputSceneIndex()->GetChildPrimPaths(primPath); +} + +void +PathInterfaceSceneIndexBase::_PrimsAdded( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::AddedPrimEntries &entries) +{ + _SendPrimsAdded(entries); +} + +void +PathInterfaceSceneIndexBase::_PrimsDirtied( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::DirtiedPrimEntries &entries) +{ + _SendPrimsDirtied(entries); +} + +void +PathInterfaceSceneIndexBase::_PrimsRemoved( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::RemovedPrimEntries &entries) +{ + _SendPrimsRemoved(entries); +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.h b/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.h new file mode 100644 index 0000000000..193d865662 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpPathInterfaceSceneIndex.h @@ -0,0 +1,66 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_PATH_INTERFACE_SCENE_INDEX_H +#define FVP_PATH_INTERFACE_SCENE_INDEX_H + +#include "flowViewport/api.h" +#include "flowViewport/sceneIndex/fvpPathInterface.h" + +#include + +namespace FVP_NS_DEF { + +/// \class PathInterfaceSceneIndexBase +/// +/// A simple pass-through filtering scene index that adds support for the path +/// interface. Derived classes need only implement the +/// PathInterface::SceneIndexPath() virtual. +/// +class PathInterfaceSceneIndexBase + : public PXR_NS::HdSingleInputFilteringSceneIndexBase, public PathInterface +{ +public: + + FVP_API + PXR_NS::HdSceneIndexPrim GetPrim(const PXR_NS::SdfPath &primPath) const override; + + FVP_API + PXR_NS::SdfPathVector GetChildPrimPaths(const PXR_NS::SdfPath &primPath) const override; + +protected: + + FVP_API + PathInterfaceSceneIndexBase( + const PXR_NS::HdSceneIndexBaseRefPtr &inputSceneIndex); + + FVP_API + void _PrimsAdded( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::AddedPrimEntries &entries) override; + + FVP_API + void _PrimsRemoved( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::RemovedPrimEntries &entries) override; + + FVP_API + void _PrimsDirtied( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::DirtiedPrimEntries &entries) override; +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.cpp b/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.cpp new file mode 100644 index 0000000000..8f27cbe642 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.cpp @@ -0,0 +1,129 @@ +// +// Copyright 2022 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpRenderIndexProxy.h" +#include "flowViewport/sceneIndex/fvpMergingSceneIndex.h" + +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace { + +// Copy-pasted and adapted from USD 0.23.08 pxr/imaging/hd/renderIndex.cpp. +// PPT, 1-Sep-2023. +HdSceneIndexBaseRefPtr +_GetInputScene(const HdPrefixingSceneIndexRefPtr &prefixingScene) +{ + const auto inputScenes = prefixingScene->GetInputScenes(); + if (inputScenes.size() == 1) { + return inputScenes[0]; + } + TF_CODING_ERROR("Expected exactly one scene index from " + "HdPrefixingSceneIndex::GetInputScenes"); + return TfNullPtr; +} + +} + +namespace FVP_NS_DEF { + +RenderIndexProxy::RenderIndexProxy(PXR_NS::HdRenderIndex* renderIndex) : + _renderIndex(renderIndex), _mergingSceneIndex(MergingSceneIndex::New()) +{ + TF_AXIOM(_renderIndex); + TF_AXIOM(_mergingSceneIndex); + _mergingSceneIndex->SetDisplayName("Flow Viewport Merging Scene Index"); +} + +void RenderIndexProxy::InsertSceneIndex( + const PXR_NS::HdSceneIndexBaseRefPtr& inputScene, + const PXR_NS::SdfPath& scenePathPrefix, + bool needsPrefixing /* = true */ +) +{ + // Copy-pasted and adapted from USD 0.23.08 + // HdRenderIndex::InsertSceneIndex() code, to preserve prefixing scene + // index creation capability. PPT, 31-Aug-2023. + auto resolvedScene = inputScene; + if (needsPrefixing && scenePathPrefix != SdfPath::AbsoluteRootPath()) { + resolvedScene = HdPrefixingSceneIndex::New(inputScene, scenePathPrefix); + } + + _mergingSceneIndex->AddInputScene(resolvedScene, scenePathPrefix); +} + +void RenderIndexProxy::RemoveSceneIndex( + const HdSceneIndexBaseRefPtr &inputScene +) +{ + // Copy-pasted and adapted from USD 0.23.08 + // HdRenderIndex::RemoveSceneIndex() code, to preserve prefixing scene + // index removal capability. PPT, 1-Sep-2023. + + const auto resolvedScenes = _mergingSceneIndex->GetInputScenes(); + + // Two cases: + // - Given scene index was added by InsertSceneIndex with + // scenePathPrefix = "/". We find it just by going over the + // input scenes of _mergingSceneIndex. + // - Given scene index was added by InsertSceneIndex with + // non-trivial scenePathPrefix. We need to find the HdPrefixingSceneIndex + // among the input scenes of _mergingSceneIndex that was constructed + // from the given scene index. + for (const auto& resolvedScene : resolvedScenes) { + if (inputScene == resolvedScene) { + _mergingSceneIndex->RemoveInputScene(resolvedScene); + return; + } + if (HdPrefixingSceneIndexRefPtr const prefixingScene = + TfDynamic_cast(resolvedScene)) { + if (inputScene == _GetInputScene(prefixingScene)) { + _mergingSceneIndex->RemoveInputScene(resolvedScene); + return; + } + } + } +} + +PXR_NS::HdSceneIndexBaseRefPtr RenderIndexProxy::GetMergingSceneIndex() const +{ + return _mergingSceneIndex; +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.h b/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.h new file mode 100644 index 0000000000..5865e439b8 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpRenderIndexProxy.h @@ -0,0 +1,96 @@ +// +// Copyright 2022 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_RENDER_INDEX_PROXY_H +#define FVP_RENDER_INDEX_PROXY_H + +#include "flowViewport/api.h" + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE +class HdRenderIndex; +PXR_NAMESPACE_CLOSE_SCOPE + +namespace FVP_NS_DEF { + +/// \class RenderIndexProxy +/// +/// Class to protect access to the render index, and provide a merging +/// scene index under Flow viewport control. +/// +/// The merging scene index accessed through the Hydra render has hard-coded +/// downstream filtering scene indices. This render index proxy provides its +/// own merging scene index, after which we can easily insert downstream +/// filtering scene indices. +/// +/// FLOW_VIEWPORT_TODO At time of writing, the renderIndex data member is +/// unused. Re-evaluate the responsibilities, future extension, and naming of +/// this class. PPT, 22-Sep-2023. + +class RenderIndexProxy +{ +public: + + FVP_API + RenderIndexProxy(PXR_NS::HdRenderIndex* renderIndex); + + FVP_API + void InsertSceneIndex( + const PXR_NS::HdSceneIndexBaseRefPtr& inputScene, + const PXR_NS::SdfPath& scenePathPrefix, + bool needsPrefixing = true); + + FVP_API + void RemoveSceneIndex(const PXR_NS::HdSceneIndexBaseRefPtr &inputScene); + + // Return the additional Flow Viewport merging scene index onto which input + // scenes are added. Returned as a base scene index to preserve + // encapsulation. + FVP_API + PXR_NS::HdSceneIndexBaseRefPtr GetMergingSceneIndex() const; + +private: + + PXR_NS::HdRenderIndex* const _renderIndex{nullptr}; + PXR_NS::HdMergingSceneIndexRefPtr _mergingSceneIndex; +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpSelectionInterface.cpp b/lib/flowViewport/sceneIndex/fvpSelectionInterface.cpp new file mode 100644 index 0000000000..bb5b385f37 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpSelectionInterface.cpp @@ -0,0 +1,22 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpSelectionInterface.h" + +namespace FVP_NS_DEF { + +SelectionInterface::~SelectionInterface() {} + +} diff --git a/lib/flowViewport/sceneIndex/fvpSelectionInterface.h b/lib/flowViewport/sceneIndex/fvpSelectionInterface.h new file mode 100644 index 0000000000..3ab73a2c78 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpSelectionInterface.h @@ -0,0 +1,63 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_SELECTION_INTERFACE_H +#define FVP_SELECTION_INTERFACE_H + +#include "flowViewport/api.h" + +#include + +PXR_NAMESPACE_OPEN_SCOPE +class SdfPath; +PXR_NAMESPACE_CLOSE_SCOPE + +namespace FVP_NS_DEF { + +/// \class SelectionInterface +/// +/// A pure interface class to allow querying a scene index for selected status +/// without inspecting data sources in scene prims. To be used as a mix-in +/// class for scene indices. +/// +class SelectionInterface +{ +public: + + //! Returns whether the prim path is a fully selected prim. + //! If no such path exists, false is returned. + //! \return Whether the argument path is a fully selected prim. + FVP_API + virtual bool IsFullySelected(const PXR_NS::SdfPath& primPath) const = 0; + + //! Returns whether the prim path is a fully selected prim, or has an + //! ancestor that is a fully selected prim. If no such path exists, false + //! is returned. + //! \return Whether the argument path is a fully selected prim, or has a + //! fully selected ancestor. + FVP_API + virtual bool HasFullySelectedAncestorInclusive(const PXR_NS::SdfPath& primPath) const = 0; + +protected: + + FVP_API + SelectionInterface() = default; + + FVP_API + virtual ~SelectionInterface(); +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.cpp b/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.cpp new file mode 100644 index 0000000000..3236f13b08 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.cpp @@ -0,0 +1,372 @@ +// +// Copyright 2022 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpSelectionSceneIndex.h" +#include "flowViewport/sceneIndex/fvpPathInterface.h" + +#include "flowViewport/debugCodes.h" + +#include "pxr/imaging/hd/retainedDataSource.h" +#include "pxr/imaging/hd/selectionSchema.h" +#include "pxr/imaging/hd/selectionsSchema.h" + +#include +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace { +const HdDataSourceLocatorSet selectionsSchemaDefaultLocator{HdSelectionsSchema::GetDefaultLocator()}; +} + +namespace FVP_NS_DEF { + +namespace SelectionSceneIndex_Impl +{ + +struct _PrimSelectionState +{ + // Container data sources conforming to HdSelectionSchema + std::vector selectionSources; + + HdDataSourceBaseHandle GetVectorDataSource() { + return HdSelectionsSchema::BuildRetained( + selectionSources.size(), + selectionSources.data()); + } +}; + +struct _Selection +{ + bool IsSelected(const SdfPath& primPath) const + { + return pathToState.find(primPath) != pathToState.end(); + } + + bool HasSelectedAncestorInclusive(const SdfPath& primPath) const + { + // FLOW_VIEWPORT_TODO Prefix tree would be much higher performance + // than iterating over the whole selection, especially for a large + // selection. PPT, 13-Sep-2023. + for(const auto& entry : pathToState) { + if (primPath.HasPrefix(entry.first)) { + return true; + } + } + return false; + } + + // Maps prim path to data sources to be returned by the vector data + // source at locator selections. + std::map pathToState; +}; + +class _PrimSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(_PrimSource); + + _PrimSource(HdContainerDataSourceHandle const &inputSource, + _SelectionSharedPtr const selection, + const SdfPath &primPath) + : _inputSource(inputSource) + , _selection(selection) + , _primPath(primPath) + { + } + + TfTokenVector GetNames() override + { + TfTokenVector names = _inputSource->GetNames(); + if (_selection->IsSelected(_primPath)) { + names.push_back(HdSelectionsSchemaTokens->selections); + } + return names; + } + + HdDataSourceBaseHandle Get(const TfToken &name) override + { + if (name == HdSelectionsSchemaTokens->selections) { + auto it = _selection->pathToState.find(_primPath); + return (it != _selection->pathToState.end()) ? + it->second.GetVectorDataSource() : nullptr; + } + + return _inputSource->Get(name); + } + +private: + HdContainerDataSourceHandle const _inputSource; + _SelectionSharedPtr const _selection; + const SdfPath _primPath; +}; + +} + +using namespace SelectionSceneIndex_Impl; + +SelectionSceneIndexRefPtr +SelectionSceneIndex::New(HdSceneIndexBaseRefPtr const &inputSceneIndex) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::New() called.\n"); + return TfCreateRefPtr(new SelectionSceneIndex(inputSceneIndex)); +} + +SelectionSceneIndex:: +SelectionSceneIndex(HdSceneIndexBaseRefPtr const &inputSceneIndex) + : HdSingleInputFilteringSceneIndexBase(inputSceneIndex) + , _selection(std::make_shared<_Selection>()) + , _inputSceneIndexPathInterface(dynamic_cast(&*_GetInputSceneIndex())) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::SelectionSceneIndex() called.\n"); + + TF_AXIOM(_inputSceneIndexPathInterface); +} + +HdSceneIndexPrim +SelectionSceneIndex::GetPrim(const SdfPath &primPath) const +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::GetPrim() called.\n"); + + HdSceneIndexPrim result = _GetInputSceneIndex()->GetPrim(primPath); + if (!result.dataSource) { + return result; + } + + result.dataSource = _PrimSource::New( + result.dataSource, _selection, primPath); + + return result; +} + +SdfPathVector +SelectionSceneIndex::GetChildPrimPaths(const SdfPath &primPath) const +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::GetChildPrimPaths() called.\n"); + + return _GetInputSceneIndex()->GetChildPrimPaths(primPath); +} + +void +SelectionSceneIndex::AddSelection(const Ufe::Path& appPath) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::AddSelection(const Ufe::Path& %s) called.\n", Ufe::PathString::string(appPath).c_str()); + + HdSelectionSchema::Builder selectionBuilder; + selectionBuilder.SetFullySelected( + HdRetainedTypedSampledDataSource::New(true)); + + // Call our input scene index to convert the application path to a scene + // index path. + auto sceneIndexPath = SceneIndexPath(appPath); + + if (sceneIndexPath.IsEmpty()) { + return; + } + + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg(" Adding %s to the Hydra selection.\n", sceneIndexPath.GetText()); + + _selection->pathToState[sceneIndexPath].selectionSources.push_back( + selectionBuilder.Build()); + + _SendPrimsDirtied({{sceneIndexPath, selectionsSchemaDefaultLocator}}); +} + +void SelectionSceneIndex::RemoveSelection(const Ufe::Path& appPath) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::RemoveSelection(const Ufe::Path& %s) called.\n", Ufe::PathString::string(appPath).c_str()); + + // Call our input scene index to convert the application path to a scene + // index path. If there is no path, or the path is not selected, early out. + auto sceneIndexPath = SceneIndexPath(appPath); + + if (sceneIndexPath.IsEmpty() || + (_selection->pathToState.erase(sceneIndexPath) != 1)) { + return; + } + + HdSceneIndexObserver::DirtiedPrimEntries entry; + entry.emplace_back(sceneIndexPath, selectionsSchemaDefaultLocator); + + _SendPrimsDirtied(entry); +} + +void +SelectionSceneIndex::ClearSelection() +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::ClearSelection() called.\n"); + + if (_selection->pathToState.empty()) { + return; + } + + HdSceneIndexObserver::DirtiedPrimEntries entries; + entries.reserve(_selection->pathToState.size()); + for (const auto &pathAndSelections : _selection->pathToState) { + entries.emplace_back(pathAndSelections.first, selectionsSchemaDefaultLocator); + } + + _selection->pathToState.clear(); + + _SendPrimsDirtied(entries); +} + +void SelectionSceneIndex::ReplaceSelection(const Ufe::Selection& selection) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::ReplaceSelection() called.\n"); + + HdSelectionSchema::Builder selectionBuilder; + selectionBuilder.SetFullySelected( + HdRetainedTypedSampledDataSource::New(true)); + + // Process the selection replace by performing dirty notification of the + // existing selection state. We could do this more efficiently by + // accounting for overlapping previous and new selections. + HdSceneIndexObserver::DirtiedPrimEntries entries; + entries.reserve(_selection->pathToState.size() + selection.size()); + for (const auto &pathAndSelections : _selection->pathToState) { + entries.emplace_back(pathAndSelections.first, selectionsSchemaDefaultLocator); + } + + _selection->pathToState.clear(); + + for (const auto& snItem : selection) { + // Call our input scene index to convert the application path to a scene + // index path. + auto sceneIndexPath = SceneIndexPath(snItem->path()); + + if (sceneIndexPath.IsEmpty()) { + continue; + } + + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg(" Adding %s to the Hydra selection.\n", sceneIndexPath.GetText()); + + _selection->pathToState[sceneIndexPath].selectionSources.push_back( + selectionBuilder.Build()); + + entries.emplace_back(sceneIndexPath, selectionsSchemaDefaultLocator); + } + + _SendPrimsDirtied(entries); +} + +bool SelectionSceneIndex::IsFullySelected(const SdfPath& primPath) const +{ + return _selection->IsSelected(primPath); +} + +bool SelectionSceneIndex::HasFullySelectedAncestorInclusive(const SdfPath& primPath) const +{ + return _selection->HasSelectedAncestorInclusive(primPath); +} + +SdfPath SelectionSceneIndex::SceneIndexPath(const Ufe::Path& appPath) const +{ + auto sceneIndexPath = _inputSceneIndexPathInterface->SceneIndexPath(appPath); + + if (sceneIndexPath.IsEmpty()) { + TF_WARN("SelectionSceneIndex::SceneIndexPath(%s) returned an empty path, Hydra selection will be incorrect", Ufe::PathString::string(appPath).c_str()); + } + + return sceneIndexPath; +} + +SdfPathVector SelectionSceneIndex::GetFullySelectedPaths() const +{ + SdfPathVector fullySelectedPaths; + fullySelectedPaths.reserve(_selection->pathToState.size()); + for(const auto& entry : _selection->pathToState) { + fullySelectedPaths.emplace_back(entry.first); + } + return fullySelectedPaths; +} + +void +SelectionSceneIndex::_PrimsAdded( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::AddedPrimEntries &entries) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::_PrimsAdded() called.\n"); + + _SendPrimsAdded(entries); +} + +void +SelectionSceneIndex::_PrimsDirtied( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::DirtiedPrimEntries &entries) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::_PrimsDirtied() called.\n"); + + _SendPrimsDirtied(entries); +} + +void +SelectionSceneIndex::_PrimsRemoved( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::RemovedPrimEntries &entries) +{ + TF_DEBUG(FVP_SELECTION_SCENE_INDEX) + .Msg("SelectionSceneIndex::_PrimsRemoved() called.\n"); + + if (!_selection->pathToState.empty()) { + for (const auto &entry : entries) { + auto it = _selection->pathToState.lower_bound(entry.primPath); + while (it != _selection->pathToState.end() && + it->first.HasPrefix(entry.primPath)) { + it = _selection->pathToState.erase(it); + } + } + } + + _SendPrimsRemoved(entries); +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.h b/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.h new file mode 100644 index 0000000000..c65b973420 --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpSelectionSceneIndex.h @@ -0,0 +1,153 @@ +// +// Copyright 2022 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_SELECTION_SCENE_INDEX_H +#define FVP_SELECTION_SCENE_INDEX_H + +#include "flowViewport/api.h" +#include "flowViewport/sceneIndex/fvpSelectionInterface.h" +#include "flowViewport/sceneIndex/fvpPathInterface.h" + +#include + +#include + +#include + +UFE_NS_DEF { +class Path; +class Selection; +} + +namespace FVP_NS_DEF { + +class PathInterface; + +// Pixar declarePtrs.h TF_DECLARE_REF_PTRS macro unusable, places resulting +// type in PXR_NS. +class SelectionSceneIndex; +typedef PXR_NS::TfRefPtr SelectionSceneIndexRefPtr; +typedef PXR_NS::TfRefPtr SelectionSceneIndexConstRefPtr; + +namespace SelectionSceneIndex_Impl +{ +using _SelectionSharedPtr = std::shared_ptr; +} + +/// \class SelectionSceneIndex +/// +/// A simple scene index adding HdSelectionsSchema to all prims selected +/// in the application. +/// +class SelectionSceneIndex final + : public PXR_NS::HdSingleInputFilteringSceneIndexBase + , public SelectionInterface + , public PathInterface +{ +public: + FVP_API + static SelectionSceneIndexRefPtr New( + PXR_NS::HdSceneIndexBaseRefPtr const &inputSceneIndex); + + FVP_API + PXR_NS::HdSceneIndexPrim GetPrim(const PXR_NS::SdfPath &primPath) const override; + + FVP_API + PXR_NS::SdfPathVector GetChildPrimPaths(const PXR_NS::SdfPath &primPath) const override; + + /// Given a path (including usd proxy path inside a native instance) of + /// a USD prim, determine the corresponding prim in the Usd scene index + /// (filtered by the UsdImagingNiPrototypePropagatingSceneIndex) and + /// populate its selections data source. + FVP_API + void AddSelection(const Ufe::Path& appPath); + + FVP_API + void RemoveSelection(const Ufe::Path& appPath); + + FVP_API + void ReplaceSelection(const Ufe::Selection& selection); + + /// Reset the scene index selection state. + FVP_API + void ClearSelection(); + + //! Selection interface overrides. + //@{ + FVP_API + bool IsFullySelected(const PXR_NS::SdfPath& primPath) const override; + + FVP_API + bool HasFullySelectedAncestorInclusive(const PXR_NS::SdfPath& primPath) const override; + //@} + + //! Path interface override. Forwards the call to the input scene index, + //! and warns about empty return paths. + //@{ + FVP_API + PXR_NS::SdfPath SceneIndexPath(const Ufe::Path& appPath) const override; + //@} + + FVP_API + PXR_NS::SdfPathVector GetFullySelectedPaths() const; + +protected: + void _PrimsAdded( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::AddedPrimEntries &entries) override; + + void _PrimsRemoved( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::RemovedPrimEntries &entries) override; + + void _PrimsDirtied( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::DirtiedPrimEntries &entries) override; + + +private: + SelectionSceneIndex( + const PXR_NS::HdSceneIndexBaseRefPtr &inputSceneIndex); + + SelectionSceneIndex_Impl::_SelectionSharedPtr _selection; + + const PathInterface* const _inputSceneIndexPathInterface; +}; + +} + +#endif diff --git a/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.cpp b/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.cpp new file mode 100644 index 0000000000..e8f691378c --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.cpp @@ -0,0 +1,186 @@ +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.h" + +#include "flowViewport/debugCodes.h" + +#include +#include +#include +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace { +const HdRetainedContainerDataSourceHandle sSelectedDisplayStyleDataSource + = HdRetainedContainerDataSource::New( + HdLegacyDisplayStyleSchemaTokens->displayStyle, + HdRetainedContainerDataSource::New( + HdLegacyDisplayStyleSchemaTokens->reprSelector, + HdRetainedTypedSampledDataSource>::New( + { HdReprTokens->refinedWireOnSurf, HdReprTokens->wireOnSurf, TfToken() }))); + +const HdRetainedContainerDataSourceHandle sUnselectedDisplayStyleDataSource + = HdRetainedContainerDataSource::New( + HdLegacyDisplayStyleSchemaTokens->displayStyle, + HdRetainedContainerDataSource::New( + HdLegacyDisplayStyleSchemaTokens->reprSelector, + HdRetainedTypedSampledDataSource>::New( + { HdReprTokens->refined, HdReprTokens->refined, TfToken() }))); + +const HdDataSourceLocator reprSelectorLocator( + HdLegacyDisplayStyleSchemaTokens->displayStyle, + HdLegacyDisplayStyleSchemaTokens->reprSelector); +} + +namespace FVP_NS_DEF { + +HdSceneIndexBaseRefPtr +WireframeSelectionHighlightSceneIndex::New(HdSceneIndexBaseRefPtr const &inputSceneIndex) +{ + return TfCreateRefPtr(new WireframeSelectionHighlightSceneIndex(inputSceneIndex)); +} + +const HdDataSourceLocator& WireframeSelectionHighlightSceneIndex::ReprSelectorLocator() +{ + return reprSelectorLocator; +} + +WireframeSelectionHighlightSceneIndex:: +WireframeSelectionHighlightSceneIndex(HdSceneIndexBaseRefPtr const &inputSceneIndex) + : PassThroughSelectionInterfaceSceneIndexBase(inputSceneIndex) +{} + +HdSceneIndexPrim +WireframeSelectionHighlightSceneIndex::GetPrim(const SdfPath &primPath) const +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::GetPrim(%s) called.\n", primPath.GetText()); + + auto prim = _GetInputSceneIndex()->GetPrim(primPath); + + // If prim is in excluded hierarchy, don't provide selection highlighting + // for it. Selection highlighting is done only on meshes. HYDRA-569: this + // means that HdsiImplicitSurfaceSceneIndex must be used before this scene + // index to convert implicit surfaces (e.g. USD cube / cone / sphere / + // capsule primitive types) to meshes. + if (!isExcluded(primPath) && prim.primType == HdPrimTypeTokens->mesh) { + prim.dataSource = HdOverlayContainerDataSource::New( + { prim.dataSource, HasFullySelectedAncestorInclusive(primPath) ? + sSelectedDisplayStyleDataSource : + sUnselectedDisplayStyleDataSource }); + } + return prim; + +} + +SdfPathVector +WireframeSelectionHighlightSceneIndex::GetChildPrimPaths(const SdfPath &primPath) const +{ + return _GetInputSceneIndex()->GetChildPrimPaths(primPath); +} + +void +WireframeSelectionHighlightSceneIndex::_PrimsAdded( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::AddedPrimEntries &entries) +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::_PrimsAdded() called.\n"); + + _SendPrimsAdded(entries); +} + +void +WireframeSelectionHighlightSceneIndex::_PrimsDirtied( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::DirtiedPrimEntries &entries) +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::_PrimsDirtied() called.\n"); + + HdSceneIndexObserver::DirtiedPrimEntries highlightEntries; + for (const auto& entry : entries) { + // If the dirtied prim is excluded, don't provide selection + // highlighting for it. + if (!isExcluded(entry.primPath) && + entry.dirtyLocators.Contains( + HdSelectionsSchema::GetDefaultLocator())) { + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg(" %s selections locator dirty.\n", entry.primPath.GetText()); + // All mesh prims recursively under the selection dirty prim have a + // dirty wireframe selection highlight. + dirtySelectionHighlightRecursive(entry.primPath, &highlightEntries); + } + } + + if (!highlightEntries.empty()) { + // Append all incoming dirty entries. + highlightEntries.reserve(highlightEntries.size()+entries.size()); + highlightEntries.insert( + highlightEntries.end(), entries.begin(), entries.end()); + _SendPrimsDirtied(highlightEntries); + } + else { + _SendPrimsDirtied(entries); + } +} + +void +WireframeSelectionHighlightSceneIndex::_PrimsRemoved( + const HdSceneIndexBase &sender, + const HdSceneIndexObserver::RemovedPrimEntries &entries) +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg("WireframeSelectionHighlightSceneIndex::_PrimsRemoved() called.\n"); + + _SendPrimsRemoved(entries); +} + +void WireframeSelectionHighlightSceneIndex::dirtySelectionHighlightRecursive( + const SdfPath& primPath, + HdSceneIndexObserver::DirtiedPrimEntries* highlightEntries +) +{ + TF_DEBUG(FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX) + .Msg(" marking %s wireframe highlight locator dirty.\n", primPath.GetText()); + highlightEntries->emplace_back(primPath, HdDataSourceLocatorSet(reprSelectorLocator)); + for (const auto& childPath : GetChildPrimPaths(primPath)) { + dirtySelectionHighlightRecursive(childPath, highlightEntries); + } +} + +void WireframeSelectionHighlightSceneIndex::addExcludedSceneRoot( + const PXR_NS::SdfPath& sceneRoot +) +{ + _excludedSceneRoots.emplace(sceneRoot); +} + +bool WireframeSelectionHighlightSceneIndex::isExcluded( + const PXR_NS::SdfPath& sceneRoot +) const +{ + for (const auto& excluded : _excludedSceneRoots) { + if (sceneRoot.HasPrefix(excluded)) { + return true; + } + } + return false; +} + +} diff --git a/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.h b/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.h new file mode 100644 index 0000000000..34a7ae590b --- /dev/null +++ b/lib/flowViewport/sceneIndex/fvpWireframeSelectionHighlightSceneIndex.h @@ -0,0 +1,100 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX_H +#define FVP_WIREFRAME_SELECTION_HIGHLIGHT_SCENE_INDEX_H + +#include "flowViewport/api.h" +#include "flowViewport/sceneIndex/fvpPassThroughSelectionInterfaceSceneIndex.h" + +#include + +#include + +namespace FVP_NS_DEF { + +// Pixar declarePtrs.h TF_DECLARE_REF_PTRS macro unusable, places resulting +// type in PXR_NS. +class WireframeSelectionHighlightSceneIndex; +typedef PXR_NS::TfRefPtr WireframeSelectionHighlightSceneIndexRefPtr; +typedef PXR_NS::TfRefPtr WireframeSelectionHighlightSceneIndexConstRefPtr; + +/// \class WireframeSelectionHighlightSceneIndex +/// +/// Uses Hydra HdRepr to add wireframe representation to selected objects +/// and their descendants. +/// +class WireframeSelectionHighlightSceneIndex + : public PassThroughSelectionInterfaceSceneIndexBase +{ +public: + + FVP_API + static PXR_NS::HdSceneIndexBaseRefPtr New( + const PXR_NS::HdSceneIndexBaseRefPtr& inputSceneIndex + ); + + FVP_API + static + const PXR_NS::HdDataSourceLocator& ReprSelectorLocator(); + + FVP_API + PXR_NS::HdSceneIndexPrim GetPrim(const PXR_NS::SdfPath &primPath) const override; + + FVP_API + PXR_NS::SdfPathVector GetChildPrimPaths(const PXR_NS::SdfPath &primPath) const override; + + // Add a scene root to the set of excluded scene roots. These are Hydra + // scene index prim hierarchies for which wireframe selection highlighting + // of meshes is not desired. + FVP_API + void addExcludedSceneRoot(const PXR_NS::SdfPath& sceneRoot); + +protected: + + FVP_API + WireframeSelectionHighlightSceneIndex( + const PXR_NS::HdSceneIndexBaseRefPtr& inputSceneIndex + ); + + FVP_API + void _PrimsAdded( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::AddedPrimEntries &entries) override; + + FVP_API + void _PrimsRemoved( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::RemovedPrimEntries &entries) override; + + FVP_API + void _PrimsDirtied( + const PXR_NS::HdSceneIndexBase &sender, + const PXR_NS::HdSceneIndexObserver::DirtiedPrimEntries &entries) override; + +private: + + void dirtySelectionHighlightRecursive( + const PXR_NS::SdfPath& primPath, + PXR_NS::HdSceneIndexObserver::DirtiedPrimEntries* highlightEntries + ); + + bool isExcluded(const PXR_NS::SdfPath& sceneRoot) const; + + std::set _excludedSceneRoots; +}; + +} + +#endif diff --git a/lib/flowViewport/selection/CMakeLists.txt b/lib/flowViewport/selection/CMakeLists.txt new file mode 100644 index 0000000000..f9ca43d0c4 --- /dev/null +++ b/lib/flowViewport/selection/CMakeLists.txt @@ -0,0 +1,31 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + fvpSelectionTracker.cpp + fvpSelectionTask.cpp +) + +set(HEADERS + fvpSelectionTracker.h + fvpSelectionTask.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/selection +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/flowViewport/selection +) diff --git a/lib/flowViewport/selection/fvpSelectionTask.cpp b/lib/flowViewport/selection/fvpSelectionTask.cpp new file mode 100644 index 0000000000..cf67e09d16 --- /dev/null +++ b/lib/flowViewport/selection/fvpSelectionTask.cpp @@ -0,0 +1,120 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "flowViewport/selection/fvpSelectionTask.h" +#include "flowViewport/selection/fvpSelectionTracker.h" + +#include "flowViewport/debugCodes.h" +#include "flowViewport/tokens.h" + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +// -------------------------------------------------------------------------- // + +SelectionTask::SelectionTask() + : HdTask(Id()) + , _lastVersion(-1) +{ + TF_DEBUG(FVP_SELECTION_TASK) + .Msg("SelectionTask::SelectionTask() called.\n"); +} + +SelectionTask::~SelectionTask() = default; + +void +SelectionTask::Sync(HdSceneDelegate* , + HdTaskContext* ctx, + HdDirtyBits* dirtyBits) +{ + TF_DEBUG(FVP_SELECTION_TASK) + .Msg("SelectionTask::Sync() called.\n"); + + HD_TRACE_FUNCTION(); + + if ((*dirtyBits) & HdChangeTracker::DirtyParams) { + // We track the version of selection tracker in the task to see if we + // need need to update. We don't have access to the selection tracker + // (as it is in the task context) so we reset the version to -1 to + // cause a version mismatch and force the update. + _lastVersion = -1; + } + + *dirtyBits = HdChangeTracker::Clean; +} + +void +SelectionTask::Prepare( + HdTaskContext* ctx, + HdRenderIndex* renderIndex +) +{ + TF_DEBUG(FVP_SELECTION_TASK) + .Msg("SelectionTask::Prepare() called.\n"); + + Fvp::SelectionTrackerSharedPtr snTracker; + if (!_GetTaskContextData(ctx, FvpTokens->fvpSelectionState, &snTracker)) { + return; + } + + auto trackerVersion = snTracker->GetVersion(); + + if (trackerVersion == _lastVersion) { + return; + } + + _lastVersion = trackerVersion; + + // FLOW_VIEWPORT_TODO Get the selection from the selection tracker here, + // compute selection-derived buffers here, and place them back in the + // context for downstream tasks to use. PPT, 27-Sep-2023. +} + +void +SelectionTask::Execute(HdTaskContext* ctx) +{ + TF_DEBUG(FVP_SELECTION_TASK) + .Msg("SelectionTask::Execute() called.\n"); + + HD_TRACE_FUNCTION(); + HF_MALLOC_TAG_FUNCTION(); + + // Note that selectionTask comes after renderTask. +} + +} diff --git a/lib/flowViewport/selection/fvpSelectionTask.h b/lib/flowViewport/selection/fvpSelectionTask.h new file mode 100644 index 0000000000..f8cc719f7a --- /dev/null +++ b/lib/flowViewport/selection/fvpSelectionTask.h @@ -0,0 +1,102 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_SELECTION_TASK_H +#define FVP_SELECTION_TASK_H + +#include "flowViewport/api.h" + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE +class HdRenderIndex; +class HdSceneDelegate; +PXR_NAMESPACE_CLOSE_SCOPE + +namespace FVP_NS_DEF { + +/// \class SelectionTask +/// +/// Placeholder for +/// https://github.com/PixarAnimationStudios/OpenUSD/blob/release/pxr/imaging/hdx/selectionTask.h +/// +class SelectionTask : public PXR_NS::HdTask +{ +public: + FVP_API + + SelectionTask(); + + FVP_API + ~SelectionTask() override; + + FVP_API + const PXR_NS::SdfPath& Id() { + static auto id = PXR_NS::SdfPath("FlowViewportSelectionTask"); + return id; + } + + /// Sync the render pass resources + FVP_API + void Sync(PXR_NS::HdSceneDelegate* delegate, + PXR_NS::HdTaskContext* ctx, + PXR_NS::HdDirtyBits* dirtyBits) override; + + + /// Prepare the tasks resources + FVP_API + void Prepare(PXR_NS::HdTaskContext* ctx, + PXR_NS::HdRenderIndex* renderIndex) override; + + /// Execute render pass task + FVP_API + void Execute(PXR_NS::HdTaskContext* ctx) override; + + +private: + int _lastVersion; + + SelectionTask(const SelectionTask &) = delete; + SelectionTask &operator =(const SelectionTask &) = delete; +}; + +} + +#endif //FVP_SELECTION_TASK_H + diff --git a/lib/flowViewport/selection/fvpSelectionTracker.cpp b/lib/flowViewport/selection/fvpSelectionTracker.cpp new file mode 100644 index 0000000000..f063c50597 --- /dev/null +++ b/lib/flowViewport/selection/fvpSelectionTracker.cpp @@ -0,0 +1,116 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +/// Placeholder for +/// +/// https://github.com/PixarAnimationStudios/OpenUSD/blob/release/pxr/imaging/hdx/selectionTracker.cpp +/// +/// which is Hydra Storm-centric. To be revised. PPT, 27-Sep-2023. + +#include "flowViewport/selection/fvpSelectionTracker.h" + +#include "flowViewport/debugCodes.h" + +#include +#include +#include +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace FVP_NS_DEF { + +class SelectionTracker::_Selection +{ +public: + // Selection task has initial selection version at -1, so match that. + _Selection() : _lastVersion(-1) {} + + // Returns the selection from the scene index. + HdSelectionSharedPtr + GetSelection(const HdRenderIndex * const index) + { + // Tell scene index observer what scene index to observe. + // FLOW_VIEWPORT_TODO Why are we setting the scene index onto the + // observer at every call? PPT, 27-Sep-2023. + _observer.SetSceneIndex(index->GetTerminalSceneIndex()); + + // Recompute if changed since last time it was computed. + if (_lastVersion != GetVersion()) { + _selection = _observer.GetSelection(); + _lastVersion = GetVersion(); + } + return _selection; + } + + // Version number for the selection. + int GetVersion() const { + return _observer.GetVersion(); + } + +private: + // Cache the selection. The version of the selection cached here + // is stored as _lastVersion. + HdSelectionSharedPtr _selection; + int _lastVersion; + + HdxSelectionSceneIndexObserver _observer; +}; + +SelectionTracker::SelectionTracker() + : _selection(std::make_unique<_Selection>()) +{} + +SelectionTracker::~SelectionTracker() = default; + +int +SelectionTracker::GetVersion() const +{ + return _selection->GetVersion(); +} + +HdSelectionSharedPtr SelectionTracker::GetSelection(const HdRenderIndex* index) const +{ + TF_DEBUG(FVP_SELECTION_TRACKER) + .Msg("SelectionTracker::GetSelection() called.\n"); + + return _selection->GetSelection(index); +} + +} diff --git a/lib/flowViewport/selection/fvpSelectionTracker.h b/lib/flowViewport/selection/fvpSelectionTracker.h new file mode 100644 index 0000000000..0e9c541c5a --- /dev/null +++ b/lib/flowViewport/selection/fvpSelectionTracker.h @@ -0,0 +1,101 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_SELECTION_TRACKER_H +#define FVP_SELECTION_TRACKER_H + +#include "flowViewport/api.h" + +#include +#include // For HdSelectionSharedPtr typedef + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class HdRenderIndex; + +PXR_NAMESPACE_CLOSE_SCOPE + +namespace FVP_NS_DEF { + +using SelectionTrackerSharedPtr = + std::shared_ptr; + +/// Placeholder for +/// +/// https://github.com/PixarAnimationStudios/OpenUSD/blob/release/pxr/imaging/hdx/selectionTracker.h +/// +/// which is Hydra Storm-centric. To be revised. PPT, 27-Sep-2023. + +/// \class SelectionTracker +/// +/// The selection tracker owns the HdSelection and the selection scene index +/// observer that maintains the selection up to date. +/// +/// HdxSelectionTask takes SelectionTracker as a task parameter, to inject +/// the selection into the list of tasks. + +class SelectionTracker +{ +public: + FVP_API + SelectionTracker(); + FVP_API + virtual ~SelectionTracker(); + + /// Returns a monotonically increasing version number, which increments + /// whenever the selection has changed. Note that this number may + /// overflow and become negative, thus clients should use a not-equal + /// comparison. + FVP_API + int GetVersion() const; + + FVP_API + PXR_NS::HdSelectionSharedPtr GetSelection(const PXR_NS::HdRenderIndex *index) const; + +private: + + // A helper class to obtain the selection computed by querying the scene + // indices (with the HdxSelectionSceneIndexObserver). + class _Selection; + std::unique_ptr<_Selection> _selection; +}; + +} + +#endif //FVP_SELECTION_TRACKER_H diff --git a/lib/flowViewport/tokens.cpp b/lib/flowViewport/tokens.cpp new file mode 100644 index 0000000000..7d05fe6384 --- /dev/null +++ b/lib/flowViewport/tokens.cpp @@ -0,0 +1,53 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "flowViewport/tokens.h" + +// FLOW_VIEWPORT_TODO Figure out how to put tokens into non-Pixar namespace. +// The following does not work: +// +// PXR_NAMESPACE_USING_DIRECTIVE +// +// namespace FVP_NS_DEF { +// TF_DEFINE_PUBLIC_TOKENS(FvpTokens, FVP_TOKENS); +// } +// +// PPT, 18-Sep-2023. + +PXR_NAMESPACE_OPEN_SCOPE +TF_DEFINE_PUBLIC_TOKENS(FvpTokens, FVP_TOKENS); +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/flowViewport/tokens.h b/lib/flowViewport/tokens.h new file mode 100644 index 0000000000..81f11add3d --- /dev/null +++ b/lib/flowViewport/tokens.h @@ -0,0 +1,64 @@ +// +// Copyright 2016 Pixar +// +// Licensed under the Apache License, Version 2.0 (the "Apache License") +// with the following modification; you may not use this file except in +// compliance with the Apache License and the following modification to it: +// Section 6. Trademarks. is deleted and replaced with: +// +// 6. Trademarks. This License does not grant permission to use the trade +// names, trademarks, service marks, or product names of the Licensor +// and its affiliates, except as required to comply with Section 4(c) of +// the License and to reproduce the content of the NOTICE file. +// +// You may obtain a copy of the Apache License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the Apache License with the above modification is +// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the Apache License for the specific +// language governing permissions and limitations under the Apache License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef FVP_TOKENS_H +#define FVP_TOKENS_H + +#include "flowViewport/api.h" + +#include "pxr/pxr.h" +#include "pxr/base/tf/staticTokens.h" + +PXR_NAMESPACE_OPEN_SCOPE +#define FVP_TOKENS (fvpSelectionState) + +TF_DECLARE_PUBLIC_TOKENS(FvpTokens, FVP_API, FVP_TOKENS); + +PXR_NAMESPACE_CLOSE_SCOPE + +// FLOW_VIEWPORT_TODO Figure out how to put tokens into non-Pixar namespace. +// The following does not work: +// +// #define FVP_TOKENS (fvpSelectionState) +// +// namespace FVP_NS_DEF { +// TF_DECLARE_PUBLIC_TOKENS(FvpTokens, FVP_API, FVP_TOKENS); +// } +// +// PPT, 18-Sep-2023. + +#endif //FVP_TOKENS_H diff --git a/lib/mayaHydra/CMakeLists.txt b/lib/mayaHydra/CMakeLists.txt new file mode 100644 index 0000000000..df12f980bd --- /dev/null +++ b/lib/mayaHydra/CMakeLists.txt @@ -0,0 +1,3 @@ +add_subdirectory(mayaPlugin) +add_subdirectory(hydraExtensions) +add_subdirectory(ufeExtensions) diff --git a/lib/mayaHydra/hydraExtensions/CMakeLists.txt b/lib/mayaHydra/hydraExtensions/CMakeLists.txt new file mode 100644 index 0000000000..e8997ec658 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/CMakeLists.txt @@ -0,0 +1,214 @@ +set(TARGET_NAME mayaHydraLib) + +add_library(${TARGET_NAME} SHARED) + +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + debugCodes.cpp + hydraUtils.cpp + interfaceImp.cpp + mayaUtils.cpp + mixedUtils.cpp + mayaHydraSceneProducer.cpp +) + +set(HEADERS + api.h + debugCodes.h + hydraUtils.h + interface.h + interfaceImp.h + mayaUtils.h + mixedUtils.h + mayaHydraSceneProducer.h +) + +# ----------------------------------------------------------------------------- +# compiler configuration +# ----------------------------------------------------------------------------- +target_compile_definitions(${TARGET_NAME} + PRIVATE + MAYAHYDRALIB_EXPORT + $<$:OSMac_> + # Copy-pasted from lib/mayaUsd/CMakeLists.txt +) + +mayaHydra_compile_config(${TARGET_NAME}) + +if(NOT BUILD_MAYAUSD_LIBRARY) + # ----------------------------------------------------------------------------- + # include directories + # ----------------------------------------------------------------------------- + target_include_directories(${TARGET_NAME} + PUBLIC + ${MAYA_INCLUDE_DIRS} + ${PYTHON_INCLUDE_DIR} + ${PXR_INCLUDE_DIRS} + ${BOOST_INCLUDE_DIR} + ${CMAKE_BINARY_DIR}/include + PRIVATE + $<$:${UFE_INCLUDE_DIR}> + ) + + target_compile_definitions(${TARGET_NAME} + PUBLIC + PXR_PLUGINPATH_NAME=${PXR_OVERRIDE_PLUGINPATH_NAME} + $<$:TBB_USE_DEBUG> + $<$:BOOST_DEBUG_PYTHON> + $<$:BOOST_LINKING_PYTHON> + PRIVATE + MAYAUSD_MACROS_EXPORT + MAYAUSD_CORE_EXPORT + MFB_PACKAGE_NAME="${PROJECT_NAME}" + MFB_ALT_PACKAGE_NAME="${PROJECT_NAME}" + MFB_PACKAGE_MODULE="${PROJECT_NAME}" + $<$:OSMac_> + $<$:LINUX> + $<$:GL_GLEXT_PROTOTYPES> + $<$:GLX_GLXEXT_PROTOTYPES> + $<$:WANT_MATERIALX_BUILD> + $<$:BUILD_MAYAHYDRALIB> + + # this flag is needed when building maya-usd in Maya + $<$:WIN32> + ) +endif() + +if(DEFINED MAYAUSD_VERSION) + target_compile_definitions(${TARGET_NAME} + PRIVATE + MAYAUSD_VERSION=${MAYAUSD_VERSION} + ) +endif() +if(DEFINED MAYAHYDRA_VERSION) + target_compile_definitions(${TARGET_NAME} + PRIVATE + MAYAHYDRA_VERSION=${MAYAHYDRA_VERSION} + ) +endif() +if(DEFINED MAYAHYDRA_CUT_ID) + target_compile_definitions(${TARGET_NAME} + PRIVATE + MAYAHYDRA_CUT_ID=${MAYAHYDRA_CUT_ID} + ) +endif() + + # ----------------------------------------------------------------------------- + # link libraries + # ----------------------------------------------------------------------------- +if(BUILD_MAYAUSD_LIBRARY) + target_link_libraries(${TARGET_NAME} + PUBLIC + mayaUsd + PRIVATE + $<$:hio> + ) +else() + target_link_libraries(${TARGET_NAME} + PUBLIC + ar + gf + hd + hdx + js + kind + plug + sdf + tf + trace + usd + usdGeom + usdImaging + usdImagingGL + usdLux + usdRender + usdRi + usdShade + usdSkel + usdUtils + ufeExtensions + flowViewport + ${MAYA_LIBRARIES} + PRIVATE + $<$:hio> + ${PYTHON_LIBRARIES} + $<$:${UFE_LIBRARY}> + ) +endif() + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +set(SRCFILE ${CMAKE_CURRENT_SOURCE_DIR}/mayaHydra.h.src) +set(DSTFILE ${CMAKE_BINARY_DIR}/include/mayaHydraLib/mayaHydra.h) +configure_file(${SRCFILE} ${DSTFILE}) + +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME} +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +set(LIBFILENAME ${CMAKE_SHARED_LIBRARY_PREFIX}${TARGET_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}) +set(PLUG_INFO_PATH "plugInfo.json") +set(SHADER_DEFS_PATH +"shaderDefs.usda" +) + +set(PLUG_INFO_LIBRARY_PATH "../../../${LIBFILENAME}") +set(PLUG_INFO_RESOURCE_PATH "resources") + +configure_file(${PLUG_INFO_PATH} ${CMAKE_CURRENT_BINARY_DIR}/${PLUG_INFO_PATH}) + +# configure shader files +foreach(X ${SHADER_DEFS_PATH}) + configure_file(${X} ${CMAKE_CURRENT_BINARY_DIR}/${X}) +endforeach(X) + + +install(TARGETS ${TARGET_NAME} + LIBRARY + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib + RUNTIME + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib +) + +install(FILES + ${CMAKE_CURRENT_BINARY_DIR}/${PLUG_INFO_PATH} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/lib/usd/mayaHydraLib/resources +) + +# install shaders +foreach(X ${SHADER_DEFS_PATH}) + install(FILES + ${CMAKE_CURRENT_BINARY_DIR}/${X} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/lib/usd/mayaHydraLib/resources + ) +endforeach(X) + + +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/mayaHydraLib +) + +if(IS_WINDOWS) + install(FILES $ + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib OPTIONAL) +endif() + +# ----------------------------------------------------------------------------- +# subdirectories +# ----------------------------------------------------------------------------- +add_subdirectory(adapters) +add_subdirectory(delegates) +add_subdirectory(sceneIndex) diff --git a/lib/mayaHydra/hydraExtensions/adapters/CMakeLists.txt b/lib/mayaHydra/hydraExtensions/adapters/CMakeLists.txt new file mode 100644 index 0000000000..4459b895c3 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/CMakeLists.txt @@ -0,0 +1,59 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + adapter.cpp + adapterDebugCodes.cpp + adapterRegistry.cpp + aiSkydomeLightAdapter.cpp + areaLightAdapter.cpp + cameraAdapter.cpp + dagAdapter.cpp + renderItemAdapter.cpp + directionalLightAdapter.cpp + lightAdapter.cpp + materialAdapter.cpp + materialNetworkConverter.cpp + mayaAttrs.cpp + meshAdapter.cpp + nurbsCurveAdapter.cpp + pointLightAdapter.cpp + shapeAdapter.cpp + spotLightAdapter.cpp + tokens.cpp +) + +set(HEADERS + adapter.h + adapterDebugCodes.h + adapterRegistry.h + cameraAdapter.h + constantShadowMatrix.h + dagAdapter.h + renderItemAdapter.h + lightAdapter.h + materialAdapter.h + materialNetworkConverter.h + mayaAttrs.h + shapeAdapter.h + tokens.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/adapters +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/mayaHydraLib/adapters +) diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/adapter.cpp new file mode 100644 index 0000000000..989544ceb4 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapter.cpp @@ -0,0 +1,124 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "adapter.h" + +#include +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_REGISTRY_FUNCTION(TfType) { TfType::Define(); } + +namespace { + +void _preRemoval(MObject& node, void* clientData) +{ + TF_UNUSED(node); + auto* adapter = reinterpret_cast(clientData); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Pre-removal callback triggered for prim (%s)\n", adapter->GetID().GetText()); + adapter->GetSceneProducer()->RemoveAdapter(adapter->GetID()); +} + +void _nameChanged(MObject& node, const MString& str, void* clientData) +{ + TF_UNUSED(node); + TF_UNUSED(str); + auto* adapter = reinterpret_cast(clientData); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Name-changed callback triggered for prim (%s)\n", adapter->GetID().GetText()); + adapter->RemoveCallbacks(); + adapter->GetSceneProducer()->RecreateAdapterOnIdle(adapter->GetID(), adapter->GetNode()); +} + +} // namespace + +// MayaHydraAdapter is the base class for all adapters. An adapter is used to translate from Maya +// data to hydra data. +MayaHydraAdapter::MayaHydraAdapter( + const MObject& node, + const SdfPath& id, + MayaHydraSceneProducer* producer) + : _id(id) + , _sceneProducer(producer) + , _node(node) +{ +} + +MayaHydraAdapter::~MayaHydraAdapter() { RemoveCallbacks(); } + +void MayaHydraAdapter::AddCallback(MCallbackId callbackId) { _callbacks.push_back(callbackId); } + +void MayaHydraAdapter::RemoveCallbacks() +{ + if (_callbacks.empty()) { + return; + } + + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Removing all adapter callbacks for prim (%s).\n", GetID().GetText()); + for (auto c : _callbacks) { + MMessage::removeCallback(c); + } + std::vector().swap(_callbacks); +} + +VtValue MayaHydraAdapter::Get(const TfToken& key) +{ + TF_UNUSED(key); + return {}; +}; + +bool MayaHydraAdapter::HasType(const TfToken& typeId) const +{ + TF_UNUSED(typeId); + return false; +} + +void MayaHydraAdapter::CreateCallbacks() +{ + if (_node != MObject::kNullObj) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating generic adapter callbacks for prim (%s).\n", GetID().GetText()); + + MStatus status; + auto id = MNodeMessage::addNodePreRemovalCallback(_node, _preRemoval, this, &status); + if (status) { + AddCallback(id); + } + id = MNodeMessage::addNameChangedCallback(_node, _nameChanged, this, &status); + if (status) { + AddCallback(id); + } + } +} + +MStatus MayaHydraAdapter::Initialize() +{ + auto status = MayaAttrs::initialize(); + if (status) { + MayaHydraMaterialNetworkConverter::initialize(); + } + return status; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapter.h b/lib/mayaHydra/hydraExtensions/adapters/adapter.h new file mode 100644 index 0000000000..d055e74a7f --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapter.h @@ -0,0 +1,106 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#ifndef MAYAHYDRALIB_ADAPTER_H +#define MAYAHYDRALIB_ADAPTER_H + +#include +#include + +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraAdapter is the base class for all adapters. An adapter is used to translate from + * Maya data to hydra data. + */ + +class MayaHydraAdapter +{ +public: + MAYAHYDRALIB_API + MayaHydraAdapter(const MObject& node, const SdfPath& id, MayaHydraSceneProducer* producer); + MAYAHYDRALIB_API + virtual ~MayaHydraAdapter(); + + const SdfPath& GetID() const { return _id; } + MayaHydraSceneProducer* GetSceneProducer() const { return _sceneProducer; } + MAYAHYDRALIB_API + void AddCallback(MCallbackId callbackId); + MAYAHYDRALIB_API + virtual void RemoveCallbacks(); + MAYAHYDRALIB_API + virtual VtValue Get(const TfToken& key); + const MObject& GetNode() const { return _node; } + MAYAHYDRALIB_API + virtual bool IsSupported() const = 0; + MAYAHYDRALIB_API + virtual bool HasType(const TfToken& typeId) const; + MAYAHYDRALIB_API + virtual bool GetVisible() { return true; } + + MAYAHYDRALIB_API + virtual void CreateCallbacks(); + virtual void MarkDirty(HdDirtyBits dirtyBits) = 0; + virtual void RemovePrim() = 0; + virtual void Populate() = 0; + + MAYAHYDRALIB_API + static MStatus Initialize(); + + bool IsPopulated() const { return _isPopulated; } + + MAYAHYDRALIB_API + virtual HdMeshTopology GetMeshTopology() { return {}; } + MAYAHYDRALIB_API + virtual HdBasisCurvesTopology GetBasisCurvesTopology() { return {}; } + MAYAHYDRALIB_API + virtual TfToken GetRenderTag() const { return TfToken(); } + MAYAHYDRALIB_API + virtual GfMatrix4d GetTransform() { return GfMatrix4d(); } + MAYAHYDRALIB_API + virtual HdPrimvarDescriptorVector GetPrimvarDescriptors(HdInterpolation interpolation) + { + return HdPrimvarDescriptorVector(); + } + MAYAHYDRALIB_API + virtual bool GetDoubleSided() const { return true; } + MAYAHYDRALIB_API + virtual HdCullStyle GetCullStyle() const { return HdCullStyleNothing; } + MAYAHYDRALIB_API + virtual HdDisplayStyle GetDisplayStyle() { return { 0, false, false }; } + +protected: + SdfPath _id; + std::vector _callbacks; + MayaHydraSceneProducer* _sceneProducer; + MObject _node; + + bool _isPopulated = false; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.cpp b/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.cpp new file mode 100644 index 0000000000..7c0ffc8ae2 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.cpp @@ -0,0 +1,72 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "adapterDebugCodes.h" + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// Some variables to enable debug printing information + +TF_REGISTRY_FUNCTION(TfDebug) +{ + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_CALLBACKS, + "Print information adding and removal of adapter callbacks."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_CURVE_PLUG_DIRTY, + "Print information when a nurbs curve prim is dirtied due to a plug " + "being dirtied."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_CURVE_UNHANDLED_PLUG_DIRTY, + "Print information when a nurbs curve prim is NOT dirtied, even though " + "a plug was dirtied."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_DAG_HIERARCHY, "Print information related to dag hierarchy changes."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_DAG_PLUG_DIRTY, "Print information about the dag node plug dirtying."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_GET, "Print information about 'Get' calls to the adapter."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE, + "Print information about 'LightParamValue' " + "calls to the light adapters."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_IMAGEPLANES, "Print information about drawing image planes."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_LIGHT_SHADOWS, "Print information about shadow rendering."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_MATERIALS, "Print information about converting materials."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_MESH_PLUG_DIRTY, + "Print information about the mesh plug dirtying handled."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_ADAPTER_MESH_UNHANDLED_PLUG_DIRTY, + "Print information about unhandled mesh plug dirtying."); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.h b/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.h new file mode 100644 index 0000000000..128f65c585 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapterDebugCodes.h @@ -0,0 +1,46 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_ADAPTER_DEBUG_CODES_H +#define MAYAHYDRALIB_ADAPTER_DEBUG_CODES_H + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +//! \brief Some variables to enable debug printing information for adapters + +// clang-format off +TF_DEBUG_CODES( + MAYAHYDRALIB_ADAPTER_CALLBACKS, + MAYAHYDRALIB_ADAPTER_CURVE_PLUG_DIRTY, + MAYAHYDRALIB_ADAPTER_CURVE_UNHANDLED_PLUG_DIRTY, + MAYAHYDRALIB_ADAPTER_DAG_HIERARCHY, + MAYAHYDRALIB_ADAPTER_DAG_PLUG_DIRTY, + MAYAHYDRALIB_ADAPTER_GET, + MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE, + MAYAHYDRALIB_ADAPTER_IMAGEPLANES, + MAYAHYDRALIB_ADAPTER_LIGHT_SHADOWS, + MAYAHYDRALIB_ADAPTER_MATERIALS, + MAYAHYDRALIB_ADAPTER_MESH_PLUG_DIRTY, + MAYAHYDRALIB_ADAPTER_MESH_UNHANDLED_PLUG_DIRTY, + MAYAHYDRALIB_ADAPTER_MATERIALS_PRINT_PARAMETERS_VALUES + ); +// clang-format on + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_ADAPTER_DEBUG_CODES_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.cpp b/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.cpp new file mode 100644 index 0000000000..fcb25ffa2d --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.cpp @@ -0,0 +1,136 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "adapterRegistry.h" + +#if defined(MAYAUSD_VERSION) +#include +#endif + +#include +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_INSTANTIATE_SINGLETON(MayaHydraAdapterRegistry); + +// An adapter is used to translate from Maya data to hydra data. MayaHydraAdapterRegistry is used to +// register/retrieve the adapters. +void MayaHydraAdapterRegistry::RegisterShapeAdapter( + const TfToken& type, + ShapeAdapterCreator creator) +{ + GetInstance()._dagAdapters.insert({ type, creator }); +} + +MayaHydraAdapterRegistry::ShapeAdapterCreator +MayaHydraAdapterRegistry::GetShapeAdapterCreator(const MDagPath& dag) +{ + MFnDependencyNode depNode(dag.node()); + ShapeAdapterCreator ret = nullptr; + TfMapLookup(GetInstance()._dagAdapters, TfToken(depNode.typeName().asChar()), &ret); + + return ret; +} + +void MayaHydraAdapterRegistry::RegisterLightAdapter( + const TfToken& type, + LightAdapterCreator creator) +{ + GetInstance()._lightAdapters.insert({ type, creator }); +} + +MayaHydraAdapterRegistry::LightAdapterCreator +MayaHydraAdapterRegistry::GetLightAdapterCreator(const MDagPath& dag) +{ + return GetLightAdapterCreator(dag.node()); +} + +MayaHydraAdapterRegistry::LightAdapterCreator +MayaHydraAdapterRegistry::GetLightAdapterCreator(const MObject& node) +{ + MFnDependencyNode depNode(node); + LightAdapterCreator ret = nullptr; + TfMapLookup(GetInstance()._lightAdapters, TfToken(depNode.typeName().asChar()), &ret); + return ret; +} + +void MayaHydraAdapterRegistry::RegisterCameraAdapter( + const TfToken& type, + CameraAdapterCreator creator) +{ + GetInstance()._cameraAdapters.insert({ type, creator }); +} + +MayaHydraAdapterRegistry::CameraAdapterCreator +MayaHydraAdapterRegistry::GetCameraAdapterCreator(const MDagPath& dag) +{ + MFnDependencyNode depNode(dag.node()); + CameraAdapterCreator ret = nullptr; + TfMapLookup(GetInstance()._cameraAdapters, TfToken(depNode.typeName().asChar()), &ret); + return ret; +} + +void MayaHydraAdapterRegistry::RegisterMaterialAdapter( + const TfToken& type, + MaterialAdapterCreator creator) +{ + GetInstance()._materialAdapters.insert({ type, creator }); +} + +MayaHydraAdapterRegistry::MaterialAdapterCreator +MayaHydraAdapterRegistry::GetMaterialAdapterCreator(const MObject& node) +{ + MFnDependencyNode depNode(node); + MaterialAdapterCreator ret = nullptr; + TfMapLookup(GetInstance()._materialAdapters, TfToken(depNode.typeName().asChar()), &ret); + return ret; +} + +void MayaHydraAdapterRegistry::LoadAllPlugin() +{ + static std::once_flag loadAllOnce; + std::call_once(loadAllOnce, []() { + TfRegistryManager::GetInstance().SubscribeTo(); + + const TfType& adapterType = TfType::Find(); + if (adapterType.IsUnknown()) { + TF_CODING_ERROR("Could not find MayaHydraAdapter type"); + return; + } + + std::set adapterTypes; + adapterType.GetAllDerivedTypes(&adapterTypes); + + PlugRegistry& plugReg = PlugRegistry::GetInstance(); + + for (auto& subType : adapterTypes) { + const PlugPluginPtr pluginForType = plugReg.GetPluginForType(subType); + if (!pluginForType) { + TF_CODING_ERROR("Could not find plugin for '%s'", subType.GetTypeName().c_str()); + return; + } + pluginForType->Load(); + } + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.h b/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.h new file mode 100644 index 0000000000..34fb71f29b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/adapterRegistry.h @@ -0,0 +1,95 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_ADAPTER_REGISTRY_H +#define MAYAHYDRALIB_ADAPTER_REGISTRY_H + +#include +#include +#include +#include +#include +#include + +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief An adapter is used to translate from Maya data to hydra data. MayaHydraAdapterRegistry is + * used to register/retrieve the adapters. + */ + +class MayaHydraAdapterRegistry : public TfSingleton +{ + friend class TfSingleton; + MAYAHYDRALIB_API + MayaHydraAdapterRegistry() = default; + +public: + using ShapeAdapterCreator + = std::function; + MAYAHYDRALIB_API + static void RegisterShapeAdapter(const TfToken& type, ShapeAdapterCreator creator); + + MAYAHYDRALIB_API + static ShapeAdapterCreator GetShapeAdapterCreator(const MDagPath& dag); + + using LightAdapterCreator + = std::function; + MAYAHYDRALIB_API + static void RegisterLightAdapter(const TfToken& type, LightAdapterCreator creator); + + MAYAHYDRALIB_API + static LightAdapterCreator GetLightAdapterCreator(const MDagPath& dag); + + MAYAHYDRALIB_API + static LightAdapterCreator GetLightAdapterCreator(const MObject& dag); + + using MaterialAdapterCreator = std::function< + MayaHydraMaterialAdapterPtr(const SdfPath&, MayaHydraSceneProducer*, const MObject&)>; + MAYAHYDRALIB_API + static void RegisterMaterialAdapter(const TfToken& type, MaterialAdapterCreator creator); + + MAYAHYDRALIB_API + static MaterialAdapterCreator GetMaterialAdapterCreator(const MObject& node); + + using CameraAdapterCreator + = std::function; + MAYAHYDRALIB_API + static void RegisterCameraAdapter(const TfToken& type, CameraAdapterCreator creator); + + MAYAHYDRALIB_API + static CameraAdapterCreator GetCameraAdapterCreator(const MDagPath& dag); + + /// Find all MayaHydraAdapter plug-ins, and load them all + MAYAHYDRALIB_API + static void LoadAllPlugin(); + +private: + std::unordered_map _dagAdapters; + std::unordered_map _lightAdapters; + std::unordered_map _materialAdapters; + std::unordered_map _cameraAdapters; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_ADAPTER_REGISTRY_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/aiSkydomeLightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/aiSkydomeLightAdapter.cpp new file mode 100644 index 0000000000..2cc3c5f0a7 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/aiSkydomeLightAdapter.cpp @@ -0,0 +1,248 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraAiSkyDomeLightAdapter is used to handle the translation from an Arnold skydome + * light to hydra. + */ +class MayaHydraAiSkyDomeLightAdapter : public MayaHydraLightAdapter +{ + ///To be able to create a dummy texture (see _dummyTexturePath) + static MHWRender::MTextureManager* _pTextureManager; + + ///Temp folder path from the OS + static std::string _tmpFolderPath; + + /**_dummyTexturePath is the fullpath filename for a dummy 1x1 texture file used when there + * is no texture connected to the color of the Arnold sky dome light + * as Hydra always wants a texture and ignores the color if no texture is present. + */ + std::string _dummyTextureFullPathFilename; + ///This is only the filename of the dummy texture to be saved + std::string _dummyTextureFilenameOnly; + + ///Is the color attribute of the sky dome light connected to something ? + bool _colorIsConnected = false; + +public: + MayaHydraAiSkyDomeLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraLightAdapter(producer, dag) + { + //Init static variables if needed + if (! _pTextureManager){ + MHWRender::MRenderer* const renderer = MHWRender::MRenderer::theRenderer(); + _pTextureManager = renderer ? renderer->getTextureManager() : nullptr; + } + + if (_tmpFolderPath.empty()){ + const char *tmpDir = getenv("TMPDIR"); + if (tmpDir) { + _tmpFolderPath = tmpDir; + }else { + tmpDir = getenv("TEMP"); + if (tmpDir) { + _tmpFolderPath = tmpDir; + } + } + } + + _dummyTextureFilenameOnly = TfStringPrintf("/HydraAiSkyDomeLightTex__%p__tmp.png", this); + } + + virtual ~MayaHydraAiSkyDomeLightAdapter() + { + // Delete the dummy texture files if they exist + if (! _dummyTextureFullPathFilename.empty()){ + auto dummyTextureFullPathFilename1 = _tmpFolderPath + _dummyTextureFilenameOnly; + std::remove(dummyTextureFullPathFilename1.c_str()); + } + } + + const TfToken& LightType() const override { return HdPrimTypeTokens->domeLight; } + + VtValue GetLightParamValue(const TfToken& paramName) override + { + MStatus status; + MFnDependencyNode light(GetNode(), &status); + if (ARCH_UNLIKELY(!status)) { + return {}; + } + + // We are not using precomputed attributes here, because we don't have + // a guarantee that mtoa will be loaded before mayaHydra. + if (paramName == HdLightTokens->color) { + const auto plug = light.findPlug("color", true); + MPlugArray conns; + plug.connectedTo(conns, true, false); + _colorIsConnected = (conns.length() > 0); + if (_colorIsConnected){ + return VtValue(GfVec3f(1.0f, 1.0f, 1.0f));// When there is a connection, return a white color. + } + + // If no texture is found then get unconnected plug value and make a 1x1 texture of constant color using it. + float r = 0.5f; + float g = 0.5f; + float b = 0.5f; + if (!plug.isNull()) + { + plug.child(0).getValue(r); + plug.child(1).getValue(g); + plug.child(2).getValue(b); + } + + const float rClamped = (r <= 1.f) ? r : 1.0f; + const float gClamped = (g <= 1.f) ? g : 1.0f; + const float bClamped = (b <= 1.f) ? b : 1.0f; + + unsigned char texData[4]; + texData[0] = (unsigned char)(255 * rClamped); + texData[1] = (unsigned char)(255 * gClamped); + texData[2] = (unsigned char)(255 * bClamped); + texData[3] = 255; + + //Create a 1 x 1 constant color texture + MHWRender::MTextureDescription desc; + desc.setToDefault2DTexture(); + desc.fWidth = 1; + desc.fHeight = 1; + desc.fFormat = MHWRender::kR8G8B8A8_UNORM; + if (_pTextureManager){ + auto pTexture = _pTextureManager->acquireTexture("", desc, texData);//In memory + if (pTexture && (!_tmpFolderPath.empty()) && (!_dummyTextureFilenameOnly.empty())){ + pTexture->setHasAlpha(true); + _dummyTextureFullPathFilename = _tmpFolderPath + _dummyTextureFilenameOnly; + // This texture will be used in the HdLightTokens->textureFile parameter + _pTextureManager->saveTexture(pTexture, MString(_dummyTextureFullPathFilename.c_str())); + } + } + return VtValue(GfVec3f(r,g,b)); + } else if (paramName == HdLightTokens->intensity) { + return VtValue(light.findPlug("intensity", true).asFloat()); + } else if (paramName == HdLightTokens->diffuse) { + MPlug aiDiffuse = light.findPlug("aiDiffuse", true, &status); + if (status == MS::kSuccess) { + return VtValue(aiDiffuse.asFloat()); + } + } else if (paramName == HdLightTokens->specular) { + MPlug aiSpecular = light.findPlug("aiSpecular", true, &status); + if (status == MS::kSuccess) { + return VtValue(aiSpecular.asFloat()); + } + } else if (paramName == HdLightTokens->exposure) { + return VtValue(light.findPlug("aiExposure", true).asFloat()); + } else if (paramName == HdLightTokens->normalize) { + return VtValue(light.findPlug("aiNormalize", true).asBool()); + } else if (paramName == HdLightTokens->textureFormat) { + const auto format = light.findPlug("format", true).asShort(); + // mirrored_ball : 0 + // angular : 1 + // latlong : 2 + if (format == 0) { + return VtValue(UsdLuxTokens->mirroredBall); + } else if (format == 2) { + return VtValue(UsdLuxTokens->latlong); + } else { + return VtValue(UsdLuxTokens->automatic); + } + } else if (paramName == HdLightTokens->textureFile) { + // Be aware that dome lights in HdStorm always need a texture to work correctly, + // the color is not used if no texture is present. + + if (!_colorIsConnected){ + if (!_dummyTextureFullPathFilename.empty()){ + // Update Hydra texture resource everytime Domelight color is tweaked. + auto resourceReg = GetSceneProducer()->GetRenderIndex().GetResourceRegistry(); + if (TF_VERIFY(resourceReg, "Unable to update AikSkyDomelights constant color")) + resourceReg->ReloadResource(TfToken("texture"), _dummyTextureFullPathFilename); + // SdfAssetPath requires both "path" and "resolvedPath" + return VtValue(SdfAssetPath(_dummyTextureFullPathFilename, _dummyTextureFullPathFilename)); + } + // this will produce a warning but hopefully is an edge case as + // it means we were not able to create a dummy texture + return VtValue(SdfAssetPath()); + } + + MPlugArray conns; + light.findPlug("color", true).connectedTo(conns, true, false); + if (conns.length() < 1) { + // Should never happen as it has been tested before with its equivalent : _colorIsConnected + return VtValue(SdfAssetPath()); + } + MFnDependencyNode file(conns[0].node(), &status); + if (ARCH_UNLIKELY( + !status || (file.typeName() != MayaHydraAdapterTokens->file.GetText()))) { + // Be aware that dome lights in HdStorm always need a texture to work correctly, + // the color is not used if no texture is present. + if (! _dummyTextureFullPathFilename.empty()){ + // SdfAssetPath requires both "path" "resolvedPath" + return VtValue(SdfAssetPath(_dummyTextureFullPathFilename, _dummyTextureFullPathFilename)); + } else { + return VtValue(SdfAssetPath());// this will produce a warning but hopefully is an edge case + } + } + + const char* fileTextureName + = file.findPlug(MayaAttrs::file::fileTextureName, true).asString().asChar(); + // SdfAssetPath requires both "path" "resolvedPath" + return VtValue(SdfAssetPath(fileTextureName, fileTextureName)); + + } else if (paramName == HdLightTokens->enableColorTemperature) { + return VtValue(false); + } + return {}; + } +}; + +//Static variables from MayaHydraAiSkyDomeLightAdapter +MHWRender::MTextureManager* MayaHydraAiSkyDomeLightAdapter::_pTextureManager = nullptr; +std::string MayaHydraAiSkyDomeLightAdapter::_tmpFolderPath; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, domeLight) +{ + MayaHydraAdapterRegistry::RegisterLightAdapter( + TfToken("aiSkyDomeLight"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraLightAdapterPtr { + return MayaHydraLightAdapterPtr(new MayaHydraAiSkyDomeLightAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/areaLightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/areaLightAdapter.cpp new file mode 100644 index 0000000000..69e4539611 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/areaLightAdapter.cpp @@ -0,0 +1,81 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraAreaLightAdapter is used to handle the translation from a Maya area light to + * hydra. + */ +class MayaHydraAreaLightAdapter : public MayaHydraLightAdapter +{ +public: + MayaHydraAreaLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraLightAdapter(producer, dag) + { + } + + void _CalculateLightParams(GlfSimpleLight& light) override { light.SetSpotCutoff(90.0f); } + + const TfToken& LightType() const override + { + if (GetSceneProducer()->IsHdSt()) { + return HdPrimTypeTokens->simpleLight; + } else { + return HdPrimTypeTokens->rectLight; + } + } + + VtValue GetLightParamValue(const TfToken& paramName) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE) + .Msg( + "Called MayaHydraAreaLightAdapter::GetLightParamValue(%s) - %s\n", + paramName.GetText(), + GetDagPath().partialPathName().asChar()); + + if (paramName == HdLightTokens->width) { + return VtValue(2.0f); + } else if (paramName == HdLightTokens->height) { + return VtValue(2.0f); + } + return MayaHydraLightAdapter::GetLightParamValue(paramName); + } +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, pointLight) +{ + MayaHydraAdapterRegistry::RegisterLightAdapter( + TfToken("areaLight"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraLightAdapterPtr { + return MayaHydraLightAdapterPtr(new MayaHydraAreaLightAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.cpp new file mode 100644 index 0000000000..8972cda352 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.cpp @@ -0,0 +1,303 @@ +// +// Copyright 2021 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "cameraAdapter.h" + +#include +#include +#include +#include + +#include +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +namespace { + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, camera) +{ + MayaHydraAdapterRegistry::RegisterCameraAdapter( + HdPrimTypeTokens->camera, + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraCameraAdapterPtr { + return MayaHydraCameraAdapterPtr(new MayaHydraCameraAdapter(producer, dag)); + }); +} + +} // namespace + +// MayaHydraCameraAdapter is used to handle the translation from a Maya camera to hydra. +MayaHydraCameraAdapter::MayaHydraCameraAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraShapeAdapter(producer->GetPrimPath(dag, true), producer, dag) +{ +} + +MayaHydraCameraAdapter::~MayaHydraCameraAdapter() { } + +TfToken MayaHydraCameraAdapter::CameraType() { return HdPrimTypeTokens->camera; } + +bool MayaHydraCameraAdapter::IsSupported() const +{ + return GetSceneProducer()->GetRenderIndex().IsSprimTypeSupported(CameraType()); +} + +void MayaHydraCameraAdapter::Populate() +{ + if (_isPopulated) { + return; + } + GetSceneProducer()->InsertSprim(this, CameraType(), GetID(), HdCamera::AllDirty); + _isPopulated = true; +} + +void MayaHydraCameraAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + if (_isPopulated && dirtyBits != 0) { + dirtyBits = dirtyBits & HdCamera::AllDirty; + GetSceneProducer()->MarkSprimDirty(GetID(), dirtyBits); + } +} + +void MayaHydraCameraAdapter::CreateCallbacks() +{ + MStatus status; + auto dag = GetDagPath(); + auto obj = dag.node(); + + auto paramsChanged = MNodeMessage::addNodeDirtyCallback( + obj, + +[](MObject& obj, void* clientData) { + auto* adapter = reinterpret_cast(clientData); + // Dirty everything rather than track complex param and fit to projection dependencies. + adapter->MarkDirty(HdCamera::DirtyParams | HdCamera::DirtyWindowPolicy); + }, + reinterpret_cast(this), + &status); + if (status) { + AddCallback(paramsChanged); + } + + auto xformChanged = MDagMessage::addWorldMatrixModifiedCallback( + dag, + +[](MObject& transformNode, MDagMessage::MatrixModifiedFlags& modified, void* clientData) { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty(HdCamera::DirtyTransform); + adapter->InvalidateTransform(); + }, + reinterpret_cast(this), + &status); + if (status) { + AddCallback(xformChanged); + } + + // Skip over MayaHydraShapeAdapter's CreateCallbacks + MayaHydraAdapter::CreateCallbacks(); +} + +void MayaHydraCameraAdapter::RemovePrim() +{ + if (!_isPopulated) { + return; + } + GetSceneProducer()->RemoveSprim(CameraType(), GetID()); + _isPopulated = false; +} + +bool MayaHydraCameraAdapter::HasType(const TfToken& typeId) const { return typeId == CameraType(); } + +VtValue MayaHydraCameraAdapter::Get(const TfToken& key) { return MayaHydraShapeAdapter::Get(key); } + +VtValue MayaHydraCameraAdapter::GetCameraParamValue(const TfToken& paramName) +{ + constexpr double mayaInchToHydraCentimeter = 0.254; + constexpr double mayaInchToHydraMillimeter = 0.0254; + constexpr double mayaFocaLenToHydra = 0.01; + + MStatus status; + + auto convertFit = [&](const MFnCamera& camera) -> CameraUtilConformWindowPolicy { + const auto mayaFit = camera.filmFit(&status); + if (mayaFit == MFnCamera::kHorizontalFilmFit) + return CameraUtilConformWindowPolicy::CameraUtilMatchHorizontally; + if (mayaFit == MFnCamera::kVerticalFilmFit) + return CameraUtilConformWindowPolicy::CameraUtilMatchVertically; + + const auto fitMatcher = camera.horizontalFilmAperture() > camera.verticalFilmAperture() + ? MFnCamera::kOverscanFilmFit + : MFnCamera::kFillFilmFit; + return mayaFit == fitMatcher ? CameraUtilConformWindowPolicy::CameraUtilMatchHorizontally + : CameraUtilConformWindowPolicy::CameraUtilMatchVertically; + }; + + auto apertureConvert = [&](const MFnCamera& camera, double glApertureX, double glApertureY) { + const auto usdFit = convertFit(camera); + const double aperture = usdFit == CameraUtilConformWindowPolicy::CameraUtilMatchHorizontally + ? camera.horizontalFilmAperture() + : camera.verticalFilmAperture(); + const double glAperture + = usdFit == CameraUtilConformWindowPolicy::CameraUtilMatchHorizontally ? glApertureX + : glApertureY; + return (0.02 / aperture) * (aperture / glAperture); + }; + + auto viewParameters = [&](const MFnCamera& camera, + const GfVec4d* viewport, + double& apertureX, + double& apertureY, + double& offsetX, + double& offsetY) -> MStatus { + double aspectRatio = viewport + ? ((*viewport)[2] - (*viewport)[0]) / ((*viewport)[3] - (*viewport)[1]) + : camera.aspectRatio(); + return camera.getViewParameters( + aspectRatio, apertureX, apertureY, offsetX, offsetY, true, false, true); + }; + + auto hadError = [&](MStatus& status) -> bool { + if (ARCH_LIKELY(status)) + return false; + TF_WARN( + "Error in MayaHydraCameraAdapter::GetCameraParamValue(%s): %s", + paramName.GetText(), + status.errorString().asChar()); + return false; + }; + + MFnCamera camera(GetDagPath(), &status); + if (hadError(status)) + return {}; + + const bool isOrtho = camera.isOrtho(&status); + if (hadError(status)) { + return {}; + } + + if (paramName == HdCameraTokens->shutterOpen) { + // No motion samples, instantaneous shutter + if (!GetSceneProducer()->GetParams().motionSamplesEnabled()) + return VtValue(double(0)); + return VtValue(double(GetSceneProducer()->GetCurrentTimeSamplingInterval().GetMin())); + } + if (paramName == HdCameraTokens->shutterClose) { + // No motion samples, instantaneous shutter + if (!GetSceneProducer()->GetParams().motionSamplesEnabled()) + return VtValue(double(0)); + const auto shutterAngle = camera.shutterAngle(&status); + if (hadError(status)) + return {}; + auto constexpr maxRadians = M_PI * 2.0; + auto shutterClose = std::min(std::max(0.0, shutterAngle), maxRadians) / maxRadians; + auto interval = GetSceneProducer()->GetCurrentTimeSamplingInterval(); + return VtValue(double(interval.GetMin() + interval.GetSize() * shutterClose)); + } + + // Don't bother with anything else for orthographic cameras + if (isOrtho) { + return {}; + } + if (paramName == HdCameraTokens->focusDistance) { + auto focusDistance = camera.focusDistance(&status); + if (hadError(status)) + return {}; + return VtValue(float(focusDistance * mayaInchToHydraCentimeter)); + } + if (paramName == HdCameraTokens->focalLength) { + const double aspectRatio = _viewport + ? (((*_viewport)[2] - (*_viewport)[0]) / ((*_viewport)[3] - (*_viewport)[1])) + : camera.aspectRatio(); + + double left, right, bottom, top; + status = camera.getViewingFrustum(aspectRatio, left, right, bottom, top, true, false, true); + + const double cameraNear = camera.nearClippingPlane(); + + const double focalLen + = (convertFit(camera) == CameraUtilConformWindowPolicy::CameraUtilMatchVertically) + ? (2.0 * cameraNear) / (top - bottom) + : (2.0 * cameraNear) / (right - left); + return VtValue(float(focalLen * mayaFocaLenToHydra)); + } + if (paramName == HdCameraTokens->fStop) { + // For USD/Hydra fStop=0 should disable depthOfField + if (!camera.isDepthOfField()) + return VtValue(0.f); + const auto fStop = camera.fStop(&status); + if (hadError(status)) + return {}; + return VtValue(float(fStop)); + } + if (paramName == HdCameraTokens->horizontalAperture) { + double apertureX, apertureY, offsetX, offsetY; + status = viewParameters(camera, _viewport.get(), apertureX, apertureY, offsetX, offsetY); + if (hadError(status)) + return {}; + return VtValue(float(apertureX * apertureConvert(camera, apertureX, apertureY))); + } + if (paramName == HdCameraTokens->verticalAperture) { + double apertureX, apertureY, offsetX, offsetY; + status = viewParameters(camera, _viewport.get(), apertureX, apertureY, offsetX, offsetY); + if (hadError(status)) + return {}; + return VtValue(float(apertureY * apertureConvert(camera, apertureX, apertureY))); + } + if (paramName == HdCameraTokens->horizontalApertureOffset) { + double apertureX, apertureY, offsetX, offsetY; + status = viewParameters(camera, _viewport.get(), apertureX, apertureY, offsetX, offsetY); + if (hadError(status)) + return {}; + return VtValue(float(offsetX * mayaInchToHydraMillimeter)); + } + if (paramName == HdCameraTokens->verticalApertureOffset) { + double apertureX, apertureY, offsetX, offsetY; + status = viewParameters(camera, _viewport.get(), apertureX, apertureY, offsetX, offsetY); + if (hadError(status)) + return {}; + return VtValue(float(offsetY * mayaInchToHydraMillimeter)); + } + if (paramName == HdCameraTokens->windowPolicy) { + const auto windowPolicy = convertFit(camera); + if (hadError(status)) + return {}; + return VtValue(windowPolicy); + } + if (paramName == HdCameraTokens->projection) { + if (isOrtho) { + return VtValue(HdCamera::Orthographic); + } else { + return VtValue(HdCamera::Perspective); + } + } + + return {}; +} + +void MayaHydraCameraAdapter::SetViewport(const GfVec4d& viewport) +{ + if (!_viewport) { + _viewport.reset(new GfVec4d); + } + *_viewport = viewport; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.h new file mode 100644 index 0000000000..575c972732 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/cameraAdapter.h @@ -0,0 +1,81 @@ +// +// Copyright 2021 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_CAMERA_ADAPTER_H +#define MAYAHYDRALIB_CAMERA_ADAPTER_H + +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraCameraAdapter is used to handle the translation from a Maya camera to hydra. + */ +class MayaHydraCameraAdapter : public MayaHydraShapeAdapter +{ +public: + MAYAHYDRALIB_API + MayaHydraCameraAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag); + + MAYAHYDRALIB_API + virtual ~MayaHydraCameraAdapter(); + + MAYAHYDRALIB_API + bool IsSupported() const override; + + MAYAHYDRALIB_API + void MarkDirty(HdDirtyBits dirtyBits) override; + + MAYAHYDRALIB_API + void Populate() override; + + MAYAHYDRALIB_API + void RemovePrim() override; + + MAYAHYDRALIB_API + bool HasType(const TfToken& typeId) const override; + + MAYAHYDRALIB_API + VtValue Get(const TfToken& key) override; + + MAYAHYDRALIB_API + VtValue GetCameraParamValue(const TfToken& paramName); + + MAYAHYDRALIB_API + void CreateCallbacks() override; + + MAYAHYDRALIB_API + void SetViewport(const GfVec4d& viewport); + +protected: + static TfToken CameraType(); + + /// The use of a pointer here helps us track whether this camera is (or has ever been) + /// the active viewport camera. NOTE: it's possible that _viewport will be out of date + /// after switching to a new camera and resizing the viewport, but _viewport will eventually + /// be re-synched before any output/pixels of the stale size is requested. + std::unique_ptr _viewport; +}; + +using MayaHydraCameraAdapterPtr = std::shared_ptr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_CAMERA_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/constantShadowMatrix.h b/lib/mayaHydra/hydraExtensions/adapters/constantShadowMatrix.h new file mode 100644 index 0000000000..851734bedf --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/constantShadowMatrix.h @@ -0,0 +1,56 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_SHADOW_MATRIX_H +#define MAYAHYDRALIB_SHADOW_MATRIX_H + +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraConstantShadowMatrix is used to provide a constant shadow matrix for hydra. + */ +class MayaHydraConstantShadowMatrix : public HdxShadowMatrixComputation +{ +public: + explicit MayaHydraConstantShadowMatrix(const GfMatrix4d& mat) + : _shadowMatrix(mat) + { + } + + inline std::vector + Compute(const CameraUtilFraming& framing, CameraUtilConformWindowPolicy policy) override + { + return { _shadowMatrix }; + } + + inline std::vector + Compute(const GfVec4f& viewport, CameraUtilConformWindowPolicy policy) override + { + return { _shadowMatrix }; + } + +private: + GfMatrix4d _shadowMatrix; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_SHADOW_MATRIX_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.cpp new file mode 100644 index 0000000000..2fb92a7c64 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.cpp @@ -0,0 +1,342 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "dagAdapter.h" + +#include +#include +#include + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +// clang-format off +TF_DEFINE_PRIVATE_TOKENS( + _tokens, + + (translate) + (rotate) + (scale) + (instanceTransform) + (instancer) +); +// clang-format on + +namespace { + +void _TransformNodeDirty(MObject& node, MPlug& plug, void* clientData) +{ + auto* adapter = reinterpret_cast(clientData); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_DAG_PLUG_DIRTY) + .Msg( + "Dag adapter marking prim (%s) dirty because .%s plug was " + "dirtied.\n", + adapter->GetID().GetText(), + plug.partialName().asChar()); + if (plug == MayaAttrs::dagNode::visibility || plug == MayaAttrs::dagNode::intermediateObject + || plug == MayaAttrs::dagNode::overrideEnabled + || plug == MayaAttrs::dagNode::overrideVisibility) { + // Unfortunately, during this callback, we can't actually + // query the new object's visiblity - the plug dirty hasn't + // really propagated yet. So we just mark our own _visibility + // as dirty, and unconditionally dirty the hd bits + + // If we're currently invisible, it's possible we were + // skipping transform updates (see below), so need to mark + // that dirty as well... + if (adapter->IsVisible(false)) { + // Transform can change while dag path is hidden. + adapter->MarkDirty(HdChangeTracker::DirtyVisibility | HdChangeTracker::DirtyTransform); + adapter->InvalidateTransform(); + } else { + adapter->MarkDirty(HdChangeTracker::DirtyVisibility); + } + // We use IsVisible(checkDirty=false) because we need to make sure we + // DON'T update visibility from within this callback, since the change + // has't propagated yet + } else if (adapter->IsVisible(false)) { + adapter->MarkDirty(HdChangeTracker::DirtyTransform); + adapter->InvalidateTransform(); + } +} + +void _HierarchyChanged(MDagPath& child, MDagPath& parent, void* clientData) +{ + TF_UNUSED(child); + auto* adapter = reinterpret_cast(clientData); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_DAG_HIERARCHY) + .Msg( + "Dag hierarchy changed for prim (%s) because %s had parent %s " + "added/removed.\n", + adapter->GetID().GetText(), + child.partialPathName().asChar(), + parent.partialPathName().asChar()); + adapter->RemoveCallbacks(); + adapter->RemovePrim(); + adapter->GetSceneProducer()->RecreateAdapterOnIdle(adapter->GetID(), adapter->GetNode()); +} + +void _InstancerNodeDirty(MObject& node, MPlug& plug, void* clientData) +{ + auto* adapter = reinterpret_cast(clientData); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_DAG_PLUG_DIRTY) + .Msg( + "Dag instancer adapter marking prim (%s) dirty because %s plug was " + "dirtied.\n", + adapter->GetID().GetText(), + plug.partialName().asChar()); + adapter->MarkDirty( + HdChangeTracker::DirtyInstancer | HdChangeTracker::DirtyInstanceIndex + | HdChangeTracker::DirtyPrimvar); +} + +const auto _instancePrimvarDescriptors = HdPrimvarDescriptorVector { + { _tokens->instanceTransform, HdInterpolationInstance, HdPrimvarRoleTokens->none }, +}; + +} // namespace + +// MayaHydraDagAdapter is the adapter base class for any dag object. +MayaHydraDagAdapter::MayaHydraDagAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MDagPath& dagPath) + : MayaHydraAdapter(dagPath.node(), id, producer) + , _dagPath(dagPath) +{ + // We shouldn't call virtual functions in constructors. + _isVisible = GetDagPath().isVisible(); + _visibilityDirty = false; + _isInstanced = _dagPath.isInstanced() && _dagPath.instanceNumber() == 0; +} + +GfMatrix4d MayaHydraDagAdapter::GetTransform() +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraDagAdapter::GetTransform() - %s\n", + _dagPath.partialPathName().asChar()); + + if (_invalidTransform) { + if (IsInstanced()) { + _transform.SetIdentity(); + } else { + _transform = GetGfMatrixFromMaya(_dagPath.inclusiveMatrix()); + } + _invalidTransform = false; + } + + return _transform; +} + +size_t +MayaHydraDagAdapter::SampleTransform(size_t maxSampleCount, float* times, GfMatrix4d* samples) +{ + return GetSceneProducer()->SampleValues(maxSampleCount, times, samples, [&]() -> GfMatrix4d { + return GetGfMatrixFromMaya(_dagPath.inclusiveMatrix()); + }); +} + +void MayaHydraDagAdapter::CreateCallbacks() +{ + MStatus status; + + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating dag adapter callbacks for prim (%s).\n", GetID().GetText()); + + MDagPathArray dags; + if (MDagPath::getAllPathsTo(GetDagPath().node(), dags)) { + const auto numDags = dags.length(); + auto dagNodeDirtyCallback = numDags > 1 ? _InstancerNodeDirty : _TransformNodeDirty; + for (auto i = decltype(numDags) { 0 }; i < numDags; ++i) { + auto dag = dags[i]; + for (; dag.length() > 0; dag.pop()) { + MObject obj = dag.node(); + if (obj != MObject::kNullObj) { + auto id = MNodeMessage::addNodeDirtyPlugCallback( + obj, dagNodeDirtyCallback, this, &status); + if (status) { + AddCallback(id); + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg( + "- Added _InstancerNodeDirty callback for " + "dagPath (%s).\n", + dag.partialPathName().asChar()); + _AddHierarchyChangedCallbacks(dag); + } + } + } + } + MayaHydraAdapter::CreateCallbacks(); +} + +void MayaHydraDagAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + if (dirtyBits != 0) { + GetSceneProducer()->GetRenderIndex().GetChangeTracker().MarkRprimDirty(GetID(), dirtyBits); + if (IsInstanced()) { + GetSceneProducer()->GetRenderIndex().GetChangeTracker().MarkInstancerDirty(GetInstancerID(), dirtyBits); + } + if (dirtyBits & HdChangeTracker::DirtyVisibility) { + _visibilityDirty = true; + } + } +} + +void MayaHydraDagAdapter::RemovePrim() +{ + if (!_isPopulated) { + return; + } + GetSceneProducer()->RemoveRprim(GetID()); + if (_isInstanced) { + GetSceneProducer()->GetRenderIndex().RemoveInstancer(GetID().AppendProperty(_tokens->instancer)); + } + _isPopulated = false; +} + +bool MayaHydraDagAdapter::UpdateVisibility() +{ + if (ARCH_UNLIKELY(!GetDagPath().isValid())) { + return false; + } + const auto visible = _GetVisibility(); + _visibilityDirty = false; + if (visible != _isVisible) { + _isVisible = visible; + return true; + } + return false; +} + +bool MayaHydraDagAdapter::IsVisible(bool checkDirty) +{ + if (checkDirty && _visibilityDirty) { + UpdateVisibility(); + } + return _isVisible; +} + +VtIntArray MayaHydraDagAdapter::GetInstanceIndices(const SdfPath& prototypeId) +{ + if (!IsInstanced()) { + return {}; + } + MDagPathArray dags; + if (!MDagPath::getAllPathsTo(GetDagPath().node(), dags)) { + return {}; + } + const auto numDags = dags.length(); + VtIntArray ret; + ret.reserve(numDags); + for (auto i = decltype(numDags) { 0 }; i < numDags; ++i) { + if (dags[i].isValid() && dags[i].isVisible()) { + ret.push_back(static_cast(ret.size())); + } + } + return ret; +} + +void MayaHydraDagAdapter::_AddHierarchyChangedCallbacks(MDagPath& dag) +{ + MStatus status; + auto id = MDagMessage::addParentAddedDagPathCallback(dag, _HierarchyChanged, this, &status); + if (status) { + AddCallback(id); + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("- Added parent added callback for dagPath (%s).\n", dag.partialPathName().asChar()); + + // We need a parent removed callback, even for non-instances, + // because when an object is removed from the scene due to an + // undo, no pre-removal (or about-to-delete, or destroyed) + // callbacks are triggered. The parent-removed callback IS + // triggered, though, so it's a way to catch deletion due to + // undo... + id = MDagMessage::addParentRemovedDagPathCallback(dag, _HierarchyChanged, this, &status); + if (status) { + AddCallback(id); + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("- Added parent removed callback for dagPath (%s).\n", dag.partialPathName().asChar()); +} + +SdfPath MayaHydraDagAdapter::GetInstancerID() const +{ + if (!_isInstanced) { + return {}; + } + + return GetID().AppendProperty(_tokens->instancer); +} + +HdPrimvarDescriptorVector +MayaHydraDagAdapter::GetInstancePrimvarDescriptors(HdInterpolation interpolation) const +{ + if (interpolation == HdInterpolationInstance) { + return _instancePrimvarDescriptors; + } else { + return {}; + } +} + +bool MayaHydraDagAdapter::_GetVisibility() const { return GetDagPath().isVisible(); } + +VtValue MayaHydraDagAdapter::GetInstancePrimvar(const TfToken& key) +{ + if (key == _tokens->instanceTransform) { + MDagPathArray dags; + if (!MDagPath::getAllPathsTo(GetDagPath().node(), dags)) { + return {}; + } + const auto numDags = dags.length(); + VtArray ret; + ret.reserve(numDags); + for (auto i = decltype(numDags) { 0 }; i < numDags; ++i) { + if (dags[i].isValid() && dags[i].isVisible()) { + ret.push_back(GetGfMatrixFromMaya(dags[i].inclusiveMatrix())); + } + } + return VtValue(ret); + } + return {}; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.h new file mode 100644 index 0000000000..eb72c953bb --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/dagAdapter.h @@ -0,0 +1,100 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DG_ADAPTER_H +#define MAYAHYDRALIB_DG_ADAPTER_H + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraDagAdapter is the adapter base class for any dag object. + */ +class MayaHydraDagAdapter : public MayaHydraAdapter +{ +protected: + MAYAHYDRALIB_API + MayaHydraDagAdapter(const SdfPath& id, MayaHydraSceneProducer* producer, const MDagPath& dagPath); + +public: + MAYAHYDRALIB_API + virtual ~MayaHydraDagAdapter() = default; + MAYAHYDRALIB_API + virtual bool GetVisible() override { return IsVisible(); } + MAYAHYDRALIB_API + virtual void CreateCallbacks() override; + MAYAHYDRALIB_API + virtual void MarkDirty(HdDirtyBits dirtyBits) override; + MAYAHYDRALIB_API + virtual void RemovePrim() override; + MAYAHYDRALIB_API + GfMatrix4d GetTransform() override; + MAYAHYDRALIB_API + size_t SampleTransform(size_t maxSampleCount, float* times, GfMatrix4d* samples); + MAYAHYDRALIB_API + bool UpdateVisibility(); + bool IsVisible(bool checkDirty = true); + const MDagPath& GetDagPath() const { return _dagPath; } + void InvalidateTransform() { _invalidTransform = true; } + bool IsInstanced() const { return _isInstanced; } + MAYAHYDRALIB_API + SdfPath GetInstancerID() const; + MAYAHYDRALIB_API + virtual VtIntArray GetInstanceIndices(const SdfPath& prototypeId); + MAYAHYDRALIB_API + HdPrimvarDescriptorVector GetInstancePrimvarDescriptors(HdInterpolation interpolation) const; + MAYAHYDRALIB_API + VtValue GetInstancePrimvar(const TfToken& key); + +protected: + MAYAHYDRALIB_API + void _AddHierarchyChangedCallbacks(MDagPath& dag); + MAYAHYDRALIB_API + virtual bool _GetVisibility() const; + +private: + MDagPath _dagPath; + GfMatrix4d _transform; + bool _isVisible = true; + bool _visibilityDirty = true; + bool _invalidTransform = true; + bool _isInstanced = false; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DG_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/directionalLightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/directionalLightAdapter.cpp new file mode 100644 index 0000000000..4b56274957 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/directionalLightAdapter.cpp @@ -0,0 +1,118 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraDirectionalLightAdapter is used to handle the translation from a Maya directional + * light to hydra. + */ +class MayaHydraDirectionalLightAdapter : public MayaHydraLightAdapter +{ +public: + MayaHydraDirectionalLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraLightAdapter(producer, dag) + { + } + + const TfToken& LightType() const override + { + if (GetSceneProducer()->IsHdSt()) { + return HdPrimTypeTokens->simpleLight; + } else { + return HdPrimTypeTokens->distantLight; + } + } + + void _CalculateLightParams(GlfSimpleLight& light) override + { + // Directional lights point toward -Z, but we need the opposite + // for the position so the light acts as a directional light. + const auto direction = GfVec4f(0.0, 0.0, 1.0, 0.0) * GetTransform(); + light.SetHasShadow(true); + light.SetPosition({ direction[0], direction[1], direction[2], 0.0f }); + } + + VtValue Get(const TfToken& key) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraDirectionalLightAdapter::Get(%s) - %s\n", + key.GetText(), + GetDagPath().partialPathName().asChar()); + + if (key == HdLightTokens->shadowParams) { + HdxShadowParams shadowParams; + MFnDirectionalLight mayaLight(GetDagPath()); + if (!GetShadowsEnabled(mayaLight)) { + shadowParams.enabled = false; + return VtValue(shadowParams); + } + + _CalculateShadowParams(mayaLight, shadowParams); + // Use the radius as the "blur" amount, for PCSS + shadowParams.blur = mayaLight.shadowRadius(); + return VtValue(shadowParams); + } + + return MayaHydraLightAdapter::Get(key); + } + + VtValue GetLightParamValue(const TfToken& paramName) override + { + if (paramName == HdLightTokens->angle) { + MStatus status; + MFnDependencyNode lightNode(GetNode(), &status); + if (ARCH_UNLIKELY(!status)) { + return VtValue(0.0f); + } + return VtValue( + lightNode.findPlug(MayaAttrs::directionalLight::lightAngle, true).asFloat()); + } else { + return MayaHydraLightAdapter::GetLightParamValue(paramName); + } + } +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, pointLight) +{ + MayaHydraAdapterRegistry::RegisterLightAdapter( + TfToken("directionalLight"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraLightAdapterPtr { + return MayaHydraLightAdapterPtr(new MayaHydraDirectionalLightAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.cpp new file mode 100644 index 0000000000..96fa2aea20 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.cpp @@ -0,0 +1,371 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "lightAdapter.h" + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +namespace { + +void _changeVisibility( + MNodeMessage::AttributeMessage msg, + MPlug& plug, + MPlug& otherPlug, + void* clientData) +{ + TF_UNUSED(msg); + TF_UNUSED(otherPlug); + if (plug == MayaAttrs::dagNode::visibility) { + auto* adapter = reinterpret_cast(clientData); + if (adapter->UpdateVisibility()) { + adapter->RemovePrim(); + adapter->Populate(); + adapter->InvalidateTransform(); + } + } +} + +void _dirtyTransform(MObject& node, void* clientData) +{ + TF_UNUSED(node); + auto* adapter = reinterpret_cast(clientData); + if (adapter->IsVisible()) { + adapter->MarkDirty( + HdLight::DirtyTransform | HdLight::DirtyParams | HdLight::DirtyShadowParams); + adapter->InvalidateTransform(); + } +} + +void _dirtyParams(MObject& node, void* clientData) +{ + TF_UNUSED(node); + auto* adapter = reinterpret_cast(clientData); + if (adapter->IsVisible()) { + adapter->MarkDirty(HdLight::DirtyParams | HdLight::DirtyShadowParams); + adapter->InvalidateTransform(); + } +} + +const MString defaultLightSet("defaultLightSet"); + +} // namespace + +// MayaHydraLightAdapter is the base class for any light adapter used to handle the translation from +// a light to hydra. + +MayaHydraLightAdapter::MayaHydraLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraDagAdapter(producer->GetPrimPath(dag, true), producer, dag) +{ + // This should be avoided, not a good idea to call virtual functions + // directly or indirectly in a constructor. + UpdateVisibility(); + _shadowProjectionMatrix.SetIdentity(); +} + +MayaHydraLightAdapter::~MayaHydraLightAdapter() +{ +} + +bool MayaHydraLightAdapter::IsSupported() const +{ + return GetSceneProducer()->GetRenderIndex().IsSprimTypeSupported(LightType()); +} + +void MayaHydraLightAdapter::Populate() +{ + if (_isPopulated) { + return; + } + if (IsVisible() && _isLightingOn) { + GetSceneProducer()->InsertSprim(this, LightType(), GetID(), HdLight::AllDirty); + _isPopulated = true; + } +} + +void MayaHydraLightAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + if (_isPopulated && dirtyBits != 0) { + GetSceneProducer()->MarkSprimDirty(GetID(), dirtyBits); + } +} + +void MayaHydraLightAdapter::RemovePrim() +{ + if (!_isPopulated) { + return; + } + GetSceneProducer()->RemoveSprim(LightType(), GetID()); + _isPopulated = false; +} + +bool MayaHydraLightAdapter::HasType(const TfToken& typeId) const { return typeId == LightType(); } + +VtValue MayaHydraLightAdapter::Get(const TfToken& key) +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraLightAdapter::Get(%s) - %s\n", + key.GetText(), + GetDagPath().partialPathName().asChar()); + + if (key == HdLightTokens->params) { + MFnLight mayaLight(GetDagPath()); + GlfSimpleLight light; + const auto color = mayaLight.color(); + const auto intensity = mayaLight.intensity(); + MPoint pt(0.0, 0.0, 0.0, 1.0); + const auto inclusiveMatrix = GetDagPath().inclusiveMatrix(); + const auto position = pt * inclusiveMatrix; + // This will return zero / false if the plug is nonexistent. + const auto decayRate + = mayaLight.findPlug(MayaAttrs::nonAmbientLightShapeNode::decayRate, true).asShort(); + const auto emitDiffuse + = mayaLight.findPlug(MayaAttrs::nonAmbientLightShapeNode::emitDiffuse, true).asBool(); + const auto emitSpecular + = mayaLight.findPlug(MayaAttrs::nonAmbientLightShapeNode::emitSpecular, true).asBool(); + MVector pv(0.0, 0.0, -1.0); + const auto lightDirection = (pv * inclusiveMatrix).normal(); + light.SetHasShadow(false); + const GfVec4f zeroColor(0.0f, 0.0f, 0.0f, 1.0f); + const GfVec4f lightColor( + color.r * intensity, color.g * intensity, color.b * intensity, 1.0f); + light.SetDiffuse(emitDiffuse ? lightColor : zeroColor); + light.SetAmbient(zeroColor); + light.SetSpecular(emitSpecular ? lightColor : zeroColor); + light.SetShadowResolution(1024); + light.SetID(GetID()); + light.SetPosition(GfVec4f(position.x, position.y, position.z, position.w)); + light.SetSpotDirection(GfVec3f(lightDirection.x, lightDirection.y, lightDirection.z)); + if (decayRate == 0) { + light.SetAttenuation(GfVec3f(1.0f, 0.0f, 0.0f)); + } else if (decayRate == 1) { + light.SetAttenuation(GfVec3f(0.0f, 1.0f, 0.0f)); + } else if (decayRate == 2) { + light.SetAttenuation(GfVec3f(0.0f, 0.0f, 1.0f)); + } +#if PXR_VERSION < 2308 + light.SetTransform( + GetGfMatrixFromMaya(GetDagPath().inclusiveMatrixInverse())); +#else + light.SetTransform( + GetGfMatrixFromMaya(GetDagPath().inclusiveMatrix())); +#endif + _CalculateLightParams(light); + return VtValue(light); + } else if (key == HdTokens->transform) { + return VtValue(MayaHydraDagAdapter::GetTransform()); + } else if (key == HdLightTokens->shadowCollection) { + // Exclude prims that should not be lighted by only + // taking the primitives whose root path is GetSceneProducer()->GetLightedPrimsRootPath() + const SdfPath lightedPrimsRootPath = GetSceneProducer()->GetLightedPrimsRootPath(); + HdRprimCollection coll( + HdTokens->geometry, + HdReprSelector(HdReprTokens->refined), + lightedPrimsRootPath); + return VtValue(coll); + } else if (key == HdLightTokens->shadowParams) { + HdxShadowParams shadowParams; + MFnLight mayaLight(GetDagPath()); + const bool bLightHasShadowsenabled = mayaLight.useRayTraceShadows(); + if (!bLightHasShadowsenabled) { + shadowParams.enabled = false; + } else { + _CalculateShadowParams(mayaLight, shadowParams); + } + return VtValue(shadowParams); + } + return {}; +} + +VtValue MayaHydraLightAdapter::GetLightParamValue(const TfToken& paramName) +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE) + .Msg( + "Called MayaHydraLightAdapter::GetLightParamValue(%s) - %s\n", + paramName.GetText(), + GetDagPath().partialPathName().asChar()); + + MFnLight light(GetDagPath()); + if (paramName == HdLightTokens->color || paramName == HdTokens->displayColor) { + const auto color = light.color(); + return VtValue(GfVec3f(color.r, color.g, color.b)); + } else if (paramName == HdLightTokens->intensity) { + return VtValue(light.intensity()); + } else if (paramName == HdLightTokens->exposure) { + return VtValue(0.0f); + } else if (paramName == HdLightTokens->normalize) { + return VtValue(true); + } else if (paramName == HdLightTokens->enableColorTemperature) { + return VtValue(false); + } else if (paramName == HdLightTokens->diffuse) { + return VtValue(light.lightDiffuse() ? 1.0f : 0.0f); + } else if (paramName == HdLightTokens->specular) { + return VtValue(light.lightSpecular() ? 1.0f : 0.0f); + } + return {}; +} + +void MayaHydraLightAdapter::CreateCallbacks() +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating light adapter callbacks for prim (%s).\n", GetID().GetText()); + + MStatus status; + auto dag = GetDagPath(); + auto obj = dag.node(); + auto id = MNodeMessage::addNodeDirtyCallback(obj, _dirtyParams, this, &status); + if (status) { + AddCallback(id); + } + dag.pop(); + for (; dag.length() > 0; dag.pop()) { + // The adapter itself will free the callbacks, so we don't have to worry + // about passing raw pointers to the callbacks. Hopefully. + obj = dag.node(); + if (obj != MObject::kNullObj) { + id = MNodeMessage::addAttributeChangedCallback(obj, _changeVisibility, this, &status); + if (status) { + AddCallback(id); + } + id = MNodeMessage::addNodeDirtyCallback(obj, _dirtyTransform, this, &status); + if (status) { + AddCallback(id); + } + _AddHierarchyChangedCallbacks(dag); + } + } + MayaHydraAdapter::CreateCallbacks(); +} + +void MayaHydraLightAdapter::SetShadowProjectionMatrix(const GfMatrix4d& matrix) +{ + if (!GfIsClose(_shadowProjectionMatrix, matrix, 0.0001)) { + MarkDirty(HdLight::DirtyShadowParams); + _shadowProjectionMatrix = matrix; + } +} + +void MayaHydraLightAdapter::_CalculateShadowParams(MFnLight& light, HdxShadowParams& params) +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_LIGHT_SHADOWS) + .Msg( + "Called MayaHydraLightAdapter::_CalculateShadowParams - %s\n", + GetDagPath().partialPathName().asChar()); + + const auto dmapResolutionPlug + = light.findPlug(MayaAttrs::nonExtendedLightShapeNode::dmapResolution, true); + const auto dmapBiasPlug = light.findPlug(MayaAttrs::nonExtendedLightShapeNode::dmapBias, true); + const auto dmapFilterSizePlug + = light.findPlug(MayaAttrs::nonExtendedLightShapeNode::dmapFilterSize, true); + + params.enabled = true; + params.resolution = dmapResolutionPlug.isNull() + ? GetSceneProducer()->GetParams().maximumShadowMapResolution + : std::min( + GetSceneProducer()->GetParams().maximumShadowMapResolution, dmapResolutionPlug.asInt()); + + params.shadowMatrix + = std::make_shared(GetTransform() * _shadowProjectionMatrix); + params.bias = dmapBiasPlug.isNull() ? -0.001 : -dmapBiasPlug.asFloat(); + params.blur = dmapFilterSizePlug.isNull() ? 0.0 + : (static_cast(dmapFilterSizePlug.asInt())) + / static_cast(params.resolution); + + if (TfDebug::IsEnabled(MAYAHYDRALIB_ADAPTER_LIGHT_SHADOWS)) { + std::cout << "Resulting HdxShadowParams:\n"; + std::cout << params << "\n"; + } +} + +bool MayaHydraLightAdapter::_GetVisibility() const +{ + if (!GetDagPath().isVisible()) { + return false; + } + // Shapes are not part of the default light set. + if (!GetNode().hasFn(MFn::kLight)) { + return true; + } + MStatus status; + MFnDependencyNode node(GetDagPath().transform(), &status); + if (ARCH_UNLIKELY(!status)) { + return true; + } + auto p = node.findPlug(MayaAttrs::dagNode::instObjGroups, true); + if (ARCH_UNLIKELY(p.isNull())) { + return true; + } + const auto numElements = p.numElements(); + MPlugArray conns; + for (auto i = decltype(numElements) { 0 }; i < numElements; ++i) { + auto ep = p[i]; // == elementByPhysicalIndex + if (!ep.connectedTo(conns, false, true) || conns.length() < 1) { + continue; + } + const auto numConns = conns.length(); + for (auto j = decltype(numConns) { 0 }; j < numConns; ++j) { + MFnDependencyNode otherNode(conns[j].node(), &status); + if (!status) { + continue; + } + if (otherNode.name() == defaultLightSet) { + return true; + } + } + } + return false; +} + +void MayaHydraLightAdapter::SetLightingOn(bool isLightingOn) +{ + if (_isLightingOn != isLightingOn) { + _isLightingOn = isLightingOn; + + RemovePrim(); + Populate(); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.h new file mode 100644 index 0000000000..8c570f640b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/lightAdapter.h @@ -0,0 +1,89 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_LIGHT_ADAPTER_H +#define MAYAHYDRALIB_LIGHT_ADAPTER_H + +#include + +#include +#include +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraLightAdapter is the base class for any light adapter used to handle the + * translation from a light to hydra. + */ +class MayaHydraLightAdapter : public MayaHydraDagAdapter +{ +public: + inline bool GetShadowsEnabled(MFnNonExtendedLight& light) + { + return light.useDepthMapShadows() || light.useRayTraceShadows(); + } + + MAYAHYDRALIB_API + MayaHydraLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag); + MAYAHYDRALIB_API + virtual ~MayaHydraLightAdapter(); + MAYAHYDRALIB_API + virtual const TfToken& LightType() const = 0; + MAYAHYDRALIB_API + bool IsSupported() const override; + MAYAHYDRALIB_API + void Populate() override; + MAYAHYDRALIB_API + void MarkDirty(HdDirtyBits dirtyBits) override; + MAYAHYDRALIB_API + virtual void RemovePrim() override; + MAYAHYDRALIB_API + bool HasType(const TfToken& typeId) const override; + MAYAHYDRALIB_API + virtual VtValue GetLightParamValue(const TfToken& paramName); + MAYAHYDRALIB_API + VtValue Get(const TfToken& key) override; + MAYAHYDRALIB_API + virtual void CreateCallbacks() override; + MAYAHYDRALIB_API + void SetShadowProjectionMatrix(const GfMatrix4d& matrix); + MAYAHYDRALIB_API + void SetLightingOn(bool isLightingOn); + +protected: + MAYAHYDRALIB_API + virtual void _CalculateLightParams(GlfSimpleLight& light) { } + MAYAHYDRALIB_API + void _CalculateShadowParams(MFnLight& light, HdxShadowParams& params); + MAYAHYDRALIB_API + bool _GetVisibility() const override; + + GfMatrix4d _shadowProjectionMatrix; + bool _isLightingOn = true; +}; + +using MayaHydraLightAdapterPtr = std::shared_ptr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_LIGHT_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.cpp new file mode 100644 index 0000000000..f603dd80b2 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.cpp @@ -0,0 +1,314 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#include "materialAdapter.h" + +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +namespace { + +const VtValue _emptyValue; +const TfToken _emptyToken; +const TfTokenVector _stSamplerCoords = { TfToken("st") }; + +} // namespace + +/* MayaHydraMaterialAdapter is used to handle the translation from a Maya material to hydra. + If you are looking for how we translate the Maya shaders to hydra and how we do the parameters + mapping, please see MayaHydraMaterialNetworkConverter::initialize(). +*/ + +MayaHydraMaterialAdapter::MayaHydraMaterialAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MObject& node) + : MayaHydraAdapter(node, id, producer) +{ +} + +bool MayaHydraMaterialAdapter::IsSupported() const +{ + return GetSceneProducer()->GetRenderIndex().IsSprimTypeSupported(HdPrimTypeTokens->material); +} + +bool MayaHydraMaterialAdapter::HasType(const TfToken& typeId) const +{ + return typeId == HdPrimTypeTokens->material; +} + +void MayaHydraMaterialAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + GetSceneProducer()->MarkSprimDirty(GetID(), dirtyBits); +} + +void MayaHydraMaterialAdapter::RemovePrim() +{ + if (!_isPopulated) { + return; + } + GetSceneProducer()->RemoveSprim(HdPrimTypeTokens->material, GetID()); + _isPopulated = false; +} + +void MayaHydraMaterialAdapter::Populate() +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg("MayaHydraMaterialAdapter::Populate() - %s\n", GetID().GetText()); + if (_isPopulated) { + return; + } + GetSceneProducer()->InsertSprim(this, HdPrimTypeTokens->material, GetID(), HdMaterial::AllDirty); + _isPopulated = true; +} + +void MayaHydraMaterialAdapter::EnableXRayShadingMode(bool enable) +{ + _enableXRayShadingMode = enable; + MarkDirty(HdMaterial::DirtyParams); +} + +VtValue MayaHydraMaterialAdapter::GetMaterialResource() +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg("MayaHydraMaterialAdapter::GetMaterialResource()\n"); + return GetPreviewMaterialResource(GetID()); +} + +VtValue MayaHydraMaterialAdapter::GetPreviewMaterialResource(const SdfPath& materialID) +{ + HdMaterialNetworkMap map; + HdMaterialNetwork network; + HdMaterialNode node; + node.path = materialID; + node.identifier + = UsdImagingTokens->UsdPreviewSurface; // We translate to a USD preview surface material + map.terminals.push_back(node.path); + for (const auto& it : MayaHydraMaterialNetworkConverter::GetPreviewShaderParams()) { + node.parameters.emplace(it.name, it.fallbackValue); + } + network.nodes.push_back(node); + map.map.emplace(HdMaterialTerminalTokens->surface, network); + return VtValue(map); +} + +/** + * \brief MayaHydraShadingEngineAdapter is used to handle the translation from a Maya shading engine + * to hydra. + */ +class MayaHydraShadingEngineAdapter : public MayaHydraMaterialAdapter +{ +public: + typedef MayaHydraMaterialNetworkConverter::PathToMobjMap PathToMobjMap; + + MayaHydraShadingEngineAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MObject& obj) + : MayaHydraMaterialAdapter(id, producer, obj) + , _surfaceShaderCallback(0) + { + _CacheNodeAndTypes(); + } + + ~MayaHydraShadingEngineAdapter() override + { + if (_surfaceShaderCallback != 0) { + MNodeMessage::removeCallback(_surfaceShaderCallback); + } + } + + void CreateCallbacks() override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating shading engine adapter callbacks for prim (%s).\n", GetID().GetText()); + + MStatus status; + auto obj = GetNode(); + auto id = MNodeMessage::addNodeDirtyCallback(obj, _DirtyMaterialParams, this, &status); + if (ARCH_LIKELY(status)) { + AddCallback(id); + } + _CreateSurfaceMaterialCallback(); + MayaHydraAdapter::CreateCallbacks(); + } + + void Populate() override + { + MayaHydraMaterialAdapter::Populate(); +#ifdef MAYAHYDRALIB_OIT_ENABLED + _isTranslucent = IsTranslucent(); +#endif + } + +private: + static void _DirtyMaterialParams(MObject& /*node*/, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->_CreateSurfaceMaterialCallback(); + adapter->MarkDirty(HdMaterial::AllDirty); + } + + static void _DirtyShaderParams(MObject& /*node*/, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty(HdMaterial::AllDirty); + if (adapter->GetSceneProducer()->IsHdSt()) { + adapter->GetSceneProducer()->MaterialTagChanged(adapter->GetID()); + } + } + + void _CacheNodeAndTypes() + { + _surfaceShader = MObject::kNullObj; + _surfaceShaderType = _emptyToken; + MStatus status; + MFnDependencyNode node(GetNode(), &status); + if (ARCH_UNLIKELY(!status)) { + return; + } + + auto p = node.findPlug(MayaAttrs::shadingEngine::surfaceShader, true); + MPlugArray conns; + p.connectedTo(conns, true, false); + if (conns.length() > 0) { + _surfaceShader = conns[0].node(); + MFnDependencyNode surfaceNode(_surfaceShader, &status); + if (ARCH_UNLIKELY(!status)) { + return; + } + _surfaceShaderType = TfToken(surfaceNode.typeName().asChar()); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + "Found surfaceShader %s[%s]\n", + surfaceNode.name().asChar(), + _surfaceShaderType.GetText()); + } + } + + void _CreateSurfaceMaterialCallback() + { + _CacheNodeAndTypes(); + if (_surfaceShaderCallback != 0) { + MNodeMessage::removeCallback(_surfaceShaderCallback); + _surfaceShaderCallback = 0; + } + + if (_surfaceShader != MObject::kNullObj) { + _surfaceShaderCallback + = MNodeMessage::addNodeDirtyCallback(_surfaceShader, _DirtyShaderParams, this); + } + } + + VtValue GetMaterialResource() override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg("MayaHydraShadingEngineAdapter::GetMaterialResource(): %s\n", GetID().GetText()); + MayaHydraMaterialNetworkConverter::MayaHydraMaterialNetworkConverterInit initStruct( + GetID(), _enableXRayShadingMode, &_materialPathToMobj); + + MayaHydraMaterialNetworkConverter converter(initStruct); + if (!converter.GetMaterial(_surfaceShader)) { + return GetPreviewMaterialResource(GetID()); + } + + HdMaterialNetworkMap materialNetworkMap; + materialNetworkMap.map[HdMaterialTerminalTokens->surface] = initStruct._materialNetwork; + if (!initStruct._materialNetwork.nodes.empty()) { + materialNetworkMap.terminals.push_back(initStruct._materialNetwork.nodes.back().path); + } + + // HdMaterialNetwork displacementNetwork; + // materialNetworkMap.map[HdMaterialTerminalTokens->displacement] = + // displacementNetwork; + + return VtValue(materialNetworkMap); + }; + +#ifdef MAYAHYDRALIB_OIT_ENABLED + bool UpdateMaterialTag() override + { + if (IsTranslucent() != _isTranslucent) { + _isTranslucent = !_isTranslucent; + return true; + } + return false; + } + + bool IsTranslucent() + { + if (_surfaceShaderType == MayaHydraAdapterTokens->usdPreviewSurface + || _surfaceShaderType == MayaHydraAdapterTokens->pxrUsdPreviewSurface) { + MFnDependencyNode node(_surfaceShader); + const auto plug = node.findPlug(MayaHydraAdapterTokens->opacity.GetText(), true); + if (!plug.isNull() && (plug.asFloat() < 1.0f || plug.isConnected())) { + return true; + } + } + return false; + } + +#endif // MAYAHYDRALIB_OIT_ENABLED + + PathToMobjMap _materialPathToMobj; + + MObject _surfaceShader; + TfToken _surfaceShaderType; + // So they live long enough + + MCallbackId _surfaceShaderCallback; +#ifdef MAYAHYDRALIB_OIT_ENABLED + bool _isTranslucent = false; +#endif +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, shadingEngine) +{ + MayaHydraAdapterRegistry::RegisterMaterialAdapter( + TfToken("shadingEngine"), + [](const SdfPath& id, + MayaHydraSceneProducer* producer, + const MObject& obj) -> MayaHydraMaterialAdapterPtr { + return MayaHydraMaterialAdapterPtr( + new MayaHydraShadingEngineAdapter(id, producer, obj)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.h new file mode 100644 index 0000000000..19e9062f7f --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/materialAdapter.h @@ -0,0 +1,78 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_MATERIAL_ADAPTER_H +#define MAYAHYDRALIB_MATERIAL_ADAPTER_H + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraMaterialAdapter is used to handle the translation from a Maya material to hydra. + * If you are looking for how we translate the Maya shaders to hydra and how we do the parameters + * mapping, please see MayaHydraMaterialNetworkConverter::initialize(). + */ +class MayaHydraMaterialAdapter : public MayaHydraAdapter +{ +public: + MAYAHYDRALIB_API + MayaHydraMaterialAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MObject& node); + MAYAHYDRALIB_API + virtual ~MayaHydraMaterialAdapter() = default; + + MAYAHYDRALIB_API + bool IsSupported() const override; + + MAYAHYDRALIB_API + bool HasType(const TfToken& typeId) const override; + + MAYAHYDRALIB_API + void MarkDirty(HdDirtyBits dirtyBits) override; + MAYAHYDRALIB_API + void RemovePrim() override; + MAYAHYDRALIB_API + void Populate() override; + + MAYAHYDRALIB_API + void EnableXRayShadingMode(bool enable); + + MAYAHYDRALIB_API + virtual VtValue GetMaterialResource(); + + /// \brief Updates the material tag for the material. + /// \return True if the material tag have changed, false otherwise. + virtual bool UpdateMaterialTag() { return false; } + + MAYAHYDRALIB_API + static VtValue GetPreviewMaterialResource(const SdfPath& materialID); + +protected: + /// Are we in viewport XRay shading mode ? + bool _enableXRayShadingMode = false; +}; + +using MayaHydraMaterialAdapterPtr = std::shared_ptr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_MATERIAL_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.cpp b/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.cpp new file mode 100644 index 0000000000..db5ae71361 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.cpp @@ -0,0 +1,1029 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#include "materialNetworkConverter.h" + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +/* This file contains how we translate the Maya shaders (Blinn, Lambert, Standard Surface etc.) to + hydra and how we do the parameters mapping. See MayaHydraMaterialNetworkConverter::initialize() + for that purpose. +*/ + +namespace { + +// Print to the output window the type and value of each parameter from the std::map +void DebugPrintParameters(const std::map& params) +{ + cout << "\n"; // Add a line + // Print all parameters types and values + for (auto param : params) { + std::string valueAsString = ConvertVtValueToString(param.second); + cout << "Material parameters : (" << param.first.GetText() << " - " << valueAsString.c_str() + << ")\n"; + } +} + +const TfToken _useSpecularWorkflowToken("useSpecularWorkflow"); +const TfToken _specularColorToken("specularColor"); +const TfToken _opacityToken("opacity"); + +constexpr float defaultTextureMemoryLimit = 1e8f; +constexpr float xRayOpacityValue + = 0.3f; // Hardcoded value taken from OGSMayaRenderItem::UpdateExtraOpacityParam + +// Lists of preferred shader output names, from SdfValueTypeName to list of +// preferred output names for that type. The list that has an empty token for +// SdfValueTypeName is used as a default. +const std::vector>> preferredOutputNamesByType { + { SdfValueTypeNames->Float3, + { MayaHydraAdapterTokens->result, + MayaHydraAdapterTokens->out, + MayaHydraAdapterTokens->output, + MayaHydraAdapterTokens->rgb, + MayaHydraAdapterTokens->xyz } }, + { SdfValueTypeNames->Float2, + { MayaHydraAdapterTokens->result, + MayaHydraAdapterTokens->out, + MayaHydraAdapterTokens->output, + MayaHydraAdapterTokens->st, + MayaHydraAdapterTokens->uv } }, + { SdfValueTypeNames->Float, + { MayaHydraAdapterTokens->result, + MayaHydraAdapterTokens->out, + MayaHydraAdapterTokens->output, + MayaHydraAdapterTokens->r, + MayaHydraAdapterTokens->x } } +}; + +// Default set of preferred output names, if type not in +// preferredOutputNamesByType +const std::vector defaultPreferredOutputNames { MayaHydraAdapterTokens->result, + MayaHydraAdapterTokens->out, + MayaHydraAdapterTokens->output }; + +SdfValueTypeName GetStandardTypeName(SdfValueTypeName type) +{ + // Will map, ie, Vector3f to Float3, TexCoord2f to Float2 + return SdfGetValueTypeNameForValue(type.GetDefaultValue()); +} + +const std::vector& +GetPreferredOutputNames(SdfValueTypeName type, bool useStandardType = true) +{ + for (const auto& typeAndNames : preferredOutputNamesByType) { + if (typeAndNames.first == type) { + return typeAndNames.second; + } + } + + if (useStandardType) { + // If we were given, ie, Vector3f, check to see if there's an entry for + // Float3 + auto standardType = GetStandardTypeName(type); + if (type != standardType) { + return GetPreferredOutputNames(standardType, false); + } + } + return defaultPreferredOutputNames; +} + +TfToken GetOutputName(const HdMaterialNode& material, SdfValueTypeName type) +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + "GetOutputName(%s - %s, %s)\n", + material.path.GetText(), + material.identifier.GetText(), + type.GetAsToken().GetText()); + auto& shaderReg = SdrRegistry::GetInstance(); + if (SdrShaderNodeConstPtr sdrNode = shaderReg.GetShaderNodeByIdentifier(material.identifier)) { + // First, get the list off all outputs of the correct type. + std::vector validOutputs; + auto outputNames = sdrNode->GetOutputNames(); + + auto addMatchingOutputs = [&](SdfValueTypeName matchingType) { + for (const auto& outName : outputNames) { + auto* sdrInfo = sdrNode->GetShaderOutput(outName); + if (sdrInfo && sdrInfo->GetTypeAsSdfType().first == matchingType) { + validOutputs.push_back(outName); + } + } + }; + + addMatchingOutputs(type); + if (validOutputs.empty()) { + auto standardType = GetStandardTypeName(type); + if (standardType != type) { + addMatchingOutputs(standardType); + } + } + + // If there's only one, use that + if (validOutputs.size() == 1) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + " found exactly one output of correct type in " + "registry: " + "%s\n", + validOutputs[0].GetText()); + return validOutputs[0]; + } + + // Then see if any preferred names are found + if (!validOutputs.empty()) { + const auto& preferredNames = GetPreferredOutputNames(type); + for (const auto& preferredName : preferredNames) { + if (std::find(validOutputs.begin(), validOutputs.end(), preferredName) + != validOutputs.end()) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + " found preferred name of correct type in " + "registry: %s\n", + preferredName.GetText()); + return preferredName; + } + } + // No preferred names were found, use the first valid name + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + " found no preferred names of correct type in " + "registry, returning first valid name: %s\n", + validOutputs[0].GetText()); + return validOutputs[0]; + } + } + + // We either couldn't find the entry in the SdrRegistry, or there were + // no outputs of the right type - make a guess, use the first preferred + // name + const auto& preferredNames = GetPreferredOutputNames(type); + if (TF_VERIFY(!preferredNames.empty())) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg( + " found no valid entries in registry, returning guess: " + "%s\n", + preferredNames[0].GetText()); + return preferredNames[0]; + } + + // We should never get here - preferredNames should never be empty! + return MayaHydraAdapterTokens->result; +} + +class MayaHydraGenericMaterialAttrConverter : public MayaHydraMaterialAttrConverter +{ +public: + /// Generic attr converter has no fixed type + SdfValueTypeName GetType() override { return SdfValueTypeName(); } + + TfToken GetPlugName(const TfToken& usdName) override { return usdName; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + return MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, paramName.GetText(), type, fallback, outPlug); + } +}; + +class MayaHydraNewDefaultMaterialAttrConverter : public MayaHydraMaterialAttrConverter +{ +public: + template + MayaHydraNewDefaultMaterialAttrConverter(const T& defaultValue) + : _defaultValue(defaultValue) + { + } + + SdfValueTypeName GetType() override { return SdfGetValueTypeNameForValue(_defaultValue); } + + TfToken GetPlugName(const TfToken& usdName) override { return usdName; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + return MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, paramName.GetText(), type, &_defaultValue, outPlug); + } + + const VtValue _defaultValue; +}; + +class MayaHydraRemappingMaterialAttrConverter : public MayaHydraMaterialAttrConverter +{ +public: + MayaHydraRemappingMaterialAttrConverter( + const TfToken& remappedName, + const SdfValueTypeName& type) + : _remappedName(remappedName) + , _type(type) + { + } + + SdfValueTypeName GetType() override { return _type; } + + TfToken GetPlugName(const TfToken& usdName) override { return _remappedName; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + return MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, _remappedName.GetText(), type, fallback, outPlug); + } + +protected: + const TfToken& _remappedName; + const SdfValueTypeName& _type; +}; + +class MayaHydraScaledRemappingMaterialAttrConverter : public MayaHydraRemappingMaterialAttrConverter +{ +public: + MayaHydraScaledRemappingMaterialAttrConverter( + const TfToken& remappedName, + const TfToken& scaleName, + const SdfValueTypeName& type) + : MayaHydraRemappingMaterialAttrConverter(remappedName, type) + , _scaleName(scaleName) + { + } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + return MayaHydraMaterialNetworkConverter::ConvertMayaAttrToScaledValue( + node, _remappedName.GetText(), _scaleName.GetText(), type, fallback, outPlug); + } + +private: + const TfToken& _scaleName; +}; + +class MayaHydraComputedMaterialAttrConverter : public MayaHydraMaterialAttrConverter +{ +public: + /// Classes which derive from this use some sort of calculation to get + /// the right value for the node, and so don't have a single plug that + /// can be hooked into a node network. + TfToken GetPlugName(const TfToken& usdName) override { return TfToken(); } +}; + +class MayaHydraFixedMaterialAttrConverter : public MayaHydraComputedMaterialAttrConverter +{ +public: + template + MayaHydraFixedMaterialAttrConverter(const T& value) + : _value(value) + { + } + + SdfValueTypeName GetType() override { return SdfGetValueTypeNameForValue(_value); } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + return _value; + } + +private: + const VtValue _value; +}; + +class MayaHydraUvAttrConverter : public MayaHydraMaterialAttrConverter +{ +public: + MayaHydraUvAttrConverter() + : _value(GfVec2f(0.0f, 0.0f)) + { + } + + SdfValueTypeName GetType() override { return SdfValueTypeNames->TexCoord2f; } + + TfToken GetPlugName(const TfToken& usdName) override { return MayaHydraAdapterTokens->uvCoord; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + if (outPlug) { + // Find a connected place2dTexture node, and set that as the + // outPlug, so that the place2dTexture node will trigger + // creation of a UsdPrimvarReader_float2 + MStatus status; + MPlugArray connections; + status = node.getConnections(connections); + if (status) { + for (size_t i = 0, len = connections.length(); i < len; ++i) { + MPlug source = connections[i].source(); + if (source.isNull()) { + continue; + } + if (source.node().hasFn(MFn::kPlace2dTexture)) { + outPlug->append(connections[i]); + break; + } + } + } + } + return _value; + } + +private: + const VtValue _value; +}; // namespace + +class MayaHydraCosinePowerMaterialAttrConverter : public MayaHydraComputedMaterialAttrConverter +{ +public: + SdfValueTypeName GetType() override { return SdfValueTypeNames->Float; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + VtValue cosinePower = MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, "cosinePower", type, nullptr, outPlug); + if (!cosinePower.IsHolding()) { + if (fallback) { + return *fallback; + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg("MayaHydraCosinePowerMaterialAttrConverter::GetValue(): " + "No float plug found with name: cosinePower and no " + "fallback given"); + return VtValue(); + } else { + // In the maya UI, cosinePower goes from 2.0 to 100.0 ... + // so for now, we just do a dumb linear mapping from that onto + // 1 to 0 for roughness + float roughnessFloat = 1.0f - (cosinePower.UncheckedGet() - 2.0f) / 98.0f; + return VtValue(roughnessFloat); + } + } +}; + +class MayaHydraTransmissionMaterialAttrConverter : public MayaHydraComputedMaterialAttrConverter +{ +public: + SdfValueTypeName GetType() override { return SdfValueTypeNames->Float; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + VtValue transmission = MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, "transmission", type, nullptr, outPlug); + // Combine transmission and Geometry-->Opacity R,G and B attributes + VtValue geometryOpacityR = MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, "opacityR", type, nullptr, outPlug); + VtValue geometryOpacityG = MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, "opacityG", type, nullptr, outPlug); + VtValue geometryOpacityB = MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + node, "opacityB", type, nullptr, outPlug); + + if (!transmission.IsHolding()) { + if (fallback) { + return *fallback; + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg("MayaHydraTransmissionMaterialAttrConverter::GetValue(): " + "No float plug found with name: transmission and no " + "fallback given"); + return VtValue(); + } + + float val = 1.0f - transmission.UncheckedGet(); + if (val < 1.0e-4f) { + // Clamp lower value as an opacity of 0.0 in hydra makes the object fully transparent, + // but in VP2 we still see the specular highlight if any, avoiding 0.0 leads to the same + // effect in hydra. + val = 1.0e-4f; + } + + float fGeometryOpacity = 1.0f; + if (geometryOpacityR.IsHolding() && geometryOpacityG.IsHolding() + && geometryOpacityB.IsHolding()) { + // Take the average as there is only 1 parameter in hydra + fGeometryOpacity = (1.0f / 3.0f) + * (geometryOpacityR.UncheckedGet() + geometryOpacityG.UncheckedGet() + + geometryOpacityB.UncheckedGet()); + } + + val *= fGeometryOpacity; + + return VtValue(val); + } +}; + +class MayaHydraFilenameMaterialAttrConverter : public MayaHydraComputedMaterialAttrConverter +{ +public: + SdfValueTypeName GetType() override { return SdfValueTypeNames->Asset; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + auto path = GetFileTexturePath(node); + return VtValue(SdfAssetPath(path.GetText(), path.GetText())); + } +}; + +class MayaHydraWrapMaterialAttrConverter : public MayaHydraComputedMaterialAttrConverter +{ +public: + MayaHydraWrapMaterialAttrConverter(MObject& wrapAttr, MObject& mirrorAttr) + : _wrapAttr(wrapAttr) + , _mirrorAttr(mirrorAttr) + { + } + + SdfValueTypeName GetType() override { return SdfValueTypeNames->Token; } + + VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr) override + { + if (node.findPlug(_wrapAttr, true).asBool()) { + if (node.findPlug(_mirrorAttr, true).asBool()) { + return VtValue(UsdHydraTokens->mirror); + } else { + return VtValue(UsdHydraTokens->repeat); + } + } else { + return VtValue(UsdHydraTokens->clamp); + } + } + +private: + MObject _wrapAttr; + MObject _mirrorAttr; +}; + +auto _genericAttrConverter = std::make_shared(); + +typedef std::unordered_map + NameToNodeConverterMap; + +/// _nodeConverters contains how we translate from a Maya shader to hydra and the parameters mapping +/// we use. +NameToNodeConverterMap _nodeConverters; + +} // namespace + +/*static*/ +void MayaHydraMaterialNetworkConverter::initialize() +{ + // Define different converters for translating from specific Maya attributes types to hydra + auto colorConverter = std::make_shared( + MayaHydraAdapterTokens->color, + MayaHydraAdapterTokens->diffuse, + SdfValueTypeNames->Vector3f); + auto incandescenceConverter = std::make_shared( + MayaHydraAdapterTokens->incandescence, SdfValueTypeNames->Vector3f); + auto eccentricityConverter = std::make_shared( + MayaHydraAdapterTokens->eccentricity, SdfValueTypeNames->Float); + auto uvConverter = std::make_shared(); + + // Standard surface: + auto baseColorConverter = std::make_shared( + MayaHydraAdapterTokens->baseColor, + MayaHydraAdapterTokens->base, + SdfValueTypeNames->Vector3f); + auto emissionColorConverter = std::make_shared( + MayaHydraAdapterTokens->emissionColor, + MayaHydraAdapterTokens->emission, + SdfValueTypeNames->Vector3f); + auto specularColorConverter = std::make_shared( + MayaHydraAdapterTokens->specularColor, + MayaHydraAdapterTokens->specular, + SdfValueTypeNames->Vector3f); + auto specularIORConverter = std::make_shared( + MayaHydraAdapterTokens->specularIOR, SdfValueTypeNames->Float); + auto specularRoughnessConverter = std::make_shared( + MayaHydraAdapterTokens->specularRoughness, SdfValueTypeNames->Float); + auto metallicConverter = std::make_shared( + MayaHydraAdapterTokens->metalness, SdfValueTypeNames->Float); + auto coatConverter = std::make_shared( + MayaHydraAdapterTokens->coat, SdfValueTypeNames->Float); + auto coatRoughnessConverter = std::make_shared( + MayaHydraAdapterTokens->coatRoughness, SdfValueTypeNames->Float); + auto transmissionToOpacity = std::make_shared(); + + auto fixedZeroFloat = std::make_shared(0.0f); + auto fixedOneFloat = std::make_shared(1.0f); + auto fixedZeroInt = std::make_shared(0); + auto fixedOneInt = std::make_shared(1); + auto fixedStToken + = std::make_shared(MayaHydraAdapterTokens->st); + + auto cosinePowerToRoughness = std::make_shared(); + auto filenameConverter = std::make_shared(); + + auto wrapUConverter = std::make_shared( + MayaAttrs::file::wrapU, MayaAttrs::file::mirrorU); + auto wrapVConverter = std::make_shared( + MayaAttrs::file::wrapV, MayaAttrs::file::mirrorV); + + auto textureMemoryConverter + = std::make_shared(defaultTextureMemoryLimit); + + // In the following code we define how we translate from a Maya shader to hydra and how we do + // the parameters mapping + _nodeConverters = { + { MayaHydraAdapterTokens->usdPreviewSurface, { UsdImagingTokens->UsdPreviewSurface, {} } }, + { MayaHydraAdapterTokens->pxrUsdPreviewSurface, + { UsdImagingTokens->UsdPreviewSurface, {} } }, + { MayaHydraAdapterTokens->lambert, + { UsdImagingTokens + ->UsdPreviewSurface, // Maya Lambert shader translated to a UsdPreviewSurface with + // the following UsdPreviewSurface parameters mapped + { + { MayaHydraAdapterTokens->diffuseColor, colorConverter }, + { MayaHydraAdapterTokens->emissiveColor, incandescenceConverter }, + { MayaHydraAdapterTokens->roughness, fixedOneFloat }, + { MayaHydraAdapterTokens->metallic, fixedZeroFloat }, + { MayaHydraAdapterTokens->useSpecularWorkflow, fixedZeroInt }, + } } }, + { MayaHydraAdapterTokens->blinn, + { UsdImagingTokens + ->UsdPreviewSurface, // Maya Blinn shader translated to a UsdPreviewSurface with the + // following UsdPreviewSurface parameters mapped + { + { MayaHydraAdapterTokens->diffuseColor, colorConverter }, + { MayaHydraAdapterTokens->emissiveColor, incandescenceConverter }, + { MayaHydraAdapterTokens->roughness, eccentricityConverter }, + { MayaHydraAdapterTokens->metallic, fixedZeroFloat }, + { MayaHydraAdapterTokens->useSpecularWorkflow, fixedOneInt }, + } } }, + { MayaHydraAdapterTokens->phong, + { UsdImagingTokens + ->UsdPreviewSurface, // Maya Phong shader translated to a UsdPreviewSurface with the + // following UsdPreviewSurface parameters mapped + { + { MayaHydraAdapterTokens->diffuseColor, colorConverter }, + { MayaHydraAdapterTokens->emissiveColor, incandescenceConverter }, + { MayaHydraAdapterTokens->roughness, cosinePowerToRoughness }, + { MayaHydraAdapterTokens->metallic, fixedZeroFloat }, + { MayaHydraAdapterTokens->useSpecularWorkflow, fixedOneInt }, + } } }, + { MayaHydraAdapterTokens->standardSurface, + { UsdImagingTokens->UsdPreviewSurface, // Maya Standard surface shader translated to a + // UsdPreviewSurface with the following + // UsdPreviewSurface parameters mapped + { + { MayaHydraAdapterTokens->diffuseColor, baseColorConverter }, + { MayaHydraAdapterTokens->emissiveColor, emissionColorConverter }, + { MayaHydraAdapterTokens->specularColor, specularColorConverter }, + { MayaHydraAdapterTokens->ior, specularIORConverter }, + { MayaHydraAdapterTokens->roughness, specularRoughnessConverter }, + { MayaHydraAdapterTokens->clearcoat, coatConverter }, + { MayaHydraAdapterTokens->clearcoatRoughness, coatRoughnessConverter }, + { MayaHydraAdapterTokens->opacity, transmissionToOpacity }, + { MayaHydraAdapterTokens->metallic, metallicConverter }, + } } }, + { MayaHydraAdapterTokens->file, + { UsdImagingTokens->UsdUVTexture, // Maya file translated to a UsdUVTexture with the + // following UsdUVTexture parameters mapped + { + { MayaHydraAdapterTokens->file, filenameConverter }, + { MayaHydraAdapterTokens->st, uvConverter }, + { UsdHydraTokens->wrapS, wrapUConverter }, + { UsdHydraTokens->wrapT, wrapVConverter }, + { UsdHydraTokens->textureMemory, textureMemoryConverter }, + } } }, + { MayaHydraAdapterTokens + ->place2dTexture, // Maya place2dTexture translated to a UsdPrimvarReader_float2 with + // the following UsdPrimvarReader_float2 parameters mapped + { UsdImagingTokens->UsdPrimvarReader_float2, + { + { MayaHydraAdapterTokens->varname, fixedStToken }, + } } }, + }; +} + +MayaHydraMaterialNodeConverter::MayaHydraMaterialNodeConverter( + const TfToken& identifier, + const NameToAttrConverterMap& attrConverters) + : _attrConverters(attrConverters) + , _identifier(identifier) +{ +} + +MayaHydraMaterialAttrConverter::RefPtr +MayaHydraMaterialNodeConverter::GetAttrConverter(const TfToken& paramName) +{ + auto it = _attrConverters.find(paramName); + if (it == _attrConverters.end()) { + return _genericAttrConverter; + } + return it->second; +} + +MayaHydraMaterialNodeConverter* +MayaHydraMaterialNodeConverter::GetNodeConverter(const TfToken& nodeType) +{ + auto it = _nodeConverters.find(nodeType); + if (it == _nodeConverters.end()) { + return nullptr; + } + return &(it->second); +} + +MayaHydraShaderParam::MayaHydraShaderParam( + const TfToken& name, + const VtValue& value, + const SdfValueTypeName& type) + : name(name) + , fallbackValue(value) + , type(type) +{ +} + +MayaHydraMaterialNetworkConverter::MayaHydraMaterialNetworkConverter( + MayaHydraMaterialNetworkConverterInit& init) + : _network(init._materialNetwork) + , _prefix(init._prefix) + , _pathToMobj(init._pathToMobj) + , _enableXRayShadingMode(init._enableXRayShadingMode) +{ +} + +HdMaterialNode* MayaHydraMaterialNetworkConverter::GetMaterial(const MObject& mayaNode) +{ + MStatus status; + MFnDependencyNode node(mayaNode, &status); + if (ARCH_UNLIKELY(!status)) { + return nullptr; + } + const auto* chr = node.name().asChar(); + if (chr == nullptr || chr[0] == '\0') { + return nullptr; + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg("MayaHydraMaterialNetworkConverter::GetMaterial(node=%s)\n", chr); + std::string nodeNameString(chr); + SanitizeNameForSdfPath(nodeNameString); + const auto materialPath = _prefix.AppendChild(TfToken(nodeNameString)); + + auto findResult = std::find_if( + _network.nodes.begin(), + _network.nodes.end(), + [&materialPath](const HdMaterialNode& m) -> bool { return m.path == materialPath; }); + if (findResult != _network.nodes.end()) { + return &(*findResult); + } + + auto* nodeConverter + = MayaHydraMaterialNodeConverter::GetNodeConverter(TfToken(node.typeName().asChar())); + if (!nodeConverter) { + return nullptr; + } + HdMaterialNode material {}; + material.path = materialPath; + material.identifier = nodeConverter->GetIdentifier(); + if (material.identifier == UsdImagingTokens->UsdPreviewSurface) { + for (const auto& param : MayaHydraMaterialNetworkConverter::GetPreviewShaderParams()) { + this->ConvertParameter( + node, *nodeConverter, material, param.name, param.type, ¶m.fallbackValue); + } + + // If we are using a specular color which is not white, the UsdPreviewsurface specular + // workflow must be enabled to use the specular color which is done by setting the + // UsdPreviewSurface param "useSpecularWork" to 1 + { + const auto it = material.parameters.find(_specularColorToken); + if (it != material.parameters.cend()) { + const VtValue& specColorVal = it->second; + if (!specColorVal.IsEmpty() + && specColorVal.UncheckedGet() != GfVec3f(1, 1, 1)) { + material.parameters[_useSpecularWorkflowToken] = VtValue(1); + } + } + } + + if (_enableXRayShadingMode) { + // Multiply current opacity by hardcoded xRayOpacityValue + const auto it = material.parameters.find(_opacityToken); + if (it != material.parameters.cend()) { + const VtValue& opacityVal = it->second; + if (!opacityVal.IsEmpty()) { + material.parameters[_opacityToken] + = VtValue(opacityVal.UncheckedGet() * xRayOpacityValue); + } + } + } + + if (TfDebug::IsEnabled(MAYAHYDRALIB_ADAPTER_MATERIALS_PRINT_PARAMETERS_VALUES)) { + // DEBUG to print material parameters type and value to the output window + DebugPrintParameters(material.parameters); + } + + } else { + for (auto& nameAttrConverterPair : nodeConverter->GetAttrConverters()) { + auto& name = nameAttrConverterPair.first; + auto& attrConverter = nameAttrConverterPair.second; + this->ConvertParameter(node, *nodeConverter, material, name, attrConverter->GetType()); + + if (name == MayaHydraAdapterTokens->varname + && (material.identifier == UsdImagingTokens->UsdPrimvarReader_float + || material.identifier == UsdImagingTokens->UsdPrimvarReader_float2 + || material.identifier == UsdImagingTokens->UsdPrimvarReader_float3 + || material.identifier == UsdImagingTokens->UsdPrimvarReader_float4)) { + VtValue& primVarName = material.parameters[name]; + if (TF_VERIFY(primVarName.IsHolding())) { + AddPrimvar(primVarName.UncheckedGet()); + } else { + TF_WARN("Converter identified as a UsdPrimvarReader*, but " + "it's " + "varname did not hold a TfToken"); + } + } + } + } + if (_pathToMobj) { + (*_pathToMobj)[materialPath] = mayaNode; + } + _network.nodes.push_back(material); + return &_network.nodes.back(); +} + +void MayaHydraMaterialNetworkConverter::AddPrimvar(const TfToken& primvar) +{ + if (std::find(_network.primvars.begin(), _network.primvars.end(), primvar) + == _network.primvars.end()) { + _network.primvars.push_back(primvar); + } +} + +void MayaHydraMaterialNetworkConverter::ConvertParameter( + MFnDependencyNode& node, + MayaHydraMaterialNodeConverter& nodeConverter, + HdMaterialNode& material, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback) +{ + MPlugArray plugArray; + VtValue val; + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS).Msg("ConvertParameter(%s)\n", paramName.GetText()); + + auto attrConverter = nodeConverter.GetAttrConverter(paramName); + if (attrConverter) { + // Using an array of MPlug in plugArray, as some settings may have 2 or more attributes that + // should be taken into consideration for connections. For example : specular has a specular + // color and specular weight attributes, both should be considered. So after calling + // attrConverter->GetValue, the plugArray will contain all dependents MPlug for connections. + val = attrConverter->GetValue(node, paramName, type, fallback, &plugArray); + } else if (fallback) { + val = *fallback; + } else { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "MayaHydraMaterialNetworkConverter::ConvertParameter(): " + "No attrConverter found with name: %s and no fallback " + "given", + paramName.GetText()); + val = VtValue(); + } + + material.parameters[paramName] = val; + + /*plugArray contains all dependents MPlug we should consider for connections. + Usually it contains 1 or 2 MPlug (2 is when dealing with a weighted attribute), + it can have more than 2 when dealing with the transmission which is combined with opacityR, + opacityG and opacityB attributes. But a limitation we have at this time is that if both the + color and the weight attributes have a connection, one of both connections will be ignored by + hydra as we have only one parameter in the UsdPreviewSurface which will have both connections + and hydra only considers the last connection added. There is no blending node we could use with + the UsdPreviewSurface. We would need the StandardSurface to be in hydra or use MaterialX to + build a shading network to handle this case with a multiply node for example. + */ + for (auto plug : plugArray) { + if (plug.isNull()) { + return; + } + + MPlug source = plug.source(); + if (!source.isNull()) { + auto* sourceMat = GetMaterial(source.node()); + if (!sourceMat) { + return; + } + const auto& sourceMatPath = sourceMat->path; + if (sourceMatPath.IsEmpty()) { + return; + } + HdMaterialRelationship rel; + rel.inputId = sourceMatPath; + rel.inputName = GetOutputName(*sourceMat, type); + rel.outputId = material.path; + rel.outputName = paramName; + _network.relationships.push_back(rel); + } + } +} + +VtValue MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue( + MFnDependencyNode& node, + const MString& plugName, + const SdfValueTypeName& type, + const VtValue* fallback, + MPlugArray* outPlug) +{ + MStatus status; + auto p = node.findPlug(plugName, true, &status); + VtValue val; + if (status) { + if (outPlug) { + outPlug->append(p); + } + val = ConvertPlugToValue(p, type, fallback); + } else if (fallback) { + val = *fallback; + } else { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "MayaHydraMaterialNetworkConverter::ConvertMayaAttrToValue(): " + "No plug found with name: %s and no fallback given", + plugName.asChar()); + val = VtValue(); + } + return val; +} + +VtValue MayaHydraMaterialNetworkConverter::ConvertMayaAttrToScaledValue( + MFnDependencyNode& node, + const MString& plugName, + const MString& scaleName, + const SdfValueTypeName& type, + const VtValue* fallback, + MPlugArray* outPlug) +{ + VtValue val = ConvertMayaAttrToValue(node, plugName, type, fallback, outPlug); + MStatus status; + auto p = node.findPlug(scaleName, true, &status); + if (status) { + if (!p.isNull() && outPlug) { + outPlug->append(p); + } + if (type.GetType() == SdfValueTypeNames->Vector3f.GetType()) { + val = val.UncheckedGet() * p.asFloat(); + } else if (type == SdfValueTypeNames->Float) { + val = val.UncheckedGet() * p.asFloat(); + } else if (type.GetType() == SdfValueTypeNames->Float2.GetType()) { + val = val.UncheckedGet() * p.asFloat(); + } + } else { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "MayaHydraMaterialNetworkConverter::ConvertMayaAttrToScaledValue(): " + "No scaling plug found with name: %s", + scaleName.asChar()); + } + return val; +} + +VtValue MayaHydraMaterialNetworkConverter::ConvertPlugToValue( + const MPlug& plug, + const SdfValueTypeName& type, + const VtValue* fallback) +{ + if (type.GetType() == SdfValueTypeNames->Vector3f.GetType()) { + return VtValue( + GfVec3f(plug.child(0).asFloat(), plug.child(1).asFloat(), plug.child(2).asFloat())); + } else if (type == SdfValueTypeNames->Float) { + return VtValue(plug.asFloat()); + } else if (type.GetType() == SdfValueTypeNames->Float2.GetType()) { + return VtValue(GfVec2f(plug.child(0).asFloat(), plug.child(1).asFloat())); + } else if (type == SdfValueTypeNames->Int) { + return VtValue(plug.asInt()); + } + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "MayaHydraMaterialNetworkConverter::ConvertPlugToValue(): do not " + "know how to handle type: %s (cpp type: %s)\n", + type.GetAsToken().GetText(), + type.GetCPPTypeName().c_str()); + if (fallback) { + return *fallback; + } + return {}; +}; + +std::mutex _previewShaderParams_mutex; +bool _previewShaderParams_initialized = false; +MayaHydraShaderParams _previewShaderParams; +static std::map _defaultShaderParams; + +const MayaHydraShaderParams& MayaHydraMaterialNetworkConverter::GetPreviewShaderParams() +{ + if (!_previewShaderParams_initialized) { + std::lock_guard lock(_previewShaderParams_mutex); + // Once we have the lock, recheck to make sure it's still + // uninitialized... + if (!_previewShaderParams_initialized) { + auto& shaderReg = SdrRegistry::GetInstance(); + SdrShaderNodeConstPtr sdrNode + = shaderReg.GetShaderNodeByIdentifier(UsdImagingTokens->UsdPreviewSurface); + if (TF_VERIFY(sdrNode)) { + auto inputNames = sdrNode->GetInputNames(); + _previewShaderParams.reserve(inputNames.size()); + + for (auto& inputName : inputNames) { + auto property = sdrNode->GetInput(inputName); + if (!TF_VERIFY(property)) { + continue; + } + _previewShaderParams.emplace_back( + inputName, property->GetDefaultValue(), property->GetTypeAsSdfType().first); + } + std::sort( + _previewShaderParams.begin(), + _previewShaderParams.end(), + [](const MayaHydraShaderParam& a, const MayaHydraShaderParam& b) -> bool { + return a.name < b.name; + }); + _previewShaderParams_initialized = true; + } + } + } + return _previewShaderParams; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.h b/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.h new file mode 100644 index 0000000000..7f08003ab1 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/materialNetworkConverter.h @@ -0,0 +1,210 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#ifndef MAYAHYDRALIB_MATERIAL_NETWORK_CONVERTER_H +#define MAYAHYDRALIB_MATERIAL_NETWORK_CONVERTER_H + +#include + +#include +#include +#include +#include + +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * The MayaHydraMaterialNetworkConverter class contains how we translate the Maya shaders to hydra + * and how we do the parameters mapping, please see MayaHydraMaterialNetworkConverter::initialize() + * for that purpose. + */ + +struct MayaHydraShaderParam +{ + TfToken name; + VtValue fallbackValue; + + SdfValueTypeName type; + + MAYAHYDRALIB_API + MayaHydraShaderParam(const TfToken& name, const VtValue& value, const SdfValueTypeName& type); +}; + +using MayaHydraShaderParams = std::vector; + +/// Class which provides basic name and value translation for an attribute. +/// Used by both MayaHydraMaterialNetworkConverter (for to-usd file export +/// translation) and MayaHydraMaterialAdapter (for translation to Hydra). +class MayaHydraMaterialAttrConverter +{ +public: + typedef std::shared_ptr RefPtr; + + virtual ~MayaHydraMaterialAttrConverter() {}; + + /// Returns the default type for this attr converter - if an + /// implementation returns an invalid type, this indicates the attr + /// converter's type is undefined / variable. + virtual SdfValueTypeName GetType() = 0; + + /// If there is a simple, one-to-one mapping from the usd/hydra attribute + /// we are trying to "get", and a corresponding maya plug, AND the value + /// can be used "directly", then this should return the name of the maya + /// plug. Otherwise it should return an empty token. + /// By returning an empty token, we indicate that we want to set a value, + /// but that we don't wish to set up any network connections (ie, textures, + /// etc.) + MAYAHYDRALIB_API + virtual TfToken GetPlugName(const TfToken& usdName) = 0; + + /// Returns the value computed from maya for the usd/hydra attribute + // MAYAHYDRALIB_API + // virtual VtValue GetValue( + // const MayaHydraShaderParam& destParam, MFnDependencyNode& node, + // MPlug* outPlug = nullptr) = 0; + + MAYAHYDRALIB_API + virtual VtValue GetValue( + MFnDependencyNode& node, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug + = nullptr) // Some parameters have more than one MPlug to look at such as transmission, + // specular etc. which have a weight and a color + = 0; +}; + +/// Class which provides basic name and value translation for a maya node +/// type. Used by both MayaHydraMaterialNetworkConverter (for to-usd file +/// export translation) and MayaHydraMaterialAdapter (for translation to Hydra). +class MayaHydraMaterialNodeConverter +{ +public: + typedef std:: + unordered_map + NameToAttrConverterMap; + + MAYAHYDRALIB_API + MayaHydraMaterialNodeConverter( + const TfToken& identifier, + const NameToAttrConverterMap& attrConverters); + + inline TfToken GetIdentifier() { return _identifier; } + + /// Try to find the correct attribute converter to use for the given + /// param; if nothing is found, will usually return a generic converter, + /// that will look for an attribute on the maya node with the same name, and + /// use that if possible. + MAYAHYDRALIB_API + MayaHydraMaterialAttrConverter::RefPtr GetAttrConverter(const TfToken& paramName); + + inline NameToAttrConverterMap& GetAttrConverters() { return _attrConverters; } + + MAYAHYDRALIB_API + static MayaHydraMaterialNodeConverter* GetNodeConverter(const TfToken& nodeType); + +private: + NameToAttrConverterMap _attrConverters; + mutable TfToken _identifier; +}; + +class MayaHydraMaterialNetworkConverter +{ +public: + typedef std::unordered_map PathToMobjMap; + + struct MayaHydraMaterialNetworkConverterInit + { + MayaHydraMaterialNetworkConverterInit( + const SdfPath& prefix, + bool enableXRayShadingMode, + PathToMobjMap* pathToMobj) + : _prefix(prefix) + , _enableXRayShadingMode(enableXRayShadingMode) + , _pathToMobj(pathToMobj) + { + } + MayaHydraMaterialNetworkConverterInit() = delete; + + HdMaterialNetwork _materialNetwork; + const SdfPath& _prefix; + bool _enableXRayShadingMode; + PathToMobjMap* _pathToMobj; // Can be a nullptr + }; + + MAYAHYDRALIB_API + MayaHydraMaterialNetworkConverter(MayaHydraMaterialNetworkConverterInit& init); + + MAYAHYDRALIB_API + HdMaterialNode* GetMaterial(const MObject& mayaNode); + + MAYAHYDRALIB_API + void AddPrimvar(const TfToken& primvar); + + MAYAHYDRALIB_API + void ConvertParameter( + MFnDependencyNode& node, + MayaHydraMaterialNodeConverter& nodeConverter, + HdMaterialNode& material, + const TfToken& paramName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr); + + MAYAHYDRALIB_API static VtValue ConvertMayaAttrToValue( + MFnDependencyNode& node, + const MString& plugName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr); + + MAYAHYDRALIB_API static VtValue ConvertMayaAttrToScaledValue( + MFnDependencyNode& node, + const MString& plugName, + const MString& scaleName, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr, + MPlugArray* outPlug = nullptr); + + MAYAHYDRALIB_API + static void initialize(); + + MAYAHYDRALIB_API + static VtValue ConvertPlugToValue( + const MPlug& plug, + const SdfValueTypeName& type, + const VtValue* fallback = nullptr); + + MAYAHYDRALIB_API + static const MayaHydraShaderParams& GetPreviewShaderParams(); + +private: + HdMaterialNetwork& _network; + const SdfPath& _prefix; + PathToMobjMap* _pathToMobj; + bool _enableXRayShadingMode = false; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_MATERIAL_NETWORK_CONVERTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.cpp b/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.cpp new file mode 100644 index 0000000000..aeaf09bf8f --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.cpp @@ -0,0 +1,284 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include + +#include + +#include + +#define SET_NODE_CLASS(nodeTypeName) \ + using namespace nodeTypeName; \ + MNodeClass nodeClass(#nodeTypeName); \ + if (!TF_VERIFY(nodeClass.typeId() != 0)) { \ + return MStatus::kFailure; \ + } + +#define SET_ATTR_OBJ(attr) \ + setAttrObj(attr, nodeClass, #attr); \ + if (!TF_VERIFY(status)) { \ + return status; \ + } + +PXR_NAMESPACE_OPEN_SCOPE + +namespace MayaAttrs { + +namespace node { + +MObject message; + +} // namespace node + +namespace dagNode { + +MObject visibility; +MObject worldMatrix; +MObject intermediateObject; +MObject instObjGroups; +MObject overrideEnabled; +MObject overrideVisibility; + +} // namespace dagNode + +namespace nonAmbientLightShapeNode { + +MObject decayRate; +MObject emitDiffuse; +MObject emitSpecular; + +} // namespace nonAmbientLightShapeNode + +namespace nonExtendedLightShapeNode { + +MObject dmapResolution; +MObject dmapBias; +MObject dmapFilterSize; +MObject useDepthMapShadows; + +} // namespace nonExtendedLightShapeNode + +namespace spotLight { + +MObject coneAngle; +MObject dropoff; + +} // namespace spotLight + +namespace directionalLight { + +MObject lightAngle; +} + +namespace surfaceShape { + +MObject doubleSided; + +} // namespace surfaceShape + +// mesh + +namespace mesh { + +MObject pnts; +MObject inMesh; +MObject uvPivot; +MObject displaySmoothMesh; +MObject smoothLevel; + +} // namespace mesh + +// nurbsCurve + +namespace nurbsCurve { + +MObject controlPoints; + +} // namespace nurbsCurve + +namespace shadingEngine { + +MObject surfaceShader; + +} // namespace shadingEngine + +namespace file { + +MObject computedFileTextureNamePattern; +MObject fileTextureName; +MObject fileTextureNamePattern; +MObject uvTilingMode; +MObject uvCoord; +MObject wrapU; +MObject wrapV; +MObject mirrorU; +MObject mirrorV; + +} // namespace file + +namespace imagePlane { + +MObject imageName; +MObject useFrameExtension; +MObject frameOffset; +MObject frameExtension; +MObject displayMode; + +MObject fit; +MObject coverage; +MObject coverageOrigin; +MObject depth; +MObject rotate; +MObject size; +MObject offset; +MObject width; +MObject height; +MObject imageCenter; + +} // namespace imagePlane + +MStatus initialize() +{ + MStatus status; + + auto setAttrObj = [&status](MObject& attrObj, MNodeClass& nodeClass, const MString& name) { + attrObj = nodeClass.attribute(name, &status); + if (!TF_VERIFY(status)) { + return; + } + if (!TF_VERIFY(!attrObj.isNull())) { + status = MS::kFailure; + MString errMsg("Error finding '"); + errMsg += nodeClass.typeName(); + errMsg += "."; + errMsg += name; + errMsg += "' attribute"; + status.perror(errMsg); + return; + } + }; + { + SET_NODE_CLASS(node); + + SET_ATTR_OBJ(message); + } + { + SET_NODE_CLASS(dagNode); + + SET_ATTR_OBJ(visibility); + SET_ATTR_OBJ(worldMatrix); + SET_ATTR_OBJ(intermediateObject); + SET_ATTR_OBJ(instObjGroups); + SET_ATTR_OBJ(overrideEnabled); + SET_ATTR_OBJ(overrideVisibility); + } + + { + SET_NODE_CLASS(nonAmbientLightShapeNode); + + SET_ATTR_OBJ(decayRate); + SET_ATTR_OBJ(emitDiffuse); + SET_ATTR_OBJ(emitSpecular); + } + + { + SET_NODE_CLASS(nonExtendedLightShapeNode); + + SET_ATTR_OBJ(dmapResolution); + SET_ATTR_OBJ(dmapBias); + SET_ATTR_OBJ(dmapFilterSize); + } + + { + SET_NODE_CLASS(spotLight); + + SET_ATTR_OBJ(coneAngle); + SET_ATTR_OBJ(dropoff); + } + + { + SET_NODE_CLASS(directionalLight); + + SET_ATTR_OBJ(lightAngle); + } + + { + SET_NODE_CLASS(surfaceShape); + + SET_ATTR_OBJ(doubleSided); + } + + { + SET_NODE_CLASS(mesh); + + SET_ATTR_OBJ(pnts); + SET_ATTR_OBJ(inMesh); + SET_ATTR_OBJ(uvPivot); + SET_ATTR_OBJ(displaySmoothMesh); + SET_ATTR_OBJ(smoothLevel); + } + + { + SET_NODE_CLASS(nurbsCurve); + + SET_ATTR_OBJ(controlPoints); + } + + { + SET_NODE_CLASS(shadingEngine); + + SET_ATTR_OBJ(surfaceShader); + } + + { + SET_NODE_CLASS(file); + + SET_ATTR_OBJ(computedFileTextureNamePattern); + SET_ATTR_OBJ(fileTextureName); + SET_ATTR_OBJ(fileTextureNamePattern); + SET_ATTR_OBJ(uvTilingMode); + SET_ATTR_OBJ(uvCoord); + SET_ATTR_OBJ(wrapU); + SET_ATTR_OBJ(wrapV); + SET_ATTR_OBJ(mirrorU); + SET_ATTR_OBJ(mirrorV); + } + + { + SET_NODE_CLASS(imagePlane); + + SET_ATTR_OBJ(displayMode); + SET_ATTR_OBJ(imageName); + SET_ATTR_OBJ(useFrameExtension); + SET_ATTR_OBJ(frameOffset); + SET_ATTR_OBJ(frameExtension); + SET_ATTR_OBJ(fit); + SET_ATTR_OBJ(coverage); + SET_ATTR_OBJ(coverageOrigin); + SET_ATTR_OBJ(depth); + SET_ATTR_OBJ(rotate); + SET_ATTR_OBJ(size); + SET_ATTR_OBJ(offset); + SET_ATTR_OBJ(width); + SET_ATTR_OBJ(height); + SET_ATTR_OBJ(imageCenter); + } + + return MStatus::kSuccess; +} + +} // namespace MayaAttrs + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.h b/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.h new file mode 100644 index 0000000000..5122936766 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/mayaAttrs.h @@ -0,0 +1,169 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_ATTRS_H +#define MAYAHYDRALIB_ATTRS_H + +#include + +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +#ifdef __GNUC__ +#pragma GCC visibility push(hidden) +#endif + +namespace MayaAttrs { + +namespace node { + +extern MObject message; + +} // namespace node + +namespace dagNode { + +using namespace node; +extern MObject visibility; +extern MObject worldMatrix; +extern MObject intermediateObject; +extern MObject instObjGroups; +extern MObject overrideEnabled; +extern MObject overrideVisibility; + +} // namespace dagNode + +namespace nonAmbientLightShapeNode { + +using namespace dagNode; +extern MObject decayRate; +extern MObject emitDiffuse; +extern MObject emitSpecular; + +} // namespace nonAmbientLightShapeNode + +namespace nonExtendedLightShapeNode { + +using namespace nonAmbientLightShapeNode; +extern MObject dmapResolution; +extern MObject dmapBias; +extern MObject dmapFilterSize; + +} // namespace nonExtendedLightShapeNode + +namespace spotLight { + +using namespace nonExtendedLightShapeNode; +extern MObject coneAngle; +extern MObject dropoff; + +} // namespace spotLight + +namespace directionalLight { + +using namespace nonExtendedLightShapeNode; +extern MObject lightAngle; + +} // namespace directionalLight + +namespace surfaceShape { + +using namespace dagNode; +extern MObject doubleSided; + +} // namespace surfaceShape + +// mesh + +namespace mesh { + +using namespace surfaceShape; +extern MObject pnts; +extern MObject inMesh; +extern MObject uvPivot; +extern MObject displaySmoothMesh; +extern MObject smoothLevel; + +} // namespace mesh + +// curve + +namespace nurbsCurve { + +using namespace surfaceShape; +extern MObject controlPoints; + +} // namespace nurbsCurve + +namespace shadingEngine { + +using namespace node; +extern MObject surfaceShader; + +} // namespace shadingEngine + +namespace file { + +using namespace node; +extern MObject computedFileTextureNamePattern; +extern MObject fileTextureName; +extern MObject fileTextureNamePattern; +extern MObject uvTilingMode; +extern MObject uvCoord; +extern MObject wrapU; +extern MObject wrapV; +extern MObject mirrorU; +extern MObject mirrorV; + +} // namespace file + +namespace imagePlane { + +using namespace dagNode; +extern MObject imageName; +extern MObject useFrameExtension; +extern MObject frameOffset; +extern MObject frameExtension; +extern MObject displayMode; + +extern MObject fit; +extern MObject coverage; +extern MObject coverageOrigin; +extern MObject depth; +extern MObject rotate; +extern MObject size; +extern MObject offset; +extern MObject width; +extern MObject height; +extern MObject imageCenter; + +} // namespace imagePlane + +MAYAHYDRALIB_API +MStatus initialize(); + +} // namespace MayaAttrs + +#ifdef __GNUC__ +#pragma GCC visibility pop +#endif + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_ATTRS_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/meshAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/meshAdapter.cpp new file mode 100644 index 0000000000..a873b21be2 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/meshAdapter.cpp @@ -0,0 +1,471 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * This file contains the MayaHydraMeshAdapter class to translate from a Maya mesh to hydra. + * Please note that, as of May 2023, this is optionally used by mayaHydra, with + * a compile-time switch (see sceneDelegate.h). + * + * We can also translate from a MRenderitem to Hydra using the + * MayaHydraRenderItemAdapter class. + */ + +namespace { + +const std::pair _dirtyBits[] { + { MayaAttrs::mesh::pnts, + // This is useful when the user edits the mesh. + HdChangeTracker::DirtyPoints | HdChangeTracker::DirtyExtent + | HdChangeTracker::DirtySubdivTags }, + { MayaAttrs::mesh::inMesh, + // We are tracking topology changes and uv changes separately + HdChangeTracker::DirtyPoints | HdChangeTracker::DirtyExtent + | HdChangeTracker::DirtySubdivTags }, + { MayaAttrs::mesh::worldMatrix, HdChangeTracker::DirtyTransform }, + { MayaAttrs::mesh::doubleSided, HdChangeTracker::DirtyDoubleSided }, + { MayaAttrs::mesh::intermediateObject, HdChangeTracker::DirtyVisibility }, + { MayaAttrs::mesh::uvPivot, + // Tracking manual edits to uvs. + HdChangeTracker::DirtyPrimvar }, + { MayaAttrs::mesh::displaySmoothMesh, HdChangeTracker::DirtyDisplayStyle }, + { MayaAttrs::mesh::smoothLevel, HdChangeTracker::DirtyDisplayStyle } +}; + +} // namespace + +/** + * \brief MayaHydraMeshAdapter is used to handle the translation from a Maya mesh to hydra. + * Please note that, at this time, this is not used by the hydra plugin, we translate from a + * renderitem to hydra using the MayaHydraRenderItemAdapter class. + */ +class MayaHydraMeshAdapter : public MayaHydraShapeAdapter +{ +public: + MayaHydraMeshAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraShapeAdapter(producer->GetPrimPath(dag, false), producer, dag) + { + } + + ~MayaHydraMeshAdapter() = default; + + void Populate() override + { + if (_isPopulated) { + return; + } + GetSceneProducer()->InsertRprim(this, HdPrimTypeTokens->mesh, GetID(), GetInstancerID()); + _isPopulated = true; + } + + void AddBuggyCallback(MCallbackId id) { _buggyCallbacks.append(id); } + + void CreateCallbacks() override + { + MStatus status; + auto obj = GetNode(); + if (obj != MObject::kNullObj) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating mesh adapter callbacks for prim (%s).\n", GetID().GetText()); + + auto id + = MNodeMessage::addNodeDirtyPlugCallback(obj, NodeDirtiedCallback, this, &status); + if (status) { + AddCallback(id); + } + id = MNodeMessage::addAttributeChangedCallback( + obj, AttributeChangedCallback, this, &status); + if (status) { + AddCallback(id); + } + id = MPolyMessage::addPolyTopologyChangedCallback( + obj, TopologyChangedCallback, this, &status); + if (status) { + AddCallback(id); + } + bool wantModifications[3] = { true, true, true }; + id = MPolyMessage::addPolyComponentIdChangedCallback( + obj, wantModifications, 3, ComponentIdChanged, this, &status); + if (status) { + AddBuggyCallback(id); + } + id = MPolyMessage::addUVSetChangedCallback(obj, UVSetChangedCallback, this, &status); + if (status) { + AddBuggyCallback(id); + } + } + MayaHydraDagAdapter::CreateCallbacks(); + } + + MAYAHYDRALIB_API + void RemoveCallbacks() override + { + if (_buggyCallbacks.length() > 0) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Removing buggy PolyComponentIdChangedCallbacks\n"); + if (_node != MObject::kNullObj && MObjectHandle(_node).isValid()) { + MMessage::removeCallbacks(_buggyCallbacks); + } + _buggyCallbacks.clear(); + } + MayaHydraAdapter::RemoveCallbacks(); + } + + bool IsSupported() const override + { + return GetSceneProducer()->GetRenderIndex().IsRprimTypeSupported(HdPrimTypeTokens->mesh); + } + + VtValue GetUVs() + { + MStatus status; + MFnMesh mesh(GetDagPath(), &status); + if (ARCH_UNLIKELY(!status)) { + return {}; + } + VtArray uvs; + uvs.reserve(static_cast(mesh.numFaceVertices())); + for (MItMeshPolygon pit(GetDagPath()); !pit.isDone(); pit.next()) { + const auto vertexCount = pit.polygonVertexCount(); + for (auto i = decltype(vertexCount) { 0 }; i < vertexCount; ++i) { + float2 uv = { 0.0f, 0.0f }; + pit.getUV(i, uv); + uvs.push_back(GfVec2f(uv[0], uv[1])); + } + } + + return VtValue(uvs); + } + + VtValue GetPoints(const MFnMesh& mesh) + { + MStatus status; + const auto* rawPoints = reinterpret_cast(mesh.getRawPoints(&status)); + if (ARCH_UNLIKELY(!status)) { + return {}; + } + VtVec3fArray ret; + ret.assign(rawPoints, rawPoints + mesh.numVertices()); + return VtValue(ret); + } + + VtValue Get(const TfToken& key) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraMeshAdapter::Get(%s) - %s\n", + key.GetText(), + GetDagPath().partialPathName().asChar()); + + if (key == HdTokens->points) { + MStatus status; + MFnMesh mesh(GetDagPath(), &status); + if (ARCH_UNLIKELY(!status)) { + return {}; + } + return GetPoints(mesh); + } else if (key == MayaHydraAdapterTokens->st) { + return GetUVs(); + } + return {}; + } + + size_t SamplePrimvar(const TfToken& key, size_t maxSampleCount, float* times, VtValue* samples) + override + { + if (maxSampleCount < 1) { + return 0; + } + + if (key == HdTokens->points) { + MStatus status; + MFnMesh mesh(GetDagPath(), &status); + if (ARCH_UNLIKELY(!status)) { + return 0; + } + return GetSceneProducer()->SampleValues( + maxSampleCount, times, samples, [&]() -> VtValue { return GetPoints(mesh); }); + } else if (key == MayaHydraAdapterTokens->st) { + times[0] = 0.0f; + samples[0] = GetUVs(); + return 1; + } + return 0; + } + + HdMeshTopology GetMeshTopology() override + { + MFnMesh mesh(GetDagPath()); + const auto numPolygons = mesh.numPolygons(); + VtIntArray faceVertexCounts; + faceVertexCounts.reserve(static_cast(numPolygons)); + VtIntArray faceVertexIndices; + faceVertexIndices.reserve(static_cast(mesh.numFaceVertices())); + for (MItMeshPolygon pit(GetDagPath()); !pit.isDone(); pit.next()) { + const auto vc = pit.polygonVertexCount(); + faceVertexCounts.push_back(vc); + for (auto i = decltype(vc) { 0 }; i < vc; ++i) { + faceVertexIndices.push_back(pit.vertexIndex(i)); + } + } + + return HdMeshTopology( + (GetSceneProducer()->GetParams().displaySmoothMeshes || GetDisplayStyle().refineLevel > 0) + ? PxOsdOpenSubdivTokens->catmullClark + : PxOsdOpenSubdivTokens->none, + + UsdGeomTokens->rightHanded, + faceVertexCounts, + faceVertexIndices); + } + + HdDisplayStyle GetDisplayStyle() override + { + MStatus status; + MFnDependencyNode node(GetNode(), &status); + if (ARCH_UNLIKELY(!status)) { + return { 0, false, false }; + } + const auto displaySmoothMesh + = node.findPlug(MayaAttrs::mesh::displaySmoothMesh, true).asShort(); + if (displaySmoothMesh == 0) { + return { 0, false, false }; + } + const auto smoothLevel + = std::max(0, node.findPlug(MayaAttrs::mesh::smoothLevel, true).asInt()); + return { smoothLevel, false, false }; + } + + PxOsdSubdivTags GetSubdivTags() override + { + PxOsdSubdivTags tags; + if (GetDisplayStyle().refineLevel < 1) { + return tags; + } + + MStatus status; + MFnMesh mesh(GetNode(), &status); + if (ARCH_UNLIKELY(!status)) { + return tags; + } + MUintArray creaseVertIds; + MDoubleArray creaseVertValues; + mesh.getCreaseVertices(creaseVertIds, creaseVertValues); + const auto creaseVertIdCount = creaseVertIds.length(); + if (!TF_VERIFY(creaseVertIdCount == creaseVertValues.length())) { + return tags; + } + + MUintArray creaseEdgeIds; + MDoubleArray creaseEdgeValues; + mesh.getCreaseEdges(creaseEdgeIds, creaseEdgeValues); + const auto creaseEdgeIdCount = creaseEdgeIds.length(); + if (!TF_VERIFY(creaseEdgeIdCount == creaseEdgeIds.length())) { + return tags; + } + + if (creaseVertIdCount > 0) { + VtIntArray cornerIndices(creaseVertIdCount); + VtFloatArray cornerWeights(creaseVertIdCount); + for (auto i = decltype(creaseVertIdCount) { 0 }; i < creaseVertIdCount; ++i) { + cornerIndices[i] = static_cast(creaseVertIds[i]); + cornerWeights[i] = static_cast(creaseVertValues[i]); + } + + tags.SetCornerIndices(cornerIndices); + tags.SetCornerWeights(cornerWeights); + } + + if (creaseEdgeIdCount > 0) { + VtIntArray edgeIndices(creaseEdgeIdCount * 2); + VtFloatArray edgeWeights(creaseEdgeIdCount); + int edgeVertices[2] = { 0, 0 }; + for (auto i = decltype(creaseEdgeIdCount) { 0 }; i < creaseEdgeIdCount; ++i) { + mesh.getEdgeVertices(creaseEdgeIds[i], edgeVertices); + edgeIndices[i * 2] = edgeVertices[0]; + edgeIndices[i * 2 + 1] = edgeVertices[1]; + edgeWeights[i] = static_cast(creaseEdgeValues[i]); + } + + tags.SetCreaseIndices(edgeIndices); + tags.SetCreaseLengths(VtIntArray(creaseEdgeIdCount, 2)); + tags.SetCreaseWeights(edgeWeights); + } + + tags.SetVertexInterpolationRule(UsdGeomTokens->edgeAndCorner); + tags.SetFaceVaryingInterpolationRule(UsdGeomTokens->cornersPlus1); + tags.SetTriangleSubdivision(UsdGeomTokens->catmullClark); + + return tags; + } + + HdPrimvarDescriptorVector GetPrimvarDescriptors(HdInterpolation interpolation) override + { + if (interpolation == HdInterpolationVertex) { + HdPrimvarDescriptor desc; + desc.name = UsdGeomTokens->points; + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->point; + return { desc }; + } else if (interpolation == HdInterpolationFaceVarying) { + // UVs are face varying in maya. + MFnMesh mesh(GetDagPath()); + if (mesh.numUVs() > 0) { + HdPrimvarDescriptor desc; + desc.name = MayaHydraAdapterTokens->st; + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->textureCoordinate; + return { desc }; + } + } + return {}; + } + + bool GetDoubleSided() const override + { + MFnMesh mesh(GetDagPath()); + auto p = mesh.findPlug(MayaAttrs::mesh::doubleSided, true); + if (ARCH_UNLIKELY(p.isNull())) { + return true; + } + bool doubleSided = true; + p.getValue(doubleSided); + return doubleSided; + } + + bool HasType(const TfToken& typeId) const override { return typeId == HdPrimTypeTokens->mesh; } + +private: + static void NodeDirtiedCallback(MObject& node, MPlug& plug, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + for (const auto& it : _dirtyBits) { + if (it.first == plug) { + adapter->MarkDirty(it.second); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MESH_PLUG_DIRTY) + .Msg( + "Marking prim dirty with bits %u because %s plug was " + "dirtied.\n", + it.second, + plug.partialName().asChar()); + return; + } + } + + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MESH_UNHANDLED_PLUG_DIRTY) + .Msg( + "%s (%s) plug dirtying was not handled by " + "MayaHydraMeshAdapter::NodeDirtiedCallback.\n", + plug.name().asChar(), + plug.partialName().asChar()); + } + + // For material assignments for now. + static void AttributeChangedCallback( + MNodeMessage::AttributeMessage msg, + MPlug& plug, + MPlug& otherPlug, + void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + if (plug == MayaAttrs::mesh::instObjGroups) { + adapter->MarkDirty(HdChangeTracker::DirtyMaterialId); + } else { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MESH_UNHANDLED_PLUG_DIRTY) + .Msg( + "%s (%s) plug dirtying was not handled by " + "MayaHydraMeshAdapter::attributeChangedCallback.\n", + plug.name().asChar(), + plug.name().asChar()); + } + } + + static void TopologyChangedCallback(MObject& node, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty( + HdChangeTracker::DirtyTopology | HdChangeTracker::DirtyPrimvar + | HdChangeTracker::DirtyPoints); + } + + static void ComponentIdChanged(MUintArray componentIds[], unsigned int count, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty( + HdChangeTracker::DirtyTopology | HdChangeTracker::DirtyPrimvar + | HdChangeTracker::DirtyPoints); + } + + static void UVSetChangedCallback( + MObject& node, + const MString& name, + MPolyMessage::MessageType type, + void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty(HdChangeTracker::DirtyPrimvar); + } + + // Maya has a bug with removing some MPolyMessage callbacks. Known + // problem callbacks include: + // MPolyMessage::addPolyComponentIdChangedCallback + // MPolyMessage::addUVSetChangedCallback + // Reproduction code can be found here: + // https://gist.github.com/elrond79/668d9809873125f608e0f7360fff7fac + // To work around this, we register these callbacks specially, and only + // remove them if the underlying node is currently valid. + MCallbackIdArray _buggyCallbacks; +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, mesh) +{ + MayaHydraAdapterRegistry::RegisterShapeAdapter( + TfToken("mesh"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraShapeAdapterPtr { + return MayaHydraShapeAdapterPtr(new MayaHydraMeshAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/nurbsCurveAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/nurbsCurveAdapter.cpp new file mode 100644 index 0000000000..a852d6ed8b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/nurbsCurveAdapter.cpp @@ -0,0 +1,249 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * This file contains the MayaHydraNurbsCurveAdapter class to translate from a Maya NURBS curve to + * hydra. Please note that, at this time, this is not used by the hydra plugin, we translate from a + * renderitem to hydra using the MayaHydraRenderItemAdapter class. + */ + +namespace { + +const std::array, 4> _dirtyBits { { + { MayaAttrs::nurbsCurve::controlPoints, + HdChangeTracker::DirtyPoints | HdChangeTracker::DirtyExtent }, + { MayaAttrs::nurbsCurve::worldMatrix, HdChangeTracker::DirtyTransform }, + { MayaAttrs::nurbsCurve::doubleSided, HdChangeTracker::DirtyDoubleSided }, + { MayaAttrs::nurbsCurve::intermediateObject, HdChangeTracker::DirtyVisibility }, +} }; + +} // namespace + +/** + * \brief MayaHydraNurbsCurveAdapter is used to handle the translation from a Maya NURBS curve to + * hydra. Please note that, at this time, this is not used by the hydra plugin, we translate from a + * renderitem to hydra using the MayaHydraRenderItemAdapter class. + */ +class MayaHydraNurbsCurveAdapter : public MayaHydraShapeAdapter +{ +public: + MayaHydraNurbsCurveAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraShapeAdapter(producer->GetPrimPath(dag, false), producer, dag) + { + } + + ~MayaHydraNurbsCurveAdapter() = default; + + bool IsSupported() const override + { + return GetSceneProducer()->GetRenderIndex().IsRprimTypeSupported(HdPrimTypeTokens->basisCurves); + } + + void Populate() override { GetSceneProducer()->InsertRprim(this, HdPrimTypeTokens->basisCurves, GetID()); } + + void CreateCallbacks() override + { + MStatus status; + auto obj = GetNode(); + if (obj != MObject::kNullObj) { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CALLBACKS) + .Msg("Creating nurbs curve adapter callbacks for prim (%s).\n", GetID().GetText()); + + auto id + = MNodeMessage::addNodeDirtyPlugCallback(obj, NodeDirtiedCallback, this, &status); + if (status) { + AddCallback(id); + } + id = MNodeMessage::addAttributeChangedCallback( + obj, AttributeChangedCallback, this, &status); + if (status) { + AddCallback(id); + } + id = MPolyMessage::addPolyTopologyChangedCallback( + obj, TopologyChangedCallback, this, &status); + if (status) { + AddCallback(id); + } + bool wantModifications[3] = { true, true, true }; + id = MPolyMessage::addPolyComponentIdChangedCallback( + obj, wantModifications, 3, ComponentIdChanged, this, &status); + if (status) { + AddCallback(id); + } + } + MayaHydraDagAdapter::CreateCallbacks(); + } + + VtValue Get(const TfToken& key) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraNurbsCurveAdapter::Get(%s) - %s\n", + key.GetText(), + GetDagPath().partialPathName().asChar()); + + if (key == HdTokens->points) { + MFnNurbsCurve curve(GetDagPath()); + MStatus status; + MPointArray pointArray; + status = curve.getCVs(pointArray); + if (!status) { + return {}; + } + VtVec3fArray ret(pointArray.length()); + const auto pointCount = pointArray.length(); + for (auto i = decltype(pointCount) { 0 }; i < pointCount; i++) { + const auto pt = pointArray[i]; + ret[i] = GfVec3f( + static_cast(pt.x), static_cast(pt.y), static_cast(pt.z)); + } + return VtValue(ret); + } + return {}; + } + + HdBasisCurvesTopology GetBasisCurvesTopology() override + { + MFnNurbsCurve curve(GetDagPath()); + const auto pointCount = curve.numCVs(); + + VtIntArray curveVertexCounts; + const auto numIndices = (pointCount - 1) * 2; + curveVertexCounts.push_back(numIndices); + VtIntArray curveIndices(static_cast(numIndices)); + for (auto i = decltype(numIndices) { 0 }; i < numIndices / 2; i++) { + curveIndices[i * 2] = i; + curveIndices[i * 2 + 1] = i + 1; + } + + return HdBasisCurvesTopology( + HdTokens->linear, + HdTokens->bezier, + HdTokens->segmented, + curveVertexCounts, + curveIndices); + } + + HdPrimvarDescriptorVector GetPrimvarDescriptors(HdInterpolation interpolation) override + { + if (interpolation == HdInterpolationVertex) { + HdPrimvarDescriptor desc; + desc.name = UsdGeomTokens->points; + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->point; + return { desc }; + } + return {}; + } + + TfToken GetRenderTag() const override { return HdRenderTagTokens->guide; } + +private: + static void NodeDirtiedCallback(MObject& node, MPlug& plug, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + for (const auto& it : _dirtyBits) { + if (it.first == plug) { + adapter->MarkDirty(it.second); + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CURVE_PLUG_DIRTY) + .Msg( + "Marking prim dirty with bits %u because %s plug was " + "dirtied.\n", + it.second, + plug.partialName().asChar()); + return; + } + } + + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CURVE_UNHANDLED_PLUG_DIRTY) + .Msg( + "%s (%s) plug dirtying was not handled by " + "MayaHydraNurbsCurveAdapter::NodeDirtiedCallback.\n", + plug.name().asChar(), + plug.partialName().asChar()); + } + + // For material assignments for now. + static void AttributeChangedCallback( + MNodeMessage::AttributeMessage msg, + MPlug& plug, + MPlug& otherPlug, + void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + if (plug == MayaAttrs::mesh::instObjGroups) { + adapter->MarkDirty(HdChangeTracker::DirtyMaterialId); + } else { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_CURVE_UNHANDLED_PLUG_DIRTY) + .Msg( + "%s (%s) plug dirtying was not handled by " + "MayaHydraNurbsCurveAdapter::attributeChangedCallback.\n", + plug.name().asChar(), + plug.name().asChar()); + } + } + + static void TopologyChangedCallback(MObject& node, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty( + HdChangeTracker::DirtyTopology | HdChangeTracker::DirtyPrimvar + | HdChangeTracker::DirtyPoints); + } + + static void ComponentIdChanged(MUintArray componentIds[], unsigned int count, void* clientData) + { + auto* adapter = reinterpret_cast(clientData); + adapter->MarkDirty( + HdChangeTracker::DirtyTopology | HdChangeTracker::DirtyPrimvar + | HdChangeTracker::DirtyPoints); + } +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, mesh) +{ + MayaHydraAdapterRegistry::RegisterShapeAdapter( + TfToken("nurbsCurve"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraShapeAdapterPtr { + return MayaHydraShapeAdapterPtr(new MayaHydraNurbsCurveAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/pointLightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/pointLightAdapter.cpp new file mode 100644 index 0000000000..00644bc636 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/pointLightAdapter.cpp @@ -0,0 +1,87 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraPointLightAdapter is used to handle the translation from a Maya point light to + * hydra. + */ +class MayaHydraPointLightAdapter : public MayaHydraLightAdapter +{ +public: + MayaHydraPointLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraLightAdapter(producer, dag) + { + } + + const TfToken& LightType() const override + { + if (GetSceneProducer()->IsHdSt()) { + return HdPrimTypeTokens->simpleLight; + } else { + return HdPrimTypeTokens->sphereLight; + } + } + + VtValue GetLightParamValue(const TfToken& paramName) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE) + .Msg( + "Called MayaHydraPointLightAdapter::GetLightParamValue(%s) - %s\n", + paramName.GetText(), + GetDagPath().partialPathName().asChar()); + + MFnPointLight light(GetDagPath()); + if (paramName == HdLightTokens->radius) { + const float radius = light.shadowRadius(); + return VtValue(radius); + } else if (paramName == UsdLuxTokens->treatAsPoint) { + const bool treatAsPoint = (light.shadowRadius() == 0.0); + return VtValue(treatAsPoint); + } + return MayaHydraLightAdapter::GetLightParamValue(paramName); + } +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, pointLight) +{ + MayaHydraAdapterRegistry::RegisterLightAdapter( + TfToken("pointLight"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraLightAdapterPtr { + return MayaHydraLightAdapterPtr(new MayaHydraPointLightAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.cpp new file mode 100644 index 0000000000..d5c24f8191 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.cpp @@ -0,0 +1,476 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "renderItemAdapter.h" + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +#define PLUG_THIS_PLUGIN \ + PlugRegistry::GetInstance().GetPluginWithName(TF_PP_STRINGIZE(MFB_PACKAGE_NAME)) + +/* + * MayaHydraRenderItemAdapter is used to translate from a render item to hydra. + * This is where we translate from Maya shapes (such as meshes) to hydra using their vertex and + * index buffers, look for "MVertexBuffer" and "MIndexBuffer" in this file to get more information. + */ +MayaHydraRenderItemAdapter::MayaHydraRenderItemAdapter( + const MDagPath& dagPath, + const SdfPath& slowId, + int fastId, + MayaHydraSceneProducer* producer, + const MRenderItem& ri) + : MayaHydraAdapter(MObject(), slowId, producer) + , _dagPath(dagPath) + , _primitive(ri.primitive()) + , _name(ri.name()) + , _fastId(fastId) +{ + _InsertRprim(this); +} + +MayaHydraRenderItemAdapter::~MayaHydraRenderItemAdapter() { _RemoveRprim(); } + +TfToken MayaHydraRenderItemAdapter::GetRenderTag() const { return HdRenderTagTokens->geometry; } + +void MayaHydraRenderItemAdapter::UpdateTransform(const MRenderItem& ri) +{ + MMatrix matrix; + if (ri.getMatrix(matrix) == MStatus::kSuccess) { + _transform[0] = GetGfMatrixFromMaya(matrix); + if (GetSceneProducer()->GetParams().motionSamplesEnabled()) { + MDGContextGuard guard(MAnimControl::currentTime() + 1.0); + _transform[1] = GetGfMatrixFromMaya(matrix); + } else { + _transform[1] = _transform[0]; + } + } +} + +bool MayaHydraRenderItemAdapter::IsSupported() const +{ + switch (_primitive) { + case MHWRender::MGeometry::Primitive::kTriangles: + return GetSceneProducer()->GetRenderIndex().IsRprimTypeSupported(HdPrimTypeTokens->mesh); + case MHWRender::MGeometry::Primitive::kLines: + case MHWRender::MGeometry::Primitive::kLineStrip: + return GetSceneProducer()->GetRenderIndex().IsRprimTypeSupported(HdPrimTypeTokens->basisCurves); + case MHWRender::MGeometry::Primitive::kPoints: + return GetSceneProducer()->GetRenderIndex().IsRprimTypeSupported(HdPrimTypeTokens->points); + default: return false; + } +} + +void MayaHydraRenderItemAdapter::_InsertRprim(MayaHydraAdapter* adapter) +{ + switch (GetPrimitive()) { + case MHWRender::MGeometry::Primitive::kTriangles: + GetSceneProducer()->InsertRprim(adapter, HdPrimTypeTokens->mesh, GetID(), {}); + break; + case MHWRender::MGeometry::Primitive::kLines: + case MHWRender::MGeometry::Primitive::kLineStrip: + GetSceneProducer()->InsertRprim(adapter, HdPrimTypeTokens->basisCurves, GetID(), {}); + break; + case MHWRender::MGeometry::Primitive::kPoints: + GetSceneProducer()->InsertRprim(adapter, HdPrimTypeTokens->points, GetID(), {}); + break; + default: + assert(false); // unexpected/unsupported primitive type + break; + } +} + +void MayaHydraRenderItemAdapter::_RemoveRprim() { GetSceneProducer()->RemoveRprim(GetID());} + +// We receive in that function the changes made in the Maya viewport between the last frame rendered +// and the current frame +void MayaHydraRenderItemAdapter::UpdateFromDelta(const UpdateFromDeltaData& data) +{ + if (_primitive != MHWRender::MGeometry::Primitive::kTriangles + && _primitive != MHWRender::MGeometry::Primitive::kLines + && _primitive != MHWRender::MGeometry::Primitive::kLineStrip) { + return; + } + + const bool positionsHaveBeenReset + = (0 == _positions.size()); // when positionsHaveBeenReset is true we need to recompute the + // geometry and topology as our data has been cleared + using MVS = MDataServerOperation::MViewportScene; + // const bool isNew = flags & MViewportScene::MVS_new; //not used yet + const bool visible = data._flags & MVS::MVS_visible; + const bool matrixChanged = data._flags & MVS::MVS_changedMatrix; + const bool geomChanged = (data._flags & MVS::MVS_changedGeometry) || positionsHaveBeenReset; + const bool topoChanged = (data._flags & MVS::MVS_changedTopo) || positionsHaveBeenReset; + const bool visibChanged = data._flags & MVS::MVS_changedVisibility; + const bool effectChanged = data._flags & MVS::MVS_changedEffect; + + HdDirtyBits dirtyBits = 0; + + if (data._wireframeColor != _wireframeColor) { + _wireframeColor = data._wireframeColor; + dirtyBits |= HdChangeTracker::DirtyPrimvar; // displayColor primVar + } + + const bool displayStatusChanged = (_displayStatus != data._displayStatus); + _displayStatus = data._displayStatus; + const bool hideOnPlayback = data._ri.isHideOnPlayback(); + if (hideOnPlayback != _isHideOnPlayback) { + _isHideOnPlayback = hideOnPlayback; + dirtyBits |= HdChangeTracker::DirtyVisibility; + } + + //Special case for aiSkydomeLight which is visible only when it is selected + if (_isArnoldSkyDomeLightTriangleShape && displayStatusChanged){ + SetVisible(IsRenderItemSelected()); + dirtyBits |= HdChangeTracker::DirtyVisibility; + } else + if (visibChanged) { + SetVisible(visible); + dirtyBits |= HdChangeTracker::DirtyVisibility; + } + + if (effectChanged) { + dirtyBits |= HdChangeTracker::DirtyMaterialId; + } + if (matrixChanged) { + dirtyBits |= HdChangeTracker::DirtyTransform; + } + if (geomChanged) { + dirtyBits |= HdChangeTracker::DirtyPoints; + } + if (topoChanged) { + dirtyBits |= (HdChangeTracker::DirtyTopology | HdChangeTracker::DirtyPrimvar); + } + + MGeometry* geom = nullptr; + if (geomChanged | topoChanged) { + geom = data._ri.geometry(); + } + VtIntArray vertexIndices; + VtIntArray vertexCounts; + + // Vertices + MVertexBuffer* verts = nullptr; + if (geomChanged && geom && geom->vertexBufferCount() > 0) { + // Assume first stream contains the positions. We do not handle multiple streams for now. + verts = geom->vertexBuffer(0); + if (verts) { + int vertCount = 0; + const unsigned int originalVertexCount = verts->vertexCount(); + if (topoChanged) { + vertCount = originalVertexCount; + } else { + // Keep the previously-determined vertex count in case it was truncated. + const size_t positionSize = _positions.size(); + if (positionSize > 0 && positionSize <= originalVertexCount) { + vertCount = positionSize; + } else { + vertCount = originalVertexCount; + } + } + + _positions.clear(); + //_positions.resize(vertCount); + // map() is usually just reading from the software copy of the vp2 buffers. It was also + // showing up in vtune that it was sometimes mapping OpenGL buffers to read from, which + // is slow. Disabling processing of non-triangle render made that disappear. Maybe + // something like joint render items point to hardware only buffers? + const auto* vertexPositions = reinterpret_cast(verts->map()); + // NOTE: Looking at MayaHydraMeshAdapter::GetPoints notice assign(vertexPositions, + // vertexPositions + vertCount) Why are we not multiplying with sizeof(GfVec3f) to + // calculate the offset ? The following happens when I try to do it : Invalid Hydra prim + // - Vertex primvar points has 288 elements, while its topology references only upto + // element index 24. + if (TF_VERIFY(vertexPositions)) { + _positions.assign(vertexPositions, vertexPositions + vertCount); + } + verts->unmap(); + } + } + + // Indices + // Note that a Primitive::kLineStrip index buffer is unavailable. The following section is + // skipped. In such case, indices are implicitly defined below. + MIndexBuffer* indices = nullptr; + if (topoChanged && geom && geom->indexBufferCount() > 0) { + // Assume first stream contains the positions. + indices = geom->indexBuffer(0); + if (indices) { + int indexCount = indices->size(); + vertexIndices.resize(indexCount); + int* indicesData = (int*)indices->map(); + // USD spamming the "topology references only upto element" message is super + // slow. Scanning the index array to look for an incompletely used vertex + // buffer is innefficient, but it's better than the spammy warning. Cause of + // the incompletely used vertex buffer is unclear. Maya scene data just is + // that way sometimes. + int maxIndex = 0; + for (int i = 0; i < indexCount; i++) { + if (indicesData[i] > maxIndex) { + maxIndex = indicesData[i]; + } + } + + // VtArray operator[] is oddly expensive, ~10ms per frame here. Replace with assign(). + // for (int i = 0; i < indexCount; i++) vertexIndices[i] = indicesData[i]; + vertexIndices.assign(indicesData, indicesData + indexCount); + + if (maxIndex < (int64_t)_positions.size() - 1) { + _positions.resize(maxIndex + 1); + } + + switch (GetPrimitive()) { + case MHWRender::MGeometry::Primitive::kTriangles: + vertexCounts.resize(indexCount / 3); + vertexCounts.assign(indexCount / 3, 3); + + // UVs + if (indexCount > 0) { + MVertexBuffer* mvb = nullptr; + for (int vbIdx = 0; vbIdx < geom->vertexBufferCount(); vbIdx++) { + mvb = geom->vertexBuffer(vbIdx); + if (!mvb) + continue; + + const MVertexBufferDescriptor& desc = mvb->descriptor(); + + if (desc.semantic() != MGeometry::Semantic::kTexture) + continue; + + // Hydra expects a uv coordinate for each face-index, not 1 per vertex. + // e.g. a cube expects 36 uvs not 24. + _uvs.clear(); + _uvs.resize(indices->size()); + float* uvs = (float*)mvb->map(); + for (int i = 0; i < indexCount; ++i) { + _uvs[i].Set(&uvs[indicesData[i] * 2]); + } + mvb->unmap(); + break; + } + } + break; + case MHWRender::MGeometry::Primitive::kLines: + vertexCounts.resize(indexCount); + vertexCounts.assign(indexCount / 2, 2); + break; + + default: + assert(false); // unexpected/unsupported primitive type + break; + } + indices->unmap(); + } + } + + if (topoChanged) { + switch (GetPrimitive()) { + case MGeometry::Primitive::kTriangles: + _topology.reset(new HdMeshTopology( + (GetSceneProducer()->GetParams().displaySmoothMeshes + || GetDisplayStyle().refineLevel > 0) + ? PxOsdOpenSubdivTokens->catmullClark + : PxOsdOpenSubdivTokens->none, + UsdGeomTokens->rightHanded, + vertexCounts, + vertexIndices)); + break; + case MGeometry::Primitive::kLines: + case MGeometry::Primitive::kLineStrip: { + TfToken curveTopoType(HdTokens->segmented); + if (GetPrimitive() == MGeometry::Primitive::kLineStrip) { + // Line strips indices are implicitly defined: + // When using line strips, the GPU will draw a connected series of lines between the + // vertices specified by the indices. When specifying indices for a line strip, you + // only need to specify the order of the vertices that you want connected. This is + // implicit in Hydra when specifying an empty index buffer. + curveTopoType = HdTokens->nonperiodic; + vertexCounts.assign(1, _positions.size()); + vertexIndices = VtIntArray(); + } + _topology.reset(new HdBasisCurvesTopology( + HdTokens->linear, + // basis type is ignored, due to linear curve type + {}, + curveTopoType, + vertexCounts, + vertexIndices)); + break; + } + default: break; + } + } + + MarkDirty(dirtyBits); +} + +HdMeshTopology MayaHydraRenderItemAdapter::GetMeshTopology() +{ + return _topology ? *static_cast(_topology.get()) : HdMeshTopology(); +} + +HdBasisCurvesTopology MayaHydraRenderItemAdapter::GetBasisCurvesTopology() +{ + return _topology ? *static_cast(_topology.get()) + : HdBasisCurvesTopology(); +} + +VtValue MayaHydraRenderItemAdapter::Get(const TfToken& key) +{ + if (key == HdTokens->points) { + return VtValue(_positions); + } + if (key == MayaHydraAdapterTokens->st) { + return VtValue(_uvs); + } + if (key == HdTokens->displayColor) { + return VtValue(GfVec4f( + _wireframeColor[0], _wireframeColor[1], _wireframeColor[2], _wireframeColor[3])); + } + + return {}; +} + +void MayaHydraRenderItemAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + if (dirtyBits != 0) { + GetSceneProducer()->MarkRprimDirty(GetID(), dirtyBits); + } +} + +HdPrimvarDescriptorVector +MayaHydraRenderItemAdapter::GetPrimvarDescriptors(HdInterpolation interpolation) +{ + HdPrimvarDescriptor desc; + + // Vertices + if (interpolation == HdInterpolationVertex) { + desc.name = UsdGeomTokens->points; + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->point; + return { desc }; + } else if (interpolation == HdInterpolationFaceVarying) { + // UVs are face varying in maya. + if (_primitive == MGeometry::Primitive::kTriangles) { + desc.name = MayaHydraAdapterTokens->st; + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->textureCoordinate; + return { desc }; + } + } else if (interpolation == HdInterpolationConstant) { + switch(_primitive){ + case MGeometry::Primitive::kPoints: //Fall into + case MGeometry::Primitive::kLines: //Fall into + case MGeometry::Primitive::kLineStrip: //Fall into + case MGeometry::Primitive::kAdjacentLines: //Fall into + case MGeometry::Primitive::kAdjacentLineStrip: + { + desc.name = HdTokens->displayColor;//Use display color only for lines/points (avoid triangles) + desc.interpolation = interpolation; + desc.role = HdPrimvarRoleTokens->color; + return { desc }; + } + break; + default: + break; + } + } + + return {}; +} + +VtValue MayaHydraRenderItemAdapter::GetMaterialResource() { return {}; } + +bool MayaHydraRenderItemAdapter::GetVisible() +{ + // Assuming that, if the playback is in the active view only + // (MAnimControl::kPlaybackViewActive), we are called because we are in the active view + if (_isHideOnPlayback) { + // MAYA-127216: Remove dependency on parent class MayaHydraAdapter. This will let us use + // MayaHydraSceneDelegate directly + auto sceneProducer = static_cast(GetSceneProducer()); + return !sceneProducer->GetPlaybackRunning(); + } + + return _visible; +} + +void MayaHydraRenderItemAdapter::SetPlaybackChanged() +{ + // There was a change in the playblack state, it started or stopped running so update any + // primitive that is dependent on this + if (_isHideOnPlayback) { + MarkDirty(HdChangeTracker::DirtyVisibility); + } +} + +bool MayaHydraRenderItemAdapter::IsRenderItemSelected() const +{ + return (MHWRender::DisplayStatus::kActive == _displayStatus) || + (MHWRender::DisplayStatus::kLead == _displayStatus); +} + +HdCullStyle MayaHydraRenderItemAdapter::GetCullStyle() const +{ + // HdCullStyleNothing means no culling, HdCullStyledontCare means : let the renderer choose + // between back or front faces culling. We don't want culling, since we want to see the + // backfaces being unlit with MayaHydraSceneDelegate::GetDoubleSided returning false. + return _isArnoldSkyDomeLightTriangleShape ? HdCullStyleFront : HdCullStyleNothing; +} + +/////////////////////////////////////////////////////////////////////// +// TF_REGISTRY +/////////////////////////////////////////////////////////////////////// + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, renderItem) { } + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.h new file mode 100644 index 0000000000..bbec8f1969 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/renderItemAdapter.h @@ -0,0 +1,216 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_RENDER_ITEM_ADAPTER_H +#define MAYAHYDRALIB_RENDER_ITEM_ADAPTER_H + +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +namespace { +std::string kRenderItemTypeName = "renderItem"; + +static constexpr const char* kPointSize = "pointSize"; + +static const SdfPath kInvalidMaterial = SdfPath("InvalidMaterial"); +} // namespace + +using MayaHydraRenderItemAdapterPtr = std::shared_ptr; + +/** + * \brief MayaHydraRenderItemAdapter is used to translate from a render item to hydra. + * This is where we translate from Maya shapes (such as meshes) to hydra. + */ +class MayaHydraRenderItemAdapter : public MayaHydraAdapter +{ +public: + MAYAHYDRALIB_API + MayaHydraRenderItemAdapter( + const MDagPath& dagPath, + const SdfPath& slowId, + int fastId, + MayaHydraSceneProducer* producer, + const MRenderItem& ri); + + MAYAHYDRALIB_API + virtual ~MayaHydraRenderItemAdapter(); + + MAYAHYDRALIB_API + virtual void RemovePrim() override { } + + MAYAHYDRALIB_API + virtual void Populate() override { } + + MAYAHYDRALIB_API + bool HasType(const TfToken& typeId) const override { return typeId == HdPrimTypeTokens->mesh; } + + MAYAHYDRALIB_API + virtual bool IsSupported() const override; + + MAYAHYDRALIB_API + bool GetDoubleSided() const override { return false; }; + + MAYAHYDRALIB_API + HdCullStyle GetCullStyle() const override; + + MAYAHYDRALIB_API + virtual void MarkDirty(HdDirtyBits dirtyBits) override; + + MAYAHYDRALIB_API + VtValue Get(const TfToken& key) override; + + MAYAHYDRALIB_API + VtValue GetMaterialResource(); + + MAYAHYDRALIB_API + void SetPlaybackChanged(); + + MAYAHYDRALIB_API + bool GetVisible() override; + + MAYAHYDRALIB_API + void SetVisible(bool val) { _visible = val; } + + MAYAHYDRALIB_API + const MColor& GetWireframeColor() const { return _wireframeColor; } + + MAYAHYDRALIB_API + MHWRender::DisplayStatus GetDisplayStatus() const { return _displayStatus; } + + MAYAHYDRALIB_API + GfMatrix4d GetTransform() override { return _transform[0]; } + + MAYAHYDRALIB_API + void InvalidateTransform() { } + + MAYAHYDRALIB_API + bool IsInstanced() const { return false; } + + MAYAHYDRALIB_API + HdPrimvarDescriptorVector GetPrimvarDescriptors(HdInterpolation interpolation) override; + + MAYAHYDRALIB_API + void UpdateTransform(const MRenderItem& ri); + + /// Class used to pass data to the UpdateFromDelta method, so we can extend the parameters in + /// the future if needed. + class UpdateFromDeltaData + { + public: + UpdateFromDeltaData( + MRenderItem& ri, + unsigned int flags, + const MColor& wireframeColor, + MHWRender::DisplayStatus displayStatus) + : _ri(ri) + , _flags(flags) + , _wireframeColor(wireframeColor) + , _displayStatus(displayStatus) + { + } + + MRenderItem& _ri; + unsigned int _flags; + const MColor& _wireframeColor; + MHWRender::DisplayStatus _displayStatus; + }; + + /// We receive in that function the changes made in the Maya viewport between the last frame + /// rendered and the current frame + MAYAHYDRALIB_API + void UpdateFromDelta(const UpdateFromDeltaData& data); + + MAYAHYDRALIB_API + HdMeshTopology GetMeshTopology() override; + + MAYAHYDRALIB_API + HdBasisCurvesTopology GetBasisCurvesTopology() override; + + MAYAHYDRALIB_API + virtual TfToken GetRenderTag() const override; + + MAYAHYDRALIB_API + SdfPath& GetMaterial() { return _material; } + + MAYAHYDRALIB_API + void SetMaterial(const SdfPath& val) { _material = val; } + + MAYAHYDRALIB_API + int GetFastID() const { return _fastId; } + + MAYAHYDRALIB_API + const MDagPath& GetDagPath() const { return _dagPath; } + + MAYAHYDRALIB_API + MGeometry::Primitive GetPrimitive() const { return _primitive; } + + MAYAHYDRALIB_API + const char* Name() const { return _name.asChar(); } + + MAYAHYDRALIB_API + void SetIsRenderITemAnaiSkydomeLightTriangleShape(bool val) {_isArnoldSkyDomeLightTriangleShape = val;} + + MAYAHYDRALIB_API + bool GetIsRenderITemAnaiSkydomeLightTriangleShape() const {return _isArnoldSkyDomeLightTriangleShape;} + + MAYAHYDRALIB_API + bool IsRenderItemSelected() const; + +private: + MAYAHYDRALIB_API + void _RemoveRprim(); + + MAYAHYDRALIB_API + void _InsertRprim(MayaHydraAdapter* adapter); + + SdfPath _material; + MDagPath _dagPath; + std::unique_ptr _topology = nullptr; + VtVec3fArray _positions = {}; + VtVec2fArray _uvs = {}; + MGeometry::Primitive _primitive; + MString _name; + GfMatrix4d _transform[2]; + int _fastId = 0; + bool _visible = false; + MColor _wireframeColor = { 1.f, 1.f, 1.f, 1.f }; + bool _isHideOnPlayback = false; + MHWRender::DisplayStatus _displayStatus = MHWRender::DisplayStatus::kNoStatus; + bool _isArnoldSkyDomeLightTriangleShape = false; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_RENDER_ITEM_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.cpp new file mode 100644 index 0000000000..0c297c8f45 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.cpp @@ -0,0 +1,156 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "shapeAdapter.h" + +#include +#include + +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +/* + * MayaHydraShapeAdapter is an adapter to translate from Maya shapes to hydra + * Please note that, at this time, this is not used by the hydra plug-in, we translate from a + * renderitem to hydra using the MayaHydraRenderItemAdapter class. + */ +MayaHydraShapeAdapter::MayaHydraShapeAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MDagPath& dagPath) + : MayaHydraDagAdapter(id, producer, dagPath) +{ + _CalculateExtent(); +} + +void MayaHydraShapeAdapter::_CalculateExtent() +{ + MStatus status; + MFnDagNode dagNode(GetDagPath(), &status); + if (ARCH_LIKELY(status)) { + const auto bb = dagNode.boundingBox(); + const auto mn = bb.min(); + const auto mx = bb.max(); + _extent.SetMin({ mn.x, mn.y, mn.z }); + _extent.SetMax({ mx.x, mx.y, mx.z }); + _extentDirty = false; + } +}; + +size_t MayaHydraShapeAdapter::SamplePrimvar( + const TfToken& key, + size_t maxSampleCount, + float* times, + VtValue* samples) +{ + if (maxSampleCount < 1) { + return 0; + } + times[0] = 0.0f; + samples[0] = Get(key); + return 1; +} + +HdMeshTopology MayaHydraShapeAdapter::GetMeshTopology() { return {}; }; + +HdBasisCurvesTopology MayaHydraShapeAdapter::GetBasisCurvesTopology() { return {}; }; + +HdDisplayStyle MayaHydraShapeAdapter::GetDisplayStyle() { return { 0, false, false }; } + +PxOsdSubdivTags MayaHydraShapeAdapter::GetSubdivTags() { return {}; } + +void MayaHydraShapeAdapter::MarkDirty(HdDirtyBits dirtyBits) +{ + MayaHydraDagAdapter::MarkDirty(dirtyBits); + if (dirtyBits & HdChangeTracker::DirtyPoints) { + _extentDirty = true; + } +} + +MObject MayaHydraShapeAdapter::GetMaterial() +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraShapeAdapter::GetMaterial() - %s\n", + GetDagPath().partialPathName().asChar()); + + MStatus status; + MFnDagNode dagNode(GetDagPath(), &status); + if (!status) { + return MObject::kNullObj; + } + + auto instObjGroups = dagNode.findPlug(MayaAttrs::dagNode::instObjGroups, true); + if (instObjGroups.isNull()) { + return MObject::kNullObj; + } + + MPlugArray conns; + instObjGroups.elementByLogicalIndex(0).connectedTo(conns, false, true); + + const auto numConnections = conns.length(); + if (numConnections == 0) { + return MObject::kNullObj; + } + for (auto i = decltype(numConnections) { 0 }; i < numConnections; ++i) { + auto sg = conns[i].node(); + if (sg.apiType() == MFn::kShadingEngine) { + return sg; + } + } + + return MObject::kNullObj; +} + +const GfRange3d& MayaHydraShapeAdapter::GetExtent() +{ + if (_extentDirty) { + _CalculateExtent(); + } + return _extent; +} + +TfToken MayaHydraShapeAdapter::GetRenderTag() const { return HdTokens->geometry; } + +void MayaHydraShapeAdapter::PopulateSelectedPaths( + const MDagPath& selectedDag, + SdfPathVector& selectedSdfPaths, + std::unordered_set& selectedMasters, + const HdSelectionSharedPtr& selection) +{ + VtIntArray indices(1); + if (IsInstanced()) { + indices[0] = selectedDag.instanceNumber(); + selection->AddInstance(HdSelection::HighlightModeSelect, _id, indices); + if (selectedMasters.find(_id) == selectedMasters.end()) { + selectedSdfPaths.push_back(_id); + selectedMasters.insert(_id); + } + } else { + selection->AddRprim(HdSelection::HighlightModeSelect, _id); + selectedSdfPaths.push_back(_id); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.h b/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.h new file mode 100644 index 0000000000..8494c8f02c --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/shapeAdapter.h @@ -0,0 +1,95 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_SHAPE_ADAPTER_H +#define MAYAHYDRALIB_SHAPE_ADAPTER_H + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraShapeAdapter is an adapter to translate from Maya shapes to hydra + * Please note that, at this time, this is not used by the hydra plug-in, we translate from a + * renderitem to hydra using the MayaHydraRenderItemAdapter class. + */ +class MayaHydraShapeAdapter : public MayaHydraDagAdapter +{ +protected: + MAYAHYDRALIB_API + MayaHydraShapeAdapter( + const SdfPath& id, + MayaHydraSceneProducer* producer, + const MDagPath& dagPath); + +public: + MAYAHYDRALIB_API + virtual ~MayaHydraShapeAdapter() = default; + + MAYAHYDRALIB_API + virtual size_t + SamplePrimvar(const TfToken& key, size_t maxSampleCount, float* times, VtValue* samples); + MAYAHYDRALIB_API + virtual HdMeshTopology GetMeshTopology() override; + MAYAHYDRALIB_API + virtual HdBasisCurvesTopology GetBasisCurvesTopology() override; + MAYAHYDRALIB_API + virtual HdDisplayStyle GetDisplayStyle() override; + MAYAHYDRALIB_API + virtual PxOsdSubdivTags GetSubdivTags(); + MAYAHYDRALIB_API + virtual HdPrimvarDescriptorVector GetPrimvarDescriptors(HdInterpolation interpolation) override + { + return {}; + } + MAYAHYDRALIB_API + virtual void MarkDirty(HdDirtyBits dirtyBits) override; + + MAYAHYDRALIB_API + virtual MObject GetMaterial(); + MAYAHYDRALIB_API + virtual bool GetDoubleSided() const override { return true; }; + + MAYAHYDRALIB_API + const GfRange3d& GetExtent(); + + MAYAHYDRALIB_API + virtual TfToken GetRenderTag() const override; + + MAYAHYDRALIB_API + virtual void PopulateSelectedPaths( + const MDagPath& selectedDag, + SdfPathVector& selectedSdfPaths, + std::unordered_set& selectedMasters, + const HdSelectionSharedPtr& selection); + +protected: + MAYAHYDRALIB_API + void _CalculateExtent(); + +private: + GfRange3d _extent; + bool _extentDirty; +}; + +using MayaHydraShapeAdapterPtr = std::shared_ptr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_SHAPE_ADAPTER_H diff --git a/lib/mayaHydra/hydraExtensions/adapters/spotLightAdapter.cpp b/lib/mayaHydra/hydraExtensions/adapters/spotLightAdapter.cpp new file mode 100644 index 0000000000..92f8b73bf4 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/spotLightAdapter.cpp @@ -0,0 +1,167 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +namespace { +void GetSpotCutoffAndSoftness(MFnSpotLight& mayaLight, float& cutoffOut, float& softnessOut) +{ + // Divided by two. + auto coneAngle = static_cast(GfRadiansToDegrees(mayaLight.coneAngle())) * 0.5f; + auto penumbraAngle = static_cast(GfRadiansToDegrees(mayaLight.penumbraAngle())); + cutoffOut = coneAngle + penumbraAngle; + softnessOut = cutoffOut == 0 ? 0 : penumbraAngle / cutoffOut; +} + +float GetSpotCutoff(MFnSpotLight& mayaLight) +{ + float cutoff; + float softness; + GetSpotCutoffAndSoftness(mayaLight, cutoff, softness); + return cutoff; +} + +float GetSpotSoftness(MFnSpotLight& mayaLight) +{ + float cutoff; + float softness; + GetSpotCutoffAndSoftness(mayaLight, cutoff, softness); + return softness; +} + +float GetSpotFalloff(MFnSpotLight& mayaLight) { return static_cast(mayaLight.dropOff()); } +} // namespace + +/** + * \brief MayaHydraSpotLightAdapter is used to handle the translation from a Maya spot light to + * hydra. + */ +class MayaHydraSpotLightAdapter : public MayaHydraLightAdapter +{ +public: + MayaHydraSpotLightAdapter(MayaHydraSceneProducer* producer, const MDagPath& dag) + : MayaHydraLightAdapter(producer, dag) + { + } + + const TfToken& LightType() const override + { + if (GetSceneProducer()->IsHdSt()) { + return HdPrimTypeTokens->simpleLight; + } else { + return HdPrimTypeTokens->sphereLight; + } + } + +protected: + void _CalculateLightParams(GlfSimpleLight& light) override + { + MStatus status; + MFnSpotLight mayaLight(GetDagPath(), &status); + if (TF_VERIFY(status)) { + light.SetHasShadow(true); + light.SetSpotCutoff(GetSpotCutoff(mayaLight)); + light.SetSpotFalloff(GetSpotFalloff(mayaLight)); + } + } + + VtValue Get(const TfToken& key) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET) + .Msg( + "Called MayaHydraSpotLightAdapter::Get(%s) - %s\n", + key.GetText(), + GetDagPath().partialPathName().asChar()); + + if (key == HdLightTokens->shadowParams) { + HdxShadowParams shadowParams; + MFnSpotLight mayaLight(GetDagPath()); + if (!GetShadowsEnabled(mayaLight)) { + shadowParams.enabled = false; + return VtValue(shadowParams); + } + + _CalculateShadowParams(mayaLight, shadowParams); + // Use the radius as the "blur" amount, for PCSS + shadowParams.blur = mayaLight.shadowRadius(); + return VtValue(shadowParams); + } + + return MayaHydraLightAdapter::Get(key); + } + + VtValue GetLightParamValue(const TfToken& paramName) override + { + TF_DEBUG(MAYAHYDRALIB_ADAPTER_GET_LIGHT_PARAM_VALUE) + .Msg( + "Called MayaHydraSpotLightAdapter::GetLightParamValue(%s) - %s\n", + paramName.GetText(), + GetDagPath().partialPathName().asChar()); + + MStatus status; + MFnSpotLight light(GetDagPath(), &status); + if (TF_VERIFY(status)) { + if (paramName == HdLightTokens->radius) { + const float radius = light.shadowRadius(); + return VtValue(radius); + } else if (paramName == UsdLuxTokens->treatAsPoint) { + const bool treatAsPoint = (light.shadowRadius() == 0.0); + return VtValue(treatAsPoint); + } else if (paramName == UsdLuxTokens->inputsShapingConeAngle) { + return VtValue(GetSpotCutoff(light)); + } else if (paramName == UsdLuxTokens->inputsShapingConeSoftness) { + return VtValue(GetSpotSoftness(light)); + } else if (paramName == UsdLuxTokens->inputsShapingFocus) { + return VtValue(GetSpotFalloff(light)); + } + } + return MayaHydraLightAdapter::GetLightParamValue(paramName); + } +}; + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraAdapterRegistry, pointLight) +{ + MayaHydraAdapterRegistry::RegisterLightAdapter( + TfToken("spotLight"), + [](MayaHydraSceneProducer* producer, const MDagPath& dag) -> MayaHydraLightAdapterPtr { + return MayaHydraLightAdapterPtr(new MayaHydraSpotLightAdapter(producer, dag)); + }); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/tokens.cpp b/lib/mayaHydra/hydraExtensions/adapters/tokens.cpp new file mode 100644 index 0000000000..9d1331b0f5 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/tokens.cpp @@ -0,0 +1,28 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// clang-format off +TF_DEFINE_PUBLIC_TOKENS( + MayaHydraAdapterTokens, + + MAYAHYDRALIB_ADAPTER_TOKENS +); +// clang-format on + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/adapters/tokens.h b/lib/mayaHydra/hydraExtensions/adapters/tokens.h new file mode 100644 index 0000000000..5d4b8457f5 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/adapters/tokens.h @@ -0,0 +1,92 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_ADAPTER_TOKENS_H +#define MAYAHYDRALIB_ADAPTER_TOKENS_H + +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraAdapterTokens contains all the hydra tokens used by this plug-in. + */ + +// clang-format off +#define MAYAHYDRALIB_ADAPTER_TOKENS \ + (roughness) \ + (clearcoat) \ + (clearcoatRoughness) \ + (emissiveColor) \ + (specular) \ + (specularColor) \ + (metallic) \ + (useSpecularWorkflow) \ + (occlusion) \ + (ior) \ + (normal) \ + (opacity) \ + (diffuse) \ + (diffuseColor) \ + (displacement) \ + (base) \ + (baseColor) \ + (emission) \ + (emissionColor) \ + (metalness) \ + (specularIOR) \ + (specularRoughness) \ + (coat) \ + (coatRoughness) \ + (transmission) \ + (lambert) \ + (blinn) \ + (phong) \ + (standardSurface) \ + (file) \ + (place2dTexture) \ + (fileTextureName) \ + (color) \ + (incandescence) \ + (out) \ + (output) \ + (st) \ + (uv) \ + (uvCoord) \ + (rgb) \ + (r) \ + (xyz) \ + (x) \ + (varname) \ + (result) \ + (eccentricity) \ + (usdPreviewSurface) \ + (pxrUsdPreviewSurface) \ + (MayaHydraLambertShader) \ + (MayaHydraPhongShader) \ + (MayaHydraBlinnShader) \ + (MayaHydraStippleShader) \ + (MayaHydraSolidColorShader) + +// clang-format on + +TF_DECLARE_PUBLIC_TOKENS(MayaHydraAdapterTokens, MAYAHYDRALIB_API, MAYAHYDRALIB_ADAPTER_TOKENS); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_ADAPTER_TOKENS_H diff --git a/lib/mayaHydra/hydraExtensions/api.h b/lib/mayaHydra/hydraExtensions/api.h new file mode 100644 index 0000000000..10f692c3be --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/api.h @@ -0,0 +1,40 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_API_H +#define MAYAHYDRALIB_API_H + +#ifdef __GNUC__ +#define MAYAHYDRALIB_API_EXPORT __attribute__((visibility("default"))) +#define MAYAHYDRALIB_API_IMPORT +#elif defined(_WIN32) || defined(_WIN64) +#define MAYAHYDRALIB_API_EXPORT __declspec(dllexport) +#define MAYAHYDRALIB_API_IMPORT __declspec(dllimport) +#else +#define MAYAHYDRALIB_API_EXPORT +#define MAYAHYDRALIB_API_IMPORT +#endif + +#if defined(MAYAHYDRALIB_STATIC) +#define MAYAHYDRALIB_API +#else +#if defined(MAYAHYDRALIB_EXPORT) +#define MAYAHYDRALIB_API MAYAHYDRALIB_API_EXPORT +#else +#define MAYAHYDRALIB_API MAYAHYDRALIB_API_IMPORT +#endif +#endif + +#endif // MAYAHYDRALIB_API_H diff --git a/lib/mayaHydra/hydraExtensions/debugCodes.cpp b/lib/mayaHydra/hydraExtensions/debugCodes.cpp new file mode 100644 index 0000000000..83cf58da73 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/debugCodes.cpp @@ -0,0 +1,29 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "debugCodes.h" + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// Some variables to enable debug printing information +TF_REGISTRY_FUNCTION(TfDebug) +{ + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_SCENE_INDEX, "Print info about the Maya Hydra scene index."); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/debugCodes.h b/lib/mayaHydra/hydraExtensions/debugCodes.h new file mode 100644 index 0000000000..01f244105c --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/debugCodes.h @@ -0,0 +1,46 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DEBUG_CODES_H +#define MAYAHYDRALIB_DEBUG_CODES_H + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// clang-format off +TF_DEBUG_CODES( + MAYAHYDRALIB_SCENE_INDEX +); +// clang-format on + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DEBUG_CODES_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/CMakeLists.txt b/lib/mayaHydra/hydraExtensions/delegates/CMakeLists.txt new file mode 100644 index 0000000000..b11a9fe287 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/CMakeLists.txt @@ -0,0 +1,42 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + delegate.cpp + delegateCtx.cpp + delegateDebugCodes.cpp + delegateRegistry.cpp + sceneDelegate.cpp + defaultLightDelegate.cpp + testDelegate.cpp +) + +set(HEADERS + delegate.h + delegateCtx.h + delegateDebugCodes.h + delegateRegistry.h + params.h + sceneDelegate.h + defaultLightDelegate.h + testDelegate.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/delegates +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/mayaHydraLib/delegates +) diff --git a/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.cpp b/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.cpp new file mode 100644 index 0000000000..f150605f76 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.cpp @@ -0,0 +1,228 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "defaultLightDelegate.h" + +#include + +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +namespace { +bool AreLightsParamsWeUseDifferent(const GlfSimpleLight& light1, const GlfSimpleLight& light2) +{ + // We only update 3 parameters in the default light : position, diffuse and specular. We don't + // use the primitive's transform. + return (light1.GetPosition() != light2.GetPosition()) + || // Position (in which we actually store a direction, updated when rotating the view for + // example) + (light1.GetDiffuse() != light2.GetDiffuse()) + || (light1.GetSpecular() != light2.GetSpecular()); +} +} // namespace + +// clang-format off +TF_DEFINE_PRIVATE_TOKENS( + _tokens, + + (DefaultMayaLight) +); +// clang-format on + +// MtohDefaultLightDelegate is a separate Hydra custom scene delegate to handle the default lighting +// from Maya. We use another Hydra custom scene delegate to handle the other parts of the Maya +// scene. See sceneDelegate.h in the mayaHydraLib project. If you want to know how to add a custom +// scene index to this plug-in, then please see the registration.cpp file in the mayaHydraLib +// project. +MtohDefaultLightDelegate::MtohDefaultLightDelegate(const InitData& initData) + : HdSceneDelegate(initData.renderIndex, initData.delegateID) + , MayaHydraDelegate(initData) + , _lightPath(initData.delegateID.AppendChild(_tokens->DefaultMayaLight)) +{ +} + +MtohDefaultLightDelegate::~MtohDefaultLightDelegate() { RemovePrim(); } + +void MtohDefaultLightDelegate::Populate() +{ + if (_isPopulated) { + return; + } + if (!_isLightingOn) { + return; + } + _isSupported = IsHdSt() ? GetRenderIndex().IsSprimTypeSupported(HdPrimTypeTokens->simpleLight) + : GetRenderIndex().IsSprimTypeSupported(HdPrimTypeTokens->distantLight); + if (ARCH_UNLIKELY(!_isSupported)) { + return; + } + if (IsHdSt()) { + GetRenderIndex().InsertSprim(HdPrimTypeTokens->simpleLight, this, _lightPath); + } else { + GetRenderIndex().InsertSprim(HdPrimTypeTokens->distantLight, this, _lightPath); + } + GetRenderIndex().GetChangeTracker().SprimInserted(_lightPath, HdLight::AllDirty); + _isPopulated = true; +} + +void MtohDefaultLightDelegate::RemovePrim() +{ + if (!_isPopulated) { + return; + } + if (ARCH_UNLIKELY(!_isSupported)) { + return; + } + if (IsHdSt()) { + GetRenderIndex().RemoveSprim(HdPrimTypeTokens->simpleLight, _lightPath); + } else { + GetRenderIndex().RemoveSprim(HdPrimTypeTokens->distantLight, _lightPath); + } + _isPopulated = false; +} + +void MtohDefaultLightDelegate::SetDefaultLight(const GlfSimpleLight& light) +{ + if (!_isPopulated) { + return; + } + if (ARCH_UNLIKELY(!_isSupported)) { + return; + } + + // We only update 3 parameters in the default light : position (in which we store a direction), + // diffuse and specular + // We don't never update the transform for the default light + const bool lightsParamsWeUseAreDifferent = AreLightsParamsWeUseDifferent(_light, light); + if (lightsParamsWeUseAreDifferent) { + // Update our light + _light.SetDiffuse(light.GetDiffuse()); + _light.SetSpecular(light.GetSpecular()); + _light.SetPosition(light.GetPosition()); + GetRenderIndex().GetChangeTracker().MarkSprimDirty(_lightPath, HdLight::DirtyParams); + } +} + +GfMatrix4d MtohDefaultLightDelegate::GetTransform(const SdfPath& id) +{ + TF_UNUSED(id); + + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_TRANSFORM) + .Msg("MtohDefaultLightDelegate::GetTransform(%s)\n", id.GetText()); + + // We have to rotate the distant to match the simple light's direction + // stored in it's position. Otherwise, the matrix needs to be an identity + // matrix. + if (!IsHdSt()) { + const auto position = _light.GetPosition(); + GfTransform transform; + transform.SetRotation( + GfRotation(GfVec3d(0.0, 0.0, -1.0), GfVec3d(-position[0], -position[1], -position[2]))); + return transform.GetMatrix(); + } + return GfMatrix4d(1.0); +} + +VtValue MtohDefaultLightDelegate::Get(const SdfPath& id, const TfToken& key) +{ + TF_UNUSED(id); + + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET) + .Msg("MtohDefaultLightDelegate::Get(%s, %s)\n", id.GetText(), key.GetText()); + + if (key == HdLightTokens->params) { + return VtValue(_light); + } else if (key == HdTokens->transform) { + return VtValue(GfMatrix4d( + 1.0)); // We don't use the transform but use the position param of the GlfsimpleLight + // Hydra might crash when this is an empty VtValue. + } else if (key == HdLightTokens->shadowCollection) { + if (!_solidPrimitivesRootPaths.empty()) { + // Exclude lines/points primitives from casting shadows by only taking the primitives + // whose root path belongs to _solidPrimitivesRootPaths + HdRprimCollection coll(HdTokens->geometry, HdReprSelector(HdReprTokens->refined)); + coll.SetRootPaths(_solidPrimitivesRootPaths); + return VtValue(coll); + } else { + HdRprimCollection coll(HdTokens->geometry, HdReprSelector(HdReprTokens->refined)); + return VtValue(coll); + } + } else if (key == HdLightTokens->shadowParams) { + HdxShadowParams shadowParams; + shadowParams.enabled = false; + return VtValue(shadowParams); + } + return {}; +} + +VtValue MtohDefaultLightDelegate::GetLightParamValue(const SdfPath& id, const TfToken& paramName) +{ + TF_UNUSED(id); + + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_LIGHT_PARAM_VALUE) + .Msg( + "MtohDefaultLightDelegate::GetLightParamValue(%s, %s)\n", + id.GetText(), + paramName.GetText()); + + if (paramName == HdLightTokens->color || paramName == HdTokens->displayColor) { + const auto diffuse = _light.GetDiffuse(); + return VtValue(GfVec3f(diffuse[0], diffuse[1], diffuse[2])); + } else if (paramName == HdLightTokens->intensity) { + return VtValue(1.0f); + } else if (paramName == HdLightTokens->diffuse) { + return VtValue(1.0f); + } else if (paramName == HdLightTokens->specular) { + return VtValue(0.0f); + } else if (paramName == HdLightTokens->exposure) { + return VtValue(0.0f); + } else if (paramName == HdLightTokens->normalize) { + return VtValue(true); + } else if (paramName == HdLightTokens->angle) { + return VtValue(0.0f); + } else if (paramName == HdLightTokens->shadowEnable) { + return VtValue(false); + } else if (paramName == HdLightTokens->shadowColor) { + return VtValue(GfVec3f(0.0f, 0.0f, 0.0f)); + } else if (paramName == HdLightTokens->enableColorTemperature) { + return VtValue(false); + } + return {}; +} + +bool MtohDefaultLightDelegate::GetVisible(const SdfPath& id) +{ + TF_UNUSED(id); + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_VISIBLE) + .Msg("MtohDefaultLightDelegate::GetVisible(%s)\n", id.GetText()); + return true; +} + +void MtohDefaultLightDelegate::SetLightingOn(bool isLightingOn) +{ + if (_isLightingOn != isLightingOn) { + _isLightingOn = isLightingOn; + + RemovePrim(); + Populate(); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.h b/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.h new file mode 100644 index 0000000000..2e154a0fa4 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/defaultLightDelegate.h @@ -0,0 +1,74 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#ifndef MTOH_DEFAULT_LIGHT_DELEGATE_H +#define MTOH_DEFAULT_LIGHT_DELEGATE_H + +#include + +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +//! \brief MtohDefaultLightDelegate is a separate Hydra custom scene delegate to handle the default +//! lighting from Maya. We use another Hydra custom scene delegate to handle the other parts of the +//! Maya scene. See sceneDelegate.h in the mayaHydraLib project. If you want to know how to add a +//! custom scene index to this plug-in, then please see the registration.cpp file in the +//! mayaHydraLib project. + +class MtohDefaultLightDelegate + : public HdSceneDelegate + , public MayaHydraDelegate +{ +public: + MtohDefaultLightDelegate(const InitData& initData); + + ~MtohDefaultLightDelegate() override; + + void Populate() override; + void SetDefaultLight(const GlfSimpleLight& light); + void SetLightingOn(bool isLightingOn); + void SetSolidPrimitivesRootPaths(const SdfPathVector& solidPrimitivesPaths) + { + _solidPrimitivesRootPaths = solidPrimitivesPaths; + } + void RemovePrim(); + +protected: + GfMatrix4d GetTransform(const SdfPath& id) override; + VtValue Get(const SdfPath& id, const TfToken& key) override; + VtValue GetLightParamValue(const SdfPath& id, const TfToken& paramName) override; + bool GetVisible(const SdfPath& id) override; + +private: + GlfSimpleLight _light; + SdfPath _lightPath; + bool _isSupported = false; + /// Is used to avoid lighting any non solid wireframe prim (such as line/points prims) + SdfPathVector _solidPrimitivesRootPaths; + bool _isPopulated = false; + bool _isLightingOn = true; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MTOH_DEFAULT_LIGHT_DELEGATE_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegate.cpp b/lib/mayaHydra/hydraExtensions/delegates/delegate.cpp new file mode 100644 index 0000000000..a9fed73957 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegate.cpp @@ -0,0 +1,47 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "delegate.h" + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_REGISTRY_FUNCTION(TfType) { TfType::Define(); } + +MayaHydraDelegate::MayaHydraDelegate(const InitData& initData) + : _mayaDelegateID(initData.delegateID) + , _name(initData.name) + , _engine(initData.engine) + , _taskController(initData.taskController) + , _isHdSt(initData.isHdSt) + , _producer(initData.producer) +{ +} + +void MayaHydraDelegate::SetParams(const MayaHydraParams& params) { _params = params; } + +void MayaHydraDelegate::SetCameraForSampling(SdfPath const& camID) +{ + _cameraPathForSampling = camID; +} + +GfInterval MayaHydraDelegate::GetCurrentTimeSamplingInterval() const +{ + return GfInterval(_params.motionSampleStart, _params.motionSampleEnd); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegate.h b/lib/mayaHydra/hydraExtensions/delegates/delegate.h new file mode 100644 index 0000000000..61b5bf66c5 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegate.h @@ -0,0 +1,204 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DELEGATE_H +#define MAYAHYDRALIB_DELEGATE_H + +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneProducer; + +/** + * \brief MayaHydraDelegate is the base class for delegate classes. + */ +class MayaHydraDelegate +{ +public: + /// Structure passed to initialize this class + struct InitData + { + inline InitData( + TfToken nameIn, + HdEngine& engineIn, + HdRenderIndex* renderIndexIn, + HdRendererPlugin* rendererPluginIn, + HdxTaskController* taskControllerIn, + const SdfPath& delegateIDIn, + bool isHdStIn, + MayaHydraSceneProducer* producerIn = nullptr) + : name(nameIn) + , engine(engineIn) + , renderIndex(renderIndexIn) + , rendererPlugin(rendererPluginIn) + , taskController(taskControllerIn) + , delegateID(delegateIDIn) + , isHdSt(isHdStIn) + , producer(producerIn) + { + } + + TfToken name; + HdEngine& engine; + HdRenderIndex* renderIndex; + HdRendererPlugin* rendererPlugin; + HdxTaskController* taskController; + SdfPath delegateID; + bool isHdSt; + MayaHydraSceneProducer* producer; + }; + + MAYAHYDRALIB_API + MayaHydraDelegate(const InitData& initData); + MAYAHYDRALIB_API + virtual ~MayaHydraDelegate() = default; + + virtual void Populate() = 0; + virtual void PreFrame(const MHWRender::MDrawContext& context) { } + virtual void PostFrame() { } + + MAYAHYDRALIB_API + virtual void SetParams(const MayaHydraParams& params); + const MayaHydraParams& GetParams() const { return _params; } + + const SdfPath& GetMayaDelegateID() { return _mayaDelegateID; } + TfToken GetName() { return _name; } + bool IsHdSt() { return _isHdSt; } + + virtual void PopulateSelectedPaths( + const MSelectionList& mayaSelection, + SdfPathVector& selectedSdfPaths, + const HdSelectionSharedPtr& selection) + { + } + + virtual bool AddPickHitToSelectionList( + const HdxPickHit& hit, + const MHWRender::MSelectionInfo& selectInfo, + MSelectionList& mayaSelection, + MPointArray& worldSpaceHitPts) + { + return false; + } + + void SetLightsEnabled(const bool enabled) { _lightsEnabled = enabled; } + bool GetLightsEnabled() { return _lightsEnabled; } + + inline HdEngine& GetEngine() { return _engine; } + inline HdxTaskController* GetTaskController() { return _taskController; } + + /// Calls that mirror UsdImagingDelegate + + /// Setup for the shutter open and close to be used for motion sampling. + MAYAHYDRALIB_API + void SetCameraForSampling(SdfPath const& id); + + /// Returns the current interval that will be used when using the + /// sample* API in the scene delegate. + MAYAHYDRALIB_API + GfInterval GetCurrentTimeSamplingInterval() const; + + /// Common function to return templated sample types + template + size_t SampleValues(size_t maxSampleCount, float* times, T* samples, Getter getValue) + { + if (ARCH_UNLIKELY(maxSampleCount == 0)) { + return 0; + } + // Fast path 1 sample at current-frame + if (maxSampleCount == 1 + || (!GetParams().motionSamplesEnabled() && GetParams().motionSampleStart == 0)) { + times[0] = 0.0f; + samples[0] = getValue(); + return 1; + } + + const GfInterval shutter = GetCurrentTimeSamplingInterval(); + // Shutter for [-1, 1] (size 2) should have a step of 2 for 2 samples, and 1 for 3 samples + // For sample size of 1 tStep is unused and we match USD and to provide t=shutterOpen + // sample. + const double tStep = maxSampleCount > 1 ? (shutter.GetSize() / (maxSampleCount - 1)) : 0; + const MTime mayaTime = MAnimControl::currentTime(); + size_t nSamples = 0; + double relTime = shutter.GetMin(); + + for (size_t i = 0; i < maxSampleCount; ++i) { + T sample; + { + MDGContextGuard guard(mayaTime + relTime); + sample = getValue(); + } + // We compare the sample to the previous in order to reduce sample count on output. + // Goal is to reduce the amount of samples/keyframes the Hydra delegate has to absorb. + if (!nSamples || sample != samples[nSamples - 1]) { + samples[nSamples] = std::move(sample); + times[nSamples] = relTime; + ++nSamples; + } + relTime += tStep; + } + return nSamples; + } + + MayaHydraSceneProducer* GetProducer() { return _producer; }; + +private: + MayaHydraParams _params; + + // Note that because there may not be a 1-to-1 relationship between + // a MayaHydraDelegate and a HdSceneDelegate, this may be different than + // "the" scene delegate id. In the case of MayaHydraSceneDelegate, + // which inherits from HdSceneDelegate, they are the same; but for, ie, + // MayaHydraALProxyDelegate, for which there are multiple HdSceneDelegates + // for each MayaHydraDelegate, the _mayaDelegateID is different from each + // HdSceneDelegate's id. + const SdfPath _mayaDelegateID; + SdfPath _cameraPathForSampling; + TfToken _name; + HdEngine& _engine; + HdxTaskController* _taskController; + bool _isHdSt = false; + bool _lightsEnabled = true; + + MayaHydraSceneProducer* _producer = nullptr; +}; + +using MayaHydraDelegatePtr = std::shared_ptr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DELEGATE_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.cpp b/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.cpp new file mode 100644 index 0000000000..48e674672a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.cpp @@ -0,0 +1,248 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#include "delegateCtx.h" + +#include +#include + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +namespace { + +static const SdfPath lightedObjectsPath = SdfPath(std::string("Lighted")); + +template SdfPath toSdfPath(const T& src); +template<> inline SdfPath toSdfPath(const MDagPath& dag) { + return DagPathToSdfPath(dag, false, false); +} +template<> inline SdfPath toSdfPath(const MRenderItem& ri) { + return RenderItemToSdfPath(ri, false); +} + +template SdfPath maybePrepend(const T& src, const SdfPath& inPath); +template<> inline SdfPath maybePrepend( + const MDagPath& , const SdfPath& inPath +) { + return inPath; +} +template<> inline SdfPath maybePrepend( + const MRenderItem& ri, const SdfPath& inPath +) { + // Prepend Maya node name, for organisation and readability. + std::string dependNodeNameString (MFnDependencyNode(ri.sourceDagPath().node()).name().asChar()); + SanitizeNameForSdfPath(dependNodeNameString); + return SdfPath(dependNodeNameString).AppendPath(inPath); +} + +///Returns false if this object should not be lighted, true if it should be lighted +template bool shouldBeLighted(const T& src); +//Template specialization for MDagPath +template<> inline bool shouldBeLighted(const MDagPath& dag) { + return (MFnDependencyNode(dag.node()).typeName().asChar() == TfToken("mesh")); +} +//Template specialization for MRenderItem +template<> inline bool shouldBeLighted(const MRenderItem& ri) { + + //Special case to recognize the Arnold skydome light + if (MayaHydraDelegateCtx::isRenderItem_aiSkyDomeLightTriangleShape(ri)){ + return false;//Don't light the sky dome light shape + } + + return (MHWRender::MGeometry::Primitive::kLines != ri.primitive() + && MHWRender::MGeometry::Primitive::kLineStrip != ri.primitive() + && MHWRender::MGeometry::Primitive::kPoints != ri.primitive()); +} + +template +SdfPath GetMayaPrimPath(const T& src) +{ + SdfPath mayaPath = toSdfPath(src); + if (mayaPath.IsEmpty() || mayaPath.IsAbsoluteRootPath()) + return {}; + + // We cannot append an absolute path (I.e : starting with "/") + if (mayaPath.IsAbsolutePath()) { + mayaPath = mayaPath.MakeRelativePath(SdfPath::AbsoluteRootPath()); + } + + mayaPath = maybePrepend(src, mayaPath); + + if (shouldBeLighted(src)) { + // Use a specific prefix when it's not an object that needs to interact with lights and shadows to be able + // We filter the objects that don't have this prefix in lights HdLightTokens->shadowCollection parameter + mayaPath = lightedObjectsPath.AppendPath(mayaPath); + } + + return mayaPath; +} + +SdfPath _GetRenderItemMayaPrimPath(const MRenderItem& ri) +{ + if (ri.InternalObjectId() == 0) + return {}; + + return GetMayaPrimPath(ri); +} + +SdfPath _GetPrimPath(const SdfPath& base, const MDagPath& dg) +{ + return base.AppendPath(GetMayaPrimPath(dg)); +} + +SdfPath _GetRenderItemPrimPath(const SdfPath& base, const MRenderItem& ri) +{ + return base.AppendPath(_GetRenderItemMayaPrimPath(ri)); +} + +SdfPath _GetRenderItemShaderPrimPath(const SdfPath& base, const MRenderItem& ri) +{ + return _GetRenderItemPrimPath(base, ri); +} + +SdfPath _GetMaterialPath(const SdfPath& base, const MObject& obj) +{ + MStatus status; + MFnDependencyNode node(obj, &status); + if (!status) { + return {}; + } + const auto* chr = node.name().asChar(); + if (chr == nullptr || chr[0] == '\0') { + return {}; + } + + std::string nodeName(chr); + SanitizeNameForSdfPath(nodeName); + return base.AppendPath(SdfPath(nodeName)); +} + +} // namespace + +// MayaHydraDelegateCtx is a set of common functions, and it is the aggregation of our +// MayaHydraDelegate base class and the hydra custom scene delegate class : HdSceneDelegate. +MayaHydraDelegateCtx::MayaHydraDelegateCtx(const InitData& initData) + : HdSceneDelegate(initData.renderIndex, initData.delegateID) + , MayaHydraDelegate(initData) + , _rprimPath(initData.delegateID.AppendPath(SdfPath(std::string("rprims")))) + , _sprimPath(initData.delegateID.AppendPath(SdfPath(std::string("sprims")))) + , _materialPath(initData.delegateID.AppendPath(SdfPath(std::string("materials")))) +{ + GetChangeTracker().AddCollection(TfToken("visible")); +} + +void MayaHydraDelegateCtx::InsertRprim( + const TfToken& typeId, + const SdfPath& id, + const SdfPath& instancerId) +{ + if (!instancerId.IsEmpty()) { + GetRenderIndex().InsertInstancer(this, instancerId); + } + GetRenderIndex().InsertRprim(typeId, this, id); +} + +void MayaHydraDelegateCtx::InsertSprim( + const TfToken& typeId, + const SdfPath& id, + HdDirtyBits initialBits) +{ + GetRenderIndex().InsertSprim(typeId, this, id); + GetChangeTracker().SprimInserted(id, initialBits); +} + +void MayaHydraDelegateCtx::RemoveRprim(const SdfPath& id) { GetRenderIndex().RemoveRprim(id); } + +void MayaHydraDelegateCtx::RemoveSprim(const TfToken& typeId, const SdfPath& id) +{ + GetRenderIndex().RemoveSprim(typeId, id); +} + +void MayaHydraDelegateCtx::RemoveInstancer(const SdfPath& id) +{ + GetRenderIndex().RemoveInstancer(id); +} + +SdfPath MayaHydraDelegateCtx::GetRprimPath() const { return _rprimPath; } + +SdfPath MayaHydraDelegateCtx::GetPrimPath(const MDagPath& dg, bool isSprim) +{ + if (isSprim) { + return _GetPrimPath(_sprimPath, dg); + } else { + return _GetPrimPath(_rprimPath, dg); + } +} + +SdfPath MayaHydraDelegateCtx::GetRenderItemPrimPath(const MRenderItem& ri) +{ + return _GetRenderItemPrimPath(_rprimPath, ri); +} + +SdfPath MayaHydraDelegateCtx::GetRenderItemShaderPrimPath(const MRenderItem& ri) +{ + return _GetRenderItemShaderPrimPath(_rprimPath, ri); +} + +SdfPath MayaHydraDelegateCtx::GetMaterialPath(const MObject& obj) +{ + return _GetMaterialPath(_materialPath, obj); +} + +SdfPath MayaHydraDelegateCtx::GetLightedPrimsRootPath() const +{ + return _rprimPath.AppendPath(lightedObjectsPath); +} + +bool MayaHydraDelegateCtx::isRenderItem_aiSkyDomeLightTriangleShape(const MRenderItem& renderItem) +{ + static const std::string _aiSkyDomeLight ("aiSkyDomeLight"); + + const auto prim = renderItem.primitive(); + MDagPath dag = renderItem.sourceDagPath(); + if( dag.isValid() && (MHWRender::MGeometry::Primitive::kTriangles == prim) && (MHWRender::MRenderItem::DecorationItem == renderItem.type()) ){ + std::string fpName = dag.fullPathName().asChar(); + if (fpName.find(_aiSkyDomeLight) != std::string::npos) { + //This render item is a aiSkyDomeLight + return true; + } + } + + return false; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.h b/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.h new file mode 100644 index 0000000000..7d8dc665fb --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateCtx.h @@ -0,0 +1,110 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DELEGATE_BASE_H +#define MAYAHYDRALIB_DELEGATE_BASE_H + +#include + +#include +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE +/** + * \brief MayaHydraDelegateCtx is a set of common functions, and it is the aggregation of our + * MayaHydraDelegate base class and the hydra custom scene delegate class : HdSceneDelegate. + */ +class MayaHydraDelegateCtx + : public HdSceneDelegate + , public MayaHydraDelegate +{ +protected: + MAYAHYDRALIB_API + MayaHydraDelegateCtx(const InitData& initData); + +public: + enum RebuildFlags : uint32_t + { + RebuildFlagPrim = 1 << 1, + RebuildFlagCallbacks = 1 << 2, + }; + + using HdSceneDelegate::GetRenderIndex; + HdChangeTracker& GetChangeTracker() { return GetRenderIndex().GetChangeTracker(); } + + MAYAHYDRALIB_API + void InsertRprim(const TfToken& typeId, const SdfPath& id, const SdfPath& instancerId = {}); + MAYAHYDRALIB_API + void InsertSprim(const TfToken& typeId, const SdfPath& id, HdDirtyBits initialBits); + MAYAHYDRALIB_API + void RemoveRprim(const SdfPath& id); + MAYAHYDRALIB_API + void RemoveSprim(const TfToken& typeId, const SdfPath& id); + MAYAHYDRALIB_API + void RemoveInstancer(const SdfPath& id); + + /** + * @brief Is used to identify a Maya RenderItem as an aiSkydomeLight triangle shape. + * + * @param[in] renderItem is the Maya RenderItem which you want to test. + * + * @return returns true if it is an aiSkydomeLight triangle shape, false if not. + */ + static bool isRenderItem_aiSkyDomeLightTriangleShape(const MRenderItem& renderItem); + + virtual void RemoveAdapter(const SdfPath& id) { } + virtual void RecreateAdapter(const SdfPath& id, const MObject& obj) { } + virtual void RecreateAdapterOnIdle(const SdfPath& id, const MObject& obj) { } + virtual void RebuildAdapterOnIdle(const SdfPath& id, uint32_t flags) { } + virtual void UpdateDisplayStatusMaterial( + MHWRender::DisplayStatus displayStatus, + const MColor& wireframecolor) + { + } + + /// \brief Notifies the scene delegate when a material tag changes. + /// + /// \param id Id of the Material that changed its tag. + virtual void MaterialTagChanged(const SdfPath& id) { } + MAYAHYDRALIB_API + SdfPath GetPrimPath(const MDagPath& dg, bool isSprim); + MAYAHYDRALIB_API + SdfPath GetRenderItemPrimPath(const MRenderItem& ri); + MAYAHYDRALIB_API + SdfPath GetRenderItemShaderPrimPath(const MRenderItem& ri); + MAYAHYDRALIB_API + SdfPath GetMaterialPath(const MObject& obj); + + MAYAHYDRALIB_API + SdfPath + GetLightedPrimsRootPath() const; /// Get the root path for lighted objects, objects that don't have this in their SdfPath are not lighted + + MAYAHYDRALIB_API + SdfPath GetRprimPath() const; + +private: + SdfPath _rprimPath; + SdfPath _sprimPath; + SdfPath _materialPath; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DELEGATE_BASE_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.cpp b/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.cpp new file mode 100644 index 0000000000..390b8e572c --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.cpp @@ -0,0 +1,125 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "delegateDebugCodes.h" + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// Some variables to enable debug printing information for our custom scene delegate + +TF_REGISTRY_FUNCTION(TfDebug) +{ + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET, "Print information about 'Get' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_CULL_STYLE, + "Print information about 'GetCullStyle' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_CURVE_TOPOLOGY, + "Print information about 'GetCurveTopology' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_DISPLAY_STYLE, + "Print information about 'GetDisplayStyle' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_DOUBLE_SIDED, + "Print information about 'GetDoubleSided' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_EXTENT, + "Print information about 'GetExtent' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_INSTANCER_ID, + "Print information about 'GetInstancerId' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_INSTANCE_INDICES, + "Print information about GetInstanceIndices calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_LIGHT_PARAM_VALUE, + "Print information about 'GetLightParamValue' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_MATERIAL_ID, + "Print information about 'GetMaterialId' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_MATERIAL_RESOURCE, + "Print information about 'GetMaterialResource' calls to the " + "delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_MESH_TOPOLOGY, + "Print information about 'GetMeshTopology' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_PRIMVAR_DESCRIPTORS, + "Print information about 'GetPrimvarDescriptors' calls to the " + "delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_RENDER_TAG, + "Print information about 'GetRenderTag' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_SUBDIV_TAGS, + "Print information about 'GetSubdivTags' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_TRANSFORM, + "Print information about 'GetTransform' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_GET_VISIBLE, + "Print information about 'GetVisible' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_INSERTDAG, + "Print information about 'InsertDag' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_IS_ENABLED, + "Print information about 'IsEnabled' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER, + "Print information when the delegate recreates adapters."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_REGISTRY, + "Print information about registration of MayaHydraDelegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_SAMPLE_PRIMVAR, + "Print information about 'SamplePrimvar' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_SAMPLE_TRANSFORM, + "Print information about 'SampleTransform' calls to the delegates."); + + TF_DEBUG_ENVIRONMENT_SYMBOL( + MAYAHYDRALIB_DELEGATE_SELECTION, + "Print information about mayaHydraLib delegate selection."); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.h b/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.h new file mode 100644 index 0000000000..ca002ad31a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateDebugCodes.h @@ -0,0 +1,58 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DELEGATE_DEBUG_CODES_H +#define MAYAHYDRALIB_DELEGATE_DEBUG_CODES_H + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +//! \brief Some variables to enable debug printing information for our custom scene delegate + +// clang-format off +TF_DEBUG_CODES( + MAYAHYDRALIB_DELEGATE_GET, + MAYAHYDRALIB_DELEGATE_GET_CULL_STYLE, + MAYAHYDRALIB_DELEGATE_GET_CURVE_TOPOLOGY, + MAYAHYDRALIB_DELEGATE_GET_DISPLAY_STYLE, + MAYAHYDRALIB_DELEGATE_GET_DOUBLE_SIDED, + MAYAHYDRALIB_DELEGATE_GET_EXTENT, + MAYAHYDRALIB_DELEGATE_GET_INSTANCER_ID, + MAYAHYDRALIB_DELEGATE_GET_INSTANCE_INDICES, + MAYAHYDRALIB_DELEGATE_GET_LIGHT_PARAM_VALUE, + MAYAHYDRALIB_DELEGATE_GET_MATERIAL_ID, + MAYAHYDRALIB_DELEGATE_GET_MATERIAL_RESOURCE, + MAYAHYDRALIB_DELEGATE_GET_MESH_TOPOLOGY, + MAYAHYDRALIB_DELEGATE_GET_PRIMVAR_DESCRIPTORS, + MAYAHYDRALIB_DELEGATE_GET_RENDER_TAG, + MAYAHYDRALIB_DELEGATE_GET_SUBDIV_TAGS, + MAYAHYDRALIB_DELEGATE_GET_TRANSFORM, + MAYAHYDRALIB_DELEGATE_GET_VISIBLE, + MAYAHYDRALIB_DELEGATE_INSERTDAG, + MAYAHYDRALIB_DELEGATE_IS_ENABLED, + MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER, + MAYAHYDRALIB_DELEGATE_REGISTRY, + MAYAHYDRALIB_DELEGATE_SAMPLE_PRIMVAR, + MAYAHYDRALIB_DELEGATE_SAMPLE_TRANSFORM, + MAYAHYDRALIB_DELEGATE_SELECTION, + MAYAHYDRALIB_DELEGATE_PRINT_LIGHTS_PARAMETERS_VALUES +); +// clang-format on + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DELEGATE_DEBUG_CODES_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.cpp b/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.cpp new file mode 100644 index 0000000000..2fabc3a581 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.cpp @@ -0,0 +1,123 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "delegateRegistry.h" + +#include + +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_INSTANTIATE_SINGLETON(MayaHydraDelegateRegistry); + +// MayaHydraDelegateRegistry is a singleton class to handle hydra delegates + +void MayaHydraDelegateRegistry::RegisterDelegate(const TfToken& name, DelegateCreator creator) +{ + auto& instance = GetInstance(); + for (auto it : instance._delegates) { + if (name == std::get<0>(it)) { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_REGISTRY) + .Msg( + "MayaHydraDelegateRegistry::RegisterDelegate(%s) - existing " + "delegate\n", + name.GetText()); + return; + } + } + + TF_DEBUG(MAYAHYDRALIB_DELEGATE_REGISTRY) + .Msg("MayaHydraDelegateRegistry::RegisterDelegate(%s) - new delegate\n", name.GetText()); + instance._delegates.emplace_back(name, creator); +} + +std::vector MayaHydraDelegateRegistry::GetDelegateNames() +{ + LoadAllDelegates(); + const auto& instance = GetInstance(); + std::vector ret; + ret.reserve(instance._delegates.size()); + for (auto it : instance._delegates) { + ret.push_back(std::get<0>(it)); + } + return ret; +} + +std::vector +MayaHydraDelegateRegistry::GetDelegateCreators() +{ + LoadAllDelegates(); + const auto& instance = GetInstance(); + std::vector ret; + ret.reserve(instance._delegates.size()); + for (auto it : instance._delegates) { + ret.push_back(std::get<1>(it)); + } + return ret; +} + +void MayaHydraDelegateRegistry::SignalDelegatesChanged() +{ + for (const auto& s : GetInstance()._signals) { + s(); + } +} + +void MayaHydraDelegateRegistry::LoadAllDelegates() +{ + static std::once_flag loadAllOnce; + std::call_once(loadAllOnce, _LoadAllDelegates); +} + +void MayaHydraDelegateRegistry::InstallDelegatesChangedSignal(DelegatesChangedSignal signal) +{ + GetInstance()._signals.emplace_back(signal); +} + +void MayaHydraDelegateRegistry::_LoadAllDelegates() +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_REGISTRY) + .Msg("MayaHydraDelegateRegistry::_LoadAllDelegates()\n"); + + TfRegistryManager::GetInstance().SubscribeTo(); + + const TfType& delegateType = TfType::Find(); + if (delegateType.IsUnknown()) { + TF_CODING_ERROR("Could not find MayaHydraDelegate type"); + return; + } + + std::set delegateTypes; + delegateType.GetAllDerivedTypes(&delegateTypes); + + PlugRegistry& plugReg = PlugRegistry::GetInstance(); + + for (auto& subType : delegateTypes) { + const PlugPluginPtr pluginForType = plugReg.GetPluginForType(subType); + if (!pluginForType) { + TF_CODING_ERROR("Could not find plugin for '%s'", subType.GetTypeName().c_str()); + return; + } + pluginForType->Load(); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.h b/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.h new file mode 100644 index 0000000000..177647b72d --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/delegateRegistry.h @@ -0,0 +1,76 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_DELEGATE_REGISTRY_H +#define MAYAHYDRALIB_DELEGATE_REGISTRY_H + +#include + +#include +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraDelegateRegistry is a singleton class to handle hydra delegates + */ +class MayaHydraDelegateRegistry : public TfSingleton +{ + friend class TfSingleton; + MAYAHYDRALIB_API + MayaHydraDelegateRegistry() = default; + +public: + /// function creates and returns a pointer to a MayaHydraDelegate - may return + /// a nullptr indicate failure, or that the delegate is currently disabled + using DelegateCreator = std::function; + + MAYAHYDRALIB_API + static void RegisterDelegate(const TfToken& name, DelegateCreator creator); + MAYAHYDRALIB_API + static std::vector GetDelegateNames(); + MAYAHYDRALIB_API + static std::vector GetDelegateCreators(); + + /// Signal that some delegate types are now either valid or invalid. + /// ie, say some delegate type is only useful / works when a certain maya + /// plug-in is loaded - you would call this every time that plugin was loaded + /// or unloaded. + MAYAHYDRALIB_API + static void SignalDelegatesChanged(); + + /// Find all MayaHydraDelegate plug-ins, and load them all + MAYAHYDRALIB_API + static void LoadAllDelegates(); + + using DelegatesChangedSignal = std::function; + + MAYAHYDRALIB_API + static void InstallDelegatesChangedSignal(DelegatesChangedSignal signal); + +private: + static void _LoadAllDelegates(); + + std::vector> _delegates; + std::vector _signals; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_DELEGATE_REGISTRY_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/params.h b/lib/mayaHydra/hydraExtensions/delegates/params.h new file mode 100644 index 0000000000..bff5a017a1 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/params.h @@ -0,0 +1,47 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_PARAMS_H +#define MAYAHYDRALIB_PARAMS_H + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraParams are the global parameters for this plug-in. + * Please note that we add to these parameters the parameters read from the chosen hydra render + * delegate and expose them in our UI with the MEL function mtohRenderOverride_AddAttribute see + * renderGlobals.cpp. + */ +struct MayaHydraParams +{ + bool proxyPurpose = true; + bool renderPurpose = false; + bool guidePurpose = false; + int textureMemoryPerTexture = 4 * 1024 * 1024; + int maximumShadowMapResolution = 2048; + float motionSampleStart = 0; + float motionSampleEnd = 0; + bool displaySmoothMeshes = true; + + bool motionSamplesEnabled() const { return motionSampleStart != 0 || motionSampleEnd != 0; } +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_PARAMS_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.cpp b/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.cpp new file mode 100644 index 0000000000..4f13ba0e7c --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.cpp @@ -0,0 +1,1612 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#include "sceneDelegate.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +int _profilerCategory = MProfiler::addCategory( + "MayaHydraSceneDelegate (mayaHydra)", + "Events for MayaHydraSceneDelegate"); + +#if PXR_VERSION < 2308 +#error USD version v0.23.08+ required +#endif + +#if MAYA_API_VERSION < 20240000 +#error Maya API version 2024+ required +#endif + +namespace { + +// Pixar macros require Pixar namespace. +PXR_NAMESPACE_USING_DIRECTIVE + +TF_DEFINE_ENV_SETTING(MAYA_HYDRA_USE_MESH_ADAPTER, false, + "Use mesh adapter instead of MRenderItem for Maya meshes."); + +bool useMeshAdapter() { + static bool uma = TfGetEnvSetting(MAYA_HYDRA_USE_MESH_ADAPTER); + return uma; +} + +bool filterMesh(const MRenderItem& ri) { + return useMeshAdapter() ? + // Filter our mesh render items, and let the mesh adapter handle Maya + // meshes. The MRenderItem::name() for meshes is "StandardShadedItem", + // their MRenderItem::type() is InternalMaterialItem, but + // this type can also be used for other purposes, e.g. face groups, so + // using the name is more appropriate. + (ri.name() == "StandardShadedItem") : false; +} + +} + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +SdfPath MayaHydraSceneDelegate::_fallbackMaterial; +SdfPath MayaHydraSceneDelegate::_mayaDefaultMaterialPath; // Common to all scene delegates +VtValue MayaHydraSceneDelegate::_mayaDefaultMaterial; + +namespace { + +void _onDagNodeAdded(MObject& obj, void* clientData) +{ + reinterpret_cast(clientData)->OnDagNodeAdded(obj); +} + +void _onDagNodeRemoved(MObject& obj, void* clientData) +{ + reinterpret_cast(clientData)->OnDagNodeRemoved(obj); +} + +const MString defaultLightSet("defaultLightSet"); + +void _connectionChanged(MPlug& srcPlug, MPlug& destPlug, bool made, void* clientData) +{ + TF_UNUSED(made); + const auto srcObj = srcPlug.node(); + if (!srcObj.hasFn(MFn::kTransform)) { + return; + } + const auto destObj = destPlug.node(); + if (!destObj.hasFn(MFn::kSet)) { + return; + } + if (srcPlug != MayaAttrs::dagNode::instObjGroups) { + return; + } + MStatus status; + MFnDependencyNode destNode(destObj, &status); + if (ARCH_UNLIKELY(!status)) { + return; + } + if (destNode.name() != defaultLightSet) { + return; + } + auto* delegate = reinterpret_cast(clientData); + MDagPath dag; + status = MDagPath::getAPathTo(srcObj, dag); + if (ARCH_UNLIKELY(!status)) { + return; + } + unsigned int shapesBelow = 0; + dag.numberOfShapesDirectlyBelow(shapesBelow); + for (auto i = decltype(shapesBelow) { 0 }; i < shapesBelow; ++i) { + auto dagCopy = dag; + dagCopy.extendToShapeDirectlyBelow(i); + delegate->UpdateLightVisibility(dagCopy); + } +} + +template inline bool _FindAdapter(const SdfPath&, F) { return false; } + +template +inline bool _FindAdapter(const SdfPath& id, F f, const M0& m0, const M&... m) +{ + auto* adapterPtr = TfMapLookupPtr(m0, id); + if (adapterPtr == nullptr) { + return _FindAdapter(id, f, m...); + } else { + f(static_cast(adapterPtr->get())); + return true; + } +} + +template inline bool _RemoveAdapter(const SdfPath&, F) { return false; } + +template +inline bool _RemoveAdapter(const SdfPath& id, F f, M0& m0, M&... m) +{ + auto* adapterPtr = TfMapLookupPtr(m0, id); + if (adapterPtr == nullptr) { + return _RemoveAdapter(id, f, m...); + } else { + f(static_cast(adapterPtr->get())); + m0.erase(id); + return true; + } +} + +template inline R _GetDefaultValue() { return {}; } + +// This will be nicer to use with automatic parameter deduction for lambdas in +// C++14. +template inline R _GetValue(const SdfPath&, F) +{ + return _GetDefaultValue(); +} + +template +inline R _GetValue(const SdfPath& id, F f, const M0& m0, const M&... m) +{ + auto* adapterPtr = TfMapLookupPtr(m0, id); + if (adapterPtr == nullptr) { + return _GetValue(id, f, m...); + } else { + return f(static_cast(adapterPtr->get())); + } +} + +template inline void _MapAdapter(F) +{ + // Do nothing. +} + +template +inline void _MapAdapter(F f, const M0& m0, const M&... m) +{ + for (auto& it : m0) { + f(static_cast(it.second.get())); + } + _MapAdapter(f, m...); +} + +} // namespace + +// clang-format off +TF_DEFINE_PRIVATE_TOKENS( + _tokens, + + (MayaHydraSceneDelegate) + ((MayaDefaultMaterial, "__maya_default_material__")) + (diffuseColor) + (emissiveColor) + (roughness) + (MayaHydraMeshPoints) + (constantLighting) +); +// clang-format on + +TF_REGISTRY_FUNCTION(TfType) +{ + TfType::Define>(); +} + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraDelegateRegistry, MayaHydraSceneDelegate) +{ + MayaHydraDelegateRegistry::RegisterDelegate( + _tokens->MayaHydraSceneDelegate, + [](const MayaHydraDelegate::InitData& initData) -> MayaHydraDelegatePtr { + return std::static_pointer_cast( + std::make_shared(initData)); + }); +} + +// MayaHydraSceneDelegate is an Hydra custom scene delegate used to translate from a Maya scene to +// hydra. If you want to know how to add a custom scene index to this plug-in, then please see the +// registration.cpp file. +MayaHydraSceneDelegate::MayaHydraSceneDelegate(const InitData& initData) + : MayaHydraDelegateCtx(initData) +{ + // TfDebug::Enable(MAYAHYDRALIB_DELEGATE_GET_MATERIAL_ID);//Enable this line to print to the + // output window all SceneDelegate::GetMaterialID(...) calls + // TfDebug::Enable(MAYAHYDRALIB_DELEGATE_GET); //Enable this line to print to the output window + // all SceneDelegate::Get(...) calls + + // Enable the following line to print to the output window the materials parameters type and + // values when there is a change in one of them. + // TfDebug::Enable(MAYAHYDRALIB_ADAPTER_MATERIALS_PRINT_PARAMETERS_VALUES); + + // Enable the following line to print to the output window the lights parameters type and + // values. TfDebug::Enable(MAYAHYDRALIB_DELEGATE_PRINT_LIGHTS_PARAMETERS_VALUES); + + static std::once_flag once; + std::call_once(once, []() { + _mayaDefaultMaterialPath = SdfPath::AbsoluteRootPath().AppendChild( + _tokens->MayaDefaultMaterial); // Is an absolute path, not linked to a scene delegate + _mayaDefaultMaterial = MayaHydraSceneDelegate::CreateMayaDefaultMaterial(); + _fallbackMaterial = SdfPath::EmptyPath(); // Empty path for hydra fallback material + }); +} + +MayaHydraSceneDelegate::~MayaHydraSceneDelegate() +{ + for (auto callback : _callbacks) { + MMessage::removeCallback(callback); + } + _MapAdapter( + [](MayaHydraAdapter* a) { a->RemoveCallbacks(); }, + _renderItemsAdapters, + _shapeAdapters, + _lightAdapters, + _materialAdapters); +} + +VtValue MayaHydraSceneDelegate::CreateMayaDefaultMaterial() +{ + static const MColor kDefaultGrayColor = MColor(0.5f, 0.5f, 0.5f) * 0.8f; + + HdMaterialNetworkMap networkMap; + HdMaterialNetwork network; + HdMaterialNode node; + node.identifier = UsdImagingTokens->UsdPreviewSurface; + node.path = _mayaDefaultMaterialPath; + node.parameters.insert( + { _tokens->diffuseColor, + VtValue(GfVec3f(kDefaultGrayColor[0], kDefaultGrayColor[1], kDefaultGrayColor[2])) }); + network.nodes.push_back(std::move(node)); + networkMap.map.insert({ HdMaterialTerminalTokens->surface, std::move(network) }); + networkMap.terminals.push_back(_mayaDefaultMaterialPath); + return VtValue(networkMap); +} + +void MayaHydraSceneDelegate::_AddRenderItem(const MayaHydraRenderItemAdapterPtr& ria) +{ + const SdfPath& primPath = ria->GetID(); + _renderItemsAdaptersFast.insert({ ria->GetFastID(), ria }); + _renderItemsAdapters.insert({ primPath, ria }); +} + +void MayaHydraSceneDelegate::_RemoveRenderItem(const MayaHydraRenderItemAdapterPtr& ria) +{ + const SdfPath& primPath = ria->GetID(); + _renderItemsAdaptersFast.erase(ria->GetFastID()); + _renderItemsAdapters.erase(primPath); +} + +void MayaHydraSceneDelegate::HandleCompleteViewportScene( + const MDataServerOperation::MViewportScene& scene, + MFrameContext::DisplayStyle displayStyle) +{ + const bool playbackRunning = MAnimControl::isPlaying(); + + if (_isPlaybackRunning != playbackRunning) { + // The value has changed, we are calling SetPlaybackChanged so that every render item that + // has its visibility dependent on the playback should dirty its hydra visibility flag so + // its gets recomputed. + for (auto it = _renderItemsAdapters.begin(); it != _renderItemsAdapters.end(); it++) { + it->second->SetPlaybackChanged(); + } + + _isPlaybackRunning = playbackRunning; + } + + // First loop to get rid of removed items + constexpr int kInvalidId = 0; + for (size_t i = 0; i < scene.mRemovalCount; i++) { + int fastId = scene.mRemovals[i]; + if (fastId == kInvalidId) + continue; + MayaHydraRenderItemAdapterPtr ria = nullptr; + if (_GetRenderItem(fastId, ria)) { + _RemoveRenderItem(ria); + } + assert(ria != nullptr); + } + + // My version, does minimal update + // This loop could, in theory, be parallelized. Unclear how large the gains would be, but maybe + // nothing to lose unless there is some internal contention in USD. + for (size_t i = 0; i < scene.mCount; i++) { + auto flags = scene.mFlags[i]; + if (flags == 0) { + continue; + } + + auto& ri = *scene.mItems[i]; + + // Meshes can optionally be handled by the mesh adapter, rather than by + // render items. + if (filterMesh(ri)) { + continue; + } + + int fastId = ri.InternalObjectId(); + MayaHydraRenderItemAdapterPtr ria = nullptr; + if (!_GetRenderItem(fastId, ria)) { + const SdfPath slowId = GetRenderItemPrimPath(ri); + if (slowId.IsEmpty()){ + continue; + } + // MAYA-128021: We do not currently support maya instances. + MDagPath dagPath(ri.sourceDagPath()); + ria = std::make_shared(dagPath, slowId, fastId, GetProducer(), ri); + + //Update the render item adapter if this render item is an aiSkydomeLight shape + ria->SetIsRenderITemAnaiSkydomeLightTriangleShape(isRenderItem_aiSkyDomeLightTriangleShape(ri)); + + _AddRenderItem(ria); + } + + SdfPath material; + MObject shadingEngineNode; + if (!_GetRenderItemMaterial(ri, material, shadingEngineNode)) { + if (material != kInvalidMaterial) { + _CreateMaterial(material, shadingEngineNode); + } + } + + if (flags & MDataServerOperation::MViewportScene::MVS_changedEffect) { + ria->SetMaterial(material); + } + + MColor wireframeColor; + MHWRender::DisplayStatus displayStatus = MHWRender::kNoStatus; + + MDagPath dagPath = ri.sourceDagPath(); + if (dagPath.isValid()) { + wireframeColor = MGeometryUtilities::wireframeColor( + dagPath); // This is a color managed VP2 color, it will need to be unmanaged at some + // point + displayStatus = MGeometryUtilities::displayStatus(dagPath); + } + + const MayaHydraRenderItemAdapter::UpdateFromDeltaData data( + ri, flags, wireframeColor, displayStatus); + ria->UpdateFromDelta(data); + if (flags & MDataServerOperation::MViewportScene::MVS_changedMatrix) { + ria->UpdateTransform(ri); + } + } +} + +void MayaHydraSceneDelegate::Populate() +{ + MayaHydraAdapterRegistry::LoadAllPlugin(); + auto& renderIndex = GetRenderIndex(); + MStatus status; + MItDag dagIt(MItDag::kDepthFirst); + dagIt.traverseUnderWorld(true); + if (useMeshAdapter()) { + for (; !dagIt.isDone(); dagIt.next()) { + MDagPath path; + dagIt.getPath(path); + InsertDag(path); + } + } + else { + for (; !dagIt.isDone(); dagIt.next()) { + MObject node = dagIt.currentItem(&status); + if (status != MS::kSuccess) + continue; + OnDagNodeAdded(node); + } + } + + auto id = MDGMessage::addNodeAddedCallback(_onDagNodeAdded, "dagNode", this, &status); + if (status) { + _callbacks.push_back(id); + } + id = MDGMessage::addNodeRemovedCallback(_onDagNodeRemoved, "dagNode", this, &status); + if (status) { + _callbacks.push_back(id); + } + id = MDGMessage::addConnectionCallback(_connectionChanged, this, &status); + if (status) { + _callbacks.push_back(id); + } + + // Adding materials sprim to the render index. + if (renderIndex.IsSprimTypeSupported(HdPrimTypeTokens->material)) { + renderIndex.InsertSprim(HdPrimTypeTokens->material, this, _mayaDefaultMaterialPath); + } +} + +MayaHydraSceneDelegate::LightDagPathMap +MayaHydraSceneDelegate::_GetActiveLightPaths() const +{ + LightDagPathMap activeLightPaths; + activeLightPaths.reserve(_lightAdapters.size()); + // By the time this function is called _lightAdapters should already have been populated + // with both Maya and Arnold light adapters. The adapters contain the DagPath information + // we store it here in unordered_map for fast retrieval + for(const auto& entry : _lightAdapters) + { + const auto& dagpath = entry.second->GetDagPath(); + activeLightPaths.emplace(dagpath.fullPathName().asChar(), + dagpath); + } + return activeLightPaths; +} + +// +void MayaHydraSceneDelegate::PreFrame(const MHWRender::MDrawContext& context) +{ + bool useDefaultMaterial + = (context.getDisplayStyle() & MHWRender::MFrameContext::kDefaultMaterial); + if (useDefaultMaterial != _useDefaultMaterial) { + _useDefaultMaterial = useDefaultMaterial; + if (useMeshAdapter()) { + for (const auto& shape : _shapeAdapters) + shape.second->MarkDirty(HdChangeTracker::DirtyMaterialId); + } + } + + const bool xRayEnabled = (context.getDisplayStyle() & MHWRender::MFrameContext::kXray); + if (xRayEnabled != _xRayEnabled) { + _xRayEnabled = xRayEnabled; + for (auto& matAdapter : _materialAdapters) + matAdapter.second->EnableXRayShadingMode(_xRayEnabled); + } + + if (!_materialTagsChanged.empty()) { + if (IsHdSt()) { + for (const auto& id : _materialTagsChanged) { + if (_GetValue( + id, + [](MayaHydraMaterialAdapter* a) { return a->UpdateMaterialTag(); }, + _materialAdapters)) { + auto& renderIndex = GetRenderIndex(); + for (const auto& rprimId : renderIndex.GetRprimIds()) { + const auto* rprim = renderIndex.GetRprim(rprimId); + if (rprim != nullptr && rprim->GetMaterialId() == id) { + RebuildAdapterOnIdle( + rprim->GetId(), MayaHydraDelegateCtx::RebuildFlagPrim); + } + } + } + } + } + _materialTagsChanged.clear(); + } + + if (!_lightsToAdd.empty()) { + for (auto& lightToAdd : _lightsToAdd) { + MDagPath dag; + MStatus status = MDagPath::getAPathTo(lightToAdd.first, dag); + if (!status) { + return; + } + CreateLightAdapter(dag); + } + _lightsToAdd.clear(); + } + + if (useMeshAdapter() && !_addedNodes.empty()) { + for (const auto& obj : _addedNodes) { + if (obj.isNull()) { + continue; + } + MDagPath dag; + MStatus status = MDagPath::getAPathTo(obj, dag); + if (!status) { + return; + } + // We need to check if there is an instanced shape below this dag + // and insert it as well, because they won't be inserted. + if (dag.hasFn(MFn::kTransform)) { + const auto childCount = dag.childCount(); + for (auto child = decltype(childCount) { 0 }; child < childCount; ++child) { + auto dagCopy = dag; + dagCopy.push(dag.child(child)); + if (dagCopy.isInstanced() && dagCopy.instanceNumber() > 0) { + AddNewInstance(dagCopy); + } + } + } else { + InsertDag(dag); + } + } + _addedNodes.clear(); + } + + // We don't need to rebuild something that's already being recreated. + // Since we have a few elements, linear search over vectors is going to + // be okay. + if (!_adaptersToRecreate.empty()) { + for (const auto& it : _adaptersToRecreate) { + RecreateAdapter(std::get<0>(it), std::get<1>(it)); + for (auto itr = _adaptersToRebuild.begin(); itr != _adaptersToRebuild.end(); ++itr) { + if (std::get<0>(it) == std::get<0>(*itr)) { + _adaptersToRebuild.erase(itr); + break; + } + } + } + _adaptersToRecreate.clear(); + } + if (!_adaptersToRebuild.empty()) { + for (const auto& it : _adaptersToRebuild) { + _FindAdapter( + std::get<0>(it), + [&](MayaHydraAdapter* a) { + if (std::get<1>(it) & MayaHydraDelegateCtx::RebuildFlagCallbacks) { + a->RemoveCallbacks(); + a->CreateCallbacks(); + } + if (std::get<1>(it) & MayaHydraDelegateCtx::RebuildFlagPrim) { + a->RemovePrim(); + a->Populate(); + } + }, + _shapeAdapters, + _lightAdapters, + _materialAdapters); + } + _adaptersToRebuild.clear(); + } + if (!IsHdSt()) { + return; + } + + LightDagPathMap activeLightPaths = _GetActiveLightPaths(); + constexpr auto considerAllSceneLights = MHWRender::MDrawContext::kFilteredIgnoreLightLimit; + MStatus status; + const auto numLights = context.numberOfActiveLights(considerAllSceneLights, &status); + + if ((!status || numLights == 0) && (0 == activeLightPaths.size())) { + _MapAdapter( + [](MayaHydraLightAdapter* a) { a->SetLightingOn(false); }, + _lightAdapters); // Turn off all lights + return; + } + + MIntArray intVals; + MMatrix matrixVal; + for (auto i = decltype(numLights) { 0 }; i < numLights; ++i) { + auto* lightParam = context.getLightParameterInformation(i, considerAllSceneLights); + if (lightParam == nullptr) { + continue; + } + const auto lightPath = lightParam->lightPath(); + if (!lightPath.isValid()) { + continue; + } + if (IsUfeItemFromMayaUsd(lightPath)) { + // If this is a UFE light created by maya-usd, it will have already added it to Hydra + continue; + } + + // we do a fast look up here for any new lights that may have been added + auto found = activeLightPaths.find(lightPath.fullPathName().asChar()); + if (found == activeLightPaths.end()) + activeLightPaths.emplace(lightPath.fullPathName().asChar(), + lightPath); + + if (!lightParam->getParameter(MHWRender::MLightParameterInformation::kShadowOn, intVals) + || intVals.length() < 1 || intVals[0] != 1) { + continue; + } + + if (lightParam->getParameter( + MHWRender::MLightParameterInformation::kShadowViewProj, matrixVal)) { + _FindAdapter( + GetPrimPath(lightPath, true), + [&matrixVal](MayaHydraLightAdapter* a) { + a->SetShadowProjectionMatrix(GetGfMatrixFromMaya(matrixVal)); + }, + _lightAdapters); + } + } + + // Turn on active lights, turn off non-active lights, and add non-created active lights. + _MapAdapter( + [&](MayaHydraLightAdapter* a) { + auto lgtAdapter = activeLightPaths.find(a->GetDagPath().fullPathName().asChar()); + if (lgtAdapter != activeLightPaths.end()) { + a->SetLightingOn(true); + activeLightPaths.erase(lgtAdapter); + } else { + a->SetLightingOn(false); + } + }, + _lightAdapters); + for(const auto& entry : activeLightPaths) { + CreateLightAdapter(entry.second); + } +} + +void MayaHydraSceneDelegate::RemoveAdapter(const SdfPath& id) +{ + if (!_RemoveAdapter( + id, + [](MayaHydraAdapter* a) { + a->RemoveCallbacks(); + a->RemovePrim(); + }, + _renderItemsAdapters, + _shapeAdapters, + _lightAdapters, + _materialAdapters)) { + TF_WARN( + "MayaHydraSceneDelegate::RemoveAdapter(%s) -- Adapter does not exists", id.GetText()); + } +} + +void MayaHydraSceneDelegate::RecreateAdapterOnIdle(const SdfPath& id, const MObject& obj) +{ + // We expect this to be a small number of objects, so using a simple linear + // search and a vector is generally a good choice. + for (auto& it : _adaptersToRecreate) { + if (std::get<0>(it) == id) { + std::get<1>(it) = obj; + return; + } + } + _adaptersToRecreate.emplace_back(id, obj); +} + +void MayaHydraSceneDelegate::MaterialTagChanged(const SdfPath& id) +{ + if (std::find(_materialTagsChanged.begin(), _materialTagsChanged.end(), id) + == _materialTagsChanged.end()) { + _materialTagsChanged.push_back(id); + } +} + +void MayaHydraSceneDelegate::RebuildAdapterOnIdle(const SdfPath& id, uint32_t flags) +{ + // We expect this to be a small number of objects, so using a simple linear + // search and a vector is generally a good choice. + for (auto& it : _adaptersToRebuild) { + if (std::get<0>(it) == id) { + std::get<1>(it) |= flags; + return; + } + } + _adaptersToRebuild.emplace_back(id, flags); +} + +void MayaHydraSceneDelegate::RecreateAdapter(const SdfPath& id, const MObject& obj) +{ + if (_RemoveAdapter( + id, + [](MayaHydraAdapter* a) { + a->RemoveCallbacks(); + a->RemovePrim(); + }, + _lightAdapters)) { + if (MObjectHandle(obj).isValid()) { + OnDagNodeAdded(obj); + } else { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER) + .Msg( + "Light prim (%s) not re-created because node no " + "longer valid\n", + id.GetText()); + } + return; + } + + if (useMeshAdapter() && _RemoveAdapter( + id, + [](MayaHydraAdapter* a) { + a->RemoveCallbacks(); + a->RemovePrim(); + }, + _shapeAdapters)) { + MFnDagNode dgNode(obj); + MDagPath path; + dgNode.getPath(path); + if (path.isValid() && MObjectHandle(obj).isValid()) { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER) + .Msg( + "Shape prim (%s) re-created for dag path (%s)\n", + id.GetText(), + path.fullPathName().asChar()); + InsertDag(path); + } else { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER) + .Msg( + "Shape prim (%s) not re-created because node no " + "longer valid\n", + id.GetText()); + } + return; + } + + if (_RemoveAdapter( + id, + [](MayaHydraMaterialAdapter* a) { + a->RemoveCallbacks(); + a->RemovePrim(); + }, + _materialAdapters)) { + auto& renderIndex = GetRenderIndex(); + auto& changeTracker = renderIndex.GetChangeTracker(); + for (const auto& rprimId : renderIndex.GetRprimIds()) { + const auto* rprim = renderIndex.GetRprim(rprimId); + if (rprim != nullptr && rprim->GetMaterialId() == id) { + changeTracker.MarkRprimDirty(rprimId, HdChangeTracker::DirtyMaterialId); + } + } + if (MObjectHandle(obj).isValid()) { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER) + .Msg( + "Material prim (%s) re-created for node (%s)\n", + id.GetText(), + MFnDependencyNode(obj).name().asChar()); + _CreateMaterial(GetMaterialPath(obj), obj); + } else { + TF_DEBUG(MAYAHYDRALIB_DELEGATE_RECREATE_ADAPTER) + .Msg( + "Material prim (%s) not re-created because node no " + "longer valid\n", + id.GetText()); + } + + } else { + TF_WARN( + "MayaHydraSceneDelegate::RecreateAdapterOnIdle(%s) -- Adapter does " + "not exists", + id.GetText()); + } +} + +MayaHydraLightAdapterPtr MayaHydraSceneDelegate::GetLightAdapter(const SdfPath& id) +{ + auto iter = _lightAdapters.find(id); + return iter == _lightAdapters.end() ? nullptr : iter->second; +} + +MayaHydraMaterialAdapterPtr MayaHydraSceneDelegate::GetMaterialAdapter(const SdfPath& id) +{ + auto iter = _materialAdapters.find(id); + return iter == _materialAdapters.end() ? nullptr : iter->second; +} + +template +AdapterPtr MayaHydraSceneDelegate::_CreateAdapter( + const MDagPath& dag, + const std::function& adapterCreator, + Map& adapterMap, + bool isSprim) +{ + // Filter for whether we should even attempt to create the adapter + + if (!adapterCreator) { + return {}; + } + + if (IsUfeItemFromMayaUsd(dag)) { + // UFE items that have a Hydra representation will be added to Hydra by maya-usd + return {}; + } + + // Attempt to create the adapter + + TF_DEBUG(MAYAHYDRALIB_DELEGATE_INSERTDAG) + .Msg( + "MayaHydraSceneDelegate::_CreateAdapter::" + "found %s: %s\n", + MFnDependencyNode(dag.node()).typeName().asChar(), + dag.fullPathName().asChar()); + + const auto id = GetPrimPath(dag, isSprim); + if (TfMapLookupPtr(adapterMap, id) != nullptr) { + return {}; + } + auto adapter = adapterCreator(GetProducer(), dag); + if (adapter == nullptr || !adapter->IsSupported()) { + return {}; + } + adapter->Populate(); + adapter->CreateCallbacks(); + adapterMap.insert({ id, adapter }); + return adapter; +} + +MayaHydraLightAdapterPtr MayaHydraSceneDelegate::CreateLightAdapter(const MDagPath& dagPath) +{ + auto lightCreatorFunc = MayaHydraAdapterRegistry::GetLightAdapterCreator(dagPath); + return _CreateAdapter(dagPath, lightCreatorFunc, _lightAdapters, true); +} + +MayaHydraCameraAdapterPtr MayaHydraSceneDelegate::CreateCameraAdapter(const MDagPath& dagPath) +{ + auto cameraCreatorFunc = MayaHydraAdapterRegistry::GetCameraAdapterCreator(dagPath); + return _CreateAdapter(dagPath, cameraCreatorFunc, _cameraAdapters, true); +} + +MayaHydraShapeAdapterPtr MayaHydraSceneDelegate::CreateShapeAdapter(const MDagPath& dagPath) { + auto shapeCreatorFunc = MayaHydraAdapterRegistry::GetShapeAdapterCreator(dagPath); + return _CreateAdapter(dagPath, shapeCreatorFunc, _shapeAdapters); +} + +namespace { +bool GetShadingEngineNode(const MRenderItem& ri, MObject& shadingEngineNode) +{ + MDagPath dagPath = ri.sourceDagPath(); + if (dagPath.isValid()) { + MFnDagNode dagNode(dagPath.node()); + MObjectArray sets, comps; + dagNode.getConnectedSetsAndMembers(dagPath.instanceNumber(), sets, comps, true); + assert(sets.length() == comps.length()); + for (uint32_t i = 0; i < sets.length(); ++i) { + const MObject& object = sets[i]; + if (object.apiType() == MFn::kShadingEngine) { + // To support per-face shading, find the shading node matched with the render item + const MObject& comp = comps[i]; + MObject shadingComp = ri.shadingComponent(); + if (shadingComp.isNull() || comp.isNull() + || MFnComponent(comp).isEqual(shadingComp)) { + shadingEngineNode = object; + return true; + } + } + } + } + return false; +} +} // namespace + +bool MayaHydraSceneDelegate::_GetRenderItemMaterial( + const MRenderItem& ri, + SdfPath& material, + MObject& shadingEngineNode) +{ + if (MHWRender::MGeometry::Primitive::kLines == ri.primitive() + || MHWRender::MGeometry::Primitive::kLineStrip == ri.primitive()) { + material = _fallbackMaterial; // Use fallbackMaterial + constantLighting + displayColor + return true; + } + + if (GetShadingEngineNode(ri, shadingEngineNode)) + // Else try to find associated material node if this is a material shader. + // NOTE: The existing maya material support in hydra expects a shading engine node + { + material = GetMaterialPath(shadingEngineNode); + if (TfMapLookupPtr(_materialAdapters, material) != nullptr) { + return true; + } + } + + return false; +} + +// Analogous to MayaHydraSceneDelegate::InsertDag +bool MayaHydraSceneDelegate::_GetRenderItem(int fastId, MayaHydraRenderItemAdapterPtr& ria) +{ + // Using SdfPath as the hash table key is extremely slow. The cost appears to be GetPrimPath, + // which would depend on MdagPath, which is a wrapper on TdagPath. TdagPath is a very slow + // class and best to avoid in any performance- critical area. Simply workaround for the + // prototype is an additional lookup index based on InternalObjectID. Long term goal would be + // that the plug-in rarely, if ever, deals with TdagPath. + MayaHydraRenderItemAdapterPtr* result = TfMapLookupPtr(_renderItemsAdaptersFast, fastId); + + if (result != nullptr) { + // adapter already exists, return it + ria = *result; + return true; + } + + return false; +} + +void MayaHydraSceneDelegate::OnDagNodeAdded(const MObject& obj) +{ + if (obj.isNull()) + return; + + if (IsUfeItemFromMayaUsd(obj)) { + // UFE items that have a Hydra representation will be added to Hydra by maya-usd + return; + } + + // When not using the mesh adapter we care only about lights for this + // callback. It is used to create a LightAdapter when adding a new light + // in the scene for Hydra rendering. + if (auto lightFn = MayaHydraAdapterRegistry::GetLightAdapterCreator(obj)) { + _lightsToAdd.push_back({ obj, lightFn }); + } + else if (useMeshAdapter()) { + _addedNodes.push_back(obj); + } +} + +void MayaHydraSceneDelegate::OnDagNodeRemoved(const MObject& obj) +{ + const auto it + = std::remove_if(_lightsToAdd.begin(), _lightsToAdd.end(), [&obj](const auto& item) { + return item.first == obj; + }); + + if (it != _lightsToAdd.end()) { + _lightsToAdd.erase(it, _lightsToAdd.end()); + } + else if (useMeshAdapter()) { + const auto it = std::remove_if(_addedNodes.begin(), _addedNodes.end(), [&obj](const auto& item) { return item == obj; }); + + if (it != _addedNodes.end()) { + _addedNodes.erase(it, _addedNodes.end()); + } + } +} + +void MayaHydraSceneDelegate::InsertDag(const MDagPath& dag) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_INSERTDAG) + .Msg( + "MayaHydraSceneDelegate::InsertDag::" + "GetLightsEnabled()=%i\n", + GetLightsEnabled()); + // We don't care about transforms. + if (dag.hasFn(MFn::kTransform)) { + return; + } + + MFnDagNode dagNode(dag); + if (dagNode.isIntermediateObject()) { + return; + } + + if (IsUfeItemFromMayaUsd(dag)) { + // UFE items that have a Hydra representation will be added to Hydra by maya-usd + return; + } + + // Custom lights don't have MFn::kLight. + if (GetLightsEnabled()) { + if (CreateLightAdapter(dag)) + return; + } + if (CreateCameraAdapter(dag)) { + return; + } + // We are inserting a single prim and + // instancer for every instanced mesh. + if (dag.isInstanced() && dag.instanceNumber() > 0) { + return; + } + + auto adapter = CreateShapeAdapter(dag); + if (adapter) { + auto material = adapter->GetMaterial(); + if (material != MObject::kNullObj) { + const auto materialId = GetMaterialPath(material); + if (TfMapLookupPtr(_materialAdapters, materialId) == nullptr) { + _CreateMaterial(materialId, material); + } + } + } +} + +void MayaHydraSceneDelegate::UpdateLightVisibility(const MDagPath& dag) +{ + const auto id = GetPrimPath(dag, true); + _FindAdapter( + id, + [](MayaHydraLightAdapter* a) { + if (a->UpdateVisibility()) { + a->RemovePrim(); + a->Populate(); + a->InvalidateTransform(); + } + }, + _lightAdapters); +} + +// +void MayaHydraSceneDelegate::AddNewInstance(const MDagPath& dag) +{ + MDagPathArray dags; + MDagPath::getAllPathsTo(dag.node(), dags); + const auto dagsLength = dags.length(); + if (dagsLength == 0) { + return; + } + const auto masterDag = dags[0]; + const auto id = GetPrimPath(masterDag, false); + std::shared_ptr masterAdapter; + if (!TfMapLookup(_shapeAdapters, id, &masterAdapter) || masterAdapter == nullptr) { + return; + } + // If dags is 1, we have to recreate the adapter. + if (dags.length() == 1 || !masterAdapter->IsInstanced()) { + RecreateAdapterOnIdle(id, masterDag.node()); + } else { + // If dags is more than one, trigger rebuilding callbacks next call and + // mark dirty. + RebuildAdapterOnIdle(id, MayaHydraDelegateCtx::RebuildFlagCallbacks); + masterAdapter->MarkDirty( + HdChangeTracker::DirtyInstancer | HdChangeTracker::DirtyInstanceIndex + | HdChangeTracker::DirtyPrimvar); + } +} + +void MayaHydraSceneDelegate::SetParams(const MayaHydraParams& params) +{ + const auto& oldParams = GetParams(); + if (oldParams.displaySmoothMeshes != params.displaySmoothMeshes) { + // I couldn't find any other way to turn this on / off. + // I can't convert HdRprim to HdMesh easily and no simple way + // to get the type of the HdRprim from the render index. + // If we want to allow creating multiple rprims and returning an id + // to a subtree, we need to use the HasType function and the mark dirty + // from each adapter. + _MapAdapter( + [](MayaHydraRenderItemAdapter* a) { + if (a->HasType(HdPrimTypeTokens->mesh) || a->HasType(HdPrimTypeTokens->basisCurves) + || a->HasType(HdPrimTypeTokens->points)) { + a->MarkDirty(HdChangeTracker::DirtyTopology); + } + }, + _renderItemsAdapters); + _MapAdapter( + [](MayaHydraDagAdapter* a) { + if (a->HasType(HdPrimTypeTokens->mesh)) { + a->MarkDirty(HdChangeTracker::DirtyTopology); + } + }, + _shapeAdapters); + } + if (oldParams.motionSampleStart != params.motionSampleStart + || oldParams.motionSampleEnd != params.motionSampleEnd) { + _MapAdapter( + [](MayaHydraRenderItemAdapter* a) { + if (a->HasType(HdPrimTypeTokens->mesh) || a->HasType(HdPrimTypeTokens->basisCurves) + || a->HasType(HdPrimTypeTokens->points)) { + a->InvalidateTransform(); + a->MarkDirty(HdChangeTracker::DirtyPoints | HdChangeTracker::DirtyTransform); + } + }, + _renderItemsAdapters); + _MapAdapter( + [](MayaHydraDagAdapter* a) { + if (a->HasType(HdPrimTypeTokens->mesh)) { + a->MarkDirty(HdChangeTracker::DirtyPoints); + } else if (a->HasType(HdPrimTypeTokens->camera)) { + a->MarkDirty(HdCamera::DirtyParams); + } + a->InvalidateTransform(); + a->MarkDirty(HdChangeTracker::DirtyTransform); + }, + _shapeAdapters, + _lightAdapters, + _cameraAdapters); + } + // We need to trigger rebuilding shaders. + if (oldParams.textureMemoryPerTexture != params.textureMemoryPerTexture) { + _MapAdapter( + [](MayaHydraMaterialAdapter* a) { a->MarkDirty(HdMaterial::AllDirty); }, + _materialAdapters); + } + if (oldParams.maximumShadowMapResolution != params.maximumShadowMapResolution) { + _MapAdapter( + [](MayaHydraLightAdapter* a) { a->MarkDirty(HdLight::AllDirty); }, _lightAdapters); + } + MayaHydraDelegate::SetParams(params); +} + +//! \brief Try to obtain maya object corresponding to HdxPickHit and add it to a maya selection +//! list \return whether the conversion was a success +bool MayaHydraSceneDelegate::AddPickHitToSelectionList( + const HdxPickHit& hit, + const MHWRender::MSelectionInfo& selectInfo, + MSelectionList& selectionList, + MPointArray& worldSpaceHitPts) +{ + SdfPath hitId = hit.objectId; + // validate that hit is indeed a maya item. Alternatively, the rprim hit could be an rprim + // defined by a scene index such as maya usd. + if (hitId.HasPrefix(GetRprimPath())) { + _FindAdapter( + hitId, + [&selectionList, &worldSpaceHitPts, &hit](MayaHydraRenderItemAdapter* a) { + // prepare the selection path of the hit item, the transform path is expected if available + const auto& itemPath = a->GetDagPath(); + MDagPath selectPath; + if (MS::kSuccess != MDagPath::getAPathTo(itemPath.transform(), selectPath)) { + selectPath = itemPath; + } + selectionList.add(selectPath); + worldSpaceHitPts.append( + hit.worldSpaceHitPoint[0], + hit.worldSpaceHitPoint[1], + hit.worldSpaceHitPoint[2]); + }, + _renderItemsAdapters); + return true; + } + + return false; +} + +HdMeshTopology MayaHydraSceneDelegate::GetMeshTopology(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_MESH_TOPOLOGY) + .Msg("MayaHydraSceneDelegate::GetMeshTopology(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraAdapter* a) -> HdMeshTopology { return a->GetMeshTopology(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +HdBasisCurvesTopology MayaHydraSceneDelegate::GetBasisCurvesTopology(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_CURVE_TOPOLOGY) + .Msg("MayaHydraSceneDelegate::GetBasisCurvesTopology(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraAdapter* a) -> HdBasisCurvesTopology { return a->GetBasisCurvesTopology(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +PxOsdSubdivTags MayaHydraSceneDelegate::GetSubdivTags(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_SUBDIV_TAGS) + .Msg("MayaHydraSceneDelegate::GetSubdivTags(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraShapeAdapter* a) -> PxOsdSubdivTags { return a->GetSubdivTags(); }, + _shapeAdapters); +} + +GfRange3d MayaHydraSceneDelegate::GetExtent(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_EXTENT) + .Msg("MayaHydraSceneDelegate::GetExtent(%s)\n", id.GetText()); + return _GetValue( + id, [](MayaHydraShapeAdapter* a) -> GfRange3d { return a->GetExtent(); }, _shapeAdapters); +} + +GfMatrix4d MayaHydraSceneDelegate::GetTransform(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_TRANSFORM) + .Msg("MayaHydraSceneDelegate::GetTransform(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraAdapter* a) -> GfMatrix4d { return a->GetTransform(); }, + _shapeAdapters, + _renderItemsAdapters, + _cameraAdapters, + _lightAdapters); +} + +size_t MayaHydraSceneDelegate::SampleTransform( + const SdfPath& id, + size_t maxSampleCount, + float* times, + GfMatrix4d* samples) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_SAMPLE_TRANSFORM) + .Msg( + "MayaHydraSceneDelegate::SampleTransform(%s, %u)\n", + id.GetText(), + static_cast(maxSampleCount)); + return _GetValue( + id, + [maxSampleCount, times, samples](MayaHydraDagAdapter* a) -> size_t { + return a->SampleTransform(maxSampleCount, times, samples); + }, + _shapeAdapters, + _cameraAdapters, + _lightAdapters); +} + +bool MayaHydraSceneDelegate::IsEnabled(const TfToken& option) const +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_IS_ENABLED) + .Msg("MayaHydraSceneDelegate::IsEnabled(%s)\n", option.GetText()); + // Maya scene can't be accessed on multiple threads, + // so I don't think this is safe to enable. + if (option == HdOptionTokens->parallelRprimSync) { + return false; + } + + TF_WARN("MayaHydraSceneDelegate::IsEnabled(%s) -- Unsupported option.\n", option.GetText()); + return false; +} + +VtValue MayaHydraSceneDelegate::Get(const SdfPath& id, const TfToken& key) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET) + .Msg("MayaHydraSceneDelegate::Get(%s, %s)\n", id.GetText(), key.GetText()); + + if (useMeshAdapter() && id.IsPropertyPath()) { + return _GetValue( + id.GetPrimPath(), + [&key](MayaHydraDagAdapter* a) -> VtValue { return a->GetInstancePrimvar(key); }, + _shapeAdapters); + } + + return _GetValue( + id, + [&key](MayaHydraAdapter* a) -> VtValue { return a->Get(key); }, + _shapeAdapters, + _renderItemsAdapters, + _cameraAdapters, + _lightAdapters, + _materialAdapters); +} + +size_t MayaHydraSceneDelegate::SamplePrimvar( + const SdfPath& id, + const TfToken& key, + size_t maxSampleCount, + float* times, + VtValue* samples) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_SAMPLE_PRIMVAR) + .Msg( + "MayaHydraSceneDelegate::SamplePrimvar(%s, %s, %u)\n", + id.GetText(), + key.GetText(), + static_cast(maxSampleCount)); + + if (!useMeshAdapter()) { + return HdSceneDelegate::SamplePrimvar(id, key, maxSampleCount, times, samples); + } + + if (maxSampleCount < 1) { + return 0; + } + if (id.IsPropertyPath()) { + times[0] = 0.0f; + samples[0] = _GetValue( + id.GetPrimPath(), + [&key](MayaHydraDagAdapter* a) -> VtValue { return a->GetInstancePrimvar(key); }, + _shapeAdapters); + return 1; + } + + return _GetValue( + id, + [&key, maxSampleCount, times, samples](MayaHydraShapeAdapter* a) -> size_t { + return a->SamplePrimvar(key, maxSampleCount, times, samples); + }, + _shapeAdapters); +} + +TfToken MayaHydraSceneDelegate::GetRenderTag(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_RENDER_TAG) + .Msg("MayaHydraSceneDelegate::GetRenderTag(%s)\n", id.GetText()); + return _GetValue( + id.GetPrimPath(), + [](MayaHydraAdapter* a) -> TfToken { return a->GetRenderTag(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +HdPrimvarDescriptorVector +MayaHydraSceneDelegate::GetPrimvarDescriptors(const SdfPath& id, HdInterpolation interpolation) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_PRIMVAR_DESCRIPTORS) + .Msg( + "MayaHydraSceneDelegate::GetPrimvarDescriptors(%s, %i)\n", id.GetText(), interpolation); + + if (useMeshAdapter() && id.IsPropertyPath()) { + return _GetValue( + id.GetPrimPath(), + [&interpolation](MayaHydraDagAdapter* a) -> HdPrimvarDescriptorVector { + return a->GetInstancePrimvarDescriptors(interpolation); + }, + _shapeAdapters); + } + + return _GetValue( + id, + [&interpolation](MayaHydraAdapter* a) -> HdPrimvarDescriptorVector { + return a->GetPrimvarDescriptors(interpolation); + }, + _shapeAdapters, + _renderItemsAdapters); +} + +VtValue MayaHydraSceneDelegate::GetLightParamValue(const SdfPath& id, const TfToken& paramName) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_LIGHT_PARAM_VALUE) + .Msg( + "MayaHydraSceneDelegate::GetLightParamValue(%s, %s)\n", + id.GetText(), + paramName.GetText()); + + const VtValue val = _GetValue( + id, + [¶mName](MayaHydraLightAdapter* a) -> VtValue { + return a->GetLightParamValue(paramName); + }, + _lightAdapters); + + if (TfDebug::IsEnabled(MAYAHYDRALIB_DELEGATE_PRINT_LIGHTS_PARAMETERS_VALUES)) { + // Print the lights parameters to the output window + std::string valueAsString = ConvertVtValueToString(val); + cout << "Light : " << id.GetText() << " Parameter : " << paramName.GetText() + << " Value : " << valueAsString << endl; + } + + return val; +} + +VtValue +MayaHydraSceneDelegate::GetCameraParamValue(const SdfPath& cameraId, const TfToken& paramName) +{ + return _GetValue( + cameraId, + [¶mName](MayaHydraCameraAdapter* a) -> VtValue { + return a->GetCameraParamValue(paramName); + }, + _cameraAdapters); +} + +VtIntArray +MayaHydraSceneDelegate::GetInstanceIndices(const SdfPath& instancerId, const SdfPath& prototypeId) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_INSTANCE_INDICES) + .Msg( + "MayaHydraSceneDelegate::GetInstanceIndices(%s, %s)\n", + instancerId.GetText(), + prototypeId.GetText()); + return _GetValue( + instancerId.GetPrimPath(), + [&prototypeId](MayaHydraDagAdapter* a) -> VtIntArray { + return a->GetInstanceIndices(prototypeId); + }, + _shapeAdapters); +} + +SdfPathVector MayaHydraSceneDelegate::GetInstancerPrototypes(SdfPath const& instancerId) +{ + return { instancerId.GetPrimPath() }; +} + +SdfPath MayaHydraSceneDelegate::GetInstancerId(const SdfPath& primId) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_INSTANCER_ID) + .Msg("MayaHydraSceneDelegate::GetInstancerId(%s)\n", primId.GetText()); + // Instancers don't have any instancers yet. + if (primId.IsPropertyPath()) { + return SdfPath(); + } + return _GetValue( + primId, + [](MayaHydraDagAdapter* a) -> SdfPath { return a->GetInstancerID(); }, + _shapeAdapters); +} + +GfMatrix4d MayaHydraSceneDelegate::GetInstancerTransform(SdfPath const& instancerId) +{ + return GfMatrix4d(1.0); +} + +SdfPath MayaHydraSceneDelegate::GetScenePrimPath( + const SdfPath& rprimPath, + int instanceIndex, + HdInstancerContext* instancerContext) +{ + return rprimPath; +} + +bool MayaHydraSceneDelegate::GetVisible(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_VISIBLE) + .Msg("MayaHydraSceneDelegate::GetVisible(%s)\n", id.GetText()); + + return _GetValue( + id, + [](MayaHydraAdapter* a) -> bool { return a->GetVisible(); }, + _shapeAdapters, + _renderItemsAdapters, + _lightAdapters); +} + +bool MayaHydraSceneDelegate::GetDoubleSided(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_DOUBLE_SIDED) + .Msg("MayaHydraSceneDelegate::GetDoubleSided(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraAdapter* a) -> bool { return a->GetDoubleSided(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +HdCullStyle MayaHydraSceneDelegate::GetCullStyle(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_CULL_STYLE) + .Msg("MayaHydraSceneDelegate::GetCullStyle(%s)\n", id.GetText()); + + return _GetValue( + id, + [](MayaHydraAdapter* a) -> HdCullStyle { return a->GetCullStyle(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +HdDisplayStyle MayaHydraSceneDelegate::GetDisplayStyle(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_DISPLAY_STYLE) + .Msg("MayaHydraSceneDelegate::GetDisplayStyle(%s)\n", id.GetText()); + return _GetValue( + id, + [](MayaHydraAdapter* a) -> HdDisplayStyle { return a->GetDisplayStyle(); }, + _shapeAdapters, + _renderItemsAdapters); +} + +SdfPath MayaHydraSceneDelegate::GetMaterialId(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_MATERIAL_ID) + .Msg("MayaHydraSceneDelegate::GetMaterialId(%s)\n", id.GetText()); + + if (_useDefaultMaterial) { + return _mayaDefaultMaterialPath; + } + + auto result = TfMapLookupPtr(_renderItemsAdapters, id); + if (result != nullptr) { + auto& renderItemAdapter = *result; + + // Check if this render item is a wireframe primitive + if (MHWRender::MGeometry::Primitive::kLines == renderItemAdapter->GetPrimitive() + || MHWRender::MGeometry::Primitive::kLineStrip == renderItemAdapter->GetPrimitive()) { + return _fallbackMaterial; + } + + auto& material = renderItemAdapter->GetMaterial(); + + if (material == kInvalidMaterial) { + return _fallbackMaterial; + } + + if (TfMapLookupPtr(_materialAdapters, material) != nullptr) { + return material; + } + } + + if (useMeshAdapter()) { + auto shapeAdapter = TfMapLookupPtr(_shapeAdapters, id); + if (shapeAdapter == nullptr) { + return _fallbackMaterial; + } + auto material = shapeAdapter->get()->GetMaterial(); + if (material == MObject::kNullObj) { + return _fallbackMaterial; + } + auto materialId = GetMaterialPath(material); + if (TfMapLookupPtr(_materialAdapters, materialId) != nullptr) { + return materialId; + } + + return _CreateMaterial(materialId, material) ? materialId : _fallbackMaterial; + } + + return _fallbackMaterial; +} + +VtValue MayaHydraSceneDelegate::GetMaterialResource(const SdfPath& id) +{ + TF_DEBUG(MAYAHYDRALIB_DELEGATE_GET_MATERIAL_RESOURCE) + .Msg("MayaHydraSceneDelegate::GetMaterialResource(%s)\n", id.GetText()); + + if (id == _mayaDefaultMaterialPath) { + return _mayaDefaultMaterial; + } + + if (id == _fallbackMaterial) { + return MayaHydraMaterialAdapter::GetPreviewMaterialResource(id); + } + + auto ret = _GetValue( + id, + [](MayaHydraMaterialAdapter* a) -> VtValue { return a->GetMaterialResource(); }, + _materialAdapters); + return ret.IsEmpty() ? MayaHydraMaterialAdapter::GetPreviewMaterialResource(id) : ret; +} + +bool MayaHydraSceneDelegate::_CreateMaterial(const SdfPath& id, const MObject& obj) +{ + TF_DEBUG(MAYAHYDRALIB_ADAPTER_MATERIALS) + .Msg("MayaHydraSceneDelegate::_CreateMaterial(%s)\n", id.GetText()); + + auto materialCreator = MayaHydraAdapterRegistry::GetMaterialAdapterCreator(obj); + if (materialCreator == nullptr) { + return false; + } + auto materialAdapter = materialCreator(id, GetProducer(), obj); + if (materialAdapter == nullptr || !materialAdapter->IsSupported()) { + return false; + } + + if (_xRayEnabled) { + materialAdapter->EnableXRayShadingMode(_xRayEnabled); // Enable XRay shading mode + } + materialAdapter->Populate(); + materialAdapter->CreateCallbacks(); + _materialAdapters.emplace(id, std::move(materialAdapter)); + return true; +} + +SdfPath MayaHydraSceneDelegate::SetCameraViewport(const MDagPath& camPath, const GfVec4d& viewport) +{ + const SdfPath camID = GetPrimPath(camPath, true); + auto&& cameraAdapter = TfMapLookupPtr(_cameraAdapters, camID); + if (cameraAdapter) { + (*cameraAdapter)->SetViewport(viewport); + return camID; + } + return {}; +} + +VtValue MayaHydraSceneDelegate::GetShadingStyle(SdfPath const& id) +{ + if (auto&& ri = TfMapLookupPtr(_renderItemsAdapters, id)) { + auto primitive = (*ri)->GetPrimitive(); + if (MHWRender::MGeometry::Primitive::kLines == primitive + || MHWRender::MGeometry::Primitive::kLineStrip == primitive) { + return VtValue( + _tokens + ->constantLighting); // Use fallbackMaterial + constantLighting + displayColor + } + } + return MayaHydraDelegateCtx::GetShadingStyle(id); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.h b/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.h new file mode 100644 index 0000000000..838827eaab --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/sceneDelegate.h @@ -0,0 +1,311 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +#ifndef MAYAHYDRALIB_SCENE_DELEGATE_H +#define MAYAHYDRALIB_SCENE_DELEGATE_H + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +/* + * Notes. + * + * To remove the need of casting between different adapter types or + * making the base adapter class too heavy I decided to use 3 different set + * or map types. This adds a bit of extra code to the RemoveAdapter function + * but simplifies the rest of the functions significantly (and no downcasting!). + * + * All this would be probably way nicer / easier with C++14 and the polymorphic + * lambdas. + * + * This also optimizes other things, like it's easier to separate functionality + * that only affects shapes, lights or materials. + */ + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief Hydra custom scene delegate. + * + * MayaHydraSceneDelegate is a Hydra custom scene delegate used to translate + * from a Maya scene to Hydra. + * + * If you want to know how to add a custom scene index to this plug-in, then please see the + * registration.cpp file. + */ + +class MayaHydraSceneDelegate : public MayaHydraDelegateCtx +{ +public: + template using AdapterMap = std::unordered_map; + + MAYAHYDRALIB_API + MayaHydraSceneDelegate(const InitData& initData); + + MAYAHYDRALIB_API + ~MayaHydraSceneDelegate() override; + + MAYAHYDRALIB_API + void Populate() override; + + MAYAHYDRALIB_API + void PreFrame(const MHWRender::MDrawContext& context) override; + + MAYAHYDRALIB_API + void RemoveAdapter(const SdfPath& id) override; + + MAYAHYDRALIB_API + void RecreateAdapter(const SdfPath& id, const MObject& obj) override; + + MAYAHYDRALIB_API + void RecreateAdapterOnIdle(const SdfPath& id, const MObject& obj) override; + + MAYAHYDRALIB_API + void RebuildAdapterOnIdle(const SdfPath& id, uint32_t flags) override; + + /// \brief Notifies the scene delegate when a material tag changes. + /// + /// This function is only affects the render index when its using HdSt. + /// HdSt requires rebuilding the shapes whenever the tags affecting + /// translucency change. + /// + /// \param id Id of the Material that changed its tag. + MAYAHYDRALIB_API + void MaterialTagChanged(const SdfPath& id) override; + + MAYAHYDRALIB_API + MayaHydraLightAdapterPtr GetLightAdapter(const SdfPath& id); + + MAYAHYDRALIB_API + MayaHydraMaterialAdapterPtr GetMaterialAdapter(const SdfPath& id); + + MAYAHYDRALIB_API + void InsertDag(const MDagPath& dag); + + void OnDagNodeAdded(const MObject& obj); + + void OnDagNodeRemoved(const MObject& obj); + + MAYAHYDRALIB_API + void UpdateLightVisibility(const MDagPath& dag); + + MAYAHYDRALIB_API + void AddNewInstance(const MDagPath& dag); + + MAYAHYDRALIB_API + void SetParams(const MayaHydraParams& params) override; + + MAYAHYDRALIB_API + SdfPath SetCameraViewport(const MDagPath& camPath, const GfVec4d& viewport); + + MAYAHYDRALIB_API + void HandleCompleteViewportScene( + const MDataServerOperation::MViewportScene& scene, + MFrameContext::DisplayStyle ds); + + MAYAHYDRALIB_API + bool AddPickHitToSelectionList( + const HdxPickHit& hit, + const MHWRender::MSelectionInfo& selectInfo, + MSelectionList& selectionList, + MPointArray& worldSpaceHitPts) override; + + bool GetPlaybackRunning() const { return _isPlaybackRunning; } + +protected: + + MAYAHYDRALIB_API + HdMeshTopology GetMeshTopology(const SdfPath& id) override; + + MAYAHYDRALIB_API + HdBasisCurvesTopology GetBasisCurvesTopology(const SdfPath& id) override; + + MAYAHYDRALIB_API + PxOsdSubdivTags GetSubdivTags(const SdfPath& id) override; + + MAYAHYDRALIB_API + GfRange3d GetExtent(const SdfPath& id) override; + + MAYAHYDRALIB_API + GfMatrix4d GetTransform(const SdfPath& id) override; + + MAYAHYDRALIB_API + size_t + SampleTransform(const SdfPath& id, size_t maxSampleCount, float* times, GfMatrix4d* samples) + override; + + MAYAHYDRALIB_API + bool GetVisible(const SdfPath& id) override; + + MAYAHYDRALIB_API + bool IsEnabled(const TfToken& option) const override; + + MAYAHYDRALIB_API + bool GetDoubleSided(const SdfPath& id) override; + + MAYAHYDRALIB_API + HdCullStyle GetCullStyle(const SdfPath& id) override; + + MAYAHYDRALIB_API + VtValue GetShadingStyle(const SdfPath& id) override; + + MAYAHYDRALIB_API + HdDisplayStyle GetDisplayStyle(const SdfPath& id) override; + // TfToken GetReprName(const SdfPath& id) override; + + MAYAHYDRALIB_API + VtValue Get(const SdfPath& id, const TfToken& key) override; + + MAYAHYDRALIB_API + size_t SamplePrimvar( + const SdfPath& id, + const TfToken& key, + size_t maxSampleCount, + float* times, + VtValue* samples) override; + + MAYAHYDRALIB_API + TfToken GetRenderTag(SdfPath const& id) override; + + MAYAHYDRALIB_API + HdPrimvarDescriptorVector + GetPrimvarDescriptors(const SdfPath& id, HdInterpolation interpolation) override; + + MAYAHYDRALIB_API + VtValue GetLightParamValue(const SdfPath& id, const TfToken& paramName) override; + + MAYAHYDRALIB_API + VtValue GetCameraParamValue(const SdfPath& cameraId, const TfToken& paramName) override; + + MAYAHYDRALIB_API + VtIntArray GetInstanceIndices(const SdfPath& instancerId, const SdfPath& prototypeId) override; + + MAYAHYDRALIB_API + SdfPathVector GetInstancerPrototypes(SdfPath const& instancerId) override; + + MAYAHYDRALIB_API + SdfPath GetInstancerId(const SdfPath& primId) override; + + MAYAHYDRALIB_API + GfMatrix4d GetInstancerTransform(SdfPath const& instancerId) override; + + MAYAHYDRALIB_API + SdfPath GetScenePrimPath( + const SdfPath& rprimPath, + int instanceIndex, + HdInstancerContext* instancerContext) override; + + MAYAHYDRALIB_API + SdfPath GetMaterialId(const SdfPath& id) override; + + MAYAHYDRALIB_API + VtValue GetMaterialResource(const SdfPath& id) override; + +private: + template + AdapterPtr _CreateAdapter( + const MDagPath& dag, + const std::function& adapterCreator, + Map& adapterMap, + bool isSprim = false); + + MayaHydraLightAdapterPtr CreateLightAdapter(const MDagPath& dagPath); + MayaHydraCameraAdapterPtr CreateCameraAdapter(const MDagPath& dagPath); + MayaHydraShapeAdapterPtr CreateShapeAdapter(const MDagPath& dagPath); + + MAYAHYDRALIB_API + bool _GetRenderItem(int fastId, MayaHydraRenderItemAdapterPtr& adapter); + + + using LightDagPathMap = std::unordered_map; + LightDagPathMap _GetActiveLightPaths() const; + + MAYAHYDRALIB_API + void _AddRenderItem(const MayaHydraRenderItemAdapterPtr& ria); + + MAYAHYDRALIB_API + void _RemoveRenderItem(const MayaHydraRenderItemAdapterPtr& ria); + + MAYAHYDRALIB_API + bool + _GetRenderItemMaterial(const MRenderItem& ri, SdfPath& material, MObject& shadingEngineNode); + + static VtValue CreateMayaDefaultMaterial(); + + bool _CreateMaterial(const SdfPath& id, const MObject& obj); + + /// \brief Unordered Map storing the shape adapters. + AdapterMap _shapeAdapters; + + /// \brief Unordered Map storing the render item adapters. + AdapterMap _renderItemsAdapters; + std::unordered_map _renderItemsAdaptersFast; + + /// \brief Unordered Map storing the light adapters. + AdapterMap _lightAdapters; + /// \brief Unordered Map storing the camera adapters. + AdapterMap _cameraAdapters; + /// \brief Unordered Map storing the material adapters. + AdapterMap _materialAdapters; + std::vector _callbacks; + std::vector> _adaptersToRecreate; + std::vector> _adaptersToRebuild; + // Nodes accumulated during _onDagNodeAdded() callback. + std::vector _addedNodes; + + using LightAdapterCreator + = std::function; + std::vector> _lightsToAdd; + + std::vector _materialTagsChanged; + + /// _fallbackMaterial is an SdfPath used when there is no material assigned to a Maya object + static SdfPath _fallbackMaterial; + /// _mayaDefaultMaterialPath is common to all scene delegates, it's the SdfPath of + /// _mayaDefaultMaterial + static SdfPath _mayaDefaultMaterialPath; + /// _mayaDefaultMaterial is an hydra material used to override all materials from the scene when + /// _useDefaultMaterial is true + static VtValue _mayaDefaultMaterial; + + bool _useDefaultMaterial = false; + bool _xRayEnabled = false; + bool _isPlaybackRunning = false; +}; + +typedef std::shared_ptr MayaSceneDelegateSharedPtr; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_SCENE_DELEGATE_H diff --git a/lib/mayaHydra/hydraExtensions/delegates/testDelegate.cpp b/lib/mayaHydra/hydraExtensions/delegates/testDelegate.cpp new file mode 100644 index 0000000000..527043fd64 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/testDelegate.cpp @@ -0,0 +1,64 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "testDelegate.h" + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_DEFINE_ENV_SETTING( + MAYAHYDRALIB_TEST_DELEGATE_FILE, + "", + "Path for MayaHydraTestDelegate to load"); + +// clang-format off +TF_DEFINE_PRIVATE_TOKENS( + _tokens, + + (MayaHydraTestDelegate) +); +// clang-format on + +TF_REGISTRY_FUNCTION_WITH_TAG(MayaHydraDelegateRegistry, MayaHydraTestDelegate) +{ + if (!TfGetEnvSetting(MAYAHYDRALIB_TEST_DELEGATE_FILE).empty()) { + MayaHydraDelegateRegistry::RegisterDelegate( + _tokens->MayaHydraTestDelegate, + [](const MayaHydraDelegate::InitData& initData) -> MayaHydraDelegatePtr { + return std::static_pointer_cast( + std::make_shared(initData)); + }); + } +} + +/* + * MayaHydraTestDelegate could be used as a test scene delegate, it is not used any more. + */ +MayaHydraTestDelegate::MayaHydraTestDelegate(const InitData& initData) + : MayaHydraDelegate(initData) +{ + _delegate.reset(new UsdImagingDelegate(initData.renderIndex, initData.delegateID)); +} + +void MayaHydraTestDelegate::Populate() +{ + _stage = UsdStage::Open(TfGetEnvSetting(MAYAHYDRALIB_TEST_DELEGATE_FILE)); + _delegate->Populate(_stage->GetPseudoRoot()); +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/delegates/testDelegate.h b/lib/mayaHydra/hydraExtensions/delegates/testDelegate.h new file mode 100644 index 0000000000..3f3c3e4ec0 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/delegates/testDelegate.h @@ -0,0 +1,48 @@ +// +// Copyright 2019 Luma Pictures +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MAYAHYDRALIB_TEST_DELEGATE_H +#define MAYAHYDRALIB_TEST_DELEGATE_H + +#include + +#include +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +/** + * \brief MayaHydraTestDelegate could be used as a test scene delegate, it is not used any more. + */ +class MayaHydraTestDelegate : public MayaHydraDelegate +{ +public: + MayaHydraTestDelegate(const InitData& initData); + + void Populate() override; + +private: + std::unique_ptr _delegate; + UsdStageRefPtr _stage; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_TEST_DELEGATE_H diff --git a/lib/mayaHydra/hydraExtensions/hydraUtils.cpp b/lib/mayaHydra/hydraExtensions/hydraUtils.cpp new file mode 100644 index 0000000000..900aed93ff --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/hydraUtils.cpp @@ -0,0 +1,248 @@ +// +// Copyright 2019 Luma Pictures +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "hydraUtils.h" + +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +// This is the delimiter that Maya uses to identify levels of hierarchy in the +// Maya DAG. +constexpr char MayaDagDelimiter[] = "|"; + +// This is the delimiter that Maya uses to separate levels of namespace in +// Maya node names. +constexpr char MayaNamespaceDelimiter[] = ":"; + +namespace MAYAHYDRA_NS_DEF { + +std::string ConvertVtValueToString(const VtValue& val) +{ + if (val.IsEmpty()) { + return "No Value!"; + } + + std::ostringstream ss; + if (val.IsHolding()) { + const bool v = val.UncheckedGet(); + ss << "bool : "; + ss << v; + } else if (val.IsHolding()) { + const TfToken elem = val.UncheckedGet(); + ss << "TfToken : " << elem.GetText(); + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + ss << std::to_string(elem) << " , "; + } + ss << ")"; + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + ss << std::to_string(elem) << " , "; + } + ss << ")"; + } else if (val.IsHolding()) { + const float v = val.UncheckedGet(); + ss << "float : "; + ss << v; + } else if (val.IsHolding()) { + const int v = val.UncheckedGet(); + ss << "int : "; + ss << v; + } else if (val.IsHolding()) { + const GfVec2f v = val.UncheckedGet(); + ss << "GfVec2f : (" << v[0] << " , " << v[1] << ")"; + } else if (val.IsHolding()) { + const GfVec3f v = val.UncheckedGet(); + ss << "GfVec3f : (" << v[0] << " , " << v[1] << " , " << v[2] << ")"; + } else if (val.IsHolding()) { + const auto v = val.UncheckedGet(); + ss << "GfVec3d : (" << v[0] << " , " << v[1] << " , " << v[2] << ")"; + } else if (val.IsHolding()) { + const SdfAssetPath v = val.UncheckedGet(); + const std::string assetPath = v.GetAssetPath(); + ss << "SdfAssetPath : \"" << assetPath << "\""; + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + ss << elem.GetText() << " , "; + } + ss << ")"; + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + auto strVec3f = "(" + std::to_string(elem[0]) + ", " + std::to_string(elem[1]) + ", " + + std::to_string(elem[2]) + ")"; + ss << strVec3f + " , "; + } + ss << ")"; + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + auto strVec3f = "(" + std::to_string(elem[0]) + ", " + std::to_string(elem[1]) + ", " + + std::to_string(elem[2]) + ")"; + ss << strVec3f + " , "; + } + ss << ")"; + } else if (val.IsHolding>()) { + auto arrayType = val.UncheckedGet>(); + ss << "VtArray : ("; + for (auto elem : arrayType) { + auto quathh = "(" + std::to_string(elem.GetReal()) + ", " + + std::to_string(elem.GetImaginary()[0]) + ", " + + std::to_string(elem.GetImaginary()[1]) + ", " + + std::to_string(elem.GetImaginary()[2]) + ")"; + ss << quathh + " , "; + } + ss << ")"; + } else if (val.IsHolding()) { + auto elem = val.UncheckedGet(); + auto quathh = "(" + std::to_string(elem.GetReal()) + ", " + + std::to_string(elem.GetImaginary()[0]) + ", " + std::to_string(elem.GetImaginary()[1]) + + ", " + std::to_string(elem.GetImaginary()[2]) + ")"; + ss << "GfQuath : " << quathh; + } else if (val.IsHolding()) { + auto mat4d = val.UncheckedGet(); + + double data[4][4]; + mat4d.Get(data); + auto strMat4d = std::string("(") + "{" + std::to_string(data[0][0]) + ", " + + std::to_string(data[0][1]) + ", " + std::to_string(data[0][2]) + ", " + + std::to_string(data[0][3]) + "}, " + "{" + std::to_string(data[1][0]) + ", " + + std::to_string(data[1][1]) + ", " + std::to_string(data[1][2]) + ", " + + std::to_string(data[1][3]) + "}, " + "{" + std::to_string(data[2][0]) + ", " + + std::to_string(data[2][1]) + ", " + std::to_string(data[2][2]) + ", " + + std::to_string(data[2][3]) + "}, " + "{" + std::to_string(data[3][0]) + ", " + + std::to_string(data[3][1]) + ", " + std::to_string(data[3][2]) + ", " + + std::to_string(data[3][3]) + "}" + ")"; + ss << "GfMatrix4d : " << strMat4d; + } + + std::string valueString = ss.str(); + if (valueString.size() > 0) { + return valueString; + } + + // Unknown + return "* Unknown Type *"; +} + +std::string StripNamespaces(const std::string& nodeName, const int nsDepth) +{ + if (nodeName.empty() || nsDepth == 0) { + return nodeName; + } + + std::stringstream ss; + + const std::vector nodeNameParts + = PXR_NS::TfStringSplit(nodeName, MayaDagDelimiter); + + const bool isAbsolute = PXR_NS::TfStringStartsWith(nodeName, MayaDagDelimiter); + + for (size_t i = 0u; i < nodeNameParts.size(); ++i) { + if (i == 0u && isAbsolute) { + // If nodeName was absolute, the first element in nodeNameParts + // will be empty, so just skip it. The output path will be made + // absolute with the next iteration. + continue; + } + + if (i != 0u) { + ss << MayaDagDelimiter; + } + + const std::vector nsNameParts + = PXR_NS::TfStringSplit(nodeNameParts[i], MayaNamespaceDelimiter); + + const size_t nodeNameIndex = nsNameParts.size() - 1u; + + auto startIter = nsNameParts.begin(); + if (nsDepth < 0) { + // If nsDepth is negative, we don't keep any namespaces, so advance + // startIter to the last element in the vector, which is just the + // node name. + startIter += nodeNameIndex; + } else { + // Otherwise we strip as many namespaces as possible up to nsDepth, + // but no more than what would leave us with just the node name. + startIter += std::min(static_cast(nsDepth), nodeNameIndex); + } + + ss << PXR_NS::TfStringJoin(startIter, nsNameParts.end(), MayaNamespaceDelimiter); + } + + return ss.str(); +} + +// Elements of the path will be sanitized such that it is a valid SdfPath. +// This means it will replace Maya's namespace delimiter (':') with +// underscores ('_'). +// A SdfPath in Pixar USD is considered invalid if it does not conform to the rules for path names. +// Some common issues that can make a path invalid include: Starting with a number : Path names +// must start with a letter, not a number. Including spaces or special characters : Path names can +// only contain letters, numbers, and the characters _, -, and : . +void SanitizeNameForSdfPath(std::string& inoutPathString, bool doStripNamespaces /*= false*/) +{ + if (doStripNamespaces) { + // Drop namespaces instead of making them part of the path. + inoutPathString = StripNamespaces(inoutPathString); + } + + std::replace( + inoutPathString.begin(), + inoutPathString.end(), + MayaDagDelimiter[0], + PXR_NS::SdfPathTokens->childDelimiter.GetString()[0]); + std::replace(inoutPathString.begin(), inoutPathString.end(), MayaNamespaceDelimiter[0], '_'); + std::replace(inoutPathString.begin(), inoutPathString.end(), ',', '_'); + std::replace(inoutPathString.begin(), inoutPathString.end(), ';', '_'); +} + +SdfPath MakeRelativeToParentPath(const SdfPath& path) +{ + return path.MakeRelativePath(path.GetParentPath()); +} + +bool GetXformMatrixFromPrim(const HdSceneIndexPrim& prim, GfMatrix4d& outMatrix) +{ + HdContainerDataSourceHandle xformContainer + = HdContainerDataSource::Cast(prim.dataSource->Get(HdXformSchemaTokens->xform)); + if (!xformContainer) { + return false; + } + HdXformSchema xform = HdXformSchema(xformContainer); + if (!xform.GetMatrix()) { + return false; + } + outMatrix = xform.GetMatrix()->GetValue(0).Get(); + return true; +} + +} // namespace MAYAHYDRA_NS_DEF diff --git a/lib/mayaHydra/hydraExtensions/hydraUtils.h b/lib/mayaHydra/hydraExtensions/hydraUtils.h new file mode 100644 index 0000000000..fbb925ec57 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/hydraUtils.h @@ -0,0 +1,93 @@ +// +// Copyright 2019 Luma Pictures +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_HYDRA_UTILS_H +#define MAYAHYDRALIB_HYDRA_UTILS_H + +#include +#include + +#include +#include +#include +#include + +#include + +namespace MAYAHYDRA_NS_DEF { + +/** + * @brief Return the \p VtValue type and value as a string for debugging purposes. + * + * @param[in] val is the \p VtValue to be converted. + * + * @return The \p VtValue type and value in string form. + */ +MAYAHYDRALIB_API +std::string ConvertVtValueToString(const pxr::VtValue& val); + +/** + * @brief Strip \p nsDepth namespaces from \p nodeName. + * + * This will turn "taco:foo:bar" into "foo:bar" for \p nsDepth == 1, or "taco:foo:bar" into + * "bar" for \p nsDepth > 1. If \p nsDepth is -1, all namespaces are stripped. + * + * @param[in] nodeName is the node name from which to strip namespaces. + * @param[in] nsDepth is the namespace depth to strip. + * + * @return The stripped version of \p nodeName. + */ +MAYAHYDRALIB_API +std::string StripNamespaces(const std::string& nodeName, const int nsDepth = -1); + +/** + * @brief Replaces the invalid characters for SdfPath in-place in \p inOutPathString. + * + * @param[in,out] inOutPathString is the path string to sanitize. + * @param[in] doStripNamespaces determines whether to strip namespaces or not. + */ +MAYAHYDRALIB_API +void SanitizeNameForSdfPath(std::string& inOutPathString, bool doStripNamespaces = false); + +/** + * @brief Get the given SdfPath without its parent path. + * + * The result is the last element of the original SdfPath. + * + * @param[in] path is the SdfPath from which to remove the parent path. + * + * @return The path without its parent path. + */ +MAYAHYDRALIB_API +pxr::SdfPath MakeRelativeToParentPath(const pxr::SdfPath& path); + +/** + * @brief Get the Hydra Xform matrix from a given prim. + * + * This method makes no guarantee on whether the matrix is flattened or not. + * + * @param[in] prim is the Hydra prim in the SceneIndex of which to get the transform matrix. + * @param[out] outMatrix is the transform matrix of the prim. + * + * @return True if the operation succeeded, false otherwise. + */ +MAYAHYDRALIB_API +bool GetXformMatrixFromPrim(const pxr::HdSceneIndexPrim& prim, pxr::GfMatrix4d& outMatrix); + +} // namespace MAYAHYDRA_NS_DEF + +#endif // MAYAHYDRALIB_HYDRA_UTILS_H diff --git a/lib/mayaHydra/hydraExtensions/interface.h b/lib/mayaHydra/hydraExtensions/interface.h new file mode 100644 index 0000000000..2aaa7e4907 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/interface.h @@ -0,0 +1,69 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_INTERFACE_H +#define MAYAHYDRALIB_INTERFACE_H + +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +using SceneIndicesVector = std::vector; + +/// In order to access this interface, call the function GetMayaHydraLibInterface() +class MayaHydraLibInterface +{ +public: + /** + * @brief Register a terminal scene index into the Hydra plugin. + * + * @param[in] sceneIndex is a pointer to the SceneIndex to be registered. + */ + virtual void RegisterTerminalSceneIndex(HdSceneIndexBasePtr sceneIndex) = 0; + + /** + * @brief Unregister a terminal scene index from the Hydra plugin. + * + * @param[in] sceneIndex is a pointer to the SceneIndex to be unregistered. + */ + virtual void UnregisterTerminalSceneIndex(HdSceneIndexBasePtr sceneIndex) = 0; + + /** + * @brief Clear the list of registered terminal scene indices + * + * This does not delete them, but just unregisters them. + */ + virtual void ClearTerminalSceneIndices() = 0; + + /** + * @brief Retrieve the list of registered terminal scene indices from the Hydra plugin. + * + * @return A const reference to the vector of registered terminal scene indices. + */ + virtual const SceneIndicesVector& GetTerminalSceneIndices() const = 0; +}; + +/// Access the MayaHydraLibInterface instance +MAYAHYDRALIB_API +MayaHydraLibInterface& GetMayaHydraLibInterface(); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_INTERFACE_H diff --git a/lib/mayaHydra/hydraExtensions/interfaceImp.cpp b/lib/mayaHydra/hydraExtensions/interfaceImp.cpp new file mode 100644 index 0000000000..ccfb841170 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/interfaceImp.cpp @@ -0,0 +1,49 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "interfaceImp.h" + +PXR_NAMESPACE_OPEN_SCOPE + +MayaHydraLibInterface& GetMayaHydraLibInterface() +{ + static MayaHydraLibInterfaceImp libInterface; + return libInterface; +} + +void MayaHydraLibInterfaceImp::RegisterTerminalSceneIndex(HdSceneIndexBasePtr sceneIndex) +{ + auto foundSceneIndex = std::find(_sceneIndices.begin(), _sceneIndices.end(), sceneIndex); + if (foundSceneIndex == _sceneIndices.end()) { + _sceneIndices.push_back(sceneIndex); + } +} + +void MayaHydraLibInterfaceImp::UnregisterTerminalSceneIndex(HdSceneIndexBasePtr sceneIndex) +{ + auto foundSceneIndex = std::find(_sceneIndices.begin(), _sceneIndices.end(), sceneIndex); + if (foundSceneIndex != _sceneIndices.end()) { + _sceneIndices.erase(foundSceneIndex); + } +} + +void MayaHydraLibInterfaceImp::ClearTerminalSceneIndices() { _sceneIndices.clear(); } + +const SceneIndicesVector& MayaHydraLibInterfaceImp::GetTerminalSceneIndices() const +{ + return _sceneIndices; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/interfaceImp.h b/lib/mayaHydra/hydraExtensions/interfaceImp.h new file mode 100644 index 0000000000..bdcef76e5b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/interfaceImp.h @@ -0,0 +1,46 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_INTERFACE_IMP_H +#define MAYAHYDRALIB_INTERFACE_IMP_H + +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraLibInterfaceImp : public MayaHydraLibInterface +{ +public: + MayaHydraLibInterfaceImp() = default; + virtual ~MayaHydraLibInterfaceImp() = default; + + void RegisterTerminalSceneIndex(HdSceneIndexBasePtr) override; + void UnregisterTerminalSceneIndex(HdSceneIndexBasePtr) override; + void ClearTerminalSceneIndices() override; + const SceneIndicesVector& GetTerminalSceneIndices() const override; + +private: + SceneIndicesVector _sceneIndices; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIB_INTERFACE_IMP_H diff --git a/lib/mayaHydra/hydraExtensions/mayaHydra.h.src b/lib/mayaHydra/hydraExtensions/mayaHydra.h.src new file mode 100644 index 0000000000..aa2fc2e422 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mayaHydra.h.src @@ -0,0 +1,50 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_MAYAHYDRA_H +#define MAYAHYDRALIB_MAYAHYDRA_H + +#define MAYAHYDRA_MAJOR_VERSION ${MAYAHYDRA_MAJOR_VERSION} +#define MAYAHYDRA_MINOR_VERSION ${MAYAHYDRA_MINOR_VERSION} +#define MAYAHYDRA_PATCH_LEVEL ${MAYAHYDRA_PATCH_LEVEL} +#define MAYAHYDRA_API_VERSION (MAYAHYDRA_MAJOR_VERSION * 10000 + MAYAHYDRA_MINOR_VERSION * 100 + MAYAHYDRA_PATCH_LEVEL) + +// MayaHydra public namespace string will never change. +#define MAYAHYDRA_NS MayaHydra +// C preprocessor trickery to expand arguments. +#define MAYAHYDRA_CONCAT(A, B) MAYAHYDRA_CONCAT_IMPL(A, B) +#define MAYAHYDRA_CONCAT_IMPL(A, B) A##B +// Versioned namespace includes the major version number. +#define MAYAHYDRA_VERSIONED_NS MAYAHYDRA_CONCAT(MAYAHYDRA_NS, _v${MAYAHYDRA_MAJOR_VERSION}) + +namespace MAYAHYDRA_VERSIONED_NS {} + +// With a using namespace declaration, pull in the versioned namespace into the +// MayaHydra public namespace, to allow client code to use the plain MayaHydra +// namespace, e.g. MayaHydra::Class. +namespace MAYAHYDRA_NS { + using namespace MAYAHYDRA_VERSIONED_NS; +} + +// Macro to place the MayaHydra symbols in the versioned namespace, which is how +// they will appear in the shared library, e.g. MayaHydra_v1::Class. +#ifdef DOXYGEN +#define MAYAHYDRA_NS_DEF MAYAHYDRA_NS +#else +#define MAYAHYDRA_NS_DEF MAYAHYDRA_VERSIONED_NS +#endif + +#endif // MAYAHYDRALIB_MAYAHYDRA_H diff --git a/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.cpp b/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.cpp new file mode 100644 index 0000000000..bcbcb0ea06 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.cpp @@ -0,0 +1,466 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraSceneProducer.h" + +#include +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE + +TF_DEFINE_ENV_SETTING(MAYA_HYDRA_ENABLE_NATIVE_SCENE_INDEX, true, + "Enable scene index for Maya native scene."); + +bool enableMayaNativeSceneIndex() { + static bool enable = TfGetEnvSetting(MAYA_HYDRA_ENABLE_NATIVE_SCENE_INDEX); + return enable; +} + +MayaHydraSceneProducer::MayaHydraSceneProducer( + Fvp::RenderIndexProxy& renderIndexProxy, + const SdfPath& id, + MayaHydraDelegate::InitData& initData, + bool lightEnabled +) : _renderIndexProxy(renderIndexProxy) +{ + if (enableMayaNativeSceneIndex()) + { + initData.name = TfToken("MayaHydraSceneIndex"); + initData.delegateID = id.AppendChild( + TfToken(TfStringPrintf("_Index_MayaHydraSceneIndex_%p", this))); + initData.producer = this; + _sceneIndex = MayaHydraSceneIndex::New(initData, lightEnabled); + TF_VERIFY(_sceneIndex, "Maya Hydra scene index not found, check mayaHydra plugin installation."); + } + else + { + SdfPathVector solidPrimsRootPaths; + auto delegateNames = MayaHydraDelegateRegistry::GetDelegateNames(); + auto creators = MayaHydraDelegateRegistry::GetDelegateCreators(); + TF_VERIFY(delegateNames.size() == creators.size()); + for (size_t i = 0, n = creators.size(); i < n; ++i) { + const auto& creator = creators[i]; + if (creator == nullptr) { + continue; + } + initData.name = delegateNames[i]; + initData.delegateID = id.AppendChild( + TfToken(TfStringPrintf("_Delegate_%s_%lu_%p", delegateNames[i].GetText(), i, this))); + initData.producer = this; + auto newDelegate = creator(initData); + if (newDelegate) { + // Call SetLightsEnabled before the delegate is populated + newDelegate->SetLightsEnabled(lightEnabled); + _sceneDelegate + = std::dynamic_pointer_cast(newDelegate); + if (TF_VERIFY( + _sceneDelegate, + "Maya Hydra scene delegate not found, check mayaHydra plugin installation.")) { + solidPrimsRootPaths.push_back(_sceneDelegate->GetLightedPrimsRootPath()); + } + _delegates.emplace_back(std::move(newDelegate)); + } + } + + initData.delegateID + = id.AppendChild(TfToken(TfStringPrintf("_DefaultLightDelegate_%p", this))); + _defaultLightDelegate.reset(new MtohDefaultLightDelegate(initData)); + // Set the scene delegate SolidPrimitivesRootPaths for the lines and points primitives to be + // ignored by the default light + _defaultLightDelegate->SetSolidPrimitivesRootPaths(solidPrimsRootPaths); + } +} + +MayaHydraSceneProducer::~MayaHydraSceneProducer() +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->GetRenderIndex().RemoveSceneIndex(_sceneIndex); + } + _delegates.clear(); +} + +void MayaHydraSceneProducer::HandleCompleteViewportScene(const MDataServerOperation::MViewportScene& scene, MFrameContext::DisplayStyle ds) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->HandleCompleteViewportScene(scene, ds); + } + else + { + return _sceneDelegate->HandleCompleteViewportScene(scene, ds); + } +} + +void MayaHydraSceneProducer::Populate() +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->Populate(); + // Call InsertSceneIndex before prims are added to scene index, would it be better to call later? + _renderIndexProxy.InsertSceneIndex(_sceneIndex, SdfPath::AbsoluteRootPath()); + } + else + { + for (auto& it : _delegates) { + it->Populate(); + } + + if (_defaultLightDelegate && !_sceneDelegate->GetLightsEnabled()) { + _defaultLightDelegate->Populate(); + } + } +} + +SdfPath MayaHydraSceneProducer::SetCameraViewport(const MDagPath& camPath, const GfVec4d& viewport) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->SetCameraViewport(camPath, viewport); + } + else + { + return _sceneDelegate->SetCameraViewport(camPath, viewport); + } +} + +void MayaHydraSceneProducer::SetLightsEnabled(const bool enabled) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->SetLightsEnabled(enabled); + } + else + { + return _sceneDelegate->SetLightsEnabled(enabled); + } +} + +void MayaHydraSceneProducer::SetDefaultLightEnabled(const bool enabled) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->SetDefaultLightEnabled(enabled); + } + else + { + _defaultLightDelegate->SetLightingOn(enabled); + } +} + + +void MayaHydraSceneProducer::SetDefaultLight(const GlfSimpleLight& light) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->SetDefaultLight(light); + } + else + { + _defaultLightDelegate->SetDefaultLight(light); + } +} + +const MayaHydraParams& MayaHydraSceneProducer::GetParams() const +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetParams(); + } + else + { + return _sceneDelegate->GetParams(); + } +} + +void MayaHydraSceneProducer::SetParams(const MayaHydraParams& params) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->SetParams(params); + } + else + { + for (auto& it : _delegates) { + it->SetParams(params); + } + } +} + +bool MayaHydraSceneProducer::AddPickHitToSelectionList( + const HdxPickHit& hit, + const MHWRender::MSelectionInfo& selectInfo, + MSelectionList& selectionList, + MPointArray& worldSpaceHitPts) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->AddPickHitToSelectionList( + hit, + selectInfo, + selectionList, + worldSpaceHitPts); + } + else + { + return _sceneDelegate->AddPickHitToSelectionList( + hit, + selectInfo, + selectionList, + worldSpaceHitPts); + } +} + +HdRenderIndex& MayaHydraSceneProducer::GetRenderIndex() +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetRenderIndex(); + } + else + { + return _sceneDelegate->GetRenderIndex(); + } +} + +bool MayaHydraSceneProducer::IsHdSt() const +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->IsHdSt(); + } + else + { + return _sceneDelegate->IsHdSt(); + } +} + +bool MayaHydraSceneProducer::GetPlaybackRunning() const +{ + if (enableMayaNativeSceneIndex()) + { + return false; + } + else + { + return _sceneDelegate->GetPlaybackRunning(); + } +} + +SdfPath MayaHydraSceneProducer::GetPrimPath(const MDagPath& dg, bool isSprim) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetPrimPath(dg, isSprim); + } + else + { + return _sceneDelegate->GetPrimPath(dg, isSprim); + } +} + +void MayaHydraSceneProducer::InsertRprim( + MayaHydraAdapter* adapter, + const TfToken& typeId, + const SdfPath& id, + const SdfPath& instancerId) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->InsertPrim(adapter, typeId, id); + } + else + { + return _sceneDelegate->InsertRprim(typeId, id, instancerId); + } +} + +void MayaHydraSceneProducer::RemoveRprim(const SdfPath& id) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->RemovePrim(id); + } + else + { + _sceneDelegate->RemoveRprim(id); + } +} + +void MayaHydraSceneProducer::MarkRprimDirty(const SdfPath& id, HdDirtyBits dirtyBits) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->MarkPrimDirty(id, dirtyBits); + } + else + { + _sceneDelegate->GetRenderIndex().GetChangeTracker().MarkRprimDirty(id, dirtyBits); + } +} + +void MayaHydraSceneProducer::InsertSprim( + MayaHydraAdapter* adapter, + const TfToken& typeId, + const SdfPath& id, + HdDirtyBits initialBits) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->InsertPrim(adapter, typeId, id); + } + else + { + _sceneDelegate->InsertSprim(typeId, id, initialBits); + } +} + +void MayaHydraSceneProducer::RemoveSprim(const TfToken& typeId, const SdfPath& id) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->RemovePrim(id); + } + else + { + _sceneDelegate->RemoveSprim(typeId, id); + } +} + +void MayaHydraSceneProducer::MarkSprimDirty(const SdfPath& id, HdDirtyBits dirtyBits) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->MarkPrimDirty(id, dirtyBits); + } + else + { + _sceneDelegate->GetRenderIndex().GetChangeTracker().MarkSprimDirty(id, dirtyBits); + } +} + +SdfPath MayaHydraSceneProducer::GetDelegateID(TfToken name) const +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetDelegateID(name); + } + else + { + for (auto& delegate : _delegates) { + if (delegate->GetName() == name) { + return delegate->GetMayaDelegateID(); + } + } + return SdfPath(); + } +} + +void MayaHydraSceneProducer::PreFrame(const MHWRender::MDrawContext& drawContext) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->PreFrame(drawContext); + } + else + { + for (auto& it : _delegates) { + it->PreFrame(drawContext); + } + } +} + +void MayaHydraSceneProducer::PostFrame() +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->PostFrame(); + } + else + { + for (auto& it : _delegates) { + it->PostFrame(); + } + } +} + +void MayaHydraSceneProducer::RemoveAdapter(const SdfPath& id) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->RemoveAdapter(id); + } + else + { + return _sceneDelegate->RemoveAdapter(id); + } +} + +void MayaHydraSceneProducer::RecreateAdapterOnIdle(const SdfPath& id, const MObject& obj) +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->RecreateAdapterOnIdle(id, obj); + } + else + { + return _sceneDelegate->RecreateAdapterOnIdle(id, obj); + } +} + +SdfPath MayaHydraSceneProducer::GetLightedPrimsRootPath() const +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetLightedPrimsRootPath(); + } + else + { + return _sceneDelegate->GetLightedPrimsRootPath(); + } +} + +void MayaHydraSceneProducer::MaterialTagChanged(const SdfPath& id) +{ + if (enableMayaNativeSceneIndex()) + { + _sceneIndex->MaterialTagChanged(id); + } + else + { + _sceneDelegate->MaterialTagChanged(id); + } +} + +GfInterval MayaHydraSceneProducer::GetCurrentTimeSamplingInterval() const +{ + if (enableMayaNativeSceneIndex()) + { + return _sceneIndex->GetCurrentTimeSamplingInterval(); + } + else + { + return _sceneDelegate->GetCurrentTimeSamplingInterval(); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.h b/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.h new file mode 100644 index 0000000000..1eacf77273 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mayaHydraSceneProducer.h @@ -0,0 +1,205 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRASCENEPRODUCER_H +#define MAYAHYDRASCENEPRODUCER_H + +#include + +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace FVP_NS_DEF { +class RenderIndexProxy; +} + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneDelegate; +class MayaHydraAdapter; +class MtohDefaultLightDelegate; + +/** + * \brief MayaHydraSceneProducer is used to produce the hydra scene from Maya native scene. + * Under the hood, the work is delegated to MayaHydraSceneIndex or MayaHydraSceneDelegate, depends on + * if MAYA_HYDRA_ENABLE_NATIVE_SCENE_INDEX is enabled or not. + * Note that MayaHydraSceneDelegate could be deprecated in the future. + */ +class MAYAHYDRALIB_API MayaHydraSceneProducer +{ +public: + MayaHydraSceneProducer( + Fvp::RenderIndexProxy& renderIndexProxy, + const SdfPath& id, + MayaHydraDelegate::InitData& initData, + bool lightEnabled); + ~MayaHydraSceneProducer(); + + // Propogate scene changes from Maya to Hydra + void HandleCompleteViewportScene(const MDataServerOperation::MViewportScene& scene, MFrameContext::DisplayStyle ds); + + // Populate primitives from Maya + void Populate(); + + // Add hydra pick points and items to Maya's selection list + bool AddPickHitToSelectionList( + const HdxPickHit& hit, + const MHWRender::MSelectionInfo& selectInfo, + MSelectionList& selectionList, + MPointArray& worldSpaceHitPts); + + // Insert a Rprim to hydra scene + void InsertRprim( + MayaHydraAdapter* adapter, + const TfToken& typeId, + const SdfPath& id, + const SdfPath& instancerId = {}); + + // Remove a Rprim from hydra scene + void RemoveRprim(const SdfPath& id); + + // Mark a Rprim in hydra scene as dirty + void MarkRprimDirty(const SdfPath& id, HdDirtyBits dirtyBits); + + // Insert a Sprim to hydra scene + void InsertSprim( + MayaHydraAdapter* adapter, + const TfToken& typeId, + const SdfPath& id, + HdDirtyBits initialBits); + + // Remove a Sprim from hydra scene + void RemoveSprim(const TfToken& typeId, const SdfPath& id); + + // Mark a Sprim in hydra scene as dirty + void MarkSprimDirty(const SdfPath& id, HdDirtyBits dirtyBits); + + // Operation that's performed on rendering a frame + void PreFrame(const MHWRender::MDrawContext& drawContext); + void PostFrame(); + + const MayaHydraParams& GetParams() const; + void SetParams(const MayaHydraParams& params); + + // Adapter operations + void RemoveAdapter(const SdfPath& id); + void RecreateAdapterOnIdle(const SdfPath& id, const MObject& obj); + + // Update viewport info to camera + SdfPath SetCameraViewport(const MDagPath& camPath, const GfVec4d& viewport); + + // Enable or disable lighting + void SetLightsEnabled(const bool enabled); + void SetDefaultLightEnabled(const bool enabled); + void SetDefaultLight(const GlfSimpleLight& light); + + bool IsHdSt() const; + + bool GetPlaybackRunning() const; + + SdfPath GetPrimPath(const MDagPath& dg, bool isSprim); + + HdRenderIndex& GetRenderIndex(); + + SdfPath GetLightedPrimsRootPath() const; + + GfInterval GetCurrentTimeSamplingInterval() const; + + // Return the id of underlying delegate by name (MayaHydraSceneIndex or MayaHydraSceneDelegate) + SdfPath GetDelegateID(TfToken name) const; + + MayaHydraSceneIndexRefPtr GetSceneIndex() const { return _sceneIndex; } + + void MaterialTagChanged(const SdfPath& id); + + // Common function to return templated sample types + template + size_t SampleValues(size_t maxSampleCount, float* times, T* samples, Getter getValue) + { + if (ARCH_UNLIKELY(maxSampleCount == 0)) { + return 0; + } + // Fast path 1 sample at current-frame + if (maxSampleCount == 1 + || (!GetParams().motionSamplesEnabled() && GetParams().motionSampleStart == 0)) { + times[0] = 0.0f; + samples[0] = getValue(); + return 1; + } + + const GfInterval shutter = GetCurrentTimeSamplingInterval(); + // Shutter for [-1, 1] (size 2) should have a step of 2 for 2 samples, and 1 for 3 samples + // For sample size of 1 tStep is unused and we match USD and to provide t=shutterOpen + // sample. + const double tStep = maxSampleCount > 1 ? (shutter.GetSize() / (maxSampleCount - 1)) : 0; + const MTime mayaTime = MAnimControl::currentTime(); + size_t nSamples = 0; + double relTime = shutter.GetMin(); + + for (size_t i = 0; i < maxSampleCount; ++i) { + T sample; + { + MDGContextGuard guard(mayaTime + relTime); + sample = getValue(); + } + // We compare the sample to the previous in order to reduce sample count on output. + // Goal is to reduce the amount of samples/keyframes the Hydra delegate has to absorb. + if (!nSamples || sample != samples[nSamples - 1]) { + samples[nSamples] = std::move(sample); + times[nSamples] = relTime; + ++nSamples; + } + relTime += tStep; + } + return nSamples; + } + +private: + + // + // Delegates, depends on if MAYA_HYDRA_ENABLE_NATIVE_SCENE_INDEX is enabled or not. + // + // SceneDelegate + std::shared_ptr _sceneDelegate; + std::vector _delegates; + std::unique_ptr _defaultLightDelegate; + + // SceneIndex + MayaHydraSceneIndexRefPtr _sceneIndex; + Fvp::RenderIndexProxy& _renderIndexProxy; +}; + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRASCENEPRODUCER_H diff --git a/lib/mayaHydra/hydraExtensions/mayaUtils.cpp b/lib/mayaHydra/hydraExtensions/mayaUtils.cpp new file mode 100644 index 0000000000..d5f308e84b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mayaUtils.cpp @@ -0,0 +1,73 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaUtils.h" + +#include +#include +#include +#include +#include +#include + +namespace MAYAHYDRA_NS_DEF { + +MStatus GetDagPathFromNodeName(const MString& nodeName, MDagPath& outDagPath) +{ + MSelectionList selectionList; + MStatus status = selectionList.add(nodeName); + if (status) { + status = selectionList.getDagPath(0, outDagPath); + } + return status; +} + +MStatus GetMayaMatrixFromDagPath(const MDagPath& dagPath, MMatrix& outMatrix) +{ + MStatus status; + outMatrix = dagPath.inclusiveMatrix(&status); + return status; +} + +bool IsUfeItemFromMayaUsd(const MDagPath& dagPath, MStatus* returnStatus) +{ + static const MString ufeRuntimeAttributeName = "ufeRuntime"; + static const MString mayaUsdUfeRuntimeName = "USD"; + + MFnDagNode dagNode(dagPath); + MStatus ufePlugSearchStatus; + MPlug ufeRuntimePlug = dagNode.findPlug(ufeRuntimeAttributeName, false, &ufePlugSearchStatus); + if (returnStatus) { + *returnStatus = ufePlugSearchStatus; + } + return ufePlugSearchStatus && ufeRuntimePlug.asString() == mayaUsdUfeRuntimeName; +} + +bool IsUfeItemFromMayaUsd(const MObject& obj, MStatus* returnStatus) +{ + MDagPath dagPath; + MStatus dagPathSearchStatus = MDagPath::getAPathTo(obj, dagPath); + if (!dagPathSearchStatus) { + if (returnStatus) { + *returnStatus = dagPathSearchStatus; + } + return false; + } + + return IsUfeItemFromMayaUsd(dagPath, returnStatus); +} + +} // namespace MAYAHYDRA_NS_DEF diff --git a/lib/mayaHydra/hydraExtensions/mayaUtils.h b/lib/mayaHydra/hydraExtensions/mayaUtils.h new file mode 100644 index 0000000000..40a913e05a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mayaUtils.h @@ -0,0 +1,99 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_MAYA_UTILS_H +#define MAYAHYDRALIB_MAYA_UTILS_H + +#include +#include + +#include + +#include + +namespace MAYAHYDRA_NS_DEF { + +// Names of color tables for indexed colors +const std::string kActiveColorTableName = "active"; +const std::string kDormantColorTableName = "dormant"; + +// Color names +const std::string kLeadColorName = "lead"; +const std::string kPolymeshActiveColorName = "polymeshActive"; +const std::string kPolyVertexColorName = "polyVertex"; +const std::string kPolyEdgeColorName = "polyEdge"; +const std::string kPolyFaceColorName = "polyFace"; + +/** + * @brief Get the DAG path of a node from the Maya scene graph using its name + * + * @param[in] nodeName is the name of the node to get the DAG path of. + * @param[out] outDagPath is the DAG path of the node in the Maya scene graph. + * + * @return The resulting status of the operation. + */ +MAYAHYDRALIB_API +MStatus GetDagPathFromNodeName(const MString& nodeName, MDagPath& outDagPath); + +/** + * @brief Get the Maya transform matrix of a node from its DAG path + * + * The output transform matrix is the resultant ("flattened") matrix from it and + * its parents' transforms. + * + * @param[in] dagPath is the DAG path of the node in the Maya scene graph. + * @param[out] outMatrix is the Maya transform matrix of the node. + * + * @return The resulting status of the operation. + */ +MAYAHYDRALIB_API +MStatus GetMayaMatrixFromDagPath(const MDagPath& dagPath, MMatrix& outMatrix); + +/** + * @brief Determines whether a given DAG path points to a UFE item created by maya-usd + * + * UFE stands for Universal Front End : the goal of the Universal Front End is to create a + * DCC-agnostic component that will allow a DCC to browse and edit data in multiple data models. + * + * @param[in] dagPath is the DAG path of the node in the Maya scene graph. + * @param[out] returnStatus is an optional output variable to return whether the operation was + * successful. Default value is nullptr (not going to store the result status). + * + * @return True if the item pointed to by dagPath is a UFE item created by maya-usd, false + * otherwise. + */ +MAYAHYDRALIB_API +bool IsUfeItemFromMayaUsd(const MDagPath& dagPath, MStatus* returnStatus = nullptr); + +/** + * @brief Determines whether a given object is a UFE item created by maya-usd + * + * UFE stands for Universal Front End : the goal of the Universal Front End is to create a + * DCC-agnostic component that will allow a DCC to browse and edit data in multiple data models. + * + * @param[in] obj is the object representing the DAG node. + * @param[out] returnStatus is an optional output variable to return whether the operation was + * successful. Default value is nullptr (not going to store the result status). + * + * @return True if the item represented by obj is a UFE item created by maya-usd, false + * otherwise. + */ +MAYAHYDRALIB_API +bool IsUfeItemFromMayaUsd(const MObject& obj, MStatus* returnStatus = nullptr); + +} // namespace MAYAHYDRA_NS_DEF + +#endif // MAYAHYDRALIB_MAYA_UTILS_H diff --git a/lib/mayaHydra/hydraExtensions/mixedUtils.cpp b/lib/mayaHydra/hydraExtensions/mixedUtils.cpp new file mode 100644 index 0000000000..b796aaa46a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mixedUtils.cpp @@ -0,0 +1,175 @@ +// +// Copyright 2019 Luma Pictures +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mixedUtils.h" + +#include +#include + +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +namespace MAYAHYDRA_NS_DEF { + +TfToken GetFileTexturePath(const MFnDependencyNode& fileNode) +{ + if (fileNode.findPlug(MayaAttrs::file::uvTilingMode, true).asShort() != 0) { + const TfToken ret { + fileNode.findPlug(MayaAttrs::file::fileTextureNamePattern, true).asString().asChar() + }; + return ret.IsEmpty() + ? TfToken { fileNode.findPlug(MayaAttrs::file::computedFileTextureNamePattern, true) + .asString() + .asChar() } + : ret; + } else { + const TfToken ret { MRenderUtil::exactFileTextureName(fileNode.object()).asChar() }; + return ret.IsEmpty() ? TfToken { fileNode.findPlug(MayaAttrs::file::fileTextureName, true) + .asString() + .asChar() } + : ret; + } +} + +bool IsShape(const MDagPath& dagPath) +{ + if (dagPath.hasFn(MFn::kTransform)) { + return false; + } + + // go to the parent + MDagPath parentDagPath = dagPath; + parentDagPath.pop(); + if (!parentDagPath.hasFn(MFn::kTransform)) { + return false; + } + + unsigned int numberOfShapesDirectlyBelow = 0; + parentDagPath.numberOfShapesDirectlyBelow(numberOfShapesDirectlyBelow); + return (numberOfShapesDirectlyBelow == 1); +} + +SdfPath DagPathToSdfPath( + const MDagPath& dagPath, + const bool mergeTransformAndShape, + const bool stripNamespaces) +{ + std::string name = dagPath.fullPathName().asChar(); + SanitizeNameForSdfPath(name, stripNamespaces); + SdfPath usdPath(name); + + if (mergeTransformAndShape && IsShape(dagPath)) { + usdPath = usdPath.GetParentPath(); + } + + return usdPath; +} + +SdfPath RenderItemToSdfPath(const MRenderItem& ri, const bool stripNamespaces) +{ + std::string internalObjectId( + "_" + std::to_string(ri.InternalObjectId())); // preventively prepend item id by underscore + std::string name(ri.name().asChar() + internalObjectId); + // Try to sanitize maya path to be used as an sdf path. + SanitizeNameForSdfPath(name, stripNamespaces); + // Path names must start with a letter, not a number + // If a number is found, prepend the path with an underscore + char digit = name[0]; + if (std::isdigit(digit)) { + name.insert(0, "_"); + } + + SdfPath sdfPath(name); + if (!TF_VERIFY( + !sdfPath.IsEmpty(), + "Render item using invalid SdfPath '%s'. Using item's id instead.", + name.c_str())) { + // If failed to include render item's name as an SdfPath simply use the item id. + return SdfPath(internalObjectId); + } + return sdfPath; +} + +bool getRGBAColorPreferenceValue(const std::string& colorName, PXR_NS::GfVec4f& outColor) +{ + MDoubleArray rgbaColorValues; + bool wasCommandSuccessful = MGlobal::executeCommand( + MString("displayRGBColor -q -a ") + MString(colorName.c_str()), rgbaColorValues); + if (!wasCommandSuccessful || rgbaColorValues.length() != 4) { + return false; + } + outColor[0] = static_cast(rgbaColorValues[0]); + outColor[1] = static_cast(rgbaColorValues[1]); + outColor[2] = static_cast(rgbaColorValues[2]); + outColor[3] = static_cast(rgbaColorValues[3]); + return true; +} + +bool getIndexedColorPreferenceIndex( + const std::string& colorName, + const std::string& tableName, + size_t& outIndex) +{ + MIntArray indexInPalette; + std::string getIndexCommand = "displayColor -q -" + tableName + " " + colorName; + bool wasCommandSuccessful + = MGlobal::executeCommand(MString(getIndexCommand.c_str()), indexInPalette); + if (!wasCommandSuccessful || indexInPalette.length() != 1) { + return false; + } + outIndex = indexInPalette[0]; + return true; +} + +bool getColorPreferencesPaletteColor( + const std::string& tableName, + size_t index, + PXR_NS::GfVec4f& outColor) +{ + MDoubleArray rgbColorValues; + std::string getColorCommand = "colorIndex -q -" + tableName + " " + std::to_string(index); + bool wasCommandSuccessful + = MGlobal::executeCommand(MString(getColorCommand.c_str()), rgbColorValues); + if (!wasCommandSuccessful || rgbColorValues.length() != 3) { + return false; + } + outColor[0] = static_cast(rgbColorValues[0]); + outColor[1] = static_cast(rgbColorValues[1]); + outColor[2] = static_cast(rgbColorValues[2]); + outColor[3] = 1.0f; + return true; +} + +bool getIndexedColorPreferenceValue( + const std::string& colorName, + const std::string& tableName, + PXR_NS::GfVec4f& outColor) +{ + size_t colorIndex = 0; + if (getIndexedColorPreferenceIndex(colorName, tableName, colorIndex)) { + return getColorPreferencesPaletteColor(tableName, colorIndex, outColor); + } + return false; +} + +} // namespace MAYAHYDRA_NS_DEF diff --git a/lib/mayaHydra/hydraExtensions/mixedUtils.h b/lib/mayaHydra/hydraExtensions/mixedUtils.h new file mode 100644 index 0000000000..3da3ae577a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/mixedUtils.h @@ -0,0 +1,178 @@ +// +// Copyright 2019 Luma Pictures +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIB_MIXED_UTILS_H +#define MAYAHYDRALIB_MIXED_UTILS_H + +#include +#include + +#include +#include +#include + +#include +#include +#include +#include + +namespace MAYAHYDRA_NS_DEF { + +/** + * @brief Converts a Maya matrix to a double precision GfMatrix. + * + * @param[in] mayaMat Maya `MMatrix` to be converted. + * + * @return `GfMatrix4d` equal to \p mayaMat. + */ +inline pxr::GfMatrix4d GetGfMatrixFromMaya(const MMatrix& mayaMat) +{ + pxr::GfMatrix4d mat; + memcpy(mat.GetArray(), mayaMat[0], sizeof(double) * 16); + return mat; +} + +/** + * @brief Converts a Maya float matrix to a double precision GfMatrix. + * + * @param[in] mayaMat Maya `MFloatMatrix` to be converted. + * + * @return `GfMatrix4d` equal to \p mayaMat. + */ +inline pxr::GfMatrix4d GetGfMatrixFromMaya(const MFloatMatrix& mayaMat) +{ + pxr::GfMatrix4d mat; + for (unsigned i = 0; i < 4; ++i) { + for (unsigned j = 0; j < 4; ++j) + mat[i][j] = mayaMat(i, j); + } + return mat; +} + +/** + * @brief Returns the texture file path from a "file" shader node. + * + * @param[in] fileNode "file" shader node. + * + * @return Full path to the texture pointed used by the file node. `` tags are kept intact. + */ +MAYAHYDRALIB_API +pxr::TfToken GetFileTexturePath(const MFnDependencyNode& fileNode); + +/** + * @brief Determines whether or not a given DagPath refers to a shape. + * + * @param[in] dagPath is the DagPath to the potential shape. + * + * @return True if the dagPath refers to a shape, false otherwise. + */ +MAYAHYDRALIB_API +bool IsShape(const MDagPath& dagPath); + +/** + * @brief Converts the given Maya MDagPath \p dagPath into an SdfPath. + * + * Elements of the path will be sanitized such that it is a valid SdfPath. If \p mergeTransformAndShape + * is true and \p dagPath is a shape node, it will return the parent SdfPath of the shape's SdfPath, + * such that the transform and the shape have the same SdfPath. + * + * @param[in] dagPath is the DAG path to convert to an SdfPath. + * @param[in] mergeTransformAndShape determines whether or not to consider the transform and shape + * paths as one. + * @param[in] stripNamespaces determines whether or not to strip namespaces from the path. + * + * @return The SdfPath corresponding to the given DAG path. + */ +MAYAHYDRALIB_API +pxr::SdfPath DagPathToSdfPath( + const MDagPath& dagPath, + const bool mergeTransformAndShape, + const bool stripNamespaces); + +/** + * @brief Creates an SdfPath from the given Maya MRenderItem. + * + * Elements of the path will be sanitized such that it is a valid SdfPath. + * + * @param[in] ri is the MRenderItem to create the SdfPath for. + * @param[in] stripNamespaces determines whether or not to strip namespaces from the path. + * + * @return The SdfPath corresponding to the given MRenderItem. + */ +MAYAHYDRALIB_API +pxr::SdfPath RenderItemToSdfPath(const MRenderItem& ri, const bool stripNamespaces); + +/** + * @brief Retrieves an RGB color preference from Maya. + * + * @param[in] colorName is the color name in Maya to get the color value of. + * @param[out] outColor is the color that will be populated if retrieved from Maya. + * + * @return True if the color retrieval was successful and outColor was populated, false otherwise. + */ +MAYAHYDRALIB_API +bool getRGBAColorPreferenceValue(const std::string& colorName, PXR_NS::GfVec4f& outColor); + +/** + * @brief Retrieves an indexed color preference's index from Maya. + * + * @param[in] colorName is the color name in Maya to get the color index of. + * @param[in] tableName is to indicate for which table/palette we want to get the index. + * @param[out] outIndex is the index value that will be populated if retrieved from Maya. + * + * @return True if the color index retrieval was successful and outIndex was populated, false + * otherwise. + */ +MAYAHYDRALIB_API +bool getIndexedColorPreferenceIndex( + const std::string& colorName, + const std::string& tableName, + size_t& outIndex); + +/** + * @brief Retrieves a palette color from Maya's color settings. + * + * @param[in] tableName is to indicate which table/palette we want to get the color from. + * @param[in] index is to indicate which color to get from the palette. + * @param[out] outColor is the color that will be populated if retrieved from Maya. + * + * @return True if the color retrieval was successful and outColor was populated, false otherwise. + */ +MAYAHYDRALIB_API +bool getColorPreferencesPaletteColor( + const std::string& tableName, + size_t index, + PXR_NS::GfVec4f& outColor); + +/** + * @brief Retrieves an indexed/paletted color preference from Maya. + * + * @param[in] colorName is the color name in Maya to get the color value of. + * @param[in] tableName is to indicate which table/palette we want to get the color from. + * @param[out] outColor is the color that will be populated if retrieved from Maya. + * + * @return True if the color retrieval was successful and outColor was populated, false otherwise. + */ +MAYAHYDRALIB_API +bool getIndexedColorPreferenceValue( + const std::string& colorName, + const std::string& tableName, + PXR_NS::GfVec4f& outColor); + +} // namespace MAYAHYDRA_NS_DEF + +#endif // MAYAHYDRALIB_MIXED_UTILS_H diff --git a/lib/mayaHydra/hydraExtensions/plugInfo.json b/lib/mayaHydra/hydraExtensions/plugInfo.json new file mode 100644 index 0000000000..769e5efd2e --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/plugInfo.json @@ -0,0 +1,137 @@ +{ + "Plugins": [ + { + "Info": { + "Types": { + # Delegates + "MayaHydraDelegate": { + "displayName": "Base type for all scene delegates in Hydra for Maya." + }, + "MayaHydraSceneDelegate": { + "bases": [ + "MayaHydraDelegate" + ], + "displayName": "Maya Shapes in Hydra for Maya." + }, + "MayaHydraProxyDelegate": { + "bases": [ + "MayaHydraDelegate" + ], + "displayName": "Proxy Shapes in Hydra for Maya" + }, + # Adapters + "MayaHydraAdapter": { + "displayName": "Base type for all node adapters in Hydra for Maya." + }, + "MayaHydraRenderItemAdapter": { + "bases": [ + "MayaHydraAdapter" + ], + "displayName": "Render item in Hydra for Maya." + }, + "MayaHydraDagAdapter": { + "bases": [ + "MayaHydraAdapter" + ], + "displayName": "Dag nodes in Hydra for Maya." + }, + "MayaHydraLightAdapter": { + "bases": [ + "MayaHydraDagAdapter" + ], + "displayName": "Lights in Hydra for Maya." + }, + "MayaHydraAreaLightAdapter": { + "bases": [ + "MayaHydraLightAdapter" + ], + "displayName": "Area lights in Hydra for Maya." + }, + "MayaHydraPointLightAdapter": { + "bases": [ + "MayaHydraLightAdapter" + ], + "displayName": "Point lights in Hydra for Maya." + }, + "MayaHydraSpotLightAdapter": { + "bases": [ + "MayaHydraLightAdapter" + ], + "displayName": "Spot lights in Hydra for Maya." + }, + "MayaHydraDirectionalLightAdapter": { + "bases": [ + "MayaHydraLightAdapter" + ], + "displayName": "Directional lights in Hydra for Maya." + }, + "MayaHydraShapeAdapter": { + "bases": [ + "MayaHydraDagAdapter" + ], + "displayName": "Shapes in Hydra for Maya." + }, + "MayaHydraCameraAdapter": { + "bases": [ + "MayaHydraShapeAdapter" + ], + "displayName": "Cameras in Hydra for Maya." + }, + "MayaHydraMeshAdapter": { + "bases": [ + "MayaHydraShapeAdapter" + ], + "displayName": "Meshes in Hydra for Maya." + }, + "MayaHydraNurbsCurveAdapter": { + "bases": [ + "MayaHydraShapeAdapter" + ], + "displayName": "Nurbs Curves in Hydra for Maya." + }, + "MayaHydraImagePlaneAdapter": { + "bases": [ + "MayaHydraShapeAdapter" + ], + "displayName": "ImagePlanes in Hydra for Maya." + }, + "MayaHydraAiSkyDomeLightAdapter": { + "bases": [ + "MayaHydraLightAdapter" + ], + "displayName": "Ai SkyDome Light in Hydra for Maya." + }, + "MayaHydraProxyAdapter": { + "bases": [ + "MayaHydraDagAdapter" + ], + "displayName": "Proxy Shapes in Hydra for Maya." + }, + # Materials + "MayaHydraMaterialAdapter": { + "bases": [ + "MayaHydraAdapter" + ], + "displayName": "Base adapter for materials." + }, + "MayaHydraShadingEngineAdapter": { + "bases": [ + "MayaHydraMaterialAdapter" + ], + "displayName": "Adapter for the shading engine that translates everything to a Preview Surface." + }, + "MayaHydraImagePlaneMaterialAdapter": { + "bases": [ + "MayaHydraMaterialAdapter" + ], + "displayName": "Adapter for the image plane texture." + } + } + }, + "LibraryPath": "@PLUG_INFO_LIBRARY_PATH@", + "Name": "@TARGET_NAME@", + "Root": ".", + "Type": "library" + } + ] +} diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/CMakeLists.txt b/lib/mayaHydra/hydraExtensions/sceneIndex/CMakeLists.txt new file mode 100644 index 0000000000..541d1cc9af --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/CMakeLists.txt @@ -0,0 +1,43 @@ +# ----------------------------------------------------------------------------- +# sources +# ----------------------------------------------------------------------------- +target_sources(${TARGET_NAME} + PRIVATE + registration.cpp + mayaHydraSceneIndex.cpp + mayaHydraDataSource.cpp + mayaHydraPrimvarDataSource.cpp + mayaHydraDisplayStyleDataSource.cpp + mayaHydraCameraDataSource.cpp + mayaHydraLightDataSource.cpp + mayaHydraDefaultLightDataSource.cpp +) + +set(HEADERS + registration.h + mayaHydraSceneIndex.h + mayaHydraDataSource.h + mayaHydraPrimvarDataSource.h + mayaHydraDisplayStyleDataSource.h + mayaHydraCameraDataSource.h + mayaHydraLightDataSource.h + mayaHydraDefaultLightDataSource.h +) + +# ----------------------------------------------------------------------------- +# promoted headers +# ----------------------------------------------------------------------------- +mayaUsd_promoteHeaderList( + HEADERS + ${HEADERS} + BASEDIR + ${TARGET_NAME}/sceneIndex +) + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(FILES ${HEADERS} + DESTINATION + ${CMAKE_INSTALL_PREFIX}/include/mayaHydraLib/sceneIndex +) diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.cpp new file mode 100644 index 0000000000..e33c612b5b --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.cpp @@ -0,0 +1,229 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraCameraDataSource.h" + +#include +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// ---------------------------------------------------------------------------- + +template +class MayaHydraTypedCameraParamValueDataSource : public HdTypedSampledDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraTypedCameraParamValueDataSource); + + MayaHydraTypedCameraParamValueDataSource( + const SdfPath& id, + const TfToken& key, + MayaHydraCameraAdapter* adapter) + : _id(id) + , _key(key) + , _adapter(adapter) + { + } + + bool GetContributingSampleTimesForInterval( + HdSampledDataSource::Time startTime, + HdSampledDataSource::Time endTime, + std::vector* outSampleTimes) override + { + return MayaHydraPrimvarValueDataSource::New( + _key, _adapter)->GetContributingSampleTimesForInterval( + startTime, endTime, outSampleTimes); + } + + T GetTypedValue(HdSampledDataSource::Time shutterOffset) override + { + VtValue v; + if (shutterOffset == 0.0f) { + v = _adapter->GetCameraParamValue(_key); + } + else { + v = MayaHydraPrimvarValueDataSource::New( + _key, _adapter)->GetValue(shutterOffset); + } + + if (v.IsHolding()) { + return v.UncheckedGet(); + } + + return T(); + } + + VtValue GetValue(HdSampledDataSource::Time shutterOffset) override + { + if (shutterOffset == 0.0f) { + return _adapter->GetCameraParamValue(_key); + } + + return VtValue(GetTypedValue(shutterOffset)); + } + +private: + SdfPath _id; + TfToken _key; + MayaHydraCameraAdapter* _adapter; +}; + + +class MayaHydraCameraParamValueDataSource : public HdSampledDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraCameraParamValueDataSource); + + MayaHydraCameraParamValueDataSource( + const SdfPath& id, + const TfToken& key, + MayaHydraCameraAdapter* adapter) + : _id(id) + , _key(key) + , _adapter(adapter) + { + } + + bool GetContributingSampleTimesForInterval( + HdSampledDataSource::Time startTime, + HdSampledDataSource::Time endTime, + std::vector* outSampleTimes) override + { + return MayaHydraPrimvarValueDataSource::New( + _key, _adapter)->GetContributingSampleTimesForInterval( + startTime, endTime, outSampleTimes); + } + + VtValue GetValue(HdSampledDataSource::Time shutterOffset) override + { + if (shutterOffset == 0.0f) { + return _adapter->GetCameraParamValue(_key); + } + + return MayaHydraPrimvarValueDataSource::New( + _key, _adapter)->GetValue(shutterOffset); + } + +private: + SdfPath _id; + TfToken _key; + MayaHydraCameraAdapter* _adapter; +}; + +// ---------------------------------------------------------------------------- + +MayaHydraCameraDataSource::MayaHydraCameraDataSource( + const SdfPath& id, + TfToken type, + MayaHydraAdapter* adapter) + : _id(id) + , _type(type) + , _adapter(adapter) +{ +} + + +TfTokenVector +MayaHydraCameraDataSource::GetNames() +{ + TfTokenVector results; + + results.push_back(HdCameraSchemaTokens->projection); + results.push_back(HdCameraSchemaTokens->horizontalAperture); + results.push_back(HdCameraSchemaTokens->verticalAperture); + results.push_back(HdCameraSchemaTokens->horizontalApertureOffset); + results.push_back(HdCameraSchemaTokens->verticalApertureOffset); + results.push_back(HdCameraSchemaTokens->focalLength); + results.push_back(HdCameraSchemaTokens->clippingRange); + results.push_back(HdCameraSchemaTokens->clippingPlanes); + + return results; + +} + +HdDataSourceBaseHandle +MayaHydraCameraDataSource::Get(const TfToken& name) +{ + MayaHydraCameraAdapter* camAdapter = dynamic_cast(_adapter); + if (!camAdapter) { + return nullptr; + } + + if (name == HdCameraSchemaTokens->projection) { + VtValue v = camAdapter->GetCameraParamValue(name); + + HdCamera::Projection proj = HdCamera::Perspective; + if (v.IsHolding()) { + proj = v.UncheckedGet(); + } + return HdRetainedTypedSampledDataSource::New( + proj == HdCamera::Perspective ? + HdCameraSchemaTokens->perspective : + HdCameraSchemaTokens->orthographic); + } + else if (name == HdCameraSchemaTokens->clippingRange) { + VtValue v = camAdapter->GetCameraParamValue(name); + + GfRange1f range; + if (v.IsHolding()) { + range = v.UncheckedGet(); + } + return HdRetainedTypedSampledDataSource::New( + GfVec2f(range.GetMin(), range.GetMax())); + } + else if (name == HdCameraTokens->windowPolicy) { + VtValue v = camAdapter->GetCameraParamValue(name); + + CameraUtilConformWindowPolicy wp = CameraUtilDontConform; + if (v.IsHolding()) { + wp = v.UncheckedGet(); + } + return HdRetainedTypedSampledDataSource< + CameraUtilConformWindowPolicy>::New(wp); + } + else if (name == HdCameraSchemaTokens->clippingPlanes) { + const VtValue v = camAdapter->GetCameraParamValue(HdCameraTokens->clipPlanes); + VtArray array; + if (v.IsHolding>()) { + const std::vector& vec = + v.UncheckedGet>(); + array.resize(vec.size()); + for (size_t i = 0; i < vec.size(); i++) { + array[i] = vec[i]; + } + } + return HdRetainedTypedSampledDataSource>::New( + array); + } + else if (std::find(HdCameraSchemaTokens->allTokens.begin(), + HdCameraSchemaTokens->allTokens.end(), name) + != HdCameraSchemaTokens->allTokens.end()) { + // all remaining HdCameraSchema members are floats and should + // be returned as a typed data source for schema conformance. + return MayaHydraTypedCameraParamValueDataSource::New( + _id, name, camAdapter); + } + else { + return MayaHydraCameraParamValueDataSource::New( + _id, name, camAdapter); + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.h b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.h new file mode 100644 index 0000000000..0ea97eaae3 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraCameraDataSource.h @@ -0,0 +1,56 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRACAMERADATASOURCE_H +#define MAYAHYDRACAMERADATASOURCE_H + +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraAdapter; +/** + * \brief A container data source representing data unique to camera + */ + class MayaHydraCameraDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraCameraDataSource); + + // ------------------------------------------------------------------------ + // HdContainerDataSource implementations + TfTokenVector GetNames() override; + HdDataSourceBaseHandle Get(const TfToken& name) override; + +private: + MayaHydraCameraDataSource( + const SdfPath& id, + TfToken type, + MayaHydraAdapter* adapter); + + SdfPath _id; + TfToken _type; + MayaHydraAdapter* _adapter = nullptr; +}; + +HD_DECLARE_DATASOURCE_HANDLES(MayaHydraCameraDataSource); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRACAMERADATASOURCE_H diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.cpp new file mode 100644 index 0000000000..f18b37291a --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.cpp @@ -0,0 +1,424 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraDataSource.h" + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + + +MayaHydraDataSource::MayaHydraDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex, + MayaHydraAdapter* adapter) + : _id(id) + , _type(type) + , _sceneIndex(sceneIndex) + , _adapter(adapter) +{ +} + +TfTokenVector +MayaHydraDataSource::GetNames() +{ + TfTokenVector result; + + if (_type == HdPrimTypeTokens->mesh) { + result.push_back(HdMeshSchemaTokens->mesh); + } + + if (_type == HdPrimTypeTokens->basisCurves) { + result.push_back(HdBasisCurvesSchemaTokens->basisCurves); + } + + result.push_back(HdPrimvarsSchemaTokens->primvars); + + if (HdPrimTypeIsGprim(_type)) { + result.push_back(HdMaterialBindingsSchema::GetSchemaToken()); + result.push_back(HdLegacyDisplayStyleSchemaTokens->displayStyle); + result.push_back(HdVisibilitySchemaTokens->visibility); + result.push_back(HdXformSchemaTokens->xform); + } + + if (HdPrimTypeIsLight(_type)) { + result.push_back(HdMaterialSchemaTokens->material); + result.push_back(HdXformSchemaTokens->xform); + result.push_back(HdLightSchemaTokens->light); + } + + if (_type == HdPrimTypeTokens->material) { + result.push_back(HdMaterialSchemaTokens->material); + } + + if (_type == HdPrimTypeTokens->camera) { + result.push_back(HdCameraSchemaTokens->camera); + result.push_back(HdXformSchemaTokens->xform); + } + + return result; +} + +HdDataSourceBaseHandle +MayaHydraDataSource::Get(const TfToken& name) +{ + if (name == HdMeshSchemaTokens->mesh) { + if (_type == HdPrimTypeTokens->mesh) { + auto topology = _adapter->GetMeshTopology(); + return HdMeshSchema::Builder() + .SetTopology( + HdMeshTopologySchema::Builder() + .SetFaceVertexCounts( + HdRetainedTypedSampledDataSource::New( + topology.GetFaceVertexCounts())) + .SetFaceVertexIndices( + HdRetainedTypedSampledDataSource::New( + topology.GetFaceVertexIndices())) + .SetOrientation( + HdRetainedTypedSampledDataSource::New( + HdMeshTopologySchemaTokens->rightHanded)) + .Build()) + .SetSubdivisionScheme( + HdRetainedTypedSampledDataSource::New(topology.GetScheme())) + .SetDoubleSided( + HdRetainedTypedSampledDataSource::New(_adapter->GetDoubleSided())) + .Build(); + } + } + else if (name == HdBasisCurvesSchemaTokens->basisCurves) { + if (_type == HdPrimTypeTokens->basisCurves) { + auto topology = _adapter->GetBasisCurvesTopology(); + return HdBasisCurvesSchema::Builder() + .SetTopology( + HdBasisCurvesTopologySchema::Builder() + .SetCurveVertexCounts( + HdRetainedTypedSampledDataSource::New( + topology.GetCurveVertexCounts())) + .SetCurveIndices( + HdRetainedTypedSampledDataSource::New( + topology.GetCurveIndices())) + .SetBasis( + HdRetainedTypedSampledDataSource::New( + topology.GetCurveBasis())) + .SetType( + HdRetainedTypedSampledDataSource::New( + topology.GetCurveType())) + .SetWrap( + HdRetainedTypedSampledDataSource::New( + topology.GetCurveWrap())) + .Build()) + .Build(); + } + } + else if (name == HdPrimvarsSchemaTokens->primvars) { + return _GetPrimvarsDataSource(); + } + else if (name == + HdMaterialBindingsSchema::GetSchemaToken() + ) { + return _GetMaterialBindingDataSource(); + } + else if (name == HdXformSchemaTokens->xform) { + auto xform = _adapter->GetTransform(); + return HdXformSchema::Builder() + .SetMatrix( + HdRetainedTypedSampledDataSource::New(xform)) + .Build(); + } + else if (name == HdMaterialSchemaTokens->material) { + return _GetMaterialDataSource(); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->displayStyle) { + return MayaHydraDisplayStyleDataSource::New(_id, _type, _sceneIndex, _adapter); + } + else if (name == HdVisibilitySchemaTokens->visibility) { + return _GetVisibilityDataSource(); + } + else if (name == HdCameraSchemaTokens->camera) { + return MayaHydraCameraDataSource::New(_id, _type, _adapter); + } + else if (name == HdLightSchemaTokens->light) { + return MayaHydraLightDataSource::New(_id, _type, _adapter); + } + + return nullptr; +} + +HdDataSourceBaseHandle MayaHydraDataSource::_GetVisibilityDataSource() +{ + bool vis = _adapter->GetVisible(); + if (vis) { + static const HdContainerDataSourceHandle visOn = + HdVisibilitySchema::BuildRetained( + HdRetainedTypedSampledDataSource::New(true)); + return visOn; + } + else { + static const HdContainerDataSourceHandle visOff = + HdVisibilitySchema::BuildRetained( + HdRetainedTypedSampledDataSource::New(false)); + return visOff; + } +} + +HdDataSourceBaseHandle MayaHydraDataSource::_GetPrimvarsDataSource() +{ + if (_primvarsBuilt.load()) { + return HdContainerDataSource::AtomicLoad(_primvars); + } + + MayaHydraPrimvarsDataSourceHandle primvarsDs; + + for (size_t interpolation = HdInterpolationConstant; + interpolation < HdInterpolationCount; ++interpolation) { + + HdPrimvarDescriptorVector v = _adapter->GetPrimvarDescriptors((HdInterpolation)interpolation); + + TfToken interpolationToken = _InterpolationAsToken( + (HdInterpolation)interpolation); + + for (const auto& primvarDesc : v) { + if (!primvarsDs) { + primvarsDs = MayaHydraPrimvarsDataSource::New(_adapter); + } + primvarsDs->AddDesc( + primvarDesc.name, interpolationToken, primvarDesc.role, + primvarDesc.indexed); + } + } + + HdContainerDataSourceHandle ds = primvarsDs; + HdContainerDataSource::AtomicStore(_primvars, ds); + _primvarsBuilt.store(true); + + return primvarsDs; +} + +TfToken MayaHydraDataSource::_InterpolationAsToken(HdInterpolation interpolation) +{ + switch (interpolation) { + case HdInterpolationConstant: + return HdPrimvarSchemaTokens->constant; + case HdInterpolationUniform: + return HdPrimvarSchemaTokens->uniform; + case HdInterpolationVarying: + return HdPrimvarSchemaTokens->varying; + case HdInterpolationVertex: + return HdPrimvarSchemaTokens->vertex; + case HdInterpolationFaceVarying: + return HdPrimvarSchemaTokens->faceVarying; + case HdInterpolationInstance: + return HdPrimvarSchemaTokens->instance; + + default: + return HdPrimvarSchemaTokens->constant; + } +} + +HdDataSourceBaseHandle +MayaHydraDataSource::_GetMaterialBindingDataSource() +{ + const SdfPath path = _sceneIndex->GetMaterialId(_id); + if (path.IsEmpty()) { + return nullptr; + } + static const TfToken purposes[] = { + HdMaterialBindingsSchemaTokens->allPurpose + }; + HdDataSourceBaseHandle const materialBindingSources[] = { + HdMaterialBindingSchema::Builder() + .SetPath( + HdRetainedTypedSampledDataSource::New(path)) + .Build() + }; + + return + HdMaterialBindingsSchema::BuildRetained( + TfArraySize(purposes), purposes, materialBindingSources); +} + +static bool +_ConvertHdMaterialNetworkToHdDataSources( + const HdMaterialNetworkMap &hdNetworkMap, + HdContainerDataSourceHandle *result) +{ + HD_TRACE_FUNCTION(); + + TfTokenVector terminalsNames; + std::vector terminalsValues; + std::vector nodeNames; + std::vector nodeValues; + + for (auto const &iter: hdNetworkMap.map) { + const TfToken &terminalName = iter.first; + const HdMaterialNetwork &hdNetwork = iter.second; + + if (hdNetwork.nodes.empty()) { + continue; + } + + terminalsNames.push_back(terminalName); + + // Transfer over individual nodes. + // Note that the same nodes may be shared by multiple terminals. + // We simply overwrite them here. + for (const HdMaterialNode &node : hdNetwork.nodes) { + std::vector paramsNames; + std::vector paramsValues; + + for (const auto &p : node.parameters) { + paramsNames.push_back(p.first); + paramsValues.push_back( + HdRetainedTypedSampledDataSource::New(p.second) + ); + } + + // Accumulate array connections to the same input + TfDenseHashMap, TfToken::HashFunctor> + connectionsMap; + + TfSmallVector cNames; + TfSmallVector cValues; + + for (const HdMaterialRelationship &rel : hdNetwork.relationships) { + if (rel.outputId == node.path) { + TfToken outputPath = rel.inputId.GetToken(); + TfToken outputName = TfToken(rel.inputName.GetString()); + + HdDataSourceBaseHandle c = + HdMaterialConnectionSchema::BuildRetained( + HdRetainedTypedSampledDataSource::New( + outputPath), + HdRetainedTypedSampledDataSource::New( + outputName)); + + connectionsMap[ + TfToken(rel.outputName.GetString())].push_back(c); + } + } + + cNames.reserve(connectionsMap.size()); + cValues.reserve(connectionsMap.size()); + + // NOTE: not const because HdRetainedSmallVectorDataSource needs + // a non-const HdDataSourceBaseHandle* + for (auto &entryPair : connectionsMap) { + cNames.push_back(entryPair.first); + cValues.push_back( + HdRetainedSmallVectorDataSource::New( + entryPair.second.size(), entryPair.second.data())); + } + + nodeNames.push_back(node.path.GetToken()); + nodeValues.push_back( + HdMaterialNodeSchema::BuildRetained( + HdRetainedContainerDataSource::New( + paramsNames.size(), + paramsNames.data(), + paramsValues.data()), + HdRetainedContainerDataSource::New( + cNames.size(), + cNames.data(), + cValues.data()), + HdRetainedTypedSampledDataSource::New( + node.identifier), + nullptr /*renderContextNodeIdentifiers*/ + , nullptr /* nodeTypeInfo */ + )); + } + + terminalsValues.push_back( + HdMaterialConnectionSchema::BuildRetained( + HdRetainedTypedSampledDataSource::New( + hdNetwork.nodes.back().path.GetToken()), + HdRetainedTypedSampledDataSource::New( + terminalsNames.back())) + ); + } + + HdContainerDataSourceHandle nodesDefaultContext = + HdRetainedContainerDataSource::New( + nodeNames.size(), + nodeNames.data(), + nodeValues.data()); + + HdContainerDataSourceHandle terminalsDefaultContext = + HdRetainedContainerDataSource::New( + terminalsNames.size(), + terminalsNames.data(), + terminalsValues.data()); + + // Create the material network, potentially one per network selector + HdDataSourceBaseHandle network = HdMaterialNetworkSchema::BuildRetained( + nodesDefaultContext, + terminalsDefaultContext); + + TfToken defaultContext = HdMaterialSchemaTokens->universalRenderContext; + *result = HdMaterialSchema::BuildRetained( + 1, + &defaultContext, + &network); + + return true; +} + +HdDataSourceBaseHandle +MayaHydraDataSource::_GetMaterialDataSource() +{ + VtValue materialContainer = _sceneIndex->GetMaterialResource(_id); + + if (!materialContainer.IsHolding()) { + return nullptr; + } + + HdMaterialNetworkMap hdNetworkMap = + materialContainer.UncheckedGet(); + HdContainerDataSourceHandle materialDS = nullptr; + if (!_ConvertHdMaterialNetworkToHdDataSources( + hdNetworkMap, + &materialDS) ) { + return nullptr; + } + return materialDS; +} +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.h b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.h new file mode 100644 index 0000000000..8f6653f1d9 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDataSource.h @@ -0,0 +1,69 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRADATASOURCE_H +#define MAYAHYDRADATASOURCE_H + +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraAdapter; +class MayaHydraSceneIndex; + +/** + * \brief A container data source representing data unique to render item + */ + class MayaHydraDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraDataSource); + + // ------------------------------------------------------------------------ + // HdContainerDataSource implementations + TfTokenVector GetNames() override; + HdDataSourceBaseHandle Get(const TfToken& name) override; + +private: + MayaHydraDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex, + MayaHydraAdapter* adapter); + + HdDataSourceBaseHandle _GetVisibilityDataSource(); + HdDataSourceBaseHandle _GetPrimvarsDataSource(); + TfToken _InterpolationAsToken(HdInterpolation interpolation); + HdDataSourceBaseHandle _GetMaterialBindingDataSource(); + HdDataSourceBaseHandle _GetMaterialDataSource(); +private: + SdfPath _id; + TfToken _type; + MayaHydraSceneIndex* _sceneIndex = nullptr; + MayaHydraAdapter* _adapter = nullptr; + + std::atomic_bool _primvarsBuilt{false}; + HdContainerDataSourceAtomicHandle _primvars; +}; + +HD_DECLARE_DATASOURCE_HANDLES(MayaHydraDataSource); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRADATASOURCE_H diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.cpp new file mode 100644 index 0000000000..09cfbe1fe9 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.cpp @@ -0,0 +1,177 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraDefaultLightDataSource.h" + +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +// ---------------------------------------------------------------------------- + +class MayaHydraSimpleLightDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraSimpleLightDataSource); + + MayaHydraSimpleLightDataSource( + const SdfPath& id, + MayaHydraSceneIndex* sceneIndex) + : _id(id) + , _sceneIndex(sceneIndex) + { + } + + TfTokenVector GetNames() override + { + TfTokenVector result = { + HdTokens->filters, + HdTokens->lightLink, + HdTokens->shadowLink, + HdTokens->lightFilterLink, + HdTokens->isLight + }; + return result; + } + + HdDataSourceBaseHandle Get(const TfToken& name) override + { + VtValue v; + if (name == HdLightTokens->params) { + v = VtValue(_sceneIndex->GetDefaultLight()); + } + else if (name == HdTokens->transform) { + v = VtValue(GfMatrix4d( + 1.0)); // We don't use the transform but use the position param of the GlfsimpleLight + // Hydra might crash when this is an empty VtValue. + } + else if (name == HdLightTokens->shadowCollection) { + // Exclude lines/points primitives from casting shadows by only taking the primitives + // whose root path belongs to LightedPrimsRootPath + HdRprimCollection coll(HdTokens->geometry, HdReprSelector(HdReprTokens->refined)); + coll.SetRootPaths({ _sceneIndex->GetLightedPrimsRootPath() }); + v = VtValue(coll); + } + else if (name == HdLightTokens->shadowParams) { + HdxShadowParams shadowParams; + shadowParams.enabled = false; + v = VtValue(shadowParams); + } + else + { + v = _GetLightParamValue(name); + } + + // Wrap as data source + if (name == HdLightTokens->params) { + return HdRetainedSampledDataSource::New(v); + } + else if (name == HdLightTokens->shadowParams) { + return HdRetainedSampledDataSource::New(v); + } + else if (name == HdLightTokens->shadowCollection) { + return HdRetainedSampledDataSource::New(v); + } + else { + return HdCreateTypedRetainedDataSource(v); + } + } + + VtValue _GetLightParamValue(const TfToken& paramName) + { + if (paramName == HdLightTokens->color || paramName == HdTokens->displayColor) { + const auto diffuse = _sceneIndex->GetDefaultLight().GetDiffuse(); + return VtValue(GfVec3f(diffuse[0], diffuse[1], diffuse[2])); + } + else if (paramName == HdLightTokens->intensity) { + return VtValue(1.0f); + } + else if (paramName == HdLightTokens->diffuse) { + return VtValue(1.0f); + } + else if (paramName == HdLightTokens->specular) { + return VtValue(0.0f); + } + else if (paramName == HdLightTokens->exposure) { + return VtValue(0.0f); + } + else if (paramName == HdLightTokens->normalize) { + return VtValue(true); + } + else if (paramName == HdLightTokens->angle) { + return VtValue(0.0f); + } + else if (paramName == HdLightTokens->shadowEnable) { + return VtValue(false); + } + else if (paramName == HdLightTokens->shadowColor) { + return VtValue(GfVec3f(0.0f, 0.0f, 0.0f)); + } + else if (paramName == HdLightTokens->enableColorTemperature) { + return VtValue(false); + } + return {}; + } + +private: + SdfPath _id; + MayaHydraSceneIndex* _sceneIndex = nullptr; +}; + +// ---------------------------------------------------------------------------- + +MayaHydraDefaultLightDataSource::MayaHydraDefaultLightDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex) + : _id(id) + , _type(type) + , _sceneIndex(sceneIndex) +{ +} + + +TfTokenVector +MayaHydraDefaultLightDataSource::GetNames() +{ + TfTokenVector result = { + HdXformSchemaTokens->xform, + HdLightSchemaTokens->light, + }; + return result; +} + +HdDataSourceBaseHandle +MayaHydraDefaultLightDataSource::Get(const TfToken& name) +{ + if (name == HdLightSchemaTokens->light) { + return MayaHydraSimpleLightDataSource::New(_id, _sceneIndex); + } + else if (name == HdXformSchemaTokens->xform) { + auto xform = GfMatrix4d(1.0); + return HdXformSchema::Builder() + .SetMatrix( + HdRetainedTypedSampledDataSource::New(xform)) + .Build(); + } + return nullptr; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.h b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.h new file mode 100644 index 0000000000..fc89b5d4a3 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDefaultLightDataSource.h @@ -0,0 +1,62 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRADEFAULTLIGHTDATASOURCE_H +#define MAYAHYDRADEFAULTLIGHTDATASOURCE_H + +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraSceneIndex; + +/** + * \brief A container data source representing data unique to light + */ + class MayaHydraDefaultLightDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraDefaultLightDataSource); + + // ------------------------------------------------------------------------ + // HdContainerDataSource implementations + TfTokenVector GetNames() override; + HdDataSourceBaseHandle Get(const TfToken& name) override; + +private: + MayaHydraDefaultLightDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex); + + VtValue _GetLightParamValue(const TfToken& paramName); + + SdfPath _id; + TfToken _type; + + const GlfSimpleLight* _light = nullptr; + MayaHydraSceneIndex* _sceneIndex = nullptr; +}; + +HD_DECLARE_DATASOURCE_HANDLES(MayaHydraDefaultLightDataSource); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRADEFAULTLIGHTDATASOURCE_H diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.cpp new file mode 100644 index 0000000000..dead0b832d --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.cpp @@ -0,0 +1,184 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraDisplayStyleDataSource.h" + +#include +#include + +#include "pxr/imaging/hd/legacyDisplayStyleSchema.h" +#include + +PXR_NAMESPACE_OPEN_SCOPE + +MayaHydraDisplayStyleDataSource::MayaHydraDisplayStyleDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex, + MayaHydraAdapter* adapter) + : _id(id) + , _type(type) + , _sceneIndex(sceneIndex) + , _adapter(adapter) + , _displayStyleRead(false) +{ +} + + +TfTokenVector +MayaHydraDisplayStyleDataSource::GetNames() +{ + TfTokenVector results; + results.push_back(HdLegacyDisplayStyleSchemaTokens->refineLevel); + results.push_back(HdLegacyDisplayStyleSchemaTokens->flatShadingEnabled); + results.push_back(HdLegacyDisplayStyleSchemaTokens->displacementEnabled); + results.push_back(HdLegacyDisplayStyleSchemaTokens->occludedSelectionShowsThrough); + results.push_back(HdLegacyDisplayStyleSchemaTokens->pointsShadingEnabled); + results.push_back(HdLegacyDisplayStyleSchemaTokens->materialIsFinal); + results.push_back(HdLegacyDisplayStyleSchemaTokens->shadingStyle); + results.push_back(HdLegacyDisplayStyleSchemaTokens->reprSelector); + results.push_back(HdLegacyDisplayStyleSchemaTokens->cullStyle); + return results; +} + +HdDataSourceBaseHandle +MayaHydraDisplayStyleDataSource::Get(const TfToken& name) +{ + if (name == HdLegacyDisplayStyleSchemaTokens->refineLevel) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return (_displayStyle.refineLevel != 0) + ? HdRetainedTypedSampledDataSource::New( + _displayStyle.refineLevel) + : nullptr; + } + else if (name == HdLegacyDisplayStyleSchemaTokens->flatShadingEnabled) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return HdRetainedTypedSampledDataSource::New( + _displayStyle.flatShadingEnabled); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->displacementEnabled) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return HdRetainedTypedSampledDataSource::New( + _displayStyle.displacementEnabled); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->occludedSelectionShowsThrough) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return HdRetainedTypedSampledDataSource::New( + _displayStyle.occludedSelectionShowsThrough); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->pointsShadingEnabled) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return HdRetainedTypedSampledDataSource::New( + _displayStyle.pointsShadingEnabled); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->materialIsFinal) { + if (!_displayStyleRead) { + _displayStyle = _adapter->GetDisplayStyle(); + _displayStyleRead = true; + } + return HdRetainedTypedSampledDataSource::New( + _displayStyle.materialIsFinal); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->shadingStyle) { + TfToken shadingStyle = _sceneIndex->GetShadingStyle(_id) + .GetWithDefault(); + if (shadingStyle.IsEmpty()) { + return nullptr; + } + return HdRetainedTypedSampledDataSource::New(shadingStyle); + } + else if (name == HdLegacyDisplayStyleSchemaTokens->reprSelector) { + HdReprSelector repr; + // TODO: Get the reprSelector + //HdSceneIndexPrim prim = _sceneIndex->GetPrim(_id); + //if (HdLegacyDisplayStyleSchema styleSchema = + // HdLegacyDisplayStyleSchema::GetFromParent(prim.dataSource)) { + + // if (HdTokenArrayDataSourceHandle ds = + // styleSchema.GetReprSelector()) { + // VtArray ar = ds->GetTypedValue(0.0f); + // ar.resize(HdReprSelector::MAX_TOPOLOGY_REPRS); + // repr = HdReprSelector(ar[0], ar[1], ar[2]); + // } + //} + HdTokenArrayDataSourceHandle reprSelectorDs = nullptr; + bool empty = true; + for (size_t i = 0; i < HdReprSelector::MAX_TOPOLOGY_REPRS; ++i) { + if (!repr[i].IsEmpty()) { + empty = false; + break; + } + } + if (!empty) { + VtArray array(HdReprSelector::MAX_TOPOLOGY_REPRS); + for (size_t i = 0; i < HdReprSelector::MAX_TOPOLOGY_REPRS; ++i) { + array[i] = repr[i]; + } + reprSelectorDs = + HdRetainedTypedSampledDataSource>::New( + array); + } + return reprSelectorDs; + } + else if (name == HdLegacyDisplayStyleSchemaTokens->cullStyle) { + HdCullStyle cullStyle = _adapter->GetCullStyle(); + if (cullStyle == HdCullStyleDontCare) { + return nullptr; + } + TfToken cullStyleToken; + switch (cullStyle) { + case HdCullStyleNothing: + cullStyleToken = HdCullStyleTokens->nothing; + break; + case HdCullStyleBack: + cullStyleToken = HdCullStyleTokens->back; + break; + case HdCullStyleFront: + cullStyleToken = HdCullStyleTokens->front; + break; + case HdCullStyleBackUnlessDoubleSided: + cullStyleToken = HdCullStyleTokens->backUnlessDoubleSided; + break; + case HdCullStyleFrontUnlessDoubleSided: + cullStyleToken = HdCullStyleTokens->frontUnlessDoubleSided; + break; + default: + cullStyleToken = HdCullStyleTokens->dontCare; + break; + } + return HdRetainedTypedSampledDataSource::New(cullStyleToken); + } + else { + return nullptr; + } +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.h b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.h new file mode 100644 index 0000000000..e0104418fd --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraDisplayStyleDataSource.h @@ -0,0 +1,64 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRADISPLAYSTYLEDATASOURCE_H +#define MAYAHYDRADISPLAYSTYLEDATASOURCE_H + +#include +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraAdapter; +class MayaHydraSceneIndex; + +/** + * \brief A container data source representing data unique to display style + */ + class MayaHydraDisplayStyleDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraDisplayStyleDataSource); + + // ------------------------------------------------------------------------ + // HdContainerDataSource implementations + TfTokenVector GetNames() override; + HdDataSourceBaseHandle Get(const TfToken& name) override; + +private: + MayaHydraDisplayStyleDataSource( + const SdfPath& id, + TfToken type, + MayaHydraSceneIndex* sceneIndex, + MayaHydraAdapter* adapter); + + SdfPath _id; + TfToken _type; + MayaHydraSceneIndex* _sceneIndex = nullptr; + MayaHydraAdapter* _adapter = nullptr; + + HdDisplayStyle _displayStyle; + bool _displayStyleRead; +}; + +HD_DECLARE_DATASOURCE_HANDLES(MayaHydraDisplayStyleDataSource); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRADISPLAYSTYLEDATASOURCE_H diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.cpp new file mode 100644 index 0000000000..f2d1f812d7 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.cpp @@ -0,0 +1,89 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraLightDataSource.h" + +#include + +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +MayaHydraLightDataSource::MayaHydraLightDataSource( + const SdfPath& id, + TfToken type, + MayaHydraAdapter* adapter) + : _id(id) + , _type(type) + , _adapter(adapter) +{ +} + + +TfTokenVector +MayaHydraLightDataSource::GetNames() +{ + TfTokenVector result = { + HdTokens->filters, + HdTokens->lightLink, + HdTokens->shadowLink, + HdTokens->lightFilterLink, + HdTokens->isLight, + }; + return result; +} + +HdDataSourceBaseHandle +MayaHydraLightDataSource::Get(const TfToken& name) +{ + VtValue v; + if (_UseGet(name)) { + v = _adapter->Get(name); + } + else { + MayaHydraLightAdapter* lightAdapter = dynamic_cast(_adapter); + if (!lightAdapter) { + return nullptr; + } + v = lightAdapter->GetLightParamValue(name); + } + + if (name == HdLightTokens->params) { + return HdRetainedSampledDataSource::New(v); + } + else if (name == HdLightTokens->shadowParams) { + return HdRetainedSampledDataSource::New(v); + } + else if (name == HdLightTokens->shadowCollection) { + return HdRetainedSampledDataSource::New(v); + } + else { + return HdCreateTypedRetainedDataSource(v); + } +} + +bool MayaHydraLightDataSource::_UseGet(const TfToken& name) const { + if (name == HdLightTokens->params || + name == HdLightTokens->shadowParams || + name == HdLightTokens->shadowCollection) { + return true; + } + + return false; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.h b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.h new file mode 100644 index 0000000000..30052db8b0 --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraLightDataSource.h @@ -0,0 +1,58 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRALIGHTDATASOURCE_H +#define MAYAHYDRALIGHTDATASOURCE_H + +#include +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +class MayaHydraAdapter; +/** + * \brief A container data source representing data unique to light + */ + class MayaHydraLightDataSource : public HdContainerDataSource +{ +public: + HD_DECLARE_DATASOURCE(MayaHydraLightDataSource); + + // ------------------------------------------------------------------------ + // HdContainerDataSource implementations + TfTokenVector GetNames() override; + HdDataSourceBaseHandle Get(const TfToken& name) override; + +private: + MayaHydraLightDataSource( + const SdfPath& id, + TfToken type, + MayaHydraAdapter* adapter); + + bool _UseGet(const TfToken& name) const; + + SdfPath _id; + TfToken _type; + MayaHydraAdapter* _adapter = nullptr; +}; + +HD_DECLARE_DATASOURCE_HANDLES(MayaHydraLightDataSource); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRALIGHTDATASOURCE_H diff --git a/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraPrimvarDataSource.cpp b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraPrimvarDataSource.cpp new file mode 100644 index 0000000000..d35e5fb64f --- /dev/null +++ b/lib/mayaHydra/hydraExtensions/sceneIndex/mayaHydraPrimvarDataSource.cpp @@ -0,0 +1,90 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "mayaHydraPrimvarDataSource.h" + +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +MayaHydraPrimvarsDataSource::MayaHydraPrimvarsDataSource( + MayaHydraAdapter* adapter) + : _adapter(adapter) +{ +} + +void MayaHydraPrimvarsDataSource::AddDesc( + const TfToken& name, + const TfToken& interpolation, + const TfToken& role, bool indexed) +{ + _entries[name] = { interpolation, role, indexed }; +} + +TfTokenVector MayaHydraPrimvarsDataSource::GetNames() +{ + TfTokenVector result; + result.reserve(_entries.size()); + for (const auto& pair : _entries) { + result.push_back(pair.first); + } + return result; +} + +HdDataSourceBaseHandle MayaHydraPrimvarsDataSource::Get(const TfToken& name) +{ + _EntryMap::const_iterator it = _entries.find(name); + if (it == _entries.end()) { + return nullptr; + } + + // Need to handle indexed case? + assert(!(*it).second.indexed); + return HdPrimvarSchema::Builder() + .SetPrimvarValue(MayaHydraPrimvarValueDataSource::New( + name, _adapter)) + .SetInterpolation(HdPrimvarSchema::BuildInterpolationDataSource( + (*it).second.interpolation)) + .SetRole(HdPrimvarSchema::BuildRoleDataSource( + (*it).second.role)) + .Build(); +} + +MayaHydraPrimvarValueDataSource::MayaHydraPrimvarValueDataSource( + const TfToken& primvarName, + MayaHydraAdapter* adapter) + : _primvarName(primvarName) + , _adapter(adapter) +{ +} + +VtValue MayaHydraPrimvarValueDataSource::GetValue(Time shutterOffset) +{ + return _adapter->Get(_primvarName); +} + +bool MayaHydraPrimvarValueDataSource::GetContributingSampleTimesForInterval( + Time startTime, Time endTime, + std::vector

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

6IZ$0rZ?i5HbGnI(X|j5OM9A|6!oRp*Haw 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�SI5 z=@^Fir~^PkJe)_egnLffUj?4lNv5Nm+@mY9XDc#h4k)NDsMQ9siF*!)-m~>JS{E#W z-8n}xD>=grplKgjpJ@*w+mukWZg<#~9c1BwG~Nz1_^9nA2<^$a>E`zPw_iVd@4e^V zdic$YZ=QLFiAX_W7G=4tX%zMeoKDwW^_=JBO*rP{aT+X+t3%UKCzj|$aA?>$IV>m$ z?oldmpr6gB+m{|eL&yF7@&m7#{piO5i+%+L zQ!$&4s|x=Z-_GJ|BevNu!dPsUK_*U9K-5Kh@<|MK`m94CE=*q)Fu;Zm5rbh0`pi4$3$!Hx{@X1X_!01dXn=bTu;zrc;v7K5t z51us=OrTdsB@p3_UikDnJX6LUZ?d9tyGftW;VdV^CR>KcHbUj`DPEu@(PdZKdKQ%O z;p@*mK4k1;VOczpD4iEVosrTW>9JQL1AAfziIEpUS?4*4_$8sP{nlV7L^8nF>ql=? z|K7-hoTCNR;PEk|7CN4YVt*YXq4sa^$TZzVZgq5u3;;HAAXA!)D{{q4Y>Ek|b78+H zsdyNvEXGCi;`~67P0(UV3%gBqIQZHc>pGh3@S{&YiZF&i@F_1s1U7;=JGS~|M{Z60Nq2%2ao>q^9gCQWmMLWA9Xq*m$Y*FZT#2B1%!9}A2Bb{Xl z93)&suxvBKaYNmvKm_RUni@LmKV(A|Rjw%=TWqG|1w4*|E>ua)=ut%mkLU0UkbCo5 zG$3gPoeU?_F5dOaq{*I+HX5WPFc&D2e6NA}5aS!AC&^G$WB-|vhgS$yz|Uq1iEhadhnHoKIZ z3ve4nSG=i1)z}O8HQVSN+>Yn8uE+a}I=FR!YZGm=jmkw&gvGNp0}IwSHIB;ZagXL# zH<_Vp!7EPbkWJ7YHSN(e`LZqx@>xUp5H;JeH9Q*@Xm)Q`*Nq4~U4iELo9?QDWT4rh zead8ND`#6|e{7A6*p_EuEs+{!8qnB^(>2_+MZ;v2ucR3A%v_Go1Pdz4L^7e`VCQ`w ze0)=&lYw9!O>98IRpTutk34h_WH|X4cmb&WHdRpeSO&Yc*kD&NxlV8wY6ef&w^=0J z$l!2CID3WdvZwqfU)_^%YmX=Vkq;V79q&1AU?w-Va;E;PzfS9Bi~2Y5bMyo?-j5CL zY0IOt@*30q9z0u_$|NosUlNA82?~ky67Rt|PuZw0gG}%1V9=GZnNq*{|NQC`)h5W3 zd110&G=;#P^<~`zInt4*X9BSDfEmSUV6kP2ic*>RhO+`x#T(7|MQ~2Gpj5$nIl->~ z*Rxtk>&U?Ks4rBDFl1P=z}k8roJ6)u;yBOzx~gASDtK$n8qFD*s)tDAab zbXSh14m-|C(kR1S0E-Ag#MJoEG3w~L4LUhJbXec*Vyg;`U8iL2?4)v6y1->>NVY1V zfM3Hoxq~THutno!cwn8~6(MH;Xbt9G^%|X~lonWmMXrf=un-~x6mU6~9XTgkHdC8TH{p>`8GjAexkTR{ z8-dSJ8c$>r!+2X@N$_VBn+#tz3JIQ302)qquiXf*UivCwS;BzE>b?Eu+h6|h`|o`9 z&E1P<|M|_Io~?cOWV&?0pmbS!3Zv&Ib{++5Hg(bCbRP=AkRRalOeyBX{n8f>UcC^x z!tXr%@4o%cm;cZI{eOMp*=3L2F}MZz9CS7Xdk6=u*UVd>)0O6C5s}s)|RQC>aU(8!4o- zbA0>=+9%OyY8rW4!>T^8;siXSN?}La2^kIKtCJmi<#6?#K-^5BELoYc7{`QExfl)X zcJkpf8IHaH@kdwp3X9c0y4*9?x<*f4$|r;0RAfA`akk3LZ0BZAyLBN%l2Yr5U~HFJ zx=C%}J2ICHFHFOqZeLV;Yj@$<=9ny)C9ja>4qmvEC`m?xeN`w(Xyeu}<#D_?V zy~>ao8mF|_xE>v|zF_T$@|_={Z%!~69Cy1=I#+2!$mogJ$Vuoa5%L8V3Sdl}NacKYiE_93CnWRVraQtCkeR_TxTx~)DZ>fHaZK-Rgf4hQ;i(%RwS)uRHM zohNJlbDCS$QQNVyomTzZhmaXr?F2UKk=ohI zE~qS_%dyU|o?+stCQHMrRNK+k)m@p)ItMoTgM1tWZ(Sqj>yUVy#)4#UNVje#vN;n$ zz(sCktj*zVatP14Ark+{Ke|II=f~E9QT(q{sbibmW|T~HqDqcI%bs4BObA&s;z)o? zg_keFCl+DWAMDqXm1vw$RqI-Mynf>^zW(`-e)>iDAOHLB|M>aj84W-oQUG$efRXG4 z$MNXfufSn0AaLy@J0olCv(ZhR(u=n6u%TH& z*F4;hEl%boG{`vA6I}J_G`h$kpWwjfRl%qIMZ~MKBrjNd!=Vl@dsN3x%YOPZ0y0$t z9a~MKU`5k{X?6G-9~GRyPN2a$Q%qB@05YB84Q=X?*|#|Yn!(CntAFl^2)@`bc^~}_ zki;Qdz9o~w|dkG3lT8&3Ar zk>ibElLcxnq8bkE;od`F)47Tu&h&u(5E=~ z%B=;}cN6du3h>DV?1a4yKDC22vTGa7L`zp0x_IXGwwtn#=$pmBswL;y=BJntNIg<~eZb1oSDS8?ARjH;coQ_NN*2FSED2G zrw|2{#@Pf+L2YxQ!J5wnLp6eqj2ZnFj?o65&WO>5V4Fu z;8^K?)7C_D$=g3ueT*KOnw;Sn>yFMxD<>j#~yBV>%sMbeWd^g(s z%Xi-S^4+(;|J93cUp%|}^Piuc%u?ObmT+txCOp=!;Ij~;`@AkNnF`qGZaT~I$Sl7! z{eY!CWF~sq-Uq+@@cDc1zxUe-ok!*$dwU}r7`fmSd=df~a{9%LbT>P%j?7jmo0Nf) z6FXY4tp28HiM?i~B<`Z+>@uPc0O061I@yR#0-n3JEwLH}g`2 zd39sdV0Rrv*?@M@VJdplS}B6>Zt=Qn#r!=gp5N)bcfSh#AsQX%i=T&buDKZo!Sd=w zBG13VV?AT`hgh9K%x(&qj7=X;_2dU1o|3_aC(&6G#$6(rBMbF#0 z*tK5%tvt}kW`ZWhy5{jETVKHxDCudlAl$1KrZ}%Y{^*lH%OohYnp14ZQXHY5 z>_;{d^P8(`gb{F$Q=c&eeVt9en-fo zMukWn5NlpA?Yrr%0Bd!EtE<(V5pn1v?BkXFbZy5BKMBfs-2|u=V@TTB-N|-DvCaF! z%W!j!<8vvSqf}@2iuA4Iz=pG*$ls&Q_fCgODRh~&V-$9NT>xhwjCw`!xse52WUaR1 zNWkg(1lRh{&2xhA=CUi_jGD0>kPJan8rSN}}QyxxwLcuvR8uVt?4hr5d%-==RoC5;e~;n|yx z_jv4Bz(g>b;XxmLS|{0>L;=d8Z3dLgtnvg~XQy5Co5Xq0DVz^FW`B5i`n@;4{@G7{ z^2L28fB5&`e{L#OXz-!Y6O?mMK`MFExhctWGaP4qyud#jE7&Z!Cwu(<;;-L-{^_Ux z_1Ev#QSl)i?tDXGb@Y_1{vq=x8@4ey#IR|_jRe9odz(y>yL0^Y+4Lx3#VH7)k{ebR zjPbk={`;2?s;Qn5!NMz#$73+O|6<7saw=ng&ON=k+Et2&uG18|6Z}~+scs6aYa87p ztFvvsHW@D3XM#aN$g!V_!mHyQ3&Ev#kAqp~F&u9!!C4aLob2$rbyKjy)wRMu<}lPX zaDrL8$teFe9ppt^$0)0xo;=<6h7S2%e4%@@41!@eCadhyjnwNdv#qnOeHJyl6c^}c zk3^!cP!M^8l^)|N?~|65=!9goij_e@Xc)bP!i}c++c;H{&sD)6(5}7FIGy( zI^S-^;Ou#G^huV=A9s=L0dR->4iaKg`M*EtBb2=@Q1@B|;3fa`2(d$IM>hs8O| zf=Kl#YmYOTa4O5#U)^>+=0tsy8(VjSr%x!#j1V%5aCYH%gT=~)4?|%8tJ8ymU|NS5T>z`-iRoFTZr;l&ezmBTd+jiJMAsxVpF&_**V%-- z_Dc+s`(EK%O&uBD>8vwTui^OJW&k!#2nY-x|9jXZ@aRrZ#X1ZD zQhw$6X2GBjz&mCWTkkUpq92ibG@Uw|k(iv42R)^0wzwOlJLmhmwJ> zZZeb{`;eiSH{8MST(_w0MA90^?9M||$vOE*h&S_5U7g)tY(=2rO|+cp!cjAwEb(K@ z<2}7%-DqU?7Dze`CtN<8j!TmAA6Iv;PBy)Jo@#5R)KPHDM@y=<*I{0sRNe~!%^^%5pBCJ zzNrWdfB6wxn`)jYLr3fKq?n0w?-X%Cg%|WX6KZL(e@q;IaQG z-I~&TsB0bqO844rt1CXjOBbj)QvHakec6Dx^e(&oK z-v8kFAO7QapMU$G|M_f=hmSQpiHsm_hu5C-);ZZ4lI}r&)%)PX_nv?DkN^1VH{bff zms{frwr#_1xKwxRIFzI)k$1<6P{0isx0}e z(+62c4=nP?y*gVPfdlOqd+cPkks$jQeQcpN3(5hW)RO%kO&{)aPsRNL3HT5Gd=i;> z>Rjh^HlzrA=`^~@cflK<@d6(``)-Xedon?2`i5q79+C_2caxpYp!zy1GAK6A9)eW; zKmr?1m$e@sc#4MILr;E;Q99m#jR>!*Y5N)e&;y94{ zKiVu9<|UB;K7|F}-z@K#%(j9${yW`4k2 zt1i6t>eWa^rnYLul^jq-~}Vp>tr_?sorAJjQIq9wjt`!`$qnP0k(FYz8_!? zyX$qm#d`Ot}nTW4O#>_gaXZ+PS9}X zR8YCg=$>}{wReO-<5)R<|LU0cVb zW^g$=!J1q2r~=v2zVSfD4|A&c>`Eoko(xFE=mcEzs1?oE1pM_UBw99h3aUE=R}lOe z!|2UG#)o!YcP-v1+&Q{AqU+<63kH*+bId7c%*RuKkMnJ9p{$6JX=@74dx5lGXH!PH zz`tHSxWJGx(v0T7^pCB?=0*{60wOKeiP>G1HLlaRhhOlZOL_`dwJVE{jP7jg za81Arl9y1c%p2Ngmh5iMd$+y29 z_s2%EC0x<30K(q(_PxGA={lI|oX_t0Fs-xuOk_5SKkIVekBzan{|Zjnu%(LV|#ungzPi-L~VOTLI0AeBe3EUwlu zX%WZ*@x|EYoD@N6N9l}Ujv9Ib&Hkf{r8*B+iZwT=qw`tdW9Xv_K11V^IW5Z(xccYm z!bT=@6hpxrVN(D>N5e9$aFeSh*Z^;|Uv^w=-6no_Ls4zE{Zajty~n8wZYfY-WAvgA z80$y?lmSFG2PcCt&*-++)9(zAfk(@BM4`4aqo;Fc=|u+%Sb^)m-Tai3oi{}07>qeO z@3Q3Em`UQxj`ygA;B$UcU4w;RyyJ6`)^0pmcd!Ka=vV5ow$`ggzi;g5d!FBT|+T!nCSbOM|#R4!nVJSvUDd?nY7ZbAqnZ!I?Mx=w zkq++@v-bF%3a(61^qCzJ?`j~p4tW*YjtU(ZZEWDBZjmlc8@w+o=vJHdx3bPQ=dr9eHS9y+3B zCImB)<&^M6&r42Kq}K_^?)q}n;R+@<*+;vaimtXG9Ae-((~Ki}C`xdoRJghMhK)0v z`1Kdw?PX6X+a9=16rga%n|R7z*XR@lG zYr*6U_-2gs!{XwXGoD9MT!KwBZHk6$WdFg^S<6a0OoA*h>yc>n!p&;Rbf{ljnm;_1_`S;gEqoLH&Ckx9oJNmO6^$u{_-Ve+f$gq56b=Jf@x z!=tBBvq3hxX{Tt8ibt=|n#@gQ3>N7^yr4!N0qNH~xQUu&G}#2`n|g>BKCAOD3QR{5 zw`r8l2#;>)t1N_F*JCVElPQ2}9dR7uM*yTAHjJ9&xhEK;aEU;tT|PXq9eQ%V%X!=m zDH#RF<7lqKsKG|O;hp_ZUu*=;6n)qEdOkTm==ah=%7DnMf1AOOBrGOmpV43>$x@N2 z6Rn!$o8)JQreIArn(A4HnT%}y*vtj~!pRO^nFbFA{~J%y?81^Cvkv6a@od2v{smlo zcJU?x_6U`WJn*>N6{<#0>|*OE9qnGN@N{_S>6ccmZ|!2Aji&K*k;^(8{!zV$wX%&+ zqLVNBG6_>vrkC+D+l)z5ko=Rc1Hnf4BpSDSvUq5;vUna1dw3{V9ZMpwKK=BQ0Ahxb z)FAH7(-Mn-$j3I~;N*;hK{0~rEYm$r5+TAm>3F+0G&BmyNiZve*{GpkQ%?*2wb9Eh zpbA0*djcfT$!N3p62Jh~VzYTuSRDmn4rlbl$Mxf~C2I^M%BUfxNL+5t%R2HvyWLY#{jJT^1YQfF zUTVWIHwxl}QD_?~dLHQG+>g@N9v=^HRRu$s$!OamJK5#{lAC85nT4Bd&glZr)=&hV z8A$lTffqrMam?vPTX5J!N&od_uAXQYvQ0zg;iRiNPc)`ZEy7(e{JMrx3GMe;8$aMr2Hn`X*gY0Clt3RR- z@q&S=SI;gAY4~dwkt#&TNW2`&D;4 z*T5i~kt3x_&uemm5POkD^V7XxEBS5B2S`Q(PqYeB%-M)uXMa~?$wYZ(^dysP`!4-! zw*+a^(>glF*7cC?9V=jx$?K=Xx6w;~rgFptPa0+eIs;p7bsu7u4#kWCxdA~^<2O2V z&Z|@LMuJbkW^lvBhSF!tpEsa`a_(%4BoZyU&5p;RYlV9}ChcG&c)bHEMfZ4@(vv+<0|$ zo4l%bSJpoAD?wKg<;$6yk36pj<*C0_BrdG!dFJn)+?9otKnI_+h6 z(ZEhjHSXrH>O}1tRQ1st-~ayCKmXu^=fC^i??3T9s~4x^UxO;D z#qh2k-DNjMx*^ZjjbzVKD|?n%rWsk-ajxv=1c+_mO?Y!8S@MoV5hJ`d(Z>yZ!L8ZE1| z*m4_;uJcVY&{?qCjmq#(PPIAP&ExZ6yK)qjX!93whpI+`cc*}1^x9TuB^BZsnplFW$~kclKX-y*Qz&BYT$OjyG? z9ZMM16-?FXeDjMP=ecuU27ZRhvH&oT5|Fcw20q4pLm+}r8V8#qy1uzy0Vy6mlEj*V zx2j1kWa(!A|o5_h8qTpO@Zn@;pRC4$-ex0h>Y7 zfzb!r@JqikOi2_-nS0<`6 zGrAE#fBKUjfAQ`Q-}(}TTii!!I>icE7{R=dFm#zJr8t|~oUT*~mV_uC1;%xl31d5O z=nM>Us4+HP7QiKH8v$nX0{gicl#s(go*kQF^g<%Z7K?c4IaM4N*VS>KfE`17#c#ab z1bh$pQ-@Gc$o>M$E^fm+THOK?EbBwtCd_{MQO$^MeCU{hy(ybUg*xpm;0szyjq%f8 z@H>lMRvR9+jKuk1lA6#06%C6e>HlFn{gTUg9OryrUl=@3lMpCs{^&kOY8+L+who4Q;fK|Ebm`)BxeL7z4=fcU$GaUAg@Nm5L z*2@~CvB_X-i=D|EL%u=c`f=FFh0VevPDp+_6$?CAav?dm1k+VC@D;Uwy>*9^uK)Kv!Jv(CW*dOF&04GpzlCZUYLs=qkVmF*xX z(A`Df8M6lvkUFJoG`b>;EPL_hG%uit1sHRgP5zpJQ8P13vEr{7!xaKhsaa`+9 zhL^SkrxNs$SFRUx~tnHQJw8=kIvDgvjWqHKC3&4fsc*?0_~nRr*7X9@wJb~ zi>!6@TLXH48@;>RJlT2F&!(9x=9dDj;I~Jr_ymVN_>d$LR|)lqh%OO8yzt>;(oDf- zG-f0Cv3M@XSc7=|N+%Pn^mr3|?5WE}H|-ht?cOW!M($)OaG`IVBKph##M7pI5EX!} z5jsJ7V8%!B%E;5q!$!RM+H~FdiAe`9Z7?M^bAt1tLD%}T7Kf(lj_&H)-n-)j{K~uP z?Ft|R4qit5Ki0prW$$Z)u=J1h`DH(^qy5+%`MHqCfmb^l3dnu7M0Y2=uKszt zbQhvu;wrf@+i>-dAAP)O9c=0#W*j>?!r6PIDUyRJn3XcV2Ut+=8XtJuDI%iB$&ZQ1?yN=Ae!9p(=XVYW~dIU2$)v=rmN|Jy%(=Xw&Nk6JkAsY{xHyAxbKqlrVm_C zCVP>SHuwmG4qxoQFJU_E=@`1ofe6mEUXHDH!TkdZBqPyCrC<6a$4z$$7+qWsAEYV> zcxNm7$j=>XKSsmTf8=z02IgoWJI(vujiHtKZc{*CdSOzG+WODzE+@Om=3WYPNa<^O z&+P$&l?W%#$(<}v<3I5NZ`pqGdDygL)HD%RZd3$@k7+gZO^28XkP+#gqA}k;yi}v| z%DO=2wok(?1|8e?cGl>`AK>FDcnDqclm3mGg4Gpdp5x%%C8tuIwJ~y;_XL;xZ0D9% zv#Y~E@WU5sRT>I=U z06HP!muE4gkoZ!B^oAH^9%ox31Hep>*7hRh;s&0 zN0L+t+KoqDo-$-nq#_8fW74BBERe=)&P(1Asy~t4?D;bpflLG1vt|NLgJxEl>yPoV>FSZ}aBj=u1*`Z997NuLEZ9nB!oFaG;Zr<^C5 z7!mL+vSP0lZ^!Ev)w7#3O3WJ46K4~Mt~pPz4ops#au%KA2T2r|e5Q@)L*{)6hCXi( zFC@QY{^&(?(~r(0nG5uBH+c#=SLucmlfMqqPPWO0tk*#aCU_%;VYX?0*EWUHDctll z1-Pa%x+`T$iz2Eo|Kp(_%l zlR(vB(z)G_0bn2C>qOR3E#lwJ&cG?YtYg@2CRehRBwZ=q>mZULc?hUgOMmP=8p2`r ziR2d~qt^zM4ZJI3I|!qKZ7U|5XaIwVcs>Rte@pbxL%#q-fRTD87v4%@ew_)rf-`!j z%Xkm=?mvoF{yg0H?th)Tsk$|kEQZ)u1kjm~o4+$v-t}zH2x>Rdg;OGiBq7PcgKw^L zyx9EGr}3){7s0@f3}1~4UfZ9SH5a$|C)@G9nESGphos~LU^XA%;B6YHGMi_XEOgo5 z8EF;cW&`xpMW5>I2@}c2RN@WY5F+<0?$R~bc2rr@B!SQ*06aRl@DQ!0GTE(}gT1hi zjB6wEZ7TXg4?Vc6{W{-snO&T9>%)tYuzHhS2lLxxd-dr*emp=FXTb}DJjBAv<_IHd zB&|`sWezN;=olW7=^|-FMi8Tp2*$C1hf}Z_b)$^gYe6r1sAt5VjwDzKa?jHV7KB^i zoIny3nCR$4FUg$d_|a^R@X39W9oMqy>JmJf1+n-LuqUX_Q&}`5!0JHfBajoQ%#PqD zD0OGl2k-q|oz*hLYNm{1tb~$)Q_y;Azv0o@5d4M<=pipWz{iPrUwm&jgPNVL4*;tx zlW+A+1sQUhkKcNSI;$6MYaRxcO=~KA}4p0xmLV^z^!a>9)YWXYo#E3Jb{YT2*gap6fOTE)3sMcm8!&B<`lqD<>JO%8SeG8S3(k_Zj~Ua%~H zhGq73rZ&J{5jCCBL2Wqi=KmoevTQ^r@VEqLcosDPa^BD2^cNa(WQOiHLG3a?y(3SIgmf%HFM`RJ!ouK_BU5K$>N!0L7bsf{44K2D! z^#nS0AEUEZ@{XF>!D5g$$$=?ucN+bZAO8#feOcyqJl57a8s$qWW+Rmgp!eSUGQsq( zgV))@o)VC}cOxaF;Sx|yWtqa+HvSanA)xpZp0_E6{>c#C;PaVa>lF70qt|t2?D*v! z^H{9m-`Xs!;4wSmpM0RR$zI29cB00keL)?LBc<#b&*24R{XIJROLrtvGz})+>5uHl zieQ5moTDeO(Mm7DV{fvC6!5SaAquL1R`KI>G4xoCsvS7%Kpg`nfeOymC;#+W0^(dA zL{=OcCV?PfVJei(R z(jEp!o(RiG4*lV^xQPpV)y~uwWZ-N|ZFDo%^}&=*P?{t|Mi~hU=8ney9I-Y=csV8{ zf;ADny3c9iXPD|Jwmwyrr{Q>k8T!rn(0?4G;1V5z?~zP6;uGyMxBGM*m6^(Al#^(d zU2;r-LyFvUlO16d9!)#w?5q1Khg{tqM^k)k>dG;QwWWU!wJBJUHC>nvqd;3^V^!nT z$8Xe}c6~RUg!bwfAzI5o2Kwz-xh;5Ej9y2O9?!H4eFY;}XWEghKueASFShal`|auP zO!Kn$=n0X|;bCM5=i@gH4014v;R@qVwn-%EyPpHsL3k*9yC!F!g}vZRq+REu$kZX& z)FB{kx=cIH-jD8M@5AQb@woZ?(9zZ;g`aSpghfZxNqp#}HyVuxqsu*s7uf!m_*J$? zw*=>mM1mFE-EgfExsq2UVg_AM(0sP9jRS;+rmcS_WN$dc>)vR&;1ifev}YSN+1D~6 z_&(t+f91RhVQW=_9Q@huY^cBQMoH;QRk~E4c-=4McQ=SuWqzZ>ggBZ+vB{c`F^B!f zcLn?8r%@k)9iWG)pl$IT?Ii&zXCtieD?_64jaa&{1dtQoNEh;*`62m+hn#D>DPFNs zNmVyWO;1}l3f^J~+VGoB?t%}>Vs&*S%DNu@p(vKY?UY(>5CPs74s4KaJMOEwz%>DaF0_c@NK6E<{?b7bAUN^=ONjgcE{UB z7TiP>qg|-JDgx7j2ByI&WtCx|K>(WJ2zt>l3JYEy*ay!ZeUR1IWkQ8jn4#d3tX>Ev z`b%e96V9OQ{`Td7L?%6RATcDQ+MC-Co@cYVKM_)3ZjU{U(x9M$j zy?t#~c;YTu9fvC%zH5_SvquwhZ;S_Mv9u~-5g+uhBkbXdWHRp&?si!Ur& zuRYpbhlk9`bNZrZTqi?@rlWxuR(2=oZlpL`0{|Z)aHhgIj1DaKD;?UJNI2#h3p&wO z6ohuc2b@(QIf3EuL|$k#&rY|?*x~Jev1&WTVqpEiizpPZJFjMSj;4)P_^=H&rhb;V z;V(l18(&@q=LKuz#%>=yg*0DfbbUZHn%JfY&;_<)NVTG)GRNdymT2|^7m^P4OFolR z@;sSw^=A{61;-bbM4{^1vYBn87sIk$L2=l$i|EA}Q#SFlt>`SLUp8UQuag~bbWAqE zio3wkA>9W$+U#CsG}Co@HmxP;m|~{6V6edjMYAN+%KTP0KweQB8@1(QH|dujm}0GD zHa7dInRcEYPA?Z4Qfy%JRXDn=8|-M89m$RMLOYJyuW2y+ZSBV>I(j!Wt_>+{r<)@& zqk5`&F3z@jr#~=jTzNA6__I0?ZPqv>IocT^#Mo0#oIv$9$}=p^)FBU>@lA*cP_Q}k z%@q;M&Y#B6-GY~N#L%5#%Eq^|2o7y4W^ZA7bQ`tj#;<_%Pex~oNxC2dAVHCsH&=>t>8pb83N(ijiFq|qW5J0bS(%_9O%cPf){;c!T5EEuEH|| zx~*OE1Lai$e{xeMPbQU32{F2N3PjKo>pPsO2r)yU5Qb=xj&m~p4jH@-fivulfP;}D zF+lNn2=w6@f5+)%X^>9FWEX5|faV&$Ws97@z!bjhDxT2UHR2BThS>3E!Ef@el6ywz zS7c*Kn~$#Twy)B!$Hhd`5{~3|wgCpW^X&$rqpqQeLh#|y>@aakaULIR(2g1rP_Wl! zV89xRtbUIo$%gQqgE=Z~Cv38;9_LazMGUyK}&s;QmqfJ z1jCnS-TfL3DzA2kP=sah;YG%wLCK&<6vD%1w=pC-GnLt<1VwuS+`U`HgRLV{PM=Nv zEFiE;zIQs9ZLeOH_(xnHvN~%e*kSX@3n(KEezjUh??+ZF{q?q*y9X^ogEQD|zru@U{`9PVIF~wZZCESp-$QEt_@=gN_~`zsp47a0nTtR{kavt!1c9Tr zg0Y=Q*RM{mxD0_aCpgnkH?_lP90W7wVsv-7GLFCsUfNOYc;vB|56*+K##0g}zfCz1 zYL|RY))O7t)|nv<9EP-vgoXH15NyrX?RIT<=As`UasY#(RYhN=Tl8XdI6yE3#?La; zI!H4Nl@W=O^GHM3f?bCQ!bUeCus}k>cp&yqBIGutVreqbKc;Wk7a|zdO7))jxv4nG@(b-?^$rw-J^(D!T z(!&49cHeLeCt8u{uMdBC*ap52U#zty<_EuECplD%K-3RhC0=L8tirJIoqW&hawP;+ zo1Qn-i{234@eM&F8=-%*M((@YS^yrwoG$12Yjb4TPNqUVw zum0O-pVXMbBP*jJ9KF-#)M`J*30CJ3jwvv4-!;t9m%>4qGsc{M2KxBi(SEeVkX*|s ztH(y{+VP}abDY>Vgq(%N?302JVUGBV;Pu38gmFW-C;pOsS@6jeOV$Jr=ujUFRX2wo z9SgLzL%GMN!)cmjQviKmmd)}*Lw(_crgu{jLoFasII7S+BtamP{)OWh$|3;0R5rtp za9=mpZFftuWu!IfH#l&ky~fKj!$lgDWyb?do?JX<4~LeV+BWFyW`S@40S%R{bu=tX zU^KzTPycm(sex<`0#-P)#a#mf`8q7}vu?3F^s9;1ZEp@AniB2W@G@aAau=k~BoZvc zPoC7dIbfBeY1uE&^244wC!f7HI|58W-7y(a_{ocz4jTlPGre_+Zud;p(?a%1E zA~4um)hAfA1n&gx@Z3d5umyv&lMKB6(6xAxMON=@Lxnq>>0Fm$6T`V@hlrD%U$iA- zr4aH{?Z|z?a#=oylcYv>F2Q*>O{*{F$z z$$j6hMe{b=z#D?Ci!_oC{A~EU$C?CdQ@JC~4z6NYZRV?N*vv>gEs12S`2%0J({%Sp z6nysp&gh)V$O=BQ3eog3`qp76cSrZ^jvTwl##B#NbiP{yDytbrc0Fy3qYm;}vZC*5 zlgll66elELRdG=QbK*d!Y$}`Cj6lace$&|K%O{!nVxi$jzxa&s;aM!D)A5p@r31RU z<80*3e%W*~TvDMy{$gAB)vrGN1ickT663EolWihO4|864QlkuGL^7(VS(ZaE0fP*m zgFE`V9`ex$cFKvZ422UXWCpMyDMO421`Zy!F@e|sE>j59GSn%v`u*KR%Y6!2hQK%p z*3f$VhLOS5Pg>;WtW}%A8IDr|C7YHLE{Xce=q91kC7Mx-ADQ27+zBMd-gA16+()e} zZumeUp0x*&E4wT}9i!O7U==oZ+VPpgW~0H)D#%}!Kvri#Q~<96qes-y`<$QhHzEo? zU2Rx?m#nHo^fFnnHwudqqljhP+7RgI?uWyv8?#VIfo8PyT)XpjLo%vJXO=y9bpF$8 z!}IE(zrTm+3Kh5BkxqbH;Ei66ym@0pOIs~}s4KVRxNR_NS z+QjchE{v_t%IIM)@5u((5rpjI6_~yO*;K})?C)HY5C{K;+{>e=3>|=<#7l=LrOE2`wZgf7jhN7Qc^Dj2T z_Dsnv8=u4XTY#EjCWj6tr&s3&m|zEk-m)hFl%F7ILAL7>5xQHavUb&>Pqu;Z>Kk1z z(ATNB%-=m8WdX8}bpa_PegvLPufDG1Pfw-|Rh!(6PIWMoN!FM>>|U+p#D3ZQc1TsW zV3;g)mxN{nEdCrHitpNO`WZepPdI3x19j+OSHOeLf8f+QJ$oY|jV=)#uSVXrhu?2@ zMN-@7Kn?J9zW4=HGBNcA&Cv;L@Q~~2^o~yOne6co=4i{#Oc$f2Kbv&WV5eoZ2rE8= zg}&NvDwfD-u-`p6vTEXoAZ|0vp1yENMDV+jMEKqQ@Z<%!aFAVJfnxR8O4ryN`tfxP zF#PIAXV)wn=uGe(*+eysNZ7r*XNgR7U48WF$4H*Cxei7di_GIJ;%cmBY$7HEp@TO9 zDiS#kh(8BVbKH{nGL z9yiAy{esA*4}(SYf)p53)W7MEVKCY)p7z}iBGV$(b7#IM1aTF{Hbeo~7RUkw{Tzp> z#)Z%6VI96Y@q$r8n$=dG{NVy~k0>YGiZ|a|mECj)MvBvI-=GxgKyq@Rl<#?KJ2_uHKQ^UPcvjvjsFJC-#O9X2`Bo-?UzJ1ryG}#@T>u+_`mw`;_bH zspygJX?LEC8@ZUGpLq2ni`{f*Ja8=2K!Wkq+gFvv&43-}j79*-o01C7%FoV6Y^T%d zP2dVH`b;rYC_pBKfW^m#fu!wv*J>|&ouLM2ozP&agO{$OL7j^D2vK?Jga9Q#PhS=8 z9XwUvPA4KD`|Nm+eyJ^p$5(U=N4DNkj_2z~x5>{mML2_t%G$$JS- zoy(?S;6YT?$rmlA#Wi_KHRxf|W!6;t70N41&#Mh1MYtiz5TNg?>(Z3H-_ z$szrqqjNEYeAzhP!k{&e$=n_2FDlBojY!dLJa?U*@1o)AlaD?Laa9XehL1CJJu*ZH z;&2WTa4}-+`2ckU*Oe4=8_|2*MGS2oEshSjsd>c7IF}s`bPPuHri%pFu5Y1z4tl?* zuUg~qpaGeat){DcBr)Zqg>n{bCg-CVuw{g-ivf{(#X$-d2a7O^i4@+ILtAwi>FU*kluMEQ`f- z>>+&?)=kN5js$6FN?w9L!xZSr9X7>y)3%14Cnt8qci@dqGGgF+X^wzV88`yK%{qJZ z_e=j?(m+0#qfh|{6C67{7Stz~A&>)~tjcHkVoe14Nh_FpD_or)eTXPWyddV|#{e5i>aJKpZDct+MWySBDh;j2B6uu931zwYKQ_7YE%Ee8$UcHYr*7|tAd z^3qvn7rIpa)5++98*R4PZjm-YEsKYO%)>tk(B12e8V%i&K=bK&IY>9$R@jpiT_Ot zl0m)q5*~0S?EZ2-)3oqy2ND?vho#*`WZDy95MJ%R=m_3Med&oknu0_pn_n{3*@9_b zcg~x7x?xMeM4G2HNI=hYI2jt{9Gmp$BU#gv-B6W}#S1$y?VTo=TkQhCe#0jY&Ggtn zI*I&j-ZtZElOtLdOK}^H6ews{?MwwDVSdoHY*8KXb*z-_fv9`lR6Z~XC>wtvCDYC& zJ?xO0?ZUz-nUE1PALZl2loPleR~c37z*jfHppk;$NOv${j?LKK?|m^k72Eqa3hQf= z%+)8KeHx%J#VGWqr&!p-mfca5;f8hsI`j3A3m-l?v>;I7rW#xJ8=;&9j;_P!g;cvCZXIjG@Y?1! zlOMTTU%9WlyPYATH#^9{*#R9yMA}Ucia_q)UidRdPV*b$smxh8A>Kyd=-k5YU^)$_ z;1wRyl`PaZg)-XGyO&#SIw?LFIvJltDOrFO+y$a|=`?6T-1bGs5je+kqO|(1a;Vzu z3V07RkOCh13Rrt@c2)SxK6J$C`|QS0QIRg_w>teq7rECe9tGkB3uH&fl9)(ZrdZ{5 z*2!E@=eK@0HA>LIf0(+I)OeDCdp8ENwtL%RGC>+D1tXo#xtaFxzW32|eKQJO)HX_w zj>F4&U)wEM*%B^rqG3T9UOg>H*^T{YMu%L_@L(^csy=y5VwVIASkawyxzR>kF>Q)R zhdvwS3!}$*2eV7y8mZWg2R(xO(8cP3ND;wO&hp*Mb^1;JRb#UZEFh2r5}0}m{+V#Y z1s2PzF4-Y=@~Ir%wb5yLnUqnGMDy(6j9{qAFPqk>N~7rU1%K+=u+H}{FxlSd?vx%n zmb`e0*6BPd$Om^Wbrx&qzM&weCyQK*&k{5rR>xm2ThU_g3 zq((CkXGd)`WVgV0^n7Qdb=!FNFtJJ?*F(R)&k2J{*NpnUU{2uhGcbbFpVfWs$QCi`^= z@I1cdNzuSy2Q0DGzC*@G59hbaU7QW114hj{8h@AkdK3+?bgu0fs9mGKSGVap81YUP z@FL;d!Cdxb4U6tIsq8Ee>$u}F8G2NRjT>3&Fh*L{A`bgH%)X`sH=WS+hhVrle?5M( zKU1y2rhjmhjm9`jURx)Lj`g)gxD9`LSq zzOCHq;ygc`0K!F9V&|s1z%e3cR|>T^5~|&7WQIhie!{U%rR&bK{{u4mDzkY+a<)5; zYd~BZr4>|O6Dl|qwt{d7k^^-6=%#bs44sfS8JQm4)^9Q^=-efn-BVGW*%G<)74^v? ze%aH^ttx_p5xF3Kg7?X$6?vB2I+N=1*}G^Y&plQn8ZBOv|HGqN*eW&5`Wnp{U7fA( zozqir_X>*Y>}E{9Leisak_|*WAuhR{pb9x^{Ib02b}mkA)&$z<;=>ym^jSw=9SO^} zJ7cRKy9BM>>{Jr5PQ!Q(V5*F!N?041k`(c3)!2Z?ZT4+)~l;U`{{WdC;W8ztF3`_5pA++ zT;2v?7$b_3(jw=^G~>aA44!c>QxD!2sIv%jb{@M;?i9A5HHS9+WN2QyG&8;QJ-xi3 z-OnvS;zYTy&{5?aANc~^cJL6IS%m6uXg`57?HDW|Y)fPCws4tG;BEIugokovCNkII zVFVq@9PI$>Y)Ta|Oe+vzwW7U};BEmcImpCAiNQzL@%*w*Bp_6CGIJ!M1MrR)Z1I{w zYa;`ADLbF)lXK;i_^_jCwzw>siPyF<_IbR@te?{Y59I5RHiKn+~_@+V%gcOP-(u;9?v?BdOE5a){ZYp z~I;hs1VPGzHO{2B=dSLN#Lsp4tw;b|XmuSLa# zDH#)EypeI6W=yfN>D0AkCP7_&;*Es0kj?pQNXZxxlaK(Uf!sO)*ApQ)z!tYH!QC=A zjFDsnQ9@NtV74Zpr*7my^!}}v){&?L-5qMZSXO!mn=a_Q_ z1{B=L*$#|g5aXzfr`Khod-p`V2r!dd_#aVs0& z*EwvOGI^h51@Y0>$T@|i$M|IkfMwJ3qu{Och<@u7>-6S}lm7AFz|13}lkc-* z^Y*bl7{oHQ-G_mVcf4%2suNhaOfPzN+HNuf!@ro*0WO2C8efyRjL|Oak>IAcm6K^a z&=>!j?6S8NxmLotX&162Z-;vMX!N6hd0EHXP=0pFE-Xpyz5kU@Ws2FYQAj+6@Yq-I zHnL3~=Mm|2OfK+*XgGogY4z#vj09;pngPoI1YULRJk%F%eKMZJuc3?f;iiw^pS$9B z77f82n65jBsWXb9 zss8QAQ78JeWuS~}#&L99$ELj?>uWBZ7*aDU2-l{2vN~HMW!U0|DVmX5m9KeAT;Nh4 zVfta~D$eKa$yJbNI9ZI0fMd+eleO*AL}1#CTr^3w4su7`rzA{qvY3t8!FdnC`@+Dfc6DuWZX`p3_aSZgq(Ef zWF$cBewc?mfIs{lrVCl(?ro7kfD;VtY``DA-*=}ezBNeqU0*zn4);~SVJZYvyzGgv z;nHFWjZyiy;SFptU%`t4CLJNy;a6DD-vYDaaHt~~E?A|C-B`y%!41D)%RXROSm~V1 zcdcq<@>YFzA-LG_v#F=}q=)5?@xS1bEVUW))#YQCC6dPjHk+v3hvj8}PpX&TgPVGkMboM6h1!}LfT{n=hmU*Rof@KIC z!i`-7Go4pX12@kZk%%4As><5&FE1(zr1}jMI5J|fyWuD!44`*(e~o&)3wzl)+e zO`#qay{SSGG`Z|4oyn5X?50ezo6#dHIaN;Cx9}A#N}|Ugb9N-KYHLZHZBk+o9Xz9x ze6vo8g3%jZ(Lq(ej8qn+g30L7UYlE-J+?MvPZk@U=~NZY+AGV_*7^6tV99q1nat2p za0xz{O+TZhc3{%y-q6+c*Yx+69A@0X>Y7P9YeflUb(dY~aFSn!cY0tiI<xgz0Mwtm)GNw03*^OZ=?E3C2+u9(F@7(FekK z2{fEM*lU)FK@c%QH{DVg?2>YFf}fnvSLkqog`Cne3XKTIS2S(9NS%IkjJH1251Pmm z9qEp*eh32Tb94rCe$Z#rPQmp){oP!aZOh2n8!3Zh{e`R~Y#3lS)-r(j>aaIO6J1op zkB%O+g;OAm$0d%#4YP|se|ybmmkb4OQ_fvyBiqaaK^^#}7W>&#aiam5i@#kT%-}5> zuikhI=4J@g!9+6nZUihiR*zbiAo828u@BwZy|<%g^b_1oUsiufTDUctZ&iCc?E(!A zq1hy4ig|Z_lOLZQs{VZ8Cy&JQNLf83=g9 zRi*fdl7htTMv30&;CvgpCgAnT=z|@};o`20P|&pA(NRCqjDuuxnMmK&*UnEdQ8G22 zF9=0bhIVuMl`{zRMFY0n6!moz4^=Jz%wVdP)9OMzP90MVX3@QAqB1SY6FrO`ZFUsw zHw2eOpesez!KYjt^42p8=hZav1g! zwZW&?LRH7p^g3BO+bAvgHOn5-Nlt*#>Bgf<*g{M#`$-8hSc{Wvg#DRP`kG$dHd-7- zv-a#`dhx{>>+{QE~Gw7$Ez{=;jT&cuz1;Dp@aw`lbt>IaLz6vMpKGl+q;LsIjbP} zP;5%9Yit28bhjYeWp;nYSli(Lv6DdBB; z0HgZbsDkOP&usA7DbP3v!EOs-d7xaXZp~+BVI0wup`Y;M{YBg(o8T;n>)fg@!P_)1 z`p`R6eA4t}F%W6d&n8V#nJMA3;)IT3Q+4o@QT$Cl>C~FUcA5r*@A+X@cbE8KAuC_A zBJr|#wwK05leJV&kA7z(IO=#)W{%;Q{Nm|d>u{;0Q>lVZ`!%{Gpo`dRM;73)%PJFC z<-6Z1-SKH~dHB;`|NY~7w;GMsfKI?jc?n<=Wrxhd;`AFp8G=mdrekVX?LFpvev^Rn zj?d>;I71kt1V^2AJ=K|@Upw=Po4X7Sg;Ardjsht4j5o%sOHm&fT4qjDS?n&StTBku z@Kwx6Oeqo6hW}f2s!f8{DPG@Q`4LimIBG@e=GZbF*!)`JT7FX*G!^V|ABN23*kFch}IrX_Ziareq6_@8PK^ z*k_<%ZVHWzIlBV(({=hq+4P1O!6qKYGm6VjH+2FJe$_uVcjMHqB^!LAHG%?h4&>o~ zM$LBH;O2~6V3|m!>o^J~T-ED4(7Ol^7&Wb(Fq6*yCk*<->S>P#Q$|0_PVN5LqcHN9 z?e7=Nrk~(TcJdXrO2^a1IzPDBNhJYOZBv5u6Lsr|GgRAXS=5QB@9BaqkjKes57k>_ z?0O%Ww!pi^?#WV{js)TA?QZhH(QS6!HAax$lV+&M39icG@i{-S?RTAJav=b&bJJMp zh#t7vEx1*A6a1om*=o$*Go-T1Y(mjfxFjlc$ls>h*&;oN1-DD+e6f+CnTGJ}X5cO ze`^%a*20&b;ryD$Lc62_J(3Q6z;^sS5?L`y9N6N2?R03SdlA9D#YQrmQIu2#IsWjC zhTzxHSgXS;pP5<14}^MT*15rzc+7Sdl`5Bz?Ow4OxwmT5So7y}Y+>GLta>8(H@A$S zD#@Ugk=SnaU&fp;D_{0rn+@NPN*TdkcD?{dXAIDAK4Ei2u&9qvGt3Ef>mURjrs~2e ztC}Ed;Sc?GF>E^~LFL?YzAD`oy?!P@_;pzEoZ!)-{p>%Ua5!XD;K&P35h@TGPVeDk z9bbLrDXBWBrV)mNREE9xtP~unNibe#LdgY^O_9~e=wzX9Mj1mhGhs5tf=Ql4vA`WK zBAV#{BO|iGQ+D*YOjk8#&EuM9NsNjrg$1-4aggm)K$8`qDqW@c8TPoT$@?k6S!nU5idB+JGIXe=@NUy8asZ!UL z!J;z<6{KyD;FAHqp@AucaLuo@54S1bDHL7Wqp_kJ4OjO0z8Qr13j-vS>R6%&Wz(|- z74|9k-OjYy*9K|<$Ov~GPCh6oiZL=)3#;OA)n@a{T3KanNkvT9x}|^hAfmE5oTFP-~J5!_AIAKr0)ed!AU)+!`I1+kq$s* zuatn%uB_b+wWF_gvmC@1G&dDQtdnZ61jY_1+LRh%{@s?>sqDdY#4|@hA2>;av8u2D zL%`~lW$R^M<9$XP?}U5L#;RVIDSA6JDtqr64zRe|cH>mOJLC(Nc6CfPWLe$I@;e2m zV?1)I&y=hir@;$??*+dCWb}{^o#;4r=QfPNT863Ib$SXgd)&>QA!W+i(KGy}b1hYNwMHP~ducPY!gw3^o}GhS}A2VfDi@Fo{1podX8%f><)C!a7m9uFi`M z0Kwl4Qk;}i_jhyA?5i@esVx(v#XV3kRgkoDfznhjI}pd{^G3?}rL!sodmSSHW!f@MMa&7PBE86J1^~954TKI1PNXU1Ejg{^_`6;rwm8(#VA^ zfzKcNkHY_$xuoWTdpZc&uRerIcRpc~RO|Ut4lwS9m6|ZLyYMr4#zK=1PJR zN-<;)wZ)$w@?i5iFMdLX{A1T6XELgNG>o6FDRXZ0V@5+KPu~mN(>;1kBzF|gVhSI6 zc;o;vE&89oId(ri zgB9hgLu})B9hDKUdI*RX-+^;a)K4=k1$MB=We<;xM)ogQPxa%4Ogl+#;6MHM&%b_q zi&)cBddCgR0q21M@{U}+*p@MNh0+!}tNSq4jAU6H$|G=&93$if*Y;cof59(-c~Yf| z(UkxMj4B4_I+_&1RtP~ga_dmQM*kkx7UP>orF7Ppqwg6);iW7TUBE1`7Y))$!?b9H zkQk=TjuL=j*iGL&aDf@fuYeAgY)<$lc#C%M?~dcjyqw2+0vRokjJ8Do8lFg4hg251 zkwPGcCxh|w9dr7O7hmK-K?3MxB_M}&>rmqvk0gOkJj`{&8$P_p%eMH2oAbzI$ej~k zqeCMTqTAc19beH$Y~(1et0lvJm+g{MzcYnJb#e(GUOT6|3kI1l54+a=i=GT^cQPvs zw&@?&W|)kLp5`1~ub!a4$K$6AMrGqu$$=E znwTj>{A3HJ9|VLwc{-Z!{+-4)=|gV22W8eN3Ak)crv#2$Z71L8eRh9X0gPSNVlXzE z4u9@j&7*Dl++p>ry`Y8WI{&9%L5$BW=p&)_9(%MW-A1SIB$k-mn*1}6C2V9%$uDoS z6I%Tx$t%c8JpPSawcRuC1bWvKGQQd8W)TPp%>oL(OhW~Gj{&I;yW7^)*?w3eatUfa zyT$Ne9F_4BT=&M%F>NH|&;^5ZrkJk7i}{C4H!7~BfJR3h+}ee?NpKiMm&rV9NS{fY zY=euQt)*}RElt0rm_0g3auv;00cLc8&IhpKF;L%sPj0)5IXG;fYz7J2h!h?jIzASI zB7>;-=gwV2|JG_m!$;@!XCshouELn@=Zvz79nZs9Iihj^giRQbYu$y_&nWHGSuhZ; z<9LV@sOork@4<|J^Q=?p1Uf5XOw>VPD)+lT=dxvg0y0u^Fx&YU;AlzV0tZlp@$F}s zS=bl`KP4Pgt`pR*Hns-ejfD81Y{wj{9-riJ8Zg6Nf!`wghV!*w90?f%y4@q;psM)a z-QZOdSQ!HG`QT1g{t`cvQ?jeZ6lNPLYB$=7mTjpF2CJOJvm}Z%qFGkjiJ-Enf>NCY zDLioi70D`gCPG$-17Zp%NBrob>8b_1(bsWCMwB|*?I3f39S>K)V#=Mp8$pEUonF!d z-RWQn|6k`Mjrh=Eyy34Bwlt^7@Q9IdcpB*?7G%hF@F5^28wLq}cyvKoN-(O2PdxQ2 zv%b6N2X6J@MFL-3a$|Vc=PN_olv~#Y(>>96)UzM>ga1=-3)^@n``XRuXopSPrIU3D zCM5-e!8%d8qPu?BIhP3y)g^~>JhJqWy>4oAl0hFjP+~fWwV5RNwa&xbIeM2tc7-G- z#scU~(b8SVl04w)_+eju1YjNL7F9X@n6JHnMRwJ(eleRi+MK;q=W7%M5@h17hNzG)`vfXUcT7;@hnmz9=p22*maO}(u`5c*(1z2#Q(fXE(@i{yJ ziqo@I9c4OKMs`cuetK%2Z|J63Y=pn-@veLwzs@tb+DRJVo#tGR9-~8}TE1-V-&7Tn zw&`Ye7ESU0)oE_S$YKb4puOt(fjnKnMay>(HnyUDTqf7wC+0Vg!D7pVIQ4w-ER zM)Dc1%@B}>SOABeFnu;E)7v9>BxM7fP6tC@g9}@FFyk;|J^Mq&5`VEK8kT%IM^7{$ z*`d@K$|WT~{`&I*eRW5OQDeMho*_yUf@(e9@SssT9zrO8etWIVv`6O&{W_Pf6CA}R zd^S7dH#`#*7@%)37#g5`)iUv3);wj@-ZYix@9$NL!F-|7YVMuVf>{g#b;KtieB>}d zSm@t8bC+bj!`VlGU`)5gw@;)=Y#2C0-kMCOf;9tNaR!p8CO`(r-uI|`oH5*J^Xwpi z$dH-e%kpa#FxlvW!xT|>p(_5P!;M&*v*hTteJ9`QPH@BkZ}1nW;Ho|vE~o^rHk*@Y z91t;<)tAG)6c`HT6%o;<4ui=dSRUfGe&|rad>Y8gAcrdpWGY}QZn`K zfJ|ock=2{B!^``-=`Yz>L8CJ8clY-Q0T;gx%dl-B?c^QyV0?`NcjqiEBwGPWCU|%1 zCyOltCP>Z{eD%JjLpOCt9uc__S#*Gxn1e+ET_blR=0Nxwm;O~?3UKlum(Dk}M%rW= zY#F^!NH;jw`7Oi6m-38m2~Hybj}TwSPhcqL`y6Ww5_l$|XtZnt8Wr3D`ghM-WSK~fGj?wK>>v!||rhm%pH;oePOS~epKjC>( zsnw;&-H}WlWwcLE`TvKtgK$w0SbW-iy?~LQ;&_>MwBA~ZB(2l&L*MaNd!s;|-_t9e zgP*K7z@RU@1aBNpm(fo}u4O+Gw{>bUBtut+me-MzYhq>-=w7EwN0<-h#Sy$C@nQ?T z3amO!uNm0-ShU?eD?x_;t@gj0&Sm45ctJrx8! zSy7Bb72lPaxmc{M-i#p{$r;ey`tTl{Xf@-*_MZOx<3H<3?pab*NFW`JXqLM9OMyEH z$LfffNVKVp!Zvr@FM3@6QqHn)-y9}@-_5)CjjlbE&s0X=6L5TR1jGDgpcqQ9^&wqL z`3Pds3FceMn=5kK+VdYWKWjmO`Wb69&dEkzv^wsZ@8JYffMJZar4R-}pj-H=jEXj| zS~nmFpk=|Z4r1Fo8JMh}0nl8)yuim08BTXllar|;GD+A*-E)q;mpEa#Nsx1ZXXnHP zH$0Ku(Y~E*qnF}3Mh1SfRU^3)5i%o5#A!rljLsGJK*_vR#?XpIX zjFD`kS*I+zEpRZVu4~W0$kkevjxjDyw-t`&U61ZyZs$zHWl!{6yDE&FBKH>K``(l! zq6$`m#gh0o9H_#F`$0`H6r3K52BFy>}EtXqGjBlPB&PdY<#6_m1~GJRJm> z$$pYTk4ToRVuYLM>tiH_aN6O4^QklN|5mwSHGEI`n&;Y*$65|$_e ziJmtVeHu96^ZVpe1E=U%XOryEW%I_THHT=sHJz%Ex!}!D)3KjztXVLR_mZeC&922X zals>dUgAA1j3)j9C%@ned&PsqkB)|)9N>$&B~*RLWwxCB2fTR0rfMGr1hqQft)GT> z9e#EZ!W!e3)w~@%vq5ZO*CxW*GV}Tt&wP5ZBX~rAZ!3hQ6N8OE|N3i{Oby83x|)Dg z2ttUQljs(gq_PVFgal0IjIMJwVtBjELxik=!QgLR;sAq3%i3ca!E>CN5uhNY?yI&D zn|sF<1F6n)J!C!2=Jz=N0x^T`kD`GgxL`UyVPzzNKr7ZM+(PvQGG!G-D^s!#z}RmG z6QNf>eixjOA2UnUklPBAnU8w>dBcrzip6#R7-a3QKOLUqh^2-vkkO6~g2 zSv~P(=Yv&!v~{uvs%BT4p6%M6k^z6$bDPT9Wp#tc_DrGc*mc@`0c#tj!}pN49Zq0- zQpDb&kwQNjae=TAYNL_e92nmc_J!K&jAinW^o-ZwB`eyVAE2+Vtx={7ad_7;)-|2{ z;D`4aAABI%Ji3TJV?)u}djZvAUGgLkwg`xbvuWLFFT8aqvvahNA6(!hhrPR|*tOd+ z62#M=fBZG$B7nOg%%AB8$(Nolz@39<2(vxI7{QhAA*sjBqAUb0Jj`(~yf^T+OJm+{|T{_4dvnZ4RR*g0s}+)X{TALF!9nS?R-iZ09&IJj_j zwc?G2()?~N11lbOV^nqPY&%cMMoBXYSkP%pYPcfu3#N|2+_ubV&jPA0%D(v7LK^-i z%9yL&0!3gkgC`Dh*(?S26Tc^mj#JF$vN>dMP!hSw*+e z?5Xqj^k!NCE|o@qvM~bKXe5Ud;7w;UT7Y*@>D&^`3bd!xh z5sd=M7R<*(b`~A812FqG1+Zz!cp)n`fPDXe!jC|`H4JsixD)ku{F$=2wZyK=e#1q+ z>#DM(Oc<-ye%I+2vvmSQc!8u#KCY0sIj|n32bnJG8lp|ocg@JcE5#)e@_7{5sMBB z|HxNyuFkjNkC545Hq77d>Egk~Dk_F6o)XNasn|ojBL+`z%US| z$D#WIONDCRR8huTo3Tf6HDV}&1=&RYWV2UbMsE#j8W3M=-lY9nmTq-<>&V@@mhH#i*f{P>vZr?;g3j`!Da_H z?a;Ld5|}sg)X7wDn+?jwwk}aQvsfJXd0QvQC##)p0xH?U*Z1G)VwPA^6D~L~v0xLN z?BW*21;VdRx7vL#i3yQ+i2vGMYz138+$yu)i9FPSPahtsvP?O>S7QNHXD8g^PeLLg z#N~Kf>)MCg_l+!~u@2XkkSSC^1}AweS&J^>5=4Lfb|W1!{a(W04tUzay~K%t;qyq3 zD))Q#f3mGzGP7>j*YQ%>o`W?zhbR7lIZ%CctO1JmP2U0Au_sV#zA{?qn^v|~(uK)U z=RNYOgU4jGS4N~0K4P>Pac8r&ul97$WvPqQVtF=9NAKxEEpf#?JgKm=_y#w^Ja%Yt zQ=F?NeuVOMLeFoBRp57>V7ii!!I%8*hU?@asagjgq<(PVFQ~#pL27+BQ}<5lX(W~>*YT(nJGVxrtrxE+%wwEX$BV_?Kia(vytIy=jT|I z|NjNYoM|*L%G%e20a5Zs4+K=U4hWf;KG-8Y4EqgfiEz9Jt8*9@aD%;0>Q`CEvXri5 zfT&xqU1zsv@1Q2U8?pJ(!^6*5&6{=zF-n%b(hFRpEqa2rV4HCYY{+!FA9JDz#nSS$)|xF}!frWS@&f+tIm^O)%^L<4>d0ce2Md@sg$BarX#S z*|t<2LTzruS)KhuoM{n2KK8ivTMT~CKYNMZT7ZsK!EyFC8j^`d07Rc_+wv0ddpZPS z)T5txHKMBOY(IE{-24UxdWh!LSGGWZwu>g$hhx-Ko@{R|GuS#{$sMMWnYnerXO>YL zM@&KtaOXM=J65>+V8S6X*nPN#>;(I4bkxHW4zJ5yuoVwn0X8~_ zR()HB!PwIV!WWgpw%M0!QQf3VE}>pbi0|Ftx=cSGaTq;1B|B2Ty=3G3H#%LTFnWSJ zG7suv2T;*CIVTPABjZ>n9f3O0jRw^vJ`4A?rN|{e9roM5C}sPddFGd+y%38XgHidv zQ&gH~vz;)a^9^(*puH=l>sulS<^~XMxiGy3YQ%dI32}bS$~^t=KmXf7kU}tFb^Ozd zenQM~W~51Yi(hfE5eVTtzebed84Hd9ATpGj{~wi`1KtJGoE>851ce-iRnxM(kMHK? z!-WPMhHu$Y3{5HE1%?6`psXpZ)elaLM`fiv*eUixeo>nAcWI zN4y&ggKzt*J8>7F`Ww%Ns@2>3szHsG>ty3M{`VM+>dl7>Ml+h1Bs3P2LqTeL*=y@Z zLa_S)=$6Au29kuQt(eYYtzeJeWH~^|VVN45!o4hz3xnUekpTFc8VEka--0fOq00rx zGvbJ;F8qCiPmkILoRse-OZ5w$94s94MV^crh~<<(Y*A0z5xxpQzJLW68)E2vr<4mo z!AVUr#|4e*lkG5!EP+p=5UfPD&i)R5d$o_=i^I?y{5gEMw?3E5myNC}jf2{8mxiPy zVlT1k+GLQPWw-P3$7AZ>s;t93EknV6*m`!+HFmqilE@zLx#FgAk}10+pN~2Myyzf_ zOENV&lIMk!eiDN3C1;gMjJCFr&wUti7!G#n>&CRMq^7eI!Pz{yk8%uxkIgxzsljle z2VV<-^J6-dD4cxR0sokdWjF5cFzq6eTVw&~YZ ze^`VhD7n}jyY*E?@QsgP1e4W!_^4@kNfvb5q6{9puP#_y(=GwIktUDJANv_qUoqgn zf8YB+7W7CuLbkQjA;al72z@7r5Hi0_zo=KyEzHn=FseIasC=DS=ZqVoI6Wh;{Ruqe zG4$*A9JKly-iMNDJSQ;5Kirt^!g@OjWe8-)h|pbH;NfRRMsOg?w1IO9I@NP>f<{jY z3|}n+9Uz|sy&w=?0b|dPsV#6@h@IbGNA#Iv8Fr&oMrwBiS?q;Uj37tweK%@#g;DVO ze)%Ml1l-s7+4@P<81IISQ-G0FaxU;OE~A&Yj-P0mI5-vt$hE?1*|{Zv9Z%F;*PkPp zBUXbx76Zw=&RVuX0DW15|VbanHqtP7oZ>CjY^9co`TfnqkH&Br2DrpG?I zE@PWRtam5djIZFa1wk-V09`<$zYrd;bFZzy|4;lPI(XB6u$EXEV;xOs3(l`PAJZ(GYN*Dh0`N#*f!jz13;k@A-{tA81CRTE3(x!}``$zP zl9>jkDR0Uz`JtCSBVy}e+~^99ae>##R6XL}sG3VSC!S`OLlYE!{;P)q|xUar?kl*q9m&B#ML`oPn#pGM@n zFkTp?Ko@`t*lrgFlOOnI1sx$p2oLApz0>#El@0@Y1@dUTn1h#Of^WHPWu|P6P#Yh3 zAVB0+7fvwf%i6G#Xq;WrEnj#R-_)tBHsT20cC&2y*kfp-)9x(nzp$GT6xBJsPEIVf z-ilh!qGQLq8F!t0xclCETX?p)2oK{u+0ZS@B*S%HG;cJ$N8+Qsx)(2xA3MN?n{Gn7 z*&ecb=+cuT$j#=G?KJz(pMM1~z!l20i2u`wA|UITvlNapQ|+QNKajE$yc!BlY7=CI z5x1!nQVS_VXH4%w;UtXD;w8bmPvHy`ET$5p0k)M>ekTI2lY)qkMBiS{Lzrquv&DB_ zVj4vzC1na@#*$DIutuEdf-onG4!knb_NT41yH+c8c8yyIc#MDAaj}{_ie<3XR9r< zj%01Yke)V5g%fRr+&S&pjS{=7i_^UoQfnQuzPlu0omM*9O_tS~OxB)!!i?{0Jsg_@ z?{Bmvhnubnu?{LY>%5o>J0Qa@lNUK`S6~;1NibCpJ+o1G3NY#Hf)$$zbb-)=6m^I? zNihXI+i_DF1mG@sRkx#HSFaif5OONh zxopEr4RCZFeLkP?(NXpE6TiV*-`0Tyff|+ZI~@VPWt$eW@iBdW=ELl8@gg1?Qf}=a z+Sj>~aoIbuf<<&-B(#nbK-W!|dN&U_g8`q61utUhOqd3u`<^EkoXC8(Q;>{jn_XLj zWa%{FZzcje!KT6ft-cDrrw_m2i*dQK5x>EP!b^0;fOzFoG|b0i#3mQ=U~lVitAlKZ zi!at|gFnwwaq*Gy7&k?$cwPLPA9Riepr8J#197&DDCFijQ_%xhpR$=Pf_H5K4%rsB z2H}(t_Ey&C=(b9K;$r{ggKwtnG?$h^z$8)$U*4c32WuFT! zpSwd?N5OEWQ1H5DGr=CyQAD4eRl2?31?{H>-lm6iVw=cjQ`6=Sqn&JPn*;<$kIdiN zK%OFKnl2kHqi-q>V|Agq7^V5nxbiP{Wt4QSE0`EZ$KG1M= z5k?)b8BX)#Anep?T8E#vTem0 zrIr7Wrg-9uO1(E^8!vSF5qmy}-{2&d$w!Ziz71(foy^!WQ0F^rR9mFn`dIR13n&|N zu@e7#v0CyJiTtRK@>kgLGPc>l+0NogHGmMzbcjE*4cDv$UdK|4>OKAaF%wW{EGQts97R0yUlj4K=anWf?0@dA>H=^uc zM9AisQ71UeEhA%?aS_DoBoNb74$+$d$yyua2lIrt&Zx@PFl2hS#V|f5i0X8G&p#@v zuP&OSMq5mXD}t^O%FR934o#Kefub4pGU9@kkpQ8tQwiR(O2LyHYJ43~^y{?T>=&#_ zIM@>tM~ogF5814fAX`;xyPLS8aS2P;<%%Wm$)0hV1Iz$IsMww`%uGpSfGlhyZNm0V1IkfuqHzWsV$v;=_cBFwz|&sRys2dJUI(e z$!V4QRHmMe4gwwNHBJZ7e;q)K27}BHH)EE?(+2v}|3;mpqQi~els4Pijh+IcU>dC5 zJ5jr@-Aae6ns5bc>s$pYQ&jl+G<3gOl^>2MnNHZo8G4S@@N)r02TN|*9q8-W*u;sX zV~j@+c01k(i+u(wN`rGa&N^~&cBJ80I)7nPUpJt-;6J+p+SJTwE)n?BxklSy63s8` z8#1P#1 zR`g#+uFC-z2C6;&`RAVb7M%m=FMV%R-><+&1vV~hDip3kidXjO?<-k+s^imfW*dAR zo$T6(@!R+JjoQ6ye0Py2BXn#8%rL8`$>%TY!0d_NR4=eg>^j!z;A&IJo8lswVny&j zcttRmh!)lH_VwJg@7BO{)Z#1I)m|r@nXC@}xae+jjK9xDcr(c|kiB8JJ&mVtJ81HZ>rnG(-um=!&Z6xCoq;LKiju_({ottiARLkHH&Rd#=~s z{V|ov813{SLf2)XI-5;z<%~MaO%p`VWutuy1Psl2f8;cI1h3k)6-NJO0q@thx4~gx zGlJ@F>LT@_nXLC%4e|*GIosmtrT?4D4wB4C_i0WQ&5-RzO|;fz51h*}1?4@|4cr{U zt5)}XEUmyrC*5-B%5Z6EN#z{%VMXq zD3}w)pFn0G-1Hyp>sW#luh)@w1;HJg()yk+#q#t?m)q&npI97HEG1($;6{xybUdcp zka>j?1CqH((#AAI+yjDXew2cw8Ch4niSE8TZe#awU z!8aTB8|xz2=4*Lh9Z zU}XpSPP)}tmsL9R7T z_Su3hTJfHJ;rN^3x@#0)P+3RTaDZ$#-}+JJlk-=wGMxmH1S>S#SI)`j*%5?h#~Zzb zcbNy9f0AWXCSYzSMFtg~54~q?_Drs>KSK0^B;A6&>8JuD3Na{O;-I^qB&9%acs=IB zw_O|6e(%~kwfMce#mT$>>169yLJ+ zgE!tM6qlV}P`|8S8GO}`CT1VH>6)T8Rg;`#ZMzv0t5x8Twzb(_x)CVnN%wjeI*hS& z%0V25Yp80USf_V#fj^i6g|6%poE>#7>(im1K^%Q}1uq>bo1X;rrX{Nq|B38$Q~BO` zAG~#7^rkvq6U3%sbiix_FKDmklCNN!e%w~gPDUR+RqxkpG}(qp&rBqmj0o8*zoSyvwLjpQiAgcAL*ee09g;p?xG#pEg?)IYRi(JNI=CJ^q{TjI690X7-Hm{f z0Q()VB@nYMa$>8Cd-<}CWz$6FKHC_tOC!u%JJB+{wi zg^6Whgiu4*37*upJAyfM$`O|ZBLaddP{ssej%F zvG3nuE*dxGL<(8_WYHhP(Q#jPP(4y7&v5+u_B#Er33yyz8?3U(>aBWI&+z->z~f)1 zR_DGfQ173!SlFF{yKXpdk9OwWH>U^>{Pc=tBbyA#mSYaVh)sXZ&^f%32O$jqpsf6E zma3E95DgnT`r3+mE`^MZSa3I|Yh((fLYorMqK}wN78&B9?2A4+>o22|Z6~j)qj=BD z($2@|Rfa7qqDu}U5U5N(b_?3_%&><@W`Y0!`ZlFNM|+qoSv;KnTpmKiWL#9gXbTW8P$4(2H3*Mb{)?0C<#sqN-NIo1qE766pGwa-Qf z$&M8xo2e(b&%*m*{rUaoQaLn3 zi4XaX`W#^P@}L<(rhK@TskeSpuMT8fc2OoGF#Dq+G`oV5Gf;Z+W=b)4ovw~T4Q1KZ z7C;|&QBS54lz5Km%uhfNU`VNUwb@i2$Av#SF2Hx(xp+(Q7RT4g)I>*R!w0io_9i&~ z9&c*Kbt}77DmY?9+p8i%3L6ls7paE{67!jPS-^Idr46t*g;#0ZC8;!P&hZ zY)LVm;2^UKHeH$wa5^6j$Hjp2l_i5Qwhm$+vZ8D-$*QX^kw?InKt1bh`F^&(a`qhG z3H4hsZ+Z`xsV$a)#wW>O3BzsC0G_4OeE^c@ZQ%}9=hjBICW9w&Gk!LsM6Q4R=Z_SC zKmzP*8i-6&OB4uD4JP18BOwI@lbrS2dxVJGdNBm11cuLKHrF1~1)u)nzcMg~E|83# zlZY7v+A*!@RB+Wd;TreOY5Hs(Wr3%Q3_Jtc+%Q4cRxsMsMhFEl%_fLwxAW`st-lC* z5e)HzhrwpjDnI@@i}RI!HcXcxpkMvnyciu8tPCmkP`#YO@b>2S(anWY9HS40Avn7V zXmyj*BRiug3*vK$>Qc6Kk?+6#zI6rS-~eD&weyEU1qWH#!>L9y{M8o+bdr0PlLg)` zLu^4a!d*;e0&?xiSb($G0}Oc!CecAxMjVXW*x2xzc{8d#^G9%|%+4>Xg{bqY3zRy_ zo9J`vH<2?!lP%SBGBM%LOe=ED@Fd)$fF$LuyCf6HLu^$8lrwaJi~5EFXqd^ z?U>xQCW9aTl9`xF+}82r_}kCllbv7@UTg_4;nCs#_UoT;W)tzI;~*A>xA)v6>p+E* z%z`E0#HcL>$l{>jGzxY?Pm8LqLNtygsf-VzBq;Xe->1wi>0wWFbGOL~M>`>w9@ z2UNSGkFRPsa*f;^*BRG3eS{A`6@PVBWCiD~r8+-*GTT0$ldEk;ThHzKk~X;D4$d}a zpcTx;J_)x?NZg1HA^>ta>W|iEi|C7|Z5CjY(J{N9b$|IumB^&3{Ps3?u&v-0C)g!F z7t{8@Mo$N*GT(jr@%LX>5z_(v8k!ouy=xfQ3Aq+?hJscg3Q{RHVm5DBEdiPcIIp?h z-NiU7WfZ{~SV|^=gvesU!GN)_0FSM;uMQ!bE*nEw!(+jM!Ei_%e0xn8AO8v|9RQte zMN4%#!S{l#*4t_*C!@U<1=>#U@@`1i4^}dH8g6()C0)ZCD?I0oT zj-TBGsWX5okeHqGjI>T7Zc{lG0jYCJGow$zhT@QgfAhZ6ef9A%SV^E_8O-z%EK?<^ zsV3TSmP{Fe@IC+~ii4v{suJ7-k=1-CXya|at zU8gWB2<*ue7mjo;nM$xj*?|$$b6K%nb?elZtiu_dk);)G&_%@=5 zVazOZugnO=IdyJ#Qn+?AVjzu>gcVyD(Yb5{9U+hU7_aWv@I0eaoAax^ZH>$g&aTaY z(4?WBRyGG7=T33(Qr5QQMGxUFo2UVJ-tVZUYP?vOI6J;Rg~3DPk{(_wk6a zt|?GMXn?OhMA zF1XYdeQ5={*-moGW{xa!HLZy#&R#Kj^P^~+GP6!ac)HaLe~> z_cZ_=n(EUB*-qaj2V_h&p3_Aza0!5>;l8FpzLJ8Q?mqh@m<1_%>U7fR@(CdRYj@AS z!oNv0oWb>>e$ith=x&md;lcle6~GTRHo# zzW8C$8HMz+WKG*i?)>jP-cs{oi;bPYp=%>^0-0XX+-ZkIzmZIQZdx`1jMm7_?#T1o zn#6;LAx=6PYY(fi+8=LV+hlbyR3v~>{9CKZTuk*=TBCn_vk{b!gJ9xLVyq2%eX%`W z2Sz91uRNm2oX_r!iu`^9FSVV6Jkw;fc9Vm?I$cYhp2gGKEHsH0unxd13tac-)&d=fF0W3Vn=Vw`& ztaMXU)vHGE;)a3!_SRXle01tcOj*roc~USL{DNUZo_ND~+FFRd5m_blxQ+&27F(^0 z2)K3hoM77mYl8-Pl>daeoo-byeG&-;NOlssaBJ~gCmNJK(Up$4qRwt3pm;*AT_88Y zEnwD8`|X^G*0o*dguf)?o4nHpy_mYd$4%FTcNo6?_TH!qp1T)abi|hkqt9T^hjk=un0zW59ixtZuI^stt6n-M zE}ZR|VoT71+qJ=^!&yp)^&EV_E-mJ_nGyQ!*oeU>+_v+@ndryo*F$61*@KZxW2XYv zMI7O)Ejy(*i4lBXF?>wdfS>Qlz+|6zaU+IV$C-?DDDj6ENxAEe#FWi4Q~=-8k3avK zgUXUTt7m)sA)zKli+}}NSxikrjz-P-Bg5U$wC};Go!;zgE^($1vs+6WWz~`GfLo(= z1{KSi?euVC7-gv|L}FlZd-T+V=DV(*sU-mzO#;pKO?RE}w3euld?9j@mtpU-_7CI%wq&L8VUBc z0am!Dzh-Y7&exHBM0GY|eKh7xacnhiDY`nO;2Ha)Nw{GpHkUFGFr5+B)rj-w`_f^ zib=S7(G*=E-Htj_on4PM)g%hL0kZE-E3AI9qdSYe>d=vsO0jda5|4*kyd^POaIok4~1qMngBaqdNC>@-~X_13@HTOhxA zLUHW|v9cSLX!d@i5AdQLyaj1~NB_|apN}v89{2^!-8>nsCUCOI*NT{+Acc*U5N^N_f<{b&V>PSkl$3ga2noNaK@~1w4P(VNJMw!27+n zGIer2zh)=sqQjH@W;YBY(ghnw+L8@1sy4f6x9GeZUCEe^_Ep^D1sR|2iKZL zTr5}*Ry0VeHaZIxnb{okX*Q@Hz^e~N<6^b%{=&bF81-zMf|JOgAN%MIFL#%NE^&@@ zbim6V1GU(IN2%PV;p63QT)fZIk3W7?shX5f76+(Bvf8EOGnCDpeSh1vc`~+>{)W&o zyeDN&QGEwWCxk)HR>KJmL0MSsd?$!Hn=N_*S+z|=bj8p?>0rdX>hI_9W&HrqU4^(cAY+iQoCx0YKg9c&$q+?wz#PTbk>%*7YO0%m_8OD*+t*3-Gb-q z5|VAn%^9~48{9o;GnhjGOI%f759yH!k`0qoujAk z>EY-ZhjcoFk1to=qO0J(k!Fl;+BG@u$p@kX96=7~j@L>Dm3{Z<{pC4to@(|&eS8d(n@3~`&__}(54`XKez zqOfYGQI0}b8RPgS=-CXJu}k&v0Ylf>OKsWrZZzgo0%GM0td-r`LrLFS9!?qe;DEOv z9K`fsl(hJxQf1`(l5Ou9XW$mohKqtDO~+@8I9pKP4_6=?{wtiXgRk!5qmGE(g-Zgo z&T)2n{4S1+Pi=!5-$R{_Bx`iNo4v{0R~_))`!CkHm0&?Aov^9uOj6Z*4aVr@JGK7^ zR)e{Nqmv~KHpNWXT_w;=#qLDsSR8oJx*`MEZTlKl>>uI&`Y0H8Opx=s0I5yVT^c~j z9btM{vQCIH9wtc2=4ZL@Hx7LD?_?_JVaDz8*JRj615Z!@GHC9 zVq&G{S@rnfR-r#NoJCIu&ZfSO-EDh8W@CnUIrz;sZ;Qflg!4>w{gka%8d18h0 zjWV%UTbas8?|6?CH$Boh^b6wPp(hw5PcO>4)OWPYW;>n;m)X`Z7+jHltFX2b1XVoR zW7!ew=`}ia#KGtK8=*ztI!tk7 zBT!|0(DUNX=D=L=?9x3lCwuI48wUE`j84ar0Oo&Bspvv>A*L?*L8UU#;=k$_?=nVo zZyL)oc6B6iMyG)&-43fIgJQpkH=l8^K^& zb%LqwgcGaeaCg0QaYXmSNH_fuW9y9GZbABWG=wCg z36HBx|D8`^8LDhmVj^gKZFQu;1yaf-q>i`u{Sj1@;8>fVFDbD);4Ns#X2R`pAf^f! z!GbShQ}`S>rNKEP=vxQ((^D>4bTn)zj$lgtscXkCIi$RoI*tVy_$p>>=o2i1FVJjW zGQ1M(8Ex>_L99bg-i)X+#t}Qoq3W6^oIP`>_6(4%;#UL-+oo(d>>g)Sov%PK5fA4X z8d+^!$XvBR5-Q^cHr5 z$+5O}12h3xMpRV|pN4(#ZjlOFUSt7&n4BCQwx)N454HbWaw#Vdbl#vXq&q$24gVPOw|ME#s!TeRw zHyz2h$#%QAvUjrq&b8$?rnP4CwdeOf)(gmFBe3uYzUUEbXFu`st60A&uhf!TvmI=89Aqe&*dx}-IGw-q&tQ8z$^veqpfx}m&_j| zT82Rp#b@l$lVCH2fKHy)xzC2lGG3C|>=JNsQh7XX3i!W&{T)7xM0vhl3cgp$WeNa`K=Tofl<7cLG;gt>nxUJR5dkZ%l-Y+UldXQo(!rd5N>)JFDZEIl~FD_S7$a9ETf^>Dm@@y2aGQ5l82cC zaiyUZond^;`d~g@(~n7shSVTe?U`^FcBLbNnPr3W+W< zC+@ZF8YwfO=>o3mumj$@t<+syqTaDrhif~pUV_Xj`aI&qR27Vk&bsEDA8f=$9_+0a zZtgVY`k6=1uXMmA_ss64`s;kUzMElZU;UNLjZ^1TT-|NBc}XT6OHQ7C{P&M3Vm=W< z4i@M!Fe4o!%*t`hb^UHG3;^&23qgJvL@@qWb=(Q;eE@qV6}AW`A^a;`4zcr&8C5o* zu?5!!ft-E;GML!!U`8UZ`2xCm)G=oW0Qd)n}Y3PgB37JrbFPaiGGPlUTj)oS`(SSworUGiufg>tjyFR0M z7|0knSu^hOo&jv=TqQc1PSCVYp>E5Cu5Yh??TK>tCj5E#gg{M>IAjKb_fok^5B}_JOYKVmmvk`W{^OPZ*l-k=}^Oc`#wowz5*_h6fF`J7P6z90p5x~g<+Z`@w#QPk%<2_CGpI{Ge6<}R9 z84h-Js-4crb9O;y;rV56=-LIxD3XA{t@J(F(2ao4FSzw}9*_v^eOFOVvb-+4#xqLF z*CtE!{#Mp5(IUslX=E3mV4`)M9bd)I@732)XFO;#o3Lb$V59Mw?N~hS*zQPs3Fd3+ z6!ghs$tR>_M8;0SS(9u;QWpG#y;&;)?(!_8>-H097Yow`xtoUc^j;vA00_SEuD)%^ z0&CE^>KxC}2S&qsa$cfc>m>jIx!|Auh`m~hZ?iSD)+u+5wGq7J<#@p(Sw>jO)TYW2 zmkm~59loV2LV+Yw0$~J*MRE~CMVC2%kq8?I-)&o$6N4q9^{M^rul6rua)MTV?`t9J zs{H-ej~U{Ij^F>UdR2iC@G8e71y_A|j@dql-QB_|$ph3Xns~0?+AnF(4K-5IpfjQj|yMcMJ$0%5a775){j6#Dl&+BW>dzN^3B!H z+0P17IGEv5Cwlg{l7t60N4rcdJorzD%UX;g&SIu8Vj4cmtiFBnAHP@rZUjcrUF01M ziB8cq1K?%-x{JRIkTc$mK0rrPe^|a96=Z`h*E?qP(c%5oGO;;zzhpyG0wiDDc357| z$SGevGBaVW<6?ws$U#po7ZBiJqNk6|-=kn&j`k#~O4c*+Zm8Q=Ea_Y=P9tWvB4exk z9lc_t;2E^9kr(Y>K&K-P3~qn%y#T_y0YE_f@9uA{rbq5;WR`BKwm_NZvyJ)GJ7=pr zp7uN=Mjss);Ixl;!RZO#?~~)ET>6{dd@u$h98*qzVLmRJEBlT2~;0aswAzs{;3iqSDarTfr}7H6v}hg1^aiNBkw ztUnN>XFkc0T>S1{rUlgF8LcAbo^!_!&>t; zp`3fmD@;Xgluai{brBySMl*Qi@yl?vasf8=>_cCs*Y=#TjyGDuiv~H@*L2aQ#@%F` zAG-^^n*B+PzV!yMWV2H$G%9G9 z@yjd|;j*3D%HmdkN{`lfkqNFJ&wDJ`uU^&Rf*;*MncT)3xpn2Ky7!m^*)Y1hDjFh! zyaK|oeL9O~e4TMldT65K2y{0++1>Ocxoh_}8OU~Ku=K*QI!2pC&t;bBiJ{IIfywKv zuXuLZMlspY?iXRGlbOf0zgG_(q@AH^NQXM72;Gf#_3$#WiYvz_S*$pCm;pOZVwYhp z+Yc85H$|3ubArlW+&EYJmpX>zWs%Yp6FY<36vq-H^oQUj1e>}FpWwaDtz*6_t4Ao? z3DY@0U#Zwj$HpOAvrjg#JA`WyM+Dtb-vOFaYa5hh`k3J5oUq`%z5RHN3sVStgt&Qy z~aLwmJD{7tS4h~WlQnVbr?zO>Ayez?Am4Wdr?hi3pNS}S|$^x2N>yn2uj_o zXLDy&-INDKR%aapdrEjNMqltk$Y4_1w!c=Fu>4O3{bwjSdl_vmTF^OTAt=U=+1JYK z)7BB_y%`mum`-|(*a=qe{|Lt0<o7(GRS1~OJ;`O=rUiy@$#~CD1y4`l+!TqUdvG^*j>loh7GJ~l*5Qqc$Ow5FX*V;S z5n@1!6BpA{RoFwgSG}^|J9^s*Iw~(o(V6H-jbit-S!A-EwNK{Jg`Z6`g^R<(<=SkO zEGpW)K-0G#>-4`lXM_aI<_%Lib{UfM*GPVMk~hL1Y(GT=zOl}qtkR<~j(79dVmSDl zAPf(GpVS2%I-k+7J-!qkgha#qdwyP>u6JeitWWG-AA$^)1auuqZ8y5+&*UEL&~I1c z5`%vBSeDARE)xx851piw&m+;bYIR%^r2vnw*$gqFg2D2;I%7dj`&wPVGeT0QD)bfQ z;mi&yV6zeiF-(y8=0#LkU?Qu7@9OmJ54&`w6G60OQ9^d3pc)28XU8^wkBG^ip9321 zy@j>wzxYSx*0wt95R{eiAk@hKn+1ZQ)CuVsKF}FZ8#vsTz>v=XcLbQ;jObm2+f+Rqh`QC_E z+5W+ma2aN(VlJ4qIio@F2}cUJkw(`f@pkP1tQu20+H;m@e60aKko&FRf<2#e)V?)J zaWCkN55B{Km@^z3g(w<=4hlksl>QBZ|q&_e;NN-?-BbYkt z&{PaoQZLx*`vhZ#C$7mt@f5hV`K-V3sP%2lMYp|{U0dSN`DX!B(justswlJ%x7flN z_vX%^`z+9-(G*)wfA=zJ6TSTAq5F*(@tj|6ieeoeZC7uAyLKdJBKR6~9Mtxo@bjGb zW`E?=RkY48`m-f>p)UTM%XBx=t0G@X)N6-u`q14l5ca`e34ou^LueU*MwL7}o+&dcuBY6ErGg_()koHlqjzN$|(x&jT*^bE^Zl+96 zMh-Qo8`pkD?dN=HLJiVmz$)pcG61Jd^ZzL^{qf1rb$Rhk` zN5cYf?Kg|V*3!wI3#Se~$P*sMlpgS=Ll9Tc@-?EO#~ZoeqwB%)?AApT{>WaFhbizX z_o;IgAJKuXu1_aMPjw#f+Bm^x_B!8MKp%P8(!p?Z?3k*ZxN7|h>aW(NaB=0!^l57nZg7Cf{SpBF9@Mlz34Nl za9qhLY8tP+vW+IdrJQ?4qs?eY;f%6&fH5|mO%w$4f^NLmv5`dqL#pFL zS|K4*8ndg4!Z;Wu@BIM*Bv1}9N!MvEP>sIY(lo;uk856gYeoc5(bSGE0Y}iQ0)+sX z(sUW4tS>=O(12=&(iK%1n_8zb0qn9QTnM7%>_otl#WTzezoD%a=a@YJ!qEFOLYRTm zZSe2?1Of?NsB`$Q{=~RPMQk*(I!F7>cb{ISq4Aq+q7j%V&8gKS#~KaQ*0*)9jzDGdTqA_u7Arj<=b@E)ifNc)OJ}i54TCJ&r%!t}HOS z!#a8^oh-nZ3@Tf)vAOWo-WT$>$pLI|g17f~2uy<1=+DLlCG=EpPro&_C2-Z=2R=li z#PLvoS;=-!b*^Jq{=qiFp))(bx`4WMnRKwufaHtBo68U2rV_jM9*&wGzk*P7ZIl@< zz7VP5>|43^ope=c?u-VwAG2xjL$7g@LIkrNYHLYaVmSBu4Jlbl%yW(e* z@e(qy+056pivYI=Pj)&V_fX1Yx`$M1HoJ)1+K+EGU-@Ja4vEXx48inSP+d?%>*$q{ ztB=3eMvZlQ3c#H~pr4G{SoNK=)zKzdI-SV=7o+)q1#;i_X`$<3*j8d&+tICaRCnfH z#q8wbU}dXYIb?k=Iq1UZR~C}j-BVRya5+c^q0C}5Mp`()D{Y}y>IsOboWecZzn0!f4WQu=!O;}7ox zp)3`!7o!Q1F*30IhoIoTTrYtzK3l{D-lOD}~$HvA(r9f~0ts}7E@jUf7a3$d-jrWy!B(A^MxpDR~iKFCQn3bIvi z-ORz6fKuSraUy#=NYICh;bIsJ)HDHsMi+TSbJ8lX>XpOM-!iam?*+WS&f)lXo>PYw zo!kA@C`8aj=dzxD>k5)DS!H132i+%M{40J9GXA*s! zJY0}m(;2zhwP*TG@*p$6%6M77TN{BPm*mOznpQ)@deAwF z*Wpvqg6}%rWK3=I6=bo^^n>r@DB%#~+15O#Q+9Leui-M)8g^YLu=ih7CJ503lMv97 z5G4O*xXw2kuie%Y`8l};%T%SHk6mEciNpvLe6Yt&fd_XFd+c~gNPk<5PA;frM=aJT z3%N^>!llWknghWGVsbW&Rx(4!}d>M*Qo4#YJ^4_@jYdPEK>1%J&$X$y7a^ju)220R>eKRxPU=`B>=ZCHf>xnwf?( zMY<_zc3nkkoCM*-CvX`XxV4H5UILo$0iABdc=XfHVGV!1D0|^s68%m4WeZP#|M=@c zbQd53AWR&_gZ=dW)20+AW<1tGJz#g2GS&!}@Q74hun-j-ikmQNzq=)CYfe<~Cp#IO zsg!|d48fwWW#?o5easQkgfKQ3US;x}$}g~z8)0-Fo{d)eY#k*yvD{z5(|r^Ss-Mk) z*5+RkMni5?@X~w3pMqr6A-kZQ(#Qs`9QSVEs$R(Nnf$?+Q*gLsn#akyXCxo*3B3%q z)b>3*TZf>~V3@|))KU1px|9upAJbiWd>QtQPNHq}2y&HM#M`=1bW*xtC?JCojs+6g zZ1u9Y<b>lfGo2fMOVI>*lgRk}@^(=wNS1E^t=|Pj;3F4g`;mWI|Wuw5h1d7l39X zAJK5ZDF!Fgt(jG4uNnYRK(4=j7`z9@1>_tMPZ7$Y2FuhUxbcwMqTDoFIwL=HNp^k) zUvLSpXM+B!C(%hVnsvT$yXO|U{-94B)H&_FGMlobq>;V0o0En&`CqU_KB=0vLca`9 zT2hE;2yQj13glg{-khY*d{Q9jboZ(lZG>OJW*egI=da(BS+*JsR*)2g zhsjqWN0pwbWE8o3qxcrz6?o%AXO`6Zsf_H|N+QJS$?H#Xk}4X{oRrMln(#ZQ!wZ{ljBP|6I6NL<~D-_i^J_du&Mdn=Z-TW)1pB(=>|y?D9Q1xUw~c7x^Ji3;2Bu72@qN$lhL(_er6{csQt278qQN|xRjCjbsI=dR2MLQoRs zAg9k7;Jr`(7-ULd6qH7*jx8=~xtFR`%|2|5k_aU*=p|b^8F~vXheJ5pfd2Iw-jS#x zs0jwsgemk-31lk(xe#3;TjoMY*r>vFG#wRGtgD#vSi{#8kdQefav9h7EF*Z%kk)Zj zW(ZG+0^Z(15&S3|pC0hI5fHHj*Y-HJz__;CrxP1&7}jX&8+jG@ouMxC^IP4dR9pNU z{1}SQc$h*f4DRkd5$yWG+J?sT#VP0H_+@YdGC7FZrU8C>5KhJirg>(^_=&Q3%Sn=t zMoMN27NdRRIT%|v(sm=DD3lC@Zw_b#LVrsugV8njj(hA#5*;sl4?((ObJV)2tmv?| z7IpoESAZI?UGH1{>7nB|Om3wE;Slt;a60+aW{y}Jx+o}2n_ZATY>pj|YfrBIhG_b) zzRPp|*+$sdM0M5Y54(?md<7hG^U01UQvnP>opav;(SmOdIGY=s;F5cEOFrOQ5*8@* zfEa|eUjmi^2);T}<#S$Y?>I1h3ZTI;8d%_B_j5{o_UU*g7k%p-qsRKq-ME$~MVFpn z6nD<1MmyQV6TT|WZj7dNx%~WPsKtF$C&I4L8&yw|PQ(X+MLOV9|3oYx_*EWjNbaj{M9C%=^W^4@6L<$m~%gl(N{ za+$k@=cC#>YjiAu6Bx-cfD6`7S?|Z&I%`SIO=E3QQ8RaMR0S6uSv-?hh+0}5@!Ij9kXXtbv?hHe0_Kcg` z0D?g_CSsxsem1Q$$NSN(zhnkz6Px)huNg0uZPqCowkAZ^V)c(7e*_zf9F= zKsHC8GA3v>J9b_3EkYAgPRK}?o%$ZEimUih3Vc0u0Kn|pO(ij^6ryvj@{D-jy2mip zX0QJY=jlK&Xgrc}SqR7fHXoP*D=_;ZGxfF@94%W<;z)>!SH(K)WvdKcG2O+3dMbF_ z0-S4`Ood@1fOQ5UdMWYr6fd`9sB1Es4Er;|Q_pOp?**-gJR^M<4%TCIQe_>=GCA2b zgC~OyvF6d(*U6pz><;#Dc@@CyPM!YY1HXC+ZAeQQTIkiF3_+gjwe=x^>|r|+3Qw3_ z5UN2fxnS2F(X{~GF|)170^pZE!Y#`qLml*FT~ZT=3lMt=R*jN;O=Nu~3q|rnu7G%Q*z-Bp5zA|2uMnUy#;=>_p_&^hGT0IRyjRvlfKDaU$1p=1%OQvrVJFK zm5m2ta7VBXE_mw{^yiqa_I!@6bd2zXd+VghVu6jasz*|rdKvz%r_Tk$zNwf^QS_#% ziYa&z#EdN2lu_CR+;m<2+cD++>>bkw9t5gd?j}S!X{3BN%aK?0`3BqW7@r<3?JMfXF!5H0 zk8_nRV#&%9#sXAjLF+?xXzH-6ehcCZAesckmtg zL6g*u(b)m|Q*dbruH)G|6l%W>1(T2|JkwkPK%GfAE+dLt4i^ABPXuBCS0{x>#=a~y zo`Tkoz%2ly)l^*V$;Z^vj9LARzM}1RaxFX;EbyQ1kxmG^9Dng9*QPdseKe8-u_mAe z*)Hx*=IKsbHpCjwKGfe$Y@Ka!7dm`*OlCYsAoQ#+AF#jZ04C|jYI@TU9Q;O?w%75n z2X;5q(Mj*o#g4$K?VdndJ<{H&7${#zt}W2&Vsv^IMUsKZRKW%fZ1l2lw1HEXhxX;Q zQJh{fxz6#}|KPqPV?ljy*^6M?gx4`}^xy~#V)}x-cF`0|)uFGWbJJjGJSEIdUsUQc z22yLEOID`4fNV4&5a75v0@!Ws?sEFxv^AUU%X))k;GW}^z)fR>YWunRCTOl>_vCle zUr|clFKlm;g^}`oT5pS9@xSP5Z118k7$L(P2 z?{@MzhBN+*bR&eX*}cf}-Xbu|SlPj1+u_i*x?v0-Ic85w5Rq+U@~{U~Wv7yxOZ@m; zZOnE=&vy2SX_K7lc*ydxPbTP!$4g$*DG+QcyzuW)@o>ds4f|zefw@d9n~RH#;8!p= zn1EZNQw(_e>(3uE8UnGnLcfR`QVdpx<~d}x>PDllyYwxlk0!dIeJQhF&yBuEz zd;ox|JqBfFgf0N-1k|sdj)#DwzRP3Y9h8h5E*4E>emfgH#yL*% zfqhtGe>yTdN)Dd6_oFkJaJc@3u4}bz^w%Fe{PZO;tAI%)$8Vo&u9LT@$u+mn~kz@Mi}GOMgZHpPA`oQ;sUq)P(0T~26Q zX>_7DLp2q-ku|vz%J{2B?IZ#97~2rCMJ)2KQxPY?*!$eog%N)r((&}2ke0kGL3XVM zAMXxoSXWE_^MUIu?5^4qFW5`?{UJrtA=r4_2;J4BKe9)|#VxyO=Xf=mPtEw_r4z}& za;~a;(q=n4C^`{i(!FQ5(e2ZZA3qZO8OP?%w=sO@W7RM5nj4k9~PzEOx&_nnM8N@N=O@g%^FGJf8XU6bk0-SR;RXrig5O@os zYo~MIh}v{LnkZawG_nye^}Lkz_zjkp)l*7&(aw<`hm{Gc!Hs6rs+(?`@BcCS))PbNS*zAa?GjnxEXy51r?H2am`Ys`}MzVOsM93wOOP0l#aBYMh z@jr7``0(K$8L`tE)s}r6j*VCzsv)PzuFqzP#I=ls&D%V)Cz^Me*Bo~&JQ}C(I=OA3 z?yGtR6YLFa_4%z)e4IvC?eGoG!wH`L{_$t{L%;z2W*Ce_EL)Zdd1xWsqGfF<%3B48 zfJS+fstIvh`o7o4^=*W)g;emOa1K~-P$tkD;;n-q%_`lS!vdG$C*dG&!J0!RxCLOx zqx}?Rl*K5|8EemR9>NTK^-@s9Qe3nz>k#Tl4E};Z6pW6UC8J;r{fMo5baamS!7-*p z!i*OIAq(5+fm3ph3F91QJ8Qt}`ec`~^szIpuFOOPd~ziJj>j@1e`L6vR86w68)>l_ z-p_lvmuzQqv&owB$xC*?L3Df+J|1Iw4q82P_*>U;&dE{xpeq56Hnwax7!qgT3#P&A zyjG!8x8LAQ=Hy^L9b8IsG5E`5tNZlAn7YO`$WDhevX$jn>4L1*@$+Wcbw8WN!An=k zU1zenT|E4(8~O?quJ_*v!j6-wtvxbLi2N#3zcoALRF^H*&ixAE7 zIR^yKaUXm&09O}{VGYp2X!4IT5DMHJuK;g5H=g42JsGkvor~Rg9VM%?YcOy6@{w_N zyy+%^0nX8BGAtRO&v$-=KlM>{_CM_`xb1Zcb~0&pHe;4$yXfIUe`wL~F1W?;oCJ zyUAYBiH7}kMwa7b4d+jLdPEvNJEYSzk-gYfPv#5a)j=a;GSb_J4HW?U`px4V_i`(p z?|ioEK_&A&i6OeNuk%^bRlsqlIFrUA6n3i;s{6 zOuAb8I{)Oi8!9u2*{o{X*>f^{`tjF~5wyof)X2>_&zHI2D3YdRjM6F+&0}Dz!o-2*#dFU3B;V`-v4kzis2gp z9Ih{X5Y{eqY~h-5KAZa`)PnXFw;fYNV93WD@YV^yj5clTC6f7YtO-B3Tja_xH;sb6 z>LYzDW>gEJNoCuBgD3Gs{o}U)8$yL$TQ^q+XN#w2WxWVaa0G2xQ?TygXzC{chAs@1 zm7UR%b2y0(UD^cGT=gqCxE@x2sB*Oin&poVo9WzeU%3jLcZ?<&=INV*$>z77vfp$q zF-SLRz0d`_0ftc*%-_=$Ud?H5R}p(sK|&(b)mC4#dw`;2&R4*^mucZC*{y;TmQmF;~*T(U)XJ|q>cbx#b`adnRz!PEr*^Smolv&2N0x;7{HXKn z`6GT5$>k={{V58OgiLVn%vdo1i(7QQ9g*?lEu5zL@Q)UDcC?aJfQW+~s*6u_b=ALG zs#V*QzbqB6Vo*?`Uoz6Qbh!~l@}ie-duvk(YUCI-YOVb?wp6&}n_WsW!AS;Lc}Mi) zt20R`nWtAzd)}*UBLV4~H4@CoWxA%r&Vx_aZWEA1Z3bkjQV!*Y4J z$S;E_p=$*4cbHL;aP&iuS78T^-Qq(p{=VfZdcaS+)xfe<)cz8Ub zwN*V@Hzmss!I5oFhsn44o1Td24gcd|?`_wyRlZK=!zilnksvt1UpwLuux5v>5NZ7< zgwEq0Rij6ns}szQ_XwJPHMJ=1)Q6qyCgI?|3Z&JAhlqr?=*CEO`*;Aqp;amTbQ!4r`8H`VMm|?7=;J?X@&&z<< zDRGqa@L}W~9Gg$ruA_2VeMsIwfP#w>+m*v0G=dZn1OfRA47*KI@U3!!~Q2Su-H+OrxT^~yJSUf z7t81yy~#&>dP&9`9I&OZG6T>~26R{q z>JmNhZvzUp;TPRq>9W{W0>>x+^ga-bE+YIBAYD;}WfTIf$_O*%L>8e6Rtk*KH%3vW zhLp{nasDX(H_NNs6wBt;I4)?z6E!=It{KhMHPx~EK-SY?MbL0%=3S*De$zJEVczBb zn9ZibIufi*aTo!Z()hHJ&2aic{SDpwZq*oOc{)68C~rut?$6i zu=Xq{q>aq#>of%au78Bb$ZE?aVp(PVs`%S$GT@wuN^dc*ZO3GcewkJmI1`wgj*4f& z$?hFaMaG8341J1!$dr!YF-J|Wce4foSWG!p!X>fxI4YH`1Hk!T@~#*>XW`GRl7F zK__vWEPNl#EhN_wdsyan-Jt{Q$;-7dGX21%NzXlybNkM^fMqz!n9Xr{s6V4bGBLqE~{qg7jT?6)2Z(k+z?>E&5-2mCQ z055QF9&viE9evdvae!}!K@2jEIpGX1W-$^d=}FLx*9l0MyR{oCm!<=X^$MMsypsk(Hn525XMz zIMUAZ7*BKvi0eGeoBLK4-xlE}kB-+QC^tSO{IPpm_~Ij)ci%!VcR79nH5uq2bUKq) z*KBeSBiw<{NN<5xHXputwI;Pi(;R0(HyKr9>ReQcUhv&GH)o-voNyg*5`kwef-%~n z^;Nk*6%gX)lwaA?vbo5{{0hk>~f&@K!whmyyS+Kl2agACes$ zG+muK=e^F9OyR7KbrhNRb`;fN9t3dNhhTAb&&ffGtjTo}!7pQZ$YrggeOp#3=5cLP zQ6qtG1tlAmz?gcp9R!AL5wEQsdNY!2;}*&3<7~U%WRy+Gi22KNw$4t`>q}E|=6{?% z9g72W%r}4jPkMSQXjZw?(WWiiLOt^(dOF!i|K$>qy&gce?7At?wdZ9g9i2u))!J+# zG`ayGt9Td0YHliU58LFO+0SB2mlnrJF2oxl&i}E(_)yicFc43-&A7xVB%3zRw%PPe zZw7OtHQZC9bxtL_S!ax$o+Kbk%EHYSj3P;Z%Fx70MVzD<5H~cSFecF0^$7`W zUr8}q1AkUpf}I>xTM|eY89SNOMB{KLw;6=m!_vWF1wjevPR@V*=YLOt+f%HU&3HBx z4}b+6!g5=@6nec@$9unWRW>>h*|Huo!L<81DNcqsqXw$$+Xc~&j6-&B``(O|f~H_W z79o^gy|{VFfIdI%20F#BvzXFia-!)6p9~Q_@aA-$bsv~;X24xQ=N7{$IJ{ec44`QU z%%gjw$F2uv@WDe4S{-9`R|vcEDH3vq@oB9iyz#OHef1K9QPe*fh}Vl=_R^KoD7@_5 z4H=Z#DpY(lOd-mb9AJ)GDJ5;8QP9h{_lz85bWNt?^|=f5=;R1lkEuMt6+gS7vAWh9 z1eM)9_mK6uR&pSlfXXb!qX5~p@qs-wCih$J>u()F-(BC@M|j`Uul0<$U=+hKEOTOJ zo6QVQyhW_eAz;B(mJOgc+R0YCK&SuYsZJHi5*|+Cy$heoh-_6}_8DwJ`l-xp^@HnX zGaos^cK?LqrZGLHBmUF%wj%-pFWs}Rz;5cV0J^E1;2LQNE&`Bs9GUaS=9-hpB4?E0 zn|^BI(e=-D1ms)u@#ENg+(E?2MnyFxqK(cyc(UNUsa(#4ve`9zdg!S0ZC@ozx_IqM zoiT?uiUa-Zo}X2t01G$WucM=jXkhQO>7t~K?TnABF`AtZXPu)d<k>hq`wlIB{^_|7x?(HWVq|uraa0!f>aMR!^m8>n0`#lx@H>_ZgKzwbV&zR9y|)@dY4bimeaKWB?%N%E5cIZD2u zBhO&diMX~7h5tpfPH|VPn*kpz(+y#V@`WRbRCrGr#(3p<0*cRaJN-&PH-7$sdncl3|!11 z9|ngSS>Vl2hr2QcG4}x@(BmUoDU)O1yyhh-{?&##O@r~riG}=ziwiuAc%)ZvmW1=F z2fzqTyJg4z#+51$dO--D;Ub0azL8pQHxXF1NNN#you~kJ!09cxv>CuTarFiNjc0;) zyTvll^NfDTpz~zG3QWt9o1;&AtDT2S1}tE0S4(g`ad%UMt_I6UZJl4kH38CmyD0_h z8nA3yqWZg`N;Z@{)afjqgWI>*6r^YquJIqPXwt!rrRvwLN`SJpc=GM5V~GGh>E&!g z7MITT#6_QFtetN@JD7BFH%;jvcaeSoPg zd#ct?@gV#9On_)1>uX;XJ}8b2x=~X{bQKXxL=b2i{rPr(F7DAu{A@ZICZi8O>@0+6 z;lJ81FRAXVzy9ptoSNVlkP`lOXMA`p2P6l4``0;V z1iORQ-KhkQu?<5R>l`-$j#6+4G$IH@eM;X=P=pY{GboNqLR}+BL$+}4+wiWkQ(SNu zCkKuAT2Clj(91a|N2 z0C{o=r~pV|w>D9o@C=optq!f!4*zIjX3Jvw-N=Q-G#a9)b%YVN^$mf;=ECtz9_Wge z9PZgd3J|ps7_G02PUi^;epJ*n{m^At0Sfm|??rgcU}uO0;EIKl!8S^&mX1Uhvk@qo z1qq`yi{mM4Vk|==2R3zE`6`Of%lJ7SeuCSN&UqbpgbD6MWqY9oIKiJzIS-vs=cXdq zj<0?e%%V@nnT^Vz6(qC$&S;P@Mr zE)p0H38lqEz%orjc5qefJ^B%Dc4nu8=`v zN0o#{A`$6#H-BcnQCM~9(GD(C7mEYzNSQWaTOi!iwJ*u!3%BhR&45PVLjk)Pn3U5! z+BbEh7TbmsfBhJ-Z^KZdT|et=5e3KWncN=!+-P`Hqm}O-n~F9ygzx0Sls2_Iv1pa{ zbzD^$eOo^xw{RrqJ=hc#V30wzbf%kDt*uUlVD}A8v`PHbyNyQqgS+UDiE#xMZjbx_DD3Bbo}jc?lXXZgNBg^kQKS>(i}O4 zZp2;#nvQ!@D5DeW@d6zT7u!2y4UAGRNEv>s?|5^-GUgbZZfk1Zn7p=sx%L6nqyn%y zJ`fYaroNoXYH;TFAw(bR4{IM`Tfnb^tVK}x_PyX7pG>ISP(#hDe+I{4)|}G70vn77 zm|&~3;asC7VJvkOpXXIgPP zyOQlX>d{k}4AC;>c%B{Y0}i^AZbik6V7{-b--9>jUF<{HM_shAm&o)sK+|RWA;(!z zQpgsgaL^G+x9Tu5e4%qm87HrJN!ivz1Op>hofKU)%GC+QG{`{GSs=2)ZLHJfm|aW>B4zcvWypWSZs%Em^5j?DRv`Oc$# z_MiixN!+_1#)$Ymf$|zpSeAf%`P?=m1ZTdQEKCvK;i+xM++8T;Z35w)v>zJ=3r6 zCAJS5tCp=+FPmQ{jizAFFQP-N-ewzH(U?L_unI0ZW+Vj zC^S?nhH*iooI9M`*B@-bj6l$I&c+xhETfu~)gp{wSH@X3cNL%>7D{8M{tLMrf3QGxw7WM@(5*b|)#}S2383BL^GQ&IHohxGB#Z z?w;BjJ&cmPmz9ynNn-5;ni(fU4$L`GSIc10p_330C3w?WG}6`QulGob2)Io<8V1j@ zgUYajud}obI+BAG4ak;l2lu9oqPIG;L2E~e$Z(dl*a&(qyAK7OZ zo_uwvlU0AFJ2DEgqfc2Kxqq+6`T5xZ9I}t)m{rJlgT42OtIG&num0wDgYo%fu35l4 zerCJ$upo(>e(=Gz=%Hhv(diNW_o#8G{Og)6y!{B!INnp0!Ss`?Of5*fIE^f?F6_6r z^y$ua*_~E`-llm&hQ>`fvN28zdDr)fgLsn}avFhK;vi$wkxag-hKh}$Vf51tyDWGL zJRVfKSp(4*HQCo5T|P!*2@UHV$cybOWj}nNu9#%}G(_s-d(H5{A$zyrHLXSQI( zAy$)@Ez72nn2yeFBj=?x%px6vU;V!7yyz;QjuCR(6davJ`+~>pAs^}bJ=(qkc3hop z;9w8I+%#R+CZG6t`qQIKT#pEiw&rjYP2e2M-Q#T8a5^U!2J0~z(v7xfQWfm^f5V-I-? z7A>vpQ3t=1arCnn&KLEkzwm6xOO{7@^)ou7gH*Za3I$fC8JKKDpts$!6KyKo=761` zG5XglnVP!nAMnuk4bKIzbIx?!&#iI!$g zF370*lleNhY%U&fY%{y3GyKzt(k`2A76_IELWjbZj;*YdE(GCiWlUD#+E&#_2Wy$N zL?t{?%mkqtKPt@@h$K18+SB9HUL)LPys?Xi4xYUVKAxw$bR){H(*?U;V7^j)^W-|o zIG@gRQl?%u8(@@E;1!VA%AOv;_hRU_0l&pYx+BY~8+Bb$31@|1CIdw}&<~wQG%Zb1 zpX?|JUIlZ83tr~hH`)oy0_~EJIFklq3V)tdcsTX`*})U?veN!>1;Cz-q}s{gV?o? zXoiPuLmdr;qv55egLt&p#L**;6fY!H8HJ}!--tiHz<{nQi-TxnIQ>nw(LP@5^KH>Wx1-6R2W1!vs@ct#I|$ zh-9L1cg{|rATt=*cC{uv*CIg^cpsexfl}E`(V?;9;B5C!4kIb&hNV@jOPb3tQX(2p zhLvd#M|F+<_h{_kY_X3~WO(GV%)PeXKHqY{*Ev7lHtd#tR13!&eQ-32w%BJO?)yKJ zhoL5Ua_DV~4As5k{gi593s>YV5X$_lVV!*i&*XVHWfx=?rnTAQ(h=0hkoC5=(qo+# z-pORrOGXKG&Or-CRtu)V?b>ceOGX|PC z=K>&636`}DjA=BDR5@K__s0L^(`Ny6uq(*)8sD1Sj`7+4Qi{CtO&s z-h875rqGnnAo;emGre(@ab>jZ0u>n)11 zXKN_g7a8jyjZSw%n|A)yW6h_dBMyKYuFP>dCLf+$tz;nRI={|gbN1oby3Jyzn$a8F zi7nKzu;h9{i%lm3K1n9%f(K@JW=~&G)aiJ;4Qt2#$d+D~U4OqMZ}gIdB(UF)0NxrE zdibpC$sUU;RKVQWjAn3>`4Z&mP@zswZH@a?gc`VG>~^PuEupL4kQg~cOmNZ2W~|=i zMDZA;;E>5i^jY<~^l++cw2||TCMd^apd^lhJbA5y5nqk+A_xj)KHBkRKEH&u?{$0$ z-PGPC?WHQSv$N5MG{T#LI#r?$9ZfF}1F&i4Be^{N`JX?B6sTDNB$^~#ok>Yrf=Ds@ z#$3jj1u)7zhC!VnsaJh^8&g9TQ@`E?LL^j|gT`Q@9YHiNJ0Fvbt#ZF;*l2)Yh!Yb8 zn6Q@-(6*Y$*W3VaBxY6G&~-IcC|vsk_?8m?qe8_rQM=a~hpJ>G;R`2#UHuzxe` zORzF9_dl$|iMOrqP(lu%ew!WH}izlE$bv{nPPU^^?Kv#ycHhKS}9$ z$T?o^7vk3|uDZ#|k1vDM@Z7W0(LuX;Z(u@pk zgU*UPEN-I_`h%zAK#iA8{h+l!@V$YNP3&pE9g^q7f>;)8cn=0yu#=Eo<{eID=;7$# zvbpH6Znc-=+v}hGN9lCQ);Dd1-`XnkzUhpotg&mRkw{N8>zh5!nPv-B7*K;Cf)jxWk{g6Gqvqv^o-E4SSyqKc>D5;5N8InU5`_Wyo@7>$n68jaO6GNB`h2wBntD+7B|wAIHG)7=aOIy-0L zMipeX*$fxfn|B<~*qkqZ7F&v2eDkw-W0XE0A4Ad5*GBA;Ry3W2WBvZzS=p?0{JCo`bSbj6$S?{vir4?3fkP?NvoO?v zZ+A)f$e)9VjMg`1h@j!4U`E+QrwXImO`EKARZ?b17`9-Ta+Y;&ora#OHiIOAn6Nm% z1=8Sopv8i$z!h+!*q(UzB%~Le)gH1Iy+&9YGRNCQh9=Gw6rW|}wZStdT+qP_9aLr~ z2wAx{r(6)ZX$Cj*+1^RUWG*vYP}M2cXbzG9w)T<3u{l6l038eB;M$Zx9_uKwSJNB! zlxD$z?8!DgO~nD3{Ml6U2{sn2WRZdrV={7JA@Cn=rvmX zR=@7B;1OY!#RItVz1gy({M2UXfUV}bev9dex12J0Il z&p|kL&{B-@IZFbuD`cG$G_~CgQqJqeC0;;PeFnh5333_4(ppS+-nIJ1D^C#HhKTs! z6;_Jh!#LrH8MLmY?s9j5R%DIDbhQ`tP_((!byyp11P8KhCwUA86E!EQ_g7X{(l!h3bxjHd5(N4x-g(>R%3@>)kLfIPUb8SIxFbaGEM zrqd|s+HvXrQU+E>Mpp?^M{^bM(|O;f^%h_dQQ4->!Y#hW!OUeN%If$L#J1kiA9`Qv zJm;{<=+-r$xGZ_bn_iGN_QAg~`sl0U3&^|Ka5BWbKw$%d4xo0Xw6G^AraRd)TF*z1{z-wau7r(LN$?N# z+6Fr^OPo$VRI{m|hadSz$C0xcfORk(yIJ&S5)bgnC`N^F#u51O4m6VaV*?sNzWcDOb<_g z{`WuOdJQmXGKI*>fVz}SZ~l#_X4Dw(+?+PUcBlBJWgympOuLAg?N0f*V1yewGu8@& zvHAbLcRXf)`^~$|=#g_SWB*`G(?GDbi$O~1bb@m!Tt0aDwJEAo)eJXavbs5N%@>er z=WpA4gZX5>G>Drko$+?P`Wv1z$}!Hsc?5;zwEdkURF@G(!Zr%5YX3KK>&q71eoUW} z8OvYBGQ+`EWgM82E5PPuH8@0XDEqt7!UYMf@~zRu*v1FU_+*R`CWu%&u^9s28`h#k zyQ&&Vd<}~)^M(5y+9Dp2K6M}#Y#Jn*PcQw$U&>toUf?ZoDeL#6&NPhUgF(YbDjwO7 z-UXImxy}}5X+|_~RAzhb^M}Vk1c&G6dGmaXY%+1Jl9I=lBajD~-9M0e4*kHAMj z{WjX#eM<<4PEMgiAX|>8a}&_Wrw}<=W5=EKw!kj<(08^l=|`iEaW{gpbAE#Obnza% z+v<5RaY$CTNL#UY&j*X$yi{+UOFttN3_SIJq0*FQc(yj=L4#{bR&4yHLuPy7-D`yf z1pHRr$k%nawoVZD3k2a|FEaAIL*p@W9|aJk2XdM}fUiEjlF^n+@Xg2RKKq(&RlW@d zTkM;eMtxmSwgcUNpQ;MEN^ZLevjTN%Ui_fH$~$*o znuUCco@E96;*8z;pw-c)09O#lz&TENyY*s5Cizz{de{dA$^g@+rJ~KSRK{FPIg1ZV zGS~`uWHqAsSl`L#>Bo;h&xvNx{b|kUBZ^__4F%@8dG*xwsBiU2M+d+lewJPlmI3>{ zX)7i|CWBcJO`tMi(>7mQJeLKWxDqPk;HKwvd~_&~(+>w((2l_k%Y&`6DnI}LP$Nl1 zK~y6ke`$&E){6!~Sx_TN-j&=_nZG`;)e zT_HZnQ5ImzAhK{!S#`?X7VLtt5m(?ntI_iLWLDi})*LDx1kl|{O3u-6qZsu%M)R_| z6luCG6Y4BFf~y{RWRT`Xr)+6c!ABkVZw+AFbUexI!tX9d(K42B?+Kn!!?7>Al#q~N`eI96 zxWpiC8EW6_=rRZ_?7o=c>Mz+C9lfESJOtMv*7|tx_;mK>B}2dlJ^9~Dww4&I-N|-s zIIh5h$E_~_M%Lj^)>BpW_8VNfyeHs-7yZFCZ3g!}+8q7KIc07`3f9R}C#gO(@gR}d zBiQ3*qqpEKNdUJ*$Lus4qN7Y~xF_fIV-a*yI5A}!jTAfENaCEncCqqB`}&T`H(HDj z3+UH^KL-m@tSC7x+pl9M@9Bchl7r5KX?l%-phm!F?a;-$L$1r~l3 zy-S?IP8pm2?RQ}4BsjZgjPCjqmG~O@^fev%R!=a;^JZvVE7{rh-th1{sz%2o)j;7k z{~V2ylIwsGMuWv_!h6ZmqIsQ1GJQ5mt9%0hj3o%Rt1o!^^FMZB5lW^NY{P7h#UV}r zHDK_Zv!2W}$Y%%$dUIojr`55|?>#t)IxYm4#3kDZUJxfpWei-fi~@WFrN4T~y|*}` zeHJjr`Xd9)Sm0xH8c+p#NqqBxt|b8Wf`=Q%4{Z zw6&4#2!ab_0#oPT&1HpJC$Z7UT*oYa^^?UqvCvU!CGpK*FDM|c|2aDVopz4n8%ac8 z_|eYT85MbK(TwsoOs~we9>vA?UWr;ejBoRT!<@<#k<124TM9=HanL8KYOCWDfSwCF zdnQknJx+RC4h5TC#uPRAffOATnZDU`hw$C`v>C|l2BQZsYQt8y4R*Xe%E*Az z)#X_a=Z#GFDp&MDrv@CsUeZP; z@ZwK4vFy9^+HIk@x=)R8$->n6rm!cAY(fGdxPaBqMh5Y0+SH>(W^3e?zwLQd5unqa zYvf0!2getk$iZPbq6QhdY9vbUTkG;t=$qTBguB@0^=q z=sl4ltPDiw2zHh;L)WP-kcU_BB=nn}A}|MMgbQTJ&yaA&I^{&GDqS9bWzcTi(vetL zf4IVwBakorqPGS*0x&7fLZkq)`eYTYQG>)jXfjj}c6B!VU4Te$bOd%Il-@-PM4FFt*)z)aiVhy^_@(gH|JIzLJ#1# z@2?fptstJ=MF`n0IMeBauNyi;fAaR{ek4I(r@juB>}PT5T<4)d%;-P{vW(u;#Eg*P zX_LGK@O!b>*$Fxrx;9JCtLHHg?tGU_?9P1-HGgB%0$o@u^Es&nsIn`mZdzKMaI6!K zX44?I{gh3hx4c=MY6GQ{xDf`+tDp!Zw;zzo+4(ZY8IdtbZR)VuuF=g`C%jkUj-JYF zLymst%|An&e0f4LBXhX+{m`$Vv+(ETm4;%H&c{mDcFnYpG^akQNAvKqkjP`c?tMzd;va{n9|W#Z90yTreAn<1zVpA zj>qL}G!+k$4A&O~`A~4kefQ5q2h;Sb+rH8@+4N^LpZx4JLX&tT_Ej1^=nmp^M?;<0 zapDEfeCA&41kWXK-``iO`q|!q1iQcOz`}FEtC-PlqL)E20U^F7sGob@v74p^uWdksL0 zP`)jrwV5MwJZ)F*Jp(ucmwbXTiYS+{f$gn;$sPFkUH$4V!1W_>m^~nlHq*y@fx_Cz zI=_C5mewf%J9$+KSFpzu#{|Ehs-1y4h=;0nUK?bEYEs1JsJ?=Ais5?NUD#bohDW=L zNvLa?XOy4w&vtlM?FNN?g#(_wb+-g$okPuC?`qcy5M3vC@LU+p1w{2chCk(v9(s_B zeFY!ea;p()Iy)H+FnM-;T2y38PVhx#*|5b`xRWH%iBEBY^OAns}Up zfu_i*p6x4}?qZk9c0Gv%Z#>~Ldg^D0W&Ce5fZ}v@_e6o691^4TAk3N~(qTvXWPelC z(B?0dskN;5C7E=xQEc^xqn~>u(#g*Br=Yn6DHx27n5F^z`sq5`;4?-e{DH_Bj?lcD zduzpi=@WqYXNQm~vVGX;{fjT?5lUX5$PoB^Fir1x62^P9Y@}Sh^Bs5}@eeb>86ACe zGi0Fq+oxcmAHh07vA2y3A_`Zx+o{^co6RbaWB8Y(3C`FjGsJgtW`#ETi5GL3eQwH; zjYik#<8!F&n+&3%18SZ>8}9}!d|;fiw$SddN{ioUNgrZOiWyByqIHOY+kbKV#~*(z zr~$sIg^2Vgo4E{Iu*N^42)&A-*Q+Ci0bYinoB}JK#QK0GNQjt}(l;Tl$F0ts#cra` zw?!~5F@k$iX9en2pMc&1IU^y}jGF9>deF^5j7BguUPcdr#9;OJI1&JBy3Wzzth}K! zMOA;h`U#F;`+2Jp#bHaPuAR@a7vNjx zXEf-z12N|bO6^iwa2BAu&O{PHi0e>lc=NVu3eG|M^%niwlQHLqlqL?RQNbRs!dRl3 zv!Y?U*s9}rQ$F~bAsWTh%|@5!OUX)V9o$Bb$(ar|N}xwSH6dMn*rr0M?xAZ4_-2rU zUjQP*aB)oQW2U$C`;>UuQ*_II$tBpMuXcD@7O$*I9yhXp6kMHK)KABz3M^#ro$`|z z2Z&B`gSYwwdb{`p+XQP<;%C*~f@b+S;!Q}6FsZ|%4JOHD6q*Wh7f-iXN3QZzuaNHZA`PoLe zZ747b9#83tOn^-O;=pe9jSjwJx;mkE{yAIRTF)F&+@nK5HrZ8&?%4vUOM)Wlg1GvT z-uXz+rl@d9o&!nfT49T)@mcKlO+V2Igi(ai$KZeQI?#Z3z-HV;%a{36pyxQYv?nw+ z%XuP=w31`?8DBRg$QIITw5T(XY-lpeCyTZ7kCnrnd>=Q*dU+hwro`3v(P~G^*px?^ zC_ICs=*~8*JpmhjZq6_=$(S*Xpvv57YP4X;I7S7kGL(%97(G}Kjp;oB6CB3~pp`z< zX)#yumr*`{ly?(OlHN`#L1qfCZg_TQZ&I=^W^>?WGUoKi$FvjLHceBVEiiv3OokUc zN?W%7`H}+%!AqT{lq!#$6W~AL`~BDZ_}zPfC&)Sl*;ugn`qvPZh$xMNQou`fY+GC) zWgySBt9;2851ApxM|kY|U_D(;!9gJ_#$fo;>1_kOY-a($xoXic$4n5mC^#qnJ?|!d zH=mtDVrv~kwf3~-KufYNLl>N=mqA2_fTx_GRb=EO60Fnz$#k98TEs;GLr39SM_nXy z8bxG{b@tJD+b%hb3dwBQ9fA10>7=cLs$w*c=mH*G2>de%aY|K^{9Q&V&Cl z(WvY#*?0UaoD)sgv@sZc`uo2>mOTIojJfkKfC4aLsSkinWz=?#i-`WJw&4Fxm<*~q z%Z8!x?XXUPhnSYxVsG`CIOWjG5Oh3WPyk0iM73#(YOjr5Dj7qdq9Fk#5KJy-fFom@ zZo9UOmZCBW+}!A?rZE7I_$8wp47^l#7Eq-Z_n@H7@&Ox(CtaM!L2u7|G@;o|gSTsE zLdh8Vh7+~%vFRgo!NH(-cy%gAWAJ~kZWleqh+}TFGGk;Yb);lbwzUVX)rwQP0b1DE zUVWBPlV1@^e_I&ud!6NE869wU;!y<~?QnwNY=~Pq|D)6e{_yX?e(}pk>H0=eI)VPC z|G7#RnP7FRgY*!eY&(S9j;-C5+wtCBsEi&p2qt>(ZgGZNo#eAoZn72PEjm7>hh>d; z)hV-~uPhI5(`CE>-#^zRr{HflNzVD19GB^FHlqbh(H%M%LG8Vle4Ts0g2JXi)||Y^ zsgr25ez6yXIiC?#uTD~6+EhU_F9FMGci#yft3r1I`8poDEBW9{WOf0)deNk#s`=UF z9QN?qcfT!{v(pOkhDJHHZ?X{D=4T7%Emr) zr>!{$JKgN+GYYr92dn=d0YZxiq@x#nzk9_&`1s*QXyNjx6m1~AS19nA{v%phSd20* z0aN$nF1jJo)@`)X;f>Dd=;^l)Pe!1l1rHKiu(&Pv$zePW7UqH-{rvM5&&dK%SNcv? zo3@RIf8)1i)nm682eZx1Rn|7zk$*RTR&6mPyzvkVU8~MtfBrosTsC_FVCxhNBEX#5 zJKSb2A-il^J=YWb6!{P=gq|rCVCF;)90T%6*#hbefdN*?z+&+BMAt!}wlV?XgsIcW z=p)!7m!P)kg6P}LV=`7&@$6mhQ4f(}V{kaX1;Opus0|a)!9YP!hU44vHn}Fe9D;BJ zAQ`=YL?`H=`b{NF4v);mUl#IG+r2R8y~a7(MqTi%^OXk%@U60(gnV=l=+jZ{rCYN7 z&gXE19G+#@wZ3`Z=&}2$;Np=dyN*^evY)*=o^iElqsctEFv@7#c4zc0fWv({uBcBY z72P`jY-Td)cRFM2bL{H!Fow7mv9b|f9jll8kc3g+B?^*==#93$o1ouxtcJJFDbDhq8BpfW#6fByS zm2*CUkRx!$%E;YKWc!47_d$UdevFz*Fp?&#ts%tQ>B10DwRtEYwP}uj%PAzvEPgwTm&pt%n?mRqC*J+_j&`<43+5ApXnhTQK6k< zdy^jDpkKzZQ9v?;=)<#&f=3Uw?~4iXYtU~dV~g@RpH7ASp&p#A|1k2{lRQ{fURikY zeShGhG}_pRRny9)z9vMjy&&Gis8r(G|FV|MkDo z7Cc$NgQ%GGK4e2JS&OZp_NY%-fnuo(f>G@5Y)Y8eF9IguB*gIr62&2&=u+ks8gj;H zq~bmSQ)af}d7%kt=cK5O^gtTUn^&FGV<`P?&4fd4O{Q=4jcV{fI0Qg2Zqf^3$3zq{ z;iP~CwD@rO1bI#<*hq5hjW>NY1%*SPmg(Jt4^4q2pCAi*3lg;@!-X%-wWFM)@=f6| zz;IibUk8OIJVwB>J`|(Pk?UPzOk}l@%E#*klVt%ye6snccG4U&%TngZ;Ucf~ggQxb z3%(}N8RYSO!7LM&5i3#2mpUrc1Xs-r@xJ8Dx3tz zSF~$TM!6jzu5p5bf*nN5tZQJC0vzD&$WVd}MWg$YHk#9+M%i}j7;d98*=$5oOO}i# z=&*eDram0(Nnv~hW}7K$rJ|s+WB=idNR;TeX>1(iJXlv@d?HVWk$g;*RV`Za9t>%) zZ$6dGHnj_Sqcc-@avzD6?r)3GoD!)cx-31umU;Hsw8m)7Zp{B=PexI02wMjkWLBrm zQA4Ctb&#R8bn&&T_Y zX5Wx+)UW;I%14c?(IOE{M?>eE?4u*l^8;ueIK{avl+@*9gQN3uc}d0W0F&8nUlIrF zs>>&1LudHQD5@Cho*8R=HWtj%&{bQ>?vj*_FDVXniN>be*-i9{t!Oo+5UInz&I~R6 zS@u--P-C4RK@<^~B(V-)X=-{A}^MPA^u)xu7oCgVC=m;f8 zWl#%1*u(b&FY2lG9x0KdC!4^{I0ab6UWPOVtBd*V{sA{7l7&anP$1fOvt~4px2l^) z-J|JUB@6u2Xc*fZjijGN<+mx!I*g1IT9%689E@J6?+0)b#f z)BA)j*(mTPM~~~qO@|wCb=Y|Ta<<#jS$zScOS>qT#ovbUkL=uMkFQ}@WLudU+Q{r&>DlIF7JMRBvMjrE#PHYY zAn@LeGgj!ZMgpJAScmc4|0OKZv;<|g9$$;PwQ&O~dw`LR0t8mP&i*sv#gWR;CLzE# z^D|O1<;g-e_0?r^I$Q5MSe_nGd<{04_NSeIB`I-IJMz4B-~%8MINlm9nsvCv4_!X; z7W9KVJ)tv1OOQ8x8$EPl3M$Ibfmjb!{aUwO0zwD*h>-}Ov%~1vh#*)-YXbQec{vYz zU<}o9KNG7V4?Ic`*%%+$4y|aYW*2E0Z%K$wf-O~_?#Q6>@0$isyHNE{s8N46SBrV- z*6u8&Bir_Z`}E_lA2E$V3J4)(bp$bW-VHFW_@g|6L{y+yP6n}MvyfeH8tcJeEH}5A zA%`57+D}nB!O04Et|CjvAuQt%3{(a^7Y zcn%CEphg?+&pQ4Dk&`x(=DV!32BtLdyU{_)l^qzt`12t;5?S{`I0b7jv={K^?qu9G zP~emuY*8-0$2nRU2i`4or~GxUoTfUv39CA)%y7uBcF)l=ywz3q?eSpBb<=plTYPUJ zFGI9w?EZ@;t6MIDmuJfH%X4n}>M`5r4?I|+%v!m&0FEHFMgD`AOTypv=OTU*Zt)A^a^ zTNWy?IM?c(|h{C3mR`zN4#{M zj5hjwIF(LRXC=VC)S)Nh*@}5(9hmXk?8kZE{KFFv*vg>NPe&7~BRcmDNGM2g(=dFA z&CU^%rT!=Z-Fgk@3tnKr*}9zeWcqAVZ-D};F5fOYu(KBO>v;Fmq+d2MnMa#leBbjc zwx&Z81IX9*VRWx<_`nokwgV`>zDD?_2aopbmW*|HtTi;=)1i=vF#gzEuHkvSso`^ux zbR4RNtYjf!*mMtFO#;b5yhtZ9f3n|NMe%BzhoX_sYy?C~e6>*to7zp3V~q5U@AKJ# zu4C;gmFj5VVPE`&L>VC}1ns`9UwXv)lCxm%>CSb`;*Qw7^~><^-EA>PR~3hkt!Tf1 zg5LP++oHcjgq^GVT+E}MP4SZPAAkLyQ7W?*xN;rd*~N}(v@A&t&P>UKTRlGu0QKOA zrTOOj60fs$A#s3TN=$F-cKxT|mGbwJD=V3!(a$ZCKTPSecexCZ2`h%644?X;AAp~ zjk*}a;MTV9n*+~iI4wH1xn5@8HOeQqYZ4C)Dv5Hmb^f4rBe3f8qs`fq?=L%&;J{$= zaC1=gRQ;ZTZN~-=svQk(8)-%rs?{I6jKTH3-F!CPQd~r=)r|J&!`HIr>JP1IbEx3p zvmmUkt+V7Tj9AL3hnPxvs;$82-2!n&DeHQ2V=9`pW*or)a%&B2E*J~S?gkE^jzI>R z*(VRMY8g&JTvB#Afc?~mQP$#y)s1+&zK3t7PnQEzSuH|+i4X=~MJ`5Mo63uaiV1vtT1?;OkNE~PCtI9g zdQ+A2!{FdWY{l{XL|Z}(52Z60yo?}X10cFe6i!F1q2tQwJk#M~FsU{=CriTXd{agR z{%tg&kMOHuXQ{xuXKVF6JyvclWg?9x^(6-j+SPfpi*v|=BQC@T9o`I0R3-Plz$;t7 zn<(=&a3`}4qG>oJ`r(CVn+n)m?7C*exF9c1X;P`3bF18$FFQ_`w$$dHhEW^Bn7|&i zRjzEupom|a&4{mP+cbQ2K5bqg>Ui2sw6$*p@}D&tz$#JvZqk#=G*D4xPW@u0zZqD4w@3vyg?cTR5;5p__M(&aWO;aPqzE8xzsA%$w}N z@=2)cP=m7$U^HYdYnr+!9P6o$8mf{&C@y>*u0v_0Ha0$V+z&bjFWBnT=wLD)1KQ}2 zihO^)^`HLW+ByKa$Y?k#L;k3PpyUBEBC9N9IzKHprI!J0`U@odgaK|3?^}?nY!lqd zIqJip>5q(LtfNs+m*54`P18UcKHU~a(y?gA85i{WUSLkil1fR!d-Y$@15|(PfDxf* zYXyl~xqxa7?JxE{Hmb;oHbRpj9zeyxal=zY^!jGLWE$bLne;dS-89*(+_(_ z!QQ~hqq8B;etR=y1|yl{Q>IVHru~hOH+6DGGKNPxS^^^Zkzdb2}_@sxTYk%XsU|%ucZ3*w=#L?wx|EM&o;Fw;h68;CLpNP9!j?wHIX7FZoz>e7zm}*r!qC5+WJw{R>Q`(|q)}~XVw~1-Sv)zt@ zIXbstfIv3d)E*HW5GTnQS^e_}yG{py(yblr!+{+n&)6njqx7!0IZ6G55~oE;PX7V0=mJa7beoSx)^k5-1I zJ*hE}N9TE?jHAJyW5LDU?%1i|6cSVT-bjbk$%Z3Pes~ycSfHzUnRGZdMVE|X=z4pB zBYaTIsRM^s^l$BgYy%DEbY$u%fibF@1Um}*V}LW_cbS_8oSA_0=-4{Gol2bV z96(zg>^V&}@DRP5UK4z9L1qJm#xqia+u9E&V1#t9T!(p#q#5~m9a}ab@QjB9gDm(K zoZv-^i?gN_J^R``|Ak-MRA9800_~2d9AMSZ!ZS0g2 zU^cDN-}f?R^4UFD0lX1uhvXk@-lW45|4jw1GcV|D6ndDZ^P46$OY?j!h3GqX=LRfp{;tc|HL>r$_8%A*CCzD9s5 zpha5-9e*VjW$2sHn7>Aw&YBrOZwWbx@6v! z;id@(?0SzA!G^Z^lxd&|nQc6DGtz=<7U8oDi=0u$CYFSS`+1`SYC8P^tCl2j+IVcC zH)6Hi_RWfZxS#{e2=n~9@9>Kerd}6W-LMF6O{hfqIH%XJqH`m|&e-(RpMU&078h{F zU~EtV0ca3_Z?Bcrn{O_8&r%T#wED;gBB?eo+I3|o$EGdRQ0zF^vQH)}C_z5EjsS{3 z_v;vugg!%Xa}qUBd4Ph$cpYw79D14RJq)q(;Br6swutxidjd@BNlj;@3S|6S<81-m zR8H+X{_1gwaewscCsRtLhga725;($8I*;70>tR<`D2(S9XKlmp*PU(J|-9M4S7uz}>u3$?L!nm5wRWQnu zp-S0y4@JR14G)y_rnDOLN#}ICmi>>p8CviLbGAUUcvswYI0gBbT4rVxT2S5fV9M?< z;W$}u1Xf+6r(qEgBI_;spLNp7Z?ew06ZOe7TG`H~X@Uutgf?j{nHcV1DbZ$As3cI+ zb+EOet~n~2tE-d!xp#=yVCcxQhS6Y6=C#8u*#7)~GAnBwyfZxMKJg9DY#Ds^kR0|I z?etK$S)hh%cb8B7^ye6n+TcE?Y*gu(uozvJalLA2P?y2G z#2$6b^oOp?5N2nFd+$5C`;jzAZj6OJ2g|I$(?8)z26VHB?^S;f zMXh6`n{}Y{!=L-!4ly(aN7lOyIR2swY*Q(0V%u3M6;FIZZ%;=L+us&y!p?8$bh8)G z1VeJNQETo`lILJB^mpM0@2o;>IH$*un#Ty7ku2vr%$9AAn~g60{Y> zf%dfiCC@tkjZQ%E6Q0;$m*Y=z-8CH)y2Zr`k~{j@x0p6|O{OJ_V7XQ;5O)s?zLdO# z3yT|eR+61cNVN4gpZTXGgzIWQNS@ckX-@<9+b|L62|K+wlk`FGPErVa%xJCSc_Tr z&hmtiQ5lvY=6bpafookSJVEb!Qy!#8k$nv}9{WLFzcdl1-!N z%(^pNfY=?z;kdo;oV?}_CI1^_18(cCkQi%I#u7eM7#^49?nb+6Zue2gwmLcodsZ9_ z9GXTBwJXa6bUYNl2lr(J=0X|9Uh-nJ7rs%R?6zH#95VI=Rq{q~js`>io@y~H2YNxi-96rW3;Cm-$EJN>wJc5QG$4;tEKXtCQUbc?d_&i1k(K!#)H zULf-RjPpfy&}B5@b#%-EWG+8Zui+_8*V0K{3FSWA8 zjpoo$`wu>bkNAayjP^9%hyS^cZ5A@wl1=z4cq!{aI-WKg5mhi>lCJ@qD=u~IBg8rj zNuQXoSpW`#)){#1Ki}^r#^|Ghr~m%pPV}zT>zGoYM-EOX_A77hxVoMYYH?goJ)j}m z-sw653Ns80YKK7uyz*rPHIV@b)!UuI$PpQwpN31}SvD89%M=8U!O6WC(*$u!@Yz&b zaL=*2rgxRybWKoRpbgIU^mG1B3Cn#ffvIksaXNwy_RLIx>l!M-U9cOKgdfb<-R_?` zMz!$!0)@VgB>Tqd|ohkT#r7E!9*Fj*UfG|+^4!i>_(EDDz;>6m*GWIu!(RBpUcFL)+G<7GE1T+jkd4M$NWBhniE(_6#Qb4vY{6lYo8C&g%krx+{bUP99`I?i=@YhF z)yXYhbV8==*v3Y}GXBmAo7J!FYy-`w82swI@v2kML;_>Cn`$``$GJcRx?*@s4YXc}a8a^vh`E%?at@zv&a)WT?KU zL~M$OPxlP}N5P81-0y5R!bVZD&&oLEGOyfWQi5Xw$9xAYhtOx7{Z;qOx4M<9LOJUY z@Yy#*c$oaobpWeIs1zMYhJsgtjO?WUIs&NfJ<)iHrasil8T~^J`7H}!ELA-t&R~Mg z4Ddnb@8CG79$Cu-_K-UoPX;bc=CucEI_mf<^JJuU((HkO5xv0hub8TX;dK1J$Ub21 zH`xu41<`f997vv0RY}K)4~MC%wWmu^r#mnwpviLc{qule^p#89C)}p$kP0yPq?H

LF$sS!=}7;Rk2YnTn_8{M(qYN+bwmrZj5u60wfW($*)~bj%uNQm-G=BPi@?$C z0$hJLpC1TpJp%W38pY4Ht8*^4Q2EVUci1smPwTs#o=;F|C1k*l8lL{Uj-`Gb{YH*= z!^rqsU}Y1NM3)E?%;e(*Nn7)qzG^3M^2^y9x~Em*>M{!`5vi(g2?Mc+ai);iF=}np z_{cT{Mr=z4eD=W4&X?lkw`^;pr~oc$Dd-q=x{(rAiP<{q;20S&NHIiCK#!q2ngXix z)6ws-FA*i17F5L-G>&tAj*fUtg$sfcbTF$t9=f7ta*VqUbs{fIN2I@AvJI*aRo(Ui&Ue7X$~vu^mW)jLyu>Xkd_{N1RiN8@_ZN!x7F+16Z6V0!~0V5lYCz zqh25kC6Ro)Zo1OQV+%kT&GwMz;A(LGbH)d(a10lL^=H15lED>31*v6I6U^b=V-bSy zu5%ecm?}kYV}i6GxX2_)V&+@>(9I|i$xw()Gv&xx_MBLUw5Qk&pOeqI0~sEF zj8uli8PLZ;mZ>MlO%q{;fxylA9J-9^*YM$wosMsm(0Op3ZSQ^d3c$z>ZroNsdZ3N2 z1=N^3n@R==jy;X$@WOxhLQVW<1B^6<*i3+jK}WJrbXe>}n+}ug?SAvjEVi*HtEvwOj7Q*OS4eVqrn@uQ>l|Ly5+6x+ycC<-UBo$b49(@p#SFP7Med%g=fl4WT) zBnVu9Aow#gn?4y`{8m+Vrrql}))p8FeUTvpL2J%RGM;8%)thY#B)3VVGTH#l!1deW zLiza@CPbDA!J(6LEi&^XQ~@+E@3jqdAA z?ChFluD;}t_pC6C{O+5a0M!0j51saA-~5)R5xTnve#d*?kq@6f$;?YaGpj8P5amHP z-WE8idpuR1F@~pUwgn}1IIfyUJ8f1n9KL`1r!NUH){q^8Kcjo6YZL(nKatl3Yy+(gBH(k`yHe2T$M7sN^bgO5q46{@oH27&xl_G2xyzG=*7w3MRlA7r}yR4 zOMswV&E+`FY)^A#HSY~?qnWFjL2iMkq)lHisOYG3T2)8#_>`!^XywTB0`BkRb{Vr& zMWyh<3Fin7Jf*GZO#yw$pqT)c~@ca>|$8sm&b%dJkeM*NRV{hKr5Jl8nF z-Cd~MBHp>#*9=LGnqXy0)7C&c*$k`CI#-8Lri+<4-Yu}FC(%E^8TU(_%dW2(D>|9u zlWsTketujljrHlB4k}J}`SACbMj&g~)eEcPm;9C8L(j@gg@PK8tl^O@I-h_4h<@_v zg$dHvJtG1h9oKVe16%B+r_QXSEgxC+k~c~_%-P;eLzrwVcQSEeyVV6JT-v;5MPje- z%0+iO2XP>spV(B-87)ifDGu$ zr(@HO3_?qw^a7!tsxZlGf)fCw>v7@GQ|rn6d)l!PD<8sM&@%wVlm{k{NGbccX>X#L- z?s;U7ku^G<-PQ~nP2+iXa!Lq8atjvlX097}5t*UlNmQ7;>@!|Kw82{P3ulSu56}7K`}Yc2#XPU=lgHoltM>?3ZC+ zIBmV5H1|YPZtXpM|H#Q7!~_~NvuiO_ls%csVuz5M@;U)Sjm>O{k$4qMclZ2oY)51C zl#Leo<4b^}9;1wnAUnSr=^75hBIDJT;FH39@A|dUcSjEF4w|*p&Y{ysKF6_XRJc)A zoLYC*0r3s&w#3M%hv~fpe1{L)v0L^vj6JFGf}%cUff8N^O64jZ&w`51h>QhQy#+<& zw)SQ^j;RQ~p1%L*-zC;y%NraCN|A)4@{WNeQh78c!(j~lY5Ww}@#ZG|#mHTR6(<=7 zL6Ub0bfFb+If&7&!2PFK(Gox^ueW8a3^z`<=1?9z>MIA*Ipr6P#OHc)$8vQX%Nc2n zuCf4HgQBQp5W!OvXQ^XwFb7`UABMvznHCfISMqp{j``3Tqy}jOg1;IA89s}C-E#;( zIxirTi$fr<4k3C*NShic*(pxq|kPok%1s<#ImTb zAF!e`4EO42%HAAjj)P?m~qlZmbK6so1^pioNQHSpLq?kj|0Iwmi%}rU9wnjY+*d0Bx_Z)|veyZOzl!Or`phM7LrV)W@ zlU+t@(YpYFTzadFWWi`U{`BI4#>=yjGE5E2hU->#DZfl+TK%mnSioCBaO5Z9fvfaCZWTX%BDcGF$Vf(d=xe*O=8Lk{0iT@Yt9)JIr!QY$!use7ly?2f;L`@m z4xiDkrboIuH)5^KtrOE_dCOOmn=A^CXL;!-z9+U!&Sxw0b+Sg+MYvn%R71e-W{dH+ zU0;p%7wncIIKkiUtM%NYGM&%up#v|@Td#aMq|t5;9Ho;YNFi$E(62oQ^u(GS9o zAX7u&kYX}@kN-(r`5CA@S&4agVchPpYaM2d6)*8;jxlyLVE;tK>-lvYy}!7^5t$kp zhrW?j>3s=WNHJqXfAfp@FNZRm+f5lrFK-6KA@7=-onxRinE2PBf5EY~byUS2~`J3SJ-ZY#z8e-_Ye)PMnItmlim44Te`NZ!&D{XL6_mCY)F~ z`6oPw!)vrS%^Q_OKOl_ZI1AHt^Pzqx|85rO90SSeT!h^3B0cppK6bL}4d#f;4>uUr ziO)`;1pkUTKdd{ki{3a!zuAb$F}dD0mytXmnLe={u6=VHWS{fRVd)SkQU@#RT^p06 z>1d84`mk_r^~U2iIP4X!nCjK({b#q&oT9pw4pZS6$|kK zcWtIu^|>BMJ6=FbKM+j3wJ#XP@B2&b@GV;kOF{0}><`9|E? zO^^=PdnQf4dtZAbPYKz`d`V?xHU&6eHYUR-dfG0mmqnmb{p(bqKxgb(T?Ltf%Pn$O z|9nn-Zu+t^Ml3_AGH_?Bo84v~z1H5#>ROTUSsiPi9rKI$7H|n3@M`Vv?rRt|%n5SY zR=gX1>I|&;seZjc^%&VY63h$6HoFmi*2os1@lD_U`4-zLtarXmo}THw6P zIz@8Q$7A5LPDWq3<(c5sJOkzIXY!TpdrCdS9pr(!Esa%uj+hcT0%s8{|M(|N`J=l# za=!mn62QxgVV2d`7U#%DaeX+DyQfxVOXu`wduodI6c%vfZ-#k_-?q`}9j{?w6b85B zCC)g@e(MBmznkIELvOf1S-VhX&*l(~RmH9m`B`jzEo1z^YKxaC%FQO7A)V))YS^G!@907q!A3 zrs09EI<6Bd-^A8do};JySt}>RW|<4e1(z+bmb#A99CZ9KwM0I-91B3|P}YgVKgOZE zDZ53tVIHq@EOeLx&rS#&uL~xUAlbGkTE%7zMC_iWf?F`MwRLnAc`YyeZaVMutI?W< zMOG#q{nn?@j^P_guriESdiBR+ZFKG5E6i>eR&AxWsXPZ-E#)sA1d}D1)g# ze-{t&H_3Y&bLkE1CqF&oYnvoGVF%gMf4_a-9PN5ey^C4Yl%pWqTyNCDc5whd=H9-< zoCb$AL0grIVt_*!iz~ioGyS;hPCs*%(a^Cpa^wGG)6fy(XV-R4Pe`h zD>pzG2nOnn=3t_^>5}-crL!D!25gfN=|(>m_jEik894ps826iuKnnWlr8<4D_g}qd zh&qocHM&8y^uJ)#@vodQ!{BKc$|4Ck0)BKg2>P&%x$oY1*nNX)>9?KEU;j$Ni!HQS z!0`o_$iD{MlWm{zHRshIp`6?82+%|Gzc6^P!%-QPau&*5=aF8jZ}afIp3^%iMaO%5 zDf*q$NEKrv=<29V`*n5~Vfp4_$cD#_i12}iGV3N$vYLzp9-{Z9dy{Rh^#p z)O*?et|wvfu;ur+7##mpv?^efts04LRzaXdZ?7Gf$rwZ@;x!WrhXrT7@$H%@)A5|` z0-kJQ&o4^;1ve{MeWkncDZn;0<+ zV87F4C-Jn~-Mo>=c^7IHOvU@wUT1eVO0B@?-ufUkPX>CwEztd@_t|!|-9Ilv9d<5^ zs+Y+axp~B`;Bf&;jNKyp*|1Jx_+D+(Z*UN2^6WxeBajVCW~32KQM;NG(U%H`k{ib% zT>h&dkY*B7XyiHVhNGL}QJ!Kbogp=HDc2cdpD9x$8ihH&@PKk|A5V+& zIOKs)&1EEjg~)Rt@we%iw|;4u^~?dXD6%NAGnD~81FA|l#nZP83PrD4aHNOuuYMQo4VHo#7Av2xD-6z19& zL}ys)@xnJ!rSBd>HH;a9tYx~n{+LTfBT2@fGu&0;qS%iVQcbNX}aig%GB-N`D`Oi+alZYbW8;>xWlbqwS9v0>C zdD9N2{M22-)Pu6MjNqvrw4)I{1B+BV8Eq+NPj@|!YC3p0V*^o7ZL;){Zw&fsVLPAG+>S4INC4_Go9D4x0N=b z*DC^nvL5qPU)+!=#crrsF^bv3=jP>H?Y-_~MhyvTCOQVs68}|nGKH=V4jr(JZ+&>@ zN@rxs^d4C8YPbq&BODPkC(jsFK!J8FSaAIPp5yBrIY}@B(wLJ$$EksKszkpd-hW0S znq*u$S)GOZXWA{XMZ6gw_r}>}6vTo9{njyU0SXHB^&folo7xf~-8^TfA~<6UTXWSq zc@7ubNx3p^v96I%E7812Gd=gYNIf3t#&3Fn7scsGd!Zu@rmR;MGPzsqwroaO9`$@q zZ$zKt?1!}}%pmt1{!rWf$#NdX!L=3CN1;=_YCPp<;ApPFJ5`$7+w;PU?v*IwQif#o z`DT=aCO)cAfUws{S1%_|Z;9o?xF6~%PZA+B8{4A*(#r~IglOy7@7M9-NGqQ0Ix;;F zA36-+JUVV|3WVk2x05S+(x;2VNaGX<>(D_tJ5EeGrV>tWd-6AgOtxe+MJHS$3!scT z*;8a1jU2WcWmQBl#hwLd3k-Ts)%4GH*rRrsfo~+#d2JrA_!`z|^&!?L5AIJ+Z#b

HQDmR+vd=;lNhB?t#r7B4`{ti zZ4QSLZcVwwJ`WYH%sL&No(@_-o8mTlN=9|jC%J~mI?Ga>8^-E5`{5gAG0}bc{{34; z8Cf~}sVqj_eq+7nl4Gck*DZh+HKdrEOT5O6;dM3<;GE-TaCasUHmOB|N3>GD>Ts$3 zc3h-@QKyv0bk~f%Mq!`&Q!2cB@iX#(1a{fwdx&2C=q~)ZV~#eIMw1c$Dq_=^ImHd7 z0i^+Q(regPmUIAeLEFvuX%6^PF1ke`FwubKILTD0YXiVWGZR!lzCexV-lE`$l!gYC zuK3C=$_gh2!!ebk@9@o_-oo?dc9n#6RGrgl*8qkA=Tkbo-Nu>|!SgMS(Xf)`PuxYV zd8I|Q5x2}C@1JLrSBBlu^5){J*R?&S z&=we_CzyGTzPV@jmBPWEk76n%Tb%xVV?RDbl`x#0!MhXv8m&EYn(jK>Vr`g5HBzw4 z^|5c|qj`>&Y;fiv(&ywm9YR;Y;I0mHPMpWe=tQc0MqMb?*E+X;Y)uPxrbCBq)=?46 z`+^;M3*6{cF!Zn~N)wE3BtOJ@)jycI$kB|hj@FsAFTBRODX&;7?QHd9YI}DikY^?G z8k;hxj?TlE)Fsqut6O>Pu;aeRax#a=^tC%RXhYH52(Q0ExsE$s&qwr|fc!}H2{iX) zRMmLNFXkx!i*>l+*QYvd(=K96xb+SjC5tj=UgXDfNpse)lRw=E^oCvl-0%1|Q&d4R zZz`^|J&TCWri}8&3!dS$0BZDWx3a~*vTIlHVj~QyHez}UBM^E?*Sjh`x<)rBKmFYZ z#Kkpw7lNR|R988QRisxkwZ!?0Mi|E!&*(H*H`kgH!x7lzw>v>pUejBR3fXi_ADH7z}7n6Sj@oI4DNTNMsY)!e}>vghi>-Ya;6;ysSo=)nW+tvl&RSrW-|IGNe1F)_9a6n21?xIJ=;#hBN@!A$z1#jmMi@ zMc&pIHp-D6pE|0ybi8?DJcTRC$mF}C<9!#Fb*Ljn*HpwtZl!aycE`=h!#Z}~ZR1Wa zyDY){`{pp%T=gm+;Z!zBtQ!hE$g&Y;+>PJy(E0K^*P%1%=Q;r$gKK~OY)XmgP4883 zcR%@B2mV+J3#YoS-(1f-#u6qbzRlvUZUkO=VAa9cp82tEAbX zaBw5Ej@cIsH?6_J^|>`vLD%_$%|MhN6Gq}|TiV3^qs9o7hi#XD1=`=Oas)c?N64X~ z3iN~1{fUG&Qs};WH+70WY@F~CpYBG9)eVbRPS8<&T;O&(*_Pq>q^Dbhldo^>6IX1` z6q2?vGmVC7;M1XTih2dQY^B`Q@p19B41Hr;x-uErob|A*o}z#~E}1GW7Hu5N2O85z)bF<~q*{%u$6dyz1Z@W?V6WpP$6STqLGn6}L=VspyEFm{`RVQz{r z;NAXXtT4!vcL_QtmeETi(qHeS2Hn9HqeUSOd7~LFqi2-S%{Upkoi)gr-iq7dxSSD1 z^TV;}0JfW{I$^*t%QWMfZgLveC@TBV(@kM`tOZ3uu`qH#bXf zrX0>1%iDW+uj9)h`MtDgR_`Lb(soByznv0&XWX!7Lt_=OWcv(@Y|2gDo(px>p;nh$ z&dcvLB%*nBYczGDbg&zPGF1G_~%dYLl;KkbB(%M z+QoDhKC6w$WK&N#Lp-iZivpX8_W6+NsH1t!-r*~_gT<*Pod6zoF;s05qtPcoHWjA~ zx~1dp0E_=bg$sV^cRRxn?O>x{>RBEgUwYDJI*o7BR3hJX2xZebs%IILNp>_C?xOuV zEgkJI^Odzz)4rze@=B9&Bl_qF3|urhZy9}dWdyFH6fC3beN$_EtXFlJVvoPAQ~Umr zPpdt%R3jKys&Cr0_Drk=O0X~rQ5VdR#C~;d3-hP literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_selected.png b/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_selected.png new file mode 100644 index 0000000000000000000000000000000000000000..06e7660601884bd67f715f36ec8423bc622df3b8 GIT binary patch literal 4146 zcmeHK_fyl^7Dr?iM5M?HQj&;WT#yojD-u+sxCllBW1&WoA|wH+0Rl=}DG7@-5m4Es zg(gxHz93;}qVi&Z(DM)h0h15{1VYLS^L~2&!TZ5GbLY(5JNL|)&;86fpUV%yJtlm-Q~wRNyG{&$U?YWyrCqx{Xz>f+7tm#j$^ z=@H{%4>vEkIC-!wdHBc$gbKpxpJ%`Nom6XhE4Y=MeBw@2MMJJ>TYbYN1fu+nPq|1x z$_Js^(N^Z6S`4{zElY6yb)ue8Sa;>jvb?;E>W&={J&26e&&!`6{0|c% zzQ;&pmtRGE-Q=S!$Rcqc%8>=Hi(qq0Idrs~atUj8c2*oCQCo(5Pe3`?dK`gF_`&|{ zHu9nd9EVI?l@wtJBtRFFfT3I4J<5_V&oKNwh z-d0ojd{v?6I7mEAI+GeV;^1WjOxD;&ho4(1BXIJ^^G8{tZ`>eJ$$&VWh;fmNAJV67 zEY%1zI$2szeJvYpI88ML@F{xA<*jh-)LD_y?@S@O{kLP4g|W4Fj_TlfBLq-jjuPJR zK*05E>_Wo*b3lmSi;9VAgwJ7G`-Qf4i}h;~!QD$#KLvTxjW^t;9C&E4ZGBu>Q%%a% z=m1CbhYF39h@0uVA86&kq6Dz6cxP1=m)G$#oVG>XPqxKrRAa54onMsuf?oJ9Y^)BN z>h#D4ojyiq1fFFs4}jIxTsm56k!f|FRTgUvaEp}p#yUN3u%`_=S)F)-a2H+mGF(lR zo5~oU1I9@`ZFAt58AUPwmzuLXJfUJSiTSprc_ffMaS+x#0Ow}H_H>0HPL91=)w#Q5 zrkk?4w%XEDw6)gxFbgsTP?$eLw>~i>^xYIV7-T9-^?RP?KiZg+#(~F7I_g|n*$8eA z4QHF6qvrKgG&Eo-b~N|vU?tj!+a$rOgwn#2q9|amU7MJkNhd9VlU8DTtL0UP8f3V49HjQeVSvo&lV&tR2fQ0o;yWMrkJhm=sEPuDt7R#Ip}`N z5_@&M{Ey8_!>xX09XCl4Q(%&W=r+~a>I*gsfd}Gu6^u}Ls$v-4Mp8W}q)5NL+A(<-=^_^TxYLWY; z*~Ze?fwwLj>!uAdfaUBkaG`z<`4E1AQg}PnERY;KB|K1bkHFO|ZA~PauO4p?Z+53b z<2_sW!yEG>F#jX{!M+_&?JbdMuwSsgOV8tS&Xh*aLL$Y*cLx1O*JU8YI( z9)r0Y+Qw24N& zBmM66`=G8HJm5w(|JcVc1Re_n(3;I>w^$sNHe#w28BYLpPqBOn%v;)YLpX&z^M_l1 zV+}ais2fsv(R4)LBwVvd|8-s=65FmBv6o=TZ?151X00{v?<9_on4CubIp;Vjpt z=rP_~!DD5qxI4N8k7&>^Rs8^o#!he!n=P=pKcesda9N^6+1zTRH{1 z?oOc{uLVVYqcJBL^6^kzuPqEkTt8(%>1i841+%nzR_gXEyZm8^EWj$4esVsM*l#BAd1`Fl z1nlYaIG5HiT95FxfzUH-(Sk+t1=3tGaZ^1zPk*@A4O!Xkv_-f#>Ba`?^8?BpYfS2$ z9A-odWp8ph&QK9MeRlr0r*DN9=Gy(w3w$K0%dbpiJsZZ#D>@g(@@cj>b6cFyc5A4i zGoqm}Q}u&K*x0m08kZ5ps$oOlh-sZQyEx#!U$fhCRGE=AtwH zlz+@eTh4?F5RxAH!Y>T^H80``@ni z=yvbW7&p@9ca+70hP~Z^M#gc_{VFc#m*lY!su4aalIWpuQ4~Wo4mw#(6a+lIGTGat zng{05#-U+tNDGMrF(|HKzshH-@Dc5kga*0ht~t1RzL*d?Xk_qrXN!t-8dGeBgOsbV>R&3jC*~*ugMmeypTO^2 z0@Ph2E?5upGOI4G$Pv{r->YSPVCGJ56m0B)rGR;=`6c;Ix`p5S#hK92*$H|1ho!I; z&C+^5^Ag3ykQWsWD5ajn)sm{S1+<<()r`%@*&)iGGh#8>b2jD5Df7IOm9uWQuTXm3 z7?zPcS^_&J+-|)*FKuasIN^eJdt=!1$jMJli75Fl3{KPck#Y|AF=xh3JE!#Y=*RRk zMQq@xZq8gxMJNTTX7y`UP5>7=ZIp1k3uT1ZfXbs ztNY@FYXI+t?)hN0P*ghgF7*Szs=F(;&&kQ*3;8?vnX$FMAJuz+!qJI3DevEM=9&ZL zf-<3m{O;n!&cR>(k3cGiuU$4StdX+fhFK#GeHgI{f*&Yew{K-cH??3e`=nilVrM(Y z?4T6_O&W<9Mei+#>`&d>^pjeVb`mXHLFxzEeih(W6oSCA17W*+rG zYwP1S-|yj4NeZsMvQx0_Sx%n*>1KXfy8jPfn<3dLGkNtu_Um5Qw)3~U;$T&M$>Yg? E0Kp5vPXGV_ literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_unselected.png b/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/cube_unselected.png new file mode 100644 index 0000000000000000000000000000000000000000..0353a7ec33ffc414e3eada11e98e673d74308f37 GIT binary patch literal 3075 zcmeHJ`&ZIi76%uTXt#WprKYIYu9>M788<~4Lr^ncq?vdzOUv^5xRscof~9HsLhU-5 zrK5O7^OdQX*9fICCu(Bw0VaykXc|gX3SL2EzRvtK|H1swUhAyA*4caQv(M+8^Esc} z!GYLq`sVr|5NMm9FFF(i0@F4JbPF&;WxKrwoL({pa~1IawMj?j8xUw~k01Kar>@_Z zO>a#wJ)v_;qR_tqvt8h?YioC+ZSw!>K;RK8xPpqXX=L=9%MX`5T(9x$19fA?%@1o{ zKj`Y}s@3XHO{O)Yx}bsk5C}D&3Nkb@1lj87pwK8#J`@Tv0)t_;FcA7%@EZvKa})Zs z3wkQCGXwengIRJILAC8V!59lxM7M~@PGe1&6ch1s%EtPdR7J;@ z5H*C9ay=+c{dMlucs$+gb_hanW2E2n&TvhloY@i0CL~#VX6)Md@OfSjI=ZHmiN?2P zAWt6AAFk1m`s4N+GnRlRHB#o+{sE`;k~nj(to0`urR+4h_`)udP-ulq7HNhA*TP&3 z=_2VOYj}pW7&LZJ7aU`r>cLg8>LU8tOF|Ke667J(h<{6?$md_>LoUEz0raO6OgarY zIAKyv!pr4*GMDsTF8p!=A)PV!Px$CK&ba^y{xQ|DC3AyFzKoC#yWppp<_xFc67NtZ{d#xZ$7wfJm5QmX3_@mAW>Z~@RIR`9y;tfpjoOoE^am^2@|S+R zCB8w*%s#VORJ}b*wkI>W?fWq8(&W{j=C4ckc5UInx#qroP6e|GxQHIs;x26ti(bNa zfLw@yaNq*M_0cqiFnyST@M3*v-Y*RSt`y?4lO^9Zt>!U%y_9ReaV?{QoP(c_qplL!zPA}eDcxgyx8u& zBhLopXSVt^mMNXfdL^#8L#>dUODm0!Do*reCYra@aLkLmxf}P0cMslk6m>;-{yyLY zBoqoz+y%OriHl00(-s3_#)Yeea1tII6RyL7lfaj-*7z?AMcyPk8voQtJYgfzxE^}RvAwc;FDK!GeYgllP2%_q z&e#s@yPzu9BH%Y@(eT|>s$3GUKAeW$Z8UzA{Ap{ zZv~5%!rhqzn<$Lq&d4HtZ(}oAn))VEDT~#y8Wq={26=pOB}RIu3V2M52by{%O{@61 zpW6 zrX%gSE1rqZ+*G@YP|^dI*+A%3!CE-_aw?3QWl`0GOC#5=I&@Cl$xAgT_C_HJiQI=` zXH53O^o(bcDNVdPoJAYMBlN+NV1ezSd&e9{PqUg^Nfa5c{AZuz9tc|?z%5XVua5~Z zj$}r=!1kQ<4&BIbJ7e`NZD?T`=3`9j+1IMfLYRXkG-`-D@ z$L-b!zq=nRN=x?0xw(n2$fS&jY}k`q=sjYQ-Rd)YH`PG$4hqRRKB4btK|3c?dkHWw zd~{S#>irbRvr^jeGvsUn`OA!IghBAY82J@%yATWEND47_ai*P3Sla|^$EYzo8Zx@v z{g~C}o*$8H`T|kP!cY>=0BaAkBv_P@kg=UaxA64!niJ}blxqOVhvYS&WBFSeGOA+^ z9xLiC%1cQRL7}~Y>2(y1N-@Vo?oZJ3mczv%Xab9*P*e1tfbZuHwJA24wsdIHI@oE| zMZ0Ax90QxWOwlNWa{#&avIBnFkqEDlEzk2Pt96_rJJ{!Bd#IV28GGuq?(EyHi){5j z-@Pz4Y`TkjN?!dWW!hT?)p}}hd61(OR8-#C0k?vMZ}R1vhWq~!va`u`?CCV&OF)nS N=I0ZLZudTWPf=h{VPkSg8l^^qU^EqsW`xnQfX2n) aCdSBXtaJ65wzdGvYX(nOKbLh*2~7ZjKibRy literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/flat_orange_bad.png b/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/flat_orange_bad.png new file mode 100644 index 0000000000000000000000000000000000000000..3a7f78a0bf918b1084ae8fd5e04af9038ca96f86 GIT binary patch literal 2015 zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL9Be?5hW%z|fD~Jjx4R3&e-K=-cll(X2xoyu zWHAE+w=f7ZGR&GI0Tg5}@$_|Nf6U4y$|&<@-2z^qNRFqAV@SoVw|AralwAcJE_R&w zbY|NPN$=#?Ga;!TCWb!UCgI?r;P`va{Ec_}C;qt@{Nc9*Ry99x-C{VUVQGksuEC3kNa^$<-VKxhLI0X zNL)3{a$RDaG^L}V;ls1V{n~P}_xt<)zOYu36`Wegrjjfzxme3>?T(OTnz{l4_e#GQ zT0fdmcr5wY_FC2_GqiO|`^?-zbBvQ$_pR^m>WGoPdOL%KrDeNJt;?E-H}1!`|Lr=_ z^WObF_p6zb{f3V>hUQEyJhM+eKRNWttXE7dE#`*5OQl}l-pBOmwOxE!YVP7krwt#U z4l}e)DmyN7J@V#OGa=Ektj2~9eS$let=%VY{H@sQ^{u5_&-VR2QN8xb4AIYpe`}5= zuQ?tnc~KYWw>{1M^{@LY*LJU&@#grfcUEb~;v_d)Z@XxieqVXYdTrU)-+crb86OI> zKmNsBKlgRS`rch-v1?a+k(|`mvDeyd?&98`QuEH()cYSbQ`%&_g_))0{M<<=Vs001 zKXqeulCc-OJZ+hMsZ~!=$_~?H`woX$Bu1J($-BpmVBkw>F=(Tk6oveLu`)i?1+24^>sSXRlZ6_ z=Za>~&M_l{8bX9C76VoB-ez#5Mo7<)aeonUe^|LYf`}5W#-ZnrN_lBlj ziTECW^~8(MCclpN?>!rnEq8rNSY3rIFumQGbuDW7>CMX5c04gvy1x1mBNLPUx{I#& zAD)n};|C_?c16a2wdY-a4;}2H8d~)5iNNTOJFw8fcHq30jY>UE(IAGsn4Tv zIg4775211v7A{Cp0c0Zas2>%m(F=0`_Uf>q{y=_;#omp7jXxd$R>cgSu6{1-oD!M< DXGI)= literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_selected.png b/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_selected.png new file mode 100644 index 0000000000000000000000000000000000000000..3226a7a33b5443e04724e8465fa0112cf1e524bd GIT binary patch literal 4080 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 literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_unselected.png b/test/lib/mayaUsd/render/mayaToHydra/MtohBasicRenderTest/lambertDefaultMaterial/cube_unselected.png new file mode 100644 index 0000000000000000000000000000000000000000..2dda3819357ad1d70160806af9ea00c06e40a7ad GIT binary patch literal 2821 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 literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_123.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_123.png new file mode 100644 index 0000000000000000000000000000000000000000..9741969af002c86b93f2baadd5eafca6bb0f89a6 GIT binary patch literal 2660 zcmeHJ`%@BF6b2+kUDMc>(iB|4&0IyXM)46+A;=Wz?li6DBj2^AqGBp(ijAcvNVt`v ztF+Gan9WTbHpMI{Qy%%sHBelY6tXc*#Ya-2F3$e6|G>`fOn2tYnYrJ&bIv{IyK}$N zkmF=43p)z{0ANK4B7P450IzKs#MHRM6npg;ojHv}q8t6+Ys96w4*;-$QHY05vQIjhJa#aZ#Y^~jJcXO_HgiPI{;MK|(sY_i}YnS7{Sw0oAj2cBG|C;dHy`00j z3mi?&HV)W}aw$D6&EpAN8qCL2r9z=lF)%PNg5Q&D2Q1|08}v;hTuj=ok#<4)>*2zrMDrkjv#O zJ^~`Bqn_*^ptct1^oxY7K@w3uy_ZulV+&dN0gve<=(6HCOiL*N8`Um5s3m ze@p_~HptYK#}u4JfqB$;mWmXMMQBzv{y<=*AXF4l@!}D=yw&~&P777$!8}2plU%x# z>GY)y%uQG_TG!Un;?3?vpF@;N@~q-}d^6^DCpx;;G6gxMhdQv5(0SQ0w4|Ec=Xda6 zEh*~xALfv<%22e0=EIn-jX{92Mlasr(wFeY6Hng_y?=~C#+glMvJ@q z>w(75D%(Q6UZ+`^hjP4W;|(*YXA-E5sB31ZacF%#76xC7T+# zW8}lhetrNy(ScEmwU%{tbsaF$2^)xRjF|=(bscqvAH49pFAk?iIr7+@5D7%4QmGDv zQ$nNVMtE&weQkvmOQX&EBE8QzmXb+PZ)p-cbJrVfg3!mE(L}V6EwL2LK)^z|Na$jzXmNx~}smPPu)%_9emrjGh`Q1IS@M{r$ zS9|b_QIlNMLYc84Va^CO2@xQzs*N}N8F@-@0Ms$*;NU=&`s1+KVF@?6+m>~K&ze49 zQ^y?MZn4bZWi1z(d<*3eiMZFn;Q^c9<-kowC%Y@%2X$R#yUuN(1-u@@Yxg3Pjapj0 zNeI>?=C>)ojj%XoaUemI%?~K3D-XMe_xCr5)#3{)Np5^gJq=ux`;#9J%0AOyM^!%* zr3{Jl`jP53+05(!Fc|}!=yNhS)Vx^0qS@n>p`yl$ntA~ zmOqBiz!?QaasKPdPiDzp#*4k#;lzlS4`=6r(GxP)_#XU}PLSf7j0a9cCBQsWCwgz+ z2fvw`ecs%(;32Xs%f-U4<1h&cEFP(RNhQr2du{@Yq4NsUXp-9v#JF}Q{xq&7#ns>5 znAR>PXLDnH#d|>diHtr=X7&pZPWUdnW24-@QLl#9H!tTpjlZ#L^9kU=Y|5p6&RgFJb;ZjK@5^`ekj6l24_aZ z1GS7O$fah%pe7(0$tZXL2?$6KGb4m6MuCJt$bLHi?Y~>KRa<3Ob-k+h>wfj?*YEXv zT@)S~XsBd?Q8O z-Q5zoTt0gAKoKI}f7*IRp+FI1(q@>Y@IG`Kk7wT5>sU7^j9418Giyv(?~F&97Jc7z zyQZEZz-u3!zn8H#0$v&79s4gMpXau;`DD&PN1XOkPBj2*-C@>Ld4o2_DQ~fiK3==E zLAm1wyiZv>xCm@DF$RsH(=z6yQmF&pwnatiHCN=#H*hE;LWBGo5NBaLsRhrYn0vvJV(f)u)O~iMnz~@ed<&Ayt>CIZHK8uUpL3wg*a|E}HF24}ej;jpyj(zm~&kuM!-iEVMg2EBJo>nQGm}vCkon$Dw zh0)U9?o~<=SnQ0K^hFnPC{*~DEf?c1U%6t0t|LLWXE%(I-SHL+4=i4E!k~FRSL32F zvb4qM=dKyQJW8-KWoxnxyU8D1pn1)-1V^JF`^0k>v&9NLE)B72YyBbm*srSly~`~5 zP;i;fmAI}l0AwVS$;M+hgf?&+Xq6<)q}Knjba^r|eR<&y0|uj(?d?lEaIQGe-;eNC z(Ad7ZZ9f}|`ylVXHO^!(b}Q}jSi9hPbYmkmT~K7?{hL94aG64#2(H-xjhE#t&)y%8 z#YhSZ3w4^MdUCVv(*6z=+IY{+7l&^rxo0jWtDBzeP1n{d;>4=IPaPlSz^isb#|!Jt z#c4G;t4kA`EVaiJf~0s6hdwOk9E>=W^$&XoD_CcMIQcq#K{HReC~Z zTVhuh@5VS8JEX{t$7^jVr3O+9^%s+{!nf-tVU5@C0#@{)a7#rSOexO6wF; zoC&bj8|Uzbzin2&!s?!Bs&#ma?2*^SD2K3twd!$Ac`iR+a%qyk-&t@ZS;SGS+F3hL zKT{}Rj_qnacrV9nK%?#Yn`;_li2<^Y-%Lv&sro0K?&2*Mcq)%66}DWUY1E>sTt1(S z_^Ws;Q6rFHEA#`O`(`XUmKTn6b*|_kG11%#&olQcoABGFxv2YPXGD+VVw3%-+wPvf zN1G4fl*@)h%rW5_vv@&|_v*hWE*>!mZZ5c$Cp(~LaPQ(M>nrc$PF$uFwDeEE&X$uK6)1e&ME&(-C$ z@cG#amOQgdcv%k_^0xn0Sbbw-qXd+N6AQdiskA9ar&WE48cATjF4+6P>CmvS!i|Av zIOV40xSH#ZxQU4gq4Mc3b|$&0YA}Qi+uGXVusEF2c(o}3mIlztlu|3U%BO2Q){0Cfo2-&MvcI4o6tlC9iz-!NuX3`tJJtgphYGYm zA_2Zb8@~qhgUX71>XK&W<~#}TI-vA?tA_791sWCvAu%anR&S`VFfF>0)&K)4e zEl5jBO6qh@Z*d^CS|AGIx`qFegtwi}*vx;|t7yr8@}>$UTpI{c1fur=VTRc~Ttx<@o7^If=9U)+ zV(ou$B!g#hQHE3VeOpivyPzC(0sxn3(UotA+S&gO1?H^U%S%( dY|Z}&d-0KLYu=RI{rPVY>>ujabQ%|OqS)+ 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 literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_1235.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_1235.png new file mode 100644 index 0000000000000000000000000000000000000000..dc1725c7fc3f3e2762f8f0e6bf0436bca5197bf2 GIT binary patch literal 2986 zcmeHJ`!^eC8cx!zMp4&gT`Cblx2d#j#d>V$Y@i7x;Jg89x`rCjoY}* z*_qMV5%-!PjZ222N!V?t39YyVnQ2^dS(hNPY3D!KpLWmbcg}a-^L^iYp7WgVeV_L{ z)Jqow^p05^0{{SefjGav0RX_qA2viwb3++);c9|*vcG?tCjP$Op89DL0MNM+=;w1K ztA1U0yVxWE`eHHNGssyh8WA(74$v{Y41vf@x@8-?^UoCY#fefUjBm7=39Ha>mA6{` z{{27t(^FDXvb6x}e1z1HdT|$oqmf;0Fi<@JAr<8#oMb1q`Nq`RVch z?-Pcu9>_UJ4xlYl+&mAkssM`)Rr?%48d)h{NhR-W2xmq|M@w0*ZZ05D(}8RZN)U_1 zL9kR_Z@PppXL2~4ZZGwLqBuod=5<1Pb*R}+OCO5$*xz7oPA1I{unKGhC4O;H3Ylmu z9^u?~e^+aA6-KN!tMp(Z3TEeD&%byc-~ z{`rq_%Nd_1Me^2LYBt4Te6Bb8CI?&)QfCn7LPK?zj>LMdiWTlaAfsctclOE4{(!c= z6P0euru-aJG_W>nxyR{aKIaU)Ts+6h%`~S5^07vxTNdM7cllfM1Q@Y_(veN|VWaST z4BD6)t{9ih4Um~7_w*thV8p0c+DWd6spwAkHY)KTSJDKlJ00mD5aEH>B*w7T9G8?h z%Ej7SXFI}RBs*G7=r>dB?6^*kMV$9+fBpjqq^_zph;M0cZz360)KKEq4kM<~z!(MJ z1{aLx)P&;@FlG3Fnci+8g>*@9r~H1f`f}rrd}*|}rOD3P>u^^T!m@L_&cLdMv+4_2 zIjX&_RG|=#SXIl@$OpT!;d#q{Ivt5Y%72v*al1e3d)*+bdhWa;Ab&JhKpaWW!3D`@ z%loi7|Nde|n1PMxoP(1=U32#23`WCPB6$*2)Gt(S`nFCF;GTL zb6MC`o?fZTl`agwInH24d^fd*&+KMzh5cqeBr`VHUQ*A%RsAGx^P`W)*q<=-4OIHK zow=5`C#MYz-1I=~DDGsBq=+gBX}{bnnx+1VP;4Pd^{W4aIk!rV&^~!f9qxh|5-%q5*%P#eg2pCv!2{>r7Uq>73h88!^vm9e*KJ+WbxqlGx8A==<(=R9 zxlms%7Ed*LGHE-1IB*5AhnEAJ{X`QL4jh7G8)ggH)IBafD^H*N6;M`9>Byw|Y**%J zbZAtje6~Zp?k218txG`In@a+2<%w)yR(=V_Y1dv`O*?l#u33L2D0JZ!?pV!&$0nlf zbE_eUW9QH=_?_`-oVQ%H=3A+HT7PhKjE8@|CbZ&zRYdni-G)g}j8=PVrRz|Nk-_%a z;w5uI!>$?AL2YM|MkFpBv2xhanr5DpB`3x3;Cr)e*P-SZSi=oUPzjR>c{$N`dBq>broAN1-I@~8W%EkR4@->3Eh;BYIFJGj77rkL_V3Iy%Z zNMPq9OuaR5(LcU0YQZ_&@8~CE@yT8NFI&t*(09ST`I$bEVILfK;Dw8RZRc<1{|Ei5 BRE7Wm literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_3.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_3.png new file mode 100644 index 0000000000000000000000000000000000000000..7cdb3cf91afa1c9a77ac56f3bca62b84550e4a93 GIT binary patch literal 1912 zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL9Be?5hW%z|fD~uGn_DrEhJagI+s-jCuxoj` zIEGZrd3$HKpN_kT!^MZXDoGqa{`)ICKG}27E?KlZCzkylXT;uVA+KgGn_wlGyK2|W z#kX(YJ}=Ag$Nl*IT1JLH_M99Hoec~VI7X?V8U%m3ckkZ)m!aBSy=vk`Sxv_wVmBP_M|@JzP`S1_P_u9xl>zH^Cx%C zyk9RG{26DP8SdD#=T2;F>{~xSzi){%9`|&fnDzPJzkTw;vn==R+xJdRUVi_x17e&U zv$r)B)Ya{~ziHOvuH%UgVJ>lTao-+B7|#5;A@pBO(&aQ&!{jD|i8J1=y16Urc6jpM z_~^)NRwL$ZJ`Ojwh&Dy~o{3o9+j;b1#PjPmjEvC{Id^hA&(`)^o-4a@ByrmYzHmit z1#LOwa^4$8jER@iL=F3&&zn2O^SM0nG)v1_Yo3EgQoDmE^sRLMFn-yPnX%n^%41Od>FMg{vd$@?2>|A3 BPu~Cl literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_35.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/instances_35.png new file mode 100644 index 0000000000000000000000000000000000000000..ece346ae074c72ce218466e37fd74364a3904fe1 GIT binary patch literal 2266 zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL9Be?5hW%z|fD~uGn_DrEhJagI+s-jCaP)Y( zIEGZrd3*PCp0KAx!^IYB4zUi_?mhn(Z*<&{bWpbK{^C<#TolgM96Ay7P&3*5_s`GI zM}L2Rzy5opt&l)jF6+aC&Ft;YVVh$=-e5~{eR6rZe|y228E5`&2>(};bUjVYFuBQK z;*7(g#V@jC_es5eUaFCO)*e&Gu(Ki@9V!?{tgEeb;|PiJkM74 z-#qtlMdx9eL$}U3g*k-Tq}?}7IM31&J3I17%)ILV|Nb7HG2_h6{?23HWKPT~J+m)+ zM&pKZrbF+1dDLXG*L;>W7M6TH=lGlN4-dB=o^c%_rK866`qSO7_skT8dv^ZHI`+nT zfBRlxfgL}77u^$`Xv#d(IV$_|Zy9O#a7Keg3G06Wpl_$? z-PD_5vGB$`8+mO7-H;fq*zi(3nx<)7YtkW@;lzX%1yE6Y%RQMBUA1(KH%D>>a z`A@&TzJC3^uT9*(n#yQ-PQPnU`}KQHds?h2e}7MQoB4DO$=B=u+_bCy_NKqyY=JTJ zLyc#bu5VR%`1$$y_3{6+W`5Z!c2(-NP}GAGPLBO|W1jtba&q$FSKD6S>v}8nQd{A6 z>ZR*IUQyKd$OA8h1?qN-Hbq8fy;~Zg82Rx=oZ9YKwOVQ6^EVqVKaW0@|NX_q#ji#4 z^7SU}OOrPj*S@1S)sESsmg!L7pXsWX>c#rDW+)c?Vrf|%aeD6kIh{F+3qKoNt2)-! zaG+8^pf71{%}{KvnC7rEKFm-P+T8TEjLqo%+1GF1B>$eE7M^hM zC(wc{B}2iRY$;8fYF*xRwNHLt5C?QhwD*$rzwNfEpR-jGJ2$2PV0+|9xN;kE)EQ%)MyZlrUKG3!at@HXOxymt$v{o PYN2?#`njxgN@xNAvKF!J literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_0.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_0.png new file mode 100644 index 0000000000000000000000000000000000000000..f90702779733d0150465c509af06caab6472f1d7 GIT binary patch literal 1513 zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL9Be?5hW%z|fD~uGn_DrEhJagI+s-jCu+H{$ zaSW-L^Y*GE?*RjW0~@xP^x2;iJ^X2=+D(p6r{~Q~XFTBj{51o^zmpCO9y}}!M@Fg9 mAQ(*rqZwheETD05_=G9tKTE-b>=|D{1-7TFpUXO@geCxt^O9KL6`iYiewqQ5n7J`mJ17@o(H^F~4LovfAb~p0Yc& zf9~PCdEcuT80ueU+?Qr}@LxrNL4}FQL1~m44FcMxf)A~)U%!6JWLLiFJ@0|?;^N}X zYuw%4-`|UmjeVQz|3Q|MW6#dryQ8md-n_Z_@ZrPd-{a!r>+kO0vE#?{>gM+z4mbAv z{{8#umcpv4U2Bx(ie=qw;^N}!*7Ec7pRTH_+xL9#;W|!^+2&4#MMXsgH8p!$lH=r= zjn+SZ{=E74@#D7drM~$C&5&#>b#b$~#yIo+b;HKBpGCxGoBN3vb14hNM;=eEVc+Hl zRI_~fa(`u`YhPcN+8irSc>Ra7`|NDTi1Q9@=hGA(7aLn_C}leItk&a{Ep8KWYje(pZvR z9wj?mTi$PPyOcFrSs~|M-uCEg9}jmH2kiT&IoZ-;esTZL${SXUjpnZ>)QAZ@Z1~Q? zVmD*qj=g(hubn=9x;Bq}pDtsgJ_q}sng3nb6$Awsgcun|snH-9>Z#x-<3o1IJ1mwP QH-aiUPgg&ebxsLQ0QosQvH$=8 literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_12.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_12.png new file mode 100644 index 0000000000000000000000000000000000000000..ab89f6c68e19478ed3b41b0f890357c4ca2d87f2 GIT binary patch literal 2136 zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL9Be?5hW%z|fD~uGn_DrEhJagI+s-jCaBz9L zIEGZrd3*P6Rhg=`Z(J@VD~LM8MR^N7T?7C zwP)s~Jp1$N)vI4T4F4`2-I&kN(C+EL;K9Pu;4(^$1_5nT!2|w#dnz~YZrB|w_nR#t z``+H_?b7;sdU=2Uyt=x2@m=AZi46yO{pZ`gJ*MtIZ_eEL^Y8z<{q^hD_xtUuzrA_z zoh@2LVa^`?_&_s(7aEdAhs87Ig0 zyf><*k2cCrOw5bfG4Gf8z4uS-Z`jE{etPe%{52E#=(Ue`$!)4*;uV zIdL~jOY@B5$JQ~L>g@<6zNY$vv}w(=(mqz3;5duCaA*h)n|qLC3)xwdJ-(Wq+^jOiq57 zC@8>})%Ni1?d{w5RoY#5Xu#0Om6B1QsNjXjoTo5N2@TlHUuYs2;l6JF!p^(h_1?YrxqF^_p67i(&*ypX zt^ACdylKPs4ImI`Q))`m7a$P$ht&;R4_JiixLyDPcoa$=fd8)Hk6QZxTUKh)r(f}9 z)1xC7HXrjNTa%&*_(-_($0fldeqzGJiQAr)?fbZ*J&(~hulk7R3$?xE*($^BCxyQT z(Wun@L7}n|4M z+VWJLKRedfw8izEP<|mArpOunIuo z`AKQ!HpP7{CMVs%&-Q1BI~)#@3YWY~p0=z1bPJ{#le`5N(ka%>FS=6L#aoAZVdW=L zh;7j|(?Ukv6FA&`9jJzbvc^~qx(XKc=U5V+M5u8_4jm`cTM(C#;`r@175C~CO|5=d zd!m?3Cx9~-&%y1X^Z_h-0k;4T268wr4n8e8xWd)4B?(%V5{GVRhk(nf5tWDvk9!8z zaTN5uF)WmY z>GrUPK0bn_?7R&o1;Ao@$-P_nB1Fw@TEo=jwG3VWj4|$<&?!|=bCI14N`XR!Noxe9 z6K~#J%mdXEh8Nehs|tKxkG*(8Tg@CJ70k1e=;8cT9UWnVMuMqQN{IHf-7KG#YFeZ&B-g9m<1znnj ztZ_#C)383HF}AwrP%JZc0;BTdIKsm2;;+bMe#re&KV+k{qpT$qp2jQBJpO&`?%zb; zknLTbgx)ftuwA>cOGb|B-xht%qrgnwa2{9|@}N)^>Z-Z4IG-*d_Lj8MAP`G+hByUK zU;MIr;L9`eIEZHQmQ0r2$Fg?Cf39LYXH#Xyf*MZ&aF(*!x*csXXbo5$0ykriI41}1 z8a&ock@9J{IUNE~?+BmEDy~{6$5A0 z7XokUc1Z-doddbuBc4JZB-~7cXs)9;snz`KWcH1~t^4!ep&AtnQB$NEXRfUk&-qDl zzs_MR?r}xFYF~|i!3Zjm#Q4sQ(2w2e^yLOt?ZwxpMk(YJc-^sIX!Ts=UbNf-E$3i% zcR9j)azjcy&>^u_Twxo?6a>IbXLC@>jXkp9(^^(jv64ZIFw^Yh$GLfx*KFG33t@xU zaQ$K3wAcTw5mwH*^oMFfpPse=ay24=P)VZw{=N9;^vHTHic@yf%%q?hQpA-9%QLxp zv#q(0iwaz2?2O?+LXm5#p4niGM_7ddkx(8=Yb1`!nCjAJ$V+rhgzqp!Lp{u&%P`f0 sl6Lij*K&Z?Y*ODBTDw>AKjQk8Agk8c!#-0nw)*)>JxWb#O3b+YH;5M_8UO$Q literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_1234.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_1234.png new file mode 100644 index 0000000000000000000000000000000000000000..8e978bb7f23ca60f7e9c71a02ce49e23e1c836b5 GIT binary patch literal 2913 zcmeHJdr;F?7EWS-@Tdf-3#18&Xtf*Lf@TR6$pZ=mM+st)mpm6G!$V$%SAZnEHn56- z?ilI{p~~8-6he8pg%DyoBxSK+0C^|`AywYQm;^!s$^IPw+&_0_XLiOrbLY&QJ9F+g z_dDOYU)32ZP61&2~E zUUwRd?j(KTS~Drz-rio{3lfg1Mvs6*yN56!KPZ%4$p*pg;2;+WgzQHKRl;B(J1`jI z^5O9R?-AaZIXF0sfswgA8cWfwP&X=-dXEn~JbLq#@}M@Be);=>;r$c7n1lDB97rUahUI#NvCj z!~Ao9kWF_+O)f?HQ|{2cCA=+~i#BrVI1Y~g(d7)TW|xh;6$69Ke2Phn;xtVV7%ff| z*XA1K>egmRTfz{uJm?{u?#956a$6*)!VTd2U2q({n@XM%?Dd^1c_v|N3iWVXI9x~1 zI~ONhnaerNcpdU+J(oOFVg`ZS_^Y4Uos793$r_JQ=}?%~ITih;ge{_=s*cO+++&5= z%T`1+0gk_9nhrCpA`Pi%lO}Qs0f5V}@gbaUS3XHWUB6XVX~`E%jpiQHT+Y-R6w(?R$s^(*I6*D%XOVux3r$8ta!bA@0_zO$&y(~U z^{EKfj!IfFg1iZ^5?isL(a`gq%zbSpkejc zVSdhr_|NK;09nWn{?>+H`kVsM@% zY;F0BfT2x6@VpCt`h8iH0@NwV6p_?HbdXCD=JS*p)d*tGj4MRE#x$%y3MNh3XKt}f zn&q+S5lbBdB2G)8^!N1M$EPFET)BR}B110tHsFNmagQZzE_z+(@t4GzjKrN$xpA1q zGHx$wIvPyz4bbw)HKlIyK(fLksnRD9xHOYiW#WWjEJYroH@$ z>)ebnDX4fl1u50m?h<}2$=L@N{>(%&Mm`rNy@;rxar9?^Nw4{M{0Bn&k5Gy)v46w4?eJs-+cm<8}Zej4I4QwQ9SK#cjm)ZFPe+f zu;yCLY}U|}>ilANb%nv7Tz`%;zpY-(6Bfu*UBuum4x3*Y$J)(7m~@Jo*L>4ngxgk#c>(kYxeufN=hCeoR|F|BbP0*!9q~wqpBsg zM!BL?crvNvh)?1Ul{<}_c21Zl20nMJ%L*99)#?T8cLN)@1R7UmOyDF#5fE{dCG1Jl zo~OsAcvV+@oLC;MEEp{HAb{42ERDAbtiC9?^yup`Wn254MQVZ_DrvN%DoZ7})^qfs z5zn-3rXjV)p`oGV@-91geF!DNv^mQp3=_SO3yUK*+;_eGvI1tt-tErIN;n;#}-|c)YAO|6wfIG zex4CHaETe+F;~K;P>8SY zvi=SQ`#HGC9u>C0jbldYKYsb}gE9Hw&i=!q8uxr!R(-8mKR5QyD4+DF`nCGTm;M`b CcnZh> literal 0 HcmV?d00001 diff --git a/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_123456.png b/test/lib/mayaUsd/render/mayaToHydra/MtohDagChangesTest/lambertDefaultMaterial/instances_123456.png new file mode 100644 index 0000000000000000000000000000000000000000..28e759cc0f2003026cfdea41ef430eda4d6f7b30 GIT binary patch literal 3564 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 A zO1d{RTT9Wo1~DcUYPi+S5jJgAG&N#@)O^buh~BTWzwDo|KXhi!JTsr?%rl>P&gVSm zQ+W8h;Lq*R_8<`G^AK{-5fBJmytZMs0Fv9e?>P|cm?Tm<5dU05JESi_AbVLz(6>i3 zMN`_TR82S}f81t@NyNBs)pL_=>?h_yueC?h*ck(KVXbY|x#qCO37^z=E3?{7?S`_h z@%NSlP)ol?ge=r_a#vHQ^+w|kPNY*|ZVR^lFh8#pY!8Auj zj+L{NDfpHUaqP2O8wI1qZAywnQp>e`n99*UyYAKg!A2iWbQ)Tk8JlrwNcQNtwH?Pm zRu-v__{2yeRMv@}q8{tx11LoP{G>`~(M{TQZY*qi7G}CSFj3-hJ>c2AzVGxBTpvnw zdeT=L0MtNTw7&Y!!>zcm0Z8lCZJNo2dBTMcXEm*l=txAq8hy^AX?T69Twhl6OJhAt zr80Ch=r~$6Kg(x7L|Oc__-tYCm_6C~a(yQ3Gx2&v{?|8?Ol7Nq^dnxf^5J_nG(37y zsxzO7CIdb37R`108Jy3F{slckO8m;j;MwU70n?lo?JmUB#NQ3Z>P!u=-oWyxV@;F- zBuoqtpsXHGh(tvUZ+^*jO#7+5W#D$pwPo|--2=uuScn(_Fs#~FHBd2ZE$k0wyothl z#p!R<^u?WbbRbP~Z|xQg6v{4k=#EC6AaD(e1%C!RwGf5{>h}K>%nS;3kWZK7%*6N^ zQ(li$D!+FM!xq=W-)T%usO+@=~QI2W#{|)&RvoC{Jsw9 zzR(!Boj?iW(eA}%P7{zaso_ADOwqvR5*#(O4J6Du@!p}6MdTUnN!?PNYCrp7$2bZ> zq5~{SDZL}Ve%ug6YJ$&E8=AZGD4Q;M3*;!4Ty*uIGc3gxhCc#{PFHfh%kP)X%|{liXqWvZq#RyE+x$g= zMVG|6!#2BTTen0-rw3p~zF0*5%XunSp7UZT;CAEE39o=X+eoegK)x{yqw&FXa!f$V z?gqt;_=38TCBPS3QxRR6Y7KqflHXO%X0 zQ6$W03*t#yg3ezStG_eY^#_y}cf+@pS^1>#eyV}cvrv)Q;s{aVpGTak5SMt0i|d!X zmim3o-mIQ@J6L-^N?1c7@5E7qU2iwP@&(gQa3)IP!)N3xtzRNvp};iuz(lJqrgtG{ zkuUFVMLp{LddO_|H-ej^p{0Mx&AGie5`7H(ZXa_18*#^C$mXy! zaSW-L^Y*GEFN31Mfdh|A*44{qERc6HTD)NYx|u&29$cxnW@s>%:OSMac_> + $<$:LINUX> + $<$:TBB_USE_DEBUG> + $<$:GTEST_LINKED_AS_SHARED_LIBRARY> +) + +# ----------------------------------------------------------------------------- +# include directories +# ----------------------------------------------------------------------------- +target_include_directories(${TARGET_NAME} + PUBLIC + ${GTEST_INCLUDE_DIRS} +) + +# ----------------------------------------------------------------------------- +# link libraries +# ----------------------------------------------------------------------------- +target_link_libraries(${TARGET_NAME} + PUBLIC + mayaHydraLib + ${GTEST_LIBRARIES} +) + +# ----------------------------------------------------------------------------- +# properties +# ----------------------------------------------------------------------------- +maya_set_plugin_properties(${TARGET_NAME}) + +# ----------------------------------------------------------------------------- +# run-time search paths +# ----------------------------------------------------------------------------- +if(IS_MACOSX OR IS_LINUX) + mayaUsd_init_rpath(rpath "lib/maya") + mayaUsd_add_rpath(rpath "${CMAKE_INSTALL_PREFIX}/lib/gtest") + mayaUsd_install_rpath(rpath ${TARGET_NAME}) +endif() + +# ----------------------------------------------------------------------------- +# install +# ----------------------------------------------------------------------------- +install(TARGETS ${TARGET_NAME} + DESTINATION ${CMAKE_INSTALL_PREFIX}/lib/maya) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/README.md b/test/lib/mayaUsd/render/mayaToHydra/cpp/README.md new file mode 100644 index 0000000000..263d24419b --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/README.md @@ -0,0 +1,33 @@ +# C++ Test Framework for Maya Hydra + +This test framework allows to run C++ tests from an interactive maya. The [`mayaHydraCppTests` Plugin](./mayaHydraCppTestsCmd.h) provides a command called `mayaHydraCppTest` which allows to run only a subset of tests. This command can be called from Python scripts. + +## mayaHydraCppTests Plugin Usage + +The plugin registers a single command which runs the tests: + +``` +mayaHydraCppTest [flags] +``` +Currently, the only flag available is `-f` (long flag version `-filter`), which takes a String and allows to filter tests. For example, to run all tests in the `CppFramework` test suite only, one would call `mayaHydraCppTest -f CppFramework.*`. To run only the `pass` test, one can call `mayaHydraCppTest -f CppFramework.pass`. By default, all tests will run and no filter will be applied.Refer to [GoogleTest documentation](https://github.com/google/googletest/blob/main/docs/advanced.md#running-a-subset-of-the-tests) for more information on filters. + +Currently, the plugin can only be invoked from Python tests. See [Known defects](#known-defects) section for more details. + +## Contributing to tests + +To introduce a new C++ test: + +1. Create a new C++ test suite using GoogleTest. See [testCppFramework.cpp](./testCppFramework.cpp). +2. Add the C++ test suite file to the target_sources of [this directory's CMakeLists.txt](./CMakeLists.txt) +3. Create a corresponding Python test suite that will interactively call the C++ tests from Maya. This Python script can setup the Maya scene before running the C++ tests. [testCppFramework.py](./testCppFramework.py) shows how to load the plugin and run its command. +4. Add the Python test suite file to `TEST_SCRIPT_FILES` in the [parent directory's CMakeLists.txt](../CMakeLists.txt). Notice `testCppFramework` being added as `cpp/testCppFramework.py`. + +To facilitate writing C++ tests, different utility files exist. Here are some non-exhaustive descriptions of them and their contents: +* #include <[mayaHydraLib/hydraUtils.h](../../../../../../lib/mayaHydra/hydraExtensions/hydraUtils.h)> : Utils using only USD/Hydra types and concepts. +* #include <[mayaHydraLib/mayaUtils.h](../../../../../../lib/mayaHydra/hydraExtensions/mayaUtils.h)> : Utils using only Maya types and concepts. +* #include <[mayaHydraLib/mixedUtils.h](../../../../../../lib/mayaHydra/hydraExtensions/mixedUtils.h)> : Utils using both USD/Hydra and Maya types and concepts. Includes utilities to translate from one representation to the other. +* #include "[testUtils.h](./testUtils.h)" : Utils for testing purposes. Includes utilities to get the terminal scene indices and inspect the contents of a SceneIndex. + +## Known defects +The `MayaHydraCppTests` plugin cannot be loaded directly from Maya because Maya's PATH environment variable currently is missing GTest's DLL path. This is a known defect that should be fixed in the future. However, the plugin does serve its main purpose: to launch Maya Hydra C++ tests from an interactive Maya via Python scripts. + diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.cpp new file mode 100644 index 0000000000..d3b8a526d4 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.cpp @@ -0,0 +1,92 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include "mayaHydraCppTestsCmd.h" +#include +#include +#include +#include +#include +#include + +namespace +{ +constexpr auto _filter = "-f"; +constexpr auto _filterLong = "-filter"; + +} + +MSyntax mayaHydraCppTestCmd::createSyntax() +{ + MSyntax syntax; + syntax.addFlag(_filter, _filterLong, MSyntax::kString); + return syntax; +} + +std::vector constructGoogleTestArgs(const MArgDatabase& database) +{ + std::vector args; + args.emplace_back("mayahydra_tests"); + + MString filter = "*"; + + if (database.isFlagSet("-f")) { + if (database.getFlagArgument("-f", 0, filter)) { } + } + + ::testing::GTEST_FLAG(filter) = filter.asChar(); + + return args; +} + +MStatus mayaHydraCppTestCmd::doIt( const MArgList& args ) +{ + MStatus status; + MArgDatabase db(syntax(), args, &status); + if (!status) { + return status; + } + + std::vector arguments = constructGoogleTestArgs(db); + + char** argv = new char*[arguments.size()]; + int32_t argc(arguments.size()); + for (int32_t i = 0; i < argc; ++i) { + argv[i] = (char*)arguments[i].c_str(); + } + + // By default, if no filter flag is given, all tests will run + ::testing::InitGoogleTest(&argc, argv); + if (RUN_ALL_TESTS() == 0 && ::testing::UnitTest::GetInstance()->test_to_run_count() > 0) { + MGlobal::displayInfo("This test passed."); + return MS::kSuccess; + } + + MGlobal::displayInfo("This test failed."); + return MS::kFailure; +} + +MStatus initializePlugin( MObject obj ) +{ + MFnPlugin plugin( obj, "Autodesk", "1.0", "Any" ); + plugin.registerCommand( "mayaHydraCppTest", mayaHydraCppTestCmd::creator, mayaHydraCppTestCmd::createSyntax ); + return MS::kSuccess; +} + +MStatus uninitializePlugin( MObject obj ) +{ + MFnPlugin plugin( obj ); + plugin.deregisterCommand( "mayaHydraCppTest" ); + return MS::kSuccess; +} \ No newline at end of file diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.h b/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.h new file mode 100644 index 0000000000..03e230e7e2 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/mayaHydraCppTestsCmd.h @@ -0,0 +1,31 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#ifndef MH_CPPTESTS_CMD +#define MH_CPPTESTS_CMD + +#include + +class mayaHydraCppTestCmd : public MPxCommand +{ +public: + static void* creator() { return new mayaHydraCppTestCmd(); } + static MSyntax createSyntax(); + + static const MString name; + + MStatus doIt(const MArgList& args) override; +}; + +#endif // MH_CPPTESTS_CMD \ No newline at end of file diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.cpp new file mode 100644 index 0000000000..172fcaf3b7 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.cpp @@ -0,0 +1,319 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include +#include + +#include +#include + +#include +#include +#include +#include + +#include +#include + +// TODO : When/if we figure out how to put tokens in a custom namespace, +// remove this and add the relevant namespace at the callsites. +using PXR_NS::FvpColorPreferencesTokens; + +using namespace MayaHydra; + +namespace { + +class ColorPreferencesTestObserver : public Fvp::Observer +{ +public: + ~ColorPreferencesTestObserver() override = default; + + void operator()(const Fvp::Notification& notification) override + { + _notifications.push_back(notification.staticCast()); + } + + const std::vector& getReceivedNotifications() { return _notifications; } + +private: + std::vector _notifications; +}; + +double kDefaultColorDifferenceTolerance = 1e-4; +double kDefaultColorComponentShift = 0.1; + +bool colorsAreClose( + const PXR_NS::GfVec4f color1, + const PXR_NS::GfVec4f color2, + double tolerance = kDefaultColorDifferenceTolerance) +{ + return PXR_NS::GfIsClose(color1, color2, tolerance); +} + +void changeRGBAColor(const std::string& colorName, const PXR_NS::GfVec4f& colorValue) +{ + std::stringstream colorValueStringStream; + colorValueStringStream << colorValue[0] << " " << colorValue[1] << " " << colorValue[2] << " " + << colorValue[3]; + std::string command = "displayRGBColor " + colorName + " " + colorValueStringStream.str(); + MGlobal::executeCommand(MString(command.c_str())); +} + +void changeColorIndex(const std::string& colorName, const std::string& tableName, size_t newIndex) +{ + std::string changeIndexCommand + = "displayColor -" + tableName + " " + colorName + " " + std::to_string(newIndex); + MGlobal::executeCommand(MString(changeIndexCommand.c_str())); +} + +void changePaletteColor( + const std::string& colorName, + const std::string& tableName, + const PXR_NS::GfVec4f& colorValue) +{ + std::stringstream colorValueStringStream; + colorValueStringStream << colorValue[0] << " " << colorValue[1] << " " << colorValue[2]; + + size_t indexInPalette = 0; + if (!getIndexedColorPreferenceIndex(colorName, tableName, indexInPalette)) { + return; + } + + std::string changePaletteColorCommand = "colorIndex -" + tableName + " " + + std::to_string(indexInPalette) + " " + colorValueStringStream.str(); + MGlobal::executeCommand(MString(changePaletteColorCommand.c_str())); +} + +PXR_NS::GfVec4f getShiftedColorComponents( + const PXR_NS::GfVec4f& colorToShift, + size_t nbComponentsToShift, + double shift = kDefaultColorComponentShift) +{ + PXR_NS::GfVec4f shiftedColor = colorToShift; + for (size_t iColorComponent = 0; iColorComponent < nbComponentsToShift; iColorComponent++) { + shiftedColor[iColorComponent] += shift; + if (shiftedColor[iColorComponent] > 1.0) { + // Keep the resulting value within [0,1] + shiftedColor[iColorComponent] -= std::floor(shiftedColor[iColorComponent]); + } + } + return shiftedColor; +} + +PXR_NS::GfVec4f getShiftedRGBComponents( + const PXR_NS::GfVec4f& colorToShift, + double shift = kDefaultColorComponentShift) +{ + return getShiftedColorComponents(colorToShift, 3, shift); +} + +PXR_NS::GfVec4f getShiftedRGBAComponents( + const PXR_NS::GfVec4f& colorToShift, + double shift = kDefaultColorComponentShift) +{ + return getShiftedColorComponents(colorToShift, 4, shift); +} + +size_t getShiftedColorIndex(size_t originalColorIndex) +{ + // We want to cycle through the colors in the palette. However, their indices are 1-based, so we + // shift back to 0-based for the modulo and then back to 1-based. We do -1 to make the index + // 0-based, +1 to increment it, % by the number of colors to keep it within range, then +1 to + // make it 1-based again. The first -1 and +1 cancel out, but I left them to be explicit. + MStatus isValid3dView; + M3dView active3dView = M3dView::active3dView(&isValid3dView); + EXPECT_TRUE(isValid3dView); + MStatus isValidColorCount; + unsigned int nbColorsInTable = active3dView.numActiveColors(&isValidColorCount); + EXPECT_TRUE(isValidColorCount); + size_t newColorIndex = ((originalColorIndex - 1 + 1) % nbColorsInTable) + 1; + return newColorIndex; +} + +} // namespace + +TEST(ColorPreferences, rgbaColorNotification) +{ + PXR_NS::GfVec4f initialColor; + EXPECT_TRUE(getRGBAColorPreferenceValue(kLeadColorName, initialColor)); + + // Hook up the observer to the Flow color preferences + auto observer = std::make_shared(); + Fvp::ColorPreferences::getInstance().addObserver(observer); + + // Change the Maya color to trigger a notification + PXR_NS::GfVec4f newColor = getShiftedRGBAComponents(initialColor); + changeRGBAColor(kLeadColorName, newColor); + + // Must explicitly remove observer, otherwise invalid read can occur. The observer's + // shared_ptr control block is allocated in the current test DLL, but can linger around + // in the UFE DLL through a weak_ptr if not explicitly removed. If the test DLL gets + // unloaded, and the UFE DLL tries to delete the control block, an invalid read will occur. + Fvp::ColorPreferences::getInstance().removeObserver(observer); + + // Check notification info + ASSERT_EQ(observer->getReceivedNotifications().size(), 1u); + Fvp::ColorChanged notification = observer->getReceivedNotifications().back(); + EXPECT_EQ(notification.token(), FvpColorPreferencesTokens->wireframeSelection); + EXPECT_TRUE(colorsAreClose(notification.oldColor(), initialColor)); + EXPECT_TRUE(colorsAreClose(notification.newColor(), newColor)); +} + +TEST(ColorPreferences, indexedColorNotification) +{ + PXR_NS::GfVec4f initialColor; + EXPECT_TRUE( + getIndexedColorPreferenceValue(kPolyVertexColorName, kActiveColorTableName, initialColor)); + + // Hook up the observer to the Flow color preferences + auto observer = std::make_shared(); + Fvp::ColorPreferences::getInstance().addObserver(observer); + + // Change the Maya color index to trigger a notification + size_t previousColorIndex; + EXPECT_TRUE(getIndexedColorPreferenceIndex( + kPolyVertexColorName, kActiveColorTableName, previousColorIndex)); + size_t newColorIndex = getShiftedColorIndex(previousColorIndex); + changeColorIndex(kPolyVertexColorName, kActiveColorTableName, newColorIndex); + + // Must explicitly remove observer, otherwise invalid read can occur. The observer's + // shared_ptr control block is allocated in the current test DLL, but can linger around + // in the UFE DLL through a weak_ptr if not explicitly removed. If the test DLL gets + // unloaded, and the UFE DLL tries to delete the control block, an invalid read will occur. + Fvp::ColorPreferences::getInstance().removeObserver(observer); + + PXR_NS::GfVec4f newColor; + EXPECT_TRUE( + getIndexedColorPreferenceValue(kPolyVertexColorName, kActiveColorTableName, newColor)); + + // Check notification info + ASSERT_EQ(observer->getReceivedNotifications().size(), 1u); + Fvp::ColorChanged notification = observer->getReceivedNotifications().back(); + EXPECT_EQ(notification.token(), FvpColorPreferencesTokens->vertexSelection); + EXPECT_TRUE(colorsAreClose(notification.oldColor(), initialColor)); + EXPECT_TRUE(colorsAreClose(notification.newColor(), newColor)); +} + +TEST(ColorPreferences, paletteColorNotification) +{ + PXR_NS::GfVec4f initialColor; + EXPECT_TRUE( + getIndexedColorPreferenceValue(kPolyVertexColorName, kActiveColorTableName, initialColor)); + + // Hook up the observer to the Flow color preferences + auto observer = std::make_shared(); + Fvp::ColorPreferences::getInstance().addObserver(observer); + + // Change the Maya color in the palette to trigger a notification + PXR_NS::GfVec4f newColor = getShiftedRGBComponents(initialColor); + changePaletteColor(kPolyVertexColorName, kActiveColorTableName, newColor); + + // Must explicitly remove observer, otherwise invalid read can occur. The observer's + // shared_ptr control block is allocated in the current test DLL, but can linger around + // in the UFE DLL through a weak_ptr if not explicitly removed. If the test DLL gets + // unloaded, and the UFE DLL tries to delete the control block, an invalid read will occur. + Fvp::ColorPreferences::getInstance().removeObserver(observer); + + // Check notification info + ASSERT_EQ(observer->getReceivedNotifications().size(), 1u); + Fvp::ColorChanged notification = observer->getReceivedNotifications().back(); + EXPECT_EQ(notification.token(), FvpColorPreferencesTokens->vertexSelection); + EXPECT_TRUE(colorsAreClose(notification.oldColor(), initialColor)); + EXPECT_TRUE(colorsAreClose(notification.newColor(), newColor)); +} + +TEST(ColorPreferences, rgbaColorQuery) +{ + // Query color from Maya + PXR_NS::GfVec4f initialMayaColor; + EXPECT_TRUE(getRGBAColorPreferenceValue(kLeadColorName, initialMayaColor)); + + // Query color from Flow Viewport + PXR_NS::GfVec4f initialFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->wireframeSelection, initialFvpColor)); + + // Check that the queried colors match + EXPECT_TRUE(colorsAreClose(initialFvpColor, initialMayaColor)); + + // Change the Maya color + PXR_NS::GfVec4f newMayaColor = getShiftedRGBAComponents(initialMayaColor); + changeRGBAColor(kLeadColorName, newMayaColor); + + // Check that a new Flow Viewport color query is correct + PXR_NS::GfVec4f newFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->wireframeSelection, newFvpColor)); + EXPECT_TRUE(colorsAreClose(newFvpColor, newMayaColor)); +} + +TEST(ColorPreferences, indexedColorQuery) +{ + // Query color from Maya + PXR_NS::GfVec4f initialMayaColor; + EXPECT_TRUE(getIndexedColorPreferenceValue( + kPolyVertexColorName, kActiveColorTableName, initialMayaColor)); + + // Query color from Flow Viewport + PXR_NS::GfVec4f initialFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->vertexSelection, initialFvpColor)); + + // Check that the queried colors match + EXPECT_TRUE(colorsAreClose(initialFvpColor, initialMayaColor)); + + // Change the Maya palette index of the color + size_t previousColorIndex; + EXPECT_TRUE(getIndexedColorPreferenceIndex( + kPolyVertexColorName, kActiveColorTableName, previousColorIndex)); + size_t newColorIndex = getShiftedColorIndex(previousColorIndex); + changeColorIndex(kPolyVertexColorName, kActiveColorTableName, newColorIndex); + + // Compare Maya and Flow Viewport-retrieved colors + PXR_NS::GfVec4f newMayaColor; + EXPECT_TRUE( + getColorPreferencesPaletteColor(kActiveColorTableName, newColorIndex, newMayaColor)); + PXR_NS::GfVec4f newFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->vertexSelection, newFvpColor)); + EXPECT_TRUE(colorsAreClose(newFvpColor, newMayaColor)); +} + +TEST(ColorPreferences, paletteColorQuery) +{ + // Query color from Maya + PXR_NS::GfVec4f initialMayaColor; + EXPECT_TRUE(getIndexedColorPreferenceValue( + kPolyVertexColorName, kActiveColorTableName, initialMayaColor)); + + // Query color from Flow Viewport + PXR_NS::GfVec4f initialFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->vertexSelection, initialFvpColor)); + + // Check that the queried colors match + EXPECT_TRUE(colorsAreClose(initialFvpColor, initialMayaColor)); + + // Change the Maya color in the palette + PXR_NS::GfVec4f newMayaColor = getShiftedRGBComponents(initialMayaColor); + changePaletteColor(kPolyVertexColorName, kActiveColorTableName, newMayaColor); + + // Check that a new Flow Viewport color query is correct + PXR_NS::GfVec4f newFvpColor; + EXPECT_TRUE(Fvp::ColorPreferences::getInstance().getColor( + FvpColorPreferencesTokens->vertexSelection, newFvpColor)); + EXPECT_TRUE(colorsAreClose(newFvpColor, newMayaColor)); +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.py new file mode 100644 index 0000000000..82c8be485c --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testColorPreferences.py @@ -0,0 +1,69 @@ +# +# Copyright 2023 Autodesk, Inc. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +import maya.cmds as cmds +import maya.mel as mel + +import fixturesUtils +import mayaUtils +import mtohUtils + +import unittest + +from testUtils import PluginLoaded + +class TestColorPreferences(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + # Utils + def resetColorPrefs(self): + mel.eval("displayRGBColor -rf; displayColor -rf; colorIndex -rf;") + + def setUp(self): + self.resetColorPrefs() + self.setHdStormRenderer() + + def tearDown(self): + self.resetColorPrefs() + + def runCppTest(self, testFilter: str): + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f=testFilter) + + # Individual tests + def test_rgbaColorNotification(self): + self.runCppTest("ColorPreferences.rgbaColorNotification") + + @unittest.skipUnless(mayaUtils.hydraFixLevel() > 1, "Requires DisplayColorChanged event bugfix.") + def test_indexedColorNotification(self): + self.runCppTest("ColorPreferences.indexedColorNotification") + + def test_paletteColorNotification(self): + self.runCppTest("ColorPreferences.paletteColorNotification") + + def test_rgbaColorQuery(self): + self.runCppTest("ColorPreferences.rgbaColorQuery") + + @unittest.skipUnless(mayaUtils.hydraFixLevel() > 1, "Requires DisplayColorChanged event bugfix.") + def test_indexedColorQuery(self): + self.runCppTest("ColorPreferences.indexedColorQuery") + + def test_paletteColorQuery(self): + self.runCppTest("ColorPreferences.paletteColorQuery") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.cpp new file mode 100644 index 0000000000..24a67fbd3c --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.cpp @@ -0,0 +1,25 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +#include + +// Test to ensure the C++ Testing Framework works as expected + +TEST (CppFramework, pass) { + EXPECT_EQ (1, 1); +} + +TEST (CppFramework, fail) { + EXPECT_EQ (0, 1); +} \ No newline at end of file diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.py new file mode 100644 index 0000000000..ef031acf33 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testCppFramework.py @@ -0,0 +1,39 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import fixturesUtils +import mtohUtils +from testUtils import PluginLoaded + +# Test to ensure the C++ Testing Framework works as expected + +class TestCppFramework(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def test_Pass(self): + self.setHdStormRenderer() + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f="CppFramework.pass") + + + def test_Fail(self): + self.setHdStormRenderer() + with PluginLoaded('mayaHydraCppTests'): + with self.assertRaises(RuntimeError): + cmds.mayaHydraCppTest(f="CppFramework.fail") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) \ No newline at end of file diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.cpp new file mode 100644 index 0000000000..27a4711c5e --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.cpp @@ -0,0 +1,69 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "testUtils.h" + +#include +#include + +#include + +#include + +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +using namespace MayaHydra; + +TEST(MayaSceneFlattening, childHasFlattenedTransform) +{ + // Setup inspector for the first scene index + const SceneIndicesVector& sceneIndices = GetTerminalSceneIndices(); + ASSERT_GT(sceneIndices.size(), static_cast(0)); + SceneIndexInspector inspector(sceneIndices.front()); + + // Retrieve the child cube prim + FindPrimPredicate findCubePrimPredicate + = [](const HdSceneIndexBasePtr& sceneIndex, const SdfPath& primPath) -> bool { + HdSceneIndexPrim prim = sceneIndex->GetPrim(primPath); + if (prim.primType != HdPrimTypeTokens->mesh) { + return false; + } + bool parentIsCube + = MakeRelativeToParentPath(primPath.GetParentPath()).GetAsString() == "childCubeShape"; + return parentIsCube; + }; + PrimEntriesVector foundPrims = inspector.FindPrims(findCubePrimPredicate, 1); + ASSERT_EQ(foundPrims.size(), static_cast(1)); + HdSceneIndexPrim cubePrim = foundPrims.front().prim; + + // Extract the Hydra xform matrix from the cube prim + GfMatrix4d cubeHydraMatrix; + ASSERT_TRUE(GetXformMatrixFromPrim(cubePrim, cubeHydraMatrix)); + + // Retrieve the child cube Maya DAG path + MDagPath cubeDagPath; + ASSERT_TRUE(GetDagPathFromNodeName("childCube", cubeDagPath)); + + // Extract the Maya matrix from the cube DAG path + MMatrix cubeMayaMatrix; + ASSERT_TRUE(GetMayaMatrixFromDagPath(cubeDagPath, cubeMayaMatrix)); + + // Make sure that both the Hydra and Maya flattened transforms match + EXPECT_TRUE(MatricesAreClose(cubeHydraMatrix, cubeMayaMatrix)) + << "Hydra matrix " << cubeHydraMatrix << " was not close enough to Maya Matrix " + << cubeMayaMatrix; +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.py new file mode 100644 index 0000000000..b38ef0d2cf --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaSceneFlattening.py @@ -0,0 +1,39 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import fixturesUtils +import mtohUtils +from testUtils import PluginLoaded + +class TestMayaSceneFlattening(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def setupParentChildScene(self): + self.setHdStormRenderer() + cmds.polySphere(name="parentSphere") + cmds.polyCube(name="childCube") + cmds.parent("childCube", "parentSphere") + cmds.move(1, 2, 3, "childCube", relative=True) + cmds.move(-6, -4, -2, "parentSphere", relative=True) + cmds.refresh() + + def test_ChildHasFlattenedTransform(self): + self.setupParentChildScene() + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f="MayaSceneFlattening.childHasFlattenedTransform") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.cpp new file mode 100644 index 0000000000..33a819d406 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.cpp @@ -0,0 +1,52 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "testUtils.h" + +#include + +#include + +#include + +#include + +PXR_NAMESPACE_USING_DIRECTIVE + +using namespace MayaHydra; + +bool IsUfeLight(const HdSceneIndexBasePtr& sceneIndex, const SdfPath& primPath) +{ + bool isUfeLightPrim = primPath.GetElementString().find("ufeLightProxy") < std::string::npos; + if (!isUfeLightPrim) { + return false; + } + // The shape prim is used to display the light's wireframe and is the only prim we want to allow + // for UFE lights + bool isShapePrim = primPath.GetElementString().find("ufeLightProxyShape") < std::string::npos; + return !isShapePrim; +} + +TEST(MayaUsdUfeItems, skipUsdUfeLights) +{ + // Setup inspector for the first scene index + const SceneIndicesVector& sceneIndices = GetTerminalSceneIndices(); + ASSERT_GT(sceneIndices.size(), static_cast(0)); + SceneIndexInspector inspector(sceneIndices.front()); + + // Find UFE lights + PrimEntriesVector ufeLights = inspector.FindPrims(&IsUfeLight, 1); + EXPECT_EQ(ufeLights.size(), static_cast(0)); +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.py new file mode 100644 index 0000000000..744d2ae29a --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMayaUsdUfeItems.py @@ -0,0 +1,47 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds + +import fixturesUtils +import mtohUtils +import unittest + +from testUtils import PluginLoaded + +class TestMayaUsdUfeItems(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def setupUsdStage(self): + import mayaUsd + import mayaUsd_createStageWithNewLayer + from pxr import UsdGeom, UsdLux + + self.setHdStormRenderer() + cmds.refresh() + usdProxyShapeUfePathString = mayaUsd_createStageWithNewLayer.createStageWithNewLayer() + usdStage = mayaUsd.lib.GetPrim(usdProxyShapeUfePathString).GetStage() + UsdGeom.Cylinder.Define(usdStage, "/USDCylinder") + UsdLux.RectLight.Define(usdStage, "/USDRectLight") + cmds.refresh() + + @unittest.skipUnless(mtohUtils.checkForMayaUsdPlugin(), "Requires Maya USD Plugin.") + def test_SkipMayaUsdUfeLights(self): + self.setupUsdStage() + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f="MayaUsdUfeItems.skipUsdUfeLights") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.cpp new file mode 100644 index 0000000000..cea06271bf --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.cpp @@ -0,0 +1,138 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "testUtils.h" + +#include +#include + +#include + +#include + +#include + +#include + +PXR_NAMESPACE_USING_DIRECTIVE +using namespace MayaHydra; + +TEST(FlowViewport, mergingSceneIndex) +{ + // The Flow Viewport custom merging scene index is in the scene index tree. + const auto& sceneIndices = GetTerminalSceneIndices(); + ASSERT_GT(sceneIndices.size(), static_cast(0)); + + auto isFvpMergingSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Merging Scene Index"); + auto mergingSiBase = findSceneIndexInTree( + sceneIndices.front(), isFvpMergingSceneIndex); + ASSERT_TRUE(mergingSiBase); + + // The custom merging scene index has the MayaHydraSceneIndex as a child, + // to produce Maya Dag data. + auto mergingSi = TfDynamic_cast(mergingSiBase); + ASSERT_TRUE(mergingSi); + + auto producers = mergingSi->GetInputScenes(); + auto isMayaProducerSceneIndex = SceneIndexDisplayNamePred( + "MayaHydraSceneIndex"); + auto found = std::find_if( + producers.begin(), producers.end(), isMayaProducerSceneIndex); + + ASSERT_NE(found, producers.end()); + auto mayaSi = TfDynamic_cast(*found); + ASSERT_TRUE(mayaSi); + + // The Flow Viewport merging scene index supports the Flow Viewport path + // interface, and forwards the call to the Maya scene index, which will + // translate a Maya path into a scene index SdfPath. + + // Get the path to the sphere defined in the Python driver for this test. + // We know this is a single-segment UFE path, and that its tail is + // "aSphere". + auto mayaPath = Ufe::PathString::path("|aSphere"); + ASSERT_EQ(mayaPath.nbSegments(), 1u); + ASSERT_EQ(mayaPath.back().string(), "aSphere"); + + // The Maya data producer scene index supports the path interface. Ask it + // to translate the application path into a scene index path. + auto sceneIndexPath = mayaSi->SceneIndexPath(mayaPath); + + // Regardless of prefix, the scene index path tail component will match the + // Maya node name + ASSERT_EQ(sceneIndexPath.GetName(), mayaPath.back().string()); + + // If we ask the terminal scene index for a prim at that path, there must be + // one. Prims that exist have a non-null data source. + SdfPath nonExistentPrimPath("/foo/bar"); + auto nonExistentPrim = sceneIndices.front()->GetPrim(nonExistentPrimPath); + ASSERT_FALSE(nonExistentPrim.dataSource); + + auto spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + + // Flow Viewport merging scene index must give the same scene index path + // answer as the Maya data producer scene index. + ASSERT_EQ(mergingSi->SceneIndexPath(mayaPath), sceneIndexPath); +} + +TEST(FlowViewport, mergingSceneIndexAddRemove) +{ + // Same setup as mergingSceneIndex test. + const auto& sceneIndices = GetTerminalSceneIndices(); + auto isFvpMergingSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Merging Scene Index"); + auto mergingSiBase = findSceneIndexInTree( + sceneIndices.front(), isFvpMergingSceneIndex); + auto mergingSi = TfDynamic_cast(mergingSiBase); + auto producers = mergingSi->GetInputScenes(); + auto isMayaProducerSceneIndex = SceneIndexDisplayNamePred( + "MayaHydraSceneIndex"); + auto found = std::find_if( + producers.begin(), producers.end(), isMayaProducerSceneIndex); + auto mayaSi = TfDynamic_cast(*found); + auto mayaPath = Ufe::PathString::path("|aSphere"); + auto sceneIndexPath = mayaSi->SceneIndexPath(mayaPath); + + // With Maya scene index in the merging scene index, the sphere prim has + // a valid scene index path. + auto spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + spherePrim = mergingSi->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + + // Remove the Maya scene index from the Flow Viewport merging scene index. + ASSERT_EQ(mergingSi->GetInputScenes().size(), 1u); + mergingSi->RemoveInputScene(mayaSi); + ASSERT_EQ(mergingSi->GetInputScenes().size(), 0u); + + // Without the Maya scene index in the merging scene index, the sphere prim + // is no longer in the Hydra scene index scene. + spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_FALSE(spherePrim.dataSource); + spherePrim = mergingSi->GetPrim(sceneIndexPath); + ASSERT_FALSE(spherePrim.dataSource); + + // Add the Maya scene index back to the Flow Viewport merging scene index, + // sphere prim will reappear. We know that the Maya scene index is added + // with the absolute root path as scene root, so duplicate that here. + mergingSi->AddInputScene(mayaSi, SdfPath::AbsoluteRootPath()); + ASSERT_EQ(mergingSi->GetInputScenes().size(), 1u); + spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + spherePrim = mergingSi->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.py new file mode 100644 index 0000000000..5956ce27ca --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testMergingSceneIndex.py @@ -0,0 +1,36 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import fixturesUtils +import mtohUtils +from testUtils import PluginLoaded + +class TestMergingSceneIndex(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def setupScene(self): + self.setHdStormRenderer() + cmds.polySphere(name="aSphere") + cmds.refresh() + + def test_mergingSceneIndex(self): + self.setupScene() + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f="FlowViewport.mergingSceneIndex") + cmds.mayaHydraCppTest(f="FlowViewport.mergingSceneIndexAddRemove") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.cpp new file mode 100644 index 0000000000..8c9748cfa9 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.cpp @@ -0,0 +1,202 @@ +// Copyright 2023 Autodesk +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "testUtils.h" + +#include + +#include +#include + +#include + +#include +#include +#include + +#include + +#include + +PXR_NAMESPACE_USING_DIRECTIVE +using namespace MayaHydra; + +TEST(FlowViewport, selectionSceneIndex) +{ + // The Flow Viewport selection scene index is in the scene index tree. + const auto& sceneIndices = GetTerminalSceneIndices(); + ASSERT_GT(sceneIndices.size(), static_cast(0)); + + auto isFvpSelectionSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Selection Scene Index"); + auto selectionSiBase = findSceneIndexInTree( + sceneIndices.front(), isFvpSelectionSceneIndex); + ASSERT_TRUE(selectionSiBase); + + auto selectionSi = TfDynamic_cast(selectionSiBase); + ASSERT_TRUE(selectionSi); + + // Get the path to the sphere defined in the Python driver for this test. + // We know this is a single-segment UFE path, and that its tail is + // "aSphere". + const auto mayaPath = Ufe::PathString::path("|aSphere"); + ASSERT_EQ(mayaPath.nbSegments(), 1u); + ASSERT_EQ(mayaPath.back().string(), "aSphere"); + + // Clear the Maya selection. Can be done equivalently either through the + // Maya MSelectionList interface or the UFE selection interface. + ASSERT_EQ(MGlobal::clearSelectionList(), MS::kSuccess); + + // The sphere prim in the Hydra scene index scene has no selection data + // source. First, translate the application path into a scene index path. + const auto sceneIndexPath = selectionSi->SceneIndexPath(mayaPath); + ASSERT_EQ(sceneIndexPath.GetName(), mayaPath.back().string()); + + // Next, check that there is no selections data source on the prim. + auto spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + auto dataSourceNames = spherePrim.dataSource->GetNames(); + ASSERT_EQ(std::find(dataSourceNames.begin(), dataSourceNames.end(), HdSelectionsSchemaTokens->selections), dataSourceNames.end()); + + // Selection scene index says the prim is not selected. + ASSERT_FALSE(selectionSi->IsFullySelected(sceneIndexPath)); + + // On selection, the prim is given a selections data source. + MSelectionList sphereSn; + ASSERT_EQ(sphereSn.add("|aSphere"), MS::kSuccess); + ASSERT_EQ(MGlobal::setActiveSelectionList(sphereSn), MS::kSuccess); + + spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + dataSourceNames = spherePrim.dataSource->GetNames(); + ASSERT_NE(std::find(dataSourceNames.begin(), dataSourceNames.end(), HdSelectionsSchemaTokens->selections), dataSourceNames.end()); + + auto snDataSource = spherePrim.dataSource->Get(HdSelectionsSchemaTokens->selections); + ASSERT_TRUE(snDataSource); + auto selectionsSchema = HdSelectionsSchema::GetFromParent(spherePrim.dataSource); + ASSERT_TRUE(selectionsSchema); + + // Only one selection in the selections schema. + ASSERT_EQ(selectionsSchema.GetNumElements(), 1u); + auto selectionSchema = selectionsSchema.GetElement(0); + + // Prim is fully selected. + auto ds = selectionSchema.GetFullySelected(); + ASSERT_TRUE(ds); + ASSERT_TRUE(ds->GetTypedValue(0.0f)); + + // Selection scene index says the prim is selected. + ASSERT_TRUE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_TRUE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); + + // The shape under the sphere transform is not selected, but it has a + // selected ancestor. + auto mayaShapePath = Ufe::PathString::path("|aSphere|aSphereShape"); + const auto sceneIndexShapePath = selectionSi->SceneIndexPath(mayaShapePath); + + auto sphereShapePrim = sceneIndices.front()->GetPrim(sceneIndexShapePath); + ASSERT_TRUE(sphereShapePrim.dataSource); + dataSourceNames = sphereShapePrim.dataSource->GetNames(); + ASSERT_EQ(std::find(dataSourceNames.begin(), dataSourceNames.end(), HdSelectionsSchemaTokens->selections), dataSourceNames.end()); + ASSERT_FALSE(selectionSi->IsFullySelected(sceneIndexShapePath)); + // HYDRA-626: cannot check for selected ancestor, as shape is in the + // "Lighted" hierarchy, and its selected parent transform is not. + // ASSERT_TRUE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexShapePath)); + + // Remove the sphere from the selection: no longer a selections data source. + ASSERT_EQ(MGlobal::clearSelectionList(), MS::kSuccess); + + spherePrim = sceneIndices.front()->GetPrim(sceneIndexPath); + ASSERT_TRUE(spherePrim.dataSource); + dataSourceNames = spherePrim.dataSource->GetNames(); + ASSERT_EQ(std::find(dataSourceNames.begin(), dataSourceNames.end(), HdSelectionsSchemaTokens->selections), dataSourceNames.end()); + ASSERT_FALSE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_FALSE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); +} + +TEST(FlowViewport, selectionSceneIndexDirty) +{ + // The Flow Viewport selection scene index is in the scene index tree. + const auto& sceneIndices = GetTerminalSceneIndices(); + auto isFvpSelectionSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Selection Scene Index"); + auto selectionSiBase = findSceneIndexInTree( + sceneIndices.front(), isFvpSelectionSceneIndex); + auto selectionSi = TfDynamic_cast(selectionSiBase); + + // The selection scene index observer builds its selection by tracking dirty + // notifications on HdSelectionsSchema::GetDefaultLocator(). Use it to + // ensure dirty notifications are correct. + HdxSelectionSceneIndexObserver ssio; + ssio.SetSceneIndex(selectionSiBase); + + // Clear the Maya selection. + MGlobal::clearSelectionList(); + + // Selection scene index observer should report an empty selection. + auto hdSn = ssio.GetSelection(); + ASSERT_TRUE(hdSn->GetAllSelectedPrimPaths().empty()); + + // Select the sphere + MSelectionList sphereSn; + sphereSn.add("|aSphere"); + const auto mayaPath = Ufe::PathString::path("|aSphere"); + const auto sceneIndexPath = selectionSi->SceneIndexPath(mayaPath); + + MGlobal::setActiveSelectionList(sphereSn); + hdSn = ssio.GetSelection(); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths().size(), 1u); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths()[0], sceneIndexPath); + ASSERT_TRUE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_TRUE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); + + // Remove the sphere from the selection. + MGlobal::clearSelectionList(); + hdSn = ssio.GetSelection(); + ASSERT_TRUE(hdSn->GetAllSelectedPrimPaths().empty()); + ASSERT_FALSE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_FALSE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); + + // Add it back. + MGlobal::setActiveSelectionList(sphereSn); + hdSn = ssio.GetSelection(); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths().size(), 1u); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths()[0], sceneIndexPath); + ASSERT_TRUE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_TRUE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); + + // Delete the sphere: the selection should be empty. + // + // Attempting to delete with MDGModifier crashes with an assert: + // + // ASSERTION: TdependGraph::getInstance().containsNode(object) + // File: Z:\worktrees\master\Maya\src\OGSMayaBridge\ObjectManagement\OGSDagItem.cpp Line: 1696 + // + // MObject sphereObj; + // ASSERT_EQ(sphereSn.getDependNode(0, sphereObj), MS::kSuccess); + // MDGModifier dgMod; + // ASSERT_EQ(dgMod.deleteNode(sphereObj), MS::kSuccess); + // ASSERT_EQ(dgMod.doIt(), MS::kSuccess); + // + // Use the MEL delete command instead. PPT, 19-Oct-2023. + // + ASSERT_EQ(MGlobal::executeCommand("delete |aSphere"), MS::kSuccess); + ASSERT_EQ(MGlobal::executeCommand("refresh"), MS::kSuccess); + + hdSn = ssio.GetSelection(); + ASSERT_TRUE(hdSn->GetAllSelectedPrimPaths().empty()); + ASSERT_FALSE(selectionSi->IsFullySelected(sceneIndexPath)); + ASSERT_FALSE(selectionSi->HasFullySelectedAncestorInclusive(sceneIndexPath)); +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.py new file mode 100644 index 0000000000..282fdf4a7d --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testSelectionSceneIndex.py @@ -0,0 +1,36 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import fixturesUtils +import mtohUtils +from testUtils import PluginLoaded + +class TestSelectionSceneIndex(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def setupScene(self): + self.setHdStormRenderer() + cmds.polySphere(name="aSphere") + cmds.refresh() + + def test_selectionSceneIndex(self): + self.setupScene() + with PluginLoaded('mayaHydraCppTests'): + cmds.mayaHydraCppTest(f="FlowViewport.selectionSceneIndex") + cmds.mayaHydraCppTest(f="FlowViewport.selectionSceneIndexDirty") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.cpp new file mode 100644 index 0000000000..19df5b7f36 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.cpp @@ -0,0 +1,243 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#include "testUtils.h" + +#include +#include +#include + +#include +#include +#include + +#include +#include + +#include + +PXR_NAMESPACE_OPEN_SCOPE +// Bring the MayaHydra namespace into scope. +// The following code currently lives inside the pxr namespace, but it would make more sense to +// have it inside the MayaHydra namespace. This using statement allows us to use MayaHydra symbols +// from within the pxr namespace as if we were in the MayaHydra namespace. +// Remove this once the code has been moved to the MayaHydra namespace. +using namespace MayaHydra; + +const SceneIndicesVector& GetTerminalSceneIndices() +{ + return GetMayaHydraLibInterface().GetTerminalSceneIndices(); +} + +bool MatricesAreClose(const GfMatrix4d& hydraMatrix, const MMatrix& mayaMatrix, double tolerance) +{ + return GfIsClose(hydraMatrix, GetGfMatrixFromMaya(mayaMatrix), tolerance); +} + +SceneIndexInspector::SceneIndexInspector(HdSceneIndexBasePtr sceneIndex) + : _sceneIndex(sceneIndex) +{ +} + +SceneIndexInspector::~SceneIndexInspector() { } + +HdSceneIndexBasePtr SceneIndexInspector::GetSceneIndex() { return _sceneIndex; } + +PrimEntriesVector SceneIndexInspector::FindPrims(FindPrimPredicate predicate, size_t maxPrims) const +{ + PrimEntriesVector searchResults; + _FindPrims(predicate, SdfPath::AbsoluteRootPath(), searchResults, maxPrims); + return searchResults; +} + +void SceneIndexInspector::_FindPrims( + FindPrimPredicate predicate, + const SdfPath& primPath, + PrimEntriesVector& primEntries, + size_t maxPrims) const +{ + HdSceneIndexPrim prim = _sceneIndex->GetPrim(primPath); + if (predicate(_sceneIndex, primPath)) { + primEntries.push_back({ primPath, prim }); + if (maxPrims > 0 && primEntries.size() >= maxPrims) { + return; + } + } else { + auto childPaths = _sceneIndex->GetChildPrimPaths(primPath); + for (auto childPath : childPaths) { + _FindPrims(predicate, childPath, primEntries, maxPrims); + if (maxPrims > 0 && primEntries.size() >= maxPrims) { + return; + } + } + } +} + +void SceneIndexInspector::WriteHierarchy(std::ostream& outStream) const +{ + _WritePrimHierarchy(SdfPath::AbsoluteRootPath(), "", "", outStream); +} + +void SceneIndexInspector::_WritePrimHierarchy( + SdfPath primPath, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const +{ + HdSceneIndexPrim prim = _sceneIndex->GetPrim(primPath); + + outStream << selfPrefix << "@ Prim : " << MakeRelativeToParentPath(primPath) + << " --- Type : " << prim.primType.GetString() << "\n"; + + _WriteContainerDataSource( + prim.dataSource, "", childrenPrefix + "|___", childrenPrefix + " ", outStream); + + auto childPaths = _sceneIndex->GetChildPrimPaths(primPath); + for (auto childPath : childPaths) { + bool isLastChild = childPath == childPaths.back(); + _WritePrimHierarchy( + childPath, + childrenPrefix + "|___", + childrenPrefix + (isLastChild ? " " : "| "), + outStream); + } +} + +void SceneIndexInspector::_WriteContainerDataSource( + HdContainerDataSourceHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const +{ + if (!dataSource) { + return; + } + + outStream << selfPrefix << "# ContainerDataSource : " << dataSourceName << "\n"; + + auto childNames = dataSource->GetNames(); + for (auto childName : childNames) { + bool isLastChild = childName == childNames.back(); + auto child = dataSource->Get(childName); + if (auto childContainer = HdContainerDataSource::Cast(child)) { + _WriteContainerDataSource( + childContainer, + childName.GetString(), + childrenPrefix + "|___", + childrenPrefix + (isLastChild ? " " : "| "), + outStream); + } else if (auto childVector = HdVectorDataSource::Cast(child)) { + _WriteVectorDataSource( + childVector, + childName.GetString(), + childrenPrefix + "|___", + childrenPrefix + (isLastChild ? " " : "| "), + outStream); + } else { + _WriteLeafDataSource(child, childName, childrenPrefix + "|___", outStream); + } + } +} + +void SceneIndexInspector::_WriteVectorDataSource( + HdVectorDataSourceHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const +{ + if (!dataSource) { + return; + } + + outStream << selfPrefix << "# VectorDataSource : " << dataSourceName << "\n"; + + auto numElements = dataSource->GetNumElements(); + for (size_t iElement = 0; iElement < numElements; iElement++) { + std::string childName = "Element " + std::to_string(iElement); + bool isLastElement = iElement == numElements - 1; + auto child = dataSource->GetElement(iElement); + if (auto childContainer = HdContainerDataSource::Cast(child)) { + _WriteContainerDataSource( + childContainer, + childName, + childrenPrefix + "|___", + childrenPrefix + (isLastElement ? " " : "| "), + outStream); + } else if (auto childVector = HdVectorDataSource::Cast(child)) { + _WriteVectorDataSource( + childVector, + childName, + childrenPrefix + "|___", + childrenPrefix + (isLastElement ? " " : "| "), + outStream); + } else { + _WriteLeafDataSource(child, childName, childrenPrefix + "|___", outStream); + } + } +} + +void SceneIndexInspector::_WriteLeafDataSource( + HdDataSourceBaseHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::ostream& outStream) const +{ + std::string dataSourceDescription; + if (auto blockDataSource = HdBlockDataSource::Cast(dataSource)) { + dataSourceDescription = "BlockDataSource"; + } else if (auto sampledDataSource = HdSampledDataSource::Cast(dataSource)) { + dataSourceDescription + = "SampledDataSource -> " + sampledDataSource->GetValue(0).GetTypeName(); + } else if ( + auto extComputationCallbackDataSource + = HdExtComputationCallbackDataSource::Cast(dataSource)) { + dataSourceDescription = "ExtComputationCallbackDataSource"; + } else { + dataSourceDescription = "Unidentified data source type"; + } + outStream << selfPrefix << dataSourceDescription << " : " << dataSourceName << "\n"; +} + +HdSceneIndexBaseRefPtr findSceneIndexInTree( + const HdSceneIndexBaseRefPtr& sceneIndex, + const std::function& predicate +) +{ + if (predicate(sceneIndex)) { + return sceneIndex; + } + auto filteringSi = TfDynamic_cast( + sceneIndex); + // End recursion at leaf scene indices, which are not filtering scene + // indices. + if (!filteringSi) { + return {}; + } + + auto sceneIndices = filteringSi->GetInputScenes(); + if (!sceneIndices.empty()) { + for (const auto& childSceneIndex : sceneIndices) { + if (auto si = findSceneIndexInTree(childSceneIndex, predicate)) { + return si; + } + } + } + return {}; +} + +PXR_NAMESPACE_CLOSE_SCOPE diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.h b/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.h new file mode 100644 index 0000000000..69b9f8bbb5 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testUtils.h @@ -0,0 +1,179 @@ +// +// Copyright 2023 Autodesk, Inc. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +#ifndef MAYAHYDRA_TEST_UTILS_H +#define MAYAHYDRA_TEST_UTILS_H + +#include +#include + +#include +#include + +#include +#include +#include + +PXR_NAMESPACE_OPEN_SCOPE + +constexpr double DEFAULT_TOLERANCE = std::numeric_limits::epsilon(); + +using SceneIndicesVector = std::vector; + +/** + * @brief Retrieve the list of registered terminal scene indices from the Hydra plugin + * + * @return A reference to the vector of registered terminal scene indices. + */ +const SceneIndicesVector& GetTerminalSceneIndices(); + +/** + * @brief Compare a Hydra and a Maya matrix and return whether they are similar + * + * Compare a Hydra and a Maya matrix and return whether the difference between each of their + * corresponding elements is less than or equal to the given tolerance. + * + * @param[in] hydraMatrix is the Hydra matrix + * @param[in] mayaMatrix is the Maya matrix + * @param[in] tolerance is the maximum allowed difference between two corresponding elements of the + * matrices. The default value is the epsilon for double precision floating-point numbers. + * + * @return True if the two matrices are similar enough given the tolerance, false otherwise. + */ +bool MatricesAreClose( + const GfMatrix4d& hydraMatrix, + const MMatrix& mayaMatrix, + double tolerance = DEFAULT_TOLERANCE); + +struct PrimEntry +{ + SdfPath primPath; + HdSceneIndexPrim prim; +}; + +using FindPrimPredicate + = std::function; + +using PrimEntriesVector = std::vector; + +class SceneIndexInspector +{ +public: + SceneIndexInspector(HdSceneIndexBasePtr sceneIndex); + ~SceneIndexInspector(); + + /** + * @brief Retrieve the underlying scene index of this inspector + * + * The returned pointer is non-owning. + * + * @return A pointer to the underlying scene index of this inspector. + */ + HdSceneIndexBasePtr GetSceneIndex(); + + /** + * @brief Retrieve all prims that match the given predicate, up until the maximum amount + * + * A maximum amount of 0 means unlimited (all matching prims will be returned). + * + * @param[in] predicate is the callable predicate used to determine whether a given prim is + * desired + * @param[in] maxPrims is the maximum amount of prims to be retrieved. The default value is 0 + * (unlimited). + * + * @return A vector of the prim entries that matched the given predicate. + */ + PrimEntriesVector FindPrims(FindPrimPredicate predicate, size_t maxPrims = 0) const; + + /** + * @brief Print the scene index's hierarchy in a tree-like format + * + * Print the scene index's hierarchy in a tree-like format, down to the individual data + * source level. + * + * @param[out] outStream is the stream in which to print the hierarchy + */ + void WriteHierarchy(std::ostream& outStream) const; + +private: + void _FindPrims( + FindPrimPredicate predicate, + const SdfPath& primPath, + PrimEntriesVector& primEntries, + size_t maxPrims) const; + + void _WritePrimHierarchy( + SdfPath primPath, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const; + + void _WriteContainerDataSource( + HdContainerDataSourceHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const; + + void _WriteVectorDataSource( + HdVectorDataSourceHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::string childrenPrefix, + std::ostream& outStream) const; + + void _WriteLeafDataSource( + HdDataSourceBaseHandle dataSource, + std::string dataSourceName, + std::string selfPrefix, + std::ostream& outStream) const; + + HdSceneIndexBasePtr _sceneIndex; +}; + +class SceneIndexDisplayNamePred { + const std::string _name; +public: + SceneIndexDisplayNamePred(const std::string& name) : _name(name) {} + + /** + * @brief Predicate to match a scene index display name string. + * + * @param[in] sceneIndex The scene index to test. + * + * @return True if the argument scene index matches the display name string, false otherwise. + */ + bool operator()(const HdSceneIndexBaseRefPtr& sceneIndex) { + return sceneIndex->GetDisplayName() == _name; + } +}; + +/** + * @brief Find the first scene index matching argument predicate in depth first search. + * + * @param[in] sceneIndex The root of the scene index tree to search. + * @param[in] predicate The predicate that determines a match. + * + * @return Scene index pointer if the predicate succeeds, otherwise nullptr. + */ +HdSceneIndexBaseRefPtr findSceneIndexInTree( + const HdSceneIndexBaseRefPtr& sceneIndex, + const std::function& predicate +); + +PXR_NAMESPACE_CLOSE_SCOPE + +#endif // MAYAHYDRA_TEST_UTILS_H diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.cpp b/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.cpp new file mode 100644 index 0000000000..3b3b553413 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.cpp @@ -0,0 +1,245 @@ + +#include "testUtils.h" + +#include + +#include +#include + +#include +#include + +#include +#include +#include +#include + +#include + +PXR_NAMESPACE_USING_DIRECTIVE +using namespace MayaHydra; + +namespace { + +/// \class WfShSiObserver +/// +/// Wireframe selection highlight scene index observer to observe dirty +/// notifications on prims. +/// +class WfShSiObserver : public HdSceneIndexObserver +{ +public: + WfShSiObserver(const HdSceneIndexBaseRefPtr& sceneIndex) + : _sceneIndex(sceneIndex) + { + _sceneIndex->AddObserver(HdSceneIndexObserverPtr(this)); + } + + void PrimsDirtied( + const HdSceneIndexBase& sender, + const DirtiedPrimEntries& entries + ) override + { + for (const auto& entry : entries) { + if (entry.dirtyLocators.Contains(Fvp::WireframeSelectionHighlightSceneIndex::ReprSelectorLocator())) { + _dirtiedPrims.insert(entry.primPath); + } + } + } + + const SdfPathSet& DirtiedPrims() const { return _dirtiedPrims; } + void ClearDirtiedPrims() { _dirtiedPrims.clear(); } + + // Don't care about other dirty notifications. + void PrimsAdded(const HdSceneIndexBase&, const AddedPrimEntries&) override + {} + void PrimsRemoved( + const HdSceneIndexBase&, const RemovedPrimEntries& + ) override {} + void PrimsRenamed( + const HdSceneIndexBase&, const RenamedPrimEntries& + ) override {} + +private: + + HdSceneIndexBaseRefPtr _sceneIndex; + + SdfPathSet _dirtiedPrims; +}; + +bool hasSelectionHighlight(const HdSceneIndexPrim& prim) +{ + if (!prim.dataSource) { + return false; + } + + auto taDs = HdTypedSampledDataSource>::Cast( + HdContainerDataSource::Get( + prim.dataSource, + Fvp::WireframeSelectionHighlightSceneIndex::ReprSelectorLocator() + )); + if (!taDs) { + return false; + } + + static VtArray expected({HdReprTokens->refinedWireOnSurf, HdReprTokens->wireOnSurf, TfToken()}); + + return taDs->GetValue(0) == expected; +} + +} + +TEST(FlowViewport, wireframeSelectionHighlightSceneIndex) +{ + // The Flow Viewport wireframe selection highlight scene index is in the + // scene index tree. + const auto& si = GetTerminalSceneIndices(); + ASSERT_GT(si.size(), static_cast(0)); + + auto isFvpWireframeSelectionHighlightSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Wireframe Selection Highlight Scene Index"); + auto wireframeSiBase = findSceneIndexInTree( + si.front(), isFvpWireframeSelectionHighlightSceneIndex); + ASSERT_TRUE(wireframeSiBase); +} + +TEST(FlowViewport, wireframeSelectionHighlightSceneIndexDirty) +{ + // Python scene setup should have created the following scene: + // + // |sphereAndCube + // |sphereAndCubeShape + // /sphereAndCubeParent + // /sphere + // /cube + // |cylinderAndCone + // |cylinderAndConeShape + // /cylinderAndConeParent + // /cylinder + // /cone + + auto scParentPath = Ufe::PathString::path( + "|sphereAndCube|sphereAndCubeShape,/sphereAndCubeParent"); + auto ccParentPath = Ufe::PathString::path( + "|cylinderAndCone|cylinderAndConeShape,/cylinderAndConeParent"); + + auto scItem = Ufe::Hierarchy::createItem(scParentPath); + auto ccItem = Ufe::Hierarchy::createItem(ccParentPath); + auto scHierarchy = Ufe::Hierarchy::hierarchy(scItem); + auto ccHierarchy = Ufe::Hierarchy::hierarchy(ccItem); + + ASSERT_EQ(scHierarchy->children().size(), 2u); + ASSERT_EQ(ccHierarchy->children().size(), 2u); + + const auto& si = GetTerminalSceneIndices(); + auto isFvpMergingSceneIndex = SceneIndexDisplayNamePred( + "Flow Viewport Merging Scene Index"); + auto mergingSi = TfDynamic_cast( + findSceneIndexInTree(si.front(), isFvpMergingSceneIndex)); + + // See testSelectionSceneIndex.cpp for selection scene index observer + // comments. + HdxSelectionSceneIndexObserver ssio; + ssio.SetSceneIndex(si.front()); + WfShSiObserver wfshsio(si.front()); + + // Maya selection API doesn't understand USD data, which can only be + // represented through UFE, so use UFE API to modify Maya selection. + auto sn = Ufe::GlobalSelection::get(); + sn->clear(); + + // Nothing is selected, no wireframe selection highlight repr locator is + // dirty. + auto hdSn = ssio.GetSelection(); + const auto& shDirtiedPrims = wfshsio.DirtiedPrims(); + ASSERT_TRUE(hdSn->GetAllSelectedPrimPaths().empty()); + ASSERT_EQ(shDirtiedPrims.size(), 0u); + + // Select the sphere. + auto spherePath = scParentPath + "sphere"; + auto sphereItem = Ufe::Hierarchy::createItem(spherePath); + sn->append(sphereItem); + + // Find the sphere in the Hydra scene index scene. + auto sphereSiPath = mergingSi->SceneIndexPath(spherePath); + auto cubeSiPath = mergingSi->SceneIndexPath(scParentPath + "cube"); + + // Sphere is selected. + hdSn = ssio.GetSelection(); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths().size(), 1u); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths()[0], sphereSiPath); + + // Sphere is a mesh, so its repr selector locator will be marked dirty. + ASSERT_EQ(shDirtiedPrims.size(), 1u); + ASSERT_NE(shDirtiedPrims.find(sphereSiPath), shDirtiedPrims.end()); + + // Pull on prim, sphere repr selector has been set by wireframe selection + // highlighting. + auto spherePrim = si.front()->GetPrim(sphereSiPath); + ASSERT_TRUE(hasSelectionHighlight(spherePrim)); + + // Cube is not selected and thus has no highlighting. + auto cubePrim = si.front()->GetPrim(cubeSiPath); + ASSERT_FALSE(hasSelectionHighlight(cubePrim)); + + wfshsio.ClearDirtiedPrims(); + ASSERT_EQ(shDirtiedPrims.size(), 0u); + + // Select the cone and cylinder parent. + Ufe::Selection newSn; + newSn.append(ccItem); + + sn->replaceWith(newSn); + + auto ccSiPath = mergingSi->SceneIndexPath(ccParentPath); + auto coneSiPath = mergingSi->SceneIndexPath(ccParentPath + "cone"); + auto cylinderSiPath = mergingSi->SceneIndexPath(ccParentPath + "cylinder"); + + // Cone and cylinder parent is selected. + // Cone is not selected. + // Cylinder is not selected. + // Sphere is not selected. + hdSn = ssio.GetSelection(); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths().size(), 1u); + ASSERT_EQ(hdSn->GetAllSelectedPrimPaths()[0], ccSiPath); + + // Sphere repr selector locator is dirty. + // Cube repr selector locator is NOT dirty. + // Cone and cylinder parent repr selector locator is dirty. + // Cone repr selector locator is dirty. + // Cylinder repr selector locator is dirty. + ASSERT_NE(shDirtiedPrims.find(sphereSiPath), shDirtiedPrims.end()); + ASSERT_EQ(shDirtiedPrims.find(cubeSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(ccSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(coneSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(cylinderSiPath), shDirtiedPrims.end()); + + wfshsio.ClearDirtiedPrims(); + + // Cone and cylinder parent is selected but has no highlight repr, as it is + // not a mesh. Cone and cylinder have selection lighlight repr. + ASSERT_FALSE(hasSelectionHighlight(si.front()->GetPrim(ccSiPath))); + ASSERT_TRUE(hasSelectionHighlight(si.front()->GetPrim(coneSiPath))); + ASSERT_TRUE(hasSelectionHighlight(si.front()->GetPrim(cylinderSiPath))); + + // Clear selection. + sn->clear(); + hdSn = ssio.GetSelection(); + ASSERT_TRUE(hdSn->GetAllSelectedPrimPaths().empty()); + + // Sphere and cube repr selector locators are NOT dirty, as these were not + // in the selection, nor did they have a selected ancestor. + // Cone and cylinder parent repr selector locator is dirty. + // Cone and cylinder repr selector locators are dirty, as they had a + // selected ancestor. + ASSERT_EQ(shDirtiedPrims.find(sphereSiPath), shDirtiedPrims.end()); + ASSERT_EQ(shDirtiedPrims.find(cubeSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(ccSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(coneSiPath), shDirtiedPrims.end()); + ASSERT_NE(shDirtiedPrims.find(cylinderSiPath), shDirtiedPrims.end()); + + // Selection cleared: no more selection highlighting. + ASSERT_FALSE(hasSelectionHighlight(si.front()->GetPrim(ccSiPath))); + ASSERT_FALSE(hasSelectionHighlight(si.front()->GetPrim(coneSiPath))); + ASSERT_FALSE(hasSelectionHighlight(si.front()->GetPrim(cylinderSiPath))); +} diff --git a/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.py b/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.py new file mode 100644 index 0000000000..0e2a595606 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/cpp/testWireframeSelectionHighlightSceneIndex.py @@ -0,0 +1,30 @@ +import maya.cmds as cmds +import fixturesUtils +import mtohUtils +import mayaUtils +from testUtils import PluginLoaded + +import unittest + +class TestWireframeSelectionHighlightSceneIndex(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + @unittest.skipUnless(mtohUtils.checkForMayaUsdPlugin(), "Requires Maya USD Plugin.") + def test_wireframeSelectionHighlightSceneIndex(self): + self.setHdStormRenderer() + with PluginLoaded('mayaHydraCppTests'): + cmds.refresh() + cmds.mayaHydraCppTest( + f="FlowViewport.wireframeSelectionHighlightSceneIndex") + + testFile = mayaUtils.openTestScene( + "testWireframeSelectionHighlight", + "testSelectionHighlightHierarchy.ma") + cmds.refresh() + + cmds.mayaHydraCppTest( + f="FlowViewport.wireframeSelectionHighlightSceneIndexDirty") + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testImageDiffing.py b/test/lib/mayaUsd/render/mayaToHydra/testImageDiffing.py new file mode 100644 index 0000000000..71c7cdc515 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testImageDiffing.py @@ -0,0 +1,93 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import os.path +import sys +import unittest + +import maya.cmds as cmds +import maya.mel + +import fixturesUtils +import mtohUtils + +class TestImageDiffing(mtohUtils.MtohTestCase): + """Test the image diffing setup to make sure it works and returns the expected results.""" + + _file = __file__ + + # Image files for comparison + COLORED_STRIPES_REFERENCE = "colored_stripes.png" + COLORED_STRIPES_ONE_PIXEL_OFF = "colored_stripes_one_pixel_off.png" + COLORED_STRIPES_SLIGHT_NOISE = "colored_stripes_slight_noise.png" + + CUBE_SCENE_REFERENCE = "cube_scene.png" + CUBE_SCENE_ONE_PIXEL_OFF = "cube_scene_one_pixel_off.png" + CUBE_SCENE_SLIGHT_NOISE = "cube_scene_slight_noise.png" + + # Thresholds used in image diffing to tolerate differences only if they are rare + RARE_DIFFS_FAIL_THRESHOLD = 0 + RARE_DIFFS_FAIL_PERCENT = 5 + + # Thresholds used in image diffing to tolerate differences only if they are small + SMALL_DIFFS_FAIL_THRESHOLD = 0.1 + SMALL_DIFFS_FAIL_PERCENT = 0 + + # Defines the camera distance used when rendering the cube scene + CUBE_SCENE_CAMERA_DISTANCE = 2 + + def setupCubeScene(self): + self.setHdStormRenderer() + self.makeCubeScene(self.CUBE_SCENE_CAMERA_DISTANCE) + + def test_assertImagesClose(self): + # Closeness criteria : tolerate differences only if they are rare + self.assertImagesClose(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_ONE_PIXEL_OFF, + self.RARE_DIFFS_FAIL_THRESHOLD, self.RARE_DIFFS_FAIL_PERCENT) + with self.assertRaises(AssertionError): + self.assertImagesClose(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_SLIGHT_NOISE, + self.RARE_DIFFS_FAIL_THRESHOLD, self.RARE_DIFFS_FAIL_PERCENT) + + # Closeness criteria : tolerate differences only if they are small + self.assertImagesClose(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_SLIGHT_NOISE, + self.SMALL_DIFFS_FAIL_THRESHOLD, self.SMALL_DIFFS_FAIL_PERCENT) + with self.assertRaises(AssertionError): + self.assertImagesClose(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_ONE_PIXEL_OFF, + self.SMALL_DIFFS_FAIL_THRESHOLD, self.SMALL_DIFFS_FAIL_PERCENT) + + def test_assertImagesEqual(self): + self.assertImagesEqual(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_REFERENCE) + with self.assertRaises(AssertionError): + self.assertImagesEqual(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_ONE_PIXEL_OFF) + with self.assertRaises(AssertionError): + self.assertImagesEqual(self.COLORED_STRIPES_REFERENCE, self.COLORED_STRIPES_SLIGHT_NOISE) + + def test_assertSnapshotClose(self): + self.setupCubeScene() + + # Closeness criteria : tolerate differences only if they are rare + self.assertSnapshotClose(self.CUBE_SCENE_ONE_PIXEL_OFF, self.RARE_DIFFS_FAIL_THRESHOLD, self.RARE_DIFFS_FAIL_PERCENT) + with self.assertRaises(AssertionError): + self.assertSnapshotClose(self.CUBE_SCENE_SLIGHT_NOISE, self.RARE_DIFFS_FAIL_THRESHOLD, self.RARE_DIFFS_FAIL_PERCENT) + + # Closeness criteria : tolerate differences only if they are small + self.assertSnapshotClose(self.CUBE_SCENE_SLIGHT_NOISE, self.SMALL_DIFFS_FAIL_THRESHOLD, self.SMALL_DIFFS_FAIL_PERCENT) + with self.assertRaises(AssertionError): + self.assertSnapshotClose(self.CUBE_SCENE_ONE_PIXEL_OFF, self.SMALL_DIFFS_FAIL_THRESHOLD, self.SMALL_DIFFS_FAIL_PERCENT) + + # We do not test for assertSnapshotEqual, as we don't have a scene that renders identically between renderer architectures. + # Since we can't guarantee equality of renders in general, the use of this method is discouraged. + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testMeshes.py b/test/lib/mayaUsd/render/mayaToHydra/testMeshes.py new file mode 100644 index 0000000000..9b6e51682f --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testMeshes.py @@ -0,0 +1,66 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import maya.mel as mel + +import fixturesUtils +import mtohUtils +import mayaUtils +from testUtils import PluginLoaded + +import unittest + +class TestMeshes(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def matchingRprims(self, rprims, matching): + return len([rprim for rprim in rprims if matching in rprim]) + + @unittest.skipUnless(mayaUtils.hydraFixLevel() > 0, "Requires Data Server render item lifescope fix.") + def test_sweepMesh(self): + self.setHdStormRenderer() + with PluginLoaded('sweep'): + mel.eval("performSweepMesh 0") + cmds.refresh() + + # There should be a single rprim from the sweep shape. + rprims = self.getIndex() + self.assertEqual(1, self.matchingRprims(rprims, 'sweepShape')) + + # Change the scale profile. + cmds.setAttr("sweepMeshCreator1.scaleProfileX", 2) + cmds.refresh() + + # Should still be a single rprim from the sweep shape. + rprims = self.getIndex() + self.assertEqual(1, self.matchingRprims(rprims, 'sweepShape')) + + def test_meshSolid(self): + '''Test that meshes are under the common Solid root for lighting / shadowing.''' + self.setHdStormRenderer() + cmds.polySphere(r=1, sx=20, sy=20, ax=[0, 1, 0], cuv=2 , ch=1) + cmds.refresh() + + # There should be two rprims from the poly sphere, one for the mesh and + # another wireframe for selection highlighting. + rprims = self.getIndex() + self.assertEqual(2, len(rprims)) + + # Only the mesh rprim is a child of the "Lighted" hierarchy. + self.assertEqual(1, self.matchingRprims(rprims, 'Lighted')) + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testMtohBasicRender.py b/test/lib/mayaUsd/render/mayaToHydra/testMtohBasicRender.py new file mode 100644 index 0000000000..deb6b518b9 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testMtohBasicRender.py @@ -0,0 +1,79 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Test to make sure that our snapshot-comparison tools work + +import os.path +import sys +import unittest + +import maya.cmds as cmds +import maya.mel + +import fixturesUtils +import mtohUtils + +class TestSnapshot(mtohUtils.MtohTestCase): + """Tests whether our snapshot rendering works with basic Viewport 2.0""" + + _file = __file__ + + def test_flat_orange(self): + cmds.file(new=1, f=1) + + activeEditor = cmds.playblast(activeEditor=1) + + # Note that we use the default viewport2 renderer, because we're not testing + # whether hdmaya works with this test - just whether we can make a snapshot + + cmds.modelEditor( + activeEditor, e=1, + rendererName='vp2Renderer') + cmds.modelEditor( + activeEditor, e=1, + rendererOverrideName="") + + cube = cmds.polyCube()[0] + shader = cmds.shadingNode("surfaceShader", asShader=1) + cmds.select(cube) + cmds.hyperShade(assign=shader) + + COLOR = (.75, .5, .25) + cmds.setAttr('{}.outColor'.format(shader), type='float3', *COLOR) + + cmds.setAttr('persp.rotate', 0, 0, 0, type='float3') + cmds.setAttr('persp.translate', 0, .25, .7, type='float3') + + self.assertSnapshotEqual("flat_orange.png") + self.assertRaises(AssertionError, + self.assertSnapshotEqual, "flat_orange_bad.png") + +class TestMayaHydraRender(mtohUtils.MtohTestCase): + _file = __file__ + + def test_cube(self): + imageVersion = None + if maya.mel.eval("defaultShaderName") != "standardSurface1": + imageVersion = 'lambertDefaultMaterial' + + self.makeCubeScene(camDist=6) + self.assertSnapshotEqual("cube_unselected.png", imageVersion) + cmds.select(self.cubeTrans) + self.assertSnapshotEqual("cube_selected.png", imageVersion) + + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) + diff --git a/test/lib/mayaUsd/render/mayaToHydra/testMtohCommand.py b/test/lib/mayaUsd/render/mayaToHydra/testMtohCommand.py new file mode 100644 index 0000000000..07b4feb2fb --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testMtohCommand.py @@ -0,0 +1,120 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import sys +import unittest + +import maya.cmds as cmds +import maya.mel as mel + +import fixturesUtils +import mtohUtils + +class TestCommand(unittest.TestCase): + _file = __file__ + _has_embree = None + + @classmethod + def setUpClass(cls): + loaded = cmds.loadPlugin('mayaHydra', quiet=True) + if loaded != ['mayaHydra']: + raise RuntimeError('mayaHydra plugin load failed.') + + @classmethod + def has_embree(cls): + import pxr.Plug + if cls._has_embree is None: + plug_reg = pxr.Plug.Registry() + cls._has_embree = bool(plug_reg.GetPluginWithName('hdEmbree')) + return cls._has_embree + + def test_invalidFlag(self): + self.assertRaises(TypeError, cmds.mayaHydra, nonExistantFlag=1) + + def test_listRenderers(self): + renderers = cmds.mayaHydra(listRenderers=1) + self.assertEqual(renderers, cmds.mayaHydra(lr=1)) + self.assertIn(mtohUtils.HD_STORM, renderers) + if self.has_embree(): + self.assertIn("HdEmbreeRendererPlugin", renderers) + + def test_listActiveRenderers(self): + activeRenderers = cmds.mayaHydra(listActiveRenderers=1) + self.assertEqual(activeRenderers, cmds.mayaHydra(lar=1)) + self.assertEqual(activeRenderers, []) + + activeEditor = cmds.playblast(ae=1) + cmds.modelEditor( + activeEditor, e=1, + rendererOverrideName=mtohUtils.HD_STORM_OVERRIDE) + cmds.refresh(f=1) + + activeRenderers = cmds.mayaHydra(listActiveRenderers=1) + self.assertEqual(activeRenderers, cmds.mayaHydra(lar=1)) + self.assertEqual(activeRenderers, [mtohUtils.HD_STORM]) + + if self.has_embree(): + cmds.modelEditor( + activeEditor, e=1, + rendererOverrideName="mtohRenderOverride_HdEmbreeRendererPlugin") + cmds.refresh(f=1) + + activeRenderers = cmds.mayaHydra(listActiveRenderers=1) + self.assertEqual(activeRenderers, cmds.mayaHydra(lar=1)) + self.assertEqual(activeRenderers, ["HdEmbreeRendererPlugin"]) + + cmds.modelEditor(activeEditor, rendererOverrideName="", e=1) + cmds.refresh(f=1) + + activeRenderers = cmds.mayaHydra(listActiveRenderers=1) + self.assertEqual(activeRenderers, cmds.mayaHydra(lar=1)) + self.assertEqual(activeRenderers, []) + + def test_getRendererDisplayName(self): + # needs at least one arg + self.assertRaises(RuntimeError, mel.eval, + "mayaHydra -getRendererDisplayName") + + displayName = cmds.mayaHydra(renderer=mtohUtils.HD_STORM, + getRendererDisplayName=True) + self.assertEqual(displayName, cmds.mayaHydra(r=mtohUtils.HD_STORM, gn=True)) + self.assertEqual(displayName, "GL") + + if self.has_embree(): + displayName = cmds.mayaHydra(renderer="HdEmbreeRendererPlugin", + getRendererDisplayName=True) + self.assertEqual(displayName, cmds.mayaHydra(r="HdEmbreeRendererPlugin", + gn=True)) + self.assertEqual(displayName, "Embree") + + def test_listDelegates(self): + delegates = cmds.mayaHydra(listDelegates=1) + self.assertEqual(delegates, cmds.mayaHydra(ld=1)) + self.assertIn("MayaHydraSceneDelegate", delegates) + + def test_createRenderGlobals(self): + for flag in ("createRenderGlobals", "crg"): + cmds.file(f=1, new=1) + self.assertFalse(cmds.objExists( + "defaultRenderGlobals.mtohMotionSampleStart")) + cmds.mayaHydra(**{flag: 1}) + self.assertTrue(cmds.objExists( + "defaultRenderGlobals.mtohMotionSampleStart")) + self.assertFalse(cmds.getAttr( + "defaultRenderGlobals.mtohMotionSampleStart")) + + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testMtohDagChanges.py b/test/lib/mayaUsd/render/mayaToHydra/testMtohDagChanges.py new file mode 100644 index 0000000000..2115672c26 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testMtohDagChanges.py @@ -0,0 +1,322 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import sys +import unittest + +import maya.cmds as cmds +import maya.mel + +import fixturesUtils +import mtohUtils + +class TestDagChanges(mtohUtils.MtohTestCase): + _file = __file__ + + def setUp(self): + self.makeCubeScene() + + self.grp1 = cmds.createNode('transform', name='group1') + self.grp1Rprim = self.rprimPath(self.grp1) + + self.grp2 = cmds.createNode('transform', name='group2') + self.grp2Rprim = self.rprimPath(self.grp2) + + self.imageVersion = None + if maya.mel.eval("defaultShaderName") != "standardSurface1": + self.imageVersion = 'lambertDefaultMaterial' + + def test_reparent_transform(self): + cmds.parent(self.cubeTrans, self.grp1) + grp1ShapeRprim = self.rprimPath(self.cubeShape) + self.assertEqual( + grp1ShapeRprim, + self.rprimPath("|{self.grp1}|{self.cubeTrans}|{self.cubeShape}" + .format(self=self))) + cmds.refresh() + index = self.getIndex() + self.assertIn(grp1ShapeRprim, index) + self.assertNotIn(self.cubeRprim, index) + + cmds.parent(self.grp1, self.grp2) + grp2ShapeRprim = self.rprimPath(self.cubeShape) + self.assertEqual( + grp2ShapeRprim, + self.rprimPath("|{self.grp2}|{self.grp1}|{self.cubeTrans}|{self.cubeShape}" + .format(self=self))) + cmds.refresh() + index = self.getIndex() + self.assertIn(grp2ShapeRprim, index) + self.assertNotIn(grp1ShapeRprim, index) + self.assertNotIn(self.cubeRprim, index) + + cmds.parent(self.cubeTrans, world=True) + origShapePrim = self.rprimPath(self.cubeShape) + self.assertEqual(origShapePrim, self.cubeRprim) + cmds.refresh() + index = self.getIndex() + self.assertIn(self.cubeRprim, index) + self.assertNotIn(grp2ShapeRprim, index) + self.assertNotIn(grp1ShapeRprim, index) + + def test_reparent_shape(self): + cmds.parent(self.cubeShape, self.grp1, shape=1, r=1) + grp1ShapeRprim = self.rprimPath(self.cubeShape) + self.assertEqual( + grp1ShapeRprim, + self.rprimPath("|{self.grp1}|{self.cubeShape}" + .format(self=self))) + cmds.refresh() + index = self.getIndex() + self.assertIn(grp1ShapeRprim, index) + self.assertNotIn(self.cubeRprim, index) + + cmds.parent(self.cubeShape, self.cubeTrans, shape=1, r=1) + origShapePrim = self.rprimPath(self.cubeShape) + self.assertEqual(origShapePrim, self.cubeRprim) + cmds.refresh() + index = self.getIndex() + self.assertIn(self.cubeRprim, index) + self.assertNotIn(grp1ShapeRprim, index) + + def test_new_shape(self): + otherCube = cmds.polyCube()[0] + otherCubeShape = cmds.listRelatives(otherCube, fullPath=1)[0] + otherRprim = self.rprimPath(otherCubeShape) + cmds.refresh() + index = self.getIndex() + self.assertIn(self.cubeRprim, index) + self.assertIn(otherRprim, index) + + def test_instances(self): + undoWasEnabled = cmds.undoInfo(q=1, state=1) + + cmds.undoInfo(state=0) + try: + pCube2 = cmds.createNode('transform', name='pCube2') + + cmds.setAttr('{}.tz'.format(self.grp1), 5) + cmds.setAttr('{}.tx'.format(self.grp2), 8) + cmds.setAttr('{}.ty'.format(pCube2), 5) + + # No instances to start + # (1) |pCube1|pCubeShape1 + self.assertSnapshotEqual("instances_1.png", self.imageVersion) + + # Add |group1|pCube1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + cmds.parent(self.cubeTrans, self.grp1, add=1, r=1) + cmds.select(clear=1) + self.assertSnapshotEqual("instances_12.png", self.imageVersion) + + # Add |pCube2|pCubeShape1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + cmds.parent(self.cubeShape, pCube2, add=1, r=1, shape=1) + cmds.select(clear=1) + self.assertSnapshotEqual("instances_123.png", self.imageVersion) + + # Add |group2|group1|pCube1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (4) |group2||group1|pCube1|pCubeShape1 + cmds.parent(self.grp1, self.grp2, add=1, r=1) + cmds.select(clear=1) + self.assertSnapshotEqual("instances_1234.png", self.imageVersion) + + # Add |group1|pCube2 instance + # (1) |pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (4) |group2||group1|pCube1|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + # (6) |group2||group1|pCube2|pCubeShape1 + cmds.parent(pCube2, self.grp1, add=1, r=1) + cmds.select(clear=1) + self.assertSnapshotEqual("instances_123456.png", self.imageVersion) + + # Delete group2 + # [no shapes] + cmds.undoInfo(state=1) + cmds.undoInfo(openChunk=1) + try: + cmds.delete(self.grp2) + self.assertNotIn(self.cubeRprim, self.getIndex()) + self.assertSnapshotEqual("instances_0.png", self.imageVersion) + finally: + cmds.undoInfo(closeChunk=1) + + # Undo group2 deletion + # (1) |pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (4) |group2||group1|pCube1|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + # (6) |group2||group1|pCube2|pCubeShape1 + cmds.undo() + self.assertSnapshotEqual("instances_123456.png", self.imageVersion) + + # Remove |group2|group1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + cmds.parent('|{self.grp2}|{self.grp1}'.format(self=self), + removeObject=1) + self.assertSnapshotEqual("instances_1235.png", self.imageVersion) + + # Remove pCube2|pCubeShape1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + cmds.undoInfo(openChunk=1) + try: + cmds.parent('|{pCube2}|{self.cubeShape}'.format(self=self, + pCube2=pCube2), + removeObject=1, shape=1) + self.assertSnapshotEqual("instances_12.png", self.imageVersion) + finally: + cmds.undoInfo(closeChunk=1) + + # Undo remove pCube2|pCubeShape1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + cmds.undo() + self.assertSnapshotEqual("instances_1235.png", self.imageVersion) + + # Remove pCube1|pCubeShape1 (the "master" instance) + # (3) |pCube2|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + cmds.undoInfo(openChunk=1) + try: + cmds.parent('|{self.cubeTrans}|{self.cubeShape}'.format(self=self), + removeObject=1, shape=1) + self.assertSnapshotEqual("instances_35.png", self.imageVersion) + finally: + cmds.undoInfo(closeChunk=1) + + # Undo remove pCube1|pCubeShape1 (the "master" instance) + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + # (3) |pCube2|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + cmds.undo() + + # the playblast command is entered into the undo queue, so we + # need to disable without flusing the queue, so we can test redo + cmds.undoInfo(stateWithoutFlush=0) + try: + self.assertSnapshotEqual("instances_1235.png", self.imageVersion) + finally: + cmds.undoInfo(stateWithoutFlush=1) + + # Redo remove pCube2|pCubeShape1 instance + # (3) |pCube2|pCubeShape1 + # (5) |group1|pCube2|pCubeShape1 + cmds.redo() + self.assertSnapshotEqual("instances_35.png", self.imageVersion) + + # Remove |group1|pCube2 instance + # (3) |pCube2|pCubeShape1 + cmds.parent('{self.grp1}|{pCube2}'.format(self=self, + pCube2=pCube2), + removeObject=1) + self.assertSnapshotEqual("instances_3.png", self.imageVersion) + finally: + cmds.undoInfo(state=undoWasEnabled) + + def test_move(self): + self.assertSnapshotEqual("instances_1.png", self.imageVersion) + cmds.setAttr('{}.ty'.format(self.cubeTrans), 5) + self.assertSnapshotEqual("instances_3.png", self.imageVersion) + + def test_instance_move(self): + # Add |group1|pCube1 instance + # (1) |pCube1|pCubeShape1 + # (2) |group1|pCube1|pCubeShape1 + cmds.parent(self.cubeTrans, self.grp1, add=1, r=1) + cmds.select(clear=1) + + # because we haven't moved anything, it should initially look like only + # one cube... + self.assertSnapshotEqual("instances_1.png", self.imageVersion) + + cmds.setAttr('{}.tz'.format(self.grp1), 5) + # Now that we moved one, it should look like 2 cubes + self.assertSnapshotEqual("instances_12.png", self.imageVersion) + + +class TestUndo(mtohUtils.MtohTestCase): + _file = __file__ + + def test_node_creation_undo(self): + undoWasEnabled = cmds.undoInfo(q=1, state=1) + + self.imageVersion = None + if maya.mel.eval("defaultShaderName") != "standardSurface1": + self.imageVersion = 'lambertDefaultMaterial' + + cmds.undoInfo(state=0) + try: + cmds.file(new=1, f=1) + self.setBasicCam() + + self.setHdStormRenderer() + + cmds.undoInfo(state=1) + cmds.undoInfo(openChunk=1) + try: + cubeTrans = cmds.polyCube() + cubeShape = cmds.listRelatives(cubeTrans)[0] + cubeRprim = self.rprimPath(cubeShape) + cmds.select(clear=1) + cmds.refresh() + self.assertEqual([cubeRprim], self.getIndex()) + self.assertSnapshotEqual("instances_1.png", self.imageVersion) + finally: + cmds.undoInfo(closeChunk=1) + + cmds.undo() + + # the playblast command is entered into the undo queue, so we + # need to disable without flusing the queue, so we can test redo + cmds.undoInfo(stateWithoutFlush=0) + try: + cmds.refresh() + self.assertEqual([], self.getIndex()) + self.assertSnapshotEqual("instances_0.png", self.imageVersion) + finally: + cmds.undoInfo(stateWithoutFlush=1) + + cmds.redo() + + cmds.undoInfo(stateWithoutFlush=0) + try: + cmds.refresh() + self.assertEqual([cubeRprim], self.getIndex()) + self.assertSnapshotEqual("instances_1.png", self.imageVersion) + finally: + cmds.undoInfo(stateWithoutFlush=1) + + finally: + cmds.undoInfo(state=undoWasEnabled) + + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testNamespaces.py b/test/lib/mayaUsd/render/mayaToHydra/testNamespaces.py new file mode 100644 index 0000000000..c3c753324a --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testNamespaces.py @@ -0,0 +1,53 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds + +import fixturesUtils +import mtohUtils + +class TestNamespaces(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + def matchingRprims(self, rprims, matching): + return len([rprim for rprim in rprims if matching in rprim]) + + def test_namespaces(self): + '''Test that Maya objects in namespaces are supported.''' + self.setHdStormRenderer() + + # Create a namespace and set it current + cmds.namespace(add='A') + cmds.namespace(set='A') + + # Create an object in the namespace + polyObjs = cmds.polySphere(r=1, sx=20, sy=20, ax=[0, 1, 0], cuv=2 , ch=1) + self.assertEqual(polyObjs[0], 'A:pSphere1') + + cmds.refresh() + + # There should be two rprims from the poly sphere, one for the mesh and + # another wireframe for selection highlighting. + rprims = self.getIndex() + self.assertEqual(2, len(rprims)) + + # Path sanitizing should leave the node name intact. + self.assertEqual(2, self.matchingRprims(rprims, 'pSphereShape1')) + + # Set the namespace back to the root. + cmds.namespace(set=':') + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testNewSceneWithStage.py b/test/lib/mayaUsd/render/mayaToHydra/testNewSceneWithStage.py new file mode 100644 index 0000000000..044a6988af --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testNewSceneWithStage.py @@ -0,0 +1,51 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds + +import fixturesUtils +import mtohUtils + +import unittest + +class TestStage(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + @unittest.skipUnless(mtohUtils.checkForMayaUsdPlugin(), "Requires Maya USD Plugin.") + def test_newFileWithUsdStage(self): + import mayaUsd_createStageWithNewLayer + import mayaUsd.lib + + self.setHdStormRenderer() + # create an empty USD Stage + psPathStr = mayaUsd_createStageWithNewLayer.createStageWithNewLayer() + + # create a new Maya scene. + # See https://jira.autodesk.com/browse/HYDRA-496 + # prior to this fix, Maya would crash + cmds.file( f=True, new=True ) + + # sanity check + self.setHdStormRenderer() + # start with a maya sphere + sphere = cmds.polySphere() + cmds.refresh() + rprims = self.getIndex() + # we expect non-zero rprims + rprimsCount = len(rprims) + self.assertGreater(rprimsCount, 0) + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testRendererSwitching.py b/test/lib/mayaUsd/render/mayaToHydra/testRendererSwitching.py new file mode 100644 index 0000000000..a74c4d3246 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testRendererSwitching.py @@ -0,0 +1,35 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds + +import fixturesUtils +import mtohUtils + +class TestRendererSwitching(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + # We found a crash that happened when switching from VP2 -> Hydra -> VP2 -> Hydra. + # The crash was related to Hydra resources management, where some resources were not + # properly unloaded. The issue would then only become apparent when trying to recreate + # the resources by switching to Hydra a second time : this is what this test is for. + def test_RendererSwitching(self): + cmds.polySphere() # The crash only occurred when the scene was not empty, so create a dummy sphere. + self.setHdStormRenderer() + self.setViewport2Renderer() + self.setHdStormRenderer() + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testSceneBrowser.py b/test/lib/mayaUsd/render/mayaToHydra/testSceneBrowser.py new file mode 100644 index 0000000000..ffcd33d1f9 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testSceneBrowser.py @@ -0,0 +1,54 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import maya.mel as mel + +import fixturesUtils +import mtohUtils +import unittest + +from testUtils import PluginLoaded + +class TestSceneBrowser(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + SCENE_BROWSER_TEST_PLUGIN_NAME = 'mayaHydraSceneBrowserTest' + SCENE_BROWSER_TEST_PLUGIN_COMMAND = SCENE_BROWSER_TEST_PLUGIN_NAME + + def setupScene(self): + import mayaUsd + import mayaUsd_createStageWithNewLayer + from pxr import UsdGeom, UsdLux + + self.setHdStormRenderer() + cmds.refresh() + usdProxyShapeUfePathString = mayaUsd_createStageWithNewLayer.createStageWithNewLayer() + usdStage = mayaUsd.lib.GetPrim(usdProxyShapeUfePathString).GetStage() + UsdGeom.Cylinder.Define(usdStage, "/USDCylinder") + UsdLux.RectLight.Define(usdStage, "/USDRectLight") + cmds.polySphere(name="MayaTorus") + cmds.directionalLight(name="MayaDirectionalLight") + cmds.refresh() + + @unittest.skipUnless(mtohUtils.checkForMayaUsdPlugin() and mtohUtils.checkForPlugin(SCENE_BROWSER_TEST_PLUGIN_NAME), + f'Requires mayaUSD and {SCENE_BROWSER_TEST_PLUGIN_NAME} plugins.') + def test_SceneBrowser(self): + self.setupScene() + with PluginLoaded(self.SCENE_BROWSER_TEST_PLUGIN_NAME): + mel.eval(self.SCENE_BROWSER_TEST_PLUGIN_COMMAND) + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testStageAddPrim.py b/test/lib/mayaUsd/render/mayaToHydra/testStageAddPrim.py new file mode 100644 index 0000000000..f4d0e1abbf --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testStageAddPrim.py @@ -0,0 +1,66 @@ +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds + +import fixturesUtils +import mtohUtils +import mayaUtils +from testUtils import PluginLoaded + +import unittest +import sys + +class TestStage(mtohUtils.MayaHydraBaseTestCase): + # MayaHydraBaseTestCase.setUpClass requirement. + _file = __file__ + + @unittest.skipUnless(mtohUtils.checkForMayaUsdPlugin(), "Requires Maya USD Plugin.") + def test_addPrim(self): + import mayaUsd_createStageWithNewLayer + import mayaUsd.lib + + self.setHdStormRenderer() + # empty scene and expect zero prims + rprims = self.getIndex() + self.assertEqual(0, len(rprims)) + + # start with a maya sphere + sphere = cmds.polySphere() + cmds.refresh() + rprims = self.getIndex() + # we expect non-zero rprims(two here) + rprimsBefore = len(rprims) + self.assertGreater(rprimsBefore, 0) + + # add an empty USD Stage. + psPathStr = mayaUsd_createStageWithNewLayer.createStageWithNewLayer() + + # duplicate the sphere into Stage as USD data + self.assertTrue(mayaUsd.lib.PrimUpdaterManager.duplicate(cmds.ls(sphere[0], long=True)[0], psPathStr)) + cmds.refresh() + rprims = self.getIndex() + rprimsAfter = len(rprims) + self.assertGreater(rprimsAfter, rprimsBefore) + cmds.delete(sphere[0]) + + cmds.refresh() + rprims = self.getIndex() + # we expect a non-zero rprim count(one here) + rprimsAfter = len(rprims) + self.assertGreater(rprimsAfter, 0) + + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/lib/mayaUsd/render/mayaToHydra/testVisibility.py b/test/lib/mayaUsd/render/mayaToHydra/testVisibility.py new file mode 100644 index 0000000000..be6159d426 --- /dev/null +++ b/test/lib/mayaUsd/render/mayaToHydra/testVisibility.py @@ -0,0 +1,179 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import maya.cmds as cmds +import maya.mel + +import fixturesUtils +import mtohUtils + +class TestVisibility(mtohUtils.MtohTestCase): + _file = __file__ + + IMAGEDIFF_FAIL_THRESHOLD = 0.01 + IMAGEDIFF_FAIL_PERCENT = 0.1 + + def setUp(self): + self.makeCubeScene(camDist=6) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeTrans))) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeShape))) + self.imageVersion = None + if maya.mel.eval("defaultShaderName") != "standardSurface1": + self.imageVersion = 'lambertDefaultMaterial' + + def test_toggleTransVis(self): + # because snapshotting is slow, we only use it in this test - otherwise + # we assume the results of `listRenderIndex=..., visibileOnly=1` are + # sufficient + + cubeUnselectedImg = "cube_unselected.png" + nothingImg = "nothing.png" + + cmds.refresh() + self.assertIn( + self.cubeRprim, + self.getVisibleIndex()) + self.assertSnapshotClose(cubeUnselectedImg, self.IMAGEDIFF_FAIL_THRESHOLD, self.IMAGEDIFF_FAIL_PERCENT, + imageVersion=self.imageVersion) + + cmds.setAttr("{}.visibility".format(self.cubeTrans), False) + self.assertFalse(cmds.getAttr("{}.visibility".format(self.cubeTrans))) + cmds.refresh() + self.assertNotIn( + self.cubeRprim, + self.getVisibleIndex()) + self.assertSnapshotClose(nothingImg, self.IMAGEDIFF_FAIL_THRESHOLD, self.IMAGEDIFF_FAIL_PERCENT) + cmds.setAttr("{}.visibility".format(self.cubeTrans), True) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeTrans))) + cmds.refresh() + self.assertIn( + self.cubeRprim, + self.getVisibleIndex()) + self.assertSnapshotClose(cubeUnselectedImg, self.IMAGEDIFF_FAIL_THRESHOLD, self.IMAGEDIFF_FAIL_PERCENT, + imageVersion=self.imageVersion) + + def test_toggleShapeVis(self): + cmds.setAttr("{}.visibility".format(self.cubeShape), False) + self.assertFalse(cmds.getAttr("{}.visibility".format(self.cubeShape))) + cmds.refresh() + self.assertNotIn( + self.cubeRprim, + self.getVisibleIndex()) + + cmds.setAttr("{}.visibility".format(self.cubeShape), True) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeShape))) + cmds.refresh() + self.assertIn( + self.cubeRprim, + self.getVisibleIndex()) + + def test_toggleBothVis(self): + cmds.setAttr("{}.visibility".format(self.cubeTrans), False) + self.assertFalse(cmds.getAttr("{}.visibility".format(self.cubeTrans))) + cmds.setAttr("{}.visibility".format(self.cubeShape), False) + self.assertFalse(cmds.getAttr("{}.visibility".format(self.cubeShape))) + cmds.refresh() + self.assertNotIn( + self.cubeRprim, + self.getVisibleIndex()) + + cmds.setAttr("{}.visibility".format(self.cubeTrans), True) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeTrans))) + cmds.setAttr("{}.visibility".format(self.cubeShape), True) + self.assertTrue(cmds.getAttr("{}.visibility".format(self.cubeShape))) + cmds.refresh() + self.assertIn( + self.cubeRprim, + self.getVisibleIndex()) + + def doHierarchicalVisibilityTest(self, makeNodeVis, makeNodeInvis, prep=None): + lowGroup = cmds.group(self.cubeTrans, name='lowGroup') + midGroup = cmds.group(lowGroup, name='midGroup') + highGroup = cmds.group(midGroup, name='highGroup') + hier = [midGroup, midGroup, lowGroup, self.cubeTrans, self.cubeShape] + cmds.select(clear=1) + cmds.refresh() + + # Hierarchy changes (like grouping above) cause the render index to be + # rebuilt, so re-read our cubeRprim from the render index. + self.cubeRprim = self.getIndex()[0] + visIndex = [self.cubeRprim] + self.assertEqual(self.getVisibleIndex(), visIndex) + + if prep is not None: + for obj in hier: + prep(obj) + self.assertEqual(self.getVisibleIndex(), visIndex) + + for obj in hier: + makeNodeInvis(obj) + cmds.refresh() + self.assertEqual(self.getVisibleIndex(), []) + makeNodeVis(obj) + cmds.refresh() + self.assertEqual(self.getVisibleIndex(), visIndex) + + def test_hierarchicalVisibility(self): + def makeNodeVis(obj): + cmds.setAttr("{}.visibility".format(obj), True) + + def makeNodeInvis(obj): + cmds.setAttr("{}.visibility".format(obj), False) + + self.doHierarchicalVisibilityTest(makeNodeVis, makeNodeInvis) + + def test_hierarchicalIntermediateObject(self): + def makeNodeVis(obj): + cmds.setAttr("{}.intermediateObject".format(obj), False) + + def makeNodeInvis(obj): + cmds.setAttr("{}.intermediateObject".format(obj), True) + + self.doHierarchicalVisibilityTest(makeNodeVis, makeNodeInvis) + + def test_hierarchicalOverrideEnabled(self): + def makeNodeVis(obj): + cmds.setAttr("{}.overrideEnabled".format(obj), False) + + def makeNodeInvis(obj): + cmds.setAttr("{}.overrideEnabled".format(obj), True) + + def prep(obj): + # set the overrideVisibility to False - as long as the + # overrideEnabled is NOT set, the object should still + # be visible + cmds.setAttr("{}.overrideVisibility".format(obj), False) + cmds.setAttr("{}.overrideEnabled".format(obj), False) + + self.doHierarchicalVisibilityTest(makeNodeVis, makeNodeInvis, prep=prep) + + def test_hierarchicalOverrideVisibility(self): + def makeNodeVis(obj): + cmds.setAttr("{}.overrideVisibility".format(obj), True) + + def makeNodeInvis(obj): + cmds.setAttr("{}.overrideVisibility".format(obj), False) + + def prep(obj): + # set the overrideEnabled to True - as long as the + # overrideVisibility is True, the object should still + # be visible + cmds.setAttr("{}.overrideEnabled".format(obj), True) + cmds.setAttr("{}.overrideVisibility".format(obj), True) + + self.doHierarchicalVisibilityTest(makeNodeVis, makeNodeInvis, prep=prep) + +if __name__ == '__main__': + fixturesUtils.runTests(globals()) diff --git a/test/testSamples/testWireframeSelectionHighlight/testSelectionHighlightHierarchy.ma b/test/testSamples/testWireframeSelectionHighlight/testSelectionHighlightHierarchy.ma new file mode 100644 index 0000000000..288746a3d0 --- /dev/null +++ b/test/testSamples/testWireframeSelectionHighlight/testSelectionHighlightHierarchy.ma @@ -0,0 +1,284 @@ +//Maya ASCII 2025ff01 scene +//Name: testSelectionHighlightHierarchy.ma +//Last modified: Mon, Oct 30, 2023 02:04:41 PM +//Codeset: 1252 +requires maya "2025ff01"; +requires -nodeType "mayaUsdLayerManager" -nodeType "mayaUsdProxyShape" -dataType "pxrUsdStageData" + "mayaUsdPlugin" "0.26.0"; +currentUnit -l centimeter -a degree -t film; +fileInfo "application" "maya"; +fileInfo "product" "Maya 2025"; +fileInfo "version" "Preview Release 149"; +fileInfo "cutIdentifier" "202309182026-000000"; +fileInfo "osv" "Windows 10 Enterprise v2009 (Build: 19045)"; +fileInfo "UUID" "0886BD1F-4B27-A043-E7BD-F3BC198F086B"; +createNode transform -s -n "persp"; + rename -uid "F710F780-4D29-D494-C575-71ADEBC5C45E"; + setAttr ".v" no; + setAttr ".t" -type "double3" 28 21 28 ; + setAttr ".r" -type "double3" -27.938352729602379 44.999999999999972 -5.172681101354183e-14 ; +createNode camera -s -n "perspShape" -p "persp"; + rename -uid "144A1FED-4013-3A15-E5D4-64BBC76AED44"; + setAttr -k off ".v" no; + setAttr ".fl" 34.999999999999993; + setAttr ".coi" 44.82186966202994; + setAttr ".imn" -type "string" "persp"; + setAttr ".den" -type "string" "persp_depth"; + setAttr ".man" -type "string" "persp_mask"; + setAttr ".hc" -type "string" "viewSet -p %camera"; +createNode transform -s -n "top"; + rename -uid "60782678-4EE9-EC09-E5F7-3499068E9E13"; + setAttr ".v" no; + setAttr ".t" -type "double3" 0 1000.1 0 ; + setAttr ".r" -type "double3" -90 0 0 ; +createNode camera -s -n "topShape" -p "top"; + rename -uid "4BA06B5B-4E32-3150-4D35-D89954E25047"; + setAttr -k off ".v" no; + setAttr ".rnd" no; + setAttr ".coi" 1000.1; + setAttr ".ow" 30; + setAttr ".imn" -type "string" "top"; + setAttr ".den" -type "string" "top_depth"; + setAttr ".man" -type "string" "top_mask"; + setAttr ".hc" -type "string" "viewSet -t %camera"; + setAttr ".o" yes; +createNode transform -s -n "front"; + rename -uid "596CF505-493C-445D-EC4A-C18414DAAEBA"; + setAttr ".v" no; + setAttr ".t" -type "double3" 0 0 1000.1 ; +createNode camera -s -n "frontShape" -p "front"; + rename -uid "F52F4350-4ADC-9E1E-392A-2AB44A2B2D9C"; + setAttr -k off ".v" no; + setAttr ".rnd" no; + setAttr ".coi" 1000.1; + setAttr ".ow" 30; + setAttr ".imn" -type "string" "front"; + setAttr ".den" -type "string" "front_depth"; + setAttr ".man" -type "string" "front_mask"; + setAttr ".hc" -type "string" "viewSet -f %camera"; + setAttr ".o" yes; +createNode transform -s -n "side"; + rename -uid "7B243DE2-4274-B8BD-8E1D-75A95F03C71A"; + setAttr ".v" no; + setAttr ".t" -type "double3" 1000.1 0 0 ; + setAttr ".r" -type "double3" 0 90 0 ; +createNode camera -s -n "sideShape" -p "side"; + rename -uid "46E34478-4CB0-5938-BC37-0DAB7F1B160F"; + setAttr -k off ".v" no; + setAttr ".rnd" no; + setAttr ".coi" 1000.1; + setAttr ".ow" 30; + setAttr ".imn" -type "string" "side"; + setAttr ".den" -type "string" "side_depth"; + setAttr ".man" -type "string" "side_mask"; + setAttr ".hc" -type "string" "viewSet -s %camera"; + setAttr ".o" yes; +createNode transform -n "sphereAndCube"; + rename -uid "93FFBF48-4A02-4834-8AEC-B9880746AF01"; +createNode mayaUsdProxyShape -n "sphereAndCubeShape" -p "sphereAndCube"; + rename -uid "C0F9037C-4064-F1FA-BB27-149A7FDCFA5F"; + addAttr -r false -ci true -h true -sn "forceCompute" -ln "forceCompute" -min 0 + -max 1 -at "bool"; + addAttr -ci true -h true -sn "usdStageLoadRules" -ln "usdStageLoadRules" -dt "string"; + addAttr -ci true -h true -sn "usdStageTargetLayer" -ln "usdStageTargetLayer" -dt "string"; + setAttr -k off ".v"; + setAttr ".covm[0]" 0 1 1; + setAttr ".cdvm[0]" 0 1 1; + setAttr ".oslid" -type "string" "anon:000001A374D17F00:anonymousLayer1-session.usda"; + setAttr ".orlid" -type "string" "anon:000001A374D193A0:anonymousLayer1"; + setAttr ".usdStageLoadRules" -type "string" ""; + setAttr ".usdStageTargetLayer" -type "string" "anon:000001A374D193A0:anonymousLayer1"; +createNode transform -n "cylinderAndCone"; + rename -uid "112071EE-4266-A4C0-F9D5-9E91EAC5C863"; +createNode mayaUsdProxyShape -n "cylinderAndConeShape" -p "cylinderAndCone"; + rename -uid "1054B18C-41D3-7753-4777-9BB3145C2EC4"; + addAttr -r false -ci true -h true -sn "forceCompute" -ln "forceCompute" -min 0 + -max 1 -at "bool"; + addAttr -ci true -h true -sn "usdStageLoadRules" -ln "usdStageLoadRules" -dt "string"; + addAttr -ci true -h true -sn "usdStageTargetLayer" -ln "usdStageTargetLayer" -dt "string"; + setAttr -k off ".v"; + setAttr ".covm[0]" 0 1 1; + setAttr ".cdvm[0]" 0 1 1; + setAttr ".oslid" -type "string" "anon:000001A373F7E250:anonymousLayer1-session.usda"; + setAttr ".orlid" -type "string" "anon:000001A373F7DA10:anonymousLayer1"; + setAttr ".usdStageLoadRules" -type "string" ""; + setAttr ".usdStageTargetLayer" -type "string" "anon:000001A373F7DA10:anonymousLayer1"; +createNode lightLinker -s -n "lightLinker1"; + rename -uid "67FB831C-4A6C-0C6C-6A19-76B7AD69002D"; + setAttr -s 2 ".lnk"; + setAttr -s 2 ".slnk"; +createNode shapeEditorManager -n "shapeEditorManager"; + rename -uid "C7CA44F6-4D35-8625-8C23-8ABFECCD735C"; +createNode poseInterpolatorManager -n "poseInterpolatorManager"; + rename -uid "EBAE41BA-400D-B038-B6A8-7683789FE259"; +createNode displayLayerManager -n "layerManager"; + rename -uid "C921B5DF-4F81-6A35-39B0-AD82CCEAD367"; +createNode displayLayer -n "defaultLayer"; + rename -uid "F81E8D95-4774-69D6-A85B-96BA480A538C"; + setAttr ".ufem" -type "stringArray" 0 ; +createNode renderLayerManager -n "renderLayerManager"; + rename -uid "6A3C2C0A-43B6-F7CF-95AB-558438B14E62"; +createNode renderLayer -n "defaultRenderLayer"; + rename -uid "44CD1AF6-4BAA-8527-FF29-05AFBD8D352B"; + setAttr ".g" yes; +createNode mayaUsdLayerManager -n "mayaUsdLayerManager1"; + rename -uid "294C406B-4DA6-ABDE-1077-B3B924B0D3C6"; + setAttr ".sst" -type "string" ""; + setAttr -s 4 ".lyr"; + setAttr ".lyr[0].id" -type "string" "anon:000001A374D17F00:anonymousLayer1-session.usda"; + setAttr ".lyr[0].fid" -type "string" "usda"; + setAttr ".lyr[0].szd" -type "string" ""; + setAttr ".lyr[0].ann" yes; + setAttr ".lyr[1].id" -type "string" "anon:000001A374D193A0:anonymousLayer1"; + setAttr ".lyr[1].fid" -type "string" "sdf"; + setAttr ".lyr[1].szd" -type "string" ( + "#sdf 1.4.32\n\ndef Xform \"sphereAndCubeParent\"\n{\n def Mesh \"sphere\" (\n kind = \"component\"\n )\n {\n uniform bool doubleSided = 1\n float3[] extent = [(-1.0000002, -1, -1.0000005), (1, 1, 1.0000001)]\n int[] faceVertexCounts = [4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3]\n" + + " int[] faceVertexIndices = [0, 1, 21, 20, 1, 2, 22, 21, 2, 3, 23, 22, 3, 4, 24, 23, 4, 5, 25, 24, 5, 6, 26, 25, 6, 7, 27, 26, 7, 8, 28, 27, 8, 9, 29, 28, 9, 10, 30, 29, 10, 11, 31, 30, 11, 12, 32, 31, 12, 13, 33, 32, 13, 14, 34, 33, 14, 15, 35, 34, 15, 16, 36, 35, 16, 17, 37, 36, 17, 18, 38, 37, 18, 19, 39, 38, 19, 0, 20, 39, 20, 21, 41, 40, 21, 22, 42, 41, 22, 23, 43, 42, 23, 24, 44, 43, 24, 25, 45, 44, 25, 26, 46, 45, 26, 27, 47, 46, 27, 28, 48, 47, 28, 29, 49, 48, 29, 30, 50, 49, 30, 31, 51, 50, 31, 32, 52, 51, 32, 33, 53, 52, 33, 34, 54, 53, 34, 35, 55, 54, 35, 36, 56, 55, 36, 37, 57, 56, 37, 38, 58, 57, 38, 39, 59, 58, 39, 20, 40, 59, 40, 41, 61, 60, 41, 42, 62, 61, 42, 43, 63, 62, 43, 44, 64, 63, 44, 45, 65, 64, 45, 46, 66, 65, 46, 47, 67, 66, 47, 48, 68, 67, 48, 49, 69, 68, 49, 50, 70, 69, 50, 51, 71, 70, 51, 52, 72, 71, 52, 53, 73, 72, 53, 54, 74, 73, 54, 55, 75, 74, 55, 56, 76, 75, 56, 57, 77, 76, 57, 58, 78, 77, 58, 59, 79, 78, 59, 40, 60, 79, 60, 61, 81, 80, 61, 62, 82, 81, 62, 63, 83, 82, 63, 64, 84, 83, 64, 65, 85, 84, 65, 66, 86, 85, 66, 67, 87, 86, 67, 68, 88, 87, 68, 69, 89, 88, 69, 70, 90, 89, 70, 71, 91, 90, 71, 72, 92, 91, 72, 73, 93, 92, 73, 74, 94, 93, 74, 75, 95, 94, 75, 76, 96, 95, 76, 77, 97, 96, 77, 78, 98, 97, 78, 79, 99, 98, 79, 60, 80, 99, 80, 81, 101, 100, 81, 82, 102, 101, 82, 83, 103, 102, 83, 84, 104, 103, 84, 85, 105, 104, 85, 86, 106, 105, 86, 87, 107, 106, 87, 88, 108, 107, 88, 89, 109, 108, 89, 90, 110, 109, 90, 91, 111, 110, 91, 92, 112, 111, 92, 93, 113, 112, 93, 94, 114, 113, 94, 95, 115, 114, 95, 96, 116, 115, 96, 97, 117, 116, 97, 98, 118, 117, 98, 99, 119, 118, 99, 80, 100, 119, 100, 101, 121, 120, 101, 102, 122, 121, 102, 103, 123, 122, 103, 104, 124, 123, 104, 105, 125, 124, 105, 106, 126, 125, 106, 107, 127, 126, 107, 108, 128, 127, 108, 109, 129, 128, 109, 110, 130, 129, 110, 111, 131, 130, 111, 112, 132, 131, 112, 113, 133, 132, 113, 114, 134, 133, 114, 115, 135, 134, 115, 116, 136, 135, 116, 117, 137, 136, 117, 118, 138, 137, 118, 119, 139, 138, 119, 100, 120, 139, 120, 121, 141, 140, 121, 122, 142, 141, 122, 123, 143, 142, 123, 124, 144, 143, 124, 125, 145, 144, 125, 126, 146, 145, 126, 127, 147, 146, 127, 128, 148, 147, 128, 129, 149, 148, 129, 130, 150, 149, 130, 131, 151, 150, 131, 132, 152, 151, 132, 133, 153, 152, 133, 134, 154, 153, 134, 135, 155, 154, 135, 136, 156, 155, 136, 137, 157, 156, 137, 138, 158, 157, 138, 139, 159, 158, 139, 120, 140, 159, 140, 141, 161, 160, 141, 142, 162, 161, 142, 143, 163, 162, 143, 144, 164, 163, 144, 145, 165, 164, 145, 146, 166, 165, 146, 147, 167, 166, 147, 148, 168, 167, 148, 149, 169, 168, 149, 150, 170, 169, 150, 151, 171, 170, 151, 152, 172, 171, 152, 153, 173, 172, 153, 154, 174, 173, 154, 155, 175, 174, 155, 156, 176, 175, 156, 157, 177, 176, 157, 158, 178, 177, 158, 159, 179, 178, 159, 140, 160, 179, 160, 161, 181, 180, 161, 162, 182, 181, 162, 163, 183, 182, 163, 164, 184, 183, 164, 165, 185, 184, 165, 166, 186, 185, 166, 167, 187, 186, 167, 168, 188, 187, 168, 169, 189, 188, 169, 170, 190, 189, 170, 171, 191, 190, 171, 172, 192, 191, 172, 173, 193, 192, 173, 174, 194, 193, 174, 175, 195, 194, 175, 176, 196, 195, 176, 177, 197, 196, 177, 178, 198, 197, 178, 179, 199, 198, 179, 160, 180, 199, 180, 181, 201, 200, 181, 182, 202, 201, 182, 183, 203, 202, 183, 184, 204, 203, 184, 185, 205, 204, 185, 186, 206, 205, 186, 187, 207, 206, 187, 188, 208, 207, 188, 189, 209, 208, 189, 190, 210, 209, 190, 191, 211, 210, 191, 192, 212, 211, 192, 193, 213, 212, 193, 194, 214, 213, 194, 195, 215, 214, 195, 196, 216, 215, 196, 197, 217, 216, 197, 198, 218, 217, 198, 199, 219, 218, 199, 180, 200, 219, 200, 201, 221, 220, 201, 202, 222, 221, 202, 203, 223, 222, 203, 204, 224, 223, 204, 205, 225, 224, 205, 206, 226, 225, 206, 207, 227, 226, 207, 208, 228, 227, 208, 209, 229, 228, 209, 210, 230, 229, 210, 211, 231, 230, 211, 212, 232, 231, 212, 213, 233, 232, 213, 214, 234, 233, 214, 215, 235, 234, 215, 216, 236, 235, 216, 217, 237, 236, 217, 218, 238, 237, 218, 219, 239, 238, 219, 200, 220, 239, 220, 221, 241, 240, 221, 222, 242, 241, 222, 223, 243, 242, 223, 224, 244, 243, 224, 225, 245, 244, 225, 226, 246, 245, 226, 227, 247, 246, 227, 228, 248, 247, 228, 229, 249, 248, 229, 230, 250, 249, 230, 231, 251, 250, 231, 232, 252, 251, 232, 233, 253, 252, 233, 234, 254, 253, 234, 235, 255, 254, 235, 236, 256, 255, 236, 237, 257, 256, 237, 238, 258, 257, 238, 239, 259, 258, 239, 220, 240, 259, 240, 241, 261, 260, 241, 242, 262, 261, 242, 243, 263, 262, 243, 244, 264, 263, 244, 245, 265, 264, 245, 246, 266, 265, 246, 247, 267, 266, 247, 248, 268, 267, 248, 249, 269, 268, 249, 250, 270, 269, 250, 251, 271, 270, 251, 252, 272, 271, 252, 253, 273, 272, 253, 254, 274, 273, 254, 255, 275, 274, 255, 256, 276, 275, 256, 257, 277, 276, 257, 258, 278, 277, 258, 259, 279, 278, 259, 240, 260, 279, 260, 261, 281, 280, 261, 262, 282, 281, 262, 263, 283, 282, 263, 264, 284, 283, 264, 265, 285, 284, 265, 266, 286, 285, 266, 267, 287, 286, 267, 268, 288, 287, 268, 269, 289, 288, 269, 270, 290, 289, 270, 271, 291, 290, 271, 272, 292, 291, 272, 273, 293, 292, 273, 274, 294, 293, 274, 275, 295, 294, 275, 276, 296, 295, 276, 277, 297, 296, 277, 278, 298, 297, 278, 279, 299, 298, 279, 260, 280, 299, 280, 281, 301, 300, 281, 282, 302, 301, 282, 283, 303, 302, 283, 284, 304, 303, 284, 285, 305, 304, 285, 286, 306, 305, 286, 287, 307, 306, 287, 288, 308, 307, 288, 289, 309, 308, 289, 290, 310, 309, 290, 291, 311, 310, 291, 292, 312, 311, 292, 293, 313, 312, 293, 294, 314, 313, 294, 295, 315, 314, 295, 296, 316, 315, 296, 297, 317, 316, 297, 298, 318, 317, 298, 299, 319, 318, 299, 280, 300, 319, 300, 301, 321, 320, 301, 302, 322, 321, 302, 303, 323, 322, 303, 304, 324, 323, 304, 305, 325, 324, 305, 306, 326, 325, 306, 307, 327, 326, 307, 308, 328, 327, 308, 309, 329, 328, 309, 310, 330, 329, 310, 311, 331, 330, 311, 312, 332, 331, 312, 313, 333, 332, 313, 314, 334, 333, 314, 315, 335, 334, 315, 316, 336, 335, 316, 317, 337, 336, 317, 318, 338, 337, 318, 319, 339, 338, 319, 300, 320, 339, 320, 321, 341, 340, 321, 322, 342, 341, 322, 323, 343, 342, 323, 324, 344, 343, 324, 325, 345, 344, 325, 326, 346, 345, 326, 327, 347, 346, 327, 328, 348, 347, 328, 329, 349, 348, 329, 330, 350, 349, 330, 331, 351, 350, 331, 332, 352, 351, 332, 333, 353, 352, 333, 334, 354, 353, 334, 335, 355, 354, 335, 336, 356, 355, 336, 337, 357, 356, 337, 338, 358, 357, 338, 339, 359, 358, 339, 320, 340, 359, 340, 341, 361, 360, 341, 342, 362, 361, 342, 343, 363, 362, 343, 344, 364, 363, 344, 345, 365, 364, 345, 346, 366, 365, 346, 347, 367, 366, 347, 348, 368, 367, 348, 349, 369, 368, 349, 350, 370, 369, 350, 351, 371, 370, 351, 352, 372, 371, 352, 353, 373, 372, 353, 354, 374, 373, 354, 355, 375, 374, 355, 356, 376, 375, 356, 357, 377, 376, 357, 358, 378, 377, 358, 359, 379, 378, 359, 340, 360, 379, 1, 0, 380, 2, 1, 380, 3, 2, 380, 4, 3, 380, 5, 4, 380, 6, 5, 380, 7, 6, 380, 8, 7, 380, 9, 8, 380, 10, 9, 380, 11, 10, 380, 12, 11, 380, 13, 12, 380, 14, 13, 380, 15, 14, 380, 16, 15, 380, 17, 16, 380, 18, 17, 380, 19, 18, 380, 0, 19, 380, 360, 361, 381, 361, 362, 381, 362, 363, 381, 363, 364, 381, 364, 365, 381, 365, 366, 381, 366, 367, 381, 367, 368, 381, 368, 369, 381, 369, 370, 381, 370, 371, 381, 371, 372, 381, 372, 373, 381, 373, 374, 381, 374, 375, 381, 375, 376, 381, 376, 377, 381, 377, 378, 381, 378, 379, 381, 379, 360, 381]\n" + + " point3f[] points = [(0.14877813, -0.98768836, -0.048340943), (0.12655823, -0.98768836, -0.09194993), (0.09194993, -0.98768836, -0.12655823), (0.048340935, -0.98768836, -0.14877811), (0, -0.98768836, -0.15643455), (-0.048340935, -0.98768836, -0.1487781), (-0.09194992, -0.98768836, -0.1265582), (-0.12655818, -0.98768836, -0.0919499), (-0.14877807, -0.98768836, -0.048340924), (-0.15643452, -0.98768836, 0), (-0.14877807, -0.98768836, 0.048340924), (-0.12655818, -0.98768836, 0.091949895), (-0.091949895, -0.98768836, 0.12655817), (-0.048340924, -0.98768836, 0.14877805), (-4.6621107e-9, -0.98768836, 0.15643449), (0.04834091, -0.98768836, 0.14877804), (0.09194988, -0.98768836, 0.12655815), (0.12655815, -0.98768836, 0.09194989), (0.14877804, -0.98768836, 0.048340913), (0.15643448, -0.98768836, 0), (0.29389283, -0.95105654, -0.095491566), (0.25000018, -0.95105654, -0.18163574), (0.18163574, -0.95105654, -0.25000015), (0.09549155, -0.95105654, -0.2938928), (0, -0.95105654, -0.30901715), (-0.09549155, -0.95105654, -0.29389277), (-0.18163571, -0.95105654, -0.2500001), (-0.2500001, -0.95105654, -0.1816357), (-0.2938927, -0.95105654, -0.09549153), (-0.30901706, -0.95105654, 0), (-0.2938927, -0.95105654, 0.09549153), (-0.25000006, -0.95105654, 0.18163568), (-0.18163568, -0.95105654, 0.25000006), (-0.09549153, -0.95105654, 0.29389268), (-9.209424e-9, -0.95105654, 0.30901703), (0.0954915, -0.95105654, 0.29389265), (0.18163563, -0.95105654, 0.25000003), (0.25, -0.95105654, 0.18163565), (0.29389265, -0.95105654, 0.095491506), (0.309017, -0.95105654, 0), (0.43177092, -0.8910065, -0.14029087), (0.3672863, -0.8910065, -0.2668491), (0.2668491, -0.8910065, -0.36728626), (0.14029086, -0.8910065, -0.43177086), (0, -0.8910065, -0.45399073), (-0.14029086, -0.8910065, -0.43177083), (-0.26684904, -0.8910065, -0.36728618), (-0.36728615, -0.8910065, -0.266849), (-0.43177077, -0.8910065, -0.14029081), (-0.45399064, -0.8910065, 0), (-0.43177077, -0.8910065, 0.14029081), (-0.36728612, -0.8910065, 0.26684898), (-0.26684898, -0.8910065, 0.36728612), (-0.14029081, -0.8910065, 0.4317707), (-1.3529972e-8, -0.8910065, 0.45399058), (0.14029078, -0.8910065, 0.43177068), (0.26684892, -0.8910065, 0.3672861), (0.36728606, -0.8910065, 0.26684895), (0.43177065, -0.8910065, 0.1402908), (0.45399052, -0.8910065, 0), (0.55901736, -0.809017, -0.18163574), (0.47552857, -0.809017, -0.3454917), (0.3454917, -0.809017, -0.47552854), (0.18163572, -0.809017, -0.5590173), (0, -0.809017, -0.58778554), (-0.18163572, -0.809017, -0.55901724), (-0.34549165, -0.809017, -0.47552842), (-0.4755284, -0.809017, -0.3454916), (-0.5590171, -0.809017, -0.18163566), (-0.58778536, -0.809017, 0), (-0.5590171, -0.809017, 0.18163566), (-0.47552836, -0.809017, 0.34549156), (-0.34549156, -0.809017, 0.47552833), (-0.18163566, -0.809017, 0.55901706), (-1.7517365e-8, -0.809017, 0.5877853), (0.18163562, -0.809017, 0.55901706), (0.3454915, -0.809017, 0.4755283), (0.47552827, -0.809017, 0.34549153), (0.559017, -0.809017, 0.18163563), (0.58778524, -0.809017, 0), (0.67249894, -0.70710677, -0.21850814), (0.5720618, -0.70710677, -0.41562718), (0.41562718, -0.70710677, -0.5720617), (0.21850812, -0.70710677, -0.6724989), (0, -0.70710677, -0.7071071), (-0.21850812, -0.70710677, -0.6724988), (-0.4156271, -0.70710677, -0.5720616), (-0.57206154, -0.70710677, -0.41562706), (-0.6724987, -0.70710677, -0.21850805), (-0.70710695, -0.70710677, 0), (-0.6724987, -0.70710677, 0.21850805), (-0.57206154, -0.70710677, 0.415627), (-0.415627, -0.70710677, 0.5720615), (-0.21850805, -0.70710677, 0.6724986), (-2.1073424e-8, -0.70710677, 0.7071068), (0.21850799, -0.70710677, 0.6724986), (0.4156269, -0.70710677, 0.5720614), (0.5720614, -0.70710677, 0.41562697), (0.6724985, -0.70710677, 0.21850802), (0.70710677, -0.70710677, 0), (0.7694214, -0.58778524, -0.25000015), (0.65450895, -0.58778524, -0.47552854), (0.47552854, -0.58778524, -0.6545089), (0.25000012, -0.58778524, -0.7694213), (0, -0.58778524, -0.80901736), (-0.25000012, -0.58778524, -0.7694212), (-0.47552845, -0.58778524, -0.65450877), (-0.6545087, -0.58778524, -0.4755284), (-0.7694211, -0.58778524, -0.25000006), (-0.8090172, -0.58778524, 0), (-0.7694211, -0.58778524, 0.25000006), (-0.65450865, -0.58778524, 0.47552836), (-0.47552836, -0.58778524, 0.6545086), (-0.25000006, -0.58778524, 0.769421), (-2.4110586e-8, -0.58778524, 0.8090171), (0.24999999, -0.58778524, 0.769421), (0.47552827, -0.58778524, 0.65450853), (0.65450853, -0.58778524, 0.4755283), (0.7694209, -0.58778524, 0.25), (0.809017, -0.58778524, 0), (0.8473981, -0.45399052, -0.27533633), (0.7208399, -0.45399052, -0.5237208), (0.5237208, -0.45399052, -0.72083986), (0.2753363, -0.45399052, -0.847398), (0, -0.45399052, -0.89100695), (-0.2753363, -0.45399052, -0.847398), (-0.5237207, -0.45399052, -0.7208397), (-0.7208396, -0.45399052, -0.5237206), (-0.8473978, -0.45399052, -0.2753362), (-0.89100677, -0.45399052, 0), (-0.8473978, -0.45399052, 0.2753362), (-0.7208396, -0.45399052, 0.5237206), (-0.5237206, -0.45399052, 0.72083956), (-0.2753362, -0.45399052, 0.8473977), (-2.6554064e-8, -0.45399052, 0.89100665), (0.27533615, -0.45399052, 0.8473976), (0.5237205, -0.45399052, 0.7208395), (0.72083944, -0.45399052, 0.52372056), (0.84739757, -0.45399052, 0.27533618), (0.8910065, -0.45399052, 0), (0.9045091, -0.30901697, -0.2938928), (0.7694214, -0.30901697, -0.55901736), (0.55901736, -0.30901697, -0.76942134), (0.29389277, -0.30901697, -0.904509), (0, -0.30901697, -0.951057), (-0.29389277, -0.30901697, -0.90450895), (-0.55901724, -0.30901697, -0.7694212), (-0.76942116, -0.30901697, -0.5590172), (-0.90450877, -0.30901697, -0.2938927), (-0.9510568, -0.30901697, 0), (-0.90450877, -0.30901697, 0.2938927), (-0.7694211, -0.30901697, 0.5590171), (-0.5590171, -0.30901697, 0.76942104), (-0.2938927, -0.30901697, 0.90450865), (-2.8343694e-8, -0.30901697, 0.95105666), (0.29389262, -0.30901697, 0.9045086), (0.559017, -0.30901697, 0.769421), (0.7694209, -0.30901697, 0.55901706), (0.90450853, -0.30901697, 0.29389265), (0.95105654, -0.30901697, 0), (0.93934804, -0.15643437, -0.30521268), (0.7990572, -0.15643437, -0.580549), (0.580549, -0.15643437, -0.7990571), (0.30521265, -0.15643437, -0.9393479), (0, -0.15643437, -0.98768884), (-0.30521265, -0.15643437, -0.93934786), (-0.5805489, -0.15643437, -0.79905695), (-0.7990569, -0.15643437, -0.5805488), (-0.9393477, -0.15643437, -0.30521256), (-0.9876886, -0.15643437, 0), (-0.9393477, -0.15643437, 0.30521256), (-0.7990568, -0.15643437, 0.58054876), (-0.58054876, -0.15643437, 0.79905677), (-0.30521256, -0.15643437, 0.93934757), (-2.9435407e-8, -0.15643437, 0.9876885), (0.30521247, -0.15643437, 0.93934757), (0.58054864, -0.15643437, 0.7990567), (0.79905665, -0.15643437, 0.5805487), (0.9393475, -0.15643437, 0.3052125), (0.98768836, -0.15643437, 0), (0.95105714, 0, -0.30901718), (0.80901754, 0, -0.5877856), (0.5877856, 0, -0.8090175), (0.30901715, 0, -0.951057), (0, 0, -1.0000005), (-0.30901715, 0, -0.95105696), (-0.5877855, 0, -0.8090173), (-0.80901724, 0, -0.5877854), (-0.9510568, 0, -0.30901706), (-1.0000002, 0, 0), (-0.9510568, 0, 0.30901706), (-0.8090172, 0, 0.58778536), (-0.58778536, 0, 0.8090171), (-0.30901706, 0, 0.95105666), (-2.9802322e-8, 0, 1.0000001), (0.30901697, 0, 0.9510566), (0.58778524, 0, 0.80901706), (0.809017, 0, 0.5877853), (0.95105654, 0, 0.309017), (1, 0, 0), (0.93934804, 0.15643437, -0.30521268), (0.7990572, 0.15643437, -0.580549), (0.580549, 0.15643437, -0.7990571), (0.30521265, 0.15643437, -0.9393479), (0, 0.15643437, -0.98768884), (-0.30521265, 0.15643437, -0.93934786), (-0.5805489, 0.15643437, -0.79905695), (-0.7990569, 0.15643437, -0.5805488), (-0.9393477, 0.15643437, -0.30521256), (-0.9876886, 0.15643437, 0), (-0.9393477, 0.15643437, 0.30521256), (-0.7990568, 0.15643437, 0.58054876), (-0.58054876, 0.15643437, 0.79905677), (-0.30521256, 0.15643437, 0.93934757), (-2.9435407e-8, 0.15643437, 0.9876885), (0.30521247, 0.15643437, 0.93934757), (0.58054864, 0.15643437, 0.7990567), (0.79905665, 0.15643437, 0.5805487), (0.9393475, 0.15643437, 0.3052125), (0.98768836, 0.15643437, 0), (0.9045091, 0.30901697, -0.2938928), (0.7694214, 0.30901697, -0.55901736), (0.55901736, 0.30901697, -0.76942134), (0.29389277, 0.30901697, -0.904509), (0, 0.30901697, -0.951057), (-0.29389277, 0.30901697, -0.90450895), (-0.55901724, 0.30901697, -0.7694212), (-0.76942116, 0.30901697, -0.5590172), (-0.90450877, 0.30901697, -0.2938927), (-0.9510568, 0.30901697, 0), (-0.90450877, 0.30901697, 0.2938927), (-0.7694211, 0.30901697, 0.5590171), (-0.5590171, 0.30901697, 0.76942104), (-0.2938927, 0.30901697, 0.90450865), (-2.8343694e-8, 0.30901697, 0.95105666), (0.29389262, 0.30901697, 0.9045086), (0.559017, 0.30901697, 0.769421), (0.7694209, 0.30901697, 0.55901706), (0.90450853, 0.30901697, 0.29389265), (0.95105654, 0.30901697, 0), (0.8473981, 0.45399052, -0.27533633), (0.7208399, 0.45399052, -0.5237208), (0.5237208, 0.45399052, -0.72083986), (0.2753363, 0.45399052, -0.847398), (0, 0.45399052, -0.89100695), (-0.2753363, 0.45399052, -0.847398), (-0.5237207, 0.45399052, -0.7208397), (-0.7208396, 0.45399052, -0.5237206), (-0.8473978, 0.45399052, -0.2753362), (-0.89100677, 0.45399052, 0), (-0.8473978, 0.45399052, 0.2753362), (-0.7208396, 0.45399052, 0.5237206), (-0.5237206, 0.45399052, 0.72083956), (-0.2753362, 0.45399052, 0.8473977), (-2.6554064e-8, 0.45399052, 0.89100665), (0.27533615, 0.45399052, 0.8473976), (0.5237205, 0.45399052, 0.7208395), (0.72083944, 0.45399052, 0.52372056), (0.84739757, 0.45399052, 0.27533618), (0.8910065, 0.45399052, 0), (0.7694214, 0.58778524, -0.25000015), (0.65450895, 0.58778524, -0.47552854), (0.47552854, 0.58778524, -0.6545089), (0.25000012, 0.58778524, -0.7694213), (0, 0.58778524, -0.80901736), (-0.25000012, 0.58778524, -0.7694212), (-0.47552845, 0.58778524, -0.65450877), (-0.6545087, 0.58778524, -0.4755284), (-0.7694211, 0.58778524, -0.25000006), (-0.8090172, 0.58778524, 0), (-0.7694211, 0.58778524, 0.25000006), (-0.65450865, 0.58778524, 0.47552836), (-0.47552836, 0.58778524, 0.6545086), (-0.25000006, 0.58778524, 0.769421), (-2.4110586e-8, 0.58778524, 0.8090171), (0.24999999, 0.58778524, 0.769421), (0.47552827, 0.58778524, 0.65450853), (0.65450853, 0.58778524, 0.4755283), (0.7694209, 0.58778524, 0.25), (0.809017, 0.58778524, 0), (0.67249894, 0.70710677, -0.21850814), (0.5720618, 0.70710677, -0.41562718), (0.41562718, 0.70710677, -0.5720617), (0.21850812, 0.70710677, -0.6724989), (0, 0.70710677, -0.7071071), (-0.21850812, 0.70710677, -0.6724988), (-0.4156271, 0.70710677, -0.5720616), (-0.57206154, 0.70710677, -0.41562706), (-0.6724987, 0.70710677, -0.21850805), (-0.70710695, 0.70710677, 0), (-0.6724987, 0.70710677, 0.21850805), (-0.57206154, 0.70710677, 0.415627), (-0.415627, 0.70710677, 0.5720615), (-0.21850805, 0.70710677, 0.6724986), (-2.1073424e-8, 0.70710677, 0.7071068), (0.21850799, 0.70710677, 0.6724986), (0.4156269, 0.70710677, 0.5720614), (0.5720614, 0.70710677, 0.41562697), (0.6724985, 0.70710677, 0.21850802), (0.70710677, 0.70710677, 0), (0.55901736, 0.809017, -0.18163574), (0.47552857, 0.809017, -0.3454917), (0.3454917, 0.809017, -0.47552854), (0.18163572, 0.809017, -0.5590173), (0, 0.809017, -0.58778554), (-0.18163572, 0.809017, -0.55901724), (-0.34549165, 0.809017, -0.47552842), (-0.4755284, 0.809017, -0.3454916), (-0.5590171, 0.809017, -0.18163566), (-0.58778536, 0.809017, 0), (-0.5590171, 0.809017, 0.18163566), (-0.47552836, 0.809017, 0.34549156), (-0.34549156, 0.809017, 0.47552833), (-0.18163566, 0.809017, 0.55901706), (-1.7517365e-8, 0.809017, 0.5877853), (0.18163562, 0.809017, 0.55901706), (0.3454915, 0.809017, 0.4755283), (0.47552827, 0.809017, 0.34549153), (0.559017, 0.809017, 0.18163563), (0.58778524, 0.809017, 0), (0.43177092, 0.8910065, -0.14029087), (0.3672863, 0.8910065, -0.2668491), (0.2668491, 0.8910065, -0.36728626), (0.14029086, 0.8910065, -0.43177086), (0, 0.8910065, -0.45399073), (-0.14029086, 0.8910065, -0.43177083), (-0.26684904, 0.8910065, -0.36728618), (-0.36728615, 0.8910065, -0.266849), (-0.43177077, 0.8910065, -0.14029081), (-0.45399064, 0.8910065, 0), (-0.43177077, 0.8910065, 0.14029081), (-0.36728612, 0.8910065, 0.26684898), (-0.26684898, 0.8910065, 0.36728612), (-0.14029081, 0.8910065, 0.4317707), (-1.3529972e-8, 0.8910065, 0.45399058), (0.14029078, 0.8910065, 0.43177068), (0.26684892, 0.8910065, 0.3672861), (0.36728606, 0.8910065, 0.26684895), (0.43177065, 0.8910065, 0.1402908), (0.45399052, 0.8910065, 0), (0.29389283, 0.95105654, -0.095491566), (0.25000018, 0.95105654, -0.18163574), (0.18163574, 0.95105654, -0.25000015), (0.09549155, 0.95105654, -0.2938928), (0, 0.95105654, -0.30901715), (-0.09549155, 0.95105654, -0.29389277), (-0.18163571, 0.95105654, -0.2500001), (-0.2500001, 0.95105654, -0.1816357), (-0.2938927, 0.95105654, -0.09549153), (-0.30901706, 0.95105654, 0), (-0.2938927, 0.95105654, 0.09549153), (-0.25000006, 0.95105654, 0.18163568), (-0.18163568, 0.95105654, 0.25000006), (-0.09549153, 0.95105654, 0.29389268), (-9.209424e-9, 0.95105654, 0.30901703), (0.0954915, 0.95105654, 0.29389265), (0.18163563, 0.95105654, 0.25000003), (0.25, 0.95105654, 0.18163565), (0.29389265, 0.95105654, 0.095491506), (0.309017, 0.95105654, 0), (0.14877813, 0.98768836, -0.048340943), (0.12655823, 0.98768836, -0.09194993), (0.09194993, 0.98768836, -0.12655823), (0.048340935, 0.98768836, -0.14877811), (0, 0.98768836, -0.15643455), (-0.048340935, 0.98768836, -0.1487781), (-0.09194992, 0.98768836, -0.1265582), (-0.12655818, 0.98768836, -0.0919499), (-0.14877807, 0.98768836, -0.048340924), (-0.15643452, 0.98768836, 0), (-0.14877807, 0.98768836, 0.048340924), (-0.12655818, 0.98768836, 0.091949895), (-0.091949895, 0.98768836, 0.12655817), (-0.048340924, 0.98768836, 0.14877805), (-4.6621107e-9, 0.98768836, 0.15643449), (0.04834091, 0.98768836, 0.14877804), (0.09194988, 0.98768836, 0.12655815), (0.12655815, 0.98768836, 0.09194989), (0.14877804, 0.98768836, 0.048340913), (0.15643448, 0.98768836, 0), (0, -1, 0), (0, 1, 0)]\n" + + " color3f[] primvars:displayColor = [(0.13320851, 0.13320851, 0.13320851)] (\n customData = {\n dictionary Maya = {\n bool generated = 1\n }\n }\n )\n" + + " texCoord2f[] primvars:st = [(0, 0.05), (0.05, 0.05), (0.1, 0.05), (0.15, 0.05), (0.2, 0.05), (0.25, 0.05), (0.3, 0.05), (0.35000002, 0.05), (0.40000004, 0.05), (0.45000005, 0.05), (0.50000006, 0.05), (0.5500001, 0.05), (0.6000001, 0.05), (0.6500001, 0.05), (0.7000001, 0.05), (0.7500001, 0.05), (0.80000013, 0.05), (0.85000014, 0.05), (0.90000015, 0.05), (0.95000017, 0.05), (1.0000001, 0.05), (0, 0.1), (0.05, 0.1), (0.1, 0.1), (0.15, 0.1), (0.2, 0.1), (0.25, 0.1), (0.3, 0.1), (0.35000002, 0.1), (0.40000004, 0.1), (0.45000005, 0.1), (0.50000006, 0.1), (0.5500001, 0.1), (0.6000001, 0.1), (0.6500001, 0.1), (0.7000001, 0.1), (0.7500001, 0.1), (0.80000013, 0.1), (0.85000014, 0.1), (0.90000015, 0.1), (0.95000017, 0.1), (1.0000001, 0.1), (0, 0.15), (0.05, 0.15), (0.1, 0.15), (0.15, 0.15), (0.2, 0.15), (0.25, 0.15), (0.3, 0.15), (0.35000002, 0.15), (0.40000004, 0.15), (0.45000005, 0.15), (0.50000006, 0.15), (0.5500001, 0.15), (0.6000001, 0.15), (0.6500001, 0.15), (0.7000001, 0.15), (0.7500001, 0.15), (0.80000013, 0.15), (0.85000014, 0.15), (0.90000015, 0.15), (0.95000017, 0.15), (1.0000001, 0.15), (0, 0.2), (0.05, 0.2), (0.1, 0.2), (0.15, 0.2), (0.2, 0.2), (0.25, 0.2), (0.3, 0.2), (0.35000002, 0.2), (0.40000004, 0.2), (0.45000005, 0.2), (0.50000006, 0.2), (0.5500001, 0.2), (0.6000001, 0.2), (0.6500001, 0.2), (0.7000001, 0.2), (0.7500001, 0.2), (0.80000013, 0.2), (0.85000014, 0.2), (0.90000015, 0.2), (0.95000017, 0.2), (1.0000001, 0.2), (0, 0.25), (0.05, 0.25), (0.1, 0.25), (0.15, 0.25), (0.2, 0.25), (0.25, 0.25), (0.3, 0.25), (0.35000002, 0.25), (0.40000004, 0.25), (0.45000005, 0.25), (0.50000006, 0.25), (0.5500001, 0.25), (0.6000001, 0.25), (0.6500001, 0.25), (0.7000001, 0.25), (0.7500001, 0.25), (0.80000013, 0.25), (0.85000014, 0.25), (0.90000015, 0.25), (0.95000017, 0.25), (1.0000001, 0.25), (0, 0.3), (0.05, 0.3), (0.1, 0.3), (0.15, 0.3), (0.2, 0.3), (0.25, 0.3), (0.3, 0.3), (0.35000002, 0.3), (0.40000004, 0.3), (0.45000005, 0.3), (0.50000006, 0.3), (0.5500001, 0.3), (0.6000001, 0.3), (0.6500001, 0.3), (0.7000001, 0.3), (0.7500001, 0.3), (0.80000013, 0.3), (0.85000014, 0.3), (0.90000015, 0.3), (0.95000017, 0.3), (1.0000001, 0.3), (0, 0.35000002), (0.05, 0.35000002), (0.1, 0.35000002), (0.15, 0.35000002), (0.2, 0.35000002), (0.25, 0.35000002), (0.3, 0.35000002), (0.35000002, 0.35000002), (0.40000004, 0.35000002), (0.45000005, 0.35000002), (0.50000006, 0.35000002), (0.5500001, 0.35000002), (0.6000001, 0.35000002), (0.6500001, 0.35000002), (0.7000001, 0.35000002), (0.7500001, 0.35000002), (0.80000013, 0.35000002), (0.85000014, 0.35000002), (0.90000015, 0.35000002), (0.95000017, 0.35000002), (1.0000001, 0.35000002), (0, 0.40000004), (0.05, 0.40000004), (0.1, 0.40000004), (0.15, 0.40000004), (0.2, 0.40000004), (0.25, 0.40000004), (0.3, 0.40000004), (0.35000002, 0.40000004), (0.40000004, 0.40000004), (0.45000005, 0.40000004), (0.50000006, 0.40000004), (0.5500001, 0.40000004), (0.6000001, 0.40000004), (0.6500001, 0.40000004), (0.7000001, 0.40000004), (0.7500001, 0.40000004), (0.80000013, 0.40000004), (0.85000014, 0.40000004), (0.90000015, 0.40000004), (0.95000017, 0.40000004), (1.0000001, 0.40000004), (0, 0.45000005), (0.05, 0.45000005), (0.1, 0.45000005), (0.15, 0.45000005), (0.2, 0.45000005), (0.25, 0.45000005), (0.3, 0.45000005), (0.35000002, 0.45000005), (0.40000004, 0.45000005), (0.45000005, 0.45000005), (0.50000006, 0.45000005), (0.5500001, 0.45000005), (0.6000001, 0.45000005), (0.6500001, 0.45000005), (0.7000001, 0.45000005), (0.7500001, 0.45000005), (0.80000013, 0.45000005), (0.85000014, 0.45000005), (0.90000015, 0.45000005), (0.95000017, 0.45000005), (1.0000001, 0.45000005), (0, 0.50000006), (0.05, 0.50000006), (0.1, 0.50000006), (0.15, 0.50000006), (0.2, 0.50000006), (0.25, 0.50000006), (0.3, 0.50000006), (0.35000002, 0.50000006), (0.40000004, 0.50000006), (0.45000005, 0.50000006), (0.50000006, 0.50000006), (0.5500001, 0.50000006), (0.6000001, 0.50000006), (0.6500001, 0.50000006), (0.7000001, 0.50000006), (0.7500001, 0.50000006), (0.80000013, 0.50000006), (0.85000014, 0.50000006), (0.90000015, 0.50000006), (0.95000017, 0.50000006), (1.0000001, 0.50000006), (0, 0.5500001), (0.05, 0.5500001), (0.1, 0.5500001), (0.15, 0.5500001), (0.2, 0.5500001), (0.25, 0.5500001), (0.3, 0.5500001), (0.35000002, 0.5500001), (0.40000004, 0.5500001), (0.45000005, 0.5500001), (0.50000006, 0.5500001), (0.5500001, 0.5500001), (0.6000001, 0.5500001), (0.6500001, 0.5500001), (0.7000001, 0.5500001), (0.7500001, 0.5500001), (0.80000013, 0.5500001), (0.85000014, 0.5500001), (0.90000015, 0.5500001), (0.95000017, 0.5500001), (1.0000001, 0.5500001), (0, 0.6000001), (0.05, 0.6000001), (0.1, 0.6000001), (0.15, 0.6000001), (0.2, 0.6000001), (0.25, 0.6000001), (0.3, 0.6000001), (0.35000002, 0.6000001), (0.40000004, 0.6000001), (0.45000005, 0.6000001), (0.50000006, 0.6000001), (0.5500001, 0.6000001), (0.6000001, 0.6000001), (0.6500001, 0.6000001), (0.7000001, 0.6000001), (0.7500001, 0.6000001), (0.80000013, 0.6000001), (0.85000014, 0.6000001), (0.90000015, 0.6000001), (0.95000017, 0.6000001), (1.0000001, 0.6000001), (0, 0.6500001), (0.05, 0.6500001), (0.1, 0.6500001), (0.15, 0.6500001), (0.2, 0.6500001), (0.25, 0.6500001), (0.3, 0.6500001), (0.35000002, 0.6500001), (0.40000004, 0.6500001), (0.45000005, 0.6500001), (0.50000006, 0.6500001), (0.5500001, 0.6500001), (0.6000001, 0.6500001), (0.6500001, 0.6500001), (0.7000001, 0.6500001), (0.7500001, 0.6500001), (0.80000013, 0.6500001), (0.85000014, 0.6500001), (0.90000015, 0.6500001), (0.95000017, 0.6500001), (1.0000001, 0.6500001), (0, 0.7000001), (0.05, 0.7000001), (0.1, 0.7000001), (0.15, 0.7000001), (0.2, 0.7000001), (0.25, 0.7000001), (0.3, 0.7000001), (0.35000002, 0.7000001), (0.40000004, 0.7000001), (0.45000005, 0.7000001), (0.50000006, 0.7000001), (0.5500001, 0.7000001), (0.6000001, 0.7000001), (0.6500001, 0.7000001), (0.7000001, 0.7000001), (0.7500001, 0.7000001), (0.80000013, 0.7000001), (0.85000014, 0.7000001), (0.90000015, 0.7000001), (0.95000017, 0.7000001), (1.0000001, 0.7000001), (0, 0.7500001), (0.05, 0.7500001), (0.1, 0.7500001), (0.15, 0.7500001), (0.2, 0.7500001), (0.25, 0.7500001), (0.3, 0.7500001), (0.35000002, 0.7500001), (0.40000004, 0.7500001), (0.45000005, 0.7500001), (0.50000006, 0.7500001), (0.5500001, 0.7500001), (0.6000001, 0.7500001), (0.6500001, 0.7500001), (0.7000001, 0.7500001), (0.7500001, 0.7500001), (0.80000013, 0.7500001), (0.85000014, 0.7500001), (0.90000015, 0.7500001), (0.95000017, 0.7500001), (1.0000001, 0.7500001), (0, 0.80000013), (0.05, 0.80000013), (0.1, 0.80000013), (0.15, 0.80000013), (0.2, 0.80000013), (0.25, 0.80000013), (0.3, 0.80000013), (0.35000002, 0.80000013), (0.40000004, 0.80000013), (0.45000005, 0.80000013), (0.50000006, 0.80000013), (0.5500001, 0.80000013), (0.6000001, 0.80000013), (0.6500001, 0.80000013), (0.7000001, 0.80000013), (0.7500001, 0.80000013), (0.80000013, 0.80000013), (0.85000014, 0.80000013), (0.90000015, 0.80000013), (0.95000017, 0.80000013), (1.0000001, 0.80000013), (0, 0.85000014), (0.05, 0.85000014), (0.1, 0.85000014), (0.15, 0.85000014), (0.2, 0.85000014), (0.25, 0.85000014), (0.3, 0.85000014), (0.35000002, 0.85000014), (0.40000004, 0.85000014), (0.45000005, 0.85000014), (0.50000006, 0.85000014), (0.5500001, 0.85000014), (0.6000001, 0.85000014), (0.6500001, 0.85000014), (0.7000001, 0.85000014), (0.7500001, 0.85000014), (0.80000013, 0.85000014), (0.85000014, 0.85000014), (0.90000015, 0.85000014), (0.95000017, 0.85000014), (1.0000001, 0.85000014), (0, 0.90000015), (0.05, 0.90000015), (0.1, 0.90000015), (0.15, 0.90000015), (0.2, 0.90000015), (0.25, 0.90000015), (0.3, 0.90000015), (0.35000002, 0.90000015), (0.40000004, 0.90000015), (0.45000005, 0.90000015), (0.50000006, 0.90000015), (0.5500001, 0.90000015), (0.6000001, 0.90000015), (0.6500001, 0.90000015), (0.7000001, 0.90000015), (0.7500001, 0.90000015), (0.80000013, 0.90000015), (0.85000014, 0.90000015), (0.90000015, 0.90000015), (0.95000017, 0.90000015), (1.0000001, 0.90000015), (0, 0.95000017), (0.05, 0.95000017), (0.1, 0.95000017), (0.15, 0.95000017), (0.2, 0.95000017), (0.25, 0.95000017), (0.3, 0.95000017), (0.35000002, 0.95000017), (0.40000004, 0.95000017), (0.45000005, 0.95000017), (0.50000006, 0.95000017), (0.5500001, 0.95000017), (0.6000001, 0.95000017), (0.6500001, 0.95000017), (0.7000001, 0.95000017), (0.7500001, 0.95000017), (0.80000013, 0.95000017), (0.85000014, 0.95000017), (0.90000015, 0.95000017), (0.95000017, 0.95000017), (1.0000001, 0.95000017), (0.025, 0), (0.075, 0), (0.125, 0), (0.17500001, 0), (0.22500001, 0), (0.275, 0), (0.32500002, 0), (0.375, 0), (0.425, 0), (0.47500002, 0), (0.525, 0), (0.575, 0), (0.625, 0), (0.675, 0), (0.72499996, 0), (0.775, 0), (0.825, 0), (0.875, 0), (0.925, 0), (0.97499996, 0), (0.025, 1), (0.075, 1), (0.125, 1), (0.17500001, 1), (0.22500001, 1), (0.275, 1), (0.32500002, 1), (0.375, 1), (0.425, 1), (0.47500002, 1), (0.525, 1), (0.575, 1), (0.625, 1), (0.675, 1), (0.72499996, 1), (0.775, 1), (0.825, 1), (0.875, 1), (0.925, 1), (0.97499996, 1)] (\n" + + " customData = {\n dictionary Maya = {\n token name = \"map1\"\n }\n }\n interpolation = \"faceVarying\"\n )\n" + + " int[] primvars:st:indices = [0, 1, 22, 21, 1, 2, 23, 22, 2, 3, 24, 23, 3, 4, 25, 24, 4, 5, 26, 25, 5, 6, 27, 26, 6, 7, 28, 27, 7, 8, 29, 28, 8, 9, 30, 29, 9, 10, 31, 30, 10, 11, 32, 31, 11, 12, 33, 32, 12, 13, 34, 33, 13, 14, 35, 34, 14, 15, 36, 35, 15, 16, 37, 36, 16, 17, 38, 37, 17, 18, 39, 38, 18, 19, 40, 39, 19, 20, 41, 40, 21, 22, 43, 42, 22, 23, 44, 43, 23, 24, 45, 44, 24, 25, 46, 45, 25, 26, 47, 46, 26, 27, 48, 47, 27, 28, 49, 48, 28, 29, 50, 49, 29, 30, 51, 50, 30, 31, 52, 51, 31, 32, 53, 52, 32, 33, 54, 53, 33, 34, 55, 54, 34, 35, 56, 55, 35, 36, 57, 56, 36, 37, 58, 57, 37, 38, 59, 58, 38, 39, 60, 59, 39, 40, 61, 60, 40, 41, 62, 61, 42, 43, 64, 63, 43, 44, 65, 64, 44, 45, 66, 65, 45, 46, 67, 66, 46, 47, 68, 67, 47, 48, 69, 68, 48, 49, 70, 69, 49, 50, 71, 70, 50, 51, 72, 71, 51, 52, 73, 72, 52, 53, 74, 73, 53, 54, 75, 74, 54, 55, 76, 75, 55, 56, 77, 76, 56, 57, 78, 77, 57, 58, 79, 78, 58, 59, 80, 79, 59, 60, 81, 80, 60, 61, 82, 81, 61, 62, 83, 82, 63, 64, 85, 84, 64, 65, 86, 85, 65, 66, 87, 86, 66, 67, 88, 87, 67, 68, 89, 88, 68, 69, 90, 89, 69, 70, 91, 90, 70, 71, 92, 91, 71, 72, 93, 92, 72, 73, 94, 93, 73, 74, 95, 94, 74, 75, 96, 95, 75, 76, 97, 96, 76, 77, 98, 97, 77, 78, 99, 98, 78, 79, 100, 99, 79, 80, 101, 100, 80, 81, 102, 101, 81, 82, 103, 102, 82, 83, 104, 103, 84, 85, 106, 105, 85, 86, 107, 106, 86, 87, 108, 107, 87, 88, 109, 108, 88, 89, 110, 109, 89, 90, 111, 110, 90, 91, 112, 111, 91, 92, 113, 112, 92, 93, 114, 113, 93, 94, 115, 114, 94, 95, 116, 115, 95, 96, 117, 116, 96, 97, 118, 117, 97, 98, 119, 118, 98, 99, 120, 119, 99, 100, 121, 120, 100, 101, 122, 121, 101, 102, 123, 122, 102, 103, 124, 123, 103, 104, 125, 124, 105, 106, 127, 126, 106, 107, 128, 127, 107, 108, 129, 128, 108, 109, 130, 129, 109, 110, 131, 130, 110, 111, 132, 131, 111, 112, 133, 132, 112, 113, 134, 133, 113, 114, 135, 134, 114, 115, 136, 135, 115, 116, 137, 136, 116, 117, 138, 137, 117, 118, 139, 138, 118, 119, 140, 139, 119, 120, 141, 140, 120, 121, 142, 141, 121, 122, 143, 142, 122, 123, 144, 143, 123, 124, 145, 144, 124, 125, 146, 145, 126, 127, 148, 147, 127, 128, 149, 148, 128, 129, 150, 149, 129, 130, 151, 150, 130, 131, 152, 151, 131, 132, 153, 152, 132, 133, 154, 153, 133, 134, 155, 154, 134, 135, 156, 155, 135, 136, 157, 156, 136, 137, 158, 157, 137, 138, 159, 158, 138, 139, 160, 159, 139, 140, 161, 160, 140, 141, 162, 161, 141, 142, 163, 162, 142, 143, 164, 163, 143, 144, 165, 164, 144, 145, 166, 165, 145, 146, 167, 166, 147, 148, 169, 168, 148, 149, 170, 169, 149, 150, 171, 170, 150, 151, 172, 171, 151, 152, 173, 172, 152, 153, 174, 173, 153, 154, 175, 174, 154, 155, 176, 175, 155, 156, 177, 176, 156, 157, 178, 177, 157, 158, 179, 178, 158, 159, 180, 179, 159, 160, 181, 180, 160, 161, 182, 181, 161, 162, 183, 182, 162, 163, 184, 183, 163, 164, 185, 184, 164, 165, 186, 185, 165, 166, 187, 186, 166, 167, 188, 187, 168, 169, 190, 189, 169, 170, 191, 190, 170, 171, 192, 191, 171, 172, 193, 192, 172, 173, 194, 193, 173, 174, 195, 194, 174, 175, 196, 195, 175, 176, 197, 196, 176, 177, 198, 197, 177, 178, 199, 198, 178, 179, 200, 199, 179, 180, 201, 200, 180, 181, 202, 201, 181, 182, 203, 202, 182, 183, 204, 203, 183, 184, 205, 204, 184, 185, 206, 205, 185, 186, 207, 206, 186, 187, 208, 207, 187, 188, 209, 208, 189, 190, 211, 210, 190, 191, 212, 211, 191, 192, 213, 212, 192, 193, 214, 213, 193, 194, 215, 214, 194, 195, 216, 215, 195, 196, 217, 216, 196, 197, 218, 217, 197, 198, 219, 218, 198, 199, 220, 219, 199, 200, 221, 220, 200, 201, 222, 221, 201, 202, 223, 222, 202, 203, 224, 223, 203, 204, 225, 224, 204, 205, 226, 225, 205, 206, 227, 226, 206, 207, 228, 227, 207, 208, 229, 228, 208, 209, 230, 229, 210, 211, 232, 231, 211, 212, 233, 232, 212, 213, 234, 233, 213, 214, 235, 234, 214, 215, 236, 235, 215, 216, 237, 236, 216, 217, 238, 237, 217, 218, 239, 238, 218, 219, 240, 239, 219, 220, 241, 240, 220, 221, 242, 241, 221, 222, 243, 242, 222, 223, 244, 243, 223, 224, 245, 244, 224, 225, 246, 245, 225, 226, 247, 246, 226, 227, 248, 247, 227, 228, 249, 248, 228, 229, 250, 249, 229, 230, 251, 250, 231, 232, 253, 252, 232, 233, 254, 253, 233, 234, 255, 254, 234, 235, 256, 255, 235, 236, 257, 256, 236, 237, 258, 257, 237, 238, 259, 258, 238, 239, 260, 259, 239, 240, 261, 260, 240, 241, 262, 261, 241, 242, 263, 262, 242, 243, 264, 263, 243, 244, 265, 264, 244, 245, 266, 265, 245, 246, 267, 266, 246, 247, 268, 267, 247, 248, 269, 268, 248, 249, 270, 269, 249, 250, 271, 270, 250, 251, 272, 271, 252, 253, 274, 273, 253, 254, 275, 274, 254, 255, 276, 275, 255, 256, 277, 276, 256, 257, 278, 277, 257, 258, 279, 278, 258, 259, 280, 279, 259, 260, 281, 280, 260, 261, 282, 281, 261, 262, 283, 282, 262, 263, 284, 283, 263, 264, 285, 284, 264, 265, 286, 285, 265, 266, 287, 286, 266, 267, 288, 287, 267, 268, 289, 288, 268, 269, 290, 289, 269, 270, 291, 290, 270, 271, 292, 291, 271, 272, 293, 292, 273, 274, 295, 294, 274, 275, 296, 295, 275, 276, 297, 296, 276, 277, 298, 297, 277, 278, 299, 298, 278, 279, 300, 299, 279, 280, 301, 300, 280, 281, 302, 301, 281, 282, 303, 302, 282, 283, 304, 303, 283, 284, 305, 304, 284, 285, 306, 305, 285, 286, 307, 306, 286, 287, 308, 307, 287, 288, 309, 308, 288, 289, 310, 309, 289, 290, 311, 310, 290, 291, 312, 311, 291, 292, 313, 312, 292, 293, 314, 313, 294, 295, 316, 315, 295, 296, 317, 316, 296, 297, 318, 317, 297, 298, 319, 318, 298, 299, 320, 319, 299, 300, 321, 320, 300, 301, 322, 321, 301, 302, 323, 322, 302, 303, 324, 323, 303, 304, 325, 324, 304, 305, 326, 325, 305, 306, 327, 326, 306, 307, 328, 327, 307, 308, 329, 328, 308, 309, 330, 329, 309, 310, 331, 330, 310, 311, 332, 331, 311, 312, 333, 332, 312, 313, 334, 333, 313, 314, 335, 334, 315, 316, 337, 336, 316, 317, 338, 337, 317, 318, 339, 338, 318, 319, 340, 339, 319, 320, 341, 340, 320, 321, 342, 341, 321, 322, 343, 342, 322, 323, 344, 343, 323, 324, 345, 344, 324, 325, 346, 345, 325, 326, 347, 346, 326, 327, 348, 347, 327, 328, 349, 348, 328, 329, 350, 349, 329, 330, 351, 350, 330, 331, 352, 351, 331, 332, 353, 352, 332, 333, 354, 353, 333, 334, 355, 354, 334, 335, 356, 355, 336, 337, 358, 357, 337, 338, 359, 358, 338, 339, 360, 359, 339, 340, 361, 360, 340, 341, 362, 361, 341, 342, 363, 362, 342, 343, 364, 363, 343, 344, 365, 364, 344, 345, 366, 365, 345, 346, 367, 366, 346, 347, 368, 367, 347, 348, 369, 368, 348, 349, 370, 369, 349, 350, 371, 370, 350, 351, 372, 371, 351, 352, 373, 372, 352, 353, 374, 373, 353, 354, 375, 374, 354, 355, 376, 375, 355, 356, 377, 376, 357, 358, 379, 378, 358, 359, 380, 379, 359, 360, 381, 380, 360, 361, 382, 381, 361, 362, 383, 382, 362, 363, 384, 383, 363, 364, 385, 384, 364, 365, 386, 385, 365, 366, 387, 386, 366, 367, 388, 387, 367, 368, 389, 388, 368, 369, 390, 389, 369, 370, 391, 390, 370, 371, 392, 391, 371, 372, 393, 392, 372, 373, 394, 393, 373, 374, 395, 394, 374, 375, 396, 395, 375, 376, 397, 396, 376, 377, 398, 397, 1, 0, 399, 2, 1, 400, 3, 2, 401, 4, 3, 402, 5, 4, 403, 6, 5, 404, 7, 6, 405, 8, 7, 406, 9, 8, 407, 10, 9, 408, 11, 10, 409, 12, 11, 410, 13, 12, 411, 14, 13, 412, 15, 14, 413, 16, 15, 414, 17, 16, 415, 18, 17, 416, 19, 18, 417, 20, 19, 418, 378, 379, 419, 379, 380, 420, 380, 381, 421, 381, 382, 422, 382, 383, 423, 383, 384, 424, 384, 385, 425, 385, 386, 426, 386, 387, 427, 387, 388, 428, 388, 389, 429, 389, 390, 430, 390, 391, 431, 391, 392, 432, 392, 393, 433, 393, 394, 434, 394, 395, 435, 395, 396, 436, 396, 397, 437, 397, 398, 438]\n" + + " token visibility = \"inherited\"\n float3 xformOp:rotateXYZ = (0, -0, 0)\n float3 xformOp:scale = (1, 1, 1)\n double3 xformOp:translate = (-7.602017665895586, 0, 0)\n uniform token[] xformOpOrder = [\"xformOp:translate\", \"xformOp:rotateXYZ\", \"xformOp:scale\"]\n }\n\n def Mesh \"cube\" (\n kind = \"component\"\n )\n {\n uniform bool doubleSided = 1\n float3[] extent = [(-0.5, -0.5, -0.5), (0.5, 0.5, 0.5)]\n int[] faceVertexCounts = [4, 4, 4, 4, 4, 4]\n int[] faceVertexIndices = [0, 1, 3, 2, 2, 3, 5, 4, 4, 5, 7, 6, 6, 7, 1, 0, 1, 7, 5, 3, 6, 0, 2, 4]\n point3f[] points = [(-0.5, -0.5, 0.5), (0.5, -0.5, 0.5), (-0.5, 0.5, 0.5), (0.5, 0.5, 0.5), (-0.5, 0.5, -0.5), (0.5, 0.5, -0.5), (-0.5, -0.5, -0.5), (0.5, -0.5, -0.5)]\n color3f[] primvars:displayColor = [(0.13320851, 0.13320851, 0.13320851)] (\n customData = {\n dictionary Maya = {\n bool generated = 1\n }\n }\n )\n texCoord2f[] primvars:st = [(0.375, 0), (0.625, 0), (0.375, 0.25), (0.625, 0.25), (0.375, 0.5), (0.625, 0.5), (0.375, 0.75), (0.625, 0.75), (0.375, 1), (0.625, 1), (0.875, 0), (0.875, 0.25), (0.125, 0), (0.125, 0.25)] (\n" + + " customData = {\n dictionary Maya = {\n token name = \"map1\"\n }\n }\n interpolation = \"faceVarying\"\n )\n int[] primvars:st:indices = [0, 1, 3, 2, 2, 3, 5, 4, 4, 5, 7, 6, 6, 7, 9, 8, 1, 10, 11, 3, 12, 0, 2, 13]\n token visibility = \"inherited\"\n float3 xformOp:rotateXYZ = (0, -0, 0)\n float3 xformOp:scale = (1, 1, 1)\n double3 xformOp:translate = (-2.4045492443284795, 0, 0)\n uniform token[] xformOpOrder = [\"xformOp:translate\", \"xformOp:rotateXYZ\", \"xformOp:scale\"]\n\n def GeomSubset \"back\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [2]\n }\n\n def GeomSubset \"bottom\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [3]\n }\n\n def GeomSubset \"front\"\n {\n uniform token elementType = \"face\"\n" + + " uniform token familyName = \"componentTag\"\n int[] indices = [0]\n }\n\n def GeomSubset \"left\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [5]\n }\n\n def GeomSubset \"right\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [4]\n }\n\n def GeomSubset \"top\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [1]\n }\n }\n}\n\n"); + setAttr ".lyr[1].ann" yes; + setAttr ".lyr[2].id" -type "string" "anon:000001A373F7E250:anonymousLayer1-session.usda"; + setAttr ".lyr[2].fid" -type "string" "usda"; + setAttr ".lyr[2].szd" -type "string" ""; + setAttr ".lyr[2].ann" yes; + setAttr ".lyr[3].id" -type "string" "anon:000001A373F7DA10:anonymousLayer1"; + setAttr ".lyr[3].fid" -type "string" "sdf"; + setAttr ".lyr[3].szd" -type "string" ( + "#sdf 1.4.32\n\ndef Xform \"cylinderAndConeParent\"\n{\n def Mesh \"cylinder\" (\n kind = \"component\"\n )\n {\n uniform bool doubleSided = 1\n float3[] extent = [(-1.0000002, -1, -1.0000005), (1, 1, 1.0000001)]\n int[] faceVertexCounts = [4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3]\n int[] faceVertexIndices = [0, 1, 21, 20, 1, 2, 22, 21, 2, 3, 23, 22, 3, 4, 24, 23, 4, 5, 25, 24, 5, 6, 26, 25, 6, 7, 27, 26, 7, 8, 28, 27, 8, 9, 29, 28, 9, 10, 30, 29, 10, 11, 31, 30, 11, 12, 32, 31, 12, 13, 33, 32, 13, 14, 34, 33, 14, 15, 35, 34, 15, 16, 36, 35, 16, 17, 37, 36, 17, 18, 38, 37, 18, 19, 39, 38, 19, 0, 20, 39, 1, 0, 40, 2, 1, 40, 3, 2, 40, 4, 3, 40, 5, 4, 40, 6, 5, 40, 7, 6, 40, 8, 7, 40, 9, 8, 40, 10, 9, 40, 11, 10, 40, 12, 11, 40, 13, 12, 40, 14, 13, 40, 15, 14, 40, 16, 15, 40, 17, 16, 40, 18, 17, 40, 19, 18, 40, 0, 19, 40, 20, 21, 41, 21, 22, 41, 22, 23, 41, 23, 24, 41, 24, 25, 41, 25, 26, 41, 26, 27, 41, 27, 28, 41, 28, 29, 41, 29, 30, 41, 30, 31, 41, 31, 32, 41, 32, 33, 41, 33, 34, 41, 34, 35, 41, 35, 36, 41, 36, 37, 41, 37, 38, 41, 38, 39, 41, 39, 20, 41]\n" + + " point3f[] points = [(0.95105714, -1, -0.30901718), (0.80901754, -1, -0.5877856), (0.5877856, -1, -0.8090175), (0.30901715, -1, -0.951057), (0, -1, -1.0000005), (-0.30901715, -1, -0.95105696), (-0.5877855, -1, -0.8090173), (-0.80901724, -1, -0.5877854), (-0.9510568, -1, -0.30901706), (-1.0000002, -1, 0), (-0.9510568, -1, 0.30901706), (-0.8090172, -1, 0.58778536), (-0.58778536, -1, 0.8090171), (-0.30901706, -1, 0.95105666), (-2.9802322e-8, -1, 1.0000001), (0.30901697, -1, 0.9510566), (0.58778524, -1, 0.80901706), (0.809017, -1, 0.5877853), (0.95105654, -1, 0.309017), (1, -1, 0), (0.95105714, 1, -0.30901718), (0.80901754, 1, -0.5877856), (0.5877856, 1, -0.8090175), (0.30901715, 1, -0.951057), (0, 1, -1.0000005), (-0.30901715, 1, -0.95105696), (-0.5877855, 1, -0.8090173), (-0.80901724, 1, -0.5877854), (-0.9510568, 1, -0.30901706), (-1.0000002, 1, 0), (-0.9510568, 1, 0.30901706), (-0.8090172, 1, 0.58778536), (-0.58778536, 1, 0.8090171), (-0.30901706, 1, 0.95105666), (-2.9802322e-8, 1, 1.0000001), (0.30901697, 1, 0.9510566), (0.58778524, 1, 0.80901706), (0.809017, 1, 0.5877853), (0.95105654, 1, 0.309017), (1, 1, 0), (0, -1, 0), (0, 1, 0)]\n" + + " color3f[] primvars:displayColor = [(0.13320851, 0.13320851, 0.13320851)] (\n customData = {\n dictionary Maya = {\n bool generated = 1\n }\n }\n )\n" + + " texCoord2f[] primvars:st = [(0.64860266, 0.107966065), (0.626409, 0.064408496), (0.5918415, 0.02984102), (0.54828393, 0.0076473355), (0.5, -7.4505806e-8), (0.45171607, 0.0076473504), (0.4081585, 0.02984105), (0.37359107, 0.064408526), (0.3513974, 0.107966095), (0.34374997, 0.15625), (0.3513974, 0.2045339), (0.37359107, 0.24809146), (0.40815854, 0.28265893), (0.4517161, 0.3048526), (0.5, 0.3125), (0.5482839, 0.3048526), (0.59184146, 0.28265893), (0.62640893, 0.24809146), (0.6486026, 0.2045339), (0.65625, 0.15625), (0.375, 0.3125), (0.3875, 0.3125), (0.39999998, 0.3125), (0.41249996, 0.3125), (0.42499995, 0.3125), (0.43749994, 0.3125), (0.44999993, 0.3125), (0.46249992, 0.3125), (0.4749999, 0.3125), (0.4874999, 0.3125), (0.49999988, 0.3125), (0.51249987, 0.3125), (0.52499986, 0.3125), (0.53749985, 0.3125), (0.54999983, 0.3125), (0.5624998, 0.3125), (0.5749998, 0.3125), (0.5874998, 0.3125), (0.5999998, 0.3125), (0.6124998, 0.3125), (0.62499976, 0.3125), (0.375, 0.6875), (0.3875, 0.6875), (0.39999998, 0.6875), (0.41249996, 0.6875), (0.42499995, 0.6875), (0.43749994, 0.6875), (0.44999993, 0.6875), (0.46249992, 0.6875), (0.4749999, 0.6875), (0.4874999, 0.6875), (0.49999988, 0.6875), (0.51249987, 0.6875), (0.52499986, 0.6875), (0.53749985, 0.6875), (0.54999983, 0.6875), (0.5624998, 0.6875), (0.5749998, 0.6875), (0.5874998, 0.6875), (0.5999998, 0.6875), (0.6124998, 0.6875), (0.62499976, 0.6875), (0.64860266, 0.79546607), (0.626409, 0.7519085), (0.5918415, 0.717341), (0.54828393, 0.69514734), (0.5, 0.68749994), (0.45171607, 0.69514734), (0.4081585, 0.71734107), (0.37359107, 0.75190854), (0.3513974, 0.79546607), (0.34374997, 0.84375), (0.3513974, 0.89203393), (0.37359107, 0.93559146), (0.40815854, 0.97015893), (0.4517161, 0.9923526), (0.5, 1), (0.5482839, 0.9923526), (0.59184146, 0.97015893), (0.62640893, 0.93559146), (0.6486026, 0.89203393), (0.65625, 0.84375), (0.5, 0.15625), (0.5, 0.84375)] (\n" + + " customData = {\n dictionary Maya = {\n token name = \"map1\"\n }\n }\n interpolation = \"faceVarying\"\n )\n int[] primvars:st:indices = [20, 21, 42, 41, 21, 22, 43, 42, 22, 23, 44, 43, 23, 24, 45, 44, 24, 25, 46, 45, 25, 26, 47, 46, 26, 27, 48, 47, 27, 28, 49, 48, 28, 29, 50, 49, 29, 30, 51, 50, 30, 31, 52, 51, 31, 32, 53, 52, 32, 33, 54, 53, 33, 34, 55, 54, 34, 35, 56, 55, 35, 36, 57, 56, 36, 37, 58, 57, 37, 38, 59, 58, 38, 39, 60, 59, 39, 40, 61, 60, 1, 0, 82, 2, 1, 82, 3, 2, 82, 4, 3, 82, 5, 4, 82, 6, 5, 82, 7, 6, 82, 8, 7, 82, 9, 8, 82, 10, 9, 82, 11, 10, 82, 12, 11, 82, 13, 12, 82, 14, 13, 82, 15, 14, 82, 16, 15, 82, 17, 16, 82, 18, 17, 82, 19, 18, 82, 0, 19, 82, 80, 79, 83, 79, 78, 83, 78, 77, 83, 77, 76, 83, 76, 75, 83, 75, 74, 83, 74, 73, 83, 73, 72, 83, 72, 71, 83, 71, 70, 83, 70, 69, 83, 69, 68, 83, 68, 67, 83, 67, 66, 83, 66, 65, 83, 65, 64, 83, 64, 63, 83, 63, 62, 83, 62, 81, 83, 81, 80, 83]\n token visibility = \"inherited\"\n" + + " float3 xformOp:rotateXYZ = (0, -0, 0)\n float3 xformOp:scale = (1, 1, 1)\n double3 xformOp:translate = (2.1286509189718723, 0, 0)\n uniform token[] xformOpOrder = [\"xformOp:translate\", \"xformOp:rotateXYZ\", \"xformOp:scale\"]\n\n def GeomSubset \"bottom\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39]\n }\n\n def GeomSubset \"sides\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]\n }\n\n def GeomSubset \"top\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59]\n }\n }\n" + + "\n def Mesh \"cone\" (\n kind = \"component\"\n )\n {\n uniform bool doubleSided = 1\n float3[] extent = [(-1.0000002, -1, -1.0000005), (1, 1, 1.0000001)]\n int[] faceVertexCounts = [20, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3]\n int[] faceVertexIndices = [0, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 1, 20, 1, 2, 20, 2, 3, 20, 3, 4, 20, 4, 5, 20, 5, 6, 20, 6, 7, 20, 7, 8, 20, 8, 9, 20, 9, 10, 20, 10, 11, 20, 11, 12, 20, 12, 13, 20, 13, 14, 20, 14, 15, 20, 15, 16, 20, 16, 17, 20, 17, 18, 20, 18, 19, 20, 19, 0, 20]\n point3f[] points = [(0.95105714, -1, -0.30901718), (0.80901754, -1, -0.5877856), (0.5877856, -1, -0.8090175), (0.30901715, -1, -0.951057), (0, -1, -1.0000005), (-0.30901715, -1, -0.95105696), (-0.5877855, -1, -0.8090173), (-0.80901724, -1, -0.5877854), (-0.9510568, -1, -0.30901706), (-1.0000002, -1, 0), (-0.9510568, -1, 0.30901706), (-0.8090172, -1, 0.58778536), (-0.58778536, -1, 0.8090171), (-0.30901706, -1, 0.95105666), (-2.9802322e-8, -1, 1.0000001), (0.30901697, -1, 0.9510566), (0.58778524, -1, 0.80901706), (0.809017, -1, 0.5877853), (0.95105654, -1, 0.309017), (1, -1, 0), (0, 1, 0)]\n" + + " color3f[] primvars:displayColor = [(0.13320851, 0.13320851, 0.13320851)] (\n customData = {\n dictionary Maya = {\n bool generated = 1\n }\n }\n )\n texCoord2f[] primvars:st = [(0.7377643, 0.1727457), (0.7022544, 0.1030536), (0.64694643, 0.04774563), (0.5772543, 0.012235746), (0.5, -1.1920929e-7), (0.4227457, 0.012235761), (0.35305363, 0.047745675), (0.2977457, 0.103053644), (0.26223582, 0.17274573), (0.24999994, 0.25), (0.26223582, 0.32725427), (0.2977457, 0.39694634), (0.35305366, 0.4522543), (0.42274573, 0.48776418), (0.5, 0.5), (0.57725424, 0.48776415), (0.6469463, 0.45225427), (0.70225424, 0.3969463), (0.7377641, 0.32725424), (0.75, 0.25), (0.25, 0.5), (0.275, 0.5), (0.3, 0.5), (0.32500002, 0.5), (0.35000002, 0.5), (0.37500003, 0.5), (0.40000004, 0.5), (0.42500004, 0.5), (0.45000005, 0.5), (0.47500005, 0.5), (0.50000006, 0.5), (0.52500004, 0.5), (0.55, 0.5), (0.575, 0.5), (0.59999996, 0.5), (0.62499994, 0.5), (0.6499999, 0.5), (0.6749999, 0.5), (0.69999987, 0.5), (0.72499985, 0.5), (0.7499998, 0.5), (0.5, 1)] (\n" + + " customData = {\n dictionary Maya = {\n token name = \"map1\"\n }\n }\n interpolation = \"faceVarying\"\n )\n int[] primvars:st:indices = [0, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 20, 21, 41, 21, 22, 41, 22, 23, 41, 23, 24, 41, 24, 25, 41, 25, 26, 41, 26, 27, 41, 27, 28, 41, 28, 29, 41, 29, 30, 41, 30, 31, 41, 31, 32, 41, 32, 33, 41, 33, 34, 41, 34, 35, 41, 35, 36, 41, 36, 37, 41, 37, 38, 41, 38, 39, 41, 39, 40, 41]\n token visibility = \"inherited\"\n float3 xformOp:rotateXYZ = (0, -0, 0)\n float3 xformOp:scale = (1, 1, 1)\n double3 xformOp:translate = (5.750569244240079, 0, 0)\n uniform token[] xformOpOrder = [\"xformOp:translate\", \"xformOp:rotateXYZ\", \"xformOp:scale\"]\n\n def GeomSubset \"bottom\"\n {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [0]\n }\n\n def GeomSubset \"sides\"\n" + + " {\n uniform token elementType = \"face\"\n uniform token familyName = \"componentTag\"\n int[] indices = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20]\n }\n }\n}\n\n"); + setAttr ".lyr[3].ann" yes; +createNode script -n "uiConfigurationScriptNode"; + rename -uid "41986B8B-4726-6A76-8ED6-20B46472EB53"; + setAttr ".b" -type "string" ( + "// Maya Mel UI Configuration File.\n//\n// This script is machine generated. Edit at your own risk.\n//\n//\n\nglobal string $gMainPane;\nif (`paneLayout -exists $gMainPane`) {\n\n\tglobal int $gUseScenePanelConfig;\n\tint $useSceneConfig = $gUseScenePanelConfig;\n\tint $nodeEditorPanelVisible = stringArrayContains(\"nodeEditorPanel1\", `getPanel -vis`);\n\tint $nodeEditorWorkspaceControlOpen = (`workspaceControl -exists nodeEditorPanel1Window` && `workspaceControl -q -visible nodeEditorPanel1Window`);\n\tint $menusOkayInPanels = `optionVar -q allowMenusInPanels`;\n\tint $nVisPanes = `paneLayout -q -nvp $gMainPane`;\n\tint $nPanes = 0;\n\tstring $editorName;\n\tstring $panelName;\n\tstring $itemFilterName;\n\tstring $panelConfig;\n\n\t//\n\t// get current state of the UI\n\t//\n\tsceneUIReplacement -update $gMainPane;\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"modelPanel\" (localizedPanelLabel(\"Top View\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tmodelPanel -edit -l (localizedPanelLabel(\"Top View\")) -mbv $menusOkayInPanels $panelName;\n" + + "\t\t$editorName = $panelName;\n modelEditor -e \n -camera \"|top\" \n -useInteractiveMode 0\n -displayLights \"default\" \n -displayAppearance \"smoothShaded\" \n -activeOnly 0\n -ignorePanZoom 0\n -wireframeOnShaded 0\n -headsUpDisplay 1\n -holdOuts 1\n -selectionHiliteDisplay 1\n -useDefaultMaterial 0\n -bufferMode \"double\" \n -twoSidedLighting 0\n -backfaceCulling 0\n -xray 0\n -jointXray 0\n -activeComponentsXray 0\n -displayTextures 0\n -smoothWireframe 0\n -lineWidth 1\n -textureAnisotropic 0\n -textureHilight 1\n -textureSampling 2\n -textureDisplay \"modulate\" \n -textureMaxSize 16384\n -fogging 0\n -fogSource \"fragment\" \n -fogMode \"linear\" \n -fogStart 0\n -fogEnd 100\n -fogDensity 0.1\n -fogColor 0.5 0.5 0.5 1 \n" + + " -depthOfFieldPreview 1\n -maxConstantTransparency 1\n -rendererName \"vp2Renderer\" \n -objectFilterShowInHUD 1\n -isFiltered 0\n -colorResolution 256 256 \n -bumpResolution 512 512 \n -textureCompression 0\n -transparencyAlgorithm \"frontAndBackCull\" \n -transpInShadows 0\n -cullingOverride \"none\" \n -lowQualityLighting 0\n -maximumNumHardwareLights 1\n -occlusionCulling 0\n -shadingModel 0\n -useBaseRenderer 0\n -useReducedRenderer 0\n -smallObjectCulling 0\n -smallObjectThreshold -1 \n -interactiveDisableShadows 0\n -interactiveBackFaceCull 0\n -sortTransparent 1\n -controllers 1\n -nurbsCurves 1\n -nurbsSurfaces 1\n -polymeshes 1\n -subdivSurfaces 1\n -planes 1\n -lights 1\n -cameras 1\n -controlVertices 1\n" + + " -hulls 1\n -grid 1\n -imagePlane 1\n -joints 1\n -ikHandles 1\n -deformers 1\n -dynamics 1\n -particleInstancers 1\n -fluids 1\n -hairSystems 1\n -follicles 1\n -nCloths 1\n -nParticles 1\n -nRigids 1\n -dynamicConstraints 1\n -locators 1\n -manipulators 1\n -pluginShapes 1\n -dimensions 1\n -handles 1\n -pivots 1\n -textures 1\n -strokes 1\n -motionTrails 1\n -clipGhosts 1\n -bluePencil 1\n -greasePencils 0\n -excludeObjectPreset \"All\" \n -shadows 0\n -captureSequenceNumber -1\n -width 1\n -height 1\n -sceneRenderFilter 0\n $editorName;\n modelEditor -e -viewSelected 0 $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"modelPanel\" (localizedPanelLabel(\"Side View\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tmodelPanel -edit -l (localizedPanelLabel(\"Side View\")) -mbv $menusOkayInPanels $panelName;\n\t\t$editorName = $panelName;\n modelEditor -e \n -camera \"|side\" \n -useInteractiveMode 0\n -displayLights \"default\" \n -displayAppearance \"smoothShaded\" \n -activeOnly 0\n -ignorePanZoom 0\n -wireframeOnShaded 0\n -headsUpDisplay 1\n -holdOuts 1\n -selectionHiliteDisplay 1\n -useDefaultMaterial 0\n -bufferMode \"double\" \n -twoSidedLighting 0\n -backfaceCulling 0\n -xray 0\n -jointXray 0\n -activeComponentsXray 0\n -displayTextures 0\n -smoothWireframe 0\n -lineWidth 1\n -textureAnisotropic 0\n -textureHilight 1\n -textureSampling 2\n -textureDisplay \"modulate\" \n -textureMaxSize 16384\n -fogging 0\n" + + " -fogSource \"fragment\" \n -fogMode \"linear\" \n -fogStart 0\n -fogEnd 100\n -fogDensity 0.1\n -fogColor 0.5 0.5 0.5 1 \n -depthOfFieldPreview 1\n -maxConstantTransparency 1\n -rendererName \"vp2Renderer\" \n -objectFilterShowInHUD 1\n -isFiltered 0\n -colorResolution 256 256 \n -bumpResolution 512 512 \n -textureCompression 0\n -transparencyAlgorithm \"frontAndBackCull\" \n -transpInShadows 0\n -cullingOverride \"none\" \n -lowQualityLighting 0\n -maximumNumHardwareLights 1\n -occlusionCulling 0\n -shadingModel 0\n -useBaseRenderer 0\n -useReducedRenderer 0\n -smallObjectCulling 0\n -smallObjectThreshold -1 \n -interactiveDisableShadows 0\n -interactiveBackFaceCull 0\n -sortTransparent 1\n -controllers 1\n -nurbsCurves 1\n" + + " -nurbsSurfaces 1\n -polymeshes 1\n -subdivSurfaces 1\n -planes 1\n -lights 1\n -cameras 1\n -controlVertices 1\n -hulls 1\n -grid 1\n -imagePlane 1\n -joints 1\n -ikHandles 1\n -deformers 1\n -dynamics 1\n -particleInstancers 1\n -fluids 1\n -hairSystems 1\n -follicles 1\n -nCloths 1\n -nParticles 1\n -nRigids 1\n -dynamicConstraints 1\n -locators 1\n -manipulators 1\n -pluginShapes 1\n -dimensions 1\n -handles 1\n -pivots 1\n -textures 1\n -strokes 1\n -motionTrails 1\n -clipGhosts 1\n -bluePencil 1\n -greasePencils 0\n -excludeObjectPreset \"All\" \n -shadows 0\n -captureSequenceNumber -1\n -width 1\n -height 1\n -sceneRenderFilter 0\n" + + " $editorName;\n modelEditor -e -viewSelected 0 $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"modelPanel\" (localizedPanelLabel(\"Front View\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tmodelPanel -edit -l (localizedPanelLabel(\"Front View\")) -mbv $menusOkayInPanels $panelName;\n\t\t$editorName = $panelName;\n modelEditor -e \n -camera \"|front\" \n -useInteractiveMode 0\n -displayLights \"default\" \n -displayAppearance \"smoothShaded\" \n -activeOnly 0\n -ignorePanZoom 0\n -wireframeOnShaded 0\n -headsUpDisplay 1\n -holdOuts 1\n -selectionHiliteDisplay 1\n -useDefaultMaterial 0\n -bufferMode \"double\" \n -twoSidedLighting 0\n -backfaceCulling 0\n -xray 0\n -jointXray 0\n -activeComponentsXray 0\n -displayTextures 0\n" + + " -smoothWireframe 0\n -lineWidth 1\n -textureAnisotropic 0\n -textureHilight 1\n -textureSampling 2\n -textureDisplay \"modulate\" \n -textureMaxSize 16384\n -fogging 0\n -fogSource \"fragment\" \n -fogMode \"linear\" \n -fogStart 0\n -fogEnd 100\n -fogDensity 0.1\n -fogColor 0.5 0.5 0.5 1 \n -depthOfFieldPreview 1\n -maxConstantTransparency 1\n -rendererName \"vp2Renderer\" \n -objectFilterShowInHUD 1\n -isFiltered 0\n -colorResolution 256 256 \n -bumpResolution 512 512 \n -textureCompression 0\n -transparencyAlgorithm \"frontAndBackCull\" \n -transpInShadows 0\n -cullingOverride \"none\" \n -lowQualityLighting 0\n -maximumNumHardwareLights 1\n -occlusionCulling 0\n -shadingModel 0\n -useBaseRenderer 0\n -useReducedRenderer 0\n" + + " -smallObjectCulling 0\n -smallObjectThreshold -1 \n -interactiveDisableShadows 0\n -interactiveBackFaceCull 0\n -sortTransparent 1\n -controllers 1\n -nurbsCurves 1\n -nurbsSurfaces 1\n -polymeshes 1\n -subdivSurfaces 1\n -planes 1\n -lights 1\n -cameras 1\n -controlVertices 1\n -hulls 1\n -grid 1\n -imagePlane 1\n -joints 1\n -ikHandles 1\n -deformers 1\n -dynamics 1\n -particleInstancers 1\n -fluids 1\n -hairSystems 1\n -follicles 1\n -nCloths 1\n -nParticles 1\n -nRigids 1\n -dynamicConstraints 1\n -locators 1\n -manipulators 1\n -pluginShapes 1\n -dimensions 1\n -handles 1\n -pivots 1\n -textures 1\n -strokes 1\n -motionTrails 1\n -clipGhosts 1\n" + + " -bluePencil 1\n -greasePencils 0\n -excludeObjectPreset \"All\" \n -shadows 0\n -captureSequenceNumber -1\n -width 1\n -height 1\n -sceneRenderFilter 0\n $editorName;\n modelEditor -e -viewSelected 0 $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"modelPanel\" (localizedPanelLabel(\"Persp View\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tmodelPanel -edit -l (localizedPanelLabel(\"Persp View\")) -mbv $menusOkayInPanels $panelName;\n\t\t$editorName = $panelName;\n modelEditor -e \n -camera \"|persp\" \n -useInteractiveMode 0\n -displayLights \"default\" \n -displayAppearance \"smoothShaded\" \n -activeOnly 0\n -ignorePanZoom 0\n -wireframeOnShaded 0\n -headsUpDisplay 1\n -holdOuts 1\n -selectionHiliteDisplay 1\n -useDefaultMaterial 0\n" + + " -bufferMode \"double\" \n -twoSidedLighting 0\n -backfaceCulling 0\n -xray 0\n -jointXray 0\n -activeComponentsXray 0\n -displayTextures 0\n -smoothWireframe 0\n -lineWidth 1\n -textureAnisotropic 0\n -textureHilight 1\n -textureSampling 2\n -textureDisplay \"modulate\" \n -textureMaxSize 16384\n -fogging 0\n -fogSource \"fragment\" \n -fogMode \"linear\" \n -fogStart 0\n -fogEnd 100\n -fogDensity 0.1\n -fogColor 0.5 0.5 0.5 1 \n -depthOfFieldPreview 1\n -maxConstantTransparency 1\n -rendererName \"vp2Renderer\" \n -rendererOverrideName \"mayaHydraRenderOverride_HdStormRendererPlugin\" \n -objectFilterShowInHUD 1\n -isFiltered 0\n -colorResolution 256 256 \n -bumpResolution 512 512 \n -textureCompression 0\n -transparencyAlgorithm \"frontAndBackCull\" \n" + + " -transpInShadows 0\n -cullingOverride \"none\" \n -lowQualityLighting 0\n -maximumNumHardwareLights 1\n -occlusionCulling 0\n -shadingModel 0\n -useBaseRenderer 0\n -useReducedRenderer 0\n -smallObjectCulling 0\n -smallObjectThreshold -1 \n -interactiveDisableShadows 0\n -interactiveBackFaceCull 0\n -sortTransparent 1\n -controllers 1\n -nurbsCurves 1\n -nurbsSurfaces 1\n -polymeshes 1\n -subdivSurfaces 1\n -planes 1\n -lights 1\n -cameras 1\n -controlVertices 1\n -hulls 1\n -grid 1\n -imagePlane 1\n -joints 1\n -ikHandles 1\n -deformers 1\n -dynamics 1\n -particleInstancers 1\n -fluids 1\n -hairSystems 1\n -follicles 1\n -nCloths 1\n -nParticles 1\n -nRigids 1\n" + + " -dynamicConstraints 1\n -locators 1\n -manipulators 1\n -pluginShapes 1\n -dimensions 1\n -handles 1\n -pivots 1\n -textures 1\n -strokes 1\n -motionTrails 1\n -clipGhosts 1\n -bluePencil 1\n -greasePencils 0\n -excludeObjectPreset \"All\" \n -shadows 0\n -captureSequenceNumber -1\n -width 1185\n -height 778\n -sceneRenderFilter 0\n $editorName;\n modelEditor -e -viewSelected 0 $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"outlinerPanel\" (localizedPanelLabel(\"ToggledOutliner\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\toutlinerPanel -edit -l (localizedPanelLabel(\"ToggledOutliner\")) -mbv $menusOkayInPanels $panelName;\n\t\t$editorName = $panelName;\n outlinerEditor -e \n -docTag \"isolOutln_fromSeln\" \n" + + " -showShapes 1\n -showAssignedMaterials 0\n -showTimeEditor 1\n -showReferenceNodes 1\n -showReferenceMembers 1\n -showAttributes 0\n -showConnected 0\n -showAnimCurvesOnly 0\n -showMuteInfo 0\n -organizeByLayer 1\n -organizeByClip 1\n -showAnimLayerWeight 1\n -autoExpandLayers 1\n -autoExpand 0\n -autoExpandAllAnimatedShapes 1\n -showDagOnly 0\n -showAssets 1\n -showContainedOnly 1\n -showPublishedAsConnected 0\n -showParentContainers 0\n -showContainerContents 1\n -ignoreDagHierarchy 0\n -expandConnections 0\n -showUpstreamCurves 1\n -showUnitlessCurves 1\n -showCompounds 1\n -showLeafs 1\n -showNumericAttrsOnly 0\n -highlightActive 1\n -autoSelectNewObjects 0\n -doNotSelectNewObjects 0\n -dropIsParent 1\n" + + " -transmitFilters 0\n -setFilter \"defaultSetFilter\" \n -showSetMembers 1\n -allowMultiSelection 1\n -alwaysToggleSelect 0\n -directSelect 0\n -isSet 0\n -isSetMember 0\n -showUfeItems 1\n -displayMode \"DAG\" \n -expandObjects 0\n -setsIgnoreFilters 1\n -containersIgnoreFilters 0\n -editAttrName 0\n -showAttrValues 0\n -highlightSecondary 0\n -showUVAttrsOnly 0\n -showTextureNodesOnly 0\n -attrAlphaOrder \"default\" \n -animLayerFilterOptions \"allAffecting\" \n -sortOrder \"none\" \n -longNames 0\n -niceNames 1\n -showNamespace 1\n -showPinIcons 0\n -mapMotionTrails 0\n -ignoreHiddenAttribute 0\n -ignoreOutlinerColor 0\n -renderFilterVisible 0\n -renderFilterIndex 0\n -selectionOrder \"chronological\" \n -expandAttribute 0\n" + + " -ufeFilter \"USD\" \"AllPurposes\" -ufeFilterValue 0\n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"outlinerPanel\" (localizedPanelLabel(\"Outliner\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\toutlinerPanel -edit -l (localizedPanelLabel(\"Outliner\")) -mbv $menusOkayInPanels $panelName;\n\t\t$editorName = $panelName;\n outlinerEditor -e \n -showShapes 0\n -showAssignedMaterials 0\n -showTimeEditor 1\n -showReferenceNodes 0\n -showReferenceMembers 0\n -showAttributes 0\n -showConnected 0\n -showAnimCurvesOnly 0\n -showMuteInfo 0\n -organizeByLayer 1\n -organizeByClip 1\n -showAnimLayerWeight 1\n -autoExpandLayers 1\n -autoExpand 0\n -autoExpandAllAnimatedShapes 1\n -showDagOnly 1\n -showAssets 1\n -showContainedOnly 1\n" + + " -showPublishedAsConnected 0\n -showParentContainers 0\n -showContainerContents 1\n -ignoreDagHierarchy 0\n -expandConnections 0\n -showUpstreamCurves 1\n -showUnitlessCurves 1\n -showCompounds 1\n -showLeafs 1\n -showNumericAttrsOnly 0\n -highlightActive 1\n -autoSelectNewObjects 0\n -doNotSelectNewObjects 0\n -dropIsParent 1\n -transmitFilters 0\n -setFilter \"defaultSetFilter\" \n -showSetMembers 1\n -allowMultiSelection 1\n -alwaysToggleSelect 0\n -directSelect 0\n -showUfeItems 1\n -displayMode \"DAG\" \n -expandObjects 0\n -setsIgnoreFilters 1\n -containersIgnoreFilters 0\n -editAttrName 0\n -showAttrValues 0\n -highlightSecondary 0\n -showUVAttrsOnly 0\n -showTextureNodesOnly 0\n -attrAlphaOrder \"default\" \n" + + " -animLayerFilterOptions \"allAffecting\" \n -sortOrder \"none\" \n -longNames 0\n -niceNames 1\n -showNamespace 1\n -showPinIcons 0\n -mapMotionTrails 0\n -ignoreHiddenAttribute 0\n -ignoreOutlinerColor 0\n -renderFilterVisible 0\n -ufeFilter \"USD\" \"AllPurposes\" -ufeFilterValue 0\n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"graphEditor\" (localizedPanelLabel(\"Graph Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Graph Editor\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = ($panelName+\"OutlineEd\");\n outlinerEditor -e \n -showShapes 1\n -showAssignedMaterials 0\n -showTimeEditor 1\n -showReferenceNodes 0\n -showReferenceMembers 0\n -showAttributes 1\n" + + " -showConnected 1\n -showAnimCurvesOnly 1\n -showMuteInfo 0\n -organizeByLayer 1\n -organizeByClip 1\n -showAnimLayerWeight 1\n -autoExpandLayers 1\n -autoExpand 1\n -autoExpandAllAnimatedShapes 1\n -showDagOnly 0\n -showAssets 1\n -showContainedOnly 0\n -showPublishedAsConnected 0\n -showParentContainers 0\n -showContainerContents 0\n -ignoreDagHierarchy 0\n -expandConnections 1\n -showUpstreamCurves 1\n -showUnitlessCurves 1\n -showCompounds 0\n -showLeafs 1\n -showNumericAttrsOnly 1\n -highlightActive 0\n -autoSelectNewObjects 1\n -doNotSelectNewObjects 0\n -dropIsParent 1\n -transmitFilters 1\n -setFilter \"0\" \n -showSetMembers 0\n" + + " -allowMultiSelection 1\n -alwaysToggleSelect 0\n -directSelect 0\n -showUfeItems 1\n -displayMode \"DAG\" \n -expandObjects 0\n -setsIgnoreFilters 1\n -containersIgnoreFilters 0\n -editAttrName 0\n -showAttrValues 0\n -highlightSecondary 0\n -showUVAttrsOnly 0\n -showTextureNodesOnly 0\n -attrAlphaOrder \"default\" \n -animLayerFilterOptions \"allAffecting\" \n -sortOrder \"none\" \n -longNames 0\n -niceNames 1\n -showNamespace 1\n -showPinIcons 1\n -mapMotionTrails 1\n -ignoreHiddenAttribute 0\n -ignoreOutlinerColor 0\n -renderFilterVisible 0\n $editorName;\n\n\t\t\t$editorName = ($panelName+\"GraphEd\");\n animCurveEditor -e \n -displayValues 0\n -snapTime \"integer\" \n" + + " -snapValue \"none\" \n -showPlayRangeShades \"on\" \n -lockPlayRangeShades \"off\" \n -smoothness \"fine\" \n -resultSamples 1\n -resultScreenSamples 0\n -resultUpdate \"delayed\" \n -showUpstreamCurves 1\n -keyMinScale 1\n -stackedCurvesMin -1\n -stackedCurvesMax 1\n -stackedCurvesSpace 0.2\n -preSelectionHighlight 1\n -limitToSelectedCurves 0\n -constrainDrag 0\n -valueLinesToggle 0\n -highlightAffectedCurves 0\n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"dopeSheetPanel\" (localizedPanelLabel(\"Dope Sheet\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Dope Sheet\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = ($panelName+\"OutlineEd\");\n" + + " outlinerEditor -e \n -showShapes 1\n -showAssignedMaterials 0\n -showTimeEditor 1\n -showReferenceNodes 0\n -showReferenceMembers 0\n -showAttributes 1\n -showConnected 1\n -showAnimCurvesOnly 1\n -showMuteInfo 0\n -organizeByLayer 1\n -organizeByClip 1\n -showAnimLayerWeight 1\n -autoExpandLayers 1\n -autoExpand 1\n -autoExpandAllAnimatedShapes 1\n -showDagOnly 0\n -showAssets 1\n -showContainedOnly 0\n -showPublishedAsConnected 0\n -showParentContainers 0\n -showContainerContents 0\n -ignoreDagHierarchy 0\n -expandConnections 1\n -showUpstreamCurves 1\n -showUnitlessCurves 0\n -showCompounds 0\n -showLeafs 1\n -showNumericAttrsOnly 1\n" + + " -highlightActive 0\n -autoSelectNewObjects 0\n -doNotSelectNewObjects 1\n -dropIsParent 1\n -transmitFilters 0\n -setFilter \"0\" \n -showSetMembers 1\n -allowMultiSelection 1\n -alwaysToggleSelect 0\n -directSelect 0\n -showUfeItems 1\n -displayMode \"DAG\" \n -expandObjects 0\n -setsIgnoreFilters 1\n -containersIgnoreFilters 0\n -editAttrName 0\n -showAttrValues 0\n -highlightSecondary 0\n -showUVAttrsOnly 0\n -showTextureNodesOnly 0\n -attrAlphaOrder \"default\" \n -animLayerFilterOptions \"allAffecting\" \n -sortOrder \"none\" \n -longNames 0\n -niceNames 1\n -showNamespace 1\n -showPinIcons 0\n -mapMotionTrails 1\n -ignoreHiddenAttribute 0\n" + + " -ignoreOutlinerColor 0\n -renderFilterVisible 0\n $editorName;\n\n\t\t\t$editorName = ($panelName+\"DopeSheetEd\");\n dopeSheetEditor -e \n -displayValues 0\n -snapTime \"integer\" \n -snapValue \"none\" \n -outliner \"dopeSheetPanel1OutlineEd\" \n -showSummary 1\n -showScene 0\n -hierarchyBelow 0\n -showTicks 1\n -selectionWindow 0 0 0 0 \n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"timeEditorPanel\" (localizedPanelLabel(\"Time Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Time Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"clipEditorPanel\" (localizedPanelLabel(\"Trax Editor\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Trax Editor\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = clipEditorNameFromPanel($panelName);\n clipEditor -e \n -displayValues 0\n -snapTime \"none\" \n -snapValue \"none\" \n -initialized 0\n -manageSequencer 0 \n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"sequenceEditorPanel\" (localizedPanelLabel(\"Camera Sequencer\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Camera Sequencer\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = sequenceEditorNameFromPanel($panelName);\n clipEditor -e \n -displayValues 0\n -snapTime \"none\" \n -snapValue \"none\" \n -initialized 0\n" + + " -manageSequencer 1 \n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"hyperGraphPanel\" (localizedPanelLabel(\"Hypergraph Hierarchy\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Hypergraph Hierarchy\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = ($panelName+\"HyperGraphEd\");\n hyperGraph -e \n -graphLayoutStyle \"hierarchicalLayout\" \n -orientation \"horiz\" \n -mergeConnections 0\n -zoom 1\n -animateTransition 0\n -showRelationships 1\n -showShapes 0\n -showDeformers 0\n -showExpressions 0\n -showConstraints 0\n -showConnectionFromSelected 0\n -showConnectionToSelected 0\n -showConstraintLabels 0\n -showUnderworld 0\n" + + " -showInvisible 0\n -transitionFrames 1\n -opaqueContainers 0\n -freeform 0\n -imagePosition 0 0 \n -imageScale 1\n -imageEnabled 0\n -graphType \"DAG\" \n -heatMapDisplay 0\n -updateSelection 1\n -updateNodeAdded 1\n -useDrawOverrideColor 0\n -limitGraphTraversal -1\n -range 0 0 \n -iconSize \"smallIcons\" \n -showCachedConnections 0\n $editorName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"hyperShadePanel\" (localizedPanelLabel(\"Hypershade\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Hypershade\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"visorPanel\" (localizedPanelLabel(\"Visor\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Visor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"nodeEditorPanel\" (localizedPanelLabel(\"Node Editor\")) `;\n\tif ($nodeEditorPanelVisible || $nodeEditorWorkspaceControlOpen) {\n\t\tif (\"\" == $panelName) {\n\t\t\tif ($useSceneConfig) {\n\t\t\t\t$panelName = `scriptedPanel -unParent -type \"nodeEditorPanel\" -l (localizedPanelLabel(\"Node Editor\")) -mbv $menusOkayInPanels `;\n\n\t\t\t$editorName = ($panelName+\"NodeEditorEd\");\n nodeEditor -e \n -allAttributes 0\n -allNodes 0\n -autoSizeNodes 1\n -consistentNameSize 1\n -createNodeCommand \"nodeEdCreateNodeCommand\" \n -connectNodeOnCreation 0\n -connectOnDrop 0\n -copyConnectionsOnPaste 0\n -connectionStyle \"bezier\" \n -defaultPinnedState 0\n" + + " -additiveGraphingMode 0\n -connectedGraphingMode 1\n -settingsChangedCallback \"nodeEdSyncControls\" \n -traversalDepthLimit -1\n -keyPressCommand \"nodeEdKeyPressCommand\" \n -nodeTitleMode \"name\" \n -gridSnap 0\n -gridVisibility 1\n -crosshairOnEdgeDragging 0\n -popupMenuScript \"nodeEdBuildPanelMenus\" \n -showNamespace 1\n -showShapes 1\n -showSGShapes 0\n -showTransforms 1\n -useAssets 1\n -syncedSelection 1\n -extendToShapes 1\n -showUnitConversions 0\n -editorMode \"default\" \n -hasWatchpoint 0\n $editorName;\n\t\t\t}\n\t\t} else {\n\t\t\t$label = `panel -q -label $panelName`;\n\t\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Node Editor\")) -mbv $menusOkayInPanels $panelName;\n\n\t\t\t$editorName = ($panelName+\"NodeEditorEd\");\n nodeEditor -e \n" + + " -allAttributes 0\n -allNodes 0\n -autoSizeNodes 1\n -consistentNameSize 1\n -createNodeCommand \"nodeEdCreateNodeCommand\" \n -connectNodeOnCreation 0\n -connectOnDrop 0\n -copyConnectionsOnPaste 0\n -connectionStyle \"bezier\" \n -defaultPinnedState 0\n -additiveGraphingMode 0\n -connectedGraphingMode 1\n -settingsChangedCallback \"nodeEdSyncControls\" \n -traversalDepthLimit -1\n -keyPressCommand \"nodeEdKeyPressCommand\" \n -nodeTitleMode \"name\" \n -gridSnap 0\n -gridVisibility 1\n -crosshairOnEdgeDragging 0\n -popupMenuScript \"nodeEdBuildPanelMenus\" \n -showNamespace 1\n -showShapes 1\n -showSGShapes 0\n -showTransforms 1\n -useAssets 1\n -syncedSelection 1\n" + + " -extendToShapes 1\n -showUnitConversions 0\n -editorMode \"default\" \n -hasWatchpoint 0\n $editorName;\n\t\t\tif (!$useSceneConfig) {\n\t\t\t\tpanel -e -l $label $panelName;\n\t\t\t}\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"createNodePanel\" (localizedPanelLabel(\"Create Node\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Create Node\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"polyTexturePlacementPanel\" (localizedPanelLabel(\"UV Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"UV Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"renderWindowPanel\" (localizedPanelLabel(\"Render View\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Render View\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"shapePanel\" (localizedPanelLabel(\"Shape Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tshapePanel -edit -l (localizedPanelLabel(\"Shape Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextPanel \"posePanel\" (localizedPanelLabel(\"Pose Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tposePanel -edit -l (localizedPanelLabel(\"Pose Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"dynRelEdPanel\" (localizedPanelLabel(\"Dynamic Relationships\")) `;\n\tif (\"\" != $panelName) {\n" + + "\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Dynamic Relationships\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"relationshipPanel\" (localizedPanelLabel(\"Relationship Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Relationship Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"referenceEditorPanel\" (localizedPanelLabel(\"Reference Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Reference Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"dynPaintScriptedPanelType\" (localizedPanelLabel(\"Paint Effects\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Paint Effects\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"scriptEditorPanel\" (localizedPanelLabel(\"Script Editor\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Script Editor\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"profilerPanel\" (localizedPanelLabel(\"Profiler Tool\")) `;\n\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Profiler Tool\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\t$panelName = `sceneUIReplacement -getNextScriptedPanel \"contentBrowserPanel\" (localizedPanelLabel(\"Content Browser\")) `;\n" + + "\tif (\"\" != $panelName) {\n\t\t$label = `panel -q -label $panelName`;\n\t\tscriptedPanel -edit -l (localizedPanelLabel(\"Content Browser\")) -mbv $menusOkayInPanels $panelName;\n\t\tif (!$useSceneConfig) {\n\t\t\tpanel -e -l $label $panelName;\n\t\t}\n\t}\n\n\n\tif ($useSceneConfig) {\n string $configName = `getPanel -cwl (localizedPanelLabel(\"Current Layout\"))`;\n if (\"\" != $configName) {\n\t\t\tpanelConfiguration -edit -label (localizedPanelLabel(\"Current Layout\")) \n\t\t\t\t-userCreated false\n\t\t\t\t-defaultImage \"vacantCell.xP:/\"\n\t\t\t\t-image \"\"\n\t\t\t\t-sc false\n\t\t\t\t-configString \"global string $gMainPane; paneLayout -e -cn \\\"single\\\" -ps 1 100 100 $gMainPane;\"\n\t\t\t\t-removeAllPanels\n\t\t\t\t-ap false\n\t\t\t\t\t(localizedPanelLabel(\"Persp View\")) \n\t\t\t\t\t\"modelPanel\"\n" + + "\t\t\t\t\t\"$panelName = `modelPanel -unParent -l (localizedPanelLabel(\\\"Persp View\\\")) -mbv $menusOkayInPanels `;\\n$editorName = $panelName;\\nmodelEditor -e \\n -cam `findStartUpCamera persp` \\n -useInteractiveMode 0\\n -displayLights \\\"default\\\" \\n -displayAppearance \\\"smoothShaded\\\" \\n -activeOnly 0\\n -ignorePanZoom 0\\n -wireframeOnShaded 0\\n -headsUpDisplay 1\\n -holdOuts 1\\n -selectionHiliteDisplay 1\\n -useDefaultMaterial 0\\n -bufferMode \\\"double\\\" \\n -twoSidedLighting 0\\n -backfaceCulling 0\\n -xray 0\\n -jointXray 0\\n -activeComponentsXray 0\\n -displayTextures 0\\n -smoothWireframe 0\\n -lineWidth 1\\n -textureAnisotropic 0\\n -textureHilight 1\\n -textureSampling 2\\n -textureDisplay \\\"modulate\\\" \\n -textureMaxSize 16384\\n -fogging 0\\n -fogSource \\\"fragment\\\" \\n -fogMode \\\"linear\\\" \\n -fogStart 0\\n -fogEnd 100\\n -fogDensity 0.1\\n -fogColor 0.5 0.5 0.5 1 \\n -depthOfFieldPreview 1\\n -maxConstantTransparency 1\\n -rendererName \\\"vp2Renderer\\\" \\n -rendererOverrideName \\\"mayaHydraRenderOverride_HdStormRendererPlugin\\\" \\n -objectFilterShowInHUD 1\\n -isFiltered 0\\n -colorResolution 256 256 \\n -bumpResolution 512 512 \\n -textureCompression 0\\n -transparencyAlgorithm \\\"frontAndBackCull\\\" \\n -transpInShadows 0\\n -cullingOverride \\\"none\\\" \\n -lowQualityLighting 0\\n -maximumNumHardwareLights 1\\n -occlusionCulling 0\\n -shadingModel 0\\n -useBaseRenderer 0\\n -useReducedRenderer 0\\n -smallObjectCulling 0\\n -smallObjectThreshold -1 \\n -interactiveDisableShadows 0\\n -interactiveBackFaceCull 0\\n -sortTransparent 1\\n -controllers 1\\n -nurbsCurves 1\\n -nurbsSurfaces 1\\n -polymeshes 1\\n -subdivSurfaces 1\\n -planes 1\\n -lights 1\\n -cameras 1\\n -controlVertices 1\\n -hulls 1\\n -grid 1\\n -imagePlane 1\\n -joints 1\\n -ikHandles 1\\n -deformers 1\\n -dynamics 1\\n -particleInstancers 1\\n -fluids 1\\n -hairSystems 1\\n -follicles 1\\n -nCloths 1\\n -nParticles 1\\n -nRigids 1\\n -dynamicConstraints 1\\n -locators 1\\n -manipulators 1\\n -pluginShapes 1\\n -dimensions 1\\n -handles 1\\n -pivots 1\\n -textures 1\\n -strokes 1\\n -motionTrails 1\\n -clipGhosts 1\\n -bluePencil 1\\n -greasePencils 0\\n -excludeObjectPreset \\\"All\\\" \\n -shadows 0\\n -captureSequenceNumber -1\\n -width 1185\\n -height 778\\n -sceneRenderFilter 0\\n $editorName;\\nmodelEditor -e -viewSelected 0 $editorName\"\n" + + "\t\t\t\t\t\"modelPanel -edit -l (localizedPanelLabel(\\\"Persp View\\\")) -mbv $menusOkayInPanels $panelName;\\n$editorName = $panelName;\\nmodelEditor -e \\n -cam `findStartUpCamera persp` \\n -useInteractiveMode 0\\n -displayLights \\\"default\\\" \\n -displayAppearance \\\"smoothShaded\\\" \\n -activeOnly 0\\n -ignorePanZoom 0\\n -wireframeOnShaded 0\\n -headsUpDisplay 1\\n -holdOuts 1\\n -selectionHiliteDisplay 1\\n -useDefaultMaterial 0\\n -bufferMode \\\"double\\\" \\n -twoSidedLighting 0\\n -backfaceCulling 0\\n -xray 0\\n -jointXray 0\\n -activeComponentsXray 0\\n -displayTextures 0\\n -smoothWireframe 0\\n -lineWidth 1\\n -textureAnisotropic 0\\n -textureHilight 1\\n -textureSampling 2\\n -textureDisplay \\\"modulate\\\" \\n -textureMaxSize 16384\\n -fogging 0\\n -fogSource \\\"fragment\\\" \\n -fogMode \\\"linear\\\" \\n -fogStart 0\\n -fogEnd 100\\n -fogDensity 0.1\\n -fogColor 0.5 0.5 0.5 1 \\n -depthOfFieldPreview 1\\n -maxConstantTransparency 1\\n -rendererName \\\"vp2Renderer\\\" \\n -rendererOverrideName \\\"mayaHydraRenderOverride_HdStormRendererPlugin\\\" \\n -objectFilterShowInHUD 1\\n -isFiltered 0\\n -colorResolution 256 256 \\n -bumpResolution 512 512 \\n -textureCompression 0\\n -transparencyAlgorithm \\\"frontAndBackCull\\\" \\n -transpInShadows 0\\n -cullingOverride \\\"none\\\" \\n -lowQualityLighting 0\\n -maximumNumHardwareLights 1\\n -occlusionCulling 0\\n -shadingModel 0\\n -useBaseRenderer 0\\n -useReducedRenderer 0\\n -smallObjectCulling 0\\n -smallObjectThreshold -1 \\n -interactiveDisableShadows 0\\n -interactiveBackFaceCull 0\\n -sortTransparent 1\\n -controllers 1\\n -nurbsCurves 1\\n -nurbsSurfaces 1\\n -polymeshes 1\\n -subdivSurfaces 1\\n -planes 1\\n -lights 1\\n -cameras 1\\n -controlVertices 1\\n -hulls 1\\n -grid 1\\n -imagePlane 1\\n -joints 1\\n -ikHandles 1\\n -deformers 1\\n -dynamics 1\\n -particleInstancers 1\\n -fluids 1\\n -hairSystems 1\\n -follicles 1\\n -nCloths 1\\n -nParticles 1\\n -nRigids 1\\n -dynamicConstraints 1\\n -locators 1\\n -manipulators 1\\n -pluginShapes 1\\n -dimensions 1\\n -handles 1\\n -pivots 1\\n -textures 1\\n -strokes 1\\n -motionTrails 1\\n -clipGhosts 1\\n -bluePencil 1\\n -greasePencils 0\\n -excludeObjectPreset \\\"All\\\" \\n -shadows 0\\n -captureSequenceNumber -1\\n -width 1185\\n -height 778\\n -sceneRenderFilter 0\\n $editorName;\\nmodelEditor -e -viewSelected 0 $editorName\"\n" + + "\t\t\t\t$configName;\n\n setNamedPanelLayout (localizedPanelLabel(\"Current Layout\"));\n }\n\n panelHistory -e -clear mainPanelHistory;\n sceneUIReplacement -clear;\n\t}\n\n\ngrid -spacing 5 -size 12 -divisions 5 -displayAxes yes -displayGridLines yes -displayDivisionLines yes -displayPerspectiveLabels no -displayOrthographicLabels no -displayAxesBold yes -perspectiveLabelPosition axis -orthographicLabelPosition edge;\nviewManip -drawCompass 0 -compassAngle 0 -frontParameters \"\" -homeParameters \"\" -selectionLockParameters \"\";\n}\n"); + setAttr ".st" 3; +createNode script -n "sceneConfigurationScriptNode"; + rename -uid "1BF9E70A-481A-AD17-FDA5-A1990923BCF8"; + setAttr ".b" -type "string" "playbackOptions -min -20 -max 10 -ast -20 -aet 10 "; + setAttr ".st" 6; +select -ne :time1; + setAttr ".o" -20; + setAttr ".unw" -20; +select -ne :hardwareRenderingGlobals; + setAttr ".otfna" -type "stringArray" 22 "NURBS Curves" "NURBS Surfaces" "Polygons" "Subdiv Surface" "Particles" "Particle Instance" "Fluids" "Strokes" "Image Planes" "UI" "Lights" "Cameras" "Locators" "Joints" "IK Handles" "Deformers" "Motion Trails" "Components" "Hair Systems" "Follicles" "Misc. UI" "Ornaments" ; + setAttr ".otfva" -type "Int32Array" 22 0 1 1 1 1 1 + 1 1 1 0 0 0 0 0 0 0 0 0 + 0 0 0 0 ; + setAttr ".fprt" yes; + setAttr ".rtfm" 1; +select -ne :renderPartition; + setAttr -s 2 ".st"; +select -ne :renderGlobalsList1; +select -ne :defaultShaderList1; + setAttr -s 5 ".s"; +select -ne :postProcessList1; + setAttr -s 2 ".p"; +select -ne :defaultRenderingList1; +select -ne :standardSurface1; + setAttr ".bc" -type "float3" 0.40000001 0.40000001 0.40000001 ; + setAttr ".sr" 0.5; +select -ne :initialShadingGroup; + setAttr ".ro" yes; +select -ne :initialParticleSE; + setAttr ".ro" yes; +select -ne :defaultRenderGlobals; + addAttr -ci true -sn "mtohMotionSampleStart" -ln "mtohMotionSampleStart" -at "float"; + addAttr -ci true -sn "mtohMotionSampleEnd" -ln "mtohMotionSampleEnd" -at "float"; + addAttr -ci true -sn "mayaHydraRenderPurpose" -ln "mayaHydraRenderPurpose" -min + 0 -max 1 -at "bool"; + addAttr -ci true -sn "mayaHydraProxyPurpose" -ln "mayaHydraProxyPurpose" -dv 1 -min + 0 -max 1 -at "bool"; + addAttr -ci true -sn "mayaHydraGuidePurpose" -ln "mayaHydraGuidePurpose" -min 0 + -max 1 -at "bool"; + addAttr -ci true -sn "mtohTextureMemoryPerTexture" -ln "mtohTextureMemoryPerTexture" + -dv 4096 -min 1 -max 262144 -smn 16384 -at "long"; + addAttr -ci true -sn "mtohMaximumShadowMapResolution" -ln "mtohMaximumShadowMapResolution" + -dv 2048 -min 32 -max 8192 -at "long"; + addAttr -ci true -sn "HdStormRendererPlugin__enableTinyPrimCulling" -ln "HdStormRendererPlugin__enableTinyPrimCulling" + -min 0 -max 1 -at "bool"; + addAttr -ci true -sn "HdStormRendererPlugin__volumeRaymarchingStepSize" -ln "HdStormRendererPlugin__volumeRaymarchingStepSize" + -dv 1 -at "float"; + addAttr -ci true -sn "HdStormRendererPlugin__volumeRaymarchingStepSizeLighting" + -ln "HdStormRendererPlugin__volumeRaymarchingStepSizeLighting" -dv 10 -at "float"; + addAttr -ci true -sn "HdStormRendererPlugin__volumeMaxTextureMemoryPerField" -ln "HdStormRendererPlugin__volumeMaxTextureMemoryPerField" + -dv 128 -at "float"; + addAttr -ci true -sn "HdStormRendererPlugin__maxLights" -ln "HdStormRendererPlugin__maxLights" + -dv 16 -at "long"; + addAttr -ci true -h true -sn "dss" -ln "defaultSurfaceShader" -dt "string"; + setAttr ".dss" -type "string" "standardSurface1"; +select -ne :defaultResolution; + setAttr ".pa" 1; +select -ne :defaultColorMgtGlobals; + setAttr ".cfe" yes; + setAttr ".cfp" -type "string" "/OCIO-configs/Maya2022-default/config.ocio"; + setAttr ".vtn" -type "string" "ACES 1.0 SDR-video (sRGB)"; + setAttr ".vn" -type "string" "ACES 1.0 SDR-video"; + setAttr ".dn" -type "string" "sRGB"; + setAttr ".wsn" -type "string" "ACEScg"; + setAttr ".otn" -type "string" "ACES 1.0 SDR-video (sRGB)"; + setAttr ".potn" -type "string" "ACES 1.0 SDR-video (sRGB)"; +select -ne :hardwareRenderGlobals; + setAttr ".ctrs" 256; + setAttr ".btrs" 512; +connectAttr ":time1.o" "sphereAndCubeShape.tm"; +connectAttr ":time1.o" "cylinderAndConeShape.tm"; +relationship "link" ":lightLinker1" ":initialShadingGroup.message" ":defaultLightSet.message"; +relationship "link" ":lightLinker1" ":initialParticleSE.message" ":defaultLightSet.message"; +relationship "shadowLink" ":lightLinker1" ":initialShadingGroup.message" ":defaultLightSet.message"; +relationship "shadowLink" ":lightLinker1" ":initialParticleSE.message" ":defaultLightSet.message"; +connectAttr "layerManager.dli[0]" "defaultLayer.id"; +connectAttr "renderLayerManager.rlmi[0]" "defaultRenderLayer.rlid"; +connectAttr "defaultRenderLayer.msg" ":defaultRenderingList1.r" -na; +// End of testSelectionHighlightHierarchy.ma diff --git a/test/testUtils/__init__.py b/test/testUtils/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/test/testUtils/cachingUtils.py b/test/testUtils/cachingUtils.py new file mode 100644 index 0000000000..7684966a4a --- /dev/null +++ b/test/testUtils/cachingUtils.py @@ -0,0 +1,162 @@ +#!/usr/bin/env python + +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +""" + Helper functions regarding Cached Playback. +""" +import maya.cmds as cmds + +from maya.debug.emModeManager import emModeManager +from maya.plugin.evaluator.CacheEvaluatorManager import CacheEvaluatorManager, CACHE_STANDARD_MODE_EVAL + +class NonCachingScope(object): + ''' + Scope object responsible for setting up non cached mode and restoring default settings after + ''' + def __enter__(self): + '''Enter the scope, setting up the evaluator managers and initial states''' + self.em_mgr = emModeManager() + self.em_mgr.setMode('emp') + self.em_mgr.setMode('-cache') + + return self + + def __init__(self, unit_test): + '''Initialize everything to be empty - only use the "with" syntax with this object''' + self.em_mgr = None + self.unit_test = unit_test + + def __exit__(self,exit_type,value,traceback): + '''Exit the scope, restoring all of the state information''' + if self.em_mgr: + self.em_mgr.restore_state() + self.em_mgr = None + + def verifyScopeSetup(self): + ''' + Meta-test to check that the scope was defined correctly + :param unit_test: The test object from which this method was called + ''' + self.unit_test.assertTrue( cmds.evaluationManager( mode=True, query=True )[0] == 'parallel' ) + if cmds.pluginInfo('cacheEvaluator', loaded=True, query=True): + self.unit_test.assertFalse( cmds.evaluator( query=True, en=True, name='cache' ) ) + + def checkValidFrames(self, expected_valid_frames, layers_mask = 0b01): + return True + + def waitForCache(self, wait_time=5): + return + + @staticmethod + def is_caching_scope(): + ''' + Method to determine whether caching is on or off in this object's scope + :return: False, since this is the non-caching scope + ''' + return False + +class CachingScope(object): + ''' + Scope object responsible for setting up caching and restoring original setup after + ''' + def __enter__(self): + '''Enter the scope, setting up the evaluator managers and initial states''' + self.em_mgr = emModeManager() + self.em_mgr.setMode('emp') + self.em_mgr.setMode('+cache') + # Enable idle build to make sure we can rebuild the graph when waiting. + self.em_mgr.idle_action = emModeManager.idle_action_build + + # Setup caching options + self.cache_mgr = CacheEvaluatorManager() + self.cache_mgr.save_state() + self.cache_mgr.plugin_loaded = True + self.cache_mgr.enabled = True + self.cache_mgr.cache_mode = CACHE_STANDARD_MODE_EVAL + self.cache_mgr.resource_guard = False + self.cache_mgr.fill_mode = 'syncAsync' + + # Setup autokey options + self.auto_key_state = cmds.autoKeyframe(q=True, state=True) + self.auto_key_chars = cmds.autoKeyframe(q=True, characterOption=True) + cmds.autoKeyframe(e=True, state=False) + + self.waitForCache() + + return self + + def __init__(self, unit_test): + '''Initialize everything to be empty - only use the "with" syntax with this object''' + self.em_mgr = None + self.cache_mgr = None + self.auto_key_state = None + self.auto_key_chars = None + self.unit_test = unit_test + + def __exit__(self,exit_type,value,traceback): + '''Exit the scope, restoring all of the state information''' + if self.cache_mgr: + self.cache_mgr.restore_state() + if self.em_mgr: + self.em_mgr.restore_state() + cmds.autoKeyframe(e=True, state=self.auto_key_state, characterOption=self.auto_key_chars) + + def verifyScopeSetup(self): + ''' + Meta-test to check that the scope was defined correctly + :param unit_test: The test object from which this method was called + ''' + self.unit_test.assertTrue( cmds.evaluationManager( mode=True, query=True )[0] == 'parallel' ) + self.unit_test.assertTrue( cmds.pluginInfo('cacheEvaluator', loaded=True, query=True) ) + self.unit_test.assertTrue( cmds.evaluator( query=True, en=True, name='cache' ) ) + + def checkValidFrames(self, expected_valid_frames, layers_mask = 0b01): + ''' + :param unit_test: The test object from which this method was called + :param expected_valid_frames: The list of frames the text expected to be cached + :return: True if the cached frame list matches the expected frame list + ''' + current_valid_frames = list(self.cache_mgr.get_valid_frames(layers_mask)) + if len(expected_valid_frames) == len(current_valid_frames): + for current, expected in zip(current_valid_frames,expected_valid_frames): + if current[0] != expected[0] or current[1] != expected[1]: + self.unit_test.fail( "{} != {} (current,expected)".format( current_valid_frames, expected_valid_frames) ) + return False + + return True + self.unit_test.fail( "{} != {} (current,expected)".format( current_valid_frames, expected_valid_frames) ) + return False + + def waitForCache(self, wait_time=5): + ''' + Fill the cache in the background, waiting for a maximum time + :param unit_test: The test object from which this method was called + :param wait_time: Time the test is willing to wait for cache completion (in seconds) + ''' + cmds.currentTime( cmds.currentTime(q=True) ) + cmds.currentTime( cmds.currentTime(q=True) ) + cache_is_ready = cmds.cacheEvaluator( waitForCache=wait_time ) + self.unit_test.assertTrue( cache_is_ready ) + + @staticmethod + def is_caching_scope(): + ''' + Method to determine whether caching is on or off in this object's scope + :return: True, since this is the caching scope + ''' + return True diff --git a/test/testUtils/fixturesUtils.py b/test/testUtils/fixturesUtils.py new file mode 100644 index 0000000000..717be63fea --- /dev/null +++ b/test/testUtils/fixturesUtils.py @@ -0,0 +1,150 @@ +# +# Copyright 2020 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +import os +import shutil +import sys +import unittest + +def _setUpClass(modulePathName, pluginName, initializeStandalone): + ''' + Common code for setUpClass() and readOnlySetUpClass() + ''' + if initializeStandalone: + from maya import standalone + standalone.initialize('usd') + + if pluginName: + import maya.cmds as cmds + cmds.loadPlugin(pluginName, quiet=True) + + realPath = os.path.realpath(modulePathName) + return os.path.split(realPath) + +def setUpClass(modulePathName, pluginName, initializeStandalone=True, suffix=''): + ''' + Test class setup. + + This function: + - (Optionally) Initializes Maya standalone use. + - Creates (or empties) a test output directory based on the argument. + - Changes the current working directory to the test output directory. + - (Optionally) Loads the plugin + - Returns the original directory from the argument. + ''' + (testDir, testFile) = _setUpClass(modulePathName, pluginName, + initializeStandalone) + outputName = os.path.splitext(testFile)[0]+suffix+'Output' + + outputPath = os.path.join(os.path.abspath('.'), outputName) + if os.path.exists(outputPath): + # Remove previous test run output. + shutil.rmtree(outputPath) + + os.mkdir(outputPath) + os.chdir(outputPath) + + return testDir + +def tearDownClass(pluginName): + ''' + Test class teardown. + + This function: + - Changes the current working directory to the main test directory. + - (Optionally) Unloads the plugin + + Required when running multiple test classes using fixturesUtils to + avoid nested test directories. + ''' + + if pluginName: + import maya.cmds as cmds + cmds.unloadPlugin(pluginName, force=True) + + # Exit into the main test directory + os.chdir("..") + +def readOnlySetUpClass(modulePathName, pluginName, initializeStandalone=True): + ''' + Test class import setup for tests that do not write to the file system. + + This function: + - (Optionally) Initializes Maya standalone use. + - (Optionally) Loads the plugin + - Returns the original directory from the argument. + ''' + (testDir, testFile) = _setUpClass(modulePathName, pluginName, + initializeStandalone) + + return testDir + +def loadTestsFromDict(namespace_dict): + ''' + Returns a unittest.TestSuite object with tests loaded from the given dict + + Similar to unittest.TestLoader.loadTestsFromModule, but works off a dict + rather than a module object. Useful when running from inside a "script" + context where there is no module, but are globals(). + + Examples + -------- + >>> testSuite = loadTestsFromDict(globals()) + ''' + # just piggy-back on loadTestsFromModule, since all it does is dir() and + # getattr checks... + class DummyModule(object): + __name__ + + dummyModule = DummyModule() + + for name, val in namespace_dict.items(): + if not name.startswith('__'): + setattr(dummyModule, name, val) + return unittest.TestLoader().loadTestsFromModule(dummyModule) + +def runTests(globals_dict, stream=sys.__stderr__, + verbosity=1): + ''' + Run the unittests within the given namespace + + Intended usage: + import fixturesUtils + if __name__ == '__main__': + fixturesUtils.runTests(globals()) + ''' + import maya.cmds as cmds + suite = loadTestsFromDict(globals_dict) + runner = unittest.TextTestRunner(stream=stream, verbosity=verbosity) + results = runner.run(suite) + if results.wasSuccessful(): + exitCode = 0 + else: + exitCode = 1 + + # cmds.quit will not flush the streams - make sure we do so! + # ...flush all of the standard ones just to be sure, as well as the stream + # given (which probably means it will be flushed twice, but that's fine) + sys.stdout.flush() + sys.stderr.flush() + sys.__stdout__.flush() + sys.__stderr__.flush() + stream.flush() + + # maya running interactively will absorb much of the output. comment out the + # following to prevent maya from exiting and open the script editor to look + # at failures. + cmds.quit(abort=True, exitCode=exitCode) diff --git a/test/testUtils/imageUtils.py b/test/testUtils/imageUtils.py new file mode 100644 index 0000000000..cbdf360d07 --- /dev/null +++ b/test/testUtils/imageUtils.py @@ -0,0 +1,244 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import os +import maya.cmds as cmds +import maya.mel +import mayaUtils +import subprocess + +KNOWN_FORMATS = { + 'gif': 0, + 'tif': 3, + 'tiff': 3, + 'sgi': 5, + 'iff': 7, + 'jpg': 8, + 'jpeg': 8, + 'tga': 19, + 'bmp': 20, + 'png': 32, +} +def resetDefaultLightIntensity(): + """If the current Maya version supports setting the default light intensity, + then restore it to 1 so snapshots look equal across versions.""" + if maya.mel.eval("optionVar -exists defaultLightIntensity"): + maya.mel.eval("optionVar -fv defaultLightIntensity 1") + if cmds.attributeQuery('defaultLightIntensity', node='hardwareRenderingGlobals', exists=True): + cmds.setAttr('hardwareRenderingGlobals.defaultLightIntensity', 1.0) +resetDefaultLightIntensity() + +def snapshot(outputPath, width=400, height=None, hud=False, grid=False, camera=None): + resetDefaultLightIntensity() + cmds.displayRGBColor('background', 0.36, 0.36, 0.36) + + if height is None: + height = width + + outputExt = os.path.splitext(outputPath)[1].lower().lstrip('.') + + formatNum = KNOWN_FORMATS.get(outputExt) + if formatNum is None: + raise ValueError("input image had unrecognized extension: {}" + .format(outputExt)) + + # if given relative path, make it relative to current dir (the test + # temp base), rather than the workspace dir + outputPath = os.path.abspath(outputPath) + + # save the old output image format + oldFormat = cmds.getAttr("defaultRenderGlobals.imageFormat") + # save the hud setting + oldHud = cmds.headsUpDisplay(q=1, layoutVisibility=1) + # save the grid setting + oldGrid = cmds.grid(q=1, toggle=1) + # save the old view transform + oldColorTransform = cmds.colorManagementPrefs(q=1, outputTarget="playblast", + outputTransformName=1) + + # Some environments use legacy synColor transforms with 2022 and above. + # Find whether the color config should be Raw or Raw legacy + # However depending on the MAYA_COLOR_MANAGEMENT_SYNCOLOR env var or the loaded + # configs, this may be under a different names. So procedurally find it. + colorTransforms = cmds.colorManagementPrefs(q=1, outputTransformNames=True) + if "Raw" in colorTransforms: + newColorTransform = "Raw" + elif "Raw (legacy)" in colorTransforms: + newColorTransform = "Raw (legacy)" + else: + # RAW should be reliably raw-like in most configs, so find the first ending in RAW + newColorTransform = [c for c in colorTransforms if c.startswith("Raw ")] + if newColorTransform: + newColorTransform = newColorTransform[0] + else: + raise RuntimeError("Could not find Raw color space in available color transforms") + + # Some environments have locked color policies that prevent changing color policies + # so we must disable and restore this accordingly. + lockedColorTransforms = os.environ.get("MAYA_COLOR_MANAGEMENT_POLICY_LOCK") == '1' + if lockedColorTransforms: + os.environ['MAYA_COLOR_MANAGEMENT_POLICY_LOCK'] = '0' + + + # Find the current model panel for playblasting + # to make sure the desired camera is set, if any + panel = mayaUtils.activeModelPanel() + oldCamera = cmds.modelPanel(panel, q=True, cam=True) + if camera: + cmds.modelEditor(panel, edit=True, camera=camera) + + + # do these in a separate try/finally from color management, because + # color management seems a bit more finicky + cmds.setAttr("defaultRenderGlobals.imageFormat", formatNum) + cmds.headsUpDisplay(layoutVisibility=hud) + cmds.grid(toggle=grid) + try: + cmds.colorManagementPrefs(e=1, outputTarget="playblast", + outputTransformName=newColorTransform) + try: + cmds.refresh() + cmds.playblast(cf=outputPath, viewer=False, format="image", + frame=cmds.currentTime(q=1), offScreen=1, + widthHeight=(width, height), percent=100) + finally: + cmds.colorManagementPrefs(e=1, outputTarget="playblast", + outputTransformName=oldColorTransform) + finally: + cmds.setAttr("defaultRenderGlobals.imageFormat", oldFormat) + cmds.headsUpDisplay(layoutVisibility=oldHud) + cmds.grid(toggle=oldGrid) + if lockedColorTransforms: + os.environ['MAYA_COLOR_MANAGEMENT_POLICY_LOCK'] = '1' + + if camera: + cmds.lookThru(panel, oldCamera) + +def imageDiff(imagePath1, imagePath2, verbose, fail, failpercent, hardfail, + warn, warnpercent, hardwarn, perceptual): + """ Returns the completed process instance after running idiff. + + imagePath1 -- First image to compare. + imagePath2 -- Second image to compare. + verbose -- If enabled, the image diffing command will be printed to log. + fail -- The threshold for the acceptable difference (relatively to the mean of + the two values) of a pixel for failure. + failpercent -- The percentage of pixels that can be different before failure. + hardfail -- Triggers a failure if any pixels are above this threshold (if the absolute + difference is below this threshold). + warn -- The threshold for the acceptable difference of a pixel for a warning. + warnpercent -- The percentage of pixels that can be different before a warning. + hardwarn -- Triggers a warning if any pixels are above this threshold. + perceptual -- Performs an additional test to see if two images are visually different. + If enabled, test overall will fail if more than the "fail percentage" failed + the perceptual test. + + By default, if any pixels differ between the images, the comparison will fail. + If, for example, we set fail=0.004, failpercent=10 and hardfail=0.25, the comparison will + fail if more than 10% of the pixels differ by 0.004, or if any pixel differs by more than + 0.25 (just above a 1/255 threshold). + + For more information, see https://github.com/OpenImageIO/oiio/blob/cb6475c0dd72b9c49d862d98c6cd2da4509d5f37/src/doc/idiff.rst#L1 + """ + import platform + + imageDiff = os.environ['IMAGE_DIFF_TOOL'] + + cmdArgs = [] + if warn is not None: + cmdArgs.extend(['-warn', str(warn)]) + if warnpercent is not None: + cmdArgs.extend(['-warnpercent', str(warnpercent)]) + if hardwarn is not None: + cmdArgs.extend(['-hardwarn', str(hardwarn)]) + if fail is not None: + cmdArgs.extend(['-fail', str(fail)]) + if failpercent is not None: + cmdArgs.extend(['-failpercent', str(failpercent)]) + if hardfail is not None: + cmdArgs.extend(['-hardfail', str(hardfail)]) + if perceptual: + cmdArgs.extend(['-p']) + cmd = [imageDiff] + cmd.extend(cmdArgs) + cmd.extend([imagePath1, imagePath2]) + + if verbose: + import sys + sys.__stdout__.write("\nimage diffing with {0}".format(cmd)) + sys.__stdout__.flush() + + # LD_LIBRARY_PATH(or PATH or DYLD_LIBRARY_PATH) needs to be set for the idiff executable because its + # RPATH is absolute rather than relative to ORIGIN, meaning the RPATH + # points to the absolute path on the machine where idiff was built. + # This absence of relative paths for RPATH comes from OpenImageIO. + # We introduce a second workaround to avoid Maya using usd's libpng, + # because both use incompatible versions of libpng. This is done by + # setting LD_LIBRARY_PATH to IDIFF_LD_LIBRARY_PATH only when we run + # idiff using Python's subprocess module. + + if sys.platform == "darwin": + os.environ["DYLD_LIBRARY_PATH"] = os.environ['IDIFF_LD_LIBRARY_PATH'] + elif sys.platform.startswith("linux"): + os.environ["LD_LIBRARY_PATH"] = os.environ['IDIFF_LD_LIBRARY_PATH'] + + # Run idiff command + proc = subprocess.run(cmd, shell=False, env=os.environ.copy(), stdout=subprocess.PIPE) + return proc + +class ImageDiffingTestCase: + '''Mixin class for unit tests that require image comparison.''' + + def assertImagesClose(self, imagePath1, imagePath2, fail, failpercent, hardfail=None, + warn=None, warnpercent=None, hardwarn=None, perceptual=False): + """ + The method will return idiff's return code if the comparison passes with + a return code of 0 or 1. + 0 -- OK: the images match within the warning and error thresholds. + 1 -- Warning: the errors differ a little, but within error thresholds. + + The assertion will fail if the return code is 2, 3 or 4. + 2 -- Failure: the errors differ a lot, outside error thresholds. + 3 -- The images were not the same size and could not be compared. + 4 -- File error: could not find or open input files, etc. + """ + + proc = imageDiff(imagePath1, imagePath2, verbose=True, + fail=fail, failpercent=failpercent, hardfail=hardfail, + warn=warn, warnpercent=warnpercent, hardwarn=hardwarn, + perceptual=perceptual) + if proc.returncode not in (0, 1): + self.fail(str(proc.stdout)) + return proc.returncode + + def assertImagesEqual(self, imagePath1, imagePath2): + self.assertImagesClose(imagePath1, imagePath2, fail=None, failpercent=None) + + def assertSnapshotClose(self, refImage, fail, failpercent, hardfail=None, + warn=None, warnpercent=None, hardwarn=None, perceptual=False): + snapDir = os.path.join(os.path.abspath('.'), self._testMethodName) + if not os.path.isdir(snapDir): + os.makedirs(snapDir) + snapImage = os.path.join(snapDir, os.path.basename(refImage)) + snapshot(snapImage) + + return self.assertImagesClose(refImage, snapImage, + fail=fail, failpercent=failpercent, hardfail=hardfail, + warn=warn, warnpercent=warnpercent, hardwarn=hardwarn, + perceptual=perceptual) + + def assertSnapshotEqual(self, refImage): + '''Use of this method is discouraged, as renders can vary slightly between renderer architectures.''' + return self.assertSnapshotClose(refImage, fail=None, failpercent=None) diff --git a/test/testUtils/mayaUtils.py b/test/testUtils/mayaUtils.py new file mode 100644 index 0000000000..49132e2136 --- /dev/null +++ b/test/testUtils/mayaUtils.py @@ -0,0 +1,330 @@ +#!/usr/bin/env python + +# +# Copyright 2019 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +""" + Helper functions regarding Maya that will be used throughout the test. +""" + +from math import radians + +try: + from mayaUsd import lib as mayaUsdLib + from mayaUsd import ufe as mayaUsdUfe +except: + pass + +from maya import cmds +from maya.api import OpenMaya as om + +import ufe +import ufeUtils, testUtils + +import os +import re +import sys + +mayaSeparator = "|" + +prRe = re.compile('Preview Release ([0-9]+)') + +def loadPlugin(pluginName): + """ + Load all given plugins created or needed by maya-ufe-plugin + Args: + pluginName (str): The plugin name to load + Returns: + True if all plugins are loaded. False if a plugin failed to load + """ + try: + if not isPluginLoaded(pluginName): + cmds.loadPlugin( pluginName, quiet = True ) + return True + except: + print(sys.exc_info()[1]) + print("Unable to load %s" % pluginName) + return False + +def isPluginLoaded(pluginName): + """ + Verifies that the given plugin is loaded + Args: + pluginName (str): The plugin name to verify + Returns: + True if the plugin is loaded. False if a plugin failed to load + """ + return cmds.pluginInfo( pluginName, loaded=True, query=True) + +def isMayaUsdPluginLoaded(): + """ + Load plugins needed by UFE tests. + Returns: + True if plugins loaded successfully. False if a plugin failed to load + """ + # Load the mayaUsdPlugin first. + if not loadPlugin("mayaUsdPlugin"): + return False + + # Load the UFE support plugin, for ufeSelectCmd support. If this plugin + # isn't included in the distribution of Maya (e.g. Maya 2019 or 2020), use + # fallback test plugin. + if not (loadPlugin("ufeSupport") or loadPlugin("ufeTestCmdsPlugin")): + return False + + # The renderSetup Python plugin registers a file new callback to Maya. On + # test application exit (in TbaseApp::cleanUp()), a file new is done and + # thus the file new callback is invoked. Unfortunately, this occurs after + # the Python interpreter has been finalized, which causes a crash. Since + # renderSetup is not needed for mayaUsd tests, unload it. + rs = 'renderSetup' + if cmds.pluginInfo(rs, q=True, loaded=True): + unloaded = cmds.unloadPlugin(rs) + return (unloaded[0] == rs) + + return True + +def createUfePathSegment(mayaPath): + """ + Create a UFE path from a given maya path and return the first segment. + Args: + mayaPath (str): The maya path to use + Returns : + PathSegment of the given mayaPath + """ + if ufeUtils.ufeFeatureSetVersion() >= 2: + return ufe.PathString.path(mayaPath).segments[0] + else: + if not mayaPath.startswith("|world"): + mayaPath = "|world" + mayaPath + return ufe.PathSegment(mayaPath, mayaUsdUfe.getMayaRunTimeId(), + mayaSeparator) + +def getMayaSelectionList(): + """ + Returns the current Maya selection in a list + Returns: + A list(str) containing all selected Maya items + """ + # Remove the unicode of cmds.ls + + # TODO: HS, June 10, 2020 investigate why x needs to be encoded + if sys.version_info[0] == 2: + return [x.encode('UTF8') for x in cmds.ls(sl=True)] + else: + return [x for x in cmds.ls(sl=True)] + +def openTestScene(*args): + filePath = testUtils.getTestScene(*args) + cmds.file(filePath, force=True, open=True) + +def openTopLayerScene(): + ''' + The test scene hierarchy is represented as : + |world + |pSphere1 + |pSphereShape1 + |transform1 + |proxyShape1 + /Room_set + /Props + /Ball_1 + /Ball_2 + ... + /Ball_35 + ''' + # Open top_layer file which contains the USD scene + return openTestScene("ballset", "StandaloneScene", "top_layer.ma" ) + +def openCylinderScene(): + return openTestScene("cylinder", "usdCylinder.ma" ) + +def openTwoSpheresScene(): + return openTestScene("twoSpheres", "twoSpheres.ma" ) + +def openSphereAnimatedRadiusScene(): + return openTestScene("sphereAnimatedRadius", "sphereAnimatedRadiusProxyShape.ma" ) + +def openTreeScene(): + return openTestScene("tree", "tree.ma" ) + +def openTreeRefScene(): + return openTestScene("tree", "treeRef.ma" ) + +def openAppleBiteScene(): + return openTestScene("appleBite", "appleBite.ma" ) + +def openGroupBallsScene(): + return openTestScene("groupBalls", "ballset.ma" ) + +def openPrimitivesScene(): + return openTestScene("reorderCmd", "primitives.ma" ) + +def openPointInstancesGrid14Scene(): + return openTestScene("pointInstances", "PointInstancer_Grid_14.ma" ) + +def openPointInstancesGrid7kScene(): + return openTestScene("pointInstances", "PointInstancer_Grid_7k.ma" ) + +def openPointInstancesGrid70kScene(): + return openTestScene("pointInstances", "PointInstancer_Grid_70k.ma" ) + +def openVariantSetScene(): + return openTestScene("variantSet", "Variant.ma" ) + +def openCompositionArcsScene(): + return openTestScene("compositionArcs", "compositionArcs.ma" ) + +def openPrimPathScene(): + return openTestScene("primPath", "primPath.ma" ) + +def setMayaTranslation(aMayaItem, t): + '''Set the translation on the argument Maya scene item.''' + + aMayaPath = aMayaItem.path() + aMayaPathStr = ufe.PathString.string(aMayaPath) + aDagPath = om.MSelectionList().add(aMayaPathStr).getDagPath(0) + aFn= om.MFnTransform(aDagPath) + aFn.setTranslation(t, om.MSpace.kObject) + return (aMayaPath, aMayaPathStr, aFn, aFn.transformation().asMatrix()) + +def setMayaRotation(aMayaItem, r): + '''Set the rotation (XYZ) on the argument Maya scene item.''' + + aMayaPath = aMayaItem.path() + aMayaPathStr = ufe.PathString.string(aMayaPath) + aDagPath = om.MSelectionList().add(aMayaPathStr).getDagPath(0) + aFn = om.MFnTransform(aDagPath) + rads = [ radians(v) for v in r ] + rot = om.MEulerRotation(rads[0], rads[1], rads[2]) + aFn.setRotation(rot, om.MSpace.kTransform) + return (aMayaPath, aMayaPathStr, aFn, aFn.transformation().asMatrix()) + +def createProxyAndStage(): + """ + Create in-memory stage + """ + cmds.createNode('mayaUsdProxyShape', name='stageShape') + + shapeNode = cmds.ls(sl=True,l=True)[0] + shapeStage = mayaUsdLib.GetPrim(shapeNode).GetStage() + + cmds.select( clear=True ) + cmds.connectAttr('time1.outTime','{}.time'.format(shapeNode)) + + return shapeNode,shapeStage + +def createProxyFromFile(filePath): + """ + Load stage from file + """ + cmds.createNode('mayaUsdProxyShape', name='stageShape') + + shapeNode = cmds.ls(sl=True,l=True)[0] + cmds.setAttr('{}.filePath'.format(shapeNode), filePath, type='string') + + shapeStage = mayaUsdLib.GetPrim(shapeNode).GetStage() + + cmds.select( clear=True ) + cmds.connectAttr('time1.outTime','{}.time'.format(shapeNode)) + + return shapeNode,shapeStage + +def createSingleSphereMayaScene(directory=None): + '''Create a Maya scene with a single polygonal sphere. + Returns the file path. + ''' + + cmds.file(new=True, force=True) + cmds.CreatePolygonSphere() + tempMayaFile = 'simpleSphere.ma' + if directory is not None: + tempMayaFile = os.path.join(directory, tempMayaFile) + # Prevent Windows single backslash from being interpreted as a control + # character. + tempMayaFile = tempMayaFile.replace(os.sep, '/') + cmds.file(rename=tempMayaFile) + cmds.file(save=True, force=True, type='mayaAscii') + return tempMayaFile + +def previewReleaseVersion(): + '''Return the Maya Preview Release version. + + If the version of Maya is 2019, returns 98. + + If the version of Maya is 2020, returns 110. + + If the version of Maya is 2022, returns 122. + + If the version of Maya is current and is not a Preview Release, returns + sys.maxsize (a very large number). If the environment variable + MAYA_PREVIEW_RELEASE_VERSION_OVERRIDE is defined, return its value instead. + ''' + + if 'MAYA_PREVIEW_RELEASE_VERSION_OVERRIDE' in os.environ: + return int(os.environ['MAYA_PREVIEW_RELEASE_VERSION_OVERRIDE']) + + majorVersion = int(cmds.about(majorVersion=True)) + if majorVersion == 2019: + return 98 + elif majorVersion == 2020: + return 110 + elif majorVersion == 2022: + return 122 + + match = prRe.match(cmds.about(v=True)) + + return int(match.group(1)) if match else sys.maxsize + +def mayaMajorVersion(): + return int(cmds.about(majorVersion=True)) + +def mayaMinorVersion(): + return int(cmds.about(minorVersion=True)) + +def mayaMajorMinorVersions(): + """ + Return the Maya version as a tuple (Major, Minor). + Thanks to Python tuple comparison rules, (2022, 0) > (2021,3). + """ + return (mayaMajorVersion(), mayaMinorVersion()) + +def ufeSupportFixLevel(): + ''' + Return the fix level defined in the UFE support package. This is used + to determine the presence of a UFE-related feature or bug fix in Maya that + does not depend on a version of UFE itself. + ''' + import maya.internal.ufeSupport.utils as ufeSupportUtils + return ufeSupportUtils.fixLevel() if hasattr(ufeSupportUtils, 'fixLevel') \ + else 0 + +def hydraFixLevel(): + ''' + Return the Hydra fix level defined in Maya. + + This is used to determine the presence of a Hydra-related feature or bug + fix in Maya. + ''' + import maya.internal.ufeSupport.utils as ufeSupportUtils + return ufeSupportUtils.hydraFixLevel() if hasattr(ufeSupportUtils, 'hydraFixLevel') \ + else 0 + +def activeModelPanel(): + """Return the model panel that will be used for playblasting etc...""" + for panel in cmds.getPanel(type="modelPanel"): + if cmds.modelEditor(panel, q=1, av=1): + return panel diff --git a/test/testUtils/mtohUtils.py b/test/testUtils/mtohUtils.py new file mode 100644 index 0000000000..6f6bc666c0 --- /dev/null +++ b/test/testUtils/mtohUtils.py @@ -0,0 +1,181 @@ +# Copyright 2020 Luma Pictures +# Copyright 2023 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +import inspect +import os +import unittest + +import maya.cmds as cmds +import maya.mel + +import fixturesUtils +import testUtils +from imageUtils import ImageDiffingTestCase + +import sys + +HD_STORM = "HdStormRendererPlugin" +HD_STORM_OVERRIDE = "mayaHydraRenderOverride_" + HD_STORM + +def checkForPlugin(pluginName: str): + try: + cmds.loadPlugin(pluginName) + except: + return False + return True + +def checkForMayaUsdPlugin(): + return checkForPlugin('mayaUsdPlugin') + +class MayaHydraBaseTestCase(unittest.TestCase): + '''Base class for mayaHydra unit tests without image comparison.''' + + _file = None + + DEFAULT_CAM_DIST = 24 + + @classmethod + def setUpClass(cls): + if cls._file is None: + raise ValueError("Subclasses of MayaHydraBaseTestCase must " + "define `_file = __file__`") + fixturesUtils.readOnlySetUpClass(cls._file, 'mayaHydra', + initializeStandalone=False) + + def setHdStormRenderer(self): + self.activeEditor = cmds.playblast(activeEditor=1) + cmds.modelEditor( + self.activeEditor, e=1, + rendererOverrideName=HD_STORM_OVERRIDE) + cmds.refresh(f=1) + self.delegateId = cmds.mayaHydra(renderer=HD_STORM, + sceneDelegateId="MayaHydraSceneDelegate") + + def setViewport2Renderer(self): + self.activeEditor = cmds.playblast(activeEditor=1) + # Empty string for rendererOverrideName unsets any currently active override, thus returning to VP2 + cmds.modelEditor(self.activeEditor, e=1, rendererOverrideName="") + cmds.refresh(f=1) + self.delegateId = "" + + def setBasicCam(self, dist=DEFAULT_CAM_DIST): + cmds.setAttr('persp.rotate', -30, 45, 0, type='float3') + cmds.setAttr('persp.translate', dist, .75 * dist, dist, type='float3') + + def makeCubeScene(self, camDist=DEFAULT_CAM_DIST): + cmds.file(f=1, new=1) + self.cubeTrans = cmds.polyCube()[0] + self.cubeShape = cmds.listRelatives(self.cubeTrans)[0] + self.setHdStormRenderer() + self.assertNodeNameInIndex(self.cubeShape) + # The single Maya cube shape maps to two rprims, the first once of + # which is the shape's StandardShadedItem. The list is ordered, as the + # Hydra call made is HdRenderIndex::GetRprimIds(), which sorts + # according to std::less, which will produce + # lexicographically-ordered paths. + self.cubeRprim = self.getIndex()[0] + cmds.select(clear=1) + cmds.refresh() + self.assertVisible(self.cubeRprim) + self.setBasicCam(dist=camDist) + cmds.select(clear=True) + + # The color and specular roughness of the default standard surface changed, set + # them back to the old default value so the tests keep on working correctly. + if maya.mel.eval("defaultShaderName") == "standardSurface1": + color = (0.8, 0.8, 0.8) + cmds.setAttr("standardSurface1.baseColor", type='float3', *color) + cmds.setAttr("standardSurface1.specularRoughness", 0.4) + + def getIndex(self, **kwargs): + return cmds.mayaHydra(renderer=HD_STORM, listRenderIndex=True, **kwargs) + + def getVisibleIndex(self, **kwargs): + kwargs['visibleOnly'] = True + return self.getIndex(**kwargs) + + def assertVisible(self, rprim): + self.assertIn(rprim, self.getVisibleIndex()) + + def assertInIndex(self, rprim): + self.assertIn(rprim, self.getIndex()) + + def assertNodeNameInIndex(self, nodeName): + for rprim in self.getIndex(): + if nodeName in rprim: + return True + return False + + def trace(self, msg): + sys.__stdout__.write(msg) + sys.__stdout__.flush() + + def traceIndex(self, msg): + self.trace(msg.format(str(self.getIndex()))) + +class MtohTestCase(MayaHydraBaseTestCase, ImageDiffingTestCase): + '''Base class for mayaHydra unit tests with image comparison.''' + + _inputDir = None + + @classmethod + def setUpClass(cls): + if cls._file is None: + raise ValueError("Subclasses of MtohTestCase, must define " + "`_file = __file__`") + + inputPath = fixturesUtils.setUpClass( + cls._file, 'mayaHydra', initializeStandalone=False, + suffix=('_' + cls.__name__)) + + if cls._inputDir is None: + inputDirName = os.path.splitext(os.path.basename(cls._file))[0] + inputDirName = testUtils.stripPrefix(inputDirName, 'test') + if not inputDirName.endswith('Test'): + inputDirName += 'Test' + cls._inputDir = os.path.join(inputPath, inputDirName) + + cls._testDir = os.path.abspath('.') + + def resolveRefImage(self, refImage, imageVersion): + if not os.path.isabs(refImage): + if imageVersion: + refImage = os.path.join(self._inputDir, imageVersion, refImage) + else: + refImage = os.path.join(self._inputDir, refImage) + return refImage + + def assertImagesClose(self, image1, image2, fail, failpercent, image1Version=None, image2Version=None, + hardfail=None, warn=None, warnpercent=None, hardwarn=None, perceptual=False): + imagePath1 = self.resolveRefImage(image1, image1Version) + imagePath2 = self.resolveRefImage(image2, image2Version) + super(MtohTestCase, self).assertImagesClose(imagePath1, imagePath2, fail, failpercent, hardfail, + warn, warnpercent, hardwarn, perceptual) + + def assertImagesEqual(self, image1, image2, image1Version=None, image2Version=None): + imagePath1 = self.resolveRefImage(image1, image1Version) + imagePath2 = self.resolveRefImage(image2, image2Version) + super(MtohTestCase, self).assertImagesEqual(imagePath1, imagePath2) + + def assertSnapshotClose(self, refImage, fail, failpercent, imageVersion=None, hardfail=None, + warn=None, warnpercent=None, hardwarn=None, perceptual=False): + refImage = self.resolveRefImage(refImage, imageVersion) + super(MtohTestCase, self).assertSnapshotClose(refImage, fail, failpercent, hardfail, + warn, warnpercent, hardwarn, perceptual) + + def assertSnapshotEqual(self, refImage, imageVersion=None): + '''Use of this method is discouraged, as renders can vary slightly between renderer architectures.''' + refImage = self.resolveRefImage(refImage, imageVersion) + super(MtohTestCase, self).assertSnapshotEqual(refImage) diff --git a/test/testUtils/testUtils.py b/test/testUtils/testUtils.py new file mode 100644 index 0000000000..e638af3261 --- /dev/null +++ b/test/testUtils/testUtils.py @@ -0,0 +1,104 @@ +# +# Copyright 2019 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +""" + General test utilities. The functions here should not use Maya, Ufe or Usd. +""" + +import os +import shutil +import tempfile + +import maya.cmds as cmds + +def stripPrefix(input_str, prefix): + if input_str.startswith(prefix): + return input_str[len(prefix):] + return input_str + +def assertMatrixAlmostEqual(testCase, ma, mb, places=7): + for ra, rb in zip(ma, mb): + for a, b in zip(ra, rb): + testCase.assertAlmostEqual(a, b, places) + +def assertVectorAlmostEqual(testCase, a, b, places=7): + for va, vb in zip(a, b): + testCase.assertAlmostEqual(va, vb, places) + +def assertVectorNotAlmostEqual(testCase, a, b, places=7): + for va, vb in zip(a, b): + testCase.assertNotAlmostEqual(va, vb, places) + +def assertVectorEqual(testCase, a, b): + for va, vb in zip(a, b): + testCase.assertEqual(va, vb) + +def getTestScene(*args): + return os.path.join(os.path.dirname(os.path.realpath(__file__)), "..", "testSamples", *args) + +class TemporaryDirectory: + ''' + Context manager that creates a temporary directory and deletes it on exit, + so it's usable with "with" statement. + ''' + def __init__(self, suffix=None, prefix=None, ignore_errors=True, keep_files=False): + # Note: the default for suffix and prefix changed between Maya 2.7 and 3.X + # from empty strings to None. To be compatible with both, we won't + # pass values when we want the default, so we have these awkward if elif. + if suffix and prefix: + self.name = tempfile.mkdtemp(suffix=suffix, prefix=prefix) + elif suffix: + self.name = tempfile.mkdtemp(suffix=suffix) + elif prefix: + self.name = tempfile.mkdtemp(prefix=prefix) + else: + self.name = tempfile.mkdtemp() + self.ignore_errors = ignore_errors + self.keep_files = keep_files + + def __enter__(self): + return self.name + + def __exit__(self, exc_type, exc_value, traceback): + if self.keep_files: + return + try: + shutil.rmtree(self.name) + except: + if not self.ignore_errors: + raise + +class PluginLoaded: + ''' + Context manager that ensures a plugin is loaded, unloading it if it had not previously been loaded. + ''' + def __init__(self, name): + self.name = name + self.wasLoaded = cmds.pluginInfo(self.name, q=True, loaded=True) + + def __enter__(self): + '''Returns whether the plugin required loading.''' + if not self.wasLoaded: + cmds.loadPlugin(self.name, quiet=True) + return True + return False + + def __exit__(self, exc_type, exc_value, traceback): + if not self.wasLoaded: + # Clean out the scene to allow plugin to unload cleanly. + cmds.file(new=True, force=True) + cmds.unloadPlugin(self.name) + diff --git a/test/testUtils/ufeScripts/__init__.py b/test/testUtils/ufeScripts/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/test/testUtils/ufeScripts/ufeSelectCmd.py b/test/testUtils/ufeScripts/ufeSelectCmd.py new file mode 100644 index 0000000000..cc697f5104 --- /dev/null +++ b/test/testUtils/ufeScripts/ufeSelectCmd.py @@ -0,0 +1,302 @@ +#!/usr/bin/env python + +# +# Copyright 2019 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +#- +# =========================================================================== +# WARNING: PROTOTYPE CODE +# +# The code in this file is intended as an engineering prototype, to demonstrate +# UFE integration in Maya. Its performance and stability may make it +# unsuitable for production use. +# +# Autodesk believes production quality would be better achieved with a C++ +# version of this code. +# +# =========================================================================== +#+ + +"""Maya UFE selection command. + +Operations in Maya are performed on the selection. This imposes the +requirement that selection in Maya must be an undoable operation, so that +operations after an undo have the proper selection as input. + +Maya selections done through non-UFE UIs already go through the undoable select +command. Selections done through UFE UIs must use the undoable command in this +module. +""" + +# Code in this module was initially in the mayaUfe module, which defines +# the initialization and finalization functions for the mayaUfe plugin. +# This confusingly caused two instances of each command class to exist (as +# demonstrated by different values of id() on the class), one used by the +# command creator, and the other by the rest of the Python code. +# Separating out the command code into this module fixed the issue. PPT, +# 26-Jan-2018. + +import maya.api.OpenMaya as OpenMaya + +import maya.cmds as cmds + +import ufe.PyUfe as PyUfe + +# Messages should be localized. +kUfeSelectCmdPrivate = 'Private UFE select command %s arguments not set up.' + +def append(item): + return SelectAppendCmd.execute(item) + +def remove(item): + return SelectRemoveCmd.execute(item) + +def clear(): + SelectClearCmd.execute() + +def replaceWith(selection): + SelectReplaceWithCmd.execute(selection) + +#============================================================================== +# CLASS SelectCmdBase +#============================================================================== + +class SelectCmdBase(OpenMaya.MPxCommand): + """Base class for UFE selection commands. + + This command is intended as a base class for concrete UFE select commands. + """ + + def __init__(self): + super(SelectCmdBase, self).__init__() + self.globalSelection = PyUfe.GlobalSelection.get() + + def isUndoable(self): + return True + + def validateArgs(self): + return True + + def doIt(self, args): + # Completely ignore the MArgList argument, as it's unnecessary: + # arguments to the commands are passed in Python object form + # directly to the command's constructor. + + if self.validateArgs() is False: + self.displayWarning(kUfeSelectCmdPrivate % self.kCmdName) + else: + self.redoIt() + +#============================================================================== +# CLASS SelectAppendCmd +#============================================================================== + +class SelectAppendCmd(SelectCmdBase): + """Append an item to the UFE selection. + + This command is a private implementation detail of this module and should + not be called otherwise.""" + + kCmdName = 'ufeSelectAppend' + + # Command data. Must be set before creating an instance of the command + # and executing it. + item = None + + # Command return data. Set by doIt(). + result = None + + @staticmethod + def execute(item): + """Append the item to the UFE selection, and add an entry to the + undo queue.""" + + # Would be nice to have a guard context to restore the class data + # to its previous value (which is None). Not obvious how to write + # a Python context manager for this, as Python simply binds names + # to objects in a scope. + SelectAppendCmd.item = item + cmds.ufeSelectAppend() + result = SelectAppendCmd.result + SelectAppendCmd.item = None + SelectAppendCmd.result = None + return result + + @staticmethod + def creator(): + return SelectAppendCmd(SelectAppendCmd.item) + + def __init__(self, item): + super(SelectAppendCmd, self).__init__() + self.item = item + self.result = None + + def validateArgs(self): + return self.item is not None + + def doIt(self, args): + super(SelectAppendCmd, self).doIt(args) + # Save the result out as a class member. + SelectAppendCmd.result = self.result + + def redoIt(self): + self.result = self.globalSelection.append(self.item) + + def undoIt(self): + if self.result: + self.globalSelection.remove(self.item) + +#============================================================================== +# CLASS SelectRemoveCmd +#============================================================================== + +class SelectRemoveCmd(SelectCmdBase): + """Append an item to the UFE selection. + + This command is a private implementation detail of this module and should + not be called otherwise.""" + + kCmdName = 'ufeSelectRemove' + + # Command data. Must be set before creating an instance of the command + # and executing it. + item = None + + # Command return data. Set by doIt(). + result = None + + @staticmethod + def execute(item): + """Remove the item from the UFE selection, and add an entry to the + undo queue.""" + + # See SelectAppendCmd.execute comments. + SelectRemoveCmd.item = item + cmds.ufeSelectRemove() + result = SelectRemoveCmd.result + SelectRemoveCmd.item = None + SelectRemoveCmd.result = None + return result + + @staticmethod + def creator(): + return SelectRemoveCmd(SelectRemoveCmd.item) + + def __init__(self, item): + super(SelectRemoveCmd, self).__init__() + self.item = item + self.result = None + + def validateArgs(self): + return self.item is not None + + def doIt(self, args): + super(SelectRemoveCmd, self).doIt(args) + # Save the result out as a class member. + SelectRemoveCmd.result = self.result + + def redoIt(self): + self.result = self.globalSelection.remove(self.item) + + def undoIt(self): + # This is not a true undo! Selection.remove removes an item regardless + # of its position in the list, but append places the item in the last + # position. Saving and restoring the complete list is O(n) for n + # elements in the list; we want an O(1) solution. Selection.remove + # should return the list position of the removed element, for undo O(1) + # re-insertion using a future Selection.insert(). In C++, this list + # position would be an iterator; a matching Python binding would most + # likely require custom pybind code. PPT, 26-Jan-2018. + if self.result: + self.globalSelection.append(self.item) + +#============================================================================== +# CLASS SelectClearCmd +#============================================================================== + +class SelectClearCmd(SelectCmdBase): + """Clear the UFE selection. + + This command is a private implementation detail of this module and should + not be called otherwise.""" + + kCmdName = 'ufeSelectClear' + + @staticmethod + def execute(): + """Clear the UFE selection, and add an entry to the undo queue.""" + cmds.ufeSelectClear() + + @staticmethod + def creator(): + return SelectClearCmd() + + def __init__(self): + super(SelectClearCmd, self).__init__() + self.savedSelection = None + + def redoIt(self): + self.savedSelection = self.globalSelection + self.globalSelection.clear() + + def undoIt(self): + self.globalSelection.replaceWith(self.savedSelection) + +#============================================================================== +# CLASS SelectReplaceWithCmd +#============================================================================== + +class SelectReplaceWithCmd(SelectCmdBase): + """Replace the UFE selection with a new selection. + + This command is a private implementation detail of this module and should + not be called otherwise.""" + + kCmdName = 'ufeSelectReplaceWith' + + # Command data. Must be set before creating an instance of the command + # and executing it. + selection = None + + @staticmethod + def execute(selection): + """Replace the UFE selection with a new selection, and add an entry to + the undo queue.""" + + # See SelectAppendCmd.execute comments. + SelectReplaceWithCmd.selection = selection + cmds.ufeSelectReplaceWith() + SelectReplaceWithCmd.selection = None + + @staticmethod + def creator(): + return SelectReplaceWithCmd(SelectReplaceWithCmd.selection) + + def __init__(self, selection): + super(SelectReplaceWithCmd, self).__init__() + self.selection = selection + self.savedSelection = None + + def redoIt(self): + # No easy way to copy a Selection, so create a new one and call + # replaceWith(). selection[:], copy.copy(selection), and + # copy.deepcopy(selection) all raise Python exceptions. + self.savedSelection = PyUfe.Selection() + self.savedSelection.replaceWith(self.globalSelection) + self.globalSelection.replaceWith(self.selection) + + def undoIt(self): + self.globalSelection.replaceWith(self.savedSelection) diff --git a/test/testUtils/ufeUtils.py b/test/testUtils/ufeUtils.py new file mode 100644 index 0000000000..547d291d98 --- /dev/null +++ b/test/testUtils/ufeUtils.py @@ -0,0 +1,107 @@ +#!/usr/bin/env python + +# +# Copyright 2022 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +""" + Helper functions regarding UFE that will be used throughout the test. +""" + +import ufe + + +def getUfeGlobalSelectionList(): + """ + Returns the current UFE selection in a list + Returns: + A list(str) containing all selected UFE items + """ + selection = [] + for item in ufe.GlobalSelection.get(): + selection.append(str(item.path().back())) + return selection + +def selectPath(path, replace=False): + """ + Select a path in the UFE Global selection + Args: + path (ufe.Path): The UFE path to select + replace (bool=False): Replace the selection with + given UFE path + """ + # Do this import inside this function rather than at the top of the file + # so that the module can be imported and ufeFeatureSetVersion() can be + # called prior to maya.standalone.initialize(). + try: + from maya.internal.ufeSupport import ufeSelectCmd + except ImportError: + # Maya 2019 and 2020 don't have ufeSupport plugin, so use fallback. + from ufeScripts import ufeSelectCmd + + sceneItem = ufe.Hierarchy.createItem(path) + if replace: + selection = ufe.Selection() + selection.append(sceneItem) + ufeSelectCmd.replaceWith(selection) + else: + ufeSelectCmd.append(sceneItem) + +def createUfeSceneItem(dagPath, sdfPath=None): + """ + Make ufe item out of dag path and sdfpath + """ + ufePath = ufe.PathString.path('{},{}'.format(dagPath,sdfPath) if sdfPath != None else '{}'.format(dagPath)) + ufeItem = ufe.Hierarchy.createItem(ufePath) + return ufeItem + +def createItem(ufePathOrPathString): + '''Create a UFE scene item from a UFE path or path string.''' + path = ufe.PathString.path(ufePathOrPathString) \ + if isinstance(ufePathOrPathString, str) else ufePathOrPathString + + return ufe.Hierarchy.createItem(path) + +def createHierarchy(ufePathOrPathStringOrItem): + '''Create a UFE scene item from a UFE path, path string, or scene item.''' + item = ufePathOrPathStringOrItem if isinstance(ufePathOrPathStringOrItem, ufe.SceneItem) else createItem(ufePathOrPathStringOrItem) + return ufe.Hierarchy.hierarchy(item) + +def selectUfeItems(selectItems): + """ + Add given UFE item or list of items to a UFE global selection list + """ + ufeSelectionList = ufe.Selection() + + realListToSelect = selectItems if type(selectItems) is list else [selectItems] + for item in realListToSelect: + ufeSelectionList.append(item) + + ufe.GlobalSelection.get().replaceWith(ufeSelectionList) + +def ufeFeatureSetVersion(): + '''Return the UFE feature set version taking into account rollback to using + 0 for unreleased version after we ship a UFE. + + If you need to test for a specific UFE preview release version then check + the cmake variable UFE_PREVIEW_VERSION_NUM. This is also set as an env + for the python tests. + ''' + + # Examples: + # v2.0.0 (released version 2). + # v0.2.20 (unreleased preview version 2). + major = ufe.VersionInfo.getMajorVersion() + return ufe.VersionInfo.getMinorVersion() if major == 0 else major diff --git a/test/testUtils/usdUtils.py b/test/testUtils/usdUtils.py new file mode 100644 index 0000000000..c1c75cc343 --- /dev/null +++ b/test/testUtils/usdUtils.py @@ -0,0 +1,181 @@ +#!/usr/bin/env python + +# +# Copyright 2019 Autodesk +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +""" + Helper functions regarding USD that will be used throughout the test. +""" + +import mayaUsd.ufe +import mayaUsd.lib +import mayaUsd_createStageWithNewLayer + +import ufe +import ufeUtils + +from pxr import Usd, UsdGeom, Gf, Sdf + +usdSeparator = '/' + + +def createUfePathSegment(usdPath): + """ + Create an UFE path from a given usd path. + Args: + usdPath (str): The usd path to use + Returns : + PathSegment of the given usdPath + """ + return ufe.PathSegment(usdPath, mayaUsd.ufe.getUsdRunTimeId(), usdSeparator) + +def getPrimFromSceneItem(item): + if ufeUtils.ufeFeatureSetVersion() >= 2: + rawItem = item.getRawAddress() + prim = mayaUsd.ufe.getPrimFromRawItem(rawItem) + return prim + else: + return Usd.Prim() + +def createAnimatedHierarchy(stage): + """ + Create simple hierarchy in the stage: + /ParentA + /Sphere + /Cube + /ParenB + + Entire ParentA hierarchy will receive time samples on translate for time 1 and 100 + """ + parentA = "/ParentA" + parentB = "/ParentB" + childSphere = "/ParentA/Sphere" + childCube = "/ParentA/Cube" + + parentPrimA = stage.DefinePrim(parentA, 'Xform') + parentPrimB = stage.DefinePrim(parentB, 'Xform') + childPrimSphere = stage.DefinePrim(childSphere, 'Sphere') + childPrimCube = stage.DefinePrim(childCube, 'Cube') + + UsdGeom.XformCommonAPI(parentPrimA).SetRotate((0,0,0)) + UsdGeom.XformCommonAPI(parentPrimB).SetTranslate((1,10,0)) + + time1 = Usd.TimeCode(1.) + UsdGeom.XformCommonAPI(parentPrimA).SetTranslate((0,0,0),time1) + UsdGeom.XformCommonAPI(childPrimSphere).SetTranslate((5,0,0),time1) + UsdGeom.XformCommonAPI(childPrimCube).SetTranslate((0,0,5),time1) + + time2 = Usd.TimeCode(100.) + UsdGeom.XformCommonAPI(parentPrimA).SetTranslate((0,5,0),time2) + UsdGeom.XformCommonAPI(childPrimSphere).SetTranslate((-5,0,0),time2) + UsdGeom.XformCommonAPI(childPrimCube).SetTranslate((0,0,-5),time2) + +def createSimpleStage(): + '''Create a simple stage and layer: + + Returns a tuple of: + - proxy shape UFE path string + - proxy shape UFE path + - proxy shape UFE item + ''' + psPathStr = mayaUsd_createStageWithNewLayer.createStageWithNewLayer() + psPath = ufe.PathString.path(psPathStr) + ps = ufe.Hierarchy.createItem(psPath) + return (psPathStr, psPath, ps) + +def createSimpleXformSceneInCurrentLayer(psPathStr, ps): + '''Create a simple scene in the current stage and layer with a trivial hierarchy: + + A translation (1, 2, 3) + |_B translation (7, 8, 9) + + Returns a tuple of: + - A translation op, A translation vector + - A UFE path string, A UFE path, A UFE item + - B translation op, B translation vector + - B UFE path string, B UFE path, B UFE item + ''' + + stage = mayaUsd.lib.GetPrim(psPathStr).GetStage() + aPrim = stage.DefinePrim('/A', 'Xform') + aXformable = UsdGeom.Xformable(aPrim) + aXlateOp = aXformable.AddTranslateOp() + aXlation = Gf.Vec3d(1, 2, 3) + aXlateOp.Set(aXlation) + aUsdUfePathStr = psPathStr + ',/A' + aUsdUfePath = ufe.PathString.path(aUsdUfePathStr) + aUsdItem = ufe.Hierarchy.createItem(aUsdUfePath) + + bPrim = stage.DefinePrim('/A/B', 'Xform') + bXformable = UsdGeom.Xformable(bPrim) + bXlateOp = bXformable.AddTranslateOp() + bXlation = Gf.Vec3d(7, 8, 9) + bXlateOp.Set(bXlation) + bUsdUfePathStr = aUsdUfePathStr + '/B' + bUsdUfePath = ufe.PathString.path(bUsdUfePathStr) + bUsdItem = ufe.Hierarchy.createItem(bUsdUfePath) + + return (aXlateOp, aXlation, aUsdUfePathStr, aUsdUfePath, aUsdItem, + bXlateOp, bXlation, bUsdUfePathStr, bUsdUfePath, bUsdItem) + +def createSimpleXformScene(): + '''Create a simple scene with a trivial hierarchy: + + A translation (1, 2, 3) + |_B translation (7, 8, 9) + + Returns a tuple of: + - proxy shape UFE item + - A translation op, A translation vector + - A UFE path string, A UFE path, A UFE item + - B translation op, B translation vector + - B UFE path string, B UFE path, B UFE item + + Note: the proxy shape path and path string are not returned for compatibility with existing tests. + ''' + (psPathStr, psPath, ps) = createSimpleStage() + (aXlateOp, aXlation, aUsdUfePathStr, aUsdUfePath, aUsdItem, + bXlateOp, bXlation, bUsdUfePathStr, bUsdUfePath, bUsdItem) = createSimpleXformSceneInCurrentLayer(psPathStr, ps) + return (ps, + aXlateOp, aXlation, aUsdUfePathStr, aUsdUfePath, aUsdItem, + bXlateOp, bXlation, bUsdUfePathStr, bUsdUfePath, bUsdItem) + +def createLayeredStage(layersCount = 3): + '''Create a stage with multiple layers, by default 3 extra layers: + + Returns a tuple of: + - proxy shape UFE path string + - proxy shape UFE path + - proxy shape UFE item + - list of root layer and additional layers, from top to bottom + ''' + (psPathStr, psPath, ps) = createSimpleStage() + + import os + print(os.environ) + stage = mayaUsd.lib.GetPrim(psPathStr).GetStage() + layer = stage.GetRootLayer() + layers = [layer] + for i in range(layersCount): + newLayerName = 'Layer_%d' % (i+1) + usdFormat = Sdf.FileFormat.FindByExtension('usd') + newLayer = Sdf.Layer.New(usdFormat, newLayerName) + layer.subLayerPaths.append(newLayer.identifier) + layer = newLayer + layers.append(layer) + + return (psPathStr, psPath, ps, layers) +