但是用户喜欢使用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;
}