本文介绍了WPF:动画不流畅的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我是一个动画的TextBlock 。在60秒内,它增加字号从8磅到200PT。一切工作正常,但我的动画正在上下位作为文本的增长。为什么会出现这种情况,是有可能避免这种情况?

I am animating a TextBlock. In 60 seconds, it increases FontSize from 8pt to 200pt. Everything is working fine, except that my animation is moving up and down a bit as the text grows. Why is this happening and is it possible to avoid this?

我有一个非常简单的XAML文件:

I have a very simple XAML file:

<Window x:Class="Timer.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Width="800"
        Height="500"
        Title="MainWindow"
        Loaded="Window_Loaded">

    <Grid>

        <TextBlock
            Name="TimerTextBlock"
            HorizontalAlignment="Center"
            VerticalAlignment="Center"
            Text="00h : 00m : 00.000s" />

    </Grid>

</Window>

和同样简单code-背后:

And equally simple code-behind:

public partial class MainWindow : Window
{
    private const string timerFormat = "{0:hh'h : 'mm'm : 'ss'.'fff's'}";
    private DispatcherTimer dispatcherTimer;
    private DateTime targetTime;

    public MainWindow()
    {
        InitializeComponent();
    }

    private void Window_Loaded(object sender, RoutedEventArgs e)
    {
        targetTime = DateTime.Now.AddSeconds(60);
        double totalTime = targetTime.Subtract(DateTime.Now).TotalMilliseconds;

        DoubleAnimation animation = new DoubleAnimation();
        animation.From = TimerTextBlock.FontSize;
        animation.To = 200;
        animation.Duration = new Duration(targetTime.Subtract(DateTime.Now));
        TimerTextBlock.BeginAnimation(TextBlock.FontSizeProperty, animation);

        dispatcherTimer = new DispatcherTimer();
        dispatcherTimer.Interval = TimeSpan.FromMilliseconds(1);
        dispatcherTimer.Tick += new EventHandler(dispatcherTimer_Tick);
        dispatcherTimer.Start();
    }

    private void dispatcherTimer_Tick(object sender, EventArgs e)
    {
        if (DateTime.Compare(targetTime, DateTime.Now) > 0)
        {
            TimerTextBlock.Text =
                string.Format(timerFormat, targetTime.Subtract(DateTime.Now));
        }
    }
}

感谢您对所有的澄清。

Thank you for all the clarifications.

推荐答案

您的垂直弹跳的问题是由于字体渲染四舍五入。具体来说,WPF将避免子像素字体的高度,以使字体平滑。避免这种情况的一种方法是将文本转换为路径的几何形状,然后用一个尺度变换的动画吧。

Your vertical jumping problem is due to font rendering rounding. Specifically, WPF will avoid subpixel font height in order to enable font smoothing. One way to avoid this is to convert your text into a path geometry and then use a scale transform to animate it.

下面就是你们的榜样的替代版本没有跳跃。新的XAML是:

Here is an alternate version of your example without the jumping. The new XAML is:

<Grid>
    <Path Name="Path" HorizontalAlignment="Center" VerticalAlignment="Center"/>
</Grid>

和新的code当加载窗口:

and the new code when you load the window:

SetText("");
var transform = new ScaleTransform(1, 1);
Path.LayoutTransform = transform;
var animationX = new DoubleAnimation(1, 10, new Duration(TimeSpan.FromSeconds(60)));
transform.BeginAnimation(ScaleTransform.ScaleXProperty, animationX);
var animationY = new DoubleAnimation(1, 10, new Duration(TimeSpan.FromSeconds(60)));
transform.BeginAnimation(ScaleTransform.ScaleYProperty, animationY);

和设置被anmiated文本的新方法:

and a new method to set the text that is anmiated:

private void SetText(string text)
{
    var formatted = new FormattedText(text, CultureInfo.CurrentCulture, FlowDirection.LeftToRight, new Typeface("Lucida Console"), 12, Brushes.Black);
    Path.Data = formatted.BuildGeometry(new Point(0, 0));
    Path.Fill = Brushes.Black;
}

和您从计时器事件处理程序调用的setText。

and you have call SetText from your timer event handler.

请注意,为了避免横向跳动,你必须使用一个固定长度的文本字符串和等宽字体。

Note that to avoid horizontal jumpiness, you have to use a fixed-length text string and a constant-width font.

这篇关于WPF:动画不流畅的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!

08-21 09:31