/* $Id: gomo1.c,v 1.1 2002/03/07 15:13:48 in2 Exp $ */ #include #include #include #include "config.h" #include "pttstruct.h" #include "proto.h" #define QCAST int (*)(const void *, const void *) #define BBLANK (0) /* 空白 */ #define BBLACK (1) /* 黑子, 先手 */ #define BWHITE (2) /* 白子, 後手 */ #ifndef BRDSIZ #define BRDSIZ (15) /* 棋盤單邊大小 */ #endif extern char ku[BRDSIZ][BRDSIZ]; /* pattern and advance map */ extern unsigned char *pat, *adv; static int intrevcmp(const void *a, const void *b) { return (*(int *)b - *(int *)a); } /* x,y: 0..BRDSIZ-1 ; color: CBLACK,CWHITE ; dx,dy: -1,0,+1 */ static int gomo_getindex(int x, int y, int color, int dx, int dy) { int i, k, n; for(n = -1, i = 0, k = 1; i < 5; i++, k <<= 1) { x += dx; y += dy; if((x < 0) || (x >= BRDSIZ) || ( y < 0) || ( y >= BRDSIZ)) { n += k; break; } else if(ku[x][y] != BBLANK) { n += k; if(ku[x][y] != color) break; } } if(i >= 5) n += k; return n; } int chkwin(int style, int limit) { if(style == 0x0c) return 1 /* style */; else if(limit == 0) { if(style == 0x0b) return 1 /* style */; return 0; } if((style < 0x0c) && (style > 0x07)) return -1 /* -style */; return 0; } /* x,y: 0..BRDSIZ-1 ; color: CBLACK,CWHITE ; limit:1,0 ; dx,dy: 0,1 */ static int dirchk(int x, int y, int color, int limit, int dx, int dy) { int le, ri, loc, style; le = gomo_getindex(x, y, color, -dx, -dy); ri = gomo_getindex(x, y, color, dx, dy); loc = (le > ri) ? (((le * (le + 1)) >> 1) + ri) : (((ri * (ri + 1)) >> 1) + le); style = pat[loc]; if(limit == 0) return (style & 0x0f); style >>= 4; if((style == 3) || (style == 2)) { int i, n, tmp, nx, ny; n = adv[loc >> 1]; ((loc & 1) == 0) ? (n >>= 4) : (n &= 0x0f); ku[x][y] = color; for(i = 0; i < 2; i++) { if((tmp = (i == 0) ? (-(n >> 2)):(n & 3)) != 0) { nx = x + (le > ri ? 1 : -1) * tmp * dx; ny = y + (le > ri ? 1 : -1) * tmp * dy; if((dirchk(nx, ny, color, 0, dx, dy) == 0x06) && (chkwin(getstyle(nx, ny, color, limit), limit) >= 0)) break; } } if(i >= 2) style = 0; ku[x][y] = BBLANK; } return style; } /* 例外=F 錯誤=E 有子=D 連五=C 連六=B 雙四=A 四四=9 三三=8 */ /* 四三=7 活四=6 斷四=5 死四=4 活三=3 斷三=2 保留=1 無效=0 */ /* x,y: 0..BRDSIZ-1 ; color: CBLACK,CWHITE ; limit: 1,0 */ int getstyle(int x, int y, int color, int limit) { int i, j, dir[4], style; if((x < 0) || (x >= BRDSIZ) || ( y < 0) || (y >= BRDSIZ)) return 0x0f; if(ku[x][y] != BBLANK) return 0x0d; for(i = 0; i < 4; i++) dir[i] = dirchk(x, y, color, limit, i ? (i>>1) : -1, i ? (i&1) : 1); qsort(dir, 4, sizeof(int), (QCAST)intrevcmp); if((style = dir[0]) >= 2) { for(i = 1, j = 6 + (limit ? 1 : 0); i < 4; i++) { if((style > j) || (dir[i] < 2)) break; if(dir[i] > 3) style = 9; else if((style < 7) && (style > 3)) style = 7; else style = 8; } } return style; }