// 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 osu.Framework.Bindables; using osu.Framework.Graphics; namespace osu.Game.Screens.OnlinePlay { /// /// Utility class to track ongoing online operations' progress. /// Can be used to disable interactivity while waiting for a response from online sources. /// public partial class OngoingOperationTracker : Component { /// /// Whether there is an online operation in progress. /// public IBindable InProgress => inProgress; private readonly Bindable inProgress = new BindableBool(); private LeasedBindable? leasedInProgress; public OngoingOperationTracker() { AlwaysPresent = true; } /// /// Begins tracking a new online operation. /// /// /// An that will automatically mark the operation as ended on disposal. /// /// An operation has already been started. public IDisposable BeginOperation() { if (leasedInProgress != null) throw new InvalidOperationException("Cannot begin operation while another is in progress."); leasedInProgress = inProgress.BeginLease(true); leasedInProgress.Value = true; return new OngoingOperation(this, leasedInProgress); } private void endOperationWithKnownLease(LeasedBindable lease) { // for extra safety, marshal the end of operation back to the update thread if necessary. Scheduler.Add(() => { if (lease != leasedInProgress) return; // UnbindAll() is purposefully used instead of Return() - the two do roughly the same thing, with one difference: // the former won't throw if the lease has already been returned before. // this matters because framework can unbind the lease via the internal UnbindAllBindables(), which is not always detectable // (it is in the case of disposal, but not in the case of screen exit - at least not cleanly). leasedInProgress?.UnbindAll(); leasedInProgress = null; }, false); } private class OngoingOperation : IDisposable { private readonly OngoingOperationTracker tracker; private readonly LeasedBindable lease; public OngoingOperation(OngoingOperationTracker tracker, LeasedBindable lease) { this.tracker = tracker; this.lease = lease; } public void Dispose() { tracker.endOperationWithKnownLease(lease); } } } }