Maison > développement back-end > Tutoriel C#.Net > Méthode de saisie C#

Méthode de saisie C#

PHPz
Libérer: 2017-03-12 16:07:29
original
1573 Les gens l'ont consulté

C# Méthode de saisie

Bien que la méthode de saisie ne soit pas une nouveauté, il existe des versions dans différentes langues, mais en C# ce n'est pas courant ; cela donnera un malentendu aux gens : C# ne peut pas le faire ! En fait, si C# peut le faire, la réponse est oui – les trois méthodes fonctionneront : IMM, TSF et plug-in. IMM est utilisé pour ajuster certaines des api sous-jacentes de Windows. Cependant, il est fondamentalement inutilisable dans les nouvelles versions de Windows et constitue une méthode de fonctionnement obsolète. TSF est une nouvelle méthode recommandée par Microsoft, mais il y a trop peu d'informations par rapport à C# ; les principales informations en ligne concernent la version C, qui peut bien sûr être utilisée comme référence pour implémenter la version C#. Ici, je présente un type de plug-in (oh mon dieu, C# peut-il écrire des plug-ins ?), qui ne vaut certainement pas la peine d'être mentionné pour les experts, mais il peut être considéré comme un plug-in et une entrée méthode! Hors sujet - C# peut-il être utilisé comme plug-in ? La réponse est oui. Il existe encore de nombreuses informations sur l'APIprogrammation de C# pour Windows. Voici une brève introduction aux API qui peuvent être utilisées :

. Installé un hook pour intercepter les signaux de la souris et du clavier

public static extern int Set WindowsHookEx (int idHook, HookProc lpfn, IntPtr hInstance, int threadId);

Arrêtez d'utiliser les crochets

public static extern bool UnhookWindowsHookEx(int idHook);

Passer au crochet suivant via le crochet de message

public statique extern int AppelSuivantHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam);

Le crochet à fil doit être utilisé

static extern int GetCurrentThreadId();

Utilisez l'API WINDOWSfonction au lieu de la fonction pour obtenir l'instance actuelle afin d'éviter l'échec du hook

public static extern IntPtr GetModuleHandle(string name);

 

转换指定的虚拟键码和键盘状态的相应字符或字符

public static extern int ToAscii(int uVirtKey, //[in] 指定虚拟关键代码进行翻译。
                                        int uScanCode, // [in] 指定的硬件扫描码的关键须翻译成英文。高阶位的这个值设定的关键,如果是(不压)
                                        byte[] lpbKeyState, // [in] 指针,以256字节数组,包含当前键盘的状态。每个元素(字节)的数组包含状态的一个关键。如果高阶位的字节是一套,关键是下跌(按下)。在低比特,如果设置表明,关键是对切换。在此功能,只有肘位的CAPS LOCK键是相关的。在切换状态的NUM个锁和滚动锁定键被忽略。
                                        byte[] lpwTransKey, // [out] 指针的缓冲区收到翻译字符或字符。
                                        int fuState);

 

1.有了以上的这些api基本上就可能实现鼠标键盘的监控或者锁定等;那么首先要安装钩子:


   // 安装键盘钩子 public void Start()
        {         
            if (hKeyboardHook == 0)
            {
                KeyboardHookProcedure = new HookProc(KeyboardHookProc);

                hKeyboardHook = SetWindowsHookEx(WH_KEYBOARD_LL, KeyboardHookProcedure, GetModuleHandle(Process.GetCurrentProcess().MainModule.ModuleName), 0);                //如果SetWindowsHookEx失败
                if (hKeyboardHook == 0)
                {
                    Stop();                    throw new Exception("安装键盘钩子失败");
                }
            }
        }
Copier après la connexion

 

2.安装完后就要对获取到钩子进行处理:


private int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam)
        {            // 侦听键盘事件
            if (nCode >= 0 && wParam == 0x0100)
            {
                KeyboardHookStruct MyKeyboardHookStruct = (KeyboardHookStruct)Marshal.PtrToStructure(lParam, typeof(KeyboardHookStruct));                #region 开关                if (MyKeyboardHookStruct.vkCode == 20 || MyKeyboardHookStruct.vkCode == 160 || MyKeyboardHookStruct.vkCode == 161)
                {
                    isLocked = isLocked ? false : true;
                }                #endregion

                #region
                if (isLocked)
                {                    if (isStarted && MyKeyboardHookStruct.vkCode >= 48 && MyKeyboardHookStruct.vkCode <= 57)
                    {                        var c = int.Parse(((char)MyKeyboardHookStruct.vkCode).ToString());
                        OnSpaced(c);
                        isStarted = false;                        return 1;
                    }                    if (isStarted && MyKeyboardHookStruct.vkCode == 8)
                    {
                        OnBacked();                        return 1;
                    }                    if ((MyKeyboardHookStruct.vkCode >= 65 && MyKeyboardHookStruct.vkCode <= 90) || MyKeyboardHookStruct.vkCode == 32)
                    {                        if (MyKeyboardHookStruct.vkCode >= 65 && MyKeyboardHookStruct.vkCode <= 90)
                        {
                            Keys keyData = (Keys)MyKeyboardHookStruct.vkCode;
                            KeyEventArgs e = new KeyEventArgs(keyData);
                            KeyUpEvent(this, e);
                            isStarted = true;
                        }                        if (MyKeyboardHookStruct.vkCode == 32)
                        {
                            OnSpaced(0);
                            isStarted = false;
                        }                        return 1;
                    }                    else
                        return 0;
                }                #endregion
            }            return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam);
        }
Copier après la connexion

上面一些数字,对于刚入门的同学来说也不是什么问题,一看就明白是对哪些键做的操作。

3.停止钩子


 1 public void Stop() 2         { 3             bool retKeyboard = true; 4  5  6             if (hKeyboardHook != 0) 7             { 8                 retKeyboard = UnhookWindowsHookEx(hKeyboardHook); 9                 hKeyboardHook = 0;10             }11 12             if (!(retKeyboard))13                 throw new Exception("卸载钩子失败!");14         }
Copier après la connexion

4.注册事件


1 private void WordBoard_Load(object sender, EventArgs e)2         {3             Program.keyBordHook.KeyUpEvent += KeyBordHook_KeyUpEvent;4             Program.keyBordHook.OnSpaced += KeyBordHook_OnSpaced;5             Program.keyBordHook.OnBacked += KeyBordHook_OnBacked;6         }
Copier après la connexion

5.根据输入内容显示并进行转换


 1 private void ShowCharatar() 2         { 3             this.listView1.BeginInvoke(new Action(() => 4             { 5                 label1.Text = keys; 6  7                 try 8                 { 9                     this.listView1.Items.Clear();10                     var arr = CacheHelper.Get(keys);11                     if (arr != null)12                         for (int i = 0; i < (arr.Length > 10 ? 9 : arr.Length); i++)13                         {14                             this.listView1.Items.Add((i + 1) + "、" + arr[i]);15                         }16                 }17                 catch18                 {19                     label1.Text = keys = "";20                 }21             }));22         }
Copier après la connexion

 

6.显示输入


1 private void KeyBordHook_KeyUpEvent(object sender, KeyEventArgs e)2         {3             keys += e.KeyCode.ToString().ToLower();4             this.ShowCharatar();5         }
Copier après la connexion

 

7.空格上屏


 1 private void KeyBordHook_OnSpaced(int choose) 2         { 3             try 4             { 5                 if (CacheHelper.ContainsKey(keys)) 6                 { 7                     if (choose > 0) 8                     { 9                         choose = choose - 1;10                     }11 12                     Program.keyBordHook.Send(CacheHelper.Get(keys)[choose]);13                     label1.Text = "";14                     this.listView1.Clear();15                 }16             }17             catch18             {19 20             }21             keys = "";22         }
Copier après la connexion

8.将数据发送到激活的输入框中


1 public void Send(string msg)2         {3             if (!string.IsNullOrEmpty(msg))4             {5                 Stop();6                 SendKeys.Send("{RIGHT}" + msg);7                 Start();8             }9         }
Copier après la connexion

 

9.back键回退


1 private void KeyBordHook_OnBacked()2         {3             if (!string.IsNullOrEmpty(keys))4             {5                 keys = keys.Substring(0, keys.Length - 1);6             }7             this.ShowCharatar();8         }
Copier après la connexion

 

当然这里还可以使其他键来完善更多的功能,例如拼音的分页处理等

 

至于什么五笔、拼音就要使用词库来解决了;其中五笔比较简单,拼音就非常复杂了,各种分词、联想等...这里以五笔为主,拼音为单拼来实现基本的输入功能;所以不需要什么高深算法,简单使用MemoryCache就轻松高效搞定(有兴趣的可以来http://www.php.cn/ 上完善)

10.键词转换


  1 /*****************************************************************************************************  2  * 本代码版权归@wenli所有,All Rights Reserved (C) 2015-2017  3  *****************************************************************************************************  4  * CLR版本:4.0.30319.42000  5  * 唯一标识:8ebc884b-ee5f-45de-8638-c054b832e0ce  6  * 机器名称:WENLI-PC  7  * 联系人邮箱:wenguoli_520@qq.com  8  *****************************************************************************************************  9  * 项目名称:$projectname$ 10  * 命名空间:Wenli.IEM 11  * 类名称:CacheHelper 12  * 创建时间:2017/3/3 16:18:14 13  * 创建人:wenli 14  * 创建说明: 15  *****************************************************************************************************/ 16 using System; 17 using System.Collections.Generic; 18 using System.IO; 19 using System.Linq; 20 using System.Runtime.Caching; 21 using System.Text; 22 using System.Windows.Forms; 23  24 namespace Wenli.IEM.Helper 25 { 26     public static class CacheHelper 27     { 28         static MemoryCache _wubiCache = new MemoryCache("wubi"); 29  30         static MemoryCache _pinyinCache = new MemoryCache("pinyin"); 31  32         static CacheHelper() 33         { 34             var path = Application.StartupPath + "\\Win32\\world.dll"; 35             var arr = File.ReadAllLines(path); 36             foreach (string item in arr) 37             { 38                 var key = item.Substring(0, item.IndexOf(" ")); 39                 var value = item.Substring(item.IndexOf(" ") + 1); 40                 _wubiCache.Add(key, (object)value, DateTimeOffset.MaxValue); 41             } 42  43             // 44  45             path = Application.StartupPath + "\\Win32\\pinyin.dll"; 46             arr = File.ReadAllLines(path); 47             foreach (string item in arr) 48             { 49                 var key = item.Substring(0, item.IndexOf(" ")); 50                 var value = item.Substring(item.IndexOf(" ") + 1); 51                 _pinyinCache.Add(key, (object)value, DateTimeOffset.MaxValue); 52             } 53         } 54  55         public static string[] Get(string key) 56         { 57             if (!string.IsNullOrEmpty(key)) 58             { 59                 var str = string.Empty; 60  61                 try 62                 { 63                     if (_wubiCache.Contains(key)) 64                         str = _wubiCache[key].ToString(); 65                 } 66                 catch { } 67                 try 68                 { 69                     if (_pinyinCache.Contains(key)) 70                         str += " " + _pinyinCache[key].ToString(); 71                 } 72                 catch { } 73  74                 if (!string.IsNullOrEmpty(str)) 75                 { 76                     var arr = str.Split(new string[] { " " }, StringSplitOptions.RemoveEmptyEntries); 77                     for (int i = 0; i < arr.Length; i++) 78                     { 79                         if (arr[i].IndexOf("*") > -1) 80                         { 81                             arr[i] = arr[i].Substring(0, arr[i].IndexOf("*")); 82                         } 83                     } 84                     return arr; 85                 } 86             } 87  88             return null; 89         } 90  91  92         public static bool ContainsKey(string key) 93         { 94             if (_wubiCache.Contains(key)) 95                 return true; 96             if (_pinyinCache.Contains(key)) 97                 return true; 98             return false; 99         }100 101         public static void Clear()102         {103             _wubiCache.Dispose();104             GC.Collect(-1);105         }106     }107 }
Copier après la connexion

 

到此一个基本型的C#版外挂输入法就成功完成了,源码地址:http://www.php.cn/ 

 

 

 

 


Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:php.cn
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal