start supporting mof properly
This commit is contained in:
parent
3525f48fda
commit
74cce227a5
24
extract.nu
24
extract.nu
|
@ -1,14 +1,28 @@
|
||||||
def main [thdat: path, version: int] {
|
def main [game_folder: path, version: int] {
|
||||||
let game = if $version == 10 {
|
let game = if $version == 10 {
|
||||||
"mof"
|
{ folder: "mof", version: $version }
|
||||||
} else if $version == 11 {
|
} else if $version == 11 {
|
||||||
"sa"
|
{ folder: "sa", version: $version }
|
||||||
} else {
|
} else {
|
||||||
return (error make {msg: "version not supported", label: { text: "expected 10 or 11", span: (metadata $version).span } })
|
return (error make {msg: "version not supported", label: { text: "expected 10 or 11", span: (metadata $version).span } })
|
||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
thdat -C $"assets/($game)" -x $version $thdat
|
let assets = $"assets/($game.folder)"
|
||||||
|
let thver = $"th($game.version)"
|
||||||
|
thdat -C $assets -x $version $"($game_folder)/($thver).dat"
|
||||||
|
try {
|
||||||
|
ln -s $"($game_folder)/thbgm.dat" $"($assets)/thbgm.dat"
|
||||||
|
} catch { }
|
||||||
|
|
||||||
|
let research = $"research/dump/($game.folder)"
|
||||||
|
let replt = {|ext| str replace $ext $"t($ext)"};
|
||||||
|
let dump = {|ext|
|
||||||
|
ls $assets | get name | where {str ends-with $".($ext)"} | each {path basename}
|
||||||
|
| each {|name| run-external $"tru($ext)" "decompile" "-g" $game.version $"($assets)/($name)" "-m" $"utils/($thver).($ext)m" | save -f $"($research)/($name | do $replt $ext)"}
|
||||||
|
};
|
||||||
|
|
||||||
|
do $dump "anm"
|
||||||
|
do $dump "ecl"
|
||||||
|
do $dump "std"
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,4 +9,4 @@ proc-macro = true
|
||||||
proc-macro2 = "1.0.86"
|
proc-macro2 = "1.0.86"
|
||||||
quote = "1.0.36"
|
quote = "1.0.36"
|
||||||
syn = "2.0.72"
|
syn = "2.0.72"
|
||||||
truth = { git = "https://github.com/ExpHP/truth", version = "0.5.2" }
|
truth = { git = "https://github.com/ExpHP/truth" }
|
||||||
|
|
|
@ -103,7 +103,7 @@ impl AnmViewer {
|
||||||
&mut BinReader::from_reader(
|
&mut BinReader::from_reader(
|
||||||
&RootEmitter::new_stderr(),
|
&RootEmitter::new_stderr(),
|
||||||
"",
|
"",
|
||||||
Cursor::new(read("assets/title.anm").unwrap()),
|
// Cursor::new(read("assets/title.anm").unwrap()),
|
||||||
),
|
),
|
||||||
Game::Th11,
|
Game::Th11,
|
||||||
true,
|
true,
|
||||||
|
|
|
@ -4,14 +4,19 @@ use async_std::fs;
|
||||||
use bytemuck::{Pod, Zeroable};
|
use bytemuck::{Pod, Zeroable};
|
||||||
use glam::Vec2;
|
use glam::Vec2;
|
||||||
use nonoverlapping_interval_tree::NonOverlappingIntervalTree;
|
use nonoverlapping_interval_tree::NonOverlappingIntervalTree;
|
||||||
use truth::{context::RootEmitter, io::BinReader, AnmFile, Game};
|
use truth::{context::RootEmitter, io::BinReader, AnmFile, Game as TruthGame};
|
||||||
use wgpu::{
|
use wgpu::{
|
||||||
naga::FastHashMap, util::DeviceExt, BindGroup, BindGroupDescriptor, BindGroupEntry, BindGroupLayoutDescriptor, BindGroupLayoutEntry, Device, Extent3d, Queue, ShaderStages, Texture, TextureDescriptor, TextureDimension, TextureFormat, TextureUsages
|
naga::FastHashMap, util::DeviceExt, BindGroup, BindGroupDescriptor, BindGroupEntry, BindGroupLayoutDescriptor,
|
||||||
|
BindGroupLayoutEntry, Device, Extent3d, Queue, ShaderStages, Texture, TextureDescriptor, TextureDimension,
|
||||||
|
TextureFormat, TextureUsages,
|
||||||
};
|
};
|
||||||
use winit::dpi::PhysicalSize;
|
use winit::dpi::PhysicalSize;
|
||||||
|
|
||||||
|
use crate::utils::game::Game;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
image::{produce_image_from_entry, Image}, vm::opcodes::{Instruction, Op}
|
image::{produce_image_from_entry, Image},
|
||||||
|
vm::opcodes::{Instruction, Op},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct LoadedEntry {
|
pub struct LoadedEntry {
|
||||||
|
@ -199,12 +204,15 @@ pub struct LoadedFile {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LoadedFile {
|
impl LoadedFile {
|
||||||
pub async fn load(device: &Device, queue: &Queue, size: PhysicalSize<u32>, file_name: &str) -> Self {
|
pub async fn load(device: &Device, queue: &Queue, size: PhysicalSize<u32>, game: Game, file_name: &str) -> Self {
|
||||||
let file_data = fs::read(Path::new("./assets").join(file_name)).await.expect("failed to load anm file");
|
let file_data = fs::read(game.asset_path(file_name)).await.expect("failed to load anm file");
|
||||||
|
|
||||||
let file = AnmFile::read_from_stream(
|
let file = AnmFile::read_from_stream(
|
||||||
&mut BinReader::from_reader(&RootEmitter::new_stderr(), file_name, Cursor::new(file_data)),
|
&mut BinReader::from_reader(&RootEmitter::new_stderr(), file_name, Cursor::new(file_data)),
|
||||||
Game::Th11,
|
match game {
|
||||||
|
Game::Mof => TruthGame::Th10,
|
||||||
|
Game::Sa => TruthGame::Th11,
|
||||||
|
},
|
||||||
true,
|
true,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
|
@ -3,7 +3,10 @@ use std::sync::Arc;
|
||||||
use wgpu::{Device, Queue};
|
use wgpu::{Device, Queue};
|
||||||
use winit::{dpi::PhysicalSize, window::Window};
|
use winit::{dpi::PhysicalSize, window::Window};
|
||||||
|
|
||||||
use crate::{game::anm::LoadedFile, utils::soon::Soon};
|
use crate::{
|
||||||
|
game::anm::LoadedFile,
|
||||||
|
utils::{game::Game, soon::Soon},
|
||||||
|
};
|
||||||
|
|
||||||
use super::Manager;
|
use super::Manager;
|
||||||
|
|
||||||
|
@ -13,6 +16,7 @@ impl Manager {
|
||||||
device: Arc<Device>,
|
device: Arc<Device>,
|
||||||
queue: Arc<Queue>,
|
queue: Arc<Queue>,
|
||||||
size: PhysicalSize<u32>,
|
size: PhysicalSize<u32>,
|
||||||
|
game: Game,
|
||||||
file_name: impl Into<String>,
|
file_name: impl Into<String>,
|
||||||
) -> Soon<Arc<LoadedFile>> {
|
) -> Soon<Arc<LoadedFile>> {
|
||||||
let file_name: String = file_name.into();
|
let file_name: String = file_name.into();
|
||||||
|
@ -22,7 +26,7 @@ impl Manager {
|
||||||
|
|
||||||
let sender = self.anm_sender.clone();
|
let sender = self.anm_sender.clone();
|
||||||
Soon::new(async move {
|
Soon::new(async move {
|
||||||
let file = Arc::new(LoadedFile::load(&device, &queue, size, &file_name).await);
|
let file = Arc::new(LoadedFile::load(&device, &queue, size, game, &file_name).await);
|
||||||
|
|
||||||
sender.send((file_name, file.clone())).unwrap();
|
sender.send((file_name, file.clone())).unwrap();
|
||||||
|
|
||||||
|
@ -41,6 +45,7 @@ impl Manager {
|
||||||
device: &Device,
|
device: &Device,
|
||||||
queue: &Queue,
|
queue: &Queue,
|
||||||
window: &Window,
|
window: &Window,
|
||||||
|
game: Game,
|
||||||
file_name: impl Into<String>,
|
file_name: impl Into<String>,
|
||||||
) -> Arc<LoadedFile> {
|
) -> Arc<LoadedFile> {
|
||||||
let file_name = file_name.into();
|
let file_name = file_name.into();
|
||||||
|
@ -52,6 +57,7 @@ impl Manager {
|
||||||
device,
|
device,
|
||||||
queue,
|
queue,
|
||||||
window.inner_size(),
|
window.inner_size(),
|
||||||
|
game,
|
||||||
&file_name,
|
&file_name,
|
||||||
)));
|
)));
|
||||||
|
|
||||||
|
|
|
@ -375,9 +375,12 @@ impl Manager {
|
||||||
resolve_target: None,
|
resolve_target: None,
|
||||||
ops: Operations {
|
ops: Operations {
|
||||||
// load: LoadOp::Clear(Color {
|
// load: LoadOp::Clear(Color {
|
||||||
// r: 100. / 255.,
|
// // r: 100. / 255.,
|
||||||
// g: 149. / 255.,
|
// // g: 149. / 255.,
|
||||||
// b: 237. / 255.,
|
// // b: 237. / 255.,
|
||||||
|
// r: 1.0,
|
||||||
|
// g: 1.0,
|
||||||
|
// b: 1.0,
|
||||||
// a: 1.0,
|
// a: 1.0,
|
||||||
// }),
|
// }),
|
||||||
load: LoadOp::Load,
|
load: LoadOp::Load,
|
||||||
|
|
|
@ -204,7 +204,7 @@ impl AnmVm {
|
||||||
dst_factor: BlendFactor::OneMinusSrcAlpha,
|
dst_factor: BlendFactor::OneMinusSrcAlpha,
|
||||||
operation: BlendOperation::Add,
|
operation: BlendOperation::Add,
|
||||||
},
|
},
|
||||||
alpha: BlendComponent::REPLACE,
|
alpha: BlendComponent::OVER,
|
||||||
},
|
},
|
||||||
1 => BlendState {
|
1 => BlendState {
|
||||||
// additive blending
|
// additive blending
|
||||||
|
@ -213,7 +213,7 @@ impl AnmVm {
|
||||||
dst_factor: BlendFactor::OneMinusSrcAlpha,
|
dst_factor: BlendFactor::OneMinusSrcAlpha,
|
||||||
operation: BlendOperation::Add,
|
operation: BlendOperation::Add,
|
||||||
},
|
},
|
||||||
alpha: BlendComponent::REPLACE,
|
alpha: BlendComponent::OVER,
|
||||||
},
|
},
|
||||||
3 => BlendState {
|
3 => BlendState {
|
||||||
color: BlendComponent::REPLACE,
|
color: BlendComponent::REPLACE,
|
||||||
|
|
|
@ -2,11 +2,11 @@ use std::{io::Cursor, path::Path, sync::Arc};
|
||||||
|
|
||||||
use async_std::fs;
|
use async_std::fs;
|
||||||
use futures::future::{join3, join_all};
|
use futures::future::{join3, join_all};
|
||||||
use truth::{context::RootEmitter, io::BinReader, Game, StackEclFile};
|
use truth::{context::RootEmitter, io::BinReader, Game as TruthGame, StackEclFile};
|
||||||
use wgpu::{naga::FastHashMap, Device, Queue};
|
use wgpu::{naga::FastHashMap, Device, Queue};
|
||||||
use winit::dpi::PhysicalSize;
|
use winit::dpi::PhysicalSize;
|
||||||
|
|
||||||
use crate::game::anm;
|
use crate::{game::anm, utils::game::Game};
|
||||||
|
|
||||||
use super::vm::opcodes::Instruction;
|
use super::vm::opcodes::Instruction;
|
||||||
|
|
||||||
|
@ -21,23 +21,27 @@ impl LoadedFile {
|
||||||
device: Arc<Device>,
|
device: Arc<Device>,
|
||||||
queue: Arc<Queue>,
|
queue: Arc<Queue>,
|
||||||
size: PhysicalSize<u32>,
|
size: PhysicalSize<u32>,
|
||||||
|
game: Game,
|
||||||
file_name: impl Into<String> + Send,
|
file_name: impl Into<String> + Send,
|
||||||
) -> LoadedFile {
|
) -> LoadedFile {
|
||||||
let file_name = file_name.into();
|
let file_name = file_name.into();
|
||||||
let file_data = fs::read(Path::new("./assets").join(&file_name)).await.expect("failed to load anm file");
|
let file_data = fs::read(game.asset_path(&file_name)).await.expect("failed to load anm file");
|
||||||
let file = StackEclFile::read_from_stream(
|
let file = StackEclFile::read_from_stream(
|
||||||
&mut BinReader::from_reader(&RootEmitter::new_stderr(), &file_name, Cursor::new(file_data)),
|
&mut BinReader::from_reader(&RootEmitter::new_stderr(), &file_name, Cursor::new(file_data)),
|
||||||
Game::Th11,
|
match game {
|
||||||
|
Game::Mof => TruthGame::Th10,
|
||||||
|
Game::Sa => TruthGame::Th11,
|
||||||
|
},
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let anm_files = join_all(file.anim_list.into_iter().map(|sp| sp.value).map(|anm| {
|
let anm_files = join_all(file.anim_list.into_iter().map(|sp| sp.value).map(|anm| {
|
||||||
let (device, queue) = (device.clone(), queue.clone());
|
let (device, queue) = (device.clone(), queue.clone());
|
||||||
async move { anm::LoadedFile::load(&device, &queue, size, &anm).await }
|
async move { anm::LoadedFile::load(&device, &queue, size, game, &anm).await }
|
||||||
}));
|
}));
|
||||||
let ecl_files = join_all(file.ecli_list.into_iter().map(|sp| sp.value).map(|ecl| {
|
let ecl_files = join_all(file.ecli_list.into_iter().map(|sp| sp.value).map(|ecl| {
|
||||||
let (device, queue) = (device.clone(), queue.clone());
|
let (device, queue) = (device.clone(), queue.clone());
|
||||||
async move { LoadedFile::load(device, queue, size, ecl).await }
|
async move { LoadedFile::load(device, queue, size, game, ecl).await }
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let subs = file.subs.into_iter().map(|(k, v)| (k.value, v)).collect::<Vec<_>>();
|
let subs = file.subs.into_iter().map(|(k, v)| (k.value, v)).collect::<Vec<_>>();
|
||||||
|
|
|
@ -4,7 +4,8 @@ use anm::LoadedFile;
|
||||||
use states::GameStateMachine;
|
use states::GameStateMachine;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
engine::{EngineState, UpdateContext}, utils::soon::Soon
|
engine::{EngineState, UpdateContext},
|
||||||
|
utils::{game::Game, soon::Soon},
|
||||||
};
|
};
|
||||||
|
|
||||||
mod anm;
|
mod anm;
|
||||||
|
@ -27,17 +28,24 @@ struct GameContext<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GameContext<'_> {
|
impl GameContext<'_> {
|
||||||
pub fn start_load_anm(&mut self, file_name: impl Into<String>) -> Soon<Arc<LoadedFile>> {
|
pub fn start_load_anm(&mut self, game: Game, file_name: impl Into<String>) -> Soon<Arc<LoadedFile>> {
|
||||||
self.anm_manager.start_load_anm(
|
self.anm_manager.start_load_anm(
|
||||||
self.engine.device.clone(),
|
self.engine.device.clone(),
|
||||||
self.engine.queue.clone(),
|
self.engine.queue.clone(),
|
||||||
self.engine.window.inner_size(),
|
self.engine.window.inner_size(),
|
||||||
|
game,
|
||||||
file_name,
|
file_name,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn load_anm(&mut self, file_name: impl Into<String>) -> Arc<LoadedFile> {
|
pub fn load_anm(&mut self, game: Game, file_name: impl Into<String>) -> Arc<LoadedFile> {
|
||||||
self.anm_manager.load_anm(&self.engine.device, &self.engine.queue, &self.engine.window, file_name)
|
self.anm_manager.load_anm(
|
||||||
|
&self.engine.device,
|
||||||
|
&self.engine.queue,
|
||||||
|
&self.engine.window,
|
||||||
|
game,
|
||||||
|
file_name,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,19 +1,23 @@
|
||||||
|
use std::{rc::Rc, sync::Arc};
|
||||||
|
|
||||||
use async_std::fs::read;
|
use async_std::fs::read;
|
||||||
use format::BgmFormat;
|
use format::BgmFormat;
|
||||||
use rodio::{static_buffer::StaticSamplesBuffer, Sink, Source};
|
use rodio::{static_buffer::StaticSamplesBuffer, Sink, Source};
|
||||||
|
|
||||||
|
use crate::utils::game::Game;
|
||||||
|
|
||||||
pub mod format;
|
pub mod format;
|
||||||
|
|
||||||
pub struct BgmManager {
|
pub struct BgmManager {
|
||||||
bgm_format: BgmFormat,
|
bgm_format: BgmFormat,
|
||||||
bgm_file: &'static [u8],
|
bgm_file: &'static [u8],
|
||||||
music_sink: Sink,
|
music_sink: Arc<Sink>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BgmManager {
|
impl BgmManager {
|
||||||
pub(super) async fn new(music_sink: Sink) -> BgmManager {
|
pub(super) async fn new(music_sink: Arc<Sink>, game: Game) -> BgmManager {
|
||||||
let bgm_format = BgmFormat::new(read("assets/thbgm.fmt").await.unwrap());
|
let bgm_format = BgmFormat::new(read(game.asset_path("thbgm.fmt")).await.unwrap());
|
||||||
let bgm_file = read("./thbgm.dat").await.unwrap();
|
let bgm_file = read(game.asset_path("thbgm.dat")).await.unwrap();
|
||||||
music_sink.set_volume(0.1);
|
music_sink.set_volume(0.1);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
|
|
|
@ -2,15 +2,17 @@ use std::sync::Arc;
|
||||||
|
|
||||||
use bgm::BgmManager;
|
use bgm::BgmManager;
|
||||||
use rodio::{
|
use rodio::{
|
||||||
dynamic_mixer::{mixer, DynamicMixerController}, OutputStream, Sink
|
dynamic_mixer::{mixer, DynamicMixerController},
|
||||||
|
OutputStream, Sink,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::utils::soon::Soon;
|
use crate::utils::{game::Game, soon::Soon};
|
||||||
|
|
||||||
pub mod bgm;
|
pub mod bgm;
|
||||||
|
|
||||||
pub struct Manager {
|
pub struct Manager {
|
||||||
bgm_manager: Soon<BgmManager>,
|
bgm_manager_mof: Soon<BgmManager>,
|
||||||
|
bgm_manager_sa: Soon<BgmManager>,
|
||||||
_output_stream: OutputStream,
|
_output_stream: OutputStream,
|
||||||
_sound_sink: Sink,
|
_sound_sink: Sink,
|
||||||
_sound_controller: Arc<DynamicMixerController<i16>>,
|
_sound_controller: Arc<DynamicMixerController<i16>>,
|
||||||
|
@ -20,12 +22,13 @@ impl Manager {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
let (output_stream, stream_handle) = OutputStream::try_default().unwrap();
|
let (output_stream, stream_handle) = OutputStream::try_default().unwrap();
|
||||||
let sound_sink = Sink::try_new(&stream_handle).unwrap();
|
let sound_sink = Sink::try_new(&stream_handle).unwrap();
|
||||||
let music_sink = Sink::try_new(&stream_handle).unwrap();
|
let music_sink = Arc::new(Sink::try_new(&stream_handle).unwrap());
|
||||||
let (sound_controller, mixer) = mixer::<i16>(2, 44100);
|
let (sound_controller, mixer) = mixer::<i16>(2, 44100);
|
||||||
sound_sink.append(mixer);
|
sound_sink.append(mixer);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
bgm_manager: Soon::new(BgmManager::new(music_sink)),
|
bgm_manager_mof: Soon::new(BgmManager::new(music_sink.clone(), Game::Mof)),
|
||||||
|
bgm_manager_sa: Soon::new(BgmManager::new(music_sink, Game::Sa)),
|
||||||
_output_stream: output_stream,
|
_output_stream: output_stream,
|
||||||
_sound_sink: sound_sink,
|
_sound_sink: sound_sink,
|
||||||
_sound_controller: sound_controller,
|
_sound_controller: sound_controller,
|
||||||
|
@ -33,10 +36,15 @@ impl Manager {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_bgm_manager_loaded(&mut self) -> bool {
|
pub fn is_bgm_manager_loaded(&mut self) -> bool {
|
||||||
self.bgm_manager.is_done()
|
self.bgm_manager_mof.is_done() && self.bgm_manager_sa.is_done()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_bgm_manager(&mut self) -> &mut BgmManager {
|
pub fn get_bgm_manager(&mut self, game: Game) -> &mut BgmManager {
|
||||||
self.bgm_manager.try_borrow_mut().unwrap()
|
match game {
|
||||||
|
Game::Mof => &mut self.bgm_manager_mof,
|
||||||
|
Game::Sa => &mut self.bgm_manager_sa,
|
||||||
|
}
|
||||||
|
.try_borrow_mut()
|
||||||
|
.unwrap()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,22 @@
|
||||||
use async_std::task::{block_on, spawn};
|
use async_std::task::{block_on, spawn};
|
||||||
use sfsm::State;
|
use sfsm::State;
|
||||||
|
|
||||||
use crate::game::{enemy::loaded_file::LoadedFile, GameContext};
|
use crate::{
|
||||||
|
game::{enemy::loaded_file::LoadedFile, GameContext},
|
||||||
|
utils::game::Game,
|
||||||
|
};
|
||||||
|
|
||||||
pub struct Gameplay {
|
pub struct Gameplay {}
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Gameplay {
|
impl Gameplay {
|
||||||
pub fn new(context: &GameContext) -> Self {
|
pub fn new(context: &GameContext) -> Self {
|
||||||
block_on(LoadedFile::load(context.engine.device.clone(), context.engine.queue.clone(), context.engine.window.inner_size(), "default.ecl"));
|
block_on(LoadedFile::load(
|
||||||
|
context.engine.device.clone(),
|
||||||
|
context.engine.queue.clone(),
|
||||||
|
context.engine.window.inner_size(),
|
||||||
|
Game::Sa,
|
||||||
|
"default.ecl",
|
||||||
|
));
|
||||||
Self {}
|
Self {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,8 +5,10 @@ use sfsm::{State, TransitGuard, Transition};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
game::{
|
game::{
|
||||||
anm::{LoadedFile, Vm, VmLocation}, GameContext
|
anm::{LoadedFile, Vm, VmLocation},
|
||||||
}, utils::soon::Soon
|
GameContext,
|
||||||
|
},
|
||||||
|
utils::{game::Game, soon::Soon},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{title::TitleScreen, UPDATE_CONTEXT};
|
use super::{title::TitleScreen, UPDATE_CONTEXT};
|
||||||
|
@ -20,8 +22,8 @@ pub struct Loading {
|
||||||
|
|
||||||
impl Loading {
|
impl Loading {
|
||||||
pub fn new(context: &mut GameContext) -> Loading {
|
pub fn new(context: &mut GameContext) -> Loading {
|
||||||
let ascii = context.load_anm("ascii.anm");
|
let ascii = context.load_anm(Game::Sa, "ascii.anm");
|
||||||
let sig_anm = context.load_anm("sig.anm");
|
let sig_anm = context.load_anm(Game::Sa, "sig.anm");
|
||||||
let sig = context.anm_manager.new_vm(sig_anm, None, 0, VmLocation::new_ui());
|
let sig = context.anm_manager.new_vm(sig_anm, None, 0, VmLocation::new_ui());
|
||||||
let ascii_loading = context.anm_manager.new_vm(ascii.clone(), None, 16, VmLocation::new_ui());
|
let ascii_loading = context.anm_manager.new_vm(ascii.clone(), None, 16, VmLocation::new_ui());
|
||||||
ascii_loading.borrow_mut().origin = Vec3::new(480.0, 392.0, 0.0);
|
ascii_loading.borrow_mut().origin = Vec3::new(480.0, 392.0, 0.0);
|
||||||
|
@ -30,7 +32,7 @@ impl Loading {
|
||||||
_sig: sig,
|
_sig: sig,
|
||||||
_ascii_loading: ascii_loading,
|
_ascii_loading: ascii_loading,
|
||||||
|
|
||||||
title_anm: context.start_load_anm("title.anm").into(),
|
title_anm: context.start_load_anm(Game::Sa, "title.anm").into(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,7 +13,7 @@ use super::GameContext;
|
||||||
|
|
||||||
pub(super) static UPDATE_CONTEXT: ContextMut<GameContext, GameStateMachine> = ContextMut::new();
|
pub(super) static UPDATE_CONTEXT: ContextMut<GameContext, GameStateMachine> = ContextMut::new();
|
||||||
|
|
||||||
add_state_machine!(Machine, Gameplay, {Loading, TitleScreen, Gameplay}, {
|
add_state_machine!(Machine, Loading, {Loading, TitleScreen, Gameplay}, {
|
||||||
Loading => TitleScreen,
|
Loading => TitleScreen,
|
||||||
TitleScreen => TitleScreen
|
TitleScreen => TitleScreen
|
||||||
});
|
});
|
||||||
|
@ -25,8 +25,8 @@ impl GameStateMachine {
|
||||||
UPDATE_CONTEXT.scoped(context, || {
|
UPDATE_CONTEXT.scoped(context, || {
|
||||||
let mut machine = GameStateMachine(Machine::new());
|
let mut machine = GameStateMachine(Machine::new());
|
||||||
|
|
||||||
// UPDATE_CONTEXT.with(|context| machine.0.start(Loading::new(context))).unwrap();
|
UPDATE_CONTEXT.with(|context| machine.0.start(Loading::new(context))).unwrap();
|
||||||
UPDATE_CONTEXT.with(|context| machine.0.start(Gameplay::new(context))).unwrap();
|
// UPDATE_CONTEXT.with(|context| machine.0.start(Gameplay::new(context))).unwrap();
|
||||||
|
|
||||||
machine
|
machine
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,8 +1,12 @@
|
||||||
use sfsm::{State, TransitGuard, Transition};
|
use sfsm::{State, TransitGuard, Transition};
|
||||||
use winit::keyboard::KeyCode;
|
use winit::keyboard::KeyCode;
|
||||||
|
|
||||||
use crate::game::{
|
use crate::{
|
||||||
anm::{Vm, VmLocation}, GameContext
|
game::{
|
||||||
|
anm::{Vm, VmLocation},
|
||||||
|
GameContext,
|
||||||
|
},
|
||||||
|
utils::game::Game,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{loading::Loading, UPDATE_CONTEXT};
|
use super::{loading::Loading, UPDATE_CONTEXT};
|
||||||
|
@ -13,12 +17,14 @@ pub struct TitleScreen {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TitleScreen {
|
impl TitleScreen {
|
||||||
fn new(context: &mut GameContext, from_startup: bool) -> Self {
|
fn new(context: &mut GameContext, _: bool) -> Self {
|
||||||
let title = context.load_anm("title.anm");
|
let title = context.load_anm(Game::Mof, "title.anm");
|
||||||
|
|
||||||
// 79, 83
|
// 79, 83
|
||||||
let splash = context.anm_manager.new_vm(title.clone(), Some(8), 79, VmLocation::new_ui());
|
// let splash = context.anm_manager.new_vm(title.clone(), Some(8), 79, VmLocation::new_ui());
|
||||||
let logo = context.anm_manager.new_vm(title.clone(), Some(8), 83, VmLocation::new_ui());
|
// let logo = context.anm_manager.new_vm(title.clone(), Some(8), 83, VmLocation::new_ui());
|
||||||
|
let splash = context.anm_manager.new_vm(title.clone(), Some(8), 88, VmLocation::new_ui());
|
||||||
|
let logo = context.anm_manager.new_vm(title.clone(), Some(8), 90, VmLocation::new_ui());
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
_logo: logo,
|
_logo: logo,
|
||||||
|
@ -40,7 +46,8 @@ impl State for TitleScreen {
|
||||||
impl From<Loading> for TitleScreen {
|
impl From<Loading> for TitleScreen {
|
||||||
fn from(_: Loading) -> Self {
|
fn from(_: Loading) -> Self {
|
||||||
UPDATE_CONTEXT.with(|context| {
|
UPDATE_CONTEXT.with(|context| {
|
||||||
context.sound_manager.get_bgm_manager().play_track("th11_00.wav");
|
context.sound_manager.get_bgm_manager(Game::Mof).play_track("th10_02.wav");
|
||||||
|
// context.sound_manager.get_bgm_manager_sa().play_track("th11_00.wav");
|
||||||
TitleScreen::new(context, true)
|
TitleScreen::new(context, true)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
20
src/utils/game.rs
Normal file
20
src/utils/game.rs
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
||||||
|
pub enum Game {
|
||||||
|
Mof,
|
||||||
|
Sa,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Game {
|
||||||
|
fn asset_path_str(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Game::Mof => "./assets/mof",
|
||||||
|
Game::Sa => "./assets/sa",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn asset_path<P: AsRef<Path>>(&self, path: P) -> PathBuf {
|
||||||
|
Path::new(self.asset_path_str()).join(path)
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,2 +1,3 @@
|
||||||
pub mod context;
|
pub mod context;
|
||||||
pub mod soon;
|
pub mod soon;
|
||||||
|
pub mod game;
|
||||||
|
|
Loading…
Reference in a new issue