// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. #nullable disable using System; using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; using osu.Framework; using osu.Framework.Allocation; using osu.Framework.Extensions.EnumExtensions; using osu.Framework.Graphics; using osu.Framework.Platform; using osu.Game.Beatmaps; using osu.Game.Collections; using osu.Game.IO; using osu.Game.Overlays; using osu.Game.Overlays.Settings.Sections.Maintenance; using osu.Game.Scoring; using osu.Game.Skinning; namespace osu.Game.Database { /// /// Handles migration of legacy user data from osu-stable. /// public class LegacyImportManager : Component { [Resolved] private SkinManager skins { get; set; } [Resolved] private BeatmapManager beatmaps { get; set; } [Resolved] private ScoreManager scores { get; set; } [Resolved] private CollectionManager collections { get; set; } [Resolved(canBeNull: true)] private OsuGame game { get; set; } [Resolved] private IDialogOverlay dialogOverlay { get; set; } [Resolved(canBeNull: true)] private DesktopGameHost desktopGameHost { get; set; } private StableStorage cachedStorage; public bool SupportsImportFromStable => RuntimeInfo.IsDesktop; public void UpdateStorage(string stablePath) => cachedStorage = new StableStorage(stablePath, desktopGameHost); public virtual async Task GetImportCount(StableContent content, CancellationToken cancellationToken) { var stableStorage = GetCurrentStableStorage(); if (stableStorage == null) return 0; cancellationToken.ThrowIfCancellationRequested(); switch (content) { case StableContent.Beatmaps: return await new LegacyBeatmapImporter(beatmaps).GetAvailableCount(stableStorage); case StableContent.Skins: return await new LegacySkinImporter(skins).GetAvailableCount(stableStorage); case StableContent.Collections: return await collections.GetAvailableCount(stableStorage); case StableContent.Scores: return await new LegacyScoreImporter(scores).GetAvailableCount(stableStorage); default: throw new ArgumentException($"Only one {nameof(StableContent)} flag should be specified."); } } public async Task ImportFromStableAsync(StableContent content, bool interactiveLocateIfNotFound = true) { var stableStorage = GetCurrentStableStorage(); if (stableStorage == null) { if (!interactiveLocateIfNotFound) return; var taskCompletionSource = new TaskCompletionSource(TaskCreationOptions.RunContinuationsAsynchronously); Schedule(() => dialogOverlay.Push(new StableDirectoryLocationDialog(taskCompletionSource))); string stablePath = await taskCompletionSource.Task.ConfigureAwait(false); UpdateStorage(stablePath); stableStorage = GetCurrentStableStorage(); } var importTasks = new List(); Task beatmapImportTask = Task.CompletedTask; if (content.HasFlagFast(StableContent.Beatmaps)) importTasks.Add(beatmapImportTask = new LegacyBeatmapImporter(beatmaps).ImportFromStableAsync(stableStorage)); if (content.HasFlagFast(StableContent.Skins)) importTasks.Add(new LegacySkinImporter(skins).ImportFromStableAsync(stableStorage)); if (content.HasFlagFast(StableContent.Collections)) importTasks.Add(beatmapImportTask.ContinueWith(_ => collections.ImportFromStableAsync(stableStorage), TaskContinuationOptions.OnlyOnRanToCompletion)); if (content.HasFlagFast(StableContent.Scores)) importTasks.Add(beatmapImportTask.ContinueWith(_ => new LegacyScoreImporter(scores).ImportFromStableAsync(stableStorage), TaskContinuationOptions.OnlyOnRanToCompletion)); await Task.WhenAll(importTasks.ToArray()).ConfigureAwait(false); } public StableStorage GetCurrentStableStorage() { if (cachedStorage != null) return cachedStorage; var stableStorage = game?.GetStorageForStableInstall(); if (stableStorage != null) return cachedStorage = stableStorage; return null; } } [Flags] public enum StableContent { Beatmaps = 1 << 0, Scores = 1 << 1, Skins = 1 << 2, Collections = 1 << 3, All = Beatmaps | Scores | Skins | Collections } }