1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-27 07:53:06 +08:00
osu-lazer/osu.Game/Collections/DrawableCollectionListItem.cs

242 lines
8.7 KiB
C#
Raw Normal View History

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-06-17 15:37:17 +08:00
#nullable disable
2020-09-05 02:52:07 +08:00
using System;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
2020-09-05 02:52:07 +08:00
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
using osu.Framework.Input.Events;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.UserInterface;
using osu.Game.Overlays;
using osuTK;
using osuTK.Graphics;
namespace osu.Game.Collections
{
2020-09-08 15:50:51 +08:00
/// <summary>
/// Visualises a <see cref="BeatmapCollection"/> inside a <see cref="DrawableCollectionList"/>.
/// </summary>
2020-09-05 02:55:43 +08:00
public class DrawableCollectionListItem : OsuRearrangeableListItem<BeatmapCollection>
2020-09-05 02:52:07 +08:00
{
private const float item_height = 35;
2020-09-08 12:02:58 +08:00
private const float button_width = item_height * 0.75f;
2020-09-08 16:05:43 +08:00
/// <summary>
/// Whether the <see cref="BeatmapCollection"/> currently exists inside the <see cref="CollectionManager"/>.
2020-09-08 16:05:43 +08:00
/// </summary>
public IBindable<bool> IsCreated => isCreated;
private readonly Bindable<bool> isCreated = new Bindable<bool>();
2020-09-08 15:50:51 +08:00
/// <summary>
/// Creates a new <see cref="DrawableCollectionListItem"/>.
/// </summary>
/// <param name="item">The <see cref="BeatmapCollection"/>.</param>
/// <param name="isCreated">Whether <paramref name="item"/> currently exists inside the <see cref="CollectionManager"/>.</param>
public DrawableCollectionListItem(BeatmapCollection item, bool isCreated)
2020-09-05 02:52:07 +08:00
: base(item)
{
this.isCreated.Value = isCreated;
ShowDragHandle.BindTo(this.isCreated);
2020-09-05 02:52:07 +08:00
}
protected override Drawable CreateContent() => new ItemContent(Model)
{
IsCreated = { BindTarget = isCreated }
};
2020-09-05 02:52:07 +08:00
2020-09-08 15:50:51 +08:00
/// <summary>
/// The main content of the <see cref="DrawableCollectionListItem"/>.
/// </summary>
2020-09-05 02:52:07 +08:00
private class ItemContent : CircularContainer
{
public readonly Bindable<bool> IsCreated = new Bindable<bool>();
private readonly IBindable<string> collectionName;
2020-09-05 02:52:07 +08:00
private readonly BeatmapCollection collection;
2020-09-09 14:39:15 +08:00
[Resolved(CanBeNull = true)]
private CollectionManager collectionManager { get; set; }
private Container textBoxPaddingContainer;
2020-09-05 02:52:07 +08:00
private ItemTextBox textBox;
public ItemContent(BeatmapCollection collection)
{
this.collection = collection;
RelativeSizeAxes = Axes.X;
Height = item_height;
Masking = true;
collectionName = collection.Name.GetBoundCopy();
2020-09-05 02:52:07 +08:00
}
[BackgroundDependencyLoader]
2022-01-15 08:06:39 +08:00
private void load()
2020-09-05 02:52:07 +08:00
{
Children = new Drawable[]
{
new DeleteButton(collection)
{
Anchor = Anchor.CentreRight,
Origin = Anchor.CentreRight,
IsCreated = { BindTarget = IsCreated },
2020-09-05 02:52:07 +08:00
IsTextBoxHovered = v => textBox.ReceivePositionalInputAt(v)
},
textBoxPaddingContainer = new Container
2020-09-05 02:52:07 +08:00
{
RelativeSizeAxes = Axes.Both,
2020-09-08 12:02:58 +08:00
Padding = new MarginPadding { Right = button_width },
2020-09-05 02:52:07 +08:00
Children = new Drawable[]
{
textBox = new ItemTextBox
{
RelativeSizeAxes = Axes.Both,
Size = Vector2.One,
CornerRadius = item_height / 2,
PlaceholderText = IsCreated.Value ? string.Empty : "Create a new collection"
2020-09-05 02:52:07 +08:00
},
}
},
};
}
protected override void LoadComplete()
{
base.LoadComplete();
// Bind late, as the collection name may change externally while still loading.
textBox.Current = collection.Name;
collectionName.BindValueChanged(_ => createNewCollection(), true);
IsCreated.BindValueChanged(created => textBoxPaddingContainer.Padding = new MarginPadding { Right = created.NewValue ? button_width : 0 }, true);
}
private void createNewCollection()
{
if (IsCreated.Value)
return;
if (string.IsNullOrEmpty(collectionName.Value))
return;
// Add the new collection and disable our placeholder. If all text is removed, the placeholder should not show back again.
2020-09-09 14:39:15 +08:00
collectionManager?.Collections.Add(collection);
textBox.PlaceholderText = string.Empty;
// When this item changes from placeholder to non-placeholder (via changing containers), its textbox will lose focus, so it needs to be re-focused.
Schedule(() => GetContainingInputManager().ChangeFocus(textBox));
IsCreated.Value = true;
}
2020-09-05 02:52:07 +08:00
}
private class ItemTextBox : OsuTextBox
{
protected override float LeftRightPadding => item_height / 2;
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
BackgroundUnfocused = colours.GreySeaFoamDarker.Darken(0.5f);
BackgroundFocused = colours.GreySeaFoam;
2020-09-05 02:52:07 +08:00
}
}
2020-09-07 21:47:19 +08:00
public class DeleteButton : CompositeDrawable
2020-09-05 02:52:07 +08:00
{
public readonly IBindable<bool> IsCreated = new Bindable<bool>();
2020-09-05 02:52:07 +08:00
public Func<Vector2, bool> IsTextBoxHovered;
[Resolved(CanBeNull = true)]
private IDialogOverlay dialogOverlay { get; set; }
2020-09-05 02:52:07 +08:00
2020-09-09 14:39:15 +08:00
[Resolved(CanBeNull = true)]
private CollectionManager collectionManager { get; set; }
2020-09-05 02:52:07 +08:00
private readonly BeatmapCollection collection;
private Drawable fadeContainer;
2020-09-05 02:52:07 +08:00
private Drawable background;
public DeleteButton(BeatmapCollection collection)
{
this.collection = collection;
2020-09-08 12:02:58 +08:00
RelativeSizeAxes = Axes.Y;
Width = button_width + item_height / 2; // add corner radius to cover with fill
2020-09-05 02:52:07 +08:00
}
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
InternalChild = fadeContainer = new Container
2020-09-05 02:52:07 +08:00
{
RelativeSizeAxes = Axes.Both,
Alpha = 0.1f,
Children = new[]
2020-09-05 02:52:07 +08:00
{
background = new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colours.Red
},
new SpriteIcon
{
Anchor = Anchor.CentreRight,
Origin = Anchor.Centre,
X = -button_width * 0.6f,
Size = new Vector2(10),
Icon = FontAwesome.Solid.Trash
}
2020-09-05 02:52:07 +08:00
}
};
}
protected override void LoadComplete()
{
base.LoadComplete();
IsCreated.BindValueChanged(created => Alpha = created.NewValue ? 1 : 0, true);
}
2020-09-05 02:52:07 +08:00
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => base.ReceivePositionalInputAt(screenSpacePos) && !IsTextBoxHovered(screenSpacePos);
protected override bool OnHover(HoverEvent e)
{
fadeContainer.FadeTo(1f, 100, Easing.Out);
2020-09-05 02:52:07 +08:00
return false;
}
protected override void OnHoverLost(HoverLostEvent e)
{
fadeContainer.FadeTo(0.1f, 100);
2020-09-05 02:52:07 +08:00
}
protected override bool OnClick(ClickEvent e)
{
background.FlashColour(Color4.White, 150);
if (collection.BeatmapHashes.Count == 0)
deleteCollection();
else
dialogOverlay?.Push(new DeleteCollectionDialog(collection, deleteCollection));
2020-09-05 02:52:07 +08:00
return true;
}
2020-09-09 14:39:15 +08:00
private void deleteCollection() => collectionManager?.Collections.Remove(collection);
2020-09-05 02:52:07 +08:00
}
}
}