2019-01-24 08:43:03 +00: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.
2018-04-13 09:19:50 +00:00
2016-10-07 08:07:23 +00:00
using System ;
2016-11-14 08:23:33 +00:00
using osu.Framework.Allocation ;
2017-02-07 12:45:59 +00:00
using osu.Framework.Audio ;
using osu.Framework.Audio.Sample ;
2017-05-23 16:17:09 +00:00
using osu.Framework.Audio.Track ;
2020-03-11 01:18:41 +00:00
using osu.Framework.Extensions.Color4Extensions ;
2016-10-07 08:07:23 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2017-06-20 05:54:23 +00:00
using osu.Framework.Graphics.Shapes ;
2016-10-07 08:07:23 +00:00
using osu.Framework.Graphics.Sprites ;
2016-11-14 08:23:33 +00:00
using osu.Framework.Graphics.Textures ;
2018-10-02 03:02:47 +00:00
using osu.Framework.Input.Events ;
2020-01-09 04:43:44 +00:00
using osu.Framework.Utils ;
2017-05-23 16:17:09 +00:00
using osu.Game.Beatmaps.ControlPoints ;
2016-12-01 11:21:14 +00:00
using osu.Game.Graphics.Backgrounds ;
2017-05-23 03:29:43 +00:00
using osu.Game.Graphics.Containers ;
2020-08-04 12:53:00 +00:00
using osu.Game.Overlays ;
2018-11-20 07:51:59 +00:00
using osuTK ;
using osuTK.Graphics ;
using osuTK.Input ;
2018-04-13 09:19:50 +00:00
2016-11-14 08:23:33 +00:00
namespace osu.Game.Screens.Menu
2016-10-07 08:07:23 +00:00
{
/// <summary>
/// osu! logo and its attachments (pulsing, visualiser etc.)
/// </summary>
2017-05-23 03:29:43 +00:00
public class OsuLogo : BeatSyncedContainer
2016-10-07 08:07:23 +00:00
{
2020-03-11 01:18:41 +00:00
public readonly Color4 OsuPink = Color4Extensions . FromHex ( @"e967a1" ) ;
2018-04-13 09:19:50 +00:00
2017-11-01 11:54:58 +00:00
private const double transition_length = 300 ;
2018-04-13 09:19:50 +00:00
2017-03-23 04:41:50 +00:00
private readonly Sprite logo ;
private readonly CircularContainer logoContainer ;
private readonly Container logoBounceContainer ;
2017-05-23 03:29:43 +00:00
private readonly Container logoBeatContainer ;
2017-05-23 16:45:01 +00:00
private readonly Container logoAmplitudeContainer ;
2017-03-23 04:41:50 +00:00
private readonly Container logoHoverContainer ;
2020-06-08 21:45:40 +00:00
private readonly MenuLogoVisualisation visualizer ;
2018-04-13 09:19:50 +00:00
2017-11-03 08:54:35 +00:00
private readonly IntroSequence intro ;
2018-04-13 09:19:50 +00:00
2021-01-19 08:11:40 +00:00
private Sample sampleClick ;
private Sample sampleBeat ;
2021-02-12 08:22:15 +00:00
private Sample sampleDownbeat ;
2018-04-13 09:19:50 +00:00
2017-03-23 04:41:50 +00:00
private readonly Container colourAndTriangles ;
2017-05-24 04:29:12 +00:00
private readonly Triangles triangles ;
2018-04-13 09:19:50 +00:00
2017-11-25 14:29:08 +00:00
/// <summary>
/// Return value decides whether the logo should play its own sample for the click action.
/// </summary>
public Func < bool > Action ;
2018-04-13 09:19:50 +00:00
2019-04-05 03:02:47 +00:00
/// <summary>
/// The size of the logo Sprite with respect to the scale of its hover and bounce containers.
/// </summary>
/// <remarks>Does not account for the scale of this <see cref="OsuLogo"/></remarks>
2019-03-28 06:40:58 +00:00
public float SizeForFlow = > logo = = null ? 0 : logo . DrawSize . X * logo . Scale . X * logoBounceContainer . Scale . X * logoHoverContainer . Scale . X ;
2018-04-13 09:19:50 +00:00
2019-04-05 04:56:08 +00:00
public bool IsTracking { get ; set ; }
2019-04-05 03:02:47 +00:00
2017-03-23 04:41:50 +00:00
private readonly Sprite ripple ;
2018-04-13 09:19:50 +00:00
2017-03-23 04:41:50 +00:00
private readonly Container rippleContainer ;
2018-04-13 09:19:50 +00:00
2016-12-01 11:21:14 +00:00
public bool Triangles
{
2019-02-28 04:58:19 +00:00
set = > colourAndTriangles . FadeTo ( value ? 1 : 0 , transition_length , Easing . OutQuint ) ;
2016-12-01 11:21:14 +00:00
}
2018-04-13 09:19:50 +00:00
2018-09-26 05:01:15 +00:00
public override bool ReceivePositionalInputAt ( Vector2 screenSpacePos ) = > logoContainer . ReceivePositionalInputAt ( screenSpacePos ) ;
2018-04-13 09:19:50 +00:00
2016-10-07 08:07:23 +00:00
public bool Ripple
{
2019-02-28 04:58:19 +00:00
get = > rippleContainer . Alpha > 0 ;
set = > rippleContainer . FadeTo ( value ? 1 : 0 , transition_length , Easing . OutQuint ) ;
2016-10-07 08:07:23 +00:00
}
2018-04-13 09:19:50 +00:00
2020-11-23 05:27:02 +00:00
private const float visualizer_default_alpha = 0.5f ;
2017-03-23 04:41:50 +00:00
private readonly Box flashLayer ;
2018-04-13 09:19:50 +00:00
2017-05-15 10:36:03 +00:00
private readonly Container impactContainer ;
2018-04-13 09:19:50 +00:00
2017-06-28 17:19:04 +00:00
private const double early_activation = 60 ;
2018-04-13 09:19:50 +00:00
2018-09-08 07:31:00 +00:00
public override bool IsPresent = > base . IsPresent | | Scheduler . HasPendingTasks ;
2016-10-07 08:07:23 +00:00
public OsuLogo ( )
{
2017-06-28 17:19:04 +00:00
EarlyActivationMilliseconds = early_activation ;
2018-04-13 09:19:50 +00:00
2016-10-07 08:07:23 +00:00
Origin = Anchor . Centre ;
2018-04-13 09:19:50 +00:00
2016-10-22 08:50:42 +00:00
AutoSizeAxes = Axes . Both ;
2018-04-13 09:19:50 +00:00
2016-10-07 08:07:23 +00:00
Children = new Drawable [ ]
{
2017-11-03 08:54:35 +00:00
intro = new IntroSequence
{
RelativeSizeAxes = Axes . Both ,
} ,
2017-05-23 06:29:23 +00:00
logoHoverContainer = new Container
2016-10-07 08:07:23 +00:00
{
2016-10-22 08:50:42 +00:00
AutoSizeAxes = Axes . Both ,
2016-10-07 08:07:23 +00:00
Children = new Drawable [ ]
{
2022-04-28 13:58:32 +00:00
logoBounceContainer = new DragContainer
2016-10-07 08:07:23 +00:00
{
2016-10-22 09:40:04 +00:00
AutoSizeAxes = Axes . Both ,
2016-12-01 11:21:14 +00:00
Children = new Drawable [ ]
2016-10-16 17:25:08 +00:00
{
2017-05-23 06:29:23 +00:00
rippleContainer = new Container
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
RelativeSizeAxes = Axes . Both ,
Children = new Drawable [ ]
{
ripple = new Sprite
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2019-08-21 04:29:50 +00:00
Blending = BlendingParameters . Additive ,
2017-05-23 06:29:23 +00:00
Alpha = 0
}
}
} ,
2017-05-23 16:45:01 +00:00
logoAmplitudeContainer = new Container
2016-12-01 11:21:14 +00:00
{
2016-12-01 12:29:14 +00:00
AutoSizeAxes = Axes . Both ,
2016-12-01 11:21:14 +00:00
Children = new Drawable [ ]
{
2017-05-23 16:45:01 +00:00
logoBeatContainer = new Container
2016-12-01 11:21:14 +00:00
{
2017-05-23 03:29:43 +00:00
AutoSizeAxes = Axes . Both ,
2016-12-01 11:52:26 +00:00
Children = new Drawable [ ]
{
2020-06-08 21:45:40 +00:00
visualizer = new MenuLogoVisualisation
2017-06-19 00:33:50 +00:00
{
RelativeSizeAxes = Axes . Both ,
Origin = Anchor . Centre ,
Anchor = Anchor . Centre ,
2020-11-23 05:27:02 +00:00
Alpha = visualizer_default_alpha ,
2017-06-19 00:33:50 +00:00
Size = new Vector2 ( 0.96f )
} ,
2018-03-01 07:01:32 +00:00
new Container
2016-12-01 11:52:26 +00:00
{
2017-05-23 16:45:01 +00:00
AutoSizeAxes = Axes . Both ,
2016-12-01 12:29:14 +00:00
Children = new Drawable [ ]
{
2017-05-23 16:45:01 +00:00
logoContainer = new CircularContainer
2016-12-01 12:29:14 +00:00
{
2017-05-23 03:29:43 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2017-05-23 16:45:01 +00:00
RelativeSizeAxes = Axes . Both ,
Scale = new Vector2 ( 0.88f ) ,
Masking = true ,
2017-05-23 03:29:43 +00:00
Children = new Drawable [ ]
{
2017-05-23 16:45:01 +00:00
colourAndTriangles = new Container
2017-05-23 03:29:43 +00:00
{
RelativeSizeAxes = Axes . Both ,
2017-05-23 16:45:01 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
Children = new Drawable [ ]
{
new Box
{
RelativeSizeAxes = Axes . Both ,
Colour = OsuPink ,
} ,
2017-05-24 04:05:11 +00:00
triangles = new Triangles
2017-05-23 16:45:01 +00:00
{
TriangleScale = 4 ,
2020-03-11 01:18:41 +00:00
ColourLight = Color4Extensions . FromHex ( @"ff7db7" ) ,
ColourDark = Color4Extensions . FromHex ( @"de5b95" ) ,
2017-05-23 16:45:01 +00:00
RelativeSizeAxes = Axes . Both ,
} ,
}
2017-05-23 03:29:43 +00:00
} ,
2017-05-23 16:45:01 +00:00
flashLayer = new Box
2017-05-23 03:29:43 +00:00
{
RelativeSizeAxes = Axes . Both ,
2019-08-21 04:29:50 +00:00
Blending = BlendingParameters . Additive ,
2017-05-23 16:45:01 +00:00
Colour = Color4 . White ,
Alpha = 0 ,
2017-05-23 03:29:43 +00:00
} ,
2017-05-23 16:45:01 +00:00
} ,
2016-12-01 12:29:14 +00:00
} ,
2017-05-23 16:45:01 +00:00
logo = new Sprite
2016-12-01 12:29:14 +00:00
{
2017-05-23 16:45:01 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2016-12-01 12:29:14 +00:00
} ,
2017-05-23 16:45:01 +00:00
}
2016-12-01 11:52:26 +00:00
} ,
2017-05-23 16:45:01 +00:00
impactContainer = new CircularContainer
2017-01-27 04:50:22 +00:00
{
2017-05-23 03:29:43 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2017-05-23 16:45:01 +00:00
Alpha = 0 ,
BorderColour = Color4 . White ,
RelativeSizeAxes = Axes . Both ,
BorderThickness = 10 ,
Masking = true ,
Children = new Drawable [ ]
{
new Box
{
RelativeSizeAxes = Axes . Both ,
AlwaysPresent = true ,
Alpha = 0 ,
}
}
2017-05-23 03:29:43 +00:00
}
}
2017-05-15 10:36:03 +00:00
}
}
2016-10-07 08:07:23 +00:00
}
}
}
}
}
} ;
}
2018-04-13 09:19:50 +00:00
2017-11-13 09:43:05 +00:00
/// <summary>
2019-08-30 17:48:45 +00:00
/// Schedule a new external animation. Handled queueing and finishing previous animations in a sane way.
2017-11-13 09:43:05 +00:00
/// </summary>
/// <param name="action">The animation to be performed</param>
/// <param name="waitForPrevious">If true, the new animation is delayed until all previous transforms finish. If false, existing transformed are cleared.</param>
2017-11-21 02:49:42 +00:00
public void AppendAnimatingAction ( Action action , bool waitForPrevious )
2017-11-13 09:43:05 +00:00
{
2018-01-15 18:42:17 +00:00
void runnableAction ( )
2017-11-13 09:43:05 +00:00
{
if ( waitForPrevious )
this . DelayUntilTransformsFinished ( ) . Schedule ( action ) ;
else
{
ClearTransforms ( ) ;
action ( ) ;
}
2018-01-15 18:42:17 +00:00
}
2018-04-13 09:19:50 +00:00
2017-11-13 09:43:05 +00:00
if ( IsLoaded )
runnableAction ( ) ;
else
2018-01-15 20:27:00 +00:00
Schedule ( runnableAction ) ;
2017-11-13 09:43:05 +00:00
}
2018-04-13 09:19:50 +00:00
2016-11-12 10:44:16 +00:00
[BackgroundDependencyLoader]
2018-08-30 22:04:40 +00:00
private void load ( TextureStore textures , AudioManager audio )
2016-10-07 08:07:23 +00:00
{
2019-05-28 08:06:01 +00:00
sampleClick = audio . Samples . Get ( @"Menu/osu-logo-select" ) ;
sampleBeat = audio . Samples . Get ( @"Menu/osu-logo-heartbeat" ) ;
2021-02-10 09:14:32 +00:00
sampleDownbeat = audio . Samples . Get ( @"Menu/osu-logo-downbeat" ) ;
2018-04-13 09:19:50 +00:00
2018-08-30 22:04:40 +00:00
logo . Texture = textures . Get ( @"Menu/logo" ) ;
ripple . Texture = textures . Get ( @"Menu/logo" ) ;
2016-11-01 14:24:14 +00:00
}
2018-04-13 09:19:50 +00:00
2017-05-23 16:45:01 +00:00
private int lastBeatIndex ;
2018-04-13 09:19:50 +00:00
2020-06-23 04:49:18 +00:00
protected override void OnNewBeat ( int beatIndex , TimingControlPoint timingPoint , EffectControlPoint effectPoint , ChannelAmplitudes amplitudes )
2017-05-23 03:29:43 +00:00
{
2017-05-23 16:17:09 +00:00
base . OnNewBeat ( beatIndex , timingPoint , effectPoint , amplitudes ) ;
2018-04-13 09:19:50 +00:00
2017-05-23 16:45:01 +00:00
lastBeatIndex = beatIndex ;
2018-04-13 09:19:50 +00:00
2021-10-27 04:04:41 +00:00
double beatLength = timingPoint . BeatLength ;
2018-04-13 09:19:50 +00:00
2017-05-23 16:45:01 +00:00
float amplitudeAdjust = Math . Min ( 1 , 0.4f + amplitudes . Maximum ) ;
2018-04-13 09:19:50 +00:00
2017-05-23 16:17:09 +00:00
if ( beatIndex < 0 ) return ;
2018-04-13 09:19:50 +00:00
2017-07-07 05:59:17 +00:00
if ( IsHovered )
2021-02-10 09:14:32 +00:00
{
this . Delay ( early_activation ) . Schedule ( ( ) = >
{
2022-01-22 16:27:27 +00:00
if ( beatIndex % timingPoint . TimeSignature . Numerator = = 0 )
2022-04-15 11:32:41 +00:00
{
sampleDownbeat ? . Play ( ) ;
}
2021-02-10 09:14:32 +00:00
else
2022-04-15 11:32:41 +00:00
{
var channel = sampleBeat . GetChannel ( ) ;
channel . Frequency . Value = 0.95 + RNG . NextDouble ( 0.1 ) ;
channel . Play ( ) ;
}
2021-02-10 09:14:32 +00:00
} ) ;
}
2018-04-13 09:19:50 +00:00
2017-07-16 15:28:20 +00:00
logoBeatContainer
2020-11-23 05:27:02 +00:00
. ScaleTo ( 1 - 0.02f * amplitudeAdjust , early_activation , Easing . Out ) . Then ( )
2017-07-22 18:50:25 +00:00
. ScaleTo ( 1 , beatLength * 2 , Easing . OutQuint ) ;
2018-04-13 09:19:50 +00:00
2017-05-23 06:29:23 +00:00
ripple . ClearTransforms ( ) ;
2017-07-16 15:28:20 +00:00
ripple
. ScaleTo ( logoAmplitudeContainer . Scale )
2017-07-22 18:50:25 +00:00
. ScaleTo ( logoAmplitudeContainer . Scale * ( 1 + 0.04f * amplitudeAdjust ) , beatLength , Easing . OutQuint )
. FadeTo ( 0.15f * amplitudeAdjust ) . FadeOut ( beatLength , Easing . OutQuint ) ;
2018-04-13 09:19:50 +00:00
2017-05-23 16:17:09 +00:00
if ( effectPoint . KiaiMode & & flashLayer . Alpha < 0.4f )
2017-05-23 07:27:01 +00:00
{
flashLayer . ClearTransforms ( ) ;
2017-07-16 15:28:20 +00:00
flashLayer
2020-11-23 05:27:02 +00:00
. FadeTo ( 0.2f * amplitudeAdjust , early_activation , Easing . Out ) . Then ( )
2017-07-16 15:28:20 +00:00
. FadeOut ( beatLength ) ;
2018-04-13 09:19:50 +00:00
2017-07-16 15:28:20 +00:00
visualizer . ClearTransforms ( ) ;
visualizer
2020-11-23 05:27:02 +00:00
. FadeTo ( visualizer_default_alpha * 1.8f * amplitudeAdjust , early_activation , Easing . Out ) . Then ( )
. FadeTo ( visualizer_default_alpha , beatLength ) ;
2017-05-23 07:27:01 +00:00
}
2017-05-23 03:29:43 +00:00
}
2018-04-13 09:19:50 +00:00
2017-11-03 08:54:35 +00:00
public void PlayIntro ( )
{
2017-11-08 05:31:11 +00:00
const double length = 3150 ;
const double fade = 200 ;
2018-04-13 09:19:50 +00:00
2017-11-03 08:54:35 +00:00
logoHoverContainer . FadeOut ( ) . Delay ( length ) . FadeIn ( fade ) ;
intro . Show ( ) ;
intro . Start ( length ) ;
intro . Delay ( length + fade ) . FadeOut ( ) ;
}
2018-04-13 09:19:50 +00:00
2020-08-04 12:53:00 +00:00
[Resolved]
private MusicController musicController { get ; set ; }
2017-05-23 16:45:01 +00:00
protected override void Update ( )
{
base . Update ( ) ;
2018-04-13 09:19:50 +00:00
2017-05-24 04:05:28 +00:00
const float scale_adjust_cutoff = 0.4f ;
2017-05-24 04:05:11 +00:00
const float velocity_adjust_cutoff = 0.98f ;
2017-09-11 09:31:36 +00:00
const float paused_velocity = 0.5f ;
2018-04-13 09:19:50 +00:00
2020-08-07 11:51:56 +00:00
if ( musicController . CurrentTrack . IsRunning )
2017-09-11 09:31:36 +00:00
{
2021-10-27 04:04:41 +00:00
float maxAmplitude = lastBeatIndex > = 0 ? musicController . CurrentTrack . CurrentAmplitudes . Maximum : 0 ;
2020-07-27 06:10:32 +00:00
logoAmplitudeContainer . Scale = new Vector2 ( ( float ) Interpolation . Damp ( logoAmplitudeContainer . Scale . X , 1 - Math . Max ( 0 , maxAmplitude - scale_adjust_cutoff ) * 0.04f , 0.9f , Time . Elapsed ) ) ;
2018-04-13 09:19:50 +00:00
2017-09-11 09:31:36 +00:00
if ( maxAmplitude > velocity_adjust_cutoff )
triangles . Velocity = 1 + Math . Max ( 0 , maxAmplitude - velocity_adjust_cutoff ) * 50 ;
else
triangles . Velocity = ( float ) Interpolation . Damp ( triangles . Velocity , 1 , 0.995f , Time . Elapsed ) ;
}
2017-05-24 04:05:11 +00:00
else
2017-09-11 09:31:36 +00:00
{
triangles . Velocity = paused_velocity ;
}
2017-05-23 16:45:01 +00:00
}
2018-04-13 09:19:50 +00:00
2018-09-26 05:01:15 +00:00
public override bool HandlePositionalInput = > base . HandlePositionalInput & & Action ! = null & & Alpha > 0.2f ;
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override bool OnMouseDown ( MouseDownEvent e )
2016-10-07 08:07:23 +00:00
{
2018-10-02 03:02:47 +00:00
if ( e . Button ! = MouseButton . Left ) return false ;
2018-07-12 03:15:22 +00:00
2017-07-22 18:50:25 +00:00
logoBounceContainer . ScaleTo ( 0.9f , 1000 , Easing . Out ) ;
2016-10-07 08:07:23 +00:00
return true ;
}
2018-04-13 09:19:50 +00:00
2020-01-20 09:17:21 +00:00
protected override void OnMouseUp ( MouseUpEvent e )
2016-10-07 08:07:23 +00:00
{
2020-01-20 09:17:21 +00:00
if ( e . Button ! = MouseButton . Left ) return ;
2018-07-12 03:15:22 +00:00
2017-07-22 18:50:25 +00:00
logoBounceContainer . ScaleTo ( 1f , 500 , Easing . OutElastic ) ;
2016-10-07 08:07:23 +00:00
}
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override bool OnClick ( ClickEvent e )
2016-10-07 08:07:23 +00:00
{
2017-11-25 14:29:08 +00:00
if ( Action ? . Invoke ( ) ? ? true )
sampleClick . Play ( ) ;
2018-04-13 09:19:50 +00:00
2017-02-25 12:12:39 +00:00
flashLayer . ClearTransforms ( ) ;
2017-01-27 04:50:22 +00:00
flashLayer . Alpha = 0.4f ;
2017-07-22 18:50:25 +00:00
flashLayer . FadeOut ( 1500 , Easing . OutExpo ) ;
2016-10-07 08:07:23 +00:00
return true ;
}
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override bool OnHover ( HoverEvent e )
2016-10-07 08:07:23 +00:00
{
2017-07-22 18:50:25 +00:00
logoHoverContainer . ScaleTo ( 1.1f , 500 , Easing . OutElastic ) ;
2016-10-07 08:07:23 +00:00
return true ;
}
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override void OnHoverLost ( HoverLostEvent e )
2016-10-07 08:07:23 +00:00
{
2017-07-22 18:50:25 +00:00
logoHoverContainer . ScaleTo ( 1 , 500 , Easing . OutElastic ) ;
2016-10-07 08:07:23 +00:00
}
2018-04-13 09:19:50 +00:00
2017-05-15 10:36:03 +00:00
public void Impact ( )
{
2017-07-22 18:50:25 +00:00
impactContainer . FadeOutFromOne ( 250 , Easing . In ) ;
2017-05-15 10:36:03 +00:00
impactContainer . ScaleTo ( 0.96f ) ;
impactContainer . ScaleTo ( 1.12f , 250 ) ;
}
2022-04-28 13:58:32 +00:00
private class DragContainer : Container
{
2022-04-29 03:09:11 +00:00
public override bool DragBlocksClick = > false ;
2022-04-28 13:58:32 +00:00
protected override bool OnDragStart ( DragStartEvent e ) = > true ;
protected override void OnDrag ( DragEvent e )
{
Vector2 change = e . MousePosition - e . MouseDownPosition ;
// Diminish the drag distance as we go further to simulate "rubber band" feeling.
change * = change . Length < = 0 ? 0 : MathF . Pow ( change . Length , 0.6f ) / change . Length ;
this . MoveTo ( change ) ;
}
protected override void OnDragEnd ( DragEndEvent e )
{
this . MoveTo ( Vector2 . Zero , 800 , Easing . OutElastic ) ;
base . OnDragEnd ( e ) ;
}
}
2016-10-07 08:07:23 +00:00
}
2017-05-23 17:53:21 +00:00
}