c语言控制台图形

编程

一、设置句柄与窗口信息

在Windows操作系统下用C语言编写控制台的窗口界面首先要获取当前标准输入和标准输出设备的句柄。通过调用函数GetStdHandle可以获取当前标准输入以及输出设备的句柄。函数原型为:

1
2
3
4
5
6
7
HANDLE GetStdHandle(DWORD nStdHandle);  
/*
其中,nStdHandle可以是
STD_INPUT_HANDLE 标准输入设备句柄
STD_OUTPUT_HANDLE 标准输出设备句柄
STD_ERROR_HANDLE 标准错误设备句柄
*/

需要说明的是,句柄 是Windows最常用的一个概念。它通常用来标识 Windows 资源(如菜单、 图标、窗口等)和设备等对象。虽然可以把句柄理解为是一个指针变量类型,但它不是对象所在的地址指针,而是作为Windows系统内部表的索引值来使用的。

调用相关文本界面控制的 API 函数可分为三类。

  • 用于控制台窗口操作的函数(包括窗口的缓冲区大小、窗口前景字符和背景颜色、窗口标题、大小和位置等)

  • 用于控制台输入输出的函数(包括字符属性操作函数)

  • 其他的函数并为最后一类。通过调用CloseHandle函数来关闭输入输出句柄。

1.1 示例程序:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
#include <stdio.h>  
#include <windows.h>
#include <conio.h>

int main(int argc,char *argv[])
{
HANDLE handle_out; //定义一个句柄
CONSOLE_SCREEN_BUFFER_INFO screen_info; //定义窗口缓冲区信息结构体
COORD pos = {0, 0}; //定义一个坐标结构体
handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
GetConsoleScreenBufferInfo(handle_out, &screen_info); //获取窗口信息
_getch(); //输入一个字符,不会显示到屏幕上
/*
向整个缓冲区填充字符'A'
其中填充的开始处为pos,坐标为{0, 0},也就是屏幕最左上角的字符处
填充个数为screen_info.dwSize.X(缓冲区宽度,也就是横坐标最大值加1) * screen_info.dwSize.Y(缓冲区高度,也就是纵坐标最大值加1)
因此可以达到向整个缓冲区填充字符'A'的效果
*/
FillConsoleOutputCharacter(handle_out, 'A', screen_info.dwSize.X * screen_info.dwSize.Y, pos, NULL);
CloseHandle(handle_out); //关闭标准输出设备句柄
return 0;
}

程序中,COORD 和 CONSOLE_SCREEN_BUFFER_ INFO 是 wincon.h 定义的控制台结构体类型
原型如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//坐标结构体  
typedef struct _COORD
{
SHORT X;
SHORT Y;
}COORD;

//控制台窗口信息结构体
typedef struct _CONSOLE_SCREEN_BUFFER_INFO
{
COORD dwSize; //缓冲区大小
COORD dwCursorPosition; //当前光标位置
WORD wAttributes; //字符属性
SMALL_RECT srWindow; //当前窗口显示的大小和位置
COORD dwMaximumWindowSize; // 最大的窗口缓冲区大小
}CONSOLE_SCREEN_BUFFER_INFO;

NOTE:

还需要说明的是,虽然在 C++ 中,iostream.h 定义了 cin 和 cout 的标准输入和输出流对象

但它们只能实现基本的输入输出操作,对于控制台窗口界面的控制却无能为力,而且不能与 stdio.h 和 conio.h 友好相处,因为 iostream.h 和它们是 C++ 两套不同的输入 输出操作方式,使用时要特别注意。


二、窗口缓冲区的设置

2.1 控制台窗口操作的API函数:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
//获取控制台窗口信息  
GetConsoleScreenBufferInfo();

//获取控制台窗口标题
GetConsoleTitle();

//更改指定缓冲区大小
SetConsoleScreenBufferSize();

//设置控制台窗口标题
SetConsoleTitle();

//设置控制台窗口信息
SetConsoleWindowInfo();

2.2 示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#include <stdio.h>  
#include <stdlib.h>
#include <Windows.h>
#include <conio.h>
#define N 255

int main()
{
HANDLE handle_out; //定义一个句柄
CONSOLE_SCREEN_BUFFER_INFO scbi; //定义一个窗口缓冲区信息结构体
COORD size = {80, 25}; //定义一个坐标结构体
char strtitle[N];
handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
GetConsoleScreenBufferInfo(handle_out, &scbi); //获得窗口缓冲区信息
GetConsoleTitle(strtitle, N); //获得当前窗口标题
printf("当前窗口标题为:%s\n", strtitle);
_getch();
SetConsoleTitle("控制台窗口操作"); //设置窗口标题为“控制台窗口操作”
GetConsoleTitle(strtitle, N); //获得当前窗口标题
printf("当前窗口标题为:%s\n", strtitle);
_getch();
SetConsoleScreenBufferSize(handle_out, size); // 重新设置缓冲区大小
_getch();
SMALL_RECT rc = {0, 0, 80-1, 25-1}; // 重置窗口位置和大小
SetConsoleWindowInfo(handle_out, 1, &rc);
CloseHandle(handle_out); //关闭标准输出设备句柄
return 0;
}

其中,SetConsoleScreenBufferSize 函数指定 新的控制台屏幕缓冲区的大小,以字符 为单位。

指定的宽度和高度不能小于控制台屏幕缓冲区窗口的宽度和高度。指定的大小也不能小于系统允许的最小大小。这个最低取决于控制台当前的字体大小 (由用户选定)。

三、文本属性

设置文本属性使用函数 SetConsoleTextAttribute ,原型如下:

1
2
3
4
BOOL SetConsoleTextAttribute(   // 设置WriteConsole等函数的字符属性  
HANDLE hConsoleOutput, // 句柄
WORD wAttributes // 文本属性
);

3.1 文本属性:

其实就是颜色属性,有 背景色前景色 (就是字符的颜色)两类

每一类只提供三原色 (绿)和 加强色(灰色) ,可与其他颜色搭配使用,使颜色变亮,后面会提到。

最后还有一个反色, 示例 C++ 代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
#include <stdio.h>  
#include <stdlib.h>
#include <windows.h>
#include <conio.h>
/*
基本文本属性
FOREGROUND_BLUE 蓝色
FOREGROUND_GREEN 绿色
FOREGROUND_RED 红色
FOREGROUND_INTENSITY 加强
BACKGROUND_BLUE 蓝色背景
BACKGROUND_GREEN 绿色背景
BACKGROUND_RED 红色背景
BACKGROUND_INTENSITY 背景色加强
COMMON_LVB_REVERSE_VIDEO 反色
*/

const WORD FORE_BLUE = FOREGROUND_BLUE; //蓝色文本属性
const WORD FORE_GREEN = FOREGROUND_GREEN; //绿色文本属性
const WORD FORE_RED = FOREGROUND_RED; //红色文本属性
const WORD FORE_PURPLE = FORE_BLUE | FORE_RED; //紫色文本属性
const WORD FORE_CYAN = FORE_BLUE | FORE_GREEN; //青色文本属性
const WORD FORE_YELLOW = FORE_RED | FORE_GREEN; //黄色文本属性
const WORD FORE_GRAY = FOREGROUND_INTENSITY; //灰色文本属性
const WORD BACK_BLUE = BACKGROUND_BLUE; //蓝色背景属性
const WORD BACK_GREEN = BACKGROUND_GREEN; //绿色背景属性
const WORD BACK_RED = BACKGROUND_RED; //绿色背景属性
const WORD BACK_PURPLE = BACK_BLUE | BACK_RED; //紫色背景属性
const WORD BACK_CYAN = BACK_BLUE | BACK_GREEN; //青色背景属性
const WORD BACK_YELLOW = BACK_RED | BACK_GREEN; //黄色背景属性
const WORD BACK_GRAY = BACKGROUND_INTENSITY; //灰色背景属性

int main()
{
HANDLE handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
CONSOLE_SCREEN_BUFFER_INFO csbi; //定义窗口缓冲区信息结构体
GetConsoleScreenBufferInfo(handle_out, &csbi); //获得窗口缓冲区信息
SetConsoleTextAttribute(handle_out, FORE_BLUE);
printf("蓝色字符\n");
SetConsoleTextAttribute(handle_out, FORE_RED);
printf("红色字符\n");
SetConsoleTextAttribute(handle_out, FORE_GREEN);
printf("绿色字符\n");
SetConsoleTextAttribute(handle_out, FORE_PURPLE);
printf("紫色字符\n");
SetConsoleTextAttribute(handle_out, FORE_CYAN);
printf("青色字符\n");
SetConsoleTextAttribute(handle_out, FORE_YELLOW);
printf("黄色字符\n");
SetConsoleTextAttribute(handle_out, FORE_GRAY);
printf("灰色字符\n");
SetConsoleTextAttribute(handle_out, FORE_GREEN | FORE_BLUE | FORE_RED);
printf("白色字符\n");
SetConsoleTextAttribute(handle_out, BACK_BLUE);
printf("蓝色背景\n");
SetConsoleTextAttribute(handle_out, BACK_RED);
printf("红色背景\n");
SetConsoleTextAttribute(handle_out, BACK_GREEN);
printf("绿色背景\n");
SetConsoleTextAttribute(handle_out, BACK_PURPLE);
printf("紫色背景\n");
SetConsoleTextAttribute(handle_out, BACK_CYAN);
printf("青色背景\n");
SetConsoleTextAttribute(handle_out, BACK_YELLOW);
printf("黄色背景\n");
SetConsoleTextAttribute(handle_out, BACK_GRAY);
printf("灰色背景\n");
SetConsoleTextAttribute(handle_out, BACK_BLUE | BACK_RED | BACK_GREEN);
printf("白色背景\n");
SetConsoleTextAttribute(handle_out, BACK_GREEN | FORE_RED); //示例:绿色背景红色字符
printf("绿色背景与红色字符的混合\n");
SetConsoleTextAttribute(handle_out, FOREGROUND_INTENSITY | FORE_RED); //示例:亮红色字符
printf("亮色的生成,与加强色融合\n");
return 0;
}

上述示例程序最好用 C++ 来编译,因为有 C语言 的编译器或者 IDE 不支持上述的定义常量的方式。需要从这个示例中了解的是三原
色的混合是用 C语言 位运算中的按位或 | 运算符,背景颜色与字符颜色的同时定义也是使用这个运算符融合。

另外,将任意颜色与对应的加强色(灰色,有前景和背景两种,需要对应)融合后会成为对应颜色的高亮版,比如红色字符与前景加强色的融合会结合成亮红色。

至于反色,大家可以试试,当我设置了文本属性为反色后,输入字符都不显示了,但是下标还在移动,我估计反色将白色字符变成了黑色字符,与黑色背景一样,所以没有显示出来。至于反色与其他的组合以及其他的颜色组合,还需要大家一起探索、、、


四、文本属性

4.1 常用的文本输出函数

如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
BOOL FillConsoleOutputAttribute(    // 填充字符属性  
HANDLE hConsoleOutput, // 句柄
WORD wAttribute, // 文本属性
DWORD nLength, // 个数
COORD dwWriteCoord, // 开始位置
LPDWORD lpNumberOfAttrsWritten // 返回填充的个数
);
BOOL FillConsoleOutputCharacter( // 填充指定数据的字符
HANDLE hConsoleOutput, // 句柄
TCHAR cCharacter, // 字符
DWORD nLength, // 字符个数
COORD dwWriteCoord, // 起始位置
LPDWORD lpNumberOfCharsWritten // 已写个数
);
BOOL WriteConsoleOutputCharacter( // 在指定位置处写指定数量的字符
HANDLE hConsoleOutput, // 句柄
LPCTSTR lpCharacter, // 字符串
DWORD nLength, // 字符个数
COORD dwWriteCoord, // 起始位置
LPDWORD lpNumberOfCharsWritten // 已写个数
);

另外再介绍一个表示区域的结构体,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
typedef struct _SMALL_RECT  //表示矩形区域的结构体  
{
SHORT Left; //左边界
SHORT Top; //上边界
SHORT Right; //右边界
SHORT Bottom; //下边界
} SMALL_RECT;
/*
微软官方的说法是
Left 区域的左上顶点的X坐标
Top 区域的左上顶点的Y坐标
Right 区域的右下顶点的X坐标
Bottom 区域的右下顶点的Y坐标
*/

4.2 示例程序

如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include <stdio.h>  
#include <stdlib.h>
#include <Windows.h>
#include <conio.h>

int main()
{
char *str = "Hello World!"; //定义输出信息
int len = strlen(str), i;
WORD shadow = BACKGROUND_INTENSITY; //阴影属性
WORD text = BACKGROUND_GREEN | BACKGROUND_INTENSITY; //文本属性
HANDLE handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
CONSOLE_SCREEN_BUFFER_INFO csbi; //定义窗口缓冲区信息结构体
GetConsoleScreenBufferInfo(handle_out, &csbi); //获得窗口缓冲区信息
SMALL_RECT rc; //定义一个文本框输出区域
COORD posText; //定义文本框的起始坐标
COORD posShadow; //定义阴影框的起始坐标
//确定区域的边界
rc.Top = 8; //上边界
rc.Bottom = rc.Top + 4; //下边界
rc.Left = (csbi.dwSize.X - len) / 2 - 2; //左边界,为了让输出的字符串居中
rc.Right = rc.Left + len + 4; //右边界
//确定文本框起始坐标
posText.X = rc.Left;
posText.Y = rc.Top;
//确定阴影框的起始坐标
posShadow.X = posText.X + 1;
posShadow.Y = posText.Y + 1;
for (i=0; i<5; ++i) //先输出阴影框
{
FillConsoleOutputAttribute(handle_out, shadow, len + 4, posShadow, NULL);
posShadow.Y++;
}
for (i=0; i<5; ++i) //在输出文本框,其中与阴影框重合的部分会被覆盖掉
{
FillConsoleOutputAttribute(handle_out, text, len + 4, posText, NULL);
posText.Y++;
}
//设置文本输出处的坐标
posText.X = rc.Left + 2;
posText.Y = rc.Top + 2;
WriteConsoleOutputCharacter(handle_out, str, len, posText, NULL); //输出字符串
SetConsoleTextAttribute(handle_out, csbi.wAttributes); // 恢复原来的属性
CloseHandle(handle_out);
return 0;
}

以上样例在Code::Blocks 13.12中编译通过。


五、文本移动

5.1文本移动的函数

如下:

1
2
3
4
5
6
7
8
BOOL ScrollConsoleScreenBuffer(             //文本移动函数  
HANDLE hConsoleOutput, //句柄
const SMALL_RECT *lpScrollRectangle, //移动区域
const SMALL_RECT *lpClipRectangle, //裁剪区域,如果为NULL,那么将代表整个屏幕缓冲区
COORD dwDestinationOrigin, //移动到的位置,这个点将成为移动区域的左上顶点
const CHAR_INFO *lpFill //空出区域的填充字符
);

5.2 文本移动样例:

下面来看一个移动文本的样例程序,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#include <stdio.h>  
#include <conio.h>
#include <Windows.h>
#include <stdlib.h>

int main()
{
HANDLE handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
CONSOLE_SCREEN_BUFFER_INFO csbi; //定义窗口缓冲区信息结构体
SMALL_RECT scroll; //定义移动区域
COORD pos = {0, 5}; //移动位置
CHAR_INFO chFill; //定义填充字符
GetConsoleScreenBufferInfo(handle_out, &csbi); //获得窗口缓冲区信息
//定义填充字符的各个参数及属性
chFill.Char.AsciiChar = ' ';
chFill.Attributes = csbi.wAttributes;
//输出文本
printf("00000000000000000000000000000\n");
printf("11111111111111111111111111111\n");
printf("22222222222222222222222222222\n");
printf("33333333333333333333333333333\n");
//确定区域
scroll.Left = 1;
scroll.Top = 1;
scroll.Right = 10;
scroll.Bottom = 2;
ScrollConsoleScreenBuffer(handle_out, &scroll, NULL, pos, &chFill); //移动文本
return 0;
}

在上面的样例程序中,裁剪区域是整个控制台窗口的屏幕缓冲区

现在如果我们把裁剪区域设定为与移动区域一样,也就是说 ScrollConsoleScreenBuffer 函数的第三个参数也改成 &scroll ,那么结果会怎么样呢?

为什么会发生这种现象呢?

很明显示因为裁剪区域的设定问题,现在我们把裁剪区域依旧设定成移动区域,但是我们只把移动区域下移一行而不是移动在别的位置,看看会有什么现象发生?

现在我们应该可以猜想出结论了,别急,再做一个实验,现在我们将裁减区域又重新改为整个屏幕缓冲区,看看会有什么样的现象发生?

再来最后一个实验,我们将裁减区域减小为移动区域的上半部分,继续执行下移一行的操作,看看最终结果会怎么样?

5.3 结论

现在,我们可以得出几个结论:

  • 一、 裁减区域以外的区域不会受文本移动的影响。具体是:

    • 裁减区域以外的区域不会被移动过来的区域覆盖,

    • 裁减区域以外的区域被移动到他处之后原区域不发生变化,因此不需要填充字符。

总的归纳来说也就是原来是什么样子,文本移动后还是什么样子,不会改变。

  • 二、 裁减区域以内的区域受文本移动的影响。具体是:

    • 当裁减区域以内的区域被移动到他处造成该区域为空时会被设定的字符填充,

    • 裁减区域以内的区域会被移动过来的区域覆盖。

总的归纳来说也就是完全受文本移动的影响,移动过来就被覆盖,被移走就由设定的字符来填充


六、键盘事件

输入事件中的键盘事件通常有字符事件和按键事件,这些事件的附带信息构成了键盘输入的信息

而想要读取这些信息,是要通过 API 函数 ReadConsoleInput 来获取的,函数原型如下:

1
2
3
4
5
6
7
BOOL ReadConsoleInput(              //读取输入信息  
HANDLE hConsoleInput, //句柄
PINPUT_RECORD lpBuffer, //输入事件结构体的指针
DWORD nLength, //要读取的记录数
LPDWORD lpNumberOfEventsRead //用来接受成功读取记录数的指针
); //如果该函数成功调用,返回非零值
//输入事件结构体的指针可以是结构体数组的首地址,这样就可以一次性读取多个记录数。

下面介绍几个和读取键盘输入事件有关的结构体,各结构体原型如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
typedef struct _INPUT_RECORD    //输入事件结构体  
{
WORD EventType; //事件类型
union
{
KEY_EVENT_RECORD KeyEvent; //按键事件
MOUSE_EVENT_RECORD MouseEvent; //鼠标事件
WINDOW_BUFFER_SIZE_RECORD WindowBufferSizeEvent;
MENU_EVENT_RECORD MenuEvent;
FOCUS_EVENT_RECORD FocusEvent;
} Event; //具体的事件
} INPUT_RECORD;
/*
其中事件类型EventType的值有5种
KEY_EVENT 代表Event包含一个KEY_EVENT_RECODE结构体
MOUSE_EVENT 代表Event包含一个MOUSE_EVENT_RECODE结构体
WINDOW_BUFFER_SIZE_EVENT 代表Event包含一个WINDOW_BUFFER_SIZE_EVENT_RECORD结构体
MENU_EVENT 代表Event包含一个MENU_EVENT_RECORD结构体
FOCUS_EVENT 代表Event包含一个FOCUS_EVENT_RECORD结构体
*/

typedef struct _KEY_EVENT_RECORD //键盘事件结构体
{
BOOL bKeyDown; //按键状态,true代表键按下,false代表键释放
WORD wRepeatCount; //按键次数
WORD wVirtualKeyCode; //虚拟键
WORD wVirtualScanCode; //虚拟键扫描码
union
{
WCHAR UnicodeChar; //解释成Unicode宽字符
CHAR AsciiChar; //解释成ASCII码字符
} uChar;
DWORD dwControlKeyState; //控制键状态
} KEY_EVENT_RECORD;
/*
控制键各状态的值
ENHANCED_KEY 扩展键被按下
LEFT_ALT_PRESSED 左Alt键被按下
LEFT_CTRL_PRESSED 左Ctrl键被按下
RIGHT_ALT_PRESSED 右Alt键被按下
RIGHT_CTRL_PRESSED 右Ctrl键被按下
NUMLOCK_ON 数字锁定被打开
SCROLLLOCK_ON 滚动锁定被打开
CAPSLOCK_ON 大写锁定被打开
SHIFT_PRESSED Shift键被按下
*/

当输入事件为键盘事件时,事件类型就为键盘事件,为其他事件时,事件类型就为对应的事件。

另外,键盘上每个有意义的键都对应着一个唯一的扫描码,虽然扫描码可以作为键的标识,但是它是依赖于具体的设备的。

因此,在应用程序中,使用的往往是与具体设备无关的虚拟键代码。这种虚拟键代码是一种与具体设备无关的键盘编码。而控制键状态比如大写锁定开启状态,Ctrl键按下状态等

6.1 部分常用虚拟键代码表

下面是部分常用虚拟键代码表:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/* 
虚拟键代码 值 键名称
-----------------------------------------------------
VK_BACK 0x08 退格键
VK_TAB 0x09 Tab键
VK_RETURN 0x0D 回车键
VK_SHIFT 0x10 Shift键
VK_LSHIFT 0xA0 左Shift键
VK_RSHIFT 0xA1 右Shift键
VK_CONTROL 0x11 Ctrl键
VK_LCONTROL 0xA2 左Ctrl键
VK_RCONTROL 0xA3 右Ctrl键
VK_MENU 0x12 Alt键
VK_LMENU 0xA4 左Alt键
VK_RMENU 0xA5 右Alt键
VK_PAUSE 0x13 Pause键
VK_CAPITAL 0x14 Caps Lock键
VK_NUMLOCK 0x90 Num Lock键
VK_SCROLL 0x91 Scroll Lock键
VK_ESCAPE 0x1B Esc键
VK_SPACE 0x20 空格键
VK_PRIOR 0x21 Page Up键
VK_NEXT 0x22 Page Down键
VK_END 0x23 End键
VK_HOME 0x24 Home键
VK_LEFT 0x25 左方向键
VK_UP 0x26 上方向键
VK_RIGHT 0x27 右方向键
VK_DOWN 0x28 下方向键
VK_DELETE 0x2E Delete键
VK_INSERT 0x2D Insert键
'0' 0x30 0键(非小键盘)
'1' 0x31 1键(非小键盘)
'2' 0x32 2键(非小键盘)
... ... ...
'9' 0x39 9键(非小键盘)
'A' 0x41 A键
'B' 0x42 B键
... ... ...
'Z' 0x5A Z键
VK_SLEEP 0x5F Sleep键
VK_NUMPAD0 0x60 小键盘0键
VK_NUMPAD1 0x61 小键盘1键
VK_NUMPAD2 0x62 小键盘2键
... ... ...
VK_NUMPAD9 0x69 小键盘9键
VK_MULTIPLY 0x6A 小键盘乘键*
VK_ADD 0x6B 小键盘加键+
VK_SUBTRACT 0x6D 小键盘减键-
VK_DIVIDE 0x6F 小键盘除键/
VK_DECIMAL 0x6E 小键盘点键.
VK_F1 0x70 F1键
VK_F2 0x71 F2键
... ... ...
VK_F12 0x7B F12键
VK_F13 0x7C F13键 注:别问我,我也不知道什么电脑有这么多键
... ... ...
VK_F24 0x87 F24键
VK_OEM_1 0xBA ;:键
VK_OEM_2 0xBF /?键
VK_OEM_3 0xC0 ·~键
VK_OEM_4 0xDB [{键
VK_OEM_5 0xDC \|键
VK_OEM_6 0xDD ]}键
VK_OEM_7 0xDE '"键
VK_OEM_PLUS 0xBB =+键
VK_OEM_MINUS 0xBD -_键
VK_OEM_COMMA 0xBC ,<键
VK_OEM_PERIOD 0xBE .>键
*/

以上是部分常用的微软虚拟键盘码表,想要知道更详细的,请参见MSDN。其中各个虚拟键的具体使用情况根据各人编译器或IDE等的不同而有所差异。

6.2 样例程序:

实现按 Esc 键就输出 Esc

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
#include <stdio.h>  
#include <stdlib.h>
#include <windows.h>
#include <conio.h>
#define true 1
#define false 0

int main()
{
HANDLE handle_in = GetStdHandle(STD_INPUT_HANDLE); //获得标准输入设备句柄
INPUT_RECORD keyrec; //定义输入事件结构体
DWORD res; //定义返回记录
for (;;)
{
ReadConsoleInput(handle_in, &keyrec, 1, &res); //读取输入事件
if (keyrec.EventType == KEY_EVENT) //如果当前事件是键盘事件
{
if (keyrec.Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE) //当前事件的虚拟键为Esc键
{
printf("Esc ");
}
}
}
return 0;
}

在上面的样例程序中,当你按下Esc键后又马上释放,程序会输出两次Esc,因为有两次事件的虚拟键代码都是Esc键的代码,一次是按下,一次是释放。

如果要实现按下键后出现反应,释放不出现反应,需要监听事件类型,将条件改为:

1
2
3
if (keyrec.Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE  
&& keyrec.Event.KeyEvent.bKeyDown == true) //表示当前为键按下而不是键释放

即可

根据控制键的状态我们可以实现不同的状态输出不同的值以及组合键的实现,下面的样例程序在大写锁定打开时输入 A 键则输出大写字母 A ,否则输出小写字母 a 。而在 Shift 键被按下的状态是则输出 Shift + A 以及 Shift + a 。样例程序如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
#include <stdio.h>  
#include <stdlib.h>
#include <windows.h>
#include <conio.h>
#define true 1
#define false 0

int main()
{
HANDLE handle_in = GetStdHandle(STD_INPUT_HANDLE); //获得标准输入设备句柄
INPUT_RECORD keyrec; //定义输入事件结构体
DWORD res; //定义返回记录
for (;;)
{
ReadConsoleInput(handle_in, &keyrec, 1, &res); //读取输入事件
if (keyrec.EventType == KEY_EVENT) //如果当前事件是键盘事件
{
if (keyrec.Event.KeyEvent.wVirtualKeyCode == 'A'
&& keyrec.Event.KeyEvent.bKeyDown == true) //当按下字母A键时
{
if (keyrec.Event.KeyEvent.dwControlKeyState & SHIFT_PRESSED) //Shift键为按下状态
{
printf("Shift+");
}
if (keyrec.Event.KeyEvent.dwControlKeyState & CAPSLOCK_ON) //大写锁定为打开状态
{
printf("A ");
}
else //大写锁定关闭状态
{
printf("a ");
}
}
}
}
return 0;
}

由上例需要了解到的是,各个控制键状态的的确定并不是使用等于符号==而是按位与&运算符,因为在同一时刻可能有多种控制键状态值,比如各种锁定都被打开且各种控制键也被同时按下。使用&运算符则显得尤其高明,方便查询各个控制键的状态而不使之出现冲突。呵呵,不服不行啊,感慨一下,还是要多学习一下别人高明的地方,比如灵活运用位运算符实现各种功能等等······


七、鼠标事件

上次讲的是键盘事件,这次我们介绍鼠标事件。下面先介绍下鼠标事件的结构体以及相关信息。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
typedef struct _MOUSE_EVENT_RECORD      //鼠标事件结构体  
{
COORD dwMousePosition; //当前鼠标在控制台窗口缓冲区的位置
DWORD dwButtonState; //鼠标按键的状态
DWORD dwControlKeyState; //控制键状态
DWORD dwEventFlags; //鼠标事件类型
} MOUSE_EVENT_RECORD;
/*
其中鼠标按键状态dwButtonState可能的值有
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
FROM_LEFT_1ST_BUTTON_PRESSED 最左边的鼠标键被按下 一般来说就是鼠标左键
FROM_LEFT_2ND_BUTTON_PRESSED 左起第二个鼠标键被按下 一般来说是鼠标中键,就是滚轮键
FROM_LEFT_3RD_BUTTON_PRESSED 左起第三个鼠标键被按下
FROM_LEFT_4TH_BUTTON_PRESSED 左起第四个鼠标键被按下
RIGHTMOST_BUTTON_PRESSED 最右边的鼠标键被按下 一般来说是鼠标右键
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
控制键状态dwControlKeyState与键盘事件的一样
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
ENHANCED_KEY 扩展键被按下
LEFT_ALT_PRESSED 左Alt键被按下
LEFT_CTRL_PRESSED 左Ctrl键被按下
RIGHT_ALT_PRESSED 右Alt键被按下
RIGHT_CTRL_PRESSED 右Ctrl键被按下
NUMLOCK_ON 数字锁定被打开
SCROLLLOCK_ON 滚动锁定被打开
CAPSLOCK_ON 大写锁定被打开
SHIFT_PRESSED Shift键被按下
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
鼠标事件类型dwEventFlags有以下几种
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
DOUBLE_CLICK 双击,第一击只作为普通按键事件,第二击才作为双击事件
MOUSE_HWHEELED 水平鼠标滚轮移动
MOUSE_MOVED 鼠标移动
MOUSE_WHEELED 垂直鼠标滚轮移动
0 当鼠标有键被按下或者释放
*/

7.1 样例程序

实现在控制台窗口缓冲区的最下面一行显示当前鼠标在缓冲区的坐标,单击左键在当前鼠标位置输出字母 A ,单击右键则输出字母 B ,双击任何鼠标键退出的功能。程序如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
#include <stdio.h>  
#include <windows.h>
#include <conio.h>

HANDLE handle_in;
HANDLE handle_out;
CONSOLE_SCREEN_BUFFER_INFO csbi; //定义窗口缓冲区信息结构体

void DisplayMousePosition(COORD pos); //显示鼠标所在位置

void gotoxy(int x, int y); //将光标移到坐标为(x,y)的位置

int main()
{
handle_in = GetStdHandle(STD_INPUT_HANDLE); //获得标准输入设备句柄
handle_out = GetStdHandle(STD_OUTPUT_HANDLE); //获得标准输出设备句柄
INPUT_RECORD mouserec; //定义输入事件结构体
DWORD res; //用于存储读取记录
COORD pos; //用于存储鼠标当前位置
COORD size = {80, 25}; //窗口缓冲区大小
GetConsoleScreenBufferInfo(handle_out, &csbi); //获得窗口缓冲区信息
SetConsoleScreenBufferSize(handle_out, size); //设置窗口缓冲区大小
for (;;)
{
ReadConsoleInput(handle_in, &mouserec, 1, &res); //读取输入事件
pos = mouserec.Event.MouseEvent.dwMousePosition; //获得当前鼠标位置
gotoxy(0, 24); //在第25行显示鼠标位置
DisplayMousePosition(pos); //显示鼠标位置
if (mouserec.EventType == MOUSE_EVENT) //如果当前为鼠标事件
{
gotoxy(pos.X, pos.Y);
//单击鼠标左键,输出字符A
if (mouserec.Event.MouseEvent.dwButtonState == FROM_LEFT_1ST_BUTTON_PRESSED)
{
putchar('A');
}
//单击鼠标右键,输出字符B
if (mouserec.Event.MouseEvent.dwButtonState == RIGHTMOST_BUTTON_PRESSED)
{
putchar('B');
}
//双击退出
if (mouserec.Event.MouseEvent.dwEventFlags == DOUBLE_CLICK)
{
break;
}
}
}
CloseHandle(handle_out);
CloseHandle(handle_in);
return 0;
}

void DisplayMousePosition(COORD pos)
{
COORD dis = {0, 24}; //在第24行显示鼠标位置
WORD att = FOREGROUND_GREEN | FOREGROUND_INTENSITY; //文本属性
GetConsoleScreenBufferInfo(handle_out, &csbi); //获得窗口缓冲区信息
printf("X = %3d, Y = %3d", (int)pos.X, (int)pos.Y);
FillConsoleOutputAttribute(handle_out, att, 16, dis, NULL); //填充文本属性
return;
}

void gotoxy(int x, int y)
{
COORD pos = {x, y};
SetConsoleCursorPosition(handle_out, pos);
}

附上用本程序写的Hello world!的图:

注意:当使用system函数后鼠标事件无法正常发生。

————————————————
版权声明:本文为CSDN博主「KKK_Kiral」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/liluo_2951121599/article/details/66474233

Author: 哒琳

Permalink: http://blog.jieis.cn/2019/2421baf4-b555-4793-8f75-afa5d3617229.html

Comments