using System.Collections.Generic; using System.Linq; using UniJSON; namespace UniVRM10 { public static class MigrationVrmSpringBone { static IEnumerable EnumJoint(List nodes, UniGLTF.glTFNode node) { yield return node; if (node.children != null && node.children.Length > 0) { foreach (var x in EnumJoint(nodes, nodes[node.children[0]])) { yield return x; } } } static float[] ReverseZ(JsonNode xyz) { return new float[]{ xyz["x"].GetSingle(), xyz["y"].GetSingle(), -xyz["z"].GetSingle(), }; } public static UniGLTF.Extensions.VRMC_springBone.VRMC_springBone Migrate(UniGLTF.glTF gltf, JsonNode sa) { var colliderNodes = new List(); foreach (var x in sa["colliderGroups"].ArrayItems()) { var node = x["node"].GetInt32(); colliderNodes.Add(node); var gltfNode = gltf.nodes[node]; var collider = new UniGLTF.Extensions.VRMC_node_collider.VRMC_node_collider() { Shapes = new List(), }; // { // "node": 14, // "colliders": [ // { // "offset": { // "x": 0.025884293, // "y": -0.120000005, // "z": 0 // }, // "radius": 0.05 // }, // { // "offset": { // "x": -0.02588429, // "y": -0.120000005, // "z": 0 // }, // "radius": 0.05 // }, // { // "offset": { // "x": 0, // "y": -0.0220816135, // "z": 0 // }, // "radius": 0.08 // } // ] // }, foreach (var y in x["colliders"].ArrayItems()) { collider.Shapes.Add(new UniGLTF.Extensions.VRMC_node_collider.ColliderShape { Sphere = new UniGLTF.Extensions.VRMC_node_collider.ColliderShapeSphere { Offset = ReverseZ(y["offset"]), Radius = y["radius"].GetSingle() } }); } if (!(gltfNode.extensions is UniGLTF.glTFExtensionExport extensions)) { extensions = new UniGLTF.glTFExtensionExport(); gltfNode.extensions = extensions; } var f = new JsonFormatter(); UniGLTF.Extensions.VRMC_node_collider.GltfSerializer.Serialize(f, collider); extensions.Add(UniGLTF.Extensions.VRMC_node_collider.VRMC_node_collider.ExtensionName, f.GetStoreBytes()); } var springBone = new UniGLTF.Extensions.VRMC_springBone.VRMC_springBone { Springs = new List(), }; foreach (var x in sa["boneGroups"].ArrayItems()) { // { // "comment": "", // "stiffiness": 2, // "gravityPower": 0, // "gravityDir": { // "x": 0, // "y": -1, // "z": 0 // }, // "dragForce": 0.7, // "center": -1, // "hitRadius": 0.02, // "bones": [ // 97, // 99, // 101, // 113, // 114 // ], // "colliderGroups": [ // 3, // 4, // 5 // ] // }, foreach (var y in x["bones"].ArrayItems()) { var comment = x.GetObjectValueOrDefault("comment", ""); var spring = new UniGLTF.Extensions.VRMC_springBone.Spring { Name = comment, Colliders = x["colliderGroups"].ArrayItems().Select(z => colliderNodes[z.GetInt32()]).ToArray(), Joints = new List(), }; foreach (var z in EnumJoint(gltf.nodes, gltf.nodes[y.GetInt32()])) { spring.Joints.Add(new UniGLTF.Extensions.VRMC_springBone.SpringBoneJoint { Node = gltf.nodes.IndexOf(z), DragForce = x["dragForce"].GetSingle(), GravityDir = ReverseZ(x["gravityDir"]), GravityPower = x["gravityPower"].GetSingle(), HitRadius = x["hitRadius"].GetSingle(), Stiffness = x["stiffiness"].GetSingle(), }); } springBone.Springs.Add(spring); } } return springBone; } } }