pong game using ncurses
bounce2d2.c
/*
* bounce2d 1.0
* bounce a character (default is 'o') around the screen
* defined by some parameters
* user input: s slow down x component, S: slow y component
* f speed up x component, F: speed y component
* Q quit
* blocks on read, but timer tick sends SIGALRM caught by ball_move
* build: cc bounce2d.c set_ticker.c -lcurses -o bounce2d
*/
#include <curses.h>
#include <string.h>
#include <signal.h>
#include "bounce.h" struct ppball the_ball; /** the main loop **/ //int flap_pos = RIGHT_EDGE / 2 - LEFT_EDGE;
int flap_pos =;
int old_pos = ;
void set_up();
void wrap_up();
void move_flap();
int bounce_or_lose(struct ppball *); int main()
{
// printf("LEFT_EDGE: %d, RIGHT_EDGE: %d\n",LEFT_EDGE, RIGHT_EDGE);
// printf("TOP_EDGE: %d, BOT_EDGE: %d\n",TOP_ROW, BOT_ROW);
int c;
set_up();
while (((c = getchar())) != 'Q')
{
if (c == 'f') the_ball.x_ttm--;
else if (c == 's') the_ball.x_ttm++;
else if (c == 'F') the_ball.y_ttm--;
else if (c == 'S') the_ball.y_ttm++;
else if (c == 'a'){
if (flap_pos > LEFT_EDGE){
old_pos = flap_pos;
flap_pos -= FLAP_SPEED;
move_flap();
}
}else if (c == 'd'){
if (flap_pos + (int)strlen(FLAP) < RIGHT_EDGE){
old_pos = flap_pos;
flap_pos += FLAP_SPEED;
move_flap();
}
}
}
wrap_up();
// printf("LINES: %d, COLS: %d\n",LINES, COLS);
return ;
} void set_up()
/*
* init structure and other stuff
*/
{
void ball_move(int);
the_ball.y_pos = Y_INIT;
the_ball.x_pos = X_INIT;
the_ball.y_ttg = the_ball.y_ttm = Y_TIM;
the_ball.x_ttg = the_ball.x_ttm = X_TIM;
the_ball.y_dir = ;
the_ball.x_dir = ;
the_ball.symbol = DFL_SYMBOL;
the_ball.x_moved = the_ball.y_moved = false; initscr();
noecho();
crmode(); signal(SIGINT, SIG_IGN);
mvaddch(the_ball.y_pos, the_ball.x_pos, the_ball.symbol);
move_flap();
signal(SIGALRM, ball_move);
set_ticker( / TICKS_PER_SEC);
} void wrap_up()
{
set_ticker();
endwin();
} void move_flap()
{
move(BOT_ROW + , old_pos);
addstr(FLAP); //FLAP is blank. Here it is used to clear its old existence. move(BOT_ROW + , flap_pos);
standout();
addstr(FLAP);
standend();
refresh();
} void ball_move(int signum)
{
int y_cur, x_cur, moved; signal(SIGALRM, SIG_IGN);
x_cur = the_ball.x_pos;
y_cur = the_ball.y_pos;
moved = ; if (the_ball.y_ttm > && the_ball.y_ttg-- == ){
the_ball.y_pos += the_ball.y_dir; /* move */
the_ball.y_ttg = the_ball.y_ttm; /* reset */
the_ball.y_moved = ;
moved = ;
} if (the_ball.x_ttm > && the_ball.x_ttg-- == ){
the_ball.x_pos += the_ball.x_dir; /* move */
the_ball.x_ttg = the_ball.x_ttm; /* reset */
the_ball.x_moved = ;
moved = ;
} if (moved){
mvaddch(y_cur, x_cur, BLANK);
mvaddch(y_cur, x_cur, BLANK);
mvaddch(the_ball.y_pos, the_ball.x_pos, the_ball.symbol);
if(bounce_or_lose(&the_ball)){
// signal(SIGALRM, SIG_IGN);
move(LINES / , COLS / );
addstr("GAME OVER");
refresh();
return;
}
move(LINES-, COLS-);
if (the_ball.x_moved && the_ball.y_moved){
refresh();
the_ball.x_moved = the_ball.y_moved = false; /* reset */
}
}
signal(SIGALRM, ball_move);
} int bounce_or_lose(struct ppball *bp)
/*
* 1 lose
* 0 not lose
*/
{
int return_val = ; if (bp->y_pos == TOP_ROW){
bp->y_dir = ;
}else if (bp->y_pos == BOT_ROW){
bp->y_dir = -;
if (!(bp->x_pos >= flap_pos && bp->x_pos <= (flap_pos + (int)strlen(FLAP)))){
return_val = ;
}
} if (bp->x_pos == LEFT_EDGE){
bp->x_dir = ;
}else if (bp->x_pos == RIGHT_EDGE){
bp->x_dir = -;
}
return return_val;
}
bounce.h
#define BLANK ' '
#define DFL_SYMBOL 'o'
#define TOP_ROW 0
#define BOT_ROW 20
#define LEFT_EDGE 0
#define RIGHT_EDGE 81
#define X_INIT 3
#define Y_INIT 5
#define TICKS_PER_SEC 50
#define Y_TIM 8
#define X_TIM 8
#define FLAP " "
//#define FLAP_LEN 21
#define FLAP_SPEED 1 struct ppball {
int x_ttg; // x 轴下次重画还要等待多少个计时器
int y_ttg; // y 轴下次重画还要等待多少个计时器
int x_ttm; // x 轴移动需要等待的信号间隔
int y_ttm; // y 轴移动絮叨等待的信号间隔
int y_pos;
int x_pos;
int y_dir;
int x_dir;
int x_moved;
int y_moved;
char symbol;
};
set_ticker.c
#include <stdio.h>
#include <sys/time.h>
#include <signal.h>
#include <stdlib.h> /*
* set_ticker.c
* set_ticker( number_of_milliseconds )
* arranges for the interval timer to issue
* SIGALRM's at regular intervals
* returns -1 on error, 0 for ok
*
* arg in milliseconds, converted into micro seoncds
*/ set_ticker( n_msecs )
{
struct itimerval new_timeset;
long n_sec, n_usecs; n_sec = n_msecs / ;
n_usecs = ( n_msecs % ) * 1000L ; new_timeset.it_interval.tv_sec = n_sec; /* set reload */
new_timeset.it_interval.tv_usec = n_usecs; /* new ticker value */
new_timeset.it_value.tv_sec = n_sec ; /* store this */
new_timeset.it_value.tv_usec = n_usecs ; /* and this */ return setitimer(ITIMER_REAL, &new_timeset, NULL);
}
====================================================================================================
A different version:
#include <curses.h>
#include <sys/time.h>
#include <signal.h> #define RIGHT COLS-1 /*球所能到达的当前屏幕最大水平范围*/
#define BOTTOM LINES-1 /*球所能到达的当前屏幕最大垂直范围*/
#define BOARD_LENGTH 10 /*挡板长度*/
#define LEFT 0 /*当前屏幕的最左边*/
#define TOP 0 /*当前屏幕的最上边*/
char BALL= 'O'; /*球的形状*/
char BLANK= ' '; /*覆盖球走过的轨迹*/ int left_board; /*挡板左侧坐标*/
int right_board; /*挡板右侧坐标*/
int is_lose=; int hdir; /*控制球水平运动的变量*/
int vdir; /*控制球垂直运动的变量*/
int pos_X; /*球的横坐标*/
int pos_Y; /*球的纵坐标*/ int delay=;
void moveBall();
void init();
void control(); int main()
{
//初始化 curses
initscr();
crmode(); /*中断模式*/
noecho(); /*关闭回显*/ move(,);
attron(A_BOLD);
addstr("Welcome to the BallGame!");
move(,);
attroff(A_BOLD);
addstr("Help:");
move(,);
addstr("'N':Start a new game.");
move(,);
addstr("'Q':Quit game.");
move(,);
addstr("'KEY_LEFT' :Control baffle left shift.");
move(,);
addstr("'KEY_RIGHT':Control baffle right shift.");
move(,);
addstr("'KEY_UP' :Control of the ball speed.");
move(,);
addstr("'KEY_DOWN' :Control of the ball reducer.");
int flag=;
char choice;
move(,);
addstr("Please choose your choice!(n/q):");
refresh();
choice=getch();
while(flag){
if(choice=='q'||choice=='Q'||choice=='n'||choice=='N')
flag=;
else choice=getch();
}
if(choice=='n'||choice=='N'){ /*开始游戏*/
clear();
move(,);
addstr("BallGame will start! Are you ready?");
refresh();
sleep();
control();
}
else if(choice=='q'||choice=='Q'){ /*退出游戏*/
clear();
move(,);
addstr("You quit the game successfully!");
refresh();
sleep();
endwin();
}
endwin(); /*结束 curses*/
return ;
} void init(){
int i,j;
clear();
if(start_color()==OK){ /*改变球和挡板的颜色*/
attron(A_BOLD); /*打开粗体*/
init_pair(,COLOR_YELLOW,COLOR_BLACK);
attron(COLOR_PAIR());
}
//初始球
pos_X =; /*球初始的横坐标*/
pos_Y = BOTTOM-; /*球初始的纵坐标*/
//初始化球的运动方向,朝右上方运动
hdir=;
vdir=-; //初始挡板
left_board=;
right_board=left_board+BOARD_LENGTH;
for(i=left_board;i<=right_board;i++){ /*显示挡板*/
move(BOTTOM,i);
addch('-');
} //初始刷新时间
signal(SIGALRM,moveBall);
set_ticker(delay); keypad(stdscr,TRUE); /*打开 keypad 键盘响应*/
attroff(A_BLINK); /*关闭 A_BLINK 属性*/ is_lose=;
move(pos_Y,pos_X);
addch(BALL);
move(LINES-, COLS-);
refresh();
usleep(); /*睡眠*/
move(LINES-,COLS-);
refresh();
} void moveBall(){
if(is_lose) return;
signal(SIGALRM,moveBall);
move(pos_Y,pos_X);
addch(BLANK);
pos_X += hdir;
pos_Y += vdir;
//改变球的方向时
if(pos_X >= RIGHT) { /*当球横坐标大于右边边缘时,球反弹朝左运动*/
hdir = -;
beep(); /*球撞墙时,发出声音*/
}
if(pos_X <= LEFT) { /*当球横坐标大于左边边缘时,球反弹朝右运动*/
hdir = ;
beep(); /*球撞墙时,发出声音*/
}
if(pos_Y <= TOP) { /*当球纵坐标大于顶部边缘时,球反弹朝下运动*/
vdir = ;
beep(); /*球撞墙时,发出声音*/
} //当球在底部的时候进行额外的处理
if(pos_Y >= BOTTOM-){
if(pos_X>=left_board&&pos_X<=right_board) /*球在挡板处*/
vdir=-;
else{ /*球不在挡板处*/
is_lose=;
move(pos_Y,pos_X);
addch(BALL);
move(LINES-, COLS-);
refresh();
usleep(delay*); /*睡眠*/
move(pos_Y,pos_X);
addch(BLANK);
pos_X += hdir;
pos_Y += vdir;
move(pos_Y,pos_X);
addch(BALL);
move(LINES-, COLS-);
refresh();
}
}
//不改变球的方向时
move(pos_Y,pos_X);
addch(BALL);
move(LINES-, COLS-);
refresh();
}
void control(){
init();
int cmd;
while ()
{
if(!is_lose){
cmd=getch();
if(cmd=='q'||cmd=='Q'||cmd==) break; //强制退出游戏
//挡板左移
if(cmd==KEY_LEFT){
if(left_board>){
move(BOTTOM,right_board);
addch(' ');
right_board--;
left_board--;
move(BOTTOM,left_board);
addch('-');
move(BOTTOM,RIGHT);
refresh();
}
}
//挡板右移
else if(cmd==KEY_RIGHT){
if(right_board<RIGHT){
move(BOTTOM,left_board);
addch(' ');
right_board++;
left_board++;
move(BOTTOM,right_board);
addch('-');
move(BOTTOM,RIGHT);
refresh();
}
}
//给球加速
else if(cmd==KEY_UP){
delay/=;
set_ticker(delay);
}
//给球减速
else if(cmd==KEY_DOWN){
delay*=;
set_ticker(delay);
} }
else{
//输掉球后的处理
int flag=;
char choice;
move(,);
addstr("Game Over!try again?(y/n):");
refresh();
choice=getch(); while(flag){
if(choice=='y'||choice=='Y'||choice=='n'||choice=='N')
flag=;
else choice=getch();
}
if(choice=='y'||choice=='Y'){ /*游戏重新开始*/
delay=; /*恢复球的初始速度*/
init();
continue;
}
else if(choice=='n'||choice=='N'){ /*结束游戏*/
break;
}
}
}
}
//设置定时器
int set_ticker(int n_msecs){
struct itimerval new_timeset;
long n_sec,n_usecs;
n_sec=n_msecs/;
n_usecs=(n_msecs%)*1000L;
new_timeset.it_interval.tv_sec=n_sec;
new_timeset.it_interval.tv_usec=n_usecs;
new_timeset.it_value.tv_sec=n_sec;
new_timeset.it_value.tv_usec=n_usecs;
return setitimer(ITIMER_REAL,&new_timeset,NULL);
}
pong game using ncurses的更多相关文章
- HDU 2492 Ping pong (树状数组)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2492 Ping pong Problem Description N(3<=N<=2000 ...
- POJ3928Ping pong[树状数组 仿逆序对]
Ping pong Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 3109 Accepted: 1148 Descrip ...
- UVALive 4329 Ping pong
Ping pong Time Limit: 3000MS Memory Limit: Unknown 64bit IO Fo ...
- ncurses库的一些函数
为了实现一个简单的聊天程序,如果使用普通的输入输出函数,会很凌乱.so,便想着能不能用下 ncurses这个字符图形库 总结一下,就是这样. 使用ncurses时,先需要初始化窗口,程序结束时,主动调 ...
- Mini projects #4 ---- Pong
课程全名:An Introduction to Interactive Programming in Python,来自 Rice University 授课教授:Joe Warren, Scott ...
- POJ 3928 Ping pong(树状数组)
Ping pong Time Limit: 1000MS ...
- (转) Deep Reinforcement Learning: Pong from Pixels
Andrej Karpathy blog About Hacker's guide to Neural Networks Deep Reinforcement Learning: Pong from ...
- Linux编译内核提示'make menuconfig' requires the ncurses libraries错误
原来使用的ubuntu 11.10系统由于误操作,导致系统崩溃,重新安装了ubuntu 11.10: 在编译内核的时候,提示如下错误: dingq@wd-u1110:~/hwsvn/2sw/1prj_ ...
- LA4329 Ping pong(树状数组与组合原理)
N (3N20000)ping pong players live along a west-east street(consider the street as a line segment). E ...
随机推荐
- 杂谈:Windows操作系统的介绍与对Win8操作系统市场反响冷淡原因的分析
Windows操作系统,毫无疑问是操作系统市场上的霸主,也正因为Windows操作系统的诞生让电脑的操作性能变得更加平民化,深的用户的喜爱.至今身边的人也是选择windows操作系统的居多,这篇文章也 ...
- [Python 从入门到放弃] 3. BIF(内建函数)
BIF (built-in functions) Python中提供了70多个内建函数,具备大量的现成功能. BIF不需要专门导入,可以直接使用,拿来就用 1.print() # 在屏幕上打印输出 如 ...
- Linux 上SSH 服务的配置和管理
0.前期准备:清空防火墙,关闭SELinux. [root@localhost ~]# iptables -F #清空防火墙 [root@localhost ~]# /etc/init.d/iptab ...
- 揭开Future的神秘面纱——任务执行
前言 此文承接之前的博文 解开Future的神秘面纱之取消任务 补充一些任务执行的一些细节,并从全局介绍程序的运行情况. 系列目录 揭开Future的神秘面纱——任务取消 揭开Future的神秘面纱— ...
- ExecutorService接口概要
ExecutorService接口继承于Executor接口,主要提供以下额外功能: 管理终结 产生Future对象,用于跟踪一个或多个任务的进度. ExecutorService可以被shut ...
- JVM内存初学 堆、栈、方法区
转自: http://www.open-open.com/lib/view/open1432200119489.html 这两天看了一下深入浅出JVM这本书,推荐给高级的java程序员去看,对你了解J ...
- Spring 环境与profile(二)——Properties with Spring
1. 简述 Spring profile用例,分3个场景(Test, Dev, Prod)相对Spring 环境与profile(一)——超简用例多了根据具体的profile获取对应的Properti ...
- spring配置文件引入properties文件:<context:property-placeholder>标签使用总结
一.问题描述: 1.有些参数在某些阶段中是常量,比如: (1)在开发阶段我们连接数据库时的连接url.username.password.driverClass等 (2)分布式应用中client端访问 ...
- TensorFlow-实战Google深度学习框架 笔记(上)
TensorFlow TensorFlow 是一种采用数据流图(data flow graphs),用于数值计算的开源软件库.在 Tensorflow 中,所有不同的变量和运算都是储存在计算图,所以在 ...
- UVa 122 Trees on the level(链式二叉树的建立和层次遍历)
题目链接: https://cn.vjudge.net/problem/UVA-122 /* 问题 给出每个节点的权值和路线,输出该二叉树的层次遍历序列. 解题思路 根据输入构建链式二叉树,再用广度优 ...