feat: SongCreator 씬 완성 — Beat Sage URL 지원, info.dat 메타데이터 자동 추출

- BeatSageUploader: audio_url 지원(UploadFromUrl), PollAndDownload 공통화, ZIP 500 오류 3회 재시도
- BeatSageConverter: info.dat 파싱(SongMetadata), BPM 자동 감지 → 노트 타이밍 변환에 적용
- SongCreatorManager: title/BPM 필수 입력 제거, 난이도 4개 자동 선택, GenerateFlowFromUrl 버그 수정
- NasPublisher: audioPath null 허용(URL 흐름에서 로컬 파일 없는 경우 스킵)
- .gitignore/.gitattributes 초기 설정

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-21 23:37:34 +09:00
commit 4dad9e5d5b
1068 changed files with 175146 additions and 0 deletions
@@ -0,0 +1,114 @@
using DamageSystem;
using UnityEngine;
using VRBeats.ScriptableEvents;
namespace VRBeats
{
public class VR_BeatCube : MonoBehaviour
{
[SerializeField] private float minCutSpeed = 0.5f;
[SerializeField] private OnSliceAction sliceAction = null;
[SerializeField] private GameEvent onCorrectSlice = null;
[SerializeField] private GameEvent onIncorrectSlice = null;
[SerializeField] private GameEvent onPlayerMiss = null;
private MaterialBindings materialBindings = null;
private ColorSide thisColorSide = ColorSide.Right;
private Transform player = null;
private VR_BeatCubeSpawneable thisSpawneable = null;
private bool canBeKilled = true;
private bool spawnComplete = false;
private bool destroyed = false;
public float MinCutSpeed { get { return minCutSpeed; } }
public Direction HitDirection { get { return thisSpawneable.HitDirection; } }
public ColorSide ThisColorSide { get { return thisColorSide; } }
private void Awake()
{
player = VR_BeatManager.instance.Player.transform;
}
public void Start()
{
thisSpawneable = GetComponent<VR_BeatCubeSpawneable>();
thisSpawneable.onSpawnComplete += delegate { spawnComplete = true; };
materialBindings = GetComponent<MaterialBindings>();
thisColorSide = thisSpawneable.ColorSide;
Color color = VR_BeatManager.instance.GetColorFromColorSide(thisColorSide);
materialBindings.SetEmmisiveColor( color );
}
private void OnDestroy()
{
destroyed = true;
}
public void OnCut(DamageInfo info)
{
canBeKilled = false;
//notify to whoever is listening that the player did a correct/incorrect slice
if ( IsCutIntentValid(info as BeatDamageInfo) )
{
onCorrectSlice.Invoke();
}
else
{
onIncorrectSlice.Invoke();
}
}
private bool IsCutIntentValid(BeatDamageInfo info)
{
if (info == null) return false;
if (info.velocity < minCutSpeed) return false;
//no matter the hit direction as soon as we have the right velocity for a cube that has a dot
if (HitDirection == Direction.Center)
return true;
float cutAngle = Vector2.Angle(transform.up, info.hitDir);
return info.colorSide == ThisColorSide && cutAngle < 80.0f;
}
private void Update()
{
if(spawnComplete)
transform.position += Vector3.forward * thisSpawneable.Speed * Time.deltaTime;
if ( ShouldKillCube() )
{
Kill();
}
}
private bool ShouldKillCube()
{
return canBeKilled && transform.position.z < player.position.z - 2.0f;
}
public void Kill()
{
onPlayerMiss.Invoke();
canBeKilled = false;
transform.ScaleTween(Vector3.zero, 2.0f).SetEase(Ease.EaseOutExpo).SetOnComplete( delegate
{
if(!destroyed)
Destroy(gameObject);
} );
}
}
}