WinUI3 模拟 iPad 高级感动画:高斯模糊渐变 + 侧边划入

手机录制的效果掉帧,实际的视觉效果极其丝滑流畅。

这个效果模拟了ipad里面的 右滑屏幕 时的效果,主页高斯模糊渐变,左侧工具条滑动进入。

高斯模糊动画:

cs 复制代码
private void Canvas_Draw(CanvasControl sender, CanvasDrawEventArgs args)
{
    var ds = args.DrawingSession;

    if (sourceImage == null)
        return;  
    if (isAnimating)
    {
        UpdateAnimation();
    } 
    DrawBlurredImage(ds, sender); 
    sender.Invalidate();
}

private void UpdateAnimation()
{
    float deltaTime = 1f / 60f; // 假设60fps
    animationProgress += (deltaTime / animationDuration) * animationSpeed;

    if (isReverseAnimation)
    {
        // 反向动画:从目标值回到0
        float t = Math.Clamp(animationProgress, 0f, 1f);
        float easedT = EaseInOutCubic(t);
        currentBlurAmount = targetBlurAmount * (1f - easedT);

        if (animationProgress >= 1f)
        {
            isAnimating = false;
            animationProgress = 0f;
            currentBlurAmount = 0f; 
        }
    }
    else
    {
        // 正向动画:从0到目标值
        float t = Math.Clamp(animationProgress, 0f, 1f);
        float easedT = EaseInOutCubic(t);
        currentBlurAmount = targetBlurAmount * easedT;

        if (animationProgress >= 1f)
        {
            isAnimating = false;
            animationProgress = 0f;
            currentBlurAmount = targetBlurAmount; 
        }
    } 
}

private void DrawBlurredImage(CanvasDrawingSession ds, CanvasControl sender)
{
    if (sourceImage == null)
        return;

    var canvasWidth = (float)sender.ActualWidth;
    var canvasHeight = (float)sender.ActualHeight;

    // 计算图像居中位置
    float imageWidth = (float)sourceImage.Size.Width;
    float imageHeight = (float)sourceImage.Size.Height;
    float x = (canvasWidth - imageWidth) / 2;
    float y = (canvasHeight - imageHeight) / 2;

    if (currentBlurAmount > 0.1f)
    {
        // 应用高斯模糊效果
        using (var blurEffect = new GaussianBlurEffect
        {
            Source = sourceImage,
            BlurAmount = currentBlurAmount,
            Optimization = EffectOptimization.Speed,
            BorderMode = EffectBorderMode.Soft
        })
        {
            ds.DrawImage(blurEffect, x, y);
        }
    }
    else
    {
        // 模糊值太小时直接绘制原图
        ds.DrawImage(sourceImage, x, y);
    }
}

左侧划入动画:

cs 复制代码
// 侧边栏滑入动画 
private void AnimateSidebarSlideIn()
{
    if (isSidebarAnimated)
        return;

    // 获取 Compositor 和 Visual
    var compositor = ElementCompositionPreview.GetElementVisual(SidebarContainer).Compositor;
    var visual = ElementCompositionPreview.GetElementVisual(SidebarContainer);

    // 创建位移动画
    var offsetAnimation = compositor.CreateVector3KeyFrameAnimation();
    offsetAnimation.Duration = TimeSpan.FromSeconds(0.4);

    // 起始位置
    offsetAnimation.InsertKeyFrame(0.0f, new Vector3(-300, 0, 0));

    // 结束位置,使用自定义缓动函数
    var easing = compositor.CreateCubicBezierEasingFunction(
        new Vector2(0.25f, 0.1f),  
        new Vector2(0.25f, 1.0f)   
    );
    offsetAnimation.InsertKeyFrame(1.0f, new Vector3(20, 0, 0), easing);
 
    visual.StartAnimation("Offset", offsetAnimation);
    isSidebarAnimated = true;
}

// 侧边栏滑出动画(反向) 
private void AnimateSidebarSlideOut()
{
    if (!isSidebarAnimated)
        return;

    // 获取 Compositor 和 Visual
    var compositor = ElementCompositionPreview.GetElementVisual(SidebarContainer).Compositor;
    var visual = ElementCompositionPreview.GetElementVisual(SidebarContainer);

    // 创建位移动画
    var offsetAnimation = compositor.CreateVector3KeyFrameAnimation();
    offsetAnimation.Duration = TimeSpan.FromSeconds(0.4);

    // 起始位置
    offsetAnimation.InsertKeyFrame(0.0f, new Vector3(20, 0, 0));

    // 结束位置,使用自定义缓动函数
    var easing = compositor.CreateCubicBezierEasingFunction(
        new Vector2(0.75f, 0.0f),  
        new Vector2(0.75f, 0.9f)     
    );
    offsetAnimation.InsertKeyFrame(1.0f, new Vector3(-300, 0, 0), easing);
 
    visual.StartAnimation("Offset", offsetAnimation);
    isSidebarAnimated = false;
}

再次来看一下进入进出的效果:

相关推荐
用户21991679703916 小时前
C# 14 中的新增功能
c#
垂葛酒肝汤7 小时前
放置挂机游戏的离线和在线收益unity实现
游戏·unity·c#
爱说实话8 小时前
C# 20260112
开发语言·c#
无风听海9 小时前
C#中实现类的值相等时需要保留null==null为true的语义
开发语言·c#
云草桑9 小时前
海外运单核心泡货计费术语:不计泡、计全泡、比例分泡
c#·asp.net·net·计泡·海运
精神小伙就是猛9 小时前
C# Task/ThreadPool async/await对比Golang GMP
开发语言·golang·c#
工程师00710 小时前
C#状态机
开发语言·c#·状态模式·状态机
开开心心_Every10 小时前
离线黑白照片上色工具:操作简单效果逼真
java·服务器·前端·学习·edge·c#·powerpoint