Wpf登陸驗證方式(2)-文字點選的實現

實現思路:express

1.隨機生成四個漢字數組

2.區域從左到右均分紅四個區域,每一個區域隨機一個點app

3.在隨機的四個點上顯示漢字,漢字隨機旋轉必定角度。dom

4.對這四個漢字進行隨機排序,而後選擇前三個做爲點擊順序,並顯示在界面下方指引用戶。函數

5.點擊漢字顯示一個定位標記,內部有個文字,按照點擊順序從1標記到3.this

截圖以下:編碼

 

按照順序點擊後spa

 

 

 XAML代碼以下code

<UserControl x:Class="Util.Controls.TextClickVerify"
             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:controls="clr-namespace:Util.Controls"
             mc:Ignorable="d" 
             d:DesignHeight="450" d:DesignWidth="800">

    <UserControl.Resources>
        <!--Button模板-->
        <ControlTemplate x:Key="DefaultButton_Template" TargetType="{x:Type Button}">
            <Border x:Name="border" Background="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, Path=Background}" 
                                    Height="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, Path=Height}" 
                                    CornerRadius="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, Path=(controls:ControlAttachProperty.CornerRadius)}" 
                                    BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}"
                                    Width="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, Path=Width}">
                <!--Text-->
                <Grid VerticalAlignment="Center"
                        Margin="{TemplateBinding Padding}"
                        HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}">
                    <ContentPresenter RecognizesAccessKey="True" VerticalAlignment="Center"  SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}"/>
                </Grid>
            </Border>
            <!--觸發器-->
            <ControlTemplate.Triggers>
                <!--設置鼠標進入時的背景、前景樣式-->
                <Trigger Property="IsMouseOver" Value="True">
                    <Setter Property="Background" Value="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, 
                                Path=(controls:ControlAttachProperty.MouseOverBackground)}" TargetName="border" />
                    <Setter Property="Foreground" Value="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, 
                                Path=(controls:ControlAttachProperty.MouseOverForeground)}"/>
                </Trigger>
                <!--鼠標按下時的前景、背景樣式-->
                <Trigger Property="IsPressed" Value="True">
                    <Setter Property="Background" Value="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, 
                                Path=(controls:ControlAttachProperty.PressedBackground)}" TargetName="border" />
                    <Setter Property="Foreground" Value="{Binding RelativeSource={RelativeSource Mode=TemplatedParent}, 
                                Path=(controls:ControlAttachProperty.PressedForeground)}" />
                </Trigger>
                <Trigger Property="IsEnabled" Value="false">
                    <Setter Property="Opacity" Value="0.5" TargetName="border"/>
                </Trigger>
            </ControlTemplate.Triggers>
        </ControlTemplate>

        <Style x:Key="DefaultButtonStyle" TargetType="{x:Type Button}">
            <Setter Property="Background" Value="{DynamicResource AccentColorBrush}" />
            <Setter Property="Foreground" Value="{DynamicResource WhiteBrush}" />
            <Setter Property="controls:ControlAttachProperty.MouseOverBackground" Value="{DynamicResource GrayBrush8}" />
            <Setter Property="controls:ControlAttachProperty.MouseOverForeground" Value="{StaticResource BlackBrush}" />
            <Setter Property="controls:ControlAttachProperty.PressedBackground" Value="{DynamicResource BlackBrush}" />
            <Setter Property="controls:ControlAttachProperty.PressedForeground" Value="{DynamicResource WhiteBrush}" />
            <Setter Property="HorizontalContentAlignment" Value="Center" />
            <Setter Property="controls:ControlAttachProperty.CornerRadius" Value="0" />
            <Setter Property="Padding" Value="0" />
            <Setter Property="Content" Value="{x:Null}" />
            <Setter Property="MinHeight" Value="22" />
            <Setter Property="Template" Value="{StaticResource DefaultButton_Template}"/>
            <Setter Property="BorderThickness" Value="1"/>
        </Style>
    </UserControl.Resources>
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="*"/>
            <RowDefinition Height="30"/>
        </Grid.RowDefinitions>
        <Canvas x:Name="myCanvas">
         
        </Canvas>

        <TextBlock x:Name="txtInfo" Grid.Row="1" VerticalAlignment="Center" HorizontalAlignment="Center"/>
        <Button x:Name="btnReset" Grid.Row="1" Visibility="Collapsed" Style="{StaticResource DefaultButtonStyle}"/>
    </Grid>
</UserControl>

  cs代碼以下:component

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Controls.Primitives;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;

namespace Util.Controls
{

    /// <summary>
    /// TextClickVerify.xaml 的交互邏輯
    /// </summary>
    public partial class TextClickVerify : UserControl
    {
        public TextClickVerify()
        {
            InitializeComponent();

            this.Loaded += TextClickVerify_Loaded; ;
            myCanvas.MouseLeftButtonDown += MyCanvas_MouseLeftButtonDown;
            btnReset.Click += BtnReset_Click;
        }

        private void BtnReset_Click(object sender, RoutedEventArgs e)
        {
            Restart();
        }


        private void MyCanvas_MouseLeftButtonDown(object sender, MouseButtonEventArgs e)
        {
            if (btnReset.Visibility == Visibility.Visible)
            {
                Restart();
                return;
            }

            var position = e.GetPosition(myCanvas);
            if (e.OriginalSource.GetType() == typeof(Grid))
            {
                Grid grid = (Grid)e.OriginalSource;
                if (grid.Tag.ToString() == strs.FirstOrDefault())
                {
                    AddPath(4 - strs.Count, position.X, position.Y);
                    strs.RemoveAt(0);
                    if (strs.Count == 0)
                    {
                        Result = true;           
                        RaiseResultChanged(Result);
                        txtInfo.Visibility = Visibility.Collapsed;
                        btnReset.Visibility = Visibility.Visible;
                        btnReset.Content = "驗證成功";
                        btnReset.Background = Brushes.Green;
                    }
                }
                else
                {
                    AddPath(4 - strs.Count, position.X, position.Y);
                    RaiseResultChanged(Result);
                    txtInfo.Visibility = Visibility.Collapsed;
                    btnReset.Visibility = Visibility.Visible;
                    btnReset.Content = "驗證失敗,請重試";
                    btnReset.Background = Brushes.Red;
                }
            }
            else
            {
                AddPath(4 - strs.Count, position.X, position.Y);
                RaiseResultChanged(Result);
                txtInfo.Visibility = Visibility.Collapsed;
                btnReset.Visibility = Visibility.Visible;
                btnReset.Content = "驗證失敗,請重試";
                btnReset.Background = Brushes.Red;
            }
        }

        public bool Result
        {
            get { return (bool)GetValue(ResultProperty); }
            set { SetValue(ResultProperty, value); }
        }

        public static readonly DependencyProperty ResultProperty =
            DependencyProperty.Register("Result", typeof(bool), typeof(TextClickVerify), new PropertyMetadata(false));

        public string ImageUri
        {
            get { return (string)GetValue(ImageUriProperty); }
            set { SetValue(ImageUriProperty, value); }
        }

        public static readonly DependencyProperty ImageUriProperty =
            DependencyProperty.Register("ImageUri", typeof(string), typeof(TextClickVerify), new PropertyMetadata(null));

        #region Routed Event
        public static readonly RoutedEvent ResultChangedEvent = EventManager.RegisterRoutedEvent("ResultChanged", RoutingStrategy.Bubble, typeof(ResultChangedEventHandler), typeof(TextClickVerify));
        public event ResultChangedEventHandler ResultChanged
        {
            add { AddHandler(ResultChangedEvent, value); }
            remove { RemoveHandler(ResultChangedEvent, value); }
        }
        void RaiseResultChanged(bool result)
        {
            var arg = new RoutedEventArgs(ResultChangedEvent, result);
            RaiseEvent(arg);
        }
        #endregion

        private void TextClickVerify_Loaded(object sender, RoutedEventArgs e)
        {
            Restart();
        }

        private List<string> strs;
        private void Restart()
        {
            if (!myCanvas.IsVisible)
                return;

            Result = false;

            Random ran = new Random();

            BitmapImage image = GetBitmapImage();
            SetBackground(image);

            //獲取GB2312編碼頁(表) 
            Encoding gb = Encoding.GetEncoding("gb2312");

            string str1;
            string str2;
            string str3;
            string str4;

                //調用函數產生4個隨機中文漢字編碼 
                object[] bytes = ChineseCode.CreateRegionCode(4);

                //根據漢字編碼的字節數組解碼出中文漢字 
                str1 = gb.GetString((byte[])Convert.ChangeType(bytes[0], typeof(byte[])));
                str2 = gb.GetString((byte[])Convert.ChangeType(bytes[1], typeof(byte[])));
                str3 = gb.GetString((byte[])Convert.ChangeType(bytes[2], typeof(byte[])));
                str4 = gb.GetString((byte[])Convert.ChangeType(bytes[3], typeof(byte[])));
            
            strs = new List<string>();
            strs.Add(str1);
            strs.Add(str2);
            strs.Add(str3);
            strs.Add(str4);
            strs = strs.OrderBy(p => ran.NextDouble()).Take(3).ToList();

            int width = (int)(myCanvas.ActualWidth - 30);
            int height = (int)(myCanvas.ActualHeight - 40);
            var brush = Application.Current.FindResource("AccentColorBrush") as Brush;


            myCanvas.Children.Clear();
            AddChild(str1, 0, brush, width, height, ran);
            AddChild(str2, 1, brush, width, height, ran);
            AddChild(str3, 2, brush, width, height, ran);
            AddChild(str4, 3, brush, width, height, ran);

            txtInfo.Visibility = Visibility.Visible;
            txtInfo.Text = $"請依次點擊\"{strs[0]}\"\"{strs[1]}\"\"{strs[2]}\"";
            btnReset.Visibility = Visibility.Collapsed;
            btnReset.Background = Brushes.Transparent;
        }
     
        public void AddChild(string str, int index, Brush brush, int width, int height, Random ran)
        {
            Grid grid = new Grid();
            grid.Tag = str;
            OutlineText outlinetext = new OutlineText()
            {
                FontSize = 30,
                Text = str,
                FontWeight = FontWeights.Bold,
                Fill = new SolidColorBrush (Color.FromRgb(Convert.ToByte(ran.Next(0, 255)), Convert.ToByte(ran.Next(0, 255)), Convert.ToByte(ran.Next(0, 255)))),//brush,
                IsHitTestVisible = false,
            };
            grid.Children.Add(outlinetext);

            SetLeft(grid, ran.Next((int)(width * index / 4)  , (int)(width * (index + 1)/ 4)));
            SetTop(grid, ran.Next(0, (int)height));
            RotateTransform rtf = new RotateTransform(ran.Next(0, 360), 15, 20);
            grid.RenderTransform = rtf;
          
            grid.Background = new SolidColorBrush(Colors.Transparent);
            myCanvas.Children.Add(grid);
        }

        private void AddPath(int number, double left, double top)
        {
            Grid grid = new Grid();

            Path path = new Path();
            path.Fill = Application.Current.FindResource("AccentColorBrush") as Brush;// Application.Current.FindResource("BlackBrush") as Brush;
            path.Stroke = Application.Current.FindResource("WhiteBrush") as Brush;
            string sData = "M12,2A7,7 0 0,0 5,9C5,14.25 12,22 12,22C12,22 19,14.25 19,9A7,7 0 0,0 12,2Z";
            var converter = TypeDescriptor.GetConverter(typeof(Geometry));
            path.Data = (Geometry)converter.ConvertFrom(sData);
            path.Height = 40;
            path.Width = 30;
            path.StrokeThickness = 2;
            path.Stretch = Stretch.Fill;
            path.HorizontalAlignment = HorizontalAlignment.Center;
            path.VerticalAlignment = VerticalAlignment.Center;

            grid.Children.Add(path);

            TextBlock text = new TextBlock();
            text.Text = number.ToString();
            text.Foreground = Application.Current.FindResource("WhiteBrush") as Brush;
            text.HorizontalAlignment = HorizontalAlignment.Center;
            text.VerticalAlignment = VerticalAlignment.Center;
            text.Margin = new Thickness(0, 0, 0, 2);
            grid.Children.Add(text);

            myCanvas.Children.Add(grid);
            SetLeft(grid, left - path.Width / 2);
            SetTop(grid, top - path.Height);
        }

        private BitmapImage GetBitmapImage()
        {           
            Random ran = new Random();
            int value = ran.Next(1, 3);

            // Create source.
            BitmapImage image = new BitmapImage();
            // BitmapImage.UriSource must be in a BeginInit/EndInit block.
            image.BeginInit();
            image.UriSource = new Uri(ImageUri ?? $"pack://application:,,,/Util.Controls;component/Resources/{value}.jpg");
            image.DecodePixelWidth = (int)myCanvas.ActualWidth;
            image.DecodePixelHeight = (int)myCanvas.ActualHeight;
            image.EndInit();

            return image;
        }

        private void SetBackground(BitmapImage image)
        {
            ImageBrush ib = new ImageBrush();
            ib.ImageSource = image;

            myCanvas.Background = ib;
        }
       

        private void SetVerCenter(FrameworkElement element)
        {
            double top = (myCanvas.ActualHeight - element.ActualHeight) / 2;
            Canvas.SetTop(element, top);
        }

        private void SetLeft(FrameworkElement element, double left)
        {
            Canvas.SetLeft(element, left);
        }

        private void SetTop(FrameworkElement element, double top)
        {
            Canvas.SetTop(element, top);
        }



    }
}

  隨機漢字的代碼,網上找的一個

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Util.Controls
{
    class ChineseCode
    {
        public static void Test()
        {
            //獲取GB2312編碼頁(表) 
            Encoding gb = Encoding.GetEncoding("gb2312");

            //調用函數產生4個隨機中文漢字編碼 
            object[] bytes = CreateRegionCode(4);

            //根據漢字編碼的字節數組解碼出中文漢字 
            string str1 = gb.GetString((byte[])Convert.ChangeType(bytes[0], typeof(byte[])));
            string str2 = gb.GetString((byte[])Convert.ChangeType(bytes[1], typeof(byte[])));
            string str3 = gb.GetString((byte[])Convert.ChangeType(bytes[2], typeof(byte[])));
            string str4 = gb.GetString((byte[])Convert.ChangeType(bytes[3], typeof(byte[])));

            //輸出的控制檯 
            Console.WriteLine(str1 + str2 + str3 + str4);
        }


        /**/
        /* 
        此函數在漢字編碼範圍內隨機建立含兩個元素的十六進制字節數組,每一個字節數組表明一個漢字,並將 
        四個字節數組存儲在object數組中。 
        參數:strlength,表明須要產生的漢字個數 
        */
        public static object[] CreateRegionCode(int strlength)
        {
            //定義一個字符串數組儲存漢字編碼的組成元素 
            string[] rBase = new String[16] { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };

            Random rnd = new Random();

            //定義一個object數組用來 
            object[] bytes = new object[strlength];

            /**/
            /*每循環一次產生一個含兩個元素的十六進制字節數組,並將其放入bject數組中 
            每一個漢字有四個區位碼組成 
            區位碼第1位和區位碼第2位做爲字節數組第一個元素 
            區位碼第3位和區位碼第4位做爲字節數組第二個元素 
            */
            for (int i = 0; i < strlength; i++)
            {
                //區位碼第1位 
                int r1 = rnd.Next(11, 14);
                string str_r1 = rBase[r1].Trim();

                //區位碼第2位 
                rnd = new Random(r1 * unchecked((int)DateTime.Now.Ticks) + i);//更換隨機數發生器的 

                //種子避免產生重複值 
                int r2;
                if (r1 == 13)
                {
                    r2 = rnd.Next(0, 7);
                }
                else
                {
                    r2 = rnd.Next(0, 16);
                }
                string str_r2 = rBase[r2].Trim();

                //區位碼第3位 
                rnd = new Random(r2 * unchecked((int)DateTime.Now.Ticks) + i);
                int r3 = rnd.Next(10, 16);
                string str_r3 = rBase[r3].Trim();

                //區位碼第4位 
                rnd = new Random(r3 * unchecked((int)DateTime.Now.Ticks) + i);
                int r4;
                if (r3 == 10)
                {
                    r4 = rnd.Next(1, 16);
                }
                else if (r3 == 15)
                {
                    r4 = rnd.Next(0, 15);
                }
                else
                {
                    r4 = rnd.Next(0, 16);
                }
                string str_r4 = rBase[r4].Trim();

                //定義兩個字節變量存儲產生的隨機漢字區位碼 
                byte byte1 = Convert.ToByte(str_r1 + str_r2, 16);
                byte byte2 = Convert.ToByte(str_r3 + str_r4, 16);
                //將兩個字節變量存儲在字節數組中 
                byte[] str_r = new byte[] { byte1, byte2 };

                //將產生的一個漢字的字節數組放入object數組中 
                bytes.SetValue(str_r, i);

            }

            return bytes;

        }
    }
}

 

控件使用方法

<util:TextClickVerify x:Name="verify2" Width="300" Height="300">
                        <i:Interaction.Triggers>
                            <i:EventTrigger EventName="ResultChanged">
                                <i:InvokeCommandAction Command="{Binding ResultChangedComamnd}" CommandParameter="{Binding Path=Result,ElementName=verify2}"/>
                            </i:EventTrigger>
                        </i:Interaction.Triggers>
</util:TextClickVerify>

  

 

好了,至此結束。

相關文章
相關標籤/搜索