1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-06 14:27:31 +08:00
osu-lazer/osu.Game/Online/DownloadTrackingComposite.cs

197 lines
7.1 KiB
C#
Raw Normal View History

2019-01-31 18:17:42 +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.
using System;
using JetBrains.Annotations;
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
using osu.Framework.Graphics.Containers;
using osu.Game.Database;
using osu.Game.Online.API;
namespace osu.Game.Online
{
/// <summary>
2019-11-17 20:48:23 +08:00
/// A component which tracks a <typeparamref name="TModel"/> through potential download/import/deletion.
/// </summary>
public abstract class DownloadTrackingComposite<TModel, TModelManager> : CompositeDrawable
2019-06-12 01:53:40 +08:00
where TModel : class, IEquatable<TModel>
2019-06-12 03:11:17 +08:00
where TModelManager : class, IModelDownloader<TModel>
{
2019-06-26 23:29:09 +08:00
protected readonly Bindable<TModel> Model = new Bindable<TModel>();
2020-02-14 21:59:51 +08:00
[Resolved(CanBeNull = true)]
protected TModelManager Manager { get; private set; }
/// <summary>
2019-11-17 20:48:23 +08:00
/// Holds the current download state of the <typeparamref name="TModel"/>, whether is has already been downloaded, is in progress, or is not downloaded.
/// </summary>
protected readonly Bindable<DownloadState> State = new Bindable<DownloadState>();
2019-12-10 17:08:11 +08:00
protected readonly BindableNumber<double> Progress = new BindableNumber<double> { MinValue = 0, MaxValue = 1 };
protected DownloadTrackingComposite(TModel model = null)
{
2019-06-26 23:29:09 +08:00
Model.Value = model;
}
private IBindable<WeakReference<TModel>> managedUpdated;
2020-05-19 15:44:22 +08:00
private IBindable<WeakReference<TModel>> managerRemoved;
private IBindable<WeakReference<ArchiveDownloadRequest<TModel>>> managerDownloadBegan;
private IBindable<WeakReference<ArchiveDownloadRequest<TModel>>> managerDownloadFailed;
[BackgroundDependencyLoader(true)]
2020-02-14 21:14:00 +08:00
private void load()
{
2019-06-26 23:29:09 +08:00
Model.BindValueChanged(modelInfo =>
{
if (modelInfo.NewValue == null)
attachDownload(null);
else if (IsModelAvailableLocally())
State.Value = DownloadState.LocallyAvailable;
else
attachDownload(Manager?.GetExistingDownload(modelInfo.NewValue));
}, true);
if (Manager == null)
return;
managerDownloadBegan = Manager.DownloadBegan.GetBoundCopy();
2020-05-19 15:44:22 +08:00
managerDownloadBegan.BindValueChanged(downloadBegan);
managerDownloadFailed = Manager.DownloadFailed.GetBoundCopy();
2020-05-19 15:44:22 +08:00
managerDownloadFailed.BindValueChanged(downloadFailed);
managedUpdated = Manager.ItemUpdated.GetBoundCopy();
managedUpdated.BindValueChanged(itemUpdated);
managerRemoved = Manager.ItemRemoved.GetBoundCopy();
2020-05-19 15:44:22 +08:00
managerRemoved.BindValueChanged(itemRemoved);
}
/// <summary>
2021-02-04 07:20:18 +08:00
/// Checks that a database model matches the one expected to be downloaded.
/// </summary>
/// <example>
/// For online play, this could be used to check that the databased model matches the online beatmap.
/// </example>
/// <param name="databasedModel">The model in database.</param>
protected virtual bool VerifyDatabasedModel([NotNull] TModel databasedModel) => true;
/// <summary>
/// Whether the given model is available in the database.
/// By default, this calls <see cref="IModelDownloader{TModel}.IsAvailableLocally"/>,
/// but can be overriden to add additional checks for verifying the model in database.
/// </summary>
protected virtual bool IsModelAvailableLocally() => Manager?.IsAvailableLocally(Model.Value) == true;
2020-05-19 15:44:22 +08:00
private void downloadBegan(ValueChangedEvent<WeakReference<ArchiveDownloadRequest<TModel>>> weakRequest)
2019-08-05 16:58:16 +08:00
{
2020-05-19 15:44:22 +08:00
if (weakRequest.NewValue.TryGetTarget(out var request))
{
Schedule(() =>
{
if (request.Model.Equals(Model.Value))
attachDownload(request);
});
}
}
2019-08-05 16:58:16 +08:00
2020-05-19 15:44:22 +08:00
private void downloadFailed(ValueChangedEvent<WeakReference<ArchiveDownloadRequest<TModel>>> weakRequest)
2019-08-05 16:58:16 +08:00
{
2020-05-19 15:44:22 +08:00
if (weakRequest.NewValue.TryGetTarget(out var request))
{
Schedule(() =>
{
if (request.Model.Equals(Model.Value))
attachDownload(null);
});
}
}
2019-08-05 16:58:16 +08:00
2019-06-26 20:52:37 +08:00
private ArchiveDownloadRequest<TModel> attachedRequest;
2019-06-26 20:52:37 +08:00
private void attachDownload(ArchiveDownloadRequest<TModel> request)
{
if (attachedRequest != null)
{
attachedRequest.Failure -= onRequestFailure;
attachedRequest.DownloadProgressed -= onRequestProgress;
attachedRequest.Success -= onRequestSuccess;
}
attachedRequest = request;
if (attachedRequest != null)
{
2019-01-31 18:08:45 +08:00
if (attachedRequest.Progress == 1)
{
Progress.Value = 1;
State.Value = DownloadState.Importing;
2019-01-31 18:08:45 +08:00
}
else
{
Progress.Value = attachedRequest.Progress;
State.Value = DownloadState.Downloading;
2019-01-31 18:08:45 +08:00
attachedRequest.Failure += onRequestFailure;
attachedRequest.DownloadProgressed += onRequestProgress;
attachedRequest.Success += onRequestSuccess;
}
}
else
{
State.Value = DownloadState.NotDownloaded;
}
}
private void onRequestSuccess(string _) => Schedule(() => State.Value = DownloadState.Importing);
private void onRequestProgress(float progress) => Schedule(() => Progress.Value = progress);
private void onRequestFailure(Exception e) => Schedule(() => attachDownload(null));
private void itemUpdated(ValueChangedEvent<WeakReference<TModel>> weakItem)
2020-05-19 15:44:22 +08:00
{
if (weakItem.NewValue.TryGetTarget(out var item))
{
Schedule(() =>
{
if (!item.Equals(Model.Value))
return;
if (!VerifyDatabasedModel(item))
{
State.Value = DownloadState.NotDownloaded;
return;
}
State.Value = DownloadState.LocallyAvailable;
});
}
2020-05-19 15:44:22 +08:00
}
2020-05-19 15:44:22 +08:00
private void itemRemoved(ValueChangedEvent<WeakReference<TModel>> weakItem)
{
if (weakItem.NewValue.TryGetTarget(out var item))
{
Schedule(() =>
{
if (item.Equals(Model.Value))
State.Value = DownloadState.NotDownloaded;
});
}
2020-05-19 15:44:22 +08:00
}
2019-06-26 23:29:38 +08:00
#region Disposal
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
State.UnbindAll();
attachDownload(null);
}
#endregion
}
}