@plastic-software/three 0.182.0 → 0.183.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (299) hide show
  1. package/LICENSE +1 -1
  2. package/build/three.cjs +11521 -10878
  3. package/build/three.core.js +11732 -11340
  4. package/build/three.core.min.js +2 -2
  5. package/build/three.module.js +510 -263
  6. package/build/three.module.min.js +2 -2
  7. package/build/three.tsl.js +7 -11
  8. package/build/three.tsl.min.js +2 -2
  9. package/build/three.webgpu.js +3072 -2607
  10. package/build/three.webgpu.min.js +2 -2
  11. package/build/three.webgpu.nodes.js +3071 -2607
  12. package/build/three.webgpu.nodes.min.js +2 -2
  13. package/examples/jsm/Addons.js +0 -3
  14. package/examples/jsm/animation/CCDIKSolver.js +2 -2
  15. package/examples/jsm/controls/ArcballControls.js +3 -3
  16. package/examples/jsm/controls/OrbitControls.js +103 -0
  17. package/examples/jsm/effects/AnaglyphEffect.js +102 -7
  18. package/examples/jsm/environments/ColorEnvironment.js +59 -0
  19. package/examples/jsm/environments/RoomEnvironment.js +1 -0
  20. package/examples/jsm/exporters/EXRExporter.js +1 -1
  21. package/examples/jsm/exporters/GLTFExporter.js +131 -4
  22. package/examples/jsm/exporters/USDZExporter.js +22 -3
  23. package/examples/jsm/helpers/AnimationPathHelper.js +302 -0
  24. package/examples/jsm/helpers/ViewHelper.js +67 -8
  25. package/examples/jsm/inspector/Inspector.js +21 -5
  26. package/examples/jsm/inspector/tabs/Console.js +39 -5
  27. package/examples/jsm/inspector/tabs/Parameters.js +16 -0
  28. package/examples/jsm/inspector/ui/Style.js +25 -1
  29. package/examples/jsm/libs/meshopt_decoder.module.js +6 -5
  30. package/examples/jsm/lines/LineMaterial.js +6 -0
  31. package/examples/jsm/loaders/3MFLoader.js +2 -2
  32. package/examples/jsm/loaders/AMFLoader.js +2 -2
  33. package/examples/jsm/loaders/ColladaLoader.js +24 -4026
  34. package/examples/jsm/loaders/EXRLoader.js +5 -5
  35. package/examples/jsm/loaders/FBXLoader.js +2 -2
  36. package/examples/jsm/loaders/GCodeLoader.js +34 -8
  37. package/examples/jsm/loaders/GLTFLoader.js +122 -171
  38. package/examples/jsm/loaders/KMZLoader.js +5 -5
  39. package/examples/jsm/loaders/KTX2Loader.js +5 -5
  40. package/examples/jsm/loaders/LWOLoader.js +7 -39
  41. package/examples/jsm/loaders/NRRDLoader.js +2 -2
  42. package/examples/jsm/loaders/PCDLoader.js +3 -2
  43. package/examples/jsm/loaders/USDLoader.js +100 -40
  44. package/examples/jsm/loaders/UltraHDRLoader.js +182 -30
  45. package/examples/jsm/loaders/VRMLLoader.js +77 -0
  46. package/examples/jsm/loaders/VTKLoader.js +37 -24
  47. package/examples/jsm/loaders/collada/ColladaComposer.js +2950 -0
  48. package/examples/jsm/loaders/collada/ColladaParser.js +1962 -0
  49. package/examples/jsm/loaders/usd/USDAParser.js +447 -366
  50. package/examples/jsm/loaders/usd/USDCParser.js +1841 -6
  51. package/examples/jsm/loaders/usd/USDComposer.js +4041 -0
  52. package/examples/jsm/materials/LDrawConditionalLineNodeMaterial.js +2 -2
  53. package/examples/jsm/objects/LensflareMesh.js +1 -1
  54. package/examples/jsm/objects/Sky.js +76 -4
  55. package/examples/jsm/objects/SkyMesh.js +114 -7
  56. package/examples/jsm/objects/Water.js +4 -3
  57. package/examples/jsm/objects/Water2.js +5 -3
  58. package/examples/jsm/objects/WaterMesh.js +5 -7
  59. package/examples/jsm/physics/JoltPhysics.js +7 -5
  60. package/examples/jsm/physics/RapierPhysics.js +6 -4
  61. package/examples/jsm/postprocessing/EffectComposer.js +7 -5
  62. package/examples/jsm/postprocessing/RenderTransitionPass.js +1 -1
  63. package/examples/jsm/renderers/CSS3DRenderer.js +1 -1
  64. package/examples/jsm/renderers/SVGRenderer.js +2 -2
  65. package/examples/jsm/shaders/GTAOShader.js +19 -6
  66. package/examples/jsm/shaders/HalftoneShader.js +12 -1
  67. package/examples/jsm/shaders/PoissonDenoiseShader.js +6 -2
  68. package/examples/jsm/shaders/SAOShader.js +17 -4
  69. package/examples/jsm/shaders/SSAOShader.js +11 -1
  70. package/examples/jsm/shaders/SSRShader.js +6 -5
  71. package/examples/jsm/shaders/VignetteShader.js +1 -1
  72. package/examples/jsm/tsl/display/AfterImageNode.js +1 -1
  73. package/examples/jsm/tsl/display/AnaglyphPassNode.js +456 -16
  74. package/examples/jsm/tsl/display/AnamorphicNode.js +1 -1
  75. package/examples/jsm/tsl/display/BilateralBlurNode.js +364 -0
  76. package/examples/jsm/tsl/display/BloomNode.js +5 -5
  77. package/examples/jsm/tsl/display/CRT.js +150 -0
  78. package/examples/jsm/tsl/display/DenoiseNode.js +1 -1
  79. package/examples/jsm/tsl/display/DepthOfFieldNode.js +1 -1
  80. package/examples/jsm/tsl/display/DotScreenNode.js +1 -1
  81. package/examples/jsm/tsl/display/FXAANode.js +2 -2
  82. package/examples/jsm/tsl/display/GTAONode.js +2 -2
  83. package/examples/jsm/tsl/display/GaussianBlurNode.js +11 -2
  84. package/examples/jsm/tsl/display/GodraysNode.js +624 -0
  85. package/examples/jsm/tsl/display/LensflareNode.js +1 -1
  86. package/examples/jsm/tsl/display/Lut3DNode.js +1 -1
  87. package/examples/jsm/tsl/display/OutlineNode.js +3 -3
  88. package/examples/jsm/tsl/display/ParallaxBarrierPassNode.js +2 -2
  89. package/examples/jsm/tsl/display/PixelationPassNode.js +5 -5
  90. package/examples/jsm/tsl/display/RGBShiftNode.js +2 -2
  91. package/examples/jsm/tsl/display/RetroPassNode.js +263 -0
  92. package/examples/jsm/tsl/display/SMAANode.js +2 -2
  93. package/examples/jsm/tsl/display/SSAAPassNode.js +2 -2
  94. package/examples/jsm/tsl/display/SSGINode.js +2 -2
  95. package/examples/jsm/tsl/display/SSRNode.js +7 -7
  96. package/examples/jsm/tsl/display/SSSNode.js +2 -2
  97. package/examples/jsm/tsl/display/Shape.js +29 -0
  98. package/examples/jsm/tsl/display/SobelOperatorNode.js +2 -2
  99. package/examples/jsm/tsl/display/StereoPassNode.js +1 -2
  100. package/examples/jsm/tsl/display/TRAANode.js +9 -12
  101. package/examples/jsm/tsl/display/TransitionNode.js +1 -1
  102. package/examples/jsm/tsl/display/depthAwareBlend.js +80 -0
  103. package/examples/jsm/tsl/math/Bayer.js +40 -1
  104. package/examples/jsm/utils/LDrawUtils.js +1 -1
  105. package/package.json +11 -19
  106. package/src/Three.Core.js +1 -1
  107. package/src/Three.TSL.js +5 -9
  108. package/src/Three.WebGPU.Nodes.js +2 -0
  109. package/src/Three.WebGPU.js +3 -0
  110. package/src/Three.js +1 -0
  111. package/src/animation/AnimationAction.js +1 -1
  112. package/src/animation/AnimationClip.js +1 -1
  113. package/src/animation/AnimationMixer.js +6 -0
  114. package/src/animation/KeyframeTrack.js +46 -7
  115. package/src/animation/PropertyMixer.js +4 -4
  116. package/src/audio/Audio.js +1 -1
  117. package/src/audio/AudioListener.js +5 -3
  118. package/src/cameras/Camera.js +32 -2
  119. package/src/cameras/CubeCamera.js +20 -0
  120. package/src/constants.js +30 -1
  121. package/src/core/Clock.js +7 -0
  122. package/src/core/Object3D.js +56 -4
  123. package/src/core/RenderTarget.js +3 -4
  124. package/src/extras/PMREMGenerator.js +4 -8
  125. package/src/geometries/TorusGeometry.js +8 -3
  126. package/src/helpers/CameraHelper.js +3 -0
  127. package/src/helpers/DirectionalLightHelper.js +4 -1
  128. package/src/helpers/HemisphereLightHelper.js +3 -0
  129. package/src/helpers/PointLightHelper.js +0 -24
  130. package/src/helpers/SpotLightHelper.js +3 -0
  131. package/src/lights/LightShadow.js +15 -3
  132. package/src/lights/webgpu/IESSpotLight.js +2 -1
  133. package/src/loaders/Cache.js +28 -0
  134. package/src/loaders/FileLoader.js +1 -1
  135. package/src/loaders/ImageBitmapLoader.js +8 -3
  136. package/src/loaders/Loader.js +6 -0
  137. package/src/loaders/ObjectLoader.js +18 -1
  138. package/src/materials/MeshLambertMaterial.js +9 -0
  139. package/src/materials/MeshPhongMaterial.js +9 -0
  140. package/src/materials/nodes/Line2NodeMaterial.js +5 -5
  141. package/src/materials/nodes/MeshPhysicalNodeMaterial.js +2 -0
  142. package/src/materials/nodes/NodeMaterial.js +15 -24
  143. package/src/materials/nodes/manager/NodeMaterialObserver.js +9 -3
  144. package/src/math/Line3.js +3 -5
  145. package/src/math/MathUtils.js +10 -10
  146. package/src/math/Matrix4.js +35 -26
  147. package/src/math/Quaternion.js +3 -29
  148. package/src/math/Vector3.js +3 -3
  149. package/src/math/interpolants/BezierInterpolant.js +108 -0
  150. package/src/nodes/Nodes.js +87 -68
  151. package/src/nodes/TSL.js +2 -5
  152. package/src/nodes/accessors/Arrays.js +1 -1
  153. package/src/nodes/accessors/Bitangent.js +5 -5
  154. package/src/nodes/accessors/BufferAttributeNode.js +1 -1
  155. package/src/nodes/accessors/Camera.js +149 -28
  156. package/src/nodes/accessors/InstanceNode.js +105 -40
  157. package/src/nodes/accessors/Normal.js +9 -9
  158. package/src/nodes/accessors/Position.js +34 -2
  159. package/src/nodes/accessors/SceneProperties.js +53 -0
  160. package/src/nodes/accessors/SkinningNode.js +12 -24
  161. package/src/nodes/accessors/StorageBufferNode.js +0 -19
  162. package/src/nodes/accessors/StorageTextureNode.js +37 -1
  163. package/src/nodes/accessors/Tangent.js +3 -3
  164. package/src/nodes/accessors/Texture3DNode.js +6 -34
  165. package/src/nodes/accessors/TextureNode.js +58 -22
  166. package/src/nodes/accessors/UniformArrayNode.js +2 -0
  167. package/src/nodes/core/MRTNode.js +48 -2
  168. package/src/nodes/core/Node.js +29 -3
  169. package/src/nodes/core/NodeBuilder.js +115 -40
  170. package/src/nodes/core/NodeError.js +28 -0
  171. package/src/nodes/core/NodeUtils.js +5 -3
  172. package/src/nodes/core/OutputStructNode.js +12 -10
  173. package/src/nodes/core/ParameterNode.js +2 -1
  174. package/src/nodes/core/StackNode.js +9 -8
  175. package/src/nodes/core/StackTrace.js +139 -0
  176. package/src/nodes/core/StructNode.js +15 -0
  177. package/src/nodes/core/SubBuildNode.js +1 -1
  178. package/src/nodes/core/UniformNode.js +2 -1
  179. package/src/nodes/core/VarNode.js +1 -1
  180. package/src/nodes/core/VaryingNode.js +1 -18
  181. package/src/nodes/display/BlendModes.js +0 -64
  182. package/src/nodes/display/ColorAdjustment.js +17 -0
  183. package/src/nodes/display/ColorSpaceNode.js +3 -3
  184. package/src/nodes/display/NormalMapNode.js +2 -2
  185. package/src/nodes/display/PassNode.js +21 -2
  186. package/src/nodes/display/RenderOutputNode.js +3 -3
  187. package/src/nodes/display/ScreenNode.js +2 -1
  188. package/src/nodes/display/ToneMappingNode.js +1 -1
  189. package/src/nodes/display/ToonOutlinePassNode.js +2 -2
  190. package/src/nodes/display/ViewportDepthNode.js +52 -4
  191. package/src/nodes/display/ViewportTextureNode.js +21 -4
  192. package/src/nodes/fog/Fog.js +18 -35
  193. package/src/nodes/functions/PhysicalLightingModel.js +25 -3
  194. package/src/nodes/geometry/RangeNode.js +4 -2
  195. package/src/nodes/gpgpu/ComputeNode.js +5 -4
  196. package/src/nodes/gpgpu/WorkgroupInfoNode.js +2 -1
  197. package/src/nodes/lighting/EnvironmentNode.js +28 -3
  198. package/src/nodes/lighting/PointShadowNode.js +24 -12
  199. package/src/nodes/lighting/ShadowFilterNode.js +15 -43
  200. package/src/nodes/lighting/ShadowNode.js +54 -32
  201. package/src/nodes/math/ConditionalNode.js +2 -2
  202. package/src/nodes/math/MathNode.js +3 -40
  203. package/src/nodes/math/OperatorNode.js +2 -1
  204. package/src/nodes/pmrem/PMREMUtils.js +9 -15
  205. package/src/nodes/tsl/TSLCore.js +13 -10
  206. package/src/nodes/utils/DebugNode.js +11 -11
  207. package/src/nodes/utils/JoinNode.js +2 -2
  208. package/src/nodes/utils/LoopNode.js +1 -1
  209. package/src/nodes/utils/MemberNode.js +1 -1
  210. package/src/nodes/utils/RTTNode.js +1 -1
  211. package/src/nodes/utils/ReflectorNode.js +2 -3
  212. package/src/nodes/utils/SpriteSheetUV.js +35 -0
  213. package/src/nodes/utils/UVUtils.js +4 -2
  214. package/src/objects/BatchedMesh.js +22 -12
  215. package/src/objects/InstancedMesh.js +11 -0
  216. package/src/renderers/WebGLRenderer.js +34 -60
  217. package/src/renderers/common/Backend.js +21 -0
  218. package/src/renderers/common/Background.js +7 -4
  219. package/src/renderers/common/BindGroup.js +1 -9
  220. package/src/renderers/common/Bindings.js +20 -5
  221. package/src/renderers/common/BlendMode.js +143 -0
  222. package/src/renderers/common/BundleGroup.js +1 -1
  223. package/src/renderers/common/CubeRenderTarget.js +50 -6
  224. package/src/renderers/common/Geometries.js +17 -3
  225. package/src/renderers/common/Lighting.js +5 -21
  226. package/src/renderers/common/Pipelines.js +4 -4
  227. package/src/renderers/common/PostProcessing.js +8 -206
  228. package/src/renderers/common/RenderBundles.js +2 -1
  229. package/src/renderers/common/RenderContext.js +16 -0
  230. package/src/renderers/common/RenderContexts.js +33 -56
  231. package/src/renderers/common/RenderLists.js +2 -1
  232. package/src/renderers/common/RenderObject.js +2 -3
  233. package/src/renderers/common/RenderObjectPipeline.js +40 -0
  234. package/src/renderers/common/RenderObjects.js +18 -2
  235. package/src/renderers/common/RenderPipeline.js +203 -17
  236. package/src/renderers/common/Renderer.js +207 -40
  237. package/src/renderers/common/Sampler.js +4 -4
  238. package/src/renderers/common/StorageBuffer.js +13 -1
  239. package/src/renderers/common/Textures.js +16 -0
  240. package/src/renderers/common/TimestampQueryPool.js +5 -3
  241. package/src/renderers/common/Uniform.js +8 -0
  242. package/src/renderers/common/UniformsGroup.js +60 -0
  243. package/src/renderers/common/XRManager.js +2 -2
  244. package/src/renderers/common/nodes/NodeBuilderState.js +1 -1
  245. package/src/renderers/common/nodes/{Nodes.js → NodeManager.js} +18 -6
  246. package/src/renderers/common/nodes/NodeStorageBuffer.js +13 -2
  247. package/src/renderers/shaders/ShaderChunk/batching_pars_vertex.glsl.js +2 -2
  248. package/src/renderers/shaders/ShaderChunk/color_fragment.glsl.js +1 -5
  249. package/src/renderers/shaders/ShaderChunk/color_pars_fragment.glsl.js +1 -5
  250. package/src/renderers/shaders/ShaderChunk/color_pars_vertex.glsl.js +1 -5
  251. package/src/renderers/shaders/ShaderChunk/color_vertex.glsl.js +8 -10
  252. package/src/renderers/shaders/ShaderChunk/cube_uv_reflection_fragment.glsl.js +4 -4
  253. package/src/renderers/shaders/ShaderChunk/envmap_fragment.glsl.js +7 -11
  254. package/src/renderers/shaders/ShaderChunk/lights_fragment_end.glsl.js +6 -0
  255. package/src/renderers/shaders/ShaderChunk/lights_fragment_maps.glsl.js +6 -2
  256. package/src/renderers/shaders/ShaderChunk/lights_physical_pars_fragment.glsl.js +23 -1
  257. package/src/renderers/shaders/ShaderChunk/packing.glsl.js +20 -4
  258. package/src/renderers/shaders/ShaderChunk/shadowmap_pars_fragment.glsl.js +55 -24
  259. package/src/renderers/shaders/ShaderLib/meshlambert.glsl.js +2 -0
  260. package/src/renderers/shaders/ShaderLib/meshphong.glsl.js +2 -0
  261. package/src/renderers/shaders/ShaderLib/shadow.glsl.js +1 -0
  262. package/src/renderers/shaders/ShaderLib.js +4 -2
  263. package/src/renderers/shaders/UniformsLib.js +0 -3
  264. package/src/renderers/webgl/WebGLBackground.js +2 -2
  265. package/src/renderers/webgl/WebGLBindingStates.js +99 -27
  266. package/src/renderers/webgl/WebGLEnvironments.js +228 -0
  267. package/src/renderers/webgl/WebGLGeometries.js +10 -7
  268. package/src/renderers/webgl/WebGLMaterials.js +12 -0
  269. package/src/renderers/webgl/WebGLObjects.js +3 -1
  270. package/src/renderers/webgl/WebGLProgram.js +2 -2
  271. package/src/renderers/webgl/WebGLPrograms.js +10 -4
  272. package/src/renderers/webgl/WebGLRenderLists.js +15 -0
  273. package/src/renderers/webgl/WebGLShadowMap.js +5 -4
  274. package/src/renderers/webgl/WebGLState.js +12 -17
  275. package/src/renderers/webgl-fallback/WebGLBackend.js +71 -7
  276. package/src/renderers/webgl-fallback/nodes/GLSLNodeBuilder.js +98 -29
  277. package/src/renderers/webgl-fallback/utils/WebGLState.js +168 -7
  278. package/src/renderers/webgpu/WebGPUBackend.js +58 -9
  279. package/src/renderers/webgpu/WebGPURenderer.js +1 -0
  280. package/src/renderers/webgpu/nodes/WGSLNodeBuilder.js +257 -45
  281. package/src/renderers/webgpu/utils/WebGPUBindingUtils.js +8 -19
  282. package/src/renderers/webgpu/utils/WebGPUConstants.js +1 -1
  283. package/src/renderers/webgpu/utils/WebGPUPipelineUtils.js +56 -31
  284. package/src/renderers/webgpu/utils/WebGPUTexturePassUtils.js +152 -200
  285. package/src/renderers/webgpu/utils/WebGPUTextureUtils.js +25 -25
  286. package/src/renderers/webgpu/utils/WebGPUUtils.js +10 -6
  287. package/src/renderers/webxr/WebXRManager.js +2 -2
  288. package/src/textures/Texture.js +2 -2
  289. package/src/utils.js +246 -3
  290. package/examples/jsm/materials/MeshGouraudMaterial.js +0 -433
  291. package/examples/jsm/materials/MeshPostProcessingMaterial.js +0 -167
  292. package/examples/jsm/shaders/GodRaysShader.js +0 -333
  293. package/src/nodes/accessors/SceneNode.js +0 -145
  294. package/src/nodes/code/ScriptableNode.js +0 -726
  295. package/src/nodes/code/ScriptableValueNode.js +0 -253
  296. package/src/nodes/display/PosterizeNode.js +0 -65
  297. package/src/nodes/utils/SpriteSheetUVNode.js +0 -90
  298. package/src/renderers/webgl/WebGLCubeMaps.js +0 -99
  299. package/src/renderers/webgl/WebGLCubeUVMaps.js +0 -134
@@ -0,0 +1,1962 @@
1
+ import {
2
+ Color,
3
+ ColorManagement,
4
+ MathUtils,
5
+ Matrix4,
6
+ Vector3,
7
+ SRGBColorSpace
8
+ } from 'three';
9
+
10
+ /**
11
+ * Utility functions for parsing
12
+ */
13
+
14
+ function getElementsByTagName( xml, name ) {
15
+
16
+ // Non recursive xml.getElementsByTagName() ...
17
+
18
+ const array = [];
19
+ const childNodes = xml.childNodes;
20
+
21
+ for ( let i = 0, l = childNodes.length; i < l; i ++ ) {
22
+
23
+ const child = childNodes[ i ];
24
+
25
+ if ( child.nodeName === name ) {
26
+
27
+ array.push( child );
28
+
29
+ }
30
+
31
+ }
32
+
33
+ return array;
34
+
35
+ }
36
+
37
+ function parseStrings( text ) {
38
+
39
+ if ( text.length === 0 ) return [];
40
+
41
+ return text.trim().split( /\s+/ );
42
+
43
+ }
44
+
45
+ function parseFloats( text ) {
46
+
47
+ if ( text.length === 0 ) return [];
48
+
49
+ return text.trim().split( /\s+/ ).map( parseFloat );
50
+
51
+ }
52
+
53
+ function parseInts( text ) {
54
+
55
+ if ( text.length === 0 ) return [];
56
+
57
+ return text.trim().split( /\s+/ ).map( s => parseInt( s ) );
58
+
59
+ }
60
+
61
+ function parseId( text ) {
62
+
63
+ return text.substring( 1 );
64
+
65
+ }
66
+
67
+ /**
68
+ * ColladaParser handles XML parsing and converts Collada XML to intermediate data structures.
69
+ */
70
+ class ColladaParser {
71
+
72
+ constructor() {
73
+
74
+ this.count = 0;
75
+
76
+ }
77
+
78
+ generateId() {
79
+
80
+ return 'three_default_' + ( this.count ++ );
81
+
82
+ }
83
+
84
+ parse( text ) {
85
+
86
+ if ( text.length === 0 ) {
87
+
88
+ return null;
89
+
90
+ }
91
+
92
+ const xml = new DOMParser().parseFromString( text, 'application/xml' );
93
+
94
+ const collada = getElementsByTagName( xml, 'COLLADA' )[ 0 ];
95
+
96
+ const parserError = xml.getElementsByTagName( 'parsererror' )[ 0 ];
97
+ if ( parserError !== undefined ) {
98
+
99
+ // Chrome will return parser error with a div in it
100
+
101
+ const errorElement = getElementsByTagName( parserError, 'div' )[ 0 ];
102
+ let errorText;
103
+
104
+ if ( errorElement ) {
105
+
106
+ errorText = errorElement.textContent;
107
+
108
+ } else {
109
+
110
+ errorText = this.parserErrorToText( parserError );
111
+
112
+ }
113
+
114
+ console.error( 'THREE.ColladaLoader: Failed to parse collada file.\n', errorText );
115
+
116
+ return null;
117
+
118
+ }
119
+
120
+ // metadata
121
+
122
+ const version = collada.getAttribute( 'version' );
123
+ console.debug( 'THREE.ColladaLoader: File version', version );
124
+
125
+ const asset = this.parseAsset( getElementsByTagName( collada, 'asset' )[ 0 ] );
126
+
127
+ //
128
+
129
+ const library = {
130
+ animations: {},
131
+ clips: {},
132
+ controllers: {},
133
+ images: {},
134
+ effects: {},
135
+ materials: {},
136
+ cameras: {},
137
+ lights: {},
138
+ geometries: {},
139
+ nodes: {},
140
+ visualScenes: {},
141
+ kinematicsModels: {},
142
+ physicsModels: {},
143
+ kinematicsScenes: {}
144
+ };
145
+
146
+ this.library = library;
147
+ this.collada = collada;
148
+
149
+ this.parseLibrary( collada, 'library_animations', 'animation', this.parseAnimation.bind( this ) );
150
+ this.parseLibrary( collada, 'library_animation_clips', 'animation_clip', this.parseAnimationClip.bind( this ) );
151
+ this.parseLibrary( collada, 'library_controllers', 'controller', this.parseController.bind( this ) );
152
+ this.parseLibrary( collada, 'library_images', 'image', this.parseImage.bind( this ) );
153
+ this.parseLibrary( collada, 'library_effects', 'effect', this.parseEffect.bind( this ) );
154
+ this.parseLibrary( collada, 'library_materials', 'material', this.parseMaterial.bind( this ) );
155
+ this.parseLibrary( collada, 'library_cameras', 'camera', this.parseCamera.bind( this ) );
156
+ this.parseLibrary( collada, 'library_lights', 'light', this.parseLight.bind( this ) );
157
+ this.parseLibrary( collada, 'library_geometries', 'geometry', this.parseGeometry.bind( this ) );
158
+ this.parseLibrary( collada, 'library_nodes', 'node', this.parseNode.bind( this ) );
159
+ this.parseLibrary( collada, 'library_visual_scenes', 'visual_scene', this.parseVisualScene.bind( this ) );
160
+ this.parseLibrary( collada, 'library_kinematics_models', 'kinematics_model', this.parseKinematicsModel.bind( this ) );
161
+ this.parseLibrary( collada, 'library_physics_models', 'physics_model', this.parsePhysicsModel.bind( this ) );
162
+ this.parseLibrary( collada, 'scene', 'instance_kinematics_scene', this.parseKinematicsScene.bind( this ) );
163
+
164
+ return {
165
+ library: library,
166
+ asset: asset,
167
+ collada: collada
168
+ };
169
+
170
+ }
171
+
172
+ // convert the parser error element into text with each child elements text
173
+ // separated by new lines.
174
+
175
+ parserErrorToText( parserError ) {
176
+
177
+ const parts = [];
178
+ const stack = [ parserError ];
179
+
180
+ while ( stack.length ) {
181
+
182
+ const node = stack.shift();
183
+
184
+ if ( node.nodeType === Node.TEXT_NODE ) {
185
+
186
+ parts.push( node.textContent );
187
+
188
+ } else {
189
+
190
+ parts.push( '\n' );
191
+ stack.push( ...node.childNodes );
192
+
193
+ }
194
+
195
+ }
196
+
197
+ return parts.join( '' ).trim();
198
+
199
+ }
200
+
201
+ // asset
202
+
203
+ parseAsset( xml ) {
204
+
205
+ return {
206
+ unit: this.parseAssetUnit( getElementsByTagName( xml, 'unit' )[ 0 ] ),
207
+ upAxis: this.parseAssetUpAxis( getElementsByTagName( xml, 'up_axis' )[ 0 ] )
208
+ };
209
+
210
+ }
211
+
212
+ parseAssetUnit( xml ) {
213
+
214
+ if ( ( xml !== undefined ) && ( xml.hasAttribute( 'meter' ) === true ) ) {
215
+
216
+ return parseFloat( xml.getAttribute( 'meter' ) );
217
+
218
+ } else {
219
+
220
+ return 1; // default 1 meter
221
+
222
+ }
223
+
224
+ }
225
+
226
+ parseAssetUpAxis( xml ) {
227
+
228
+ return xml !== undefined ? xml.textContent : 'Y_UP';
229
+
230
+ }
231
+
232
+ // library
233
+
234
+ parseLibrary( xml, libraryName, nodeName, parser ) {
235
+
236
+ const library = getElementsByTagName( xml, libraryName )[ 0 ];
237
+
238
+ if ( library !== undefined ) {
239
+
240
+ const elements = getElementsByTagName( library, nodeName );
241
+
242
+ for ( let i = 0; i < elements.length; i ++ ) {
243
+
244
+ parser( elements[ i ] );
245
+
246
+ }
247
+
248
+ }
249
+
250
+ }
251
+
252
+ // animation
253
+
254
+ parseAnimation( xml ) {
255
+
256
+ const data = {
257
+ sources: {},
258
+ samplers: {},
259
+ channels: {}
260
+ };
261
+
262
+ let hasChildren = false;
263
+
264
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
265
+
266
+ const child = xml.childNodes[ i ];
267
+
268
+ if ( child.nodeType !== 1 ) continue;
269
+
270
+ let id;
271
+
272
+ switch ( child.nodeName ) {
273
+
274
+ case 'source':
275
+ id = child.getAttribute( 'id' );
276
+ data.sources[ id ] = this.parseSource( child );
277
+ break;
278
+
279
+ case 'sampler':
280
+ id = child.getAttribute( 'id' );
281
+ data.samplers[ id ] = this.parseAnimationSampler( child );
282
+ break;
283
+
284
+ case 'channel':
285
+ id = child.getAttribute( 'target' );
286
+ data.channels[ id ] = this.parseAnimationChannel( child );
287
+ break;
288
+
289
+ case 'animation':
290
+ // hierarchy of related animations
291
+ this.parseAnimation( child );
292
+ hasChildren = true;
293
+ break;
294
+
295
+ default:
296
+
297
+ }
298
+
299
+ }
300
+
301
+ if ( hasChildren === false ) {
302
+
303
+ // since 'id' attributes can be optional, it's necessary to generate a UUID for unique assignment
304
+
305
+ this.library.animations[ xml.getAttribute( 'id' ) || MathUtils.generateUUID() ] = data;
306
+
307
+ }
308
+
309
+ }
310
+
311
+ parseAnimationSampler( xml ) {
312
+
313
+ const data = {
314
+ inputs: {},
315
+ };
316
+
317
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
318
+
319
+ const child = xml.childNodes[ i ];
320
+
321
+ if ( child.nodeType !== 1 ) continue;
322
+
323
+ switch ( child.nodeName ) {
324
+
325
+ case 'input':
326
+ const id = parseId( child.getAttribute( 'source' ) );
327
+ const semantic = child.getAttribute( 'semantic' );
328
+ data.inputs[ semantic ] = id;
329
+ break;
330
+
331
+ }
332
+
333
+ }
334
+
335
+ return data;
336
+
337
+ }
338
+
339
+ parseAnimationChannel( xml ) {
340
+
341
+ const data = {};
342
+
343
+ const target = xml.getAttribute( 'target' );
344
+
345
+ // parsing SID Addressing Syntax
346
+
347
+ let parts = target.split( '/' );
348
+
349
+ const id = parts.shift();
350
+ let sid = parts.shift();
351
+
352
+ // check selection syntax
353
+
354
+ const arraySyntax = ( sid.indexOf( '(' ) !== - 1 );
355
+ const memberSyntax = ( sid.indexOf( '.' ) !== - 1 );
356
+
357
+ if ( memberSyntax ) {
358
+
359
+ // member selection access
360
+
361
+ parts = sid.split( '.' );
362
+ sid = parts.shift();
363
+ data.member = parts.shift();
364
+
365
+ } else if ( arraySyntax ) {
366
+
367
+ // array-access syntax. can be used to express fields in one-dimensional vectors or two-dimensional matrices.
368
+
369
+ const indices = sid.split( '(' );
370
+ sid = indices.shift();
371
+
372
+ for ( let i = 0; i < indices.length; i ++ ) {
373
+
374
+ indices[ i ] = parseInt( indices[ i ].replace( /\)/, '' ) );
375
+
376
+ }
377
+
378
+ data.indices = indices;
379
+
380
+ }
381
+
382
+ data.id = id;
383
+ data.sid = sid;
384
+
385
+ data.arraySyntax = arraySyntax;
386
+ data.memberSyntax = memberSyntax;
387
+
388
+ data.sampler = parseId( xml.getAttribute( 'source' ) );
389
+
390
+ return data;
391
+
392
+ }
393
+
394
+ // animation clips
395
+
396
+ parseAnimationClip( xml ) {
397
+
398
+ const data = {
399
+ name: xml.getAttribute( 'id' ) || 'default',
400
+ start: parseFloat( xml.getAttribute( 'start' ) || 0 ),
401
+ end: parseFloat( xml.getAttribute( 'end' ) || 0 ),
402
+ animations: []
403
+ };
404
+
405
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
406
+
407
+ const child = xml.childNodes[ i ];
408
+
409
+ if ( child.nodeType !== 1 ) continue;
410
+
411
+ switch ( child.nodeName ) {
412
+
413
+ case 'instance_animation':
414
+ data.animations.push( parseId( child.getAttribute( 'url' ) ) );
415
+ break;
416
+
417
+ }
418
+
419
+ }
420
+
421
+ this.library.clips[ xml.getAttribute( 'id' ) ] = data;
422
+
423
+ }
424
+
425
+ // controller
426
+
427
+ parseController( xml ) {
428
+
429
+ const data = {};
430
+
431
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
432
+
433
+ const child = xml.childNodes[ i ];
434
+
435
+ if ( child.nodeType !== 1 ) continue;
436
+
437
+ switch ( child.nodeName ) {
438
+
439
+ case 'skin':
440
+ // there is exactly one skin per controller
441
+ data.id = parseId( child.getAttribute( 'source' ) );
442
+ data.skin = this.parseSkin( child );
443
+ break;
444
+
445
+ case 'morph':
446
+ data.id = parseId( child.getAttribute( 'source' ) );
447
+ console.warn( 'THREE.ColladaLoader: Morph target animation not supported yet.' );
448
+ break;
449
+
450
+ }
451
+
452
+ }
453
+
454
+ this.library.controllers[ xml.getAttribute( 'id' ) ] = data;
455
+
456
+ }
457
+
458
+ parseSkin( xml ) {
459
+
460
+ const data = {
461
+ sources: {}
462
+ };
463
+
464
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
465
+
466
+ const child = xml.childNodes[ i ];
467
+
468
+ if ( child.nodeType !== 1 ) continue;
469
+
470
+ switch ( child.nodeName ) {
471
+
472
+ case 'bind_shape_matrix':
473
+ data.bindShapeMatrix = parseFloats( child.textContent );
474
+ break;
475
+
476
+ case 'source':
477
+ const id = child.getAttribute( 'id' );
478
+ data.sources[ id ] = this.parseSource( child );
479
+ break;
480
+
481
+ case 'joints':
482
+ data.joints = this.parseJoints( child );
483
+ break;
484
+
485
+ case 'vertex_weights':
486
+ data.vertexWeights = this.parseVertexWeights( child );
487
+ break;
488
+
489
+ }
490
+
491
+ }
492
+
493
+ return data;
494
+
495
+ }
496
+
497
+ parseJoints( xml ) {
498
+
499
+ const data = {
500
+ inputs: {}
501
+ };
502
+
503
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
504
+
505
+ const child = xml.childNodes[ i ];
506
+
507
+ if ( child.nodeType !== 1 ) continue;
508
+
509
+ switch ( child.nodeName ) {
510
+
511
+ case 'input':
512
+ const semantic = child.getAttribute( 'semantic' );
513
+ const id = parseId( child.getAttribute( 'source' ) );
514
+ data.inputs[ semantic ] = id;
515
+ break;
516
+
517
+ }
518
+
519
+ }
520
+
521
+ return data;
522
+
523
+ }
524
+
525
+ parseVertexWeights( xml ) {
526
+
527
+ const data = {
528
+ inputs: {}
529
+ };
530
+
531
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
532
+
533
+ const child = xml.childNodes[ i ];
534
+
535
+ if ( child.nodeType !== 1 ) continue;
536
+
537
+ switch ( child.nodeName ) {
538
+
539
+ case 'input':
540
+ const semantic = child.getAttribute( 'semantic' );
541
+ const id = parseId( child.getAttribute( 'source' ) );
542
+ const offset = parseInt( child.getAttribute( 'offset' ) );
543
+ data.inputs[ semantic ] = { id: id, offset: offset };
544
+ break;
545
+
546
+ case 'vcount':
547
+ data.vcount = parseInts( child.textContent );
548
+ break;
549
+
550
+ case 'v':
551
+ data.v = parseInts( child.textContent );
552
+ break;
553
+
554
+ }
555
+
556
+ }
557
+
558
+ return data;
559
+
560
+ }
561
+
562
+ // image
563
+
564
+ parseImage( xml ) {
565
+
566
+ const data = {
567
+ init_from: getElementsByTagName( xml, 'init_from' )[ 0 ].textContent
568
+ };
569
+
570
+ this.library.images[ xml.getAttribute( 'id' ) ] = data;
571
+
572
+ }
573
+
574
+ // effect
575
+
576
+ parseEffect( xml ) {
577
+
578
+ const data = {};
579
+
580
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
581
+
582
+ const child = xml.childNodes[ i ];
583
+
584
+ if ( child.nodeType !== 1 ) continue;
585
+
586
+ switch ( child.nodeName ) {
587
+
588
+ case 'profile_COMMON':
589
+ data.profile = this.parseEffectProfileCOMMON( child );
590
+ break;
591
+
592
+ }
593
+
594
+ }
595
+
596
+ this.library.effects[ xml.getAttribute( 'id' ) ] = data;
597
+
598
+ }
599
+
600
+ parseEffectProfileCOMMON( xml ) {
601
+
602
+ const data = {
603
+ surfaces: {},
604
+ samplers: {}
605
+ };
606
+
607
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
608
+
609
+ const child = xml.childNodes[ i ];
610
+
611
+ if ( child.nodeType !== 1 ) continue;
612
+
613
+ switch ( child.nodeName ) {
614
+
615
+ case 'newparam':
616
+ this.parseEffectNewparam( child, data );
617
+ break;
618
+
619
+ case 'technique':
620
+ data.technique = this.parseEffectTechnique( child );
621
+ break;
622
+
623
+ case 'extra':
624
+ data.extra = this.parseEffectExtra( child );
625
+ break;
626
+
627
+ }
628
+
629
+ }
630
+
631
+ return data;
632
+
633
+ }
634
+
635
+ parseEffectNewparam( xml, data ) {
636
+
637
+ const sid = xml.getAttribute( 'sid' );
638
+
639
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
640
+
641
+ const child = xml.childNodes[ i ];
642
+
643
+ if ( child.nodeType !== 1 ) continue;
644
+
645
+ switch ( child.nodeName ) {
646
+
647
+ case 'surface':
648
+ data.surfaces[ sid ] = this.parseEffectSurface( child );
649
+ break;
650
+
651
+ case 'sampler2D':
652
+ data.samplers[ sid ] = this.parseEffectSampler( child );
653
+ break;
654
+
655
+ }
656
+
657
+ }
658
+
659
+ }
660
+
661
+ parseEffectSurface( xml ) {
662
+
663
+ const data = {};
664
+
665
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
666
+
667
+ const child = xml.childNodes[ i ];
668
+
669
+ if ( child.nodeType !== 1 ) continue;
670
+
671
+ switch ( child.nodeName ) {
672
+
673
+ case 'init_from':
674
+ data.init_from = child.textContent;
675
+ break;
676
+
677
+ }
678
+
679
+ }
680
+
681
+ return data;
682
+
683
+ }
684
+
685
+ parseEffectSampler( xml ) {
686
+
687
+ const data = {};
688
+
689
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
690
+
691
+ const child = xml.childNodes[ i ];
692
+
693
+ if ( child.nodeType !== 1 ) continue;
694
+
695
+ switch ( child.nodeName ) {
696
+
697
+ case 'source':
698
+ data.source = child.textContent;
699
+ break;
700
+
701
+ }
702
+
703
+ }
704
+
705
+ return data;
706
+
707
+ }
708
+
709
+ parseEffectTechnique( xml ) {
710
+
711
+ const data = {};
712
+
713
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
714
+
715
+ const child = xml.childNodes[ i ];
716
+
717
+ if ( child.nodeType !== 1 ) continue;
718
+
719
+ switch ( child.nodeName ) {
720
+
721
+ case 'constant':
722
+ case 'lambert':
723
+ case 'blinn':
724
+ case 'phong':
725
+ data.type = child.nodeName;
726
+ data.parameters = this.parseEffectParameters( child );
727
+ break;
728
+
729
+ case 'extra':
730
+ data.extra = this.parseEffectExtra( child );
731
+ break;
732
+
733
+ }
734
+
735
+ }
736
+
737
+ return data;
738
+
739
+ }
740
+
741
+ parseEffectParameters( xml ) {
742
+
743
+ const data = {};
744
+
745
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
746
+
747
+ const child = xml.childNodes[ i ];
748
+
749
+ if ( child.nodeType !== 1 ) continue;
750
+
751
+ switch ( child.nodeName ) {
752
+
753
+ case 'emission':
754
+ case 'diffuse':
755
+ case 'specular':
756
+ case 'bump':
757
+ case 'ambient':
758
+ case 'shininess':
759
+ case 'transparency':
760
+ data[ child.nodeName ] = this.parseEffectParameter( child );
761
+ break;
762
+ case 'transparent':
763
+ data[ child.nodeName ] = {
764
+ opaque: child.hasAttribute( 'opaque' ) ? child.getAttribute( 'opaque' ) : 'A_ONE',
765
+ data: this.parseEffectParameter( child )
766
+ };
767
+ break;
768
+
769
+ }
770
+
771
+ }
772
+
773
+ return data;
774
+
775
+ }
776
+
777
+ parseEffectParameter( xml ) {
778
+
779
+ const data = {};
780
+
781
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
782
+
783
+ const child = xml.childNodes[ i ];
784
+
785
+ if ( child.nodeType !== 1 ) continue;
786
+
787
+ switch ( child.nodeName ) {
788
+
789
+ case 'color':
790
+ data[ child.nodeName ] = parseFloats( child.textContent );
791
+ break;
792
+
793
+ case 'float':
794
+ data[ child.nodeName ] = parseFloat( child.textContent );
795
+ break;
796
+
797
+ case 'texture':
798
+ data[ child.nodeName ] = { id: child.getAttribute( 'texture' ), extra: this.parseEffectParameterTexture( child ) };
799
+ break;
800
+
801
+ }
802
+
803
+ }
804
+
805
+ return data;
806
+
807
+ }
808
+
809
+ parseEffectParameterTexture( xml ) {
810
+
811
+ const data = {
812
+ technique: {}
813
+ };
814
+
815
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
816
+
817
+ const child = xml.childNodes[ i ];
818
+
819
+ if ( child.nodeType !== 1 ) continue;
820
+
821
+ switch ( child.nodeName ) {
822
+
823
+ case 'extra':
824
+ this.parseEffectParameterTextureExtra( child, data );
825
+ break;
826
+
827
+ }
828
+
829
+ }
830
+
831
+ return data;
832
+
833
+ }
834
+
835
+ parseEffectParameterTextureExtra( xml, data ) {
836
+
837
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
838
+
839
+ const child = xml.childNodes[ i ];
840
+
841
+ if ( child.nodeType !== 1 ) continue;
842
+
843
+ switch ( child.nodeName ) {
844
+
845
+ case 'technique':
846
+ this.parseEffectParameterTextureExtraTechnique( child, data );
847
+ break;
848
+
849
+ }
850
+
851
+ }
852
+
853
+ }
854
+
855
+ parseEffectParameterTextureExtraTechnique( xml, data ) {
856
+
857
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
858
+
859
+ const child = xml.childNodes[ i ];
860
+
861
+ if ( child.nodeType !== 1 ) continue;
862
+
863
+ switch ( child.nodeName ) {
864
+
865
+ case 'repeatU':
866
+ case 'repeatV':
867
+ case 'offsetU':
868
+ case 'offsetV':
869
+ data.technique[ child.nodeName ] = parseFloat( child.textContent );
870
+ break;
871
+
872
+ case 'wrapU':
873
+ case 'wrapV':
874
+
875
+ // some files have values for wrapU/wrapV which become NaN via parseInt
876
+
877
+ if ( child.textContent.toUpperCase() === 'TRUE' ) {
878
+
879
+ data.technique[ child.nodeName ] = 1;
880
+
881
+ } else if ( child.textContent.toUpperCase() === 'FALSE' ) {
882
+
883
+ data.technique[ child.nodeName ] = 0;
884
+
885
+ } else {
886
+
887
+ data.technique[ child.nodeName ] = parseInt( child.textContent );
888
+
889
+ }
890
+
891
+ break;
892
+
893
+ case 'bump':
894
+ data[ child.nodeName ] = this.parseEffectExtraTechniqueBump( child );
895
+ break;
896
+
897
+ }
898
+
899
+ }
900
+
901
+ }
902
+
903
+ parseEffectExtra( xml ) {
904
+
905
+ const data = {};
906
+
907
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
908
+
909
+ const child = xml.childNodes[ i ];
910
+
911
+ if ( child.nodeType !== 1 ) continue;
912
+
913
+ switch ( child.nodeName ) {
914
+
915
+ case 'technique':
916
+ data.technique = this.parseEffectExtraTechnique( child );
917
+ break;
918
+
919
+ }
920
+
921
+ }
922
+
923
+ return data;
924
+
925
+ }
926
+
927
+ parseEffectExtraTechnique( xml ) {
928
+
929
+ const data = {};
930
+
931
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
932
+
933
+ const child = xml.childNodes[ i ];
934
+
935
+ if ( child.nodeType !== 1 ) continue;
936
+
937
+ switch ( child.nodeName ) {
938
+
939
+ case 'double_sided':
940
+ data[ child.nodeName ] = parseInt( child.textContent );
941
+ break;
942
+
943
+ case 'bump':
944
+ data[ child.nodeName ] = this.parseEffectExtraTechniqueBump( child );
945
+ break;
946
+
947
+ }
948
+
949
+ }
950
+
951
+ return data;
952
+
953
+ }
954
+
955
+ parseEffectExtraTechniqueBump( xml ) {
956
+
957
+ const data = {};
958
+
959
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
960
+
961
+ const child = xml.childNodes[ i ];
962
+
963
+ if ( child.nodeType !== 1 ) continue;
964
+
965
+ switch ( child.nodeName ) {
966
+
967
+ case 'texture':
968
+ data[ child.nodeName ] = { id: child.getAttribute( 'texture' ), texcoord: child.getAttribute( 'texcoord' ), extra: this.parseEffectParameterTexture( child ) };
969
+ break;
970
+
971
+ }
972
+
973
+ }
974
+
975
+ return data;
976
+
977
+ }
978
+
979
+ // material
980
+
981
+ parseMaterial( xml ) {
982
+
983
+ const data = {
984
+ name: xml.getAttribute( 'name' )
985
+ };
986
+
987
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
988
+
989
+ const child = xml.childNodes[ i ];
990
+
991
+ if ( child.nodeType !== 1 ) continue;
992
+
993
+ switch ( child.nodeName ) {
994
+
995
+ case 'instance_effect':
996
+ data.url = parseId( child.getAttribute( 'url' ) );
997
+ break;
998
+
999
+ }
1000
+
1001
+ }
1002
+
1003
+ this.library.materials[ xml.getAttribute( 'id' ) ] = data;
1004
+
1005
+ }
1006
+
1007
+ // camera
1008
+
1009
+ parseCamera( xml ) {
1010
+
1011
+ const data = {
1012
+ name: xml.getAttribute( 'name' )
1013
+ };
1014
+
1015
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
1016
+
1017
+ const child = xml.childNodes[ i ];
1018
+
1019
+ if ( child.nodeType !== 1 ) continue;
1020
+
1021
+ switch ( child.nodeName ) {
1022
+
1023
+ case 'optics':
1024
+ data.optics = this.parseCameraOptics( child );
1025
+ break;
1026
+
1027
+ }
1028
+
1029
+ }
1030
+
1031
+ this.library.cameras[ xml.getAttribute( 'id' ) ] = data;
1032
+
1033
+ }
1034
+
1035
+ parseCameraOptics( xml ) {
1036
+
1037
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1038
+
1039
+ const child = xml.childNodes[ i ];
1040
+
1041
+ switch ( child.nodeName ) {
1042
+
1043
+ case 'technique_common':
1044
+ return this.parseCameraTechnique( child );
1045
+
1046
+ }
1047
+
1048
+ }
1049
+
1050
+ return {};
1051
+
1052
+ }
1053
+
1054
+ parseCameraTechnique( xml ) {
1055
+
1056
+ const data = {};
1057
+
1058
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1059
+
1060
+ const child = xml.childNodes[ i ];
1061
+
1062
+ switch ( child.nodeName ) {
1063
+
1064
+ case 'perspective':
1065
+ case 'orthographic':
1066
+
1067
+ data.technique = child.nodeName;
1068
+ data.parameters = this.parseCameraParameters( child );
1069
+
1070
+ break;
1071
+
1072
+ }
1073
+
1074
+ }
1075
+
1076
+ return data;
1077
+
1078
+ }
1079
+
1080
+ parseCameraParameters( xml ) {
1081
+
1082
+ const data = {};
1083
+
1084
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1085
+
1086
+ const child = xml.childNodes[ i ];
1087
+
1088
+ switch ( child.nodeName ) {
1089
+
1090
+ case 'xfov':
1091
+ case 'yfov':
1092
+ case 'xmag':
1093
+ case 'ymag':
1094
+ case 'znear':
1095
+ case 'zfar':
1096
+ case 'aspect_ratio':
1097
+ data[ child.nodeName ] = parseFloat( child.textContent );
1098
+ break;
1099
+
1100
+ }
1101
+
1102
+ }
1103
+
1104
+ return data;
1105
+
1106
+ }
1107
+
1108
+ // light
1109
+
1110
+ parseLight( xml ) {
1111
+
1112
+ let data = {};
1113
+
1114
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
1115
+
1116
+ const child = xml.childNodes[ i ];
1117
+
1118
+ if ( child.nodeType !== 1 ) continue;
1119
+
1120
+ switch ( child.nodeName ) {
1121
+
1122
+ case 'technique_common':
1123
+ data = this.parseLightTechnique( child );
1124
+ break;
1125
+
1126
+ }
1127
+
1128
+ }
1129
+
1130
+ this.library.lights[ xml.getAttribute( 'id' ) ] = data;
1131
+
1132
+ }
1133
+
1134
+ parseLightTechnique( xml ) {
1135
+
1136
+ const data = {};
1137
+
1138
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
1139
+
1140
+ const child = xml.childNodes[ i ];
1141
+
1142
+ if ( child.nodeType !== 1 ) continue;
1143
+
1144
+ switch ( child.nodeName ) {
1145
+
1146
+ case 'directional':
1147
+ case 'point':
1148
+ case 'spot':
1149
+ case 'ambient':
1150
+
1151
+ data.technique = child.nodeName;
1152
+ data.parameters = this.parseLightParameters( child );
1153
+ break;
1154
+
1155
+ }
1156
+
1157
+ }
1158
+
1159
+ return data;
1160
+
1161
+ }
1162
+
1163
+ parseLightParameters( xml ) {
1164
+
1165
+ const data = {};
1166
+
1167
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
1168
+
1169
+ const child = xml.childNodes[ i ];
1170
+
1171
+ if ( child.nodeType !== 1 ) continue;
1172
+
1173
+ switch ( child.nodeName ) {
1174
+
1175
+ case 'color':
1176
+ const array = parseFloats( child.textContent );
1177
+ data.color = new Color().fromArray( array );
1178
+ ColorManagement.colorSpaceToWorking( data.color, SRGBColorSpace );
1179
+ break;
1180
+
1181
+ case 'falloff_angle':
1182
+ data.falloffAngle = parseFloat( child.textContent );
1183
+ break;
1184
+
1185
+ case 'quadratic_attenuation':
1186
+ const f = parseFloat( child.textContent );
1187
+ data.distance = f ? Math.sqrt( 1 / f ) : 0;
1188
+ break;
1189
+
1190
+ }
1191
+
1192
+ }
1193
+
1194
+ return data;
1195
+
1196
+ }
1197
+
1198
+ // geometry
1199
+
1200
+ parseGeometry( xml ) {
1201
+
1202
+ const data = {
1203
+ name: xml.getAttribute( 'name' ),
1204
+ sources: {},
1205
+ vertices: {},
1206
+ primitives: []
1207
+ };
1208
+
1209
+ const mesh = getElementsByTagName( xml, 'mesh' )[ 0 ];
1210
+
1211
+ // the following tags inside geometry are not supported yet (see https://github.com/mrdoob/three.js/pull/12606): convex_mesh, spline, brep
1212
+ if ( mesh === undefined ) return;
1213
+
1214
+ for ( let i = 0; i < mesh.childNodes.length; i ++ ) {
1215
+
1216
+ const child = mesh.childNodes[ i ];
1217
+
1218
+ if ( child.nodeType !== 1 ) continue;
1219
+
1220
+ const id = child.getAttribute( 'id' );
1221
+
1222
+ switch ( child.nodeName ) {
1223
+
1224
+ case 'source':
1225
+ data.sources[ id ] = this.parseSource( child );
1226
+ break;
1227
+
1228
+ case 'vertices':
1229
+ // data.sources[ id ] = data.sources[ parseId( getElementsByTagName( child, 'input' )[ 0 ].getAttribute( 'source' ) ) ];
1230
+ data.vertices = this.parseGeometryVertices( child );
1231
+ break;
1232
+
1233
+ case 'polygons':
1234
+ console.warn( 'THREE.ColladaLoader: Unsupported primitive type: ', child.nodeName );
1235
+ break;
1236
+
1237
+ case 'lines':
1238
+ case 'linestrips':
1239
+ case 'polylist':
1240
+ case 'triangles':
1241
+ data.primitives.push( this.parseGeometryPrimitive( child ) );
1242
+ break;
1243
+
1244
+ default:
1245
+
1246
+ }
1247
+
1248
+ }
1249
+
1250
+ this.library.geometries[ xml.getAttribute( 'id' ) ] = data;
1251
+
1252
+ }
1253
+
1254
+ parseSource( xml ) {
1255
+
1256
+ const data = {
1257
+ array: [],
1258
+ stride: 3
1259
+ };
1260
+
1261
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1262
+
1263
+ const child = xml.childNodes[ i ];
1264
+
1265
+ if ( child.nodeType !== 1 ) continue;
1266
+
1267
+ switch ( child.nodeName ) {
1268
+
1269
+ case 'float_array':
1270
+ data.array = parseFloats( child.textContent );
1271
+ break;
1272
+
1273
+ case 'Name_array':
1274
+ data.array = parseStrings( child.textContent );
1275
+ break;
1276
+
1277
+ case 'technique_common':
1278
+ const accessor = getElementsByTagName( child, 'accessor' )[ 0 ];
1279
+
1280
+ if ( accessor !== undefined ) {
1281
+
1282
+ data.stride = parseInt( accessor.getAttribute( 'stride' ) );
1283
+
1284
+ }
1285
+
1286
+ break;
1287
+
1288
+ }
1289
+
1290
+ }
1291
+
1292
+ return data;
1293
+
1294
+ }
1295
+
1296
+ parseGeometryVertices( xml ) {
1297
+
1298
+ const data = {};
1299
+
1300
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1301
+
1302
+ const child = xml.childNodes[ i ];
1303
+
1304
+ if ( child.nodeType !== 1 ) continue;
1305
+
1306
+ data[ child.getAttribute( 'semantic' ) ] = parseId( child.getAttribute( 'source' ) );
1307
+
1308
+ }
1309
+
1310
+ return data;
1311
+
1312
+ }
1313
+
1314
+ parseGeometryPrimitive( xml ) {
1315
+
1316
+ const primitive = {
1317
+ type: xml.nodeName,
1318
+ material: xml.getAttribute( 'material' ),
1319
+ count: parseInt( xml.getAttribute( 'count' ) ),
1320
+ inputs: {},
1321
+ stride: 0,
1322
+ hasUV: false
1323
+ };
1324
+
1325
+ for ( let i = 0, l = xml.childNodes.length; i < l; i ++ ) {
1326
+
1327
+ const child = xml.childNodes[ i ];
1328
+
1329
+ if ( child.nodeType !== 1 ) continue;
1330
+
1331
+ switch ( child.nodeName ) {
1332
+
1333
+ case 'input':
1334
+ const id = parseId( child.getAttribute( 'source' ) );
1335
+ const semantic = child.getAttribute( 'semantic' );
1336
+ const offset = parseInt( child.getAttribute( 'offset' ) );
1337
+ const set = parseInt( child.getAttribute( 'set' ) );
1338
+ const inputname = ( set > 0 ? semantic + set : semantic );
1339
+ primitive.inputs[ inputname ] = { id: id, offset: offset };
1340
+ primitive.stride = Math.max( primitive.stride, offset + 1 );
1341
+ if ( semantic === 'TEXCOORD' ) primitive.hasUV = true;
1342
+ break;
1343
+
1344
+ case 'vcount':
1345
+ primitive.vcount = parseInts( child.textContent );
1346
+ break;
1347
+
1348
+ case 'p':
1349
+ primitive.p = parseInts( child.textContent );
1350
+ break;
1351
+
1352
+ }
1353
+
1354
+ }
1355
+
1356
+ return primitive;
1357
+
1358
+ }
1359
+
1360
+ // kinematics
1361
+
1362
+ parseKinematicsModel( xml ) {
1363
+
1364
+ const data = {
1365
+ name: xml.getAttribute( 'name' ) || '',
1366
+ joints: {},
1367
+ links: []
1368
+ };
1369
+
1370
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1371
+
1372
+ const child = xml.childNodes[ i ];
1373
+
1374
+ if ( child.nodeType !== 1 ) continue;
1375
+
1376
+ switch ( child.nodeName ) {
1377
+
1378
+ case 'technique_common':
1379
+ this.parseKinematicsTechniqueCommon( child, data );
1380
+ break;
1381
+
1382
+ }
1383
+
1384
+ }
1385
+
1386
+ this.library.kinematicsModels[ xml.getAttribute( 'id' ) ] = data;
1387
+
1388
+ }
1389
+
1390
+ parseKinematicsTechniqueCommon( xml, data ) {
1391
+
1392
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1393
+
1394
+ const child = xml.childNodes[ i ];
1395
+
1396
+ if ( child.nodeType !== 1 ) continue;
1397
+
1398
+ switch ( child.nodeName ) {
1399
+
1400
+ case 'joint':
1401
+ data.joints[ child.getAttribute( 'sid' ) ] = this.parseKinematicsJoint( child );
1402
+ break;
1403
+
1404
+ case 'link':
1405
+ data.links.push( this.parseKinematicsLink( child ) );
1406
+ break;
1407
+
1408
+ }
1409
+
1410
+ }
1411
+
1412
+ }
1413
+
1414
+ parseKinematicsJoint( xml ) {
1415
+
1416
+ let data;
1417
+
1418
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1419
+
1420
+ const child = xml.childNodes[ i ];
1421
+
1422
+ if ( child.nodeType !== 1 ) continue;
1423
+
1424
+ switch ( child.nodeName ) {
1425
+
1426
+ case 'prismatic':
1427
+ case 'revolute':
1428
+ data = this.parseKinematicsJointParameter( child );
1429
+ break;
1430
+
1431
+ }
1432
+
1433
+ }
1434
+
1435
+ return data;
1436
+
1437
+ }
1438
+
1439
+ parseKinematicsJointParameter( xml ) {
1440
+
1441
+ const data = {
1442
+ sid: xml.getAttribute( 'sid' ),
1443
+ name: xml.getAttribute( 'name' ) || '',
1444
+ axis: new Vector3(),
1445
+ limits: {
1446
+ min: 0,
1447
+ max: 0
1448
+ },
1449
+ type: xml.nodeName,
1450
+ static: false,
1451
+ zeroPosition: 0,
1452
+ middlePosition: 0
1453
+ };
1454
+
1455
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1456
+
1457
+ const child = xml.childNodes[ i ];
1458
+
1459
+ if ( child.nodeType !== 1 ) continue;
1460
+
1461
+ switch ( child.nodeName ) {
1462
+
1463
+ case 'axis':
1464
+ const array = parseFloats( child.textContent );
1465
+ data.axis.fromArray( array );
1466
+ break;
1467
+ case 'limits':
1468
+ const max = child.getElementsByTagName( 'max' )[ 0 ];
1469
+ const min = child.getElementsByTagName( 'min' )[ 0 ];
1470
+
1471
+ data.limits.max = parseFloat( max.textContent );
1472
+ data.limits.min = parseFloat( min.textContent );
1473
+ break;
1474
+
1475
+ }
1476
+
1477
+ }
1478
+
1479
+ // if min is equal to or greater than max, consider the joint static
1480
+
1481
+ if ( data.limits.min >= data.limits.max ) {
1482
+
1483
+ data.static = true;
1484
+
1485
+ }
1486
+
1487
+ // calculate middle position
1488
+
1489
+ data.middlePosition = ( data.limits.min + data.limits.max ) / 2.0;
1490
+
1491
+ return data;
1492
+
1493
+ }
1494
+
1495
+ parseKinematicsLink( xml ) {
1496
+
1497
+ const data = {
1498
+ sid: xml.getAttribute( 'sid' ),
1499
+ name: xml.getAttribute( 'name' ) || '',
1500
+ attachments: [],
1501
+ transforms: []
1502
+ };
1503
+
1504
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1505
+
1506
+ const child = xml.childNodes[ i ];
1507
+
1508
+ if ( child.nodeType !== 1 ) continue;
1509
+
1510
+ switch ( child.nodeName ) {
1511
+
1512
+ case 'attachment_full':
1513
+ data.attachments.push( this.parseKinematicsAttachment( child ) );
1514
+ break;
1515
+
1516
+ case 'matrix':
1517
+ case 'translate':
1518
+ case 'rotate':
1519
+ data.transforms.push( this.parseKinematicsTransform( child ) );
1520
+ break;
1521
+
1522
+ }
1523
+
1524
+ }
1525
+
1526
+ return data;
1527
+
1528
+ }
1529
+
1530
+ parseKinematicsAttachment( xml ) {
1531
+
1532
+ const data = {
1533
+ joint: xml.getAttribute( 'joint' ).split( '/' ).pop(),
1534
+ transforms: [],
1535
+ links: []
1536
+ };
1537
+
1538
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1539
+
1540
+ const child = xml.childNodes[ i ];
1541
+
1542
+ if ( child.nodeType !== 1 ) continue;
1543
+
1544
+ switch ( child.nodeName ) {
1545
+
1546
+ case 'link':
1547
+ data.links.push( this.parseKinematicsLink( child ) );
1548
+ break;
1549
+
1550
+ case 'matrix':
1551
+ case 'translate':
1552
+ case 'rotate':
1553
+ data.transforms.push( this.parseKinematicsTransform( child ) );
1554
+ break;
1555
+
1556
+ }
1557
+
1558
+ }
1559
+
1560
+ return data;
1561
+
1562
+ }
1563
+
1564
+ parseKinematicsTransform( xml ) {
1565
+
1566
+ const data = {
1567
+ type: xml.nodeName
1568
+ };
1569
+
1570
+ const array = parseFloats( xml.textContent );
1571
+
1572
+ switch ( data.type ) {
1573
+
1574
+ case 'matrix':
1575
+ data.obj = new Matrix4();
1576
+ data.obj.fromArray( array ).transpose();
1577
+ break;
1578
+
1579
+ case 'translate':
1580
+ data.obj = new Vector3();
1581
+ data.obj.fromArray( array );
1582
+ break;
1583
+
1584
+ case 'rotate':
1585
+ data.obj = new Vector3();
1586
+ data.obj.fromArray( array );
1587
+ data.angle = MathUtils.degToRad( array[ 3 ] );
1588
+ break;
1589
+
1590
+ }
1591
+
1592
+ return data;
1593
+
1594
+ }
1595
+
1596
+ // physics
1597
+
1598
+ parsePhysicsModel( xml ) {
1599
+
1600
+ const data = {
1601
+ name: xml.getAttribute( 'name' ) || '',
1602
+ rigidBodies: {}
1603
+ };
1604
+
1605
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1606
+
1607
+ const child = xml.childNodes[ i ];
1608
+
1609
+ if ( child.nodeType !== 1 ) continue;
1610
+
1611
+ switch ( child.nodeName ) {
1612
+
1613
+ case 'rigid_body':
1614
+ data.rigidBodies[ child.getAttribute( 'name' ) ] = {};
1615
+ this.parsePhysicsRigidBody( child, data.rigidBodies[ child.getAttribute( 'name' ) ] );
1616
+ break;
1617
+
1618
+ }
1619
+
1620
+ }
1621
+
1622
+ this.library.physicsModels[ xml.getAttribute( 'id' ) ] = data;
1623
+
1624
+ }
1625
+
1626
+ parsePhysicsRigidBody( xml, data ) {
1627
+
1628
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1629
+
1630
+ const child = xml.childNodes[ i ];
1631
+
1632
+ if ( child.nodeType !== 1 ) continue;
1633
+
1634
+ switch ( child.nodeName ) {
1635
+
1636
+ case 'technique_common':
1637
+ this.parsePhysicsTechniqueCommon( child, data );
1638
+ break;
1639
+
1640
+ }
1641
+
1642
+ }
1643
+
1644
+ }
1645
+
1646
+ parsePhysicsTechniqueCommon( xml, data ) {
1647
+
1648
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1649
+
1650
+ const child = xml.childNodes[ i ];
1651
+
1652
+ if ( child.nodeType !== 1 ) continue;
1653
+
1654
+ switch ( child.nodeName ) {
1655
+
1656
+ case 'inertia':
1657
+ data.inertia = parseFloats( child.textContent );
1658
+ break;
1659
+
1660
+ case 'mass':
1661
+ data.mass = parseFloats( child.textContent )[ 0 ];
1662
+ break;
1663
+
1664
+ }
1665
+
1666
+ }
1667
+
1668
+ }
1669
+
1670
+ // scene
1671
+
1672
+ parseKinematicsScene( xml ) {
1673
+
1674
+ const data = {
1675
+ bindJointAxis: []
1676
+ };
1677
+
1678
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1679
+
1680
+ const child = xml.childNodes[ i ];
1681
+
1682
+ if ( child.nodeType !== 1 ) continue;
1683
+
1684
+ switch ( child.nodeName ) {
1685
+
1686
+ case 'bind_joint_axis':
1687
+ data.bindJointAxis.push( this.parseKinematicsBindJointAxis( child ) );
1688
+ break;
1689
+
1690
+ }
1691
+
1692
+ }
1693
+
1694
+ this.library.kinematicsScenes[ parseId( xml.getAttribute( 'url' ) ) ] = data;
1695
+
1696
+ }
1697
+
1698
+ parseKinematicsBindJointAxis( xml ) {
1699
+
1700
+ const data = {
1701
+ target: xml.getAttribute( 'target' ).split( '/' ).pop()
1702
+ };
1703
+
1704
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1705
+
1706
+ const child = xml.childNodes[ i ];
1707
+
1708
+ if ( child.nodeType !== 1 ) continue;
1709
+
1710
+ switch ( child.nodeName ) {
1711
+
1712
+ case 'axis':
1713
+ const param = child.getElementsByTagName( 'param' )[ 0 ];
1714
+ data.axis = param.textContent;
1715
+ const tmpJointIndex = data.axis.split( 'inst_' ).pop().split( 'axis' )[ 0 ];
1716
+ data.jointIndex = tmpJointIndex.substring( 0, tmpJointIndex.length - 1 );
1717
+ break;
1718
+
1719
+ }
1720
+
1721
+ }
1722
+
1723
+ return data;
1724
+
1725
+ }
1726
+
1727
+ // nodes
1728
+
1729
+ prepareNodes( xml ) {
1730
+
1731
+ const elements = xml.getElementsByTagName( 'node' );
1732
+
1733
+ // ensure all node elements have id attributes
1734
+
1735
+ for ( let i = 0; i < elements.length; i ++ ) {
1736
+
1737
+ const element = elements[ i ];
1738
+
1739
+ if ( element.hasAttribute( 'id' ) === false ) {
1740
+
1741
+ element.setAttribute( 'id', this.generateId() );
1742
+
1743
+ }
1744
+
1745
+ }
1746
+
1747
+ }
1748
+
1749
+ parseNode( xml ) {
1750
+
1751
+ const matrix = new Matrix4();
1752
+ const vector = new Vector3();
1753
+
1754
+ const data = {
1755
+ name: xml.getAttribute( 'name' ) || '',
1756
+ type: xml.getAttribute( 'type' ),
1757
+ id: xml.getAttribute( 'id' ),
1758
+ sid: xml.getAttribute( 'sid' ),
1759
+ matrix: new Matrix4(),
1760
+ nodes: [],
1761
+ instanceCameras: [],
1762
+ instanceControllers: [],
1763
+ instanceLights: [],
1764
+ instanceGeometries: [],
1765
+ instanceNodes: [],
1766
+ transforms: {},
1767
+ transformData: {},
1768
+ transformOrder: []
1769
+ };
1770
+
1771
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1772
+
1773
+ const child = xml.childNodes[ i ];
1774
+
1775
+ if ( child.nodeType !== 1 ) continue;
1776
+
1777
+ let array;
1778
+
1779
+ switch ( child.nodeName ) {
1780
+
1781
+ case 'node':
1782
+ data.nodes.push( child.getAttribute( 'id' ) );
1783
+ this.parseNode( child );
1784
+ break;
1785
+
1786
+ case 'instance_camera':
1787
+ data.instanceCameras.push( parseId( child.getAttribute( 'url' ) ) );
1788
+ break;
1789
+
1790
+ case 'instance_controller':
1791
+ data.instanceControllers.push( this.parseNodeInstance( child ) );
1792
+ break;
1793
+
1794
+ case 'instance_light':
1795
+ data.instanceLights.push( parseId( child.getAttribute( 'url' ) ) );
1796
+ break;
1797
+
1798
+ case 'instance_geometry':
1799
+ data.instanceGeometries.push( this.parseNodeInstance( child ) );
1800
+ break;
1801
+
1802
+ case 'instance_node':
1803
+ data.instanceNodes.push( parseId( child.getAttribute( 'url' ) ) );
1804
+ break;
1805
+
1806
+ case 'matrix':
1807
+ array = parseFloats( child.textContent );
1808
+ data.matrix.multiply( matrix.fromArray( array ).transpose() );
1809
+ {
1810
+
1811
+ const sid = child.getAttribute( 'sid' );
1812
+ data.transforms[ sid ] = child.nodeName;
1813
+ data.transformData[ sid ] = { type: 'matrix', array: array };
1814
+ data.transformOrder.push( sid );
1815
+
1816
+ }
1817
+
1818
+ break;
1819
+
1820
+ case 'translate':
1821
+ array = parseFloats( child.textContent );
1822
+ vector.fromArray( array );
1823
+ data.matrix.multiply( matrix.makeTranslation( vector.x, vector.y, vector.z ) );
1824
+ {
1825
+
1826
+ const sid = child.getAttribute( 'sid' );
1827
+ data.transforms[ sid ] = child.nodeName;
1828
+ data.transformData[ sid ] = { type: 'translate', x: array[ 0 ], y: array[ 1 ], z: array[ 2 ] };
1829
+ data.transformOrder.push( sid );
1830
+
1831
+ }
1832
+
1833
+ break;
1834
+
1835
+ case 'rotate':
1836
+ array = parseFloats( child.textContent );
1837
+ {
1838
+
1839
+ const angle = MathUtils.degToRad( array[ 3 ] );
1840
+ data.matrix.multiply( matrix.makeRotationAxis( vector.fromArray( array ), angle ) );
1841
+ const sid = child.getAttribute( 'sid' );
1842
+ data.transforms[ sid ] = child.nodeName;
1843
+ data.transformData[ sid ] = { type: 'rotate', axis: [ array[ 0 ], array[ 1 ], array[ 2 ] ], angle: array[ 3 ] };
1844
+ data.transformOrder.push( sid );
1845
+
1846
+ }
1847
+
1848
+ break;
1849
+
1850
+ case 'scale':
1851
+ array = parseFloats( child.textContent );
1852
+ data.matrix.scale( vector.fromArray( array ) );
1853
+ {
1854
+
1855
+ const sid = child.getAttribute( 'sid' );
1856
+ data.transforms[ sid ] = child.nodeName;
1857
+ data.transformData[ sid ] = { type: 'scale', x: array[ 0 ], y: array[ 1 ], z: array[ 2 ] };
1858
+ data.transformOrder.push( sid );
1859
+
1860
+ }
1861
+
1862
+ break;
1863
+
1864
+ case 'extra':
1865
+ break;
1866
+
1867
+ default:
1868
+
1869
+ }
1870
+
1871
+ }
1872
+
1873
+ if ( this.hasNode( data.id ) ) {
1874
+
1875
+ console.warn( 'THREE.ColladaLoader: There is already a node with ID %s. Exclude current node from further processing.', data.id );
1876
+
1877
+ } else {
1878
+
1879
+ this.library.nodes[ data.id ] = data;
1880
+
1881
+ }
1882
+
1883
+ return data;
1884
+
1885
+ }
1886
+
1887
+ parseNodeInstance( xml ) {
1888
+
1889
+ const data = {
1890
+ id: parseId( xml.getAttribute( 'url' ) ),
1891
+ materials: {},
1892
+ skeletons: []
1893
+ };
1894
+
1895
+ for ( let i = 0; i < xml.childNodes.length; i ++ ) {
1896
+
1897
+ const child = xml.childNodes[ i ];
1898
+
1899
+ switch ( child.nodeName ) {
1900
+
1901
+ case 'bind_material':
1902
+ const instances = child.getElementsByTagName( 'instance_material' );
1903
+
1904
+ for ( let j = 0; j < instances.length; j ++ ) {
1905
+
1906
+ const instance = instances[ j ];
1907
+ const symbol = instance.getAttribute( 'symbol' );
1908
+ const target = instance.getAttribute( 'target' );
1909
+
1910
+ data.materials[ symbol ] = parseId( target );
1911
+
1912
+ }
1913
+
1914
+ break;
1915
+
1916
+ case 'skeleton':
1917
+ data.skeletons.push( parseId( child.textContent ) );
1918
+ break;
1919
+
1920
+ default:
1921
+ break;
1922
+
1923
+ }
1924
+
1925
+ }
1926
+
1927
+ return data;
1928
+
1929
+ }
1930
+
1931
+ // visual scenes
1932
+
1933
+ parseVisualScene( xml ) {
1934
+
1935
+ const data = {
1936
+ name: xml.getAttribute( 'name' ),
1937
+ children: []
1938
+ };
1939
+
1940
+ this.prepareNodes( xml );
1941
+
1942
+ const elements = getElementsByTagName( xml, 'node' );
1943
+
1944
+ for ( let i = 0; i < elements.length; i ++ ) {
1945
+
1946
+ data.children.push( this.parseNode( elements[ i ] ) );
1947
+
1948
+ }
1949
+
1950
+ this.library.visualScenes[ xml.getAttribute( 'id' ) ] = data;
1951
+
1952
+ }
1953
+
1954
+ hasNode( id ) {
1955
+
1956
+ return this.library.nodes[ id ] !== undefined;
1957
+
1958
+ }
1959
+
1960
+ }
1961
+
1962
+ export { ColladaParser, getElementsByTagName, parseStrings, parseFloats, parseInts, parseId };