考虑实现一种机械泵控件。
机械泵是工业中通常用来制造真空的一类设备,我们在绘制界面UI时希望可以生动形象地来表述一个机械泵,下面讲述了一种简单的实现。

声明一个MechanicalPumpControl的自定义控件,它继承自Control类。
对于一个MechanicalPump来说,它具有状态,这里定义一个State依赖属性来简单描述泵的状态,State等于0表示泵停止状态,State等于1表示泵运行状态。
State可以绑定到实际的泵状态数据源,当泵状态数据变化时,MechanicalPumpControl需要相应动态显示泵状态动画。
定义StartAnimation和StopAnimation两个依赖属性,来启动和停止泵状态动画。
关于控件的状态切换也可以考虑用VisualStateManager来实现,可以更方便地管理控件的状态以及用于状态过渡的逻辑,推荐用这种方式。
public class MechanicalPumpControl : Control
{
static MechanicalPumpControl()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(MechanicalPumpControl), new FrameworkPropertyMetadata(typeof(MechanicalPumpControl)));
}
public static readonly DependencyProperty StateProperty = DependencyProperty.Register(
"State",
typeof(int),
typeof(MechanicalPumpControl), new PropertyMetadata(PropertyChangedCallback));
public int State
{
get => (int)GetValue(StateProperty);
set => SetValue(StateProperty, value);
}
private static void PropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var pumpControl = d as MechanicalPumpControl;
if ((int)e.OldValue == 0 && (int)e.NewValue == 1)
{
pumpControl.StopAnimation = false;
pumpControl.StartAnimation = true;
}
else if ((int)e.OldValue == 1 && (int)e.NewValue == 0)
{
pumpControl.StopAnimation = true;
pumpControl.StartAnimation = false;
}
}
public static readonly DependencyProperty StartAnimationProperty = DependencyProperty.Register(
"StartAnimation",
typeof(bool),
typeof(MechanicalPumpControl));
public bool StartAnimation
{
get => (bool)GetValue(StartAnimationProperty);
set => SetValue(StartAnimationProperty, value);
}
public static readonly DependencyProperty StopAnimationProperty = DependencyProperty.Register(
"StopAnimation",
typeof(bool),
typeof(MechanicalPumpControl));
public bool StopAnimation
{
get => (bool)GetValue(StopAnimationProperty);
set => SetValue(StopAnimationProperty, value);
}
}
接下来编写MechanicalPumpControl的样式。
考虑绘制出不动的泵体部分,和动作的扇叶部分。当泵停止状态时,扇叶静止,当泵运转状态时,扇叶转动。
样式代码如下:
<Style TargetType="{x:Type Rectangle}" x:Key="FanRec">
<Setter Property="Canvas.Left" Value="7"/>
<Setter Property="Width" Value="48"/>
<Setter Property="Height" Value="6"/>
<Setter Property="RadiusX" Value="0.5"/>
<Setter Property="RadiusY" Value="0.5"/>
<Setter Property="Fill" Value="SlateGray"/>
<Setter Property="RenderTransform">
<Setter.Value>
<SkewTransform AngleX="-20" AngleY="-7"/>
</Setter.Value>
</Setter>
</Style>
<Style TargetType="{x:Type local:MechanicalPumpControl}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type local:MechanicalPumpControl}">
<Viewbox Stretch="Fill">
<Canvas Width="180" Height="100">
<Rectangle Width="60" Height="60" Canvas.Top="10" Stroke="Gray" StrokeThickness="3" RadiusX="3" RadiusY="3" />
<Canvas Width="60" Height="54" Canvas.Top="14" ClipToBounds="True">
<Canvas x:Name="canvas1" Width="60" Height="54">
<Rectangle x:Name="rec1" Style="{StaticResource FanRec}" Canvas.Top="6"/>
<Rectangle x:Name="rec2" Style="{StaticResource FanRec}" Canvas.Top="16"/>
<Rectangle x:Name="rec3" Style="{StaticResource FanRec}" Canvas.Top="26"/>
<Rectangle x:Name="rec4" Style="{StaticResource FanRec}" Canvas.Top="36"/>
<Rectangle x:Name="rec5" Style="{StaticResource FanRec}" Canvas.Top="46"/>
</Canvas>
<Canvas x:Name="canvas2" Width="60" Height="54">
<Rectangle x:Name="rec6" Style="{StaticResource FanRec}" Canvas.Top="56"/>
<Rectangle x:Name="rec7" Style="{StaticResource FanRec}" Canvas.Top="66"/>
<Rectangle x:Name="rec8" Style="{StaticResource FanRec}" Canvas.Top="76"/>
<Rectangle x:Name="rec9" Style="{StaticResource FanRec}" Canvas.Top="86"/>
<Rectangle x:Name="rec10" Style="{StaticResource FanRec}" Canvas.Top="96"/>
<Rectangle x:Name="rec11" Style="{StaticResource FanRec}" Canvas.Top="106"/>
</Canvas>
</Canvas>
<Rectangle Width="5" Height="40" Canvas.Left="60" Canvas.Top="20" Stroke="Gray" StrokeThickness="4"/>
<Rectangle Width="110" Height="82" Canvas.Left="65" RadiusX="3" RadiusY="3" Stroke="Gray" StrokeThickness="4"/>
<Rectangle Width="10" Height="8" Canvas.Left="75" Canvas.Top="82" Fill="Gray" />
<Rectangle Width="10" Height="8" Canvas.Left="155" Canvas.Top="82" Fill="Gray" />
<Rectangle Width="140" Height="10" Canvas.Left="40" Canvas.Top="90" Fill="Black" RadiusX="2" RadiusY="2"/>
</Canvas>
</Viewbox>
<ControlTemplate.Triggers>
<Trigger Property="StartAnimation" Value="true">
<Trigger.EnterActions>
<BeginStoryboard Name="moveStory">
<Storyboard RepeatBehavior="Forever" SpeedRatio="4">
<DoubleAnimationUsingKeyFrames
Storyboard.TargetName="canvas1"
Storyboard.TargetProperty="(Canvas.Top)"
Duration="0:0:6" >
<LinearDoubleKeyFrame Value="0" KeyTime="0:0:0"/>
<LinearDoubleKeyFrame Value="-10" KeyTime="0:0:1"/>
<LinearDoubleKeyFrame Value="-20" KeyTime="0:0:2"/>
<LinearDoubleKeyFrame Value="-30" KeyTime="0:0:3"/>
<LinearDoubleKeyFrame Value="-40" KeyTime="0:0:4"/>
<LinearDoubleKeyFrame Value="-50" KeyTime="0:0:5"/>
<LinearDoubleKeyFrame Value="-60" KeyTime="0:0:6"/>
</DoubleAnimationUsingKeyFrames>
<DoubleAnimationUsingKeyFrames
Storyboard.TargetName="canvas2"
Storyboard.TargetProperty="(Canvas.Top)"
Duration="0:0:6" >
<LinearDoubleKeyFrame Value="0" KeyTime="0:0:0"/>
<LinearDoubleKeyFrame Value="-10" KeyTime="0:0:1"/>
<LinearDoubleKeyFrame Value="-20" KeyTime="0:0:2"/>
<LinearDoubleKeyFrame Value="-30" KeyTime="0:0:3"/>
<LinearDoubleKeyFrame Value="-40" KeyTime="0:0:4"/>
<LinearDoubleKeyFrame Value="-50" KeyTime="0:0:5"/>
<LinearDoubleKeyFrame Value="-60" KeyTime="0:0:6"/>
</DoubleAnimationUsingKeyFrames>
</Storyboard>
</BeginStoryboard>
</Trigger.EnterActions>
</Trigger>
<Trigger Property="StopAnimation" Value="true">
<Trigger.EnterActions>
<StopStoryboard BeginStoryboardName="moveStory">
</StopStoryboard>
</Trigger.EnterActions>
</Trigger>
<Trigger Property="State" Value="1">
<Setter TargetName="rec1" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec2" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec3" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec4" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec5" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec6" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec7" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec8" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec9" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec10" Property="Fill" Value="SeaGreen"/>
<Setter TargetName="rec11" Property="Fill" Value="SeaGreen"/>
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
Xaml代码如下:
<Window x:Class="WpfApp1.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:WpfApp1"
mc:Ignorable="d"
Title="MainWindow" Height="768" Width="1366">
<Canvas>
<local:MechanicalPumpControl x:Name="pump" Canvas.Left="115" Canvas.Top="85"/>
<Button Content="START" Width="60" Height="30" Canvas.Left="363" Canvas.Top="94" Click="STARTButton_Click"/>
<Button Content="STOP" Width="60" Height="30" Canvas.Left="363" Canvas.Top="143" Click="STOPButton_Click"/>
</Canvas>
</Window>
后台代码如下:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private void STARTButton_Click(object sender, RoutedEventArgs e)
{
pump.State = 1;
}
private void STOPButton_Click(object sender, RoutedEventArgs e)
{
pump.State = 0;
}
}
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden
我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm
我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI
这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub
我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain
无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.