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

这篇文章主要为大家详细介绍了wpf图形解锁控件screenunlock的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

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

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

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

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

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

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

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

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

创建ScreenUnLock

public partial class ScreenUnlock : UserControl

登录后复制

定义相关属性

///   /// 验证正确的颜色  ///   private SolidColorBrush rightColor;  ///   /// 验证失败的颜色  ///   private SolidColorBrush errorColor;  ///   /// 图案是否在检查中  ///   private bool isChecking;  public static readonly DependencyProperty PointArrayProperty = DependencyProperty.Register("PointArray", typeof(IList), typeof(ScreenUnlock));  ///   /// 记忆的坐标点   ///   public IList PointArray  {   get { return GetValue(PointArrayProperty) as IList; }   set { SetValue(PointArrayProperty, value); }  }  ///   /// 当前坐标点集合  ///   private IList currentPointArray;  ///   /// 当前线集合  ///   private IList currentLineList;  ///   /// 点集合  ///   private IList ellipseList;  ///   /// 当前正在绘制的线  ///   private Line currentLine;  public static readonly DependencyProperty OperationPorperty = DependencyProperty.Register("Operation", typeof(ScreenUnLockOperationType), typeof(ScreenUnlock), new FrameworkPropertyMetadata(ScreenUnLockOperationType.Remember));  ///   /// 操作类型  ///   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));  ///   /// 坐标点大小   ///   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();  })));  ///   /// 坐标点及线条颜色  ///   public SolidColorBrush Color  {   get { return GetValue(ColorProperty) as SolidColorBrush; }   set { SetValue(ColorProperty, value); }  }     ///      /// 操作类型     ///      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();   currentLineList = new List();   ellipseList = new List();   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();  }

登录后复制

创建点

///   /// 创建点  ///   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 

创建线


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); //触发图形记忆事件      }     });    }   }  }

登录后复制

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

///   /// 两点计算一线的长度  ///   ///   ///   ///   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)²)  }  ///   /// 判断线是否经过了某个点  ///   ///   ///   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 

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


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

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


///   /// 记录经过的点  ///   ///   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);  }

登录后复制

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

登录后复制

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

///   /// 执行动画  ///   ///   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();    }   });  }

登录后复制

图形解锁的调用

                                                              

登录后复制

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

以上就是ScreenUnLock-图形解锁控件使用详解的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2493740.html

(0)
上一篇 2025年3月5日 01:37:42
下一篇 2025年2月22日 13:25:52

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • C#中发送邮件的实现方法详解

    这篇文章主要为大家详细介绍了c#实现发送邮件的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 本文实例为大家分享了Android九宫格图片展示的具体代码,供大家参考,具体内容如下 #region 发送邮件部分    private …

    编程技术 2025年3月5日
    200
  • c#接口的问题的解决办法详解

    这段时间的项目有用到接口,开始不是特别理解接口,只是单单知道接口定义非常简单,甚至觉得这个接口只是多此一举(个人开发的时候)。现在开始团队开发,才发现接口原来是这么的重要和便捷! 接下来就来谈谈我这段时间对接口使用的粗浅见解,说的对希望大家…

    编程技术 2025年3月5日
    200
  • C#中的抽象类与接口的详解

    问题出现: 我们在使用C#的抽象类和接口的时候,往往会遇到以下类似的问题,大致归纳如下: (1)抽象类和接口有什么本质的区别和联系? (2)什么时候选择使用抽象类,然啥时候使用接口最恰当呢? (3)在项目中怎样使用才能使得项目更具有可维护性…

    编程技术 2025年3月5日
    200
  • 代码详解AVL树的插入

    AVL树被称为高度平衡的二叉搜索树,尽量降低二叉树的高度,来保持二叉树的平衡,减少树的平均搜索长度。 avl树的性质:1、左子树和右子树的高度之差(绝对值)不超过1                         2、树中的每棵子树都是AV…

    2025年3月5日 编程技术
    200
  • 什么是XML?xml的实例讲解

    目录结构: contents structure [-] 什么是XML 解析XML 解析XML的两种方式 使用dom4j解析xml dom4j的部分API 打印一个XML文件的全部内容 在dom4j中应用XPath解析XML 相关的部分AP…

    2025年3月5日 编程技术
    200
  • 如何在Go中使用数据库?

    随着web应用程序和其他复杂应用程序在互联网上的广泛普及,数据库已经成为不可或缺的部分。go是一种流行的编程语言,可以用于web应用程序和其他系统开发。在本文中,我们将探讨如何在go中使用数据库。 选择数据库 在开始使用数据库之前,首先需要…

    编程技术 2025年3月4日
    200
  • 如何在Go中使用条件语句?

    在go中,条件语句是控制程序流程的关键之一。在编写代码时,我们经常需要使用条件语句来实现特定的逻辑控制。在本文中,我们将讨论在go语言中如何使用条件语句。 If语句 if语句是Go中最常见的条件语句之一。它根据一个布尔表达式的值来决定是否执…

    编程技术 2025年3月4日
    200
  • 如何使用Go语言进行数据库操作

    如何使用go语言进行数据库操作 引言:Go语言是一种高效且简洁的编程语言,拥有强大的并发能力和优秀的性能表现。在开发过程中,与数据库的交互是一个非常重要的环节。本文将介绍如何使用go语言进行数据库操作,包括连接数据库、CRUD操作以及事务处…

    编程技术 2025年3月4日
    200
  • 如何使用Go语言进行Web开发

    如何使用go语言进行web开发 近年来,Go语言(也称为Golang)在Web开发领域变得越来越流行。其快速的执行速度和简单易用的语法使其成为开发人员的首选。本文将介绍如何使用go语言进行web开发,并附上一些代码示例。 一、安装和配置Go…

    编程技术 2025年3月4日
    200
  • 字节跳动在使用Golang吗

    《字节跳动的Golang应用探秘:探寻背后的代码世界》 字节跳动,作为一家知名的互联网公司,一直以技术领先而闻名。除了在移动端和Web端开发中广泛使用的Java、Python等编程语言外,近年来,字节跳动越来越注重Golang(Go语言)的…

    2025年3月4日
    200

发表回复

登录后才能评论