mirror of
git://git.savannah.nongnu.org/eliot.git
synced 2025-01-15 03:44:04 +01:00
5441007db1
Until now there is no user-visible change, but it should make it much easier to add custom layouts.
300 lines
10 KiB
C++
300 lines
10 KiB
C++
/*****************************************************************************
|
|
* Eliot
|
|
* Copyright (C) 1999-2012 Antoine Fraboulet & Olivier Teulière
|
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
|
*
|
|
* 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 St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*****************************************************************************/
|
|
|
|
#include <cwctype> // For towupper
|
|
|
|
#include "board_search.h"
|
|
#include "dic.h"
|
|
#include "game_params.h"
|
|
#include "board.h"
|
|
#include "tile.h"
|
|
#include "rack.h"
|
|
#include "round.h"
|
|
#include "results.h"
|
|
|
|
|
|
BoardSearch::BoardSearch(const Dictionary &iDic,
|
|
const GameParams &iParams,
|
|
const Matrix<Tile> &iTilesMx,
|
|
const Matrix<Cross> &iCrossMx,
|
|
const Matrix<int> &iPointsMx,
|
|
const Matrix<bool> &iJokerMx,
|
|
bool isFirstTurn)
|
|
: m_dic(iDic), m_params(iParams), m_tilesMx(iTilesMx), m_crossMx(iCrossMx),
|
|
m_pointsMx(iPointsMx), m_jokerMx(iJokerMx), m_firstTurn(isFirstTurn)
|
|
{
|
|
}
|
|
|
|
|
|
void BoardSearch::search(Rack &iRack, Results &oResults, Coord::Direction iDir) const
|
|
{
|
|
// Handle the first turn specifically
|
|
if (m_firstTurn)
|
|
{
|
|
const int row = 8, col = 8;
|
|
Round tmpRound;
|
|
tmpRound.accessCoord().setRow(row);
|
|
tmpRound.accessCoord().setCol(col);
|
|
tmpRound.accessCoord().setDir(Coord::HORIZONTAL);
|
|
leftPart(iRack, tmpRound, oResults, m_dic.getRoot(),
|
|
row, col, std::min(iRack.getNbTiles(), (unsigned)col) - 1);
|
|
return;
|
|
}
|
|
|
|
|
|
vector<Tile> rackTiles;
|
|
iRack.getTiles(rackTiles);
|
|
vector<Tile>::const_iterator it;
|
|
|
|
for (int row = 1; row <= BOARD_DIM; row++)
|
|
{
|
|
Round partialWord;
|
|
partialWord.accessCoord().setDir(iDir);
|
|
partialWord.accessCoord().setRow(row);
|
|
int lastanchor = 0;
|
|
for (int col = 1; col <= BOARD_DIM; col++)
|
|
{
|
|
if (m_tilesMx[row][col].isEmpty() &&
|
|
(!m_tilesMx[row][col - 1].isEmpty() ||
|
|
!m_tilesMx[row][col + 1].isEmpty() ||
|
|
!m_tilesMx[row - 1][col].isEmpty() ||
|
|
!m_tilesMx[row + 1][col].isEmpty()))
|
|
{
|
|
#ifdef DONT_USE_SEARCH_OPTIMIZATION
|
|
if (!m_tilesMx[row][col - 1].isEmpty())
|
|
{
|
|
partialWord.accessCoord().setCol(lastanchor + 1);
|
|
extendRight(iRack, partialWord, oResults,
|
|
m_dic.getRoot(), row, lastanchor + 1, col);
|
|
}
|
|
else
|
|
{
|
|
partialWord.accessCoord().setCol(col);
|
|
leftPart(iRack, partialWord, oResults,
|
|
m_dic.getRoot(), row, col, col - lastanchor - 1);
|
|
}
|
|
lastanchor = col;
|
|
#else
|
|
// Optimization compared to the original Appel & Jacobson
|
|
// algorithm: skip leftPart if none of the tiles of the rack
|
|
// matches the cross mask for the current anchor
|
|
bool match = false;
|
|
for (it = rackTiles.begin(); it != rackTiles.end(); it++)
|
|
{
|
|
if (m_crossMx[row][col].check(*it))
|
|
{
|
|
match = true;
|
|
break;
|
|
}
|
|
}
|
|
if (match)
|
|
{
|
|
if (!m_tilesMx[row][col - 1].isEmpty())
|
|
{
|
|
partialWord.accessCoord().setCol(lastanchor + 1);
|
|
extendRight(iRack, partialWord, oResults,
|
|
m_dic.getRoot(), row, lastanchor + 1, col);
|
|
}
|
|
else
|
|
{
|
|
partialWord.accessCoord().setCol(col);
|
|
leftPart(iRack, partialWord, oResults,
|
|
m_dic.getRoot(), row, col, col - lastanchor - 1);
|
|
}
|
|
}
|
|
lastanchor = col;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void BoardSearch::leftPart(Rack &iRack, Round &ioPartialWord,
|
|
Results &oResults, int n, int iRow,
|
|
int iAnchor, int iLimit) const
|
|
{
|
|
extendRight(iRack, ioPartialWord, oResults, n, iRow, iAnchor, iAnchor);
|
|
|
|
if (iLimit > 0)
|
|
{
|
|
bool hasJokerInRack = iRack.in(Tile::Joker());
|
|
for (unsigned int succ = m_dic.getSucc(n); succ; succ = m_dic.getNext(succ))
|
|
{
|
|
const Tile &l = Tile(m_dic.getChar(succ));
|
|
if (iRack.in(l))
|
|
{
|
|
iRack.remove(l);
|
|
ioPartialWord.addRightFromRack(l, false);
|
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
|
leftPart(iRack, ioPartialWord, oResults,
|
|
succ, iRow, iAnchor, iLimit - 1);
|
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
|
ioPartialWord.removeRight();
|
|
iRack.add(l);
|
|
}
|
|
if (hasJokerInRack)
|
|
{
|
|
iRack.remove(Tile::Joker());
|
|
ioPartialWord.addRightFromRack(l, true);
|
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
|
leftPart(iRack, ioPartialWord, oResults,
|
|
succ, iRow, iAnchor, iLimit - 1);
|
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
|
ioPartialWord.removeRight();
|
|
iRack.add(Tile::Joker());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void BoardSearch::extendRight(Rack &iRack, Round &ioPartialWord,
|
|
Results &oResults, unsigned int iNode,
|
|
int iRow, int iCol, int iAnchor) const
|
|
{
|
|
if (m_tilesMx[iRow][iCol].isEmpty())
|
|
{
|
|
if (m_dic.isEndOfWord(iNode) && iCol > iAnchor)
|
|
{
|
|
evalMove(oResults, ioPartialWord);
|
|
}
|
|
|
|
// Optimization: avoid entering the for loop if no tile can match
|
|
if (m_crossMx[iRow][iCol].isNone())
|
|
return;
|
|
|
|
bool hasJokerInRack = iRack.in(Tile::Joker());
|
|
for (unsigned int succ = m_dic.getSucc(iNode); succ; succ = m_dic.getNext(succ))
|
|
{
|
|
const Tile &l = Tile(m_dic.getChar(succ));
|
|
if (m_crossMx[iRow][iCol].check(l))
|
|
{
|
|
if (iRack.in(l))
|
|
{
|
|
iRack.remove(l);
|
|
ioPartialWord.addRightFromRack(l, false);
|
|
extendRight(iRack, ioPartialWord, oResults,
|
|
succ, iRow, iCol + 1, iAnchor);
|
|
ioPartialWord.removeRight();
|
|
iRack.add(l);
|
|
}
|
|
if (hasJokerInRack)
|
|
{
|
|
iRack.remove(Tile::Joker());
|
|
ioPartialWord.addRightFromRack(l, true);
|
|
extendRight(iRack, ioPartialWord, oResults,
|
|
succ, iRow, iCol + 1, iAnchor);
|
|
ioPartialWord.removeRight();
|
|
iRack.add(Tile::Joker());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
const Tile &l = m_tilesMx[iRow][iCol];
|
|
wint_t upperChar = towupper(l.toChar());
|
|
for (unsigned int succ = m_dic.getSucc(iNode); succ ; succ = m_dic.getNext(succ))
|
|
{
|
|
if ((wint_t)m_dic.getChar(succ) == upperChar)
|
|
{
|
|
ioPartialWord.addRightFromBoard(l);
|
|
extendRight(iRack, ioPartialWord,
|
|
oResults, succ, iRow, iCol + 1, iAnchor);
|
|
ioPartialWord.removeRight();
|
|
// The letter will be present only once in the dictionary,
|
|
// so we can stop looping
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Computes the score of a word, coordinates may be changed to reflect
|
|
* the real direction of the word
|
|
*/
|
|
void BoardSearch::evalMove(Results &oResults, Round &iWord) const
|
|
{
|
|
int fromrack = 0;
|
|
int pts = 0;
|
|
int ptscross = 0;
|
|
int wordmul = 1;
|
|
|
|
unsigned int len = iWord.getWordLen();
|
|
|
|
int row = iWord.getCoord().getRow();
|
|
int col = iWord.getCoord().getCol();
|
|
|
|
const BoardLayout & boardLayout = m_params.getBoardLayout();
|
|
for (unsigned int i = 0; i < len; i++)
|
|
{
|
|
if (!m_tilesMx[row][col+i].isEmpty())
|
|
{
|
|
if (!m_jokerMx[row][col+i])
|
|
pts += iWord.getTile(i).getPoints();
|
|
}
|
|
else
|
|
{
|
|
int l;
|
|
if (!iWord.isJoker(i))
|
|
l = iWord.getTile(i).getPoints() *
|
|
boardLayout.getLetterMultiplier(row, col + i);
|
|
else
|
|
l = 0;
|
|
pts += l;
|
|
int wm = boardLayout.getWordMultiplier(row, col + i);
|
|
wordmul *= wm;
|
|
|
|
int t = m_pointsMx[row][col+i];
|
|
if (t >= 0)
|
|
ptscross += (t + l) * wm;
|
|
fromrack++;
|
|
}
|
|
}
|
|
|
|
// Ignore words using too many letters from the rack
|
|
if (fromrack > m_params.getLettersToPlay())
|
|
return;
|
|
|
|
pts = ptscross + pts * wordmul;
|
|
if (fromrack == m_params.getLettersToPlay())
|
|
{
|
|
pts += m_params.getBonusPoints();
|
|
iWord.setBonus(true);
|
|
}
|
|
iWord.setPoints(pts);
|
|
|
|
if (iWord.getCoord().getDir() == Coord::VERTICAL)
|
|
{
|
|
// Exchange the coordinates temporarily
|
|
iWord.accessCoord().swap();
|
|
}
|
|
oResults.add(iWord);
|
|
if (iWord.getCoord().getDir() == Coord::VERTICAL)
|
|
{
|
|
// Restore the coordinates
|
|
iWord.accessCoord().swap();
|
|
}
|
|
}
|
|
|