1
0
mirror of https://github.com/ppy/osu.git synced 2026-05-17 06:32:36 +08:00
Files
osu-lazer/osu.Game/Rulesets/Edit/ExpandingToolboxContainer.cs
T
2025-06-11 09:48:51 +02:00

132 lines
4.3 KiB
C#

// 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.
using System;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Input.Events;
using osu.Framework.Utils;
using osu.Game.Configuration;
using osu.Game.Graphics.Containers;
using osu.Game.Screens.Edit;
using osuTK;
namespace osu.Game.Rulesets.Edit
{
public partial class ExpandingToolboxContainer : ExpandingContainer
{
protected override double HoverExpansionDelay => 250;
protected override bool ExpandOnHover => expandOnHover;
private readonly Bindable<bool> contractSidebars = new Bindable<bool>();
private bool expandOnHover;
private OffsetMaintainingScrollContainer scrollContainer = null!;
[Resolved]
private Editor? editor { get; set; }
public ExpandingToolboxContainer(float contractedWidth, float expandedWidth)
: base(contractedWidth, expandedWidth)
{
RelativeSizeAxes = Axes.Y;
FillFlow.Spacing = new Vector2(5);
FillFlow.Padding = new MarginPadding { Vertical = 5 };
Expanded.Value = true;
}
[BackgroundDependencyLoader]
private void load(OsuConfigManager config)
{
config.BindWith(OsuSetting.EditorContractSidebars, contractSidebars);
}
protected override OsuScrollContainer CreateScrollContainer() => scrollContainer = new OffsetMaintainingScrollContainer();
protected override void LoadComplete()
{
base.LoadComplete();
var inputManager = GetContainingInputManager();
if (inputManager != null)
{
Expanded.BindValueChanged(_ =>
{
// When state changes from expanded -> collapsed the mouse is no longer within the toolbox so there would be no
// hovered children if we used the mouse position directly
var position = new Vector2(ScreenSpaceDrawQuad.Centre.X, inputManager.CurrentState.Mouse.Position.Y);
scrollContainer.TargetDrawable = Children.FirstOrDefault(it => it.Contains(position));
});
}
}
protected override void Update()
{
base.Update();
bool requireContracting = contractSidebars.Value || editor?.DrawSize.X / editor?.DrawSize.Y < 1.7f;
if (expandOnHover != requireContracting)
{
expandOnHover = requireContracting;
Expanded.Value = !expandOnHover;
}
}
protected override bool OnMouseDown(MouseDownEvent e) => true;
protected override bool OnClick(ClickEvent e) => true;
private partial class OffsetMaintainingScrollContainer : OsuScrollContainer
{
private Drawable? targetDrawable;
private float targetPosition;
public Drawable? TargetDrawable
{
get => targetDrawable;
set
{
targetDrawable = value;
if (value != null)
targetPosition = ToLocalSpace(value.ScreenSpaceDrawQuad.TopLeft).Y;
}
}
protected override void UpdateAfterChildren()
{
if (targetDrawable != null)
{
float currentPosition = ToLocalSpace(targetDrawable.ScreenSpaceDrawQuad.TopLeft).Y;
if (!Precision.AlmostEquals(targetPosition, currentPosition))
{
double offset = currentPosition - targetPosition;
double scrollTarget = Math.Clamp(Current + offset, 0, ScrollableExtent);
ScrollTo(scrollTarget, false, double.PositiveInfinity);
}
}
base.UpdateAfterChildren();
}
protected override void OnUserScroll(double value, bool animated = true, double? distanceDecay = null)
{
targetDrawable = null;
base.OnUserScroll(value, animated, distanceDecay);
}
}
}
}