《Programming WPF》翻译 第5章 1.不使用样式

原文:《Programming WPF》翻译 第5章 1.不使用样式

作为一个样式如何使其在WPF使用的例子,,让我们看一下TTT简单的实现,如示例5-1。

示例5-1

<!-- Window1.xaml -->

<Window

    x:Class="TicTacToe.Window1"

    xmlns="http://schemas.microsoft.com/winfx/avalon/2005"

    xmlns:x="http://schemas.microsoft.com/winfx/xaml/2005"

    Text="TicTacToe">

  <!-- the black background lets the tic-tac-toe -->

  <!-- crosshatch come through on the margins -->

  <Grid Background="Black">

    <Grid.RowDefinitions>

      <RowDefinition />

      <RowDefinition />

      <RowDefinition />

    </Grid.RowDefinitions>

    <Grid.ColumnDefinitions>

      <ColumnDefinition />

      <ColumnDefinition />

      <ColumnDefinition />

    </Grid.ColumnDefinitions>

    <Button Margin="0,0,2,2" Grid.Row="0" Grid.Column="0" x:Name="cell00" />

    <Button Margin="2,0,2,2" Grid.Row="0" Grid.Column="1" x:Name="cell01" />

    <Button Margin="2,0,0,2" Grid.Row="0" Grid.Column="2" x:Name="cell02" />

    <Button Margin="0,2,2,2" Grid.Row="1" Grid.Column="0" x:Name="cell10" />

    <Button Margin="2,2,2,2" Grid.Row="1" Grid.Column="1" x:Name="cell11" />

    <Button Margin="2,2,0,2" Grid.Row="1" Grid.Column="2" x:Name="cell12" />

    <Button Margin="0,2,2,0" Grid.Row="2" Grid.Column="0" x:Name="cell20" />

    <Button Margin="2,2,2,0" Grid.Row="2" Grid.Column="1" x:Name="cell21" />

    <Button Margin="2,2,0,0" Grid.Row="2" Grid.Column="2" x:Name="cell22" />

  </Grid>

</Window>

这个grid的外观上排列了一组9个按钮在一个3X3栅格的TTT单元中,在按钮上使用了页面空白为了TTT的交叉线阴影。对游戏逻辑的一个简单的实现,在xaml后台代码中,如示例5-2所示。

示例5-2

// Window1.xaml.cs

namespace TicTacToe {

  public partial class Window1 : Window {

    // Track the current player (X or O)

    string currentPlayer;

    // Track the list of cells for finding a winner etc.

    Button[] cells;

    public Window1( ) {

      InitializeComponent( );

      // Cache the list of buttons and handle their clicks

      this.cells = new Button[] { this.cell00, this.cell01,  };

      foreach( Button cell in this.cells ) {

        cell.Click += cell_Click;

      }

      // Initialize a new game

      NewGame( );

    }

    // Wrapper around the current player for future expansion,

    // e.g. updating status text with the current player

    string CurrentPlayer {

      get { return this.currentPlayer; }

      set { this.currentPlayer = value; }

    }

    // Use the buttons to track game state

    void NewGame( ) {

      foreach( Button cell in this.cells ) {

        cell.Content = null;

      }

      CurrentPlayer = "X";

    }

    void cell_Click(object sender, RoutedEventArgs e) {

      Button button = (Button)sender;

      // Don‘t let multiple clicks change the player for a cell

      if( button.Content != null ) { return; }

      // Set button content

      button.Content = CurrentPlayer;

      // Check for winner or a tie

      if( HasWon(this.currentPlayer) ) {

        MessageBox.Show("Winner!", "Game Over");

        NewGame( );

        return;

      }

      else if( TieGame( ) ) {

        MessageBox.Show("No Winner!", "Game Over");

        NewGame( );

        return;

      }

      // Switch player

      if( CurrentPlayer == "X" ) {

        CurrentPlayer = "O";

      }

      else {

        CurrentPlayer = "X";

      }

    }

    // Use this.cells to find a winner or a tie

    bool HasWon(string player) {}

    bool TieGame( ) {}

  }

}

我们的简单TTT逻辑使用字符串代表玩家,使用按钮来跟踪游戏状态。当点击任意一个按钮时,我们将内容设置为字符串,用来象征当前玩家以及转换玩家。当游戏结束的时候,每一个按钮上的内容都会被清除。游戏中的截图如图5-1。

图5-1

注意到图5-1中,grid的背景来自页面的空白。这些空白差不多使grid看上去像一个可绘制的TTT木板(虽然我们将来会做的更好)。然而,如果我们真的指望模仿一个手绘的游戏,我们已经对按钮上的字体大小做了设置,但并没匹配到线条的厚度。

一种修复这个问题的方法是为每一个按钮对象设置字体和宽度,如示例5-3。

示例5-3

<Button FontSize="32" FontWeight="Bold"  x:Name="cell00" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell01" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell02" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell10" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell11" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell12" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell20" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell21" />

<Button FontSize="32" FontWeight="Bold" x:Name="cell22" />

依照我的视觉敏感性,今天,虽然这样做使得X的和O的外观更好,一旦我以后想改动它,我就要负责在9个独立的地方改变这些属性,这是重复性的努力——违反了我的编码敏感性。我宁愿重制我的决定——为了以后的维护,将我的TTT单元的外观放在一个共同的地方。这是样式派得上用场的地方。

时间: 2024-12-14 11:15:39

《Programming WPF》翻译 第5章 1.不使用样式的相关文章

《Programming WPF》翻译 第5章 4.元素类型样式

原文:<Programming WPF>翻译 第5章 4.元素类型样式 命名样式非常有用,当你得到一组属性并应用到特点的元素上.然而,如果你想要应用一个统一的样式到所有确定元素类型的实例,设置TargetType而不用一个Key,如示例5-16所示. 示例5-16 <!-- no Key --> <Style TargetType="{x:Type Button}">   <Setter Property="FontSize"

《Programming WPF》翻译 第5章 2.内嵌样式

原文:<Programming WPF>翻译 第5章 2.内嵌样式 每一个“可样式化”的WPF元素都有一个Style属性,可以在内部设置这个属性--使用XAML属性-元素的语法(在第一章讨论的),如示例5-4. 示例5-4 <Button  x:Name="cell00" />   <Button.Style>     <Style>       <Setter Property="Button.FontSize"

《Programming WPF》翻译 第6章 2.资源与样式

原文:<Programming WPF>翻译 第6章 2.资源与样式 WPF的样式机制以来于资源体系来定位样式.正如你在第5章看到的,样式在元素的资源片段中定义,而且样式通过其名字被引用,正如示例6-18所示: 示例6-18 <Window x:Class="ResourcePlay.Window1" Text="ResourcePlay"     xmlns="http://schemas.microsoft.com/winfx/ava

《Programming WPF》翻译 第8章 5.创建动画过程

原文:<Programming WPF>翻译 第8章 5.创建动画过程 所有在这章使用xaml举例说明的技术,都可以在代码中使用,正如你希望的.可是,代码可以使用动画在某种程度上不可能在xaml中实现的. 在代码中创建动画需要稍微多一点的努力--比使用标记.然而,代码提供了更多的弹性.你可以在运行期计算属性,而不是在xaml中硬编码,从而支持你的动画适应环境.例如,这可能是有用的--在当前窗体的大小基于动画的参数. 使用代码一个额外的好处是我们不需要使用storyboard,替代的,我们可以创

《Programming WPF》翻译 第8章 2.Timeline

原文:<Programming WPF>翻译 第8章 2.Timeline Timeline代表了时间的延伸.它通常还描述了一个或多个在这段时间所发生的事情.例如,在前面章节描述的动画类型,都是Timeline.可哦率这样的DoubleAnimation: <DoubleAnimation From=”10” To=”300” Duration=”0:0:5” /> 正如Duration属性指出的,这代表了一个5秒的时间长度.所有类型的Timeline总是有一个开始时间和一个持续时

《Programming WPF》翻译 第9章 6.我们进行到哪里了?

原文:<Programming WPF>翻译 第9章 6.我们进行到哪里了? 只有当任何内嵌控件都没有提供你需要的底层行为时,你将要写一个自定义控件.当你写一个自定义控件,你将要使用到依赖属性系统,来提供支持数据绑定和动画的属性.你将使用routed事件结构来暴露事件.如果你想写一个没有外观的控件,允许其可视化能被替换--如内嵌控件,你必须考虑你的控件和模板之间如何进行交互.你还将要为一个提供了一组默认可视化的模板提供一个默认值.

《Programming WPF》翻译 第9章 4.模板

原文:<Programming WPF>翻译 第9章 4.模板 对一个自定义元素最后的设计考虑是,它是如何连接其可视化的.如果一个元素直接从FrameworkElement中派生,这将会适当的生成它自己的可视化.(第7章描述了如何创建一个图形外观.)尤其是,如果你创建了一个元素,是为了提供一个特定的可视化表现,该元素应该完全控制这个可视化是如何管理的,一旦你编写了一个控件,通常你不会将一个图形硬编码到里面. 记住,一个控件的工作是提供行为.可视化是由控件模板提供的.这种可视化是由控件模板提供的

《Programming WPF》翻译 第9章 2.选择一个基类

原文:<Programming WPF>翻译 第9章 2.选择一个基类 WPF提供了很多类,当创建一个自定义元素时,你可以从这些类中派生.图9-1显示了一组可能作为类--可能是合适的基类,并且说明了他们之间的继承关系.注意到,这决不是完整的继承关系图,只是简单的显示了一些你应该考虑的可能的基类. 无论你选择了哪一个基类,你的元素都会直接或间接地从FrameworkElement派生.这将提供routing事件,高级属性处理,动画,数据绑定,外观上的支持,样式,以及逻辑树的集成. 派生于Fram

《Programming WPF》翻译 第8章 4.关键帧动画

原文:<Programming WPF>翻译 第8章 4.关键帧动画 到目前为止,我们只看到简单的点到点的动画.我们使用了To和From属性或者By属性来设计动画--相对于当前的属性值.这很适合简单的动画,但是我们可以构造序列来创建更复杂的动画,这可能是非常麻烦的.幸运的是,这是没有必要的.WPF提供了动画对象,允许我们详细指出一系列时间和值. 在影视中传统的动画中,这是普通的开始--通过绘制最重要的动画步骤.这些关键帧定义了场景的基本流程,捕获了它的最重要的点.只要一旦这些关键帧是满意的,是