// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using osu.Framework.Allocation; 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.Sprites; using osuTK; using osuTK.Input; namespace osu.Game.Screens.Edit.Compose.Components { public class SelectionBox : CompositeDrawable { public const float BORDER_RADIUS = 3; public Func OnRotation; public Func OnScale; public Func OnFlip; public Func OnReverse; public Action OperationStarted; public Action OperationEnded; private bool canReverse; /// /// Whether pattern reversing support should be enabled. /// public bool CanReverse { get => canReverse; set { if (canReverse == value) return; canReverse = value; recreate(); } } private bool canRotate; /// /// Whether rotation support should be enabled. /// public bool CanRotate { get => canRotate; set { if (canRotate == value) return; canRotate = value; recreate(); } } private bool canScaleX; /// /// Whether vertical scale support should be enabled. /// public bool CanScaleX { get => canScaleX; set { if (canScaleX == value) return; canScaleX = value; recreate(); } } private bool canScaleY; /// /// Whether horizontal scale support should be enabled. /// public bool CanScaleY { get => canScaleY; set { if (canScaleY == value) return; canScaleY = value; recreate(); } } private string text; public string Text { get => text; set { if (value == text) return; text = value; if (selectionDetailsText != null) selectionDetailsText.Text = value; } } private Container dragHandles; private FillFlowContainer buttons; private OsuSpriteText selectionDetailsText; [Resolved] private OsuColour colours { get; set; } [BackgroundDependencyLoader] private void load() => recreate(); protected override bool OnKeyDown(KeyDownEvent e) { if (e.Repeat || !e.ControlPressed) return false; bool runOperationFromHotkey(Func operation) { operationStarted(); bool result = operation?.Invoke() ?? false; operationEnded(); return result; } switch (e.Key) { case Key.G: return CanReverse && runOperationFromHotkey(OnReverse); case Key.H: return CanScaleX && runOperationFromHotkey(() => OnFlip?.Invoke(Direction.Horizontal) ?? false); case Key.J: return CanScaleY && runOperationFromHotkey(() => OnFlip?.Invoke(Direction.Vertical) ?? false); } return base.OnKeyDown(e); } private void recreate() { if (LoadState < LoadState.Loading) return; InternalChildren = new Drawable[] { new Container { Name = "info text", AutoSizeAxes = Axes.Both, Children = new Drawable[] { new Box { Colour = colours.YellowDark, RelativeSizeAxes = Axes.Both, }, selectionDetailsText = new OsuSpriteText { Padding = new MarginPadding(2), Colour = colours.Gray0, Font = OsuFont.Default.With(size: 11), Text = text, } } }, new Container { Masking = true, BorderThickness = BORDER_RADIUS, BorderColour = colours.YellowDark, RelativeSizeAxes = Axes.Both, Children = new Drawable[] { new Box { RelativeSizeAxes = Axes.Both, AlwaysPresent = true, Alpha = 0 }, } }, dragHandles = new Container { RelativeSizeAxes = Axes.Both, // ensures that the centres of all drag handles line up with the middle of the selection box border. Padding = new MarginPadding(BORDER_RADIUS / 2) }, buttons = new FillFlowContainer { Y = 20, AutoSizeAxes = Axes.Both, Direction = FillDirection.Horizontal, Anchor = Anchor.BottomCentre, Origin = Anchor.Centre } }; if (CanScaleX) addXScaleComponents(); if (CanScaleX && CanScaleY) addFullScaleComponents(); if (CanScaleY) addYScaleComponents(); if (CanRotate) addRotationComponents(); if (CanReverse) addButton(FontAwesome.Solid.Backward, "Reverse pattern (Ctrl-G)", () => OnReverse?.Invoke()); } private void addRotationComponents() { const float separation = 40; addButton(FontAwesome.Solid.Undo, "Rotate 90 degrees counter-clockwise", () => OnRotation?.Invoke(-90)); addButton(FontAwesome.Solid.Redo, "Rotate 90 degrees clockwise", () => OnRotation?.Invoke(90)); AddRangeInternal(new Drawable[] { new Box { Depth = float.MaxValue, Colour = colours.YellowLight, Blending = BlendingParameters.Additive, Alpha = 0.3f, Size = new Vector2(BORDER_RADIUS, separation), Anchor = Anchor.TopCentre, Origin = Anchor.BottomCentre, }, new SelectionBoxDragHandleButton(FontAwesome.Solid.Redo, "Free rotate") { Anchor = Anchor.TopCentre, Y = -separation, HandleDrag = e => OnRotation?.Invoke(convertDragEventToAngleOfRotation(e)), OperationStarted = operationStarted, OperationEnded = operationEnded } }); } private void addYScaleComponents() { addButton(FontAwesome.Solid.ArrowsAltV, "Flip vertically (Ctrl-J)", () => OnFlip?.Invoke(Direction.Vertical)); addDragHandle(Anchor.TopCentre); addDragHandle(Anchor.BottomCentre); } private void addFullScaleComponents() { addDragHandle(Anchor.TopLeft); addDragHandle(Anchor.TopRight); addDragHandle(Anchor.BottomLeft); addDragHandle(Anchor.BottomRight); } private void addXScaleComponents() { addButton(FontAwesome.Solid.ArrowsAltH, "Flip horizontally (Ctrl-H)", () => OnFlip?.Invoke(Direction.Horizontal)); addDragHandle(Anchor.CentreLeft); addDragHandle(Anchor.CentreRight); } private void addButton(IconUsage icon, string tooltip, Action action) { buttons.Add(new SelectionBoxDragHandleButton(icon, tooltip) { OperationStarted = operationStarted, OperationEnded = operationEnded, Action = action }); } private void addDragHandle(Anchor anchor) => dragHandles.Add(new SelectionBoxDragHandle { Anchor = anchor, HandleDrag = e => OnScale?.Invoke(e.Delta, anchor), OperationStarted = operationStarted, OperationEnded = operationEnded }); private int activeOperations; private float convertDragEventToAngleOfRotation(DragEvent e) { // Adjust coordinate system to the center of SelectionBox float startAngle = MathF.Atan2(e.LastMousePosition.Y - DrawHeight / 2, e.LastMousePosition.X - DrawWidth / 2); float endAngle = MathF.Atan2(e.MousePosition.Y - DrawHeight / 2, e.MousePosition.X - DrawWidth / 2); return (endAngle - startAngle) * 180 / MathF.PI; } private void operationEnded() { if (--activeOperations == 0) OperationEnded?.Invoke(); } private void operationStarted() { if (activeOperations++ == 0) OperationStarted?.Invoke(); } } }