OpenCores
URL https://opencores.org/ocsvn/connect-6/connect-6/trunk

Subversion Repositories connect-6

[/] [connect-6/] [trunk/] [CONNECTK/] [connectk-2.0/] [src/] [ai/] [mesh.c] - Rev 4

Compare with Previous | Blame | View Log

 
/*
 
connectk -- a program to play the connect-k family of games
Copyright (C) 2007 Jeff Deitch
 
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
 
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.
 
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 
*/
 
#include "config.h"
#include <glib.h>
#include "../shared.h"
 
int cur_player_weight = 50;
int opp_player_weight = 500;
int mesh_array[19][19];
 
int get_mesh_value(int x, int y) {
	if (x < 0 || y < 0 || x >= board_size || y >= board_size) {
		/* out of bounds */
		return 0;
	} else {
		return mesh_array[x][y];
	}
}
 
int stone_weight(int x, int y) {
	if (piece_at(board, x, y) == board->turn) {
		return cur_player_weight;
	} else if (piece_at(board, x, y) == other_player(board->turn)) {
		return opp_player_weight;
	} else {
		return 0;
	}
}
 
int surrounding_weight(int x, int y) {
 
	int weight = 0;
	int n = 4;  // number of tiles in each of the 8 radials that contribute to the weight.
 
	int dx, dy, ddx, ddy, i;
        int ddxs[] = {1, 0, 1,  1};
        int ddys[] = {0, 1, 1, -1};
 
	for (i = 0; i < 4; i++) {
 
		ddx = ddxs[i];
		ddy = ddys[i];
 
		for (dx = -(ddx * n), dy = -(ddy * n); dx <= n && dy <= n; dx += ddx, dy += ddy) {
			if (dx == 0 && dy == 0) {
				continue;
			} else {
				weight += get_mesh_value(x + dx, y + dy);
			}
		}
	}
 
	return weight / (8 * n);
}
 
AIMoves *ai_mesh(const Board *b)
/* imagine the board is elastic and that each stone has a weight.  The current players stones have
a different weight than the opposite players stones.  Placing a stone on the board creates a depression,
and the more stones in an area, the deeper the depression.  This ai chooses the lowest, unplayed tile for
the next move.  The idea is to create clumps of stones.  */
{
        AIMove move;
        AIMoves *moves = aimoves_new();
 
	int i, x, y;
	int iterations = 10;
	int max_weight = 0;
 
	/* set all values to 0 */
	for (x = 0; x < board_size; x++) {
		for (y = 0; y < board_size; y++) {
			mesh_array[x][y] = 0;
		}
	}
 
	/* iteratively find the depth of each tile */
	for (i = 0; i < iterations; i++) {
		for (x = 0; x < board_size; x++) {
			for (y = 0; y < board_size; y++) {
				mesh_array[x][y] = stone_weight(x, y) + surrounding_weight(x, y);
			}
		}
	}
 
	/* find the max weight (i.e. the lowest spot on the board) */
	for (x = 0; x < board_size; x++) {
		for (y = 0; y < board_size; y++) {
			if (piece_at(b, x, y) == PIECE_NONE) {
				move.weight = mesh_array[x][y];
                                if (move.weight > max_weight)
					max_weight = move.weight;
				move.x = x;
				move.y = y;
                                aimoves_add(moves, &move);
			}
		}
	}
 
	/* if the board is empty, play in the middle */
	if (max_weight == 0) {
		move.x = board_size / 2;
		move.y = board_size / 2;
		move.weight = 1;
		aimoves_add(moves, &move);
	}
 
        moves->utility = max_weight;
 
	/* return the array */
	return moves;
}
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.