mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
The selected class and animation are now highlighted in the model viewer.
This commit is contained in:
parent
27fdefe74a
commit
7335042c53
@ -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(
|
||||||
if (this.current_obj3d && this.animation) {
|
"set_model",
|
||||||
this.animation.mixer.stopAllAction();
|
(model: NinjaObject<NinjaModel>, player_model?: PlayerModel) => {
|
||||||
this.animation.mixer.uncacheRoot(this.current_obj3d);
|
if (this.current_obj3d && this.animation) {
|
||||||
this.animation = undefined;
|
this.animation.mixer.stopAllAction();
|
||||||
|
this.animation.mixer.uncacheRoot(this.current_obj3d);
|
||||||
|
this.animation = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.current_player_model = player_model;
|
||||||
|
this.current_model = model;
|
||||||
|
this.current_bone_count = model.bone_count();
|
||||||
|
|
||||||
|
const mesh = ninja_object_to_skinned_mesh(this.current_model);
|
||||||
|
mesh.translateY(-mesh.geometry.boundingSphere.radius);
|
||||||
|
this.current_obj3d = mesh;
|
||||||
}
|
}
|
||||||
|
);
|
||||||
this.current_model = model;
|
|
||||||
this.current_bone_count = model.bone_count();
|
|
||||||
|
|
||||||
const mesh = ninja_object_to_skinned_mesh(this.current_model);
|
|
||||||
mesh.translateY(-mesh.geometry.boundingSphere.radius);
|
|
||||||
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) => {
|
||||||
|
@ -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%);
|
||||||
}
|
}
|
||||||
|
@ -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 => {
|
||||||
<li
|
const selected =
|
||||||
key={animation.id}
|
model_viewer_store.animation &&
|
||||||
onClick={() => model_viewer_store.load_animation(animation)}
|
model_viewer_store.animation.player_animation &&
|
||||||
>
|
model_viewer_store.animation.player_animation.id === animation.id;
|
||||||
{animation.name}
|
|
||||||
</li>
|
return (
|
||||||
))}
|
<li
|
||||||
|
key={animation.id}
|
||||||
|
className={selected ? "selected" : undefined}
|
||||||
|
onClick={() => model_viewer_store.load_animation(animation)}
|
||||||
|
>
|
||||||
|
{animation.name}
|
||||||
|
</li>
|
||||||
|
);
|
||||||
|
})}
|
||||||
</ul>
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
|
@ -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%);
|
||||||
}
|
}
|
||||||
|
@ -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 => {
|
||||||
<List.Item onClick={() => model_viewer_store.load_model(model)}>
|
const selected = model_viewer_store.current_player_model === model;
|
||||||
<List.Item.Meta
|
|
||||||
title={
|
return (
|
||||||
<span className="mv-ModelSelectionComponent-model">
|
<List.Item onClick={() => model_viewer_store.load_model(model)}>
|
||||||
{model.name}
|
<List.Item.Meta
|
||||||
</span>
|
title={
|
||||||
}
|
<span
|
||||||
/>
|
className={
|
||||||
</List.Item>
|
"mv-ModelSelectionComponent-model" +
|
||||||
)}
|
(selected ? " selected" : "")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{model.name}
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</List.Item>
|
||||||
|
);
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
|
Loading…
Reference in New Issue
Block a user