EasyX图形库实现 链表的星星的不断移动

EasyX图形库实现 链表的星星的不断移动

代码:

#include
#include
#include

using namespace std;

//屏幕宽高宏定义
#define SCREEN_WIDTH 640
#define SCREEN_HEIGHT 480

//星星的各个属性宏定义
#define MAX_STAR 100 //:数量
#define MAX_RADIUS 3
#define MAX_STEP 5

//星星的移动状态
enum class _STATUS {
STOP,
UP,
DOWN,
LEFT,
RIGTH,

ALL_STATUS

};

//星星结构体内部的链表 存的是其他星星结构体的地址
typedef struct _LinkNode {
struct _LinkNode *prev; //上一个星星里node的地址
struct _LinkNode *next; //下一个星星里node的地址
}_LinkNode;

//星星结构体
typedef struct {
int x;
int y;
unsigned int radius;
_STATUS status;
int step;
int color;
_LinkNode node;
}_STAR;

//初始化星星首节点
bool starInit(_STAR *&L_star) {
L_star = new _STAR;
if (!L_star) {
return false;
}

//不初始化星星属性
//这里是首节点
L_star->node.next = NULL;
L_star->node.prev = NULL;

return true;

}

//初始化星星里的属性
void initStar(_STAR *&p) {
if (!p) {
cout << “没有该星星的空间!” << endl;
return;
}

p->x = rand() % SCREEN_WIDTH;					//0 - 639
p->y = rand() % SCREEN_HEIGHT;					//0 - 479
p->radius = rand() % MAX_RADIUS + 1;			// 1 - 3
p->status = _STATUS::UP;						//状态向上
p->step = rand() % MAX_STEP + 1;				//步长 1- 5
int rgb = 255 * (p->step / MAX_STEP);			
p->color = RGB(rgb, rgb, rgb);					//颜色

}

//添加链表(前插法)
bool linkInsert_front(_LinkNode *L, _LinkNode *node) {
//L:首个星星里node地址
//node:添加星星里node地址
if (!L || !node) {
return false;
}

if (L->next) {
	L->next->prev = node;
}

node->next = L->next;
node->prev = L;
L->next = node;


return true;

}

//后插法
bool linkInsert_back(_LinkNode *L, _LinkNode *node) {
if (!L || !node) {
return false;
}

_LinkNode *last = L;

//找到最后一个的地址
while (last->next) {
	last = last->next;
}

node->next = NULL;
node->prev = last;
last->next = node;


return true;

}

//显示星空
void starDisplay(_STAR *&L_star) {
if (!L_star || !L_star->node.next) {
return;
}

//指向第一个链表节点
_LinkNode *p = L_star->node.next;

//查看距离的_STAR头的位置
int offset = offsetof(_STAR, node);

while (p) {
	_STAR *tmp = (_STAR *)((size_t)p - offset);

	//绘制图像
	setfillcolor(tmp->color);
	solidcircle(tmp->x, tmp->y, tmp->radius);

	//指向下一个星星结构体里的node地址
	p = p->next;
}

}

//星星移动
void moveStar(_STAR *&L_star) {
if (!L_star || !L_star->node.next) {
return;
}

//指向第一个星星
_LinkNode *p = L_star->node.next;

//查看距离的_STAR头的位置
int offset = offsetof(_STAR, node);

while (p) {
	_STAR *tmp = (_STAR *)((size_t)p - offset);

	//擦除原来位置的星星
	setfillcolor(BLACK);
	solidcircle(tmp->x, tmp->y, tmp->radius);

	//判断移动状态,通过步长移动坐标
	//默认都是UP
	switch (tmp->status) {
	case _STATUS::UP:
		tmp->y -= tmp->step;
		if (tmp->y <= 0)  tmp->y = SCREEN_HEIGHT;
		break;
	case _STATUS::DOWN:
		tmp->y += tmp->step;
		if (tmp->y >= SCREEN_HEIGHT)  tmp->y = 0;
		break;
	case _STATUS::LEFT:
		tmp->x -= tmp->step;
		if (tmp->x <= 0)  tmp->x = SCREEN_WIDTH;
		break;
	case _STATUS::RIGTH:
		tmp->x += tmp->step;
		if (tmp->x >= SCREEN_WIDTH)  tmp->x = 0;
		break;
	case _STATUS::ALL_STATUS:
	default:
		break;
	}

	//绘制新位置图像
	setfillcolor(tmp->color);
	solidcircle(tmp->x, tmp->y, tmp->radius);

	//遍历到下一个星星的node地址
	p = p->next;
}

}

int main(void) {
//首星星的结构体指针
_STAR *L_star = NULL;

//存其他的添加的星星结构体
_STAR *s_star = NULL;

//初始化星星首节点
starInit(L_star);

//添加MAX_STAR个星星
for (int i = 0; i < MAX_STAR; i++) {
	s_star = new _STAR;

	//初始化分配空间里的星星各个属性
	initStar(s_star);

	//分配星星里node的链表,形成双向链表[后插法]
	linkInsert_back(&(L_star->node), &(s_star->node));
}

//显示图像
initgraph(SCREEN_WIDTH, SCREEN_HEIGHT);
HWND hwnd = GetHWnd();
SetWindowText(hwnd, _T("星空图"));

//显示星空
starDisplay(L_star);

//星星不断移动
while (1) {
	//移动完所有星星
	moveStar(L_star);

	//就休息50ms
	Sleep(50);
}


system("pause");

//关闭图像
closegraph();

return 0;

}

你可能感兴趣的:(EasyX图形库实现 链表的星星的不断移动)