0

我想制作一个小型命运之轮游戏,但在制作旋转图像的事件时遇到问题

这是我现在拥有的代码,但它似乎只运行 _thread 一次

    private readonly Timer _timer;
    private Thread _thread;
    private int _angle;

    public MainWindow()
    {
        InitializeComponent();
        _timer = new Timer(500);
        _timer.Elapsed += RotateWheelEvent;
        _timer.Start();

        _angle = 0;

        SetupRotationThread();
        _thread.Start();
    }

        public void RotateWheelEvent(object sender, EventArgs args)
    {
        _thread.Abort();
        _angle = _angle + 15;
        _thread.Start();
    }

    public void SetupRotationThread()
    {
        var rotatetransform = new RotateTransform(_angle);

        _thread = new Thread(
            delegate()
                {
                    ImageWheel.Dispatcher.Invoke(
                        System.Windows.Threading.DispatcherPriority.SystemIdle,
                        TimeSpan.FromSeconds(1),
                        new Action(
                            delegate()
                                {
                                    ImageWheel.RenderTransformOrigin = new Point(0.5, 0.5);
                                    ImageWheel.RenderTransform = rotatetransform;
                                }
                            ));
                });
    }
4

2 回答 2

3

您可以只使用 aSystem.Threading.Timer而不是System.Timer每次都生成一个新线程。

        private System.Threading.Timer _timer;
        private int _angle;

        public MainWindow()
        {
            InitializeComponent();
              _timer = new System.Threading.Timer((o) => 
              {
                  _angle += 15;
                  Dispatcher.Invoke(DispatcherPriority.Background, (Action)delegate
                  {
                      ImageWheel.RenderTransformOrigin = new Point(0.5, 0.5);
                      ImageWheel.RenderTransform = new RotateTransform(_angle);
                  }); 
              }, null, 500, 500);
        }

然而,不清楚你为什么要线程化这个,你似乎正在启动一个线程,它只是将代码调用回 UI 线程,我看不到这个线程需要做的任何工作。

所以我认为你需要的就是这个

private readonly Timer _timer;
private int _angle;

    public MainWindow()
    {
        InitializeComponent();
        _timer = new Timer(500);
        _timer.Elapsed += RotateWheelEvent;
        _timer.Start();
        _angle = 0;
    }

    public void RotateWheelEvent(object sender, EventArgs args)
    {
        Dispatcher.Invoke(DispatcherPriority.Background, (Action)delegate
        {
            _angle += 15;
            ImageWheel.RenderTransformOrigin = new Point(0.5, 0.5);
            ImageWheel.RenderTransform = new RotateTransform(_angle);
        }); 
    }
于 2013-01-29T23:38:44.150 回答
1

在最坏的情况下,您应该只DispatcherTimer在代码隐藏中使用 a :

var timer = new DispatcherTimer
{
    Interval = TimeSpan.FromSeconds(0.5)
};

timer.Tick += (s, e) => ((RotateTransform)ImageWheel.RenderTransform).Angle += 15;
timer.IsEnabled = true;

但实际上根本没有必要这样做。Storyboard你可以在你的 XAML 中设置一个。就像是:

<Image x:Name="image" RenderTransformOrigin="0.5,0.5" ...>
    <Image.RenderTransform>
        <RotateTransform/>
    </Image.RenderTransform>
    <Image.Triggers>
        <EventTrigger RoutedEvent="Loaded">
            <BeginStoryboard>
                <Storyboard Duration="00:00:00.5" RepeatBehavior="Forever">
                    <DoubleAnimation Storyboard.TargetName="image" Storyboard.TargetProperty="(Image.RenderTransform).(RotateTransform.Angle)" By="15"/>
                </Storyboard>
            </BeginStoryboard>
        </EventTrigger>
    </Image.Triggers>
</Image>

其中一些语法可能不太正确(在 linux 上因此无法检查),但您明白了。你把一些非常简单的事情过于复杂了。

于 2013-01-30T00:17:59.980 回答