// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Diagnostics; using System.Linq; using JetBrains.Annotations; using Newtonsoft.Json; using osu.Game.Collections; using osu.Game.Database; using osu.Game.Models; using osu.Game.Online.API.Requests.Responses; using osu.Game.Overlays.BeatmapSet.Scores; using osu.Game.Rulesets; using osu.Game.Scoring; using Realms; namespace osu.Game.Beatmaps { /// /// A realm model containing metadata for a single beatmap difficulty. /// This should generally include anything which is required to be filtered on at song select, or anything pertaining to storage of beatmaps in the client. /// /// /// There are some legacy fields in this model which are not persisted to realm. These are isolated in a code region within the class and should eventually be migrated to `Beatmap`. /// [Serializable] [MapTo("Beatmap")] public class BeatmapInfo : RealmObject, IHasGuidPrimaryKey, IBeatmapInfo, IEquatable { [PrimaryKey] public Guid ID { get; set; } public string DifficultyName { get; set; } = string.Empty; public RulesetInfo Ruleset { get; set; } = null!; public BeatmapDifficulty Difficulty { get; set; } = null!; public BeatmapMetadata Metadata { get; set; } = null!; [JsonIgnore] [Backlink(nameof(ScoreInfo.BeatmapInfo))] public IQueryable Scores { get; } = null!; public BeatmapUserSettings UserSettings { get; set; } = null!; public BeatmapInfo(RulesetInfo? ruleset = null, BeatmapDifficulty? difficulty = null, BeatmapMetadata? metadata = null) { ID = Guid.NewGuid(); Ruleset = ruleset ?? new RulesetInfo { OnlineID = 0, ShortName = @"osu", Name = @"null placeholder ruleset" }; Difficulty = difficulty ?? new BeatmapDifficulty(); Metadata = metadata ?? new BeatmapMetadata(); UserSettings = new BeatmapUserSettings(); } [UsedImplicitly] private BeatmapInfo() { } public BeatmapSetInfo? BeatmapSet { get; set; } [Ignored] public RealmNamedFileUsage? File => BeatmapSet?.Files.FirstOrDefault(f => f.File.Hash == Hash); [Ignored] public BeatmapOnlineStatus Status { get => (BeatmapOnlineStatus)StatusInt; set => StatusInt = (int)value; } [MapTo(nameof(Status))] public int StatusInt { get; set; } = (int)BeatmapOnlineStatus.None; [Indexed] public int OnlineID { get; set; } = -1; public double Length { get; set; } public double BPM { get; set; } public string Hash { get; set; } = string.Empty; /// /// Defaults to -1 (meaning not-yet-calculated). /// Will likely be superseded with a better storage considering ruleset/mods. /// public double StarRating { get; set; } = -1; [Indexed] public string MD5Hash { get; set; } = string.Empty; public string OnlineMD5Hash { get; set; } = string.Empty; /// /// The last time of a local modification (via the editor). /// public DateTimeOffset? LastLocalUpdate { get; set; } /// /// The last time online metadata was applied to this beatmap. /// public DateTimeOffset? LastOnlineUpdate { get; set; } /// /// Whether this beatmap matches the online version, based on fetched online metadata. /// Will return true if no online metadata is available. /// public bool MatchesOnlineVersion => LastOnlineUpdate == null || MD5Hash == OnlineMD5Hash; [JsonIgnore] public bool Hidden { get; set; } public int EndTimeObjectCount { get; set; } = -1; public int TotalObjectCount { get; set; } = -1; /// /// Reset any fetched online linking information (and history). /// public void ResetOnlineInfo() { OnlineID = -1; LastOnlineUpdate = null; OnlineMD5Hash = string.Empty; if (Status != BeatmapOnlineStatus.LocallyModified) Status = BeatmapOnlineStatus.None; } #region Properties we may not want persisted (but also maybe no harm?) /// /// The time at which this beatmap was last played by the local user. /// public DateTimeOffset? LastPlayed { get; set; } public int BeatDivisor { get; set; } = 4; /// /// The time in milliseconds when last exiting the editor with this beatmap loaded. /// public double? EditorTimestamp { get; set; } [Ignored] public CountdownType Countdown { get; set; } = CountdownType.Normal; /// /// The number of beats to move the countdown backwards (compared to its default location). /// public int CountdownOffset { get; set; } #endregion public bool Equals(BeatmapInfo? other) { if (ReferenceEquals(this, other)) return true; if (other == null) return false; return ID == other.ID; } public bool Equals(IBeatmapInfo? other) => other is BeatmapInfo b && Equals(b); public bool AudioEquals(BeatmapInfo? other) => other != null && BeatmapSet != null && other.BeatmapSet != null && compareFiles(this, other, m => m.AudioFile); public bool BackgroundEquals(BeatmapInfo? other) => other != null && BeatmapSet != null && other.BeatmapSet != null && compareFiles(this, other, m => m.BackgroundFile); private static bool compareFiles(BeatmapInfo x, BeatmapInfo y, Func getFilename) { Debug.Assert(x.BeatmapSet != null); Debug.Assert(y.BeatmapSet != null); string? fileHashX = x.BeatmapSet.GetFile(getFilename(x.Metadata))?.File.Hash; string? fileHashY = y.BeatmapSet.GetFile(getFilename(y.Metadata))?.File.Hash; return fileHashX == fileHashY; } /// /// When updating a beatmap, its hashes will change. Collections currently track beatmaps by hash, so they need to be updated. /// This method will handle updating /// /// A realm instance in an active write transaction. /// The previous MD5 hash of the beatmap before update. public void TransferCollectionReferences(Realm realm, string previousMD5Hash) { var collections = realm.All().AsEnumerable().Where(c => c.BeatmapMD5Hashes.Contains(previousMD5Hash)); foreach (var c in collections) { c.BeatmapMD5Hashes.Remove(previousMD5Hash); c.BeatmapMD5Hashes.Add(MD5Hash); } } /// /// Local scores are retained separate from a beatmap's lifetime, matched via . /// Therefore we need to detach / reattach scores when a beatmap is edited or imported. /// /// A realm instance in an active write transaction. public void UpdateLocalScores(Realm realm) { // first disassociate any scores which are already attached and no longer valid. foreach (var score in Scores) score.BeatmapInfo = null; // then attach any scores which match the new hash. foreach (var score in realm.All().Where(s => s.BeatmapHash == Hash)) score.BeatmapInfo = this; } IBeatmapMetadataInfo IBeatmapInfo.Metadata => Metadata; IBeatmapSetInfo? IBeatmapInfo.BeatmapSet => BeatmapSet; IRulesetInfo IBeatmapInfo.Ruleset => Ruleset; IBeatmapDifficultyInfo IBeatmapInfo.Difficulty => Difficulty; #region Compatibility properties [Ignored] public string? Path => File?.Filename; [Ignored] public APIBeatmap? OnlineInfo { get; set; } /// /// The maximum achievable combo on this beatmap, populated for online info purposes only. /// Todo: This should never be used nor exist, but is still relied on in since can't be used yet. For now this is obsoleted until it is removed. /// [Ignored] [Obsolete("Use ScoreManager.GetMaximumAchievableComboAsync instead.")] public int? MaxCombo { get; set; } [Ignored] public int[] Bookmarks { get; set; } = Array.Empty(); public int BeatmapVersion; public BeatmapInfo Clone() => (BeatmapInfo)this.Detach().MemberwiseClone(); public override string ToString() => this.GetDisplayTitle(); #endregion } }