2019-01-24 16:43:03 +08:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 17:19:50 +08:00
using System ;
using System.Collections.Generic ;
2018-11-28 15:39:08 +08:00
using System.ComponentModel.DataAnnotations.Schema ;
using System.Linq ;
2018-09-28 17:29:49 +08:00
using Newtonsoft.Json ;
2021-07-24 04:37:08 +08:00
using osu.Framework.Localisation ;
2018-04-13 17:19:50 +08:00
using osu.Game.Beatmaps ;
2018-11-28 15:39:08 +08:00
using osu.Game.Database ;
2021-04-12 18:50:24 +08:00
using osu.Game.Online.API ;
2021-11-04 17:02:44 +08:00
using osu.Game.Online.API.Requests.Responses ;
2018-11-28 15:33:42 +08:00
using osu.Game.Rulesets ;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Mods ;
2018-11-28 15:33:42 +08:00
using osu.Game.Rulesets.Scoring ;
2020-02-04 12:17:23 +08:00
using osu.Game.Utils ;
2018-04-13 17:19:50 +08:00
2018-11-28 15:12:57 +08:00
namespace osu.Game.Scoring
2018-04-13 17:19:50 +08:00
{
2021-10-28 16:57:17 +08:00
public class ScoreInfo : IScoreInfo , IHasFiles < ScoreFileInfo > , IHasPrimaryKey , ISoftDelete , IEquatable < ScoreInfo > , IDeepCloneable < ScoreInfo >
2018-04-13 17:19:50 +08:00
{
2018-11-28 15:39:08 +08:00
public int ID { get ; set ; }
2018-04-13 17:19:50 +08:00
public ScoreRank Rank { get ; set ; }
2019-02-26 12:10:07 +08:00
public long TotalScore { get ; set ; }
2018-04-13 17:19:50 +08:00
2021-03-25 16:50:21 +08:00
[Column(TypeName = "DECIMAL(1,4)")] // TODO: This data type is wrong (should contain more precision). But at the same time, we probably don't need to be storing this in the database.
2018-04-13 17:19:50 +08:00
public double Accuracy { get ; set ; }
2021-07-24 04:37:08 +08:00
public LocalisableString DisplayAccuracy = > Accuracy . FormatAccuracy ( ) ;
2020-02-03 23:09:07 +08:00
2018-04-13 17:19:50 +08:00
public double? PP { get ; set ; }
public int MaxCombo { get ; set ; }
2018-12-14 20:09:17 +08:00
public int Combo { get ; set ; } // Todo: Shouldn't exist in here
2018-04-13 17:19:50 +08:00
2018-11-28 16:26:46 +08:00
public int RulesetID { get ; set ; }
2018-12-14 20:09:17 +08:00
[NotMapped]
public bool Passed { get ; set ; } = true ;
2021-10-28 16:57:17 +08:00
public RulesetInfo Ruleset { get ; set ; }
2018-04-13 17:19:50 +08:00
2021-04-12 18:50:24 +08:00
private APIMod [ ] localAPIMods ;
2021-10-28 17:23:52 +08:00
2018-11-30 15:11:09 +08:00
private Mod [ ] mods ;
[NotMapped]
2018-11-28 18:47:20 +08:00
public Mod [ ] Mods
2018-11-28 15:39:08 +08:00
{
2018-11-28 18:47:20 +08:00
get
2018-11-28 15:39:08 +08:00
{
2021-04-22 17:44:14 +08:00
var rulesetInstance = Ruleset ? . CreateInstance ( ) ;
if ( rulesetInstance = = null )
return mods ? ? Array . Empty < Mod > ( ) ;
2021-04-21 14:16:28 +08:00
2021-04-22 17:44:14 +08:00
Mod [ ] scoreMods = Array . Empty < Mod > ( ) ;
2021-04-21 14:16:28 +08:00
2018-11-30 17:52:31 +08:00
if ( mods ! = null )
2021-04-21 14:16:28 +08:00
scoreMods = mods ;
else if ( localAPIMods ! = null )
2021-10-29 12:02:19 +08:00
scoreMods = APIMods . Select ( m = > m . ToMod ( rulesetInstance ) ) . ToArray ( ) ;
2018-11-30 15:11:09 +08:00
2021-04-21 14:16:28 +08:00
return scoreMods ;
2018-11-30 15:11:09 +08:00
}
set
{
2021-04-12 18:50:24 +08:00
localAPIMods = null ;
2018-11-30 15:11:09 +08:00
mods = value ;
2018-11-28 15:39:08 +08:00
}
}
2021-04-12 18:50:24 +08:00
// Used for API serialisation/deserialisation.
2021-04-12 19:49:44 +08:00
[NotMapped]
2021-10-29 12:02:19 +08:00
public APIMod [ ] APIMods
2018-11-30 15:11:09 +08:00
{
2021-03-21 18:01:06 +08:00
get
{
2021-04-12 18:50:24 +08:00
if ( localAPIMods ! = null )
return localAPIMods ;
2021-03-21 18:01:06 +08:00
if ( mods = = null )
2021-04-12 18:50:24 +08:00
return Array . Empty < APIMod > ( ) ;
2021-03-21 18:01:06 +08:00
2021-04-12 18:50:24 +08:00
return localAPIMods = mods . Select ( m = > new APIMod ( m ) ) . ToArray ( ) ;
2021-03-21 18:01:06 +08:00
}
2018-11-30 15:11:09 +08:00
set
{
2021-04-12 18:50:24 +08:00
localAPIMods = value ;
2018-11-30 15:11:09 +08:00
2021-04-12 18:50:24 +08:00
// We potentially can't update this yet due to Ruleset being late-bound, so instead update on read as necessary.
2018-11-30 15:11:09 +08:00
mods = null ;
}
}
2018-11-28 18:47:20 +08:00
2021-04-12 18:50:24 +08:00
// Used for database serialisation/deserialisation.
[Column("Mods")]
2021-04-12 19:50:18 +08:00
public string ModsJson
2021-04-12 18:50:24 +08:00
{
2021-10-29 12:02:19 +08:00
get = > JsonConvert . SerializeObject ( APIMods ) ;
set = > APIMods = JsonConvert . DeserializeObject < APIMod [ ] > ( value ) ;
2021-04-12 18:50:24 +08:00
}
2018-12-22 14:17:35 +08:00
[NotMapped]
2021-11-04 17:02:44 +08:00
public APIUser User { get ; set ; }
2018-04-13 17:19:50 +08:00
2018-11-30 15:11:09 +08:00
[Column("User")]
2018-11-28 15:39:08 +08:00
public string UserString
{
get = > User ? . Username ;
2019-02-25 12:58:19 +08:00
set
{
2021-11-04 17:02:44 +08:00
User ? ? = new APIUser ( ) ;
2019-02-25 14:25:22 +08:00
User . Username = value ;
2019-02-25 12:58:19 +08:00
}
2019-02-23 16:04:02 +08:00
}
[Column("UserID")]
2020-11-06 11:40:54 +08:00
public int? UserID
2019-02-23 16:04:02 +08:00
{
2019-02-25 13:40:44 +08:00
get = > User ? . Id ? ? 1 ;
2019-02-25 12:58:19 +08:00
set
{
2021-11-04 17:02:44 +08:00
User ? ? = new APIUser ( ) ;
2019-02-25 14:25:22 +08:00
User . Id = value ? ? 1 ;
2019-02-25 12:58:19 +08:00
}
2018-11-28 15:39:08 +08:00
}
2018-11-28 16:26:39 +08:00
public int BeatmapInfoID { get ; set ; }
2021-10-04 16:35:53 +08:00
[Column("Beatmap")]
2021-10-29 12:02:19 +08:00
public BeatmapInfo BeatmapInfo { get ; set ; }
2018-04-13 17:19:50 +08:00
2018-11-28 17:52:57 +08:00
public long? OnlineScoreID { get ; set ; }
2018-04-13 17:19:50 +08:00
2018-11-28 19:16:20 +08:00
public DateTimeOffset Date { get ; set ; }
2018-04-13 17:19:50 +08:00
2021-10-29 12:02:19 +08:00
[NotMapped]
2021-10-28 17:23:52 +08:00
public Dictionary < HitResult , int > Statistics { get ; set ; } = new Dictionary < HitResult , int > ( ) ;
2018-11-28 15:39:08 +08:00
2018-11-30 15:11:09 +08:00
[Column("Statistics")]
public string StatisticsJson
2018-11-30 13:27:08 +08:00
{
get = > JsonConvert . SerializeObject ( Statistics ) ;
set
{
if ( value = = null )
2018-11-30 15:11:09 +08:00
{
Statistics . Clear ( ) ;
2018-11-30 13:27:08 +08:00
return ;
2018-11-30 15:11:09 +08:00
}
2018-11-30 13:27:08 +08:00
2018-12-05 18:44:01 +08:00
Statistics = JsonConvert . DeserializeObject < Dictionary < HitResult , int > > ( value ) ;
2018-11-30 13:27:08 +08:00
}
}
2020-06-18 21:11:03 +08:00
[NotMapped]
2020-06-19 18:58:35 +08:00
public List < HitEvent > HitEvents { get ; set ; }
2020-06-15 21:44:55 +08:00
2018-11-28 15:39:08 +08:00
public List < ScoreFileInfo > Files { get ; set ; }
2018-11-30 16:36:06 +08:00
public string Hash { get ; set ; }
2018-11-28 15:39:08 +08:00
public bool DeletePending { get ; set ; }
2018-11-30 15:11:09 +08:00
2020-07-28 21:08:10 +08:00
/// <summary>
/// The position of this score, starting at 1.
/// </summary>
[NotMapped]
2021-10-29 12:02:19 +08:00
public int? Position { get ; set ; } // TODO: remove after all calls to `CreateScoreInfo` are gone.
2020-07-28 21:08:10 +08:00
2020-09-29 17:55:06 +08:00
/// <summary>
/// Whether this <see cref="ScoreInfo"/> represents a legacy (osu!stable) score.
/// </summary>
[NotMapped]
2021-06-08 17:23:03 +08:00
public bool IsLegacyScore = > Mods . OfType < ModClassic > ( ) . Any ( ) ;
2020-09-29 17:55:06 +08:00
2020-10-07 14:34:03 +08:00
public IEnumerable < HitResultDisplayStatistic > GetStatisticsForDisplay ( )
2020-09-25 19:22:59 +08:00
{
2020-10-07 14:34:03 +08:00
foreach ( var r in Ruleset . CreateInstance ( ) . GetHitResults ( ) )
2020-09-25 19:22:59 +08:00
{
2021-07-19 03:52:16 +08:00
int value = Statistics . GetValueOrDefault ( r . result ) ;
2020-09-25 19:22:59 +08:00
2020-10-07 14:34:03 +08:00
switch ( r . result )
2020-09-25 19:22:59 +08:00
{
case HitResult . SmallTickHit :
{
2021-07-19 03:52:16 +08:00
int total = value + Statistics . GetValueOrDefault ( HitResult . SmallTickMiss ) ;
2020-09-25 19:22:59 +08:00
if ( total > 0 )
2020-10-07 14:34:03 +08:00
yield return new HitResultDisplayStatistic ( r . result , value , total , r . displayName ) ;
2020-09-25 19:22:59 +08:00
break ;
}
case HitResult . LargeTickHit :
{
2021-07-19 03:52:16 +08:00
int total = value + Statistics . GetValueOrDefault ( HitResult . LargeTickMiss ) ;
2020-09-25 19:22:59 +08:00
if ( total > 0 )
2020-10-07 14:34:03 +08:00
yield return new HitResultDisplayStatistic ( r . result , value , total , r . displayName ) ;
2020-09-25 19:22:59 +08:00
break ;
}
case HitResult . SmallTickMiss :
case HitResult . LargeTickMiss :
break ;
default :
2020-10-07 14:34:03 +08:00
yield return new HitResultDisplayStatistic ( r . result , value , null , r . displayName ) ;
2020-09-25 19:22:59 +08:00
break ;
}
}
}
2021-07-19 12:02:40 +08:00
public ScoreInfo DeepClone ( )
{
var clone = ( ScoreInfo ) MemberwiseClone ( ) ;
clone . Statistics = new Dictionary < HitResult , int > ( clone . Statistics ) ;
return clone ;
}
2021-10-04 16:35:53 +08:00
public override string ToString ( ) = > $"{User} playing {BeatmapInfo}" ;
2019-06-12 04:01:57 +08:00
2019-12-03 12:33:42 +08:00
public bool Equals ( ScoreInfo other )
{
if ( other = = null )
return false ;
if ( ID ! = 0 & & other . ID ! = 0 )
return ID = = other . ID ;
if ( OnlineScoreID . HasValue & & other . OnlineScoreID . HasValue )
return OnlineScoreID = = other . OnlineScoreID ;
if ( ! string . IsNullOrEmpty ( Hash ) & & ! string . IsNullOrEmpty ( other . Hash ) )
return Hash = = other . Hash ;
return ReferenceEquals ( this , other ) ;
}
2021-10-29 10:48:36 +08:00
2021-10-30 20:15:20 +08:00
#region Implementation of IHasOnlineID
2021-10-29 10:48:36 +08:00
public long OnlineID = > OnlineScoreID ? ? - 1 ;
2021-10-28 17:23:52 +08:00
2021-10-30 20:15:20 +08:00
#endregion
2021-10-30 21:07:34 +08:00
2021-10-30 21:08:45 +08:00
#region Implementation of IScoreInfo
2021-10-28 17:23:52 +08:00
IBeatmapInfo IScoreInfo . Beatmap = > BeatmapInfo ;
IRulesetInfo IScoreInfo . Ruleset = > Ruleset ;
bool IScoreInfo . HasReplay = > Files . Any ( ) ;
2021-10-30 21:08:45 +08:00
#endregion
2018-04-13 17:19:50 +08:00
}
}