#include <bits/stdc++.h>
#include <windows.h>
#include <conio.h>
#include <unistd.h>
#define clear() system("cls");
using namespace std;
int len = 20;
void gotoxy(int x,int y){
    COORD pos;
    pos.X=x;
    pos.Y=y;
    HANDLE handle=GetStdHandle(STD_OUTPUT_HANDLE);
    SetConsoleCursorPosition(handle,pos);
}
struct node
{
int x, y;
};
int num[100][100];
int x, y;
node head, food;
bool Food = false;
list<node> snake;
char ch = 0;
int score = 0;
static int peek_character = -1;
void changeNum();
void drawBoard();
void move();
int scanKeyboard();
void init_snake();
void fullFood();
bool getFood();
void longer();
bool gameOver();
void intro();
int main()
{
init_snake();
x = 0, y = 0;
intro();
getch();
fullFood();
drawBoard();
ch = getch();
while(ch != 'n') {
usleep(10000);
int nx = x, ny = y;
if (ch == 'w') nx = -1,ny = 0;
if (ch == 's') nx = 1, ny = 0;
if (ch == 'a') nx = 0, ny = -1;
if (ch == 'd') nx = 0, ny = 1;
if (ch == 'n') return 0;
if (nx != -x || ny != -y)
x = nx, y = ny;
if (getFood())
{
score++;
longer();
fullFood();
}
else move();

if (gameOver())
{
drawBoard();
cout << "GameOver" << endl;
return 0;
}
drawBoard();
if (kbhit()) ch = getch();
}
return 0;

}

bool gameOver()
{
head = snake.front();
if (head.x  <= 0 || head.x > len)
return true;
if (head.y <= 0 || head.y > len)
return true;
if (num[head.x][head.y] == 1)
return true;
return false;
}
void fullFood()
{
int x[10000] = {}, y[10000] = {}, cur = 0;
for (int i = 1; i <= len; i++)
{
for (int j = 1; j <= len; j++)
{
if (num[i][j] == 0)
{
cur++;
x[cur] = i;
y[cur] = j;
}
}
}
if (cur > 0)
{
srand(time(0));
int cc = rand() % cur + 1;
food.x = x[cc];
food.y = y[cc];
}
}

void changeNum()
{
    if(snake.empty()) return;
    memset(num, 0, sizeof(num));
    for(list<node>::iterator it = snake.begin(); it != snake.end(); ++it)
    {
        num[(*it).x][(*it).y] = 1;
    }
    num[food.x][food.y] = 3;
    num[(*snake.begin()).x][(*snake.begin()).y] = 2;
}

void drawBoard()
{
HANDLE hConsole = GetStdHandle(STD_OUTPUT_HANDLE);
changeNum();
gotoxy(0,0);
cout << "    ╔";
for (int i = 1; i <= len / 2; i++) cout << "════";
cout << "═╗\n";
// 输出中间部分
for (int i = 1; i <= len; i++) // 行
{
cout << "    ║";
for (int j = 1; j <= len; j++) // 列
{
if (num[i][j] == 1)
{
	SetConsoleTextAttribute(hConsole, FOREGROUND_GREEN);
	cout << "■";
	SetConsoleTextAttribute(hConsole, FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE);
}
else if (num[i][j] == 2)
{
	SetConsoleTextAttribute(hConsole, FOREGROUND_RED);
	cout << "■";
	SetConsoleTextAttribute(hConsole, FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE);
}
else if (num[i][j] == 3)
{
	SetConsoleTextAttribute(hConsole, FOREGROUND_RED | FOREGROUND_GREEN);
	cout << "■";
	SetConsoleTextAttribute(hConsole, FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE);
}
else
{
	cout << "  ";
}
}
cout << " ║" << endl;
}
cout << "    ╚";
for (int i = 1; i <= len / 2; i++) cout << "════";
cout << "═╝\n";
cout << "score:" << score << endl;

}

void move()
{
clear();
if (x == 0 && y == 0)
{
drawBoard();
return;
}
node next = snake.front();
next.y += y;
next.x += x;
snake.pop_back();
snake.push_front(next);
}

void init_snake()
{
head.x = 1, head.y = 1;
snake.push_back(head);
for (int i = 2; i <= 4; i++)
{
node a;
a.x = 1, a.y = i;
snake.push_back(a);
}
changeNum();
}

bool getFood()
{
head = snake.front();
if(num[head.x + x][head.y + y] == 3) return true;
else return false;
}

void longer()
{
node temp = snake.front();
temp.x += x;
temp.y += y;
snake.push_front(temp);
}

void intro()
{
cout << "欢迎来到贪吃蛇小游戏" << endl;
cout << "w,a,s,d控制" << endl;
cout << "按下 n 健提前结束游戏" << endl;
cout << "在移动过程中碰到了自己或者墙壁则GAMEOVER" << endl;
cout << "按下任意键开始游戏" << endl;
}