在“【WPF学习】第五十章 故事板”中讨论了如何使用代码创建简单动画,以及如何使用XAML标记构建更复杂的故事板——具有多个动画以及播放控制功能。但有时采用更复杂的故事板例程,并在代码中实现全部复杂功能是合理的。实际上,这种情况十分常见。当需要处理多个动画并且预先不知道将有多少个动画或不知道如何配置动画时,就会遇到这种情况。如果希望在不同的窗口中使用相同的动画,或者只是希望从标记中灵活地分离出所有与动画相关的细节以以方便重用,也会遇到这种情况。

  通过编写代码创建、配置和启动故事板并不难。只需要创建动画和故事板对象,并将动画添加到故事板中,然后启动故事板即可。在动画结束后可响应Storyboard.Completed事件以执行所有清理工作。

  在接下来的示例中,将看到如何创建下图中显示的游戏。在该例中,投下的一系列炸弹的速度始终不断增加。玩家必须单击每个炸弹以逐一拆除。当达到设置的极限时——默认情况下是落下5个炸弹——游戏i结束。

  在这个示例中,投下的每颗炸弹都有自己的包含两个动画的故事板。第一个动画使炸弹下落(通过为Canvas.Top属性应用动画),而第二个动画稍微前后旋转炸弹,使其就有逼真的摆动效果。如果用户单击一颗下落的炸弹,这些动画就会停止,并且会发生另外两个动画,使炸弹倾斜,悄然间离开Canvas面板的侧边。最后,每次结束一个动画,应用程序都会进行检查,一查看该动画是标示炸弹被拆除了还是落下了,并相应地更新计数。

一、创建主窗口

  BombDropper示例中的主窗口十分简单。它包含一个具有两列的Grid面板。在左侧是一个Border元素,该Border元素包含标示游戏表面的Canvas面板:

<Border BorderBrush="SteelBlue" BorderThickness="1" Margin="5">
<Grid>
<Canvas x:Name="canvasBackground" SizeChanged="canvasBackground_SizeChanged" MinWidth="50">
<Canvas.Background>
<RadialGradientBrush>
<GradientStop Color="AliceBlue" Offset="0"></GradientStop>
<GradientStop Color="White" Offset="0.7"></GradientStop>
</RadialGradientBrush>
</Canvas.Background>
</Canvas>
</Grid>
</Border>

  当第一次为Canvas面板设置尺寸或改变尺寸时(当用户改变窗口的尺寸时),会运行下面的代码变设置裁剪区域:

private void canvasBackground_SizeChanged(object sender, SizeChangedEventArgs e)
{
RectangleGeometry rect = new RectangleGeometry();
rect.Rect = new Rect(, , canvasBackground.ActualWidth, canvasBackground.ActualHeight);
canvasBackground.Clip = rect;
}

  这是需要的,否则即使Canvas面板的子元素位于显示区域外,Canvas面板也会绘制其子元素。在投弹游戏中,这会导致炸弹飞出勾勒的Canvas面板轮廓的方框。

  在主窗口的右侧是一个面板,该面板显示游戏统计信息、当前拆除的炸弹数量和落下的炸弹数量,以及一个用于开始游戏的按钮:

<Border Grid.Column="1" BorderBrush="SteelBlue" BorderThickness="1" Margin="5">
<Border.Background>
<RadialGradientBrush GradientOrigin="1,0.7" Center="1,0.7" RadiusX="1" RadiusY="1">
<GradientStop Color="Orange" Offset="0"></GradientStop>
<GradientStop Color="White" Offset="1"></GradientStop>
</RadialGradientBrush>
</Border.Background>
<StackPanel Margin="15" VerticalAlignment="Center" HorizontalAlignment="Center">
<TextBlock FontFamily="Impact" FontSize="35" Foreground="LightSteelBlue">Bomb Dropper</TextBlock>
<TextBlock x:Name="lblRate" Margin="0,30,0,0" TextWrapping="Wrap" FontFamily="Georgia" FontSize="14"></TextBlock>
<TextBlock x:Name="lblSpeed" Margin="0,30" TextWrapping="Wrap" FontFamily="Georgia" FontSize="14"></TextBlock>
<TextBlock x:Name="lblStatus"
TextWrapping="Wrap" FontFamily="Georgia" FontSize="20">No bombs have dropped.</TextBlock>
<Button x:Name="cmdStart" Padding="5" Margin="0,30" Width="80" Content="Start Game" Click="cmdStart_Click"></Button>
</StackPanel> </Border>

二、创建Bomb用户控件

  下一步创建炸弹的图形划线。尽管可使用静态图像(只要具有透明的背景就可以),但使用更灵活的WPF形状总是更好一些。通过使用形状,当改变炸弹尺寸时不会导致变形,并且可以修改图画中的单个部分或为其应用动画。这个动画中显示的炸弹是直接从Miscrosoft Word的在线剪贴画集合绘制的。炸弹剪贴画被转换为XAML,具体方法是首先将剪贴画插入到Word文档中,然后将Word文档保存为XPS文件。

  用于Bomb类的XAML稍微做了简化(删除不必要的用于包含炸弹的额外Canvas元素以及用于缩放炸弹的变换)。然后将XAML插入到新的名为Bomb的用户控件中,通过这种方法,住页面可通过创建Bomb用户控件并将其添加到布局容器(如Canvas面板)中来显示炸弹。

  在单独的用户控件中放置图形,可是的在用户界面中实例化该图形的多个副本更加容易。还可通过为用户控件添加代码来封装相关的功能。在投弹示例中,只需要为代码添加一个细节——跟踪炸弹当时是否正在下落的Boolean属性:

public partial class Bomb : UserControl
{
public Bomb()
{
InitializeComponent();
}
public bool IsFalling
{
get;
set;
}
}

  用于炸弹的标记包含RotateTransform变换,动画代码使用该变换为下落中的炸弹应用摆动效果。尽管可通过编写代码创建并添加这个RotateTransform变换,但在炸弹的XAML文件中定义该变换更加合理:

<UserControl x:Class="BombDropper.Bomb"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
>
<UserControl.RenderTransform>
<TransformGroup>
<RotateTransform Angle="20" CenterX="50" CenterY="50"></RotateTransform>
<ScaleTransform ScaleX="0.5" ScaleY="0.5"></ScaleTransform>
</TransformGroup>
</UserControl.RenderTransform>
<Canvas> <Path Data="M 11.989,50.026 L 24.381,37.08 L 19.097,53.862 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 0.46098,31.997 L 17.945,28.449 L 4.1114,39.19 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 9.9713,7.3517 L 22.075,20.49 L 5.7445,14.16 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 58.484,29.408 L 40.712,31.997 L 57.523,37.367 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 51.663,10.229 L 38.694,22.408 L 55.506,17.325 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 32.354,0.25535 L 31.682,18.092 L 40.039,2.7487 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path
Data="M 105.84,186.87 L 110.84,183.99 L 115.45,180.64 L 119.58,176.9 L 123.33,172.77 L 126.69,168.36 L 129.47,163.76 L 131.88,158.87 L 133.8,153.79 L 135.14,148.51 L 136.01,143.14 L 136.39,137.77 L 136.3,132.21 L 135.53,126.74 L 134.28,121.37 L 132.45,116 L 130.05,110.73 L 128.61,108.14 L 127.17,105.74 L 125.54,103.34 L 123.81,101.14 L 121.98,99.029 L 120.06,97.015 L 118.04,95.097 L 115.93,93.275 L 113.82,91.549 L 111.61,89.919 L 109.3,88.481 L 106.9,87.138 L 104.5,85.891 L 102,84.741 L 99.503,83.782 L 96.909,82.823 L 94.316,82.055 L 91.722,81.48 L 89.032,81.001 L 86.342,80.617 L 83.556,80.329 L 80.867,80.233 L 78.081,80.233 L 75.391,80.329 L 72.605,80.617 L 69.915,81.096 L 67.129,81.672 L 64.44,82.343 L 61.75,83.206 L 59.06,84.165 L 56.37,85.316 L 53.777,86.563 L 48.781,89.44 L 44.17,92.796 L 40.039,96.536 L 36.293,100.66 L 33.027,104.97 L 30.145,109.67 L 27.743,114.56 L 25.918,119.65 L 24.477,124.83 L 23.612,130.2 L 23.324,135.66 L 23.42,141.13 L 24.189,146.59 L 25.438,152.06 L 27.263,157.43 L 29.664,162.7 L 31.105,165.29 L 32.546,167.69 L 34.179,170.09 L 35.909,172.29 L 37.734,174.4 L 39.655,176.42 L 41.672,178.34 L 43.69,180.16 L 45.899,181.88 L 48.109,183.51 L 50.414,184.95 L 52.72,186.3 L 55.217,187.54 L 57.619,188.69 L 60.213,189.65 L 62.71,190.61 L 65.304,191.38 L 67.994,191.95 L 70.684,192.43 L 73.374,192.82 L 76.063,193.1 L 78.753,193.2 L 81.539,193.2 L 84.325,193.1 L 87.015,192.82 L 89.801,192.34 L 92.49,191.76 L 95.18,191.09 L 97.87,190.23 L 100.56,189.27 L 103.25,188.12 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 125.92,112.84 L 125.92,112.84 L 128.13,117.63 L 129.86,122.62 L 131.01,127.51 L 131.68,132.5 L 131.78,137.68 L 131.4,142.66 L 130.63,147.65 L 129.38,152.44 L 127.65,157.05 L 125.44,161.55 L 122.94,165.77 L 119.77,169.9 L 116.31,173.64 L 112.57,177.09 L 108.34,180.16 L 103.73,182.75 L 107.96,190.99 L 113.34,187.83 L 118.33,184.19 L 122.85,180.16 L 126.88,175.65 L 130.44,170.95 L 133.51,165.97 L 136.1,160.69 L 138.22,155.13 L 139.66,149.38 L 140.62,143.62 L 141,137.87 L 140.91,131.92 L 140.04,125.98 L 138.7,120.13 L 136.78,114.37 L 134.18,108.62 L 134.18,108.62 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 55.89,90.686 L 55.89,90.686 L 58.195,89.44 L 60.693,88.481 L 63.191,87.522 L 65.688,86.754 L 68.09,86.179 L 70.78,85.604 L 73.181,85.124 L 75.679,84.932 L 78.081,84.836 L 80.867,84.836 L 83.268,84.932 L 85.862,85.22 L 88.36,85.508 L 90.857,85.987 L 93.163,86.467 L 95.468,87.138 L 97.87,88.097 L 100.27,88.96 L 102.48,90.015 L 104.79,91.166 L 107,92.412 L 109.01,93.659 L 111.03,95.193 L 112.95,96.728 L 114.97,98.454 L 116.79,100.28 L 118.62,102.1 L 120.25,104.02 L 121.79,106.03 L 123.33,108.14 L 124.67,110.44 L 125.92,112.84 L 134.18,108.62 L 132.55,105.84 L 131.01,103.34 L 129.28,100.66 L 127.36,98.262 L 125.34,95.961 L 123.33,93.755 L 121.12,91.741 L 118.91,89.823 L 116.6,87.905 L 114.2,86.179 L 111.61,84.549 L 109.01,83.11 L 106.52,81.768 L 103.73,80.521 L 101.14,79.466 L 98.35,78.507 L 95.468,77.644 L 92.586,76.973 L 89.704,76.493 L 86.823,76.014 L 83.845,75.726 L 80.867,75.63 L 78.081,75.63 L 75.103,75.726 L 72.029,76.11 L 69.051,76.589 L 66.169,77.165 L 63.191,77.932 L 60.309,78.891 L 57.427,79.85 L 54.545,81.192 L 51.663,82.439 L 51.663,82.439 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 33.795,160.6 L 33.795,160.6 L 31.586,155.8 L 29.857,150.81 L 28.704,145.83 L 28.031,140.84 L 27.935,135.66 L 28.223,130.68 L 28.992,125.78 L 30.337,120.99 L 31.97,116.29 L 34.179,111.88 L 36.773,107.56 L 39.847,103.54 L 43.306,99.796 L 47.052,96.344 L 51.279,93.275 L 55.89,90.686 L 51.663,82.439 L 46.284,85.604 L 41.288,89.248 L 36.773,93.275 L 32.738,97.783 L 29.28,102.39 L 26.11,107.47 L 23.516,112.84 L 21.499,118.3 L 19.962,123.87 L 19.001,129.72 L 18.713,135.66 L 18.809,141.42 L 19.674,147.36 L 21.019,153.31 L 22.94,159.06 L 25.534,164.81 L 25.534,164.81 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 103.73,182.75 L 103.73,182.75 L 101.42,183.99 L 98.927,184.95 L 96.429,185.91 L 93.931,186.68 L 91.53,187.25 L 88.936,187.83 L 86.438,188.31 L 84.037,188.5 L 81.539,188.6 L 78.753,188.6 L 76.352,188.5 L 73.854,188.21 L 71.356,187.93 L 68.859,187.45 L 66.361,186.97 L 64.151,186.3 L 61.846,185.34 L 59.348,184.47 L 57.235,183.42 L 54.833,182.27 L 52.816,181.02 L 50.702,179.77 L 48.685,178.24 L 46.668,176.71 L 44.746,174.98 L 42.921,173.16 L 41.096,171.34 L 39.463,169.42 L 37.926,167.4 L 36.389,165.29 L 35.044,162.99 L 33.795,160.6 L 25.534,164.81 L 27.167,167.6 L 28.704,170.09 L 30.433,172.77 L 32.354,175.17 L 34.372,177.47 L 36.389,179.68 L 38.598,181.69 L 40.712,183.61 L 43.113,185.53 L 45.515,187.25 L 48.013,188.88 L 50.606,190.32 L 53.2,191.67 L 55.89,192.91 L 58.58,193.97 L 61.27,194.93 L 64.248,195.79 L 67.129,196.46 L 70.011,196.94 L 72.893,197.42 L 75.775,197.71 L 78.753,197.8 L 81.539,197.8 L 84.613,197.71 L 87.591,197.32 L 90.665,196.84 L 93.451,196.27 L 96.429,195.5 L 99.311,194.54 L 102.19,193.58 L 105.07,192.24 L 107.96,190.99 L 107.96,190.99 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 105.84,186.87 L 110.84,183.99 L 115.45,180.64 L 119.58,176.9 L 123.33,172.77 L 126.69,168.36 L 129.47,163.76 L 131.88,158.87 L 133.8,153.79 L 135.14,148.51 L 136.01,143.14 L 136.39,137.77 L 136.3,132.21 L 135.53,126.74 L 134.28,121.37 L 132.45,116 L 130.05,110.73 L 128.61,108.14 L 127.17,105.74 L 125.54,103.34 L 123.81,101.14 L 121.98,99.029 L 120.06,97.015 L 118.04,95.097 L 115.93,93.275 L 113.82,91.549 L 111.61,89.919 L 109.3,88.481 L 106.9,87.138 L 104.5,85.891 L 102,84.741 L 99.503,83.782 L 96.909,82.823 L 94.316,82.055 L 91.722,81.48 L 89.032,81.001 L 86.342,80.617 L 83.556,80.329 L 80.867,80.233 L 78.081,80.233 L 75.391,80.329 L 72.605,80.617 L 69.915,81.096 L 67.129,81.672 L 64.44,82.343 L 61.75,83.206 L 59.06,84.165 L 56.37,85.316 L 53.777,86.563 L 48.781,89.44 L 44.17,92.796 L 40.039,96.536 L 36.293,100.66 L 33.027,104.97 L 30.145,109.67 L 27.743,114.56 L 25.918,119.65 L 24.477,124.83 L 23.612,130.2 L 23.324,135.66 L 23.42,141.13 L 24.189,146.59 L 25.438,152.06 L 27.263,157.43 L 29.664,162.7 L 31.105,165.29 L 32.546,167.69 L 34.179,170.09 L 35.909,172.29 L 37.734,174.4 L 39.655,176.42 L 41.672,178.34 L 43.69,180.16 L 45.899,181.88 L 48.109,183.51 L 50.414,184.95 L 52.72,186.3 L 55.217,187.54 L 57.619,188.69 L 60.213,189.65 L 62.71,190.61 L 65.304,191.38 L 67.994,191.95 L 70.684,192.43 L 73.374,192.82 L 76.063,193.1 L 78.753,193.2 L 81.539,193.2 L 84.325,193.1 L 87.015,192.82 L 89.801,192.34 L 92.49,191.76 L 95.18,191.09 L 97.87,190.23 L 100.56,189.27 L 103.25,188.12 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 58.195,51.081 L 20.538,70.548 L 18.809,71.507 L 19.674,73.233 L 37.734,107.85 L 38.022,108.52 L 38.791,108.71 L 38.983,108.81 L 39.367,108.91 L 40.039,109.1 L 40.808,109.19 L 41.769,109.39 L 42.921,109.48 L 44.266,109.58 L 45.707,109.67 L 47.34,109.58 L 49.069,109.48 L 50.991,109.19 L 52.912,108.81 L 55.025,108.24 L 57.139,107.56 L 59.444,106.7 L 61.75,105.55 L 64.632,103.92 L 67.129,102.39 L 69.339,100.76 L 71.26,99.221 L 72.989,97.687 L 74.334,96.248 L 75.487,94.906 L 76.448,93.563 L 77.216,92.316 L 77.889,91.262 L 78.273,90.207 L 78.657,89.44 L 78.849,88.672 L 78.945,88.193 L 79.041,87.809 L 79.041,87.617 L 79.041,87.042 L 78.849,86.467 L 60.789,51.848 L 59.925,50.218 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 60.021,102.1 L 58.292,102.96 L 56.562,103.63 L 54.929,104.21 L 53.392,104.69 L 51.855,105.07 L 50.414,105.36 L 49.069,105.55 L 47.724,105.65 L 46.476,105.74 L 45.419,105.74 L 44.362,105.74 L 43.402,105.65 L 42.537,105.55 L 41.865,105.45 L 41.192,105.36 L 40.712,105.26 L 23.997,73.137 L 58.292,55.396 L 75.103,87.713 L 74.815,88.576 L 74.238,89.823 L 73.278,91.357 L 71.933,93.18 L 70.011,95.193 L 67.418,97.399 L 64.151,99.7 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 50.222,155.32 L 47.917,150.33 L 46.284,145.16 L 45.323,140.07 L 44.939,134.89 L 45.131,129.81 L 45.995,124.92 L 47.34,120.22 L 49.262,115.72 L 47.532,116 L 45.995,116.19 L 44.458,116.19 L 43.017,116.19 L 41.672,116.1 L 40.424,115.81 L 39.271,115.52 L 38.214,115.24 L 36.101,119.74 L 34.564,124.44 L 33.699,129.43 L 33.315,134.51 L 33.603,139.79 L 34.564,144.96 L 36.197,150.14 L 38.406,155.22 L 40.135,158.2 L 41.961,160.98 L 44.074,163.66 L 46.284,166.06 L 48.589,168.27 L 51.087,170.28 L 53.68,172.1 L 56.37,173.73 L 59.156,175.17 L 61.942,176.32 L 64.92,177.28 L 67.898,178.05 L 70.876,178.53 L 73.854,178.72 L 76.928,178.72 L 79.906,178.53 L 77.696,177.95 L 75.391,177.28 L 73.181,176.51 L 71.068,175.55 L 68.955,174.5 L 66.937,173.35 L 64.92,172.01 L 62.903,170.57 L 61.077,169.03 L 59.252,167.4 L 57.523,165.68 L 55.89,163.76 L 54.353,161.84 L 52.816,159.73 L 51.471,157.62 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 102.87,110.63 L 98.35,116.29 L 98.734,116.87 L 99.215,117.54 L 99.599,118.11 L 100.08,118.78 L 100.46,119.36 L 100.85,120.03 L 101.14,120.61 L 101.52,121.28 L 101.81,121.76 L 102,122.24 L 102.29,122.72 L 102.48,123.2 L 104.02,122.72 L 108.15,120.7 L 107.67,119.26 L 107.09,117.92 L 106.42,116.58 L 105.84,115.33 L 105.07,114.09 L 104.4,112.84 L 103.63,111.69 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 116.12,113.8 L 115.35,112.45 L 114.59,111.02 L 113.72,109.77 L 112.95,108.43 L 111.99,107.18 L 111.03,105.93 L 110.07,104.69 L 109.11,103.54 L 105.27,108.24 L 106.13,109.39 L 107,110.63 L 107.86,111.98 L 108.63,113.32 L 109.4,114.66 L 110.07,116.19 L 110.74,117.63 L 111.32,119.26 L 117.37,116.48 L 117.08,115.81 L 116.79,115.14 L 116.51,114.47 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 101.71,104.21 L 105.36,99.7 L 104.02,98.742 L 102.58,97.783 L 101.14,96.919 L 99.599,96.056 L 98.062,95.289 L 96.525,94.618 L 94.988,93.947 L 93.355,93.275 L 90.473,98.837 L 91.722,99.221 L 93.067,99.796 L 94.508,100.28 L 95.853,100.95 L 97.294,101.62 L 98.734,102.39 L 100.27,103.25 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 88.744,102.29 L 85.574,108.43 L 86.823,108.81 L 88.167,109.29 L 89.416,109.77 L 90.569,110.35 L 91.818,110.92 L 92.971,111.59 L 94.123,112.26 L 95.18,113.03 L 99.695,107.37 L 98.254,106.41 L 96.813,105.65 L 95.372,104.88 L 94.027,104.21 L 92.586,103.63 L 91.338,103.15 L 89.993,102.67 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 112.95,97.974 L 114.39,99.413 L 115.83,100.95 L 117.18,102.48 L 118.43,104.11 L 119.68,105.84 L 120.93,107.66 L 121.98,109.48 L 123.04,111.4 L 125.15,116.1 L 126.79,120.89 L 127.94,125.69 L 128.61,130.58 L 128.71,135.47 L 128.42,140.36 L 127.55,145.06 L 126.4,149.76 L 124.67,154.27 L 122.56,158.68 L 119.96,162.8 L 117.08,166.73 L 113.72,170.38 L 109.97,173.73 L 105.94,176.71 L 101.42,179.29 L 97.966,180.93 L 94.508,182.27 L 90.953,183.32 L 87.399,184.09 L 83.845,184.67 L 80.29,184.95 L 76.64,184.95 L 73.085,184.67 L 69.627,184.19 L 66.169,183.51 L 62.71,182.56 L 59.348,181.31 L 56.082,179.87 L 53.008,178.24 L 49.934,176.32 L 47.052,174.21 L 50.03,176.8 L 53.104,179.1 L 56.37,181.12 L 59.732,182.94 L 63.287,184.47 L 66.841,185.72 L 70.588,186.68 L 74.334,187.45 L 78.177,187.83 L 82.019,188.02 L 85.958,187.83 L 89.801,187.35 L 93.643,186.58 L 97.486,185.53 L 101.23,184.09 L 104.98,182.36 L 109.49,179.77 L 113.53,176.8 L 117.27,173.45 L 120.64,169.8 L 123.52,165.87 L 126.11,161.75 L 128.23,157.33 L 129.86,152.83 L 131.11,148.13 L 131.88,143.33 L 132.26,138.44 L 132.07,133.55 L 131.49,128.66 L 130.34,123.87 L 128.71,119.07 L 126.59,114.37 L 125.25,111.98 L 123.81,109.67 L 122.17,107.47 L 120.54,105.36 L 118.72,103.34 L 116.89,101.43 L 114.97,99.605 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 52.624,63.739 L 69.243,95.865 L 69.723,95.385 L 70.203,95.002 L 70.588,94.522 L 71.068,94.138 L 71.452,93.659 L 71.74,93.275 L 72.125,92.892 L 72.413,92.508 L 56.466,61.725 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 47.628,66.328 L 64.728,99.317 L 65.4,98.837 L 66.073,98.358 L 66.649,97.974 L 67.225,97.495 L 50.318,64.89 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 44.554,70.26 L 46.187,69.397 L 47.724,68.438 L 49.262,67.575 L 50.702,66.616 L 52.143,65.657 L 53.392,64.698 L 54.641,63.739 L 55.794,62.78 L 56.851,61.821 L 57.811,60.862 L 58.676,59.903 L 59.444,58.944 L 60.117,58.081 L 60.693,57.218 L 61.173,56.355 L 61.462,55.492 L 61.75,54.245 L 61.75,53.19 L 61.558,52.327 L 61.27,51.656 L 61.077,51.368 L 60.885,51.081 L 60.597,50.697 L 60.309,50.409 L 59.925,50.122 L 59.444,49.834 L 58.868,49.546 L 58.292,49.355 L 57.427,49.163 L 56.466,48.971 L 55.41,48.875 L 54.257,48.971 L 53.008,48.971 L 51.759,49.163 L 50.414,49.45 L 49.069,49.738 L 47.532,50.122 L 46.091,50.601 L 44.554,51.081 L 42.921,51.656 L 41.384,52.327 L 39.751,52.999 L 38.022,53.766 L 36.389,54.629 L 34.564,55.588 L 32.642,56.739 L 30.913,57.794 L 29.184,58.944 L 27.551,60.191 L 26.014,61.342 L 24.573,62.588 L 23.228,63.835 L 22.075,65.082 L 21.115,66.328 L 20.346,67.575 L 19.674,68.822 L 19.29,69.972 L 19.193,71.123 L 19.29,72.178 L 19.674,73.233 L 19.866,73.521 L 20.058,73.808 L 20.346,74.096 L 20.634,74.384 L 21.115,74.767 L 21.595,75.055 L 22.075,75.343 L 22.748,75.534 L 23.612,75.726 L 24.573,75.918 L 25.63,75.918 L 26.686,75.918 L 27.935,75.822 L 29.184,75.63 L 30.529,75.439 L 31.97,75.151 L 33.411,74.767 L 34.852,74.288 L 36.389,73.808 L 38.022,73.233 L 39.559,72.562 L 41.192,71.89 L 42.921,71.123 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 23.132,71.411 L 23.132,71.219 L 23.132,71.027 L 23.132,70.836 L 23.228,70.548 L 23.805,69.397 L 24.765,68.055 L 26.11,66.616 L 27.839,64.986 L 29.857,63.26 L 32.354,61.534 L 35.14,59.807 L 38.214,58.081 L 39.847,57.314 L 41.384,56.547 L 42.921,55.876 L 44.458,55.3 L 45.803,54.725 L 47.244,54.245 L 48.589,53.862 L 49.838,53.478 L 50.991,53.286 L 52.143,52.999 L 53.2,52.903 L 54.161,52.807 L 55.025,52.807 L 55.794,52.807 L 56.466,52.903 L 57.043,52.999 L 57.331,53.095 L 57.523,53.19 L 57.715,53.382 L 57.811,53.478 L 57.907,53.766 L 57.811,54.149 L 57.619,54.629 L 57.331,55.204 L 56.947,55.971 L 56.37,56.643 L 55.602,57.506 L 54.833,58.369 L 53.873,59.328 L 52.72,60.287 L 51.471,61.342 L 50.03,62.397 L 48.397,63.451 L 46.668,64.602 L 44.746,65.657 L 42.729,66.808 L 41.096,67.575 L 39.559,68.342 L 38.022,69.013 L 36.485,69.589 L 35.14,70.164 L 33.699,70.644 L 32.354,71.027 L 31.105,71.315 L 29.953,71.603 L 28.8,71.794 L 27.743,71.986 L 26.783,72.082 L 25.918,72.082 L 25.149,72.082 L 24.477,71.986 L 23.901,71.794 L 23.612,71.699 L 23.42,71.603 L 23.228,71.507 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 41.769,58.752 L 40.52,57.698 L 39.079,56.451 L 37.542,54.917 L 36.197,53.286 L 35.044,51.56 L 34.179,49.93 L 33.891,48.396 L 34.083,46.957 L 35.332,43.697 L 36.101,41.012 L 36.389,38.614 L 36.197,36.504 L 35.62,34.586 L 34.66,32.86 L 33.507,31.038 L 32.066,29.12 L 31.586,28.545 L 31.105,27.874 L 30.625,27.298 L 30.145,26.627 L 30.145,26.627 L 29.568,26.148 L 28.896,25.86 L 28.127,25.86 L 27.455,26.243 L 26.975,26.819 L 26.686,27.49 L 26.686,28.257 L 26.975,28.929 L 26.975,28.929 L 27.551,29.6 L 28.031,30.271 L 28.512,30.942 L 28.992,31.518 L 30.241,33.052 L 31.201,34.491 L 31.97,35.833 L 32.354,37.272 L 32.45,38.806 L 32.162,40.628 L 31.586,42.834 L 30.529,45.423 L 29.953,48.204 L 30.433,51.081 L 31.778,53.766 L 33.603,56.259 L 35.524,58.369 L 37.446,60.095 L 38.791,61.342 L 39.463,61.821 L 39.463,61.821 L 40.135,62.205 L 40.904,62.205 L 41.576,61.917 L 42.153,61.438 L 42.537,60.766 L 42.537,59.999 L 42.249,59.328 L 41.769,58.752 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFFF1900"/>
</Path.Fill>
</Path>
<Path
Data="M 59.925,139.59 L 60.597,130.2 L 27.167,127.61 L 26.398,137 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
</Canvas> </UserControl>

Bomb.XAML

  通过使用上述代码,可使用<bomb:Bomb>元素将炸弹插入到窗口中,就像为主窗口插入Title用户控件一样。然而,在这个示例中以编程方式创建炸弹更加合理。

三、投弹

  为了投弹,应用程序使用DispatcherTimer,这是一种能很好地用于WPF用户界面的计时器,因为它在用户界面线程触发事件。选择事件间隔,此后DispatcherTime会在该时间间隔内引发周期性的Tick事件。

 // Fires events on the user interface thread.
private DispatcherTimer bombTimer = new DispatcherTimer();
public MainWindow()
{
InitializeComponent();
bombTimer.Tick += bombTimer_Tick;
}

  在BombDropper游戏中,计时器最初被设置为每隔1.3秒引发一次。当用户单击按钮开始游戏时,计时器随之启动:

 // Keep track of how many are dropped and stopped.
private int droppedCount = ;
private int savedCount = ; // Initially, bombs fall every 1.3 seconds, and hit the ground after 3.5 seconds.
private double initialSecondsBetweenBombs = 1.3;
private double initialSecondsToFall = 3.5;
private double secondsBetweenBombs;
private double secondsToFall; // Start the game.
private void cmdStart_Click(object sender, RoutedEventArgs e)
{
cmdStart.IsEnabled = false; // Reset the game.
droppedCount = ;
savedCount = ;
secondsBetweenBombs = initialSecondsBetweenBombs;
secondsToFall = initialSecondsToFall; // Start bomb dropping events.
bombTimer.Interval = TimeSpan.FromSeconds(secondsBetweenBombs);
bombTimer.Start();
}

  每次引发计时器事件时,代码创建一个新的Bomb对象并设置其再Canvas面板上的位置。炸弹放在Canvas面板的顶部边缘,使其可以无缝地落入试图。炸弹的水平位置是随机的,位于Canvas面板的左侧和右侧之间:

 // Drop a bomb.
private void bombTimer_Tick(object sender, EventArgs e)
{
// Create the bomb.
Bomb bomb = new Bomb();
bomb.IsFalling = true; // Position the bomb.
Random random = new Random();
bomb.SetValue(Canvas.LeftProperty,
(double)(random.Next(, (int)(canvasBackground.ActualWidth - ))));
bomb.SetValue(Canvas.TopProperty, -100.0); ...
}

  然后代码动态创建故事板为炸弹应用动画。这里使用例两个动画:一个动画通过修改Canvas.Top附加属性使炸弹下落,另一个动画通过修改旋转变换的角度使炸弹摆动。因为Storyboard.TargetElement和Storyboard.TargetProperty是附加属性,所以必须使用Storyboard.SetTargetElement和Storyboard.SetTargetProperty()方法设置它们:

// Attach mouse click event (for defusing the bomb).
bomb.MouseLeftButtonDown += bomb_MouseLeftButtonDown; // Create the animation for the falling bomb.
Storyboard storyboard = new Storyboard();
DoubleAnimation fallAnimation = new DoubleAnimation();
fallAnimation.To = canvasBackground.ActualHeight;
fallAnimation.Duration = TimeSpan.FromSeconds(secondsToFall); Storyboard.SetTarget(fallAnimation, bomb);
Storyboard.SetTargetProperty(fallAnimation, new PropertyPath("(Canvas.Top)"));
storyboard.Children.Add(fallAnimation); // Create the animation for the bomb "wiggle."
DoubleAnimation wiggleAnimation = new DoubleAnimation();
wiggleAnimation.To = ;
wiggleAnimation.Duration = TimeSpan.FromSeconds(0.2);
wiggleAnimation.RepeatBehavior = RepeatBehavior.Forever;
wiggleAnimation.AutoReverse = true; Storyboard.SetTarget(wiggleAnimation, ((TransformGroup)bomb.RenderTransform).Children[]);
Storyboard.SetTargetProperty(wiggleAnimation, new PropertyPath("Angle"));
storyboard.Children.Add(wiggleAnimation);

  这两个动画均可使用动画缓动以得到更逼真的行为,但这个示例使用基本的线性动画以使代码保持简单。

  新创建的故事板存储在字典集合中,从而可以在其他事件处理程序中很容易地检索故事板。字典集合存储为主窗口类的一个字段:

// Make it possible to look up a storyboard based on a bomb.
private Dictionary<Bomb, Storyboard> storyboards = new Dictionary<Bomb, Storyboard>();

  下面的代码将故事板添加到跟踪集合中:

...
// Add the storyboard to the tracking collection.
storyboards.Add(bomb, storyboard);
...

  接下来,关联当故事板结束fallAnimation动画时进行响应的事件处理程序,当炸弹落地时fallAnimation动画结束。最后,启动股市板并执行动画:

 // Configure and start the storyboard.
storyboard.Duration = fallAnimation.Duration;
storyboard.Completed += storyboard_Completed;
storyboard.Begin();

  用于投弹的代码还需要最后一个细节。随着游戏的进行,游戏难度加大。更频繁地引发计时器事件,从而炸弹之间的距离越来越近,并且介绍了下落时间。为实现这些变化,每经过一定的时间间隔就调整一次计时器代码。默认情况下,BombDropper每隔15秒调整一次。下面是控制调整的字段:

// "Adjustments" happen periodically, increasing the speed of bomb
// falling and shortening the time between bombs.
private DateTime lastAdjustmentTime = DateTime.MinValue; // Perform an adjustment every 15 seconds.
private double secondsBetweenAdjustments = ;
// After every adjustment, shave 0.1 seconds off both.
private double secondsBetweenBombsReduction = 0.1;
private double secondsToFallReduction = 0.1;

  下面的代码位于DispatcherTime.Tick事件处理程序的末尾处,这些代码检查是否需要对计时器进行一次调整,并执行适当的修改:

// Perform an "adjustment" when needed.
if ((DateTime.Now.Subtract(lastAdjustmentTime).TotalSeconds >
secondsBetweenAdjustments))
{
lastAdjustmentTime = DateTime.Now; secondsBetweenBombs -= secondsBetweenBombsReduction;
secondsToFall -= secondsToFallReduction; // (Technically, you should check for 0 or negative values.
// However, in practice these won't occur because the game will
// always end first.) // Set the timer to drop the next bomb at the appropriate time.
bombTimer.Interval = TimeSpan.FromSeconds(secondsBetweenBombs); // Update the status message.
lblRate.Text = String.Format("A bomb is released every {0} seconds.",
secondsBetweenBombs);
lblSpeed.Text = String.Format("Each bomb takes {0} seconds to fall.",
secondsToFall);
}

  通过上面的代码,这款游戏已经具有以不断增加的速率投弹的功能。不过,游戏仍缺少响应炸弹落下以及被拆除的代码。

四、拦截炸弹

  用户通过在炸弹达到Canvas面板底部之前单击炸弹来进行拆除。因为每个炸弹都是单独的Bomb用户控件实例,所以拦截鼠标单击很容易——需要做的全部工作就是处理MouseLeftButtonDown事件,当单击炸弹的任意部分时会引发该事件(但如果单击背景上的某个地方,例如炸弹圈边缘的周围,不会引发该事件)。

  当单击炸弹时,第一步是获取适当的炸弹对象,并设置其IsFalling属性以指示不再下降(在处理动画完成的事件处理程序中会使用IsFalling属性)。

private void bomb_MouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
// Get the bomb.
Bomb bomb = (Bomb)sender;
bomb.IsFalling = false; // Get the bomb's current position.
Storyboard storyboard = storyboards[bomb];
double currentTop = Canvas.GetTop(bomb);
...
}

  接下来查找控制炸弹的动画的故事板,从而可以停止动画。为查找故事板,需要在游戏的跟踪集合中查找。当前,WPF没有提供任何查找影响给定元素的动画的标准方式。

// Get the bomb's current position.
Storyboard storyboard = storyboards[bomb];
// Stop the bomb from falling.
storyboard.Stop();

  单击炸弹后,使用另一个动画集将炸弹移除屏幕,将炸弹抛向上方、抛向左侧或右侧(屈居于距离哪一侧最近)。尽管可创建全新的故事板以实现该效果,但BombDropper游戏清空用于炸弹的当前故事板并为其添加新动画。处理完毕后,启动新的故事板:

// Reuse the existing storyboard, but with new animations.
// Send the bomb on a new trajectory by animating Canvas.Top
// and Canvas.Left.
storyboard.Children.Clear(); DoubleAnimation riseAnimation = new DoubleAnimation();
riseAnimation.From = currentTop;
riseAnimation.To = ;
riseAnimation.Duration = TimeSpan.FromSeconds(); Storyboard.SetTarget(riseAnimation, bomb);
Storyboard.SetTargetProperty(riseAnimation, new PropertyPath("(Canvas.Top)"));
storyboard.Children.Add(riseAnimation); DoubleAnimation slideAnimation = new DoubleAnimation();
double currentLeft = Canvas.GetLeft(bomb);
// Throw the bomb off the closest side.
if (currentLeft < canvasBackground.ActualWidth / )
{
slideAnimation.To = -;
}
else
{
slideAnimation.To = canvasBackground.ActualWidth + ;
}
slideAnimation.Duration = TimeSpan.FromSeconds();
Storyboard.SetTarget(slideAnimation, bomb);
Storyboard.SetTargetProperty(slideAnimation, new PropertyPath("(Canvas.Left)"));
storyboard.Children.Add(slideAnimation); // Start the new animation.
storyboard.Duration = slideAnimation.Duration;
storyboard.Begin();

  现在,游戏已经具有足够的代码用于投下炸弹并当用户拆除它们弹出屏幕。然而,为跟踪哪些炸弹被拆了以及那些炸弹落下了,需要响应在动画结束时引发的Storyboard.Completed事件。

五、统计炸弹和清理工作

  正如在前面看到的,BombDropper采用两种方式使用故事板:为下落的炸弹应用动画以及为拆除的炸弹应用动画。可使用不同的事件处理程序处理这些故事板的结束事件,但为使代码保持简单,BombDropper只使用一个事件处理程序。通过检查Bomb.IsFalling属性来区分爆炸的炸弹和拆除的炸弹。

// End the game at maxDropped.
private int maxDropped = ; private void storyboard_Completed(object sender, EventArgs e)
{
ClockGroup clockGroup = (ClockGroup)sender; // Get the first animation in the storyboard, and use it to find the
// bomb that's being animated.
DoubleAnimation completedAnimation = (DoubleAnimation)clockGroup.Children[].Timeline;
Bomb completedBomb = (Bomb)Storyboard.GetTarget(completedAnimation); // Determine if a bomb fell or flew off the Canvas after being clicked.
if (completedBomb.IsFalling)
{
droppedCount++;
}
else
{
savedCount++;
}
...

  无论采用哪种方式,代码都会接着更新显示测试,指示已经落下的拆除的炸弹数量:

// Update the display.
lblStatus.Text = String.Format("You have dropped {0} bombs and saved {1}.",
droppedCount, savedCount);

  接下来,代码进行检查以查看落下炸弹是否达到了最大值。如果达到了最大值,游戏结束,停止计时器并移除所有炸弹和故事板:

 // Check if it's game over.
if (droppedCount >= maxDropped)
{
bombTimer.Stop();
lblStatus.Text += "\r\n\r\nGame over."; // Find all the storyboards that are underway.
foreach (KeyValuePair<Bomb, Storyboard> item in storyboards)
{
Storyboard storyboard = item.Value;
Bomb bomb = item.Key; storyboard.Stop();
canvasBackground.Children.Remove(bomb);
}
// Empty the tracking collection.
storyboards.Clear(); // Allow the user to start a new game.
cmdStart.IsEnabled = true;
}
else
{
Storyboard storyboard = (Storyboard)clockGroup.Timeline;
storyboard.Stop(); storyboards.Remove(completedBomb);
canvasBackground.Children.Remove(completedBomb);
}

  本例完整XAML标记:

<Window x:Class="BombDropper.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="" Width="">
<Grid x:Name="LayoutRoot">
<Grid.ColumnDefinitions>
<ColumnDefinition></ColumnDefinition>
<ColumnDefinition Width=""></ColumnDefinition>
</Grid.ColumnDefinitions> <Border BorderBrush="SteelBlue" BorderThickness="" Margin="">
<Grid>
<Canvas x:Name="canvasBackground" SizeChanged="canvasBackground_SizeChanged" MinWidth="">
<Canvas.Background>
<RadialGradientBrush>
<GradientStop Color="AliceBlue" Offset=""></GradientStop>
<GradientStop Color="White" Offset="0.7"></GradientStop>
</RadialGradientBrush>
</Canvas.Background>
</Canvas>
</Grid>
</Border> <Border Grid.Column="" BorderBrush="SteelBlue" BorderThickness="" Margin="">
<Border.Background>
<RadialGradientBrush GradientOrigin="1,0.7" Center="1,0.7" RadiusX="" RadiusY="">
<GradientStop Color="Orange" Offset=""></GradientStop>
<GradientStop Color="White" Offset=""></GradientStop>
</RadialGradientBrush>
</Border.Background>
<StackPanel Margin="" VerticalAlignment="Center" HorizontalAlignment="Center">
<TextBlock FontFamily="Impact" FontSize="" Foreground="LightSteelBlue">Bomb Dropper</TextBlock>
<TextBlock x:Name="lblRate" Margin="0,30,0,0" TextWrapping="Wrap" FontFamily="Georgia" FontSize=""></TextBlock>
<TextBlock x:Name="lblSpeed" Margin="0,30" TextWrapping="Wrap" FontFamily="Georgia" FontSize=""></TextBlock>
<TextBlock x:Name="lblStatus"
TextWrapping="Wrap" FontFamily="Georgia" FontSize="">No bombs have dropped.</TextBlock>
<Button x:Name="cmdStart" Padding="" Margin="0,30" Width="" Content="Start Game" Click="cmdStart_Click"></Button>
</StackPanel> </Border>
</Grid>
</Window>

BombDropper.xaml

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Windows.Threading; namespace BombDropper
{
/// <summary>
/// MainWindow.xaml 的交互逻辑
/// </summary>
public partial class MainWindow : Window
{
// Fires events on the user interface thread.
private DispatcherTimer bombTimer = new DispatcherTimer();
public MainWindow()
{
InitializeComponent();
bombTimer.Tick += bombTimer_Tick;
}
private void canvasBackground_SizeChanged(object sender, SizeChangedEventArgs e)
{
RectangleGeometry rect = new RectangleGeometry();
rect.Rect = new Rect(, , canvasBackground.ActualWidth, canvasBackground.ActualHeight);
canvasBackground.Clip = rect;
} // "Adjustments" happen periodically, increasing the speed of bomb
// falling and shortening the time between bombs.
private DateTime lastAdjustmentTime = DateTime.MinValue; // Perform an adjustment every 15 seconds.
private double secondsBetweenAdjustments = ; // Initially, bombs fall every 1.3 seconds, and hit the ground after 3.5 seconds.
private double initialSecondsBetweenBombs = 1.3;
private double initialSecondsToFall = 3.5;
private double secondsBetweenBombs;
private double secondsToFall; // After every adjustment, shave 0.1 seconds off both.
private double secondsBetweenBombsReduction = 0.1;
private double secondsToFallReduction = 0.1; // Make it possible to look up a storyboard based on a bomb.
private Dictionary<Bomb, Storyboard> storyboards = new Dictionary<Bomb, Storyboard>(); // Start the game.
private void cmdStart_Click(object sender, RoutedEventArgs e)
{
cmdStart.IsEnabled = false; // Reset the game.
droppedCount = ;
savedCount = ;
secondsBetweenBombs = initialSecondsBetweenBombs;
secondsToFall = initialSecondsToFall; // Start bomb dropping events.
bombTimer.Interval = TimeSpan.FromSeconds(secondsBetweenBombs);
bombTimer.Start();
} // Drop a bomb.
private void bombTimer_Tick(object sender, EventArgs e)
{
// Perform an "adjustment" when needed.
if ((DateTime.Now.Subtract(lastAdjustmentTime).TotalSeconds >
secondsBetweenAdjustments))
{
lastAdjustmentTime = DateTime.Now; secondsBetweenBombs -= secondsBetweenBombsReduction;
secondsToFall -= secondsToFallReduction; // (Technically, you should check for 0 or negative values.
// However, in practice these won't occur because the game will
// always end first.) // Set the timer to drop the next bomb at the appropriate time.
bombTimer.Interval = TimeSpan.FromSeconds(secondsBetweenBombs); // Update the status message.
lblRate.Text = String.Format("A bomb is released every {0} seconds.",
secondsBetweenBombs);
lblSpeed.Text = String.Format("Each bomb takes {0} seconds to fall.",
secondsToFall);
} // Create the bomb.
Bomb bomb = new Bomb();
bomb.IsFalling = true; // Position the bomb.
Random random = new Random();
bomb.SetValue(Canvas.LeftProperty,
(double)(random.Next(, (int)(canvasBackground.ActualWidth - ))));
bomb.SetValue(Canvas.TopProperty, -100.0); // Attach mouse click event (for defusing the bomb).
bomb.MouseLeftButtonDown += bomb_MouseLeftButtonDown; // Create the animation for the falling bomb.
Storyboard storyboard = new Storyboard();
DoubleAnimation fallAnimation = new DoubleAnimation();
fallAnimation.To = canvasBackground.ActualHeight;
fallAnimation.Duration = TimeSpan.FromSeconds(secondsToFall); Storyboard.SetTarget(fallAnimation, bomb);
Storyboard.SetTargetProperty(fallAnimation, new PropertyPath("(Canvas.Top)"));
storyboard.Children.Add(fallAnimation); // Create the animation for the bomb "wiggle."
DoubleAnimation wiggleAnimation = new DoubleAnimation();
wiggleAnimation.To = ;
wiggleAnimation.Duration = TimeSpan.FromSeconds(0.2);
wiggleAnimation.RepeatBehavior = RepeatBehavior.Forever;
wiggleAnimation.AutoReverse = true; Storyboard.SetTarget(wiggleAnimation, ((TransformGroup)bomb.RenderTransform).Children[]);
Storyboard.SetTargetProperty(wiggleAnimation, new PropertyPath("Angle"));
storyboard.Children.Add(wiggleAnimation); // Add the bomb to the Canvas.
canvasBackground.Children.Add(bomb); // Add the storyboard to the tracking collection.
storyboards.Add(bomb, storyboard); // Configure and start the storyboard.
storyboard.Duration = fallAnimation.Duration;
storyboard.Completed += storyboard_Completed;
storyboard.Begin();
} private void bomb_MouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
// Get the bomb.
Bomb bomb = (Bomb)sender;
bomb.IsFalling = false; // Get the bomb's current position.
Storyboard storyboard = storyboards[bomb];
double currentTop = Canvas.GetTop(bomb); // Stop the bomb from falling.
storyboard.Stop(); // Reuse the existing storyboard, but with new animations.
// Send the bomb on a new trajectory by animating Canvas.Top
// and Canvas.Left.
storyboard.Children.Clear(); DoubleAnimation riseAnimation = new DoubleAnimation();
riseAnimation.From = currentTop;
riseAnimation.To = ;
riseAnimation.Duration = TimeSpan.FromSeconds(); Storyboard.SetTarget(riseAnimation, bomb);
Storyboard.SetTargetProperty(riseAnimation, new PropertyPath("(Canvas.Top)"));
storyboard.Children.Add(riseAnimation); DoubleAnimation slideAnimation = new DoubleAnimation();
double currentLeft = Canvas.GetLeft(bomb);
// Throw the bomb off the closest side.
if (currentLeft < canvasBackground.ActualWidth / )
{
slideAnimation.To = -;
}
else
{
slideAnimation.To = canvasBackground.ActualWidth + ;
}
slideAnimation.Duration = TimeSpan.FromSeconds();
Storyboard.SetTarget(slideAnimation, bomb);
Storyboard.SetTargetProperty(slideAnimation, new PropertyPath("(Canvas.Left)"));
storyboard.Children.Add(slideAnimation); // Start the new animation.
storyboard.Duration = slideAnimation.Duration;
storyboard.Begin();
} // Keep track of how many are dropped and stopped.
private int droppedCount = ;
private int savedCount = ; // End the game at maxDropped.
private int maxDropped = ; private void storyboard_Completed(object sender, EventArgs e)
{
ClockGroup clockGroup = (ClockGroup)sender; // Get the first animation in the storyboard, and use it to find the
// bomb that's being animated.
DoubleAnimation completedAnimation = (DoubleAnimation)clockGroup.Children[].Timeline;
Bomb completedBomb = (Bomb)Storyboard.GetTarget(completedAnimation); // Determine if a bomb fell or flew off the Canvas after being clicked.
if (completedBomb.IsFalling)
{
droppedCount++;
}
else
{
savedCount++;
} // Update the display.
lblStatus.Text = String.Format("You have dropped {0} bombs and saved {1}.",
droppedCount, savedCount); // Check if it's game over.
if (droppedCount >= maxDropped)
{
bombTimer.Stop();
lblStatus.Text += "\r\n\r\nGame over."; // Find all the storyboards that are underway.
foreach (KeyValuePair<Bomb, Storyboard> item in storyboards)
{
Storyboard storyboard = item.Value;
Bomb bomb = item.Key; storyboard.Stop();
canvasBackground.Children.Remove(bomb);
}
// Empty the tracking collection.
storyboards.Clear(); // Allow the user to start a new game.
cmdStart.IsEnabled = true;
}
else
{
Storyboard storyboard = (Storyboard)clockGroup.Timeline;
storyboard.Stop(); storyboards.Remove(completedBomb);
canvasBackground.Children.Remove(completedBomb);
}
}
}
}

BombDropper.xaml.cs

<UserControl x:Class="BombDropper.Bomb"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
>
<UserControl.RenderTransform>
<TransformGroup>
<RotateTransform Angle="20" CenterX="50" CenterY="50"></RotateTransform>
<ScaleTransform ScaleX="0.5" ScaleY="0.5"></ScaleTransform>
</TransformGroup>
</UserControl.RenderTransform>
<Canvas>
<Path Data="M 11.989,50.026 L 24.381,37.08 L 19.097,53.862 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 0.46098,31.997 L 17.945,28.449 L 4.1114,39.19 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 9.9713,7.3517 L 22.075,20.49 L 5.7445,14.16 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 58.484,29.408 L 40.712,31.997 L 57.523,37.367 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 51.663,10.229 L 38.694,22.408 L 55.506,17.325 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path Data="M 32.354,0.25535 L 31.682,18.092 L 40.039,2.7487 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFF2CC0C"/>
</Path.Fill>
</Path>
<Path
Data="M 105.84,186.87 L 110.84,183.99 L 115.45,180.64 L 119.58,176.9 L 123.33,172.77 L 126.69,168.36 L 129.47,163.76 L 131.88,158.87 L 133.8,153.79 L 135.14,148.51 L 136.01,143.14 L 136.39,137.77 L 136.3,132.21 L 135.53,126.74 L 134.28,121.37 L 132.45,116 L 130.05,110.73 L 128.61,108.14 L 127.17,105.74 L 125.54,103.34 L 123.81,101.14 L 121.98,99.029 L 120.06,97.015 L 118.04,95.097 L 115.93,93.275 L 113.82,91.549 L 111.61,89.919 L 109.3,88.481 L 106.9,87.138 L 104.5,85.891 L 102,84.741 L 99.503,83.782 L 96.909,82.823 L 94.316,82.055 L 91.722,81.48 L 89.032,81.001 L 86.342,80.617 L 83.556,80.329 L 80.867,80.233 L 78.081,80.233 L 75.391,80.329 L 72.605,80.617 L 69.915,81.096 L 67.129,81.672 L 64.44,82.343 L 61.75,83.206 L 59.06,84.165 L 56.37,85.316 L 53.777,86.563 L 48.781,89.44 L 44.17,92.796 L 40.039,96.536 L 36.293,100.66 L 33.027,104.97 L 30.145,109.67 L 27.743,114.56 L 25.918,119.65 L 24.477,124.83 L 23.612,130.2 L 23.324,135.66 L 23.42,141.13 L 24.189,146.59 L 25.438,152.06 L 27.263,157.43 L 29.664,162.7 L 31.105,165.29 L 32.546,167.69 L 34.179,170.09 L 35.909,172.29 L 37.734,174.4 L 39.655,176.42 L 41.672,178.34 L 43.69,180.16 L 45.899,181.88 L 48.109,183.51 L 50.414,184.95 L 52.72,186.3 L 55.217,187.54 L 57.619,188.69 L 60.213,189.65 L 62.71,190.61 L 65.304,191.38 L 67.994,191.95 L 70.684,192.43 L 73.374,192.82 L 76.063,193.1 L 78.753,193.2 L 81.539,193.2 L 84.325,193.1 L 87.015,192.82 L 89.801,192.34 L 92.49,191.76 L 95.18,191.09 L 97.87,190.23 L 100.56,189.27 L 103.25,188.12 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 125.92,112.84 L 125.92,112.84 L 128.13,117.63 L 129.86,122.62 L 131.01,127.51 L 131.68,132.5 L 131.78,137.68 L 131.4,142.66 L 130.63,147.65 L 129.38,152.44 L 127.65,157.05 L 125.44,161.55 L 122.94,165.77 L 119.77,169.9 L 116.31,173.64 L 112.57,177.09 L 108.34,180.16 L 103.73,182.75 L 107.96,190.99 L 113.34,187.83 L 118.33,184.19 L 122.85,180.16 L 126.88,175.65 L 130.44,170.95 L 133.51,165.97 L 136.1,160.69 L 138.22,155.13 L 139.66,149.38 L 140.62,143.62 L 141,137.87 L 140.91,131.92 L 140.04,125.98 L 138.7,120.13 L 136.78,114.37 L 134.18,108.62 L 134.18,108.62 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 55.89,90.686 L 55.89,90.686 L 58.195,89.44 L 60.693,88.481 L 63.191,87.522 L 65.688,86.754 L 68.09,86.179 L 70.78,85.604 L 73.181,85.124 L 75.679,84.932 L 78.081,84.836 L 80.867,84.836 L 83.268,84.932 L 85.862,85.22 L 88.36,85.508 L 90.857,85.987 L 93.163,86.467 L 95.468,87.138 L 97.87,88.097 L 100.27,88.96 L 102.48,90.015 L 104.79,91.166 L 107,92.412 L 109.01,93.659 L 111.03,95.193 L 112.95,96.728 L 114.97,98.454 L 116.79,100.28 L 118.62,102.1 L 120.25,104.02 L 121.79,106.03 L 123.33,108.14 L 124.67,110.44 L 125.92,112.84 L 134.18,108.62 L 132.55,105.84 L 131.01,103.34 L 129.28,100.66 L 127.36,98.262 L 125.34,95.961 L 123.33,93.755 L 121.12,91.741 L 118.91,89.823 L 116.6,87.905 L 114.2,86.179 L 111.61,84.549 L 109.01,83.11 L 106.52,81.768 L 103.73,80.521 L 101.14,79.466 L 98.35,78.507 L 95.468,77.644 L 92.586,76.973 L 89.704,76.493 L 86.823,76.014 L 83.845,75.726 L 80.867,75.63 L 78.081,75.63 L 75.103,75.726 L 72.029,76.11 L 69.051,76.589 L 66.169,77.165 L 63.191,77.932 L 60.309,78.891 L 57.427,79.85 L 54.545,81.192 L 51.663,82.439 L 51.663,82.439 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 33.795,160.6 L 33.795,160.6 L 31.586,155.8 L 29.857,150.81 L 28.704,145.83 L 28.031,140.84 L 27.935,135.66 L 28.223,130.68 L 28.992,125.78 L 30.337,120.99 L 31.97,116.29 L 34.179,111.88 L 36.773,107.56 L 39.847,103.54 L 43.306,99.796 L 47.052,96.344 L 51.279,93.275 L 55.89,90.686 L 51.663,82.439 L 46.284,85.604 L 41.288,89.248 L 36.773,93.275 L 32.738,97.783 L 29.28,102.39 L 26.11,107.47 L 23.516,112.84 L 21.499,118.3 L 19.962,123.87 L 19.001,129.72 L 18.713,135.66 L 18.809,141.42 L 19.674,147.36 L 21.019,153.31 L 22.94,159.06 L 25.534,164.81 L 25.534,164.81 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="F 1 M 103.73,182.75 L 103.73,182.75 L 101.42,183.99 L 98.927,184.95 L 96.429,185.91 L 93.931,186.68 L 91.53,187.25 L 88.936,187.83 L 86.438,188.31 L 84.037,188.5 L 81.539,188.6 L 78.753,188.6 L 76.352,188.5 L 73.854,188.21 L 71.356,187.93 L 68.859,187.45 L 66.361,186.97 L 64.151,186.3 L 61.846,185.34 L 59.348,184.47 L 57.235,183.42 L 54.833,182.27 L 52.816,181.02 L 50.702,179.77 L 48.685,178.24 L 46.668,176.71 L 44.746,174.98 L 42.921,173.16 L 41.096,171.34 L 39.463,169.42 L 37.926,167.4 L 36.389,165.29 L 35.044,162.99 L 33.795,160.6 L 25.534,164.81 L 27.167,167.6 L 28.704,170.09 L 30.433,172.77 L 32.354,175.17 L 34.372,177.47 L 36.389,179.68 L 38.598,181.69 L 40.712,183.61 L 43.113,185.53 L 45.515,187.25 L 48.013,188.88 L 50.606,190.32 L 53.2,191.67 L 55.89,192.91 L 58.58,193.97 L 61.27,194.93 L 64.248,195.79 L 67.129,196.46 L 70.011,196.94 L 72.893,197.42 L 75.775,197.71 L 78.753,197.8 L 81.539,197.8 L 84.613,197.71 L 87.591,197.32 L 90.665,196.84 L 93.451,196.27 L 96.429,195.5 L 99.311,194.54 L 102.19,193.58 L 105.07,192.24 L 107.96,190.99 L 107.96,190.99 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 105.84,186.87 L 110.84,183.99 L 115.45,180.64 L 119.58,176.9 L 123.33,172.77 L 126.69,168.36 L 129.47,163.76 L 131.88,158.87 L 133.8,153.79 L 135.14,148.51 L 136.01,143.14 L 136.39,137.77 L 136.3,132.21 L 135.53,126.74 L 134.28,121.37 L 132.45,116 L 130.05,110.73 L 128.61,108.14 L 127.17,105.74 L 125.54,103.34 L 123.81,101.14 L 121.98,99.029 L 120.06,97.015 L 118.04,95.097 L 115.93,93.275 L 113.82,91.549 L 111.61,89.919 L 109.3,88.481 L 106.9,87.138 L 104.5,85.891 L 102,84.741 L 99.503,83.782 L 96.909,82.823 L 94.316,82.055 L 91.722,81.48 L 89.032,81.001 L 86.342,80.617 L 83.556,80.329 L 80.867,80.233 L 78.081,80.233 L 75.391,80.329 L 72.605,80.617 L 69.915,81.096 L 67.129,81.672 L 64.44,82.343 L 61.75,83.206 L 59.06,84.165 L 56.37,85.316 L 53.777,86.563 L 48.781,89.44 L 44.17,92.796 L 40.039,96.536 L 36.293,100.66 L 33.027,104.97 L 30.145,109.67 L 27.743,114.56 L 25.918,119.65 L 24.477,124.83 L 23.612,130.2 L 23.324,135.66 L 23.42,141.13 L 24.189,146.59 L 25.438,152.06 L 27.263,157.43 L 29.664,162.7 L 31.105,165.29 L 32.546,167.69 L 34.179,170.09 L 35.909,172.29 L 37.734,174.4 L 39.655,176.42 L 41.672,178.34 L 43.69,180.16 L 45.899,181.88 L 48.109,183.51 L 50.414,184.95 L 52.72,186.3 L 55.217,187.54 L 57.619,188.69 L 60.213,189.65 L 62.71,190.61 L 65.304,191.38 L 67.994,191.95 L 70.684,192.43 L 73.374,192.82 L 76.063,193.1 L 78.753,193.2 L 81.539,193.2 L 84.325,193.1 L 87.015,192.82 L 89.801,192.34 L 92.49,191.76 L 95.18,191.09 L 97.87,190.23 L 100.56,189.27 L 103.25,188.12 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 58.195,51.081 L 20.538,70.548 L 18.809,71.507 L 19.674,73.233 L 37.734,107.85 L 38.022,108.52 L 38.791,108.71 L 38.983,108.81 L 39.367,108.91 L 40.039,109.1 L 40.808,109.19 L 41.769,109.39 L 42.921,109.48 L 44.266,109.58 L 45.707,109.67 L 47.34,109.58 L 49.069,109.48 L 50.991,109.19 L 52.912,108.81 L 55.025,108.24 L 57.139,107.56 L 59.444,106.7 L 61.75,105.55 L 64.632,103.92 L 67.129,102.39 L 69.339,100.76 L 71.26,99.221 L 72.989,97.687 L 74.334,96.248 L 75.487,94.906 L 76.448,93.563 L 77.216,92.316 L 77.889,91.262 L 78.273,90.207 L 78.657,89.44 L 78.849,88.672 L 78.945,88.193 L 79.041,87.809 L 79.041,87.617 L 79.041,87.042 L 78.849,86.467 L 60.789,51.848 L 59.925,50.218 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 60.021,102.1 L 58.292,102.96 L 56.562,103.63 L 54.929,104.21 L 53.392,104.69 L 51.855,105.07 L 50.414,105.36 L 49.069,105.55 L 47.724,105.65 L 46.476,105.74 L 45.419,105.74 L 44.362,105.74 L 43.402,105.65 L 42.537,105.55 L 41.865,105.45 L 41.192,105.36 L 40.712,105.26 L 23.997,73.137 L 58.292,55.396 L 75.103,87.713 L 74.815,88.576 L 74.238,89.823 L 73.278,91.357 L 71.933,93.18 L 70.011,95.193 L 67.418,97.399 L 64.151,99.7 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 50.222,155.32 L 47.917,150.33 L 46.284,145.16 L 45.323,140.07 L 44.939,134.89 L 45.131,129.81 L 45.995,124.92 L 47.34,120.22 L 49.262,115.72 L 47.532,116 L 45.995,116.19 L 44.458,116.19 L 43.017,116.19 L 41.672,116.1 L 40.424,115.81 L 39.271,115.52 L 38.214,115.24 L 36.101,119.74 L 34.564,124.44 L 33.699,129.43 L 33.315,134.51 L 33.603,139.79 L 34.564,144.96 L 36.197,150.14 L 38.406,155.22 L 40.135,158.2 L 41.961,160.98 L 44.074,163.66 L 46.284,166.06 L 48.589,168.27 L 51.087,170.28 L 53.68,172.1 L 56.37,173.73 L 59.156,175.17 L 61.942,176.32 L 64.92,177.28 L 67.898,178.05 L 70.876,178.53 L 73.854,178.72 L 76.928,178.72 L 79.906,178.53 L 77.696,177.95 L 75.391,177.28 L 73.181,176.51 L 71.068,175.55 L 68.955,174.5 L 66.937,173.35 L 64.92,172.01 L 62.903,170.57 L 61.077,169.03 L 59.252,167.4 L 57.523,165.68 L 55.89,163.76 L 54.353,161.84 L 52.816,159.73 L 51.471,157.62 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 102.87,110.63 L 98.35,116.29 L 98.734,116.87 L 99.215,117.54 L 99.599,118.11 L 100.08,118.78 L 100.46,119.36 L 100.85,120.03 L 101.14,120.61 L 101.52,121.28 L 101.81,121.76 L 102,122.24 L 102.29,122.72 L 102.48,123.2 L 104.02,122.72 L 108.15,120.7 L 107.67,119.26 L 107.09,117.92 L 106.42,116.58 L 105.84,115.33 L 105.07,114.09 L 104.4,112.84 L 103.63,111.69 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 116.12,113.8 L 115.35,112.45 L 114.59,111.02 L 113.72,109.77 L 112.95,108.43 L 111.99,107.18 L 111.03,105.93 L 110.07,104.69 L 109.11,103.54 L 105.27,108.24 L 106.13,109.39 L 107,110.63 L 107.86,111.98 L 108.63,113.32 L 109.4,114.66 L 110.07,116.19 L 110.74,117.63 L 111.32,119.26 L 117.37,116.48 L 117.08,115.81 L 116.79,115.14 L 116.51,114.47 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 101.71,104.21 L 105.36,99.7 L 104.02,98.742 L 102.58,97.783 L 101.14,96.919 L 99.599,96.056 L 98.062,95.289 L 96.525,94.618 L 94.988,93.947 L 93.355,93.275 L 90.473,98.837 L 91.722,99.221 L 93.067,99.796 L 94.508,100.28 L 95.853,100.95 L 97.294,101.62 L 98.734,102.39 L 100.27,103.25 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 88.744,102.29 L 85.574,108.43 L 86.823,108.81 L 88.167,109.29 L 89.416,109.77 L 90.569,110.35 L 91.818,110.92 L 92.971,111.59 L 94.123,112.26 L 95.18,113.03 L 99.695,107.37 L 98.254,106.41 L 96.813,105.65 L 95.372,104.88 L 94.027,104.21 L 92.586,103.63 L 91.338,103.15 L 89.993,102.67 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 112.95,97.974 L 114.39,99.413 L 115.83,100.95 L 117.18,102.48 L 118.43,104.11 L 119.68,105.84 L 120.93,107.66 L 121.98,109.48 L 123.04,111.4 L 125.15,116.1 L 126.79,120.89 L 127.94,125.69 L 128.61,130.58 L 128.71,135.47 L 128.42,140.36 L 127.55,145.06 L 126.4,149.76 L 124.67,154.27 L 122.56,158.68 L 119.96,162.8 L 117.08,166.73 L 113.72,170.38 L 109.97,173.73 L 105.94,176.71 L 101.42,179.29 L 97.966,180.93 L 94.508,182.27 L 90.953,183.32 L 87.399,184.09 L 83.845,184.67 L 80.29,184.95 L 76.64,184.95 L 73.085,184.67 L 69.627,184.19 L 66.169,183.51 L 62.71,182.56 L 59.348,181.31 L 56.082,179.87 L 53.008,178.24 L 49.934,176.32 L 47.052,174.21 L 50.03,176.8 L 53.104,179.1 L 56.37,181.12 L 59.732,182.94 L 63.287,184.47 L 66.841,185.72 L 70.588,186.68 L 74.334,187.45 L 78.177,187.83 L 82.019,188.02 L 85.958,187.83 L 89.801,187.35 L 93.643,186.58 L 97.486,185.53 L 101.23,184.09 L 104.98,182.36 L 109.49,179.77 L 113.53,176.8 L 117.27,173.45 L 120.64,169.8 L 123.52,165.87 L 126.11,161.75 L 128.23,157.33 L 129.86,152.83 L 131.11,148.13 L 131.88,143.33 L 132.26,138.44 L 132.07,133.55 L 131.49,128.66 L 130.34,123.87 L 128.71,119.07 L 126.59,114.37 L 125.25,111.98 L 123.81,109.67 L 122.17,107.47 L 120.54,105.36 L 118.72,103.34 L 116.89,101.43 L 114.97,99.605 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 52.624,63.739 L 69.243,95.865 L 69.723,95.385 L 70.203,95.002 L 70.588,94.522 L 71.068,94.138 L 71.452,93.659 L 71.74,93.275 L 72.125,92.892 L 72.413,92.508 L 56.466,61.725 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 47.628,66.328 L 64.728,99.317 L 65.4,98.837 L 66.073,98.358 L 66.649,97.974 L 67.225,97.495 L 50.318,64.89 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF383838"/>
</Path.Fill>
</Path>
<Path
Data="M 44.554,70.26 L 46.187,69.397 L 47.724,68.438 L 49.262,67.575 L 50.702,66.616 L 52.143,65.657 L 53.392,64.698 L 54.641,63.739 L 55.794,62.78 L 56.851,61.821 L 57.811,60.862 L 58.676,59.903 L 59.444,58.944 L 60.117,58.081 L 60.693,57.218 L 61.173,56.355 L 61.462,55.492 L 61.75,54.245 L 61.75,53.19 L 61.558,52.327 L 61.27,51.656 L 61.077,51.368 L 60.885,51.081 L 60.597,50.697 L 60.309,50.409 L 59.925,50.122 L 59.444,49.834 L 58.868,49.546 L 58.292,49.355 L 57.427,49.163 L 56.466,48.971 L 55.41,48.875 L 54.257,48.971 L 53.008,48.971 L 51.759,49.163 L 50.414,49.45 L 49.069,49.738 L 47.532,50.122 L 46.091,50.601 L 44.554,51.081 L 42.921,51.656 L 41.384,52.327 L 39.751,52.999 L 38.022,53.766 L 36.389,54.629 L 34.564,55.588 L 32.642,56.739 L 30.913,57.794 L 29.184,58.944 L 27.551,60.191 L 26.014,61.342 L 24.573,62.588 L 23.228,63.835 L 22.075,65.082 L 21.115,66.328 L 20.346,67.575 L 19.674,68.822 L 19.29,69.972 L 19.193,71.123 L 19.29,72.178 L 19.674,73.233 L 19.866,73.521 L 20.058,73.808 L 20.346,74.096 L 20.634,74.384 L 21.115,74.767 L 21.595,75.055 L 22.075,75.343 L 22.748,75.534 L 23.612,75.726 L 24.573,75.918 L 25.63,75.918 L 26.686,75.918 L 27.935,75.822 L 29.184,75.63 L 30.529,75.439 L 31.97,75.151 L 33.411,74.767 L 34.852,74.288 L 36.389,73.808 L 38.022,73.233 L 39.559,72.562 L 41.192,71.89 L 42.921,71.123 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF777777"/>
</Path.Fill>
</Path>
<Path
Data="M 23.132,71.411 L 23.132,71.219 L 23.132,71.027 L 23.132,70.836 L 23.228,70.548 L 23.805,69.397 L 24.765,68.055 L 26.11,66.616 L 27.839,64.986 L 29.857,63.26 L 32.354,61.534 L 35.14,59.807 L 38.214,58.081 L 39.847,57.314 L 41.384,56.547 L 42.921,55.876 L 44.458,55.3 L 45.803,54.725 L 47.244,54.245 L 48.589,53.862 L 49.838,53.478 L 50.991,53.286 L 52.143,52.999 L 53.2,52.903 L 54.161,52.807 L 55.025,52.807 L 55.794,52.807 L 56.466,52.903 L 57.043,52.999 L 57.331,53.095 L 57.523,53.19 L 57.715,53.382 L 57.811,53.478 L 57.907,53.766 L 57.811,54.149 L 57.619,54.629 L 57.331,55.204 L 56.947,55.971 L 56.37,56.643 L 55.602,57.506 L 54.833,58.369 L 53.873,59.328 L 52.72,60.287 L 51.471,61.342 L 50.03,62.397 L 48.397,63.451 L 46.668,64.602 L 44.746,65.657 L 42.729,66.808 L 41.096,67.575 L 39.559,68.342 L 38.022,69.013 L 36.485,69.589 L 35.14,70.164 L 33.699,70.644 L 32.354,71.027 L 31.105,71.315 L 29.953,71.603 L 28.8,71.794 L 27.743,71.986 L 26.783,72.082 L 25.918,72.082 L 25.149,72.082 L 24.477,71.986 L 23.901,71.794 L 23.612,71.699 L 23.42,71.603 L 23.228,71.507 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
<Path
Data="M 41.769,58.752 L 40.52,57.698 L 39.079,56.451 L 37.542,54.917 L 36.197,53.286 L 35.044,51.56 L 34.179,49.93 L 33.891,48.396 L 34.083,46.957 L 35.332,43.697 L 36.101,41.012 L 36.389,38.614 L 36.197,36.504 L 35.62,34.586 L 34.66,32.86 L 33.507,31.038 L 32.066,29.12 L 31.586,28.545 L 31.105,27.874 L 30.625,27.298 L 30.145,26.627 L 30.145,26.627 L 29.568,26.148 L 28.896,25.86 L 28.127,25.86 L 27.455,26.243 L 26.975,26.819 L 26.686,27.49 L 26.686,28.257 L 26.975,28.929 L 26.975,28.929 L 27.551,29.6 L 28.031,30.271 L 28.512,30.942 L 28.992,31.518 L 30.241,33.052 L 31.201,34.491 L 31.97,35.833 L 32.354,37.272 L 32.45,38.806 L 32.162,40.628 L 31.586,42.834 L 30.529,45.423 L 29.953,48.204 L 30.433,51.081 L 31.778,53.766 L 33.603,56.259 L 35.524,58.369 L 37.446,60.095 L 38.791,61.342 L 39.463,61.821 L 39.463,61.821 L 40.135,62.205 L 40.904,62.205 L 41.576,61.917 L 42.153,61.438 L 42.537,60.766 L 42.537,59.999 L 42.249,59.328 L 41.769,58.752 Z ">
<Path.Fill>
<SolidColorBrush Color="#FFFF1900"/>
</Path.Fill>
</Path>
<Path
Data="M 59.925,139.59 L 60.597,130.2 L 27.167,127.61 L 26.398,137 Z ">
<Path.Fill>
<SolidColorBrush Color="#FF000000"/>
</Path.Fill>
</Path>
</Canvas>
</UserControl>

Bomb.xmal

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes; namespace BombDropper
{
/// <summary>
/// Bomb.xaml 的交互逻辑
/// </summary>
public partial class Bomb : UserControl
{
public Bomb()
{
InitializeComponent();
}
public bool IsFalling
{
get;
set;
}
}
}

Bomb.xmal.cs

  现在已经完成了BombDropper游戏的代码。然而,可进行诸多改进。例如,可执行如下改进:

  •   为炸弹添加爆炸动画效果。这种效果使炸弹周围的火焰闪耀或发射在Canvas面板上四处飞溅的炸弹碎片。
  •   为背景添加动画。此改进易于实现,可添加精彩的可视化效果。例如,可创建上移的线性渐变,产生移动感,或创建在两种颜色之间过渡的效果。
  •   添加深度。实现这一改进比想象得更容易。基本技术是为炸弹设置不同尺寸。更大的炸弹应当具有更高的ZIndex值,确保大炸弹重叠在小炸弹之上,而且应为大炸弹设置更短的动画时间,从而确保他们下落得更快。还可使炸弹半透明,从而当一个炸弹下落时,仍能看到它背后的其他炸弹。
  •   添加音效。可是准确计时的声音效果以强调炸弹爆炸或拆除。
  •   使用动画缓动。如果希望炸弹在下落、弹离屏幕时加速,或更自然地摆动,可为此处使用的动画缓动函数。并且,正如所期望的,可使用代码构造缓动函数,就像在XAML中构建缓动函数一样容易。
  •   调整参数。可为修改行为提供更多细节(例如,当游戏运行时设置如何修改炸弹运动时间、轨迹以及投放频率的变量),还可插入更多随机因素(例如,使拆除的炸弹以稍有不同的方式弹离Canvas面板)

【WPF学习】第五十七章 使用代码创建故事板的更多相关文章

  1. 【WPF学习】第二十七章 Application类的任务

    上一章介绍了有关WPF应用程序中使用Application对象的方式,接下来看一下如何使用Application对象来处理一些更普通的情况,接下俩介绍如何初始化界面.如何处理命名行参数.如何处理支付窗 ...

  2. 【WPF学习】第十七章 键盘输入

    当用户按下键盘上的一个键时,就会发生一系列事件.下表根据他们的发生顺序列出了这些事件: 表 所有元素的键盘事件(按顺序) 键盘处理永远不会像上面看到的这么简单.一些控件可能会挂起这些事件中的某些事件, ...

  3. 【WPF学习】第十七章 鼠标输入

    鼠标事件执行几个关联的任务.当鼠标移到某个元素上时,可通过最基本的鼠标事件进行响应.这些事件是MouseEnter(当鼠标指针移到元素上时引发该事件)和MouseLeave(当鼠标指针离开元素时引发该 ...

  4. “全栈2019”Java第五十七章:多态与构造方法详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  5. 【WPF学习】第十三章 理解路由事件

    每个.NET开发人员都熟悉“事件”的思想——当有意义的事情发生时,由对象(如WPF元素)发送的用于通知代码的消息.WPF通过事件路由(event routing)的概念增强了.NET事件模型.事件路由 ...

  6. [汇编学习笔记][第十七章使用BIOS进行键盘输入和磁盘读写

    第十七章 使用BIOS进行键盘输入和磁盘读写 17.1 int 9 中断例程对键盘输入的处理 17.2 int 16 读取键盘缓存区 mov ah,0 int 16h 结果:(ah)=扫描码,(al) ...

  7. IntelliJ IDEA平台下JNI编程(五)—本地C代码创建Java对象及引用

    本文学习如何在C代码中创建Java对象和对象数组,前面我们学习了C代码中访问Java对象的属性和方法,其实在创建对象时本质上也就是调用构造函数,因此本文知识学习起来也很轻松.有了前面学习数组创建的方法 ...

  8. 代码初始化 故事板初始化 xib初始化总结

    对象的初始化有三种方式   // 代码创建 - (id)initWithFrame:(CGRect)frame { if (self = [super initWithFrame:frame]) { ...

  9. 【WPF学习】第二十三章 列表控件

    WPF提供了许多封装项的集合的控件,本章介绍简单的ListBox和ComboBox控件,后续哈会介绍更特殊的控件,如ListView.TreeView和ToolBar控件.所有这些控件都继承自Item ...

随机推荐

  1. 洛谷-P3369-普通平衡树(Treap)

    题目传送门 标题说平衡树,那么应该AVL,红黑树都能过,但是这次做这题主要是学习Treap,所以花了几天搞出了这题.其他方法以后再说吧 Treap(带旋转) #include <bits/std ...

  2. 数据结构与算法 python课后题(未完成)

    挖一个坑,先立个flag,后面慢慢填坑. 先放个其它人写的链接

  3. CentOS换yum源和epel源为国内源

    CentOS换源 YUM源 备份原来的repo文件 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.bac ...

  4. 转载-WebSocket协议解析

    现在,很多网站为了实现推送技术,所用的技术都是轮询.轮询是指在特定的时间间隔(如每一秒),由浏览器对服务器发起HTTP请求,然后由服务器返回数据给浏览器.由于HTTP协议是惰性的,只有客户端发起请求, ...

  5. Angular开发者指南(四)控制器

    了解控制器controller 在AngularJS中,Controller由JavaScript构造函数定义,用于扩充AngularJS Scope. 当控制器通过ng-controller指令连接 ...

  6. Metasploit MS15-020漏洞利用

    title date tags layout Metasploit MS15-020漏洞利用 2018-05-06 Metasploit post 环境: 共享机/受害者:windows xp IP: ...

  7. SQL中的一些关键字用法

    1.where 条件筛选结果 select * from `表名` where `列名`='value' 上诉语句的意思是在某表中查询某列名等于某特定值得所有列 2.Like 模糊查询 select ...

  8. 三步教你实现MyEclipse的debug远程调试

    MyEclipse远程调试程序是个神奇的东西,有时一个项目本地运行没问题可放到服务器上,同样的条件就是结果不一样:有时服务器上工程出点问题需要远程调测.于是就灰常想看一下程序在远程运行时候的状态,希望 ...

  9. 编译gcc报错make[3]: Leaving directory `/usr/local/src/gcc-7.4.0/build/gcc' make[2]: *** [all-stage1-gcc] Error 2 处理

    因业务需要安装7.4高版本gcc时报错: configure: error: in `/usr/local/src/gcc-7.4.0/build/gcc': configure: error: C+ ...

  10. 洛谷P4180【Beijing2010组队】次小生成树Tree

    题目描述: 小C最近学了很多最小生成树的算法,Prim算法.Kurskal算法.消圈算法等等.正当小C洋洋得意之时,小P又来泼小C冷水了.小P说,让小C求出一个无向图的次小生成树,而且这个次小生成树还 ...