// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using osu.Framework.Extensions.Color4Extensions; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Shapes; using osu.Game.Graphics.Backgrounds; using OpenTK.Graphics; using osu.Game.Beatmaps.ControlPoints; using osu.Framework.Audio.Track; namespace osu.Game.Rulesets.Taiko.Objects.Drawables.Pieces { /// /// A circle piece which is used uniformly through osu!taiko to visualise hitobjects. /// /// Note that this can actually be non-circle if the width is changed. See /// for a usage example. /// /// public class CirclePiece : TaikoPiece { public const float SYMBOL_SIZE = TaikoHitObject.DEFAULT_CIRCLE_DIAMETER * 0.45f; public const float SYMBOL_BORDER = 8; public const float SYMBOL_INNER_SIZE = SYMBOL_SIZE - 2 * SYMBOL_BORDER; private const double pre_beat_transition_time = 80; /// /// The colour of the inner circle and outer glows. /// public override Color4 AccentColour { get { return base.AccentColour; } set { base.AccentColour = value; background.Colour = AccentColour; resetEdgeEffects(); } } /// /// Whether Kiai mode effects are enabled for this circle piece. /// public override bool KiaiMode { get { return base.KiaiMode; } set { base.KiaiMode = value; resetEdgeEffects(); } } protected override Container Content => content; private readonly Container content; private readonly Container background; public Box FlashBox; public CirclePiece(bool isStrong = false) { EarlyActivationMilliseconds = pre_beat_transition_time; AddInternal(new Drawable[] { background = new CircularContainer { Name = "Background", Anchor = Anchor.Centre, Origin = Anchor.Centre, RelativeSizeAxes = Axes.Both, Masking = true, Children = new Drawable[] { new Box { Anchor = Anchor.Centre, Origin = Anchor.Centre, RelativeSizeAxes = Axes.Both, }, new Triangles { Anchor = Anchor.Centre, Origin = Anchor.Centre, RelativeSizeAxes = Axes.Both, ColourLight = Color4.White, ColourDark = Color4.White.Darken(0.1f) } } }, new CircularContainer { Name = "Ring", Anchor = Anchor.Centre, Origin = Anchor.Centre, RelativeSizeAxes = Axes.Both, BorderThickness = 8, BorderColour = Color4.White, Masking = true, Children = new[] { FlashBox = new Box { Anchor = Anchor.Centre, Origin = Anchor.Centre, RelativeSizeAxes = Axes.Both, Colour = Color4.White, BlendingMode = BlendingMode.Additive, Alpha = 0, AlwaysPresent = true } } }, content = new Container { RelativeSizeAxes = Axes.Both, Name = "Content", Anchor = Anchor.Centre, Origin = Anchor.Centre, } }); if (isStrong) { Size *= TaikoHitObject.STRONG_CIRCLE_DIAMETER_SCALE; //default for symbols etc. Content.Scale *= TaikoHitObject.STRONG_CIRCLE_DIAMETER_SCALE; } } protected override void Update() { base.Update(); //we want to allow for width of content to remain mapped to the area inside us, regardless of the scale applied above. Content.Width = 1 / Content.Scale.X; } private const float edge_alpha_kiai = 0.5f; private void resetEdgeEffects() { background.EdgeEffect = new EdgeEffectParameters { Type = EdgeEffectType.Glow, Colour = AccentColour.Opacity(KiaiMode ? edge_alpha_kiai : 1f), Radius = KiaiMode ? 32 : 8 }; } protected override void OnNewBeat(int beatIndex, TimingControlPoint timingPoint, EffectControlPoint effectPoint, TrackAmplitudes amplitudes) { if (!effectPoint.KiaiMode) return; if (beatIndex % (int)timingPoint.TimeSignature != 0) return; double duration = timingPoint.BeatLength * 2; background.FadeEdgeEffectTo(1, pre_beat_transition_time, EasingTypes.OutQuint); using (background.BeginDelayedSequence(pre_beat_transition_time)) background.FadeEdgeEffectTo(edge_alpha_kiai, duration, EasingTypes.OutQuint); } } }