2020-09-05 02:52:07 +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.
2022-07-27 18:35:25 +08:00
using System ;
2024-11-14 13:46:58 +08:00
using System.Collections.Generic ;
2022-07-27 15:46:23 +08:00
using System.Diagnostics ;
2020-09-08 15:43:07 +08:00
using System.Linq ;
2022-07-27 18:35:25 +08:00
using osu.Framework.Allocation ;
2020-09-08 15:43:07 +08:00
using osu.Framework.Bindables ;
2020-09-05 02:52:07 +08:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2022-07-27 18:35:25 +08:00
using osu.Game.Database ;
2020-09-05 02:52:07 +08:00
using osu.Game.Graphics.Containers ;
using osuTK ;
2022-07-27 18:35:25 +08:00
using Realms ;
2020-09-05 02:52:07 +08:00
namespace osu.Game.Collections
{
2020-09-08 15:50:51 +08:00
/// <summary>
2022-07-27 15:46:23 +08:00
/// Visualises a list of <see cref="BeatmapCollection"/>s.
2020-09-08 15:50:51 +08:00
/// </summary>
2022-11-24 13:32:20 +08:00
public partial class DrawableCollectionList : OsuRearrangeableListContainer < Live < BeatmapCollection > >
2020-09-05 02:52:07 +08:00
{
2024-11-18 22:28:30 +08:00
public new MarginPadding Padding
{
get = > base . Padding ;
set = > base . Padding = value ;
}
2020-09-08 15:43:07 +08:00
protected override ScrollContainer < Drawable > CreateScrollContainer ( ) = > scroll = new Scroll ( ) ;
2020-09-05 02:52:07 +08:00
2022-07-27 18:35:25 +08:00
[Resolved]
private RealmAccess realm { get ; set ; } = null ! ;
2022-07-28 12:50:19 +08:00
private Scroll scroll = null ! ;
private IDisposable ? realmSubscription ;
2024-11-14 13:46:58 +08:00
private Flow flow = null ! ;
public IEnumerable < Drawable > OrderedItems = > flow . FlowingChildren ;
2024-11-18 14:57:39 +08:00
public string SearchTerm
{
get = > flow . SearchTerm ;
set = > flow . SearchTerm = value ;
}
2024-11-14 13:46:58 +08:00
protected override FillFlowContainer < RearrangeableListItem < Live < BeatmapCollection > > > CreateListFillFlowContainer ( ) = > flow = new Flow
2020-09-05 02:52:07 +08:00
{
2020-09-08 15:43:07 +08:00
DragActive = { BindTarget = DragActive }
2020-09-05 02:52:07 +08:00
} ;
2022-07-27 18:35:25 +08:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2022-07-28 13:07:42 +08:00
realmSubscription = realm . RegisterForNotifications ( r = > r . All < BeatmapCollection > ( ) . OrderBy ( c = > c . Name ) , collectionsChanged ) ;
2022-07-27 18:35:25 +08:00
}
2024-11-18 14:12:28 +08:00
/// <summary>
/// When non-null, signifies that a new collection was created and should be presented to the user.
/// </summary>
private Guid ? lastCreated ;
protected override void OnItemsChanged ( )
{
base . OnItemsChanged ( ) ;
if ( lastCreated ! = null )
{
var createdItem = flow . Children . SingleOrDefault ( item = > item . Model . Value . ID = = lastCreated ) ;
if ( createdItem ! = null )
scroll . ScrollTo ( createdItem ) ;
lastCreated = null ;
}
}
2023-07-06 12:37:42 +08:00
private void collectionsChanged ( IRealmCollection < BeatmapCollection > collections , ChangeSet ? changes )
2022-07-27 18:35:25 +08:00
{
2024-11-14 13:12:13 +08:00
if ( changes = = null )
{
Items . AddRange ( collections . AsEnumerable ( ) . Select ( c = > c . ToLive ( realm ) ) ) ;
return ;
}
foreach ( int i in changes . DeletedIndices . OrderDescending ( ) )
Items . RemoveAt ( i ) ;
foreach ( int i in changes . InsertedIndices )
Items . Insert ( i , collections [ i ] . ToLive ( realm ) ) ;
2024-11-18 14:12:28 +08:00
if ( changes . InsertedIndices . Length = = 1 )
lastCreated = collections [ changes . InsertedIndices [ 0 ] ] . ID ;
2024-11-14 13:12:13 +08:00
foreach ( int i in changes . NewModifiedIndices )
2024-11-18 14:12:28 +08:00
2024-11-14 13:12:13 +08:00
{
var updatedItem = collections [ i ] ;
Items . RemoveAt ( i ) ;
Items . Insert ( i , updatedItem . ToLive ( realm ) ) ;
}
2022-07-27 18:35:25 +08:00
}
2022-07-27 14:59:36 +08:00
2022-07-27 18:35:25 +08:00
protected override OsuRearrangeableListItem < Live < BeatmapCollection > > CreateOsuDrawable ( Live < BeatmapCollection > item )
2020-09-08 15:43:07 +08:00
{
2022-07-27 14:59:36 +08:00
if ( item . ID = = scroll . PlaceholderItem . Model . ID )
2020-09-08 15:43:07 +08:00
return scroll . ReplacePlaceholder ( ) ;
return new DrawableCollectionListItem ( item , true ) ;
}
2022-07-28 12:50:19 +08:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
realmSubscription ? . Dispose ( ) ;
}
2020-09-08 15:50:51 +08:00
/// <summary>
/// The scroll container for this <see cref="DrawableCollectionList"/>.
/// Contains the main flow of <see cref="DrawableCollectionListItem"/> and attaches a placeholder item to the end of the list.
/// </summary>
/// <remarks>
/// Use <see cref="ReplacePlaceholder"/> to transfer the placeholder into the main list.
/// </remarks>
2022-11-24 13:32:20 +08:00
private partial class Scroll : OsuScrollContainer
2020-09-08 15:43:07 +08:00
{
2020-09-08 15:50:51 +08:00
/// <summary>
/// The currently-displayed placeholder item.
/// </summary>
2022-07-27 15:46:23 +08:00
public DrawableCollectionListItem PlaceholderItem { get ; private set ; } = null ! ;
2020-09-08 15:43:07 +08:00
protected override Container < Drawable > Content = > content ;
private readonly Container content ;
private readonly Container < DrawableCollectionListItem > placeholderContainer ;
public Scroll ( )
{
2024-11-18 13:38:12 +08:00
ScrollbarOverlapsContent = false ;
2020-09-08 15:43:07 +08:00
base . Content . Add ( new FillFlowContainer
{
RelativeSizeAxes = Axes . X ,
AutoSizeAxes = Axes . Y ,
LayoutDuration = 200 ,
LayoutEasing = Easing . OutQuint ,
Children = new Drawable [ ]
{
content = new Container { RelativeSizeAxes = Axes . X } ,
placeholderContainer = new Container < DrawableCollectionListItem >
{
RelativeSizeAxes = Axes . X ,
AutoSizeAxes = Axes . Y
}
}
} ) ;
ReplacePlaceholder ( ) ;
2022-07-27 15:46:23 +08:00
Debug . Assert ( PlaceholderItem ! = null ) ;
2020-09-08 15:43:07 +08:00
}
protected override void Update ( )
{
base . Update ( ) ;
// AutoSizeAxes cannot be used as the height should represent the post-layout-transform height at all times, so that the placeholder doesn't bounce around.
2024-11-18 14:57:39 +08:00
content . Height = ( ( Flow ) Child ) . Children . Sum ( c = > c . IsPresent ? c . DrawHeight + 5 : 0 ) ;
2020-09-08 15:43:07 +08:00
}
/// <summary>
/// Replaces the current <see cref="PlaceholderItem"/> with a new one, and returns the previous.
/// </summary>
2020-09-08 15:50:51 +08:00
/// <returns>The current <see cref="PlaceholderItem"/>.</returns>
2020-09-08 15:43:07 +08:00
public DrawableCollectionListItem ReplacePlaceholder ( )
{
var previous = PlaceholderItem ;
placeholderContainer . Clear ( false ) ;
2024-11-14 13:12:31 +08:00
placeholderContainer . Add ( PlaceholderItem = new NewCollectionEntryItem ( ) ) ;
2020-09-08 15:43:07 +08:00
return previous ;
}
}
2024-11-14 18:58:57 +08:00
private partial class NewCollectionEntryItem : DrawableCollectionListItem
2024-11-14 13:12:31 +08:00
{
[Resolved]
private RealmAccess realm { get ; set ; } = null ! ;
public NewCollectionEntryItem ( )
: base ( new BeatmapCollection ( ) . ToLiveUnmanaged ( ) , false )
{
}
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
TextBox . OnCommit + = ( sender , newText ) = >
{
2024-11-14 13:23:43 +08:00
if ( string . IsNullOrEmpty ( TextBox . Text ) )
return ;
realm . Write ( r = > r . Add ( new BeatmapCollection ( TextBox . Text ) ) ) ;
2024-11-14 13:12:31 +08:00
TextBox . Text = string . Empty ;
} ;
}
}
2020-09-08 15:50:51 +08:00
/// <summary>
/// The flow of <see cref="DrawableCollectionListItem"/>. Disables layout easing unless a drag is in progress.
/// </summary>
2024-11-18 14:57:39 +08:00
private partial class Flow : SearchContainer < RearrangeableListItem < Live < BeatmapCollection > > >
2020-09-08 15:43:07 +08:00
{
public readonly IBindable < bool > DragActive = new Bindable < bool > ( ) ;
public Flow ( )
{
Spacing = new Vector2 ( 0 , 5 ) ;
LayoutEasing = Easing . OutQuint ;
2024-11-18 13:38:12 +08:00
Padding = new MarginPadding { Right = 5 } ;
2020-09-08 15:43:07 +08:00
}
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
DragActive . BindValueChanged ( active = > LayoutDuration = active . NewValue ? 200 : 0 ) ;
}
}
2020-09-05 02:52:07 +08:00
}
}