侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

WPF图形解锁控件ScreenUnLock使用详解

技术  /  管理员 发布于 7年前   156

ScreenUnLock 与智能手机上的图案解锁功能一样。通过绘制图形达到解锁或记忆图形的目的。

本人突发奇想,把手机上的图形解锁功能移植到WPF中。也应用到了公司的项目中。

在创建ScreenUnLock之前,先来分析一下图形解锁的实现思路。

1.创建九宫格原点(或更多格子),每个点定义一个坐标值

2.提供图形解锁相关扩展属性和事件,方便调用者定义。比如:点和线的颜色(Color),操作模式(Check|Remember),验证正确的颜色(RightColor), 验证失败的颜色(ErrorColor), 解锁事件 OnCheckedPoint,记忆事件 OnRememberPoint 等;

3.定义MouseMove事件监听画线行为。 画线部分也是本文的核心。在画线过程中。程序需判断,线条从哪个点开始绘制,经过了哪个点(排除已经记录的点)。是否完成了绘制等等。

4.画线完成,根据操作模式处理画线完成行为。并调用相关自定义事件

大致思路如上,下面开始一步一步编写ScreenUnLock吧

创建ScreenUnLock

public partial class ScreenUnlock : UserControl

定义相关属性

/// <summary>  /// 验证正确的颜色  /// </summary>  private SolidColorBrush rightColor;  /// <summary>  /// 验证失败的颜色  /// </summary>  private SolidColorBrush errorColor;  /// <summary>  /// 图案是否在检查中  /// </summary>  private bool isChecking;  public static readonly DependencyProperty PointArrayProperty = DependencyProperty.Register("PointArray", typeof(IList<string>), typeof(ScreenUnlock));  /// <summary>  /// 记忆的坐标点   /// </summary>  public IList<string> PointArray  {   get { return GetValue(PointArrayProperty) as IList<string>; }   set { SetValue(PointArrayProperty, value); }  }  /// <summary>  /// 当前坐标点集合  /// </summary>  private IList<string> currentPointArray;  /// <summary>  /// 当前线集合  /// </summary>  private IList<Line> currentLineList;  /// <summary>  /// 点集合  /// </summary>  private IList<Ellipse> ellipseList;  /// <summary>  /// 当前正在绘制的线  /// </summary>  private Line currentLine;  public static readonly DependencyProperty OperationPorperty = DependencyProperty.Register("Operation", typeof(ScreenUnLockOperationType), typeof(ScreenUnlock), new FrameworkPropertyMetadata(ScreenUnLockOperationType.Remember));  /// <summary>  /// 操作类型  /// </summary>  public ScreenUnLockOperationType Operation  {   get { return (ScreenUnLockOperationType)GetValue(OperationPorperty); }   set { SetValue(OperationPorperty, value); }  }  public static readonly DependencyProperty PointSizeProperty = DependencyProperty.Register("PointSize", typeof(double), typeof(ScreenUnlock), new FrameworkPropertyMetadata(15.0));  /// <summary>  /// 坐标点大小   /// </summary>  public double PointSize  {   get { return Convert.ToDouble(GetValue(PointSizeProperty)); }   set { SetValue(PointSizeProperty, value); }  }  public static readonly DependencyProperty ColorProperty = DependencyProperty.Register("Color", typeof(SolidColorBrush), typeof(ScreenUnlock), new FrameworkPropertyMetadata(new SolidColorBrush(Colors.White), new PropertyChangedCallback((s, e) =>  {   (s as ScreenUnlock).Refresh();  })));  /// <summary>  /// 坐标点及线条颜色  /// </summary>  public SolidColorBrush Color  {   get { return GetValue(ColorProperty) as SolidColorBrush; }   set { SetValue(ColorProperty, value); }  }     /// <summary>     /// 操作类型     /// </summary>     public enum ScreenUnLockOperationType     {      Remember = 0, Check = 1     }

初始化ScreenUnLock

public ScreenUnlock()  {   InitializeComponent();   this.Loaded += ScreenUnlock_Loaded;   this.Unloaded += ScreenUnlock_Unloaded;   this.MouseMove += ScreenUnlock_MouseMove; //监听绘制事件  } private void ScreenUnlock_Loaded(object sender, RoutedEventArgs e)  {   isChecking = false;   rightColor = new SolidColorBrush(Colors.Green);   errorColor = new SolidColorBrush(Colors.Red);   currentPointArray = new List<string>();   currentLineList = new List<Line>();   ellipseList = new List<Ellipse>();   CreatePoint();  }  private void ScreenUnlock_Unloaded(object sender, RoutedEventArgs e)  {   rightColor = null;   errorColor = null;   if (currentPointArray != null)    this.currentPointArray.Clear();   if (currentLineList != null)    this.currentLineList.Clear();   if (ellipseList != null)    ellipseList.Clear();   this.canvasRoot.Children.Clear();  }

创建点

/// <summary>  /// 创建点  /// </summary>  private void CreatePoint()  {   canvasRoot.Children.Clear();   int row = 3, column = 3; //三行三列,九宫格   double oneColumnWidth = (this.ActualWidth == 0 ? this.Width : this.ActualWidth) / 3; //单列的宽度   double oneRowHeight = (this.ActualHeight == 0 ? this.Height : this.ActualHeight) / 3; //单列的高度   double leftDistance = (oneColumnWidth - PointSize) / 2; //单列左边距   double topDistance = (oneRowHeight - PointSize) / 2; //单列上边距   for (var i = 0; i < row; i++)   {    for (var j = 0; j < column; j++)    {     Ellipse ellipse = new Ellipse()     {      Width = PointSize,      Height = PointSize,      Fill = Color,      Tag = string.Format("{0}{1}", i, j)     };     Canvas.SetLeft(ellipse, j * oneColumnWidth + leftDistance);     Canvas.SetTop(ellipse, i * oneRowHeight + topDistance);     canvasRoot.Children.Add(ellipse);     ellipseList.Add(ellipse);    }   }  }

创建线

private Line CreateLine()  {   Line line = new Line()   {    Stroke = Color,    StrokeThickness = 2   };   return line;  }

点和线都创建都定义好了,可以开始监听绘制事件了

private void ScreenUnlock_MouseMove(object sender, System.Windows.Input.MouseEventArgs e)  {   if (isChecking) //如果图形正在检查中,不响应后续处理    return;   if (e.LeftButton == System.Windows.Input.MouseButtonState.Pressed)   {    var point = e.GetPosition(this);    HitTestResult result = VisualTreeHelper.HitTest(this, point);    Ellipse ellipse = result.VisualHit as Ellipse;    if (ellipse != null)    {     if (currentLine == null)     {      //从头开始绘制currentLine = CreateLine();      var ellipseCenterPoint = GetCenterPoint(ellipse);      currentLine.X1 = currentLine.X2 = ellipseCenterPoint.X;      currentLine.Y1 = currentLine.Y2 = ellipseCenterPoint.Y;      currentPointArray.Add(ellipse.Tag.ToString());      Console.WriteLine(string.Join(",", currentPointArray));      currentLineList.Add(currentLine);      canvasRoot.Children.Add(currentLine);     }     else     {      //遇到下一个点,排除已经经过的点      if (currentPointArray.Contains(ellipse.Tag.ToString()))       return;      OnAfterByPoint(ellipse);     }    }    else if (currentLine != null)    {     //绘制过程中     currentLine.X2 = point.X;     currentLine.Y2 = point.Y;     //判断当前Line是否经过点     ellipse = IsOnLine();     if (ellipse != null)      OnAfterByPoint(ellipse);    }   }   else   {    if (currentPointArray.Count == 0)     return;    isChecking = true;    if (currentLineList.Count + 1 != currentPointArray.Count)    {     //最后一条线的终点不在点上     //两点一线,点的个数-1等于线的条数     currentLineList.Remove(currentLine); //从已记录的线集合中删除最后一条多余的线     canvasRoot.Children.Remove(currentLine); //从界面上删除最后一条多余的线     currentLine = null;    }    if (Operation == ScreenUnLockOperationType.Check)    {     Console.WriteLine("playAnimation Check");     var result = CheckPoint(); //执行图形检查              //执行完成动画并触发检查事件     PlayAnimation(result, () =>     {      if (OnCheckedPoint != null)      {       this.Dispatcher.BeginInvoke(OnCheckedPoint, this, new CheckPointArgs() { Result = result }); //触发检查完成事件      }     });    }    else if (Operation == ScreenUnLockOperationType.Remember)    {     Console.WriteLine("playAnimation Remember");     RememberPoint(); //记忆绘制的坐标     var args = new RememberPointArgs() { PointArray = this.PointArray };             //执行完成动画并触发记忆事件     PlayAnimation(true, () =>     {      if (OnRememberPoint != null)      {       this.Dispatcher.BeginInvoke(OnRememberPoint, this, args); //触发图形记忆事件      }     });    }   }  }

判断线是否经过了附近的某个点

/// <summary>  /// 两点计算一线的长度  /// </summary>  /// <param name="pt1"></param>  /// <param name="pt2"></param>  /// <returns></returns>  private double GetLineLength(double x1, double y1, double x2, double y2)  {   return Math.Sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2)); //根据两点计算线段长度公式 √((x1-x2)²x(y1-y2)²)  }  /// <summary>  /// 判断线是否经过了某个点  /// </summary>  /// <param name="ellipse"></param>  /// <returns></returns>  private Ellipse IsOnLine()  {   double lineAB = 0; //当前画线的长度   double lineCA = 0; //当前点和A点的距离    double lineCB = 0; //当前点和B点的距离   double dis = 0;   double deciation = 1; //允许的偏差距离   lineAB = GetLineLength(currentLine.X1, currentLine.Y1, currentLine.X2, currentLine.Y2); //计算当前画线的长度   foreach (Ellipse ellipse in ellipseList)   {    if (currentPointArray.Contains(ellipse.Tag.ToString())) //排除已经经过的点     continue;    var ellipseCenterPoint = GetCenterPoint(ellipse); //取当前点的中心点    lineCA = GetLineLength(currentLine.X1, currentLine.Y1, ellipseCenterPoint.X, ellipseCenterPoint.Y); //计算当前点到线A端的长度    lineCB = GetLineLength(currentLine.X2, currentLine.Y2, ellipseCenterPoint.X, ellipseCenterPoint.Y); //计算当前点到线B端的长度    dis = Math.Abs(lineAB - (lineCA + lineCB)); //线CA的长度+线CB的长度>当前线AB的长度 说明点不在线上    if (dis <= deciation) //因为绘制的点具有一个宽度和高度,所以需设定一个允许的偏差范围,让线靠近点就命中之(吸附效果)    {     return ellipse;    }   }   return null;  }

检查点是否正确,按数组顺序逐个匹配之

/// <summary>  /// 检查坐标点是否正确  /// </summary>  /// <returns></returns>  private bool CheckPoint()  {          //PointArray:正确的坐标值数组         //currentPointArray:当前绘制的坐标值数组   if (currentPointArray.Count != PointArray.Count)    return false;   for (var i = 0; i < currentPointArray.Count; i++)   {    if (currentPointArray[i] != PointArray[i])     return false;   }   return true;  }

记录经过点,并创建一条新的线

/// <summary>  /// 记录经过的点  /// </summary>  /// <param name="ellipse"></param>  private void OnAfterByPoint(Ellipse ellipse)  {   var ellipseCenterPoint = GetCenterPoint(ellipse);   currentLine.X2 = ellipseCenterPoint.X;   currentLine.Y2 = ellipseCenterPoint.Y;   currentLine = CreateLine();   currentLine.X1 = currentLine.X2 = ellipseCenterPoint.X;   currentLine.Y1 = currentLine.Y2 = ellipseCenterPoint.Y;   currentPointArray.Add(ellipse.Tag.ToString());   Console.WriteLine(string.Join(",", currentPointArray));   currentLineList.Add(currentLine);   canvasRoot.Children.Add(currentLine);  }

/// <summary>  /// 获取原点的中心点坐标  /// </summary>  /// <param name="ellipse"></param>  /// <returns></returns>  private Point GetCenterPoint(Ellipse ellipse)  {   Point p = new Point(Canvas.GetLeft(ellipse) + ellipse.Width / 2, Canvas.GetTop(ellipse) + ellipse.Height / 2);   return p;  }

当绘制完成时,执行完成动画并触发响应模式的事件

/// <summary>  /// 执行动画  /// </summary>  /// <param name="result"></param>  private void PlayAnimation(bool result, Action callback = null)  {   Task.Factory.StartNew(() =>   {    this.Dispatcher.Invoke((Action)delegate    {     foreach (Line l in currentLineList)      l.Stroke = result ? rightColor : errorColor;     foreach (Ellipse e in ellipseList)      if (currentPointArray.Contains(e.Tag.ToString()))       e.Fill = result ? rightColor : errorColor;    });    Thread.Sleep(1500);    this.Dispatcher.Invoke((Action)delegate    {     foreach (Line l in currentLineList)      this.canvasRoot.Children.Remove(l);     foreach (Ellipse e in ellipseList)      e.Fill = Color;    });    currentLine = null;    this.currentPointArray.Clear();    this.currentLineList.Clear();    isChecking = false;   }).ContinueWith(t =>   {    try    {     if (callback != null)      callback();    }    catch (Exception ex)    {     Console.WriteLine(ex.Message);    }    finally    {     t.Dispose();    }   });  }

图形解锁的调用

<local:ScreenUnlock Width="500" Height="500"      PointArray="{Binding PointArray, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"      Operation="Check"> <!--或Remember-->      <i:Interaction.Triggers>       <i:EventTrigger EventName="OnCheckedPoint">        <Custom:EventToCommand Command="{Binding OnCheckedPoint}" PassEventArgsToCommand="True"/>       </i:EventTrigger>       <i:EventTrigger EventName="OnRememberPoint">        <Custom:EventToCommand Command="{Binding OnRememberPoint}" PassEventArgsToCommand="True"/>       </i:EventTrigger>      </i:Interaction.Triggers>     </local:ScreenUnlock>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


  • 上一条:
    VS2015下简单使用EF框架的方法
    下一条:
    基于Entity Framework自定义分页效果
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 2024.07.09日OpenAI将终止对中国等国家和地区API服务(0个评论)
    • 2024/6/9最新免费公益节点SSR/V2ray/Shadowrocket/Clash节点分享|科学上网|免费梯子(0个评论)
    • 国外服务器实现api.openai.com反代nginx配置(0个评论)
    • 2024/4/28最新免费公益节点SSR/V2ray/Shadowrocket/Clash节点分享|科学上网|免费梯子(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-07
    • 2017-08
    • 2017-09
    • 2018-01
    • 2018-07
    • 2018-08
    • 2018-09
    • 2018-12
    • 2019-01
    • 2019-02
    • 2019-03
    • 2019-04
    • 2019-05
    • 2019-06
    • 2019-07
    • 2019-08
    • 2019-09
    • 2019-10
    • 2019-11
    • 2019-12
    • 2020-01
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2020-10
    • 2020-11
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-03
    • 2022-04
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-12
    • 2024-02
    • 2024-04
    • 2024-05
    • 2024-06
    • 2025-02
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客