2019-01-24 16:43:03 +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.
2018-04-13 17:19:50 +08:00
2018-12-17 13:45:06 +08:00
using System ;
2024-11-12 00:38:31 +08:00
using System.Collections.Generic ;
using System.ComponentModel ;
2018-12-03 19:50:40 +08:00
using System.Linq ;
2024-11-12 00:38:31 +08:00
using System.Runtime.CompilerServices ;
2018-12-12 15:06:56 +08:00
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 ;
2021-11-04 17:02:44 +08:00
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
2017-05-22 11:07:15 +08:00
{
2022-02-23 16:06:40 +08:00
[JsonObject(MemberSerialization.OptIn)]
2024-11-12 00:38:31 +08:00
public partial class Room : IDependencyInjectionCandidate , INotifyPropertyChanged
2017-05-22 11:07:15 +08:00
{
2024-11-12 00:38:31 +08:00
public event PropertyChangedEventHandler ? PropertyChanged ;
2024-11-13 15:28:39 +08:00
/// <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 ) ;
}
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 ) ;
}
2024-11-12 00:38:31 +08:00
/// <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 15:28:39 +08:00
[JsonProperty("id")]
private long? roomId ;
2024-11-13 15:55:18 +08:00
[JsonProperty("name")]
private string name = string . Empty ;
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 ;
2024-11-13 14:50:01 +08:00
[JsonProperty("current_playlist_item")]
private PlaylistItem ? currentPlaylistItem ;
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-12 15:06:56 +08:00
[JsonProperty("playlist")]
2020-07-10 11:07:17 +08:00
public readonly BindableList < PlaylistItem > Playlist = new BindableList < PlaylistItem > ( ) ;
2018-12-12 15:06:56 +08:00
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
2022-02-21 18:02:21 +08:00
[JsonProperty("playlist_item_stats")]
[Cached]
public readonly Bindable < RoomPlaylistItemStats > PlaylistItemStats = new Bindable < RoomPlaylistItemStats > ( ) ;
2022-02-17 18:15:09 +08:00
[JsonProperty("difficulty_range")]
2022-02-24 15:12:15 +08:00
[Cached]
2022-02-17 18:15:09 +08:00
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
2019-02-05 18:00:01 +08:00
[Cached]
2020-07-10 11:07:17 +08:00
public readonly Bindable < RoomStatus > Status = new Bindable < RoomStatus > ( new RoomStatusOpen ( ) ) ;
2018-12-17 13:45:06 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2020-07-10 11:07:17 +08:00
public readonly Bindable < RoomAvailability > Availability = new Bindable < RoomAvailability > ( ) ;
2018-12-17 13:45:06 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2021-08-03 13:46:31 +08:00
public readonly Bindable < MatchType > Type = new Bindable < MatchType > ( ) ;
2018-12-17 13:45:06 +08:00
2021-08-03 22:08:11 +08:00
// Todo: osu-framework bug (https://github.com/ppy/osu-framework/issues/4106)
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
[JsonProperty("type")]
private MatchType type
{
get = > Type . Value ;
set = > Type . Value = value ;
}
2021-10-20 13:51:59 +08:00
[Cached]
2021-11-16 13:53:10 +08:00
public readonly Bindable < QueueMode > QueueMode = new Bindable < QueueMode > ( ) ;
2021-10-20 13:51:59 +08:00
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
[JsonProperty("queue_mode")]
2021-11-16 13:53:10 +08:00
private QueueMode queueMode
2021-10-20 13:51:59 +08:00
{
get = > QueueMode . Value ;
set = > QueueMode . Value = value ;
}
2022-03-18 21:00:39 +08:00
[Cached]
public readonly Bindable < TimeSpan > AutoStartDuration = new Bindable < TimeSpan > ( ) ;
2022-03-25 14:34:33 +08:00
[JsonProperty("auto_start_duration")]
2022-03-18 21:00:39 +08:00
private ushort autoStartDuration
{
get = > ( ushort ) AutoStartDuration . Value . TotalSeconds ;
set = > AutoStartDuration . Value = TimeSpan . FromSeconds ( value ) ;
}
2019-02-05 18:00:01 +08:00
[Cached]
2020-07-10 11:07:17 +08:00
public readonly Bindable < int? > MaxParticipants = new Bindable < int? > ( ) ;
2018-12-17 13:45:06 +08:00
2021-02-16 12:32:14 +08:00
[Cached]
[JsonProperty("current_user_score")]
public readonly Bindable < PlaylistAggregateScore > UserScore = new Bindable < PlaylistAggregateScore > ( ) ;
2021-07-07 17:53:13 +08:00
[JsonProperty("has_password")]
2024-05-03 17:10:59 +08:00
public readonly Bindable < bool > HasPassword = new Bindable < bool > ( ) ;
2021-07-07 17:53:13 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2020-02-27 18:24:13 +08:00
[JsonProperty("recent_participants")]
2021-11-04 17:02:44 +08:00
public readonly BindableList < APIUser > RecentParticipants = new BindableList < APIUser > ( ) ;
2018-12-17 13:45:06 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-26 21:25:15 +08:00
[JsonProperty("participant_count")]
2020-12-21 15:35:19 +08:00
public readonly Bindable < int > ParticipantCount = new Bindable < int > ( ) ;
2018-12-26 21:25:15 +08:00
2021-07-07 17:53:13 +08:00
#region Properties only used for room creation request
2021-07-19 19:02:14 +08:00
[Cached(Name = nameof(Password))]
2021-07-07 17:53:13 +08:00
[JsonProperty("password")]
2024-11-12 00:38:31 +08:00
public readonly Bindable < string? > Password = new Bindable < string? > ( ) ;
2021-07-07 17:53:13 +08:00
[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
2021-07-07 17:53:13 +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 ? > ( ) ;
2018-12-19 09:52:15 +08:00
// Only supports retrieval for now
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-19 09:52:15 +08:00
[JsonProperty("ends_at")]
2020-12-21 15:18:39 +08:00
public readonly Bindable < DateTimeOffset ? > EndDate = new Bindable < DateTimeOffset ? > ( ) ;
2018-12-19 09:52:15 +08:00
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
2022-08-31 18:49:04 +08:00
[Cached]
[JsonProperty("auto_skip")]
2022-09-01 13:14:22 +08:00
public readonly Bindable < bool > AutoSkip = new Bindable < bool > ( ) ;
2022-08-31 18:49:04 +08:00
2021-07-10 13:14:22 +08:00
public Room ( )
{
Password . BindValueChanged ( p = > HasPassword . Value = ! string . IsNullOrEmpty ( p . NewValue ) ) ;
}
2022-06-03 18:17:31 +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>
2020-08-16 04:06:16 +08:00
public void CopyFrom ( Room other )
{
2024-11-13 15:28:39 +08:00
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 ;
2020-08-16 04:06:16 +08:00
ChannelId . Value = other . ChannelId . Value ;
Status . Value = other . Status . Value ;
2019-02-21 17:56:34 +08:00
Availability . Value = other . Availability . Value ;
2021-07-12 17:54:07 +08:00
HasPassword . Value = other . HasPassword . Value ;
2019-02-21 17:56:34 +08:00
Type . Value = other . Type . Value ;
MaxParticipants . Value = other . MaxParticipants . Value ;
2020-08-16 04:06:16 +08:00
ParticipantCount . Value = other . ParticipantCount . Value ;
EndDate . Value = other . EndDate . Value ;
2021-02-16 12:32:14 +08:00
UserScore . Value = other . UserScore . Value ;
2021-10-22 19:14:04 +08:00
QueueMode . Value = other . QueueMode . Value ;
2022-03-18 21:00:39 +08:00
AutoStartDuration . Value = other . AutoStartDuration . Value ;
2022-02-22 14:47:00 +08:00
DifficultyRange . Value = other . DifficultyRange . Value ;
PlaylistItemStats . Value = other . PlaylistItemStats . Value ;
2024-11-12 00:38:31 +08:00
CurrentPlaylistItem = other . CurrentPlaylistItem ;
2022-08-31 18:49:04 +08:00
AutoSkip . Value = other . AutoSkip . Value ;
2020-08-16 04:06:16 +08:00
2021-09-15 16:03:26 +08:00
other . RemoveExpiredPlaylistItems ( ) ;
2021-02-17 16:33:10 +08:00
2020-02-16 15:23:46 +08:00
if ( ! Playlist . SequenceEqual ( other . Playlist ) )
{
Playlist . Clear ( ) ;
Playlist . AddRange ( other . Playlist ) ;
}
2020-08-16 04:06:16 +08:00
if ( ! RecentParticipants . SequenceEqual ( other . RecentParticipants ) )
{
RecentParticipants . Clear ( ) ;
RecentParticipants . AddRange ( other . RecentParticipants ) ;
}
2018-12-12 15:06:56 +08:00
}
2018-12-17 10:04:38 +08:00
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.
if ( ! ( Status . Value is RoomStatusEnded ) )
Playlist . RemoveAll ( i = > i . Expired ) ;
}
2022-02-21 18:02:21 +08:00
[JsonObject(MemberSerialization.OptIn)]
public class RoomPlaylistItemStats
{
[JsonProperty("count_active")]
public int CountActive ;
[JsonProperty("count_total")]
public int CountTotal ;
[JsonProperty("ruleset_ids")]
2024-11-12 00:38:31 +08:00
public int [ ] RulesetIDs = [ ] ;
2022-02-21 18:02:21 +08:00
}
2022-02-17 18:15:09 +08:00
[JsonObject(MemberSerialization.OptIn)]
public class RoomDifficultyRange
{
[JsonProperty("min")]
public double Min ;
[JsonProperty("max")]
public double Max ;
}
2024-11-12 00:38:31 +08:00
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-12-12 15:06:56 +08:00
}
2017-05-22 11:07:15 +08:00
}