2023-09-09 14:58:20 +02:00
|
|
|
|
using System.Collections.Immutable;
|
|
|
|
|
using Swan.Mapping;
|
|
|
|
|
using TaikoWebUI.Shared.Models;
|
|
|
|
|
|
|
|
|
|
namespace TaikoWebUI.Services;
|
|
|
|
|
|
|
|
|
|
public class GameDataService : IGameDataService
|
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
private string[] bodyTitles;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
private readonly HttpClient client;
|
2023-10-15 16:27:07 +02:00
|
|
|
|
private string[] faceTitles;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
2023-10-15 16:27:07 +02:00
|
|
|
|
private string[] headTitles;
|
|
|
|
|
private string[] kigurumiTitles;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
|
|
|
|
private readonly Dictionary<uint, MusicDetail> musicMap = new();
|
2023-10-15 16:27:07 +02:00
|
|
|
|
private string[] puchiTitles;
|
|
|
|
|
|
|
|
|
|
private List<int> costumeFlagArraySizes = new();
|
|
|
|
|
|
|
|
|
|
private int titleFlagArraySize;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
|
|
|
|
private ImmutableDictionary<uint, DanData> danMap = ImmutableDictionary<uint, DanData>.Empty;
|
|
|
|
|
|
|
|
|
|
private ImmutableHashSet<Title> titles = ImmutableHashSet<Title>.Empty;
|
|
|
|
|
|
|
|
|
|
public GameDataService(HttpClient client)
|
|
|
|
|
{
|
|
|
|
|
this.client = client;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task InitializeAsync(string dataBaseUrl)
|
|
|
|
|
{
|
|
|
|
|
dataBaseUrl = dataBaseUrl.TrimEnd('/');
|
|
|
|
|
var musicInfo = await GetData<MusicInfo>(dataBaseUrl, Constants.MUSIC_INFO_BASE_NAME);
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var wordList = await GetData<WordList>(dataBaseUrl, Constants.WORDLIST_BASE_NAME);
|
2023-09-09 14:58:20 +02:00
|
|
|
|
var musicOrder = await GetData<MusicOrder>(dataBaseUrl, Constants.MUSIC_ORDER_BASE_NAME);
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var donCosRewardData = await GetData<DonCosRewards>(dataBaseUrl, Constants.DON_COS_REWARD_BASE_NAME);
|
|
|
|
|
var shougouData = await GetData<Shougous>(dataBaseUrl, Constants.SHOUGOU_BASE_NAME);
|
2023-09-09 14:58:20 +02:00
|
|
|
|
var danData = await client.GetFromJsonAsync<List<DanData>>($"{dataBaseUrl}/data/dan_data.json");
|
|
|
|
|
|
|
|
|
|
danData.ThrowIfNull();
|
|
|
|
|
danMap = danData.ToImmutableDictionary(data => data.DanId);
|
|
|
|
|
|
|
|
|
|
// To prevent duplicate entries in wordlist
|
|
|
|
|
var dict = wordList.WordListEntries.GroupBy(entry => entry.Key)
|
|
|
|
|
.ToImmutableDictionary(group => group.Key, group => group.First());
|
|
|
|
|
await Task.Run(() => InitializeMusicMap(musicInfo, dict, musicOrder));
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
|
|
|
|
InitializeCostumeFlagArraySizes(donCosRewardData);
|
|
|
|
|
InitializeTitleFlagArraySize(shougouData);
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
|
|
|
|
await Task.Run(() => InitializeHeadTitles(dict));
|
|
|
|
|
await Task.Run(() => InitializeFaceTitles(dict));
|
|
|
|
|
await Task.Run(() => InitializeBodyTitles(dict));
|
|
|
|
|
await Task.Run(() => InitializePuchiTitles(dict));
|
|
|
|
|
await Task.Run(() => InitializeKigurumiTitles(dict));
|
|
|
|
|
await Task.Run(() => InitializeTitles(dict));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task<T> GetData<T>(string dataBaseUrl, string fileBaseName) where T : notnull
|
|
|
|
|
{
|
2023-10-04 20:09:37 +02:00
|
|
|
|
var data = await client.GetFromJsonAsync<T>($"{dataBaseUrl}/data/{fileBaseName}.json");
|
|
|
|
|
data.ThrowIfNull();
|
|
|
|
|
return data;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetMusicNameBySongId(uint songId)
|
|
|
|
|
{
|
|
|
|
|
return musicMap.TryGetValue(songId, out var musicDetail) ? musicDetail.SongName : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetMusicArtistBySongId(uint songId)
|
|
|
|
|
{
|
|
|
|
|
return musicMap.TryGetValue(songId, out var musicDetail) ? musicDetail.ArtistName : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public SongGenre GetMusicGenreBySongId(uint songId)
|
|
|
|
|
{
|
|
|
|
|
return musicMap.TryGetValue(songId, out var musicDetail) ? musicDetail.Genre : SongGenre.Variety;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int GetMusicIndexBySongId(uint songId)
|
|
|
|
|
{
|
|
|
|
|
return musicMap.TryGetValue(songId, out var musicDetail) ? musicDetail.Index : int.MaxValue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public DanData GetDanDataById(uint danId)
|
|
|
|
|
{
|
|
|
|
|
return danMap.GetValueOrDefault(danId, new DanData());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int GetMusicStarLevel(uint songId, Difficulty difficulty)
|
|
|
|
|
{
|
|
|
|
|
var success = musicMap.TryGetValue(songId, out var musicDetail);
|
|
|
|
|
return difficulty switch
|
|
|
|
|
{
|
|
|
|
|
Difficulty.None => throw new ArgumentException("Difficulty cannot be none"),
|
|
|
|
|
Difficulty.Easy => success ? musicDetail!.StarEasy : 0,
|
|
|
|
|
Difficulty.Normal => success ? musicDetail!.StarNormal : 0,
|
|
|
|
|
Difficulty.Hard => success ? musicDetail!.StarHard : 0,
|
|
|
|
|
Difficulty.Oni => success ? musicDetail!.StarOni : 0,
|
|
|
|
|
Difficulty.UraOni => success ? musicDetail!.StarUra : 0,
|
|
|
|
|
_ => throw new ArgumentOutOfRangeException(nameof(difficulty), difficulty, null)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetHeadTitle(uint index)
|
|
|
|
|
{
|
|
|
|
|
return index < headTitles.Length ? headTitles[index] : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetKigurumiTitle(uint index)
|
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
return index < kigurumiTitles.Length ? kigurumiTitles[index] : string.Empty;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetBodyTitle(uint index)
|
|
|
|
|
{
|
|
|
|
|
return index < bodyTitles.Length ? bodyTitles[index] : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetFaceTitle(uint index)
|
|
|
|
|
{
|
|
|
|
|
return index < faceTitles.Length ? faceTitles[index] : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetPuchiTitle(uint index)
|
|
|
|
|
{
|
|
|
|
|
return index < puchiTitles.Length ? puchiTitles[index] : string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public ImmutableHashSet<Title> GetTitles()
|
|
|
|
|
{
|
|
|
|
|
return titles;
|
|
|
|
|
}
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
|
|
|
|
public List<int> GetCostumeFlagArraySizes()
|
|
|
|
|
{
|
|
|
|
|
return costumeFlagArraySizes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeTitleFlagArraySize(Shougous? shougouData)
|
|
|
|
|
{
|
|
|
|
|
shougouData.ThrowIfNull("Shouldn't happen!");
|
|
|
|
|
titleFlagArraySize = (int)shougouData.ShougouEntries.Max(entry => entry.uniqueId) + 1;
|
|
|
|
|
}
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
|
|
|
|
private void InitializeTitles(ImmutableDictionary<string, WordListEntry> dict)
|
|
|
|
|
{
|
|
|
|
|
var set = ImmutableHashSet.CreateBuilder<Title>();
|
2023-10-15 16:27:07 +02:00
|
|
|
|
for (var i = 1; i < titleFlagArraySize; i++)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
|
|
|
|
var key = $"syougou_{i}";
|
|
|
|
|
|
|
|
|
|
var titleWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
|
|
|
|
|
|
|
|
|
set.Add(new Title{
|
|
|
|
|
TitleName = titleWordlistItem.JapaneseText,
|
|
|
|
|
TitleId = i
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
titles = set.ToImmutable();
|
|
|
|
|
}
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
|
|
|
|
private void InitializeCostumeFlagArraySizes(DonCosRewards? donCosRewardData)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
donCosRewardData.ThrowIfNull("Shouldn't happen!");
|
|
|
|
|
var kigurumiUniqueIdList = donCosRewardData.DonCosRewardEntries
|
|
|
|
|
.Where(entry => entry.cosType == "kigurumi")
|
|
|
|
|
.Select(entry => entry.uniqueId);
|
|
|
|
|
var headUniqueIdList = donCosRewardData.DonCosRewardEntries
|
|
|
|
|
.Where(entry => entry.cosType == "head")
|
|
|
|
|
.Select(entry => entry.uniqueId);
|
|
|
|
|
var bodyUniqueIdList = donCosRewardData.DonCosRewardEntries
|
|
|
|
|
.Where(entry => entry.cosType == "body")
|
|
|
|
|
.Select(entry => entry.uniqueId);
|
|
|
|
|
var faceUniqueIdList = donCosRewardData.DonCosRewardEntries
|
|
|
|
|
.Where(entry => entry.cosType == "face")
|
|
|
|
|
.Select(entry => entry.uniqueId);
|
|
|
|
|
var puchiUniqueIdList = donCosRewardData.DonCosRewardEntries
|
|
|
|
|
.Where(entry => entry.cosType == "puchi")
|
|
|
|
|
.Select(entry => entry.uniqueId);
|
|
|
|
|
|
|
|
|
|
costumeFlagArraySizes = new List<int>
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
(int)kigurumiUniqueIdList.Max() + 1,
|
|
|
|
|
(int)headUniqueIdList.Max() + 1,
|
|
|
|
|
(int)bodyUniqueIdList.Max() + 1,
|
|
|
|
|
(int)faceUniqueIdList.Max() + 1,
|
|
|
|
|
(int)puchiUniqueIdList.Max() + 1
|
|
|
|
|
};
|
2023-09-09 14:58:20 +02:00
|
|
|
|
}
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
2023-09-09 14:58:20 +02:00
|
|
|
|
private void InitializeKigurumiTitles(ImmutableDictionary<string, WordListEntry> dict)
|
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var costumeKigurumiMax = costumeFlagArraySizes[0];
|
|
|
|
|
kigurumiTitles = new string[costumeKigurumiMax];
|
|
|
|
|
for (var i = 0; i < costumeKigurumiMax; i++)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
|
|
|
|
var key = $"costume_kigurumi_{i}";
|
|
|
|
|
|
|
|
|
|
var costumeWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
2023-10-15 16:27:07 +02:00
|
|
|
|
kigurumiTitles[i] = costumeWordlistItem.JapaneseText;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
|
|
|
|
private void InitializeHeadTitles(ImmutableDictionary<string, WordListEntry> dict)
|
|
|
|
|
{
|
|
|
|
|
var costumeHeadMax = costumeFlagArraySizes[1];
|
|
|
|
|
headTitles = new string[costumeHeadMax];
|
|
|
|
|
for (var i = 0; i < costumeHeadMax; i++)
|
|
|
|
|
{
|
|
|
|
|
var key = $"costume_head_{i}";
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var costumeWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
|
|
|
|
headTitles[i] = costumeWordlistItem.JapaneseText;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-09 14:58:20 +02:00
|
|
|
|
private void InitializeBodyTitles(ImmutableDictionary<string, WordListEntry> dict)
|
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var costumeBodyMax = costumeFlagArraySizes[2];
|
|
|
|
|
bodyTitles = new string[costumeBodyMax];
|
|
|
|
|
for (var i = 0; i < costumeBodyMax; i++)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
|
|
|
|
var key = $"costume_body_{i}";
|
|
|
|
|
|
|
|
|
|
var costumeWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
|
|
|
|
bodyTitles[i] = costumeWordlistItem.JapaneseText;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-10-15 16:27:07 +02:00
|
|
|
|
|
2023-09-09 14:58:20 +02:00
|
|
|
|
private void InitializeFaceTitles(ImmutableDictionary<string, WordListEntry> dict)
|
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var costumeFaceMax = costumeFlagArraySizes[3];
|
|
|
|
|
faceTitles = new string[costumeFaceMax];
|
|
|
|
|
for (var i = 0; i < costumeFaceMax; i++)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
|
|
|
|
var key = $"costume_face_{i}";
|
|
|
|
|
|
|
|
|
|
var costumeWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
|
|
|
|
faceTitles[i] = costumeWordlistItem.JapaneseText;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-15 16:27:07 +02:00
|
|
|
|
private void InitializePuchiTitles(ImmutableDictionary<string, WordListEntry> dict)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var costumePuchiMax = costumeFlagArraySizes[4];
|
|
|
|
|
puchiTitles = new string[costumePuchiMax];
|
|
|
|
|
for (var i = 0; i < costumePuchiMax; i++)
|
2023-09-09 14:58:20 +02:00
|
|
|
|
{
|
2023-10-15 16:27:07 +02:00
|
|
|
|
var key = $"costume_puchi_{i}";
|
2023-09-09 14:58:20 +02:00
|
|
|
|
|
|
|
|
|
var costumeWordlistItem = dict.GetValueOrDefault(key, new WordListEntry());
|
2023-10-15 16:27:07 +02:00
|
|
|
|
puchiTitles[i] = costumeWordlistItem.JapaneseText;
|
2023-09-09 14:58:20 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeMusicMap(MusicInfo musicInfo, ImmutableDictionary<string, WordListEntry> dict,
|
|
|
|
|
MusicOrder musicOrder)
|
|
|
|
|
{
|
|
|
|
|
foreach (var music in musicInfo.Items)
|
|
|
|
|
{
|
|
|
|
|
var songNameKey = $"song_{music.Id}";
|
|
|
|
|
var songArtistKey = $"song_sub_{music.Id}";
|
|
|
|
|
|
|
|
|
|
var musicName = dict.GetValueOrDefault(songNameKey, new WordListEntry());
|
|
|
|
|
var musicArtist = dict.GetValueOrDefault(songArtistKey, new WordListEntry());
|
|
|
|
|
|
|
|
|
|
var musicSongId = music.SongId;
|
|
|
|
|
var musicDetail = music.CopyPropertiesToNew<MusicDetail>();
|
|
|
|
|
musicDetail.SongName = musicName.JapaneseText;
|
|
|
|
|
musicDetail.ArtistName = musicArtist.JapaneseText;
|
|
|
|
|
|
|
|
|
|
musicMap.TryAdd(musicSongId, musicDetail);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (var index = 0; index < musicOrder.Order.Count; index++)
|
|
|
|
|
{
|
|
|
|
|
var musicOrderEntry = musicOrder.Order[index];
|
|
|
|
|
var songId = musicOrderEntry.SongId;
|
|
|
|
|
if (musicMap.ContainsKey(songId))
|
|
|
|
|
{
|
|
|
|
|
musicMap[songId].Index = index;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|