WPFを使用して、WiiリモコンでゲームSimonの非常に単純なバージョンを作成しようとしています。私がこだわっているのは、GUIがシーケンスの表示を完了するまでプログラムがブロックする、ターンベースにする方法です。
これが私がこれまでに持っているコードです(主にここでの答えに基づいています:WPF-シーケンシャルアニメーションの簡単な例):
public partial class Window1 : Window
{
public enum SimonSquare { BLUE = 1, GREEN = 3, RED = 5, YELLOW = 7 };
List<int> _correctSequence;
int _currentLevel = 1;
Random random = new Random();
Wiimote _wiiMote;
List<int> _squaresEntered;
private IEnumerator<Action> _actions;
Rectangle blueRect;
Rectangle redRect;
Rectangle greenRect;
Rectangle yellowRect;
AutoResetEvent autoEvent;
public Window1()
{
InitializeComponent();
blueRect = new Rectangle() { Fill =
System.Windows.Media.Brushes.Blue, Name = "Blue"};
redRect = new Rectangle() { Fill =
System.Windows.Media.Brushes.Red, Name = "Red" };
greenRect = new Rectangle() { Fill =
System.Windows.Media.Brushes.Green, Name = "Green" };
yellowRect = new Rectangle() { Fill =
System.Windows.Media.Brushes.Yellow, Name = "Yellow" };
UniformGrid1.Children.Add(new Rectangle() { Fill =
System.Windows.Media.Brushes.LightGray });
UniformGrid1.Children.Add(blueRect);
UniformGrid1.Children.Add(new Rectangle() { Fill =
System.Windows.Media.Brushes.LightGray });
UniformGrid1.Children.Add(redRect);
UniformGrid1.Children.Add(new Rectangle() { Fill =
System.Windows.Media.Brushes.LightGray });
UniformGrid1.Children.Add(greenRect);
UniformGrid1.Children.Add(new Rectangle() { Fill =
System.Windows.Media.Brushes.LightGray });
UniformGrid1.Children.Add(yellowRect);
UniformGrid1.Children.Add(new Rectangle() { Fill =
System.Windows.Media.Brushes.LightGray });
//connectWiiRemote();
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
_actions = AnimationSequence().GetEnumerator();
autoEvent = new AutoResetEvent(false);
Thread thread = new Thread(RunNextAction);
thread.Start();
autoEvent.WaitOne(); // need to block here somehow!
int x = 5;
}
IEnumerable<Action> AnimationSequence()
{
getSequence();
foreach(int square in _correctSequence)
{
if(square == (int) SimonSquare.BLUE)
yield return () => animateCell(blueRect, Colors.Blue);
else if(square == (int) SimonSquare.RED)
yield return () => animateCell(redRect, Colors.Red);
else if (square == (int)SimonSquare.GREEN)
yield return () => animateCell(greenRect, Colors.Green);
else if (square == (int)SimonSquare.YELLOW)
yield return () => animateCell(yellowRect, Colors.Yellow);
}
}
private void animateCell(Rectangle rectangle, Color fromColor)
{
this.Dispatcher.BeginInvoke(new Action(delegate
{
Color toColor = Colors.White;
ColorAnimation ani = new ColorAnimation(toColor,
new Duration(TimeSpan.FromMilliseconds(300)));
ani.AutoReverse = true;
SolidColorBrush newBrush = new SolidColorBrush(fromColor);
ani.BeginTime = TimeSpan.FromSeconds(2);
rectangle.Fill = newBrush;
ani.Completed += (s, e) => RunNextAction();
newBrush.BeginAnimation(SolidColorBrush.ColorProperty, ani);
}));
}
private void RunNextAction()
{
if (_actions.MoveNext())
_actions.Current();
else
{
autoEvent.Set();
_currentLevel++;
}
}
private void getSequence()
{
_correctSequence = new List<int>();
int[] values =
Enum.GetValues(typeof(SimonSquare)).Cast<int>().ToArray();
for (int i = 0; i < _currentLevel + 2; i++)
{
_correctSequence.Add(values[random.Next(values.Length)]);
}
}
}
ただし、autoSetのwaitOne/setは正しく機能していません。現在、RunNextActionを1回呼び出しますが、その後、waitOneを無期限にブロックします。私は何が間違っているのですか?
編集:質問を言い換えてみましょう。ThreadingとAutoResetEventを取得すると、Window_Loadedに次のようになります。
private void Window_Loaded(object sender, RoutedEventArgs e)
{
_actions = AnimationSequence().GetEnumerator();
RunNextAction(); // shows all of the flashing squares
// need to wait here until the flashing squares are all shown
// process player's events etc.
}
上記のコードを実行すると、RunNextActionが1回呼び出され、すべての正方形が表示されるまで(独自のスレッドのように見えます)、それ自体が呼び出され続けますが、WindowLoadedメソッドは続行されます。RunNextAction()を呼び出した後、RunNextActionが完全に完了するまで、Window_Loadedをブロックする必要があります。