1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-18 08:12:55 +08:00
osu-lazer/osu.Game/Screens/OnlinePlay/Multiplayer/Spectate/PlayerArea.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

163 lines
5.9 KiB
C#
Raw Normal View History

// 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.
2021-04-16 12:28:32 +08:00
using System;
using System.Collections.Generic;
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Beatmaps;
2021-04-16 12:28:32 +08:00
using osu.Game.Graphics.UserInterface;
using osu.Game.Rulesets;
using osu.Game.Rulesets.Mods;
using osu.Game.Scoring;
using osu.Game.Screens.Play;
namespace osu.Game.Screens.OnlinePlay.Multiplayer.Spectate
{
2021-04-22 22:52:22 +08:00
/// <summary>
/// Provides an area for and manages the hierarchy of a spectated player within a <see cref="MultiSpectatorScreen"/>.
/// </summary>
2021-04-26 18:01:30 +08:00
public class PlayerArea : CompositeDrawable
{
2021-08-11 17:16:25 +08:00
/// <summary>
/// Raised after <see cref="Player.StartGameplay"/> is called on <see cref="Player"/>.
/// </summary>
2022-08-22 20:52:43 +08:00
public event Action? OnGameplayStarted;
2021-08-11 17:16:25 +08:00
2021-04-22 22:52:22 +08:00
/// <summary>
/// Whether a <see cref="Player"/> is loaded in the area.
/// </summary>
2021-06-11 17:14:37 +08:00
public bool PlayerLoaded => (stack?.CurrentScreen as Player)?.IsLoaded == true;
2021-04-22 22:52:22 +08:00
/// <summary>
/// The user id this <see cref="PlayerArea"/> corresponds to.
/// </summary>
public readonly int UserId;
2021-04-22 22:52:22 +08:00
/// <summary>
/// The <see cref="Spectate.SpectatorPlayerClock"/> used to control the gameplay running state of a loaded <see cref="Player"/>.
2021-04-22 22:52:22 +08:00
/// </summary>
public readonly SpectatorPlayerClock SpectatorPlayerClock;
2021-04-22 22:52:22 +08:00
/// <summary>
/// The clock adjustments applied by the <see cref="Player"/> loaded in this area.
/// </summary>
public IAggregateAudioAdjustment ClockAdjustmentsFromMods => clockAdjustmentsFromMods;
2021-04-22 22:52:22 +08:00
/// <summary>
/// The currently-loaded score.
/// </summary>
2022-08-22 20:52:43 +08:00
public Score? Score { get; private set; }
[Resolved]
private IBindable<WorkingBeatmap> beatmap { get; set; } = null!;
private readonly AudioAdjustments clockAdjustmentsFromMods = new AudioAdjustments();
2021-04-26 18:01:30 +08:00
private readonly BindableDouble volumeAdjustment = new BindableDouble();
private readonly Container gameplayContent;
2021-04-16 12:28:32 +08:00
private readonly LoadingLayer loadingLayer;
2022-08-22 20:52:43 +08:00
private OsuScreenStack? stack;
2022-08-24 14:07:04 +08:00
public PlayerArea(int userId, SpectatorPlayerClock clock)
{
UserId = userId;
SpectatorPlayerClock = clock;
2021-04-08 21:13:54 +08:00
RelativeSizeAxes = Axes.Both;
Masking = true;
2021-04-16 12:28:32 +08:00
2021-04-26 18:01:30 +08:00
AudioContainer audioContainer;
2021-04-16 12:28:32 +08:00
InternalChildren = new Drawable[]
{
audioContainer = new AudioContainer
{
RelativeSizeAxes = Axes.Both,
Child = gameplayContent = new DrawSizePreservingFillContainer { RelativeSizeAxes = Axes.Both },
},
2021-04-16 12:28:32 +08:00
loadingLayer = new LoadingLayer(true) { State = { Value = Visibility.Visible } }
};
2021-04-26 18:01:30 +08:00
audioContainer.AddAdjustment(AdjustableProperty.Volume, volumeAdjustment);
}
2022-08-22 20:52:43 +08:00
public void LoadScore(Score score)
{
2021-04-16 12:28:32 +08:00
if (Score != null)
2021-04-22 22:52:22 +08:00
throw new InvalidOperationException($"Cannot load a new score on a {nameof(PlayerArea)} that has an existing score.");
2021-04-16 12:28:32 +08:00
Score = score;
gameplayContent.Child = new PlayerIsolationContainer(beatmap.Value, Score.ScoreInfo.Ruleset, Score.ScoreInfo.Mods)
{
RelativeSizeAxes = Axes.Both,
Child = stack = new OsuScreenStack
{
Name = nameof(PlayerArea),
}
};
2021-08-11 17:16:25 +08:00
stack.Push(new MultiSpectatorPlayerLoader(Score, () =>
{
var player = new MultiSpectatorPlayer(Score, SpectatorPlayerClock);
2022-09-08 16:41:23 +08:00
player.OnGameplayStarted += () => OnGameplayStarted?.Invoke();
2022-09-08 16:41:23 +08:00
clockAdjustmentsFromMods.BindAdjustments(player.ClockAdjustmentsFromMods);
2021-08-11 17:16:25 +08:00
return player;
}));
2021-04-16 12:28:32 +08:00
loadingLayer.Hide();
}
2021-04-13 19:52:20 +08:00
2021-04-26 18:01:30 +08:00
private bool mute = true;
2021-04-26 18:01:30 +08:00
public bool Mute
{
2021-04-26 18:01:30 +08:00
get => mute;
set
{
mute = value;
volumeAdjustment.Value = value ? 0 : 1;
}
}
2021-04-26 18:01:30 +08:00
// Player interferes with global input, so disable input for now.
public override bool PropagatePositionalInputSubTree => false;
public override bool PropagateNonPositionalInputSubTree => false;
/// <summary>
/// Isolates each player instance from the game-wide ruleset/beatmap/mods (to allow for different players having different settings).
/// </summary>
private class PlayerIsolationContainer : Container
{
[Cached]
private readonly Bindable<RulesetInfo> ruleset = new Bindable<RulesetInfo>();
[Cached]
private readonly Bindable<WorkingBeatmap> beatmap = new Bindable<WorkingBeatmap>();
[Cached]
private readonly Bindable<IReadOnlyList<Mod>> mods = new Bindable<IReadOnlyList<Mod>>();
public PlayerIsolationContainer(WorkingBeatmap beatmap, RulesetInfo ruleset, IReadOnlyList<Mod> mods)
{
this.beatmap.Value = beatmap;
this.ruleset.Value = ruleset;
this.mods.Value = mods;
}
protected override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent)
{
var dependencies = new DependencyContainer(base.CreateChildDependencies(parent));
dependencies.CacheAs(ruleset.BeginLease(false));
dependencies.CacheAs(beatmap.BeginLease(false));
dependencies.CacheAs(mods.BeginLease(false));
return dependencies;
}
}
}
}