From e8226d94bee1e408e64d67dbeeab10e142573fec Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Mon, 1 Jul 2019 13:05:58 +0200 Subject: [PATCH] Bugfix: respect the original bone's rotation order in animation keyframes. --- src/data_formats/parsing/ninja/index.ts | 87 ++++++++++++++++------- src/data_formats/parsing/ninja/motion.ts | 8 +-- src/rendering/ModelRenderer.ts | 8 +-- src/rendering/animation.ts | 11 ++- src/stores/ModelViewerStore.ts | 33 ++++----- src/ui/model_viewer/RendererComponent.tsx | 4 +- 6 files changed, 92 insertions(+), 59 deletions(-) diff --git a/src/data_formats/parsing/ninja/index.ts b/src/data_formats/parsing/ninja/index.ts index e3177445..4c7a89b2 100644 --- a/src/data_formats/parsing/ninja/index.ts +++ b/src/data_formats/parsing/ninja/index.ts @@ -1,7 +1,7 @@ -import { BufferCursor } from '../../BufferCursor'; -import { parse_nj_model, NjModel } from './nj'; -import { parse_xj_model, XjModel } from './xj'; import { Vec3 } from '../../../domain'; +import { BufferCursor } from '../../BufferCursor'; +import { NjModel, parse_nj_model } from './nj'; +import { parse_xj_model, XjModel } from './xj'; // TODO: // - deal with multiple NJCM chunks @@ -19,22 +19,57 @@ export type NinjaVertex = { export type NinjaModel = NjModel | XjModel; -export type NinjaObject = { - evaluation_flags: { - no_translate: boolean, - no_rotate: boolean, - no_scale: boolean, - hidden: boolean, - break_child_trace: boolean, - zxy_rotation_order: boolean, - skip: boolean, - shape_skip: boolean, - }, - model?: M, - position: Vec3, - rotation: Vec3, // Euler angles in radians. - scale: Vec3, - children: NinjaObject[], +export class NinjaObject { + private bone_cache = new Map | null>(); + + constructor( + public evaluation_flags: { + no_translate: boolean, + no_rotate: boolean, + no_scale: boolean, + hidden: boolean, + break_child_trace: boolean, + zxy_rotation_order: boolean, + skip: boolean, + shape_skip: boolean, + }, + public model: M | undefined, + public position: Vec3, + public rotation: Vec3, // Euler angles in radians. + public scale: Vec3, + public children: NinjaObject[] + ) { } + + find_bone(bone_id: number): NinjaObject | undefined { + let bone = this.bone_cache.get(bone_id); + + // Strict check because null means there's no bone with this id. + if (bone === undefined) { + bone = this.find_bone_internal(this, bone_id, [0]); + this.bone_cache.set(bone_id, bone || null); + } + + return bone || undefined; + } + + private find_bone_internal( + object: NinjaObject, + bone_id: number, + id_ref: [number] + ): NinjaObject | undefined { + if (!object.evaluation_flags.skip) { + const id = id_ref[0]++; + + if (id === bone_id) { + return object; + } + } + + for (const child of object.children) { + const bone = this.find_bone_internal(child, bone_id, id_ref); + if (bone) return bone; + } + } } export function parse_nj(cursor: BufferCursor): NinjaObject[] { @@ -119,8 +154,8 @@ function parse_sibling_objects( siblings = []; } - const object: NinjaObject = { - evaluation_flags: { + const object = new NinjaObject( + { no_translate, no_rotate, no_scale, @@ -131,11 +166,11 @@ function parse_sibling_objects( shape_skip, }, model, - position: new Vec3(pos_x, pos_y, pos_z), - rotation: new Vec3(rotation_x, rotation_y, rotation_z), - scale: new Vec3(scale_x, scale_y, scale_z), - children, - }; + new Vec3(pos_x, pos_y, pos_z), + new Vec3(rotation_x, rotation_y, rotation_z), + new Vec3(scale_x, scale_y, scale_z), + children + ); return [object, ...siblings]; } diff --git a/src/data_formats/parsing/ninja/motion.ts b/src/data_formats/parsing/ninja/motion.ts index 77863685..84a98563 100644 --- a/src/data_formats/parsing/ninja/motion.ts +++ b/src/data_formats/parsing/ninja/motion.ts @@ -1,7 +1,7 @@ import { BufferCursor } from '../../BufferCursor'; import { Vec3 } from '../../../domain'; -const ANGLE_TO_RAD = 2 * Math.PI / 65536; +const ANGLE_TO_RAD = 2 * Math.PI / 0xFFFF; export type NjAction = { object_offset: number, @@ -123,7 +123,7 @@ function parse_motion(cursor: BufferCursor): NjMotion { } // NJD_MTYPE_POS_0 - if ((type & (1 << 0)) !== 0) { + if (type & (1 << 0)) { cursor.seek_start(keyframe_offsets.shift()!); const count = keyframe_counts.shift(); @@ -136,7 +136,7 @@ function parse_motion(cursor: BufferCursor): NjMotion { } // NJD_MTYPE_ANG_1 - if ((type & (1 << 1)) !== 0) { + if (type & (1 << 1)) { cursor.seek_start(keyframe_offsets.shift()!); const count = keyframe_counts.shift(); @@ -149,7 +149,7 @@ function parse_motion(cursor: BufferCursor): NjMotion { } // NJD_MTYPE_SCL_2 - if ((type & (1 << 2)) !== 0) { + if (type & (1 << 2)) { cursor.seek_start(keyframe_offsets.shift()!); const count = keyframe_counts.shift(); diff --git a/src/rendering/ModelRenderer.ts b/src/rendering/ModelRenderer.ts index e56be2e4..a6564a8e 100644 --- a/src/rendering/ModelRenderer.ts +++ b/src/rendering/ModelRenderer.ts @@ -1,7 +1,7 @@ -import { Object3D, Vector3, Clock, SkeletonHelper } from "three"; +import { autorun } from "mobx"; +import { Clock, SkeletonHelper, SkinnedMesh, Vector3 } from "three"; import { model_viewer_store } from "../stores/ModelViewerStore"; import { Renderer } from "./Renderer"; -import { autorun } from "mobx"; let renderer: ModelRenderer | undefined; @@ -13,7 +13,7 @@ export function get_model_renderer(): ModelRenderer { export class ModelRenderer extends Renderer { private clock = new Clock(); - private model?: Object3D; + private model?: SkinnedMesh; private skeleton_helper?: SkeletonHelper; constructor() { @@ -27,7 +27,7 @@ export class ModelRenderer extends Renderer { }); } - set_model(model?: Object3D) { + set_model(model?: SkinnedMesh) { if (this.model !== model) { if (this.model) { this.scene.remove(this.model); diff --git a/src/rendering/animation.ts b/src/rendering/animation.ts index f6b23198..e07a7904 100644 --- a/src/rendering/animation.ts +++ b/src/rendering/animation.ts @@ -1,9 +1,13 @@ import { AnimationClip, Euler, InterpolateLinear, InterpolateSmooth, KeyframeTrack, Quaternion, QuaternionKeyframeTrack, VectorKeyframeTrack } from "three"; import { NjAction, NjInterpolation, NjKeyframeTrackType } from "../data_formats/parsing/ninja/motion"; +import { NinjaObject, NinjaModel } from "../data_formats/parsing/ninja"; export const PSO_FRAME_RATE = 30; -export function create_animation_clip(action: NjAction): AnimationClip { +export function create_animation_clip( + object: NinjaObject, + action: NjAction +): AnimationClip { const motion = action.motion; const interpolation = motion.interpolation === NjInterpolation.Spline ? InterpolateSmooth @@ -12,6 +16,8 @@ export function create_animation_clip(action: NjAction): AnimationClip { const tracks: KeyframeTrack[] = []; motion.motion_data.forEach((motion_data, bone_id) => { + const bone = object.find_bone(bone_id); + motion_data.tracks.forEach(({ type, keyframes }) => { const times: number[] = []; const values: number[] = []; @@ -20,8 +26,9 @@ export function create_animation_clip(action: NjAction): AnimationClip { times.push(keyframe.frame / PSO_FRAME_RATE); if (type === NjKeyframeTrackType.Rotation) { + const order = bone && bone.evaluation_flags.zxy_rotation_order ? 'ZXY' : 'ZYX'; const quat = new Quaternion().setFromEuler( - new Euler(keyframe.value.x, keyframe.value.y, keyframe.value.z) + new Euler(keyframe.value.x, keyframe.value.y, keyframe.value.z, order) ); values.push(quat.x, quat.y, quat.z, quat.w); diff --git a/src/stores/ModelViewerStore.ts b/src/stores/ModelViewerStore.ts index ce9b1cd6..2429e9c4 100644 --- a/src/stores/ModelViewerStore.ts +++ b/src/stores/ModelViewerStore.ts @@ -1,6 +1,6 @@ import Logger from 'js-logger'; import { action, observable } from "mobx"; -import { AnimationAction, AnimationClip, AnimationMixer, Object3D } from "three"; +import { AnimationAction, AnimationClip, AnimationMixer, SkinnedMesh } from "three"; import { BufferCursor } from "../data_formats/BufferCursor"; import { NinjaModel, NinjaObject, parse_nj, parse_xj } from "../data_formats/parsing/ninja"; import { parse_njm_4 } from "../data_formats/parsing/ninja/motion"; @@ -30,7 +30,7 @@ class ModelViewerStore { ]; @observable.ref current_model?: NinjaObject; - @observable.ref current_model_obj3d?: Object3D; + @observable.ref current_model_obj3d?: SkinnedMesh; @observable.ref animation?: { mixer: AnimationMixer, @@ -127,11 +127,10 @@ class ModelViewerStore { const model = parse_xj(new BufferCursor(reader.result, true))[0]; this.set_model(model, file.name); } else if (file.name.endsWith('.njm')) { - this.add_animation( - create_animation_clip( - parse_njm_4(new BufferCursor(reader.result, true)) - ) - ); + if (this.current_model) { + const njm = parse_njm_4(new BufferCursor(reader.result, true)); + this.add_animation(create_animation_clip(this.current_model, njm)); + } } else { logger.error(`Unknown file extension in filename "${file.name}".`); } @@ -140,22 +139,14 @@ class ModelViewerStore { private add_to_bone( object: NinjaObject, head_part: NinjaObject, - bone_id: number, - id_ref: [number] = [0] + bone_id: number ) { - if (!object.evaluation_flags.skip) { - const id = id_ref[0]++; + const bone = object.find_bone(bone_id); - if (id === bone_id) { - object.evaluation_flags.hidden = false; - object.evaluation_flags.break_child_trace = false; - object.children.push(head_part); - return; - } - } - - for (const child of object.children) { - this.add_to_bone(child, head_part, bone_id, id_ref); + if (bone) { + bone.evaluation_flags.hidden = false; + bone.evaluation_flags.break_child_trace = false; + bone.children.push(head_part); } } diff --git a/src/ui/model_viewer/RendererComponent.tsx b/src/ui/model_viewer/RendererComponent.tsx index 805d8326..5112d852 100644 --- a/src/ui/model_viewer/RendererComponent.tsx +++ b/src/ui/model_viewer/RendererComponent.tsx @@ -1,9 +1,9 @@ import React from 'react'; -import { Object3D } from 'three'; +import { SkinnedMesh } from 'three'; import { get_model_renderer } from '../../rendering/ModelRenderer'; type Props = { - model?: Object3D + model?: SkinnedMesh } export class RendererComponent extends React.Component {