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 ;
2018-12-03 19:50:40 +08:00
using System.Linq ;
2021-12-28 15:37:16 +08:00
using JetBrains.Annotations ;
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 ;
2021-07-19 11:38:22 +08:00
using osu.Game.Utils ;
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
{
2021-07-19 11:38:22 +08:00
public class Room : IDeepCloneable < Room >
2018-04-13 17:19:50 +08:00
{
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-12 18:04:11 +08:00
[JsonProperty("id")]
2021-02-16 18:29:40 +08:00
public readonly Bindable < long? > RoomID = new Bindable < long? > ( ) ;
2018-12-12 15:06:56 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-12 15:06:56 +08:00
[JsonProperty("name")]
2020-07-10 11:07:17 +08:00
public readonly Bindable < string > Name = new Bindable < string > ( ) ;
2018-12-12 15:06:56 +08:00
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-12 15:06:56 +08:00
[JsonProperty("host")]
2021-11-04 17:02:44 +08:00
public readonly Bindable < APIUser > Host = new Bindable < APIUser > ( ) ;
2018-12-12 18:04:11 +08:00
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
2020-07-10 18:37:27 +08:00
[Cached]
2020-12-21 15:42:21 +08:00
[JsonIgnore]
2020-07-10 18:37:27 +08:00
public readonly Bindable < RoomCategory > Category = new Bindable < RoomCategory > ( ) ;
2020-12-21 15:42:21 +08:00
// Todo: osu-framework bug (https://github.com/ppy/osu-framework/issues/4106)
[JsonProperty("category")]
2020-12-21 15:56:45 +08:00
[JsonConverter(typeof(SnakeCaseStringEnumConverter))]
private RoomCategory category
2020-12-21 15:42:21 +08:00
{
2020-12-21 15:56:45 +08:00
get = > Category . Value ;
set = > Category . Value = value ;
2020-12-21 15:42:21 +08:00
}
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-13 15:06:30 +08:00
[JsonIgnore]
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]
2018-12-17 13:45:06 +08:00
[JsonIgnore]
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]
2018-12-17 13:45:06 +08:00
[JsonIgnore]
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]
2018-12-17 13:45:06 +08:00
[JsonIgnore]
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]
[JsonIgnore]
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 ;
}
2019-02-05 18:00:01 +08:00
[Cached]
2018-12-17 13:45:06 +08:00
[JsonIgnore]
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")]
public readonly BindableBool HasPassword = new BindableBool ( ) ;
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")]
public readonly Bindable < string > Password = new Bindable < string > ( ) ;
[Cached]
[JsonIgnore]
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
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
2021-07-10 13:14:22 +08:00
public Room ( )
{
Password . BindValueChanged ( p = > HasPassword . Value = ! string . IsNullOrEmpty ( p . NewValue ) ) ;
}
2020-08-10 07:16:01 +08:00
/// <summary>
2020-10-19 16:15:13 +08:00
/// Create a copy of this room without online information.
/// Should be used to create a local copy of a room for submitting in the future.
2020-08-10 07:16:01 +08:00
/// </summary>
2021-07-19 11:38:22 +08:00
public Room DeepClone ( )
2018-12-12 15:06:56 +08:00
{
2020-10-19 16:15:13 +08:00
var copy = new Room ( ) ;
2020-08-10 07:16:01 +08:00
2020-10-19 16:15:13 +08:00
copy . CopyFrom ( this ) ;
copy . RoomID . Value = null ;
2020-08-10 07:16:01 +08:00
2020-10-19 16:15:13 +08:00
return copy ;
2020-08-16 04:06:16 +08:00
}
2020-08-10 07:16:01 +08:00
2020-08-16 04:06:16 +08:00
public void CopyFrom ( Room other )
{
RoomID . Value = other . RoomID . Value ;
2020-08-10 07:16:01 +08:00
Name . Value = other . Name . Value ;
2020-12-26 19:13:28 +08:00
if ( other . Category . Value ! = RoomCategory . Spotlight )
Category . Value = other . Category . Value ;
2018-12-25 17:07:50 +08:00
2020-08-16 04:06:16 +08:00
if ( other . Host . Value ! = null & & Host . Value ? . Id ! = other . Host . Value . Id )
Host . Value = other . Host . Value ;
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 ;
2020-08-16 04:06:16 +08:00
2020-12-21 15:23:42 +08:00
if ( EndDate . Value ! = null & & DateTimeOffset . Now > = EndDate . Value )
2020-08-16 04:06:16 +08:00
Status . Value = new RoomStatusEnded ( ) ;
2018-12-27 12:30:36 +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 ) ;
}
2021-12-28 15:37:16 +08:00
#region Newtonsoft . Json implicit ShouldSerialize ( ) methods
// The properties in this region are used implicitly by Newtonsoft.Json to not serialise certain fields in some cases.
// They rely on being named exactly the same as the corresponding fields (casing included) and as such should NOT be renamed
// unless the fields are also renamed.
[UsedImplicitly]
2018-12-17 13:45:06 +08:00
public bool ShouldSerializeRoomID ( ) = > false ;
2021-12-28 15:37:16 +08:00
[UsedImplicitly]
2018-12-17 13:45:06 +08:00
public bool ShouldSerializeHost ( ) = > false ;
2021-12-28 15:37:16 +08:00
[UsedImplicitly]
2018-12-19 09:52:15 +08:00
public bool ShouldSerializeEndDate ( ) = > false ;
2021-12-28 15:37:16 +08:00
#endregion
2018-12-12 15:06:56 +08:00
}
2018-04-13 17:19:50 +08:00
}