C# 输入法 z
C# 输入法
虽说输入法不是什么新事物,各种语言版本都有,不过在C#不常见;这就会给人一种误会:C#不能做!其实C#能不能做呢,答案是肯定的——三种方式都行:IMM、TSF以及外挂式。IMM这种就是调windows的一些底层api,不过在新版本的windows中基本上已经不能用了,属于一种过时的操作方式。TSF是微软推荐的一种新方式,不过相对C#资料太少;线上主要的一些都是针对C++的版本资料,当然可以作为借鉴来实现C#版的。我这里主要介绍一种外挂式的(天啦撸,C#可以写外挂?),对于高手来说肯定不值一提,不过也算是实现了外挂及输入法!题外话——C#可以做外挂么?答案是可以的,C#针对windows的api编程资料还是很多的,下面就简单的介绍一下面可能要使用到的api:
安装了一个钩子,截取鼠标键盘等信号
public static extern int SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId);
停止使用钩子
public static extern bool UnhookWindowsHookEx(int idHook);
通过信息钩子继续下一个钩子
public static extern int CallNextHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam);
线程钩子需要用到
static extern int GetCurrentThreadId();
使用WINDOWS API函数代替获取当前实例的函数,防止钩子失效
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("安装键盘钩子失败");
}
}
}

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);
}

上面一些数字,对于刚入门的同学来说也不是什么问题,一看就明白是对哪些键做的操作。
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 }

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 }
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 catch
18 {
19 label1.Text = keys = "";
20 }
21 }));
22 }

6.显示输入
1 private void KeyBordHook_KeyUpEvent(object sender, KeyEventArgs e)
2 {
3 keys += e.KeyCode.ToString().ToLower();
4 this.ShowCharatar();
5 }
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 catch
18 {
19
20 }
21 keys = "";
22 }

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 }

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 }

当然这里还可以使其他键来完善更多的功能,例如拼音的分页处理等
至于什么五笔、拼音就要使用词库来解决了;其中五笔比较简单,拼音就非常复杂了,各种分词、联想等...这里以五笔为主,拼音为单拼来实现基本的输入功能;所以不需要什么高深算法,简单使用MemoryCache就轻松高效搞定(有兴趣的可以来https://github.com/yswenli/Wenli.IEM 上完善)
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 }

到此一个基本型的C#版外挂输入法就成功完成了,源码地址:https://github.com/yswenli/Wenli.IEM
转载请标明本文来源:http://www.cnblogs.com/yswenli/p/6528447.html
更多内容欢迎star作者的github:https://github.com/yswenli/RedisDrive
如果发现本文有什么问题和任何建议,也随时欢迎交流~
C# 输入法 z的更多相关文章
- 【Python】使用torrentParser1.03对多文件torrent的分析结果
Your environment has been set up for using Node.js 8.5.0 (x64) and npm. C:\Users\horn1>cd C:\User ...
- JavaScript 汉字与拼音互转终极方案 附JS拼音输入法
转:http://www.codeceo.com/article/javascript-pinyin.html 前言 网上关于JS实现汉字和拼音互转的文章很多,但是比较杂乱,都是互相抄来抄去,而且有的 ...
- 【转】ubuntu下最好用的输入法fcitx-sunpinyin
http://www.freetstar.com/index.php/ubuntu-most-use-friendly-fcitx-sunpinyin 今天难得折腾一会儿输入法,对于系统美化方面的 ...
- 【干货】JS版汉字与拼音互转终极方案,附简单的JS拼音输入法
前言 网上关于JS实现汉字和拼音互转的文章很多,但是比较杂乱,都是互相抄来抄去,而且有的不支持多音字,有的不支持声调,有的字典文件太大,还比如有时候我仅仅是需要获取汉字拼音首字母却要引入200kb的字 ...
- 在 Visual Studio 等编辑器/IDE中自动切换输入法,不需要手动的有没有?
使用Visual Studio写代码,经常遇到的一个问题就是切换中文输入法麻烦,输入完注释//,要切换到中文,输入完引号,要输入中文,然后还需要切换回来,有没有? 有时候中文输入法忽然失效有没有?明明 ...
- Ubuntu下安装中文输入法
搜狗输入法 for Linux 是基于Fcitx 框架(fcitx-sogoupinyin). 安装环境为Ubuntu 13.04 安装过程: 卸载Ubuntu默认的ibus输入法: sudo apt ...
- Ubuntu 安装 fcitx 输入法
fcitx 和 ibus一样都是输入法框架.下面介绍ubuntu下安装fcitx输入法. 1.先卸载系统中的输入法 2.安装. 增加ppa源:sudo add-apt-repository ppa:f ...
- Ubuntu安装Fcitx(小企鹅五笔输入法)
安装配置如下: 1. 安装 fcitx sudo apt-get install fcitx 2. 配置默认输入法为 fcitx im-switch -s fcitx // 注意无须加 sudo 3. ...
- 【原创】mac 上如何安装及切换输入法
1. 安装输入法 刚购入mac mini一台,默认只有英文输入法,而自己用习惯了sogou输入法,于是就在网上下载安装了一个sogou拼音输入 for mac. 下载完毕后,在屏幕下方的dock区域右 ...
随机推荐
- OneNote无法同时设置中英文字体设置解决办法
如果你是一位OneNote老用户,无论是2003.2007还是2010或者最新的2013版本,都一直存在一个Bug,就是无法同时设置中英文字体(比如在Word中就可以分别设置不同的).我搜了一下,在微 ...
- CCF CSP 201409-2 画图
CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201409-2 画图 问题描述 在一个定义了直角坐标系的纸上,画一个(x1,y1)到(x2,y ...
- 【LOJ】#2066. 「SDOI2016」墙上的句子
题解 我一直也不会网络流--orz 我们分析下这道题,显然和行列没啥关系,就是想给你n + m个串 那么我们对于非回文单词之外的单词,找到两两匹配的反转单词(即使另一个反转单词不会出现也要建出来) 具 ...
- MongoDB入门教程三[数据类型]
MongoDB的文档使用BSON(Binary JSON)来组织数据,BSON类似于JSON,JSON只是一种简单的表示数据的方式,只包含了6种数据类型(null.布尔.数字.字符串.数组及对象),不 ...
- 采用Post请求的方式提交参数并导出excel
一般情况下,我们都是采用get请求的方式导出excel.例如采用如下方式: var exportUrl = '/xxx;'; window.open(exportUrl); 导出excel所需的逻辑参 ...
- C# 操作 access 数据库
随笔: (1) 命名空间 using System.Data.OleDb; (2) 连接字符串 private staticstring connStr = @"Provider= ...
- 世界杯:用Python分析热门夺冠球队-(附源代码)
2018年,火热的世界杯即将拉开序幕.在比赛开始之前,我们不妨用 Python 来对参赛队伍的实力情况进行分析,并大胆的预测下本届世界杯的夺冠热门球队. 通过数据分析,可以发现很多有趣的结果,比如: ...
- nc工具学习
0x00.命令详解 基本使用 想要连接到某处:nc [-options] ip port 绑定端口等待连接:nc -l -p port ip 参数: -e prog 程序重定向,一旦连接,就执行 [ ...
- 【基础知识】.Net基础加强 第二天
第02天 .Net基础加强 1. 封装 1> 属性的封装: 属性封装字段:把变化封装一下,保留用户的使用方式 2> 把方法的多个参数封装成一个对象 3> 将一堆代码封装到一个方法中 ...
- 防止sql注入的几种方法
一.SQL注入简介 SQL注入是比较常见的网络攻击方式之一,它不是利用操作系统的BUG来实现攻击,而是针对程序员编程时的疏忽,通过SQL语句,实现无帐号登录,甚至篡改数据库. 二.SQL注入攻击的总体 ...