⚫️
고스트바둑왕(오목왕) 만들기#2
2025.01.12
1. 바둑판 화점 찍기
// 화점 위치 (오목판에서 별 모양의 점 위치)
const starPoints = [
[3, 3],
[3, 9],
[3, 15],
[9, 3],
[9, 9],
[9, 15],
[15, 3],
[15, 9],
[15, 15],
];
// 오목판 생성 함수: 시각적인 바둑판을 생성
function createCheckerboard() {
for (let i = 0; i < 18; i++) {
// 18행(테두리를 포함한 크기)을 생성
const tr = document.createElement('tr'); // 새로운 행(tr) 생성
table.appendChild(tr); // 테이블에 행 추가
for (let j = 0; j < 18; j++) {
// 각 행에 대해 18개의 셀(td) 생성
const td = document.createElement('td'); // 셀(td) 생성
td.setAttribute('class', 'square'); // CSS 스타일을 위한 클래스 지정
// 화점 위치에 별 모양의 점 추가
if (starPoints.some(([x, y]) => x === i && y === j)) {
const point = document.createElement('div');
point.setAttribute('class', 'star-point'); // 화점 스타일 지정
td.appendChild(point);
}
tr.appendChild(td); // 생성된 셀을 현재 행에 추가
}
}
}
.star-point {
position: absolute;
top: -3px;
left: -3px;
width: 6px;
height: 6px;
background-color: black;
border-radius: 50%;
}
생각해보니 바둑판을 그릴 때 바둑판의 화점을 생각못해서 화점을 찍는 JavaScript, CSS 구현
2. 3-3, 4-4 규칙 구현
// 바둑돌 배치 함수: 사용자가 돌을 놓을 때 호출
function handleStonePlacement(e) {
const target = e.target; // 클릭된 요소 가져오기
if (target.tagName !== 'TD') return; // 클릭된 요소가 셀이 아닐 경우 무시
const [row, col] = target.id.split('-').map(Number); // 클릭된 셀의 행(row)과 열(col) 추출
// 이미 돌이 놓인 경우 경고 메시지 표시 후 중단
if (boardState[row][col] !== null) {
alert('이미 돌이 놓인 자리입니다!');
return;
}
// 3-3 또는 4-4 규칙 위반 확인 (흑돌인 경우만 적용)
if (currentPlayer === 'black' && isForbiddenMove(row, col)) {
alert('이 동작은 3-3 또는 4-4 규칙을 위반합니다. 다른 위치를 선택하세요.');
return;
}
// 돌을 놓고 논리 상태 업데이트
boardState[row][col] = currentPlayer; // 현재 플레이어의 돌 배치
target.classList.add(`${currentPlayer}-stone`); // 시각적으로 돌 표시
// 승리 조건 확인
if (checkWin(row, col, currentPlayer)) {
setTimeout(() => {
alert(`${currentPlayer === 'black' ? '흑' : '백'}이 승리했습니다!`);
initializeBoard(); // 승리 후 보드 초기화
}, 100);
return;
}
// 다음 플레이어로 전환
currentPlayer = currentPlayer === 'black' ? 'white' : 'black';
}
// 금지된 동작(3-3 또는 4-4) 확인 함수
function isForbiddenMove(row, col) {
boardState[row][col] = 'black'; // 임시로 돌 배치
const isThreeThree = checkDoubleThree(row, col); // 3-3 규칙 위반 확인
const isFourFour = checkDoubleFour(row, col); // 4-4 규칙 위반 확인
boardState[row][col] = null; // 임시 배치된 돌 제거
return isThreeThree || isFourFour; // 둘 중 하나라도 위반 시 true 반환
}
// 3-3 규칙 확인 함수
function checkDoubleThree(row, col) {
let threeDirections = []; // 열린 삼(Open Three)의 방향 저장
const directions = [
[0, 1],
[1, 0],
[1, 1],
[1, -1], // 가로, 세로, 대각선 ↘, 대각선 ↙
];
for (const [dx, dy] of directions) {
if (isOpenThree(row, col, dx, dy)) {
threeDirections.push([dx, dy]); // 열린 삼 발견 시 방향 추가
}
if (threeDirections.length >= 2) return true; // 두 방향 이상 발견 시 3-3 규칙 위반
}
return false; // 3-3 규칙 위반 아님
}
// 열린 삼(Open Three) 확인 함수
function isOpenThree(row, col, dx, dy) {
let count = 1; // 현재 돌 포함 돌 개수
let openEnds = 0; // 열린 끝부분 개수
// 앞쪽 방향 탐색
let x = row + dx;
let y = col + dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x += dx;
y += dy;
}
if (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === null) {
openEnds++; // 열린 끝 발견 시 증가
}
// 뒤쪽 방향 탐색
x = row - dx;
y = col - dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x -= dx;
y -= dy;
}
if (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === null) {
openEnds++; // 열린 끝 발견 시 증가
}
// 정확히 3개의 돌이 있고 양쪽 끝이 열려 있는지 확인
return count === 3 && openEnds === 2;
}
// 4-4 규칙 확인 함수
function checkDoubleFour(row, col) {
let fourCount = 0; // 정확한 사(4)의 개수
const directions = [
[0, 1],
[1, 0],
[1, 1],
[1, -1],
];
for (const [dx, dy] of directions) {
if (isExactFour(row, col, dx, dy)) {
fourCount++;
}
if (fourCount >= 2) return true; // 두 방향 이상 발견 시 4-4 규칙 위반
}
return false; // 4-4 규칙 위반 아님
}
// 정확한 사(4) 확인 함수
function isExactFour(row, col, dx, dy) {
let count = 1; // 현재 돌 포함 돌 개수
// 앞쪽 방향 탐색
let x = row + dx;
let y = col + dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x += dx;
y += dy;
}
// 뒤쪽 방향 탐색
x = row - dx;
y = col - dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x -= dx;
y -= dy;
}
return count === 4; // 정확히 4개의 돌이 있는지 반환
}
3-3 일 때
4-4 일 때
3. 전체코드
// DOM 요소
const table = document.getElementById('table'); // 시각적인 오목판을 나타내는 HTML 요소
const go = document.getElementById('go'); // 바둑돌이 놓이는 위치를 표시할 HTML 요소
// 게임 상태
let currentPlayer = 'black'; // 현재 플레이어 ('black' 또는 'white')를 저장
let boardState; // 17x17 크기의 배열로 게임 보드의 상태를 저장 (null, 'black', 'white')
// 화점 위치 (오목판에서 별 모양의 점 위치)
const starPoints = [
[3, 3],
[3, 9],
[3, 15],
[9, 3],
[9, 9],
[9, 15],
[15, 3],
[15, 9],
[15, 15],
];
// 게임 초기화 함수: 보드를 초기화하고 이벤트 리스너를 설정
initializeBoard();
// 오목판 생성 함수: 시각적인 바둑판을 생성
function createCheckerboard() {
for (let i = 0; i < 18; i++) {
// 18행(테두리를 포함한 크기)을 생성
const tr = document.createElement('tr'); // 새로운 행(tr) 생성
table.appendChild(tr); // 테이블에 행 추가
for (let j = 0; j < 18; j++) {
// 각 행에 대해 18개의 셀(td) 생성
const td = document.createElement('td'); // 셀(td) 생성
td.setAttribute('class', 'square'); // CSS 스타일을 위한 클래스 지정
// 화점 위치에 별 모양의 점 추가
if (starPoints.some(([x, y]) => x === i && y === j)) {
const point = document.createElement('div');
point.setAttribute('class', 'star-point'); // 화점 스타일 지정
td.appendChild(point);
}
tr.appendChild(td); // 생성된 셀을 현재 행에 추가
}
}
}
// 바둑돌이 놓일 판 생성 함수: 논리적인 바둑판을 생성
function createPlacementBoard() {
for (let i = 0; i < 17; i++) {
// 17행(실제 게임 영역)을 생성
const tr = document.createElement('tr'); // 새로운 행 생성
go.appendChild(tr); // 바둑돌 배치판에 행 추가
for (let j = 0; j < 17; j++) {
// 각 행에 대해 17개의 셀(td) 생성
const td = document.createElement('td'); // 셀 생성
td.setAttribute('id', `${i}-${j}`); // 셀 ID를 "행-열" 형식으로 지정
tr.appendChild(td); // 생성된 셀을 현재 행에 추가
}
}
}
// 보드 초기화 함수: 보드와 게임 상태를 초기화
function initializeBoard() {
table.innerHTML = ''; // 기존 테이블 내용 삭제
go.innerHTML = ''; // 기존 바둑돌 배치판 내용 삭제
createCheckerboard(); // 시각적인 오목판 생성
createPlacementBoard(); // 논리적인 바둑판 생성
resetBoardState(); // 게임 상태 초기화
addEventListeners(); // 클릭 이벤트 리스너 추가
}
// 게임 상태 초기화 함수
function resetBoardState() {
// 17x17 배열 생성, 모든 셀을 null로 초기화 (돌이 놓이지 않음)
boardState = Array(17)
.fill(null)
.map(() => Array(17).fill(null));
// 바둑돌 배치판의 모든 시각적 요소 초기화
const cells = go.querySelectorAll('td');
cells.forEach((cell) => {
cell.className = ''; // CSS 클래스 제거
cell.style.backgroundColor = ''; // 인라인 스타일 초기화
});
currentPlayer = 'black'; // 현재 플레이어를 흑돌로 초기화
}
// 이벤트 리스너 추가 함수: 바둑돌 배치판에 클릭 이벤트 추가
function addEventListeners() {
go.addEventListener('click', handleStonePlacement); // 클릭 시 handleStonePlacement 실행
}
// 바둑돌 배치 함수: 사용자가 돌을 놓을 때 호출
function handleStonePlacement(e) {
const target = e.target; // 클릭된 요소 가져오기
if (target.tagName !== 'TD') return; // 클릭된 요소가 셀이 아닐 경우 무시
const [row, col] = target.id.split('-').map(Number); // 클릭된 셀의 행(row)과 열(col) 추출
// 이미 돌이 놓인 경우 경고 메시지 표시 후 중단
if (boardState[row][col] !== null) {
alert('이미 돌이 놓인 자리입니다!');
return;
}
// 3-3 또는 4-4 규칙 위반 확인 (흑돌인 경우만 적용)
if (currentPlayer === 'black' && isForbiddenMove(row, col)) {
alert('이 동작은 3-3 또는 4-4 규칙을 위반합니다. 다른 위치를 선택하세요.');
return;
}
// 돌을 놓고 논리 상태 업데이트
boardState[row][col] = currentPlayer; // 현재 플레이어의 돌 배치
target.classList.add(`${currentPlayer}-stone`); // 시각적으로 돌 표시
// 승리 조건 확인
if (checkWin(row, col, currentPlayer)) {
setTimeout(() => {
alert(`${currentPlayer === 'black' ? '흑' : '백'}이 승리했습니다!`);
initializeBoard(); // 승리 후 보드 초기화
}, 100);
return;
}
// 다음 플레이어로 전환
currentPlayer = currentPlayer === 'black' ? 'white' : 'black';
}
// 금지된 동작(3-3 또는 4-4) 확인 함수
function isForbiddenMove(row, col) {
boardState[row][col] = 'black'; // 임시로 돌 배치
const isThreeThree = checkDoubleThree(row, col); // 3-3 규칙 위반 확인
const isFourFour = checkDoubleFour(row, col); // 4-4 규칙 위반 확인
boardState[row][col] = null; // 임시 배치된 돌 제거
return isThreeThree || isFourFour; // 둘 중 하나라도 위반 시 true 반환
}
// 3-3 규칙 확인 함수
function checkDoubleThree(row, col) {
let threeDirections = []; // 열린 삼(Open Three)의 방향 저장
const directions = [
[0, 1],
[1, 0],
[1, 1],
[1, -1], // 가로, 세로, 대각선 ↘, 대각선 ↙
];
for (const [dx, dy] of directions) {
if (isOpenThree(row, col, dx, dy)) {
threeDirections.push([dx, dy]); // 열린 삼 발견 시 방향 추가
}
if (threeDirections.length >= 2) return true; // 두 방향 이상 발견 시 3-3 규칙 위반
}
return false; // 3-3 규칙 위반 아님
}
// 열린 삼(Open Three) 확인 함수
function isOpenThree(row, col, dx, dy) {
let count = 1; // 현재 돌 포함 돌 개수
let openEnds = 0; // 열린 끝부분 개수
// 앞쪽 방향 탐색
let x = row + dx;
let y = col + dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x += dx;
y += dy;
}
if (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === null) {
openEnds++; // 열린 끝 발견 시 증가
}
// 뒤쪽 방향 탐색
x = row - dx;
y = col - dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x -= dx;
y -= dy;
}
if (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === null) {
openEnds++; // 열린 끝 발견 시 증가
}
// 정확히 3개의 돌이 있고 양쪽 끝이 열려 있는지 확인
return count === 3 && openEnds === 2;
}
// 4-4 규칙 확인 함수
function checkDoubleFour(row, col) {
let fourCount = 0; // 정확한 사(4)의 개수
const directions = [
[0, 1],
[1, 0],
[1, 1],
[1, -1],
];
for (const [dx, dy] of directions) {
if (isExactFour(row, col, dx, dy)) {
fourCount++;
}
if (fourCount >= 2) return true; // 두 방향 이상 발견 시 4-4 규칙 위반
}
return false; // 4-4 규칙 위반 아님
}
// 정확한 사(4) 확인 함수
function isExactFour(row, col, dx, dy) {
let count = 1; // 현재 돌 포함 돌 개수
// 앞쪽 방향 탐색
let x = row + dx;
let y = col + dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x += dx;
y += dy;
}
// 뒤쪽 방향 탐색
x = row - dx;
y = col - dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === 'black') {
count++;
x -= dx;
y -= dy;
}
return count === 4; // 정확히 4개의 돌이 있는지 반환
}
// 승리 조건 확인 함수
function checkWin(row, col, player) {
const directions = [
[0, 1],
[1, 0],
[1, 1],
[1, -1],
];
for (const [dx, dy] of directions) {
let count = 1;
count += countStones(row, col, dx, dy, player); // 앞쪽 방향 돌 개수
count += countStones(row, col, -dx, -dy, player); // 뒤쪽 방향 돌 개수
if (count >= 5) return true; // 5개 이상의 돌이 연속되면 승리
}
return false; // 승리 조건 미충족
}
// 특정 방향으로 연속된 돌 개수 계산 함수
function countStones(row, col, dx, dy, player) {
let count = 0;
let x = row + dx;
let y = col + dy;
while (x >= 0 && x < 17 && y >= 0 && y < 17 && boardState[x][y] === player) {
count++;
x += dx;
y += dy;
}
return count; // 해당 방향으로 연속된 돌 개수 반환
}
다음에 해야 할 것
흑돌일 때 33 규칙 넣기버그 찾아서 고치기- 리팩토링하기
- (흑돌, 백돌 각각의) 스탑워치 놓기
- WebRCT 이용해서 실시간으로 채팅이나, 게임방 만들기