The selected class and animation are now highlighted in the model viewer.

This commit is contained in:
Daan Vanden Bosch 2019-07-03 14:27:44 +02:00
parent 27fdefe74a
commit 7335042c53
5 changed files with 69 additions and 35 deletions

View File

@ -32,11 +32,13 @@ class ModelViewerStore {
.fill(undefined) .fill(undefined)
.map((_, i) => new PlayerAnimation(i, `Animation ${i + 1}`)); .map((_, i) => new PlayerAnimation(i, `Animation ${i + 1}`));
@observable.ref current_player_model?: PlayerModel;
@observable.ref current_model?: NinjaObject<NinjaModel>; @observable.ref current_model?: NinjaObject<NinjaModel>;
@observable.ref current_bone_count: number = 0; @observable.ref current_bone_count: number = 0;
@observable.ref current_obj3d?: SkinnedMesh; @observable.ref current_obj3d?: SkinnedMesh;
@observable.ref animation?: { @observable.ref animation?: {
player_animation?: PlayerAnimation;
mixer: AnimationMixer; mixer: AnimationMixer;
clip: AnimationClip; clip: AnimationClip;
action: AnimationAction; action: AnimationAction;
@ -67,7 +69,7 @@ class ModelViewerStore {
load_model = async (model: PlayerModel) => { load_model = async (model: PlayerModel) => {
const object = await this.get_player_ninja_object(model); const object = await this.get_player_ninja_object(model);
this.set_model(object); this.set_model(object, model);
// Ignore the bones from the head parts. // Ignore the bones from the head parts.
this.current_bone_count = 64; this.current_bone_count = 64;
}; };
@ -76,7 +78,7 @@ class ModelViewerStore {
const nj_motion = await this.get_nj_motion(animation); const nj_motion = await this.get_nj_motion(animation);
if (this.current_model) { if (this.current_model) {
this.set_animation(create_animation_clip(this.current_model, nj_motion)); this.set_animation(create_animation_clip(this.current_model, nj_motion), animation);
} }
}; };
@ -102,7 +104,7 @@ class ModelViewerStore {
} }
}); });
set_animation = action("set_animation", (clip: AnimationClip) => { set_animation = action("set_animation", (clip: AnimationClip, animation?: PlayerAnimation) => {
if (!this.current_obj3d) return; if (!this.current_obj3d) return;
let mixer: AnimationMixer; let mixer: AnimationMixer;
@ -115,6 +117,7 @@ class ModelViewerStore {
} }
this.animation = { this.animation = {
player_animation: animation,
mixer, mixer,
clip, clip,
action: mixer.clipAction(clip), action: mixer.clipAction(clip),
@ -125,20 +128,24 @@ class ModelViewerStore {
this.animation_frame_count = Math.round(PSO_FRAME_RATE * clip.duration) + 1; this.animation_frame_count = Math.round(PSO_FRAME_RATE * clip.duration) + 1;
}); });
private set_model = action("set_model", (model: NinjaObject<NinjaModel>) => { private set_model = action(
"set_model",
(model: NinjaObject<NinjaModel>, player_model?: PlayerModel) => {
if (this.current_obj3d && this.animation) { if (this.current_obj3d && this.animation) {
this.animation.mixer.stopAllAction(); this.animation.mixer.stopAllAction();
this.animation.mixer.uncacheRoot(this.current_obj3d); this.animation.mixer.uncacheRoot(this.current_obj3d);
this.animation = undefined; this.animation = undefined;
} }
this.current_player_model = player_model;
this.current_model = model; this.current_model = model;
this.current_bone_count = model.bone_count(); this.current_bone_count = model.bone_count();
const mesh = ninja_object_to_skinned_mesh(this.current_model); const mesh = ninja_object_to_skinned_mesh(this.current_model);
mesh.translateY(-mesh.geometry.boundingSphere.radius); mesh.translateY(-mesh.geometry.boundingSphere.radius);
this.current_obj3d = mesh; this.current_obj3d = mesh;
}); }
);
// TODO: notify user of problems. // TODO: notify user of problems.
private loadend = async (file: File, reader: FileReader) => { private loadend = async (file: File, reader: FileReader) => {

View File

@ -13,6 +13,10 @@
padding: 2px 5px; padding: 2px 5px;
white-space: nowrap; white-space: nowrap;
&.selected {
color: lighten(@primary-color, 15%);
}
&:hover { &:hover {
color: lighten(@primary-color, 30%); color: lighten(@primary-color, 30%);
} }

View File

@ -1,20 +1,30 @@
import React, { Component, ReactNode } from "react"; import React, { Component, ReactNode } from "react";
import { model_viewer_store } from "../../stores/ModelViewerStore"; import { model_viewer_store } from "../../stores/ModelViewerStore";
import "./AnimationSelectionComponent.less"; import "./AnimationSelectionComponent.less";
import { observer } from "mobx-react";
@observer
export class AnimationSelectionComponent extends Component { export class AnimationSelectionComponent extends Component {
render(): ReactNode { render(): ReactNode {
return ( return (
<section className="mv-AnimationSelectionComponent"> <section className="mv-AnimationSelectionComponent">
<ul> <ul>
{model_viewer_store.animations.map(animation => ( {model_viewer_store.animations.map(animation => {
const selected =
model_viewer_store.animation &&
model_viewer_store.animation.player_animation &&
model_viewer_store.animation.player_animation.id === animation.id;
return (
<li <li
key={animation.id} key={animation.id}
className={selected ? "selected" : undefined}
onClick={() => model_viewer_store.load_animation(animation)} onClick={() => model_viewer_store.load_animation(animation)}
> >
{animation.name} {animation.name}
</li> </li>
))} );
})}
</ul> </ul>
</section> </section>
); );

View File

@ -5,6 +5,10 @@
.mv-ModelSelectionComponent-model { .mv-ModelSelectionComponent-model {
cursor: pointer; cursor: pointer;
&.selected {
color: lighten(@primary-color, 15%);
}
&:hover { &:hover {
color: lighten(@primary-color, 30%); color: lighten(@primary-color, 30%);
} }

View File

@ -11,17 +11,26 @@ export class ModelSelectionComponent extends Component {
itemLayout="horizontal" itemLayout="horizontal"
dataSource={model_viewer_store.models} dataSource={model_viewer_store.models}
size="small" size="small"
renderItem={model => ( renderItem={model => {
const selected = model_viewer_store.current_player_model === model;
return (
<List.Item onClick={() => model_viewer_store.load_model(model)}> <List.Item onClick={() => model_viewer_store.load_model(model)}>
<List.Item.Meta <List.Item.Meta
title={ title={
<span className="mv-ModelSelectionComponent-model"> <span
className={
"mv-ModelSelectionComponent-model" +
(selected ? " selected" : "")
}
>
{model.name} {model.name}
</span> </span>
} }
/> />
</List.Item> </List.Item>
)} );
}}
/> />
</section> </section>
); );