From ddc183acdde7834ccd45d49d55b8d7e68334f75f Mon Sep 17 00:00:00 2001
From: Zeo <67521686+ThatZeoMan@users.noreply.github.com>
Date: Fri, 6 Jan 2023 21:59:54 -0600
Subject: [PATCH] Unrecognized entity support (#191)
Due to some recent discoveries about a certain upcoming rhythm game, I have added support for unrecognized entities found in remix.json to be loaded and moved, and be able to save them.
---
.../Scripts/BeatmapFormats/DynamicBeatmap.cs | 58 +++++++++++++++++--
Assets/Scripts/GameManager.cs | 8 ++-
Assets/Scripts/LevelEditor/Editor.cs | 21 ++++---
3 files changed, 72 insertions(+), 15 deletions(-)
diff --git a/Assets/Scripts/BeatmapFormats/DynamicBeatmap.cs b/Assets/Scripts/BeatmapFormats/DynamicBeatmap.cs
index 543fddef1..77ef3024b 100644
--- a/Assets/Scripts/BeatmapFormats/DynamicBeatmap.cs
+++ b/Assets/Scripts/BeatmapFormats/DynamicBeatmap.cs
@@ -1,6 +1,7 @@
using System;
using System.Collections.Generic;
using System.ComponentModel;
+using System.Text.RegularExpressions;
using UnityEngine;
using Newtonsoft.Json;
@@ -344,8 +345,33 @@ namespace HeavenStudio
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));
+ var gameName = e.datamodel.Split(0);
+ var actionName = e.datamodel.Split(1);
+ game = EventCaller.instance.GetMinigame(gameName);
+ if (game == null)
+ {
+ Debug.LogWarning($"Unknown game {gameName} found in remix.json! Adding game...");
+ game = new Minigames.Minigame(gameName, DisplayName(gameName) + " \n[inferred from remix.json]", "", false, true, new List());
+ EventCaller.instance.minigames.Add(game);
+ Editor.Editor.instance.AddIcon(game);
+ }
+ action = EventCaller.instance.GetGameAction(game, actionName);
+ if (action == null)
+ {
+ Debug.LogWarning($"Unknown action {gameName}/{actionName} found in remix.json! Adding action...");
+ var parameters = new List();
+ foreach (var item in e.DynamicData)
+ {
+ var value = item.Value;
+ if (value.GetType() == typeof(long))
+ value = new EntityTypes.Integer(int.MinValue, int.MaxValue, (int)value);
+ else if (value.GetType() == typeof(double))
+ value = new EntityTypes.Float(float.NegativeInfinity, float.PositiveInfinity, (float)value);
+ parameters.Add(new Minigames.Param(item.Key, value, item.Key, "[inferred from remix.json]"));
+ }
+ action = new Minigames.GameAction(actionName, DisplayName(actionName), e.length, true, parameters);
+ game.actions.Add(action);
+ }
Dictionary dynamicData = new Dictionary();
//check each param of the action
if (action.parameters != null)
@@ -361,9 +387,9 @@ namespace HeavenStudio
{
Debug.LogWarning($"Property {param.propertyName} does not exist in the entity's dynamic data! Adding...");
if (type == typeof(EntityTypes.Integer))
- dynamicData.Add(param.propertyName, (int)param.parameter);
+ dynamicData.Add(param.propertyName, ((EntityTypes.Integer)param.parameter).val);
else if (type == typeof(EntityTypes.Float))
- dynamicData.Add(param.propertyName, (float)param.parameter);
+ dynamicData.Add(param.propertyName, ((EntityTypes.Float)param.parameter).val);
else if (type.IsEnum && param.propertyName != "ease")
dynamicData.Add(param.propertyName, (int)param.parameter);
else
@@ -408,5 +434,29 @@ namespace HeavenStudio
}
}
}
+
+ private string DisplayName(string name)
+ {
+ // "gameName" -> "Game Name"
+ // "action name" -> "Action Name"
+ if (!name.Contains(" "))
+ name = SplitCamelCase(name);
+ System.Globalization.TextInfo textInfo = new System.Globalization.CultureInfo("en-US", false).TextInfo;
+ return textInfo.ToTitleCase(name);
+ }
+
+ // https://stackoverflow.com/a/5796793
+ public static string SplitCamelCase(string str)
+ {
+ return Regex.Replace(
+ Regex.Replace(
+ str,
+ @"(\P{Ll})(\P{Ll}\p{Ll})",
+ "$1 $2"
+ ),
+ @"(\p{Ll})(\P{Ll})",
+ "$1 $2"
+ );
+ }
}
}
\ No newline at end of file
diff --git a/Assets/Scripts/GameManager.cs b/Assets/Scripts/GameManager.cs
index 876bf42a3..829e50a94 100644
--- a/Assets/Scripts/GameManager.cs
+++ b/Assets/Scripts/GameManager.cs
@@ -599,14 +599,18 @@ namespace HeavenStudio
{
if (gameInfo.fxOnly)
{
- name = Beatmap.entities.FindAll(c => {
+ var gameEntities = Beatmap.entities.FindAll(c => {
var gameName = c.datamodel.Split(0);
var newGameInfo = GetGameInfo(gameName);
if (newGameInfo == null)
return false;
else
return !newGameInfo.fxOnly;
- }).ToList()[0].datamodel.Split(0);
+ }).ToList();
+ if (gameEntities.Count != 0)
+ name = gameEntities[0].datamodel.Split(0);
+ else
+ name = "noGame";
}
else
{
diff --git a/Assets/Scripts/LevelEditor/Editor.cs b/Assets/Scripts/LevelEditor/Editor.cs
index 4e2b3617d..549fafb2d 100644
--- a/Assets/Scripts/LevelEditor/Editor.cs
+++ b/Assets/Scripts/LevelEditor/Editor.cs
@@ -100,15 +100,8 @@ namespace HeavenStudio.Editor
GameManager.instance.Init();
Timeline.Init();
- for (int i = 0; i < EventCaller.instance.minigames.Count; i++)
- {
- GameObject GameIcon_ = Instantiate(GridGameSelector.GetChild(0).gameObject, GridGameSelector);
- GameIcon_.GetComponent().sprite = GameIcon(EventCaller.instance.minigames[i].name);
- GameIcon_.GetComponent().MaskTex = GameIconMask(EventCaller.instance.minigames[i].name);
- GameIcon_.GetComponent().UnClickIcon();
- GameIcon_.gameObject.SetActive(true);
- GameIcon_.name = EventCaller.instance.minigames[i].displayName;
- }
+ foreach (var minigame in EventCaller.instance.minigames)
+ AddIcon(minigame);
Tooltip.AddTooltip(NewBTN.gameObject, "New [Ctrl+N]");
Tooltip.AddTooltip(OpenBTN.gameObject, "Open [Ctrl+O]");
@@ -128,6 +121,16 @@ namespace HeavenStudio.Editor
BuildDateDisplay.text = GlobalGameManager.buildTime;
}
+ public void AddIcon(Minigames.Minigame minigame)
+ {
+ GameObject GameIcon_ = Instantiate(GridGameSelector.GetChild(0).gameObject, GridGameSelector);
+ GameIcon_.GetComponent().sprite = GameIcon(minigame.name);
+ GameIcon_.GetComponent().MaskTex = GameIconMask(minigame.name);
+ GameIcon_.GetComponent().UnClickIcon();
+ GameIcon_.gameObject.SetActive(true);
+ GameIcon_.name = minigame.displayName;
+ }
+
public void LateUpdate()
{
#region Keyboard Shortcuts