<html xmlns:o="urn:schemas-microsoft-com:office:office" xmlns:w="urn:schemas-microsoft-com:office:word" xmlns:m="http://schemas.microsoft.com/office/2004/12/omml" xmlns="http://www.w3.org/TR/REC-html40"><head><meta http-equiv=Content-Type content="text/html; charset=utf-8"><meta name=Generator content="Microsoft Word 15 (filtered medium)"><style><!--
/* Font Definitions */
@font-face
{font-family:Wingdings;
panose-1:5 0 0 0 0 0 0 0 0 0;}
@font-face
{font-family:"Cambria Math";
panose-1:2 4 5 3 5 4 6 3 2 4;}
@font-face
{font-family:Calibri;
panose-1:2 15 5 2 2 2 4 3 2 4;}
/* Style Definitions */
p.MsoNormal, li.MsoNormal, div.MsoNormal
{margin:0in;
margin-bottom:.0001pt;
font-size:11.0pt;
font-family:"Calibri",sans-serif;}
a:link, span.MsoHyperlink
{mso-style-priority:99;
color:blue;
text-decoration:underline;}
a:visited, span.MsoHyperlinkFollowed
{mso-style-priority:99;
color:#954F72;
text-decoration:underline;}
p.MsoListParagraph, li.MsoListParagraph, div.MsoListParagraph
{mso-style-priority:34;
margin-top:0in;
margin-right:0in;
margin-bottom:0in;
margin-left:.5in;
margin-bottom:.0001pt;
font-size:11.0pt;
font-family:"Calibri",sans-serif;}
.MsoChpDefault
{mso-style-type:export-only;}
@page WordSection1
{size:8.5in 11.0in;
margin:1.0in 1.0in 1.0in 1.0in;}
div.WordSection1
{page:WordSection1;}
/* List Definitions */
@list l0
{mso-list-id:202179825;
mso-list-type:hybrid;
mso-list-template-ids:1632669066 -1 67698691 67698693 67698689 67698691 67698693 67698689 67698691 67698693;}
@list l0:level1
{mso-level-start-at:0;
mso-level-number-format:bullet;
mso-level-text:\F0D8;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Wingdings;
mso-fareast-font-family:"Times New Roman";
mso-bidi-font-family:"Times New Roman";}
@list l0:level2
{mso-level-number-format:bullet;
mso-level-text:o;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:"Courier New";}
@list l0:level3
{mso-level-number-format:bullet;
mso-level-text:\F0A7;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Wingdings;}
@list l0:level4
{mso-level-number-format:bullet;
mso-level-text:\F0B7;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Symbol;}
@list l0:level5
{mso-level-number-format:bullet;
mso-level-text:o;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:"Courier New";}
@list l0:level6
{mso-level-number-format:bullet;
mso-level-text:\F0A7;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Wingdings;}
@list l0:level7
{mso-level-number-format:bullet;
mso-level-text:\F0B7;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Symbol;}
@list l0:level8
{mso-level-number-format:bullet;
mso-level-text:o;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:"Courier New";}
@list l0:level9
{mso-level-number-format:bullet;
mso-level-text:\F0A7;
mso-level-tab-stop:none;
mso-level-number-position:left;
text-indent:-.25in;
font-family:Wingdings;}
ol
{margin-bottom:0in;}
ul
{margin-bottom:0in;}
--></style></head><body lang=EN-US link=blue vlink="#954F72"><div class=WordSection1><ul style='margin-top:0in' type=disc><li class=MsoListParagraph style='margin-left:0in;mso-list:l0 level1 lfo1'>I do think glTF allows that mesh vertices are affected by both, morph targets and joints, at least nothing is precluding it. It does not seem explicit about what effect is applied first, perhaps it is implicit that morphing is first.</li></ul><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>HAnim is clear: First find Joint(s), then find Displacer(s). What if multiple Displacer moves on same vertex, then maybe order of appearance in user code. Certainly the skin deformation must be applied in certain order - first parents, then child, down the line. Every cycle the vertex gets moved by new joint data and this file in its inial state would define a pose. I think in detailed surface animations, like maybe face, it is still mainly a simple set of basic joints, enhanced by various displacers and you want to be able to easily read the data you are authoring. . </p><p class=MsoNormal> </p><p class=MsoNormal>There is no reason to expect that any gltf would associate order about which effect(s) are applied first. </p><p class=MsoNormal>That is for author and finally the runtime. Hey, it probably was a big deal with deep negotiations to define a gltf package that takes the step of being able to associate a vertex of a mesh with data set of joints and weights. There is a lot you need to know about the target to be animated in order to use this little package in realtime runtime. Read one for a piece of hair with 500 joints and 5000 vertices. A certain binding convenience for the runtime is provided vey expllicitly. </p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>There are other package(s) somewhere that associates the joint locations and hierarchies an others. In fact, this choice of contents is a most secure way of transporting carefully-guarded animation data, because unless you get more, exactly as expected, you know almost nothing. </p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>The author has to already understand what to do with the package he gets. For example, to import a certain style of gltf, x3d importer uses those components of the hanim in place of expected user code. An author importing such a file would know that he won’t need to supply skincordindex and skincoordweight for each Joint, and maybe skin field could be empty. This could be a real authoring convenience (but now don’t change the skeleton or nothing may work until bindings fixed). So you know what you are importing (like maybe you just got new skin with different res for your old skeleton so you need new bindings) and the sender is sure of what he is sending re: target skeleton and target skin.</p><p class=MsoNormal> </p><p class=MsoNormal>If you know enough about the humanoid, you can look at this list either way or both ways, and in action. In fact, the only way you can do deep troubleshooting is look at the skin-skeleton binding(s) both ways!</p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>For text-driven authortime, let’s give a list of joints that out front shows the joint to skin vertex bindings. Due to the hierarchy this is really the first and most important authoring step (of course some authoring views may show the vertex association with a segment (or bone)). In reality, the joints mainly run the show. So, this is a way to structure a data package for one part of one step (now so simple) of best practice industry transport of one or more parts of the data set required to produce an animated humanoid? I’m very certain that for a certain set of transport operations, this a great way to save and transport a part of the show, maybe just a certain pose or the basis for further animation. </p><p class=MsoNormal><o:p> </o:p></p><ul style='margin-top:0in' type=disc><li class=MsoListParagraph style='margin-left:0in;mso-list:l0 level1 lfo1'> … what effect is applied first</li></ul><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>Of course this is critical that there is an order. And this just gives one step in vertex operations - deforming the mesh any deformation order may be possible, maybe a switch somewhere, but ‘standard’ hanim is first you get the deformation due to joint(s), then you apply the displacer(s). </p><p class=MsoNormal>As is necessary, the author must be able to depend upon the order applied since in the final result all this ultimately, depending upon your persistence, needs to be carefully hacked in user code, or whatever is presented to you as user code in whatever tool you are driving to create your humanoid. I say just recall we are looking at stuff that should qualify as industry standard best practices for inclusion in x3d. Since all the data is the same for what hanim does, it is just the names and style of organizing the data, except in the case that hanim only knows about axis-angle and not quats, which we really ought to fix in x3d. If you have skin in the game of skeletal animation, you want to create, edit, document, and transport animation using quaternions. </p><p class=MsoNormal><o:p> </o:p></p><ul style='margin-top:0in' type=disc><li class=MsoListParagraph style='margin-left:0in;mso-list:l0 level1 lfo1'>It is possible to map many of the glTF data structures to X3D and HAnim data structures but this ability is unrelated to data types or binary encoding. X3D and HAnim provide enough flexibility to fit glTF. The reverse is not always the case.</li></ul><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>Right, there is much ‘non-standard’ transport of certain elements of this highly prized professional exchange of skeleton and displacer style animations. This particular gltf example is just one that is actually one form of the most basic data you gotta send to a model you know very well to set a pose. All the bigs agree that this formulation is verifiable and secure and is practically worthless by itself. Really, to import a compete skin and skeleton and bindings and animations you need a grandlist of objects that the runtime knows it can just drop into certain buckets and go. In x3d the author defines those buckets in a certain hierarchy using certain objects with expectations of certain results. There is always some new stuff (for instance at one point there was discussion of defining some kind of event-driven runtime for an example gltf eater) and always new ways of looking at old stuff so let’s try to bring some of that into the hanim realm. <o:p></o:p></p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal>All Best Wishes,</p><p class=MsoNormal>Joe</p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal><o:p> </o:p></p><p class=MsoNormal><o:p> </o:p></p><div style='mso-element:para-border-div;border:none;border-top:solid #E1E1E1 1.0pt;padding:3.0pt 0in 0in 0in'><p class=MsoNormal style='border:none;padding:0in'><b>From: </b><a href="mailto:andreasplesch@gmail.com">Andreas Plesch</a><br><b>Sent: </b>Friday, November 30, 2018 6:20 AM<br><b>To: </b><a href="mailto:x3d-public@web3d.org">X3D Graphics public mailing list</a><br><b>Subject: </b>Re: [x3d-public] HAnim and glTF skins</p></div><p class=MsoNormal><o:p> </o:p></p><div><div><p class=MsoNormal>Joe, excellent points. See below.<br><br>> Date: Thu, 29 Nov 2018 11:47:34 -0800<br>> From: Joseph D Williams <<a href="mailto:joedwil@earthlink.net" target="_blank">joedwil@earthlink.net</a>><br>> To: John Carlson <<a href="mailto:yottzumm@gmail.com" target="_blank">yottzumm@gmail.com</a>>, GPU Group <<a href="mailto:gpugroup@gmail.com" target="_blank">gpugroup@gmail.com</a>><br>> Cc: X3D Graphics public mailing list <<a href="mailto:x3d-public@web3d.org" target="_blank">x3d-public@web3d.org</a>><br>> Subject: Re: [x3d-public] HAnim and glTF skins<br>><br>> ? Andreas - Another feature in glTF are morph targets which are independent of skins. They are similar to HAnim displacers but I have not looked very closely if it is possible to define a mapping between them.<br>><br>> The thing that maybe special in HAnim is that the displacers applied to skin are additive to other mesh animation. I think you will find that morph thingees will be used with other skin animation techniquess<br><br>I do think glTF allows that mesh vertices are affected by both, morph targets and joints, at least nothing is precluding it. It does not seem explicit about what effect is applied first, perhaps it is implicit that morphing is first.<br><br>><br>> HAnimDisplacer<br>> coordIndex [ numerical index of each parent geometry vertex<br>> to be animated in the same order as they appear<br>> in the user code for the target geometry ]<br>> displacements [ x,y,z in skeleton space<br>> maximum displacement of each vertex ]<br>> weight 0 to 1 animation control<br>> (linear interpolation)<br>> 0 = same location as parent mesh initial vertex<br>> 1 = defined maximum displacement location of parent mesh vertex<br>><br>> Parent geometry is either the Humanoid, if the Displacer is applied to the mesh in the skin node, or, the geometry of the Displacer node's parent Segment. The displacements xyz data gives a relative target maximum displacement, sort of like there is vector that points from the current vertex position to the location of maximum displacement and the current vertex is moved along that vector according to the weight input.</p><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>That is identical to morph targets in glTF. There, for some reason, the weights are defined in the Node (transform) around the morphed mesh.</p></div><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>In addition, morph targets also define normal and tangent 'displacements', so they do not need to be recomputed if available.</p></div><div><p class=MsoNormal><br>><br>> Displacer nodes require detailed knowledge of the parent mesh. Specifically, the vertex order of the parent mesh must not change between authoring and rendering. Compared to CoordinateInterpolator, for example, since all vertices of the mesh may not need to be animated for a sequence, only the vertices to be moved are included. If the mesh is being animated by another process, then those results and all displacer(s) results must be applied to the entire mesh in the same frame.</p></div><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>I think, in glTF all vertices need a corresponding target. It is probably cheaper on the GPU just to displace all than doing the work necessary to skip some. </p></div><div><p class=MsoNormal><br>><br>> In fact, what would be the problem, since the name of the node in X3D is HAnimDisplacer, the name Displacer is available for x3d, Why not figure out how to do at least a simple one for V4, or change the name. Canvas the industry and see the various names they call these things, then pick something different. The data and execution of the displacer is like other things called various names that so the same thing, as it has always been.</p></div><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>Displacer could be a GPU friendly, simple Coordinate/NormalInterpolator.</p></div><div><p class=MsoNormal><br>><br>> Also, if a good goal is to bring in some more simplified form of articulation-driven mesh animation, why not define a simple root-Joints-segments-skin hierarchy setup using some plain names for the animation nodes and skin bindings. I mean, whatever. we might as well call a joint a rotary, fixed center of rotation actuator. And, of course we need joint-like things, only the most perverse (mis)understanding of the fact that bone orientation the same as joint rotation wants to do away with joints.<br>><br>> I can see why a gltf vertex contains a list of actuators and weights while hanim uses the actuator to hold the list of vertices and weights of the mesh it animates. Some gltf code may show a minimum of 16 actuaor realtionships per vertex is ?standard? with bunches of zeros, beca the hdwr wants it that way. That is wasteful and from what I have seen x3d would not make it the prime directive that the authortime has to look like the runtime. That has always been. There is runtime and there is authortime. HAnim, is concerned with composability and clear documentation. Sure, it may depend upon how you learned it, but for authortime, if you take care of it at the jointss, then the mesh will be ok.</p></div><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>I agree that it should be the job of the engine/browser to make authoring/editing easy while still providing flexibility and power. However, the tradeoff is that this makes implementations harder or perhaps completely unfeasible in some cases.</p></div><div><p class=MsoNormal><br>><br>> The gltf binary is for under the covers, more for rendering than for authoring. The runtimes want the data in gltf form and hanim says that is fine, take the way we do it and create your binary, maybe using a style sheet, or vice-versa. Every current x3d runtime converts the x3d utf-8 text to a target binary form at some point. That is why a set of properly structured binaries might be able to be imported rather easily because the runtimes probably use that form anyway, just input in different, more abstractly-connected containers than x3d hanim. The names and readability are different while the data, the data structures, and the necessary connections are the same under the covers. I hope all the gltf and x3d data types match. gltf is fine, as long as I can edit the source.</p></div><div><p class=MsoNormal><o:p> </o:p></p></div><div><p class=MsoNormal>It is possible to map many of the glTF data structures to X3D and HAnim data structures but this ability is unrelated to data types or binary encoding. X3D and HAnim provide enough flexibility to fit glTF. The reverse is not always the case.</p></div></div></div><p class=MsoNormal style='margin-bottom:12.0pt'><br>-Andreas</p><p class=MsoNormal><o:p> </o:p></p></div></body></html>