1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-24 22:07:27 +08:00
osu-lazer/osu.Game.Rulesets.Mania/UI/ManiaHitRenderer.cs

192 lines
7.5 KiB
C#
Raw Normal View History

// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2017-05-10 13:56:39 +08:00
using System;
using System.Collections.Generic;
2017-05-10 13:56:39 +08:00
using System.Linq;
2017-05-11 13:26:00 +08:00
using OpenTK;
using OpenTK.Input;
2017-05-29 13:44:42 +08:00
using osu.Framework.Allocation;
using osu.Framework.Configuration;
using osu.Framework.Extensions.IEnumerableExtensions;
2017-05-11 13:26:00 +08:00
using osu.Framework.Graphics;
using osu.Framework.Lists;
2017-05-29 13:44:42 +08:00
using osu.Framework.MathUtils;
using osu.Game.Beatmaps;
using osu.Game.Beatmaps.ControlPoints;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Beatmaps;
using osu.Game.Rulesets.Mania.Beatmaps;
using osu.Game.Rulesets.Mania.Judgements;
using osu.Game.Rulesets.Mania.Objects;
using osu.Game.Rulesets.Mania.Objects.Drawables;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Mania.Scoring;
using osu.Game.Rulesets.Mania.Timing;
using osu.Game.Rulesets.Mania.Timing.Drawables;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Objects.Drawables;
2017-05-10 13:56:39 +08:00
using osu.Game.Rulesets.Objects.Types;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Rulesets.UI;
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Mania.UI
{
2017-05-03 11:44:19 +08:00
public class ManiaHitRenderer : HitRenderer<ManiaHitObject, ManiaJudgement>
{
2017-06-02 16:33:58 +08:00
private int? columns;
2017-06-02 17:19:21 +08:00
public int Columns
{
get { return columns ?? (int)Math.Round(Beatmap.BeatmapInfo.Difficulty.CircleSize); }
set { columns = value; }
}
2017-06-02 16:33:58 +08:00
public Dictionary<int, List<DrawableTimingChange>> HitObjectTimingChanges;
public List<DrawableTimingChange> BarlineTimingChanges;
public ManiaHitRenderer(WorkingBeatmap beatmap, bool isForCurrentRuleset)
: base(beatmap, isForCurrentRuleset)
{
generateDefaultTimingChanges();
2017-05-10 13:56:39 +08:00
}
private void generateDefaultTimingChanges()
2017-05-10 13:56:39 +08:00
{
2017-06-02 16:33:58 +08:00
if (HitObjectTimingChanges != null || BarlineTimingChanges != null)
return;
HitObjectTimingChanges = new Dictionary<int, List<DrawableTimingChange>>();
BarlineTimingChanges = new List<DrawableTimingChange>();
double lastSpeedMultiplier = 1;
double lastBeatLength = 500;
2017-05-11 21:16:50 +08:00
// Merge timing + difficulty points
var allPoints = new SortedList<ControlPoint>(Comparer<ControlPoint>.Default);
allPoints.AddRange(Beatmap.ControlPointInfo.TimingPoints);
allPoints.AddRange(Beatmap.ControlPointInfo.DifficultyPoints);
2017-05-12 21:23:32 +08:00
// Generate the timing points, making non-timing changes use the previous timing change
var timingChanges = allPoints.Select(c =>
2017-05-10 13:56:39 +08:00
{
var timingPoint = c as TimingControlPoint;
var difficultyPoint = c as DifficultyControlPoint;
if (timingPoint != null)
lastBeatLength = timingPoint.BeatLength;
2017-05-12 21:23:32 +08:00
if (difficultyPoint != null)
lastSpeedMultiplier = difficultyPoint.SpeedMultiplier;
2017-05-10 13:56:39 +08:00
return new TimingChange
{
Time = c.Time,
BeatLength = lastBeatLength,
SpeedMultiplier = lastSpeedMultiplier
};
2017-05-12 21:23:32 +08:00
});
2017-05-10 13:56:39 +08:00
2017-05-18 18:05:03 +08:00
double lastObjectTime = (Objects.LastOrDefault() as IHasEndTime)?.EndTime ?? Objects.LastOrDefault()?.StartTime ?? double.MaxValue;
2017-05-10 13:56:39 +08:00
2017-05-12 21:23:32 +08:00
// Perform some post processing of the timing changes
timingChanges = timingChanges
2017-05-10 13:56:39 +08:00
// Collapse sections after the last hit object
2017-05-12 21:23:32 +08:00
.Where(s => s.Time <= lastObjectTime)
2017-05-10 13:56:39 +08:00
// Collapse sections with the same start time
2017-05-12 21:23:32 +08:00
.GroupBy(s => s.Time).Select(g => g.Last()).OrderBy(s => s.Time)
2017-05-10 13:56:39 +08:00
// Collapse sections with the same beat length
2017-05-12 21:23:32 +08:00
.GroupBy(s => s.BeatLength * s.SpeedMultiplier).Select(g => g.First())
2017-05-10 13:56:39 +08:00
.ToList();
2017-06-02 16:33:58 +08:00
timingChanges.ForEach(t =>
{
for (int i = 0; i < Columns; i++)
{
List<DrawableTimingChange> columnTimingChanges;
if (!HitObjectTimingChanges.TryGetValue(i, out columnTimingChanges))
HitObjectTimingChanges[i] = columnTimingChanges = new List<DrawableTimingChange>();
columnTimingChanges.Add(new DrawableScrollingTimingChange(t));
}
BarlineTimingChanges.Add(new DrawableScrollingTimingChange(t));
});
}
protected override Playfield<ManiaHitObject, ManiaJudgement> CreatePlayfield()
{
var playfield = new ManiaPlayfield(Columns)
2017-05-11 13:26:00 +08:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
// Invert by default for now (should be moved to config/skin later)
Scale = new Vector2(1, -1)
2017-05-11 13:26:00 +08:00
};
2017-06-02 16:33:58 +08:00
foreach (var kvp in HitObjectTimingChanges)
{
int column = kvp.Key;
List<DrawableTimingChange> timingChanges = kvp.Value;
foreach (var change in timingChanges)
playfield.Columns.ElementAt(column).Add(change);
}
foreach (var change in BarlineTimingChanges)
playfield.Add(change);
return playfield;
}
2017-05-29 13:44:42 +08:00
[BackgroundDependencyLoader]
private void load()
{
var maniaPlayfield = (ManiaPlayfield)Playfield;
double lastObjectTime = (Objects.LastOrDefault() as IHasEndTime)?.EndTime ?? Objects.LastOrDefault()?.StartTime ?? double.MaxValue;
SortedList<TimingControlPoint> timingPoints = Beatmap.ControlPointInfo.TimingPoints;
for (int i = 0; i < timingPoints.Count; i++)
{
TimingControlPoint point = timingPoints[i];
2017-05-29 14:30:05 +08:00
// Stop on the beat before the next timing point, or if there is no next timing point stop slightly past the last object
2017-05-29 13:44:42 +08:00
double endTime = i < timingPoints.Count - 1 ? timingPoints[i + 1].Time - point.BeatLength : lastObjectTime + point.BeatLength * (int)point.TimeSignature;
int index = 0;
for (double t = timingPoints[i].Time; Precision.DefinitelyBigger(endTime, t); t += point.BeatLength, index++)
{
2017-05-29 14:01:13 +08:00
maniaPlayfield.Add(new DrawableBarLine(new BarLine
2017-05-29 13:44:42 +08:00
{
StartTime = t,
ControlPoint = point,
BeatIndex = index
}));
}
}
}
public override ScoreProcessor CreateScoreProcessor() => new ManiaScoreProcessor(this);
2017-05-03 11:44:19 +08:00
protected override BeatmapConverter<ManiaHitObject> CreateBeatmapConverter() => new ManiaBeatmapConverter();
protected override DrawableHitObject<ManiaHitObject, ManiaJudgement> GetVisualRepresentation(ManiaHitObject h)
{
var maniaPlayfield = Playfield as ManiaPlayfield;
if (maniaPlayfield == null)
return null;
Bindable<Key> key = maniaPlayfield.Columns.ElementAt(h.Column).Key;
2017-05-12 15:35:57 +08:00
var holdNote = h as HoldNote;
if (holdNote != null)
return new DrawableHoldNote(holdNote, key);
2017-05-12 15:35:57 +08:00
var note = h as Note;
if (note != null)
return new DrawableNote(note, key);
return null;
}
2017-05-11 13:26:00 +08:00
protected override Vector2 GetPlayfieldAspectAdjust() => new Vector2(1, 0.8f);
}
}