通过 ncurses 在终端创建一个冒险游戏 | Linux 中国

怎样使用 curses 函数读取键盘并操作屏幕。
-- Jim Hall

致谢
编译自 | http://www.linuxjournal.com/content/creating-adventure-game-terminal-ncurses 
 作者 | Jim Hall
 译者 | leemeans ? ? 共计翻译:5 篇 贡献时间:22 天

怎样使用 curses 函数读取键盘并操作屏幕。

之前的文章[1]介绍了 ncurses 库,并提供了一个简单的程序展示了一些将文本放到屏幕上的 curses 函数。在接下来的文章中,我将介绍如何使用其它的 curses 函数。

探险

当我逐渐长大,家里有了一台苹果 II 电脑。我和我兄弟正是在这台电脑上自学了如何用 AppleSoft BASIC 写程序。我在写了一些数学智力游戏之后,继续创造游戏。作为 80 年代的人,我已经是龙与地下城桌游的粉丝,在游戏中角色扮演一个追求打败怪物并在陌生土地上抢掠的战士或者男巫,所以我创建一个基本的冒险游戏也在情理之中。

AppleSoft BASIC 支持一种简洁的特性:在标准分辨率图形模式(GR 模式)下,你可以检测屏幕上特定点的颜色。这为创建一个冒险游戏提供了捷径。比起创建并更新周期性传送到屏幕的内存地图,我现在可以依赖 GR 模式为我维护地图,我的程序还可以在玩家的角色(LCTT 译注:此处 character 双关一个代表玩家的角色,同时也是一个字符)在屏幕四处移动的时候查询屏幕。通过这种方式,我让电脑完成了大部分艰难的工作。因此,我的自顶向下的冒险游戏使用了块状的 GR 模式图形来展示我的游戏地图。

我的冒险游戏使用了一张简单的地图,上面有一大片绿地伴着山脉从中间蔓延向下和一个在左上方的大湖。我要粗略地为桌游战役绘制这个地图,其中包含一个允许玩家穿过到远处的狭窄通道。

图 1. 一个有湖和山的简单桌游地图

你可以用 curses 绘制这个地图,并用字符代表草地、山脉和水。接下来,我描述怎样使用 curses 那样做,以及如何在 Linux 终端创建和进行类似的一个冒险游戏。

构建程序

在我的上一篇文章,我提到了大多数 curses 程序以相同的一组指令获取终端类型和设置 curses 环境:

  1. initscr();

  2. cbreak();

  3. noecho();

在这个程序,我添加了另外的语句:

  1. keypad(stdscr, TRUE);

这里的 TRUE 标志允许 curses 从用户终端读取小键盘和功能键。如果你想要在你的程序中使用上下左右方向键,你需要使用这里的 keypad(stdscr, TRUE)

这样做了之后,你现在可以开始在终端屏幕上绘图了。curses 函数包括了一系列在屏幕上绘制文本的方法。在我之前的文章中,我展示了 addch() 和 addstr() 函数以及在添加文本之前先移动到指定屏幕位置的对应函数 mvaddch() 和 mvaddstr()。为了在终端上创建这个冒险游戏的地图,你可以使用另外一组函数:vline() 和 hline(),以及它们对应的函数 mvvline() 和 mvhline()。这些 mv 函数接受屏幕坐标、一个要绘制的字符和要重复此字符的次数的参数。例如,mvhline(1, 2, '-', 20) 将会绘制一条开始于第一行第二列并由 20 个横线组成的线段。

为了以编程方式绘制地图到终端屏幕上,让我们先定义这个 draw_map() 函数:

  1. #define GRASS     ' '

  2. #define EMPTY     '.'

  3. #define WATER     '~'

  4. #define MOUNTAIN  '^'

  5. #define PLAYER    '*'

  6. void draw_map(void)

  7. {

  8.    int y, x;

  9.    /* 绘制探索地图 */

  10.    /* 背景 */

  11.    for (y = 0; y < LINES; y++) {

  12.        mvhline(y, 0, GRASS, COLS);

  13.    }

  14.    /* 山和山道 */

  15.    for (x = COLS / 2; x < COLS * 3 / 4; x++) {

  16.        mvvline(0, x, MOUNTAIN, LINES);

  17.    }

  18.    mvhline(LINES / 4, 0, GRASS, COLS);

  19.    /* 湖 */

  20.    for (y = 1; y < LINES / 2; y++) {

  21.        mvhline(y, 1, WATER, COLS / 3);

  22.    }

  23. }

在绘制这副地图时,记住填充大块字符到屏幕所使用的 mvvline() 和 mvhline() 函数。我绘制从 0 列开始的字符水平线(mvhline)以创建草地区域,直到占满整个屏幕的高度和宽度。我绘制从 0 行开始的多条垂直线(mvvline)在此上添加了山脉,绘制单行水平线添加了一条山道(mvhline)。并且,我通过绘制一系列短水平线(mvhline)创建了湖。这种绘制重叠方块的方式看起来似乎并没有效率,但是记住在我们调用 refresh() 函数之前 curses 并不会真正更新屏幕。

绘制完地图,创建游戏就还剩下进入循环让程序等待用户按下上下左右方向键中的一个然后让玩家图标正确移动了。如果玩家想要移动的地方是空的,就应该允许玩家到那里。

你可以把 curses 当做捷径使用。比起在程序中实例化一个版本的地图并复制到屏幕这么复杂,你可以让屏幕为你跟踪所有东西。inch() 函数和相关联的 mvinch() 函数允许你探测屏幕的内容。这让你可以查询 curses 以了解玩家想要移动到的位置是否被水填满或者被山阻挡。这样做你需要一个之后会用到的一个帮助函数:

  1. int is_move_okay(int y, int x)

  2. {

  3.    int testch;

  4.    /* 如果要进入的位置可以进入,返回 true */

  5.    testch = mvinch(y, x);

  6.    return ((testch == GRASS) || (testch == EMPTY));

  7. }

如你所见,这个函数探测行 x、列 y 并在空间未被占据的时候返回 true,否则返回 false

这样我们写移动循环就很容易了:从键盘获取一个键值然后根据是上下左右键移动用户字符。这里是一个这种循环的简单版本:

  1.    do {

  2.        ch = getch();

  3.        /* 测试输入的值并获取方向 */

  4.        switch (ch) {

  5.        case KEY_UP:

  6.            if ((y > 0) && is_move_okay(y - 1, x)) {

  7.                y = y - 1;

  8.            }

  9.            break;

  10.        case KEY_DOWN:

  11.            if ((y < LINES - 1) && is_move_okay(y + 1, x)) {

  12.                y = y + 1;

  13.            }

  14.            break;

  15.        case KEY_LEFT:

  16.            if ((x > 0) && is_move_okay(y, x - 1)) {

  17.                x = x - 1;

  18.            }

  19.            break;

  20.        case KEY_RIGHT

  21.            if ((x < COLS - 1) && is_move_okay(y, x + 1)) {

  22.                x = x + 1;

  23.            }

  24.            break;

  25.        }

  26.    }

  27.    while (1);

为了在游戏中使用这个循环,你需要在循环里添加一些代码来启用其它的键(例如传统的移动键 WASD),以提供让用户退出游戏和在屏幕上四处移动的方法。这里是完整的程序:

  1. /* quest.c */

  2. #include

  3. #include

  4. #define GRASS     ' '

  5. #define EMPTY     '.'

  6. #define WATER     '~'

  7. #define MOUNTAIN  '^'

  8. #define PLAYER    '*'

  9. int is_move_okay(int y, int x);

  10. void draw_map(void);

  11. int main(void)

  12. {

  13.    int y, x;

  14.    int ch;

  15.    /* 初始化curses */

  16.    initscr();

  17.    keypad(stdscr, TRUE);

  18.    cbreak();

  19.    noecho();

  20.    clear();

  21.    /* 初始化探索地图 */

  22.    draw_map();

  23.    /* 在左下角初始化玩家 */

  24.    y = LINES - 1;

  25.    x = 0;

  26.    do {

  27.    /* 默认获得一个闪烁的光标--表示玩家字符 */

  28.    mvaddch(y, x, PLAYER);

  29.    move(y, x);

  30.    refresh();

  31.    ch = getch();

  32.    /* 测试输入的键并获取方向 */

  33.    switch (ch) {

  34.    case KEY_UP:

  35.    case 'w':

  36.    case 'W':

  37.        if ((y > 0) && is_move_okay(y - 1, x)) {

  38.        mvaddch(y, x, EMPTY);

  39.        y = y - 1;

  40.        }

  41.        break;

  42.    case KEY_DOWN:

  43.    case 's':

  44.    case 'S':

  45.        if ((y < LINES - 1) && is_move_okay(y + 1, x)) {

  46.        mvaddch(y, x, EMPTY);

  47.        y = y + 1;

  48.        }

  49.        break;

  50.    case KEY_LEFT:

  51.    case 'a':

  52.    case 'A':

  53.        if ((x > 0) && is_move_okay(y, x - 1)) {

  54.        mvaddch(y, x, EMPTY);

  55.        x = x - 1;

  56.        }

  57.        break;

  58.    case KEY_RIGHT:

  59.    case 'd':

  60.    case 'D':

  61.        if ((x < COLS - 1) && is_move_okay(y, x + 1)) {

  62.        mvaddch(y, x, EMPTY);

  63.        x = x + 1;

  64.        }

  65.        break;

  66.    }

  67.    }

  68.    while ((ch != 'q') && (ch != 'Q'));

  69.    endwin();

  70.    exit(0);

  71. }

  72. int is_move_okay(int y, int x)

  73. {

  74.    int testch;

  75.    /* 当空间可以进入时返回true */

  76.    testch = mvinch(y, x);

  77.    return ((testch == GRASS) || (testch == EMPTY));

  78. }

  79. void draw_map(void)

  80. {

  81.    int y, x;

  82.    /* 绘制探索地图 */

  83.    /* 背景 */

  84.    for (y = 0; y < LINES; y++) {

  85.    mvhline(y, 0, GRASS, COLS);

  86.    }

  87.    /* 山脉和山道 */

  88.    for (x = COLS / 2; x < COLS * 3 / 4; x++) {

  89.    mvvline(0, x, MOUNTAIN, LINES);

  90.    }

  91.    mvhline(LINES / 4, 0, GRASS, COLS);

  92.    /* 湖 */

  93.    for (y = 1; y < LINES / 2; y++) {

  94.    mvhline(y, 1, WATER, COLS / 3);

  95.    }

  96. }

在完整的程序清单中,你可以看见使用 curses 函数创建游戏的完整布置:

☉ 初始化 curses 环境。
☉ 绘制地图。
☉ 初始化玩家坐标(左下角)
☉ 循环:
◈ 绘制玩家的角色。
◈ 从键盘获取键值。
◈ 对应地上下左右调整玩家坐标。
◈ 重复。
☉ 完成时关闭curses环境并退出。

开始玩

当你运行游戏时,玩家的字符在左下角初始化。当玩家在游戏区域四处移动的时候,程序创建了“一串”点。这样可以展示玩家经过了的点,让玩家避免经过不必要的路径。

图 2. 初始化在左下角的玩家

图 3. 玩家可以在游戏区域四处移动,例如湖周围和山的通道

为了创建上面这样的完整冒险游戏,你可能需要在他/她的角色在游戏区域四处移动的时候随机创建不同的怪物。你也可以创建玩家可以发现在打败敌人后可以掠夺的特殊道具,这些道具应能提高玩家的能力。

但是作为起点,这是一个展示如何使用 curses 函数读取键盘和操纵屏幕的好程序。

下一步

这是一个如何使用 curses 函数更新和读取屏幕和键盘的简单例子。按照你的程序需要做什么,curses 可以做得更多。在下一篇文章中,我计划展示如何更新这个简单程序以使用颜色。同时,如果你想要学习更多 curses,我鼓励你去读位于 Linux 文档计划的 Pradeep Padala 写的如何使用 NCURSES 编程[2]


via: http://www.linuxjournal.com/content/creating-adventure-game-terminal-ncurses

作者:Jim Hall[4] 译者:Leemeans 校对:wxy

本文由 LCTT 原创编译,Linux中国 荣誉推出


版权声明:本文为F8qG7f9YD02Pe原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/F8qG7f9YD02Pe/article/details/79386783

智能推荐

在liunx中安装elasticsearch(Elasticsearch head插件安装,kibana安装,ik分词器安装)

目录 安装Elasticsearch(单节点Linux环境) Elasticsearch head插件安装 kibana安装 安装ik分词器 安装Elasticsearch(单节点Linux环境) 我安装的是7.6.1版本以下是提供的安装包主要官网下载太慢 elasticsearch-7.6.1-linux-x86_64.tar.gz elasticsearch-analysis-ik-7.6.1...

前端小练习:jQuery酷炫照片墙

jQuery酷炫照片墙 效果展示: HTML代码: css代码: jQuery代码: 方法 解释 transform transform 属性向元素应用 2D 或 3D 转换。该属性允许我们对元素进行旋转、缩放、移动或倾斜。W3scool Math.random() 产生随机数。编程狮 translate 绘图函数编程狮 attr attr() 方法设置或返回被选元素的属性和值。编程狮 anima...

springMVC拦截器

一、     SpringMVC拦截器实现原理 用户请求到DispatherServlet中,DispatherServlet调用HandlerMapping查找Handler,HandlerMapping返回一个拦截器链(HandlerExecutionChain),springmvc中的拦截器是通过HandlerMapping发起的。 &nbs...

Unity Json反序列化

Json反序列化 结果:...

[机器学习-回归算法]Sklearn之线性回归实战

Sklearn之线性回归实战 一,前言 二,热身例子 三,贸易公司的简单例子 四,Sklearn 官网里的一个例子 参考资料 一,前言 一元线性回归的理论片请看我这个链接 二,热身例子 预测直线 y=1x1+2x2+3y = 1x_1 + 2x_2 +3y=1x1​+2x2​+3 导入LinearRegression 从Sklearn.liear_model 包里 拟合数据也可以说是训练 检验正确...

猜你喜欢

Android 开发者,你真的懂 Context 吗?

Android Context 详解 前言 一、Context是什么 二、Context结构 1、ContextImpl类介绍 2、ContextWrapper类介绍 3、ContextThemeWrapper 三、Context的数量 四、Context注意事项 五、如何正确回复以上面试题? 前言 Context 相信所有的 Android 开发人员基本上每天都在接触,因为它太常见了。但是这并不...

SpringMVC ----Json的简单交互处理

SpringMVC--Json Json的介绍 什么是JSON? JSON 和 JavaScript 对象互转 Controller返回JSON数据 Jackson 乱码 乱码的解决方法一 代码优化 乱码统一解决方法 返回json字符串统一解决 测试多个对象的集合输出 输出时间对象 抽取为工具类 FastJson fastjson 三个主要的类: JSONObject JSONArray JSON...

微信小程序自定义组件简单实现

本文将教你如何实现一个自定义的toast提示框,实现后的基本效果图如下: 小程序中一个自定义组件由 json wxml wxss js 4个文件组成的。下面我们一步一步地来创建文件及完成其中的配置: step1:创建自定义组件 首先创建一个components文件夹,用于放置所有自定义的组件,创建之后的目录结构为 其中的toastedit是我们本次...

PyTorch学习(四)--用PyTorch实现线性回归

教程视频:https://www.bilibili.com/video/BV1tE411s7QT 废话不多说,代码如下: 结果: 0 56.52023696899414 1 25.170454025268555 2 11.214292526245117 3 5.001270771026611 4 2.2352840900421143 5 1.0038176774978638 6 0.4554775...

1、Qt 的窗口组件和窗口类型

1、窗口组件 图形用户界面由不同的窗口和窗口组件组成 组件的类型 — 容器类(父组件):用于包含其它的界面组件 — 功能类(子组件):用于实现特定的交互功能 Qt 中没有父组件的顶级组件叫做窗口 QWidget QWidget 继承于 QObject 和 QPaintDevice — QObject 是所有支持 Qt 对象模型的基类 — QPaint...