全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

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.停止钩子

public void Stop()
  {
   bool retKeyboard = true;
   if (hKeyboardHook != 0)
   {
    retKeyboard = UnhookWindowsHookEx(hKeyboardHook);
    hKeyboardHook = 0;
   }
   if (!(retKeyboard))
    throw new Exception("卸载钩子失败!");
  }

4.注册事件

 private void WordBoard_Load(object sender, EventArgs e)
   {
    Program.keyBordHook.KeyUpEvent += KeyBordHook_KeyUpEvent;
    Program.keyBordHook.OnSpaced += KeyBordHook_OnSpaced;
    Program.keyBordHook.OnBacked += KeyBordHook_OnBacked;
   }

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

private void ShowCharatar()
  {
   this.listView1.BeginInvoke(new Action(() =>
   {
    label1.Text = keys;
    try
    {
     this.listView1.Items.Clear();
     var arr = CacheHelper.Get(keys);
     if (arr != null)
      for (int i = 0; i < (arr.Length > 10 ? 9 : arr.Length); i++)
      {
       this.listView1.Items.Add((i + 1) + "、" + arr[i]);
      }
    }
    catch
    {
     label1.Text = keys = "";
    }
   }));
  }

6.显示输入

 private void KeyBordHook_KeyUpEvent(object sender, KeyEventArgs e)
   {
    keys += e.KeyCode.ToString().ToLower();
    this.ShowCharatar();
   }

7.空格上屏

private void KeyBordHook_OnSpaced(int choose)
  {
   try
   {
    if (CacheHelper.ContainsKey(keys))
    {
     if (choose > 0)
     {
      choose = choose - 1;
     }
     Program.keyBordHook.Send(CacheHelper.Get(keys)[choose]);
     label1.Text = "";
     this.listView1.Clear();
    }
   }
   catch
   {
   }
   keys = "";
  }

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

public void Send(string msg)
  {
   if (!string.IsNullOrEmpty(msg))
   {
    Stop();
    SendKeys.Send("{RIGHT}" + msg);
    Start();
   }
  }

9.back键回退

private void KeyBordHook_OnBacked()
  {
   if (!string.IsNullOrEmpty(keys))
   {
    keys = keys.Substring(0, keys.Length - 1);
   }
   this.ShowCharatar();
  }

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

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

10.键词转换

/*****************************************************************************************************
 * 本代码版权归@wenli所有,All Rights Reserved (C) 2015-2017
*****************************************************************************************************
 * CLR版本:4.0.30319.42000
 * 唯一标识:8ebc884b-ee5f-45de-8638-c054b832e0ce
 * 机器名称:WENLI-PC
 * 联系人邮箱:wenguoli_520@qq.com
*****************************************************************************************************
 * 项目名称:$projectname$
 * 命名空间:Wenli.IEM
 * 类名称:CacheHelper
 * 创建时间:2017/3/3 16:18:14
 * 创建人:wenli
 * 创建说明:
*****************************************************************************************************/
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.Caching;
using System.Text;
using System.Windows.Forms;
namespace Wenli.IEM.Helper
{
 public static class CacheHelper
 {
  static MemoryCache _wubiCache = new MemoryCache("wubi");
  static MemoryCache _pinyinCache = new MemoryCache("pinyin");
  static CacheHelper()
  {
   var path = Application.StartupPath + "\\Win32\\world.dll";
   var arr = File.ReadAllLines(path);
   foreach (string item in arr)
   {
    var key = item.Substring(0, item.IndexOf(" "));
    var value = item.Substring(item.IndexOf(" ") + 1);
    _wubiCache.Add(key, (object)value, DateTimeOffset.MaxValue);
   }
   //
   path = Application.StartupPath + "\\Win32\\pinyin.dll";
   arr = File.ReadAllLines(path);
   foreach (string item in arr)
   {
    var key = item.Substring(0, item.IndexOf(" "));
    var value = item.Substring(item.IndexOf(" ") + 1);
    _pinyinCache.Add(key, (object)value, DateTimeOffset.MaxValue);
   }
  }
  public static string[] Get(string key)
  {
   if (!string.IsNullOrEmpty(key))
   {
    var str = string.Empty;
    try
    {
     if (_wubiCache.Contains(key))
      str = _wubiCache[key].ToString();
    }
    catch { }
    try
    {
     if (_pinyinCache.Contains(key))
      str += " " + _pinyinCache[key].ToString();
    }
    catch { }
    if (!string.IsNullOrEmpty(str))
    {
     var arr = str.Split(new string[] { " " }, StringSplitOptions.RemoveEmptyEntries);
     for (int i = 0; i < arr.Length; i++)
     {
      if (arr[i].IndexOf("*") > -1)
      {
       arr[i] = arr[i].Substring(0, arr[i].IndexOf("*"));
      }
     }
     return arr;
    }
   }
   return null;
  }
  public static bool ContainsKey(string key)
  {
   if (_wubiCache.Contains(key))
    return true;
   if (_pinyinCache.Contains(key))
    return true;
   return false;
  }
  public static void Clear()
  {
   _wubiCache.Dispose();
   GC.Collect(-1);
  }
 }
}

到此一个基本型的C#版*输入法就成功完成了,源码地址:https://github.com/yswenli/Wenli.IEM

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!


# C#  # 输入法  # C#中Winfrom默认输入法的设置方法  # C#设置输入法实例分析  # 五笔  # 能做  # 来实现  # 高阶  # 都是  # 鼠标键盘  # 也不  # 都有  # 还可以  # 在此  # 不需要  # 版权归  # 微软  # 用了  # 给人  # 英文  # 有兴趣  # 线上  # 三种  # 分页 


相关文章: 如何在香港免费服务器上快速搭建网站?  道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?  制作企业网站建设方案,怎样建设一个公司网站?  建站中国必看指南:CMS建站系统+手机网站搭建核心技巧解析  Python lxml的etree和ElementTree有什么区别  建站之星安装后界面空白如何解决?  如何通过西部数码建站助手快速创建专业网站?  建站之星好吗?新手能否轻松上手建站?  北京网页设计制作网站有哪些,继续教育自动播放怎么设置?  利用JavaScript实现拖拽改变元素大小  怎么用手机制作网站链接,dw怎么把手机适应页面变成网页?  如何通过多用户协作模板快速搭建高效企业网站?  制作营销网站公司,淘特是干什么用的?  如何在阿里云虚拟服务器快速搭建网站?  南京做网站制作公司,南京哈发网络有限公司,公司怎么样,做网页美工DIV+CSS待遇怎么样?  如何在自有机房高效搭建专业网站?  深圳网站制作的公司有哪些,dido官方网站?  ,有什么在线背英语单词效率比较高的网站?  *服务器网站为何频现安全漏洞?  招商网站制作流程,网站招商广告语?  如何选择高效便捷的WAP商城建站系统?  如何用西部建站助手快速创建专业网站?  如何通过.red域名打造高辨识度品牌网站?  西安大型网站制作公司,西安招聘网站最好的是哪个?  黑客如何通过漏洞一步步攻陷网站服务器?  香港服务器建站指南:免备案优势与SEO优化技巧全解析  Java解压缩zip - 解压缩多个文件或文件夹实例  专业的网站制作设计是什么,如何制作一个企业网站,建设网站的基本步骤有哪些?  c# Task.Yield 的作用是什么 它和Task.Delay(1)有区别吗  如何彻底删除建站之星生成的Banner?  整人网站在线制作软件,整蛊网站退不出去必须要打我是白痴才能出去?  建站之星后台密码如何安全设置与找回?  详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)  建站主机与服务器功能差异如何区分?  香港服务器选型指南:免备案配置与高效建站方案解析  如何彻底卸载建站之星软件?  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  建站之星后台管理系统如何操作?  网站制作免费,什么网站能看正片电影?  为什么Go需要go mod文件_Go go mod文件作用说明  ,石家庄四十八中学官网?  建站主机与虚拟主机有何区别?如何选择最优方案?  网站建设设计制作营销公司南阳,如何策划设计和建设网站?  如何快速搭建安全的FTP站点?  建站之星导航配置指南:自助建站与SEO优化全解析  如何确保FTP站点访问权限与数据传输安全?  建站主机默认首页配置指南:核心功能与访问路径优化  如何在服务器上配置二级域名建站?  学校免费自助建站系统:智能生成+拖拽设计+多端适配  如何配置支付宝与微信支付功能? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。