源码链接:https://github.com/DuelWithSelf/WPFEffects

参考:https://www.cnblogs.com/duel/p/duel_clock.html

更新一: 功能导览模块新增Binding用法示例。

更新二:仪表盘效果实现。

Binding用法与ListBox的用法一致:

Xaml定义节点样式; .cs 文件中定义数据:

 <DataTemplate x:Key="ListMenuBox.ItemTemplate" DataType="{x:Type local:CatalogOfEffect}">
<Border x:Name="BdrNavItem" Background="Transparent" Height="30"
MouseLeftButtonUp="BdrNavItem_MouseLeftButtonUp">
<TextBlock Margin="60,0,0,0" HorizontalAlignment="Left" VerticalAlignment="Center"
Foreground="White" Text="{Binding Path=Name, Mode=TwoWay}"/>
</Border>
<DataTemplate.Triggers>
<DataTrigger Binding="{Binding Path=IsSelected}" Value="True">
<Setter TargetName="BdrNavItem" Property="Background"
Value="{StaticResource ColorBrush.LightWhite}"/>
</DataTrigger>
</DataTemplate.Triggers>
</DataTemplate>
 public class BaseRecord : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged; protected void OnPropertyChanged(string prop)
{
if (this.PropertyChanged != null)
this.PropertyChanged(this, new PropertyChangedEventArgs(prop));
}
} public class CatalogOfEffect: BaseRecord
{
private string _Name;
public string Name {
get { return _Name; }
set { _Name = value; this.OnPropertyChanged("Name"); }
} private bool _IsSelected;
public bool IsSelected
{
get { return _IsSelected; }
set { _IsSelected = value; this.OnPropertyChanged("IsSelected"); }
} private string _Key;
public string Key
{
get { return _Key; }
set { _Key = value; }
}
}

引用样式:

 <CustomFrms:ListMenuBox
x:Name="LmxBinding" Text="Binding示例"
IconData="{StaticResource PathData.TagSolid}"
ItemTemplate="{StaticResource ListMenuBox.ItemTemplate}"/>

指定数据源:

  ObservableCollection<CatalogOfEffect> ltCatalogs = new System.Collections.ObjectModel.ObservableCollection<CatalogOfEffect>();
ltCatalogs.Add(new CatalogOfEffect() { Name = "淡入动效", Key="AnimFadeIn" });
ltCatalogs.Add(new CatalogOfEffect() { Name = "淡出动效", Key = "AnimFadeOut" });
ltCatalogs.Add(new CatalogOfEffect() { Name = "翻转动效", Key = "AnimFlip" });
ltCatalogs.Add(new CatalogOfEffect() { Name = "爆炸动效", Key = "AnimExpo" });
this.LmxBinding.ItemsSource = ltCatalogs;

扇形画面有很多码农通过Path、 ArcSegment等方式去构建。 Blend里面有Arc。 官方封装好的直接拿来用,省心省力。  用Path和ArcSegment的方式去实现,无非是用自己的方式再封装出一个Arc,其实没必要。

仪表盘的实现原理请参考Github上的源码。

仪表盘:代码

<UserControl x:Class="WPFEffects.Modules.Chart.ClockChart.ClockSlave2View"

xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"

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

xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"

xmlns:d="http://schemas.microsoft.com/expression/blend/2008"

xmlns:local="clr-namespace:WPFEffects.Modules.Chart.ClockChart"

xmlns:BlendCom="http://schemas.microsoft.com/expression/2010/drawing"

mc:Ignorable="d" d:DesignHeight="480" d:DesignWidth="480">

<Grid Width="370" Height="370">

<BlendCom:Arc Width="340" Height="340" StartAngle="-136"

EndAngle="136" Stretch="None" ArcThickness="60">

<BlendCom:Arc.Fill>

<LinearGradientBrush EndPoint="0.5,1" StartPoint="0.5,0">

<GradientStop Color="#3300A2FF" Offset="0"/>

<GradientStop Color="#005ED3FF" Offset="1"/>

</LinearGradientBrush>

</BlendCom:Arc.Fill>

</BlendCom:Arc>

<Grid x:Name="GdEllipse" Width="290" Height="290">

<Rectangle Width="1" Height="8" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,18.05">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="2.7"/>

<TranslateTransform X="-0.5"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

</Grid>

<Grid x:Name="GdEllipse1" Width="306" Height="306">

<Rectangle Width="1" Height="16" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,9.53">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="0"/>

<TranslateTransform X="-0.5" Y="0"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

<Rectangle Width="1" Height="16" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,9.53">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="67.5"/>

<TranslateTransform X="-0.5" Y="0"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

<Rectangle Width="1" Height="16" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,9.53">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="135"/>

<TranslateTransform X="-0.5" Y="0"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

<Rectangle Width="1" Height="16" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,9.53">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="-67.5"/>

<TranslateTransform X="-0.5" Y="0"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

<Rectangle Width="1" Height="16" VerticalAlignment="Top"

Fill="White" HorizontalAlignment="Center"

RenderTransformOrigin="0.5,9.53">

<Rectangle.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="-135"/>

<TranslateTransform X="-0.5" Y="0"/>

</TransformGroup>

</Rectangle.RenderTransform>

</Rectangle>

</Grid>

<Grid Width="270" Height="270">

<TextBlock Text="50" HorizontalAlignment="Center" VerticalAlignment="Top"

Foreground="White" FontSize="8" Margin="5"/>

<TextBlock Text="75" HorizontalAlignment="Right" VerticalAlignment="Top"

Foreground="White" FontSize="8" Margin="0,80,16,0"/>

<TextBlock Text="100" HorizontalAlignment="Right" VerticalAlignment="Bottom"

Foreground="White" FontSize="8" Margin="0,0,41,46"/>

<TextBlock Text="0" HorizontalAlignment="Left" VerticalAlignment="Bottom"

Foreground="White" FontSize="8" Margin="42,0,0,42"/>

<TextBlock Text="25" HorizontalAlignment="Left" VerticalAlignment="Top"

Foreground="White" FontSize="8" Margin="14,82,0,0"/>

</Grid>

<Grid Opacity="0">

<Grid Width="370" Height="370" x:Name="ClipProxy">

<Grid Background="Transparent"></Grid>

<BlendCom:Arc Width="370" Height="370" StartAngle="-180"

EndAngle="{Binding Path=DegreeAngle}" Fill="White"

x:Name="ArcProxy"  Stretch="None"

ArcThickness="175" >

</BlendCom:Arc>

</Grid>

</Grid>

<Grid>

<Border Width="370" Height="370" Background="Transparent">

<Border.OpacityMask>

<VisualBrush Visual="{Binding ElementName=ClipProxy}"/>

</Border.OpacityMask>

<BlendCom:Arc Width="370" Height="370" StartAngle="-135"

EndAngle="135" Stretch="None" ArcThickness="10">

<BlendCom:Arc.Fill>

<LinearGradientBrush EndPoint="1,1" StartPoint="0,1">

<GradientStop Color="#FF31FF10" Offset="0.01"/>

<GradientStop Color="#FF760F0F" Offset="1"/>

<GradientStop Color="#FFE6BF6B" Offset="0.22"/>

<GradientStop Color="#FFF7876E" Offset="0.444"/>

<GradientStop Color="#FFE22E21" Offset="0.702"/>

</LinearGradientBrush>

</BlendCom:Arc.Fill>

</BlendCom:Arc>

</Border>

<Path Stretch="Fill" Width="4" Height="95" VerticalAlignment="Top"

Margin="0,90,0,0" HorizontalAlignment="Center"

Fill="#FF109B3A" Data="M0,180 L2,184 L4,180 L2 0 v4 z"

RenderTransformOrigin="0.5,0.946">

<Path.RenderTransform>

<TransformGroup>

<ScaleTransform/>

<SkewTransform/>

<RotateTransform Angle="{Binding Path=DegreeAngle}"/>

<TranslateTransform/>

</TransformGroup>

</Path.RenderTransform>

<Path.Effect>

<DropShadowEffect Color="White" Opacity="0.5" ShadowDepth="0" BlurRadius="8"/>

</Path.Effect>

</Path>

<TextBlock x:Name="TbkValue" Text="0.0" Foreground="White" HorizontalAlignment="Center"

VerticalAlignment="Bottom" Margin="0,0,0,130"/>

</Grid>

</Grid>

</UserControl>

---------------------------------------------------------------------------------------------------------------

public partial class ClockSlave2View : UserControl

{

public double DegreeAngle

{

get { return (double)base.GetValue(DegreeAngleProperty); }

set { base.SetValue(DegreeAngleProperty, value); }

}

public static readonly DependencyProperty DegreeAngleProperty =

DependencyProperty.Register("DegreeAngle", typeof(double), typeof(ClockSlave2View),

new FrameworkPropertyMetadata(0d));

public ClockSlave2View()

{

InitializeComponent();

this.CreateNormalBorderMark();

this.DataContext = this;

this.Loaded += ClockSlave2View_Loaded;

this.Unloaded += ClockSlave2View_Unloaded;

}

private void ClockSlave2View_Unloaded(object sender, RoutedEventArgs e)

{

this.FreeSampleTimer();

}

private void ClockSlave2View_Loaded(object sender, RoutedEventArgs e)

{

this.CreateSampleTimer();

}

private void CreateNormalBorderMark()

{

this.GdEllipse.Children.Clear();

double dAngel = 270d / 100d;

for (int i = 1; i <= 100; i++)

{

if (i % 25 != 0)

{

Rectangle rect = new Rectangle();

rect.VerticalAlignment = VerticalAlignment.Top;

rect.HorizontalAlignment = HorizontalAlignment.Center;

rect.Fill = new SolidColorBrush(Colors.White);

rect.Width = 1;

rect.Height = 8;

rect.RenderTransformOrigin = new Point(0.5, 18.05);

TransformGroup transGroup = new TransformGroup();

RotateTransform rotate = new RotateTransform();

rotate.Angle = -135 + i * dAngel;

transGroup.Children.Add(rotate);

TranslateTransform translate = new TranslateTransform();

translate.X = -0.5;

transGroup.Children.Add(translate);

rect.RenderTransform = transGroup;

this.GdEllipse.Children.Add(rect);

}

}

}

private DispatcherTimer TimerSample;

private void CreateSampleTimer()

{

if(this.TimerSample == null)

{

this.TimerSample = new DispatcherTimer();

this.TimerSample.Tick += TimerSample_Tick;

this.TimerSample.Interval = TimeSpan.FromSeconds(1);

}

this.TimerSample.Start();

}

private void FreeSampleTimer()

{

if(this.TimerSample != null)

{

this.TimerSample.Stop();

this.TimerSample.Tick -= TimerSample_Tick;

}

this.TimerSample = null;

}

private void TimerSample_Tick(object sender, EventArgs e)

{

Random random = new Random();

int nData = random.Next(0, 100);

this.TbkValue.Text = nData + "";

double dAngle = -135d + 270d * nData / 100d;

DoubleAnimation anim = new DoubleAnimation(dAngle, TimeSpan.FromSeconds(0.3));

anim.EasingFunction = new ExponentialEase();

this.BeginAnimation(DegreeAngleProperty, anim);

}

}

WPF - 简单的UI框架 - 仪表盘的更多相关文章

  1. WPF - 简单的UI框架

    实现了一个简单的WPF应用程序UI框架 ,分享出来.界面效果图如下: 运行效果如下: 喜欢的可以下载源码参考:https://github.com/DuelWithSelf/WPFEffects 左侧 ...

  2. laya fgui 超简单的UI框架

    FairyGUI 超简单的UI框架 Laya使用fgui的超简单UI框架 使用场景:用于使用fgui进行layaUI开发的程序人员 整个框架分为3个模块,共有4个类: FGUIManager :FGU ...

  3. 【设计和开发一套简单自己主动化UI框架】

    !有兴趣的朋友请直接移步Github,本帖子已经不做更新,框架的详细的实现已经做了优化和代码整理,本文仅仅介绍了详细的设计思路! 目标:编写一个简单通用UI框架用于管理页面和完毕导航跳转 终于的实现效 ...

  4. 【转】发布一个基于NGUI编写的UI框架

    发布一个基于NGUI编写的UI框架 1.加载,显示,隐藏,关闭页面,根据标示获得相应界面实例 2.提供界面显示隐藏动画接口 3.单独界面层级,Collider,背景管理 4.根据存储的导航信息完成界面 ...

  5. 造轮子,模仿WPF的UI框架,还没完善。。。

    Wtf(暂时命名,随便起的 = _=),模仿WPF的框架,还没有完善,只有简单的基础元素,支持数据绑定.虽然支持mono但是mono有bug 写这个只是兴趣爱好,感觉也没多大意义了,如果这个UI框架完 ...

  6. WPF简单导航框架(Window与Page互相调用)

    相当多的WPF程序都有着丰富的页面和功能,如何使程序在不同页面间转换并降低资源占用,选择适合自己的导航框架就很重要了.最近花了一点时间做了一个简单的导航框架,并在这个过程中对Window.Page.U ...

  7. (转载)基于Unity~UGUI的简单UI框架(附UIFramework源码)

    此博客跟随siki老师的课程笔记生成,感谢siki老师的辛勤付出! 此框架功能较简单,适用于学习,可以很好的锻炼我们的设计思想 框架源码地址: UIFramework litjson.dll下载地址: ...

  8. 关于几种UI框架简单总结

    最近两年多的时间先后做过几款终端程序,UI框架从MFC转向过WxWidgets,之后再转向Qt.三种框架精通远谈不上,用起来还是没什么问题. 简单聊聊三种框架的优缺点. 1.MFC 似乎作为一种饱受批 ...

  9. 免费UI框架推荐--Charisma UI

    基于Jquery.Bootstrap的后台管理免费UI框架推荐--Charisma UI 在项目设计和开发工作中,做过一些后台管理系统的设计和开发,用的很多都是比较传统的UI框架. 老是走在这个圈子里 ...

随机推荐

  1. Centos7 firewall开放3306端口 笔记

    1. 开启端口 // zone -- 作用域 // add-port=80/tcp -- 添加端口,格式为:端口/通讯协议 // permanent -- 永久生效,没有此参数重启后失效 firewa ...

  2. OpenFaaS实战之三:Java函数

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  3. 基于小熊派Hi3861鸿蒙开发的IoT物联网学习【三】

    软件定时器:是基于系统Tick时钟中断且由软件来模拟的定时器,当经过设定的Tick时钟计数值后会触发用户定义的回调函数.定时精度与系统Tick时钟的周期有关. 定时器运行机制: cmsis_os2的A ...

  4. Select、Poll、Epoll IO复用技术

    简介 目前多进程方式实现的服务器端,一次创建多个工作子进程来给客户端提供服务, 但是创建进程会耗费大量资源,导致系统资源不足 IO复用技术就是让一个进程同时为多个客户端端提供服务 IO复用技术 之 S ...

  5. 用postman进行web端自动化测试

    概括说一下,web接口自动化测试就是模拟人的操作来进行功能自动化,主要用来跑通业务流程. 主要有两种请求方式:post和get,get请求一般用来查看网页信息:post请求一般用来更改请求参数,查看结 ...

  6. maven 标签 关于<import>标签

      标签用途:在dependecyManagement元素下用,合并此import标签上级dependency的groupId和artid中指向依赖的dependecyManagement内容   标 ...

  7. java小程序之随机任务发布器

    需求 自定义任务 随机定时 随机排序 集合输出 搭建 简单项目,所以我使用java 的maven来搭建一个控制台程序,并且使用excel来记录任务 创建任务类(sub) package com.mic ...

  8. “百度杯”CTF比赛 十月场-Getflag(md5碰撞+sql注入+网站绝对路径)

    进去md5碰撞,贴一下脚本代码 import hashlib def md5(value): return hashlib.md5(str(value).encode("utf-8" ...

  9. C++ 友元 (全局函数做友元) (类做友元) (成员函数做友元)

    1 //友元 全局函数做友元 2 /* 3 #include <iostream> 4 #include <string> 5 using namespace std; 6 7 ...

  10. 这才是做了五年Android开发该有的样子!

    程序员工作五年后一般怎样了? 五年程序员生涯对身体上的摧残就不说了,来讲讲一般会有怎样的状态吧! 优秀的一般是这样:有着明确的职业目标与规划,热爱技术,五年的工作沉淀,技术能力得到了飞速提升,每天依然 ...