win32使用Tab键输入制表符或者替换为四个空格

3342 人浏览 | 时间: 2016-01-04 19:26:25 | 作者: codexia 会员文章,禁止转载

    制表符在VS中可以用于排版,一个制表符就可以空出比较大的位置,不用给一个个的空格的敲入。然而,当你要复制这些含有制表符的代码时,粘贴在其他编辑器中,很可能会出现问题。因为不同的编辑器对制表符的处理不一样,所以就会出现原本在VS中排版很好的代码,就出现乱七八糟的排版。
    但是用户喜欢使用Tab键来快速的排版,所以我们可以将Tab键输入的制表符替换成相同宽度的空格,就可以搞定这个排版问题。因为不同的编辑器,对于空格的处理是一样的。
    在VS中我们可以通过“工具”->“选项”->“文本编辑器”->“C/C++”->“制表符”->勾选“插入空格”,然后设置制表符占用的空格的大小,缩进大小。一般设置为4。
    这样设置之后,将原有的代码全选,剪切,再粘贴,一般能够将制表符替换成空格,然后新输入的制表符都是空格替换的。使用空格替换的代码排版,复制到哪都不会变的。
    那么我们这里就来实现键入Tab键后将制表符替换为4个空格的功能。如果你不做任何操作,那么输入的就是制表符,如果编辑器不支持制表符,则就会成为黑心块。

【使用4个空格字符替换一个Tab产生的制表符的效果】
    你可以从图中看到,中间空格的宽度是一样的,有一个是比较宽的,由两个制表符产生,其实就是8个空格字符。这样用户看来,既不会觉得Tab键无效,而且效果也达到了,同时复制的话不会丢失原有的排版。
    输入文字以及可以使用退格键删除输入的字符的功能分析,见文章《win32直接在窗口中输入文字,将窗口当做编辑框》。
    因为Tab键会产生制表符,制表符也是一个字符,所以我们可以检测字符消息,就在WM_CHAR里处理了。因为有退格键和制表符以及其他的字符消息,所有就使用switch case来实现。
    注意,在检测到Tab字符时,我们不需要真正的替换,而是将这个字符解释为4个空格追加到原有的字符串即可。而追加的位置则有lstrlen函数来动态获取。追加一个,这个动态获取的字符串字符个数就加一了。所以,字符数组的索引不要加上i来做递增。如果你想直接输入制表符,则直接将'\t'字符追加到字符串末尾即可,你只要让这个字符流入default处理追加即可。
    在追加字符的时候,不要在索引里减一。因为我们本来就是要覆盖最后一个空字符。而在右击清空字符数组的时候,之前是将索引为0的位置设置为空字符。然而这个并没有达到效果。所以,要彻底清除,就使用memset函数。
    下面是完整的代码:
#include "windows.h"
#include <tchar.h>
TCHAR cjjjs[100]=_T("【C++技术网http:www.cjjjs.com】请直接输入文字");
TCHAR txt[1024]=_T("");
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
    HDC hdc;
    PAINTSTRUCT ps;
    static bool bIsCap=false;
    switch (message)
    {
    case WM_PAINT:
        hdc = BeginPaint(hwnd,&ps);
        SelectObject(hdc,GetStockObject(SYSTEM_FIXED_FONT));
        TextOut(hdc,0,100,cjjjs,lstrlen(cjjjs));
        TextOut(hdc,0,0,txt,lstrlen(txt));       
        EndPaint(hwnd,&ps);
        return 0;
     case WM_CHAR:
         {
             TCHAR ch=(TCHAR)wParam;       
             switch(ch)
             {
             case 8://退格键
                 {
                     int len=lstrlen(txt)-1;
                     txt[len]=_T('\0');
                     break;
                 }
             case 9://Tab键
                 for (int i=0;i<4;i++)
                 {
                     txt[lstrlen(txt)]=_T(' ');//因为是动态获取字符串的长度,每次新增字符后,获取的结果都不一样了。所有txt数组内的所有必须要加上i
                     //txt[lstrlen(txt)+i]=_T(' ');//错误的
                 }
                 break;
             default:
                 txt[lstrlen(txt)]=ch;//追加字符,就是要覆盖最后一个空字符,不要减一哦
             }
         }
         InvalidateRect(hwnd,NULL,TRUE);//刷新显示结果
         return 0;
     case WM_RBUTTONDOWN:
         {
             txt[0]=_T('\0');//只是填充开头一个元素为0,在输入一个之后,就会恢复之前的长度
             memset(txt,0,sizeof(txt));//这样才能将所有内存情况,真正清空字符串
             InvalidateRect(hwnd,NULL,TRUE);
         }
         return 0;
     case WM_DESTROY:
         PostQuitMessage(0);
         return 0;
     default:
         return DefWindowProc(hwnd, message, wParam, lParam);
    }
}

int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
    TCHAR ClassName[] = _T("MyClass");
    TCHAR title1[] = _T("C++技术网http://www.cjjjs.com");
    WNDCLASS wndClass;
    wndClass.cbClsExtra=0;
    wndClass.cbWndExtra=0;
    wndClass.hbrBackground= (HBRUSH)GetStockObject(WHITE_BRUSH);
    wndClass.hCursor=LoadCursor(NULL,IDC_HAND);
    wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
    wndClass.hInstance = hInstance;
    wndClass.lpfnWndProc = WinProc;
    wndClass.lpszClassName = ClassName;
    wndClass.lpszMenuName=NULL;
    wndClass.style=CS_HREDRAW|CS_VREDRAW;

    if(!RegisterClass(&wndClass))return 0;
    HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,0,0,650,400,NULL,NULL,hInstance,NULL);
    ShowWindow(hwnd,SW_SHOWNORMAL);

    MSG msg;
    while (GetMessage(&msg,NULL,0,0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    return msg.wParam;
}
当前文章为会员文章,请前往[用户中心]开通会员后继续阅读。

相关阅读