tris-in-stile-c/tris.c
Rnhmjoj f5630398bf Risolti problemi di compatibilità e risolti alcuni bug
*Abilmente esclusa int mgetchar() dalla compilazione su Windows;
*void Sleep(int milliseconds) di windows.h aggiunta;
*Risolto un bug che provocava un ciclo infinito dopo che un char rimane
nella stdin.
*Aggiornato il README.md;
2012-11-21 01:07:30 +01:00

309 lines
5.7 KiB
C
Executable File

#include "tris.h"
/*Funzioni*/
/*Legge dalla tastiera senza usare il buffer*/
UX int mgetchar(){
UX char buf=0;
UX struct termios old={0};
UX if(tcgetattr(0,&old)<0)
UX perror("tcsetattr()");
UX old.c_lflag&=~ICANON;
UX old.c_lflag&=~ECHO;
UX old.c_cc[VMIN]=1;
UX old.c_cc[VTIME]=0;
UX if(tcsetattr(0,TCSANOW,&old)<0)
UX perror("tcsetattr ICANON");
UX if(read(0,&buf,1)<0)
UX perror("read()");
UX old.c_lflag|=ICANON;
UX old.c_lflag|=ECHO;
UX if(tcsetattr(0,TCSADRAIN,&old)<0)
UX perror("tcsetattr ~ICANON");
UX return (buf);
UX}
/*Stampa la tabella del tris*/
void stampa(){
int i,k;
printf("\n\n\n");
for(i=0;i<M;i++) {
printf(RESET" ");
for(k=0;k<M;k++) {
if(tabella[i][k]==1)
printf(GIALLO" O ");
else {
if(tabella[i][k]==2)
printf(ROSSO" X ");
else
printf(BIANCO" . ");
}
}
printf("\n\n");
}
printf("\n");
}
/*Cancella la tabella del tris*/
void vuota() {
for(i=0;i<M;i++) {
for(k=0;k<M;k++)
tabella[i][k]=0;
}
}
/*Legge la mossa del giocatore*/
void leggimossa(){
int tasto=0;
fflush(stdin);
/*Se è attivo il tastierino*/
if(tastierino==1){
do{
tasto=0;
fflush(stdin);
/*2 giocatori*/
if(scelta==2)
printf(GIALLO" Fai la tua mossa:\n"RESET BIANCO);
else
printf(GIALLO" Gioca il giocatore %d.\n Fai la tua mossa:\n"RESET BIANCO,giocatore);
if(os)
tasto=mgetchar();
else
do{
tasto=getch();
}while(kbhit());
switch(tasto){
case 49:{
i=2;
k=0;
break;
}
case 50:{
i=2;
k=1;
break;
}
case 51:{
i=2;
k=2;
break;
}
case 52:{
i=1;
k=0;
break;
}
case 53:{
i=1;
k=1;
break;
}
case 54:{
i=1;
k=2;
break;
}
case 55:{
i=0;
k=0;
break;
}
case 56:{
i=0;
k=1;
break;
}
case 57:{
i=0;
k=2;
break;
}
}
if(tabella[i][k]==1||tabella[i][k]==2||i<0||k<0||i>=M||k>=M){
system(clear);
stampa();
printf(GRASSETTO ROSSO"\n Mossa non valida,riprova.\n\n"RESET BIANCO);
}
}while(tabella[i][k]==1||tabella[i][k]==2||i<0||k<0||i>=M||k>=M);
}
/*Mossa con coordinate*/
else{
do{
/*Giocatore singolo*/
if(scelta==2)
printf(GIALLO" Tocca a te.\n Inserisci coordinate della mossa"ROSSO"[Riga,colonna]:\n" RESET BIANCO);
/*2 giocatori*/
else
printf(GIALLO" Gioca il giocatore %d.\n Inserisci coordinate della mossa"ROSSO"[Riga,colonna]:\n" RESET BIANCO,giocatore);
scanf("%d,%d",&i,&k);
if(tabella[i][k]==1||tabella[i][k]==2||i<0||k<0||i>=M||k>=M){
system(clear);
stampa();
printf(GRASSETTO ROSSO"\n\n Coordinate non valide,riprova.\n\n\n"RESET BIANCO);
}
}while(tabella[i][k]==1||tabella[i][k]==2||i<0||k<0||i>=M||k>=M);
system(clear);
}
}
/*Controlla se il giocatore ha vinto*/
int controlla(int giocatore){
/*Diagonali*/
for(i=0;i<M;i++){
for(k=M;k>=0;k--)
if(tabella[i][k]==giocatore&&tabella[i+1][k-1]==giocatore&&tabella[i+1][k-2]==giocatore)
return 1;
for(k=0;k<M;k++)
if(tabella[i][k]==giocatore&&tabella[i+1][k+1]==giocatore&&tabella[i+2][k+2]==giocatore)
return 1;
}
/*Righe e colonne*/
for(i=0;i<M;i++){
for(k=0;k<M;k++)
if(tabella[i][k]==giocatore&&tabella[i][k+1]==giocatore&&tabella[i][k+2]==giocatore)
return 1;
for(k=0;k<M;k++)
if(tabella[k][i]==giocatore&&tabella[k+1][i]==giocatore&&tabella[k+2][i]==giocatore)
return 1;
}
return 0;
}
/*Minimax*/
int minimax(int giocatore, int profondita){
if(controlla(2))
return INT_MAX;
if(mossa==M*M)
return 0;
int res,tmp;
if(giocatore==1){
res=1;
for(i=0;i<M;i++){
for(k=0;k<M;k++){
if(!tabella[i][k]){
tabella[i][k]=1;
if(controlla(1)&&!profondita==20){
tabella[i][k]=0;
return INT_MIN;
}
else if((tmp=minimax(2,profondita-1))<res)
res=tmp;
else
res-=2;
tabella[i][k]=0;
}
}
}
}
else{
res=-1;
for(i=0;i<M;i++){
for(k=0;k<M;k++){
if(!tabella[i][k]){
tabella[i][k]=2;
if(controlla(2))
res+=2;
else if((tmp=minimax(1,profondita-1))>res)
res=tmp;
tabella[i][k]=0;
}
}
}
}
return res;
}
/*Mossa del computer a difficoltà facile*/
void facile() {
int a,b;
do{
randomize;
a=random(M);
b=random(M);
}while(tabella[a][b]==1||tabella[a][b]==2);
tabella[a][b]=2;
}
/*Mossa del computer a difficoltà media*/
void medio(){
int max=INT_MIN,mi=1,mk=1,t;
for(i=0;i<M;i++){
for(k=0;k<M;k++){
if(!tabella[i][k]){
tabella[i][k]=2;
t=minimax(1, 10);
if(t>max){
max=t;
mi=i;
mk=k;
}
tabella[i][k]=0;
}
}
}
stampa();
tabella[mi][mk]=2;
}
/*Mossa del computer a difficoltà difficile*/
void difficile(){
int max=INT_MIN,mi=1,mk=1,t;
for(i=0;i<M;i++){
for(k=0;k<M;k++){
if(!tabella[i][k]){
tabella[i][k]=2;
t=minimax(1,20);
if(t>max){
max=t;
mi=i;
mk=k;
}
tabella[i][k]=0;
}
}
}
tabella[mi][mk]=2;
}
/* Blocca l'esecuzione finchè non viene premuto invio*/
void pausa() {
do
getchar();
while ((c= getchar()) != '\n' && c!= EOF);
}
/*Replica di Sleep()*/
void aspetta(int t){
time_t Ti,Tf;
time(&Ti);
do{
time(&Tf);
}while(difftime(Tf,Ti)<t);
}
/*Caricamento stile Ajax*/
void spinner(int tempo) {
char spinner[] = "/-\\|";
printf(GIALLO"\n ");
for (int i=0; i<(tempo*10);i++){
putchar(spinner[i%4]);
fflush(stdout);
sleep(2000);
putchar('\b');
}
printf(RESET);
putchar(' ');
}
/*Beep*/
void beep(int t){
for(i=0;i<t*100;i++)
putchar('\a');
}