Home Reference Source Repository

js/mmd/MMDAnimator.js

  1. 'use strict'
  2.  
  3. import Animator from '../base/Animator'
  4.  
  5. /**
  6. * MMDAnimator class
  7. * @access public
  8. */
  9. export default class MMDAnimator extends Animator {
  10.  
  11. updateMotion(dhObject, elapsedTime) {
  12. const model = dhObject._model
  13. const motion = dhObject._motion
  14. const time = elapsedTime * dhObject._animationSpeed
  15.  
  16. if(dhObject._animating){
  17. dhObject._animationTime += time
  18. const frameNo = dhObject._animationTime * motion.defaultFPS
  19. if(frameNo > motion.frameLength){
  20. if(dhObject._loop){
  21. dhObject._animationTime -= motion.frameLength / motion.defaultFPS
  22. }else{
  23. dhObject._animationTime = motion.frameLength / motion.defaultFPS
  24. dhObject._animating = false
  25. }
  26. }
  27. if(dhObject._motionBlendStep){
  28. dhObject._motionBlendCount -= dhObject._motionBlendStep * motion.defaultFPS * time
  29. if(dhObject._motionBlendCount < 0){
  30. dhObject._motionBlendCount = 0
  31. dhObject._motionBlendStep = 0
  32. }
  33. }
  34. }
  35. dhObject._animationFrame = dhObject._animationTime * motion.defaultFPS
  36.  
  37. const animator = this
  38. motion.motionArray.forEach( (boneMotion, key) => {
  39. const bone = model.boneHash.get(key)
  40. if(!bone){
  41. return
  42. }
  43.  
  44. animator.setBone(dhObject, boneMotion, bone)
  45. })
  46.  
  47. if(model.faceArray.length > 0){
  48. model.faceArray[0].setFace(model)
  49. motion.faceMotionArray.forEach( (faceMotion, key) => {
  50. const face = model.faceHash.get(key)
  51. if(!face)
  52. return
  53.  
  54. animator.setFace(dhObject, faceMotion, face)
  55. })
  56. }
  57. }
  58.  
  59. setBone(dhObject, motion, bone) {
  60. if(!dhObject._motionNumCache)
  61. dhObject._motionNumCache = new Map()
  62.  
  63. let frameNo = dhObject._motionNumCache.get(bone.name)
  64. const time = dhObject._animationFrame
  65. const motionLen = motion.length
  66. let key0 = 0
  67. let key1 = 0
  68.  
  69. if(!frameNo)
  70. frameNo = 0
  71.  
  72. if(motionLen <= frameNo)
  73. frameNo = motionLen - 1
  74.  
  75. if(motion[frameNo].frameNo < time){
  76. for(; frameNo < motionLen; frameNo++){
  77. if(motion[frameNo].frameNo >= time)
  78. break
  79. }
  80. key0 = frameNo - 1
  81. key1 = frameNo
  82. }else{
  83. for(; frameNo >= 0; frameNo--){
  84. if(motion[frameNo].frameNo < time)
  85. break
  86. }
  87. key0 = frameNo
  88. key1 = frameNo + 1
  89. }
  90. if(key0 <= 0) key0 = 0
  91. if(key1 >= motionLen) key1 = motionLen - 1
  92.  
  93. // cache motion number
  94. dhObject._motionNumCache.set(bone.name, key0)
  95.  
  96. const motion0 = motion[key0]
  97. const motion1 = motion[key1]
  98. const time0 = motion0.frameNo
  99. const time1 = motion1.frameNo
  100.  
  101. const pos = bone.position
  102. const rot = bone.rotate
  103. if(time0 !== time1){
  104. const k = 127 * (time - time0) / (time1 - time0)
  105. pos.x = this.getBezierValue(motion1.interpolation[0], motion1.interpolation[4],
  106. motion1.interpolation[8], motion1.interpolation[12],
  107. motion0.position.x, motion1.position.x, k)
  108. pos.y = this.getBezierValue(motion1.interpolation[1], motion1.interpolation[5],
  109. motion1.interpolation[9], motion1.interpolation[13],
  110. motion0.position.y, motion1.position.y, k)
  111. pos.z = this.getBezierValue(motion1.interpolation[2], motion1.interpolation[6],
  112. motion1.interpolation[10], motion1.interpolation[14],
  113. motion0.position.z, motion1.position.z, k)
  114. const r = this.getBezierValue(motion1.interpolation[3], motion1.interpolation[7],
  115. motion1.interpolation[11], motion1.interpolation[15],
  116. 0, 1, k)
  117. // DEBUG
  118. //rot.slerp(motion0.rotate, motion1.rotate, r)
  119. rot.lerp(motion0.rotate, motion1.rotate, r)
  120. rot.normalize()
  121. }else{
  122. pos.setValue(motion0.position)
  123. rot.setValue(motion0.rotate)
  124. }
  125.  
  126. if(dhObject._motionBlendCount){
  127. pos.lerp(pos, bone.blendPosition, dhObject._motionBlendCount)
  128. rot.lerp(rot, bone.blendRotation, dhObject._motionBlendCount)
  129. }
  130. }
  131.  
  132. getBezierValue(bx1, by1, bx2, by2, y0, y1, k) {
  133. let r = 0
  134. let val = 0
  135. let t0 = 0
  136. let t1 = 127 / 127.0
  137. let t = 63.5 / 127.0
  138. const nx1 = bx1 / 127.0
  139. const ny1 = by1 / 127.0
  140. const nx2 = bx2 / 127.0
  141. const ny2 = by2 / 127.0
  142. const nk = k / 127.0
  143.  
  144. for(let i=0; i<8; i++){
  145. r = 1-t
  146. val = 3*t*r*(nx1*r + nx2*t) + t*t*t
  147. if(nk > val){
  148. t0 = t
  149. }else{
  150. t1 = t
  151. }
  152. t = (t0 + t1) / 2
  153. }
  154. r = 1-t
  155. val = (3*t*r*(ny1*r + ny2*t) + t*t*t)
  156.  
  157. return (y0 + (y1 - y0) * val)
  158. }
  159.  
  160. setFace(dhObject, faceMotion, face) {
  161. if(!dhObject._faceMotionNumCache)
  162. dhObject._faceMotionNumCache = new Map()
  163.  
  164. let frameNo = dhObject._faceMotionNumCache.get(face.name)
  165. let time = dhObject._animationFrame
  166. const motionLen = faceMotion.length
  167. let key0 = 0
  168. let key1 = 0
  169.  
  170. if(!frameNo)
  171. frameNo = 0
  172.  
  173. if(time > faceMotion[motionLen-1].frameNo){
  174. time = faceMotion[motionLen-1].frameNo
  175. }
  176.  
  177. if(faceMotion[frameNo].frameNo < time){
  178. for(; frameNo < motionLen; frameNo++){
  179. if(faceMotion[frameNo].frameNo >= time)
  180. break
  181. }
  182. key0 = frameNo - 1
  183. key1 = frameNo
  184. }else{
  185. for(; frameNo >= 0; frameNo--){
  186. if(faceMotion[frameNo].frameNo < time)
  187. break
  188. }
  189. key0 = frameNo
  190. key1 = frameNo + 1
  191. }
  192. if(key0 < 0) key0 = 0
  193. if(key1 >= motionLen) key1 = motionLen - 1
  194.  
  195. const motion0 = faceMotion[key0]
  196. const motion1 = faceMotion[key1]
  197. const time0 = motion0.frameNo
  198. const time1 = motion1.frameNo
  199.  
  200. let rate = 0
  201. if(time0 !== time1){
  202. const k = (time1 - time) / (time1 - time0)
  203. rate = motion0.factor * k + motion1.factor * (1.0 - k)
  204. }else{
  205. rate = motion0.factor
  206. }
  207. face.blendFace(dhObject._model, rate)
  208. }
  209. }