1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-25 23:52:54 +08:00
osu-lazer/osu.Game/Online/Rooms/Room.cs

380 lines
12 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.
2018-04-13 17:19:50 +08:00
2018-12-17 13:45:06 +08:00
using System;
using System.Collections.Generic;
using System.ComponentModel;
2018-12-03 19:50:40 +08:00
using System.Linq;
using System.Runtime.CompilerServices;
using Newtonsoft.Json;
2019-02-05 18:00:01 +08:00
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
2020-12-21 15:56:45 +08:00
using osu.Game.IO.Serialization.Converters;
using osu.Game.Online.API.Requests.Responses;
2021-11-19 13:46:53 +08:00
using osu.Game.Online.Multiplayer;
2020-12-25 12:38:11 +08:00
using osu.Game.Online.Rooms.RoomStatuses;
2018-04-13 17:19:50 +08:00
2020-12-25 12:38:11 +08:00
namespace osu.Game.Online.Rooms
2018-04-13 17:19:50 +08:00
{
[JsonObject(MemberSerialization.OptIn)]
public partial class Room : IDependencyInjectionCandidate, INotifyPropertyChanged
2018-04-13 17:19:50 +08:00
{
public event PropertyChangedEventHandler? PropertyChanged;
/// <summary>
/// The online room ID. Will be <c>null</c> while the room has not yet been created.
/// </summary>
public long? RoomID
{
get => roomId;
set => SetField(ref roomId, value);
}
2024-11-13 15:55:18 +08:00
/// <summary>
/// The room name.
/// </summary>
public string Name
{
get => name;
set => SetField(ref name, value);
}
/// <summary>
/// Sets the room password. Will be <c>null</c> after the room is created.
/// </summary>
/// <remarks>
/// To check if the room has a password, use <see cref="HasPassword"/>.
/// </remarks>
public string? Password
{
get => password;
set
{
SetField(ref password, value);
HasPassword = !string.IsNullOrEmpty(value);
}
}
/// <summary>
/// Whether the room has a password.
/// </summary>
/// <remarks>
/// To set a password, use <see cref="Password"/>.
/// </remarks>
[JsonProperty("has_password")]
public bool HasPassword
{
get => hasPassword;
private set => SetField(ref hasPassword, value);
}
2024-11-13 17:27:32 +08:00
/// <summary>
/// The room host. Will be <c>null</c> while the room has not yet been created.
/// </summary>
2024-11-13 16:32:32 +08:00
public APIUser? Host
{
get => host;
set => SetField(ref host, value);
}
2024-11-13 17:27:32 +08:00
/// <summary>
/// The room category.
/// </summary>
public RoomCategory Category
{
get => category;
set => SetField(ref category, value);
}
/// <summary>
/// The maximum number of users allowed in the room.
/// </summary>
public int? MaxParticipants
{
get => maxParticipants;
set => SetField(ref maxParticipants, value);
}
/// <summary>
/// The current number of users in the room.
/// </summary>
public int ParticipantCount
{
get => participantCount;
set => SetField(ref participantCount, value);
}
2024-11-13 17:36:47 +08:00
/// <summary>
/// The match type.
/// </summary>
public MatchType Type
{
get => type;
set => SetField(ref type, value);
}
2024-11-13 17:57:51 +08:00
/// <summary>
/// The playlist queueing mode. Only valid for multiplayer rooms.
/// </summary>
public QueueMode QueueMode
{
get => queueMode;
set => SetField(ref queueMode, value);
}
2024-11-13 18:15:29 +08:00
/// <summary>
/// Whether to automatically skip map intros. Only valid for multiplayer rooms.
/// </summary>
public bool AutoSkip
{
get => autoSkip;
set => SetField(ref autoSkip, value);
}
/// <summary>
/// The amount of time before the match is automatically started. Only valid for multiplayer rooms.
/// </summary>
public TimeSpan AutoStartDuration
{
get => TimeSpan.FromSeconds(autoStartDuration);
set => SetField(ref autoStartDuration, (ushort)value.TotalSeconds);
}
/// <summary>
/// Represents the current item selected within the room.
/// </summary>
/// <remarks>
/// Only valid for room listing requests (i.e. in the lounge screen), and may not be valid while inside the room.
/// </remarks>
public PlaylistItem? CurrentPlaylistItem
{
get => currentPlaylistItem;
set => SetField(ref currentPlaylistItem, value);
}
2024-11-13 17:48:13 +08:00
/// <summary>
/// The current room status.
/// </summary>
public RoomStatus Status
{
get => status;
set => SetField(ref status, value);
}
2024-11-13 17:52:04 +08:00
/// <summary>
/// Describes which players are able to join the room.
/// </summary>
public RoomAvailability Availability
{
get => availability;
set => SetField(ref availability, value);
}
[JsonProperty("id")]
private long? roomId;
2024-11-13 15:55:18 +08:00
[JsonProperty("name")]
private string name = string.Empty;
[JsonProperty("password")]
private string? password;
// Not serialised (internal use only).
private bool hasPassword;
2024-11-13 16:32:32 +08:00
[JsonProperty("host")]
private APIUser? host;
2024-11-13 17:27:32 +08:00
[JsonProperty("category")]
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
private RoomCategory category;
// Not yet serialised (not implemented).
private int? maxParticipants;
[JsonProperty("participant_count")]
private int participantCount;
2024-11-13 17:36:47 +08:00
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
[JsonProperty("type")]
private MatchType type;
2024-11-13 17:57:51 +08:00
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
[JsonProperty("queue_mode")]
private QueueMode queueMode;
2024-11-13 18:15:29 +08:00
[JsonProperty("auto_skip")]
private bool autoSkip;
[JsonProperty("auto_start_duration")]
private ushort autoStartDuration;
[JsonProperty("current_playlist_item")]
private PlaylistItem? currentPlaylistItem;
2024-11-13 17:48:13 +08:00
// Not serialised (see: GetRoomsRequest).
private RoomStatus status = new RoomStatusOpen();
2024-11-13 17:52:04 +08:00
// Not yet serialised (not implemented).
private RoomAvailability availability;
2019-02-05 18:00:01 +08:00
[Cached]
[JsonProperty("playlist")]
2020-07-10 11:07:17 +08:00
public readonly BindableList<PlaylistItem> Playlist = new BindableList<PlaylistItem>();
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-21 13:01:06 +08:00
[JsonProperty("channel_id")]
2020-07-10 11:07:17 +08:00
public readonly Bindable<int> ChannelId = new Bindable<int>();
2018-12-21 13:01:06 +08:00
[JsonProperty("playlist_item_stats")]
[Cached]
public readonly Bindable<RoomPlaylistItemStats> PlaylistItemStats = new Bindable<RoomPlaylistItemStats>();
[JsonProperty("difficulty_range")]
[Cached]
public readonly Bindable<RoomDifficultyRange> DifficultyRange = new Bindable<RoomDifficultyRange>();
2019-02-05 18:00:01 +08:00
[Cached]
2020-07-10 11:07:17 +08:00
public readonly Bindable<int?> MaxAttempts = new Bindable<int?>();
2018-12-13 15:06:30 +08:00
[Cached]
[JsonProperty("current_user_score")]
public readonly Bindable<PlaylistAggregateScore> UserScore = new Bindable<PlaylistAggregateScore>();
2019-02-05 18:00:01 +08:00
[Cached]
[JsonProperty("recent_participants")]
public readonly BindableList<APIUser> RecentParticipants = new BindableList<APIUser>();
2018-12-17 13:45:06 +08:00
#region Properties only used for room creation request
[Cached]
public readonly Bindable<TimeSpan?> Duration = new Bindable<TimeSpan?>();
2018-12-17 13:44:54 +08:00
[JsonProperty("duration")]
2020-12-21 15:18:39 +08:00
private int? duration
2018-12-17 13:44:54 +08:00
{
2020-12-21 15:18:39 +08:00
get => (int?)Duration.Value?.TotalMinutes;
set
{
if (value == null)
Duration.Value = null;
else
Duration.Value = TimeSpan.FromMinutes(value.Value);
}
2018-12-17 13:44:54 +08:00
}
2018-12-17 13:45:06 +08:00
#endregion
2024-05-31 17:49:56 +08:00
// Only supports retrieval for now
[Cached]
[JsonProperty("starts_at")]
public readonly Bindable<DateTimeOffset?> StartDate = new Bindable<DateTimeOffset?>();
// Only supports retrieval for now
2019-02-05 18:00:01 +08:00
[Cached]
[JsonProperty("ends_at")]
2020-12-21 15:18:39 +08:00
public readonly Bindable<DateTimeOffset?> EndDate = new Bindable<DateTimeOffset?>();
2018-12-13 15:06:30 +08:00
// Todo: Find a better way to do this (https://github.com/ppy/osu-framework/issues/1930)
[JsonProperty("max_attempts", DefaultValueHandling = DefaultValueHandling.Ignore)]
private int? maxAttempts
{
2019-02-21 17:56:34 +08:00
get => MaxAttempts.Value;
2018-12-13 15:06:30 +08:00
set => MaxAttempts.Value = value;
}
2018-12-12 18:04:11 +08:00
/// <summary>
/// Copies values from another <see cref="Room"/> into this one.
/// </summary>
/// <remarks>
/// **Beware**: This will store references between <see cref="Room"/>s.
/// </remarks>
/// <param name="other">The <see cref="Room"/> to copy values from.</param>
public void CopyFrom(Room other)
{
RoomID = other.RoomID;
2024-11-13 15:55:18 +08:00
Name = other.Name;
2020-12-26 19:13:28 +08:00
2024-11-13 17:27:32 +08:00
Category = other.Category;
2018-12-25 17:07:50 +08:00
2024-11-13 16:32:32 +08:00
if (other.Host != null && Host?.Id != other.Host.Id)
Host = other.Host;
ChannelId.Value = other.ChannelId.Value;
2024-11-13 17:48:13 +08:00
Status = other.Status;
2024-11-13 17:52:04 +08:00
Availability = other.Availability;
HasPassword = other.HasPassword;
2024-11-13 17:36:47 +08:00
Type = other.Type;
MaxParticipants = other.MaxParticipants;
ParticipantCount = other.ParticipantCount;
EndDate.Value = other.EndDate.Value;
UserScore.Value = other.UserScore.Value;
2024-11-13 17:57:51 +08:00
QueueMode = other.QueueMode;
AutoStartDuration = other.AutoStartDuration;
DifficultyRange.Value = other.DifficultyRange.Value;
PlaylistItemStats.Value = other.PlaylistItemStats.Value;
CurrentPlaylistItem = other.CurrentPlaylistItem;
2024-11-13 18:15:29 +08:00
AutoSkip = other.AutoSkip;
2021-09-15 16:03:26 +08:00
other.RemoveExpiredPlaylistItems();
2021-02-17 16:33:10 +08:00
if (!Playlist.SequenceEqual(other.Playlist))
{
Playlist.Clear();
Playlist.AddRange(other.Playlist);
}
if (!RecentParticipants.SequenceEqual(other.RecentParticipants))
{
RecentParticipants.Clear();
RecentParticipants.AddRange(other.RecentParticipants);
}
}
2021-09-15 16:03:26 +08:00
public void RemoveExpiredPlaylistItems()
{
// Todo: This is not the best way/place to do this, but the intention is to display all playlist items when the room has ended,
// and display only the non-expired playlist items while the room is still active. In order to achieve this, all expired items are removed from the source Room.
// More refactoring is required before this can be done locally instead - DrawableRoomPlaylist is currently directly bound to the playlist to display items in the room.
2024-11-13 17:48:13 +08:00
if (Status is not RoomStatusEnded)
2021-09-15 16:03:26 +08:00
Playlist.RemoveAll(i => i.Expired);
}
[JsonObject(MemberSerialization.OptIn)]
public class RoomPlaylistItemStats
{
[JsonProperty("count_active")]
public int CountActive;
[JsonProperty("count_total")]
public int CountTotal;
[JsonProperty("ruleset_ids")]
public int[] RulesetIDs = [];
}
[JsonObject(MemberSerialization.OptIn)]
public class RoomDifficultyRange
{
[JsonProperty("min")]
public double Min;
[JsonProperty("max")]
public double Max;
}
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null!)
=> PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
protected bool SetField<T>(ref T field, T value, [CallerMemberName] string propertyName = null!)
{
if (EqualityComparer<T>.Default.Equals(field, value))
return false;
field = value;
OnPropertyChanged(propertyName);
return true;
}
}
2018-04-13 17:19:50 +08:00
}