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的更多相关文章

  1. HDU 2492 Ping pong (树状数组)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2492 Ping pong Problem Description N(3<=N<=2000 ...

  2. POJ3928Ping pong[树状数组 仿逆序对]

    Ping pong Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3109   Accepted: 1148 Descrip ...

  3. UVALive 4329 Ping pong

                                      Ping pong Time Limit: 3000MS   Memory Limit: Unknown   64bit IO Fo ...

  4. ncurses库的一些函数

    为了实现一个简单的聊天程序,如果使用普通的输入输出函数,会很凌乱.so,便想着能不能用下 ncurses这个字符图形库 总结一下,就是这样. 使用ncurses时,先需要初始化窗口,程序结束时,主动调 ...

  5. Mini projects #4 ---- Pong

    课程全名:An Introduction to Interactive Programming in Python,来自 Rice University 授课教授:Joe Warren, Scott ...

  6. POJ 3928 Ping pong(树状数组)

                                                                          Ping pong Time Limit: 1000MS   ...

  7. (转) Deep Reinforcement Learning: Pong from Pixels

    Andrej Karpathy blog About Hacker's guide to Neural Networks Deep Reinforcement Learning: Pong from ...

  8. Linux编译内核提示'make menuconfig' requires the ncurses libraries错误

    原来使用的ubuntu 11.10系统由于误操作,导致系统崩溃,重新安装了ubuntu 11.10: 在编译内核的时候,提示如下错误: dingq@wd-u1110:~/hwsvn/2sw/1prj_ ...

  9. LA4329 Ping pong(树状数组与组合原理)

    N (3N20000)ping pong players live along a west-east street(consider the street as a line segment). E ...

随机推荐

  1. PHP多进程系列笔记(四)

    本节主要讲解Posix常用函数和进程池的概念,也会涉及到守护进程的知识.本节难度较低. Posix常用函数 posix_kill 向指定pid进程发送信号.成功时返回 TRUE , 或者在失败时返回 ...

  2. 解读Secondary NameNode的功能

    1.概述 最近有朋友问我Secondary NameNode的作用,是不是NameNode的备份?是不是为了防止NameNode的单点问题?确实,刚接触Hadoop,从字面上看,很容易会把Second ...

  3. SVN Hooks的介绍及使用

    阅读此篇文章你可以: 对SVN Hooks有一定的了解 获取两个最常用的SVN Hooks案例 SVN hooks介绍 Hooks 钩子,主要实现的功能就是在特定事件发生之前或者之后自动执行事先定义好 ...

  4. Java设计模式学习记录-建造者模式

    前言 今天周末,有小雨,正好也不用出门了,那就在家学习吧,经过了两周的面试,拿到了几个offer,但是都不是自己很想去的那种,要么就是几个人的初创小公司,要么就是开发企业内部系统的这种传统开发,感觉这 ...

  5. 使用Amazon AWS SNS 发送 SMS 消息 .net

    1.浏览aws 开发人员指南 https://docs.aws.amazon.com/zh_cn/sns/latest/dg/sms_publish-to-phone.html 2.安装 aws sm ...

  6. 浅谈JSONP (vue-jsonp组件 XXXtoken:报错处理)

    由于同源策略的存在,特别是前后端两个项目存在的情况下,客户端访问服务端必然存在跨域的情况,而使用jsonp,则不存在这个问题. 主要是因为jsonp是在页面中插入一段js代码,而请求返回的也是一段js ...

  7. 在JAVA中封装JSONUtil工具类及使用

    在JAVA中用json-lib-2.3-jdk15.jar包中提供了JSONObject和JSONArray基类,用于JSON的序列化和反序列化的操作.但是我们更习惯将其进一步封装,达到更好的重用. ...

  8. Android8自定义广播无法收到消息

    在Android 8之前,如果要发送自定义的静态广播,做好其他配置后,只需实例化一个Intent对象intent,然后将其作为参数传入sendBroadcast()方法中即可,例如 Intent in ...

  9. 小程序 波浪进度球 wave

    直接上代码: //index.js //获取应用实例 const app = getApp() var wave = function (ctx, oRange){ var tid; //oRange ...

  10. JS touch

    一个月没写博客了,感觉空唠唠的,有多好想写的,今天全都给补上吧,记录最近这个月的收获 https://blog.csdn.net/sinat_19327991/article/details/7382 ...