Winform输入法全角自动转半角

本文介绍了一个C#实现的ImeHelper类,用于自动将输入法从全角转换为半角,适用于各种Windows表单控件。通过在基类窗体加载时调用该类的方法,确保所有文本输入始终为半角字符。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

ImeHelper.cs

====================================================================================================================

using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
using System.Windows.Forms;

namespace ChangeImeFromFullToHalf
{
    /// <summary>
    ///  输入法帮助,全角 转换为半角
    /// </summary>
    public class ImeHelper
    {
        #region 声明一些API函数
        [DllImport("imm32.dll")]
        public static extern IntPtr ImmGetContext(IntPtr hwnd);
        [DllImport("imm32.dll")]
        public static extern bool ImmGetOpenStatus(IntPtr himc);
        [DllImport("imm32.dll")]
        public static extern bool ImmSetOpenStatus(IntPtr himc, bool b);
        [DllImport("imm32.dll")]
        public static extern bool ImmGetConversionStatus(IntPtr himc, ref int lpdw, ref int lpdw2);
        [DllImport("imm32.dll")]
        public static extern int ImmSimulateHotKey(IntPtr hwnd, int lngHotkey);
        public const int IME_CMODE_FULLSHAPE = 0x8;
        public const int IME_CHOTKEY_SHAPE_TOGGLE = 0x11;
        #endregion


        /// <summary>
        /// 重载SetIme,传入Form
        /// </summary>
        /// <param name="frm"></param>
        public static void SetIme(Form frm)
        {
            ChangeAllControl(frm);
        }


        /// <summary>
        /// 重载SetIme,传入Control
        /// </summary>
        /// <param name="ctl"></param>
        public static void SetIme(Control ctl)
        {
            ChangeAllControl(ctl);
        }


        /// <summary>
        /// 重载SetIme,传入对象句柄
        /// </summary>
        /// <param name="Handel"></param>
        public static void SetIme(IntPtr Handel)
        {
            ChangeControlIme(Handel);
        }


        private static void ChangeAllControl(Control ctl)
        {
            //在控件的的Enter事件中触发来调整输入法状态


            //遍历子控件,使每个控件都用上Enter的委托处理
            foreach (Control ctlChild in ctl.Controls)
            {
                ctlChild.Paint += new PaintEventHandler(frm_Paint);
                ctlChild.Enter += new EventHandler(ctl_Enter);
                ctlChild.TextChanged += new EventHandler(ctl_TextChanged);
                ctlChild.KeyDown += new KeyEventHandler(ctl_KeyDown);


                //DataGridView中Cell进行特殊处理
                if (ctlChild is DataGridView)
                {
                    DataGridView dgv = ctlChild as DataGridView;
                    dgv.EditingControlShowing += new DataGridViewEditingControlShowingEventHandler(dgv_EditingControlShowing);
                }


                ChangeAllControl(ctlChild);
            }
        }


        static void ctl_TextChanged(object sender, EventArgs e)
        {
            bool b = false;
            TextBox txt = null;
            ComboBox cmb = null;
            RichTextBox rtb = null;
            DataGridTextBox dgtb = null;


            if (sender is TextBox)
                txt = sender as TextBox;
            else if (sender is ComboBox)
                cmb = sender as ComboBox;
            else if (sender is RichTextBox)
                rtb = sender as RichTextBox;
            else if (sender is DataGridTextBox)
                dgtb = sender as DataGridTextBox;


            char[] c = null;
            if (b == false)
            {
                if (txt != null)
                    c = txt.Text.ToCharArray();
                else if (cmb != null)
                    c = cmb.Text.ToCharArray();
                else if (rtb != null)
                    c = rtb.Text.ToCharArray();
                else if (dgtb != null)
                    c = dgtb.Text.ToCharArray();


                if (c == null)
                    return;


                for (int i = 0; i < c.Length; i++)
                {
                    if (c[i] == 12288)
                    {
                        c[i] = (char)32;
                        continue;
                    }
                    if (c[i] > 65280 && c[i] < 65375)
                        c[i] = (char)(c[i] - 65248);
                }
            }
            b = true;


            if (txt != null)
            {
                txt.Text = new string(c);
                txt.SelectionStart = txt.Text.Length;
            }
            else if (cmb != null)
            {
                cmb.Text = new string(c);
                cmb.SelectionStart = cmb.Text.Length;
            }
            else if (rtb != null)
            {
                rtb.Text = new string(c);
                rtb.SelectionStart = rtb.Text.Length;
            }
            else if (dgtb != null)
            {
                dgtb.Text = new string(c);
                dgtb.SelectionStart = dgtb.Text.Length;
            }
        }


        static void ctl_KeyDown(object sender, KeyEventArgs e)
        {
            ChangeControlIme(sender);
        }


        public static void frm_Paint(object sender, PaintEventArgs e)
        {
            /**/
            /*有人问为什么使用Pain事件,而不用Load事件或Activated事件,是基于下列考虑:
         * 、在您的Form中,有些控件可能是运行时动态添加的
         * 、在您的Form中,使用到了非.NET的OCX控件
         * 、Form调用子Form的时候,Activated事件根本不会触发 */
            ChangeControlIme(sender);
        }


        /// <summary>
        /// 控件的Enter处理程序
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        static void ctl_Enter(object sender, EventArgs e)
        {
            ChangeControlIme(sender);
        }


        //编辑单元格控件发生事件
        static void dgv_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
        {
            DataGridView dgv = sender as DataGridView;
            //获取列序号
            int columnIndex = dgv.CurrentCell.ColumnIndex;
            //单元格转化成文本框
            TextBox tb = e.Control as TextBox;
            //委托单元格KeyPress事件
            tb.KeyPress += new KeyPressEventHandler(tb_KeyPress);
        }


        //单元格KeyPress事件
        static void tb_KeyPress(object sender, KeyPressEventArgs e)
        {
            //全角转半角
            if (e.KeyChar >= 65296 && e.KeyChar <= 65305)
            {
                e.KeyChar -= Convert.ToChar(65248);
            }
        }


        private static void ChangeControlIme(object sender)
        {
            Control ctl = (Control)sender;
            ChangeControlIme(ctl.Handle);
        }


        /// <summary>
        /// 下面这个函数才是真正检查输入法的全角半角状态
        /// </summary>
        /// <param name="h"></param>
        private static void ChangeControlIme(IntPtr h)
        {
            IntPtr HIme = ImmGetContext(h);
            if (ImmGetOpenStatus(HIme))  //如果输入法处于打开状态
            {
                int iMode = 0;
                int iSentence = 0;
                bool bSuccess = ImmGetConversionStatus(HIme, ref iMode, ref iSentence);  //检索输入法信息
                if (bSuccess)
                {
                    if ((iMode & IME_CMODE_FULLSHAPE) > 0)   //如果是全角
                    {
                        ImmSimulateHotKey(h, IME_CHOTKEY_SHAPE_TOGGLE);  //转换成半角
                    }
                }
            }
        }
    }
}


在基类窗体中进行调用

================================================================================================================

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;

namespace ChangeImeFromFullToHalf
{
    public partial class BaseFrm : Form
    {
        public BaseFrm()
        {
            InitializeComponent();
        }


        private void BaseFrm_Load(object sender, EventArgs e)
        {
           ImeHelper.SetIme(this);
        }
    }
}

基于C2000 DSP的电力电子、电机驱动和数字滤波器的仿真模型构建及其C代码实现方法。首先,在MATLAB/Simulink环境中创建电力电子系统的仿真模型,如三相逆变器,重点讨论了PWM生成模块中死区时间的设置及其对输出波形的影响。接着,深入探讨了C2000 DSP内部各关键模块(如ADC、DAC、PWM定时器)的具体配置步骤,特别是EPWM模块采用上下计数模式以确保对称波形的生成。此外,还讲解了数字滤波器的设计流程,从MATLAB中的参数设定到最终转换为适用于嵌入式系统的高效C代码。文中强调了硬件在环(HIL)和支持快速原型设计(RCP)的重要性,并分享了一些实际项目中常见的陷阱及解决方案,如PCB布局不当导致的ADC采样异常等问题。最后,针对中断服务程序(ISR)提出了优化建议,避免因ISR执行时间过长而引起的系统不稳定现象。 适合人群:从事电力电子、电机控制系统开发的技术人员,尤其是那些希望深入了解C2000 DSP应用细节的研发工程师。 使用场景及目标:①掌握利用MATLAB/Simulink进行电力电子设备仿真的技巧;②学会正确配置C2000 DSP的各项外设资源;③能够独立完成从理论设计到实际产品落地全过程中的各个环节,包括但不限于数字滤波器设计、PWM信号生成、ADC采样同步等。 其他说明:文中提供了大量实用的代码片段和技术提示,帮助读者更好地理解和实践相关知识点。同时,也提到了一些常见错误案例,有助于开发者规避潜在风险。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值