2022-08-20 20:21:30 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.ComponentModel;
|
|
|
|
using UnityEngine;
|
|
|
|
|
|
|
|
using Newtonsoft.Json;
|
2022-08-22 01:57:32 +00:00
|
|
|
using Newtonsoft.Json.Linq;
|
2022-08-20 20:21:30 +00:00
|
|
|
|
|
|
|
using HeavenStudio.Util;
|
|
|
|
|
|
|
|
namespace HeavenStudio
|
|
|
|
{
|
|
|
|
[Serializable]
|
|
|
|
public class DynamicBeatmap
|
|
|
|
{
|
2022-08-22 02:25:39 +00:00
|
|
|
public static int CurrentRiqVersion = 0;
|
2022-08-20 20:21:30 +00:00
|
|
|
public float bpm;
|
|
|
|
|
|
|
|
[JsonProperty(DefaultValueHandling = DefaultValueHandling.Populate)]
|
|
|
|
[DefaultValue(100)] public int musicVolume; // In percent (1-100)
|
|
|
|
|
|
|
|
public Dictionary<string, object> properties =
|
|
|
|
new Dictionary<string, object>() {
|
2022-08-20 23:03:51 +00:00
|
|
|
// software version (MajorMinorPatch, revision)
|
|
|
|
{"productversion", 000},
|
|
|
|
{"productsubversion", 0},
|
2022-08-23 13:19:17 +00:00
|
|
|
// file format version
|
2022-08-22 02:25:39 +00:00
|
|
|
{"riqversion", CurrentRiqVersion},
|
2022-09-04 03:17:17 +00:00
|
|
|
// mapper set properties? (future: use this to flash the button)
|
2022-08-23 13:19:17 +00:00
|
|
|
{"propertiesmodified", false},
|
2022-08-20 23:03:51 +00:00
|
|
|
|
2022-08-23 14:56:39 +00:00
|
|
|
////// CATEGORY 1: SONG INFO
|
2022-08-20 23:03:51 +00:00
|
|
|
// general chart info
|
2022-09-04 03:17:17 +00:00
|
|
|
{"remixtitle", "New Remix"}, // chart name
|
|
|
|
{"remixauthor", "Your Name"}, // charter's name
|
|
|
|
{"remixdesc", "Remix Description"}, // chart description
|
|
|
|
{"remixlevel", 1}, // chart difficulty (maybe offer a suggestion but still have the mapper determine it)
|
|
|
|
{"remixtempo", 120f}, // avg. chart tempo
|
|
|
|
{"remixtags", ""}, // chart tags
|
|
|
|
{"icontype", 0}, // chart icon (presets, custom - future)
|
|
|
|
{"iconurl", ""}, // custom icon location (future)
|
2022-08-20 23:03:51 +00:00
|
|
|
|
|
|
|
// chart song info
|
2022-09-04 03:17:17 +00:00
|
|
|
{"idolgenre", "Song Genre"}, // song genre
|
|
|
|
{"idolsong", "Song Name"}, // song name
|
|
|
|
{"idolcredit", "Artist"}, // song artist
|
2022-08-20 23:03:51 +00:00
|
|
|
|
2022-08-23 14:56:39 +00:00
|
|
|
////// CATEGORY 2: PROLOGUE AND EPILOGUE
|
2022-08-20 23:03:51 +00:00
|
|
|
// chart prologue
|
2022-09-04 03:17:17 +00:00
|
|
|
{"prologuetype", 0}, // prologue card animation (future)
|
|
|
|
{"prologuecaption", "Remix"}, // prologue card sub-title (future)
|
2022-08-20 23:03:51 +00:00
|
|
|
|
|
|
|
// chart results screen messages
|
|
|
|
{"resultcaption", "Rhythm League Notes"}, // result screen header
|
|
|
|
{"resultcommon_hi", "Good rhythm."}, // generic "Superb" message (one-liner, or second line for single-type)
|
|
|
|
{"resultcommon_ok", "Eh. Passable."}, // generic "OK" message (one-liner, or second line for single-type)
|
|
|
|
{"resultcommon_ng", "Try harder next time."}, // generic "Try Again" message (one-liner, or second line for single-type)
|
|
|
|
|
|
|
|
// the following are shown / hidden in-editor depending on the tags of the games used
|
|
|
|
{"resultnormal_hi", "You show strong fundamentals."}, // "Superb" message for normal games (two-liner)
|
|
|
|
{"resultnormal_ng", "Work on your fundamentals."}, // "Try Again" message for normal games (two-liner)
|
|
|
|
|
|
|
|
{"resultkeep_hi", "You kept the beat well."}, // "Superb" message for keep-the-beat games (two-liner)
|
|
|
|
{"resultkeep_ng", "You had trouble keeping the beat."}, // "Try Again" message for keep-the-beat games (two-liner)
|
|
|
|
|
|
|
|
{"resultaim_hi", "You had great aim."}, // "Superb" message for aim games (two-liner)
|
|
|
|
{"resultaim_ng", "Your aim was a little shaky."}, // "Try Again" message for aim games (two-liner)
|
|
|
|
|
|
|
|
{"resultrepeat_hi", "You followed the example well."}, // "Superb" message for call-and-response games (two-liner)
|
|
|
|
{"resultrepeat_ng", "Next time, follow the example better."}, // "Try Again" message for call-and-response games (two-liner)
|
2022-08-20 20:21:30 +00:00
|
|
|
};
|
2022-08-20 23:03:51 +00:00
|
|
|
|
2022-08-20 20:21:30 +00:00
|
|
|
public List<DynamicEntity> entities = new List<DynamicEntity>();
|
|
|
|
public List<TempoChange> tempoChanges = new List<TempoChange>();
|
|
|
|
public List<VolumeChange> volumeChanges = new List<VolumeChange>();
|
2022-08-23 14:56:39 +00:00
|
|
|
public List<ChartSection> beatmapSections = new List<ChartSection>();
|
2022-08-20 20:21:30 +00:00
|
|
|
public float firstBeatOffset;
|
|
|
|
|
|
|
|
[Serializable]
|
|
|
|
public class DynamicEntity : ICloneable
|
|
|
|
{
|
|
|
|
public float beat;
|
|
|
|
public int track;
|
|
|
|
[JsonProperty(DefaultValueHandling = DefaultValueHandling.Ignore)] public float length;
|
|
|
|
[JsonProperty(DefaultValueHandling = DefaultValueHandling.Ignore)] public float swing;
|
2022-08-21 23:46:45 +00:00
|
|
|
public Dictionary<string, dynamic> DynamicData = new Dictionary<string, dynamic>();
|
2022-08-20 20:21:30 +00:00
|
|
|
|
|
|
|
public string datamodel;
|
|
|
|
[JsonIgnore] public Editor.Track.TimelineEventObj eventObj;
|
|
|
|
|
|
|
|
public object Clone()
|
|
|
|
{
|
|
|
|
return this.MemberwiseClone();
|
|
|
|
}
|
|
|
|
|
|
|
|
public DynamicEntity DeepCopy()
|
|
|
|
{
|
2022-08-23 13:19:17 +00:00
|
|
|
DynamicEntity copy = (DynamicEntity)this.MemberwiseClone();
|
|
|
|
copy.DynamicData = new Dictionary<string, dynamic>(this.DynamicData);
|
|
|
|
return copy;
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
|
2022-08-21 23:46:45 +00:00
|
|
|
public dynamic this[string propertyName]
|
2022-08-20 20:21:30 +00:00
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2022-08-22 02:25:39 +00:00
|
|
|
switch (propertyName)
|
2022-08-22 01:57:32 +00:00
|
|
|
{
|
2022-08-22 02:25:39 +00:00
|
|
|
case "beat":
|
|
|
|
return beat;
|
|
|
|
case "track":
|
|
|
|
return track;
|
|
|
|
case "length":
|
|
|
|
return length;
|
|
|
|
case "swing":
|
|
|
|
return swing;
|
|
|
|
case "datamodel":
|
|
|
|
return datamodel;
|
|
|
|
default:
|
|
|
|
if (DynamicData.ContainsKey(propertyName))
|
2022-08-22 23:14:38 +00:00
|
|
|
return DynamicData[propertyName];
|
2022-08-22 01:57:32 +00:00
|
|
|
else
|
2022-08-22 02:25:39 +00:00
|
|
|
{
|
2022-08-22 23:14:38 +00:00
|
|
|
Minigames.Minigame game = EventCaller.instance.GetMinigame(datamodel.Split(0));
|
|
|
|
Minigames.Param param = EventCaller.instance.GetGameParam(game, datamodel.Split(1), propertyName);
|
2022-08-22 02:25:39 +00:00
|
|
|
return param.parameter;
|
|
|
|
}
|
2022-08-22 01:57:32 +00:00
|
|
|
}
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2022-08-23 13:24:42 +00:00
|
|
|
switch (propertyName)
|
2022-08-20 20:21:30 +00:00
|
|
|
{
|
2022-08-23 13:24:42 +00:00
|
|
|
case "beat":
|
|
|
|
case "track":
|
|
|
|
case "length":
|
|
|
|
case "swing":
|
|
|
|
case "datamodel":
|
|
|
|
UnityEngine.Debug.LogWarning($"Property name {propertyName} is reserved and cannot be set.");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (DynamicData.ContainsKey(propertyName))
|
|
|
|
DynamicData[propertyName] = value;
|
|
|
|
else
|
|
|
|
UnityEngine.Debug.LogError($"This entity does not have a property named {propertyName}! Attempted to insert value of type {value.GetType()}");
|
|
|
|
break;
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
2022-08-23 13:24:42 +00:00
|
|
|
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
}
|
2022-08-20 23:03:51 +00:00
|
|
|
|
2022-08-21 23:46:45 +00:00
|
|
|
public void CreateProperty(string name, dynamic defaultValue)
|
2022-08-20 23:03:51 +00:00
|
|
|
{
|
|
|
|
if (!DynamicData.ContainsKey(name))
|
|
|
|
DynamicData.Add(name, defaultValue);
|
|
|
|
}
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
[Serializable]
|
|
|
|
public class TempoChange : ICloneable
|
|
|
|
{
|
|
|
|
public float beat;
|
|
|
|
public float length;
|
|
|
|
public float tempo;
|
|
|
|
|
|
|
|
public object Clone()
|
|
|
|
{
|
|
|
|
return this.MemberwiseClone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[Serializable]
|
|
|
|
public class VolumeChange : ICloneable
|
|
|
|
{
|
|
|
|
public float beat;
|
|
|
|
public float length;
|
|
|
|
public float volume;
|
|
|
|
|
|
|
|
public object Clone()
|
|
|
|
{
|
|
|
|
return this.MemberwiseClone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-23 14:56:39 +00:00
|
|
|
[Serializable]
|
|
|
|
public class ChartSection : ICloneable
|
|
|
|
{
|
|
|
|
public float beat;
|
|
|
|
public bool startPerfect;
|
|
|
|
public string sectionName;
|
|
|
|
public bool isCheckpoint; // really don't think we need this but who knows
|
|
|
|
|
|
|
|
public object Clone()
|
|
|
|
{
|
|
|
|
return this.MemberwiseClone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-23 13:19:17 +00:00
|
|
|
public dynamic this[string propertyName]
|
2022-08-20 20:21:30 +00:00
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2022-08-23 13:19:17 +00:00
|
|
|
return properties[propertyName] ?? null;
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
if (properties.ContainsKey(propertyName))
|
|
|
|
{
|
|
|
|
properties[propertyName] = value;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
UnityEngine.Debug.LogError($"This beatmap does not have a property named {propertyName}! Attempted to insert value of type {value.GetType()}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-21 23:46:45 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// converts from the old "rhmania" / "tengoku" format to the new "riq" format
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="beatmap">a deserialized .rhmania or .tengoku beatmap</param>
|
|
|
|
/// <returns>a .riq beatmap</returns>
|
|
|
|
public static DynamicBeatmap BeatmapConverter(Beatmap beatmap)
|
|
|
|
{
|
|
|
|
DynamicBeatmap dynamicBeatmap = new DynamicBeatmap();
|
|
|
|
dynamicBeatmap.bpm = beatmap.bpm;
|
|
|
|
dynamicBeatmap.musicVolume = beatmap.musicVolume;
|
|
|
|
dynamicBeatmap.firstBeatOffset = beatmap.firstBeatOffset;
|
|
|
|
|
2022-08-22 23:14:38 +00:00
|
|
|
Minigames.Minigame game;
|
|
|
|
Minigames.GameAction action;
|
|
|
|
System.Type type, pType;
|
|
|
|
foreach (var e in beatmap.entities)
|
2022-08-21 23:46:45 +00:00
|
|
|
{
|
2022-08-22 23:14:38 +00:00
|
|
|
game = EventCaller.instance.GetMinigame(e.datamodel.Split(0));
|
|
|
|
action = EventCaller.instance.GetGameAction(game, e.datamodel.Split(1));
|
|
|
|
|
2022-09-08 20:29:56 +00:00
|
|
|
if (game == null || action == null)
|
|
|
|
{
|
|
|
|
//FUTURE: attempt to convert to a new entity if a converter exists for this datamodel
|
|
|
|
UnityEngine.Debug.LogError($"Could not find game or gameaction from datamodel {e.datamodel} @ beat {e.beat}, skipping entity");
|
|
|
|
continue;
|
|
|
|
}
|
2022-09-04 01:51:37 +00:00
|
|
|
// Debug.Log($"{game.name} {action.displayName} @ beat {e.beat}");
|
|
|
|
|
2022-08-22 23:14:38 +00:00
|
|
|
Dictionary<string, dynamic> dynamicData = new Dictionary<string, dynamic>();
|
|
|
|
//check each param of the action
|
|
|
|
if (action.parameters != null)
|
2022-08-21 23:46:45 +00:00
|
|
|
{
|
2022-08-22 23:14:38 +00:00
|
|
|
foreach (var param in action.parameters)
|
2022-08-21 23:46:45 +00:00
|
|
|
{
|
2022-08-22 23:14:38 +00:00
|
|
|
type = param.parameter.GetType();
|
|
|
|
pType = e[param.propertyName].GetType();
|
2022-09-04 01:51:37 +00:00
|
|
|
// Debug.Log($"adding parameter {param.propertyName} of type {type}");
|
|
|
|
if (!dynamicData.ContainsKey(param.propertyName))
|
2022-08-22 23:14:38 +00:00
|
|
|
{
|
2022-09-04 01:51:37 +00:00
|
|
|
if (pType == type)
|
|
|
|
{
|
|
|
|
dynamicData.Add(param.propertyName, e[param.propertyName]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (type == typeof(EntityTypes.Integer))
|
|
|
|
dynamicData.Add(param.propertyName, (int) e[param.propertyName]);
|
|
|
|
else if (type == typeof(EntityTypes.Float))
|
|
|
|
dynamicData.Add(param.propertyName, (float) e[param.propertyName]);
|
|
|
|
else if (type.IsEnum && param.propertyName != "ease")
|
|
|
|
dynamicData.Add(param.propertyName, (int) e[param.propertyName]);
|
|
|
|
else if (pType == typeof(Newtonsoft.Json.Linq.JObject))
|
|
|
|
dynamicData.Add(param.propertyName, e[param.propertyName].ToObject(type));
|
|
|
|
else
|
|
|
|
dynamicData.Add(param.propertyName, Convert.ChangeType(e[param.propertyName], type));
|
|
|
|
}
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-04 01:51:37 +00:00
|
|
|
Debug.LogWarning($"Property {param.propertyName} already exists in the entity's dynamic data! Skipping...");
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
2022-08-21 23:46:45 +00:00
|
|
|
}
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
2022-09-04 01:51:37 +00:00
|
|
|
|
2022-08-22 23:14:38 +00:00
|
|
|
dynamicBeatmap.entities.Add(new DynamicEntity()
|
|
|
|
{
|
|
|
|
beat = e.beat,
|
|
|
|
track = e.track,
|
|
|
|
length = e.length,
|
|
|
|
swing = e.swing,
|
|
|
|
datamodel = e.datamodel,
|
|
|
|
DynamicData = dynamicData
|
2022-08-21 23:46:45 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
foreach (var tempoChange in beatmap.tempoChanges)
|
|
|
|
{
|
|
|
|
dynamicBeatmap.tempoChanges.Add(new TempoChange()
|
|
|
|
{
|
|
|
|
beat = tempoChange.beat,
|
|
|
|
length = tempoChange.length,
|
|
|
|
tempo = tempoChange.tempo
|
|
|
|
});
|
|
|
|
}
|
|
|
|
foreach (var volumeChange in beatmap.volumeChanges)
|
|
|
|
{
|
|
|
|
dynamicBeatmap.volumeChanges.Add(new VolumeChange()
|
|
|
|
{
|
|
|
|
beat = volumeChange.beat,
|
|
|
|
length = volumeChange.length,
|
|
|
|
volume = volumeChange.volume
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return dynamicBeatmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// FUTURE: converts from a karateka mania chart ("bor") to the "riq" format
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="bor">a rawtext .bor chart</param>
|
|
|
|
/// <returns>a .riq beatmap</returns>
|
|
|
|
/// <remarks>not implemented yet</remarks>
|
|
|
|
public static DynamicBeatmap KManiaBorConverter(String bor)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// updates an "riq" beatmap
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="beatmap">old beatmap</param>
|
|
|
|
/// <param name="version">version of old beatmap</param>
|
|
|
|
/// <returns>updated beatmap</returns>
|
|
|
|
/// <remarks>not implemented yet</remarks>
|
|
|
|
public static DynamicBeatmap BeatmapUpdater(DynamicBeatmap beatmap, int version)
|
|
|
|
{
|
|
|
|
return beatmap;
|
|
|
|
}
|
|
|
|
|
2022-08-22 23:14:38 +00:00
|
|
|
/// <summary>
|
|
|
|
/// processes an riq beatmap after it is loaded
|
|
|
|
/// </summary>
|
|
|
|
public void PostProcess()
|
|
|
|
{
|
2022-09-04 03:17:17 +00:00
|
|
|
DynamicBeatmap beatmapModel = new DynamicBeatmap();
|
2022-08-22 23:14:38 +00:00
|
|
|
Minigames.Minigame game;
|
|
|
|
Minigames.GameAction action;
|
|
|
|
System.Type type, pType;
|
|
|
|
foreach (var e in entities)
|
|
|
|
{
|
|
|
|
game = EventCaller.instance.GetMinigame(e.datamodel.Split(0));
|
|
|
|
action = EventCaller.instance.GetGameAction(game, e.datamodel.Split(1));
|
|
|
|
Dictionary<string, dynamic> dynamicData = new Dictionary<string, dynamic>();
|
|
|
|
//check each param of the action
|
|
|
|
if (action.parameters != null)
|
|
|
|
{
|
|
|
|
foreach (var param in action.parameters)
|
|
|
|
{
|
2022-09-04 01:51:37 +00:00
|
|
|
if (!dynamicData.ContainsKey(param.propertyName))
|
2022-08-22 23:14:38 +00:00
|
|
|
{
|
2022-09-04 01:51:37 +00:00
|
|
|
type = param.parameter.GetType();
|
|
|
|
pType = e[param.propertyName].GetType();
|
|
|
|
if (pType == type)
|
|
|
|
{
|
|
|
|
dynamicData.Add(param.propertyName, e[param.propertyName]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (type == typeof(EntityTypes.Integer))
|
|
|
|
dynamicData.Add(param.propertyName, (int)e[param.propertyName]);
|
|
|
|
else if (type == typeof(EntityTypes.Float))
|
|
|
|
dynamicData.Add(param.propertyName, (float)e[param.propertyName]);
|
|
|
|
else if (type == typeof(EasingFunction.Ease) && pType == typeof(string))
|
|
|
|
dynamicData.Add(param.propertyName, Enum.Parse(typeof(EasingFunction.Ease), (string)e[param.propertyName]));
|
|
|
|
else if (type.IsEnum)
|
|
|
|
dynamicData.Add(param.propertyName, (int)e[param.propertyName]);
|
|
|
|
else if (pType == typeof(Newtonsoft.Json.Linq.JObject))
|
|
|
|
dynamicData.Add(param.propertyName, e[param.propertyName].ToObject(type));
|
|
|
|
else
|
|
|
|
dynamicData.Add(param.propertyName, Convert.ChangeType(e[param.propertyName], type));
|
|
|
|
}
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-04 01:51:37 +00:00
|
|
|
Debug.LogWarning($"Property {param.propertyName} already exists in the entity's dynamic data! Skipping...");
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
e.DynamicData = dynamicData;
|
|
|
|
}
|
2022-09-04 03:17:17 +00:00
|
|
|
//go thru each property of the model beatmap and add any missing keyvalue pair
|
|
|
|
foreach (var prop in beatmapModel.properties)
|
|
|
|
{
|
|
|
|
if (!properties.ContainsKey(prop.Key))
|
|
|
|
{
|
|
|
|
properties.Add(prop.Key, prop.Value);
|
|
|
|
}
|
|
|
|
}
|
2022-08-22 23:14:38 +00:00
|
|
|
}
|
2022-08-20 20:21:30 +00:00
|
|
|
}
|
|
|
|
}
|