国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > .NET > 正文

WPF圖形解鎖控件ScreenUnLock使用詳解

2024-07-10 12:54:32
字體:
來源:轉載
供稿:網友

圖形解鎖與智能手機上的模式解鎖是相同的,解鎖圖形的目的是通過繪制圖形來實現的,感興趣的話就隨錯新技術頻道小編一起來了解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)2x(y1-y2)2)  }  /// <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>

以上就是關于WPF圖形解鎖控件ScreenUnLock使用詳解,朋友們快來js.VeVb.com享受編程的樂趣吧,希望大家繼續支持我們吧!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 忻州市| 安徽省| 奎屯市| 濉溪县| 杭州市| 朔州市| 尉氏县| 宜春市| 瑞昌市| 新巴尔虎左旗| 彰化县| 裕民县| 新蔡县| 公安县| 吴堡县| 马鞍山市| 台安县| 连州市| 阜城县| 炎陵县| 工布江达县| 葫芦岛市| 岑溪市| 德昌县| 石渠县| 湾仔区| 上栗县| 临汾市| 彰武县| 吴堡县| 颍上县| 珲春市| 成安县| 芦山县| 宝应县| 洛扎县| 思茅市| 上林县| 咸丰县| 弋阳县| 界首市|