2020-02-14 14:01:45 +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.
2020-02-14 14:36:47 +08:00
using System.Collections.Generic ;
2020-02-14 14:01:45 +08:00
using System.Linq ;
using NUnit.Framework ;
2020-06-02 13:31:43 +08:00
using osu.Framework.Allocation ;
using osu.Framework.Audio ;
2020-02-14 14:01:45 +08:00
using osu.Framework.Graphics ;
2020-02-14 15:48:30 +08:00
using osu.Framework.Graphics.Containers ;
2020-06-02 13:31:43 +08:00
using osu.Framework.Platform ;
2020-02-14 15:48:30 +08:00
using osu.Framework.Testing ;
2020-06-02 13:31:43 +08:00
using osu.Game.Beatmaps ;
2020-02-14 15:48:30 +08:00
using osu.Game.Graphics.Containers ;
using osu.Game.Graphics.UserInterface ;
2020-12-25 12:38:11 +08:00
using osu.Game.Online.Rooms ;
2020-06-02 13:31:43 +08:00
using osu.Game.Overlays ;
using osu.Game.Rulesets ;
2020-02-14 16:57:25 +08:00
using osu.Game.Rulesets.Osu ;
2020-02-14 14:01:45 +08:00
using osu.Game.Rulesets.Osu.Mods ;
2020-12-25 23:50:00 +08:00
using osu.Game.Screens.OnlinePlay ;
2020-02-14 16:57:25 +08:00
using osu.Game.Tests.Beatmaps ;
2020-02-14 14:01:45 +08:00
using osuTK ;
2020-02-14 15:48:30 +08:00
using osuTK.Input ;
2020-02-14 14:01:45 +08:00
namespace osu.Game.Tests.Visual.Multiplayer
{
2020-03-23 09:01:33 +08:00
public class TestSceneDrawableRoomPlaylist : OsuManualInputManagerTestScene
2020-02-14 14:01:45 +08:00
{
2020-02-20 07:37:23 +08:00
private TestPlaylist playlist ;
2020-02-14 14:01:45 +08:00
2020-06-02 13:31:43 +08:00
private BeatmapManager manager ;
private RulesetStore rulesets ;
[BackgroundDependencyLoader]
private void load ( GameHost host , AudioManager audio )
{
Dependencies . Cache ( rulesets = new RulesetStore ( ContextFactory ) ) ;
Dependencies . Cache ( manager = new BeatmapManager ( LocalStorage , ContextFactory , rulesets , null , audio , host , Beatmap . Default ) ) ;
manager . Import ( new TestBeatmap ( new OsuRuleset ( ) . RulesetInfo ) . BeatmapInfo . BeatmapSet ) . Wait ( ) ;
}
2020-02-14 14:01:45 +08:00
[Test]
2020-02-14 14:36:47 +08:00
public void TestNonEditableNonSelectable ( )
2020-02-14 14:01:45 +08:00
{
createPlaylist ( false , false ) ;
2020-02-14 15:48:30 +08:00
moveToItem ( 0 ) ;
assertHandleVisibility ( 0 , false ) ;
assertDeleteButtonVisibility ( 0 , false ) ;
2020-02-14 15:55:05 +08:00
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "no item selected" , ( ) = > playlist . SelectedItem . Value = = null ) ;
2020-02-14 14:01:45 +08:00
}
[Test]
2020-02-14 14:36:47 +08:00
public void TestEditable ( )
2020-02-14 14:01:45 +08:00
{
createPlaylist ( true , false ) ;
2020-02-14 15:48:30 +08:00
moveToItem ( 0 ) ;
assertHandleVisibility ( 0 , true ) ;
assertDeleteButtonVisibility ( 0 , true ) ;
2020-02-14 15:55:05 +08:00
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "no item selected" , ( ) = > playlist . SelectedItem . Value = = null ) ;
2020-02-14 14:01:45 +08:00
}
[Test]
2020-02-14 14:36:47 +08:00
public void TestSelectable ( )
2020-02-14 14:01:45 +08:00
{
createPlaylist ( false , true ) ;
2020-02-14 15:48:30 +08:00
moveToItem ( 0 ) ;
assertHandleVisibility ( 0 , false ) ;
assertDeleteButtonVisibility ( 0 , false ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "item 0 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 0 ] ) ;
2020-02-14 14:01:45 +08:00
}
[Test]
2020-02-14 14:36:47 +08:00
public void TestEditableSelectable ( )
2020-02-14 14:01:45 +08:00
{
createPlaylist ( true , true ) ;
2020-02-14 15:48:30 +08:00
moveToItem ( 0 ) ;
assertHandleVisibility ( 0 , true ) ;
assertDeleteButtonVisibility ( 0 , true ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "item 0 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 0 ] ) ;
}
[Test]
public void TestSelectionNotLostAfterRearrangement ( )
{
createPlaylist ( true , true ) ;
moveToItem ( 0 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
moveToDragger ( 0 ) ;
AddStep ( "begin drag" , ( ) = > InputManager . PressButton ( MouseButton . Left ) ) ;
moveToDragger ( 1 , new Vector2 ( 0 , 5 ) ) ;
AddStep ( "end drag" , ( ) = > InputManager . ReleaseButton ( MouseButton . Left ) ) ;
AddAssert ( "item 1 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 1 ] ) ;
}
[Test]
public void TestItemRemovedOnDeletion ( )
{
PlaylistItem selectedItem = null ;
createPlaylist ( true , true ) ;
moveToItem ( 0 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddStep ( "retrieve selection" , ( ) = > selectedItem = playlist . SelectedItem . Value ) ;
moveToDeleteButton ( 0 ) ;
AddStep ( "click delete button" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "item removed" , ( ) = > ! playlist . Items . Contains ( selectedItem ) ) ;
}
[Test]
public void TestNextItemSelectedAfterDeletion ( )
{
createPlaylist ( true , true ) ;
moveToItem ( 0 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
moveToDeleteButton ( 0 ) ;
AddStep ( "click delete button" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "item 0 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 0 ] ) ;
}
[Test]
public void TestLastItemSelectedAfterLastItemDeleted ( )
{
createPlaylist ( true , true ) ;
AddWaitStep ( "wait for flow" , 5 ) ; // Items may take 1 update frame to flow. A wait count of 5 is guaranteed to result in the flow being updated as desired.
AddStep ( "scroll to bottom" , ( ) = > playlist . ChildrenOfType < ScrollContainer < Drawable > > ( ) . First ( ) . ScrollToEnd ( false ) ) ;
moveToItem ( 19 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
moveToDeleteButton ( 19 ) ;
AddStep ( "click delete button" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddAssert ( "item 18 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 18 ] ) ;
}
[Test]
public void TestSelectionResetWhenAllItemsDeleted ( )
{
createPlaylist ( true , true ) ;
AddStep ( "remove all but one item" , ( ) = >
{
playlist . Items . RemoveRange ( 1 , playlist . Items . Count - 1 ) ;
} ) ;
moveToItem ( 0 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
moveToDeleteButton ( 0 ) ;
AddStep ( "click delete button" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
2020-02-14 15:55:05 +08:00
AddAssert ( "no item selected" , ( ) = > playlist . SelectedItem . Value = = null ) ;
2020-02-14 15:48:30 +08:00
}
// Todo: currently not possible due to bindable list shortcomings (https://github.com/ppy/osu-framework/issues/3081)
// [Test]
public void TestNextItemSelectedAfterExternalDeletion ( )
{
createPlaylist ( true , true ) ;
moveToItem ( 0 ) ;
AddStep ( "click" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
AddStep ( "remove item 0" , ( ) = > playlist . Items . RemoveAt ( 0 ) ) ;
AddAssert ( "item 0 is selected" , ( ) = > playlist . SelectedItem . Value = = playlist . Items [ 0 ] ) ;
2020-02-14 14:01:45 +08:00
}
2020-02-15 23:22:14 +08:00
[Test]
public void TestChangeBeatmapAndRemove ( )
{
createPlaylist ( true , true ) ;
AddStep ( "change beatmap of first item" , ( ) = > playlist . Items [ 0 ] . BeatmapID = 30 ) ;
moveToDeleteButton ( 0 ) ;
AddStep ( "click delete button" , ( ) = > InputManager . Click ( MouseButton . Left ) ) ;
}
2020-06-02 13:31:43 +08:00
[Test]
public void TestDownloadButtonHiddenInitiallyWhenBeatmapExists ( )
{
createPlaylist ( new TestBeatmap ( new OsuRuleset ( ) . RulesetInfo ) . BeatmapInfo ) ;
AddAssert ( "download button hidden" , ( ) = > ! playlist . ChildrenOfType < BeatmapDownloadTrackingComposite > ( ) . Single ( ) . IsPresent ) ;
}
[Test]
public void TestDownloadButtonVisibleInitiallyWhenBeatmapDoesNotExist ( )
{
var byOnlineId = new TestBeatmap ( new OsuRuleset ( ) . RulesetInfo ) . BeatmapInfo ;
byOnlineId . BeatmapSet . OnlineBeatmapSetID = 1337 ; // Some random ID that does not exist locally.
var byChecksum = new TestBeatmap ( new OsuRuleset ( ) . RulesetInfo ) . BeatmapInfo ;
byChecksum . MD5Hash = "1337" ; // Some random checksum that does not exist locally.
createPlaylist ( byOnlineId , byChecksum ) ;
AddAssert ( "download buttons shown" , ( ) = > playlist . ChildrenOfType < BeatmapDownloadTrackingComposite > ( ) . All ( d = > d . IsPresent ) ) ;
}
2020-02-14 15:48:30 +08:00
private void moveToItem ( int index , Vector2 ? offset = null )
= > AddStep ( $"move mouse to item {index}" , ( ) = > InputManager . MoveMouseTo ( playlist . ChildrenOfType < OsuRearrangeableListItem < PlaylistItem > > ( ) . ElementAt ( index ) , offset ) ) ;
private void moveToDragger ( int index , Vector2 ? offset = null ) = > AddStep ( $"move mouse to dragger {index}" , ( ) = >
{
var item = playlist . ChildrenOfType < OsuRearrangeableListItem < PlaylistItem > > ( ) . ElementAt ( index ) ;
InputManager . MoveMouseTo ( item . ChildrenOfType < OsuRearrangeableListItem < PlaylistItem > . PlaylistItemHandle > ( ) . Single ( ) , offset ) ;
} ) ;
private void moveToDeleteButton ( int index , Vector2 ? offset = null ) = > AddStep ( $"move mouse to delete button {index}" , ( ) = >
{
var item = playlist . ChildrenOfType < OsuRearrangeableListItem < PlaylistItem > > ( ) . ElementAt ( index ) ;
InputManager . MoveMouseTo ( item . ChildrenOfType < IconButton > ( ) . ElementAt ( 0 ) , offset ) ;
} ) ;
private void assertHandleVisibility ( int index , bool visible )
= > AddAssert ( $"handle {index} {(visible ? " is " : " is not ")} visible" ,
( ) = > ( playlist . ChildrenOfType < OsuRearrangeableListItem < PlaylistItem > . PlaylistItemHandle > ( ) . ElementAt ( index ) . Alpha > 0 ) = = visible ) ;
private void assertDeleteButtonVisibility ( int index , bool visible )
= > AddAssert ( $"delete button {index} {(visible ? " is " : " is not ")} visible" , ( ) = > ( playlist . ChildrenOfType < IconButton > ( ) . ElementAt ( 2 + index * 2 ) . Alpha > 0 ) = = visible ) ;
2020-02-20 07:37:23 +08:00
private void createPlaylist ( bool allowEdit , bool allowSelection )
2020-02-14 14:01:45 +08:00
{
2020-02-20 07:37:23 +08:00
AddStep ( "create playlist" , ( ) = >
2020-02-14 14:01:45 +08:00
{
2020-02-20 07:37:23 +08:00
Child = playlist = new TestPlaylist ( allowEdit , allowSelection )
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
Size = new Vector2 ( 500 , 300 )
} ;
2020-02-14 14:01:45 +08:00
2020-02-20 07:37:23 +08:00
for ( int i = 0 ; i < 20 ; i + + )
2020-02-14 14:01:45 +08:00
{
2020-02-20 07:37:23 +08:00
playlist . Items . Add ( new PlaylistItem
2020-02-14 14:01:45 +08:00
{
2020-02-20 07:37:23 +08:00
ID = i ,
Beatmap = { Value = new TestBeatmap ( new OsuRuleset ( ) . RulesetInfo ) . BeatmapInfo } ,
Ruleset = { Value = new OsuRuleset ( ) . RulesetInfo } ,
RequiredMods =
{
new OsuModHardRock ( ) ,
new OsuModDoubleTime ( ) ,
new OsuModAutoplay ( )
}
} ) ;
}
} ) ;
AddUntilStep ( "wait for items to load" , ( ) = > playlist . ItemMap . Values . All ( i = > i . IsLoaded ) ) ;
}
2020-06-02 13:31:43 +08:00
private void createPlaylist ( params BeatmapInfo [ ] beatmaps )
{
AddStep ( "create playlist" , ( ) = >
{
Child = playlist = new TestPlaylist ( false , false )
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
Size = new Vector2 ( 500 , 300 )
} ;
int index = 0 ;
foreach ( var b in beatmaps )
{
playlist . Items . Add ( new PlaylistItem
{
ID = index + + ,
Beatmap = { Value = b } ,
Ruleset = { Value = new OsuRuleset ( ) . RulesetInfo } ,
RequiredMods =
{
new OsuModHardRock ( ) ,
new OsuModDoubleTime ( ) ,
new OsuModAutoplay ( )
}
} ) ;
}
} ) ;
AddUntilStep ( "wait for items to load" , ( ) = > playlist . ItemMap . Values . All ( i = > i . IsLoaded ) ) ;
}
2020-02-20 07:37:23 +08:00
private class TestPlaylist : DrawableRoomPlaylist
{
public new IReadOnlyDictionary < PlaylistItem , RearrangeableListItem < PlaylistItem > > ItemMap = > base . ItemMap ;
public TestPlaylist ( bool allowEdit , bool allowSelection )
: base ( allowEdit , allowSelection )
{
2020-02-14 14:01:45 +08:00
}
2020-02-20 07:37:23 +08:00
}
2020-02-14 14:01:45 +08:00
}
}