mirror of
git://git.savannah.nongnu.org/eliot.git
synced 2025-01-13 20:03:23 +01:00
Merged the "cppdic" branch back into HEAD.
There are too many change to list properly, here is an overview of the main changes: - the dictionary is now in C++ - the dictionary has a new format, where it is possible to specify the letters, their points, their frequency, ... It is backwards compatible. - Eliot now supports non-ASCII characters everywhere - i18n of the compdic, listdic, regexpmain binaries - i18n of the wxWidgets interface (now in english by default)
This commit is contained in:
parent
d86bd2af4a
commit
e7a8d01a8d
143 changed files with 12900 additions and 8013 deletions
58
INSTALL
58
INSTALL
|
@ -1,6 +1,7 @@
|
||||||
|
|
||||||
Installation sous Linux (Un*x) (bien/facile)
|
Installation sous Linux (Un*x) (bien/facile)
|
||||||
------------------------------
|
------------------------------
|
||||||
* Pour installer à partir de l'archive CVS :
|
* Pour installer à partir de l'archive CVS :
|
||||||
|
|
||||||
./bootstrap
|
./bootstrap
|
||||||
|
|
||||||
|
@ -8,16 +9,16 @@ Installation sous Linux (Un*x) (bien/facile)
|
||||||
make
|
make
|
||||||
make install
|
make install
|
||||||
|
|
||||||
* Pour installer à partir de l'archive tar.gz
|
* Pour installer à partir de l'archive tar.gz
|
||||||
|
|
||||||
./configure
|
./configure
|
||||||
make
|
make
|
||||||
make install
|
make install
|
||||||
|
|
||||||
|
|
||||||
il existe en fait 3 versions d'eliot, une en mode texte, une avec une
|
Il existe en fait 3 versions d'eliot, une en mode texte, une avec une
|
||||||
interface curses et une avec wxwidgets. Les modes peuvent être
|
interface curses et une avec wxwidgets. Les modes peuvent être
|
||||||
sélectionnés à l'aide de la commande configure lors de la compilation
|
sélectionnés à l'aide de la commande configure lors de la compilation
|
||||||
du programme.
|
du programme.
|
||||||
|
|
||||||
./configure --enable-text --enable-ncurses --enable-wxwidgets
|
./configure --enable-text --enable-ncurses --enable-wxwidgets
|
||||||
|
@ -27,39 +28,56 @@ du programme.
|
||||||
Installation pour Windows (moins bien/facile)
|
Installation pour Windows (moins bien/facile)
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
Il y a 2 principales façons de procéder :
|
Il y a 2 principales façons de procéder :
|
||||||
* directement depuis Windows, en utilisant Cygwin (http://www.cygwin.com/).
|
* directement depuis Windows, en utilisant Cygwin (http://www.cygwin.com/).
|
||||||
* depuis GNU/Linux, en utilisant le cross-compilateur Mingw32.
|
* depuis GNU/Linux, en utilisant le cross-compilateur Mingw32.
|
||||||
|
|
||||||
|
|
||||||
Dans les 2 cas, les étapes sont les mêmes :
|
Dans les 2 cas, les étapes sont les mêmes :
|
||||||
* installation de l'environnement de compilation (cette étape n'est pas
|
* installation de l'environnement de compilation (cette étape n'est pas
|
||||||
décrite ici, car elle ne rentre pas dans le cadre de ce document)
|
décrite ici, car elle ne rentre pas dans le cadre de ce document)
|
||||||
|
|
||||||
* compilation et installation de wxWindows (http://www.wxwidgets.org/),
|
* compilation et installation des dépendances (même remarque):
|
||||||
version 2.4.2 ou ultérieure (même remarque)
|
|
||||||
|
- wxWidgets (http://www.wxwidgets.org/), version 2.4.2 ou ultérieure, avec
|
||||||
|
support de l'unicode
|
||||||
|
|
||||||
|
- libiconv (http://www.gnu.org/software/libiconv/), de préférence compilée
|
||||||
|
en mode statique (--disable-shared --enable-static)
|
||||||
|
|
||||||
|
- boost (http://www.boost.org/). Eliot n'utilise pas de librairie de Boost
|
||||||
|
(uniquement des headers), donc il n'y a pas vraiment besoin de compiler
|
||||||
|
|
||||||
* compilation d'Eliot :
|
* compilation d'Eliot :
|
||||||
|
|
||||||
- si vous utilisez l'archive CVS, il faut générer le script 'configure'
|
- si vous utilisez l'archive CVS, il faut générer le script 'configure'
|
||||||
(aussi bien sous Cygwin que sous GNU/Linux) :
|
(aussi bien sous Cygwin que sous GNU/Linux) :
|
||||||
./bootstrap
|
./bootstrap
|
||||||
|
|
||||||
|
- à cause d'un bug de gettext, il faut appliquer un patch aux fichiers installés
|
||||||
|
dans intl/ :
|
||||||
|
- télécharger le patch ici (lien en haut à gauche) :
|
||||||
|
http://www.koders.com/noncode/fid46DF595700FEB564B6EF45BFF55067F95DCF0420.aspx
|
||||||
|
- exécuter la commande suivante :
|
||||||
|
patch -p2 < gettext-win32.patch
|
||||||
|
|
||||||
- avec Cygwin, configurer avec la ligne de commande suivante :
|
- avec Cygwin, configurer avec la ligne de commande suivante :
|
||||||
|
CPPFLAGS=-I/path/to/installs/include LDFLAGS=-L/path/to/installs/lib \
|
||||||
CC="gcc -mno-cygwin" CXX="g++ -mno-cygwin" \
|
CC="gcc -mno-cygwin" CXX="g++ -mno-cygwin" \
|
||||||
./configure --with-wx-config=/path/to/wx-config
|
./configure --enable-wxwidgets --with-wx-config=/path/to/wx-config \
|
||||||
en prenant soin d'indiquer le chemin correct vers le fichier
|
--with-boost=/path/to/boost/installs
|
||||||
'wx-config' de l'installation de wxWindows.
|
en prenant soin de remplacer les différents chemins par les bonnes valeurs.
|
||||||
Ensuite, un simple 'make' suffit pour terminer la compilation,
|
Ensuite, un simple 'make' suffit pour terminer la compilation,
|
||||||
éventuellement suivi de 'make install'.
|
éventuellement suivi de 'make install'.
|
||||||
|
|
||||||
- pour la cross-compilation depuis GNU/Linux, configurer avec la ligne
|
- pour la cross-compilation depuis GNU/Linux, configurer avec la ligne
|
||||||
de commande suivante :
|
de commande suivante :
|
||||||
|
CPPFLAGS=-I/path/to/installs/include LDFLAGS=-L/path/to/installs/lib \
|
||||||
CC=i586-mingw32msvc-gcc CXX=i586-mingw32msvc-g++ \
|
CC=i586-mingw32msvc-gcc CXX=i586-mingw32msvc-g++ \
|
||||||
./configure --host=i586-mingw32msvc --build=i386-linux \
|
./configure --host=i586-mingw32msvc --build=i386-linux \
|
||||||
--with-wx-config=/path/to/wx-config
|
--enable-wxwidgets --with-wx-config=/path/to/wx-config \
|
||||||
en prenant soin d'indiquer le chemin correct vers le fichier
|
--with-boost=/path/to/installs
|
||||||
'wx-config' de l'installation de wxWindows.
|
en prenant soin de remplacer les différents chemins par les bonnes valeurs.
|
||||||
Ensuite, un simple 'make' suffit pour terminer la compilation,
|
Ensuite, un simple 'make' suffit pour terminer la compilation,
|
||||||
éventuellement suivi de 'make install'.
|
éventuellement suivi de 'make install'.
|
||||||
|
|
||||||
|
|
30
TODO
30
TODO
|
@ -1,33 +1,19 @@
|
||||||
|
|
||||||
* ====================
|
|
||||||
* TODO Current version
|
|
||||||
* ====================
|
|
||||||
|
|
||||||
- Correct game save/load functions : Advanced format
|
|
||||||
file saving for freegames and duplicate need a serious
|
|
||||||
rewrite. We need to specifie a file format that can handle
|
|
||||||
all the information contained in a multiplayer game.
|
|
||||||
|
|
||||||
- Add "joker" type games in wxwin version of Eliot, Freegame
|
|
||||||
and Duplicate will follow
|
|
||||||
|
|
||||||
- full French i18n interface and error messages for wxwin.
|
|
||||||
|
|
||||||
|
|
||||||
* ==================
|
* ==================
|
||||||
* Next Eliot version
|
* Next Eliot version
|
||||||
* ==================
|
* ==================
|
||||||
|
|
||||||
- new dictionnary format that includes tiles
|
- Improve error handling (use exceptions more)
|
||||||
- number
|
- Correct game save/load functions: Advanced format
|
||||||
- points
|
file saving for freegames and duplicate need a serious
|
||||||
- printable equivalent
|
rewrite. We need to specify a file format that can handle
|
||||||
- other languages support using the new dictionary
|
all the information contained in a multiplayer game.
|
||||||
- new wxWidgets interface
|
- rack shuffling
|
||||||
|
- new wxWidgets or QT interface
|
||||||
- support of the different modes
|
- support of the different modes
|
||||||
- ability to choose the number and type of the players
|
- ability to choose the number and type of the players
|
||||||
- ability to display the history and score of all the players
|
- ability to display the history and score of all the players
|
||||||
-- partly done : history is now a separate class
|
-- partly done: history is now a separate class
|
||||||
- detection of blocked positions?
|
- detection of blocked positions?
|
||||||
- getopt support for all the interfaces (only ncurses is done)
|
- getopt support for all the interfaces (only ncurses is done)
|
||||||
|
|
||||||
|
|
82
configure.in
82
configure.in
|
@ -7,6 +7,8 @@ AC_INIT(eliot, 1.5)
|
||||||
AC_CONFIG_SRCDIR(wxwin/main.cc)
|
AC_CONFIG_SRCDIR(wxwin/main.cc)
|
||||||
AM_INIT_AUTOMAKE
|
AM_INIT_AUTOMAKE
|
||||||
AM_CONFIG_HEADER(config.h)
|
AM_CONFIG_HEADER(config.h)
|
||||||
|
AC_CANONICAL_HOST
|
||||||
|
AC_CANONICAL_BUILD
|
||||||
AM_OPTIONS_WXCONFIG
|
AM_OPTIONS_WXCONFIG
|
||||||
|
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
|
@ -14,6 +16,8 @@ dnl Checks for programs.
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
AC_PROG_INSTALL
|
AC_PROG_INSTALL
|
||||||
AC_PROG_CC
|
AC_PROG_CC
|
||||||
|
dnl Needed for gettext
|
||||||
|
AC_GNU_SOURCE
|
||||||
AC_PROG_CXX
|
AC_PROG_CXX
|
||||||
AC_PROG_MAKE_SET
|
AC_PROG_MAKE_SET
|
||||||
AC_PROG_RANLIB
|
AC_PROG_RANLIB
|
||||||
|
@ -23,6 +27,7 @@ if test "$YACC" = yacc ; then
|
||||||
AC_MSG_ERROR([Could not find the 'bison' program on your system])
|
AC_MSG_ERROR([Could not find the 'bison' program on your system])
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
dnl Better than AC_PROG_LEX
|
||||||
AM_PROG_LEX
|
AM_PROG_LEX
|
||||||
if test "$LEX" != "flex" ; then
|
if test "$LEX" != "flex" ; then
|
||||||
AC_MSG_ERROR([Could not find the 'flex' program on your system])
|
AC_MSG_ERROR([Could not find the 'flex' program on your system])
|
||||||
|
@ -56,15 +61,17 @@ fi
|
||||||
dnl Debug mode
|
dnl Debug mode
|
||||||
AC_ARG_ENABLE([debug],AC_HELP_STRING([--enable-debug],[debug mode (default disabled)]))
|
AC_ARG_ENABLE([debug],AC_HELP_STRING([--enable-debug],[debug mode (default disabled)]))
|
||||||
if test "${enable_debug}" = "yes"; then
|
if test "${enable_debug}" = "yes"; then
|
||||||
CFLAGS+=" -g -DDEBUG"
|
CPPFLAGS+=" -DDEBUG"
|
||||||
CXXFLAGS+=" -g -DDEBUG"
|
CFLAGS+=" -g"
|
||||||
|
CXXFLAGS+=" -g"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
dnl Profile mode
|
dnl Profile mode
|
||||||
AC_ARG_ENABLE([profile],AC_HELP_STRING([--enable-profile],[profile mode (default disabled)]))
|
AC_ARG_ENABLE([profile],AC_HELP_STRING([--enable-profile],[profile mode (default disabled)]))
|
||||||
if test "${enable_profile}" = "yes"; then
|
if test "${enable_profile}" = "yes"; then
|
||||||
CFLAGS+=" -pg -DPROFILE"
|
CPPFLAGS+=" -DPROFILE"
|
||||||
CXXFLAGS+=" -pg -DPROFILE"
|
CFLAGS+=" -pg"
|
||||||
|
CXXFLAGS+=" -pg"
|
||||||
LDFLAGS+=" -pg"
|
LDFLAGS+=" -pg"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
@ -77,6 +84,9 @@ dnl Checks for header files.
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
AC_HEADER_STDC
|
AC_HEADER_STDC
|
||||||
AC_CHECK_HEADERS(fcntl.h unistd.h sys/wait.h)
|
AC_CHECK_HEADERS(fcntl.h unistd.h sys/wait.h)
|
||||||
|
AC_CHECK_HEADERS(arpa/inet.h netinet/in.h)
|
||||||
|
AC_CHECK_HEADERS([readline/readline.h], [has_readline=1], [has_readline=0])
|
||||||
|
AM_CONDITIONAL(HAS_READLINE, test "$has_readline" = "1")
|
||||||
|
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
dnl Checks for typedefs, structures, and compiler characteristics.
|
dnl Checks for typedefs, structures, and compiler characteristics.
|
||||||
|
@ -85,22 +95,26 @@ AC_C_CONST
|
||||||
AC_TYPE_SIZE_T
|
AC_TYPE_SIZE_T
|
||||||
AC_C_BIGENDIAN
|
AC_C_BIGENDIAN
|
||||||
AC_C_INLINE
|
AC_C_INLINE
|
||||||
AC_CHECK_SIZEOF(char, 1)
|
dnl AC_CHECK_SIZEOF(char, 1)
|
||||||
AC_CHECK_SIZEOF(short, 2)
|
dnl AC_CHECK_SIZEOF(short, 2)
|
||||||
AC_CHECK_SIZEOF(int *, 4)
|
dnl AC_CHECK_SIZEOF(int *, 4)
|
||||||
AC_CHECK_SIZEOF(int, 4)
|
dnl AC_CHECK_SIZEOF(int, 4)
|
||||||
AC_CHECK_SIZEOF(long, 4)
|
dnl AC_CHECK_SIZEOF(long, 4)
|
||||||
AC_CHECK_SIZEOF(long long, 0)
|
dnl AC_CHECK_SIZEOF(long long, 0)
|
||||||
|
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
dnl Checks for library functions.
|
dnl Checks for library functions.
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
AC_FUNC_MEMCMP
|
AC_FUNC_MEMCMP
|
||||||
|
AC_CHECK_FUNCS([wcwidth])
|
||||||
|
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
dnl Checks for libraries.
|
dnl Checks for libraries.
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
|
|
||||||
|
dnl Check for the Boost libraries (in fact we only need the headers)
|
||||||
|
AX_BOOST_BASE([1.33.1])
|
||||||
|
|
||||||
dnl Check for wxWidgets
|
dnl Check for wxWidgets
|
||||||
AC_ARG_ENABLE([wxwidgets],AC_HELP_STRING([--enable-wxwidgets],[wxWidgets interface support (default disabled)]))
|
AC_ARG_ENABLE([wxwidgets],AC_HELP_STRING([--enable-wxwidgets],[wxWidgets interface support (default disabled)]))
|
||||||
if test "${enable_wxwidgets}" = "yes"
|
if test "${enable_wxwidgets}" = "yes"
|
||||||
|
@ -125,27 +139,53 @@ fi
|
||||||
AM_CONDITIONAL([BUILD_WXWIDGETS], [test "${wxWin}" = "1"])
|
AM_CONDITIONAL([BUILD_WXWIDGETS], [test "${wxWin}" = "1"])
|
||||||
|
|
||||||
dnl Check for ncurses
|
dnl Check for ncurses
|
||||||
AC_ARG_ENABLE([ncurses],AC_HELP_STRING([--enable-ncurses],[ncurses interface support (default disabled)]))
|
dnl We enable it if asked by the user, or if ncursesw is found
|
||||||
if test "${enable_ncurses}" = "yes"
|
AC_ARG_ENABLE([ncurses],AC_HELP_STRING([--enable-ncurses],
|
||||||
then
|
[ncurses interface support (default enabled if ncursesw found on your system)]))
|
||||||
AC_CHECK_HEADERS(ncurses.h, want_ncurses=1,
|
AC_CHECK_HEADERS(ncursesw/curses.h, [has_ncursesw=1], [has_ncursesw=0])
|
||||||
[AC_MSG_ERROR([Could not find the ncurses library on your system])])
|
if test "${enable_ncurses}" != "no" -a "${has_ncursesw}" = "1"; then
|
||||||
|
want_ncurses=1
|
||||||
|
else
|
||||||
|
want_ncurses=0
|
||||||
|
if test "${enable_ncurses}" = "yes"; then
|
||||||
|
AC_MSG_ERROR([Could not find the ncursesw library on your system])
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
AM_CONDITIONAL([BUILD_NCURSES], [test "${want_ncurses}" = "1"])
|
AM_CONDITIONAL([BUILD_NCURSES], [test "${want_ncurses}" = "1"])
|
||||||
|
|
||||||
dnl Enable/disable text version
|
dnl Enable/disable text version
|
||||||
AC_ARG_ENABLE([text],AC_HELP_STRING([--enable-text],[text interface support (default enabled)]))
|
AC_ARG_ENABLE([text],AC_HELP_STRING([--enable-text],[text interface support (default enabled)]))
|
||||||
if test "${enable_text}" != "no"
|
|
||||||
then
|
|
||||||
AC_CHECK_HEADERS(readline/readline.h, want_text=1,
|
|
||||||
[AC_MSG_ERROR([Could not find the readline library on your system])])
|
|
||||||
fi
|
|
||||||
AM_CONDITIONAL([BUILD_TEXT], [test "${enable_text}" != "no"])
|
AM_CONDITIONAL([BUILD_TEXT], [test "${enable_text}" != "no"])
|
||||||
|
|
||||||
dnl Internationalization macros
|
dnl Internationalization macros
|
||||||
AM_GNU_GETTEXT_VERSION(0.11.5)
|
AM_GNU_GETTEXT_VERSION(0.16.1)
|
||||||
AM_GNU_GETTEXT
|
AM_GNU_GETTEXT
|
||||||
|
|
||||||
|
dnl Iconv
|
||||||
|
dnl This test depends on AM_GNU_GETTEXT executed before
|
||||||
|
AS_IF([test "$am_cv_func_iconv" != "yes"],
|
||||||
|
[AC_MSG_ERROR([libiconv is needed for Eliot to work properly])])
|
||||||
|
|
||||||
|
|
||||||
|
dnl Information about who built eliot (useful for the dictionary)
|
||||||
|
AC_DEFINE_UNQUOTED(ELIOT_COMPILE_BY, "`whoami`", [user who ran configure])
|
||||||
|
AC_DEFINE_UNQUOTED(ELIOT_COMPILE_HOST, "`hostname`", [host which ran configure])
|
||||||
|
|
||||||
|
dnl Check the operating system
|
||||||
|
case "${host_os}" in
|
||||||
|
*mingw32* | *cygwin*)
|
||||||
|
SYS=mingw32
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
dnl nothing to do
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
if test "$SYS" = "mingw32"; then
|
||||||
|
# For ntohl, in particular
|
||||||
|
LIBS="${LIBS} -lws2_32"
|
||||||
|
fi
|
||||||
|
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
dnl Output
|
dnl Output
|
||||||
dnl --------------------------------------------------------------
|
dnl --------------------------------------------------------------
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
.deps
|
.deps
|
||||||
Makefile
|
Makefile
|
||||||
Makefile.in
|
Makefile.in
|
||||||
scanner.h
|
libdic_a-erl.cpp
|
||||||
er.c
|
libdic_a-erl.h
|
||||||
libdic_a-er.c
|
libdic_a-ery.cpp
|
||||||
libdic_a-er.h
|
libdic_a-ery.h
|
||||||
compdic
|
compdic
|
||||||
listdic
|
listdic
|
||||||
regexp
|
regexp
|
||||||
|
|
103
dic/Makefile.am
103
dic/Makefile.am
|
@ -1,6 +1,7 @@
|
||||||
# Eliot
|
# Eliot
|
||||||
# Copyright (C) 1999 Antoine Fraboulet
|
# Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
# antoine.fraboulet@free.fr
|
# 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
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -18,67 +19,71 @@
|
||||||
|
|
||||||
noinst_LIBRARIES = libdic.a
|
noinst_LIBRARIES = libdic.a
|
||||||
|
|
||||||
INCLUDES = -I$(top_srcdir)
|
localedir = $(datadir)/locale
|
||||||
|
AM_CPPFLAGS = -DLOCALEDIR=\"$(localedir)\"
|
||||||
|
|
||||||
libdic_a_CFLAGS=$(DEBUGFLAGS)
|
INCLUDES = -I$(top_srcdir) -I../intl -I$(top_srcdir)/intl $(INCICONV)
|
||||||
libdic_a_YFLAGS=
|
|
||||||
|
libdic_a_CFLAGS=
|
||||||
|
libdic_a_YFLAGS=-d
|
||||||
libdic_a_LFLAGS=
|
libdic_a_LFLAGS=
|
||||||
libdic_a_SOURCES= \
|
libdic_a_SOURCES = \
|
||||||
ery.y \
|
erl.lpp \
|
||||||
erl.l \
|
ery.ypp \
|
||||||
dic_internals.h \
|
dic_exception.cpp dic_exception.h \
|
||||||
dic_search.c dic_search.h \
|
header.cpp header.h \
|
||||||
dic.c dic.h \
|
dic_internals.h \
|
||||||
automaton.c automaton.h \
|
tile.cpp tile.h \
|
||||||
hashtable.h hashtable.c \
|
dic.cpp dic.h \
|
||||||
regexp.c regexp.h \
|
dic_search.cpp \
|
||||||
alist.h alist.c
|
encoding.cpp encoding.h \
|
||||||
|
automaton.cpp automaton.h \
|
||||||
|
regexp.cpp regexp.h
|
||||||
|
|
||||||
BUILT_SOURCES= \
|
BUILT_SOURCES= \
|
||||||
libdic_a-erl.c \
|
libdic_a-erl.cpp \
|
||||||
libdic_a-erl.h \
|
libdic_a-erl.h \
|
||||||
libdic_a-ery.c \
|
libdic_a-ery.cpp \
|
||||||
libdic_a-ery.h
|
libdic_a-ery.h
|
||||||
|
|
||||||
nodist_libdic_a_SOURCES= \
|
# This hook triggers on 'make dist' (and 'make distcheck')
|
||||||
libdic_a-erl.c \
|
# XXX: In fact, the recommended behaviour is:
|
||||||
libdic_a-erl.h \
|
# - list only libdic_a-ery.h in BUILT_SOURCES,
|
||||||
libdic_a-ery.c \
|
# - do not die with an error in configure.in if flex or bison is not found
|
||||||
|
# - do not have any dist-hook trigger
|
||||||
|
# The result is that the generated files are kept in the tarball generated with make dist,
|
||||||
|
# with still an error message for developers when the ypp or lpp file has been modified
|
||||||
|
# and bison or flex is not found.
|
||||||
|
# The problem is that, even though Automake is aware of the header generated by bison,
|
||||||
|
# it seems to have problems with the one generated by flex...
|
||||||
|
dist-hook:
|
||||||
|
-for file in $(BUILT_SOURCES) ; do rm -f $(distdir)/$$file ; done
|
||||||
|
|
||||||
|
CLEANFILES= \
|
||||||
|
libdic_a-erl.cpp \
|
||||||
|
libdic_a-erl.h \
|
||||||
|
libdic_a-ery.cpp \
|
||||||
libdic_a-ery.h
|
libdic_a-ery.h
|
||||||
|
|
||||||
CLEANFILES= \
|
|
||||||
libdic_a-erl.c \
|
|
||||||
libdic_a-erl.h \
|
|
||||||
libdic_a-ery.c \
|
|
||||||
libdic_a-ery.h
|
|
||||||
|
|
||||||
|
|
||||||
## automake workaround to generate .h file
|
|
||||||
libdic_a-erl.h: erl.l
|
|
||||||
${LEX} ${srcdir}/erl.l
|
|
||||||
|
|
||||||
#####################################
|
|
||||||
|
|
||||||
#####################################
|
#####################################
|
||||||
if BUILD_DICTOOLS
|
if BUILD_DICTOOLS
|
||||||
|
|
||||||
bin_PROGRAMS = \
|
bin_PROGRAMS = \
|
||||||
compdic \
|
compdic \
|
||||||
listdic \
|
listdic \
|
||||||
regexp
|
regexp
|
||||||
|
|
||||||
compdic_SOURCES= \
|
compdic_SOURCES=compdic.cpp \
|
||||||
dic_internals.h \
|
hashtable.h hashtable.cpp hashtable.i
|
||||||
hashtable.c hashtble.h \
|
compdic_CPPFLAGS=$(AM_CPPFLAGS) @BOOST_CPPFLAGS@
|
||||||
compdic.c
|
compdic_LDADD=libdic.a @LIBINTL@
|
||||||
|
|
||||||
listdic_SOURCES= \
|
listdic_SOURCES=listdic.cpp
|
||||||
dic_internals.h \
|
listdic_LDADD=libdic.a @LIBINTL@
|
||||||
dic.c dic.h \
|
|
||||||
listdic.c
|
|
||||||
|
|
||||||
#regexp_CFLAGS=-DDEBUG_RE
|
#regexp_CFLAGS=-DDEBUG_RE
|
||||||
regexp_SOURCES=regexpmain.c
|
regexp_SOURCES=regexpmain.cpp
|
||||||
regexp_LDADD=libdic.a
|
regexp_LDADD=libdic.a @LIBINTL@
|
||||||
|
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
201
dic/alist.c
201
dic/alist.c
|
@ -1,201 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file alist.c
|
|
||||||
* \brief List type used by automaton
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include "alist.h"
|
|
||||||
|
|
||||||
#define __UNUSED__ __attribute__((unused))
|
|
||||||
|
|
||||||
struct alist_elt_t {
|
|
||||||
void* info;
|
|
||||||
alist_elt next;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct alist_t {
|
|
||||||
int size;
|
|
||||||
void (*delete_function)(void*);
|
|
||||||
alist_elt start;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
void*
|
|
||||||
alist_elt_get_value(alist_elt e)
|
|
||||||
{
|
|
||||||
return e->info;
|
|
||||||
}
|
|
||||||
|
|
||||||
alist_elt
|
|
||||||
alist_elt_create(void* info)
|
|
||||||
{
|
|
||||||
alist_elt e;
|
|
||||||
e = (alist_elt)malloc(sizeof(struct alist_elt_t));
|
|
||||||
e->info = info;
|
|
||||||
e->next = NULL;
|
|
||||||
return e;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ************************************************** */
|
|
||||||
/* ************************************************** */
|
|
||||||
/* ************************************************** */
|
|
||||||
|
|
||||||
alist
|
|
||||||
alist_create()
|
|
||||||
{
|
|
||||||
alist l;
|
|
||||||
l = (alist)malloc(sizeof(struct alist_t));
|
|
||||||
l->size = 0;
|
|
||||||
l->start = NULL;
|
|
||||||
l->delete_function = NULL;
|
|
||||||
return l;
|
|
||||||
}
|
|
||||||
|
|
||||||
alist
|
|
||||||
alist_clone(alist l)
|
|
||||||
{
|
|
||||||
alist t;
|
|
||||||
alist_elt ptr;
|
|
||||||
t = alist_create();
|
|
||||||
for(ptr = alist_get_first(l); ptr ; ptr = alist_get_next(l,ptr))
|
|
||||||
{
|
|
||||||
alist_add(t,alist_elt_get_value(ptr));
|
|
||||||
}
|
|
||||||
return t;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
alist_set_delete (alist l, void (*f)(void*))
|
|
||||||
{
|
|
||||||
l->delete_function = f;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
alist_delete_rec(alist_elt e, void (*delete_function)(void*))
|
|
||||||
{
|
|
||||||
if (e != NULL)
|
|
||||||
{
|
|
||||||
alist_delete_rec(e->next, delete_function);
|
|
||||||
if (delete_function)
|
|
||||||
delete_function(e->info);
|
|
||||||
e->info = NULL;
|
|
||||||
free(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
alist_delete(alist l)
|
|
||||||
{
|
|
||||||
alist_delete_rec(l->start,l->delete_function);
|
|
||||||
free(l);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
alist_add(alist l, void* value)
|
|
||||||
{
|
|
||||||
alist_elt e;
|
|
||||||
e = alist_elt_create(value);
|
|
||||||
e->next = l->start;
|
|
||||||
l->start = e;
|
|
||||||
l->size ++;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
alist_is_in(alist l, void* e)
|
|
||||||
{
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(l); ptr; ptr = alist_get_next(l,ptr))
|
|
||||||
if (alist_elt_get_value(ptr) == e)
|
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
alist_equal(alist id1, alist id2)
|
|
||||||
{
|
|
||||||
alist_elt e1;
|
|
||||||
|
|
||||||
if (alist_get_size(id1) != alist_get_size(id2))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
for(e1 = alist_get_first(id1) ; e1 ; e1 = alist_get_next(id1,e1))
|
|
||||||
{
|
|
||||||
if (! alist_is_in(id2, alist_elt_get_value(e1)))
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
alist_insert(alist dst, alist src)
|
|
||||||
{
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(src); ptr ; ptr = alist_get_next(src,ptr))
|
|
||||||
{
|
|
||||||
void *e = alist_elt_get_value(ptr);
|
|
||||||
if (! alist_is_in(dst,e))
|
|
||||||
alist_add(dst,e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
alist_elt
|
|
||||||
alist_get_first(alist l)
|
|
||||||
{
|
|
||||||
return l->start;
|
|
||||||
}
|
|
||||||
|
|
||||||
alist_elt
|
|
||||||
alist_get_next(alist __UNUSED__ l, alist_elt e)
|
|
||||||
{
|
|
||||||
return e->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
void*
|
|
||||||
alist_pop_first_value(alist l)
|
|
||||||
{
|
|
||||||
void* p = NULL;
|
|
||||||
alist_elt e = l->start;
|
|
||||||
if (e)
|
|
||||||
{
|
|
||||||
l->start = e->next;
|
|
||||||
e->next = NULL;
|
|
||||||
p = e->info;
|
|
||||||
l->size --;
|
|
||||||
alist_delete_rec(e,l->delete_function);
|
|
||||||
}
|
|
||||||
return p;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
alist_get_size(alist l)
|
|
||||||
{
|
|
||||||
return l->size;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
alist_is_empty(alist l)
|
|
||||||
{
|
|
||||||
return l->size == 0;
|
|
||||||
}
|
|
98
dic/alist.h
98
dic/alist.h
|
@ -1,98 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file alist.h
|
|
||||||
* \brief List type used by automaton
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _ALIST_H_
|
|
||||||
#define _ALIST_H_
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* untyped list type element
|
|
||||||
*/
|
|
||||||
typedef struct alist_elt_t* alist_elt;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* extract the value from an alist element
|
|
||||||
* result is untyped si the user should know
|
|
||||||
* what the value type is
|
|
||||||
*/
|
|
||||||
void* alist_elt_get_value(alist_elt);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* untyped list type
|
|
||||||
*/
|
|
||||||
typedef struct alist_t* alist;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* list creation
|
|
||||||
* @returns list
|
|
||||||
*/
|
|
||||||
alist alist_create ();
|
|
||||||
alist alist_clone (alist);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* funtion to use on data during list deletion.
|
|
||||||
*/
|
|
||||||
void alist_set_delete (alist,void (*f)(void*));
|
|
||||||
|
|
||||||
/**
|
|
||||||
* delete a complete list.
|
|
||||||
*/
|
|
||||||
void alist_delete (alist);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* add a element to the list
|
|
||||||
*/
|
|
||||||
void alist_add (alist, void*);
|
|
||||||
void alist_insert (alist, alist);
|
|
||||||
/**
|
|
||||||
* get first element
|
|
||||||
*/
|
|
||||||
int alist_is_in (alist l, void* e);
|
|
||||||
int alist_equal (alist , alist);
|
|
||||||
|
|
||||||
alist_elt alist_get_first (alist);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* get next element from current
|
|
||||||
*/
|
|
||||||
alist_elt alist_get_next (alist,alist_elt);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns 0 or 1
|
|
||||||
*/
|
|
||||||
int alist_is_empty (alist);
|
|
||||||
|
|
||||||
int alist_get_size (alist);
|
|
||||||
|
|
||||||
void* alist_pop_first_value (alist);
|
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _ALIST_H_ */
|
|
679
dic/automaton.c
679
dic/automaton.c
|
@ -1,679 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file automaton.c
|
|
||||||
* \brief (Non)Deterministic Finite Automaton for Regexp
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include <assert.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
# include <sys/wait.h>
|
|
||||||
#endif
|
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
#include "dic.h"
|
|
||||||
#include "regexp.h"
|
|
||||||
#include "alist.h"
|
|
||||||
#include "automaton.h"
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
#define DMSG(a) a
|
|
||||||
#else
|
|
||||||
#define DMSG(a)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define MAX_TRANSITION_LETTERS 256
|
|
||||||
|
|
||||||
typedef struct automaton_state_t *astate;
|
|
||||||
typedef struct Automaton_t *Automaton;
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
exported functions for static automata
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
automaton automaton_build (int init_state, int *ptl, int *PS, struct search_RegE_list_t *list);
|
|
||||||
void automaton_delete (automaton a);
|
|
||||||
int automaton_get_nstate (automaton a);
|
|
||||||
int automaton_get_init (automaton a);
|
|
||||||
int automaton_get_accept (automaton a, int state);
|
|
||||||
int automaton_get_next_state (automaton a, int start, char l);
|
|
||||||
void automaton_dump (automaton a, char* filename);
|
|
||||||
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
static functions for dynamic automata
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
static Automaton s_automaton_create ();
|
|
||||||
static void s_automaton_delete (Automaton a);
|
|
||||||
|
|
||||||
static alist s_automaton_id_create (int id);
|
|
||||||
static astate s_automaton_state_create (alist id);
|
|
||||||
|
|
||||||
static void s_automaton_add_state (Automaton a, astate s);
|
|
||||||
static astate s_automaton_get_state (Automaton a, alist id);
|
|
||||||
|
|
||||||
static Automaton s_automaton_PS_to_NFA (int init_state, int *ptl, int *PS);
|
|
||||||
static Automaton s_automaton_NFA_to_DFA (Automaton a, struct search_RegE_list_t *list);
|
|
||||||
static automaton s_automaton_finalize (Automaton a);
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
static char* s_automaton_id_to_str (alist id);
|
|
||||||
static void s_automaton_dump (Automaton a, char* filename);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
data types
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
struct automaton_state_t {
|
|
||||||
alist id; // alist of int
|
|
||||||
int accept;
|
|
||||||
int id_static;
|
|
||||||
astate next[MAX_TRANSITION_LETTERS];
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Automaton_t {
|
|
||||||
int nstates;
|
|
||||||
astate init_state;
|
|
||||||
alist states; // alist of alist of int
|
|
||||||
};
|
|
||||||
|
|
||||||
struct automaton_t {
|
|
||||||
int nstates;
|
|
||||||
int init;
|
|
||||||
int *accept;
|
|
||||||
int **trans;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
exported functions for static automata
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
automaton automaton_build(int init_state, int *ptl, int *PS, struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
Automaton nfa,dfa;
|
|
||||||
automaton final;
|
|
||||||
|
|
||||||
nfa = s_automaton_PS_to_NFA(init_state,ptl,PS);
|
|
||||||
DMSG(printf("\n non deterministic automaton OK \n\n"));
|
|
||||||
DMSG(s_automaton_dump(nfa,"auto_nfa"));
|
|
||||||
|
|
||||||
dfa = s_automaton_NFA_to_DFA(nfa, list);
|
|
||||||
DMSG(printf("\n deterministic automaton OK \n\n"));
|
|
||||||
DMSG(s_automaton_dump(dfa,"auto_dfa"));
|
|
||||||
|
|
||||||
final = s_automaton_finalize(dfa);
|
|
||||||
DMSG(printf("\n final automaton OK \n\n"));
|
|
||||||
DMSG(automaton_dump(final,"auto_fin"));
|
|
||||||
|
|
||||||
s_automaton_delete(nfa);
|
|
||||||
s_automaton_delete(dfa);
|
|
||||||
return final;
|
|
||||||
}
|
|
||||||
|
|
||||||
void automaton_delete(automaton a)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
free(a->accept);
|
|
||||||
for(i=0; i <= a->nstates; i++)
|
|
||||||
free(a->trans[i]);
|
|
||||||
free(a->trans);
|
|
||||||
free(a);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int automaton_get_nstates(automaton a)
|
|
||||||
{
|
|
||||||
return a->nstates;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int automaton_get_init(automaton a)
|
|
||||||
{
|
|
||||||
return a->init;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int automaton_get_accept(automaton a, int state)
|
|
||||||
{
|
|
||||||
return a->accept[state];
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int automaton_get_next_state(automaton a, int state, char l)
|
|
||||||
{
|
|
||||||
return a->trans[state][(int)l];
|
|
||||||
}
|
|
||||||
|
|
||||||
void automaton_dump(automaton a, char* filename)
|
|
||||||
{
|
|
||||||
int i,l;
|
|
||||||
FILE* f;
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
pid_t pid;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (a == NULL)
|
|
||||||
return ;
|
|
||||||
f=fopen(filename,"w");
|
|
||||||
fprintf(f,"digraph automaton {\n");
|
|
||||||
for(i=1; i<=a->nstates; i++)
|
|
||||||
{
|
|
||||||
fprintf(f,"\t%d [label = \"%d\"",i,i);
|
|
||||||
if (i == a->init)
|
|
||||||
fprintf(f,", style = filled, color=lightgrey");
|
|
||||||
if (a->accept[i])
|
|
||||||
fprintf(f,", shape = doublecircle");
|
|
||||||
fprintf(f,"];\n");
|
|
||||||
}
|
|
||||||
fprintf(f,"\n");
|
|
||||||
for(i=1; i<=a->nstates; i++)
|
|
||||||
for(l=0; l < MAX_TRANSITION_LETTERS; l++)
|
|
||||||
if (a->trans[i][l])
|
|
||||||
{
|
|
||||||
fprintf(f,"\t%d -> %d [label = \"",i,a->trans[i][l]);
|
|
||||||
regexp_print_letter(f,l);
|
|
||||||
fprintf(f,"\"];\n");
|
|
||||||
}
|
|
||||||
fprintf(f,"fontsize=20;\n");
|
|
||||||
fprintf(f,"}\n");
|
|
||||||
fclose(f);
|
|
||||||
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
pid = fork ();
|
|
||||||
if (pid > 0) {
|
|
||||||
wait(NULL);
|
|
||||||
} else if (pid == 0) {
|
|
||||||
execlp("dotty","dotty",filename,NULL);
|
|
||||||
printf("exec dotty failed\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
void state_delete_fun(void* ps)
|
|
||||||
{
|
|
||||||
astate s = ps;
|
|
||||||
alist_delete(s->id);
|
|
||||||
free(s);
|
|
||||||
}
|
|
||||||
|
|
||||||
static Automaton s_automaton_create()
|
|
||||||
{
|
|
||||||
Automaton a;
|
|
||||||
a = (Automaton)malloc(sizeof(struct Automaton_t));
|
|
||||||
a->nstates = 0;
|
|
||||||
a->init_state = NULL;
|
|
||||||
a->states = alist_create();
|
|
||||||
alist_set_delete(a->states,state_delete_fun);
|
|
||||||
return a;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void s_automaton_delete(Automaton a)
|
|
||||||
{
|
|
||||||
alist_delete(a->states);
|
|
||||||
free(a);
|
|
||||||
}
|
|
||||||
|
|
||||||
static alist s_automaton_id_create(int id)
|
|
||||||
{
|
|
||||||
alist a = alist_create();
|
|
||||||
alist_add(a,(void*)((unsigned long)id));
|
|
||||||
return a;
|
|
||||||
}
|
|
||||||
|
|
||||||
static astate s_automaton_state_create(alist id)
|
|
||||||
{
|
|
||||||
astate s;
|
|
||||||
s = (astate)malloc(sizeof(struct automaton_state_t));
|
|
||||||
s->id = id;
|
|
||||||
s->accept = 0;
|
|
||||||
memset(s->next,0,sizeof(astate)*MAX_TRANSITION_LETTERS);
|
|
||||||
DMSG(printf("** state %s creation\n",s_automaton_id_to_str(id)));
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void s_automaton_add_state(Automaton a, astate s)
|
|
||||||
{
|
|
||||||
a->nstates ++;
|
|
||||||
alist_add(a->states,(void*)s);
|
|
||||||
DMSG(printf("** state %s added to automaton\n",s_automaton_id_to_str(s->id)));
|
|
||||||
}
|
|
||||||
|
|
||||||
static astate s_automaton_get_state(Automaton a, alist id)
|
|
||||||
{
|
|
||||||
astate s;
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(a->states) ; ptr ; ptr = alist_get_next(a->states,ptr))
|
|
||||||
{
|
|
||||||
s = alist_elt_get_value(ptr);
|
|
||||||
if (alist_equal(s->id,id))
|
|
||||||
{
|
|
||||||
//DMSG(printf("** get state %s ok\n",s_automaton_id_to_str(s->id)));
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
Automaton s_automaton_PS_to_NFA(int init_state_id, int *ptl, int *PS)
|
|
||||||
{
|
|
||||||
int p;
|
|
||||||
int maxpos = PS[0];
|
|
||||||
Automaton nfa = NULL;
|
|
||||||
alist temp_id;
|
|
||||||
alist_elt ptr;
|
|
||||||
astate temp_state,current_state;
|
|
||||||
alist L;
|
|
||||||
char used_letter[MAX_TRANSITION_LETTERS];
|
|
||||||
|
|
||||||
nfa = s_automaton_create();
|
|
||||||
L = alist_create();
|
|
||||||
|
|
||||||
/* 1: init_state = root->PP */
|
|
||||||
temp_id = s_automaton_id_create(init_state_id);
|
|
||||||
temp_state = s_automaton_state_create(temp_id);
|
|
||||||
nfa->init_state = temp_state;
|
|
||||||
s_automaton_add_state(nfa,temp_state);
|
|
||||||
alist_add(L,temp_state);
|
|
||||||
/* 2: while \exist state \in state_list */
|
|
||||||
while (! alist_is_empty(L))
|
|
||||||
{
|
|
||||||
current_state = (astate)alist_pop_first_value(L);
|
|
||||||
DMSG(printf("** current state = %s\n",s_automaton_id_to_str(current_state->id)));
|
|
||||||
memset(used_letter,0,sizeof(used_letter));
|
|
||||||
/* 3: \foreach l in \sigma | l \neq # */
|
|
||||||
for(p=1; p < maxpos; p++)
|
|
||||||
{
|
|
||||||
int current_letter = ptl[p];
|
|
||||||
if (used_letter[current_letter] == 0)
|
|
||||||
{
|
|
||||||
/* 4: int set = \cup { PS(pos) | pos \in state \wedge pos == l } */
|
|
||||||
int pos, ens = 0;
|
|
||||||
for(pos = 1; pos <= maxpos; pos++)
|
|
||||||
{
|
|
||||||
if (ptl[pos] == current_letter &&
|
|
||||||
(unsigned long)alist_elt_get_value(alist_get_first(current_state->id)) & (1 << (pos - 1)))
|
|
||||||
ens |= PS[pos];
|
|
||||||
}
|
|
||||||
/* 5: transition from current_state to temp_state */
|
|
||||||
if (ens)
|
|
||||||
{
|
|
||||||
temp_id = s_automaton_id_create(ens);
|
|
||||||
temp_state = s_automaton_get_state(nfa,temp_id);
|
|
||||||
if (temp_state == NULL)
|
|
||||||
{
|
|
||||||
temp_state = s_automaton_state_create(temp_id);
|
|
||||||
s_automaton_add_state (nfa,temp_state);
|
|
||||||
current_state->next[current_letter] = temp_state;
|
|
||||||
alist_add(L,temp_state);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
alist_delete(temp_id);
|
|
||||||
current_state->next[current_letter] = temp_state;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
used_letter[current_letter] = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
alist_delete(L);
|
|
||||||
|
|
||||||
for(ptr = alist_get_first(nfa->states); ptr ; ptr = alist_get_next(nfa->states,ptr))
|
|
||||||
{
|
|
||||||
astate s = (astate)alist_elt_get_value(ptr);
|
|
||||||
if ((unsigned long)alist_elt_get_value(alist_get_first(s->id)) & (1 << (maxpos - 1)))
|
|
||||||
s->accept = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return nfa;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
static alist s_automaton_successor(alist S, int letter, Automaton nfa, struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
alist R,r;
|
|
||||||
alist_elt ptr;
|
|
||||||
R = alist_create(); /* R = \empty */
|
|
||||||
/* \forall y \in S */
|
|
||||||
for(ptr = alist_get_first(S); ptr ; ptr = alist_get_next(S,ptr))
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
alist t, Ry; astate y,z;
|
|
||||||
|
|
||||||
i = (unsigned long)alist_elt_get_value(ptr);
|
|
||||||
t = s_automaton_id_create(i);
|
|
||||||
assert(y = s_automaton_get_state(nfa,t));
|
|
||||||
alist_delete(t);
|
|
||||||
|
|
||||||
Ry = alist_create(); /* Ry = \empty */
|
|
||||||
|
|
||||||
if ((z = y->next[letter]) != NULL) /* \delta (y,z) = l */
|
|
||||||
{
|
|
||||||
r = s_automaton_successor(z->id,RE_EPSILON,nfa, list);
|
|
||||||
alist_insert(Ry,r);
|
|
||||||
alist_delete(r);
|
|
||||||
alist_insert(Ry,z->id); /* Ry = Ry \cup succ(z) */
|
|
||||||
}
|
|
||||||
|
|
||||||
/* \epsilon transition from start node */
|
|
||||||
if ((z = y->next[RE_EPSILON]) != NULL) /* \delta (y,z) = \epsilon */
|
|
||||||
{
|
|
||||||
r = s_automaton_successor(z->id,letter,nfa, list);
|
|
||||||
alist_insert(Ry,r); /* Ry = Ry \cup succ(z) */
|
|
||||||
alist_delete(r);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (letter < RE_FINAL_TOK)
|
|
||||||
{
|
|
||||||
for(i = 0 ; i < DIC_SEARCH_REGE_LIST ; i++)
|
|
||||||
if (list->valid[i])
|
|
||||||
{
|
|
||||||
if (list->letters[i][letter] && (z = y->next[(int)list->symbl[i]]) != NULL)
|
|
||||||
{
|
|
||||||
DMSG(printf("*** letter "));
|
|
||||||
DMSG(regexp_print_letter(stdout,letter));
|
|
||||||
DMSG(printf("is in "));
|
|
||||||
DMSG(regexp_print_letter(stdout,i));
|
|
||||||
|
|
||||||
r = s_automaton_successor(z->id,RE_EPSILON,nfa, list);
|
|
||||||
alist_insert(Ry,r);
|
|
||||||
alist_delete(r);
|
|
||||||
alist_insert(Ry,z->id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
if (alist_is_empty(Ry)) /* Ry = \empty */
|
|
||||||
return Ry;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
alist_insert(R,Ry); /* R = R \cup Ry */
|
|
||||||
alist_delete(Ry);
|
|
||||||
}
|
|
||||||
|
|
||||||
return R;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void s_automaton_node_set_accept(astate s, Automaton nfa)
|
|
||||||
{
|
|
||||||
void* idx;
|
|
||||||
alist_elt ptr;
|
|
||||||
|
|
||||||
DMSG(printf("=== setting accept for node (%s) :",s_automaton_id_to_str(s->id)));
|
|
||||||
for(ptr = alist_get_first(nfa->states) ; ptr ; ptr = alist_get_next(nfa->states,ptr))
|
|
||||||
{
|
|
||||||
astate ns = (astate)alist_elt_get_value(ptr);
|
|
||||||
idx = alist_elt_get_value(alist_get_first(ns->id));
|
|
||||||
DMSG(printf("%s ",s_automaton_id_to_str(ns->id)));
|
|
||||||
if (ns->accept && alist_is_in(s->id,idx))
|
|
||||||
{
|
|
||||||
DMSG(printf("(ok) "));
|
|
||||||
s->accept = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DMSG(printf("\n"));
|
|
||||||
}
|
|
||||||
|
|
||||||
static Automaton s_automaton_NFA_to_DFA(Automaton nfa, struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
Automaton dfa = NULL;
|
|
||||||
alist temp_id;
|
|
||||||
alist_elt ptr;
|
|
||||||
astate temp_state, current_state;
|
|
||||||
alist L;
|
|
||||||
int letter;
|
|
||||||
|
|
||||||
dfa = s_automaton_create();
|
|
||||||
L = alist_create();
|
|
||||||
|
|
||||||
temp_id = alist_clone(nfa->init_state->id);
|
|
||||||
temp_state = s_automaton_state_create(temp_id);
|
|
||||||
dfa->init_state = temp_state;
|
|
||||||
s_automaton_add_state(dfa,temp_state);
|
|
||||||
alist_add(L,temp_state);
|
|
||||||
while (! alist_is_empty(L))
|
|
||||||
{
|
|
||||||
current_state = (astate)alist_pop_first_value(L);
|
|
||||||
DMSG(printf("** current state = %s\n",s_automaton_id_to_str(current_state->id)));
|
|
||||||
for(letter = 1; letter < DIC_LETTERS; letter++)
|
|
||||||
{
|
|
||||||
// DMSG(printf("*** start successor of %s\n",s_automaton_id_to_str(current_state->id)));
|
|
||||||
|
|
||||||
temp_id = s_automaton_successor(current_state->id,letter,nfa,list);
|
|
||||||
|
|
||||||
if (! alist_is_empty(temp_id))
|
|
||||||
{
|
|
||||||
|
|
||||||
DMSG(printf("*** successor of %s for ",s_automaton_id_to_str(current_state->id)));
|
|
||||||
DMSG(regexp_print_letter(stdout,letter));
|
|
||||||
DMSG(printf(" = %s\n", s_automaton_id_to_str(temp_id)));
|
|
||||||
|
|
||||||
temp_state = s_automaton_get_state(dfa,temp_id);
|
|
||||||
|
|
||||||
// DMSG(printf("*** automaton get state -%s- ok\n",s_automaton_id_to_str(temp_id)));
|
|
||||||
|
|
||||||
if (temp_state == NULL)
|
|
||||||
{
|
|
||||||
temp_state = s_automaton_state_create(temp_id);
|
|
||||||
s_automaton_add_state(dfa,temp_state);
|
|
||||||
current_state->next[letter] = temp_state;
|
|
||||||
alist_add(L,temp_state);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
alist_delete(temp_id);
|
|
||||||
current_state->next[letter] = temp_state;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
alist_delete(temp_id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for(ptr = alist_get_first(dfa->states) ; ptr ; ptr = alist_get_next(dfa->states,ptr))
|
|
||||||
{
|
|
||||||
astate s = (astate)alist_elt_get_value(ptr);
|
|
||||||
s_automaton_node_set_accept(s,nfa);
|
|
||||||
}
|
|
||||||
|
|
||||||
alist_delete(L);
|
|
||||||
return dfa;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
static automaton s_automaton_finalize(Automaton a)
|
|
||||||
{
|
|
||||||
int i,l;
|
|
||||||
automaton fa = NULL;
|
|
||||||
alist_elt ptr;
|
|
||||||
astate s;
|
|
||||||
|
|
||||||
if (a == NULL)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
/* creation */
|
|
||||||
fa = (automaton)malloc(sizeof(struct automaton_t));
|
|
||||||
fa->nstates = a->nstates;
|
|
||||||
fa->accept = (int*) malloc((fa->nstates + 1)*sizeof(int));
|
|
||||||
memset(fa->accept,0,(fa->nstates + 1)*sizeof(int));
|
|
||||||
fa->trans = (int**)malloc((fa->nstates + 1)*sizeof(int*));
|
|
||||||
for(i=0; i <= fa->nstates; i++)
|
|
||||||
{
|
|
||||||
fa->trans[i] = (int*)malloc(MAX_TRANSITION_LETTERS * sizeof(int));
|
|
||||||
memset(fa->trans[i],0,MAX_TRANSITION_LETTERS * sizeof(int));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* create new id for states */
|
|
||||||
for(i = 1 , ptr = alist_get_first(a->states); ptr ; ptr = alist_get_next(a->states,ptr), i++)
|
|
||||||
{
|
|
||||||
s = (astate)alist_elt_get_value(ptr);
|
|
||||||
s->id_static = i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* build new automaton */
|
|
||||||
for(ptr = alist_get_first(a->states); ptr ; ptr = alist_get_next(a->states,ptr))
|
|
||||||
{
|
|
||||||
s = (astate)alist_elt_get_value(ptr);
|
|
||||||
i = s->id_static;
|
|
||||||
|
|
||||||
if (s == a->init_state)
|
|
||||||
fa->init = i;
|
|
||||||
if (s->accept == 1)
|
|
||||||
fa->accept[i] = 1;
|
|
||||||
|
|
||||||
for(l=0; l < MAX_TRANSITION_LETTERS; l++)
|
|
||||||
if (s->next[l])
|
|
||||||
fa->trans[i][l] = s->next[l]->id_static;
|
|
||||||
}
|
|
||||||
|
|
||||||
return fa;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
static char* s_automaton_id_to_str(alist id)
|
|
||||||
{
|
|
||||||
static char s[250];
|
|
||||||
memset(s,0,sizeof(s));
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(id); ptr ; ptr = alist_get_next(id,ptr))
|
|
||||||
{
|
|
||||||
char tmp[50];
|
|
||||||
sprintf(tmp,"%d ",(int)alist_elt_get_value(ptr));
|
|
||||||
strcat(s,tmp);
|
|
||||||
}
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
#endif // DEBUG_AUTOMATON
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
static void s_automaton_print_nodes(FILE* f, Automaton a)
|
|
||||||
{
|
|
||||||
char * sid;
|
|
||||||
astate s;
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(a->states) ; ptr != NULL ; ptr = alist_get_next(a->states,ptr))
|
|
||||||
{
|
|
||||||
s = alist_elt_get_value(ptr);
|
|
||||||
sid = s_automaton_id_to_str(s->id);
|
|
||||||
fprintf(f,"\t\"%s\" [label = \"%s\"",sid,sid);
|
|
||||||
if (s == a->init_state)
|
|
||||||
{
|
|
||||||
fprintf(f,", style = filled, color=lightgrey");
|
|
||||||
}
|
|
||||||
if (s->accept)
|
|
||||||
{
|
|
||||||
fprintf(f,", shape = doublecircle");
|
|
||||||
}
|
|
||||||
fprintf(f,"];\n");
|
|
||||||
}
|
|
||||||
fprintf(f,"\n");
|
|
||||||
}
|
|
||||||
#endif // DEBUG_AUTOMATON
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
static void s_automaton_print_edges(FILE* f, Automaton a)
|
|
||||||
{
|
|
||||||
int letter;
|
|
||||||
char * sid;
|
|
||||||
astate s;
|
|
||||||
alist_elt ptr;
|
|
||||||
for(ptr = alist_get_first(a->states) ; ptr != NULL ; ptr = alist_get_next(a->states,ptr))
|
|
||||||
{
|
|
||||||
s = (astate)alist_elt_get_value(ptr);
|
|
||||||
for(letter=0; letter < 255; letter++)
|
|
||||||
{
|
|
||||||
if (s->next[letter])
|
|
||||||
{
|
|
||||||
sid = s_automaton_id_to_str(s->id);
|
|
||||||
fprintf(f,"\t\"%s\" -> ",sid);
|
|
||||||
sid = s_automaton_id_to_str(s->next[letter]->id);
|
|
||||||
fprintf(f,"\"%s\" [label = \"",sid);
|
|
||||||
regexp_print_letter(f,letter);
|
|
||||||
fprintf(f,"\"];\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // DEBUG_AUTOMATON
|
|
||||||
|
|
||||||
#ifdef DEBUG_AUTOMATON
|
|
||||||
static void s_automaton_dump(Automaton a, char* filename)
|
|
||||||
{
|
|
||||||
FILE* f;
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
pid_t pid;
|
|
||||||
#endif
|
|
||||||
if (a == NULL)
|
|
||||||
return;
|
|
||||||
f=fopen(filename,"w");
|
|
||||||
fprintf(f,"digraph automaton {\n");
|
|
||||||
s_automaton_print_nodes(f,a);
|
|
||||||
s_automaton_print_edges(f,a);
|
|
||||||
fprintf(f,"fontsize=20;\n");
|
|
||||||
fprintf(f,"}\n");
|
|
||||||
fclose(f);
|
|
||||||
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
pid = fork ();
|
|
||||||
if (pid > 0) {
|
|
||||||
wait(NULL);
|
|
||||||
} else if (pid == 0) {
|
|
||||||
execlp("dotty","dotty",filename,NULL);
|
|
||||||
printf("exec dotty failed\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif // DEBUG_AUTOMATON
|
|
||||||
|
|
||||||
/* ************************************************** *
|
|
||||||
* ************************************************** *
|
|
||||||
* ************************************************** */
|
|
||||||
|
|
603
dic/automaton.cpp
Normal file
603
dic/automaton.cpp
Normal file
|
@ -0,0 +1,603 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Antoine Fraboulet
|
||||||
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file automaton.c
|
||||||
|
* \brief (Non)Deterministic Finite AutomatonHelper for Regexp
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 2005
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <set>
|
||||||
|
#include <list>
|
||||||
|
#include <cassert>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#ifdef HAVE_SYS_WAIT_H
|
||||||
|
# include <sys/wait.h>
|
||||||
|
#endif
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "dic.h"
|
||||||
|
#include "regexp.h"
|
||||||
|
#include "automaton.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
#ifdef DEBUG_AUTOMATON
|
||||||
|
# define DMSG(a) (a)
|
||||||
|
#else
|
||||||
|
# define DMSG(a)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define MAX_TRANSITION_LETTERS 256
|
||||||
|
|
||||||
|
typedef struct automaton_state_t *astate;
|
||||||
|
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
Helper class, allowing to build a NFA, then a DFA
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
class AutomatonHelper
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
AutomatonHelper(astate iInitState);
|
||||||
|
~AutomatonHelper();
|
||||||
|
|
||||||
|
astate getInitState() const { return m_initState; }
|
||||||
|
#ifdef DEBUG_AUTOMATON
|
||||||
|
void dump(const string &iFileName) const;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static AutomatonHelper *ps2nfa(int iInitState, int *ptl, int *PS);
|
||||||
|
static AutomatonHelper *nfa2dfa(const AutomatonHelper &iNfa,
|
||||||
|
struct search_RegE_list_t *iList);
|
||||||
|
|
||||||
|
/// List of states
|
||||||
|
list<astate> m_states;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Initial state of the automaton
|
||||||
|
astate m_initState;
|
||||||
|
|
||||||
|
void addState(astate s);
|
||||||
|
astate getState(const set<int> &iId) const;
|
||||||
|
void printNodes(FILE* f) const;
|
||||||
|
void printEdges(FILE* f) const;
|
||||||
|
void setAccept(astate s) const;
|
||||||
|
set<int> getSuccessor(const set<int> &S, int letter, struct search_RegE_list_t *iList) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
State handling
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
static set<int> s_state_id_create(int id);
|
||||||
|
static string s_state_id_to_str(const set<int> &iId);
|
||||||
|
static astate s_state_create (const set<int> &iId);
|
||||||
|
|
||||||
|
struct automaton_state_t
|
||||||
|
{
|
||||||
|
set<int> id;
|
||||||
|
int accept;
|
||||||
|
int id_static;
|
||||||
|
astate next[MAX_TRANSITION_LETTERS];
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
Definition of the Automaton class
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
Automaton::Automaton(int iInitState, int *ptl, int *PS, struct search_RegE_list_t *iList)
|
||||||
|
{
|
||||||
|
AutomatonHelper *nfa = AutomatonHelper::ps2nfa(iInitState, ptl, PS);
|
||||||
|
DMSG(printf("\n non deterministic automaton OK \n\n"));
|
||||||
|
DMSG(nfa->dump("auto_nfa"));
|
||||||
|
|
||||||
|
AutomatonHelper *dfa = AutomatonHelper::nfa2dfa(*nfa, iList);
|
||||||
|
DMSG(printf("\n deterministic automaton OK \n\n"));
|
||||||
|
DMSG(dfa->dump("auto_dfa"));
|
||||||
|
|
||||||
|
finalize(*dfa);
|
||||||
|
DMSG(printf("\n final automaton OK \n\n"));
|
||||||
|
DMSG(automaton_dump("auto_fin"));
|
||||||
|
|
||||||
|
delete nfa;
|
||||||
|
delete dfa;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Automaton::~Automaton()
|
||||||
|
{
|
||||||
|
delete[] m_acceptors;
|
||||||
|
for (int i = 0; i <= m_nbStates; i++)
|
||||||
|
{
|
||||||
|
delete[] m_transitions[i];
|
||||||
|
}
|
||||||
|
delete[] m_transitions;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Automaton::finalize(const AutomatonHelper &iHelper)
|
||||||
|
{
|
||||||
|
/* Creation */
|
||||||
|
m_nbStates = iHelper.m_states.size();
|
||||||
|
m_acceptors = new bool[m_nbStates + 1];
|
||||||
|
memset(m_acceptors, 0, (m_nbStates + 1) * sizeof(bool));
|
||||||
|
m_transitions = new int*[m_nbStates + 1];
|
||||||
|
for (int i = 0; i <= m_nbStates; i++)
|
||||||
|
{
|
||||||
|
m_transitions[i] = new int[MAX_TRANSITION_LETTERS];
|
||||||
|
memset(m_transitions[i], 0, MAX_TRANSITION_LETTERS * sizeof(int));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create new id for states */
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
int i;
|
||||||
|
for (i = 1, it = iHelper.m_states.begin();
|
||||||
|
it != iHelper.m_states.end(); it++, i++)
|
||||||
|
{
|
||||||
|
(*it)->id_static = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Build new automaton */
|
||||||
|
for (it = iHelper.m_states.begin(); it != iHelper.m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate s = *it;
|
||||||
|
int i = s->id_static;
|
||||||
|
|
||||||
|
if (s == iHelper.getInitState())
|
||||||
|
m_init = i;
|
||||||
|
if (s->accept == 1)
|
||||||
|
m_acceptors[i] = true;
|
||||||
|
|
||||||
|
for (int l = 0; l < MAX_TRANSITION_LETTERS; l++)
|
||||||
|
{
|
||||||
|
if (s->next[l])
|
||||||
|
m_transitions[i][l] = s->next[l]->id_static;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Automaton::dump(const string &iFileName) const
|
||||||
|
{
|
||||||
|
FILE *f = fopen(iFileName.c_str(), "w");
|
||||||
|
fprintf(f, "digraph automaton {\n");
|
||||||
|
for (int i = 1; i <= m_nbStates; i++)
|
||||||
|
{
|
||||||
|
fprintf(f, "\t%d [label = \"%d\"", i, i);
|
||||||
|
if (i == m_init)
|
||||||
|
fprintf(f, ", style = filled, color=lightgrey");
|
||||||
|
if (accept(i))
|
||||||
|
fprintf(f, ", shape = doublecircle");
|
||||||
|
fprintf(f, "];\n");
|
||||||
|
}
|
||||||
|
fprintf(f, "\n");
|
||||||
|
for (int i = 1; i <= m_nbStates; i++)
|
||||||
|
{
|
||||||
|
for (int l = 0; l < MAX_TRANSITION_LETTERS; l++)
|
||||||
|
{
|
||||||
|
if (m_transitions[i][l])
|
||||||
|
{
|
||||||
|
fprintf(f, "\t%d -> %d [label = \"", i, m_transitions[i][l]);
|
||||||
|
regexp_print_letter(f, l);
|
||||||
|
fprintf(f, "\"];\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fprintf(f, "fontsize=20;\n");
|
||||||
|
fprintf(f, "}\n");
|
||||||
|
fclose(f);
|
||||||
|
|
||||||
|
#ifdef HAVE_SYS_WAIT_H
|
||||||
|
pid_t pid = fork ();
|
||||||
|
if (pid > 0)
|
||||||
|
{
|
||||||
|
wait(NULL);
|
||||||
|
}
|
||||||
|
else if (pid == 0)
|
||||||
|
{
|
||||||
|
execlp("dotty", "dotty", iFileName.c_str(), NULL);
|
||||||
|
printf("exec dotty failed\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
Definition of the state handling methods
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
static set<int> s_state_id_create(int id)
|
||||||
|
{
|
||||||
|
set<int> l;
|
||||||
|
l.insert(id);
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static string s_state_id_to_str(const set<int> &iId)
|
||||||
|
{
|
||||||
|
string s;
|
||||||
|
set<int>::const_iterator it;
|
||||||
|
for (it = iId.begin(); it != iId.end(); it++)
|
||||||
|
{
|
||||||
|
char tmp[50];
|
||||||
|
sprintf(tmp, "%d ", *it);
|
||||||
|
s += tmp;
|
||||||
|
}
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static astate s_state_create(const set<int> &iId)
|
||||||
|
{
|
||||||
|
astate s = new automaton_state_t();
|
||||||
|
// TODO: use copy constructor
|
||||||
|
s->id = iId;
|
||||||
|
s->accept = 0;
|
||||||
|
memset(s->next, 0, sizeof(astate)*MAX_TRANSITION_LETTERS);
|
||||||
|
DMSG(printf("** state %s creation\n", s_state_id_to_str(iId).c_str()));
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
Definition of the AutomatonHelper class
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
AutomatonHelper::AutomatonHelper(astate iInitState)
|
||||||
|
: m_initState(iInitState)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AutomatonHelper::~AutomatonHelper()
|
||||||
|
{
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = m_states.begin(); it != m_states.end(); it++)
|
||||||
|
{
|
||||||
|
delete *it;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void AutomatonHelper::addState(astate s)
|
||||||
|
{
|
||||||
|
m_states.push_front(s);
|
||||||
|
DMSG(printf("** state %s added to automaton\n", s_state_id_to_str(s->id).c_str()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
astate AutomatonHelper::getState(const set<int> &iId) const
|
||||||
|
{
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = m_states.begin(); it != m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate s = *it;
|
||||||
|
if (s->id == iId)
|
||||||
|
{
|
||||||
|
//DMSG(printf("** get state %s ok\n", s_state_id_to_str(s->id).c_str()));
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
* ************************************************** *
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
AutomatonHelper *AutomatonHelper::ps2nfa(int init_state_id, int *ptl, int *PS)
|
||||||
|
{
|
||||||
|
int maxpos = PS[0];
|
||||||
|
astate current_state;
|
||||||
|
char used_letter[MAX_TRANSITION_LETTERS];
|
||||||
|
|
||||||
|
|
||||||
|
/* 1: init_state = root->PP */
|
||||||
|
set<int> temp_id0 = s_state_id_create(init_state_id);
|
||||||
|
astate temp_state = s_state_create(temp_id0);
|
||||||
|
AutomatonHelper *nfa = new AutomatonHelper(temp_state);
|
||||||
|
nfa->addState(temp_state);
|
||||||
|
list<astate> L;
|
||||||
|
L.push_front(temp_state);
|
||||||
|
/* 2: while \exist state \in state_list */
|
||||||
|
while (! L.empty())
|
||||||
|
{
|
||||||
|
current_state = L.front();
|
||||||
|
L.pop_front();
|
||||||
|
DMSG(printf("** current state = %s\n", s_state_id_to_str(current_state->id).c_str()));
|
||||||
|
memset(used_letter, 0, sizeof(used_letter));
|
||||||
|
/* 3: \foreach l in \sigma | l \neq # */
|
||||||
|
for (int p = 1; p < maxpos; p++)
|
||||||
|
{
|
||||||
|
int current_letter = ptl[p];
|
||||||
|
if (used_letter[current_letter] == 0)
|
||||||
|
{
|
||||||
|
/* 4: int set = \cup { PS(pos) | pos \in state \wedge pos == l } */
|
||||||
|
int ens = 0;
|
||||||
|
for (int pos = 1; pos <= maxpos; pos++)
|
||||||
|
{
|
||||||
|
if (ptl[pos] == current_letter &&
|
||||||
|
(unsigned int)*(current_state->id.begin()) & (1 << (pos - 1)))
|
||||||
|
ens |= PS[pos];
|
||||||
|
}
|
||||||
|
/* 5: transition from current_state to temp_state */
|
||||||
|
if (ens)
|
||||||
|
{
|
||||||
|
set<int> temp_id = s_state_id_create(ens);
|
||||||
|
temp_state = nfa->getState(temp_id);
|
||||||
|
if (temp_state == NULL)
|
||||||
|
{
|
||||||
|
temp_state = s_state_create(temp_id);
|
||||||
|
nfa->addState(temp_state);
|
||||||
|
current_state->next[current_letter] = temp_state;
|
||||||
|
L.push_front(temp_state);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
current_state->next[current_letter] = temp_state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
used_letter[current_letter] = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = nfa->m_states.begin(); it != nfa->m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate s = *it;
|
||||||
|
if (*(s->id.begin()) & (1 << (maxpos - 1)))
|
||||||
|
s->accept = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nfa;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
* ************************************************** *
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
set<int> AutomatonHelper::getSuccessor(const set<int> &S,
|
||||||
|
int letter,
|
||||||
|
struct search_RegE_list_t *iList) const
|
||||||
|
{
|
||||||
|
set<int> R, r;
|
||||||
|
set<int>::const_iterator it;
|
||||||
|
for (it = S.begin(); it != S.end(); it++) /* \forall y \in S */
|
||||||
|
{
|
||||||
|
astate y, z;
|
||||||
|
|
||||||
|
set<int> t = s_state_id_create(*it);
|
||||||
|
assert(y = getState(t));
|
||||||
|
|
||||||
|
set<int> Ry; /* Ry = \empty */
|
||||||
|
|
||||||
|
if ((z = y->next[letter]) != NULL) /* \delta (y,z) = l */
|
||||||
|
{
|
||||||
|
r = getSuccessor(z->id, RE_EPSILON, iList);
|
||||||
|
Ry.insert(r.begin(), r.end());
|
||||||
|
Ry.insert(z->id.begin(), z->id.end()); /* Ry = Ry \cup succ(z) */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* \epsilon transition from start node */
|
||||||
|
if ((z = y->next[RE_EPSILON]) != NULL) /* \delta (y,z) = \epsilon */
|
||||||
|
{
|
||||||
|
r = getSuccessor(z->id, letter, iList);
|
||||||
|
Ry.insert(r.begin(), r.end()); /* Ry = Ry \cup succ(z) */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (letter < RE_FINAL_TOK)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < DIC_SEARCH_REGE_LIST; i++)
|
||||||
|
{
|
||||||
|
if (iList->valid[i])
|
||||||
|
{
|
||||||
|
if (iList->letters[i][letter] && (z = y->next[(int)iList->symbl[i]]) != NULL)
|
||||||
|
{
|
||||||
|
DMSG(printf("*** letter "));
|
||||||
|
DMSG(regexp_print_letter(stdout, letter));
|
||||||
|
DMSG(printf("is in "));
|
||||||
|
DMSG(regexp_print_letter(stdout, i));
|
||||||
|
|
||||||
|
r = getSuccessor(z->id, RE_EPSILON, iList);
|
||||||
|
Ry.insert(r.begin(), r.end());
|
||||||
|
Ry.insert(z->id.begin(), z->id.end());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
if (alist_is_empty(Ry)) /* Ry = \empty */
|
||||||
|
return Ry;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
R.insert(Ry.begin(), Ry.end()); /* R = R \cup Ry */
|
||||||
|
}
|
||||||
|
|
||||||
|
return R;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void AutomatonHelper::setAccept(astate s) const
|
||||||
|
{
|
||||||
|
DMSG(printf("=== setting accept for node (%s) :", s_state_id_to_str(s->id).c_str()));
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = m_states.begin(); it != m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate ns = *it;
|
||||||
|
int idx = *(ns->id.begin());
|
||||||
|
DMSG(printf("%s ", s_state_id_to_str(ns->id).c_str()));
|
||||||
|
if (ns->accept && (find(s->id.begin(), s->id.end(), idx) != s->id.end()))
|
||||||
|
{
|
||||||
|
DMSG(printf("(ok) "));
|
||||||
|
s->accept = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DMSG(printf("\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AutomatonHelper *AutomatonHelper::nfa2dfa(const AutomatonHelper &iNfa,
|
||||||
|
struct search_RegE_list_t *iList)
|
||||||
|
{
|
||||||
|
astate current_state;
|
||||||
|
|
||||||
|
list<astate> L;
|
||||||
|
|
||||||
|
// Clone the list
|
||||||
|
set<int> temp_id0 = iNfa.m_initState->id;
|
||||||
|
astate temp_state = s_state_create(temp_id0);
|
||||||
|
AutomatonHelper *dfa = new AutomatonHelper(temp_state);
|
||||||
|
dfa->addState(temp_state);
|
||||||
|
L.push_front(temp_state);
|
||||||
|
while (! L.empty())
|
||||||
|
{
|
||||||
|
current_state = L.front();
|
||||||
|
L.pop_front();
|
||||||
|
DMSG(printf("** current state = %s\n", s_state_id_to_str(current_state->id).c_str()));
|
||||||
|
for (int letter = 1; letter < DIC_LETTERS; letter++)
|
||||||
|
{
|
||||||
|
// DMSG(printf("*** start successor of %s\n", s_state_id_to_str(current_state->id).c_str()));
|
||||||
|
|
||||||
|
set<int> temp_id = iNfa.getSuccessor(current_state->id, letter, iList);
|
||||||
|
|
||||||
|
if (! temp_id.empty())
|
||||||
|
{
|
||||||
|
|
||||||
|
DMSG(printf("*** successor of %s for ", s_state_id_to_str(current_state->id).c_str()));
|
||||||
|
DMSG(regexp_print_letter(stdout, letter));
|
||||||
|
DMSG(printf(" = %s\n", s_state_id_to_str(temp_id).c_str()));
|
||||||
|
|
||||||
|
temp_state = dfa->getState(temp_id);
|
||||||
|
|
||||||
|
// DMSG(printf("*** automaton get state -%s- ok\n", s_state_id_to_str(temp_id).c_str()));
|
||||||
|
|
||||||
|
if (temp_state == NULL)
|
||||||
|
{
|
||||||
|
temp_state = s_state_create(temp_id);
|
||||||
|
dfa->addState(temp_state);
|
||||||
|
current_state->next[letter] = temp_state;
|
||||||
|
L.push_front(temp_state);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
current_state->next[letter] = temp_state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = dfa->m_states.begin(); it != dfa->m_states.end(); it++)
|
||||||
|
{
|
||||||
|
iNfa.setAccept(*it);
|
||||||
|
}
|
||||||
|
|
||||||
|
return dfa;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ************************************************** *
|
||||||
|
* ************************************************** *
|
||||||
|
* ************************************************** */
|
||||||
|
|
||||||
|
void AutomatonHelper::printNodes(FILE* f) const
|
||||||
|
{
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = m_states.begin(); it != m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate s = *it;
|
||||||
|
string sid = s_state_id_to_str(s->id);
|
||||||
|
fprintf(f, "\t\"%s\" [label = \"%s\"", sid.c_str(), sid.c_str());
|
||||||
|
if (s == m_initState)
|
||||||
|
{
|
||||||
|
fprintf(f, ", style = filled, color=lightgrey");
|
||||||
|
}
|
||||||
|
if (s->accept)
|
||||||
|
{
|
||||||
|
fprintf(f, ", shape = doublecircle");
|
||||||
|
}
|
||||||
|
fprintf(f, "];\n");
|
||||||
|
}
|
||||||
|
fprintf(f, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void AutomatonHelper::printEdges(FILE* f) const
|
||||||
|
{
|
||||||
|
list<astate>::const_iterator it;
|
||||||
|
for (it = m_states.begin(); it != m_states.end(); it++)
|
||||||
|
{
|
||||||
|
astate s = *it;
|
||||||
|
for (int letter = 0; letter < 255; letter++)
|
||||||
|
{
|
||||||
|
if (s->next[letter])
|
||||||
|
{
|
||||||
|
string sid = s_state_id_to_str(s->id);
|
||||||
|
fprintf(f, "\t\"%s\" -> ", sid.c_str());
|
||||||
|
sid = s_state_id_to_str(s->next[letter]->id);
|
||||||
|
fprintf(f, "\"%s\" [label = \"", sid.c_str());
|
||||||
|
regexp_print_letter(f, letter);
|
||||||
|
fprintf(f, "\"];\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef DEBUG_AUTOMATON
|
||||||
|
void AutomatonHelper::dump(const string &iFileName) const
|
||||||
|
{
|
||||||
|
FILE *f = fopen(iFileName.c_str(), "w");
|
||||||
|
fprintf(f, "digraph automaton {\n");
|
||||||
|
printNodes(f);
|
||||||
|
printEdges(f);
|
||||||
|
fprintf(f, "fontsize=20;\n");
|
||||||
|
fprintf(f, "}\n");
|
||||||
|
fclose(f);
|
||||||
|
|
||||||
|
#ifdef HAVE_SYS_WAIT_H
|
||||||
|
pid_t pid = fork();
|
||||||
|
if (pid > 0)
|
||||||
|
{
|
||||||
|
wait(NULL);
|
||||||
|
}
|
||||||
|
else if (pid == 0)
|
||||||
|
{
|
||||||
|
execlp("dotty", "dotty", iFileName.c_str(), NULL);
|
||||||
|
printf("exec dotty failed\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
104
dic/automaton.h
104
dic/automaton.h
|
@ -1,21 +1,22 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
*
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
* 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 */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* (at your option) any later version. */
|
* (at your option) any later version.
|
||||||
/* */
|
*
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
/* GNU General Public License for more details. */
|
* GNU General Public License for more details.
|
||||||
/* */
|
*
|
||||||
/* You should have received a copy of the GNU General Public License */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* along with this program; if not, write to the Free Software */
|
* along with this program; if not, write to the Free Software
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file automaton.h
|
* \file automaton.h
|
||||||
|
@ -26,51 +27,68 @@
|
||||||
|
|
||||||
#ifndef _DIC_AUTOMATON_H_
|
#ifndef _DIC_AUTOMATON_H_
|
||||||
#define _DIC_AUTOMATON_H_
|
#define _DIC_AUTOMATON_H_
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
typedef struct automaton_t *automaton;
|
class AutomatonHelper;
|
||||||
|
|
||||||
|
class Automaton
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// Constructor
|
||||||
/**
|
/**
|
||||||
* build a static deterministic finite automaton from
|
* Build a static deterministic finite automaton from
|
||||||
* "init_state", "ptl" and "PS" given by the parser
|
* "init_state", "ptl" and "PS" given by the parser
|
||||||
*/
|
*/
|
||||||
automaton automaton_build(int init_state, int *ptl, int *PS, struct search_RegE_list_t *list);
|
Automaton(int init_state, int *ptl, int *PS, struct search_RegE_list_t *iList);
|
||||||
|
|
||||||
|
/// Destructor
|
||||||
|
~Automaton();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* automaton delete function
|
* Get the number of states in the automaton.
|
||||||
*/
|
|
||||||
void automaton_delete (automaton a);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* get the number of states in the automaton
|
|
||||||
* @returns number of states
|
* @returns number of states
|
||||||
*/
|
*/
|
||||||
int automaton_get_nstate (automaton a);
|
int getNbStates() const { return m_nbStates; }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* query the id of the init state
|
* Query the id of the init state.
|
||||||
* @returns init state id
|
* @returns init state id
|
||||||
*/
|
*/
|
||||||
int automaton_get_init (automaton a);
|
int getInitId() const { return m_init; }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ask for the acceptor flag for the state
|
* Query the acceptor flag for the given state
|
||||||
* @returns boolean flag 0 or 1
|
* @return true/false
|
||||||
*/
|
*/
|
||||||
int automaton_get_accept (automaton a, int state);
|
bool accept(int state) const { return m_acceptors[state]; }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* returns the next state when the transition is taken
|
* Return the next state when the transition is taken
|
||||||
* @returns next state id (1 <= id <= nstate, 0 = invalid id)
|
* @returns next state id (1 <= id <= nstate, 0 = invalid id)
|
||||||
*/
|
*/
|
||||||
int automaton_get_next_state (automaton a, int start, char l);
|
int getNextState(int start, char l) const
|
||||||
|
{
|
||||||
|
return m_transitions[start][(int)l];
|
||||||
|
}
|
||||||
|
|
||||||
void automaton_dump (automaton a, char* filename);
|
/**
|
||||||
|
* Dump the automaton into a file (for debugging purposes)
|
||||||
|
*/
|
||||||
|
void dump(const string &iFileName) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Number of states
|
||||||
|
int m_nbStates;
|
||||||
|
|
||||||
|
/// ID of the init state
|
||||||
|
int m_init;
|
||||||
|
|
||||||
|
/// Array of booleans, one for each state
|
||||||
|
bool *m_acceptors;
|
||||||
|
|
||||||
|
/// Matrix of transitions
|
||||||
|
int **m_transitions;
|
||||||
|
|
||||||
|
void finalize(const AutomatonHelper &a);
|
||||||
|
};
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _DIC_AUTOMATON_H_ */
|
#endif /* _DIC_AUTOMATON_H_ */
|
||||||
|
|
350
dic/compdic.c
350
dic/compdic.c
|
@ -1,350 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file compdic.c
|
|
||||||
* \brief Program used to compress a dictionary
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 1999
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <time.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <ctype.h>
|
|
||||||
#include <assert.h>
|
|
||||||
|
|
||||||
#include "hashtable.h"
|
|
||||||
#include "dic_internals.h"
|
|
||||||
#include "dic.h"
|
|
||||||
|
|
||||||
//#define DEBUG_LIST
|
|
||||||
//#define DEBUG_OUTPUT
|
|
||||||
//#define DEBUG_OUTPUT_L2
|
|
||||||
#define CHECK_RECURSION
|
|
||||||
|
|
||||||
char*
|
|
||||||
load_uncompressed(const char* file_name, unsigned int *dic_size)
|
|
||||||
{
|
|
||||||
unsigned r;
|
|
||||||
char *uncompressed;
|
|
||||||
FILE* file_desc;
|
|
||||||
|
|
||||||
if ((file_desc = fopen (file_name, "r")) == NULL)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if ((uncompressed = (char*)malloc (sizeof(char)*(*dic_size))) == NULL)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
r = fread (uncompressed, 1, *dic_size, file_desc);
|
|
||||||
if (r < *dic_size)
|
|
||||||
{
|
|
||||||
/* \n is 2 chars under MS OS */
|
|
||||||
printf("\n");
|
|
||||||
printf("** The number of bytes read is less than the size of the file **\n");
|
|
||||||
printf("** this may be OK if you run a Microsoft OS but not on Unix **\n");
|
|
||||||
printf("** please check the results. **\n");
|
|
||||||
printf("\n");
|
|
||||||
*dic_size = r;
|
|
||||||
}
|
|
||||||
|
|
||||||
fclose(file_desc);
|
|
||||||
return uncompressed;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
file_length(const char* file_name)
|
|
||||||
{
|
|
||||||
struct stat stat_buf;
|
|
||||||
if (stat (file_name, &stat_buf) < 0)
|
|
||||||
return - 1;
|
|
||||||
return (int) stat_buf.st_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
skip_init_header(FILE* outfile, Dict_header *header)
|
|
||||||
{
|
|
||||||
header->unused_1 = 0;
|
|
||||||
header->unused_2 = 0;
|
|
||||||
header->root = 0;
|
|
||||||
header->nwords = 0;
|
|
||||||
header->nodesused = 1;
|
|
||||||
header->edgesused = 1;
|
|
||||||
header->nodessaved = 0;
|
|
||||||
header->edgessaved = 0;
|
|
||||||
|
|
||||||
fwrite (header, sizeof(Dict_header), 1, outfile);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
fix_header(FILE* outfile, Dict_header* header)
|
|
||||||
{
|
|
||||||
strcpy(header->ident,_COMPIL_KEYWORD_);
|
|
||||||
header->root = header->edgesused;
|
|
||||||
rewind (outfile);
|
|
||||||
#if defined(WORDS_BIGENDIAN)
|
|
||||||
#warning "**********************************************"
|
|
||||||
#warning "compdic does not run yet on bigendian machines"
|
|
||||||
#warning "**********************************************"
|
|
||||||
#else
|
|
||||||
fwrite (header, sizeof(Dict_header), 1, outfile);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
print_header_info(Dict_header *header)
|
|
||||||
{
|
|
||||||
printf("============================\n");
|
|
||||||
printf("keyword length %lu bytes\n", (unsigned long)strlen(_COMPIL_KEYWORD_));
|
|
||||||
printf("keyword size %lu bytes\n", (unsigned long)sizeof(_COMPIL_KEYWORD_));
|
|
||||||
printf("header size %lu bytes\n", (unsigned long)sizeof(Dict_header));
|
|
||||||
printf("\n");
|
|
||||||
printf("%d words\n",header->nwords);
|
|
||||||
printf("\n");
|
|
||||||
printf("root : %7d (edge)\n",header->root);
|
|
||||||
printf("root : %7lu (byte)\n",(unsigned long) header->root * sizeof(Dawg_edge));
|
|
||||||
printf("\n");
|
|
||||||
printf("nodes : %d+%d\n",header->nodesused, header->nodessaved);
|
|
||||||
printf("edges : %d+%d\n",header->edgesused, header->edgessaved);
|
|
||||||
printf("============================\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
write_node(Dawg_edge *edges, int size, int num, FILE* outfile)
|
|
||||||
{
|
|
||||||
#ifdef DEBUG_OUTPUT
|
|
||||||
int i;
|
|
||||||
printf("writing %d edges\n",num);
|
|
||||||
for(i=0; i<num; i++)
|
|
||||||
{
|
|
||||||
#ifdef DEBUG_OUTPUT_L2
|
|
||||||
printf("ptr=%2d t=%d l=%d f=%d chr=%2d (%c)\n",
|
|
||||||
edges[i].ptr, edges[i].term, edges[i].last,
|
|
||||||
edges[i].fill, edges[i].chr, edges[i].chr -1 +'a');
|
|
||||||
#endif
|
|
||||||
fwrite (edges+i, sizeof(Dawg_edge), 1, outfile);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
fwrite (edges, size, num, outfile);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#define MAX_STRING_LENGTH 200
|
|
||||||
|
|
||||||
|
|
||||||
#define MAX_EDGES 2000
|
|
||||||
/* ods3: ?? */
|
|
||||||
/* ods4: 1746 */
|
|
||||||
|
|
||||||
/* global variables */
|
|
||||||
FILE* global_outfile;
|
|
||||||
Dict_header global_header;
|
|
||||||
Hash_table global_hashtable;
|
|
||||||
|
|
||||||
char global_stringbuf[MAX_STRING_LENGTH]; /* Space for current string */
|
|
||||||
char* global_endstring; /* Marks END of current string */
|
|
||||||
char* global_input;
|
|
||||||
char* global_endofinput;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Makenode takes a prefix (as position relative to stringbuf) and
|
|
||||||
* returns an index of the start node of a dawg that recognizes all
|
|
||||||
* words beginning with that prefix. String is a pointer (relative
|
|
||||||
* to stringbuf) indicating how much of prefix is matched in the
|
|
||||||
* input.
|
|
||||||
*/
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
int current_rec =0;
|
|
||||||
int max_rec = 0;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
unsigned int
|
|
||||||
makenode(char *prefix)
|
|
||||||
{
|
|
||||||
int numedges;
|
|
||||||
Dawg_edge edges[MAX_EDGES];
|
|
||||||
Dawg_edge *edgeptr = edges;
|
|
||||||
unsigned *saved_position;
|
|
||||||
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
current_rec++;
|
|
||||||
if (current_rec > max_rec)
|
|
||||||
max_rec = current_rec;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
while (prefix == global_endstring)
|
|
||||||
{
|
|
||||||
/* More edges out of node */
|
|
||||||
edgeptr->ptr = 0;
|
|
||||||
edgeptr->term = 0;
|
|
||||||
edgeptr->last = 0;
|
|
||||||
edgeptr->fill = 0;
|
|
||||||
edgeptr->chr = 0;
|
|
||||||
|
|
||||||
(*(edgeptr++)).chr = (*global_endstring++ = *global_input++) & DIC_CHAR_MASK;
|
|
||||||
if (*global_input == '\n') /* End of a word */
|
|
||||||
{
|
|
||||||
global_header.nwords++;
|
|
||||||
edgeptr[-1].term = 1; /* Mark edge as word */
|
|
||||||
*global_endstring++ = *global_input++; /* Skip \n */
|
|
||||||
if (global_input == global_endofinput) /* At end of input? */
|
|
||||||
break;
|
|
||||||
|
|
||||||
global_endstring = global_stringbuf;
|
|
||||||
while(*global_endstring == *global_input)
|
|
||||||
{
|
|
||||||
global_endstring++;
|
|
||||||
global_input++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/* make dawg pointed to by this edge */
|
|
||||||
edgeptr[-1].ptr = makenode(prefix + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
numedges = edgeptr - edges;
|
|
||||||
if (numedges == 0)
|
|
||||||
{
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
current_rec --;
|
|
||||||
#endif
|
|
||||||
return 0; /* Special node zero - no edges */
|
|
||||||
}
|
|
||||||
|
|
||||||
edgeptr[-1].last = 1; /* Mark the last edge */
|
|
||||||
|
|
||||||
saved_position = (unsigned int*) hash_find (global_hashtable,
|
|
||||||
(void*)edges,
|
|
||||||
numedges*sizeof(Dawg_edge));
|
|
||||||
if (saved_position)
|
|
||||||
{
|
|
||||||
global_header.edgessaved += numedges;
|
|
||||||
global_header.nodessaved++;
|
|
||||||
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
current_rec --;
|
|
||||||
#endif
|
|
||||||
return *saved_position;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
unsigned int node_pos;
|
|
||||||
|
|
||||||
node_pos = global_header.edgesused;
|
|
||||||
hash_add(global_hashtable,
|
|
||||||
(void*)edges,numedges*sizeof(Dawg_edge),
|
|
||||||
(void*)(&global_header.edgesused),sizeof(global_header.edgesused));
|
|
||||||
global_header.edgesused += numedges;
|
|
||||||
global_header.nodesused++;
|
|
||||||
write_node (edges, sizeof(Dawg_edge), numedges, global_outfile);
|
|
||||||
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
current_rec --;
|
|
||||||
#endif
|
|
||||||
return node_pos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
main(int argc, char* argv[])
|
|
||||||
{
|
|
||||||
unsigned int dicsize;
|
|
||||||
char *uncompressed;
|
|
||||||
Dawg_edge rootnode = {0,0,0,0,0};
|
|
||||||
Dawg_edge specialnode = {0,0,0,0,0};
|
|
||||||
|
|
||||||
int size;
|
|
||||||
char* outfilename;
|
|
||||||
char outfilenamedefault[] = "dict.daw";
|
|
||||||
clock_t starttime, endtime;
|
|
||||||
|
|
||||||
if (argc < 2)
|
|
||||||
{
|
|
||||||
fprintf(stderr,"usage: %s uncompressed_dic [compressed_dic]\n",argv[0]);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
size = file_length (argv[1]);
|
|
||||||
if (size < 0)
|
|
||||||
{
|
|
||||||
fprintf(stderr,"Cannot stat uncompressed dictionary %s\n",argv[1]);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
dicsize = size;
|
|
||||||
outfilename = (argc == 3) ? argv[2] : outfilenamedefault;
|
|
||||||
|
|
||||||
if ((global_outfile = fopen (outfilename,"wb")) == NULL)
|
|
||||||
{
|
|
||||||
fprintf(stderr,"Cannot open output file %s\n",outfilename);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((uncompressed = load_uncompressed(argv[1], &dicsize)) == NULL)
|
|
||||||
{
|
|
||||||
fprintf(stderr,"Cannot load uncompressed dictionary into memory\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
global_input = uncompressed;
|
|
||||||
global_endofinput = global_input + dicsize;
|
|
||||||
|
|
||||||
#define SCALE 0.6
|
|
||||||
global_hashtable = hash_init((unsigned int)(dicsize * SCALE));
|
|
||||||
#undef SCALE
|
|
||||||
|
|
||||||
skip_init_header(global_outfile,&global_header);
|
|
||||||
|
|
||||||
specialnode.last = 1;
|
|
||||||
write_node(&specialnode,sizeof(specialnode),1,global_outfile);
|
|
||||||
/*
|
|
||||||
* Call makenode with null (relative to stringbuf) prefix;
|
|
||||||
* Initialize string to null; Put index of start node on output
|
|
||||||
*/
|
|
||||||
starttime=clock();
|
|
||||||
rootnode.ptr = makenode(global_endstring = global_stringbuf);
|
|
||||||
endtime=clock();
|
|
||||||
write_node(&rootnode,sizeof(rootnode),1,global_outfile);
|
|
||||||
|
|
||||||
fix_header(global_outfile,&global_header);
|
|
||||||
|
|
||||||
print_header_info(&global_header);
|
|
||||||
hash_destroy(global_hashtable);
|
|
||||||
free(uncompressed);
|
|
||||||
fclose(global_outfile);
|
|
||||||
|
|
||||||
printf(" Elapsed time is : %f s\n", 1.0*(endtime-starttime) / CLOCKS_PER_SEC);
|
|
||||||
#ifdef CHECK_RECURSION
|
|
||||||
printf(" Maximum recursion level reached : %d\n",max_rec);
|
|
||||||
#endif
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
563
dic/compdic.cpp
Normal file
563
dic/compdic.cpp
Normal file
|
@ -0,0 +1,563 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 1999-2007 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file compdic.c
|
||||||
|
* \brief Program used to compress a dictionary
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 1999
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <vector>
|
||||||
|
#include <map>
|
||||||
|
#include <boost/tokenizer.hpp>
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <ctime>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <cwctype>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
// For ntohl & Co.
|
||||||
|
#ifdef WIN32
|
||||||
|
# include <winsock2.h>
|
||||||
|
#else
|
||||||
|
# if HAVE_NETINET_IN_H
|
||||||
|
# include <netinet/in.h>
|
||||||
|
# endif
|
||||||
|
# if HAVE_ARPA_INET_H
|
||||||
|
# include <arpa/inet.h>
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
# include <libintl.h>
|
||||||
|
# define _(String) gettext(String)
|
||||||
|
#else
|
||||||
|
# define _(String) String
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "hashtable.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "header.h"
|
||||||
|
#include "dic_internals.h"
|
||||||
|
#include "dic_exception.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
//#define DEBUG_LIST
|
||||||
|
//#define DEBUG_OUTPUT
|
||||||
|
//#define DEBUG_OUTPUT_L2
|
||||||
|
#define CHECK_RECURSION
|
||||||
|
|
||||||
|
|
||||||
|
const wchar_t* load_uncompressed(const string &iFileName, unsigned int &ioDicSize)
|
||||||
|
{
|
||||||
|
ifstream file(iFileName.c_str());
|
||||||
|
if (!file.is_open())
|
||||||
|
throw DicException("Could not open file " + iFileName);
|
||||||
|
|
||||||
|
// Place the buffer in a vector to avoid worrying about memory handling
|
||||||
|
vector<char> buffer(ioDicSize);
|
||||||
|
// Load the file data, everything in one shot
|
||||||
|
file.read(&buffer.front(), ioDicSize);
|
||||||
|
file.close();
|
||||||
|
|
||||||
|
// Buffer for the wide characters (it will use at most as many characters
|
||||||
|
// as the utf-8 version)
|
||||||
|
wchar_t *wideBuf = new wchar_t[ioDicSize];
|
||||||
|
unsigned int number;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
number = readFromUTF8(wideBuf, ioDicSize, &buffer.front(),
|
||||||
|
ioDicSize, "load_uncompressed");
|
||||||
|
ioDicSize = number;
|
||||||
|
return wideBuf;
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
// Avoid leaks, and propagate the exception
|
||||||
|
delete[] wideBuf;
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void readLetters(const char *iFileName, DictHeaderInfo &ioHeaderInfo)
|
||||||
|
{
|
||||||
|
ifstream in(iFileName);
|
||||||
|
if (!in.is_open())
|
||||||
|
throw DicException("Could not open file " + string(iFileName));
|
||||||
|
|
||||||
|
// Use a more friendly type name
|
||||||
|
typedef boost::tokenizer<boost::char_separator<char> > Tokenizer;
|
||||||
|
|
||||||
|
int lineNb = 1;
|
||||||
|
string line;
|
||||||
|
while (getline(in, line))
|
||||||
|
{
|
||||||
|
// Split the lines on space characters
|
||||||
|
vector<string> tokens;
|
||||||
|
boost::char_separator<char> sep(" ");
|
||||||
|
Tokenizer tok(line, sep);
|
||||||
|
Tokenizer::iterator it;
|
||||||
|
for (it = tok.begin(); it != tok.end(); ++it)
|
||||||
|
{
|
||||||
|
tokens.push_back(*it);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore empty lines
|
||||||
|
if (tokens.empty())
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// We expect 5 fields on the line, and the first one is a letter, so
|
||||||
|
// it cannot exceed 4 bytes
|
||||||
|
if (tokens.size() != 5 || tokens[0].size() > 4)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "readLetters: Invalid line in " << iFileName;
|
||||||
|
ss << " (line " << lineNb;
|
||||||
|
throw DicException(ss.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
#define MAX_SIZE 4
|
||||||
|
char buff[MAX_SIZE];
|
||||||
|
strncpy(buff, tokens[0].c_str(), MAX_SIZE);
|
||||||
|
|
||||||
|
wstring letter = readFromUTF8(buff, tokens[0].size(), "readLetters");
|
||||||
|
|
||||||
|
if (letter.size() != 1)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "readLetters: Invalid letter at line " << lineNb;
|
||||||
|
throw DicException(ss.str());
|
||||||
|
}
|
||||||
|
#undef MAX_SIZE
|
||||||
|
|
||||||
|
ioHeaderInfo.letters += towupper(letter[0]);
|
||||||
|
|
||||||
|
ioHeaderInfo.points.push_back(atoi(tokens[1].c_str()));
|
||||||
|
ioHeaderInfo.frequency.push_back(atoi(tokens[2].c_str()));
|
||||||
|
ioHeaderInfo.vowels.push_back(atoi(tokens[3].c_str()));
|
||||||
|
ioHeaderInfo.consonants.push_back(atoi(tokens[4].c_str()));
|
||||||
|
|
||||||
|
++lineNb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Header skip_init_header(ostream &outfile, DictHeaderInfo &ioHeaderInfo)
|
||||||
|
{
|
||||||
|
ioHeaderInfo.root = 0;
|
||||||
|
ioHeaderInfo.nwords = 0;
|
||||||
|
ioHeaderInfo.nodesused = 1;
|
||||||
|
ioHeaderInfo.edgesused = 1;
|
||||||
|
ioHeaderInfo.nodessaved = 0;
|
||||||
|
ioHeaderInfo.edgessaved = 0;
|
||||||
|
|
||||||
|
Header aHeader(ioHeaderInfo);
|
||||||
|
aHeader.write(outfile);
|
||||||
|
return aHeader;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void fix_header(ostream &outfile, DictHeaderInfo &ioHeaderInfo)
|
||||||
|
{
|
||||||
|
ioHeaderInfo.root = ioHeaderInfo.edgesused;
|
||||||
|
// Go back to the beginning of the stream to overwrite the header
|
||||||
|
outfile.seekp(0, ios::beg);
|
||||||
|
#if defined(WORDS_BIGENDIAN)
|
||||||
|
#warning "**********************************************"
|
||||||
|
#warning "compdic does not run yet on bigendian machines"
|
||||||
|
#warning "**********************************************"
|
||||||
|
#else
|
||||||
|
Header aHeader(ioHeaderInfo);
|
||||||
|
aHeader.write(outfile);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Change endianness of the pointes edges, and write them to the given ostream
|
||||||
|
void write_node(uint32_t *ioEdges, unsigned int num, ostream &outfile)
|
||||||
|
{
|
||||||
|
// Handle endianness
|
||||||
|
for (unsigned int i = 0; i < num; ++i)
|
||||||
|
{
|
||||||
|
ioEdges[i] = htonl(ioEdges[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUG_OUTPUT
|
||||||
|
printf("writing %d edges\n", num);
|
||||||
|
for (int i = 0; i < num; i++)
|
||||||
|
{
|
||||||
|
#ifdef DEBUG_OUTPUT_L2
|
||||||
|
printf("ptr=%2d t=%d l=%d chr=%2d (%c)\n",
|
||||||
|
ioEdges[i].ptr, ioEdges[i].term, ioEdges[i].last,
|
||||||
|
ioEdges[i].chr, ioEdges[i].chr -1 +'a');
|
||||||
|
#endif
|
||||||
|
outfile.write((char*)(ioEdges + i), sizeof(DicEdge));
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
outfile.write((char*)ioEdges, num * sizeof(DicEdge));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#define MAX_STRING_LENGTH 200
|
||||||
|
|
||||||
|
|
||||||
|
#define MAX_EDGES 2000
|
||||||
|
/* ods3: ?? */
|
||||||
|
/* ods4: 1746 */
|
||||||
|
|
||||||
|
// Hashing function for a vector of DicEdge, based on the hashing function
|
||||||
|
// of the HashTable
|
||||||
|
struct HashVector
|
||||||
|
{
|
||||||
|
unsigned int operator()(const vector<DicEdge> &iKey) const
|
||||||
|
{
|
||||||
|
if (iKey.empty())
|
||||||
|
return 0;
|
||||||
|
return HashPtr(&iKey.front(), iKey.size() * sizeof(DicEdge));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef CHECK_RECURSION
|
||||||
|
class IncDec
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
IncDec(int &ioCounter)
|
||||||
|
: m_counter(ioCounter)
|
||||||
|
{
|
||||||
|
m_counter++;
|
||||||
|
}
|
||||||
|
|
||||||
|
~IncDec()
|
||||||
|
{
|
||||||
|
m_counter--;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
int &m_counter;
|
||||||
|
};
|
||||||
|
|
||||||
|
int current_rec = 0;
|
||||||
|
int max_rec = 0;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* global variables */
|
||||||
|
HashTable<vector<DicEdge>, unsigned int, HashVector> *global_hashtable;
|
||||||
|
|
||||||
|
wchar_t global_stringbuf[MAX_STRING_LENGTH]; /* Space for current string */
|
||||||
|
wchar_t* global_endstring; /* Marks END of current string */
|
||||||
|
const wchar_t* global_input;
|
||||||
|
const wchar_t* global_endofinput;
|
||||||
|
#ifdef CHECK_RECURSION
|
||||||
|
map<int, vector<DicEdge> > global_mapfordepth;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Makenode takes a prefix (as position relative to stringbuf) and
|
||||||
|
* returns an index of the start node of a dawg that recognizes all
|
||||||
|
* words beginning with that prefix. String is a pointer (relative
|
||||||
|
* to stringbuf) indicating how much of iPrefix is matched in the
|
||||||
|
* input.
|
||||||
|
* @param iPrefix: prefix to work on
|
||||||
|
* @param outfile: stream where to write the nodes
|
||||||
|
* @param ioHeaderInfo: information needed to build the final header, updated
|
||||||
|
* during the processing
|
||||||
|
* @param iHeader: temporary header, used only to do the conversion between
|
||||||
|
* the (wide) chars and their corresponding internal code
|
||||||
|
*/
|
||||||
|
unsigned int makenode(const wchar_t *iPrefix, ostream &outfile,
|
||||||
|
DictHeaderInfo &ioHeaderInfo, const Header &iHeader)
|
||||||
|
{
|
||||||
|
#ifdef CHECK_RECURSION
|
||||||
|
IncDec inc(current_rec);
|
||||||
|
if (current_rec > max_rec)
|
||||||
|
max_rec = current_rec;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef CHECK_RECURSION
|
||||||
|
// Instead of creating a vector, try to reuse an existing one
|
||||||
|
vector<DicEdge> &edges = global_mapfordepth[current_rec];
|
||||||
|
edges.reserve(MAX_EDGES);
|
||||||
|
edges.clear();
|
||||||
|
#else
|
||||||
|
vector<DicEdge> edges;
|
||||||
|
// Optimize allocation
|
||||||
|
edges.reserve(MAX_EDGES);
|
||||||
|
#endif
|
||||||
|
DicEdge newEdge;
|
||||||
|
|
||||||
|
while (iPrefix == global_endstring)
|
||||||
|
{
|
||||||
|
// More edges out of node
|
||||||
|
newEdge.ptr = 0;
|
||||||
|
newEdge.term = 0;
|
||||||
|
newEdge.last = 0;
|
||||||
|
newEdge.chr = iHeader.getCodeFromChar(*global_endstring++ = *global_input++);
|
||||||
|
edges.push_back(newEdge);
|
||||||
|
|
||||||
|
// End of a word?
|
||||||
|
if (*global_input == L'\n' || *global_input == L'\r')
|
||||||
|
{
|
||||||
|
ioHeaderInfo.nwords++;
|
||||||
|
*global_endstring = L'\0';
|
||||||
|
// Mark edge as word
|
||||||
|
edges.back().term = 1;
|
||||||
|
|
||||||
|
// Skip \r and/or \n
|
||||||
|
while (global_input != global_endofinput &&
|
||||||
|
(*global_input == L'\n' || *global_input == L'\r'))
|
||||||
|
{
|
||||||
|
++global_input;
|
||||||
|
}
|
||||||
|
// At the end of input?
|
||||||
|
if (global_input == global_endofinput)
|
||||||
|
break;
|
||||||
|
|
||||||
|
global_endstring = global_stringbuf;
|
||||||
|
while (*global_endstring == *global_input)
|
||||||
|
{
|
||||||
|
global_endstring++;
|
||||||
|
global_input++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Make dawg pointed to by this edge
|
||||||
|
edges.back().ptr =
|
||||||
|
makenode(iPrefix + 1, outfile, ioHeaderInfo, iHeader);
|
||||||
|
}
|
||||||
|
|
||||||
|
int numedges = edges.size();
|
||||||
|
if (numedges == 0)
|
||||||
|
{
|
||||||
|
// Special node zero - no edges
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark the last edge
|
||||||
|
edges.back().last = 1;
|
||||||
|
|
||||||
|
const unsigned int *saved_position = global_hashtable->find(edges);
|
||||||
|
if (saved_position)
|
||||||
|
{
|
||||||
|
ioHeaderInfo.edgessaved += numedges;
|
||||||
|
ioHeaderInfo.nodessaved++;
|
||||||
|
|
||||||
|
return *saved_position;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
unsigned int node_pos = ioHeaderInfo.edgesused;
|
||||||
|
global_hashtable->add(edges, ioHeaderInfo.edgesused);
|
||||||
|
ioHeaderInfo.edgesused += numedges;
|
||||||
|
ioHeaderInfo.nodesused++;
|
||||||
|
write_node(reinterpret_cast<uint32_t*>(&edges.front()),
|
||||||
|
numedges, outfile);
|
||||||
|
|
||||||
|
return node_pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void printUsage(const string &iBinaryName)
|
||||||
|
{
|
||||||
|
cout << "Usage: " << iBinaryName << " [options]" << endl
|
||||||
|
<< _("Mandatory options:") << endl
|
||||||
|
<< _(" -d, --dicname <string> Set the dictionary name and version") << endl
|
||||||
|
<< _(" -l, --letters <string> Path to the file containing the letters (see below)") << endl
|
||||||
|
<< _(" -i, --input <string> Path to the uncompressed dictionary file (encoded in UTF-8)") << endl
|
||||||
|
<< _(" -o, --output <string Path to the generated compressed dictionary file") << endl
|
||||||
|
<< _("Other options:") << endl
|
||||||
|
<< _(" -h, --help Print this help and exit") << endl
|
||||||
|
<< _("Example:") << endl
|
||||||
|
<< " " << iBinaryName << _(" -d 'ODS 5.0' -l letters.txt -i ods5.txt -o ods5.dawg") << endl
|
||||||
|
<< endl
|
||||||
|
<< _("The file containing the letters (--letters switch) must be UTF-8 encoded.") << endl
|
||||||
|
<< _("Each line corresponds to one letter, and must contain 5 fields separated with ") << endl
|
||||||
|
<< _("one or more space(s).") << endl
|
||||||
|
<< _(" - 1st field: the letter itself") << endl
|
||||||
|
<< _(" - 2nd field: the points of the letter") << endl
|
||||||
|
<< _(" - 3rd field: the frequency of the letter (how many letters of this kind in the game)") << endl
|
||||||
|
<< _(" - 4th field: 1 if the letter is considered as a vowel in Scrabble game, 0 otherwise") << endl
|
||||||
|
<< _(" - 5th field: 1 if the letter is considered as a consonant in Scrabble game, 0 otherwise") << endl
|
||||||
|
<< _("Example for french:") << endl
|
||||||
|
<< _("A 1 9 1 0") << endl
|
||||||
|
<< _("[...]") << endl
|
||||||
|
<< _("Z 10 1 0 1") << endl
|
||||||
|
<< _("? 0 2 1 1") << endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int main(int argc, char* argv[])
|
||||||
|
{
|
||||||
|
#if HAVE_SETLOCALE
|
||||||
|
// Set locale via LC_ALL
|
||||||
|
setlocale(LC_ALL, "");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
// Set the message domain
|
||||||
|
bindtextdomain(PACKAGE, LOCALEDIR);
|
||||||
|
textdomain(PACKAGE);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static const struct option long_options[] =
|
||||||
|
{
|
||||||
|
{"help", no_argument, NULL, 'h'},
|
||||||
|
{"dicname", required_argument, NULL, 'd'},
|
||||||
|
{"letters", required_argument, NULL, 'l'},
|
||||||
|
{"input", required_argument, NULL, 'i'},
|
||||||
|
{"output", required_argument, NULL, 'o'},
|
||||||
|
{0, 0, 0, 0}
|
||||||
|
};
|
||||||
|
static const char short_options[] = "hd:l:i:o:";
|
||||||
|
|
||||||
|
bool found_d = false;
|
||||||
|
bool found_l = false;
|
||||||
|
bool found_i = false;
|
||||||
|
bool found_o = false;
|
||||||
|
string inFileName;
|
||||||
|
string outFileName;
|
||||||
|
DictHeaderInfo headerInfo;
|
||||||
|
|
||||||
|
int res;
|
||||||
|
int option_index = 1;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while ((res = getopt_long(argc, argv, short_options,
|
||||||
|
long_options, &option_index)) != -1)
|
||||||
|
{
|
||||||
|
switch (res)
|
||||||
|
{
|
||||||
|
case 'h':
|
||||||
|
printUsage(argv[0]);
|
||||||
|
exit(0);
|
||||||
|
case 'd':
|
||||||
|
found_d = true;
|
||||||
|
headerInfo.dicName = convertToWc(optarg);
|
||||||
|
break;
|
||||||
|
case 'l':
|
||||||
|
found_l = true;
|
||||||
|
readLetters(optarg, headerInfo);
|
||||||
|
break;
|
||||||
|
case 'i':
|
||||||
|
found_i = true;
|
||||||
|
inFileName = optarg;
|
||||||
|
break;
|
||||||
|
case 'o':
|
||||||
|
found_o = true;
|
||||||
|
outFileName = optarg;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check mandatory options
|
||||||
|
if (!found_d || !found_l || !found_i || !found_o)
|
||||||
|
{
|
||||||
|
cerr << _("A mandatory option is missing") << endl;
|
||||||
|
printUsage(argv[0]);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct stat stat_buf;
|
||||||
|
if (stat(inFileName.c_str(), &stat_buf) < 0)
|
||||||
|
{
|
||||||
|
cerr << _("Cannot stat uncompressed dictionary ") << inFileName << endl;
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
unsigned int dicsize = (unsigned int)stat_buf.st_size;
|
||||||
|
|
||||||
|
ofstream outfile(outFileName.c_str(), ios::out | ios::binary | ios::trunc);
|
||||||
|
if (!outfile.is_open())
|
||||||
|
{
|
||||||
|
cerr << _("Cannot open output file ") << outFileName << endl;
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
clock_t startLoadTime = clock();
|
||||||
|
// FIXME: not exception safe
|
||||||
|
const wchar_t *uncompressed = load_uncompressed(inFileName, dicsize);
|
||||||
|
clock_t endLoadTime = clock();
|
||||||
|
|
||||||
|
global_input = uncompressed;
|
||||||
|
global_endofinput = global_input + dicsize;
|
||||||
|
|
||||||
|
#define SCALE 0.6
|
||||||
|
global_hashtable = new HashTable<vector<DicEdge>, unsigned int, HashVector>((unsigned int)(dicsize * SCALE));
|
||||||
|
#undef SCALE
|
||||||
|
|
||||||
|
headerInfo.dawg = true;
|
||||||
|
Header tempHeader = skip_init_header(outfile, headerInfo);
|
||||||
|
|
||||||
|
DicEdge specialnode = {0, 0, 0, 0};
|
||||||
|
specialnode.last = 1;
|
||||||
|
// Temporary variable to avoid a warning when compiling with -O2
|
||||||
|
// (there is no warning with -O0... g++ bug?)
|
||||||
|
DicEdge *tmpPtr = &specialnode;
|
||||||
|
write_node(reinterpret_cast<uint32_t*>(tmpPtr), 1, outfile);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Call makenode with null (relative to stringbuf) prefix;
|
||||||
|
* Initialize string to null; Put index of start node on output
|
||||||
|
*/
|
||||||
|
DicEdge rootnode = {0, 0, 0, 0};
|
||||||
|
global_endstring = global_stringbuf;
|
||||||
|
clock_t startBuildTime = clock();
|
||||||
|
rootnode.ptr = makenode(global_endstring, outfile, headerInfo, tempHeader);
|
||||||
|
clock_t endBuildTime = clock();
|
||||||
|
// Reuse the temporary variable
|
||||||
|
tmpPtr = &rootnode;
|
||||||
|
write_node(reinterpret_cast<uint32_t*>(tmpPtr), 1, outfile);
|
||||||
|
|
||||||
|
fix_header(outfile, headerInfo);
|
||||||
|
|
||||||
|
Header aHeader(headerInfo);
|
||||||
|
aHeader.print();
|
||||||
|
|
||||||
|
delete global_hashtable;
|
||||||
|
delete[] uncompressed;
|
||||||
|
outfile.close();
|
||||||
|
|
||||||
|
printf(_(" Load time: %.3f s\n"), 1.0 * (endLoadTime - startLoadTime) / CLOCKS_PER_SEC);
|
||||||
|
printf(_(" Compression time: %.3f s\n"), 1.0 * (endBuildTime - startBuildTime) / CLOCKS_PER_SEC);
|
||||||
|
#ifdef CHECK_RECURSION
|
||||||
|
printf(_(" Maximum recursion level reached: %d\n"), max_rec);
|
||||||
|
#endif
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
catch (std::exception &e)
|
||||||
|
{
|
||||||
|
cerr << e.what() << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
282
dic/dic.c
282
dic/dic.c
|
@ -1,282 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file dic.c
|
|
||||||
* \brief Dawg dictionary
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2002
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <ctype.h>
|
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include "dic_internals.h"
|
|
||||||
#include "dic.h"
|
|
||||||
|
|
||||||
#define __UNUSED__ __attribute__((unused))
|
|
||||||
|
|
||||||
#if defined(WORDS_BIGENDIAN)
|
|
||||||
static uint32_t swap4(uint32_t v)
|
|
||||||
{
|
|
||||||
uint32_t r;
|
|
||||||
uint8_t *pv,*pr;
|
|
||||||
|
|
||||||
pv = (uint8_t*)&v;
|
|
||||||
pr = (uint8_t*)&r;
|
|
||||||
|
|
||||||
pr[0] = pv[3];
|
|
||||||
pr[1] = pv[2];
|
|
||||||
pr[2] = pv[1];
|
|
||||||
pr[3] = pv[0];
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static int
|
|
||||||
Dic_read_convert_header(Dict_header *header, FILE* file)
|
|
||||||
{
|
|
||||||
|
|
||||||
if (fread(header,sizeof(Dict_header),1,file) != 1)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
#if defined(WORDS_BIGENDIAN)
|
|
||||||
header->root = swap4(header->root);
|
|
||||||
header->nwords = swap4(header->nwords);
|
|
||||||
header->nodesused = swap4(header->nodesused);
|
|
||||||
header->edgesused = swap4(header->edgesused);
|
|
||||||
header->nodessaved = swap4(header->nodessaved);
|
|
||||||
header->edgessaved = swap4(header->edgessaved);
|
|
||||||
#else
|
|
||||||
|
|
||||||
#endif
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
Dic_check_header(Dict_header *header, const char *path)
|
|
||||||
{
|
|
||||||
int r;
|
|
||||||
FILE* file;
|
|
||||||
if ((file = fopen(path,"rb")) == NULL)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
r = Dic_read_convert_header(header,file);
|
|
||||||
fclose(file);
|
|
||||||
|
|
||||||
return r || strcmp(header->ident,_COMPIL_KEYWORD_);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_convert_data_to_arch(Dictionary __UNUSED__ dic)
|
|
||||||
{
|
|
||||||
#if defined(WORDS_BIGENDIAN)
|
|
||||||
int i;
|
|
||||||
uint32_t* p;
|
|
||||||
p = (uint32_t*)dic->dawg;
|
|
||||||
for(i=0; i < (dic->nedges + 1); i++)
|
|
||||||
{
|
|
||||||
p[i] = swap4(p[i]);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
Dic_load(Dictionary *dic, const char* path)
|
|
||||||
{
|
|
||||||
FILE* file;
|
|
||||||
Dict_header header;
|
|
||||||
|
|
||||||
|
|
||||||
*dic = NULL;
|
|
||||||
if ((file = fopen(path,"rb")) == NULL)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
Dic_read_convert_header(&header,file);
|
|
||||||
|
|
||||||
if ((*dic = (Dictionary) malloc(sizeof(struct _Dictionary))) == NULL)
|
|
||||||
return 3;
|
|
||||||
|
|
||||||
if (((*dic)->dawg = (Dawg_edge*)malloc((header.edgesused + 1)*sizeof(Dawg_edge))) == NULL)
|
|
||||||
{
|
|
||||||
free(*dic);
|
|
||||||
*dic = NULL;
|
|
||||||
return 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fread((*dic)->dawg,sizeof(Dawg_edge),header.edgesused + 1,file) !=
|
|
||||||
(header.edgesused + 1))
|
|
||||||
{
|
|
||||||
free((*dic)->dawg);
|
|
||||||
free(*dic);
|
|
||||||
*dic = NULL;
|
|
||||||
return 5;
|
|
||||||
}
|
|
||||||
|
|
||||||
(*dic)->root = header.root;
|
|
||||||
(*dic)->nwords = header.nwords;
|
|
||||||
(*dic)->nnodes = header.nodesused;
|
|
||||||
(*dic)->nedges = header.edgesused;
|
|
||||||
|
|
||||||
Dic_convert_data_to_arch(*dic);
|
|
||||||
|
|
||||||
fclose(file);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
Dic_destroy(Dictionary dic)
|
|
||||||
{
|
|
||||||
if (dic != NULL)
|
|
||||||
{
|
|
||||||
if (dic->dawg != NULL)
|
|
||||||
free(dic->dawg);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
free(dic);
|
|
||||||
return 2;
|
|
||||||
}
|
|
||||||
free(dic);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
dic_elt_t
|
|
||||||
Dic_next(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
if (! Dic_last(d,e))
|
|
||||||
return e+1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
dic_elt_t
|
|
||||||
Dic_succ(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
return (d->dawg[e]).ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
dic_elt_t
|
|
||||||
Dic_root(Dictionary d)
|
|
||||||
{
|
|
||||||
return d->root;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
dic_code_t
|
|
||||||
Dic_chr(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
return (dic_code_t)(d->dawg[e]).chr;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
Dic_last(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
return (d->dawg[e]).last;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
Dic_word(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
return (d->dawg[e]).term;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned int
|
|
||||||
Dic_lookup(Dictionary d, dic_elt_t root, dic_code_t* s)
|
|
||||||
{
|
|
||||||
unsigned int p;
|
|
||||||
begin:
|
|
||||||
if (! *s)
|
|
||||||
return root;
|
|
||||||
if (! Dic_succ(d, root))
|
|
||||||
return 0;
|
|
||||||
p = Dic_succ(d, root);
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (Dic_chr(d, p) == *s)
|
|
||||||
{
|
|
||||||
root = p;
|
|
||||||
s++;
|
|
||||||
goto begin;
|
|
||||||
}
|
|
||||||
else if (Dic_last(d, p))
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
p = Dic_next(d, p);
|
|
||||||
} while (1);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* **************************************************************************** */
|
|
||||||
/* **************************************************************************** */
|
|
||||||
/* **************************************************************************** */
|
|
||||||
/* **************************************************************************** */
|
|
||||||
|
|
||||||
char
|
|
||||||
Dic_char(Dictionary d, dic_elt_t e)
|
|
||||||
{
|
|
||||||
char c = (d->dawg[e]).chr;
|
|
||||||
if (c)
|
|
||||||
return c + 'A' - 1;
|
|
||||||
else
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned int
|
|
||||||
Dic_char_lookup(Dictionary d, dic_elt_t root, char* s)
|
|
||||||
{
|
|
||||||
unsigned int p;
|
|
||||||
begin:
|
|
||||||
if (! *s)
|
|
||||||
return root;
|
|
||||||
if (! Dic_succ(d, root))
|
|
||||||
return 0;
|
|
||||||
p = Dic_succ(d, root);
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (Dic_char(d, p) == *s)
|
|
||||||
{
|
|
||||||
root = p;
|
|
||||||
s++;
|
|
||||||
goto begin;
|
|
||||||
}
|
|
||||||
else if (Dic_last(d, p))
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
p = Dic_next(d, p);
|
|
||||||
} while (1);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
276
dic/dic.cpp
Normal file
276
dic/dic.cpp
Normal file
|
@ -0,0 +1,276 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2002-2007 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file dic.c
|
||||||
|
* \brief Dawg dictionary
|
||||||
|
* \author Antoine Fraboulet & Olivier Teuliere
|
||||||
|
* \date 2002
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cctype>
|
||||||
|
|
||||||
|
// For ntohl & Co.
|
||||||
|
#ifdef WIN32
|
||||||
|
# include <winsock2.h>
|
||||||
|
#else
|
||||||
|
# if HAVE_NETINET_IN_H
|
||||||
|
# include <netinet/in.h>
|
||||||
|
# endif
|
||||||
|
# if HAVE_ARPA_INET_H
|
||||||
|
# include <arpa/inet.h>
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "dic.h"
|
||||||
|
#include "header.h"
|
||||||
|
#include "dic_exception.h"
|
||||||
|
#include "dic_internals.h"
|
||||||
|
#include "tile.h"
|
||||||
|
|
||||||
|
|
||||||
|
const Dictionary *Dictionary::m_dic = NULL;
|
||||||
|
|
||||||
|
|
||||||
|
// Note: duplicated in header.cpp
|
||||||
|
#if defined(WORDS_BIGENDIAN)
|
||||||
|
static uint32_t swap4(uint32_t v)
|
||||||
|
{
|
||||||
|
uint32_t r;
|
||||||
|
uint8_t *pv = (uint8_t*)&v;
|
||||||
|
uint8_t *pr = (uint8_t*)&r;
|
||||||
|
|
||||||
|
pr[0] = pv[3];
|
||||||
|
pr[1] = pv[2];
|
||||||
|
pr[2] = pv[1];
|
||||||
|
pr[3] = pv[0];
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
Dictionary::Dictionary(const string &iPath)
|
||||||
|
: m_dawg(NULL)
|
||||||
|
{
|
||||||
|
ifstream file(iPath.c_str(), ios::in | ios::binary);
|
||||||
|
|
||||||
|
if (!file.is_open())
|
||||||
|
throw DicException("Cannot open " + iPath);
|
||||||
|
|
||||||
|
// XXX: we should protect these allocations with auto_ptr
|
||||||
|
m_header = new Header(file);
|
||||||
|
m_dawg = new uint32_t[m_header->getNbEdgesUsed() + 1];
|
||||||
|
|
||||||
|
streamsize toRead = (m_header->getNbEdgesUsed() + 1) * sizeof(uint32_t);
|
||||||
|
file.read((char*)m_dawg, toRead);
|
||||||
|
if (file.gcount() != toRead)
|
||||||
|
{
|
||||||
|
delete[] m_dawg;
|
||||||
|
delete m_header;
|
||||||
|
throw DicException("Problem reading dictionary arcs");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle endianness
|
||||||
|
convertDataToArch();
|
||||||
|
|
||||||
|
initializeTiles();
|
||||||
|
|
||||||
|
// Concatenate the uppercase and lowercase letters
|
||||||
|
wstring lower = m_header->getLetters();
|
||||||
|
std::transform(lower.begin(), lower.end(), lower.begin(), towlower);
|
||||||
|
m_allLetters = m_header->getLetters() + lower;
|
||||||
|
|
||||||
|
m_dic = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Dictionary::~Dictionary()
|
||||||
|
{
|
||||||
|
delete[] m_dawg;
|
||||||
|
delete m_header;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::convertDataToArch()
|
||||||
|
{
|
||||||
|
if (m_header->getVersion() == 0)
|
||||||
|
{
|
||||||
|
#if defined(WORDS_BIGENDIAN)
|
||||||
|
for (unsigned int i = 0; i < (m_header->getNbEdgesUsed() + 1); i++)
|
||||||
|
{
|
||||||
|
m_dawg[i] = swap4(m_dawg[i]);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (unsigned int i = 0; i < (m_header->getNbEdgesUsed() + 1); i++)
|
||||||
|
{
|
||||||
|
m_dawg[i] = ntohl(m_dawg[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::initializeTiles()
|
||||||
|
{
|
||||||
|
// "Activate" the dictionary by giving the header to the Tile class
|
||||||
|
Tile::SetHeader(*m_header);
|
||||||
|
|
||||||
|
// XXX: temp
|
||||||
|
Tile::m_TheJoker = Tile(Tile::kTILE_JOKER);
|
||||||
|
|
||||||
|
m_tilesVect.reserve(m_header->getLetters().size() + 1);
|
||||||
|
// Create a tile for each letter in the dictionary header
|
||||||
|
for (unsigned int i = 0; i < m_header->getLetters().size(); ++i)
|
||||||
|
m_tilesVect.push_back(Tile(m_header->getLetters()[i]));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Dictionary::validateLetters(const wstring &iLetters,
|
||||||
|
const wstring &iAccepted) const
|
||||||
|
{
|
||||||
|
return iLetters.empty()
|
||||||
|
|| iLetters.find_first_not_of(m_allLetters + iAccepted) == string::npos;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const dic_elt_t Dictionary::getNext(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
if (!isLast(e))
|
||||||
|
return e + 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const dic_elt_t Dictionary::getSucc(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
if (m_header->getVersion() == 0)
|
||||||
|
return reinterpret_cast<const DicEdgeOld*>(m_dawg + e)->ptr;
|
||||||
|
else
|
||||||
|
return reinterpret_cast<const DicEdge*>(m_dawg + e)->ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const dic_elt_t Dictionary::getRoot() const
|
||||||
|
{
|
||||||
|
return m_header->getRoot();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const dic_code_t Dictionary::getCode(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
if (m_header->getVersion() == 0)
|
||||||
|
return reinterpret_cast<const DicEdgeOld*>(m_dawg + e)->chr;
|
||||||
|
else
|
||||||
|
return reinterpret_cast<const DicEdge*>(m_dawg + e)->chr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wchar_t Dictionary::getChar(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
return m_header->getCharFromCode(getCode(e));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Dictionary::isLast(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
if (m_header->getVersion() == 0)
|
||||||
|
return reinterpret_cast<const DicEdgeOld*>(m_dawg + e)->last;
|
||||||
|
else
|
||||||
|
return reinterpret_cast<const DicEdge*>(m_dawg + e)->last;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Dictionary::isEndOfWord(const dic_elt_t &e) const
|
||||||
|
{
|
||||||
|
if (m_header->getVersion() == 0)
|
||||||
|
return reinterpret_cast<const DicEdgeOld*>(m_dawg + e)->term;
|
||||||
|
else
|
||||||
|
return reinterpret_cast<const DicEdge*>(m_dawg + e)->term;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Dictionary::lookup(const dic_elt_t &root, const dic_code_t *s) const
|
||||||
|
{
|
||||||
|
unsigned int p;
|
||||||
|
dic_elt_t rootCopy = root;
|
||||||
|
begin:
|
||||||
|
if (! *s)
|
||||||
|
return rootCopy;
|
||||||
|
if (! getSucc(rootCopy))
|
||||||
|
return 0;
|
||||||
|
p = getSucc(rootCopy);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (getCode(p) == *s)
|
||||||
|
{
|
||||||
|
rootCopy = p;
|
||||||
|
s++;
|
||||||
|
goto begin;
|
||||||
|
}
|
||||||
|
else if (isLast( p))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
p = getNext(p);
|
||||||
|
} while (1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Dictionary::charLookup(const dic_elt_t &iRoot, const wchar_t *s) const
|
||||||
|
{
|
||||||
|
unsigned int p;
|
||||||
|
dic_elt_t rootCopy = iRoot;
|
||||||
|
begin:
|
||||||
|
if (! *s)
|
||||||
|
return rootCopy;
|
||||||
|
if (! getSucc(rootCopy))
|
||||||
|
return 0;
|
||||||
|
p = getSucc(rootCopy);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (getChar(p) == *s)
|
||||||
|
{
|
||||||
|
rootCopy = p;
|
||||||
|
s++;
|
||||||
|
goto begin;
|
||||||
|
}
|
||||||
|
else if (isLast(p))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
p = getNext(p);
|
||||||
|
} while (1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
287
dic/dic.h
287
dic/dic.h
|
@ -1,119 +1,153 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file dic.h
|
* \file dic.h
|
||||||
* \brief Dawg dictionary
|
* \brief Dawg dictionary
|
||||||
* \author Antoine Fraboulet
|
* \author Antoine Fraboulet & Olivier Teuliere
|
||||||
* \date 2002
|
* \date 2002
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _DIC_H_
|
#ifndef _DIC_H_
|
||||||
#define _DIC_H_
|
#define _DIC_H_
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
#include <string>
|
||||||
* different letters in the dictionary
|
#include <vector>
|
||||||
*/
|
#include <map>
|
||||||
#define DIC_LETTERS 27
|
|
||||||
|
#include "tile.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* max length of words (including last \0)
|
* max length of words (including last \0)
|
||||||
*/
|
*/
|
||||||
#define DIC_WORD_MAX 16
|
#define DIC_WORD_MAX 16
|
||||||
|
|
||||||
typedef struct _Dict_header Dict_header;
|
class Header;
|
||||||
typedef struct _Dictionary *Dictionary;
|
|
||||||
typedef unsigned int dic_elt_t;
|
typedef unsigned int dic_elt_t;
|
||||||
typedef unsigned char dic_code_t;
|
typedef unsigned char dic_code_t;
|
||||||
|
struct params_cross_t;
|
||||||
|
struct params_7plus1_t;
|
||||||
|
struct params_regexp_t;
|
||||||
|
struct search_RegE_list_t;
|
||||||
|
|
||||||
|
class Dictionary
|
||||||
/**
|
{
|
||||||
* Dictionary header loading from a file
|
public:
|
||||||
* @param dic : pointer to a header
|
|
||||||
* @param path : compressed dictionary path
|
|
||||||
* @return 0 ok, otherwise error
|
|
||||||
*/
|
|
||||||
int Dic_check_header(Dict_header *header, const char* path);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dictionary creation and loading from a file
|
* Dictionary creation and loading from a file
|
||||||
* @param dic : pointer to a dictionary
|
* @param path: compressed dictionary path
|
||||||
* @param path : compressed dictionary path
|
|
||||||
* @return 0 ok, 1 error
|
|
||||||
*/
|
*/
|
||||||
int Dic_load (Dictionary* dic,const char* path);
|
Dictionary(const string &path);
|
||||||
|
|
||||||
|
/// Destructor
|
||||||
|
~Dictionary();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Destroy a dictionary
|
* Return the current instance of the dictionary object
|
||||||
|
* XXX: This is ugly, but I don't see any clean way apart from carrying
|
||||||
|
* a reference to a Dictionary object in many places...
|
||||||
|
* Other more or less ugly options:
|
||||||
|
* - Make the dictionary a singleton (2 dictionaries cannot coexist...)
|
||||||
|
* - Make many classes inherit from a common base class with a dictionary
|
||||||
|
* member (possibly bad for performances)
|
||||||
|
* A new created dictionary replaces the previous instance, even if the
|
||||||
|
* previous instance is not destroyed yet
|
||||||
|
* If no dictionary object is instanciated when this method is called,
|
||||||
|
* it will probably crash...
|
||||||
*/
|
*/
|
||||||
int Dic_destroy(Dictionary dic);
|
static const Dictionary& GetDic() { return *m_dic; }
|
||||||
|
|
||||||
|
/** Give access to the dictionary header */
|
||||||
|
const Header& getHeader() const { return *m_header; }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dic_chr returns the character code associated with an element,
|
* Check whether all the given letters are present in the dictionary,
|
||||||
|
* or are one of the other accepted letters.
|
||||||
|
* Return true if this is the case, false otherwise
|
||||||
|
*/
|
||||||
|
bool validateLetters(const wstring &iLetters,
|
||||||
|
const wstring &iAccepted = L"") const;
|
||||||
|
|
||||||
|
/** Return a vector containing one of each possible tile */
|
||||||
|
const vector<Tile>& getAllTiles() const { return m_tilesVect; }
|
||||||
|
|
||||||
|
/** Return the number of different tiles (including the joker) */
|
||||||
|
unsigned int getTileNumber() const { return m_tilesVect.size(); }
|
||||||
|
|
||||||
|
/** Return a tile from its code */
|
||||||
|
const Tile &getTileFromCode(unsigned int iCode) const { return m_tilesVect[iCode - 1]; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the character code associated with an element,
|
||||||
* codes may range from 0 to 31. 0 is the null character.
|
* codes may range from 0 to 31. 0 is the null character.
|
||||||
* @returns code for the encoded character
|
* @returns code for the encoded character
|
||||||
*/
|
*/
|
||||||
dic_code_t Dic_chr (Dictionary dic, dic_elt_t elt);
|
const dic_code_t getCode(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the wide character associated with an element.
|
||||||
|
* @returns wide character for the element
|
||||||
|
*/
|
||||||
|
wchar_t getChar(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a boolean to show if there is another available
|
* Returns a boolean to show if there is another available
|
||||||
* character in the current depth (a neighbor in the tree)
|
* character in the current depth (a neighbor in the tree)
|
||||||
* @returns 0 or 1 (true)
|
* @returns 0 or 1 (true)
|
||||||
*/
|
*/
|
||||||
int Dic_last(Dictionary dic, dic_elt_t elt);
|
bool isLast(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a boolean to show if we are at the end of a word
|
* Returns a boolean to show if we are at the end of a word
|
||||||
* (see Dic_next)
|
* (see getNext)
|
||||||
* @returns 0 or 1 (true)
|
* @returns 0 or 1 (true)
|
||||||
*/
|
*/
|
||||||
int Dic_word(Dictionary dic, dic_elt_t elt);
|
bool isEndOfWord(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the root of the dictionary
|
* Returns the root of the dictionary
|
||||||
* @returns root element
|
* @returns root element
|
||||||
*/
|
*/
|
||||||
dic_elt_t Dic_root(Dictionary dic);
|
const dic_elt_t getRoot() const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the next available neighbor (see Dic_last)
|
* Returns the next available neighbor (see getLast)
|
||||||
* @returns next dictionary element at the same depth
|
* @returns next dictionary element at the same depth
|
||||||
*/
|
*/
|
||||||
dic_elt_t Dic_next(Dictionary dic, dic_elt_t elt);
|
const dic_elt_t getNext(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the first element available at the next depth
|
* Returns the first element available at the next depth
|
||||||
* in the dictionary
|
* in the dictionary
|
||||||
* @params dic : dictionary
|
|
||||||
* @params elt : current dictionary element
|
* @params elt : current dictionary element
|
||||||
* @returns next element (successor)
|
* @returns next element (successor)
|
||||||
*/
|
*/
|
||||||
dic_elt_t Dic_succ(Dictionary dic, dic_elt_t elt);
|
const dic_elt_t getSucc(const dic_elt_t &elt) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find the dictionary element matching the pattern starting
|
* Find the dictionary element matching the pattern starting
|
||||||
* from the given root node by walking the dictionary tree
|
* from the given root node by walking the dictionary tree
|
||||||
* @params dic : valid dictionary
|
|
||||||
* @params root : starting dictionary node for the search
|
* @params root : starting dictionary node for the search
|
||||||
* @params pattern : string encoded according to the dictionary codes,
|
* @params pattern : string encoded according to the dictionary codes,
|
||||||
* the pattern must be null ('\0') terminated
|
* the pattern must be null ('\0') terminated
|
||||||
|
@ -121,19 +155,11 @@ dic_elt_t Dic_succ(Dictionary dic, dic_elt_t elt);
|
||||||
* element that results from walking the dictionary according to the
|
* element that results from walking the dictionary according to the
|
||||||
* pattern
|
* pattern
|
||||||
*/
|
*/
|
||||||
unsigned int Dic_lookup(Dictionary dic, dic_elt_t root, dic_code_t* pattern);
|
unsigned int lookup(const dic_elt_t &root, const dic_code_t *pattern) const;
|
||||||
|
|
||||||
/**
|
|
||||||
* Dic_char returns the character associated with an element
|
|
||||||
* (in the range ['A'-'Z']), or the null character ('\0').
|
|
||||||
* @returns ASCII code for the character
|
|
||||||
*/
|
|
||||||
char Dic_char (Dictionary dic, dic_elt_t elt);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find the dictionary element matching the pattern starting
|
* Find the dictionary element matching the pattern starting
|
||||||
* from the given root node by walking the dictionary tree
|
* from the given root node by walking the dictionary tree
|
||||||
* @params dic : valid dictionary
|
|
||||||
* @params root : starting dictionary node for the search
|
* @params root : starting dictionary node for the search
|
||||||
* @params pattern : string made of uppercase characters in the range
|
* @params pattern : string made of uppercase characters in the range
|
||||||
* ['A'-'Z']. The pattern must be null ('\0') terminated
|
* ['A'-'Z']. The pattern must be null ('\0') terminated
|
||||||
|
@ -141,11 +167,136 @@ char Dic_char (Dictionary dic, dic_elt_t elt);
|
||||||
* element that results from walking the dictionary according to the
|
* element that results from walking the dictionary according to the
|
||||||
* pattern
|
* pattern
|
||||||
*/
|
*/
|
||||||
unsigned int Dic_char_lookup(Dictionary dic, dic_elt_t root, char* pattern);
|
unsigned int charLookup(const dic_elt_t &iRoot, const wchar_t *iPattern) const;
|
||||||
|
|
||||||
|
/// Getter for the edge at the given position
|
||||||
|
const uint32_t *getEdgeAt(const dic_elt_t &iElt) const { return m_dawg + iElt; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for a word in the dictionary
|
||||||
|
* @param iWord: lookup word
|
||||||
|
* @return true if the word is valid, false otherwise
|
||||||
|
*/
|
||||||
|
bool searchWord(const wstring &iWord) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for benjamins
|
||||||
|
* @param iWord: letters
|
||||||
|
* @param oWordList: results
|
||||||
|
*/
|
||||||
|
void searchBenj(const wstring &iWord, list<wstring> &oWordList) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for all words feasible by adding a letter in front or at the end
|
||||||
|
* @param iWord: word
|
||||||
|
* @param oWordList: results
|
||||||
|
*/
|
||||||
|
void searchRacc(const wstring &iWord, list<wstring> &oWordList) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for crosswords
|
||||||
|
* @param iMask: letters
|
||||||
|
* @param oWordList: results
|
||||||
|
*/
|
||||||
|
void searchCross(const wstring &iMask, list<wstring> &oWordList) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for all feasible word with "rack" plus one letter
|
||||||
|
* @param iRack: letters
|
||||||
|
* @param oWordlist: results
|
||||||
|
* @param joker: true if the search must be performed when a joker is in the rack
|
||||||
|
*/
|
||||||
|
void search7pl1(const wstring &iRack,
|
||||||
|
map<wchar_t, list<wstring> > &oWordList,
|
||||||
|
bool joker) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search for words matching a regular expression
|
||||||
|
* @param iRegexp: regular expression
|
||||||
|
* @param oWordList: results
|
||||||
|
* @param iList: parameters for the search (?)
|
||||||
|
*/
|
||||||
|
void searchRegExp(const wstring &iRegexp,
|
||||||
|
list<wstring> &oWordList,
|
||||||
|
struct search_RegE_list_t *iList) const;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Prevent from copying the dictionary!
|
||||||
|
Dictionary &operator=(const Dictionary&);
|
||||||
|
Dictionary(const Dictionary&);
|
||||||
|
|
||||||
|
Header *m_header;
|
||||||
|
uint32_t *m_dawg;
|
||||||
|
|
||||||
|
/** Letters of the dictionary, both in uppercase and lowercase */
|
||||||
|
wstring m_allLetters;
|
||||||
|
|
||||||
|
/// Vector of available tiles
|
||||||
|
vector<Tile> m_tilesVect;
|
||||||
|
|
||||||
|
static const Dictionary *m_dic;
|
||||||
|
|
||||||
|
void convertDataToArch();
|
||||||
|
void initializeTiles();
|
||||||
|
|
||||||
|
/// Template getter for the edge at the given position
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
const DAWG_EDGE * getEdgeAt(const dic_elt_t &iElt) const
|
||||||
|
{
|
||||||
|
return reinterpret_cast<const DAWG_EDGE*>(m_dawg + iElt);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Walk the dictionary until the end of the word
|
||||||
|
* @param s: current pointer to letters
|
||||||
|
* @param eptr: current edge in the dawg
|
||||||
|
*/
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
const DAWG_EDGE * seekEdgePtr(const wchar_t *s, const DAWG_EDGE *eptr) const;
|
||||||
|
|
||||||
|
/// Helper for searchBenj()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void searchBenjTempl(const wstring &iWord, list<wstring> &oWordList) const;
|
||||||
|
|
||||||
|
/// Helper for searchRacc()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void searchRaccTempl(const wstring &iWord, list<wstring> &oWordList) const;
|
||||||
|
|
||||||
|
/// Helper for searchCross()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void searchCrossRecTempl(struct params_cross_t *params,
|
||||||
|
list<wstring> &oWordList,
|
||||||
|
const DAWG_EDGE *edgeptr) const;
|
||||||
|
|
||||||
|
/// Helper for search7pl1()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void search7pl1Templ(const wstring &iRack,
|
||||||
|
map<wchar_t, list<wstring> > &oWordList,
|
||||||
|
bool joker) const;
|
||||||
|
|
||||||
|
/// Second helper for search7pl1()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void searchWordByLen(struct params_7plus1_t *params,
|
||||||
|
int i, const DAWG_EDGE *edgeptr) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal version of searchRegExp, needed until
|
||||||
|
* wide chars are supported by our regexp engine.
|
||||||
|
*/
|
||||||
|
void searchRegExpInner(const string &iRegexp,
|
||||||
|
list<string> &oWordList,
|
||||||
|
struct search_RegE_list_t *iList) const;
|
||||||
|
|
||||||
|
/// Helper for searchRegExp()
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void searchRegexpRecTempl(struct params_regexp_t *params,
|
||||||
|
int state,
|
||||||
|
const DAWG_EDGE *edgeptr,
|
||||||
|
list<string> &oWordList) const;
|
||||||
|
};
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _DIC_H_ */
|
#endif /* _DIC_H_ */
|
||||||
|
|
||||||
/// Local Variables:
|
/// Local Variables:
|
||||||
|
|
36
dic/dic_exception.cpp
Normal file
36
dic/dic_exception.cpp
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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 "dic_exception.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
DicException::DicException(const string &iMessage)
|
||||||
|
: m_message(iMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const char *DicException::what() const throw()
|
||||||
|
{
|
||||||
|
return m_message.c_str();
|
||||||
|
}
|
||||||
|
|
44
dic/dic_exception.h
Normal file
44
dic/dic_exception.h
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef _DIC_EXCEPTION_H_
|
||||||
|
#define _DIC_EXCEPTION_H_
|
||||||
|
|
||||||
|
#include <exception>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exception class for the dictionary.
|
||||||
|
* It simply inherits from the standard exception and overrides
|
||||||
|
* its what() method.
|
||||||
|
*/
|
||||||
|
class DicException: public std::exception
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
DicException(const std::string &iMessage);
|
||||||
|
~DicException() throw() {}
|
||||||
|
virtual const char *what() const throw();
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string m_message;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,21 +1,22 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2002-2007 Antoine Fraboulet
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
*
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
* 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 */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* (at your option) any later version. */
|
* (at your option) any later version.
|
||||||
/* */
|
*
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
/* GNU General Public License for more details. */
|
* GNU General Public License for more details.
|
||||||
/* */
|
*
|
||||||
/* You should have received a copy of the GNU General Public License */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* along with this program; if not, write to the Free Software */
|
* along with this program; if not, write to the Free Software
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file dic_internals.h
|
* \file dic_internals.h
|
||||||
|
@ -26,26 +27,10 @@
|
||||||
|
|
||||||
#ifndef _DIC_INTERNALS_H_
|
#ifndef _DIC_INTERNALS_H_
|
||||||
#define _DIC_INTERNALS_H_
|
#define _DIC_INTERNALS_H_
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
/**
|
|
||||||
* bit masking for ascii characters \n
|
|
||||||
* ('a' & CHAR) == ('A' & CHAR) == 1
|
|
||||||
*/
|
|
||||||
#define DIC_CHAR_MASK 0x1F
|
|
||||||
|
|
||||||
/**
|
|
||||||
* keyword included in dictionary headers
|
|
||||||
* implies little endian storage on words
|
|
||||||
*/
|
|
||||||
#define _COMPIL_KEYWORD_ "_COMPILED_DICTIONARY_"
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* structure of a compressed dictionary \n
|
* structure of a compressed dictionary \n
|
||||||
* \n
|
* \n
|
||||||
|
@ -60,53 +45,36 @@ extern "C"
|
||||||
* ----------------
|
* ----------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#if defined(WORDS_BIGENDIAN)
|
struct __attribute__ ((packed)) DicEdgeOld
|
||||||
struct __attribute__ ((packed)) _Dawg_edge {
|
|
||||||
uint32_t
|
|
||||||
chr : 5,
|
|
||||||
fill : 1,
|
|
||||||
last : 1,
|
|
||||||
term : 1,
|
|
||||||
ptr : 24;
|
|
||||||
};
|
|
||||||
#else
|
|
||||||
struct __attribute__ ((packed)) _Dawg_edge {
|
|
||||||
uint32_t
|
|
||||||
ptr : 24,
|
|
||||||
term : 1,
|
|
||||||
last : 1,
|
|
||||||
fill : 1,
|
|
||||||
chr : 5;
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
typedef struct _Dawg_edge Dawg_edge;
|
|
||||||
|
|
||||||
|
|
||||||
struct _Dict_header {
|
|
||||||
char ident[sizeof(_COMPIL_KEYWORD_)];
|
|
||||||
char unused_1;
|
|
||||||
char unused_2;
|
|
||||||
int32_t root;
|
|
||||||
int32_t nwords;
|
|
||||||
uint32_t edgesused;
|
|
||||||
uint32_t nodesused;
|
|
||||||
uint32_t nodessaved;
|
|
||||||
uint32_t edgessaved;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
struct _Dictionary
|
|
||||||
{
|
{
|
||||||
Dawg_edge *dawg;
|
public:
|
||||||
unsigned int root;
|
uint32_t
|
||||||
int nwords;
|
ptr : 24,
|
||||||
int nnodes;
|
term: 1,
|
||||||
int nedges;
|
last: 1,
|
||||||
|
fill: 1,
|
||||||
|
chr : 5;
|
||||||
|
bool operator==(const DicEdgeOld &iOther) const
|
||||||
|
{
|
||||||
|
return memcmp(this, &iOther, sizeof(*this)) == 0;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
struct __attribute__ ((packed)) DicEdge
|
||||||
#endif
|
{
|
||||||
|
public:
|
||||||
|
uint32_t
|
||||||
|
ptr : 24,
|
||||||
|
term: 1,
|
||||||
|
last: 1,
|
||||||
|
chr : 6;
|
||||||
|
bool operator==(const DicEdge &iOther) const
|
||||||
|
{
|
||||||
|
return memcmp(this, &iOther, sizeof(*this)) == 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
#endif /* _DIC_INTERNALS_H */
|
#endif /* _DIC_INTERNALS_H */
|
||||||
|
|
||||||
|
|
690
dic/dic_search.c
690
dic/dic_search.c
|
@ -1,690 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file dic_search.c
|
|
||||||
* \brief Dictionary lookup functions
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2002
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <ctype.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <wchar.h>
|
|
||||||
|
|
||||||
#include "dic_internals.h"
|
|
||||||
#include "dic.h"
|
|
||||||
#include "regexp.h"
|
|
||||||
#include "dic_search.h"
|
|
||||||
#include "libdic_a-ery.h" /* generated by bison */
|
|
||||||
#include "libdic_a-erl.h" /* generated by flex */
|
|
||||||
#include "automaton.h"
|
|
||||||
|
|
||||||
/**
|
|
||||||
* function prototype for bison generated parser
|
|
||||||
*/
|
|
||||||
int regexpparse(yyscan_t scanner, NODE** root,
|
|
||||||
struct search_RegE_list_t *list,
|
|
||||||
struct regexp_error_report_t *err);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Dic_seel_edgeptr
|
|
||||||
* walk the dictionary until the end of the word
|
|
||||||
* @param dic : dictionnary
|
|
||||||
* @param s : current pointer to letters
|
|
||||||
* @param eptr : current edge in the dawg
|
|
||||||
*/
|
|
||||||
static Dawg_edge*
|
|
||||||
Dic_seek_edgeptr(const Dictionary dic, const char* s, Dawg_edge *eptr)
|
|
||||||
{
|
|
||||||
if (*s)
|
|
||||||
{
|
|
||||||
Dawg_edge *p = dic->dawg + eptr->ptr;
|
|
||||||
do {
|
|
||||||
if (p->chr == (unsigned)(*s & DIC_CHAR_MASK))
|
|
||||||
return Dic_seek_edgeptr (dic,s + 1, p);
|
|
||||||
} while (!(*p++).last);
|
|
||||||
return dic->dawg;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return eptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Dic_search_word_inner : direct application of Dic_seek_edgeptr
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param word : word to lookup
|
|
||||||
* @result 0 not a valid word, 1 ok
|
|
||||||
*/
|
|
||||||
static int Dic_search_word_inner(const Dictionary dic, const char* word)
|
|
||||||
{
|
|
||||||
Dawg_edge *e;
|
|
||||||
e = Dic_seek_edgeptr(dic, word, dic->dawg + dic->root);
|
|
||||||
return e->term;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_word_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
int Dic_search_word(const Dictionary dic, const wchar_t* word)
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
char *tmp_word = malloc(wcslen(word) + 1);
|
|
||||||
sprintf(tmp_word, "%ls", word);
|
|
||||||
|
|
||||||
// Do the actual work
|
|
||||||
res = Dic_search_word_inner(dic, tmp_word);
|
|
||||||
|
|
||||||
// Release memory
|
|
||||||
free(tmp_word);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* global variables for Dic_search_word_by_len :
|
|
||||||
*
|
|
||||||
* a pointer to the structure is passed as a parameter
|
|
||||||
* so that all the search_* variables appear to the functions
|
|
||||||
* as global but the code remains re-entrant.
|
|
||||||
* Should be better to change the algorithm ...
|
|
||||||
*/
|
|
||||||
|
|
||||||
struct params_7plus1_t {
|
|
||||||
Dictionary search_dic;
|
|
||||||
int search_len;
|
|
||||||
int search_wordlistlen;
|
|
||||||
int search_wordlistlenmax;
|
|
||||||
char search_wordtst[DIC_WORD_MAX];
|
|
||||||
char search_letters[DIC_LETTERS];
|
|
||||||
char (*search_wordlist)[RES_7PL1_MAX][DIC_WORD_MAX];
|
|
||||||
};
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_search_word_by_len(struct params_7plus1_t *params, int i, Dawg_edge *edgeptr)
|
|
||||||
{
|
|
||||||
/* depth first search in the dictionary */
|
|
||||||
do {
|
|
||||||
/* we use a static array and not a real list so we have to stop if
|
|
||||||
* the array is full */
|
|
||||||
if (params->search_wordlistlen >= params->search_wordlistlenmax)
|
|
||||||
break;
|
|
||||||
|
|
||||||
/* the test is false only when reach the end-node */
|
|
||||||
if (edgeptr->chr)
|
|
||||||
{
|
|
||||||
|
|
||||||
/* is the letter available in search_letters */
|
|
||||||
if (params->search_letters[edgeptr->chr])
|
|
||||||
{
|
|
||||||
params->search_wordtst[i] = edgeptr->chr + 'A' - 1;
|
|
||||||
params->search_letters[edgeptr->chr] --;
|
|
||||||
if (i == params->search_len)
|
|
||||||
{
|
|
||||||
if ((edgeptr->term)
|
|
||||||
/* && (params->search_wordlistlen < params->search_wordlistlenmax) */)
|
|
||||||
strcpy((*params->search_wordlist)[params->search_wordlistlen++],params->search_wordtst);
|
|
||||||
}
|
|
||||||
else /* if (params->search_wordlistlen < params->search_wordlistlenmax) */
|
|
||||||
{
|
|
||||||
Dic_search_word_by_len(params,i + 1, params->search_dic->dawg + edgeptr->ptr);
|
|
||||||
}
|
|
||||||
params->search_letters[edgeptr->chr] ++;
|
|
||||||
params->search_wordtst[i] = '\0';
|
|
||||||
}
|
|
||||||
|
|
||||||
/* the letter is of course available if we have a joker available */
|
|
||||||
if (params->search_letters[0])
|
|
||||||
{
|
|
||||||
params->search_wordtst[i] = edgeptr->chr + 'a' - 1;
|
|
||||||
params->search_letters[0] --;
|
|
||||||
if (i == params->search_len)
|
|
||||||
{
|
|
||||||
if ((edgeptr->term)
|
|
||||||
/* && (params->search_wordlistlen < params->search_wordlistlenmax) */)
|
|
||||||
strcpy((*(params->search_wordlist))[params->search_wordlistlen++],params->search_wordtst);
|
|
||||||
}
|
|
||||||
else /* if (params->search_wordlistlen < params->search_wordlistlenmax) */
|
|
||||||
{
|
|
||||||
Dic_search_word_by_len(params,i + 1,params->search_dic->dawg + edgeptr->ptr);
|
|
||||||
}
|
|
||||||
params->search_letters[0] ++;
|
|
||||||
params->search_wordtst[i] = '\0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while (! (*edgeptr++).last);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_search_7pl1_inner(const Dictionary dic, const char* rack,
|
|
||||||
char buff[DIC_LETTERS][RES_7PL1_MAX][DIC_WORD_MAX],
|
|
||||||
int joker)
|
|
||||||
{
|
|
||||||
int i,j,wordlen;
|
|
||||||
const char* r = rack;
|
|
||||||
struct params_7plus1_t params;
|
|
||||||
Dawg_edge *root_edge;
|
|
||||||
|
|
||||||
for(i=0; i < DIC_LETTERS; i++)
|
|
||||||
for(j=0; j < RES_7PL1_MAX; j++)
|
|
||||||
buff[i][j][0] = '\0';
|
|
||||||
|
|
||||||
for(i=0; i<DIC_LETTERS; i++)
|
|
||||||
params.search_letters[i] = 0;
|
|
||||||
|
|
||||||
if (dic == NULL || rack == NULL || *rack == '\0')
|
|
||||||
return;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* the letters are verified and changed to the dic internal
|
|
||||||
* representation (*r & DIC_CHAR_MASK)
|
|
||||||
*/
|
|
||||||
for(wordlen=0; wordlen < DIC_WORD_MAX && *r; r++)
|
|
||||||
{
|
|
||||||
if (isalpha(*r))
|
|
||||||
{
|
|
||||||
params.search_letters[(int)*r & DIC_CHAR_MASK]++;
|
|
||||||
wordlen++;
|
|
||||||
}
|
|
||||||
else if (*r == '?')
|
|
||||||
{
|
|
||||||
if (joker)
|
|
||||||
{
|
|
||||||
params.search_letters[0]++;
|
|
||||||
wordlen++;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
strncpy(buff[0][0],"** joker **",DIC_WORD_MAX);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wordlen < 1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
root_edge = dic->dawg + (dic->dawg[dic->root].ptr);
|
|
||||||
|
|
||||||
params.search_dic = dic;
|
|
||||||
params.search_wordlistlenmax = RES_7PL1_MAX;
|
|
||||||
|
|
||||||
/* search for all the words that can be done with the letters */
|
|
||||||
params.search_len = wordlen - 1;
|
|
||||||
params.search_wordtst[wordlen]='\0';
|
|
||||||
params.search_wordlist = & buff[0];
|
|
||||||
params.search_wordlistlen = 0;
|
|
||||||
Dic_search_word_by_len(¶ms,0,root_edge);
|
|
||||||
|
|
||||||
/* search for all the words that can be done with the letters +1 */
|
|
||||||
params.search_len = wordlen;
|
|
||||||
params.search_wordtst[wordlen + 1]='\0';
|
|
||||||
for(i='a'; i <= 'z'; i++)
|
|
||||||
{
|
|
||||||
params.search_letters[i & DIC_CHAR_MASK]++;
|
|
||||||
|
|
||||||
params.search_wordlist = & buff[i & DIC_CHAR_MASK];
|
|
||||||
params.search_wordlistlen = 0;
|
|
||||||
Dic_search_word_by_len(¶ms,0,root_edge);
|
|
||||||
|
|
||||||
params.search_letters[i & DIC_CHAR_MASK]--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_7pl1_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
Dic_search_7pl1(const Dictionary dic, const wchar_t* rack,
|
|
||||||
wchar_t buff[DIC_LETTERS][RES_7PL1_MAX][DIC_WORD_MAX],
|
|
||||||
int joker)
|
|
||||||
{
|
|
||||||
int i, j, k;
|
|
||||||
char tmp_buff[DIC_LETTERS][RES_7PL1_MAX][DIC_WORD_MAX];
|
|
||||||
char *tmp_rack = malloc(wcslen(rack) + 1);
|
|
||||||
sprintf(tmp_rack, "%ls", rack);
|
|
||||||
// Do the actual work
|
|
||||||
Dic_search_7pl1_inner(dic, tmp_rack, tmp_buff, joker);
|
|
||||||
|
|
||||||
for (i = 0; i < DIC_LETTERS; i++)
|
|
||||||
{
|
|
||||||
for (j = 0; j < RES_7PL1_MAX; j++)
|
|
||||||
{
|
|
||||||
for (k = 0; k < DIC_WORD_MAX; k++)
|
|
||||||
{
|
|
||||||
buff[i][j][k] = tmp_buff[i][j][k];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(tmp_rack);
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************/
|
|
||||||
/****************************************/
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_search_Racc_inner(const Dictionary dic, const char* word,
|
|
||||||
char wordlist[RES_RACC_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
/* search_racc will try to add a letter in front and at the end of a word */
|
|
||||||
|
|
||||||
int i,wordlistlen;
|
|
||||||
Dawg_edge *edge;
|
|
||||||
char wordtst[DIC_WORD_MAX];
|
|
||||||
|
|
||||||
for(i=0; i < RES_RACC_MAX; i++)
|
|
||||||
wordlist[i][0] = 0;
|
|
||||||
|
|
||||||
if (dic == NULL || wordlist == NULL || *wordlist == '\0')
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* let's try for the front */
|
|
||||||
wordlistlen = 0;
|
|
||||||
strcpy(wordtst+1,word);
|
|
||||||
for(i='a'; i <= 'z'; i++)
|
|
||||||
{
|
|
||||||
wordtst[0] = i;
|
|
||||||
if (Dic_search_word_inner(dic,wordtst) && wordlistlen < RES_RACC_MAX)
|
|
||||||
strcpy(wordlist[wordlistlen++],wordtst);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add a letter at the end */
|
|
||||||
for(i=0; word[i]; i++)
|
|
||||||
wordtst[i] = word[i];
|
|
||||||
|
|
||||||
wordtst[i ] = '\0';
|
|
||||||
wordtst[i+1] = '\0';
|
|
||||||
|
|
||||||
edge = Dic_seek_edgeptr(dic,word,dic->dawg + dic->root);
|
|
||||||
|
|
||||||
/* points to what the next letter can be */
|
|
||||||
edge = dic->dawg + edge->ptr;
|
|
||||||
|
|
||||||
if (edge != dic->dawg)
|
|
||||||
{
|
|
||||||
do {
|
|
||||||
if (edge->term && wordlistlen < RES_RACC_MAX)
|
|
||||||
{
|
|
||||||
wordtst[i] = edge->chr + 'a' - 1;
|
|
||||||
strcpy(wordlist[wordlistlen++],wordtst);
|
|
||||||
}
|
|
||||||
} while (!(*edge++).last);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_Racc_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
Dic_search_Racc(const Dictionary dic, const wchar_t* word,
|
|
||||||
wchar_t wordlist[RES_RACC_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int i, j;
|
|
||||||
char tmp_buff[RES_RACC_MAX][DIC_WORD_MAX];
|
|
||||||
char *tmp_word = malloc(wcslen(word) + 1);
|
|
||||||
sprintf(tmp_word, "%ls", word);
|
|
||||||
// Do the actual work
|
|
||||||
Dic_search_Racc_inner(dic, tmp_word, tmp_buff);
|
|
||||||
|
|
||||||
for (i = 0; i < RES_RACC_MAX; i++)
|
|
||||||
{
|
|
||||||
for (j = 0; j < DIC_WORD_MAX; j++)
|
|
||||||
{
|
|
||||||
wordlist[i][j] = tmp_buff[i][j];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(tmp_word);
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************/
|
|
||||||
/****************************************/
|
|
||||||
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_search_Benj_inner(const Dictionary dic, const char* word,
|
|
||||||
char wordlist[RES_BENJ_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int i,wordlistlen;
|
|
||||||
char wordtst[DIC_WORD_MAX];
|
|
||||||
Dawg_edge *edge0,*edge1,*edge2,*edgetst;
|
|
||||||
|
|
||||||
for(i=0; i < RES_BENJ_MAX; i++)
|
|
||||||
wordlist[i][0] = 0;
|
|
||||||
|
|
||||||
if (dic == NULL || word == NULL || *word == '\0')
|
|
||||||
return;
|
|
||||||
|
|
||||||
wordlistlen = 0;
|
|
||||||
|
|
||||||
strcpy(wordtst+3,word);
|
|
||||||
edge0 = dic->dawg + (dic->dawg[dic->root].ptr);
|
|
||||||
do {
|
|
||||||
wordtst[0] = edge0->chr + 'a' - 1;
|
|
||||||
edge1 = dic->dawg + edge0->ptr;
|
|
||||||
do {
|
|
||||||
wordtst[1] = edge1->chr + 'a' - 1;
|
|
||||||
edge2 = dic->dawg + edge1->ptr;
|
|
||||||
do {
|
|
||||||
wordtst[2] = edge2->chr + 'a' - 1;
|
|
||||||
edgetst = Dic_seek_edgeptr(dic,word,edge2);
|
|
||||||
if (edgetst->term && wordlistlen < RES_BENJ_MAX)
|
|
||||||
strcpy(wordlist[wordlistlen++],wordtst);
|
|
||||||
} while (!(*edge2++).last);
|
|
||||||
} while (!(*edge1++).last);
|
|
||||||
} while (!(*edge0++).last);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_Benj_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
Dic_search_Benj(const Dictionary dic, const wchar_t* word,
|
|
||||||
wchar_t wordlist[RES_BENJ_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int i, j;
|
|
||||||
char tmp_buff[RES_BENJ_MAX][DIC_WORD_MAX];
|
|
||||||
char *tmp_word = malloc(wcslen(word) + 1);
|
|
||||||
sprintf(tmp_word, "%ls", word);
|
|
||||||
// Do the actual work
|
|
||||||
Dic_search_Benj_inner(dic, tmp_word, tmp_buff);
|
|
||||||
|
|
||||||
for (i = 0; i < RES_BENJ_MAX; i++)
|
|
||||||
{
|
|
||||||
for (j = 0; j < DIC_WORD_MAX; j++)
|
|
||||||
{
|
|
||||||
wordlist[i][j] = tmp_buff[i][j];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(tmp_word);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/****************************************/
|
|
||||||
/****************************************/
|
|
||||||
|
|
||||||
struct params_cross_t {
|
|
||||||
Dictionary dic;
|
|
||||||
int wordlen;
|
|
||||||
int wordlistlen;
|
|
||||||
int wordlistlenmax;
|
|
||||||
char mask[DIC_WORD_MAX];
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
Dic_search_cross_rec(struct params_cross_t *params,
|
|
||||||
char wordlist[RES_CROS_MAX][DIC_WORD_MAX],
|
|
||||||
Dawg_edge *edgeptr)
|
|
||||||
{
|
|
||||||
Dawg_edge *current = params->dic->dawg + edgeptr->ptr;
|
|
||||||
|
|
||||||
if (params->mask[params->wordlen] == '\0' && edgeptr->term)
|
|
||||||
{
|
|
||||||
if (params->wordlistlen < params->wordlistlenmax)
|
|
||||||
strcpy(wordlist[params->wordlistlen++],params->mask);
|
|
||||||
}
|
|
||||||
else if (params->mask[params->wordlen] == '.')
|
|
||||||
{
|
|
||||||
do
|
|
||||||
{
|
|
||||||
params->mask[params->wordlen] = current->chr + 'a' - 1;
|
|
||||||
params->wordlen ++;
|
|
||||||
Dic_search_cross_rec(params,wordlist,current);
|
|
||||||
params->wordlen --;
|
|
||||||
params->mask[params->wordlen] = '.';
|
|
||||||
}
|
|
||||||
while (!(*current++).last);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (current->chr == (unsigned int)(params->mask[params->wordlen] & DIC_CHAR_MASK))
|
|
||||||
{
|
|
||||||
params->wordlen ++;
|
|
||||||
Dic_search_cross_rec(params,wordlist,current);
|
|
||||||
params->wordlen --;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while (!(*current++).last);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void
|
|
||||||
Dic_search_Cros_inner(const Dictionary dic, const char* mask,
|
|
||||||
char wordlist[RES_CROS_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
struct params_cross_t params;
|
|
||||||
|
|
||||||
for(i=0; i < RES_CROS_MAX; i++)
|
|
||||||
wordlist[i][0] = 0;
|
|
||||||
|
|
||||||
if (dic == NULL || mask == NULL || *mask == '\0')
|
|
||||||
return;
|
|
||||||
|
|
||||||
for(i=0; i < DIC_WORD_MAX && mask[i]; i++)
|
|
||||||
{
|
|
||||||
if (isalpha(mask[i]))
|
|
||||||
params.mask[i] = (mask[i] & DIC_CHAR_MASK) + 'A' - 1;
|
|
||||||
else
|
|
||||||
params.mask[i] = '.';
|
|
||||||
}
|
|
||||||
params.mask[i] = '\0';
|
|
||||||
|
|
||||||
params.dic = dic;
|
|
||||||
params.wordlen = 0;
|
|
||||||
params.wordlistlen = 0;
|
|
||||||
params.wordlistlenmax = RES_CROS_MAX;
|
|
||||||
Dic_search_cross_rec(¶ms, wordlist, dic->dawg + dic->root);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_Cros_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
Dic_search_Cros(const Dictionary dic, const wchar_t* mask,
|
|
||||||
wchar_t wordlist[RES_CROS_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int i, j;
|
|
||||||
char tmp_buff[RES_CROS_MAX][DIC_WORD_MAX];
|
|
||||||
char *tmp_mask = malloc(wcslen(mask) + 1);
|
|
||||||
sprintf(tmp_mask, "%ls", mask);
|
|
||||||
// Do the actual work
|
|
||||||
Dic_search_Cros_inner(dic, tmp_mask, tmp_buff);
|
|
||||||
|
|
||||||
for (i = 0; i < RES_CROS_MAX; i++)
|
|
||||||
{
|
|
||||||
for (j = 0; j < DIC_WORD_MAX; j++)
|
|
||||||
{
|
|
||||||
wordlist[i][j] = tmp_buff[i][j];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(tmp_mask);
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************/
|
|
||||||
/****************************************/
|
|
||||||
|
|
||||||
struct params_regexp_t {
|
|
||||||
Dictionary dic;
|
|
||||||
int minlength;
|
|
||||||
int maxlength;
|
|
||||||
automaton automaton;
|
|
||||||
struct search_RegE_list_t *charlist;
|
|
||||||
char word[DIC_WORD_MAX];
|
|
||||||
int wordlen;
|
|
||||||
int wordlistlen;
|
|
||||||
int wordlistlenmax;
|
|
||||||
};
|
|
||||||
|
|
||||||
void
|
|
||||||
Dic_search_regexp_rec(struct params_regexp_t *params,
|
|
||||||
int state,
|
|
||||||
Dawg_edge *edgeptr,
|
|
||||||
char wordlist[RES_REGE_MAX][DIC_WORD_MAX])
|
|
||||||
{
|
|
||||||
int next_state;
|
|
||||||
Dawg_edge *current;
|
|
||||||
/* if we have a valid word we store it */
|
|
||||||
if (automaton_get_accept(params->automaton,state) && edgeptr->term)
|
|
||||||
{
|
|
||||||
int l = strlen(params->word);
|
|
||||||
if (params->wordlistlen < params->wordlistlenmax &&
|
|
||||||
params->minlength <= l &&
|
|
||||||
params->maxlength >= l)
|
|
||||||
{
|
|
||||||
strcpy(wordlist[params->wordlistlen++],params->word);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/* we now drive the search by exploring the dictionary */
|
|
||||||
current = params->dic->dawg + edgeptr->ptr;
|
|
||||||
do {
|
|
||||||
/* the current letter is current->chr */
|
|
||||||
next_state = automaton_get_next_state(params->automaton,state,current->chr);
|
|
||||||
/* 1 : the letter appears in the automaton as is */
|
|
||||||
if (next_state)
|
|
||||||
{
|
|
||||||
params->word[params->wordlen] = current->chr + 'a' - 1;
|
|
||||||
params->wordlen ++;
|
|
||||||
Dic_search_regexp_rec(params,next_state,current,wordlist);
|
|
||||||
params->wordlen --;
|
|
||||||
params->word[params->wordlen] = '\0';
|
|
||||||
}
|
|
||||||
} while (!(*current++).last);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
Dic_search_RegE_inner(const Dictionary dic, const char* re,
|
|
||||||
char wordlist[RES_REGE_MAX][DIC_WORD_MAX],
|
|
||||||
struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
int i,p,n,value;
|
|
||||||
int ptl[REGEXP_MAX+1];
|
|
||||||
int PS [REGEXP_MAX+1];
|
|
||||||
NODE* root;
|
|
||||||
yyscan_t scanner;
|
|
||||||
YY_BUFFER_STATE buf;
|
|
||||||
automaton a;
|
|
||||||
char stringbuf[250];
|
|
||||||
struct params_regexp_t params;
|
|
||||||
struct regexp_error_report_t report;
|
|
||||||
|
|
||||||
/* init */
|
|
||||||
for(i=0; i < RES_REGE_MAX; i++)
|
|
||||||
wordlist[i][0] = 0;
|
|
||||||
|
|
||||||
if (dic == NULL || re == NULL || *re == '\0')
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* (expr)# */
|
|
||||||
sprintf(stringbuf,"(%s)#",re);
|
|
||||||
for(i=0; i < REGEXP_MAX; i++)
|
|
||||||
{
|
|
||||||
PS[i] = 0;
|
|
||||||
ptl[i] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
report.pos1 = 0;
|
|
||||||
report.pos2 = 0;
|
|
||||||
report.msg[0] = '\0';
|
|
||||||
|
|
||||||
/* parsing */
|
|
||||||
regexplex_init( &scanner );
|
|
||||||
buf = regexp_scan_string( stringbuf, scanner );
|
|
||||||
root = NULL;
|
|
||||||
value = regexpparse( scanner , &root, list, &report);
|
|
||||||
regexp_delete_buffer(buf,scanner);
|
|
||||||
regexplex_destroy( scanner );
|
|
||||||
|
|
||||||
if (value)
|
|
||||||
{
|
|
||||||
#ifdef DEBUG_FLEX_IS_BROKEN
|
|
||||||
fprintf(stderr,"parser error at pos %d - %d : %s\n",
|
|
||||||
report.pos1, report.pos2, report.msg);
|
|
||||||
#endif
|
|
||||||
regexp_delete_tree(root);
|
|
||||||
return ;
|
|
||||||
}
|
|
||||||
|
|
||||||
n = 1;
|
|
||||||
p = 1;
|
|
||||||
regexp_parcours(root, &p, &n, ptl);
|
|
||||||
PS [0] = p - 1;
|
|
||||||
ptl[0] = p - 1;
|
|
||||||
|
|
||||||
regexp_possuivante(root,PS);
|
|
||||||
|
|
||||||
if ((a = automaton_build(root->PP,ptl,PS,list)) != NULL)
|
|
||||||
{
|
|
||||||
params.dic = dic;
|
|
||||||
params.minlength = list->minlength;
|
|
||||||
params.maxlength = list->maxlength;
|
|
||||||
params.automaton = a;
|
|
||||||
params.charlist = list;
|
|
||||||
memset(params.word,'\0',sizeof(params.word));
|
|
||||||
params.wordlen = 0;
|
|
||||||
params.wordlistlen = 0;
|
|
||||||
params.wordlistlenmax = RES_REGE_MAX;
|
|
||||||
Dic_search_regexp_rec(¶ms, automaton_get_init(a), dic->dawg + dic->root, wordlist);
|
|
||||||
|
|
||||||
automaton_delete(a);
|
|
||||||
}
|
|
||||||
regexp_delete_tree(root);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper around Dic_search_RegE_inner, until we have multibyte support in
|
|
||||||
* the dictionary
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
Dic_search_RegE(const Dictionary dic, const wchar_t* re,
|
|
||||||
wchar_t wordlist[RES_REGE_MAX][DIC_WORD_MAX],
|
|
||||||
struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
char tmp_buff[RES_REGE_MAX][DIC_WORD_MAX];
|
|
||||||
char *tmp_re = malloc(wcslen(re) + 1);
|
|
||||||
sprintf(tmp_re, "%ls", re);
|
|
||||||
// Do the actual work
|
|
||||||
Dic_search_RegE_inner(dic, tmp_re, tmp_buff, list);
|
|
||||||
|
|
||||||
for (i = 0; i < RES_REGE_MAX; i++)
|
|
||||||
{
|
|
||||||
mbstowcs(wordlist[i], tmp_buff[i], DIC_WORD_MAX);
|
|
||||||
}
|
|
||||||
free(tmp_re);
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************/
|
|
||||||
/****************************************/
|
|
||||||
|
|
566
dic/dic_search.cpp
Normal file
566
dic/dic_search.cpp
Normal file
|
@ -0,0 +1,566 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2002-2007 Antoine Fraboulet
|
||||||
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file dic_search.c
|
||||||
|
* \brief Dictionary lookup functions
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 2002
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cwchar>
|
||||||
|
#include <cwctype>
|
||||||
|
|
||||||
|
#include "dic_internals.h"
|
||||||
|
#include "dic.h"
|
||||||
|
#include "header.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "regexp.h"
|
||||||
|
#include "libdic_a-ery.h" /* generated by bison */
|
||||||
|
#include "libdic_a-erl.h" /* generated by flex */
|
||||||
|
#include "automaton.h"
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function prototype for bison generated parser
|
||||||
|
*/
|
||||||
|
int regexpparse(yyscan_t scanner, NODE** root,
|
||||||
|
struct search_RegE_list_t *iList,
|
||||||
|
struct regexp_error_report_t *err);
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
const DAWG_EDGE* Dictionary::seekEdgePtr(const wchar_t* s, const DAWG_EDGE *eptr) const
|
||||||
|
{
|
||||||
|
if (*s)
|
||||||
|
{
|
||||||
|
const DAWG_EDGE *p = getEdgeAt<DAWG_EDGE>(eptr->ptr);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (p->chr == getHeader().getCodeFromChar(*s))
|
||||||
|
return seekEdgePtr(s + 1, p);
|
||||||
|
} while (!(*p++).last);
|
||||||
|
return getEdgeAt<DAWG_EDGE>(0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return eptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Dictionary::searchWord(const wstring &iWord) const
|
||||||
|
{
|
||||||
|
if (!validateLetters(iWord))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
{
|
||||||
|
const DicEdgeOld *e =
|
||||||
|
seekEdgePtr(iWord.c_str(), getEdgeAt<DicEdgeOld>(getRoot()));
|
||||||
|
return e->term;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const DicEdge *e =
|
||||||
|
seekEdgePtr(iWord.c_str(), getEdgeAt<DicEdge>(getRoot()));
|
||||||
|
return e->term;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Global variables for searchWordByLen:
|
||||||
|
*
|
||||||
|
* A pointer to the structure is passed as a parameter
|
||||||
|
* so that all the search_* variables appear to the functions
|
||||||
|
* as global but the code remains re-entrant.
|
||||||
|
* Should be better to change the algorithm ...
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct params_7plus1_t
|
||||||
|
{
|
||||||
|
wchar_t added_char;
|
||||||
|
map<wchar_t, list<wstring> > *results;
|
||||||
|
int search_len;
|
||||||
|
wchar_t search_wordtst[DIC_WORD_MAX];
|
||||||
|
char search_letters[63];
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::searchWordByLen(struct params_7plus1_t *params,
|
||||||
|
int i, const DAWG_EDGE *edgeptr) const
|
||||||
|
{
|
||||||
|
/* depth first search in the dictionary */
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* the test is false only when reach the end-node */
|
||||||
|
if (edgeptr->chr)
|
||||||
|
{
|
||||||
|
/* is the letter available in search_letters */
|
||||||
|
if (params->search_letters[edgeptr->chr])
|
||||||
|
{
|
||||||
|
params->search_wordtst[i] = getHeader().getCharFromCode(edgeptr->chr);
|
||||||
|
params->search_letters[edgeptr->chr] --;
|
||||||
|
if (i == params->search_len)
|
||||||
|
{
|
||||||
|
if (edgeptr->term)
|
||||||
|
{
|
||||||
|
(*params->results)[params->added_char].push_back(params->search_wordtst);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
searchWordByLen(params, i + 1, getEdgeAt<DAWG_EDGE>(edgeptr->ptr));
|
||||||
|
}
|
||||||
|
params->search_letters[edgeptr->chr] ++;
|
||||||
|
params->search_wordtst[i] = L'\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
/* the letter is of course available if we have a joker available */
|
||||||
|
if (params->search_letters[0])
|
||||||
|
{
|
||||||
|
params->search_wordtst[i] = getHeader().getCharFromCode(edgeptr->chr);
|
||||||
|
params->search_letters[0] --;
|
||||||
|
if (i == params->search_len)
|
||||||
|
{
|
||||||
|
if (edgeptr->term)
|
||||||
|
{
|
||||||
|
(*params->results)[params->added_char].push_back(params->search_wordtst);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
searchWordByLen(params, i + 1, getEdgeAt<DAWG_EDGE>(edgeptr->ptr));
|
||||||
|
}
|
||||||
|
params->search_letters[0] ++;
|
||||||
|
params->search_wordtst[i] = L'\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while (! (*edgeptr++).last);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::search7pl1Templ(const wstring &iRack,
|
||||||
|
map<wchar_t, list<wstring> > &oWordList,
|
||||||
|
bool joker) const
|
||||||
|
{
|
||||||
|
if (iRack == L"" || iRack.size() > DIC_WORD_MAX)
|
||||||
|
return;
|
||||||
|
|
||||||
|
struct params_7plus1_t params;
|
||||||
|
|
||||||
|
for (unsigned int i = 0; i < sizeof(params.search_letters); i++)
|
||||||
|
params.search_letters[i] = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* the letters are verified and changed to the dic internal
|
||||||
|
* representation (using getCodeFromChar(*r))
|
||||||
|
*/
|
||||||
|
int wordlen = 0;
|
||||||
|
for (const wchar_t* r = iRack.c_str(); *r; r++)
|
||||||
|
{
|
||||||
|
if (iswalpha(*r))
|
||||||
|
{
|
||||||
|
params.search_letters[getHeader().getCodeFromChar(*r)]++;
|
||||||
|
wordlen++;
|
||||||
|
}
|
||||||
|
else if (*r == L'?')
|
||||||
|
{
|
||||||
|
if (joker)
|
||||||
|
{
|
||||||
|
params.search_letters[0]++;
|
||||||
|
wordlen++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
oWordList[0].push_back(L"** joker **");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wordlen < 1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const DAWG_EDGE *root_edge = getEdgeAt<DAWG_EDGE>(getRoot());
|
||||||
|
root_edge = getEdgeAt<DAWG_EDGE>(root_edge->ptr);
|
||||||
|
|
||||||
|
params.results = &oWordList;
|
||||||
|
|
||||||
|
/* search for all the words that can be done with the letters */
|
||||||
|
params.added_char = L'\0';
|
||||||
|
params.search_len = wordlen - 1;
|
||||||
|
params.search_wordtst[wordlen] = L'\0';
|
||||||
|
searchWordByLen(¶ms, 0, root_edge);
|
||||||
|
|
||||||
|
/* search for all the words that can be done with the letters +1 */
|
||||||
|
params.search_len = wordlen;
|
||||||
|
params.search_wordtst[wordlen + 1] = L'\0';
|
||||||
|
const wstring &letters = getHeader().getLetters();
|
||||||
|
for (unsigned int i = 0; i < letters.size(); i++)
|
||||||
|
{
|
||||||
|
params.added_char = letters[i];
|
||||||
|
unsigned int code = getHeader().getCodeFromChar(letters[i]);
|
||||||
|
params.search_letters[code]++;
|
||||||
|
|
||||||
|
searchWordByLen(¶ms, 0, root_edge);
|
||||||
|
|
||||||
|
params.search_letters[code]--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::search7pl1(const wstring &iRack,
|
||||||
|
map<wchar_t, list<wstring> > &oWordList,
|
||||||
|
bool joker) const
|
||||||
|
{
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
search7pl1Templ<DicEdgeOld>(iRack, oWordList, joker);
|
||||||
|
else
|
||||||
|
search7pl1Templ<DicEdge>(iRack, oWordList, joker);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************/
|
||||||
|
/****************************************/
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::searchRaccTempl(const wstring &iWord, list<wstring> &oWordList) const
|
||||||
|
{
|
||||||
|
if (iWord == L"")
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* search_racc will try to add a letter in front and at the end of a word */
|
||||||
|
|
||||||
|
/* let's try for the front */
|
||||||
|
wchar_t wordtst[DIC_WORD_MAX];
|
||||||
|
wcscpy(wordtst + 1, iWord.c_str());
|
||||||
|
const wstring &letters = getHeader().getLetters();
|
||||||
|
for (unsigned int i = 0; i <= letters.size(); i++)
|
||||||
|
{
|
||||||
|
wordtst[0] = letters[i];
|
||||||
|
if (searchWord(wordtst))
|
||||||
|
oWordList.push_back(wordtst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add a letter at the end */
|
||||||
|
int i;
|
||||||
|
for (i = 0; iWord[i]; i++)
|
||||||
|
wordtst[i] = iWord[i];
|
||||||
|
|
||||||
|
wordtst[i ] = '\0';
|
||||||
|
wordtst[i+1] = '\0';
|
||||||
|
|
||||||
|
const DAWG_EDGE *edge_seek =
|
||||||
|
seekEdgePtr(iWord.c_str(), getEdgeAt<DAWG_EDGE>(getRoot()));
|
||||||
|
|
||||||
|
/* points to what the next letter can be */
|
||||||
|
const DAWG_EDGE *edge = getEdgeAt<DAWG_EDGE>(edge_seek->ptr);
|
||||||
|
|
||||||
|
if (edge != getEdgeAt<DAWG_EDGE>(0))
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (edge->term)
|
||||||
|
{
|
||||||
|
wordtst[i] = getHeader().getCharFromCode(edge->chr);
|
||||||
|
oWordList.push_back(wordtst);
|
||||||
|
}
|
||||||
|
} while (!(*edge++).last);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::searchRacc(const wstring &iWord, list<wstring> &oWordList) const
|
||||||
|
{
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
searchRaccTempl<DicEdgeOld>(iWord, oWordList);
|
||||||
|
else
|
||||||
|
searchRaccTempl<DicEdge>(iWord, oWordList);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************/
|
||||||
|
/****************************************/
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::searchBenjTempl(const wstring &iWord, list<wstring> &oWordList) const
|
||||||
|
{
|
||||||
|
if (iWord == L"")
|
||||||
|
return;
|
||||||
|
|
||||||
|
wchar_t wordtst[DIC_WORD_MAX];
|
||||||
|
wcscpy(wordtst + 3, iWord.c_str());
|
||||||
|
const DAWG_EDGE *edge0, *edge1, *edge2, *edgetst;
|
||||||
|
edge0 = getEdgeAt<DAWG_EDGE>(getRoot());
|
||||||
|
edge0 = getEdgeAt<DAWG_EDGE>(edge0->ptr);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
wordtst[0] = getHeader().getCharFromCode(edge0->chr);
|
||||||
|
edge1 = getEdgeAt<DAWG_EDGE>(edge0->ptr);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
wordtst[1] = getHeader().getCharFromCode(edge1->chr);
|
||||||
|
edge2 = getEdgeAt<DAWG_EDGE>(edge1->ptr);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
edgetst = seekEdgePtr(iWord.c_str(), edge2);
|
||||||
|
if (edgetst->term)
|
||||||
|
{
|
||||||
|
wordtst[2] = getHeader().getCharFromCode(edge2->chr);
|
||||||
|
oWordList.push_back(wordtst);
|
||||||
|
}
|
||||||
|
} while (!(*edge2++).last);
|
||||||
|
} while (!(*edge1++).last);
|
||||||
|
} while (!(*edge0++).last);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::searchBenj(const wstring &iWord, list<wstring> &oWordList) const
|
||||||
|
{
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
searchBenjTempl<DicEdgeOld>(iWord, oWordList);
|
||||||
|
else
|
||||||
|
searchBenjTempl<DicEdge>(iWord, oWordList);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************/
|
||||||
|
/****************************************/
|
||||||
|
|
||||||
|
struct params_cross_t
|
||||||
|
{
|
||||||
|
int wordlen;
|
||||||
|
wchar_t mask[DIC_WORD_MAX];
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::searchCrossRecTempl(struct params_cross_t *params,
|
||||||
|
list<wstring> &oWordList,
|
||||||
|
const DAWG_EDGE *edgeptr) const
|
||||||
|
{
|
||||||
|
const DAWG_EDGE *current = getEdgeAt<DAWG_EDGE>(edgeptr->ptr);
|
||||||
|
|
||||||
|
if (params->mask[params->wordlen] == '\0' && edgeptr->term)
|
||||||
|
{
|
||||||
|
oWordList.push_back(params->mask);
|
||||||
|
}
|
||||||
|
else if (params->mask[params->wordlen] == '.')
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
params->mask[params->wordlen] = getHeader().getCharFromCode(current->chr);
|
||||||
|
params->wordlen ++;
|
||||||
|
searchCrossRecTempl(params, oWordList, current);
|
||||||
|
params->wordlen --;
|
||||||
|
params->mask[params->wordlen] = '.';
|
||||||
|
}
|
||||||
|
while (!(*current++).last);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (current->chr == getHeader().getCodeFromChar(params->mask[params->wordlen]))
|
||||||
|
{
|
||||||
|
params->wordlen ++;
|
||||||
|
searchCrossRecTempl(params, oWordList, current);
|
||||||
|
params->wordlen --;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (!(*current++).last);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::searchCross(const wstring &iMask, list<wstring> &oWordList) const
|
||||||
|
{
|
||||||
|
if (iMask == L"")
|
||||||
|
return;
|
||||||
|
|
||||||
|
struct params_cross_t params;
|
||||||
|
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < DIC_WORD_MAX && iMask[i]; i++)
|
||||||
|
{
|
||||||
|
if (iswalpha(iMask[i]))
|
||||||
|
params.mask[i] = towupper(iMask[i]);
|
||||||
|
else
|
||||||
|
params.mask[i] = '.';
|
||||||
|
}
|
||||||
|
params.mask[i] = '\0';
|
||||||
|
|
||||||
|
params.wordlen = 0;
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
{
|
||||||
|
searchCrossRecTempl(¶ms, oWordList,
|
||||||
|
getEdgeAt<DicEdgeOld>(getRoot()));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
searchCrossRecTempl(¶ms, oWordList,
|
||||||
|
getEdgeAt<DicEdge>(getRoot()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************/
|
||||||
|
/****************************************/
|
||||||
|
|
||||||
|
struct params_regexp_t
|
||||||
|
{
|
||||||
|
int minlength;
|
||||||
|
int maxlength;
|
||||||
|
Automaton *automaton_field;
|
||||||
|
struct search_RegE_list_t *charlist;
|
||||||
|
char word[DIC_WORD_MAX];
|
||||||
|
int wordlen;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void Dictionary::searchRegexpRecTempl(struct params_regexp_t *params,
|
||||||
|
int state,
|
||||||
|
const DAWG_EDGE *edgeptr,
|
||||||
|
list<string> &oWordList) const
|
||||||
|
{
|
||||||
|
int next_state;
|
||||||
|
/* if we have a valid word we store it */
|
||||||
|
if (params->automaton_field->accept(state) && edgeptr->term)
|
||||||
|
{
|
||||||
|
int l = strlen(params->word);
|
||||||
|
if (params->minlength <= l &&
|
||||||
|
params->maxlength >= l)
|
||||||
|
{
|
||||||
|
oWordList.push_back(params->word);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* we now drive the search by exploring the dictionary */
|
||||||
|
const DAWG_EDGE *current = getEdgeAt<DAWG_EDGE>(edgeptr->ptr);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* the current letter is current->chr */
|
||||||
|
next_state = params->automaton_field->getNextState(state, current->chr);
|
||||||
|
/* 1: the letter appears in the automaton as is */
|
||||||
|
if (next_state)
|
||||||
|
{
|
||||||
|
params->word[params->wordlen] = current->chr + 'a' - 1;
|
||||||
|
params->wordlen ++;
|
||||||
|
searchRegexpRecTempl(params, next_state, current, oWordList);
|
||||||
|
params->wordlen --;
|
||||||
|
params->word[params->wordlen] = '\0';
|
||||||
|
}
|
||||||
|
} while (!(*current++).last);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::searchRegExpInner(const string &iRegexp,
|
||||||
|
list<string> &oWordList,
|
||||||
|
struct search_RegE_list_t *iList) const
|
||||||
|
{
|
||||||
|
int ptl[REGEXP_MAX+1];
|
||||||
|
int PS [REGEXP_MAX+1];
|
||||||
|
|
||||||
|
/* (expr)# */
|
||||||
|
char stringbuf[250];
|
||||||
|
sprintf(stringbuf, "(%s)#", iRegexp.c_str());
|
||||||
|
for (int i = 0; i < REGEXP_MAX; i++)
|
||||||
|
{
|
||||||
|
PS[i] = 0;
|
||||||
|
ptl[i] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct regexp_error_report_t report;
|
||||||
|
report.pos1 = 0;
|
||||||
|
report.pos2 = 0;
|
||||||
|
report.msg[0] = '\0';
|
||||||
|
|
||||||
|
/* parsing */
|
||||||
|
yyscan_t scanner;
|
||||||
|
regexplex_init( &scanner );
|
||||||
|
YY_BUFFER_STATE buf = regexp_scan_string(stringbuf, scanner);
|
||||||
|
NODE *root = NULL;
|
||||||
|
int value = regexpparse(scanner , &root, iList, &report);
|
||||||
|
regexp_delete_buffer(buf, scanner);
|
||||||
|
regexplex_destroy(scanner);
|
||||||
|
|
||||||
|
if (value)
|
||||||
|
{
|
||||||
|
#ifdef DEBUG_FLEX_IS_BROKEN
|
||||||
|
fprintf(stderr, "parser error at pos %d - %d: %s\n",
|
||||||
|
report.pos1, report.pos2, report.msg);
|
||||||
|
#endif
|
||||||
|
regexp_delete_tree(root);
|
||||||
|
return ;
|
||||||
|
}
|
||||||
|
|
||||||
|
int n = 1;
|
||||||
|
int p = 1;
|
||||||
|
regexp_parcours(root, &p, &n, ptl);
|
||||||
|
PS [0] = p - 1;
|
||||||
|
ptl[0] = p - 1;
|
||||||
|
|
||||||
|
regexp_possuivante(root, PS);
|
||||||
|
|
||||||
|
Automaton *a = new Automaton(root->PP, ptl, PS, iList);
|
||||||
|
if (a)
|
||||||
|
{
|
||||||
|
struct params_regexp_t params;
|
||||||
|
params.minlength = iList->minlength;
|
||||||
|
params.maxlength = iList->maxlength;
|
||||||
|
params.automaton_field = a;
|
||||||
|
params.charlist = iList;
|
||||||
|
memset(params.word, '\0', sizeof(params.word));
|
||||||
|
params.wordlen = 0;
|
||||||
|
if (getHeader().getVersion() == 0)
|
||||||
|
{
|
||||||
|
searchRegexpRecTempl(¶ms, a->getInitId(),
|
||||||
|
getEdgeAt<DicEdgeOld>(getRoot()), oWordList);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
searchRegexpRecTempl(¶ms, a->getInitId(),
|
||||||
|
getEdgeAt<DicEdge>(getRoot()), oWordList);
|
||||||
|
}
|
||||||
|
|
||||||
|
delete a;
|
||||||
|
}
|
||||||
|
regexp_delete_tree(root);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Dictionary::searchRegExp(const wstring &iRegexp,
|
||||||
|
list<wstring> &oWordList,
|
||||||
|
struct search_RegE_list_t *iList) const
|
||||||
|
{
|
||||||
|
if (iRegexp == L"")
|
||||||
|
return;
|
||||||
|
|
||||||
|
list<string> tmpWordList;
|
||||||
|
// Do the actual work
|
||||||
|
searchRegExpInner(convertToMb(iRegexp), tmpWordList, iList);
|
||||||
|
|
||||||
|
list<string>::const_iterator it;
|
||||||
|
for (it = tmpWordList.begin(); it != tmpWordList.end(); it++)
|
||||||
|
{
|
||||||
|
oWordList.push_back(convertToWc(*it));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
131
dic/dic_search.h
131
dic/dic_search.h
|
@ -1,131 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file dic_search.h
|
|
||||||
* \brief Dictionary lookup functions
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2002
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _DIC_SEARCH_H_
|
|
||||||
#define _DIC_SEARCH_H_
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* number of results for Rack+1 search (Dic_search_7pl1)
|
|
||||||
*/
|
|
||||||
#define RES_7PL1_MAX 200
|
|
||||||
|
|
||||||
/**
|
|
||||||
* number of results for Extensions search (Dic_search_Racc)
|
|
||||||
*/
|
|
||||||
#define RES_RACC_MAX 100
|
|
||||||
|
|
||||||
/**
|
|
||||||
* number of results for Benjamin search (Dic_search_Benj)
|
|
||||||
*/
|
|
||||||
#define RES_BENJ_MAX 100
|
|
||||||
|
|
||||||
/**
|
|
||||||
* number of results for CrossWords search (Dic_search_Cros)
|
|
||||||
*/
|
|
||||||
#define RES_CROS_MAX 200
|
|
||||||
|
|
||||||
/**
|
|
||||||
* number of results for Regular Expression search (Dic_search_RegE)
|
|
||||||
*/
|
|
||||||
#define RES_REGE_MAX 200
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for a word in the dictionnary
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param path : lookup word
|
|
||||||
* @return 1 present, 0 error
|
|
||||||
*/
|
|
||||||
int Dic_search_word(Dictionary dic,
|
|
||||||
const wchar_t* path);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for all feasible word with "rack" plus one letter
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param rack : letters
|
|
||||||
* @param wordlist : results
|
|
||||||
*/
|
|
||||||
void Dic_search_7pl1(Dictionary dic,
|
|
||||||
const wchar_t* rack,
|
|
||||||
wchar_t wordlist[DIC_LETTERS][RES_7PL1_MAX][DIC_WORD_MAX],
|
|
||||||
int joker);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for all feasible word adding a letter in front or at the end
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param word : word
|
|
||||||
* @param wordlist : results
|
|
||||||
*/
|
|
||||||
void Dic_search_Racc(Dictionary dic,
|
|
||||||
const wchar_t* word,
|
|
||||||
wchar_t wordlist[RES_RACC_MAX][DIC_WORD_MAX]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for benjamins
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param rack : letters
|
|
||||||
* @param wordlist : results
|
|
||||||
*/
|
|
||||||
void Dic_search_Benj(Dictionary dic,
|
|
||||||
const wchar_t* word,
|
|
||||||
wchar_t wordlist[RES_BENJ_MAX][DIC_WORD_MAX]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for crosswords
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param rack : letters
|
|
||||||
* @param wordlist : results
|
|
||||||
*/
|
|
||||||
void Dic_search_Cros(Dictionary dic,
|
|
||||||
const wchar_t* mask,
|
|
||||||
wchar_t wordlist[RES_CROS_MAX][DIC_WORD_MAX]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search for words matching a regular expression
|
|
||||||
* @param dic : dictionary
|
|
||||||
* @param re : regular expression
|
|
||||||
* @param wordlist : results
|
|
||||||
*/
|
|
||||||
void Dic_search_RegE(Dictionary dic,
|
|
||||||
const wchar_t* re,
|
|
||||||
wchar_t wordlist[RES_REGE_MAX][DIC_WORD_MAX],
|
|
||||||
struct search_RegE_list_t *list);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Internal version of Dic_search_RegE, used inside the dictionary.
|
|
||||||
* Please use Dic_search_RegE instead from outside the dic library.
|
|
||||||
*/
|
|
||||||
void Dic_search_RegE_inner(const Dictionary dic, const char* re,
|
|
||||||
char wordlist[RES_REGE_MAX][DIC_WORD_MAX],
|
|
||||||
struct search_RegE_list_t *list);
|
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _DIC_SEARCH_H_ */
|
|
396
dic/encoding.cpp
Normal file
396
dic/encoding.cpp
Normal file
|
@ -0,0 +1,396 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file encoding.cpp
|
||||||
|
* \brief Utility functions to ease handling of wide-character strings
|
||||||
|
* \author Olivier Teuliere
|
||||||
|
* \date 2005
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstdarg>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cwchar>
|
||||||
|
#include <cwctype>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <iconv.h>
|
||||||
|
|
||||||
|
#ifdef WIN32
|
||||||
|
#include <windows.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "dic_exception.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef WIN32
|
||||||
|
// Utility function to get the last system error as a string
|
||||||
|
static string GetWin32Error()
|
||||||
|
{
|
||||||
|
char *lpMsgBuf;
|
||||||
|
DWORD dw = GetLastError();
|
||||||
|
cerr << dw << endl;
|
||||||
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL, dw,
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0, NULL);
|
||||||
|
string msg = lpMsgBuf;
|
||||||
|
LocalFree(lpMsgBuf);
|
||||||
|
return msg;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
#if !HAVE_WCWIDTH
|
||||||
|
// wcwidth replacement (for win32 in particular)
|
||||||
|
// Inspired from the gnulib package, without some of the refinements
|
||||||
|
static inline int wcwidth(wchar_t c)
|
||||||
|
{
|
||||||
|
// Assume all the printable characters have width 1
|
||||||
|
return c == 0 ? 0 : (iswprint(c) ? 1 : -1);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
int _wtoi(const wchar_t *iWStr)
|
||||||
|
{
|
||||||
|
return wcstol(iWStr, NULL, 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int _swprintf(wchar_t *wcs, size_t maxlen, const wchar_t *format, ...)
|
||||||
|
{
|
||||||
|
int res;
|
||||||
|
va_list argp;
|
||||||
|
va_start(argp, format);
|
||||||
|
#ifdef WIN32
|
||||||
|
// Mingw32 does not take the maxlen argument
|
||||||
|
res = vswprintf(wcs, format, argp);
|
||||||
|
#else
|
||||||
|
res = vswprintf(wcs, maxlen, format, argp);
|
||||||
|
#endif
|
||||||
|
va_end(argp);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wchar_t *_wcstok(wchar_t *wcs, const wchar_t *delim, wchar_t **ptr)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
// Mingw32 does not take the third argument
|
||||||
|
return wcstok(wcs, delim);
|
||||||
|
#else
|
||||||
|
return wcstok(wcs, delim, ptr);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#define _MAX_SIZE_FOR_STACK_ 30
|
||||||
|
wstring convertToWc(const string& iStr)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
// XXX: Assume the input is in UTF-8
|
||||||
|
return readFromUTF8(iStr.c_str(), iStr.size(), "convertToWc");
|
||||||
|
#else
|
||||||
|
// Get the needed length (we _can't_ use string::size())
|
||||||
|
size_t len = mbstowcs(NULL, iStr.c_str(), 0);
|
||||||
|
if (len == (size_t)-1)
|
||||||
|
return L"";
|
||||||
|
|
||||||
|
// Change the allocation method depending on the length of the string
|
||||||
|
if (len < _MAX_SIZE_FOR_STACK_)
|
||||||
|
{
|
||||||
|
// Without multi-thread, we can use static storage
|
||||||
|
static wchar_t tmp[_MAX_SIZE_FOR_STACK_];
|
||||||
|
len = mbstowcs(tmp, iStr.c_str(), len + 1);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
wchar_t *tmp = new wchar_t[len + 1];
|
||||||
|
len = mbstowcs(tmp, iStr.c_str(), len + 1);
|
||||||
|
wstring res = tmp;
|
||||||
|
delete[] tmp;
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string convertToMb(const wstring& iWStr)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
const unsigned int size = iWStr.size() * 4;
|
||||||
|
char buf[size];
|
||||||
|
// XXX: Assume the output is in UTF-8
|
||||||
|
int nb = writeInUTF8(iWStr, buf, size, "convertToMb");
|
||||||
|
return string(buf, nb);
|
||||||
|
#else
|
||||||
|
// Get the needed length (we _can't_ use wstring::size())
|
||||||
|
size_t len = wcstombs(NULL, iWStr.c_str(), 0);
|
||||||
|
if (len == (size_t)-1)
|
||||||
|
return "";
|
||||||
|
|
||||||
|
// Change the allocation method depending on the length of the string
|
||||||
|
if (len < _MAX_SIZE_FOR_STACK_)
|
||||||
|
{
|
||||||
|
// Without multi-thread, we can use static storage
|
||||||
|
static char tmp[_MAX_SIZE_FOR_STACK_];
|
||||||
|
len = wcstombs(tmp, iWStr.c_str(), len + 1);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
char *tmp = new char[len + 1];
|
||||||
|
len = wcstombs(tmp, iWStr.c_str(), len + 1);
|
||||||
|
string res = tmp;
|
||||||
|
delete[] tmp;
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#undef _MAX_SIZE_FOR_STACK_
|
||||||
|
|
||||||
|
|
||||||
|
string convertToMb(wchar_t iWChar)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
return convertToMb(wstring(1, iWChar));
|
||||||
|
#else
|
||||||
|
char res[MB_CUR_MAX + 1];
|
||||||
|
int len = wctomb(res, iWChar);
|
||||||
|
if (len == -1)
|
||||||
|
return "";
|
||||||
|
res[len] = '\0';
|
||||||
|
|
||||||
|
return res;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string truncString(const string &iStr, unsigned int iMaxWidth)
|
||||||
|
{
|
||||||
|
// Heuristic: the width of a character cannot exceed the number of
|
||||||
|
// bytes used to represent it (even in UTF-8)
|
||||||
|
if (iStr.size() <= iMaxWidth)
|
||||||
|
return iStr;
|
||||||
|
return truncAndConvert(convertToWc(iStr), iMaxWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string truncAndConvert(const wstring &iWstr, unsigned int iMaxWidth)
|
||||||
|
{
|
||||||
|
unsigned int width = 0;
|
||||||
|
unsigned int pos;
|
||||||
|
for (pos = 0; pos < iWstr.size(); ++pos)
|
||||||
|
{
|
||||||
|
int n = wcwidth(iWstr[pos]);
|
||||||
|
if (n == -1)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "truncAndConvert: non printable character: " << iWstr[pos];
|
||||||
|
// XXX: Should we throw an exception instead? Just ignore the problem?
|
||||||
|
cerr << ss.str() << endl;;
|
||||||
|
//throw DicException(ss.str());
|
||||||
|
return convertToMb(iWstr);
|
||||||
|
}
|
||||||
|
if (width + n > iMaxWidth)
|
||||||
|
break;
|
||||||
|
width += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return convertToMb(iWstr.substr(0, pos));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string truncOrPad(const string &iStr, unsigned int iWidth, char iChar)
|
||||||
|
{
|
||||||
|
wstring wstr = convertToWc(iStr);
|
||||||
|
unsigned int width = 0;
|
||||||
|
unsigned int pos;
|
||||||
|
for (pos = 0; pos < wstr.size(); ++pos)
|
||||||
|
{
|
||||||
|
int n = wcwidth(wstr[pos]);
|
||||||
|
if (n == -1)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "truncAndConvert: non printable character: " << wstr[pos];
|
||||||
|
// XXX: Should we throw an exception instead? Just ignore the problem?
|
||||||
|
cerr << ss.str() << endl;;
|
||||||
|
//throw DicException(ss.str());
|
||||||
|
return convertToMb(wstr);
|
||||||
|
}
|
||||||
|
if (width + n > iWidth)
|
||||||
|
break;
|
||||||
|
width += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iWidth > width)
|
||||||
|
return convertToMb(wstr.substr(0, pos)) + string(iWidth - width, iChar);
|
||||||
|
else
|
||||||
|
return convertToMb(wstr.substr(0, pos));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
string padAndConvert(const wstring &iWstr, unsigned int iLength,
|
||||||
|
bool iLeftPad, char c)
|
||||||
|
{
|
||||||
|
int width = 0;
|
||||||
|
for (unsigned int i = 0; i < iWstr.size(); ++i)
|
||||||
|
{
|
||||||
|
int n = wcwidth(iWstr[i]);
|
||||||
|
if (n == -1)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "padAndConvert: non printable character: " << iWstr[i];
|
||||||
|
// XXX: Should we throw an exception instead? Just ignore the problem?
|
||||||
|
cerr << ss.str() << endl;;
|
||||||
|
//throw DicException(ss.str());
|
||||||
|
return convertToMb(iWstr);
|
||||||
|
}
|
||||||
|
width += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((unsigned int)width >= iLength)
|
||||||
|
return convertToMb(iWstr);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Padding is needed
|
||||||
|
string s(iLength - width, c);
|
||||||
|
if (iLeftPad)
|
||||||
|
return s + convertToMb(iWstr);
|
||||||
|
else
|
||||||
|
return convertToMb(iWstr) + s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int readFromUTF8(wchar_t *oString, unsigned int iWideSize,
|
||||||
|
const char *iBuffer, unsigned int iBufSize,
|
||||||
|
const string &iContext)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
int res = MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, iBuffer,
|
||||||
|
iBufSize, oString, iWideSize);
|
||||||
|
if (res == 0)
|
||||||
|
{
|
||||||
|
// Retrieve the system error message for the last-error code
|
||||||
|
throw DicException("readFromUTF8: MultiByteToWideChar failed (" +
|
||||||
|
iContext + "): " + GetWin32Error());
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
#else
|
||||||
|
iconv_t handle = iconv_open("WCHAR_T", "UTF-8");
|
||||||
|
if (handle == (iconv_t)(-1))
|
||||||
|
throw DicException("readFromUTF8: iconv_open failed");
|
||||||
|
size_t inChars = iBufSize;
|
||||||
|
size_t outChars = iWideSize * sizeof(wchar_t);
|
||||||
|
// Use the ICONV_CONST trick because the declaration of iconv()
|
||||||
|
// differs depending on the implementations...
|
||||||
|
ICONV_CONST char *in = const_cast<ICONV_CONST char*>(iBuffer);
|
||||||
|
char *out = (char*)oString;
|
||||||
|
size_t res = iconv(handle, &in, &inChars, &out, &outChars);
|
||||||
|
iconv_close(handle);
|
||||||
|
// Problem during encoding conversion?
|
||||||
|
if (res == (size_t)(-1))
|
||||||
|
{
|
||||||
|
throw DicException("readFromUTF8: iconv failed (" +
|
||||||
|
iContext + "): " + string(strerror(errno)));
|
||||||
|
}
|
||||||
|
return iWideSize - outChars / sizeof(wchar_t);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wstring readFromUTF8(const char *iBuffer, unsigned int iBufSize,
|
||||||
|
const string &iContext)
|
||||||
|
{
|
||||||
|
// Temporary buffer for output
|
||||||
|
// We will have at most as many characters as in the UTF-8 string
|
||||||
|
wchar_t *wideBuf = new wchar_t[iBufSize];
|
||||||
|
unsigned int number;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
number = readFromUTF8(wideBuf, iBufSize, iBuffer, iBufSize, iContext);
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
// Make sure not to leak
|
||||||
|
delete[] wideBuf;
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
// Copy the string
|
||||||
|
wstring res(wideBuf, number);
|
||||||
|
delete[] wideBuf;
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int writeInUTF8(const wstring &iWString, char *oBuffer,
|
||||||
|
unsigned int iBufSize, const string &iContext)
|
||||||
|
{
|
||||||
|
#ifdef WIN32
|
||||||
|
int res = WideCharToMultiByte(CP_UTF8, 0, iWString.c_str(), iWString.size(),
|
||||||
|
oBuffer, iBufSize, NULL, NULL);
|
||||||
|
if (res == 0)
|
||||||
|
{
|
||||||
|
DWORD dw = GetLastError();
|
||||||
|
cerr << dw << endl;
|
||||||
|
// Retrieve the system error message for the last-error code
|
||||||
|
throw DicException("writeInUTF8: WideCharToMultiByte failed (" +
|
||||||
|
iContext + "): " + GetWin32Error());
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
#else
|
||||||
|
iconv_t handle = iconv_open("UTF-8", "WCHAR_T");
|
||||||
|
if (handle == (iconv_t)(-1))
|
||||||
|
throw DicException("writeInUTF8: iconv_open failed");
|
||||||
|
size_t length = iWString.size();
|
||||||
|
size_t inChars = sizeof(wchar_t) * length;
|
||||||
|
size_t outChars = iBufSize;
|
||||||
|
// Use the ICONV_CONST trick because the declaration of iconv()
|
||||||
|
// differs depending on the implementations...
|
||||||
|
// FIXME: bonus ugliness for doing 2 casts at once, and accessing string
|
||||||
|
// internals...
|
||||||
|
ICONV_CONST char *in = (ICONV_CONST char*)(&iWString[0]);
|
||||||
|
char *out = oBuffer;
|
||||||
|
size_t res = iconv(handle, &in, &inChars, &out, &outChars);
|
||||||
|
iconv_close(handle);
|
||||||
|
// Problem during encoding conversion?
|
||||||
|
if (res == (size_t)(-1))
|
||||||
|
{
|
||||||
|
throw DicException("writeInUTF8: iconv failed (" +
|
||||||
|
iContext + ")" + string(strerror(errno)));
|
||||||
|
}
|
||||||
|
// Return the number of written bytes
|
||||||
|
return iBufSize - outChars;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
122
dic/encoding.h
Normal file
122
dic/encoding.h
Normal file
|
@ -0,0 +1,122 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file encoding.h
|
||||||
|
* \brief Utility functions to ease manipulation of wide-character strings
|
||||||
|
* \author Olivier Teuliere
|
||||||
|
* \date 2005
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _ENCODING_H_
|
||||||
|
#define _ENCODING_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
using std::string;
|
||||||
|
using std::wstring;
|
||||||
|
|
||||||
|
|
||||||
|
/// Equivalent of atoi for wide-caracter strings
|
||||||
|
int _wtoi(const wchar_t *iWStr);
|
||||||
|
|
||||||
|
/// Equivalent of swprintf, but working also with mingw32
|
||||||
|
int _swprintf(wchar_t *wcs, size_t maxlen, const wchar_t *format, ...);
|
||||||
|
|
||||||
|
/// Equivalent of wcstok, but working also with mingw32
|
||||||
|
wchar_t *_wcstok(wchar_t *wcs, const wchar_t *delim, wchar_t **ptr);
|
||||||
|
|
||||||
|
/// Convert a multi-byte string into a wide-character string
|
||||||
|
wstring convertToWc(const string &iStr);
|
||||||
|
|
||||||
|
/// Convert a wide-character string into a multi-byte string
|
||||||
|
string convertToMb(const wstring &iWStr);
|
||||||
|
|
||||||
|
/// Convert a wide character into a multi-byte string
|
||||||
|
string convertToMb(wchar_t iWChar);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Truncate the given string to ensure that the number of columns needed
|
||||||
|
* to display it is at most iMaxWidth. If the string is already less wide,
|
||||||
|
* it is returned without truncation
|
||||||
|
*/
|
||||||
|
string truncString(const string &iStr, unsigned int iMaxWidth);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert the given string into a multi-byte one. If the number of columns
|
||||||
|
* needed to display the resulting string is more than iMaxWidth, truncate it
|
||||||
|
* on the right before conversion
|
||||||
|
*/
|
||||||
|
string truncAndConvert(const wstring &iWStr, unsigned int iMaxWidth);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make sure the displayed version of iStr has a width of iWidth.
|
||||||
|
* If the string is too long, truncate it, if it is too short, pad it
|
||||||
|
* with iChar
|
||||||
|
*/
|
||||||
|
string truncOrPad(const string &iStr, unsigned int iWidth, char iChar = ' ');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert the given string into a multi-byte one. If the number of columns
|
||||||
|
* needed to display the resulting string is less than iLength, pad it with
|
||||||
|
* the given character (defaulting to space)
|
||||||
|
*/
|
||||||
|
string padAndConvert(const wstring &iWstr, unsigned int iLength,
|
||||||
|
bool iLeftPad = true, char c = ' ');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utility function to convert a char* buffer encoded in UTF-8 into a
|
||||||
|
* wchar_t* string
|
||||||
|
* @param oString: where to write the converted string
|
||||||
|
* @param iWideSize: size available in oString (number of wchar_t)
|
||||||
|
* @param iBuffer: UTF-8 string to convert
|
||||||
|
* @param iBufSize: available size in iBuffer
|
||||||
|
* @param iContext: free text used in case of exception
|
||||||
|
* @return: number of wide chars actually written
|
||||||
|
*/
|
||||||
|
unsigned int readFromUTF8(wchar_t *oString, unsigned int iWideSize,
|
||||||
|
const char *iBuffer, unsigned int iBufSize,
|
||||||
|
const string &iContext);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Same as the other readFromUTF8 function, dealing with a wstring
|
||||||
|
* instead of a wchar_t*. Note that it performs an additional copy
|
||||||
|
* of the output string...
|
||||||
|
* @param iBuffer: UTF-8 string to convert
|
||||||
|
* @param iBufSize: available size in iBuffer
|
||||||
|
* @param iContext: free text used in case of exception
|
||||||
|
* @return: the converted wide string
|
||||||
|
*/
|
||||||
|
wstring readFromUTF8(const char *iBuffer, unsigned int iBufSize,
|
||||||
|
const string &iContext);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utility function to convert a wstring into an UTF-8 char* buffer
|
||||||
|
* @param iWString: the wide string to encode
|
||||||
|
* @param oBuffer: where to write the encoded string
|
||||||
|
* @param iBufSize: available size in oBuffer
|
||||||
|
* @param iContext: free text used in case of exception
|
||||||
|
* @return: number of bytes actually written
|
||||||
|
*/
|
||||||
|
unsigned int writeInUTF8(const wstring &iWString, char *oBuffer,
|
||||||
|
unsigned int iBufSize, const string &iContext);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
58
dic/erl.l
58
dic/erl.l
|
@ -1,58 +0,0 @@
|
||||||
%{
|
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Elit 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 "dic.h"
|
|
||||||
#include "regexp.h"
|
|
||||||
#include "libdic_a-ery.h"
|
|
||||||
|
|
||||||
#define MASK_TO_REMOVE 0x1F
|
|
||||||
|
|
||||||
%}
|
|
||||||
%option prefix="regexp"
|
|
||||||
%option outfile="lex.yy.c"
|
|
||||||
%option header-file="libdic_a-erl.h"
|
|
||||||
%option reentrant bison-bridge
|
|
||||||
%option bison-locations
|
|
||||||
%option noyywrap nounput
|
|
||||||
|
|
||||||
/* TODO : remove lexer translation */
|
|
||||||
alphabet [a-zA-Z]
|
|
||||||
%%
|
|
||||||
|
|
||||||
{alphabet} {yylval_param->c=(yytext[0]&MASK_TO_REMOVE); return LEX_CHAR;}
|
|
||||||
"[" {return LEX_L_SQBRACKET;}
|
|
||||||
"]" {return LEX_R_SQBRACKET;}
|
|
||||||
"(" {return LEX_L_BRACKET;}
|
|
||||||
")" {return LEX_R_BRACKET;}
|
|
||||||
"^" {return LEX_HAT;}
|
|
||||||
|
|
||||||
"." {return LEX_ALL;}
|
|
||||||
":v:" {return LEX_VOWL;}
|
|
||||||
":c:" {return LEX_CONS;}
|
|
||||||
":1:" {return LEX_USER1;}
|
|
||||||
":2:" {return LEX_USER2;}
|
|
||||||
|
|
||||||
"?" {return LEX_QMARK;}
|
|
||||||
"+" {return LEX_PLUS;}
|
|
||||||
"*" {return LEX_STAR;}
|
|
||||||
|
|
||||||
"#" {return LEX_SHARP;}
|
|
||||||
%%
|
|
||||||
|
|
59
dic/erl.lpp
Normal file
59
dic/erl.lpp
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
%{
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Antoine Fraboulet
|
||||||
|
* Authors: Antoine Fraboulet
|
||||||
|
*
|
||||||
|
* 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 "dic.h"
|
||||||
|
#include "regexp.h"
|
||||||
|
#include "libdic_a-ery.h"
|
||||||
|
|
||||||
|
#define MASK_TO_REMOVE 0x1F
|
||||||
|
|
||||||
|
%}
|
||||||
|
%option prefix="regexp"
|
||||||
|
%option outfile="lex.yy.c"
|
||||||
|
%option header-file="libdic_a-erl.h"
|
||||||
|
%option reentrant bison-bridge
|
||||||
|
%option bison-locations
|
||||||
|
%option noyywrap nounput
|
||||||
|
|
||||||
|
/* TODO : remove lexer translation */
|
||||||
|
alphabet [a-zA-Z]
|
||||||
|
%%
|
||||||
|
|
||||||
|
{alphabet} {yylval_param->c=(yytext[0]&MASK_TO_REMOVE); return LEX_CHAR;}
|
||||||
|
"[" {return LEX_L_SQBRACKET;}
|
||||||
|
"]" {return LEX_R_SQBRACKET;}
|
||||||
|
"(" {return LEX_L_BRACKET;}
|
||||||
|
")" {return LEX_R_BRACKET;}
|
||||||
|
"^" {return LEX_HAT;}
|
||||||
|
|
||||||
|
"." {return LEX_ALL;}
|
||||||
|
":v:" {return LEX_VOWL;}
|
||||||
|
":c:" {return LEX_CONS;}
|
||||||
|
":1:" {return LEX_USER1;}
|
||||||
|
":2:" {return LEX_USER2;}
|
||||||
|
|
||||||
|
"?" {return LEX_QMARK;}
|
||||||
|
"+" {return LEX_PLUS;}
|
||||||
|
"*" {return LEX_STAR;}
|
||||||
|
|
||||||
|
"#" {return LEX_SHARP;}
|
||||||
|
%%
|
||||||
|
|
|
@ -1,27 +1,28 @@
|
||||||
%{
|
%{
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 2005 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet
|
||||||
/* */
|
*
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
* 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 */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* (at your option) any later version. */
|
* (at your option) any later version.
|
||||||
/* */
|
*
|
||||||
/* Elit is distributed in the hope that it will be useful, */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
/* GNU General Public License for more details. */
|
* GNU General Public License for more details.
|
||||||
/* */
|
*
|
||||||
/* You should have received a copy of the GNU General Public License */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* along with this program; if not, write to the Free Software */
|
* along with this program; if not, write to the Free Software
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include "dic.h"
|
#include "dic.h"
|
||||||
#include "regexp.h"
|
#include "regexp.h"
|
163
dic/hashtable.c
163
dic/hashtable.c
|
@ -1,163 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file hashtable.c
|
|
||||||
* \brief Simple hashtable type
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 1999
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include "hashtable.h"
|
|
||||||
|
|
||||||
typedef struct _Hash_node {
|
|
||||||
struct _Hash_node *next;
|
|
||||||
void* key;
|
|
||||||
unsigned int keysize;
|
|
||||||
void* value;
|
|
||||||
unsigned int valuesize;
|
|
||||||
} Hash_node;
|
|
||||||
|
|
||||||
struct _Hash_table {
|
|
||||||
unsigned int size;
|
|
||||||
Hash_node** nodes;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
Hash_table
|
|
||||||
hash_init(unsigned int size)
|
|
||||||
{
|
|
||||||
Hash_table ht;
|
|
||||||
|
|
||||||
ht = (Hash_table) calloc(1,sizeof(struct _Hash_table));
|
|
||||||
ht->size = size;
|
|
||||||
ht->nodes = (Hash_node **) calloc (size, sizeof (Hash_node*));
|
|
||||||
return ht;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
hash_rec_free(Hash_node* node)
|
|
||||||
{
|
|
||||||
if (node)
|
|
||||||
{
|
|
||||||
if (node->next)
|
|
||||||
hash_rec_free(node->next);
|
|
||||||
if (node->key)
|
|
||||||
free(node->key);
|
|
||||||
if (node->value)
|
|
||||||
free(node->value);
|
|
||||||
free(node);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
hash_destroy(Hash_table hashtable)
|
|
||||||
{
|
|
||||||
unsigned int i;
|
|
||||||
if (hashtable)
|
|
||||||
{
|
|
||||||
for(i=0; i<hashtable->size; i++)
|
|
||||||
if (hashtable->nodes[i])
|
|
||||||
hash_rec_free(hashtable->nodes[i]);
|
|
||||||
if (hashtable->nodes)
|
|
||||||
free(hashtable->nodes);
|
|
||||||
free(hashtable);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static unsigned int
|
|
||||||
hash_key(Hash_table hashtable, void* ptr, unsigned int size)
|
|
||||||
{
|
|
||||||
unsigned int i;
|
|
||||||
unsigned int key = 0;
|
|
||||||
|
|
||||||
if (size % 4 == 0)
|
|
||||||
{
|
|
||||||
unsigned int *v = (unsigned int*)ptr;
|
|
||||||
for (i = 0; i < (size / 4); i++)
|
|
||||||
key ^= (key << 3) ^ (key >> 1) ^ v[i];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
unsigned char *v = (unsigned char*)ptr;
|
|
||||||
for (i = 0; i < size; i++)
|
|
||||||
key ^= (key << 3) ^ (key >> 1) ^ v[i];
|
|
||||||
}
|
|
||||||
key %= hashtable->size;
|
|
||||||
return key;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void*
|
|
||||||
hash_find(Hash_table hashtable, void* key, unsigned int keysize)
|
|
||||||
{
|
|
||||||
Hash_node *entry;
|
|
||||||
unsigned int h_key;
|
|
||||||
|
|
||||||
h_key = hash_key(hashtable,key,keysize);
|
|
||||||
for (entry = hashtable->nodes[h_key]; entry; entry = entry -> next)
|
|
||||||
{
|
|
||||||
if ((entry -> keysize == keysize) &&
|
|
||||||
(memcmp(entry->key,key,keysize) == 0))
|
|
||||||
{
|
|
||||||
return entry->value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static Hash_node*
|
|
||||||
new_entry(void* key, unsigned int keysize, void* value, unsigned int
|
|
||||||
valuesize)
|
|
||||||
{
|
|
||||||
Hash_node *n;
|
|
||||||
n = (Hash_node*)calloc(1,sizeof(Hash_node));
|
|
||||||
n->key = (void*)malloc(keysize);
|
|
||||||
n->value = (void*)malloc(valuesize);
|
|
||||||
n->keysize = keysize;
|
|
||||||
n->valuesize = valuesize;
|
|
||||||
memcpy(n->key,key,keysize);
|
|
||||||
memcpy(n->value,value,valuesize);
|
|
||||||
return n;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
hash_add(Hash_table hashtable,
|
|
||||||
void* key, unsigned int keysize,
|
|
||||||
void* value, unsigned int valuesize)
|
|
||||||
{
|
|
||||||
Hash_node *entry;
|
|
||||||
unsigned int h_key;
|
|
||||||
|
|
||||||
h_key = hash_key(hashtable,key,keysize);
|
|
||||||
entry = new_entry(key,keysize,value,valuesize);
|
|
||||||
entry->next = hashtable->nodes[h_key];
|
|
||||||
hashtable->nodes[h_key] = entry;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
52
dic/hashtable.cpp
Normal file
52
dic/hashtable.cpp
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 1999-2007 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file hashtable.c
|
||||||
|
* \brief Simple hashtable type
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 1999
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "hashtable.h"
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int HashPtr(const void *iPtr, unsigned int iSize)
|
||||||
|
{
|
||||||
|
unsigned int key = 0;
|
||||||
|
|
||||||
|
if (iSize % sizeof(unsigned int) == 0)
|
||||||
|
{
|
||||||
|
const unsigned int *ptr =
|
||||||
|
reinterpret_cast<const unsigned int*>(iPtr);
|
||||||
|
for (unsigned int i = 0; i < (iSize / sizeof(unsigned int)); ++i)
|
||||||
|
key ^= (key << 3) ^ (key >> 1) ^ ptr[i];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const unsigned char *ptr =
|
||||||
|
reinterpret_cast<const unsigned char*>(iPtr);
|
||||||
|
for (unsigned int i = 0; i < iSize; ++i)
|
||||||
|
key ^= (key << 3) ^ (key >> 1) ^ ptr[i];
|
||||||
|
}
|
||||||
|
return key;
|
||||||
|
}
|
||||||
|
|
100
dic/hashtable.h
100
dic/hashtable.h
|
@ -1,21 +1,23 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file hashtable.h
|
* \file hashtable.h
|
||||||
|
@ -26,21 +28,55 @@
|
||||||
|
|
||||||
#ifndef _HASHTABLE_H
|
#ifndef _HASHTABLE_H
|
||||||
#define _HASHTABLE_H
|
#define _HASHTABLE_H
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
typedef struct _Hash_table* Hash_table;
|
|
||||||
|
|
||||||
Hash_table hash_init(unsigned int);
|
/// Compute a hash for the data pointed to by iPtr
|
||||||
int hash_destroy(Hash_table);
|
/**
|
||||||
int hash_size(Hash_table);
|
* This function is useful to define the HASH_FCN template parameter
|
||||||
void* hash_find(Hash_table,void* key,unsigned keysize);
|
* of HashTable.
|
||||||
int hash_add (Hash_table,void* key,unsigned keysize,
|
*/
|
||||||
void* value,unsigned valuesize);
|
unsigned int HashPtr(const void *iPtr, unsigned int iSize);
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
class HashTable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// Constructor taking the number of records in the table
|
||||||
|
HashTable(unsigned int iSize);
|
||||||
|
|
||||||
|
/// Destructor
|
||||||
|
~HashTable();
|
||||||
|
|
||||||
|
/// Return the number of records in the table
|
||||||
|
unsigned int size() const { return m_size; }
|
||||||
|
|
||||||
|
/// Return the value corresponding to the given key, or NULL if not found
|
||||||
|
const VALUE *find(const KEY &iKey) const;
|
||||||
|
|
||||||
|
/// Add a new key/value pair (both the key and the value are copied)
|
||||||
|
void add(const KEY& iKey, const VALUE &iValue);
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Maximum number of records
|
||||||
|
unsigned int m_size;
|
||||||
|
|
||||||
|
/// Definition of a record
|
||||||
|
class Node
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Node(const KEY &iKey, const VALUE &iValue, const Node *iNext);
|
||||||
|
~Node();
|
||||||
|
KEY m_key;
|
||||||
|
VALUE m_value;
|
||||||
|
const Node *m_next;
|
||||||
|
};
|
||||||
|
|
||||||
|
/// All the nodes
|
||||||
|
const Node **m_nodes;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Include the implementation of the template
|
||||||
|
#include "hashtable.i"
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _HASHTABLE_H_ */
|
#endif /* _HASHTABLE_H_ */
|
||||||
|
|
88
dic/hashtable.i
Normal file
88
dic/hashtable.i
Normal file
|
@ -0,0 +1,88 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
|
* Authors: Antoine Fraboulet
|
||||||
|
* 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 <cstdlib>
|
||||||
|
|
||||||
|
#include "hashtable.h"
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
HashTable<KEY, VALUE, HASH_FCN>::HashTable(unsigned int iSize)
|
||||||
|
: m_size(iSize)
|
||||||
|
{
|
||||||
|
m_nodes = new const Node*[m_size];
|
||||||
|
for (unsigned int i = 0; i < m_size; ++i)
|
||||||
|
{
|
||||||
|
m_nodes[i] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
HashTable<KEY, VALUE, HASH_FCN>::~HashTable()
|
||||||
|
{
|
||||||
|
for (unsigned int i = 0; i < m_size; ++i)
|
||||||
|
delete m_nodes[i];
|
||||||
|
delete[] m_nodes;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
HashTable<KEY, VALUE, HASH_FCN>::Node::Node(const KEY &iKey, const VALUE &iValue, const Node *iNext)
|
||||||
|
: m_key(iKey), m_value(iValue), m_next(iNext)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
HashTable<KEY, VALUE, HASH_FCN>::Node::~Node()
|
||||||
|
{
|
||||||
|
delete m_next;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
const VALUE *HashTable<KEY, VALUE, HASH_FCN>::find(const KEY &iKey) const
|
||||||
|
{
|
||||||
|
HASH_FCN aHashFunc;
|
||||||
|
unsigned int h_key = aHashFunc(iKey) % m_size;
|
||||||
|
for (const Node *entry = m_nodes[h_key]; entry; entry = entry->m_next)
|
||||||
|
{
|
||||||
|
// Note: we need to be able to call == on a type KEY
|
||||||
|
if (entry->m_key == iKey)
|
||||||
|
{
|
||||||
|
return &entry->m_value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template<typename KEY, typename VALUE, typename HASH_FCN>
|
||||||
|
void HashTable<KEY, VALUE, HASH_FCN>::add(const KEY &iKey, const VALUE &iValue)
|
||||||
|
{
|
||||||
|
HASH_FCN aHashFunc;
|
||||||
|
unsigned int h_key = aHashFunc(iKey) % m_size;
|
||||||
|
const Node *entry = new Node(iKey, iValue, m_nodes[h_key]);
|
||||||
|
m_nodes[h_key] = entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
// vim: ft=cpp
|
526
dic/header.cpp
Normal file
526
dic/header.cpp
Normal file
|
@ -0,0 +1,526 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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 "config.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <sstream>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
// For ntohl & Co.
|
||||||
|
#ifdef WIN32
|
||||||
|
# include <winsock2.h>
|
||||||
|
#else
|
||||||
|
# if HAVE_NETINET_IN_H
|
||||||
|
# include <netinet/in.h>
|
||||||
|
# endif
|
||||||
|
# if HAVE_ARPA_INET_H
|
||||||
|
# include <arpa/inet.h>
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
# include <libintl.h>
|
||||||
|
# define _(String) gettext(String)
|
||||||
|
#else
|
||||||
|
# define _(String) String
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "header.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "dic_exception.h"
|
||||||
|
|
||||||
|
|
||||||
|
// Note: swap4 is duplicated in dic.cpp
|
||||||
|
#if defined(WORDS_BIGENDIAN)
|
||||||
|
static uint32_t swap4(uint32_t v)
|
||||||
|
{
|
||||||
|
uint32_t r;
|
||||||
|
uint8_t *pv = (uint8_t*)&v;
|
||||||
|
uint8_t *pr = (uint8_t*)&r;
|
||||||
|
|
||||||
|
pr[0] = pv[3];
|
||||||
|
pr[1] = pv[2];
|
||||||
|
pr[2] = pv[1];
|
||||||
|
pr[3] = pv[0];
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nothing to do on big-endian machines
|
||||||
|
# define ntohll(x) (x)
|
||||||
|
# define htonll(x) (x)
|
||||||
|
#else
|
||||||
|
static inline uint64_t htonll(uint64_t host64)
|
||||||
|
{
|
||||||
|
return (((uint64_t)htonl((host64 << 32) >> 32)) << 32) | htonl(host64 >> 32);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline uint64_t ntohll(uint64_t net64)
|
||||||
|
{
|
||||||
|
return htonll(net64);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Keyword included in dictionary headers
|
||||||
|
* Implies little endian storage on words
|
||||||
|
*/
|
||||||
|
#define _COMPIL_KEYWORD_ "_COMPILED_DICTIONARY_"
|
||||||
|
|
||||||
|
/** Old format of the header (i.e. version 0) */
|
||||||
|
struct Dict_header_old
|
||||||
|
{
|
||||||
|
/// Identification string
|
||||||
|
char ident[sizeof(_COMPIL_KEYWORD_)];
|
||||||
|
/// Version of the serialization format
|
||||||
|
uint8_t version;
|
||||||
|
/// Unused at the moment, reserved for further use
|
||||||
|
char unused;
|
||||||
|
uint32_t root;
|
||||||
|
uint32_t nwords;
|
||||||
|
/// Information about the compression
|
||||||
|
//@{
|
||||||
|
uint32_t edgesused;
|
||||||
|
uint32_t nodesused;
|
||||||
|
uint32_t nodessaved;
|
||||||
|
uint32_t edgessaved;
|
||||||
|
//@}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Do not change these values, as they impact the size of the structure!
|
||||||
|
// Note: they are chosen carefully to avoid alignment issues
|
||||||
|
#define _MAX_USER_HOST_ 32
|
||||||
|
#define _MAX_DIC_NAME_SIZE_ 30
|
||||||
|
#define _MAX_LETTERS_NB_ 63
|
||||||
|
#define _MAX_LETTERS_SIZE_ 80
|
||||||
|
|
||||||
|
/** Extension of the old format (used in version 1)*/
|
||||||
|
struct Dict_header_ext
|
||||||
|
{
|
||||||
|
// Time when the dictionary was compressed
|
||||||
|
// (number of seconds since the Epoch)
|
||||||
|
uint64_t compressDate;
|
||||||
|
// Build information
|
||||||
|
char userHost[_MAX_USER_HOST_];
|
||||||
|
// Size taken by the build information
|
||||||
|
uint32_t userHostSize;
|
||||||
|
|
||||||
|
// --- we have a multiple of 64 bytes here
|
||||||
|
|
||||||
|
// Compression algorithm (1 = DAWG, 2 = GADDAG)
|
||||||
|
uint8_t algorithm;
|
||||||
|
// Variant used in the rules (XXX: currently unused)
|
||||||
|
uint8_t variant;
|
||||||
|
|
||||||
|
// --- we have a multiple of 64 bytes here
|
||||||
|
|
||||||
|
// Dictionary official name and version (e.g.: ODS 5.0)
|
||||||
|
char dicName[_MAX_DIC_NAME_SIZE_];
|
||||||
|
// Size taken by the dictionary name
|
||||||
|
uint32_t dicNameSize;
|
||||||
|
|
||||||
|
// --- we have a multiple of 64 bytes here
|
||||||
|
|
||||||
|
// Letters used in the dictionary
|
||||||
|
// We should have: nbLetters <= lettersSize <= _MAX_LETTERS_SIZE_
|
||||||
|
// and: nbLetters <= _MAX_LETTERS_NB_
|
||||||
|
// The letters themselves, in UTF-8
|
||||||
|
char letters[_MAX_LETTERS_SIZE_];
|
||||||
|
// Size taken by the letters
|
||||||
|
uint32_t lettersSize;
|
||||||
|
// Number of letters (XXX: in theory useless, but allows a sanity check)
|
||||||
|
uint32_t nbLetters;
|
||||||
|
|
||||||
|
// --- we have a multiple of 64 bytes here
|
||||||
|
|
||||||
|
// Points of the letters (indexed by their code)
|
||||||
|
// The "+ 1" is there for struct alignment
|
||||||
|
uint8_t points[_MAX_LETTERS_NB_ + 1];
|
||||||
|
// Frequency of the letters (indexed by their code)
|
||||||
|
// The "+ 1" is there for struct alignment
|
||||||
|
uint8_t frequency[_MAX_LETTERS_NB_ + 1];
|
||||||
|
// Bitfield indicating whether letters are vowels
|
||||||
|
uint64_t vowels;
|
||||||
|
// Bitfield indicating whether letters are consonants
|
||||||
|
uint64_t consonants;
|
||||||
|
|
||||||
|
// --- we have a multiple of 64 bytes here
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Header::Header(istream &iStream)
|
||||||
|
: m_root(0), m_nbWords(0), m_nodesUsed(0), m_edgesUsed(0),
|
||||||
|
m_nodesSaved(0), m_edgesSaved(0), m_type(kDAWG)
|
||||||
|
{
|
||||||
|
// Simply delegate to the read() method
|
||||||
|
// The code is not moved here because I find it more natural to have a
|
||||||
|
// read() method symmetrical to the write() one
|
||||||
|
read(iStream);
|
||||||
|
buildMapCodeFromChar();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Header::Header(const DictHeaderInfo &iInfo)
|
||||||
|
{
|
||||||
|
// Use the latest serialization format
|
||||||
|
m_version = 1;
|
||||||
|
|
||||||
|
// Sanity checks
|
||||||
|
if (iInfo.letters.size() > _MAX_LETTERS_NB_)
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << _MAX_LETTERS_NB_;
|
||||||
|
throw DicException("Header::Header: Too many different letters for "
|
||||||
|
"the current format; only " + ss.str() +
|
||||||
|
" are supported");
|
||||||
|
}
|
||||||
|
if (iInfo.points.size() != iInfo.letters.size())
|
||||||
|
{
|
||||||
|
throw DicException("Header::Header: Different sizes for "
|
||||||
|
"iInfo.points and iInfo.letters");
|
||||||
|
}
|
||||||
|
if (iInfo.frequency.size() != iInfo.letters.size())
|
||||||
|
{
|
||||||
|
throw DicException("Header::Header: Different sizes for "
|
||||||
|
"iInfo.frequency and iInfo.letters");
|
||||||
|
}
|
||||||
|
if (iInfo.vowels.size() != iInfo.letters.size())
|
||||||
|
{
|
||||||
|
throw DicException("Header::Header: Different sizes for "
|
||||||
|
"iInfo.vowels and iInfo.letters");
|
||||||
|
}
|
||||||
|
if (iInfo.consonants.size() != iInfo.letters.size())
|
||||||
|
{
|
||||||
|
throw DicException("Header::Header: Different sizes for "
|
||||||
|
"iInfo.consonants and iInfo.letters");
|
||||||
|
}
|
||||||
|
|
||||||
|
m_compressDate = time(NULL);
|
||||||
|
m_userHost = convertToWc(ELIOT_COMPILE_BY + string("@") + ELIOT_COMPILE_HOST);
|
||||||
|
m_root = iInfo.root;
|
||||||
|
m_nbWords = iInfo.nwords;
|
||||||
|
m_nodesUsed = iInfo.nodesused;
|
||||||
|
m_edgesUsed = iInfo.edgesused;
|
||||||
|
m_nodesSaved = iInfo.nodessaved;
|
||||||
|
m_edgesSaved = iInfo.edgessaved;
|
||||||
|
m_type = iInfo.dawg ? kDAWG : kGADDAG;
|
||||||
|
m_dicName = iInfo.dicName;
|
||||||
|
m_letters = iInfo.letters;
|
||||||
|
m_points = iInfo.points;
|
||||||
|
m_frequency = iInfo.frequency;
|
||||||
|
m_vowels = iInfo.vowels;
|
||||||
|
m_consonants = iInfo.consonants;
|
||||||
|
|
||||||
|
buildMapCodeFromChar();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Header::buildMapCodeFromChar()
|
||||||
|
{
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
// We don't differentiate uppercase and lowercase letters
|
||||||
|
m_mapCodeFromChar[towlower(m_letters[i])] = i + 1;
|
||||||
|
m_mapCodeFromChar[towupper(m_letters[i])] = i + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wchar_t Header::getCharFromCode(unsigned int iCode) const
|
||||||
|
{
|
||||||
|
// Safety check
|
||||||
|
if (iCode == 0 || iCode > m_letters.size())
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << iCode;
|
||||||
|
throw DicException("Header::getCharFromCode: no letter for code " + ss.str());
|
||||||
|
}
|
||||||
|
return m_letters[iCode - 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Header::getCodeFromChar(wchar_t iChar) const
|
||||||
|
{
|
||||||
|
map<wchar_t, unsigned int>::const_iterator pair =
|
||||||
|
m_mapCodeFromChar.find(iChar);
|
||||||
|
if (pair == m_mapCodeFromChar.end())
|
||||||
|
{
|
||||||
|
throw DicException("Header::getCodeFromChar: No code for letter " +
|
||||||
|
convertToMb(iChar));
|
||||||
|
}
|
||||||
|
return pair->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Header::read(istream &iStream)
|
||||||
|
{
|
||||||
|
Dict_header_old aHeader;
|
||||||
|
iStream.read((char*)&aHeader, sizeof(Dict_header_old));
|
||||||
|
if (iStream.gcount() != sizeof(Dict_header_old))
|
||||||
|
throw DicException("Header::read: expected to read more bytes");
|
||||||
|
|
||||||
|
// Check the identification string
|
||||||
|
if (string(aHeader.ident) != _COMPIL_KEYWORD_)
|
||||||
|
throw DicException("Header::read: incorrect header keyword; is it a dictionary file?");
|
||||||
|
|
||||||
|
m_version = aHeader.version;
|
||||||
|
|
||||||
|
// Handle endianness
|
||||||
|
if (m_version == 0)
|
||||||
|
{
|
||||||
|
#if defined(WORDS_BIGENDIAN)
|
||||||
|
aHeader.root = swap4(aHeader.root);
|
||||||
|
aHeader.nwords = swap4(aHeader.nwords);
|
||||||
|
aHeader.nodesused = swap4(aHeader.nodesused);
|
||||||
|
aHeader.edgesused = swap4(aHeader.edgesused);
|
||||||
|
aHeader.nodessaved = swap4(aHeader.nodessaved);
|
||||||
|
aHeader.edgessaved = swap4(aHeader.edgessaved);
|
||||||
|
#endif
|
||||||
|
m_root = aHeader.root;
|
||||||
|
m_nbWords = aHeader.nwords;
|
||||||
|
m_nodesUsed = aHeader.nodesused;
|
||||||
|
m_edgesUsed = aHeader.edgesused;
|
||||||
|
m_nodesSaved = aHeader.nodessaved;
|
||||||
|
m_edgesSaved = aHeader.edgessaved;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_root = ntohl(aHeader.root);
|
||||||
|
m_nbWords = ntohl(aHeader.nwords);
|
||||||
|
m_nodesUsed = ntohl(aHeader.nodesused);
|
||||||
|
m_edgesUsed = ntohl(aHeader.edgesused);
|
||||||
|
m_nodesSaved = ntohl(aHeader.nodessaved);
|
||||||
|
m_edgesSaved = ntohl(aHeader.edgessaved);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_version == 0)
|
||||||
|
{
|
||||||
|
m_compressDate = 0;
|
||||||
|
m_userHost = convertToWc(_("Unknown (old format)"));
|
||||||
|
m_dicName = convertToWc(_("Unknown (old format)"));
|
||||||
|
|
||||||
|
// In version 0, the letters, points, frequency,
|
||||||
|
// vowels and consonants were hard-coded...
|
||||||
|
m_letters = convertToWc("ABCDEFGHIJKLMNOPQRSTUVWXYZ?");
|
||||||
|
|
||||||
|
static const uint8_t Frenchpoints[] =
|
||||||
|
{
|
||||||
|
// A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
||||||
|
1,3,3,2, 1,4,2,4,1,8,10,1,2,1,1,3,8,1,1,1,1,4,10,10,10,10,0
|
||||||
|
};
|
||||||
|
|
||||||
|
static const uint8_t FrenchFrequency[] =
|
||||||
|
{
|
||||||
|
// A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
||||||
|
9,2,2,3,15,2,2,2,8,1, 1,5,3,6,6,2,1,6,6,6,6,2, 1, 1, 1, 1,2
|
||||||
|
};
|
||||||
|
|
||||||
|
// The jokers and the 'Y' can be considered both as vowels or consonants
|
||||||
|
static const uint8_t FrenchVowels[] =
|
||||||
|
{
|
||||||
|
// A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
||||||
|
1,0,0,0, 1,0,0,0,1,0, 0,0,0,0,1,0,0,0,0,0,1,0, 0, 0, 1, 0,1
|
||||||
|
};
|
||||||
|
|
||||||
|
static const uint8_t FrenchConsonants[] =
|
||||||
|
{
|
||||||
|
// A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
||||||
|
0,1,1,1, 0,1,1,1,0,1, 1,1,1,1,0,1,1,1,1,1,0,1, 1, 1, 1, 1,1
|
||||||
|
};
|
||||||
|
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
m_points.push_back(Frenchpoints[i]);
|
||||||
|
m_frequency.push_back(FrenchFrequency[i]);
|
||||||
|
m_vowels.push_back(FrenchVowels[i]);
|
||||||
|
m_consonants.push_back(FrenchConsonants[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// This header doesn't use the old serialization format, so read the
|
||||||
|
// extension as well
|
||||||
|
Dict_header_ext aHeaderExt;
|
||||||
|
iStream.read((char*)&aHeaderExt, sizeof(Dict_header_ext));
|
||||||
|
if (iStream.gcount() != sizeof(Dict_header_ext))
|
||||||
|
throw DicException("Header::read: expected to read more bytes");
|
||||||
|
|
||||||
|
// Handle endianness in the extension
|
||||||
|
aHeaderExt.compressDate = ntohll(aHeaderExt.compressDate);
|
||||||
|
aHeaderExt.userHostSize = ntohl(aHeaderExt.userHostSize);
|
||||||
|
aHeaderExt.dicNameSize = ntohl(aHeaderExt.dicNameSize);
|
||||||
|
aHeaderExt.lettersSize = ntohl(aHeaderExt.lettersSize);
|
||||||
|
aHeaderExt.nbLetters = ntohl(aHeaderExt.nbLetters);
|
||||||
|
aHeaderExt.vowels = ntohll(aHeaderExt.vowels);
|
||||||
|
aHeaderExt.consonants = ntohll(aHeaderExt.consonants);
|
||||||
|
|
||||||
|
m_compressDate = aHeaderExt.compressDate;
|
||||||
|
|
||||||
|
if (aHeaderExt.algorithm == kDAWG)
|
||||||
|
m_type = kDAWG;
|
||||||
|
else if (aHeaderExt.algorithm == kGADDAG)
|
||||||
|
m_type = kGADDAG;
|
||||||
|
else
|
||||||
|
throw DicException("Header::read: unrecognized algorithm type");
|
||||||
|
|
||||||
|
m_userHost = readFromUTF8(aHeaderExt.userHost, aHeaderExt.userHostSize,
|
||||||
|
"user and host information");
|
||||||
|
|
||||||
|
// Convert the dictionary letters from UTF-8 to wchar_t*
|
||||||
|
m_dicName = readFromUTF8(aHeaderExt.dicName, aHeaderExt.dicNameSize,
|
||||||
|
"dictionary name");
|
||||||
|
|
||||||
|
// Convert the dictionary letters from UTF-8 to wchar_t*
|
||||||
|
m_letters = readFromUTF8(aHeaderExt.letters, aHeaderExt.lettersSize,
|
||||||
|
"dictionary letters");
|
||||||
|
// Safety check: correct number of letters?
|
||||||
|
if (m_letters.size() != aHeaderExt.nbLetters)
|
||||||
|
{
|
||||||
|
throw DicException("Header::read: inconsistent header");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Letters points and frequency
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
m_points.push_back(aHeaderExt.points[i]);
|
||||||
|
m_frequency.push_back(aHeaderExt.frequency[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Vowels and consonants
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
m_vowels.push_back(aHeaderExt.vowels & (1 << i));
|
||||||
|
m_consonants.push_back(aHeaderExt.consonants & (1 << i));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Header::write(ostream &oStream) const
|
||||||
|
{
|
||||||
|
Dict_header_old aHeader;
|
||||||
|
strcpy(aHeader.ident, _COMPIL_KEYWORD_);
|
||||||
|
aHeader.version = m_version;
|
||||||
|
aHeader.unused = 0;
|
||||||
|
aHeader.root = htonl(m_root);
|
||||||
|
aHeader.nwords = htonl(m_nbWords);
|
||||||
|
aHeader.nodesused = htonl(m_nodesUsed);
|
||||||
|
aHeader.edgesused = htonl(m_edgesUsed);
|
||||||
|
aHeader.nodessaved = htonl(m_nodesSaved);
|
||||||
|
aHeader.edgessaved = htonl(m_edgesSaved);
|
||||||
|
|
||||||
|
oStream.write((char*)&aHeader, sizeof(Dict_header_old));
|
||||||
|
if (!oStream.good())
|
||||||
|
throw DicException("Header::write: error when writing to file");
|
||||||
|
|
||||||
|
if (m_version != 0)
|
||||||
|
{
|
||||||
|
Dict_header_ext aHeaderExt;
|
||||||
|
aHeaderExt.compressDate = m_compressDate;
|
||||||
|
aHeaderExt.userHostSize =
|
||||||
|
writeInUTF8(m_userHost, aHeaderExt.userHost,
|
||||||
|
_MAX_USER_HOST_, "user and host information");
|
||||||
|
aHeaderExt.algorithm = m_type;
|
||||||
|
|
||||||
|
// Convert the dictionary name to UTF-8
|
||||||
|
aHeaderExt.dicNameSize =
|
||||||
|
writeInUTF8(m_dicName, aHeaderExt.dicName,
|
||||||
|
_MAX_DIC_NAME_SIZE_, "dictionary name");
|
||||||
|
|
||||||
|
// Convert the dictionary letters to UTF-8
|
||||||
|
aHeaderExt.lettersSize =
|
||||||
|
writeInUTF8(m_letters, aHeaderExt.letters,
|
||||||
|
_MAX_LETTERS_SIZE_, "dictionary letters");
|
||||||
|
aHeaderExt.nbLetters = (uint32_t)m_letters.size();
|
||||||
|
|
||||||
|
// Letters points and frequency
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
aHeaderExt.points[i] = m_points[i];
|
||||||
|
aHeaderExt.frequency[i] = m_frequency[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Vowels and consonants
|
||||||
|
aHeaderExt.vowels = 0;
|
||||||
|
aHeaderExt.consonants = 0;
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
if (m_vowels[i])
|
||||||
|
aHeaderExt.vowels |= 1 << i;
|
||||||
|
if (m_consonants[i])
|
||||||
|
aHeaderExt.consonants |= 1 << i;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle endianness in the extension
|
||||||
|
aHeaderExt.userHostSize = htonl(aHeaderExt.userHostSize);
|
||||||
|
aHeaderExt.compressDate = htonll(aHeaderExt.compressDate);
|
||||||
|
aHeaderExt.dicNameSize = htonl(aHeaderExt.dicNameSize);
|
||||||
|
aHeaderExt.lettersSize = htonl(aHeaderExt.lettersSize);
|
||||||
|
aHeaderExt.nbLetters = htonl(aHeaderExt.nbLetters);
|
||||||
|
aHeaderExt.vowels = htonll(aHeaderExt.vowels);
|
||||||
|
aHeaderExt.consonants = htonll(aHeaderExt.consonants);
|
||||||
|
|
||||||
|
// Write the extension
|
||||||
|
oStream.write((char*)&aHeaderExt, sizeof(Dict_header_ext));
|
||||||
|
if (!oStream.good())
|
||||||
|
throw DicException("Header::write: error when writing to file");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Header::print() const
|
||||||
|
{
|
||||||
|
printf(_("dictionary name: %s\n"), convertToMb(m_dicName).c_str());
|
||||||
|
if (m_version)
|
||||||
|
{
|
||||||
|
char buf[50];
|
||||||
|
strftime(buf, sizeof(buf), "%c", gmtime(&m_compressDate));
|
||||||
|
printf(_("compressed on: %s\n"), buf);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf(_("compressed on: Unknown date (old format)\n"));
|
||||||
|
}
|
||||||
|
printf(_("compressed using a binary compiled by: %s\n"), convertToMb(m_userHost).c_str());
|
||||||
|
printf(_("dictionary type: %s\n"), m_type == kDAWG ? "DAWG" : "GADDAG");
|
||||||
|
printf(_("letters: %s\n"), convertToMb(m_letters).c_str());
|
||||||
|
printf(_("number of letters: %lu\n"), (long unsigned int)m_letters.size());
|
||||||
|
printf(_("number of words: %d\n"), m_nbWords);
|
||||||
|
long unsigned int size =
|
||||||
|
sizeof(Dict_header_old) + (m_version ? sizeof(Dict_header_ext) : 0);
|
||||||
|
printf(_("header size: %lu bytes\n"), size);
|
||||||
|
printf(_("root: %d (edge)\n"), m_root);
|
||||||
|
printf(_("nodes: %d used + %d saved\n"), m_nodesUsed, m_nodesSaved);
|
||||||
|
printf(_("edges: %d used + %d saved\n"), m_edgesUsed, m_edgesSaved);
|
||||||
|
printf("===============================================\n");
|
||||||
|
printf(_("letter | points | frequency | vowel | consonant\n"));
|
||||||
|
printf("-------+--------+-----------+-------+----------\n");
|
||||||
|
for (unsigned int i = 0; i < m_letters.size(); ++i)
|
||||||
|
{
|
||||||
|
printf(" %s | %2d | %2d | %d | %d\n",
|
||||||
|
padAndConvert(wstring(1, m_letters[i]), 2).c_str(),
|
||||||
|
m_points[i], m_frequency[i], m_vowels[i], m_consonants[i]);
|
||||||
|
}
|
||||||
|
printf("===============================================\n");
|
||||||
|
}
|
||||||
|
|
191
dic/header.h
Normal file
191
dic/header.h
Normal file
|
@ -0,0 +1,191 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef _HEADER_H
|
||||||
|
#define _HEADER_H
|
||||||
|
|
||||||
|
#include <iosfwd>
|
||||||
|
#include <map>
|
||||||
|
#include <vector>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Structure used to create a Header object.
|
||||||
|
* Note: this structure doesn't pretend to map the way the data is stored in a
|
||||||
|
* file. For (de)serialization, always use a Header object
|
||||||
|
*/
|
||||||
|
struct DictHeaderInfo
|
||||||
|
{
|
||||||
|
uint32_t root;
|
||||||
|
uint32_t nwords;
|
||||||
|
uint32_t edgesused;
|
||||||
|
uint32_t nodesused;
|
||||||
|
uint32_t nodessaved;
|
||||||
|
uint32_t edgessaved;
|
||||||
|
bool dawg;
|
||||||
|
wstring dicName;
|
||||||
|
wstring letters;
|
||||||
|
vector<uint8_t> points;
|
||||||
|
vector<uint8_t> frequency;
|
||||||
|
vector<bool> vowels;
|
||||||
|
vector<bool> consonants;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dictionary header.
|
||||||
|
* There are 2 ways to create a Header object:
|
||||||
|
* - fill a DictHeaderInfo structure and give it in the constructor of the
|
||||||
|
* Header class (usually to call write() afterwards)
|
||||||
|
* - give it an input stream on a compiled dictionary
|
||||||
|
*
|
||||||
|
* The class is immutable: a Header object cannot be modified after creation.
|
||||||
|
*
|
||||||
|
* Serialization:
|
||||||
|
* Several formats of headers will be handled by this class, even though only
|
||||||
|
* the first one is handled at the moment. You can use the write() method to
|
||||||
|
* write the latest version of the header into a given file.
|
||||||
|
* When using the constructor taking an input stream, all the header versions
|
||||||
|
* are supported.
|
||||||
|
*/
|
||||||
|
class Header
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
/// Dictionary type
|
||||||
|
enum DictType
|
||||||
|
{
|
||||||
|
kDAWG = 1,
|
||||||
|
kGADDAG = 2
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor from an input stream
|
||||||
|
* @param iStream: Input stream where to read the header
|
||||||
|
*/
|
||||||
|
Header(istream &iStream);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor from a filled structure
|
||||||
|
* @param iInfo: info needed to build the header
|
||||||
|
*/
|
||||||
|
Header(const DictHeaderInfo &iInfo);
|
||||||
|
|
||||||
|
/// Return the version of the dictionary format
|
||||||
|
uint8_t getVersion() const { return m_version; }
|
||||||
|
|
||||||
|
/// Getters
|
||||||
|
//@{
|
||||||
|
unsigned int getRoot() const { return m_root; }
|
||||||
|
unsigned int getNbWords() const { return m_nbWords; }
|
||||||
|
unsigned int getNbNodesUsed() const { return m_nodesUsed; }
|
||||||
|
unsigned int getNbEdgesUsed() const { return m_edgesUsed; }
|
||||||
|
unsigned int getNbNodesSaved() const { return m_nodesSaved; }
|
||||||
|
unsigned int getNbEdgesSaved() const { return m_edgesSaved; }
|
||||||
|
DictType getType() const { return m_type; }
|
||||||
|
wstring getLetters() const { return m_letters; }
|
||||||
|
uint8_t getPoints(unsigned int iCode) const { return m_points[iCode - 1]; }
|
||||||
|
uint8_t getFrequency(unsigned int iCode) const { return m_frequency[iCode - 1]; }
|
||||||
|
bool isVowel(unsigned int iCode) const { return m_vowels[iCode - 1]; }
|
||||||
|
bool isConsonant(unsigned int iCode) const { return m_consonants[iCode - 1]; }
|
||||||
|
//@}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the letter corresponding to the given code
|
||||||
|
*/
|
||||||
|
wchar_t getCharFromCode(unsigned int iCode) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the code corresponding to the given letter
|
||||||
|
*/
|
||||||
|
unsigned int getCodeFromChar(wchar_t iChar) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Print a readable summary of the header on standard output
|
||||||
|
*/
|
||||||
|
void print() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write the header to a file, using the latest format
|
||||||
|
* @param oStream: Output stream where to write the header
|
||||||
|
* @exception: Throw a DicException in case of problem
|
||||||
|
*/
|
||||||
|
void write(ostream &oStream) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Version of the serialization
|
||||||
|
uint8_t m_version;
|
||||||
|
|
||||||
|
wstring m_userHost;
|
||||||
|
time_t m_compressDate;
|
||||||
|
|
||||||
|
uint32_t m_root;
|
||||||
|
uint32_t m_nbWords;
|
||||||
|
uint32_t m_nodesUsed;
|
||||||
|
uint32_t m_edgesUsed;
|
||||||
|
uint32_t m_nodesSaved;
|
||||||
|
uint32_t m_edgesSaved;
|
||||||
|
|
||||||
|
/// Specify whether the dictionary is a DAWG or a GADDAG
|
||||||
|
DictType m_type;
|
||||||
|
|
||||||
|
/// Dictionary name (e.g.: ODS 5.0)
|
||||||
|
wstring m_dicName;
|
||||||
|
|
||||||
|
/// The letters constituting the words of the dictionary
|
||||||
|
wstring m_letters;
|
||||||
|
|
||||||
|
/// Points of the letters
|
||||||
|
vector<uint8_t> m_points;
|
||||||
|
|
||||||
|
/// Frequency of the letters
|
||||||
|
vector<uint8_t> m_frequency;
|
||||||
|
|
||||||
|
/// Vowels
|
||||||
|
vector<bool> m_vowels;
|
||||||
|
|
||||||
|
/// Consonants
|
||||||
|
vector<bool> m_consonants;
|
||||||
|
|
||||||
|
map<wchar_t, unsigned int> m_mapCodeFromChar;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load the header from a file
|
||||||
|
* @param iStream: Input stream where to read the header
|
||||||
|
* @exception: Throw a DicException in case of problem
|
||||||
|
*/
|
||||||
|
void read(istream &iStream);
|
||||||
|
|
||||||
|
/** Build m_mapCodeFromChar */
|
||||||
|
void buildMapCodeFromChar();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#endif /* _HEADER_H */
|
||||||
|
|
||||||
|
/// Local Variables:
|
||||||
|
/// mode: c++
|
||||||
|
/// mode: hs-minor
|
||||||
|
/// c-basic-offset: 4
|
||||||
|
/// indent-tabs-mode: nil
|
||||||
|
/// End:
|
207
dic/listdic.c
207
dic/listdic.c
|
@ -1,207 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file listdic.c
|
|
||||||
* \brief Program used to list a dictionary
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 1999
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stddef.h>
|
|
||||||
#include "dic_internals.h"
|
|
||||||
#include "dic.h"
|
|
||||||
|
|
||||||
|
|
||||||
static void
|
|
||||||
print_dic_rec(FILE* out, Dictionary dic, char *buf, char* s, Dawg_edge i)
|
|
||||||
{
|
|
||||||
if (i.term) /* edge points at a complete word */
|
|
||||||
{
|
|
||||||
*s = '\0';
|
|
||||||
fprintf (out,"%s\n", buf);
|
|
||||||
}
|
|
||||||
if (i.ptr)
|
|
||||||
{ /* Compute index: is it non-zero ? */
|
|
||||||
Dawg_edge *p = dic->dawg + i.ptr;
|
|
||||||
do { /* for each edge out of this node */
|
|
||||||
*s = p->chr + 'a' - 1;
|
|
||||||
print_dic_rec (out,dic,buf,s + 1, *p);
|
|
||||||
}
|
|
||||||
while (!(*p++).last);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
dic_load(Dictionary* dic, char* filename)
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
if ((res = Dic_load(dic, filename)) != 0)
|
|
||||||
{
|
|
||||||
switch (res) {
|
|
||||||
case 1: printf("chargement: problème d'ouverture de %s\n",filename); break;
|
|
||||||
case 2: printf("chargement: mauvais en-tete de dictionnaire\n"); break;
|
|
||||||
case 3: printf("chargement: problème 3 d'allocation mémoire\n"); break;
|
|
||||||
case 4: printf("chargement: problème 4 d'alocation mémoire\n"); break;
|
|
||||||
case 5: printf("chargement: problème de lecture des arcs du dictionnaire\n"); break;
|
|
||||||
default: printf("chargement: problème non-repertorié\n"); break;
|
|
||||||
}
|
|
||||||
exit(res);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
print_dic_list(char* filename, char* out)
|
|
||||||
{
|
|
||||||
FILE* fout;
|
|
||||||
Dictionary dic;
|
|
||||||
static char buf[80];
|
|
||||||
|
|
||||||
dic_load(&dic,filename);
|
|
||||||
|
|
||||||
if (strcmp(out,"stdout") == 0)
|
|
||||||
print_dic_rec(stdout,dic,buf,buf,dic->dawg[dic->root]);
|
|
||||||
else if (strcmp(out,"stderr") == 0)
|
|
||||||
print_dic_rec(stderr,dic,buf,buf,dic->dawg[dic->root]);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if ((fout = fopen(out,"w")) == NULL)
|
|
||||||
return;
|
|
||||||
print_dic_rec(fout,dic,buf,buf,dic->dawg[dic->root]);
|
|
||||||
fclose(fout);
|
|
||||||
}
|
|
||||||
Dic_destroy(dic);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
print_header(char* filename)
|
|
||||||
{
|
|
||||||
Dict_header header;
|
|
||||||
|
|
||||||
Dic_check_header(&header,filename);
|
|
||||||
|
|
||||||
#define OO(IDENT) (unsigned long)offsetof(Dict_header,IDENT)
|
|
||||||
|
|
||||||
printf("Dictionary header information\n");
|
|
||||||
printf("0x%02lx ident : %s\n", OO(ident) ,header.ident);
|
|
||||||
printf("0x%02lx unused 1 : %6d %06x\n",OO(unused_1) ,header.unused_1 ,header.unused_1);
|
|
||||||
printf("0x%02lx unused 2 : %6d %06x\n",OO(unused_2) ,header.unused_2 ,header.unused_2);
|
|
||||||
printf("0x%02lx root : %6d %06x\n",OO(root) ,header.root ,header.root);
|
|
||||||
printf("0x%02lx words : %6d %06x\n",OO(nwords) ,header.nwords ,header.nwords);
|
|
||||||
printf("0x%02lx edges used : %6d %06x\n",OO(edgesused) ,header.edgesused ,header.edgesused);
|
|
||||||
printf("0x%02lx nodes used : %6d %06x\n",OO(nodesused) ,header.nodesused ,header.nodesused);
|
|
||||||
printf("0x%02lx nodes saved : %6d %06x\n",OO(nodessaved),header.nodessaved,header.nodessaved);
|
|
||||||
printf("0x%02lx edges saved : %6d %06x\n",OO(edgessaved),header.edgessaved,header.edgessaved);
|
|
||||||
printf("\n");
|
|
||||||
printf("sizeof(header) = 0x%lx (%lu)\n", (unsigned long)sizeof(header), (unsigned long)sizeof(header));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static void
|
|
||||||
print_node_hex(Dictionary dic, int i)
|
|
||||||
{
|
|
||||||
union edge_t {
|
|
||||||
Dawg_edge e;
|
|
||||||
uint32_t s;
|
|
||||||
} ee;
|
|
||||||
|
|
||||||
ee.e = dic->dawg[i];
|
|
||||||
|
|
||||||
printf("0x%04lx %08x |%4d ptr=%8d t=%d l=%d f=%d chr=%2d (%c)\n",
|
|
||||||
(unsigned long)i*sizeof(ee), (unsigned int)(ee.s),
|
|
||||||
i, ee.e.ptr, ee.e.term, ee.e.last, ee.e.fill, ee.e.chr, ee.e.chr +'a' -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
print_dic_hex(char* filename)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
Dictionary dic;
|
|
||||||
dic_load(&dic,filename);
|
|
||||||
|
|
||||||
printf("offs binary structure \n");
|
|
||||||
printf("---- -------- | ------------------\n");
|
|
||||||
for(i=0; i < (dic->nedges + 1); i++)
|
|
||||||
print_node_hex(dic,i);
|
|
||||||
Dic_destroy(dic);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
usage(char* name)
|
|
||||||
{
|
|
||||||
printf("usage: %s [-a|-d|-h|-l] dictionnaire\n", name);
|
|
||||||
printf(" -a : print all\n");
|
|
||||||
printf(" -h : print header\n");
|
|
||||||
printf(" -d : print dic in hex\n");
|
|
||||||
printf(" -l : print dic word list\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
|
||||||
main(int argc, char *argv[])
|
|
||||||
{
|
|
||||||
int arg_count;
|
|
||||||
int option_print_all = 0;
|
|
||||||
int option_print_header = 0;
|
|
||||||
int option_print_dic_hex = 0;
|
|
||||||
int option_print_dic_list = 0;
|
|
||||||
|
|
||||||
if (argc < 3)
|
|
||||||
{
|
|
||||||
usage(argv[0]);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
arg_count = 1;
|
|
||||||
while(argv[arg_count][0] == '-')
|
|
||||||
{
|
|
||||||
switch (argv[arg_count][1])
|
|
||||||
{
|
|
||||||
case 'a': option_print_all = 1; break;
|
|
||||||
case 'h': option_print_header = 1; break;
|
|
||||||
case 'd': option_print_dic_hex = 1; break;
|
|
||||||
case 'l': option_print_dic_list = 1; break;
|
|
||||||
default: usage(argv[0]); exit(2);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
arg_count++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (option_print_header || option_print_all)
|
|
||||||
{
|
|
||||||
print_header(argv[arg_count]);
|
|
||||||
}
|
|
||||||
if (option_print_dic_hex || option_print_all)
|
|
||||||
{
|
|
||||||
print_dic_hex(argv[arg_count]);
|
|
||||||
}
|
|
||||||
if (option_print_dic_list || option_print_all)
|
|
||||||
{
|
|
||||||
print_dic_list(argv[arg_count],"stdout");
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
187
dic/listdic.cpp
Normal file
187
dic/listdic.cpp
Normal file
|
@ -0,0 +1,187 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 1999-2007 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file listdic.c
|
||||||
|
* \brief Program used to list a dictionary
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 1999
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <cstring>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstddef>
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
# include <libintl.h>
|
||||||
|
# define _(String) gettext(String)
|
||||||
|
#else
|
||||||
|
# define _(String) String
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "header.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "dic_internals.h"
|
||||||
|
#include "dic.h"
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
static void print_dic_rec(ostream &out, const Dictionary &iDic, wchar_t *buf, wchar_t *s, DAWG_EDGE i)
|
||||||
|
{
|
||||||
|
if (i.term) /* edge points at a complete word */
|
||||||
|
{
|
||||||
|
*s = '\0';
|
||||||
|
out << convertToMb(buf) << endl;
|
||||||
|
}
|
||||||
|
if (i.ptr)
|
||||||
|
{ /* Compute index: is it non-zero ? */
|
||||||
|
const DAWG_EDGE *p = reinterpret_cast<const DAWG_EDGE*>(iDic.getEdgeAt(i.ptr));
|
||||||
|
do
|
||||||
|
{ /* for each edge out of this node */
|
||||||
|
*s = iDic.getHeader().getCharFromCode(p->chr);
|
||||||
|
print_dic_rec(out, iDic, buf, s + 1, *p);
|
||||||
|
}
|
||||||
|
while (!(*p++).last);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void print_dic_list(const Dictionary &iDic)
|
||||||
|
{
|
||||||
|
static wchar_t buf[80];
|
||||||
|
print_dic_rec(cout, iDic, buf, buf, *reinterpret_cast<const DAWG_EDGE*>(iDic.getEdgeAt(iDic.getRoot())));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
static void print_node_hex(const Dictionary &dic, int i)
|
||||||
|
{
|
||||||
|
union edge_t
|
||||||
|
{
|
||||||
|
DAWG_EDGE e;
|
||||||
|
uint32_t s;
|
||||||
|
} ee;
|
||||||
|
|
||||||
|
ee.e = *reinterpret_cast<const DAWG_EDGE*>(dic.getEdgeAt(i));
|
||||||
|
|
||||||
|
printf("0x%04lx %08x |%4d ptr=%8d t=%d l=%d chr=%2d (%c)\n",
|
||||||
|
(unsigned long)i*sizeof(ee), (unsigned int)(ee.s),
|
||||||
|
i, ee.e.ptr, ee.e.term, ee.e.last, ee.e.chr, ee.e.chr +'a' -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename DAWG_EDGE>
|
||||||
|
void print_dic_hex(const Dictionary &iDic)
|
||||||
|
{
|
||||||
|
printf(_("offset binary | structure\n"));
|
||||||
|
printf("------ -------- | --------------------\n");
|
||||||
|
for (unsigned int i = 0; i < (iDic.getHeader().getNbEdgesUsed() + 1); i++)
|
||||||
|
print_node_hex<DAWG_EDGE>(iDic, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void usage(const string &iName)
|
||||||
|
{
|
||||||
|
printf(_("usage: %s [-a|-h|-l|-x] dictionary\n"), iName.c_str());
|
||||||
|
printf(_(" -a: print all\n"));
|
||||||
|
printf(_(" -h: print header\n"));
|
||||||
|
printf(_(" -l: print dictionary word list\n"));
|
||||||
|
printf(_(" -x: print dictionary in hex\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
#if HAVE_SETLOCALE
|
||||||
|
// Set locale via LC_ALL
|
||||||
|
setlocale(LC_ALL, "");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
// Set the message domain
|
||||||
|
bindtextdomain(PACKAGE, LOCALEDIR);
|
||||||
|
textdomain(PACKAGE);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int arg_count;
|
||||||
|
int option_print_all = 0;
|
||||||
|
int option_print_header = 0;
|
||||||
|
int option_print_dic_hex = 0;
|
||||||
|
int option_print_dic_list = 0;
|
||||||
|
|
||||||
|
if (argc < 3)
|
||||||
|
{
|
||||||
|
usage(argv[0]);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
arg_count = 1;
|
||||||
|
while (argv[arg_count][0] == '-')
|
||||||
|
{
|
||||||
|
switch (argv[arg_count][1])
|
||||||
|
{
|
||||||
|
case 'a': option_print_all = 1; break;
|
||||||
|
case 'h': option_print_header = 1; break;
|
||||||
|
case 'l': option_print_dic_list = 1; break;
|
||||||
|
case 'x': option_print_dic_hex = 1; break;
|
||||||
|
default: usage(argv[0]); exit(2); break;
|
||||||
|
}
|
||||||
|
arg_count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Dictionary dic(argv[arg_count]);
|
||||||
|
|
||||||
|
if (option_print_header || option_print_all)
|
||||||
|
{
|
||||||
|
dic.getHeader().print();
|
||||||
|
}
|
||||||
|
if (option_print_dic_hex || option_print_all)
|
||||||
|
{
|
||||||
|
if (dic.getHeader().getVersion() == 0)
|
||||||
|
print_dic_hex<DicEdgeOld>(dic);
|
||||||
|
else
|
||||||
|
print_dic_hex<DicEdge>(dic);
|
||||||
|
}
|
||||||
|
if (option_print_dic_list || option_print_all)
|
||||||
|
{
|
||||||
|
if (dic.getHeader().getVersion() == 0)
|
||||||
|
print_dic_list<DicEdgeOld>(dic);
|
||||||
|
else
|
||||||
|
print_dic_list<DicEdge>(dic);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
catch (std::exception &e)
|
||||||
|
{
|
||||||
|
cerr << e.what() << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
382
dic/regexp.c
382
dic/regexp.c
|
@ -1,382 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file regexp.c
|
|
||||||
* \brief Regular Expression functions
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
# include <sys/wait.h>
|
|
||||||
#endif
|
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
#include "dic.h"
|
|
||||||
#include "regexp.h"
|
|
||||||
#include "automaton.h"
|
|
||||||
|
|
||||||
#ifndef PDBG
|
|
||||||
#ifdef DEBUG_RE2
|
|
||||||
#define PDBG(x) x
|
|
||||||
#else
|
|
||||||
#define PDBG(x)
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
NODE* regexp_createNODE(int type,char v,NODE *fg,NODE *fd)
|
|
||||||
{
|
|
||||||
NODE *x;
|
|
||||||
x=(NODE *)malloc(sizeof(NODE));
|
|
||||||
x->type = type;
|
|
||||||
x->var = v;
|
|
||||||
x->fd = fd;
|
|
||||||
x->fg = fg;
|
|
||||||
x->numero = 0;
|
|
||||||
x->position = 0;
|
|
||||||
x->annulable = 0;
|
|
||||||
x->PP = 0;
|
|
||||||
x->DP = 0;
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
void regexp_delete_tree(NODE *root)
|
|
||||||
{
|
|
||||||
if (root == NULL)
|
|
||||||
return;
|
|
||||||
regexp_delete_tree(root->fg);
|
|
||||||
regexp_delete_tree(root->fd);
|
|
||||||
free(root);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
static void print_node(FILE*, NODE *n, int detail);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* computes position, annulable, PP, DP attributes
|
|
||||||
* @param r = root
|
|
||||||
* @param p = current leaf position
|
|
||||||
* @param n = current node number
|
|
||||||
* @param ptl = position to letter
|
|
||||||
*/
|
|
||||||
|
|
||||||
void regexp_parcours(NODE* r, int *p, int *n, int ptl[])
|
|
||||||
{
|
|
||||||
if (r == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
regexp_parcours(r->fg,p,n,ptl);
|
|
||||||
regexp_parcours(r->fd,p,n,ptl);
|
|
||||||
|
|
||||||
switch (r->type)
|
|
||||||
{
|
|
||||||
case NODE_VAR:
|
|
||||||
r->position = *p;
|
|
||||||
ptl[*p] = r->var;
|
|
||||||
*p = *p + 1;
|
|
||||||
r->annulable = 0;
|
|
||||||
r->PP = 1 << (r->position - 1);
|
|
||||||
r->DP = 1 << (r->position - 1);
|
|
||||||
break;
|
|
||||||
case NODE_OR:
|
|
||||||
r->position = 0;
|
|
||||||
r->annulable = r->fg->annulable || r->fd->annulable;
|
|
||||||
r->PP = r->fg->PP | r->fd->PP;
|
|
||||||
r->DP = r->fg->DP | r->fd->DP;
|
|
||||||
break;
|
|
||||||
case NODE_AND:
|
|
||||||
r->position = 0;
|
|
||||||
r->annulable = r->fg->annulable && r->fd->annulable;
|
|
||||||
r->PP = (r->fg->annulable) ? (r->fg->PP | r->fd->PP) : r->fg->PP;
|
|
||||||
r->DP = (r->fd->annulable) ? (r->fg->DP | r->fd->DP) : r->fd->DP;
|
|
||||||
break;
|
|
||||||
case NODE_PLUS:
|
|
||||||
r->position = 0;
|
|
||||||
r->annulable = 0;
|
|
||||||
r->PP = r->fg->PP;
|
|
||||||
r->DP = r->fg->DP;
|
|
||||||
break;
|
|
||||||
case NODE_STAR:
|
|
||||||
r->position = 0;
|
|
||||||
r->annulable = 1;
|
|
||||||
r->PP = r->fg->PP;
|
|
||||||
r->DP = r->fg->DP;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
r->numero = *n;
|
|
||||||
*n = *n + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* computes possuivante
|
|
||||||
* @param r = root
|
|
||||||
* @param PS = next position
|
|
||||||
*/
|
|
||||||
|
|
||||||
void regexp_possuivante(NODE* r, int PS[])
|
|
||||||
{
|
|
||||||
int pos;
|
|
||||||
if (r == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
regexp_possuivante(r->fg,PS);
|
|
||||||
regexp_possuivante(r->fd,PS);
|
|
||||||
|
|
||||||
switch (r->type)
|
|
||||||
{
|
|
||||||
case NODE_AND:
|
|
||||||
/************************************/
|
|
||||||
/* \forall p \in DP(left) */
|
|
||||||
/* PS[p] = PS[p] \cup PP(right) */
|
|
||||||
/************************************/
|
|
||||||
for(pos=1; pos <= PS[0]; pos++)
|
|
||||||
{
|
|
||||||
if (r->fg->DP & (1 << (pos-1)))
|
|
||||||
PS[pos] |= r->fd->PP;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case NODE_PLUS:
|
|
||||||
/************************************/
|
|
||||||
/* == same as START */
|
|
||||||
/* \forall p \in DP(left) */
|
|
||||||
/* PS[p] = PS[p] \cup PP(left) */
|
|
||||||
/************************************/
|
|
||||||
for(pos=1; pos <= PS[0]; pos++)
|
|
||||||
{
|
|
||||||
if (r->DP & (1 << (pos-1)))
|
|
||||||
PS[pos] |= r->PP;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case NODE_STAR:
|
|
||||||
/************************************/
|
|
||||||
/* \forall p \in DP(left) */
|
|
||||||
/* PS[p] = PS[p] \cup PP(left) */
|
|
||||||
/************************************/
|
|
||||||
for(pos=1; pos <= PS[0]; pos++)
|
|
||||||
{
|
|
||||||
if (r->DP & (1 << (pos-1)))
|
|
||||||
PS[pos] |= r->PP;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
// DEBUG only fonctions
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
void regexp_print_PS(int PS[])
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
printf("** positions suivantes **\n");
|
|
||||||
for(i=1; i <= PS[0]; i++)
|
|
||||||
{
|
|
||||||
printf("%02d: 0x%08x\n", i, PS[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
void regexp_print_ptl(int ptl[])
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
printf("** pos -> lettre: ");
|
|
||||||
for(i=1; i <= ptl[0]; i++)
|
|
||||||
{
|
|
||||||
printf("%d=%c ",i,ptl[i]);
|
|
||||||
}
|
|
||||||
printf("\n");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
void regexp_print_letter(FILE* f, char l)
|
|
||||||
{
|
|
||||||
switch (l)
|
|
||||||
{
|
|
||||||
case RE_EPSILON: fprintf(f,"( & [%d])",l); break;
|
|
||||||
case RE_FINAL_TOK: fprintf(f,"( # [%d])",l); break;
|
|
||||||
case RE_ALL_MATCH: fprintf(f,"( . [%d])",l); break;
|
|
||||||
case RE_VOWL_MATCH: fprintf(f,"(:v: [%d])",l); break;
|
|
||||||
case RE_CONS_MATCH: fprintf(f,"(:c: [%d])",l); break;
|
|
||||||
case RE_USR1_MATCH: fprintf(f,"(:1: [%d])",l); break;
|
|
||||||
case RE_USR2_MATCH: fprintf(f,"(:2: [%d])",l); break;
|
|
||||||
default:
|
|
||||||
if (l < RE_FINAL_TOK)
|
|
||||||
fprintf(f," (%c [%d]) ",l + 'a' - 1, l);
|
|
||||||
else
|
|
||||||
fprintf(f," (liste %d)",l - RE_LIST_USER_END);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
void regexp_print_letter2(FILE* f, char l)
|
|
||||||
{
|
|
||||||
switch (l)
|
|
||||||
{
|
|
||||||
case RE_EPSILON: fprintf(f,"&"); break;
|
|
||||||
case RE_FINAL_TOK: fprintf(f,"#"); break;
|
|
||||||
case RE_ALL_MATCH: fprintf(f,"."); break;
|
|
||||||
case RE_VOWL_MATCH: fprintf(f,":v:"); break;
|
|
||||||
case RE_CONS_MATCH: fprintf(f,":c:"); break;
|
|
||||||
case RE_USR1_MATCH: fprintf(f,":1:"); break;
|
|
||||||
case RE_USR2_MATCH: fprintf(f,":2:"); break;
|
|
||||||
default:
|
|
||||||
if (l < RE_FINAL_TOK)
|
|
||||||
fprintf(f,"%c",l + 'a' - 1);
|
|
||||||
else
|
|
||||||
fprintf(f,"l%d",l - RE_LIST_USER_END);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
static void print_node(FILE* f, NODE *n, int detail)
|
|
||||||
{
|
|
||||||
if (n == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
switch (n->type)
|
|
||||||
{
|
|
||||||
case NODE_VAR:
|
|
||||||
regexp_print_letter(f,n->var);
|
|
||||||
break;
|
|
||||||
case NODE_OR:
|
|
||||||
fprintf(f,"OR");
|
|
||||||
break;
|
|
||||||
case NODE_AND:
|
|
||||||
fprintf(f,"AND");
|
|
||||||
break;
|
|
||||||
case NODE_PLUS:
|
|
||||||
fprintf(f,"+");
|
|
||||||
break;
|
|
||||||
case NODE_STAR:
|
|
||||||
fprintf(f,"*");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (detail == 2)
|
|
||||||
{
|
|
||||||
fprintf(f,"\\n pos=%d\\n annul=%d\\n PP=0x%04x\\n DP=0x%04x",
|
|
||||||
n->position,n->annulable,n->PP,n->DP);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
static void print_tree_nodes(FILE* f, NODE* n, int detail)
|
|
||||||
{
|
|
||||||
if (n == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
print_tree_nodes(f,n->fg,detail);
|
|
||||||
print_tree_nodes(f,n->fd,detail);
|
|
||||||
|
|
||||||
fprintf(f,"%d [ label=\"",n->numero);
|
|
||||||
print_node(f,n,detail);
|
|
||||||
fprintf(f,"\"];\n");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
static void print_tree_edges(FILE* f, NODE* n)
|
|
||||||
{
|
|
||||||
if (n == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
print_tree_edges(f,n->fg);
|
|
||||||
print_tree_edges(f,n->fd);
|
|
||||||
|
|
||||||
switch (n->type)
|
|
||||||
{
|
|
||||||
case NODE_OR:
|
|
||||||
fprintf(f,"%d -> %d;",n->numero,n->fg->numero);
|
|
||||||
fprintf(f,"%d -> %d;",n->numero,n->fd->numero);
|
|
||||||
break;
|
|
||||||
case NODE_AND:
|
|
||||||
fprintf(f,"%d -> %d;",n->numero,n->fg->numero);
|
|
||||||
fprintf(f,"%d -> %d;",n->numero,n->fd->numero);
|
|
||||||
break;
|
|
||||||
case NODE_PLUS:
|
|
||||||
case NODE_STAR:
|
|
||||||
fprintf(f,"%d -> %d;",n->numero,n->fg->numero);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*////////////////////////////////////////////////
|
|
||||||
////////////////////////////////////////////////*/
|
|
||||||
|
|
||||||
#ifdef DEBUG_RE
|
|
||||||
void regexp_print_tree(NODE* n, char* name, int detail)
|
|
||||||
{
|
|
||||||
FILE* f;
|
|
||||||
pid_t pid;
|
|
||||||
|
|
||||||
f=fopen(name,"w");
|
|
||||||
fprintf(f,"digraph %s {\n",name);
|
|
||||||
print_tree_nodes(f,n,detail);
|
|
||||||
print_tree_edges(f,n);
|
|
||||||
fprintf(f,"fontsize=20;\n");
|
|
||||||
fprintf(f,"}\n");
|
|
||||||
fclose(f);
|
|
||||||
|
|
||||||
#ifdef HAVE_SYS_WAIT_H
|
|
||||||
pid = fork ();
|
|
||||||
if (pid > 0) {
|
|
||||||
wait(NULL);
|
|
||||||
} else if (pid == 0) {
|
|
||||||
execlp("dotty","dotty",name,NULL);
|
|
||||||
printf("exec dotty failed\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
/// Local Variables:
|
|
||||||
/// mode: hs-minor
|
|
||||||
/// c-basic-offset: 2
|
|
||||||
/// End:
|
|
384
dic/regexp.cpp
Normal file
384
dic/regexp.cpp
Normal file
|
@ -0,0 +1,384 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 1999-2006 Antoine Fraboulet
|
||||||
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file regexp.c
|
||||||
|
* \brief Regular Expression functions
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 2005
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
#ifdef HAVE_SYS_WAIT_H
|
||||||
|
# include <sys/wait.h>
|
||||||
|
#endif
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "dic.h"
|
||||||
|
#include "regexp.h"
|
||||||
|
#include "automaton.h"
|
||||||
|
|
||||||
|
#ifndef PDBG
|
||||||
|
#ifdef DEBUG_RE2
|
||||||
|
#define PDBG(x) x
|
||||||
|
#else
|
||||||
|
#define PDBG(x)
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
NODE* regexp_createNODE(int type, char v, NODE *fg, NODE *fd)
|
||||||
|
{
|
||||||
|
NODE *x;
|
||||||
|
x=(NODE *)malloc(sizeof(NODE));
|
||||||
|
x->type = type;
|
||||||
|
x->var = v;
|
||||||
|
x->fd = fd;
|
||||||
|
x->fg = fg;
|
||||||
|
x->number = 0;
|
||||||
|
x->position = 0;
|
||||||
|
x->annulable = 0;
|
||||||
|
x->PP = 0;
|
||||||
|
x->DP = 0;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
void regexp_delete_tree(NODE *root)
|
||||||
|
{
|
||||||
|
if (root == NULL)
|
||||||
|
return;
|
||||||
|
regexp_delete_tree(root->fg);
|
||||||
|
regexp_delete_tree(root->fd);
|
||||||
|
free(root);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
static void print_node(FILE*, NODE *n, int detail);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* computes position, annulable, PP, DP attributes
|
||||||
|
* @param r = root
|
||||||
|
* @param p = current leaf position
|
||||||
|
* @param n = current node number
|
||||||
|
* @param ptl = position to letter
|
||||||
|
*/
|
||||||
|
|
||||||
|
void regexp_parcours(NODE* r, int *p, int *n, int ptl[])
|
||||||
|
{
|
||||||
|
if (r == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
regexp_parcours(r->fg, p, n, ptl);
|
||||||
|
regexp_parcours(r->fd, p, n, ptl);
|
||||||
|
|
||||||
|
switch (r->type)
|
||||||
|
{
|
||||||
|
case NODE_VAR:
|
||||||
|
r->position = *p;
|
||||||
|
ptl[*p] = r->var;
|
||||||
|
*p = *p + 1;
|
||||||
|
r->annulable = 0;
|
||||||
|
r->PP = 1 << (r->position - 1);
|
||||||
|
r->DP = 1 << (r->position - 1);
|
||||||
|
break;
|
||||||
|
case NODE_OR:
|
||||||
|
r->position = 0;
|
||||||
|
r->annulable = r->fg->annulable || r->fd->annulable;
|
||||||
|
r->PP = r->fg->PP | r->fd->PP;
|
||||||
|
r->DP = r->fg->DP | r->fd->DP;
|
||||||
|
break;
|
||||||
|
case NODE_AND:
|
||||||
|
r->position = 0;
|
||||||
|
r->annulable = r->fg->annulable && r->fd->annulable;
|
||||||
|
r->PP = (r->fg->annulable) ? (r->fg->PP | r->fd->PP) : r->fg->PP;
|
||||||
|
r->DP = (r->fd->annulable) ? (r->fg->DP | r->fd->DP) : r->fd->DP;
|
||||||
|
break;
|
||||||
|
case NODE_PLUS:
|
||||||
|
r->position = 0;
|
||||||
|
r->annulable = 0;
|
||||||
|
r->PP = r->fg->PP;
|
||||||
|
r->DP = r->fg->DP;
|
||||||
|
break;
|
||||||
|
case NODE_STAR:
|
||||||
|
r->position = 0;
|
||||||
|
r->annulable = 1;
|
||||||
|
r->PP = r->fg->PP;
|
||||||
|
r->DP = r->fg->DP;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
r->number = *n;
|
||||||
|
*n = *n + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* computes possuivante
|
||||||
|
* @param r = root
|
||||||
|
* @param PS = next position
|
||||||
|
*/
|
||||||
|
|
||||||
|
void regexp_possuivante(NODE* r, int PS[])
|
||||||
|
{
|
||||||
|
if (r == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
regexp_possuivante(r->fg, PS);
|
||||||
|
regexp_possuivante(r->fd, PS);
|
||||||
|
|
||||||
|
switch (r->type)
|
||||||
|
{
|
||||||
|
case NODE_AND:
|
||||||
|
/************************************/
|
||||||
|
/* \forall p \in DP(left) */
|
||||||
|
/* PS[p] = PS[p] \cup PP(right) */
|
||||||
|
/************************************/
|
||||||
|
for (int pos = 1; pos <= PS[0]; pos++)
|
||||||
|
{
|
||||||
|
if (r->fg->DP & (1 << (pos-1)))
|
||||||
|
PS[pos] |= r->fd->PP;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case NODE_PLUS:
|
||||||
|
/************************************/
|
||||||
|
/* == same as START */
|
||||||
|
/* \forall p \in DP(left) */
|
||||||
|
/* PS[p] = PS[p] \cup PP(left) */
|
||||||
|
/************************************/
|
||||||
|
for (int pos = 1; pos <= PS[0]; pos++)
|
||||||
|
{
|
||||||
|
if (r->DP & (1 << (pos-1)))
|
||||||
|
PS[pos] |= r->PP;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case NODE_STAR:
|
||||||
|
/************************************/
|
||||||
|
/* \forall p \in DP(left) */
|
||||||
|
/* PS[p] = PS[p] \cup PP(left) */
|
||||||
|
/************************************/
|
||||||
|
for (int pos = 1; pos <= PS[0]; pos++)
|
||||||
|
{
|
||||||
|
if (r->DP & (1 << (pos-1)))
|
||||||
|
PS[pos] |= r->PP;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
// DEBUG only fonctions
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
void regexp_print_PS(int PS[])
|
||||||
|
{
|
||||||
|
printf("** positions suivantes **\n");
|
||||||
|
for (int i = 1; i <= PS[0]; i++)
|
||||||
|
{
|
||||||
|
printf("%02d: 0x%08x\n", i, PS[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
void regexp_print_ptl(int ptl[])
|
||||||
|
{
|
||||||
|
printf("** pos -> lettre: ");
|
||||||
|
for (int i = 1; i <= ptl[0]; i++)
|
||||||
|
{
|
||||||
|
printf("%d=%c ", i, ptl[i]);
|
||||||
|
}
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
void regexp_print_letter(FILE* f, char l)
|
||||||
|
{
|
||||||
|
switch (l)
|
||||||
|
{
|
||||||
|
case RE_EPSILON: fprintf(f, "( & [%d])", l); break;
|
||||||
|
case RE_FINAL_TOK: fprintf(f, "( # [%d])", l); break;
|
||||||
|
case RE_ALL_MATCH: fprintf(f, "( . [%d])", l); break;
|
||||||
|
case RE_VOWL_MATCH: fprintf(f, "(:v: [%d])", l); break;
|
||||||
|
case RE_CONS_MATCH: fprintf(f, "(:c: [%d])", l); break;
|
||||||
|
case RE_USR1_MATCH: fprintf(f, "(:1: [%d])", l); break;
|
||||||
|
case RE_USR2_MATCH: fprintf(f, "(:2: [%d])", l); break;
|
||||||
|
default:
|
||||||
|
if (l < RE_FINAL_TOK)
|
||||||
|
fprintf(f, " (%c [%d]) ", l + 'a' - 1, l);
|
||||||
|
else
|
||||||
|
fprintf(f, " (liste %d)", l - RE_LIST_USER_END);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
void regexp_print_letter2(FILE* f, char l)
|
||||||
|
{
|
||||||
|
switch (l)
|
||||||
|
{
|
||||||
|
case RE_EPSILON: fprintf(f, "&"); break;
|
||||||
|
case RE_FINAL_TOK: fprintf(f, "#"); break;
|
||||||
|
case RE_ALL_MATCH: fprintf(f, "."); break;
|
||||||
|
case RE_VOWL_MATCH: fprintf(f, ":v:"); break;
|
||||||
|
case RE_CONS_MATCH: fprintf(f, ":c:"); break;
|
||||||
|
case RE_USR1_MATCH: fprintf(f, ":1:"); break;
|
||||||
|
case RE_USR2_MATCH: fprintf(f, ":2:"); break;
|
||||||
|
default:
|
||||||
|
if (l < RE_FINAL_TOK)
|
||||||
|
fprintf(f, "%c", l + 'a' - 1);
|
||||||
|
else
|
||||||
|
fprintf(f, "l%d", l - RE_LIST_USER_END);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
static void print_node(FILE* f, NODE *n, int detail)
|
||||||
|
{
|
||||||
|
if (n == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
switch (n->type)
|
||||||
|
{
|
||||||
|
case NODE_VAR:
|
||||||
|
regexp_print_letter(f, n->var);
|
||||||
|
break;
|
||||||
|
case NODE_OR:
|
||||||
|
fprintf(f, "OR");
|
||||||
|
break;
|
||||||
|
case NODE_AND:
|
||||||
|
fprintf(f, "AND");
|
||||||
|
break;
|
||||||
|
case NODE_PLUS:
|
||||||
|
fprintf(f, "+");
|
||||||
|
break;
|
||||||
|
case NODE_STAR:
|
||||||
|
fprintf(f, "*");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (detail == 2)
|
||||||
|
{
|
||||||
|
fprintf(f, "\\n pos=%d\\n annul=%d\\n PP=0x%04x\\n DP=0x%04x",
|
||||||
|
n->position, n->annulable, n->PP, n->DP);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
static void print_tree_nodes(FILE* f, NODE* n, int detail)
|
||||||
|
{
|
||||||
|
if (n == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
print_tree_nodes(f, n->fg, detail);
|
||||||
|
print_tree_nodes(f, n->fd, detail);
|
||||||
|
|
||||||
|
fprintf(f, "%d [ label=\"", n->number);
|
||||||
|
print_node(f, n, detail);
|
||||||
|
fprintf(f, "\"];\n");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
static void print_tree_edges(FILE *f, NODE *n)
|
||||||
|
{
|
||||||
|
if (n == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
print_tree_edges(f, n->fg);
|
||||||
|
print_tree_edges(f, n->fd);
|
||||||
|
|
||||||
|
switch (n->type)
|
||||||
|
{
|
||||||
|
case NODE_OR:
|
||||||
|
fprintf(f, "%d -> %d;", n->number, n->fg->number);
|
||||||
|
fprintf(f, "%d -> %d;", n->number, n->fd->number);
|
||||||
|
break;
|
||||||
|
case NODE_AND:
|
||||||
|
fprintf(f, "%d -> %d;", n->number, n->fg->number);
|
||||||
|
fprintf(f, "%d -> %d;", n->number, n->fd->number);
|
||||||
|
break;
|
||||||
|
case NODE_PLUS:
|
||||||
|
case NODE_STAR:
|
||||||
|
fprintf(f, "%d -> %d;", n->number, n->fg->number);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*////////////////////////////////////////////////
|
||||||
|
////////////////////////////////////////////////*/
|
||||||
|
|
||||||
|
#ifdef DEBUG_RE
|
||||||
|
void regexp_print_tree(NODE* n, const string &iName, int detail)
|
||||||
|
{
|
||||||
|
FILE *f = fopen(iName.c_str(), "w");
|
||||||
|
if (f == NULL)
|
||||||
|
return;
|
||||||
|
fprintf(f, "digraph %s {\n", iName.c_str());
|
||||||
|
print_tree_nodes(f, n, detail);
|
||||||
|
print_tree_edges(f, n);
|
||||||
|
fprintf(f, "fontsize=20;\n");
|
||||||
|
fprintf(f, "}\n");
|
||||||
|
fclose(f);
|
||||||
|
|
||||||
|
#ifdef HAVE_SYS_WAIT_H
|
||||||
|
pid_t pid = fork();
|
||||||
|
if (pid > 0)
|
||||||
|
{
|
||||||
|
wait(NULL);
|
||||||
|
}
|
||||||
|
else if (pid == 0)
|
||||||
|
{
|
||||||
|
execlp("dotty", "dotty", iName.c_str(), NULL);
|
||||||
|
printf("exec dotty failed\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/// Local Variables:
|
||||||
|
/// mode: hs-minor
|
||||||
|
/// c-basic-offset: 2
|
||||||
|
/// End:
|
79
dic/regexp.h
79
dic/regexp.h
|
@ -1,21 +1,22 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 1999-2006 Antoine Fraboulet
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
*
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
* 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 */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* (at your option) any later version. */
|
* (at your option) any later version.
|
||||||
/* */
|
*
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
/* GNU General Public License for more details. */
|
* GNU General Public License for more details.
|
||||||
/* */
|
*
|
||||||
/* You should have received a copy of the GNU General Public License */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* along with this program; if not, write to the Free Software */
|
* along with this program; if not, write to the Free Software
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file regexp.h
|
* \file regexp.h
|
||||||
|
@ -24,12 +25,8 @@
|
||||||
* \date 2005
|
* \date 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _TREE_H_
|
#ifndef _REGEXP_H_
|
||||||
#define _TREE_H_
|
#define _REGEXP_H_
|
||||||
#if defined(__cplusplus)
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define NODE_TOP 0
|
#define NODE_TOP 0
|
||||||
#define NODE_VAR 1
|
#define NODE_VAR 1
|
||||||
|
@ -38,18 +35,25 @@ extern "C"
|
||||||
#define NODE_STAR 4
|
#define NODE_STAR 4
|
||||||
#define NODE_PLUS 5
|
#define NODE_PLUS 5
|
||||||
|
|
||||||
typedef struct node {
|
|
||||||
int type;
|
typedef struct node
|
||||||
char var;
|
{
|
||||||
struct node *fg;
|
int type;
|
||||||
struct node *fd;
|
char var;
|
||||||
int numero;
|
struct node *fg;
|
||||||
int position;
|
struct node *fd;
|
||||||
int annulable;
|
int number;
|
||||||
int PP;
|
int position;
|
||||||
int DP;
|
int annulable;
|
||||||
|
int PP;
|
||||||
|
int DP;
|
||||||
} NODE;
|
} NODE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* different letters in the dictionary
|
||||||
|
*/
|
||||||
|
#define DIC_LETTERS 27
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* maximum number of accepted terminals in regular expressions
|
* maximum number of accepted terminals in regular expressions
|
||||||
*/
|
*/
|
||||||
|
@ -139,7 +143,7 @@ struct regexp_error_report_t {
|
||||||
char msg[MAX_REGEXP_ERROR_LENGTH];
|
char msg[MAX_REGEXP_ERROR_LENGTH];
|
||||||
};
|
};
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <cstdio>
|
||||||
|
|
||||||
void regexp_print_letter(FILE* f, char l);
|
void regexp_print_letter(FILE* f, char l);
|
||||||
void regexp_print_letter2(FILE* f, char l);
|
void regexp_print_letter2(FILE* f, char l);
|
||||||
|
@ -147,10 +151,7 @@ void regexp_print_PS(int PS[]);
|
||||||
void regexp_print_ptl(int ptl[]);
|
void regexp_print_ptl(int ptl[]);
|
||||||
void regexp_print_tree(NODE* n, char* name, int detail);
|
void regexp_print_tree(NODE* n, char* name, int detail);
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
#endif /* _REGEXP_H_ */
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* _TREE_H_ */
|
|
||||||
|
|
||||||
/// Local Variables:
|
/// Local Variables:
|
||||||
/// mode: c++
|
/// mode: c++
|
||||||
|
|
140
dic/regexpmain.c
140
dic/regexpmain.c
|
@ -1,140 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file regexpmain.c
|
|
||||||
* \brief Program used to test regexp
|
|
||||||
* \author Antoine Fraboulet
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include "dic.h"
|
|
||||||
#include "regexp.h"
|
|
||||||
#include "dic_search.h"
|
|
||||||
|
|
||||||
#define __UNUSED__ __attribute__((unused))
|
|
||||||
|
|
||||||
/********************************************************/
|
|
||||||
/********************************************************/
|
|
||||||
/********************************************************/
|
|
||||||
|
|
||||||
const unsigned int all_letter[DIC_LETTERS] =
|
|
||||||
{
|
|
||||||
/* 1 1 1 1 1 1 1 1 1 1 2 2 2 2 2 2 2 */
|
|
||||||
/* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 */
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
|
||||||
0,1,1,1,1, 1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1, 1, 1, 1, 1
|
|
||||||
};
|
|
||||||
|
|
||||||
const unsigned int vowels[DIC_LETTERS] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
|
||||||
0,1,0,0,0, 1,0,0,0,1,0, 0,0,0,0,1,0,0,0,0,0,1,0, 0, 0, 1, 0
|
|
||||||
};
|
|
||||||
|
|
||||||
const unsigned int consonants[DIC_LETTERS] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
|
||||||
0,0,1,1,1, 0,1,1,1,0,1, 1,1,1,1,0,1,1,1,1,1,0,1, 1, 1, 1, 1
|
|
||||||
};
|
|
||||||
|
|
||||||
void init_letter_lists(struct search_RegE_list_t *list)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
memset (list,0,sizeof(*list));
|
|
||||||
list->minlength = 1;
|
|
||||||
list->maxlength = 15;
|
|
||||||
list->valid[0] = 1; // all letters
|
|
||||||
list->symbl[0] = RE_ALL_MATCH;
|
|
||||||
list->valid[1] = 1; // vowels
|
|
||||||
list->symbl[1] = RE_VOWL_MATCH;
|
|
||||||
list->valid[2] = 1; // consonants
|
|
||||||
list->symbl[2] = RE_CONS_MATCH;
|
|
||||||
for(i=0; i < DIC_LETTERS; i++)
|
|
||||||
{
|
|
||||||
list->letters[0][i] = all_letter[i];
|
|
||||||
list->letters[1][i] = vowels[i];
|
|
||||||
list->letters[2][i] = consonants[i];
|
|
||||||
}
|
|
||||||
list->valid[3] = 0; // user defined list 1
|
|
||||||
list->symbl[3] = RE_USR1_MATCH;
|
|
||||||
list->valid[4] = 0; // user defined list 2
|
|
||||||
list->symbl[4] = RE_USR2_MATCH;
|
|
||||||
}
|
|
||||||
|
|
||||||
/********************************************************/
|
|
||||||
/********************************************************/
|
|
||||||
/********************************************************/
|
|
||||||
void
|
|
||||||
usage(int __UNUSED__ argc, char* argv[])
|
|
||||||
{
|
|
||||||
fprintf(stderr,"usage: %s dictionary\n",argv[0]);
|
|
||||||
fprintf(stderr," dictionary : path to dawg eliot dictionary\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char* argv[])
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
Dictionary dic;
|
|
||||||
char wordlist[RES_REGE_MAX][DIC_WORD_MAX];
|
|
||||||
char er[200];
|
|
||||||
strcpy(er,".");
|
|
||||||
struct search_RegE_list_t list;
|
|
||||||
|
|
||||||
if (argc < 2)
|
|
||||||
{
|
|
||||||
usage(argc,argv);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Dic_load(&dic,argv[1]))
|
|
||||||
{
|
|
||||||
fprintf(stdout,"impossible de lire le dictionnaire\n");
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (strcmp(er,""))
|
|
||||||
{
|
|
||||||
fprintf(stdout,"**************************************************************\n");
|
|
||||||
fprintf(stdout,"**************************************************************\n");
|
|
||||||
fprintf(stdout,"entrer une ER:\n");
|
|
||||||
fgets(er,sizeof(er),stdin);
|
|
||||||
/* strip \n */
|
|
||||||
er[strlen(er) - 1] = '\0';
|
|
||||||
if (strcmp(er,"") == 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
/* automaton */
|
|
||||||
init_letter_lists(&list);
|
|
||||||
Dic_search_RegE_inner(dic,er,wordlist,&list);
|
|
||||||
|
|
||||||
fprintf(stdout,"résultat:\n");
|
|
||||||
for(i=0; i<RES_REGE_MAX && wordlist[i][0]; i++)
|
|
||||||
{
|
|
||||||
fprintf(stderr,"%s\n",wordlist[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Dic_destroy(dic);
|
|
||||||
return 0;
|
|
||||||
}
|
|
169
dic/regexpmain.cpp
Normal file
169
dic/regexpmain.cpp
Normal file
|
@ -0,0 +1,169 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Antoine Fraboulet
|
||||||
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* \file regexpmain.c
|
||||||
|
* \brief Program used to test regexp
|
||||||
|
* \author Antoine Fraboulet
|
||||||
|
* \date 2005
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <exception>
|
||||||
|
#include <iostream>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
# include <libintl.h>
|
||||||
|
# define _(String) gettext(String)
|
||||||
|
#else
|
||||||
|
# define _(String) String
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "dic.h"
|
||||||
|
#include "regexp.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define __UNUSED__ __attribute__((unused))
|
||||||
|
|
||||||
|
/********************************************************/
|
||||||
|
/********************************************************/
|
||||||
|
/********************************************************/
|
||||||
|
|
||||||
|
const unsigned int all_letter[DIC_LETTERS] =
|
||||||
|
{
|
||||||
|
/* 1 1 1 1 1 1 1 1 1 1 2 2 2 2 2 2 2 */
|
||||||
|
/* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 */
|
||||||
|
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
||||||
|
0,1,1,1,1, 1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1, 1, 1, 1, 1
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsigned int vowels[DIC_LETTERS] =
|
||||||
|
{
|
||||||
|
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
||||||
|
0,1,0,0,0, 1,0,0,0,1,0, 0,0,0,0,1,0,0,0,0,0,1,0, 0, 0, 1, 0
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsigned int consonants[DIC_LETTERS] =
|
||||||
|
{
|
||||||
|
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z */
|
||||||
|
0,0,1,1,1, 0,1,1,1,0,1, 1,1,1,1,0,1,1,1,1,1,0,1, 1, 1, 1, 1
|
||||||
|
};
|
||||||
|
|
||||||
|
void init_letter_lists(struct search_RegE_list_t *iList)
|
||||||
|
{
|
||||||
|
memset (iList, 0, sizeof(*iList));
|
||||||
|
iList->minlength = 1;
|
||||||
|
iList->maxlength = 15;
|
||||||
|
iList->valid[0] = 1; // all letters
|
||||||
|
iList->symbl[0] = RE_ALL_MATCH;
|
||||||
|
iList->valid[1] = 1; // vowels
|
||||||
|
iList->symbl[1] = RE_VOWL_MATCH;
|
||||||
|
iList->valid[2] = 1; // consonants
|
||||||
|
iList->symbl[2] = RE_CONS_MATCH;
|
||||||
|
for (int i = 0; i < DIC_LETTERS; i++)
|
||||||
|
{
|
||||||
|
iList->letters[0][i] = all_letter[i];
|
||||||
|
iList->letters[1][i] = vowels[i];
|
||||||
|
iList->letters[2][i] = consonants[i];
|
||||||
|
}
|
||||||
|
iList->valid[3] = 0; // user defined list 1
|
||||||
|
iList->symbl[3] = RE_USR1_MATCH;
|
||||||
|
iList->valid[4] = 0; // user defined list 2
|
||||||
|
iList->symbl[4] = RE_USR2_MATCH;
|
||||||
|
}
|
||||||
|
|
||||||
|
/********************************************************/
|
||||||
|
/********************************************************/
|
||||||
|
/********************************************************/
|
||||||
|
void usage(const char *iBinaryName)
|
||||||
|
{
|
||||||
|
cerr << _("usage: %s dictionary") << iBinaryName << endl;
|
||||||
|
cerr << _(" dictionary: path to eliot dawg dictionary") << endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int main(int argc, char* argv[])
|
||||||
|
{
|
||||||
|
#if HAVE_SETLOCALE
|
||||||
|
// Set locale via LC_ALL
|
||||||
|
setlocale(LC_ALL, "");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if ENABLE_NLS
|
||||||
|
// Set the message domain
|
||||||
|
bindtextdomain(PACKAGE, LOCALEDIR);
|
||||||
|
textdomain(PACKAGE);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (argc != 2)
|
||||||
|
{
|
||||||
|
usage(argv[0]);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Dictionary dic(argv[1]);
|
||||||
|
|
||||||
|
char er[200];
|
||||||
|
strcpy(er, ".");
|
||||||
|
|
||||||
|
struct search_RegE_list_t regList;
|
||||||
|
while (strcmp(er, ""))
|
||||||
|
{
|
||||||
|
cout << "**************************************************************" << endl;
|
||||||
|
cout << "**************************************************************" << endl;
|
||||||
|
cout << _("enter a regular expression:") << endl;
|
||||||
|
fgets(er, sizeof(er), stdin);
|
||||||
|
/* strip \n */
|
||||||
|
er[strlen(er) - 1] = '\0';
|
||||||
|
if (strcmp(er, "") == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* automaton */
|
||||||
|
init_letter_lists(®List);
|
||||||
|
list<wstring> wordList;
|
||||||
|
dic.searchRegExp(convertToWc(er), wordList, ®List);
|
||||||
|
|
||||||
|
cout << _("result:") << endl;
|
||||||
|
list<wstring>::const_iterator it;
|
||||||
|
for (it = wordList.begin(); it != wordList.end(); it++)
|
||||||
|
{
|
||||||
|
cerr << convertToMb(*it) << endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
catch (std::exception &e)
|
||||||
|
{
|
||||||
|
std::cerr << e.what() << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
std::cerr << "Unkown exception taken" << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
150
dic/tile.cpp
Normal file
150
dic/tile.cpp
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2005-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
|
*
|
||||||
|
* 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 <sstream>
|
||||||
|
#include <string>
|
||||||
|
#include <wctype.h>
|
||||||
|
#include "tile.h"
|
||||||
|
#include "header.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
#include "dic_exception.h"
|
||||||
|
|
||||||
|
|
||||||
|
const Header * Tile::m_header = NULL;
|
||||||
|
Tile Tile::m_TheJoker;
|
||||||
|
|
||||||
|
|
||||||
|
Tile::Tile(wchar_t c)
|
||||||
|
{
|
||||||
|
if (iswalpha(c))
|
||||||
|
{
|
||||||
|
m_joker = iswlower(c);
|
||||||
|
m_char = towupper(c);
|
||||||
|
m_code = m_header->getCodeFromChar(m_char);
|
||||||
|
}
|
||||||
|
else if (c == kTILE_JOKER)
|
||||||
|
{
|
||||||
|
m_joker = true;
|
||||||
|
m_char = kTILE_JOKER;
|
||||||
|
m_code = m_header->getCodeFromChar(m_char);
|
||||||
|
}
|
||||||
|
else if (c == kTILE_DUMMY)
|
||||||
|
{
|
||||||
|
// The char and the code are chosen to be different from any possible
|
||||||
|
// real tile
|
||||||
|
m_joker = false;
|
||||||
|
m_char = kTILE_DUMMY;
|
||||||
|
m_code = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ostringstream ss;
|
||||||
|
ss << "Tile::Tile: Unknown character: " << convertToMb(c);
|
||||||
|
throw DicException(ss.str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Tile::isVowel() const
|
||||||
|
{
|
||||||
|
if (m_code == 0)
|
||||||
|
throw DicException("Tile::isVowel: Invalid tile");
|
||||||
|
return m_header->isVowel(m_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Tile::isConsonant() const
|
||||||
|
{
|
||||||
|
if (m_code == 0)
|
||||||
|
throw DicException("Tile::isConsonant: Invalid tile");
|
||||||
|
return m_header->isConsonant(m_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Tile::maxNumber() const
|
||||||
|
{
|
||||||
|
if (m_code == 0)
|
||||||
|
throw DicException("Tile::maxNumber: Invalid tile");
|
||||||
|
return m_header->getFrequency(m_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Tile::getPoints() const
|
||||||
|
{
|
||||||
|
if (m_code == 0)
|
||||||
|
throw DicException("Tile::getPoints: Invalid tile");
|
||||||
|
return m_header->getPoints(m_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wchar_t Tile::toChar() const
|
||||||
|
{
|
||||||
|
if (m_joker)
|
||||||
|
{
|
||||||
|
if (iswalpha(m_char))
|
||||||
|
return towlower(m_char);
|
||||||
|
else
|
||||||
|
return kTILE_JOKER;
|
||||||
|
}
|
||||||
|
return m_char;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int Tile::toCode() const
|
||||||
|
{
|
||||||
|
return m_code;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Tile::operator<(const Tile &iOther) const
|
||||||
|
{
|
||||||
|
if (m_joker)
|
||||||
|
return false;
|
||||||
|
else if (iOther.m_joker)
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
return m_char < iOther.m_char;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Tile::operator==(const Tile &iOther) const
|
||||||
|
{
|
||||||
|
if (m_joker || iOther.m_joker)
|
||||||
|
{
|
||||||
|
if (m_joker != iOther.m_joker)
|
||||||
|
return false;
|
||||||
|
return m_char == iOther.m_char;
|
||||||
|
}
|
||||||
|
return m_char == iOther.m_char;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Tile::operator!=(const Tile &iOther) const
|
||||||
|
{
|
||||||
|
return !(*this == iOther);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Local Variables:
|
||||||
|
/// mode: c++
|
||||||
|
/// mode: hs-minor
|
||||||
|
/// c-basic-offset: 4
|
||||||
|
/// indent-tabs-mode: nil
|
||||||
|
/// End:
|
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -25,6 +27,8 @@
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
class Header;
|
||||||
|
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
* A Tile is the internal representation
|
* A Tile is the internal representation
|
||||||
|
@ -34,17 +38,17 @@ using namespace std;
|
||||||
|
|
||||||
class Tile
|
class Tile
|
||||||
{
|
{
|
||||||
|
friend class Dictionary;
|
||||||
public:
|
public:
|
||||||
|
|
||||||
// a lowercase character always a joker
|
// a lowercase character is always a joker
|
||||||
// - this permits to detect joker in already played games
|
// - this permits to detect joker in already played games
|
||||||
// - we need to pay attention when inserting character taken
|
// - we need to pay attention when inserting characters taken
|
||||||
// from user input
|
// from user input
|
||||||
|
|
||||||
Tile(wchar_t c = 0);
|
Tile(wchar_t c = kTILE_DUMMY);
|
||||||
virtual ~Tile() {}
|
|
||||||
|
|
||||||
bool isEmpty() const { return m_dummy; }
|
bool isEmpty() const { return m_char == kTILE_DUMMY; }
|
||||||
bool isJoker() const { return m_joker; }
|
bool isJoker() const { return m_joker; }
|
||||||
bool isVowel() const;
|
bool isVowel() const;
|
||||||
bool isConsonant() const;
|
bool isConsonant() const;
|
||||||
|
@ -53,10 +57,7 @@ public:
|
||||||
wchar_t toChar() const;
|
wchar_t toChar() const;
|
||||||
unsigned int toCode() const;
|
unsigned int toCode() const;
|
||||||
|
|
||||||
static const Tile &dummy() { return m_TheDummy; }
|
|
||||||
static const Tile &Joker() { return m_TheJoker; }
|
static const Tile &Joker() { return m_TheJoker; }
|
||||||
static const list<Tile>& getAllTiles();
|
|
||||||
static const Tile &GetTileFromCode(unsigned int iCode);
|
|
||||||
|
|
||||||
bool operator <(const Tile &iOther) const;
|
bool operator <(const Tile &iOther) const;
|
||||||
bool operator ==(const Tile &iOther) const;
|
bool operator ==(const Tile &iOther) const;
|
||||||
|
@ -65,7 +66,6 @@ public:
|
||||||
private:
|
private:
|
||||||
wchar_t m_char;
|
wchar_t m_char;
|
||||||
bool m_joker;
|
bool m_joker;
|
||||||
bool m_dummy;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Internal code, used in the dictionary to represent the letter.
|
* Internal code, used in the dictionary to represent the letter.
|
||||||
|
@ -73,16 +73,17 @@ private:
|
||||||
*/
|
*/
|
||||||
int m_code;
|
int m_code;
|
||||||
|
|
||||||
// Special tiles are declared static
|
static const wchar_t kTILE_DUMMY = L'%';
|
||||||
static const Tile m_TheJoker;
|
static const wchar_t kTILE_JOKER = L'?';
|
||||||
static const Tile m_TheDummy;
|
|
||||||
|
|
||||||
/// List of available tiles
|
// Special tiles are declared static
|
||||||
static list<Tile> m_tilesList;
|
static Tile m_TheJoker;
|
||||||
/// Vector of tiles indexed by their code, for fast look-up
|
|
||||||
static vector<Tile> m_tilesVect;
|
/// Dictionary header
|
||||||
/// True when m_tilesVect is correctly initialized
|
static const Header *m_header;
|
||||||
static bool m_vectInitialized;
|
|
||||||
|
/// Update the dictionary header
|
||||||
|
static void SetHeader(const Header &iHeader) { m_header = &iHeader; }
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
208
doc/dic.txt
208
doc/dic.txt
|
@ -1,16 +1,16 @@
|
||||||
|
|
||||||
The dictionary is a directed tree. Compression is achieved by
|
The dictionary is a directed tree. Compression is achieved by
|
||||||
sharing word prefix and suffix. Search is NOT case sensitive.
|
sharing word prefix and suffix. Search is NOT case sensitive.
|
||||||
|
|
||||||
considering this 3 words dictionary:
|
considering this 3 words dictionary:
|
||||||
|
|
||||||
ABC
|
ABC
|
||||||
ADA
|
ADA
|
||||||
EDAA
|
EDAA
|
||||||
|
|
||||||
The tree will look like this:
|
The tree will look like this:
|
||||||
|
|
||||||
root
|
root
|
||||||
|
|
|
|
||||||
A----------E!
|
A----------E!
|
||||||
| |
|
| |
|
||||||
|
@ -24,41 +24,56 @@
|
||||||
------------------- 0!* (sink)
|
------------------- 0!* (sink)
|
||||||
|
|
||||||
The tree is saved using an array of 32 bits words.
|
The tree is saved using an array of 32 bits words.
|
||||||
A cell is a binary structure
|
A cell is a binary structure
|
||||||
|
|
||||||
ptr : index in the array of the first child
|
ptr : index in the array of the first child
|
||||||
term : is it the last letter of a word (*)
|
term : is it the last letter of a word (*)
|
||||||
last : is it the last child of its local root (!)
|
last : is it the last child of its local root (!)
|
||||||
fill : currently unused.
|
|
||||||
chr : guess what !
|
chr : guess what !
|
||||||
|
|
||||||
There is no pointer from a cell to its brother, it is simply the
|
There is no pointer from a cell to its brother, it is simply the
|
||||||
next cell in the array (you know you are on the last brother when
|
next cell in the array (you know you are on the last brother when
|
||||||
the flag "last" is set).
|
the flag "last" is set).
|
||||||
|
|
||||||
The way it is stored in a file is different thing! The tree is
|
The way it is stored in a file is a different thing! The tree is
|
||||||
stored bottom-up. The sink (offset 0) is the first cell of
|
stored bottom-up. The sink (offset 0) is the first cell of
|
||||||
the array.
|
the array.
|
||||||
|
|
||||||
Using compdict (which you can found in the eliot/dic directory),
|
Using compdic (which you can find in the eliot/dic directory),
|
||||||
the compiled dictionary will look like this:
|
the compiled dictionary will look like this:
|
||||||
|
|
||||||
compdict's console output:
|
compdic console output (cut in the middle):
|
||||||
============================
|
===================================================================
|
||||||
keyword length 21 bytes
|
dictionary name: ODS 4.0
|
||||||
keyword size 22 bytes
|
compressed on: mer 12 déc 2007 07:29:50 GMT
|
||||||
header size 48 bytes
|
compressed using a binary compiled by: ipkiss@ulukai
|
||||||
|
dictionary type: DAWG
|
||||||
|
letters: ABCDEFGHIJKLMNOPQRSTUVWXYZ?
|
||||||
|
number of letters: 27
|
||||||
|
number of words: 369085
|
||||||
|
header size: 360 bytes
|
||||||
|
root: 100950 (edge)
|
||||||
|
nodes: 40377 used + 418387 saved
|
||||||
|
edges: 100950 used + 601922 saved
|
||||||
|
===============================================
|
||||||
|
letter | points | frequency | vowel | consonant
|
||||||
|
-------+--------+-----------+-------+----------
|
||||||
|
A | 1 | 9 | 1 | 0
|
||||||
|
B | 3 | 2 | 0 | 1
|
||||||
|
C | 3 | 2 | 0 | 1
|
||||||
|
D | 2 | 3 | 0 | 1
|
||||||
|
[... output cut here ...]
|
||||||
|
X | 10 | 1 | 0 | 1
|
||||||
|
Y | 10 | 1 | 1 | 1
|
||||||
|
Z | 10 | 1 | 0 | 1
|
||||||
|
? | 0 | 2 | 1 | 1
|
||||||
|
===============================================
|
||||||
|
Load time: 0,060 s
|
||||||
|
Compression time: 0,170 s
|
||||||
|
Maximum recursion level reached: 16
|
||||||
|
===================================================================
|
||||||
|
|
||||||
3 words
|
binary view of the dictionary (FIXME: not up to date):
|
||||||
|
|
||||||
root : 9 (edge)
|
|
||||||
root : 36 (byte)
|
|
||||||
|
|
||||||
nodes : 7+1
|
|
||||||
edges : 9+1
|
|
||||||
============================
|
|
||||||
|
|
||||||
binary view of the dictionary:
|
|
||||||
===================================================================
|
===================================================================
|
||||||
0001 0203 0405 0607 0809 0a0b 0c0d 0e0f
|
0001 0203 0405 0607 0809 0a0b 0c0d 0e0f
|
||||||
00000000: 5f43 4f4d 5049 4c45 445f 4449 4354 494f _COMPILED_DICTIO
|
00000000: 5f43 4f4d 5049 4c45 445f 4449 4354 494f _COMPILED_DICTIO
|
||||||
|
@ -69,27 +84,85 @@ binary view of the dictionary:
|
||||||
00000050: 0600 002a 0700 0000 ...*....
|
00000050: 0600 002a 0700 0000 ...*....
|
||||||
===================================================================
|
===================================================================
|
||||||
|
|
||||||
The header structure is the following:
|
The header is made of 2 structures (for backwards compatibility
|
||||||
|
with older headers) like this:
|
||||||
|
===================================================================
|
||||||
#define _COMPIL_KEYWORD_ "_COMPILED_DICTIONARY_"
|
#define _COMPIL_KEYWORD_ "_COMPILED_DICTIONARY_"
|
||||||
|
|
||||||
typedef struct _Dict_header { // offset
|
struct Dict_header_old // offset
|
||||||
|
{
|
||||||
char ident[sizeof(_COMPIL_KEYWORD_)]; // 0x00
|
char ident[sizeof(_COMPIL_KEYWORD_)]; // 0x00
|
||||||
char unused_1; // 0x16
|
uint8_t version; // 0x16
|
||||||
char unused_2; // 0x17
|
char unused; // 0x17
|
||||||
int root; // 0x18
|
uint32_t root; // 0x18
|
||||||
int nwords; // 0x1c
|
uint32_t nwords; // 0x1c
|
||||||
unsigned int edgesused; // 0x20
|
uint32_t edgesused; // 0x20
|
||||||
unsigned int nodesused; // 0x24
|
uint32_t nodesused; // 0x24
|
||||||
unsigned int nodessaved; // 0x2c
|
uint32_t nodessaved; // 0x28
|
||||||
unsigned int edgessaved; // 0x30
|
uint32_t edgessaved; // 0x2c
|
||||||
} Dict_header;
|
};
|
||||||
|
|
||||||
binary output of the header:
|
#define _MAX_USER_HOST_ 32
|
||||||
|
#define _MAX_DIC_NAME_SIZE_ 30
|
||||||
|
#define _MAX_LETTERS_NB_ 63
|
||||||
|
#define _MAX_LETTERS_SIZE_ 80
|
||||||
|
|
||||||
|
struct Dict_header
|
||||||
|
{
|
||||||
|
uint64_t compressDate;
|
||||||
|
// Build information
|
||||||
|
char userHost[_MAX_USER_HOST_];
|
||||||
|
// Size taken by the build information
|
||||||
|
uint32_t userHostSize;
|
||||||
|
|
||||||
|
// Compression algorithm (1 = DAWG, 2 = GADDAG)
|
||||||
|
uint8_t algorithm;
|
||||||
|
// Variant used in the rules (XXX: currently unused)
|
||||||
|
uint8_t variant;
|
||||||
|
|
||||||
|
// Dictionary official name and version (e.g.: ODS 5.0)
|
||||||
|
char dicName[_MAX_DIC_NAME_SIZE_];
|
||||||
|
// Size taken by the dictionary name
|
||||||
|
uint32_t dicNameSize;
|
||||||
|
|
||||||
|
// Letters used in the dictionary
|
||||||
|
// We should have: nbLetters <= lettersSize <= _MAX_LETTERS_SIZE_
|
||||||
|
// and: nbLetters <= _MAX_LETTERS_NB_
|
||||||
|
// The letters themselves, in UTF-8
|
||||||
|
char letters[_MAX_LETTERS_SIZE_];
|
||||||
|
// Size taken by the letters
|
||||||
|
uint32_t lettersSize;
|
||||||
|
// Number of letters (XXX: in theory useless, but allows a sanity check)
|
||||||
|
uint32_t nbLetters;
|
||||||
|
|
||||||
|
// Points of the letters (indexed by their code)
|
||||||
|
// The "+ 1" is there for struct alignment
|
||||||
|
uint8_t points[_MAX_LETTERS_NB_ + 1];
|
||||||
|
// Frequency of the letters (indexedy their code)
|
||||||
|
// The "+ 1" is there for struct alignment
|
||||||
|
uint8_t frequency[_MAX_LETTERS_NB_ + 1];
|
||||||
|
// Bitfield indicating whether letters are vowels
|
||||||
|
uint64_t vowels;
|
||||||
|
// Bitfield indicating whether letters are consonants
|
||||||
|
uint64_t consonants;
|
||||||
|
}
|
||||||
|
===================================================================
|
||||||
|
|
||||||
|
In the old version of the dictionary, only the first structure was used
|
||||||
|
(with version = 0). The current format (version = 1) has the 2 structs
|
||||||
|
next to each other.
|
||||||
|
The dictionary name, the letters, and the user/host information are
|
||||||
|
stored in UTF-8. All the numbers are big endian (i.e. the output of
|
||||||
|
the htonl() function).
|
||||||
|
To avoid alignment issues, the extended header has been designed to
|
||||||
|
have multiples of 64 bits regularly.
|
||||||
|
|
||||||
|
|
||||||
|
binary output of the header (FIXME: not up to date):
|
||||||
===================================================================
|
===================================================================
|
||||||
0x00 ident : _COMPILED_DICTIONARY_
|
0x00 ident : _COMPILED_DICTIONARY_
|
||||||
0x16 unused 1 : 0 00000000
|
0x16 version : 0 00000001
|
||||||
0x17 unused 2 : 0 00000000
|
0x17 unused : 0 00000000
|
||||||
0x18 root : 9 00000009
|
0x18 root : 9 00000009
|
||||||
0x1c words : 3 00000003
|
0x1c words : 3 00000003
|
||||||
0x20 edges used : 9 00000009
|
0x20 edges used : 9 00000009
|
||||||
|
@ -98,40 +171,37 @@ binary output of the header:
|
||||||
0x2c edges saved : 1 00000001
|
0x2c edges saved : 1 00000001
|
||||||
===================================================================
|
===================================================================
|
||||||
|
|
||||||
The real array of data begins at offset 0x34. Integer are stored in a
|
The real array of data begins at offset 0x168. The array is stored
|
||||||
machine dependent way. This dictionary was compiled on a i386 and is
|
'as is' right after the header. Each array cell is a bit-structure
|
||||||
not readable on a machine with a different endianess (unless swapping
|
on 4 bytes:
|
||||||
all necessary information). The array is stored 'as is' right after
|
|
||||||
the header. Each array cell is a bit-structure on 4 bytes :
|
|
||||||
|
|
||||||
typedef struct _Dawg_edge {
|
struct DicEdge
|
||||||
unsigned int ptr : 24;
|
{
|
||||||
unsigned int term : 1;
|
unsigned int ptr : 24;
|
||||||
unsigned int last : 1;
|
unsigned int term : 1;
|
||||||
unsigned int fill : 1; // reserved (currently unused)
|
unsigned int last : 1;
|
||||||
unsigned int chr : 5;
|
unsigned int chr : 6;
|
||||||
} Dawg_edge;
|
};
|
||||||
|
|
||||||
Characters are not stored in ASCII. The order is preserved but
|
Characters are not stored in ASCII. The order of the letters given
|
||||||
we changed the values: A=1, B=2, ... This is very easy to do
|
to the compdic binary is preserved, but we changed the values: the
|
||||||
with the ASCII table as ('A' & 0x1f) == ('a' & 0x1f) == 1.
|
first letter is 1, the second one is 2, etc...
|
||||||
This may not work on machines that are not using ASCII. The dictionary
|
The dictionary can thus handle up to 64 different letters but not more.
|
||||||
can thus handle up to 32 different letters but not more.
|
The letter 0 is special (used for the sink node in particular), so
|
||||||
|
in practice there are only 63 distinct letters.
|
||||||
|
|
||||||
offs binary structure
|
offs binary structure
|
||||||
---- -------- | ------------------
|
---- -------- | ------------------
|
||||||
0x00 02000000 | 0 ptr= 0 t=0 l=1 f=0 chr=0 (`)
|
0x00 02000000 | 0 ptr= 0 t=0 l=1 chr=0 (`)
|
||||||
0x04 1b000000 | 1 ptr= 0 t=1 l=1 f=0 chr=3 (c)
|
0x04 1b000000 | 1 ptr= 0 t=1 l=1 chr=3 (c)
|
||||||
0x08 0b000000 | 2 ptr= 0 t=1 l=1 f=0 chr=1 (a)
|
0x08 0b000000 | 2 ptr= 0 t=1 l=1 chr=1 (a)
|
||||||
0x0c 10000001 | 3 ptr= 1 t=0 l=0 f=0 chr=2 (b)
|
0x0c 10000001 | 3 ptr= 1 t=0 l=0 chr=2 (b)
|
||||||
0x10 22000002 | 4 ptr= 2 t=0 l=1 f=0 chr=4 (d)
|
0x10 22000002 | 4 ptr= 2 t=0 l=1 chr=4 (d)
|
||||||
0x14 0a000002 | 5 ptr= 2 t=0 l=1 f=0 chr=1 (a)
|
0x14 0a000002 | 5 ptr= 2 t=0 l=1 chr=1 (a)
|
||||||
0x18 22000005 | 6 ptr= 5 t=0 l=1 f=0 chr=4 (d)
|
0x18 22000005 | 6 ptr= 5 t=0 l=1 chr=4 (d)
|
||||||
0x1c 08000003 | 7 ptr= 3 t=0 l=0 f=0 chr=1 (a)
|
0x1c 08000003 | 7 ptr= 3 t=0 l=0 chr=1 (a)
|
||||||
0x20 2a000006 | 8 ptr= 6 t=0 l=1 f=0 chr=5 (e)
|
0x20 2a000006 | 8 ptr= 6 t=0 l=1 chr=5 (e)
|
||||||
0x24 00000007 | 9 ptr= 7 t=0 l=0 f=0 chr=0 (`)
|
0x24 00000007 | 9 ptr= 7 t=0 l=0 chr=0 (`)
|
||||||
|
|
||||||
Strictly speaking, there is no node in the graph, only labelled edges.
|
|
||||||
|
|
||||||
|
|
||||||
|
Strictly speaking, there is no node in the graph, only labelled edges.
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
# Eliot
|
# Eliot
|
||||||
# Copyright (C) 1999 Antoine Fraboulet
|
# Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
# antoine.fraboulet@free.fr
|
# 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
|
# 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
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
@ -23,7 +24,6 @@ INCLUDES = -I$(top_srcdir)/dic
|
||||||
libgame_a_SOURCES= \
|
libgame_a_SOURCES= \
|
||||||
ai_percent.cpp ai_percent.h \
|
ai_percent.cpp ai_percent.h \
|
||||||
ai_player.h \
|
ai_player.h \
|
||||||
tile.cpp tile.h \
|
|
||||||
bag.cpp bag.h \
|
bag.cpp bag.h \
|
||||||
coord.cpp coord.h \
|
coord.cpp coord.h \
|
||||||
cross.cpp cross.h \
|
cross.cpp cross.h \
|
||||||
|
@ -31,17 +31,19 @@ libgame_a_SOURCES= \
|
||||||
board_cross.cpp \
|
board_cross.cpp \
|
||||||
board_search.cpp \
|
board_search.cpp \
|
||||||
duplicate.cpp duplicate.h \
|
duplicate.cpp duplicate.h \
|
||||||
encoding.cpp encoding.h \
|
|
||||||
freegame.cpp freegame.h \
|
freegame.cpp freegame.h \
|
||||||
game.cpp game.h \
|
game.cpp game.h \
|
||||||
game_factory.cpp game_factory.h \
|
game_factory.cpp game_factory.h \
|
||||||
game_io.cpp \
|
game_io.cpp \
|
||||||
|
move.cpp move.h \
|
||||||
player.cpp player.h \
|
player.cpp player.h \
|
||||||
pldrack.cpp pldrack.h \
|
pldrack.cpp pldrack.h \
|
||||||
rack.cpp rack.h \
|
rack.cpp rack.h \
|
||||||
results.cpp results.h \
|
results.cpp results.h \
|
||||||
round.cpp round.h \
|
round.cpp round.h \
|
||||||
|
settings.cpp settings.h \
|
||||||
training.cpp training.h \
|
training.cpp training.h \
|
||||||
turn.cpp turn.h \
|
turn.cpp turn.h \
|
||||||
history.cpp history.h
|
history.cpp history.h \
|
||||||
|
debug.h
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -21,6 +22,7 @@
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
|
#include "move.h"
|
||||||
#include "results.h"
|
#include "results.h"
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "ai_percent.h"
|
#include "ai_percent.h"
|
||||||
|
@ -37,32 +39,46 @@ AIPercent::AIPercent(int iId, float iPercent)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void AIPercent::compute(const Dictionary &iDic, Board &iBoard, int turn)
|
void AIPercent::compute(const Dictionary &iDic, Board &iBoard, bool iFirstWord)
|
||||||
{
|
{
|
||||||
m_results.clear();
|
m_results.clear();
|
||||||
|
|
||||||
Rack rack;
|
Rack rack;
|
||||||
getCurrentRack().getRack(rack);
|
getCurrentRack().getRack(rack);
|
||||||
m_results.search(iDic, iBoard, rack, turn);
|
m_results.search(iDic, iBoard, rack, iFirstWord);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool AIPercent::changesLetters() const
|
Move AIPercent::getMove() const
|
||||||
{
|
{
|
||||||
return (m_results.size() == 0);
|
if (m_results.size() == 0)
|
||||||
}
|
{
|
||||||
|
// If there is no result, simply pass the turn
|
||||||
|
// XXX: it is forbidden in duplicate mode, but well, what else to do?
|
||||||
const Round & AIPercent::getChosenRound() const
|
return Move(L"");
|
||||||
{
|
}
|
||||||
int index = (int)(m_percent * (m_results.size() - 1));
|
else
|
||||||
return m_results.get(index);
|
{
|
||||||
}
|
// If there are results, apply the algorithm
|
||||||
|
double wantedScore = m_percent * m_results.get(0).getPoints();
|
||||||
|
// Look for the first round giving at least 'wantedScore' points
|
||||||
vector<Tile> AIPercent::getChangedLetters() const
|
// Browse the results 10 by 10 (a dichotomy would be better, but this
|
||||||
{
|
// is not performance critical)
|
||||||
return vector<Tile>();
|
unsigned int index = 0;
|
||||||
|
while (index < m_results.size() &&
|
||||||
|
m_results.get(index).getPoints() > wantedScore)
|
||||||
|
{
|
||||||
|
index += 10;
|
||||||
|
}
|
||||||
|
// Now the wanted round is in the last 10 indices
|
||||||
|
if (index >= m_results.size())
|
||||||
|
index = m_results.size() - 1;
|
||||||
|
while (m_results.get(index).getPoints() < wantedScore)
|
||||||
|
{
|
||||||
|
--index;
|
||||||
|
}
|
||||||
|
return Move(m_results.get(index));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Local Variables:
|
/// Local Variables:
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -27,10 +28,10 @@
|
||||||
* This kind of AI is parameterized by a percentage p.
|
* This kind of AI is parameterized by a percentage p.
|
||||||
* The computation consists in finding all the N possible rounds for the
|
* The computation consists in finding all the N possible rounds for the
|
||||||
* current rack/board, and sorting the list.
|
* current rack/board, and sorting the list.
|
||||||
* The chosen round is the n'th element of the sorted list, such that n/N
|
* The chosen round is the one with the smallest score at least equal to
|
||||||
* is closest to the percentage p.
|
* p * best_score.
|
||||||
* A percentage of 0 should always return the best round (i.e. the one with
|
* A percentage of 1 should always return the best round (i.e. the one with
|
||||||
* the highest score), while a percentage of 1 should return the worst one.
|
* the highest score), while a percentage of 0 should return the worst one.
|
||||||
* This kind of AI will never change letters (unless it cannot play anything,
|
* This kind of AI will never change letters (unless it cannot play anything,
|
||||||
* in which case it just passes without changing letters).
|
* in which case it just passes without changing letters).
|
||||||
*/
|
*/
|
||||||
|
@ -45,17 +46,15 @@ public:
|
||||||
* This method does the actual computation. It will be called before any
|
* This method does the actual computation. It will be called before any
|
||||||
* of the following methods, so it must prepare everything for them.
|
* of the following methods, so it must prepare everything for them.
|
||||||
*/
|
*/
|
||||||
virtual void compute(const Dictionary &iDic, Board &iBoard, int turn);
|
virtual void compute(const Dictionary &iDic, Board &iBoard, bool iFirstWord);
|
||||||
/// Return true when the AI wants to change letters instead of playing a word
|
|
||||||
virtual bool changesLetters() const;
|
/// Return the move played by the AI
|
||||||
/// Return the round played by the AI (if changesLetters() returns false)
|
virtual Move getMove() const;
|
||||||
virtual const Round & getChosenRound() const;
|
|
||||||
/// Get the letters to change (if changesLetters() returns true)
|
|
||||||
virtual vector<Tile> getChangedLetters() const;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// Percentage used for this player
|
/// Percentage used for this player
|
||||||
float m_percent;
|
float m_percent;
|
||||||
|
|
||||||
/// Container for all the found solutions
|
/// Container for all the found solutions
|
||||||
Results m_results;
|
Results m_results;
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -22,10 +23,10 @@
|
||||||
|
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
|
|
||||||
|
class Dictionary;
|
||||||
class Round;
|
class Round;
|
||||||
class Board;
|
class Board;
|
||||||
class Tile;
|
class Tile;
|
||||||
typedef struct _Dictionary * Dictionary;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class is a pure interface, that must be implemented by all the AI
|
* This class is a pure interface, that must be implemented by all the AI
|
||||||
|
@ -69,22 +70,14 @@ public:
|
||||||
* This method does the actual computation. It will be called before any
|
* This method does the actual computation. It will be called before any
|
||||||
* of the following methods, so it must prepare everything for them.
|
* of the following methods, so it must prepare everything for them.
|
||||||
*/
|
*/
|
||||||
virtual void compute(const Dictionary &iDic, Board &iBoard, int turn) = 0;
|
virtual void compute(const Dictionary &iDic, Board &iBoard, bool iFirstWord) = 0;
|
||||||
/**
|
|
||||||
* Return true when the AI wants to change letters instead of playing a
|
/// Return the move played by the AI
|
||||||
* word.
|
virtual Move getMove() const = 0;
|
||||||
* Should return false in duplicate mode, as it is not allowed to change
|
|
||||||
* letters.
|
|
||||||
*/
|
|
||||||
virtual bool changesLetters() const = 0;
|
|
||||||
/// Return the round played by the AI (if changesLetters() returns false)
|
|
||||||
virtual const Round & getChosenRound() const = 0;
|
|
||||||
/// Get the letters to change (if changesLetters() returns true)
|
|
||||||
virtual vector<Tile> getChangedLetters() const = 0;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/// This class is a pure interface, forbid any direct instanciation
|
/// This class is a pure interface, forbid any direct instanciation
|
||||||
AIPlayer(int iId): Player(iId) {}
|
AIPlayer(unsigned int iId): Player(iId) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
84
game/bag.cpp
84
game/bag.cpp
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -20,22 +21,18 @@
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#include "tile.h"
|
#include <dic.h>
|
||||||
#include "bag.h"
|
#include "bag.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
#include "encoding.h"
|
||||||
|
|
||||||
|
|
||||||
Bag::Bag()
|
Bag::Bag(const Dictionary &iDic)
|
||||||
{
|
: m_dic(iDic)
|
||||||
init();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void Bag::init()
|
|
||||||
{
|
{
|
||||||
m_ntiles = 0;
|
m_ntiles = 0;
|
||||||
const list<Tile>& allTiles = Tile::getAllTiles();
|
const vector<Tile>& allTiles = m_dic.getAllTiles();
|
||||||
list<Tile>::const_iterator it;
|
vector<Tile>::const_iterator it;
|
||||||
for (it = allTiles.begin(); it != allTiles.end(); it++)
|
for (it = allTiles.begin(); it != allTiles.end(); it++)
|
||||||
{
|
{
|
||||||
m_tilesMap[*it] = it->maxNumber();
|
m_tilesMap[*it] = it->maxNumber();
|
||||||
|
@ -53,7 +50,7 @@ unsigned int Bag::in(const Tile &iTile) const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
unsigned int Bag::nVowels() const
|
unsigned int Bag::getNbVowels() const
|
||||||
{
|
{
|
||||||
map<Tile, int>::const_iterator it;
|
map<Tile, int>::const_iterator it;
|
||||||
int v = 0;
|
int v = 0;
|
||||||
|
@ -67,7 +64,7 @@ unsigned int Bag::nVowels() const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
unsigned int Bag::nConsonants() const
|
unsigned int Bag::getNbConsonants() const
|
||||||
{
|
{
|
||||||
map<Tile, int>::const_iterator it;
|
map<Tile, int>::const_iterator it;
|
||||||
int c = 0;
|
int c = 0;
|
||||||
|
@ -84,7 +81,7 @@ unsigned int Bag::nConsonants() const
|
||||||
void Bag::takeTile(const Tile &iTile)
|
void Bag::takeTile(const Tile &iTile)
|
||||||
{
|
{
|
||||||
ASSERT(in(iTile),
|
ASSERT(in(iTile),
|
||||||
(wstring(L"The bag does not contain the letter ") + iTile.toChar()).c_str());
|
"The bag does not contain the letter " + convertToMb(iTile.toChar()));
|
||||||
|
|
||||||
m_tilesMap[iTile]--;
|
m_tilesMap[iTile]--;
|
||||||
m_ntiles--;
|
m_ntiles--;
|
||||||
|
@ -94,20 +91,21 @@ void Bag::takeTile(const Tile &iTile)
|
||||||
void Bag::replaceTile(const Tile &iTile)
|
void Bag::replaceTile(const Tile &iTile)
|
||||||
{
|
{
|
||||||
ASSERT(in(iTile) < iTile.maxNumber(),
|
ASSERT(in(iTile) < iTile.maxNumber(),
|
||||||
(wstring(L"Cannot replace tile: ") + iTile.toChar()).c_str());
|
"Cannot replace tile: " + convertToMb(iTile.toChar()));
|
||||||
|
|
||||||
m_tilesMap[iTile]++;
|
m_tilesMap[iTile]++;
|
||||||
m_ntiles++;
|
m_ntiles++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Tile Bag::selectRandom()
|
Tile Bag::selectRandom() const
|
||||||
{
|
{
|
||||||
map<Tile, int>::const_iterator it;
|
|
||||||
int n;
|
|
||||||
double max = m_ntiles;
|
double max = m_ntiles;
|
||||||
|
ASSERT(max > 0, "The bag is empty");
|
||||||
|
|
||||||
n = (int)(max * rand() / (RAND_MAX + 1.0));
|
int n = (int)(max * rand() / (RAND_MAX + 1.0));
|
||||||
|
|
||||||
|
map<Tile, int>::const_iterator it;
|
||||||
for (it = m_tilesMap.begin(); it != m_tilesMap.end(); it++)
|
for (it = m_tilesMap.begin(); it != m_tilesMap.end(); it++)
|
||||||
{
|
{
|
||||||
if (n < it->second)
|
if (n < it->second)
|
||||||
|
@ -115,7 +113,49 @@ Tile Bag::selectRandom()
|
||||||
n -= it->second;
|
n -= it->second;
|
||||||
}
|
}
|
||||||
ASSERT(false, "We should not come here");
|
ASSERT(false, "We should not come here");
|
||||||
return Tile::dummy();
|
return Tile();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Tile Bag::selectRandomVowel() const
|
||||||
|
{
|
||||||
|
double max = getNbVowels();
|
||||||
|
ASSERT(max > 0, "Not enough vowels in the bag");
|
||||||
|
|
||||||
|
int n = (int)(max * rand() / (RAND_MAX + 1.0));
|
||||||
|
|
||||||
|
map<Tile, int>::const_iterator it;
|
||||||
|
for (it = m_tilesMap.begin(); it != m_tilesMap.end(); it++)
|
||||||
|
{
|
||||||
|
if (!it->first.isVowel())
|
||||||
|
continue;
|
||||||
|
if (n < it->second)
|
||||||
|
return it->first;
|
||||||
|
n -= it->second;
|
||||||
|
}
|
||||||
|
ASSERT(false, "We should not come here");
|
||||||
|
return Tile();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Tile Bag::selectRandomConsonant() const
|
||||||
|
{
|
||||||
|
double max = getNbConsonants();
|
||||||
|
ASSERT(max > 0, "Not enough consonants in the bag");
|
||||||
|
|
||||||
|
int n = (int)(max * rand() / (RAND_MAX + 1.0));
|
||||||
|
|
||||||
|
map<Tile, int>::const_iterator it;
|
||||||
|
for (it = m_tilesMap.begin(); it != m_tilesMap.end(); it++)
|
||||||
|
{
|
||||||
|
if (!it->first.isConsonant())
|
||||||
|
continue;
|
||||||
|
if (n < it->second)
|
||||||
|
return it->first;
|
||||||
|
n -= it->second;
|
||||||
|
}
|
||||||
|
ASSERT(false, "We should not come here");
|
||||||
|
return Tile();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
41
game/bag.h
41
game/bag.h
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -21,11 +22,13 @@
|
||||||
#ifndef _BAG_H_
|
#ifndef _BAG_H_
|
||||||
#define _BAG_H_
|
#define _BAG_H_
|
||||||
|
|
||||||
#include "tile.h"
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include "tile.h"
|
||||||
|
|
||||||
using std::map;
|
using std::map;
|
||||||
|
|
||||||
|
class Dictionary;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A bag stores the set of free tiles for the game.
|
* A bag stores the set of free tiles for the game.
|
||||||
|
@ -33,9 +36,7 @@ using std::map;
|
||||||
class Bag
|
class Bag
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Bag();
|
explicit Bag(const Dictionary &iDic);
|
||||||
virtual ~Bag() {}
|
|
||||||
void init();
|
|
||||||
|
|
||||||
/// Take a tile in the bag
|
/// Take a tile in the bag
|
||||||
void takeTile(const Tile &iTile);
|
void takeTile(const Tile &iTile);
|
||||||
|
@ -47,18 +48,30 @@ public:
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return how many tiles/vowels/consonants are available
|
* Return how many tiles/vowels/consonants are available
|
||||||
* Warning: b.nVowels() + b.nConsonants() != b.nTiles(),
|
* Warning: b.getNbVowels() + b.getNbConsonants() != b.getNbTiles(),
|
||||||
* because of the jokers and the 'Y'.
|
* because of the jokers and the 'Y'.
|
||||||
*/
|
*/
|
||||||
unsigned int nTiles() const { return m_ntiles; }
|
unsigned int getNbTiles() const { return m_ntiles; }
|
||||||
unsigned int nVowels() const;
|
unsigned int getNbVowels() const;
|
||||||
unsigned int nConsonants() const;
|
unsigned int getNbConsonants() const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a random available tile
|
* Return a random available tile
|
||||||
* The tile is not taken out of the bag.
|
* The tile is not taken out of the bag.
|
||||||
*/
|
*/
|
||||||
Tile selectRandom();
|
Tile selectRandom() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a random available vowel.
|
||||||
|
* The tile is not taken out of the bag.
|
||||||
|
*/
|
||||||
|
Tile selectRandomVowel() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a random available consonant.
|
||||||
|
* The tile is not taken out of the bag.
|
||||||
|
*/
|
||||||
|
Tile selectRandomConsonant() const;
|
||||||
|
|
||||||
void operator=(const Bag &iOther);
|
void operator=(const Bag &iOther);
|
||||||
|
|
||||||
|
@ -66,8 +79,12 @@ public:
|
||||||
void dumpAll() const;
|
void dumpAll() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
/// Dictionary
|
||||||
|
const Dictionary &m_dic;
|
||||||
|
|
||||||
/// Associate to each tile its number of occurrences in the bag
|
/// Associate to each tile its number of occurrences in the bag
|
||||||
map<Tile, int> m_tilesMap;
|
map<Tile, int> m_tilesMap;
|
||||||
|
|
||||||
/// Total number of tiles in the bag
|
/// Total number of tiles in the bag
|
||||||
int m_ntiles;
|
int m_ntiles;
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -82,8 +83,8 @@ const int Board::m_wordMultipliers[BOARD_REALDIM][BOARD_REALDIM] =
|
||||||
|
|
||||||
|
|
||||||
Board::Board():
|
Board::Board():
|
||||||
m_tilesRow(BOARD_REALDIM, Tile::dummy()),
|
m_tilesRow(BOARD_REALDIM, Tile()),
|
||||||
m_tilesCol(BOARD_REALDIM, Tile::dummy()),
|
m_tilesCol(BOARD_REALDIM, Tile()),
|
||||||
m_jokerRow(BOARD_REALDIM, false),
|
m_jokerRow(BOARD_REALDIM, false),
|
||||||
m_jokerCol(BOARD_REALDIM, false),
|
m_jokerCol(BOARD_REALDIM, false),
|
||||||
m_crossRow(BOARD_REALDIM, Cross()),
|
m_crossRow(BOARD_REALDIM, Cross()),
|
||||||
|
@ -154,13 +155,12 @@ bool Board::isVacant(int iRow, int iCol) const
|
||||||
void Board::addRound(const Dictionary &iDic, const Round &iRound)
|
void Board::addRound(const Dictionary &iDic, const Round &iRound)
|
||||||
{
|
{
|
||||||
Tile t;
|
Tile t;
|
||||||
int row, col;
|
|
||||||
|
|
||||||
row = iRound.getCoord().getRow();
|
int row = iRound.getCoord().getRow();
|
||||||
col = iRound.getCoord().getCol();
|
int col = iRound.getCoord().getCol();
|
||||||
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (m_tilesRow[row][col + i].isEmpty())
|
if (m_tilesRow[row][col + i].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -174,7 +174,7 @@ void Board::addRound(const Dictionary &iDic, const Round &iRound)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (m_tilesRow[row + i][col].isEmpty())
|
if (m_tilesRow[row + i][col].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -195,20 +195,18 @@ void Board::addRound(const Dictionary &iDic, const Round &iRound)
|
||||||
|
|
||||||
void Board::removeRound(const Dictionary &iDic, const Round &iRound)
|
void Board::removeRound(const Dictionary &iDic, const Round &iRound)
|
||||||
{
|
{
|
||||||
int row, col;
|
int row = iRound.getCoord().getRow();
|
||||||
|
int col = iRound.getCoord().getCol();
|
||||||
row = iRound.getCoord().getRow();
|
|
||||||
col = iRound.getCoord().getCol();
|
|
||||||
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (iRound.isPlayedFromRack(i))
|
if (iRound.isPlayedFromRack(i))
|
||||||
{
|
{
|
||||||
m_tilesRow[row][col + i] = Tile::dummy();
|
m_tilesRow[row][col + i] = Tile();
|
||||||
m_jokerRow[row][col + i] = false;
|
m_jokerRow[row][col + i] = false;
|
||||||
m_crossRow[row][col + i].setAny();
|
m_crossRow[row][col + i].setAny();
|
||||||
m_tilesCol[col + i][row] = Tile::dummy();
|
m_tilesCol[col + i][row] = Tile();
|
||||||
m_jokerCol[col + i][row] = false;
|
m_jokerCol[col + i][row] = false;
|
||||||
m_crossCol[col + i][row].setAny();
|
m_crossCol[col + i][row].setAny();
|
||||||
}
|
}
|
||||||
|
@ -216,14 +214,14 @@ void Board::removeRound(const Dictionary &iDic, const Round &iRound)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (iRound.isPlayedFromRack(i))
|
if (iRound.isPlayedFromRack(i))
|
||||||
{
|
{
|
||||||
m_tilesRow[row + i][col] = Tile::dummy();
|
m_tilesRow[row + i][col] = Tile();
|
||||||
m_jokerRow[row + i][col] = false;
|
m_jokerRow[row + i][col] = false;
|
||||||
m_crossRow[row + i][col].setAny();
|
m_crossRow[row + i][col].setAny();
|
||||||
m_tilesCol[col][row + i] = Tile::dummy();
|
m_tilesCol[col][row + i] = Tile();
|
||||||
m_jokerCol[col][row + i] = false;
|
m_jokerCol[col][row + i] = false;
|
||||||
m_crossCol[col][row + i].setAny();
|
m_crossCol[col][row + i].setAny();
|
||||||
}
|
}
|
||||||
|
@ -246,17 +244,15 @@ int Board::checkRoundAux(Matrix<Tile> &iTilesMx,
|
||||||
bool firstturn)
|
bool firstturn)
|
||||||
{
|
{
|
||||||
Tile t;
|
Tile t;
|
||||||
int row, col, i;
|
int l, p;
|
||||||
int l, p, fromrack;
|
|
||||||
int pts, ptscross, wordmul;
|
|
||||||
bool isolated = true;
|
bool isolated = true;
|
||||||
|
|
||||||
fromrack = 0;
|
unsigned int fromrack = 0;
|
||||||
pts = 0;
|
int pts = 0;
|
||||||
ptscross = 0;
|
int ptscross = 0;
|
||||||
wordmul = 1;
|
int wordmul = 1;
|
||||||
row = iRound.getCoord().getRow();
|
int row = iRound.getCoord().getRow();
|
||||||
col = iRound.getCoord().getCol();
|
int col = iRound.getCoord().getCol();
|
||||||
|
|
||||||
/* Is the word an extension of another word? */
|
/* Is the word an extension of another word? */
|
||||||
if (!iTilesMx[row][col - 1].isEmpty() ||
|
if (!iTilesMx[row][col - 1].isEmpty() ||
|
||||||
|
@ -265,7 +261,7 @@ int Board::checkRoundAux(Matrix<Tile> &iTilesMx,
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
t = iRound.getTile(i);
|
t = iRound.getTile(i);
|
||||||
if (!iTilesMx[row][col + i].isEmpty())
|
if (!iTilesMx[row][col + i].isEmpty())
|
||||||
|
@ -383,13 +379,12 @@ int Board::checkRound(Round &iRound, bool firstturn)
|
||||||
void Board::testRound(const Round &iRound)
|
void Board::testRound(const Round &iRound)
|
||||||
{
|
{
|
||||||
Tile t;
|
Tile t;
|
||||||
int row, col;
|
|
||||||
|
|
||||||
row = iRound.getCoord().getRow();
|
int row = iRound.getCoord().getRow();
|
||||||
col = iRound.getCoord().getCol();
|
int col = iRound.getCoord().getCol();
|
||||||
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
if (iRound.getCoord().getDir() == Coord::HORIZONTAL)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (m_tilesRow[row][col + i].isEmpty())
|
if (m_tilesRow[row][col + i].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -405,7 +400,7 @@ void Board::testRound(const Round &iRound)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (m_tilesRow[row + i][col].isEmpty())
|
if (m_tilesRow[row + i][col].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -430,11 +425,11 @@ void Board::removeTestRound()
|
||||||
{
|
{
|
||||||
if (m_testsRow[row][col])
|
if (m_testsRow[row][col])
|
||||||
{
|
{
|
||||||
m_tilesRow[row][col] = Tile::dummy();
|
m_tilesRow[row][col] = Tile();
|
||||||
m_testsRow[row][col] = 0;
|
m_testsRow[row][col] = 0;
|
||||||
m_jokerRow[row][col] = false;
|
m_jokerRow[row][col] = false;
|
||||||
|
|
||||||
m_tilesCol[col][row] = Tile::dummy();
|
m_tilesCol[col][row] = Tile();
|
||||||
m_jokerCol[col][row] = false;
|
m_jokerCol[col][row] = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
15
game/board.h
15
game/board.h
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -21,12 +22,13 @@
|
||||||
#ifndef _BOARD_H_
|
#ifndef _BOARD_H_
|
||||||
#define _BOARD_H_
|
#define _BOARD_H_
|
||||||
|
|
||||||
#include "tile.h"
|
|
||||||
#include "cross.h"
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
typedef struct _Dictionary*Dictionary;
|
#include "tile.h"
|
||||||
|
#include "cross.h"
|
||||||
|
|
||||||
|
class Dictionary;
|
||||||
class Rack;
|
class Rack;
|
||||||
class Round;
|
class Round;
|
||||||
class Results;
|
class Results;
|
||||||
|
@ -61,7 +63,6 @@ class Board
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Board();
|
Board();
|
||||||
virtual ~Board() {}
|
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
* Coordinates have to be BOARD_MIN <= int <= BOARD_MAX
|
* Coordinates have to be BOARD_MIN <= int <= BOARD_MAX
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -25,11 +26,14 @@
|
||||||
* \date 2005
|
* \date 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <wctype.h>
|
||||||
|
|
||||||
#include <dic.h>
|
#include <dic.h>
|
||||||
#include "tile.h"
|
#include "tile.h"
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
static void Board_checkout_tile(const Dictionary &iDic,
|
static void Board_checkout_tile(const Dictionary &iDic,
|
||||||
vector<Tile>& iTiles,
|
vector<Tile>& iTiles,
|
||||||
vector<bool> & iJoker,
|
vector<bool> & iJoker,
|
||||||
|
@ -37,7 +41,7 @@ static void Board_checkout_tile(const Dictionary &iDic,
|
||||||
int& oPoints,
|
int& oPoints,
|
||||||
int index)
|
int index)
|
||||||
{
|
{
|
||||||
int i,left;
|
int i, left;
|
||||||
unsigned int node, succ;
|
unsigned int node, succ;
|
||||||
|
|
||||||
oPoints = 0;
|
oPoints = 0;
|
||||||
|
@ -52,36 +56,36 @@ static void Board_checkout_tile(const Dictionary &iDic,
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: create temporary strings until the dictionary uses Tile objects
|
// FIXME: create temporary strings until the dictionary uses Tile objects
|
||||||
char leftTiles [BOARD_DIM + 1];
|
wchar_t leftTiles [BOARD_DIM + 1];
|
||||||
char rightTiles[BOARD_DIM + 1];
|
wchar_t rightTiles[BOARD_DIM + 1];
|
||||||
|
|
||||||
for (i = left; i < index; i++)
|
for (i = left; i < index; i++)
|
||||||
leftTiles[i - left] = toupper(iTiles[i].toChar());
|
leftTiles[i - left] = towupper(iTiles[i].toChar());
|
||||||
leftTiles[index - left] = 0;
|
leftTiles[index - left] = 0;
|
||||||
|
|
||||||
for (i = index + 1; !iTiles[i].isEmpty(); i++)
|
for (i = index + 1; !iTiles[i].isEmpty(); i++)
|
||||||
rightTiles[i - index - 1] = toupper(iTiles[i].toChar());
|
rightTiles[i - index - 1] = towupper(iTiles[i].toChar());
|
||||||
rightTiles[i - index - 1] = 0;
|
rightTiles[i - index - 1] = 0;
|
||||||
|
|
||||||
/* Tiles that can be played */
|
/* Tiles that can be played */
|
||||||
node = Dic_char_lookup(iDic, Dic_root(iDic), leftTiles);
|
node = iDic.charLookup(iDic.getRoot(), leftTiles);
|
||||||
if (node == 0)
|
if (node == 0)
|
||||||
{
|
{
|
||||||
oCross.setNone();
|
oCross.setNone();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (succ = Dic_succ(iDic, node); succ; succ = Dic_next(iDic, succ))
|
for (succ = iDic.getSucc(node); succ; succ = iDic.getNext(succ))
|
||||||
{
|
{
|
||||||
if (Dic_word(iDic, Dic_char_lookup(iDic, succ, rightTiles)))
|
if (iDic.isEndOfWord(iDic.charLookup(succ, rightTiles)))
|
||||||
oCross.insert(Tile(Dic_char(iDic, succ)));
|
oCross.insert(Tile(iDic.getChar(succ)));
|
||||||
if (Dic_last(iDic, succ))
|
if (iDic.isLast(succ))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Points on the right part */
|
/* Points on the right part */
|
||||||
/* yes, it is REALLY [index+1] */
|
/* yes, it is REALLY [index + 1] */
|
||||||
while (!iTiles[index+1].isEmpty())
|
while (!iTiles[index + 1].isEmpty())
|
||||||
{
|
{
|
||||||
index++;
|
index++;
|
||||||
if (!iJoker[index])
|
if (!iJoker[index])
|
||||||
|
@ -104,7 +108,7 @@ static void Board_check(const Dictionary &iDic,
|
||||||
if (!iTilesMx[i][j].isEmpty())
|
if (!iTilesMx[i][j].isEmpty())
|
||||||
{
|
{
|
||||||
iCrossMx[j][i].setNone();
|
iCrossMx[j][i].setNone();
|
||||||
}
|
}
|
||||||
else if (!iTilesMx[i][j - 1].isEmpty() ||
|
else if (!iTilesMx[i][j - 1].isEmpty() ||
|
||||||
!iTilesMx[i][j + 1].isEmpty())
|
!iTilesMx[i][j + 1].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -119,7 +123,7 @@ static void Board_check(const Dictionary &iDic,
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
iCrossMx[j][i].setAny();
|
iCrossMx[j][i].setAny();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -38,21 +39,17 @@ static void BoardSearchEvalMove(const Board &iBoard,
|
||||||
Matrix<bool> &iJokerMx,
|
Matrix<bool> &iJokerMx,
|
||||||
Results &iResults, Round &iWord)
|
Results &iResults, Round &iWord)
|
||||||
{
|
{
|
||||||
int i, pts, ptscross;
|
unsigned int fromrack = 0;
|
||||||
int l, t, fromrack;
|
int pts = 0;
|
||||||
int len, row, col, wordmul;
|
int ptscross = 0;
|
||||||
|
int wordmul = 1;
|
||||||
|
|
||||||
fromrack = 0;
|
unsigned int len = iWord.getWordLen();
|
||||||
pts = 0;
|
|
||||||
ptscross = 0;
|
|
||||||
wordmul = 1;
|
|
||||||
|
|
||||||
len = iWord.getWordLen();
|
int row = iWord.getCoord().getRow();
|
||||||
|
int col = iWord.getCoord().getCol();
|
||||||
|
|
||||||
row = iWord.getCoord().getRow();
|
for (unsigned int i = 0; i < len; i++)
|
||||||
col = iWord.getCoord().getCol();
|
|
||||||
|
|
||||||
for (i = 0; i < len; i++)
|
|
||||||
{
|
{
|
||||||
if (!iTilesMx[row][col+i].isEmpty())
|
if (!iTilesMx[row][col+i].isEmpty())
|
||||||
{
|
{
|
||||||
|
@ -61,6 +58,7 @@ static void BoardSearchEvalMove(const Board &iBoard,
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
int l;
|
||||||
if (!iWord.isJoker(i))
|
if (!iWord.isJoker(i))
|
||||||
l = iWord.getTile(i).getPoints() *
|
l = iWord.getTile(i).getPoints() *
|
||||||
iBoard.getLetterMultiplier(row, col + i);
|
iBoard.getLetterMultiplier(row, col + i);
|
||||||
|
@ -69,7 +67,7 @@ static void BoardSearchEvalMove(const Board &iBoard,
|
||||||
pts += l;
|
pts += l;
|
||||||
wordmul *= iBoard.getWordMultiplier(row, col + i);
|
wordmul *= iBoard.getWordMultiplier(row, col + i);
|
||||||
|
|
||||||
t = iPointsMx[row][col+i];
|
int t = iPointsMx[row][col+i];
|
||||||
if (t >= 0)
|
if (t >= 0)
|
||||||
ptscross += (t + l) * iBoard.getWordMultiplier(row, col + i);
|
ptscross += (t + l) * iBoard.getWordMultiplier(row, col + i);
|
||||||
fromrack++;
|
fromrack++;
|
||||||
|
@ -109,33 +107,40 @@ static void ExtendRight(const Board &iBoard,
|
||||||
|
|
||||||
if (iTilesMx[iRow][iCol].isEmpty())
|
if (iTilesMx[iRow][iCol].isEmpty())
|
||||||
{
|
{
|
||||||
if (Dic_word(iDic, iNode) && iCol > iAnchor)
|
if (iDic.isEndOfWord(iNode) && iCol > iAnchor)
|
||||||
|
{
|
||||||
BoardSearchEvalMove(iBoard, iTilesMx, iPointsMx, iJokerMx,
|
BoardSearchEvalMove(iBoard, iTilesMx, iPointsMx, iJokerMx,
|
||||||
iResults, ioPartialWord);
|
iResults, ioPartialWord);
|
||||||
|
}
|
||||||
|
|
||||||
for (succ = Dic_succ(iDic, iNode); succ; succ = Dic_next(iDic, succ))
|
// Optimization: avoid entering the for loop if no tile can match
|
||||||
|
if (iCrossMx[iRow][iCol].isNone())
|
||||||
|
return;
|
||||||
|
|
||||||
|
bool hasJokerInRack = iRack.in(Tile::Joker());
|
||||||
|
for (succ = iDic.getSucc(iNode); succ; succ = iDic.getNext(succ))
|
||||||
{
|
{
|
||||||
l = Tile(Dic_char(iDic, succ));
|
l = Tile(iDic.getChar(succ));
|
||||||
if (iCrossMx[iRow][iCol].check(l))
|
if (iCrossMx[iRow][iCol].check(l))
|
||||||
{
|
{
|
||||||
if (iRack.in(l))
|
if (iRack.in(l))
|
||||||
{
|
{
|
||||||
iRack.remove(l);
|
iRack.remove(l);
|
||||||
ioPartialWord.addRightFromRack(l, 0);
|
ioPartialWord.addRightFromRack(l, false);
|
||||||
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, ioPartialWord, iResults,
|
iJokerMx, iRack, ioPartialWord, iResults,
|
||||||
succ, iRow, iCol + 1, iAnchor);
|
succ, iRow, iCol + 1, iAnchor);
|
||||||
ioPartialWord.removeRightToRack(l, 0);
|
ioPartialWord.removeRightToRack(l, false);
|
||||||
iRack.add(l);
|
iRack.add(l);
|
||||||
}
|
}
|
||||||
if (iRack.in(Tile::Joker()))
|
if (hasJokerInRack)
|
||||||
{
|
{
|
||||||
iRack.remove(Tile::Joker());
|
iRack.remove(Tile::Joker());
|
||||||
ioPartialWord.addRightFromRack(l, 1);
|
ioPartialWord.addRightFromRack(l, true);
|
||||||
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, ioPartialWord, iResults,
|
iJokerMx, iRack, ioPartialWord, iResults,
|
||||||
succ, iRow, iCol + 1, iAnchor);
|
succ, iRow, iCol + 1, iAnchor);
|
||||||
ioPartialWord.removeRightToRack(l, 1);
|
ioPartialWord.removeRightToRack(l, true);
|
||||||
iRack.add(Tile::Joker());
|
iRack.add(Tile::Joker());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -144,15 +149,19 @@ static void ExtendRight(const Board &iBoard,
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
l = iTilesMx[iRow][iCol];
|
l = iTilesMx[iRow][iCol];
|
||||||
for (succ = Dic_succ(iDic, iNode); succ ; succ = Dic_next(iDic, succ))
|
wint_t upperChar = towupper(l.toChar());
|
||||||
|
for (succ = iDic.getSucc(iNode); succ ; succ = iDic.getNext(succ))
|
||||||
{
|
{
|
||||||
if (Dic_char(iDic, succ) == toupper(l.toChar()))
|
if ((wint_t)iDic.getChar(succ) == upperChar)
|
||||||
{
|
{
|
||||||
ioPartialWord.addRightFromBoard(l);
|
ioPartialWord.addRightFromBoard(l);
|
||||||
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, ioPartialWord,
|
iJokerMx, iRack, ioPartialWord,
|
||||||
iResults, succ, iRow, iCol + 1, iAnchor);
|
iResults, succ, iRow, iCol + 1, iAnchor);
|
||||||
ioPartialWord.removeRightToBoard(l);
|
ioPartialWord.removeRightToBoard(l);
|
||||||
|
// The letter will be present only once in the dictionary,
|
||||||
|
// so we can stop looping
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -177,31 +186,32 @@ static void LeftPart(const Board &iBoard,
|
||||||
|
|
||||||
if (iLimit > 0)
|
if (iLimit > 0)
|
||||||
{
|
{
|
||||||
for (succ = Dic_succ(iDic, n); succ; succ = Dic_next(iDic, succ))
|
bool hasJokerInRack = iRack.in(Tile::Joker());
|
||||||
|
for (succ = iDic.getSucc(n); succ; succ = iDic.getNext(succ))
|
||||||
{
|
{
|
||||||
l = Tile(Dic_char(iDic, succ));
|
l = Tile(iDic.getChar(succ));
|
||||||
if (iRack.in(l))
|
if (iRack.in(l))
|
||||||
{
|
{
|
||||||
iRack.remove(l);
|
iRack.remove(l);
|
||||||
ioPartialWord.addRightFromRack(l, 0);
|
ioPartialWord.addRightFromRack(l, false);
|
||||||
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
||||||
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, ioPartialWord, iResults,
|
iJokerMx, iRack, ioPartialWord, iResults,
|
||||||
succ, iRow, iAnchor, iLimit - 1);
|
succ, iRow, iAnchor, iLimit - 1);
|
||||||
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
||||||
ioPartialWord.removeRightToRack(l, 0);
|
ioPartialWord.removeRightToRack(l, false);
|
||||||
iRack.add(l);
|
iRack.add(l);
|
||||||
}
|
}
|
||||||
if (iRack.in(Tile::Joker()))
|
if (hasJokerInRack)
|
||||||
{
|
{
|
||||||
iRack.remove(Tile::Joker());
|
iRack.remove(Tile::Joker());
|
||||||
ioPartialWord.addRightFromRack(l, 1);
|
ioPartialWord.addRightFromRack(l, true);
|
||||||
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() - 1);
|
||||||
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, ioPartialWord, iResults,
|
iJokerMx, iRack, ioPartialWord, iResults,
|
||||||
succ, iRow, iAnchor, iLimit - 1);
|
succ, iRow, iAnchor, iLimit - 1);
|
||||||
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
ioPartialWord.accessCoord().setCol(ioPartialWord.getCoord().getCol() + 1);
|
||||||
ioPartialWord.removeRightToRack(l, 1);
|
ioPartialWord.removeRightToRack(l, true);
|
||||||
iRack.add(Tile::Joker());
|
iRack.add(Tile::Joker());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -221,10 +231,10 @@ static void BoardSearchAux(const Board &iBoard,
|
||||||
int row, col, lastanchor;
|
int row, col, lastanchor;
|
||||||
Round partialword;
|
Round partialword;
|
||||||
|
|
||||||
list<Tile> rackTiles;
|
vector<Tile> rackTiles;
|
||||||
iRack.getTiles(rackTiles);
|
iRack.getTiles(rackTiles);
|
||||||
list<Tile>::const_iterator it;
|
vector<Tile>::const_iterator it;
|
||||||
bool match;
|
vector<Tile>::const_iterator itEnd;
|
||||||
|
|
||||||
for (row = 1; row <= BOARD_DIM; row++)
|
for (row = 1; row <= BOARD_DIM; row++)
|
||||||
{
|
{
|
||||||
|
@ -237,7 +247,7 @@ static void BoardSearchAux(const Board &iBoard,
|
||||||
if (iTilesMx[row][col].isEmpty() &&
|
if (iTilesMx[row][col].isEmpty() &&
|
||||||
(!iTilesMx[row][col - 1].isEmpty() ||
|
(!iTilesMx[row][col - 1].isEmpty() ||
|
||||||
!iTilesMx[row][col + 1].isEmpty() ||
|
!iTilesMx[row][col + 1].isEmpty() ||
|
||||||
!iTilesMx[row - 1][col].isEmpty() ||
|
!iTilesMx[row - 1][col].isEmpty() ||
|
||||||
!iTilesMx[row + 1][col].isEmpty()))
|
!iTilesMx[row + 1][col].isEmpty()))
|
||||||
{
|
{
|
||||||
#if defined(DONT_USE_SEARCH_OPTIMIZATION)
|
#if defined(DONT_USE_SEARCH_OPTIMIZATION)
|
||||||
|
@ -261,13 +271,13 @@ static void BoardSearchAux(const Board &iBoard,
|
||||||
// Optimization compared to the original Appel & Jacobson
|
// Optimization compared to the original Appel & Jacobson
|
||||||
// algorithm: skip Leftpart if none of the tiles of the rack
|
// algorithm: skip Leftpart if none of the tiles of the rack
|
||||||
// matches the cross mask for the current anchor
|
// matches the cross mask for the current anchor
|
||||||
match = false;
|
bool match = false;
|
||||||
for (it = rackTiles.begin();
|
for (it = rackTiles.begin(); it != rackTiles.end(); it++)
|
||||||
!match && it != rackTiles.end(); it++)
|
|
||||||
{
|
{
|
||||||
if (iCrossMx[row][col].check(*it))
|
if (iCrossMx[row][col].check(*it))
|
||||||
{
|
{
|
||||||
match = true;
|
match = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (match)
|
if (match)
|
||||||
|
@ -277,14 +287,14 @@ static void BoardSearchAux(const Board &iBoard,
|
||||||
partialword.accessCoord().setCol(lastanchor + 1);
|
partialword.accessCoord().setCol(lastanchor + 1);
|
||||||
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
ExtendRight(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, partialword, iResults,
|
iJokerMx, iRack, partialword, iResults,
|
||||||
Dic_root(iDic), row, lastanchor + 1, col);
|
iDic.getRoot(), row, lastanchor + 1, col);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
partialword.accessCoord().setCol(col);
|
partialword.accessCoord().setCol(col);
|
||||||
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
LeftPart(iBoard, iDic, iTilesMx, iCrossMx, iPointsMx,
|
||||||
iJokerMx, iRack, partialword, iResults,
|
iJokerMx, iRack, partialword, iResults,
|
||||||
Dic_root(iDic), row, col, col -
|
iDic.getRoot(), row, col, col -
|
||||||
lastanchor - 1);
|
lastanchor - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -328,8 +338,8 @@ void Board::searchFirst(const Dictionary &iDic,
|
||||||
partialword.accessCoord().setDir(Coord::HORIZONTAL);
|
partialword.accessCoord().setDir(Coord::HORIZONTAL);
|
||||||
LeftPart(*this, iDic, m_tilesRow, m_crossRow,
|
LeftPart(*this, iDic, m_tilesRow, m_crossRow,
|
||||||
m_pointRow, m_jokerRow,
|
m_pointRow, m_jokerRow,
|
||||||
copyRack, partialword, oResults, Dic_root(iDic), row, col,
|
copyRack, partialword, oResults, iDic.getRoot(), row, col,
|
||||||
copyRack.nTiles() - 1);
|
copyRack.getNbTiles() - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Local Variables:
|
/// Local Variables:
|
||||||
|
|
|
@ -1,21 +1,23 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file coord.cpp
|
* \file coord.cpp
|
||||||
|
|
49
game/coord.h
49
game/coord.h
|
@ -1,21 +1,23 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file coord.h
|
* \file coord.h
|
||||||
|
@ -27,19 +29,24 @@
|
||||||
#ifndef _COORD_H
|
#ifndef _COORD_H
|
||||||
#define _COORD_H
|
#define _COORD_H
|
||||||
|
|
||||||
using std::string;
|
#include <string>
|
||||||
|
|
||||||
using std::wstring;
|
using std::wstring;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class handles coordinates of a square on the board.
|
||||||
|
* The row and column start at 1.
|
||||||
|
*/
|
||||||
class Coord
|
class Coord
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
||||||
enum Direction {VERTICAL, HORIZONTAL};
|
enum Direction {VERTICAL, HORIZONTAL};
|
||||||
|
|
||||||
// Construction, destruction
|
// Construction
|
||||||
Coord(int iRow = -1, int iCol = -1, Direction iDir = HORIZONTAL);
|
Coord(int iRow = -1, int iCol = -1, Direction iDir = HORIZONTAL);
|
||||||
Coord(const wstring &iStr);
|
Coord(const wstring &iStr);
|
||||||
virtual ~Coord() {}
|
|
||||||
|
|
||||||
// Accessors
|
// Accessors
|
||||||
void setRow(int iRow) { m_row = iRow; }
|
void setRow(int iRow) { m_row = iRow; }
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -22,26 +24,25 @@
|
||||||
|
|
||||||
#define CROSS_MASK 0xFFFFFFFF
|
#define CROSS_MASK 0xFFFFFFFF
|
||||||
|
|
||||||
|
|
||||||
Cross::Cross()
|
Cross::Cross()
|
||||||
{
|
{
|
||||||
// The default behaviour is to match everything
|
// The default behaviour is to match everything
|
||||||
setAny();
|
setAny();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Cross::setAny()
|
|
||||||
{
|
void Cross::setAny()
|
||||||
|
{
|
||||||
m_mask = CROSS_MASK;
|
m_mask = CROSS_MASK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool Cross::isAny() const
|
bool Cross::isAny() const
|
||||||
{
|
{
|
||||||
return m_mask == CROSS_MASK;
|
return m_mask == CROSS_MASK;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Cross::setNone()
|
|
||||||
{
|
|
||||||
m_mask = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
string Cross::getHexContent() const
|
string Cross::getHexContent() const
|
||||||
{
|
{
|
||||||
|
@ -51,22 +52,21 @@ string Cross::getHexContent() const
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool Cross::check(const Tile& iTile) const
|
bool Cross::check(const Tile& iTile) const
|
||||||
{
|
{
|
||||||
return (iTile.isJoker() && m_mask != 0) || (m_mask & (1 << iTile.toCode()));
|
return (iTile.isJoker() && m_mask != 0) || (m_mask & (1 << iTile.toCode()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Cross::insert(const Tile& iTile)
|
void Cross::insert(const Tile& iTile)
|
||||||
{
|
{
|
||||||
m_mask |= (1 << iTile.toCode());
|
m_mask |= (1 << iTile.toCode());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool Cross::operator==(const Cross &iOther) const
|
bool Cross::operator==(const Cross &iOther) const
|
||||||
{
|
{
|
||||||
/*
|
|
||||||
* if (isAny() || iOther.isAny())
|
|
||||||
* return isAny() && iOther.isAny();
|
|
||||||
*/
|
|
||||||
return m_mask == iOther.m_mask;
|
return m_mask == iOther.m_mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
13
game/cross.h
13
game/cross.h
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -34,13 +36,12 @@ class Cross
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Cross();
|
Cross();
|
||||||
virtual ~Cross() {}
|
|
||||||
|
|
||||||
void setAny();
|
void setAny();
|
||||||
void setNone();
|
void setNone() { m_mask = 0; }
|
||||||
|
|
||||||
bool isAny() const;
|
bool isAny() const;
|
||||||
bool isNone() const;
|
bool isNone() const { return m_mask == 0; }
|
||||||
|
|
||||||
bool check(const Tile& iTile) const;
|
bool check(const Tile& iTile) const;
|
||||||
|
|
||||||
|
|
10
game/debug.h
10
game/debug.h
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2004-2007 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,8 +19,8 @@
|
||||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
#ifndef _CONST_H_
|
#ifndef _DEBUG_H_
|
||||||
#define _CONST_H_
|
#define _DEBUG_H_
|
||||||
|
|
||||||
/**********
|
/**********
|
||||||
* General
|
* General
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,78 +18,74 @@
|
||||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#include "duplicate.h"
|
||||||
#include "dic.h"
|
#include "dic.h"
|
||||||
#include "tile.h"
|
#include "tile.h"
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
|
#include "move.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "results.h"
|
#include "results.h"
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
#include "ai_player.h"
|
#include "ai_player.h"
|
||||||
#include "duplicate.h"
|
#include "settings.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
Duplicate::Duplicate(const Dictionary &iDic): Game(iDic)
|
Duplicate::Duplicate(const Dictionary &iDic)
|
||||||
|
: Game(iDic)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Duplicate::~Duplicate()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int Duplicate::setRackRandom(int p, bool iCheck, set_rack_mode mode)
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
res = helperSetRackRandom(p, iCheck, mode);
|
|
||||||
} while (res == 2);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int Duplicate::play(const wstring &iCoord, const wstring &iWord)
|
int Duplicate::play(const wstring &iCoord, const wstring &iWord)
|
||||||
{
|
{
|
||||||
/* Perform all the validity checks, and fill a round */
|
// Perform all the validity checks, and try to fill a round
|
||||||
Round round;
|
Round round;
|
||||||
int res = checkPlayedWord(iCoord, iWord, round);
|
int res = checkPlayedWord(iCoord, iWord, round);
|
||||||
if (res != 0)
|
if (res != 0 && Settings::Instance().getBool("duplicate-reject-invalid"))
|
||||||
{
|
{
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Everything is OK, we can play the word */
|
// If we reach this point, either the move is valid and we can use the
|
||||||
playRound(round, m_currPlayer);
|
// "round" variable, or it is invalid but played nevertheless
|
||||||
|
if (res == 0)
|
||||||
|
{
|
||||||
|
// Everything is OK, we can play the word
|
||||||
|
playMove(Move(round), m_currPlayer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Record the invalid move of the player
|
||||||
|
playMove(Move(iWord, iCoord), m_currPlayer);
|
||||||
|
}
|
||||||
|
|
||||||
/* Next turn */
|
// Little hack to handle duplicate games with only AI players.
|
||||||
// XXX: Should it be done by the interface instead?
|
// This will have no effect when there is at least one human player
|
||||||
endTurn();
|
tryEndTurn();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Duplicate::duplicateAI(int n)
|
void Duplicate::playAI(unsigned int p)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= n && n < getNPlayers(), "Wrong player number");
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
ASSERT(!m_players[n]->isHuman(), "AI requested for a human player");
|
|
||||||
|
|
||||||
AIPlayer *player = static_cast<AIPlayer*>(m_players[n]);
|
AIPlayer *player = dynamic_cast<AIPlayer*>(m_players[p]);
|
||||||
player->compute(*m_dic, m_board, m_history.getSize());
|
ASSERT(player != NULL, "AI requested for a human player");
|
||||||
|
|
||||||
if (player->changesLetters())
|
player->compute(m_dic, m_board, m_history.beforeFirstRound());
|
||||||
|
const Move move = player->getMove();
|
||||||
|
if (move.getType() == Move::CHANGE_LETTERS ||
|
||||||
|
move.getType() == Move::PASS)
|
||||||
{
|
{
|
||||||
// The AI player has nothing to play. This should not happen in
|
// The AI player must be buggy...
|
||||||
// duplicate mode, otherwise the implementation of the AI is buggy...
|
ASSERT(false, "AI tried to cheat!");
|
||||||
ASSERT(false, "AI player has nothing to play!");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
playRound(player->getChosenRound(), n);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
playMove(move, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -96,165 +93,155 @@ int Duplicate::start()
|
||||||
{
|
{
|
||||||
ASSERT(getNPlayers(), "Cannot start a game without any player");
|
ASSERT(getNPlayers(), "Cannot start a game without any player");
|
||||||
|
|
||||||
|
// Arbitrary player, since they should all have the same rack
|
||||||
m_currPlayer = 0;
|
m_currPlayer = 0;
|
||||||
|
|
||||||
/* XXX: code similar with endTurnForReal() */
|
// Complete the rack for the player that just played
|
||||||
/* Complete the rack for the player that just played */
|
int res = helperSetRackRandom(m_currPlayer, true, RACK_NEW);
|
||||||
int res = setRackRandom(m_currPlayer, true, RACK_NEW);
|
// End of the game?
|
||||||
/* End of the game? */
|
|
||||||
if (res == 1)
|
if (res == 1)
|
||||||
{
|
{
|
||||||
end();
|
endGame();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
const PlayedRack& pld = m_players[m_currPlayer]->getCurrentRack();
|
const PlayedRack& pld = m_players[m_currPlayer]->getCurrentRack();
|
||||||
/* All the players have the same rack */
|
// All the players have the same rack
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
if (i != m_currPlayer)
|
if (i != m_currPlayer)
|
||||||
{
|
{
|
||||||
m_players[i]->setCurrentRack(pld);
|
m_players[i]->setCurrentRack(pld);
|
||||||
}
|
}
|
||||||
/* Nobody has played yet in this round */
|
// Nobody has played yet in this round
|
||||||
m_hasPlayed[i] = false;
|
m_hasPlayed[i] = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Next turn */
|
// Little hack to handle duplicate games with only AI players.
|
||||||
// XXX: Should it be done by the interface instead?
|
// This will have no effect when there is at least one human player
|
||||||
endTurn();
|
tryEndTurn();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
void Duplicate::tryEndTurn()
|
||||||
* This function does not terminate the turn itself, but performs some
|
|
||||||
* checks to know whether or not it should be terminated (with a call to
|
|
||||||
* endTurnForReal()).
|
|
||||||
*
|
|
||||||
* For the turn to be terminated, all the players must have played.
|
|
||||||
* Since the AI players play after the human players, we check whether
|
|
||||||
* one of the human players has not played yet:
|
|
||||||
* - if so, we have nothing to do (we are waiting for him)
|
|
||||||
* - if not (all human players have played), the AI players can play,
|
|
||||||
* and we finish the turn.
|
|
||||||
*/
|
|
||||||
int Duplicate::endTurn()
|
|
||||||
{
|
{
|
||||||
int i;
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
for (i = 0; i < getNPlayers(); i++)
|
|
||||||
{
|
{
|
||||||
if (m_players[i]->isHuman() && !m_hasPlayed[i])
|
if (m_players[i]->isHuman() && !m_hasPlayed[i])
|
||||||
{
|
{
|
||||||
/* A human player has not played... */
|
// A human player has not played...
|
||||||
m_currPlayer = i;
|
m_currPlayer = i;
|
||||||
// XXX: check return code meaning
|
// So we don't finish the turn
|
||||||
return 1;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If all the human players have played */
|
// Now that all the human players have played,
|
||||||
if (i == getNPlayers())
|
// make AI players play their turn
|
||||||
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
/* Make AI players play their turn */
|
if (!m_players[i]->isHuman())
|
||||||
for (i = 0; i < getNPlayers(); i++)
|
|
||||||
{
|
{
|
||||||
if (!m_players[i]->isHuman())
|
playAI(i);
|
||||||
{
|
|
||||||
duplicateAI(i);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Next turn */
|
|
||||||
endTurnForReal();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// XXX: check return code meaning
|
// Next turn
|
||||||
return 0;
|
endTurn();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Duplicate::playRound(const Round &iRound, int n)
|
void Duplicate::playMove(const Move &iMove, unsigned int p)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= n && n < getNPlayers(), "Wrong player number");
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
Player *player = m_players[n];
|
|
||||||
|
|
||||||
/* Update the rack and the score of the current player */
|
// Update the rack and the score of the playing player
|
||||||
player->addPoints(iRound.getPoints());
|
m_players[p]->endTurn(iMove, m_history.getSize());
|
||||||
player->endTurn(iRound, m_history.getSize());
|
|
||||||
|
|
||||||
m_hasPlayed[n] = true;
|
m_hasPlayed[p] = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
void Duplicate::endTurn()
|
||||||
* This function really changes the turn, i.e. the best word is played and
|
|
||||||
* a new rack is given to the players.
|
|
||||||
* We suppose that all the players have finished to play for this turn (this
|
|
||||||
* should have been checked by endturn())
|
|
||||||
*/
|
|
||||||
int Duplicate::endTurnForReal()
|
|
||||||
{
|
{
|
||||||
int res, i, imax;
|
// Find the player with the best score
|
||||||
|
unsigned int imax = 0;
|
||||||
/* Play the best word on the board */
|
for (unsigned int i = 1; i < getNPlayers(); i++)
|
||||||
imax = 0;
|
|
||||||
for (i = 1; i < getNPlayers(); i++)
|
|
||||||
{
|
{
|
||||||
if (m_players[i]->getLastRound().getPoints() >
|
if (m_players[i]->getLastMove().getScore() >
|
||||||
m_players[imax]->getLastRound().getPoints())
|
m_players[imax]->getLastMove().getScore())
|
||||||
{
|
{
|
||||||
imax = i;
|
imax = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
m_currPlayer = imax;
|
|
||||||
helperPlayRound(m_players[imax]->getLastRound());
|
|
||||||
|
|
||||||
/* Complete the rack for the player that just played */
|
// TODO: do something if nobody played a valid round!
|
||||||
res = setRackRandom(imax, true, RACK_NEW);
|
|
||||||
/* End of the game? */
|
// Handle solo bonus
|
||||||
if (res == 1)
|
// First check whether there are enough players in the game for the
|
||||||
|
// bonus to apply
|
||||||
|
int minNbPlayers = Settings::Instance().getInt("duplicate-solo-players");
|
||||||
|
if (getNPlayers() >= (unsigned int)minNbPlayers &&
|
||||||
|
m_players[imax]->getLastMove().getType() == Move::VALID_ROUND)
|
||||||
{
|
{
|
||||||
end();
|
int maxScore = m_players[imax]->getLastMove().getScore();
|
||||||
return 1;
|
// Find whether other players than imax have the same score
|
||||||
|
bool otherWithSameScore = false;
|
||||||
|
for (unsigned int i = imax + 1; i < getNPlayers(); i++)
|
||||||
|
{
|
||||||
|
if (m_players[i]->getLastMove().getScore() >= maxScore)
|
||||||
|
{
|
||||||
|
otherWithSameScore = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!otherWithSameScore)
|
||||||
|
{
|
||||||
|
// Give the bonus to player imax
|
||||||
|
int bonus = Settings::Instance().getInt("duplicate-solo-value");
|
||||||
|
m_players[imax]->addPoints(bonus);
|
||||||
|
// TODO: keep a trace of the solo, so the interface
|
||||||
|
// can be aware of it...
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Play the best word on the board
|
||||||
|
helperPlayMove(imax, m_players[imax]->getLastMove());
|
||||||
|
|
||||||
|
// Leave the same reliquate to all players
|
||||||
|
// This is required by the start() method which will be called to
|
||||||
|
// start the next turn
|
||||||
const PlayedRack& pld = m_players[imax]->getCurrentRack();
|
const PlayedRack& pld = m_players[imax]->getCurrentRack();
|
||||||
/* All the players have the same rack */
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
for (i = 0; i < getNPlayers(); i++)
|
|
||||||
{
|
{
|
||||||
if (i != imax)
|
if (i != imax)
|
||||||
{
|
{
|
||||||
m_players[i]->setCurrentRack(pld);
|
m_players[i]->setCurrentRack(pld);
|
||||||
}
|
}
|
||||||
/* Nobody has played yet in this round */
|
|
||||||
m_hasPlayed[i] = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* XXX: Little hack to handle the games with only AI players.
|
// Start next turn...
|
||||||
* This will have no effect when there is at least one human player */
|
start();
|
||||||
endTurn();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Duplicate::end()
|
void Duplicate::endGame()
|
||||||
{
|
{
|
||||||
m_finished = true;
|
m_finished = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Duplicate::setPlayer(int n)
|
int Duplicate::setPlayer(unsigned int p)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= n && n < getNPlayers(), "Wrong player number");
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
|
|
||||||
/* Forbid switching to an AI player */
|
// Forbid switching to an AI player
|
||||||
if (!m_players[n]->isHuman())
|
if (!m_players[p]->isHuman())
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
m_currPlayer = n;
|
m_currPlayer = p;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -34,7 +35,7 @@ using std::wstring;
|
||||||
* and rack are updated. He cannot change his word afterwards.
|
* and rack are updated. He cannot change his word afterwards.
|
||||||
* - if there is still a human player who has not played for the current
|
* - if there is still a human player who has not played for the current
|
||||||
* turn, we wait for him
|
* turn, we wait for him
|
||||||
* - if all the human players have played, it's the turn to the AI players
|
* - if all the human players have played, it's the turn of the AI players
|
||||||
* (currently handled in a loop, but we could imagine that they are running
|
* (currently handled in a loop, but we could imagine that they are running
|
||||||
* in their own thread).
|
* in their own thread).
|
||||||
* - once all the players have played, we can really end the turn:
|
* - once all the players have played, we can really end the turn:
|
||||||
|
@ -57,26 +58,73 @@ public:
|
||||||
/*************************
|
/*************************
|
||||||
* Game handling
|
* Game handling
|
||||||
*************************/
|
*************************/
|
||||||
|
/**
|
||||||
|
* In Duplicate mode, the start() method starts a new turn, and is
|
||||||
|
* automatically called when the previous turn is finished.
|
||||||
|
*
|
||||||
|
* Pre-requisite: all the players must have the same rack when this
|
||||||
|
* method is called
|
||||||
|
*/
|
||||||
virtual int start();
|
virtual int start();
|
||||||
virtual int setRackRandom(int, bool, set_rack_mode);
|
|
||||||
virtual int play(const wstring &iCoord, const wstring &iWord);
|
|
||||||
virtual int endTurn();
|
|
||||||
|
|
||||||
int setPlayer(int);
|
/**
|
||||||
// Switch to the previous human player who has not played yet
|
* See description of Game::play() for the possible return values.
|
||||||
|
* Note that if the "duplicate-reject-invalid" setting is set to false
|
||||||
|
* the method always returns 0 (the player will have 0 for this turn)
|
||||||
|
*/
|
||||||
|
virtual int play(const wstring &iCoord, const wstring &iWord);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the current player, given its ID.
|
||||||
|
* The given player ID must correspond to a human player, which did not
|
||||||
|
* play yet for this turn.
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: the player is not human
|
||||||
|
*/
|
||||||
|
int setPlayer(unsigned int p);
|
||||||
|
|
||||||
|
/// Switch to the previous human player who has not played yet
|
||||||
void prevHumanPlayer();
|
void prevHumanPlayer();
|
||||||
// Switch to the next human player who has not played yet
|
|
||||||
|
/// Switch to the next human player who has not played yet
|
||||||
void nextHumanPlayer();
|
void nextHumanPlayer();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Private constructor and destructor to force using the GameFactory class
|
// Private constructor to force using the GameFactory class
|
||||||
Duplicate(const Dictionary &iDic);
|
Duplicate(const Dictionary &iDic);
|
||||||
virtual ~Duplicate();
|
|
||||||
|
|
||||||
void playRound(const Round &iRound, int n);
|
void playMove(const Move &iMove, unsigned int p);
|
||||||
int endTurnForReal();
|
|
||||||
void end();
|
/// Make the AI player whose ID is p play its turn
|
||||||
void duplicateAI(int n);
|
void playAI(unsigned int p);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This function does not terminate the turn itself, but performs some
|
||||||
|
* checks to know whether or not it should be terminated (with a call to
|
||||||
|
* endTurn()).
|
||||||
|
*
|
||||||
|
* For the turn to be terminated, all the players must have played.
|
||||||
|
* Since the AI players play after the human players, we check whether
|
||||||
|
* one of the human players has not played yet:
|
||||||
|
* - if so, we have nothing to do (we are waiting for him/her)
|
||||||
|
* - if not (all human players have played), the AI players can play,
|
||||||
|
* and we finish the turn.
|
||||||
|
*/
|
||||||
|
void tryEndTurn();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This function really changes the turn, i.e. the best word is played,
|
||||||
|
* the game history is updated, a "solo" bonus is given if needed, and
|
||||||
|
* all racks are made equal to the one of the player who played the
|
||||||
|
* best move.
|
||||||
|
* We suppose here that all the players have finished to play for this
|
||||||
|
* turn (this should have been checked by tryEndturn())
|
||||||
|
*/
|
||||||
|
void endTurn();
|
||||||
|
|
||||||
|
/// Finish the game
|
||||||
|
void endGame();
|
||||||
|
|
||||||
// m_hasPlayed[p] is true iff player p has played for this turn
|
// m_hasPlayed[p] is true iff player p has played for this turn
|
||||||
map<int, bool> m_hasPlayed;
|
map<int, bool> m_hasPlayed;
|
||||||
|
|
|
@ -1,98 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file encoding.cpp
|
|
||||||
* \brief Utility functions to ease manipulation of wide-character strings
|
|
||||||
* \author Olivier Teuliere
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <wchar.h>
|
|
||||||
#include <wctype.h>
|
|
||||||
#include "encoding.h"
|
|
||||||
|
|
||||||
|
|
||||||
int _wtoi(const wchar_t *iWStr)
|
|
||||||
{
|
|
||||||
return wcstol(iWStr,NULL,10);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int _swprintf(wchar_t *wcs, size_t maxlen, const wchar_t *format, ...)
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
va_list argp;
|
|
||||||
va_start(argp, format);
|
|
||||||
#ifdef WIN32
|
|
||||||
// Mingw32 does not take the maxlen argument
|
|
||||||
res = vswprintf(wcs, format, argp);
|
|
||||||
#else
|
|
||||||
res = vswprintf(wcs, maxlen, format, argp);
|
|
||||||
#endif
|
|
||||||
va_end(argp);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
wstring convertToWc(const string& iStr)
|
|
||||||
{
|
|
||||||
// Get the needed length (we _can't_ use string::size())
|
|
||||||
size_t len = mbstowcs(NULL, iStr.c_str(), 0);
|
|
||||||
if (len == (size_t)-1)
|
|
||||||
return L"";
|
|
||||||
|
|
||||||
wchar_t *tmp = new wchar_t[len + 1];
|
|
||||||
len = mbstowcs(tmp, iStr.c_str(), len + 1);
|
|
||||||
wstring res = tmp;
|
|
||||||
delete[] tmp;
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
string convertToMb(const wstring& iWStr)
|
|
||||||
{
|
|
||||||
// Get the needed length (we _can't_ use wstring::size())
|
|
||||||
size_t len = wcstombs(NULL, iWStr.c_str(), 0);
|
|
||||||
if (len == (size_t)-1)
|
|
||||||
return "";
|
|
||||||
|
|
||||||
char *tmp = new char[len + 1];
|
|
||||||
len = wcstombs(tmp, iWStr.c_str(), len + 1);
|
|
||||||
string res = tmp;
|
|
||||||
delete[] tmp;
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
string convertToMb(wchar_t iWChar)
|
|
||||||
{
|
|
||||||
char res[MB_CUR_MAX + 1];
|
|
||||||
int len = wctomb(res, iWChar);
|
|
||||||
if (len == -1)
|
|
||||||
return "";
|
|
||||||
res[len] = '\0';
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,52 +0,0 @@
|
||||||
/* Eliot */
|
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
|
||||||
/* */
|
|
||||||
/* This file is part of Eliot. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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. */
|
|
||||||
/* */
|
|
||||||
/* Eliot 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 */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* \file encoding.h
|
|
||||||
* \brief Utility functions to ease manipulation of wide-character strings
|
|
||||||
* \author Olivier Teuliere
|
|
||||||
* \date 2005
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _ENCODING_H_
|
|
||||||
#define _ENCODING_H_
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
using std::string;
|
|
||||||
using std::wstring;
|
|
||||||
|
|
||||||
|
|
||||||
/// Equivalent of atoi for wide-caracter strings
|
|
||||||
int _wtoi(const wchar_t *iWStr);
|
|
||||||
|
|
||||||
/// Equivalent of swprintf, but working also with mingw32
|
|
||||||
int _swprintf(wchar_t *wcs, size_t maxlen, const wchar_t *format, ...);
|
|
||||||
|
|
||||||
/// Convert a multi-byte string into a wide-character string
|
|
||||||
wstring convertToWc(const string& iStr);
|
|
||||||
|
|
||||||
/// Convert a wide-character string into a multi-byte string
|
|
||||||
string convertToMb(const wstring& iWStr);
|
|
||||||
|
|
||||||
/// Convert a wide character into a multi-byte string
|
|
||||||
string convertToMb(wchar_t iWChar);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -19,89 +20,88 @@
|
||||||
|
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
#include <wctype.h>
|
#include <wctype.h>
|
||||||
|
|
||||||
|
#include "freegame.h"
|
||||||
#include "dic.h"
|
#include "dic.h"
|
||||||
#include "tile.h"
|
#include "tile.h"
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
|
#include "move.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "results.h"
|
#include "results.h"
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
#include "ai_player.h"
|
#include "ai_player.h"
|
||||||
#include "freegame.h"
|
#include "settings.h"
|
||||||
|
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
FreeGame::FreeGame(const Dictionary &iDic): Game(iDic)
|
FreeGame::FreeGame(const Dictionary &iDic)
|
||||||
|
: Game(iDic)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
FreeGame::~FreeGame()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int FreeGame::setRackRandom(int p, bool iCheck, set_rack_mode mode)
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
res = helperSetRackRandom(p, iCheck, mode);
|
|
||||||
} while (res == 2);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int FreeGame::play(const wstring &iCoord, const wstring &iWord)
|
int FreeGame::play(const wstring &iCoord, const wstring &iWord)
|
||||||
{
|
{
|
||||||
/* Perform all the validity checks, and fill a round */
|
// Perform all the validity checks, and try to fill a round
|
||||||
Round round;
|
Round round;
|
||||||
|
|
||||||
int res = checkPlayedWord(iCoord, iWord, round);
|
int res = checkPlayedWord(iCoord, iWord, round);
|
||||||
if (res != 0)
|
if (res != 0 && Settings::Instance().getBool("freegame-reject-invalid"))
|
||||||
{
|
{
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update the rack and the score of the current player */
|
// If we reach this point, either the move is valid and we can use the
|
||||||
m_players[m_currPlayer]->addPoints(round.getPoints());
|
// "round" variable, or it is invalid but played nevertheless
|
||||||
m_players[m_currPlayer]->endTurn(round, m_history.getSize());
|
if (res == 0)
|
||||||
|
{
|
||||||
|
Move move(round);
|
||||||
|
|
||||||
/* Everything is OK, we can play the word */
|
// Update the rack and the score of the current player
|
||||||
helperPlayRound(round);
|
m_players[m_currPlayer]->endTurn(move, m_history.getSize());
|
||||||
|
|
||||||
/* Next turn */
|
// Everything is OK, we can play the word
|
||||||
// XXX: Should it be done by the interface instead?
|
helperPlayMove(m_currPlayer, move);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Move move(iWord, iCoord);
|
||||||
|
|
||||||
|
// Record the invalid move of the player
|
||||||
|
m_players[m_currPlayer]->endTurn(move, m_history.getSize());
|
||||||
|
|
||||||
|
// Update the game
|
||||||
|
helperPlayMove(m_currPlayer, move);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Next turn
|
||||||
endTurn();
|
endTurn();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void FreeGame::freegameAI(int n)
|
void FreeGame::playAI(unsigned int p)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= n && n < getNPlayers(), "Wrong player number");
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
ASSERT(!m_players[n]->isHuman(), "AI requested for a human player");
|
ASSERT(!m_players[p]->isHuman(), "AI requested for a human player");
|
||||||
|
|
||||||
AIPlayer *player = static_cast<AIPlayer*>(m_players[n]);
|
AIPlayer *player = static_cast<AIPlayer*>(m_players[p]);
|
||||||
|
|
||||||
player->compute(*m_dic, m_board, m_history.getSize());
|
player->compute(m_dic, m_board, m_history.beforeFirstRound());
|
||||||
if (player->changesLetters())
|
const Move move = player->getMove();
|
||||||
|
if (move.getType() == Move::CHANGE_LETTERS ||
|
||||||
|
move.getType() == Move::PASS)
|
||||||
{
|
{
|
||||||
helperPass(player->getChangedLetters(), n);
|
ASSERT(checkPass(move.getChangedLetters(), p) == 0, "AI tried to cheat!");
|
||||||
endTurn();
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
const Round &round = player->getChosenRound();
|
|
||||||
/* Update the rack and the score of the current player */
|
|
||||||
player->addPoints(round.getPoints());
|
|
||||||
player->endTurn(round, m_history.getSize());
|
|
||||||
|
|
||||||
helperPlayRound(round);
|
// Update the rack and the score of the current player
|
||||||
endTurn();
|
player->endTurn(move, m_history.getSize());
|
||||||
}
|
|
||||||
|
helperPlayMove(p, move);
|
||||||
|
endTurn();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -109,19 +109,18 @@ int FreeGame::start()
|
||||||
{
|
{
|
||||||
ASSERT(getNPlayers(), "Cannot start a game without any player");
|
ASSERT(getNPlayers(), "Cannot start a game without any player");
|
||||||
|
|
||||||
/* Set the initial racks of the players */
|
// Set the initial racks of the players
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
setRackRandom(i, false, RACK_NEW);
|
helperSetRackRandom(i, false, RACK_NEW);
|
||||||
}
|
}
|
||||||
|
|
||||||
// XXX
|
|
||||||
m_currPlayer = 0;
|
m_currPlayer = 0;
|
||||||
|
|
||||||
/* If the first player is an AI, make it play now */
|
// If the first player is an AI, make it play now
|
||||||
if (!m_players[0]->isHuman())
|
if (!m_players[m_currPlayer]->isHuman())
|
||||||
{
|
{
|
||||||
freegameAI(0);
|
playAI(m_currPlayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -130,21 +129,21 @@ int FreeGame::start()
|
||||||
|
|
||||||
int FreeGame::endTurn()
|
int FreeGame::endTurn()
|
||||||
{
|
{
|
||||||
/* Complete the rack for the player that just played */
|
// Complete the rack for the player that just played
|
||||||
if (setRackRandom(m_currPlayer, false, RACK_NEW) == 1)
|
if (helperSetRackRandom(m_currPlayer, false, RACK_NEW) == 1)
|
||||||
{
|
{
|
||||||
/* End of the game */
|
// End of the game
|
||||||
end();
|
endGame();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Next player */
|
// Next player
|
||||||
nextPlayer();
|
nextPlayer();
|
||||||
|
|
||||||
/* If this player is an AI, make it play now */
|
// If this player is an AI, make it play now
|
||||||
if (!m_players[m_currPlayer]->isHuman())
|
if (!m_players[m_currPlayer]->isHuman())
|
||||||
{
|
{
|
||||||
freegameAI(m_currPlayer);
|
playAI(m_currPlayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -152,7 +151,7 @@ int FreeGame::endTurn()
|
||||||
|
|
||||||
|
|
||||||
// Adjust the scores of the players with the points of the remaining tiles
|
// Adjust the scores of the players with the points of the remaining tiles
|
||||||
void FreeGame::end()
|
void FreeGame::endGame()
|
||||||
{
|
{
|
||||||
vector<Tile> tiles;
|
vector<Tile> tiles;
|
||||||
|
|
||||||
|
@ -165,12 +164,12 @@ void FreeGame::end()
|
||||||
// We currently handle case 1, and cannot handle case 3 until timers are
|
// We currently handle case 1, and cannot handle case 3 until timers are
|
||||||
// implemented.
|
// implemented.
|
||||||
// For case 2, we need both to detect a blocked situation (not easy...) and
|
// For case 2, we need both to detect a blocked situation (not easy...) and
|
||||||
// to handle it in the end() method (very easy).
|
// to handle it in the endGame() method (very easy).
|
||||||
|
|
||||||
/* Add the points of the remaining tiles to the score of the current
|
/* Add the points of the remaining tiles to the score of the current
|
||||||
* player (i.e. the first player with an empty rack), and remove them
|
* player (i.e. the first player with an empty rack), and remove them
|
||||||
* from the score of the players who still have tiles */
|
* from the score of the players who still have tiles */
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
if (i != m_currPlayer)
|
if (i != m_currPlayer)
|
||||||
{
|
{
|
||||||
|
@ -184,73 +183,73 @@ void FreeGame::end()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Lock game */
|
// Lock game
|
||||||
m_finished = true;
|
m_finished = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int FreeGame::pass(const wstring &iToChange, int n)
|
int FreeGame::checkPass(const wstring &iToChange, unsigned int p) const
|
||||||
{
|
{
|
||||||
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
|
|
||||||
|
// Check that the game is not finished
|
||||||
if (m_finished)
|
if (m_finished)
|
||||||
return 3;
|
return 3;
|
||||||
|
|
||||||
|
// Check that the letters are valid for the current dictionary
|
||||||
|
if (!m_dic.validateLetters(iToChange))
|
||||||
|
return 4;
|
||||||
|
|
||||||
|
// It is forbidden to change letters when the bag does not contain at
|
||||||
|
// least 7 letters (this is explicitly stated in the ODS). But it is
|
||||||
|
// still allowed to pass
|
||||||
|
Bag bag(m_dic);
|
||||||
|
realBag(bag);
|
||||||
|
if (bag.getNbTiles() < 7 && !iToChange.empty())
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that the letters are all present in the player's rack
|
||||||
|
Player *player = m_players[p];
|
||||||
|
PlayedRack pld = player->getCurrentRack();
|
||||||
|
Rack rack;
|
||||||
|
pld.getRack(rack);
|
||||||
|
for (unsigned int i = 0; i < iToChange.size(); i++)
|
||||||
|
{
|
||||||
|
// Remove the letter from the rack
|
||||||
|
if (!rack.in(Tile(iToChange[i])))
|
||||||
|
{
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
rack.remove(Tile(iToChange[i]));
|
||||||
|
}
|
||||||
|
|
||||||
// According to the rules in the ODS, it is allowed to pass its turn (no
|
// According to the rules in the ODS, it is allowed to pass its turn (no
|
||||||
// need to change letters for that).
|
// need to change letters for that).
|
||||||
// TODO: However, if all the players pass their turn, the first one has to
|
// TODO: However, if all the players pass their turn, the first one has to
|
||||||
// play, or change at least one letter. To implement this behaviour, we
|
// play, or change at least one letter. To implement this behaviour, we
|
||||||
// must also take care of blocked positions, where no one _can_ play (see
|
// must also take care of blocked positions, where no one _can_ play (see
|
||||||
// also comment in the end() method).
|
// also comment in the endGame() method).
|
||||||
|
|
||||||
// Convert the string into tiles
|
return 0;
|
||||||
vector<Tile> tilesVect;
|
|
||||||
for (unsigned int i = 0; i < iToChange.size(); i++)
|
|
||||||
{
|
|
||||||
Tile tile(towupper(iToChange[i]));
|
|
||||||
tilesVect.push_back(tile);
|
|
||||||
}
|
|
||||||
|
|
||||||
int res = helperPass(tilesVect, n);
|
|
||||||
if (res == 0)
|
|
||||||
endTurn();
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int FreeGame::helperPass(const vector<Tile> &iToChange, int n)
|
int FreeGame::pass(const wstring &iToChange)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= n && n < getNPlayers(), "Wrong player number");
|
int res = checkPass(iToChange, m_currPlayer);
|
||||||
|
if (res != 0)
|
||||||
|
return res;
|
||||||
|
|
||||||
// It is forbidden to change letters when the bag does not contain at
|
Move move(iToChange);
|
||||||
// least 7 letters (this is explicitely stated in the ODS).
|
// End the player's turn
|
||||||
Bag bag;
|
m_players[m_currPlayer]->endTurn(move, m_history.getSize());
|
||||||
realBag(bag);
|
// Update the game
|
||||||
if (bag.nTiles() < 7)
|
helperPlayMove(m_currPlayer, move);
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Player *player = m_players[n];
|
// Next game turn
|
||||||
PlayedRack pld = player->getCurrentRack();
|
endTurn();
|
||||||
Rack rack;
|
|
||||||
pld.getRack(rack);
|
|
||||||
|
|
||||||
for (unsigned int i = 0; i < iToChange.size(); i++)
|
|
||||||
{
|
|
||||||
/* Remove the letter from the rack */
|
|
||||||
if (!rack.in(iToChange[i]))
|
|
||||||
{
|
|
||||||
return 2;
|
|
||||||
}
|
|
||||||
rack.remove(iToChange[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
pld.reset();
|
|
||||||
pld.setOld(rack);
|
|
||||||
|
|
||||||
player->setCurrentRack(pld);
|
|
||||||
|
|
||||||
// FIXME: the letters to change should not be in the bag while generating
|
|
||||||
// the new rack!
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -48,20 +49,54 @@ public:
|
||||||
/*************************
|
/*************************
|
||||||
* Game handling
|
* Game handling
|
||||||
*************************/
|
*************************/
|
||||||
|
/**
|
||||||
|
* Start the game.
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
*/
|
||||||
virtual int start();
|
virtual int start();
|
||||||
virtual int setRackRandom(int, bool, set_rack_mode);
|
|
||||||
|
/**
|
||||||
|
* See description of Game::play() for the possible return values.
|
||||||
|
* Note that if the "freegame-reject-invalid" setting is set to false
|
||||||
|
* the method always returns 0 (the player will have 0 for this turn)
|
||||||
|
*/
|
||||||
virtual int play(const wstring &iCoord, const wstring &iWord);
|
virtual int play(const wstring &iCoord, const wstring &iWord);
|
||||||
virtual int endTurn();
|
|
||||||
int pass(const wstring &iToChange, int n);
|
/**
|
||||||
|
* Pass the turn, changing the letters listed in iToChange.
|
||||||
|
* If you simply want to pass the turn without changing any letter,
|
||||||
|
* provide an empty string.
|
||||||
|
*
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: changing letters is not allowed if there are less than 7 tiles
|
||||||
|
* left in the bag
|
||||||
|
* 2: the rack of the current player does not contain all the
|
||||||
|
* listed letters
|
||||||
|
* 3: the game is already finished
|
||||||
|
* 4: some letters are invalid for the current dictionary
|
||||||
|
*/
|
||||||
|
int pass(const wstring &iToChange);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Private constructor and destructor to force using the GameFactory class
|
// Private constructor to force using the GameFactory class
|
||||||
FreeGame(const Dictionary &iDic);
|
FreeGame(const Dictionary &iDic);
|
||||||
virtual ~FreeGame();
|
|
||||||
|
|
||||||
void freegameAI(int n);
|
/// Make the AI player whose ID is p play its turn
|
||||||
void end();
|
void playAI(unsigned int p);
|
||||||
int helperPass(const vector<Tile> &iToChange, int n);
|
|
||||||
|
/// Finish the current turn
|
||||||
|
int endTurn();
|
||||||
|
|
||||||
|
/// Finish the game
|
||||||
|
void endGame();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether it is legal to change the letters of iToChange.
|
||||||
|
* The return codes are the same as the ones on the pass() method
|
||||||
|
*/
|
||||||
|
int checkPass(const wstring &iToChange, unsigned int p) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* _FREEGAME_H_ */
|
#endif /* _FREEGAME_H_ */
|
||||||
|
|
493
game/game.cpp
493
game/game.cpp
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -19,7 +20,6 @@
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
#include "dic.h"
|
#include "dic.h"
|
||||||
#include "dic_search.h"
|
|
||||||
#include "tile.h"
|
#include "tile.h"
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
|
@ -35,65 +35,70 @@
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
const int Game::RACK_SIZE = 7;
|
const unsigned int Game::RACK_SIZE = 7;
|
||||||
const int Game::BONUS_POINTS = 50;
|
const int Game::BONUS_POINTS = 50;
|
||||||
|
|
||||||
Game::Game(const Dictionary &iDic):
|
Game::Game(const Dictionary &iDic):
|
||||||
m_dic(&iDic)
|
m_dic(iDic), m_bag(iDic)
|
||||||
{
|
{
|
||||||
m_variant = kNONE;
|
m_variant = kNONE;
|
||||||
m_points = 0;
|
m_points = 0;
|
||||||
m_currPlayer = -1;
|
m_currPlayer = 0;
|
||||||
m_finished = false;
|
m_finished = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Game::~Game()
|
Game::~Game()
|
||||||
{
|
{
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
delete m_players[i];
|
delete m_players[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const Player& Game::getPlayer(int iNum) const
|
const Player& Game::getPlayer(unsigned int iNum) const
|
||||||
{
|
{
|
||||||
ASSERT(0 <= iNum && iNum < (int)m_players.size(), "Wrong player number");
|
ASSERT(iNum < m_players.size(), "Wrong player number");
|
||||||
return *(m_players[iNum]);
|
return *(m_players[iNum]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* This function plays a round on the board */
|
void Game::helperPlayMove(unsigned int iPlayerId, const Move &iMove)
|
||||||
int Game::helperPlayRound(const Round &iRound)
|
|
||||||
{
|
{
|
||||||
/*
|
|
||||||
* We remove tiles from the bag only when they are played
|
|
||||||
* on the board. When going back in the game, we must only
|
|
||||||
* replace played tiles.
|
|
||||||
* We test a rack when it is set but tiles are left in the bag.
|
|
||||||
*/
|
|
||||||
|
|
||||||
// History of the game
|
// History of the game
|
||||||
m_history.setCurrentRack(getCurrentPlayer().getLastRack());
|
m_history.setCurrentRack(getPlayer(iPlayerId).getLastRack());
|
||||||
m_history.playRound(m_currPlayer, m_history.getSize(), iRound);
|
m_history.playMove(iPlayerId, m_history.getSize(), iMove);
|
||||||
|
|
||||||
debug(" helper: %d points\n",iRound.getPoints());
|
// Points
|
||||||
m_points += iRound.getPoints();
|
debug(" helper: %d points\n", iMove.getScore());
|
||||||
|
m_points += iMove.getScore();
|
||||||
|
|
||||||
|
// For moves corresponding to a valid round, we have much more
|
||||||
|
// work to do...
|
||||||
|
if (iMove.getType() == Move::VALID_ROUND)
|
||||||
|
{
|
||||||
|
helperPlayRound(iPlayerId, iMove.getRound());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void Game::helperPlayRound(unsigned int iPlayerId, const Round &iRound)
|
||||||
|
{
|
||||||
// Before updating the bag and the board, if we are playing a "joker game",
|
// Before updating the bag and the board, if we are playing a "joker game",
|
||||||
// we replace in the round the joker by the letter it represents
|
// we replace in the round the joker by the letter it represents
|
||||||
// This is currently done by a succession of ugly hacks :-/
|
// This is currently done by a succession of ugly hacks :-/
|
||||||
if (m_variant == kJOKER)
|
if (m_variant == kJOKER)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (iRound.isPlayedFromRack(i) && iRound.isJoker(i))
|
if (iRound.isPlayedFromRack(i) && iRound.isJoker(i))
|
||||||
{
|
{
|
||||||
// Is the represented letter still available in the bag?
|
// Is the represented letter still available in the bag?
|
||||||
// FIXME: this way to get the represented letter sucks...
|
// XXX: this way to get the represented letter sucks...
|
||||||
Tile t(towupper(iRound.getTile(i).toChar()));
|
Tile t(towupper(iRound.getTile(i).toChar()));
|
||||||
Bag bag;
|
Bag bag(m_dic);
|
||||||
realBag(bag);
|
realBag(bag);
|
||||||
// FIXME: realBag() does not give us a real bag in this
|
// FIXME: realBag() does not give us a real bag in this
|
||||||
// particular case! This is because Player::endTurn() is called
|
// particular case! This is because Player::endTurn() is called
|
||||||
|
@ -108,12 +113,12 @@ int Game::helperPlayRound(const Round &iRound)
|
||||||
// There is a big design problem here, but i am unsure what is
|
// There is a big design problem here, but i am unsure what is
|
||||||
// the best way to fix it.
|
// the best way to fix it.
|
||||||
vector<Tile> tiles;
|
vector<Tile> tiles;
|
||||||
getPlayer(m_currPlayer).getCurrentRack().getAllTiles(tiles);
|
getPlayer(iPlayerId).getCurrentRack().getAllTiles(tiles);
|
||||||
for (unsigned int j = 0; j < tiles.size(); j++)
|
for (unsigned int j = 0; j < tiles.size(); j++)
|
||||||
{
|
{
|
||||||
bag.replaceTile(tiles[j]);
|
bag.replaceTile(tiles[j]);
|
||||||
}
|
}
|
||||||
getPlayer(m_currPlayer).getLastRack().getAllTiles(tiles);
|
getPlayer(iPlayerId).getLastRack().getAllTiles(tiles);
|
||||||
for (unsigned int j = 0; j < tiles.size(); j++)
|
for (unsigned int j = 0; j < tiles.size(); j++)
|
||||||
{
|
{
|
||||||
bag.takeTile(tiles[j]);
|
bag.takeTile(tiles[j]);
|
||||||
|
@ -128,12 +133,19 @@ int Game::helperPlayRound(const Round &iRound)
|
||||||
// rounds
|
// rounds
|
||||||
const_cast<Round&>(iRound).setJoker(i, false);
|
const_cast<Round&>(iRound).setJoker(i, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// In a joker game we should have only 1 joker in the rack
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the bag and the board
|
// Update the bag
|
||||||
for (int i = 0; i < iRound.getWordLen(); i++)
|
// We remove tiles from the bag only when they are played
|
||||||
|
// on the board. When going back in the game, we must only
|
||||||
|
// replace played tiles.
|
||||||
|
// We test a rack when it is set but tiles are left in the bag.
|
||||||
|
for (unsigned int i = 0; i < iRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (iRound.isPlayedFromRack(i))
|
if (iRound.isPlayedFromRack(i))
|
||||||
{
|
{
|
||||||
|
@ -147,65 +159,53 @@ int Game::helperPlayRound(const Round &iRound)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
m_board.addRound(*m_dic, iRound);
|
|
||||||
|
|
||||||
return 0;
|
// Update the board
|
||||||
|
m_board.addRound(m_dic, iRound);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Game::back(int n)
|
int Game::back(unsigned int n)
|
||||||
{
|
{
|
||||||
int i, j;
|
|
||||||
Player *player;
|
|
||||||
|
|
||||||
if (n < 0)
|
|
||||||
{
|
|
||||||
debug("Game::back negative argument\n");
|
|
||||||
n = -n;
|
|
||||||
}
|
|
||||||
debug("Game::back %d\n",n);
|
debug("Game::back %d\n",n);
|
||||||
for (i = 0; i < n; i++)
|
// TODO: throw an exception
|
||||||
|
if (m_history.getSize() < n)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
for (unsigned int i = 0; i < n; i++)
|
||||||
{
|
{
|
||||||
if (m_history.getSize() > 0)
|
prevPlayer();
|
||||||
|
const Move &lastMove = m_history.getPreviousTurn().getMove();
|
||||||
|
// Nothing to cancel if the move was not a valid round
|
||||||
|
if (lastMove.getType() != Move::VALID_ROUND)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const Round &lastround = lastMove.getRound();
|
||||||
|
debug("Game::back last round %s\n",
|
||||||
|
convertToMb(lastround.toString()).c_str());
|
||||||
|
/* Remove the word from the board, and put its letters back
|
||||||
|
* into the bag */
|
||||||
|
m_board.removeRound(m_dic, lastround);
|
||||||
|
for (unsigned int j = 0; j < lastround.getWordLen(); j++)
|
||||||
{
|
{
|
||||||
prevPlayer();
|
if (lastround.isPlayedFromRack(j))
|
||||||
player = m_players[m_currPlayer];
|
|
||||||
const Round &lastround = m_history.getPreviousTurn().getRound();
|
|
||||||
debug("Game::back last round %s\n",
|
|
||||||
convertToMb(lastround.toString()).c_str());
|
|
||||||
/* Remove the word from the board, and put its letters back
|
|
||||||
* into the bag */
|
|
||||||
m_board.removeRound(*m_dic, lastround);
|
|
||||||
for (j = 0; j < lastround.getWordLen(); j++)
|
|
||||||
{
|
{
|
||||||
if (lastround.isPlayedFromRack(j))
|
if (lastround.isJoker(j))
|
||||||
{
|
m_bag.replaceTile(Tile::Joker());
|
||||||
if (lastround.isJoker(j))
|
else
|
||||||
m_bag.replaceTile(Tile::Joker());
|
m_bag.replaceTile(lastround.getTile(j));
|
||||||
else
|
|
||||||
m_bag.replaceTile(lastround.getTile(j));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
/* Remove the points of this round */
|
|
||||||
player->addPoints(- lastround.getPoints());
|
|
||||||
m_points -= lastround.getPoints();
|
|
||||||
/* Remove the turns */
|
|
||||||
player->removeLastTurn();
|
|
||||||
m_history.removeLastTurn();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
/* Remove the points of this round */
|
||||||
|
m_points -= lastround.getPoints();
|
||||||
|
/* Remove the turns */
|
||||||
|
m_players[m_currPlayer]->removeLastTurn();
|
||||||
|
m_history.removeLastTurn();
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* The realBag is the current bag minus all the racks
|
|
||||||
* present in the game. It represents the actual
|
|
||||||
* letters that are left in the bag.
|
|
||||||
*/
|
|
||||||
void Game::realBag(Bag &ioBag) const
|
void Game::realBag(Bag &ioBag) const
|
||||||
{
|
{
|
||||||
vector<Tile> tiles;
|
vector<Tile> tiles;
|
||||||
|
@ -217,7 +217,7 @@ void Game::realBag(Bag &ioBag) const
|
||||||
if (getMode() == kFREEGAME)
|
if (getMode() == kFREEGAME)
|
||||||
{
|
{
|
||||||
/* In freegame mode, take the letters from all the racks */
|
/* In freegame mode, take the letters from all the racks */
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
getPlayer(i).getCurrentRack().getAllTiles(tiles);
|
getPlayer(i).getCurrentRack().getAllTiles(tiles);
|
||||||
for (unsigned int j = 0; j < tiles.size(); j++)
|
for (unsigned int j = 0; j < tiles.size(); j++)
|
||||||
|
@ -239,22 +239,36 @@ void Game::realBag(Bag &ioBag) const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
int Game::helperSetRackRandom(unsigned int p, bool iCheck, set_rack_mode mode)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= p && p < getNPlayers(), "Wrong player number");
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
|
// FIXME: RACK_MANUAL shouldn't be in the enum
|
||||||
|
ASSERT(mode != RACK_MANUAL, "Invalid rack mode");
|
||||||
|
|
||||||
int nold, min;
|
// When iCheck is true, we must make sure that there are at least 2 vowels
|
||||||
|
// and 2 consonants in the rack up to the 15th turn, and at least one of
|
||||||
|
// each starting from the 16th turn.
|
||||||
|
// So before trying to fill the rack, we'd better make sure there is a way
|
||||||
|
// to complete the rack with these constraints...
|
||||||
|
unsigned int min = 0;
|
||||||
|
if (iCheck)
|
||||||
|
{
|
||||||
|
// 2 vowels and 2 consonants are needed up to the 15th turn
|
||||||
|
if (m_history.getSize() < 15)
|
||||||
|
min = 2;
|
||||||
|
else
|
||||||
|
min = 1;
|
||||||
|
}
|
||||||
|
|
||||||
// Make a copy of the current player's rack
|
// Make a copy of the current player's rack
|
||||||
PlayedRack pld = getPlayer(p).getCurrentRack();
|
PlayedRack pld = getPlayer(p).getCurrentRack();
|
||||||
nold = pld.nOld();
|
int nold = pld.getNbOld();
|
||||||
|
|
||||||
// Create a copy of the bag in which we can do everything we want,
|
// Create a copy of the bag in which we can do everything we want,
|
||||||
// and take from it the tiles of the players rack so that "bag"
|
// and take from it the tiles of the players rack so that "bag"
|
||||||
// contains the right number of tiles.
|
// contains the right number of tiles.
|
||||||
Bag bag;
|
Bag bag(m_dic);
|
||||||
realBag(bag);
|
realBag(bag);
|
||||||
|
|
||||||
if (mode == RACK_NEW && nold != 0)
|
if (mode == RACK_NEW && nold != 0)
|
||||||
{
|
{
|
||||||
// We may have removed too many letters from the bag (i.e. the 'new'
|
// We may have removed too many letters from the bag (i.e. the 'new'
|
||||||
|
@ -286,8 +300,212 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
debug("Game::helperSetRackRandom not a random mode\n");
|
debug("Game::helperSetRackRandom not a random mode\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Get the tiles remaining on the rack
|
||||||
|
vector<Tile> tiles;
|
||||||
|
pld.getOldTiles(tiles);
|
||||||
|
ASSERT(tiles.size() < RACK_SIZE,
|
||||||
|
"Cannot complete the rack, it is already complete");
|
||||||
|
|
||||||
|
bool jokerAdded = false;
|
||||||
|
// Are we dealing with a normal game or a joker game?
|
||||||
|
if (m_variant == kJOKER)
|
||||||
|
{
|
||||||
|
// 1) Is there already a joker in the remaining letters of the rack?
|
||||||
|
bool jokerFound = false;
|
||||||
|
for (unsigned int i = 0; i < tiles.size(); i++)
|
||||||
|
{
|
||||||
|
if (tiles[i].isJoker())
|
||||||
|
{
|
||||||
|
jokerFound = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2) If there was no joker, we add one if possible
|
||||||
|
if (!jokerFound && bag.in(Tile::Joker()))
|
||||||
|
{
|
||||||
|
jokerAdded = true;
|
||||||
|
pld.addNew(Tile::Joker());
|
||||||
|
tiles.push_back(Tile::Joker());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3) Remove all the jokers from the bag, to avoid taking another one
|
||||||
|
for (unsigned int i = 0; i < bag.in(Tile::Joker()); ++i)
|
||||||
|
{
|
||||||
|
bag.takeTile(Tile::Joker());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nothing in the rack, nothing in the bag --> end of the (free)game
|
||||||
|
if (bag.getNbTiles() == 0 && pld.getNbTiles() == 0)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
// End of game condition
|
||||||
|
if (iCheck)
|
||||||
|
{
|
||||||
|
if (bag.getNbVowels() == 0 || bag.getNbConsonants() == 0 ||
|
||||||
|
bag.getNbTiles() == 1)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle reject:
|
||||||
|
// Now that the joker has been dealt with, we try to complete the rack
|
||||||
|
// with truly random tiles. If it meets the requirements (i.e. if there
|
||||||
|
// are at least "min" vowels and "min" consonants in the rack), fine.
|
||||||
|
// Otherwise, we reject the rack completely, and we try again
|
||||||
|
// to complete it, but this time we ensure by construction that the
|
||||||
|
// requirements will be met.
|
||||||
|
while (bag.getNbTiles() != 0 && pld.getNbTiles() < RACK_SIZE)
|
||||||
|
{
|
||||||
|
Tile l = bag.selectRandom();
|
||||||
|
bag.takeTile(l);
|
||||||
|
pld.addNew(l);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pld.checkRack(min, min))
|
||||||
|
{
|
||||||
|
// Bad luck... we have to reject the rack
|
||||||
|
vector<Tile> rejectedTiles;
|
||||||
|
pld.getAllTiles(rejectedTiles);
|
||||||
|
for (unsigned int i = 0; i < rejectedTiles.size(); i++)
|
||||||
|
{
|
||||||
|
bag.replaceTile(rejectedTiles[i]);
|
||||||
|
}
|
||||||
|
pld.reset();
|
||||||
|
// Do not mark the rack as rejected if it was empty
|
||||||
|
if (nold > 0)
|
||||||
|
pld.setReject();
|
||||||
|
|
||||||
|
// Restore the joker if we are in a joker game
|
||||||
|
if (jokerAdded)
|
||||||
|
pld.addNew(Tile::Joker());
|
||||||
|
|
||||||
|
// Count the needed consonants and vowels in the rack
|
||||||
|
// (i.e. minimum required, minus what we already have in the rack)
|
||||||
|
unsigned int neededVowels = min;
|
||||||
|
unsigned int neededConsonants = min;
|
||||||
|
for (unsigned int i = 0; i < tiles.size(); ++i)
|
||||||
|
{
|
||||||
|
if (neededVowels > 0 && tiles[i].isVowel())
|
||||||
|
neededVowels--;
|
||||||
|
if (neededConsonants > 0 && tiles[i].isConsonant())
|
||||||
|
neededConsonants--;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether it is possible to complete the rack properly
|
||||||
|
if (bag.getNbVowels() < neededVowels ||
|
||||||
|
bag.getNbConsonants() < neededConsonants)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// RACK_SIZE - tiles.size() is the number of letters to add to the rack
|
||||||
|
if (neededVowels > RACK_SIZE - tiles.size() ||
|
||||||
|
neededConsonants > RACK_SIZE - tiles.size())
|
||||||
|
{
|
||||||
|
// We cannot fill the rack with enough vowels or consonants!
|
||||||
|
// Actually this should never happen, but it doesn't hurt to check...
|
||||||
|
// FIXME: this test is not completely right, because it supposes no
|
||||||
|
// letter can be at the same time a vowel and a consonant
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the required vowels and consonants first
|
||||||
|
for (unsigned int i = 0; i < neededVowels; ++i)
|
||||||
|
{
|
||||||
|
Tile l = bag.selectRandomVowel();
|
||||||
|
bag.takeTile(l);
|
||||||
|
pld.addNew(l);
|
||||||
|
// Handle the case where the vowel can also be considered
|
||||||
|
// as a consonant
|
||||||
|
if (l.isConsonant() && neededConsonants > 0)
|
||||||
|
neededConsonants--;
|
||||||
|
}
|
||||||
|
for (unsigned int i = 0; i < neededConsonants; ++i)
|
||||||
|
{
|
||||||
|
Tile l = bag.selectRandomConsonant();
|
||||||
|
bag.takeTile(l);
|
||||||
|
pld.addNew(l);
|
||||||
|
}
|
||||||
|
|
||||||
|
// The difficult part is done:
|
||||||
|
// - we have handled joker games
|
||||||
|
// - we have handled the checks
|
||||||
|
// Now complete the rack with truly random letters
|
||||||
|
while (bag.getNbTiles() != 0 && pld.getNbTiles() < RACK_SIZE)
|
||||||
|
{
|
||||||
|
Tile l = bag.selectRandom();
|
||||||
|
bag.takeTile(l);
|
||||||
|
pld.addNew(l);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shuffle the new tiles, to hide the order we imposed (joker first in a
|
||||||
|
// joker game, then needed vowels, then needed consonants, and rest of the
|
||||||
|
// rack)
|
||||||
|
pld.shuffleNew();
|
||||||
|
|
||||||
|
// Post-condition check. This should never fail, of course :)
|
||||||
|
ASSERT(pld.checkRack(min, min), "helperSetRackRandom() is buggy!")
|
||||||
|
|
||||||
|
// Until now we didn't modify anything except local variables.
|
||||||
|
// Let's "commit" the changes
|
||||||
|
m_players[p]->setCurrentRack(pld);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int Game::helperSetRackRandomOld(unsigned int p, bool iCheck, set_rack_mode mode)
|
||||||
|
{
|
||||||
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
|
|
||||||
|
// Make a copy of the current player's rack
|
||||||
|
PlayedRack pld = getPlayer(p).getCurrentRack();
|
||||||
|
int nold = pld.getNbOld();
|
||||||
|
|
||||||
|
// Create a copy of the bag in which we can do everything we want,
|
||||||
|
// and take from it the tiles of the players rack so that "bag"
|
||||||
|
// contains the right number of tiles.
|
||||||
|
Bag bag(m_dic);
|
||||||
|
realBag(bag);
|
||||||
|
|
||||||
|
if (mode == RACK_NEW && nold != 0)
|
||||||
|
{
|
||||||
|
// We may have removed too many letters from the bag (i.e. the 'new'
|
||||||
|
// letters of the player)
|
||||||
|
vector<Tile> tiles;
|
||||||
|
pld.getNewTiles(tiles);
|
||||||
|
for (unsigned int i = 0; i < tiles.size(); i++)
|
||||||
|
{
|
||||||
|
bag.replaceTile(tiles[i]);
|
||||||
|
}
|
||||||
|
pld.resetNew();
|
||||||
|
}
|
||||||
|
else if (mode == RACK_NEW && nold == 0 || mode == RACK_ALL)
|
||||||
|
{
|
||||||
|
// Replace all the tiles in the bag before choosing random ones
|
||||||
|
vector<Tile> tiles;
|
||||||
|
pld.getAllTiles(tiles);
|
||||||
|
for (unsigned int i = 0; i < tiles.size(); i++)
|
||||||
|
{
|
||||||
|
bag.replaceTile(tiles[i]);
|
||||||
|
}
|
||||||
|
// RACK_NEW with an empty rack is equivalent to RACK_ALL
|
||||||
|
pld.reset();
|
||||||
|
// Do not forget to update nold, for the RACK_ALL case
|
||||||
|
nold = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
debug("Game::helperSetRackRandomOld not a random mode\n");
|
||||||
|
}
|
||||||
|
|
||||||
// Nothing in the rack, nothing in the bag --> end of the game
|
// Nothing in the rack, nothing in the bag --> end of the game
|
||||||
if (bag.nTiles() == 0 && pld.nTiles() == 0)
|
if (bag.getNbTiles() == 0 && pld.getNbTiles() == 0)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -297,17 +515,17 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
// them from the 16th turn.
|
// them from the 16th turn.
|
||||||
// So before trying to fill the rack, we'd better make sure there is a way
|
// So before trying to fill the rack, we'd better make sure there is a way
|
||||||
// to complete the rack with these constraints...
|
// to complete the rack with these constraints...
|
||||||
min = 0;
|
unsigned int min = 0;
|
||||||
if (iCheck)
|
if (iCheck)
|
||||||
{
|
{
|
||||||
int oldc, oldv;
|
unsigned int oldc, oldv;
|
||||||
|
|
||||||
if (bag.nVowels() == 0 || bag.nConsonants() == 0)
|
if (bag.getNbVowels() == 0 || bag.getNbConsonants() == 0)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
// 2 vowels and 2 consonants are needed up to the 15th turn
|
// 2 vowels and 2 consonants are needed up to the 15th turn
|
||||||
if (bag.nVowels() > 1 && bag.nConsonants() > 1
|
if (bag.getNbVowels() > 1 && bag.getNbConsonants() > 1
|
||||||
&& m_history.getSize() < 15)
|
&& m_history.getSize() < 15)
|
||||||
min = 2;
|
min = 2;
|
||||||
else
|
else
|
||||||
|
@ -360,7 +578,9 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
|
|
||||||
// 3) Complete the rack normally... but without any joker!
|
// 3) Complete the rack normally... but without any joker!
|
||||||
Tile l;
|
Tile l;
|
||||||
while (bag.nTiles() != 0 && pld.nTiles() != RACK_SIZE)
|
// FIXME: this can be an infinite loop if the only tile left in the
|
||||||
|
// bag is a joker!
|
||||||
|
while (bag.getNbTiles() != 0 && pld.getNbTiles() != RACK_SIZE)
|
||||||
{
|
{
|
||||||
l = bag.selectRandom();
|
l = bag.selectRandom();
|
||||||
if (!l.isJoker())
|
if (!l.isJoker())
|
||||||
|
@ -374,7 +594,7 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
{
|
{
|
||||||
// Get new tiles from the bag
|
// Get new tiles from the bag
|
||||||
Tile l;
|
Tile l;
|
||||||
while (bag.nTiles() != 0 && pld.nTiles() != RACK_SIZE)
|
while (bag.getNbTiles() != 0 && pld.getNbTiles() != RACK_SIZE)
|
||||||
{
|
{
|
||||||
l = bag.selectRandom();
|
l = bag.selectRandom();
|
||||||
bag.takeTile(l);
|
bag.takeTile(l);
|
||||||
|
@ -382,7 +602,7 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (iCheck && !pld.checkRack(min,min))
|
if (iCheck && !pld.checkRack(min, min))
|
||||||
return 2;
|
return 2;
|
||||||
|
|
||||||
m_players[p]->setCurrentRack(pld);
|
m_players[p]->setCurrentRack(pld);
|
||||||
|
@ -391,20 +611,10 @@ int Game::helperSetRackRandom(int p, bool iCheck, set_rack_mode mode)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the players rack can be obtained from the bag.
|
|
||||||
* Since letters are removed from the bag only when the
|
|
||||||
* round is played we need to check that ALL the racks
|
|
||||||
* are in the bag simultaneously.
|
|
||||||
*
|
|
||||||
* FIXME: since we do not check for all racks it works
|
|
||||||
* for training and duplicate but it won't work for
|
|
||||||
* freegames.
|
|
||||||
*/
|
|
||||||
bool Game::rackInBag(const Rack &iRack, const Bag &iBag) const
|
bool Game::rackInBag(const Rack &iRack, const Bag &iBag) const
|
||||||
{
|
{
|
||||||
const list<Tile>& allTiles = Tile::getAllTiles();
|
const vector<Tile>& allTiles = m_dic.getAllTiles();
|
||||||
list<Tile>::const_iterator it;
|
vector<Tile>::const_iterator it;
|
||||||
for (it = allTiles.begin(); it != allTiles.end(); it++)
|
for (it = allTiles.begin(); it != allTiles.end(); it++)
|
||||||
{
|
{
|
||||||
if (iRack.in(*it) > iBag.in(*it))
|
if (iRack.in(*it) > iBag.in(*it))
|
||||||
|
@ -413,20 +623,16 @@ bool Game::rackInBag(const Rack &iRack, const Bag &iBag) const
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the rack of the player p manually.
|
int Game::helperSetRackManual(unsigned int p, bool iCheck, const wstring &iLetters)
|
||||||
*/
|
|
||||||
int Game::helperSetRackManual(int p, bool iCheck, const wstring &iLetters)
|
|
||||||
{
|
{
|
||||||
int min, ret;
|
ASSERT(p < getNPlayers(), "Wrong player number");
|
||||||
|
|
||||||
PlayedRack pld = getPlayer(p).getCurrentRack();
|
if (!m_dic.validateLetters(iLetters, L"+"))
|
||||||
pld.reset();
|
return 3;
|
||||||
|
|
||||||
if ((ret = pld.setManual(iLetters)) > 0)
|
PlayedRack pld;
|
||||||
{
|
pld.setManual(iLetters);
|
||||||
return 1; /* add new tests */
|
|
||||||
}
|
|
||||||
|
|
||||||
Rack rack;
|
Rack rack;
|
||||||
pld.getRack(rack);
|
pld.getRack(rack);
|
||||||
|
@ -438,12 +644,13 @@ int Game::helperSetRackManual(int p, bool iCheck, const wstring &iLetters)
|
||||||
|
|
||||||
if (iCheck)
|
if (iCheck)
|
||||||
{
|
{
|
||||||
if (m_bag.nVowels() > 1 && m_bag.nConsonants() > 1
|
int min;
|
||||||
|
if (m_bag.getNbVowels() > 1 && m_bag.getNbConsonants() > 1
|
||||||
&& m_history.getSize() < 15)
|
&& m_history.getSize() < 15)
|
||||||
min = 2;
|
min = 2;
|
||||||
else
|
else
|
||||||
min = 1;
|
min = 1;
|
||||||
if (!pld.checkRack(min,min))
|
if (!pld.checkRack(min, min))
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -456,10 +663,10 @@ int Game::helperSetRackManual(int p, bool iCheck, const wstring &iLetters)
|
||||||
*********************************************************/
|
*********************************************************/
|
||||||
|
|
||||||
|
|
||||||
int Game::getNHumanPlayers() const
|
unsigned int Game::getNHumanPlayers() const
|
||||||
{
|
{
|
||||||
int count = 0;
|
unsigned int count = 0;
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
count += (getPlayer(i).isHuman() ? 1 : 0);
|
count += (getPlayer(i).isHuman() ? 1 : 0);
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
@ -474,7 +681,8 @@ void Game::addHumanPlayer()
|
||||||
|
|
||||||
void Game::addAIPlayer()
|
void Game::addAIPlayer()
|
||||||
{
|
{
|
||||||
m_players.push_back(new AIPercent(getNPlayers(), 0));
|
// TODO: allow other percentages, and even other types of AI
|
||||||
|
m_players.push_back(new AIPercent(getNPlayers(), 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -500,34 +708,15 @@ void Game::nextPlayer()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This function checks whether it is legal to play the given word at the
|
|
||||||
* given coordinates. If so, the function fills a Round object, also given as
|
|
||||||
* a parameter.
|
|
||||||
* Possible return values:
|
|
||||||
* 0: correct word, the Round can be used by the caller
|
|
||||||
* 1: no dictionary set
|
|
||||||
* 2: invalid coordinates (unreadable or out of the board)
|
|
||||||
* 3: word not present in the dictionary
|
|
||||||
* 4: not enough letters in the rack to play the word
|
|
||||||
* 5: word is part of a longer one
|
|
||||||
* 6: word overwriting an existing letter
|
|
||||||
* 7: invalid crosscheck, or word going out of the board
|
|
||||||
* 8: word already present on the board (no new letter from the rack)
|
|
||||||
* 9: isolated word (not connected to the rest)
|
|
||||||
* 10: first word not horizontal
|
|
||||||
* 11: first word not covering the H8 square
|
|
||||||
*/
|
|
||||||
int Game::checkPlayedWord(const wstring &iCoord,
|
int Game::checkPlayedWord(const wstring &iCoord,
|
||||||
const wstring &iWord, Round &oRound)
|
const wstring &iWord, Round &oRound)
|
||||||
{
|
{
|
||||||
ASSERT(getNPlayers() != 0, "Expected at least one player");
|
ASSERT(getNPlayers() != 0, "Expected at least one player");
|
||||||
|
|
||||||
int res;
|
if (!m_dic.validateLetters(iWord))
|
||||||
vector<Tile> tiles;
|
return 1;
|
||||||
Tile t;
|
|
||||||
|
|
||||||
/* Init the round with the given coordinates */
|
// Init the round with the given coordinates
|
||||||
oRound.init();
|
oRound.init();
|
||||||
oRound.accessCoord().setFromString(iCoord);
|
oRound.accessCoord().setFromString(iCoord);
|
||||||
if (!oRound.getCoord().isValid())
|
if (!oRound.getCoord().isValid())
|
||||||
|
@ -535,17 +724,18 @@ int Game::checkPlayedWord(const wstring &iCoord,
|
||||||
debug("game: incorrect coordinates\n");
|
debug("game: incorrect coordinates\n");
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check the existence of the word */
|
// Check the existence of the word
|
||||||
if (Dic_search_word(*m_dic, iWord.c_str()) == 0)
|
if (!m_dic.searchWord(iWord))
|
||||||
{
|
{
|
||||||
return 3;
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set the word */
|
// Set the word
|
||||||
// TODO: make this a Round_ function (Round_setwordfromchar for example)
|
// TODO: make this a Round_ function (Round_setwordfromchar for example)
|
||||||
// or a Tiles_ function (to transform a char* into a vector<Tile>)
|
// or a Tiles_ function (to transform a char* into a vector<Tile>)
|
||||||
// Adding a getter on the word could help too...
|
// Adding a getter on the word could help too...
|
||||||
|
vector<Tile> tiles;
|
||||||
for (unsigned int i = 0; i < iWord.size(); i++)
|
for (unsigned int i = 0; i < iWord.size(); i++)
|
||||||
{
|
{
|
||||||
tiles.push_back(Tile(iWord[i]));
|
tiles.push_back(Tile(iWord[i]));
|
||||||
|
@ -557,20 +747,21 @@ int Game::checkPlayedWord(const wstring &iCoord,
|
||||||
oRound.setJoker(i);
|
oRound.setJoker(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check the word position, compute its points,
|
// Check the word position, compute its points,
|
||||||
* and specify the origin of each letter (board or rack) */
|
// and specify the origin of each letter (board or rack)
|
||||||
res = m_board.checkRound(oRound, m_history.getSize() == 0);
|
int res = m_board.checkRound(oRound, m_history.getSize() == 0);
|
||||||
if (res != 0)
|
if (res != 0)
|
||||||
return res + 4;
|
return res + 4;
|
||||||
|
|
||||||
/* Check that the word can be formed with the tiles in the rack:
|
// Check that the word can be formed with the tiles in the rack:
|
||||||
* we first create a copy of the rack, then we remove the tiles
|
// we first create a copy of the rack, then we remove the tiles
|
||||||
* one by one */
|
// one by one
|
||||||
Rack rack;
|
Rack rack;
|
||||||
Player *player = m_players[m_currPlayer];
|
Player *player = m_players[m_currPlayer];
|
||||||
player->getCurrentRack().getRack(rack);
|
player->getCurrentRack().getRack(rack);
|
||||||
|
|
||||||
for (int i = 0; i < oRound.getWordLen(); i++)
|
Tile t;
|
||||||
|
for (unsigned int i = 0; i < oRound.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (oRound.isPlayedFromRack(i))
|
if (oRound.isPlayedFromRack(i))
|
||||||
{
|
{
|
||||||
|
|
260
game/game.h
260
game/game.h
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -28,12 +29,12 @@
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "history.h"
|
#include "history.h"
|
||||||
|
|
||||||
|
class Dictionary;
|
||||||
class Player;
|
class Player;
|
||||||
class PlayedRack;
|
class PlayedRack;
|
||||||
class Round;
|
class Round;
|
||||||
class Rack;
|
class Rack;
|
||||||
class Turn;
|
class Turn;
|
||||||
typedef struct _Dictionary * Dictionary;
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
@ -47,13 +48,17 @@ class Game
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
/// Game specs.
|
/// Game specs.
|
||||||
static const int RACK_SIZE;
|
static const unsigned int RACK_SIZE;
|
||||||
static const int BONUS_POINTS;
|
static const int BONUS_POINTS;
|
||||||
|
|
||||||
|
|
||||||
Game(const Dictionary &iDic);
|
Game(const Dictionary &iDic);
|
||||||
virtual ~Game();
|
virtual ~Game();
|
||||||
|
|
||||||
|
/***************
|
||||||
|
* Game type
|
||||||
|
***************/
|
||||||
|
|
||||||
/// Game mode: each one of these modes is implemented in an inherited class
|
/// Game mode: each one of these modes is implemented in an inherited class
|
||||||
enum GameMode
|
enum GameMode
|
||||||
{
|
{
|
||||||
|
@ -79,100 +84,118 @@ public:
|
||||||
void setVariant(GameVariant iVariant) { m_variant = iVariant; }
|
void setVariant(GameVariant iVariant) { m_variant = iVariant; }
|
||||||
GameVariant getVariant() const { return m_variant; }
|
GameVariant getVariant() const { return m_variant; }
|
||||||
|
|
||||||
/**
|
/***************
|
||||||
* Dictionary associated with the game.
|
* Various getters
|
||||||
* The dictionary can be changed during a game without problem
|
***************/
|
||||||
*/
|
|
||||||
const Dictionary & getDic() const { return *m_dic; }
|
|
||||||
void setDic(const Dictionary &iDic) { m_dic = &iDic; }
|
|
||||||
|
|
||||||
const Board& getBoard() const { return m_board; }
|
/**
|
||||||
const Bag& getBag() const { return m_bag; }
|
* Get the dictionary associated with the game.
|
||||||
const Player& getPlayer(int iNum) const;
|
* You should never create a new dictionary object while a Game
|
||||||
const Turn& getTurn(int iNum) const;
|
* object still exists
|
||||||
|
*/
|
||||||
|
const Dictionary & getDic() const { return m_dic; }
|
||||||
|
|
||||||
|
/// Get the board
|
||||||
|
const Board& getBoard() const { return m_board; }
|
||||||
|
/// Get the bag
|
||||||
|
const Bag& getBag() const { return m_bag; }
|
||||||
|
/// Get the history of the game */
|
||||||
|
const History& getHistory() const { return m_history; }
|
||||||
|
|
||||||
|
/***************
|
||||||
|
* Methods to access players.
|
||||||
|
***************/
|
||||||
|
|
||||||
|
const Player& getPlayer(unsigned int iNum) const;
|
||||||
const Player& getCurrentPlayer() const { return getPlayer(currPlayer()); };
|
const Player& getCurrentPlayer() const { return getPlayer(currPlayer()); };
|
||||||
|
unsigned int getNPlayers() const { return m_players.size(); }
|
||||||
|
unsigned int getNHumanPlayers() const;
|
||||||
|
virtual void addHumanPlayer();
|
||||||
|
// TODO: Ability to specify which kind of AI player is wanted
|
||||||
|
virtual void addAIPlayer();
|
||||||
|
unsigned int currPlayer() const { return m_currPlayer; }
|
||||||
|
|
||||||
|
/***************
|
||||||
|
* Game handling
|
||||||
|
***************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Eliot file formats
|
* Start the game.
|
||||||
|
* AI players are handled automatically, so if the game only has AI
|
||||||
|
* players, it will play until the end.
|
||||||
*/
|
*/
|
||||||
typedef enum {
|
virtual int start() = 0;
|
||||||
FILE_FORMAT_STANDARD,
|
|
||||||
FILE_FORMAT_ADVANCED
|
/**
|
||||||
} game_file_format;
|
* Method used by human players to play the word iWord at coordinates
|
||||||
|
* iCoord, and end the turn (if possible)
|
||||||
|
* Possible return values:
|
||||||
|
* 0: correct word, the Round can be used by the caller
|
||||||
|
* 1: one letter of the word is invalid in the current dictionary
|
||||||
|
* 2: invalid coordinates (unreadable or out of the board)
|
||||||
|
* 3: word not present in the dictionary
|
||||||
|
* 4: not enough letters in the rack to play the word
|
||||||
|
* 5: word is part of a longer one
|
||||||
|
* 6: word overwriting an existing letter
|
||||||
|
* 7: invalid crosscheck, or word going out of the board
|
||||||
|
* 8: word already present on the board (no new letter from the rack)
|
||||||
|
* 9: isolated word (not connected to the rest)
|
||||||
|
* 10: first word not horizontal
|
||||||
|
* 11: first word not covering the H8 square
|
||||||
|
*/
|
||||||
|
virtual int play(const wstring &iCoord, const wstring &iWord) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Go back to turn iTurn.
|
||||||
|
* We must have: iTurn < getHistory().getSize()
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: iTurn is invalid
|
||||||
|
*/
|
||||||
|
int back(unsigned int iTurn);
|
||||||
|
|
||||||
|
/***************
|
||||||
|
* Saved games handling
|
||||||
|
***************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Possible formats for the saved games
|
||||||
|
*/
|
||||||
|
enum game_file_format
|
||||||
|
{
|
||||||
|
FILE_FORMAT_STANDARD,
|
||||||
|
FILE_FORMAT_ADVANCED
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saved games handling.
|
|
||||||
*
|
|
||||||
* load() returns the loaded game, or NULL if there was a problem
|
* load() returns the loaded game, or NULL if there was a problem
|
||||||
* load() might need some more work to be robust enough to
|
* load() does need some more work to be robust enough to
|
||||||
* handle "hand written" files
|
* handle "hand written" files
|
||||||
*/
|
*/
|
||||||
static Game * load(FILE *fin, const Dictionary &iDic);
|
static Game * load(FILE *fin, const Dictionary &iDic);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a game to a File
|
* Save a game to a file
|
||||||
* Standard format is used for training games so that it is compatible
|
* Standard format is used for training games so that it is compatible
|
||||||
* with previous versions of Eliot.
|
* with previous versions of Eliot.
|
||||||
*
|
*
|
||||||
* Saving can be forced to advanced format for training games by
|
* Saving can be forced to advanced format for training games by
|
||||||
* setting the last parameter to FILE_FORMAT_ADVANCED
|
* setting the last parameter to FILE_FORMAT_ADVANCED
|
||||||
*/
|
*/
|
||||||
void save(ostream &out, game_file_format format=FILE_FORMAT_STANDARD) const;
|
void save(ostream &out, game_file_format format = FILE_FORMAT_STANDARD) const;
|
||||||
|
|
||||||
/*************************
|
/***************
|
||||||
* Playing the game
|
* Setting the rack
|
||||||
* the int parameter should be 0 <= int < getNTurns
|
***************/
|
||||||
*************************/
|
|
||||||
int back(int);
|
|
||||||
|
|
||||||
/*************************
|
|
||||||
* Set the rack for searching
|
|
||||||
*
|
|
||||||
* The int parameter is a boolean, if this parameter
|
|
||||||
* set the rack will check that there are at least
|
|
||||||
* 2 vowels and 2 consonants before the round 15.
|
|
||||||
*
|
|
||||||
* The setrackmanual parameter string has to contain
|
|
||||||
* 'a' <= char <= 'z' or 'A' <= char <= 'Z' or '?'
|
|
||||||
*
|
|
||||||
* return value
|
|
||||||
* 0 : the rack has been set
|
|
||||||
* 1 : the bag does not contain enough tiles
|
|
||||||
* 2 : the rack check was set on and failed
|
|
||||||
* 3 : the rack cannot be completed (Game_*_setrackrandom only)
|
|
||||||
*************************/
|
|
||||||
enum set_rack_mode {RACK_ALL, RACK_NEW, RACK_MANUAL};
|
enum set_rack_mode {RACK_ALL, RACK_NEW, RACK_MANUAL};
|
||||||
int setRack(int player, set_rack_mode mode, bool check, const wstring& str);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Methods to access already played words.
|
|
||||||
* The int parameter should be 0 <= int < getNTurns()
|
|
||||||
*/
|
|
||||||
const History& getHistory() const { return m_history; }
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Methods to access players.
|
|
||||||
* The int parameter should be 0 <= int < getNPlayers()
|
|
||||||
*/
|
|
||||||
int getNPlayers() const { return m_players.size(); }
|
|
||||||
int getNHumanPlayers() const;
|
|
||||||
virtual void addHumanPlayer();
|
|
||||||
// TODO: Ability to specify which kind of AI player is wanted
|
|
||||||
virtual void addAIPlayer();
|
|
||||||
int currPlayer() const { return m_currPlayer; }
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Game handling
|
|
||||||
*/
|
|
||||||
virtual int start() = 0;
|
|
||||||
virtual int play(const wstring &iCoord, const wstring &iWord) = 0;
|
|
||||||
virtual int endTurn() = 0;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/// All the players, indexed by their ID
|
/// All the players, indexed by their ID
|
||||||
vector<Player*> m_players;
|
vector<Player*> m_players;
|
||||||
/// ID of the "current" player
|
/// ID of the "current" player
|
||||||
int m_currPlayer;
|
unsigned int m_currPlayer;
|
||||||
|
|
||||||
// TODO: check what should be private and what should be protected
|
// TODO: check what should be private and what should be protected
|
||||||
// private:
|
// private:
|
||||||
|
@ -181,7 +204,7 @@ protected:
|
||||||
GameVariant m_variant;
|
GameVariant m_variant;
|
||||||
|
|
||||||
/// Dictionary currently associated to the game
|
/// Dictionary currently associated to the game
|
||||||
const Dictionary * m_dic;
|
const Dictionary & m_dic;
|
||||||
|
|
||||||
/// Bag
|
/// Bag
|
||||||
Bag m_bag;
|
Bag m_bag;
|
||||||
|
@ -203,14 +226,91 @@ protected:
|
||||||
* Helper functions
|
* Helper functions
|
||||||
*********************************************************/
|
*********************************************************/
|
||||||
|
|
||||||
int helperPlayRound(const Round &iRound);
|
/** Play a Move for the given player, updating game history */
|
||||||
int helperSetRackRandom(int p, bool iCheck, set_rack_mode mode);
|
void helperPlayMove(unsigned int iPlayerId, const Move &iMove);
|
||||||
int helperSetRackManual(int p, bool iCheck, const wstring &iLetters);
|
|
||||||
|
/**
|
||||||
|
* Set the rack randomly for the player p
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: the game is over
|
||||||
|
* 3: there is no chance to set the rack with the vowels/consonants
|
||||||
|
* constraints
|
||||||
|
*
|
||||||
|
* Completing a rack randomly is more complex than it seems, because we
|
||||||
|
* must take into account several constraints:
|
||||||
|
* - if iCheck is true, we must ensure that the rack contains a minimum
|
||||||
|
* number of vowels and consonants (2 of each in the 15 first moves of
|
||||||
|
* the game, 1 of each after)
|
||||||
|
* - the game is over if the (real) bag contains only vowels or only
|
||||||
|
* consonants, and in particular if it contains only one letter
|
||||||
|
* - some letters (in particular the joker) can count both as a vowel and
|
||||||
|
* as a consonant (but not at the same time)
|
||||||
|
* - in a joker game, the joker must be present in the rack unless there
|
||||||
|
* is no joker left in the bag. In addition, we must prevent that both
|
||||||
|
* jokers are present in the rack at the same time
|
||||||
|
* - if completing a rack doesn't meet the requirements on the vowels and
|
||||||
|
* consonants, we must reject the rack completely (but only once,
|
||||||
|
* otherwise we have no guarantee that the rejects will stop eventually).
|
||||||
|
* This also means we have to check whether completing the rack with the
|
||||||
|
* requirements is possible...
|
||||||
|
*/
|
||||||
|
int helperSetRackRandom(unsigned int p, bool iCheck, set_rack_mode mode);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the rack randomly for the player p
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: the game is over
|
||||||
|
* 2: the rack was checked and was not correct (try calling the
|
||||||
|
* function again)
|
||||||
|
* 3: there is no chance to set the rack with the vowels/consonants
|
||||||
|
* constraints
|
||||||
|
*
|
||||||
|
* @deprecated: use helperSetRackRandom instead
|
||||||
|
*/
|
||||||
|
int helperSetRackRandomOld(unsigned int p, bool iCheck, set_rack_mode mode);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the rack for the player p with the given letters
|
||||||
|
* Possible return values:
|
||||||
|
* 0: everything went fine
|
||||||
|
* 1: the bag doesn't have the wanted letters
|
||||||
|
* 2: the rack was checked for vowels/consonants and was not correct
|
||||||
|
*/
|
||||||
|
int helperSetRackManual(unsigned int p, bool iCheck, const wstring &iLetters);
|
||||||
|
|
||||||
void prevPlayer();
|
void prevPlayer();
|
||||||
void nextPlayer();
|
void nextPlayer();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the players rack can be obtained from the bag.
|
||||||
|
* Since letters are removed from the bag only when the
|
||||||
|
* round is played we need to check that ALL the racks
|
||||||
|
* are in the bag simultaneously.
|
||||||
|
*
|
||||||
|
* FIXME: since we do not check for all racks it works
|
||||||
|
* for training and duplicate but it won't work for
|
||||||
|
* freegames.
|
||||||
|
*/
|
||||||
bool rackInBag(const Rack &iRack, const Bag &iBag) const;
|
bool rackInBag(const Rack &iRack, const Bag &iBag) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The realBag is the current bag minus all the racks
|
||||||
|
* present in the game. It represents the actual
|
||||||
|
* letters that are left in the bag.
|
||||||
|
* FIXME: in Duplicate mode, this method uses m_currPlayer to find the
|
||||||
|
* rack of the player. Since not all the players played the same word,
|
||||||
|
* it is important to set m_currPlayer accurately before!
|
||||||
|
*/
|
||||||
void realBag(Bag &iBag) const;
|
void realBag(Bag &iBag) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This function checks whether it is legal to play the given word at the
|
||||||
|
* given coordinates. If so, the function fills a Round object, also given
|
||||||
|
* as a parameter.
|
||||||
|
* Possible return values: same as the play() method
|
||||||
|
*/
|
||||||
int checkPlayedWord(const wstring &iCoord,
|
int checkPlayedWord(const wstring &iCoord,
|
||||||
const wstring &iWord, Round &oRound);
|
const wstring &iWord, Round &oRound);
|
||||||
|
|
||||||
|
@ -236,6 +336,14 @@ protected:
|
||||||
*/
|
*/
|
||||||
void gameSaveFormat_15(ostream &out) const;
|
void gameSaveFormat_15(ostream &out) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Play a round on the board.
|
||||||
|
* This should only be called by helperPlayMove().
|
||||||
|
*/
|
||||||
|
void helperPlayRound(unsigned int iPlayerId, const Round &iRound);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* _GAME_H_ */
|
#endif /* _GAME_H_ */
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -17,12 +19,19 @@
|
||||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
#include <getopt.h>
|
#include <getopt.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <fstream>
|
||||||
|
#include <exception>
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include "dic.h"
|
|
||||||
#include "game_factory.h"
|
#include "game_factory.h"
|
||||||
|
#include "game.h"
|
||||||
|
#include "training.h"
|
||||||
|
#include "freegame.h"
|
||||||
|
#include "duplicate.h"
|
||||||
|
#include "dic.h"
|
||||||
|
|
||||||
|
|
||||||
GameFactory *GameFactory::m_factory = NULL;
|
GameFactory *GameFactory::m_factory = NULL;
|
||||||
|
@ -35,8 +44,7 @@ GameFactory::GameFactory(): m_dic(NULL), m_human(0), m_ai(0), m_joker(false)
|
||||||
|
|
||||||
GameFactory::~GameFactory()
|
GameFactory::~GameFactory()
|
||||||
{
|
{
|
||||||
if (m_dic)
|
delete m_dic;
|
||||||
Dic_destroy(m_dic);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,8 +58,7 @@ GameFactory *GameFactory::Instance()
|
||||||
|
|
||||||
void GameFactory::Destroy()
|
void GameFactory::Destroy()
|
||||||
{
|
{
|
||||||
if (m_factory)
|
delete m_factory;
|
||||||
delete m_factory;
|
|
||||||
m_factory = NULL;
|
m_factory = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,16 +146,20 @@ Game *GameFactory::createFromCmdLine(int argc, char **argv)
|
||||||
cerr << "dict";
|
cerr << "dict";
|
||||||
else if (!found_m)
|
else if (!found_m)
|
||||||
cerr << "mode";
|
cerr << "mode";
|
||||||
cerr << "\n";
|
cerr << endl;
|
||||||
|
|
||||||
printUsage(argv[0]);
|
printUsage(argv[0]);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3) Try to load the dictionary
|
// 3) Try to load the dictionary
|
||||||
if (Dic_load(&m_dic, m_dicStr.c_str()))
|
try
|
||||||
{
|
{
|
||||||
cerr << "Could not load dictionary '" << m_dicStr << "'\n";
|
m_dic = new Dictionary(m_dicStr);
|
||||||
|
}
|
||||||
|
catch (std::exception &e)
|
||||||
|
{
|
||||||
|
cerr << e.what() << endl;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,19 +167,19 @@ Game *GameFactory::createFromCmdLine(int argc, char **argv)
|
||||||
Game *game = NULL;
|
Game *game = NULL;
|
||||||
if (m_modeStr == "training" || m_modeStr == "t")
|
if (m_modeStr == "training" || m_modeStr == "t")
|
||||||
{
|
{
|
||||||
game = createTraining(m_dic);
|
game = createTraining(*m_dic);
|
||||||
}
|
}
|
||||||
else if (m_modeStr == "freegame" || m_modeStr == "f")
|
else if (m_modeStr == "freegame" || m_modeStr == "f")
|
||||||
{
|
{
|
||||||
game = createFreeGame(m_dic);
|
game = createFreeGame(*m_dic);
|
||||||
}
|
}
|
||||||
else if (m_modeStr == "duplicate" || m_modeStr == "d")
|
else if (m_modeStr == "duplicate" || m_modeStr == "d")
|
||||||
{
|
{
|
||||||
game = createDuplicate(m_dic);
|
game = createDuplicate(*m_dic);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
cerr << "Invalid game mode '" << m_modeStr << "'\n";
|
cerr << "Invalid game mode '" << m_modeStr << "'" << endl;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -185,21 +196,21 @@ Game *GameFactory::createFromCmdLine(int argc, char **argv)
|
||||||
return game;
|
return game;
|
||||||
}
|
}
|
||||||
|
|
||||||
Game* GameFactory::load(string filename, const Dictionary &iDic)
|
|
||||||
|
Game* GameFactory::load(const string &iFileName, const Dictionary &iDic)
|
||||||
{
|
{
|
||||||
Game* game;
|
FILE* fin = fopen(iFileName.c_str(), "r");
|
||||||
FILE* fin;
|
if (fin == NULL)
|
||||||
if ((fin = fopen(filename.c_str(), "r")) == NULL)
|
{
|
||||||
{
|
printf("Cannot open %s\n", iFileName.c_str());
|
||||||
printf("impossible d'ouvrir %s\n",
|
return NULL;
|
||||||
filename.c_str());
|
}
|
||||||
return NULL;
|
Game *game = Game::load(fin, iDic);
|
||||||
}
|
|
||||||
game = Game::load(fin,iDic);
|
|
||||||
fclose(fin);
|
fclose(fin);
|
||||||
return game;
|
return game;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void GameFactory::releaseGame(Game &iGame)
|
void GameFactory::releaseGame(Game &iGame)
|
||||||
{
|
{
|
||||||
delete &iGame;
|
delete &iGame;
|
||||||
|
@ -208,25 +219,26 @@ void GameFactory::releaseGame(Game &iGame)
|
||||||
|
|
||||||
void GameFactory::printUsage(const string &iBinaryName) const
|
void GameFactory::printUsage(const string &iBinaryName) const
|
||||||
{
|
{
|
||||||
cout << "Usage: " << iBinaryName << " [options]\n"
|
cout << "Usage: " << iBinaryName << " [options]" << endl
|
||||||
<< "Options:\n"
|
<< "Options:" << endl
|
||||||
<< " -h, --help Print this help and exit\n"
|
<< " -h, --help Print this help and exit" << endl
|
||||||
<< " -v, --version Print version information and exit\n"
|
<< " -v, --version Print version information and exit" << endl
|
||||||
<< " -m, --mode {duplicate,d,freegame,f,training,t}\n"
|
<< " -m, --mode {duplicate,d,freegame,f,training,t}" << endl
|
||||||
<< " Choose game mode (mandatory)\n"
|
<< " Choose game mode (mandatory)" << endl
|
||||||
<< " -d, --dict <string> Choose a dictionary (mandatory)\n"
|
<< " -d, --dict <string> Choose a dictionary (mandatory)" << endl
|
||||||
<< " --human Add a human player\n"
|
<< " --human Add a human player" << endl
|
||||||
<< " --ai Add a AI (Artificial Intelligence) player\n"
|
<< " --ai Add a AI (Artificial Intelligence) player" << endl
|
||||||
<< " --joker Play with the \"Joker game\" variant\n";
|
<< " --joker Play with the \"Joker game\" variant" << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void GameFactory::printVersion() const
|
void GameFactory::printVersion() const
|
||||||
{
|
{
|
||||||
cout << PACKAGE_STRING << "\n"
|
cout << PACKAGE_STRING << endl
|
||||||
<< "This program comes with NO WARRANTY, to the extent permitted by "
|
<< "This program comes with NO WARRANTY, to the extent permitted by "
|
||||||
<< "law.\nYou may redistribute it under the terms of the GNU General "
|
<< "law." << endl << "You may redistribute it under the terms of the "
|
||||||
<< "Public License;\nsee the file named COPYING for details.\n";
|
<< "GNU General Public License;" << endl
|
||||||
|
<< "see the file named COPYING for details." << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2005-2007 Olivier Teulière
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -20,10 +21,15 @@
|
||||||
#ifndef _GAME_FACTORY_H_
|
#ifndef _GAME_FACTORY_H_
|
||||||
#define _GAME_FACTORY_H_
|
#define _GAME_FACTORY_H_
|
||||||
|
|
||||||
#include "game.h"
|
#include <string>
|
||||||
#include "training.h"
|
|
||||||
#include "freegame.h"
|
using std::string;
|
||||||
#include "duplicate.h"
|
|
||||||
|
class Dictionary;
|
||||||
|
class Game;
|
||||||
|
class Training;
|
||||||
|
class FreeGame;
|
||||||
|
class Duplicate;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -52,7 +58,7 @@ public:
|
||||||
* load() might need some more work to be robust enough to
|
* load() might need some more work to be robust enough to
|
||||||
* handle "hand written" files
|
* handle "hand written" files
|
||||||
*/
|
*/
|
||||||
Game *load(string filename, const Dictionary &iDic);
|
Game *load(const string &iFileName, const Dictionary &iDic);
|
||||||
|
|
||||||
Game *createFromCmdLine(int argc, char **argv);
|
Game *createFromCmdLine(int argc, char **argv);
|
||||||
|
|
||||||
|
@ -62,13 +68,13 @@ public:
|
||||||
private:
|
private:
|
||||||
|
|
||||||
GameFactory();
|
GameFactory();
|
||||||
virtual ~GameFactory();
|
~GameFactory();
|
||||||
|
|
||||||
/// The unique instance of the class
|
/// The unique instance of the class
|
||||||
static GameFactory *m_factory;
|
static GameFactory *m_factory;
|
||||||
|
|
||||||
/// Initial dictionary (it could be changed later)
|
/// Initial dictionary (it could be changed later)
|
||||||
Dictionary m_dic;
|
Dictionary *m_dic;
|
||||||
|
|
||||||
/** Parameters specified on the command-line */
|
/** Parameters specified on the command-line */
|
||||||
//@{
|
//@{
|
||||||
|
|
171
game/game_io.cpp
171
game/game_io.cpp
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -25,12 +26,16 @@
|
||||||
* \date 2002 - 2005
|
* \date 2002 - 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "dic.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
#include "turn.h"
|
#include "turn.h"
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
#include "game.h"
|
#include "game.h"
|
||||||
#include "game_factory.h"
|
#include "game_factory.h"
|
||||||
|
#include "training.h"
|
||||||
|
#include "freegame.h"
|
||||||
|
#include "duplicate.h"
|
||||||
#include "encoding.h"
|
#include "encoding.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
@ -107,14 +112,14 @@ Game* Game::gameLoadFormat_14(FILE *fin, const Dictionary& iDic)
|
||||||
char delim[]=" \t\n\012\015";
|
char delim[]=" \t\n\012\015";
|
||||||
char *token;
|
char *token;
|
||||||
Game *pGame = NULL;
|
Game *pGame = NULL;
|
||||||
|
|
||||||
debug("Game::gameLoadFormat_14\n");
|
debug("Game::gameLoadFormat_14\n");
|
||||||
pGame = GameFactory::Instance()->createTraining(iDic);
|
pGame = GameFactory::Instance()->createTraining(iDic);
|
||||||
pGame->start();
|
pGame->start();
|
||||||
|
|
||||||
/* rack word ?bonus pts coord */
|
/* rack word ?bonus pts coord */
|
||||||
/* EUOFMIE FUMEE * 26 H 4 */
|
/* EUOFMIE FUMEE * 26 H 4 */
|
||||||
|
|
||||||
/* read all turns until total */
|
/* read all turns until total */
|
||||||
while (fgets(buff, sizeof(buff), fin))
|
while (fgets(buff, sizeof(buff), fin))
|
||||||
{
|
{
|
||||||
|
@ -192,14 +197,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
Game *pGame = NULL;
|
Game *pGame = NULL;
|
||||||
|
|
||||||
char buff[4096];
|
char buff[4096];
|
||||||
int num;
|
|
||||||
char rack[20];
|
|
||||||
char word[20];
|
|
||||||
char ref[4];
|
|
||||||
int pts;
|
|
||||||
int player;
|
|
||||||
char *pos;
|
char *pos;
|
||||||
Tile tile;
|
|
||||||
|
|
||||||
/*************/
|
/*************/
|
||||||
/* Game type */
|
/* Game type */
|
||||||
|
@ -322,9 +320,15 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int num;
|
||||||
|
char rack[20];
|
||||||
|
char tmpWord[20];
|
||||||
|
char ref[4];
|
||||||
|
int pts;
|
||||||
|
unsigned int player;
|
||||||
char bonus = 0;
|
char bonus = 0;
|
||||||
int res = sscanf(buff, " %2d | %8s | %s | %3s | %3d | %1d | %c",
|
int res = sscanf(buff, " %2d | %8s | %s | %3s | %3d | %1d | %c",
|
||||||
&num, rack, word, ref, &pts, &player, &bonus);
|
&num, rack, tmpWord, ref, &pts, &player, &bonus);
|
||||||
|
|
||||||
debug(" -- line %s",buff);
|
debug(" -- line %s",buff);
|
||||||
|
|
||||||
|
@ -335,7 +339,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
}
|
}
|
||||||
|
|
||||||
debug(" %2d | %8s | %s | %3s | %3d | %1d | %c \n",
|
debug(" %2d | %8s | %s | %3s | %3d | %1d | %c \n",
|
||||||
num, rack, word, ref, pts, player, bonus);
|
num, rack, tmpWord, ref, pts, player, bonus);
|
||||||
|
|
||||||
// Integrity checks
|
// Integrity checks
|
||||||
// TODO: add more checks
|
// TODO: add more checks
|
||||||
|
@ -344,7 +348,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
debug(" Game::load15 line -%s- points < 0 ?\n",buff);
|
debug(" Game::load15 line -%s- points < 0 ?\n",buff);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (player < 0 || player > pGame->getNPlayers())
|
if (player > pGame->getNPlayers())
|
||||||
{
|
{
|
||||||
debug(" Game::load15 line -%s- too much player (%d>%d)",buff,player,pGame->getNPlayers());
|
debug(" Game::load15 line -%s- too much player (%d>%d)",buff,player,pGame->getNPlayers());
|
||||||
continue;
|
continue;
|
||||||
|
@ -357,10 +361,11 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
|
|
||||||
// Build a rack for the correct player
|
// Build a rack for the correct player
|
||||||
PlayedRack pldrack;
|
PlayedRack pldrack;
|
||||||
if ((res = pldrack.setManual(convertToWc(rack))) > 0)
|
if (!iDic.validateLetters(convertToWc(rack)))
|
||||||
{
|
{
|
||||||
debug(" Game::load15 set rack manual returned with error %d\n",res);
|
debug(" Game::load15 rack invalid for the current dictionary\n");
|
||||||
}
|
}
|
||||||
|
pldrack.setManual(convertToWc(rack));
|
||||||
debug(" history rack %s\n", convertToMb(pldrack.toString()).c_str());
|
debug(" history rack %s\n", convertToMb(pldrack.toString()).c_str());
|
||||||
|
|
||||||
// Build a round
|
// Build a round
|
||||||
|
@ -369,6 +374,8 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
if (bonus == '*')
|
if (bonus == '*')
|
||||||
round.setBonus(1);
|
round.setBonus(1);
|
||||||
|
|
||||||
|
wstring word = convertToWc(tmpWord);
|
||||||
|
Tile tile;
|
||||||
if (isalpha(ref[0]))
|
if (isalpha(ref[0]))
|
||||||
{
|
{
|
||||||
// Horizontal word
|
// Horizontal word
|
||||||
|
@ -376,7 +383,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
round.accessCoord().setRow(ref[0] - 'A' + 1);
|
round.accessCoord().setRow(ref[0] - 'A' + 1);
|
||||||
round.accessCoord().setCol(atoi(ref + 1));
|
round.accessCoord().setCol(atoi(ref + 1));
|
||||||
|
|
||||||
for (unsigned int i = 0; i < strlen(word); i++)
|
for (unsigned int i = 0; i < word.size(); i++)
|
||||||
{
|
{
|
||||||
tile = Tile(word[i]);
|
tile = Tile(word[i]);
|
||||||
|
|
||||||
|
@ -386,8 +393,8 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
round.addRightFromRack(tile, islower(word[i]));
|
round.addRightFromRack(tile, iswlower(word[i]));
|
||||||
pGame->m_bag.takeTile((islower(word[i])) ? Tile::Joker() : tile);
|
pGame->m_bag.takeTile((iswlower(word[i])) ? Tile::Joker() : tile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -398,7 +405,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
round.accessCoord().setRow(ref[strlen(ref) - 1] - 'A' + 1);
|
round.accessCoord().setRow(ref[strlen(ref) - 1] - 'A' + 1);
|
||||||
round.accessCoord().setCol(atoi(ref));
|
round.accessCoord().setCol(atoi(ref));
|
||||||
|
|
||||||
for (unsigned int i = 0; i < strlen(word); i++)
|
for (unsigned int i = 0; i < word.size(); i++)
|
||||||
{
|
{
|
||||||
tile = Tile(word[i]);
|
tile = Tile(word[i]);
|
||||||
|
|
||||||
|
@ -408,20 +415,20 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
round.addRightFromRack(tile, islower(word[i]));
|
round.addRightFromRack(tile, iswlower(word[i]));
|
||||||
pGame->m_bag.takeTile((islower(word[i])) ? Tile::Joker() : tile);
|
pGame->m_bag.takeTile((iswlower(word[i])) ? Tile::Joker() : tile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// pGame->m_currPlayer = player;
|
// pGame->m_currPlayer = player;
|
||||||
// // Update the rack for the player
|
// // Update the rack for the player
|
||||||
// pGame->m_players[player]->setCurrentRack(pldrack);
|
// pGame->m_players[player]->setCurrentRack(pldrack);
|
||||||
// // End the turn for the current player (this creates a new rack)
|
// // End the turn for the current player (this creates a new rack)
|
||||||
// pGame->m_players[player]->endTurn(round,num - 1);
|
// pGame->m_players[player]->endTurn(round,num - 1);
|
||||||
|
|
||||||
// Play the round
|
// Play the round
|
||||||
pGame->helperPlayRound(round);
|
pGame->helperPlayRound(pGame->m_currPlayer, round);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**************************************/
|
/**************************************/
|
||||||
|
@ -452,7 +459,7 @@ Game* Game::gameLoadFormat_15(FILE *fin, const Dictionary& iDic)
|
||||||
{
|
{
|
||||||
// We don't really know whose turn it is, but at least we know that
|
// We don't really know whose turn it is, but at least we know that
|
||||||
// the game was saved while a human was to play.
|
// the game was saved while a human was to play.
|
||||||
for (int i = 0; i < pGame->getNPlayers(); i++)
|
for (unsigned int i = 0; i < pGame->getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
if (pGame->m_players[i]->isHuman())
|
if (pGame->m_players[i]->isHuman())
|
||||||
{
|
{
|
||||||
|
@ -490,21 +497,21 @@ void Game::gameSaveFormat_14(ostream &out) const
|
||||||
const string decal = " ";
|
const string decal = " ";
|
||||||
out << IDENT_STRING << endl << endl;
|
out << IDENT_STRING << endl << endl;
|
||||||
|
|
||||||
for (int i = 0; i < m_history.getSize(); i++)
|
for (unsigned int i = 0; i < m_history.getSize(); i++)
|
||||||
{
|
{
|
||||||
const Turn& turn = m_history.getTurn(i);
|
const Turn& turn = m_history.getTurn(i);
|
||||||
string rack = convertToMb(turn.getPlayedRack().toString(PlayedRack::RACK_EXTRA));
|
wstring rack = turn.getPlayedRack().toString(PlayedRack::RACK_EXTRA);
|
||||||
string word = convertToMb(turn.getRound().getWord());
|
// FIXME: this will not work if the move does not correspond to a played round!
|
||||||
string coord = convertToMb(turn.getRound().getCoord().toString(Coord::COORD_MODE_LONG));
|
const Round &round = turn.getMove().getRound();
|
||||||
|
wstring word = round.getWord();
|
||||||
|
string coord = convertToMb(round.getCoord().toString(Coord::COORD_MODE_LONG));
|
||||||
|
|
||||||
// rack [space] word [space] bonus points coord
|
// rack [space] word [space] bonus points coord
|
||||||
sprintf(line,"%s%s%s%s%c%4d %s",
|
sprintf(line,"%s%s%c%4d %s",
|
||||||
rack.c_str(),
|
padAndConvert(rack, 12, false).c_str(),
|
||||||
string(12 - rack.size(), ' ').c_str(),
|
padAndConvert(word, 16, false).c_str(),
|
||||||
word.c_str(),
|
round.getBonus() ? '*' : ' ',
|
||||||
string(16 - word.size(), ' ').c_str(),
|
round.getPoints(),
|
||||||
turn.getRound().getBonus() ? '*' : ' ',
|
|
||||||
turn.getRound().getPoints(),
|
|
||||||
coord.c_str()
|
coord.c_str()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -513,7 +520,7 @@ void Game::gameSaveFormat_14(ostream &out) const
|
||||||
|
|
||||||
out << endl;
|
out << endl;
|
||||||
out << decal << "total" << string(24,' ');
|
out << decal << "total" << string(24,' ');
|
||||||
sprintf(line,"%4d", getCurrentPlayer().getPoints());
|
sprintf(line, "%4d", getCurrentPlayer().getPoints());
|
||||||
out << line << endl;
|
out << line << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -526,7 +533,7 @@ void Game::gameSaveFormat_15(ostream &out) const
|
||||||
// Game type
|
// Game type
|
||||||
out << "Game type: " << getModeAsString() << endl;
|
out << "Game type: " << getModeAsString() << endl;
|
||||||
// Player list
|
// Player list
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
out << "Player " << i << ": ";
|
out << "Player " << i << ": ";
|
||||||
if (m_players[i]->isHuman())
|
if (m_players[i]->isHuman())
|
||||||
|
@ -542,21 +549,69 @@ void Game::gameSaveFormat_15(ostream &out) const
|
||||||
out << decal << "===|==========|=================|=====|=====|===|======" << endl;
|
out << decal << "===|==========|=================|=====|=====|===|======" << endl;
|
||||||
|
|
||||||
// Print the game itself
|
// Print the game itself
|
||||||
for (int i = 0; i < m_history.getSize(); i++)
|
for (unsigned int i = 0; i < m_history.getSize(); i++)
|
||||||
{
|
{
|
||||||
const Turn& turn = m_history.getTurn(i);
|
const Turn& turn = m_history.getTurn(i);
|
||||||
string rack = convertToMb(turn.getPlayedRack().toString(PlayedRack::RACK_EXTRA));
|
wstring rack = turn.getPlayedRack().toString(PlayedRack::RACK_EXTRA);
|
||||||
string word = convertToMb(turn.getRound().getWord());
|
const Move &move = turn.getMove();
|
||||||
string coord = convertToMb(turn.getRound().getCoord().toString());
|
switch (move.getType())
|
||||||
sprintf(line, "%2d | %8s | %s%s | %3s | %3d | %1d | %c",
|
{
|
||||||
i + 1,
|
case Move::VALID_ROUND:
|
||||||
rack.c_str(), /* pldrack */
|
{
|
||||||
word.c_str(), /* word */
|
const Round &round = move.getRound();
|
||||||
string(15 - word.size(), ' ').c_str(), /* fill spaces */
|
wstring word = round.getWord();
|
||||||
coord.c_str(), /* coord */
|
string coord = convertToMb(round.getCoord().toString());
|
||||||
turn.getRound().getPoints(),
|
sprintf(line, "%2d | %s | %s | %3s | %3d | %1d | %c",
|
||||||
turn.getPlayer(),
|
i + 1,
|
||||||
turn.getRound().getBonus() ? '*' : ' ');
|
padAndConvert(rack, 8).c_str(), /* pldrack */
|
||||||
|
padAndConvert(word, 15, false).c_str(), /* word */
|
||||||
|
coord.c_str(), /* coord */
|
||||||
|
move.getScore(),
|
||||||
|
turn.getPlayer(),
|
||||||
|
round.getBonus() ? '*' : ' ');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Move::INVALID_WORD:
|
||||||
|
{
|
||||||
|
wstring word = move.getBadWord();
|
||||||
|
string coord = convertToMb(move.getBadCoord());
|
||||||
|
sprintf(line, "%2d | %s | %s | %3s | %3d | %1d |",
|
||||||
|
i + 1,
|
||||||
|
padAndConvert(rack, 8).c_str(), /* pldrack */
|
||||||
|
padAndConvert(word, 15, false).c_str(), /* word */
|
||||||
|
coord.c_str(), /* coord */
|
||||||
|
move.getScore(),
|
||||||
|
turn.getPlayer());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Move::PASS:
|
||||||
|
{
|
||||||
|
string action = "(PASS)";
|
||||||
|
string coord = " - ";
|
||||||
|
sprintf(line, "%2d | %s | %s | %3s | %3d | %1d |",
|
||||||
|
i + 1,
|
||||||
|
padAndConvert(rack, 8).c_str(), /* pldrack */
|
||||||
|
truncOrPad(action, 15, ' ').c_str(), /* word */
|
||||||
|
coord.c_str(), /* coord */
|
||||||
|
move.getScore(),
|
||||||
|
turn.getPlayer());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Move::CHANGE_LETTERS:
|
||||||
|
{
|
||||||
|
wstring action = L"(-" + move.getChangedLetters() + L")";
|
||||||
|
string coord = " - ";
|
||||||
|
sprintf(line, "%2d | %s | %s | %3s | %3d | %1d |",
|
||||||
|
i + 1,
|
||||||
|
padAndConvert(rack, 8).c_str(), /* pldrack */
|
||||||
|
padAndConvert(action, 15, false).c_str(), /* word */
|
||||||
|
coord.c_str(), /* coord */
|
||||||
|
move.getScore(),
|
||||||
|
turn.getPlayer());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
out << decal << line << endl;
|
out << decal << line << endl;
|
||||||
}
|
}
|
||||||
|
@ -577,7 +632,7 @@ void Game::gameSaveFormat_15(ostream &out) const
|
||||||
|
|
||||||
// Print current rack for all the players
|
// Print current rack for all the players
|
||||||
out << endl;
|
out << endl;
|
||||||
for (int i = 0; i < getNPlayers(); i++)
|
for (unsigned int i = 0; i < getNPlayers(); i++)
|
||||||
{
|
{
|
||||||
wstring rack = m_players[i]->getCurrentRack().toString();
|
wstring rack = m_players[i]->getCurrentRack().toString();
|
||||||
out << "Rack " << i << ": " << convertToMb(rack) << endl;
|
out << "Rack " << i << ": " << convertToMb(rack) << endl;
|
||||||
|
|
124
game/history.cpp
124
game/history.cpp
|
@ -1,21 +1,23 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file history.cpp
|
* \file history.cpp
|
||||||
|
@ -27,7 +29,7 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "round.h"
|
#include "move.h"
|
||||||
#include "turn.h"
|
#include "turn.h"
|
||||||
#include "history.h"
|
#include "history.h"
|
||||||
#include "encoding.h"
|
#include "encoding.h"
|
||||||
|
@ -41,7 +43,7 @@
|
||||||
|
|
||||||
History::History()
|
History::History()
|
||||||
{
|
{
|
||||||
Turn* t = new Turn ();
|
Turn* t = new Turn();
|
||||||
m_history.clear();
|
m_history.clear();
|
||||||
m_history.push_back(t);
|
m_history.push_back(t);
|
||||||
}
|
}
|
||||||
|
@ -51,17 +53,14 @@ History::~History()
|
||||||
{
|
{
|
||||||
for (unsigned int i = 0; i < m_history.size(); i++)
|
for (unsigned int i = 0; i < m_history.size(); i++)
|
||||||
{
|
{
|
||||||
if (m_history[i] != NULL)
|
delete m_history[i];
|
||||||
{
|
|
||||||
delete m_history[i];
|
|
||||||
m_history[i] = NULL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int History::getSize() const
|
unsigned int History::getSize() const
|
||||||
{
|
{
|
||||||
|
ASSERT(!m_history.empty(), "Invalid history size");
|
||||||
return m_history.size() - 1;
|
return m_history.size() - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,44 +87,61 @@ const Turn& History::getPreviousTurn() const
|
||||||
|
|
||||||
const Turn& History::getTurn(unsigned int n) const
|
const Turn& History::getTurn(unsigned int n) const
|
||||||
{
|
{
|
||||||
// ASSERT(0 <= n && n < m_history.size(), "Wrong turn number");
|
|
||||||
ASSERT(n < m_history.size(), "Wrong turn number");
|
ASSERT(n < m_history.size(), "Wrong turn number");
|
||||||
return *(m_history[n]);
|
return *(m_history[n]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* This function increments the number of racks, and fills the new rack
|
bool History::beforeFirstRound() const
|
||||||
* with the unplayed tiles from the previous one.
|
|
||||||
* 03 sept 2000 : We have to sort the tiles according to the new rules
|
|
||||||
*/
|
|
||||||
void History::playRound(int player, int turn, const Round& round)
|
|
||||||
{
|
{
|
||||||
|
for (unsigned int i = 0; i < m_history.size() - 1; i++)
|
||||||
|
{
|
||||||
|
if (m_history[i]->getMove().getType() == Move::VALID_ROUND)
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void History::playMove(unsigned int iPlayer, unsigned int iTurn, const Move &iMove)
|
||||||
|
{
|
||||||
|
Turn * current_turn = m_history.back();
|
||||||
|
|
||||||
|
// Set the number and the round
|
||||||
|
current_turn->setNum(iTurn);
|
||||||
|
current_turn->setPlayer(iPlayer);
|
||||||
|
current_turn->setMove(iMove);
|
||||||
|
|
||||||
|
// Get what was the rack for the current turn
|
||||||
Rack rack;
|
Rack rack;
|
||||||
Turn * current_turn;
|
|
||||||
|
|
||||||
current_turn = m_history.back();
|
|
||||||
|
|
||||||
/* set the number and the round */
|
|
||||||
current_turn->setNum(turn);
|
|
||||||
current_turn->setPlayer(player);
|
|
||||||
current_turn->setRound(round);
|
|
||||||
|
|
||||||
/* get what was the rack for the current turn */
|
|
||||||
current_turn->getPlayedRack().getRack(rack);
|
current_turn->getPlayedRack().getRack(rack);
|
||||||
|
|
||||||
/* remove the played tiles from the rack */
|
if (iMove.getType() == Move::VALID_ROUND)
|
||||||
for (int i = 0; i < round.getWordLen(); i++)
|
|
||||||
{
|
{
|
||||||
if (round.isPlayedFromRack(i))
|
// Remove the played tiles from the rack
|
||||||
|
const Round &round = iMove.getRound();
|
||||||
|
for (unsigned int i = 0; i < round.getWordLen(); i++)
|
||||||
{
|
{
|
||||||
if (round.isJoker(i))
|
if (round.isPlayedFromRack(i))
|
||||||
rack.remove(Tile::Joker());
|
{
|
||||||
else
|
if (round.isJoker(i))
|
||||||
rack.remove(round.getTile(i));
|
rack.remove(Tile::Joker());
|
||||||
|
else
|
||||||
|
rack.remove(round.getTile(i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (iMove.getType() == Move::CHANGE_LETTERS)
|
||||||
|
{
|
||||||
|
// Remove the changed tiles from the rack
|
||||||
|
const wstring & changed = iMove.getChangedLetters();
|
||||||
|
for (unsigned int i = 0; i < changed.size(); ++i)
|
||||||
|
{
|
||||||
|
rack.remove(Tile(changed[i]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* create a new turn */
|
// Create a new turn
|
||||||
Turn * next_turn = new Turn();
|
Turn * next_turn = new Turn();
|
||||||
PlayedRack pldrack;
|
PlayedRack pldrack;
|
||||||
pldrack.setOld(rack);
|
pldrack.setOld(rack);
|
||||||
|
@ -146,11 +162,11 @@ void History::removeLastTurn()
|
||||||
delete t;
|
delete t;
|
||||||
}
|
}
|
||||||
|
|
||||||
// now we have the previous played round in back()
|
// Now we have the previous played round in back()
|
||||||
Turn* t = m_history.back();
|
Turn *t = m_history.back();
|
||||||
t->setNum(0);
|
t->setNum(0);
|
||||||
t->setPlayer(0);
|
t->setPlayer(0);
|
||||||
t->setRound(Round());
|
//t->setRound(Round());
|
||||||
#ifdef BACK_REMOVE_RACK_NEW_PART
|
#ifdef BACK_REMOVE_RACK_NEW_PART
|
||||||
t->getPlayedRound().setNew(Rack());
|
t->getPlayedRound().setNew(Rack());
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,21 +1,23 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file history.h
|
* \file history.h
|
||||||
|
@ -33,7 +35,7 @@
|
||||||
using std::wstring;
|
using std::wstring;
|
||||||
using std::vector;
|
using std::vector;
|
||||||
|
|
||||||
class Round;
|
class Move;
|
||||||
class Turn;
|
class Turn;
|
||||||
class PlayedRack;
|
class PlayedRack;
|
||||||
|
|
||||||
|
@ -41,15 +43,12 @@ class PlayedRack;
|
||||||
* History stores all the turns that have been played
|
* History stores all the turns that have been played
|
||||||
* This class is used many times in the game
|
* This class is used many times in the game
|
||||||
* - one for the complete game
|
* - one for the complete game
|
||||||
* - one for each of the players
|
* - one for each player
|
||||||
*
|
*
|
||||||
* A History is never void (getSize() can be used as the is the current turn
|
* The top of the history is an empty Turn until it has been filled
|
||||||
* number for the complete game history).
|
* and the game is up to a new turn. So a History object is never empty.
|
||||||
*
|
* However, the getSize() method only returns the number of complete
|
||||||
* History starts at zero.
|
* turns, and can therefore return 0.
|
||||||
*
|
|
||||||
* The top of the history is an empty
|
|
||||||
* Turn until it has been filled and game is up to a new round.
|
|
||||||
*
|
*
|
||||||
* getCurrentRack() can/should be used to store the current played rack.
|
* getCurrentRack() can/should be used to store the current played rack.
|
||||||
* setCurrentRack must be called whenever the current played rack is
|
* setCurrentRack must be called whenever the current played rack is
|
||||||
|
@ -58,31 +57,41 @@ class PlayedRack;
|
||||||
* History owns the turns that it stores. Do not delete a turn referenced
|
* History owns the turns that it stores. Do not delete a turn referenced
|
||||||
* by History
|
* by History
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class History
|
class History
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
History();
|
History();
|
||||||
virtual ~History();
|
~History();
|
||||||
|
|
||||||
/// get the size of the history
|
/// Get the size of the history (without the current incomplete turn)
|
||||||
int getSize() const;
|
unsigned int getSize() const;
|
||||||
|
|
||||||
/// Get the (possibly incomplete) rack
|
/// Get the (possibly incomplete) rack
|
||||||
const PlayedRack& getCurrentRack() const;
|
const PlayedRack& getCurrentRack() const;
|
||||||
|
|
||||||
/// Set the current rack
|
/// Set the current rack
|
||||||
void setCurrentRack(const PlayedRack &iPld);
|
void setCurrentRack(const PlayedRack &iPld);
|
||||||
|
|
||||||
/// Get the previous turn
|
/// Get the previous (complete) turn
|
||||||
const Turn& getPreviousTurn() const;
|
const Turn& getPreviousTurn() const;
|
||||||
|
|
||||||
/// Get turn 'n'
|
/// Get turn 'n' (starting at 0)
|
||||||
const Turn& getTurn(unsigned int) const;
|
const Turn& getTurn(unsigned int) const;
|
||||||
|
|
||||||
/// Update the "history" with the given round and complete the turn.
|
/**
|
||||||
/// A new turn is created with the remaining letters in the rack
|
* Return true if the history doesn't contain at least one move
|
||||||
void playRound(int player, int turn, const Round& round);
|
* corresponding to a valid round, false otherwise.
|
||||||
|
* Said differently, this method checks whether a word was already played
|
||||||
|
* on the board.
|
||||||
|
*/
|
||||||
|
bool beforeFirstRound() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update the history with the given move and complete the turn.
|
||||||
|
* A new turn is created with the unplayed letters in the rack
|
||||||
|
* 03 sept 2000: We have to sort the tiles according to the new rules
|
||||||
|
*/
|
||||||
|
void playMove(unsigned int player, unsigned int turn, const Move &iMove);
|
||||||
|
|
||||||
/// Remove last turn
|
/// Remove last turn
|
||||||
void removeLastTurn();
|
void removeLastTurn();
|
||||||
|
|
107
game/move.cpp
Normal file
107
game/move.cpp
Normal file
|
@ -0,0 +1,107 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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 <algorithm>
|
||||||
|
#include <wctype.h>
|
||||||
|
|
||||||
|
#include "move.h"
|
||||||
|
|
||||||
|
|
||||||
|
Move::Move(const Round &iRound)
|
||||||
|
: m_score(0), m_round(iRound)
|
||||||
|
{
|
||||||
|
m_type = VALID_ROUND;
|
||||||
|
m_score = m_round.getPoints();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Move::Move(const wstring &iWord, const wstring &iCoord)
|
||||||
|
: m_word(iWord), m_coord(iCoord)
|
||||||
|
{
|
||||||
|
m_type = INVALID_WORD;
|
||||||
|
m_score = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Move::Move(const wstring &iLetters)
|
||||||
|
: m_score(0), m_letters(iLetters)
|
||||||
|
{
|
||||||
|
// Make the letters uppercase
|
||||||
|
std::transform(m_letters.begin(), m_letters.end(),
|
||||||
|
m_letters.begin(), towupper);
|
||||||
|
|
||||||
|
if (m_letters.empty())
|
||||||
|
m_type = PASS;
|
||||||
|
else
|
||||||
|
m_type = CHANGE_LETTERS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const Round & Move::getRound() const
|
||||||
|
{
|
||||||
|
if (m_type != VALID_ROUND)
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
return m_round;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const wstring & Move::getBadWord() const
|
||||||
|
{
|
||||||
|
if (m_type != INVALID_WORD)
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
return m_word;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const wstring & Move::getBadCoord() const
|
||||||
|
{
|
||||||
|
if (m_type != INVALID_WORD)
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
return m_coord;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const wstring & Move::getChangedLetters() const
|
||||||
|
{
|
||||||
|
if (m_type != CHANGE_LETTERS &&
|
||||||
|
m_type != PASS)
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
return m_letters;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
wstring Move::toString() const
|
||||||
|
{
|
||||||
|
// TODO
|
||||||
|
return L"";
|
||||||
|
}
|
||||||
|
|
134
game/move.h
Normal file
134
game/move.h
Normal file
|
@ -0,0 +1,134 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef _MOVE_H
|
||||||
|
#define _MOVE_H
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "round.h"
|
||||||
|
|
||||||
|
using std::wstring;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A Move is what a player can do during the game:
|
||||||
|
* - play a valid word
|
||||||
|
* - play an invalid or misplaced word
|
||||||
|
* - pass the turn (freegame only)
|
||||||
|
* - change letters (freegame only)
|
||||||
|
* - play nothing (timeout) (not supported yet)
|
||||||
|
*
|
||||||
|
* Moves are useful to record what happened, even if the board doesn't keep
|
||||||
|
* a trace of the move (e.g.: an invalid move which was rejected will still
|
||||||
|
* be remembered in the player history).
|
||||||
|
*
|
||||||
|
* Currently, moves are not used by the interfaces (they are only used
|
||||||
|
* internally), but this could change in the future.
|
||||||
|
*/
|
||||||
|
class Move
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/**
|
||||||
|
* Constructor taking a (valid) round
|
||||||
|
*/
|
||||||
|
explicit Move(const Round &iRound);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor taking a word and its coordinates, corresponding
|
||||||
|
* to an invalid move by the player (invalid word, invalid coordinates,
|
||||||
|
* letters not corresponding to the rack, ...)
|
||||||
|
*/
|
||||||
|
explicit Move(const wstring &iWord, const wstring &iCoord);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor taking letters to change.
|
||||||
|
* An empty string means that the player simply passes without
|
||||||
|
* changing any letter.
|
||||||
|
* The given letters must have been already validated for correctness.
|
||||||
|
*/
|
||||||
|
explicit Move(const wstring &iLetters);
|
||||||
|
|
||||||
|
enum Type
|
||||||
|
{
|
||||||
|
VALID_ROUND,
|
||||||
|
INVALID_WORD,
|
||||||
|
PASS,
|
||||||
|
CHANGE_LETTERS
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Return the type of move
|
||||||
|
Type getType() const { return m_type; }
|
||||||
|
|
||||||
|
/// Get the score of this move (0 unless the round is valid)
|
||||||
|
int getScore() const { return m_score; };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the round associated with the move, or throw an exception
|
||||||
|
* if this move was not constructed from a valid round
|
||||||
|
*/
|
||||||
|
const Round & getRound() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the word played at this move associated with the move, or
|
||||||
|
* throw an exception if this move was not constructed from an invalid
|
||||||
|
* pair (word, coord)
|
||||||
|
*/
|
||||||
|
const wstring & getBadWord() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the coordinates of the (incorrect) word played at this move,
|
||||||
|
* or throw an exception if this move was not constructed from an
|
||||||
|
* invalid pair (cord, coord)
|
||||||
|
*/
|
||||||
|
const wstring & getBadCoord() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the changed letters (possibly an empty string if the player
|
||||||
|
* simply wanted to pass the turn), or throw an exception if this move
|
||||||
|
* does not correspond to a passed turn
|
||||||
|
*/
|
||||||
|
const wstring & getChangedLetters() const;
|
||||||
|
|
||||||
|
/// To help debugging
|
||||||
|
wstring toString() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Type of move
|
||||||
|
Type m_type;
|
||||||
|
|
||||||
|
/// Score associated with this move
|
||||||
|
int m_score;
|
||||||
|
|
||||||
|
/// Round played at this turn
|
||||||
|
Round m_round;
|
||||||
|
|
||||||
|
/// Word played (incorrectly)
|
||||||
|
wstring m_word;
|
||||||
|
|
||||||
|
/// Coordinates of the word played (incorrectly)
|
||||||
|
wstring m_coord;
|
||||||
|
|
||||||
|
/// Changed letters (or empty string for passed turn)
|
||||||
|
wstring m_letters;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2004-2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2004-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -31,14 +33,8 @@
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
Player::Player(int iId)
|
Player::Player(unsigned int iId)
|
||||||
{
|
: m_id(iId), m_score(0)
|
||||||
m_id = iId;
|
|
||||||
m_score = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Player::~Player()
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,19 +57,22 @@ const PlayedRack & Player::getLastRack() const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const Round & Player::getLastRound() const
|
const Move & Player::getLastMove() const
|
||||||
{
|
{
|
||||||
return m_history.getPreviousTurn().getRound();
|
return m_history.getPreviousTurn().getMove();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Player::endTurn(const Round &iRound, int iTurn)
|
void Player::endTurn(const Move &iMove, unsigned int iTurn)
|
||||||
{
|
{
|
||||||
m_history.playRound(m_id,iTurn,iRound);
|
addPoints(iMove.getScore());
|
||||||
|
m_history.playMove(m_id, iTurn, iMove);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Player::removeLastTurn()
|
void Player::removeLastTurn()
|
||||||
{
|
{
|
||||||
|
// Remove points of the last turn
|
||||||
|
addPoints(- m_history.getPreviousTurn().getMove().getScore());
|
||||||
m_history.removeLastTurn();
|
m_history.removeLastTurn();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 2004-2005 Eliot
|
* Eliot
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
* Copyright (C) 2004-2007 Olivier Teulière & Antoine Fraboulet
|
||||||
|
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
|
* Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -35,8 +37,8 @@ class Turn;
|
||||||
class Player
|
class Player
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Player(int iId);
|
explicit Player(unsigned int iId);
|
||||||
virtual ~Player();
|
virtual ~Player() {}
|
||||||
|
|
||||||
// Pseudo RTTI
|
// Pseudo RTTI
|
||||||
virtual bool isHuman() const = 0;
|
virtual bool isHuman() const = 0;
|
||||||
|
@ -48,8 +50,8 @@ public:
|
||||||
const PlayedRack & getCurrentRack() const;
|
const PlayedRack & getCurrentRack() const;
|
||||||
// Get the previous rack
|
// Get the previous rack
|
||||||
const PlayedRack & getLastRack() const;
|
const PlayedRack & getLastRack() const;
|
||||||
// Get the previous round (corresponding to the previous rack...)
|
/// Get the previous move (corresponding to the previous rack...)
|
||||||
const Round & getLastRound() const;
|
const Move & getLastMove() const;
|
||||||
|
|
||||||
void setCurrentRack(const PlayedRack &iPld);
|
void setCurrentRack(const PlayedRack &iPld);
|
||||||
|
|
||||||
|
@ -65,15 +67,19 @@ public:
|
||||||
void addPoints(int iPoints) { m_score += iPoints; }
|
void addPoints(int iPoints) { m_score += iPoints; }
|
||||||
int getPoints() const { return m_score; }
|
int getPoints() const { return m_score; }
|
||||||
|
|
||||||
// Update the player "history", with the given round.
|
/**
|
||||||
// A new rack is created with the remaining letters
|
* Update the player "history", with the given move.
|
||||||
void endTurn(const Round &iRound, int iTurn);
|
* A new rack is created with the remaining letters.
|
||||||
|
* The score of the player is updated with the one of the move, if it is
|
||||||
|
* meaningful.
|
||||||
|
*/
|
||||||
|
void endTurn(const Move &iMove, unsigned int iTurn);
|
||||||
|
|
||||||
wstring toString() const;
|
wstring toString() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// ID of the player
|
/// ID of the player
|
||||||
int m_id;
|
unsigned int m_id;
|
||||||
|
|
||||||
/// Score of the player
|
/// Score of the player
|
||||||
int m_score;
|
int m_score;
|
||||||
|
@ -89,8 +95,6 @@ private:
|
||||||
class HumanPlayer: public Player
|
class HumanPlayer: public Player
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
string name;
|
|
||||||
|
|
||||||
HumanPlayer(int iId): Player(iId) {}
|
HumanPlayer(int iId): Player(iId) {}
|
||||||
virtual ~HumanPlayer() {}
|
virtual ~HumanPlayer() {}
|
||||||
|
|
||||||
|
|
104
game/pldrack.cpp
104
game/pldrack.cpp
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -25,17 +26,17 @@
|
||||||
* \date 2002 - 2005
|
* \date 2002 - 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "rack.h"
|
#include <algorithm>
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
|
#include "rack.h"
|
||||||
#include "debug.h"
|
|
||||||
|
|
||||||
|
|
||||||
PlayedRack::PlayedRack()
|
PlayedRack::PlayedRack()
|
||||||
|
: m_reject(false)
|
||||||
{
|
{
|
||||||
reject = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void PlayedRack::addOld(const Tile &t)
|
void PlayedRack::addOld(const Tile &t)
|
||||||
{
|
{
|
||||||
m_oldTiles.push_back(t);
|
m_oldTiles.push_back(t);
|
||||||
|
@ -51,26 +52,22 @@ void PlayedRack::addNew(const Tile &t)
|
||||||
void PlayedRack::getOldTiles(vector<Tile> &oTiles) const
|
void PlayedRack::getOldTiles(vector<Tile> &oTiles) const
|
||||||
{
|
{
|
||||||
oTiles.clear();
|
oTiles.clear();
|
||||||
for (int i = 0; i < nOld(); i++)
|
oTiles = m_oldTiles;
|
||||||
oTiles.push_back(m_oldTiles[i]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void PlayedRack::getNewTiles(vector<Tile> &oTiles) const
|
void PlayedRack::getNewTiles(vector<Tile> &oTiles) const
|
||||||
{
|
{
|
||||||
oTiles.clear();
|
oTiles.clear();
|
||||||
for (int i = 0; i < nNew(); i++)
|
oTiles = m_newTiles;
|
||||||
oTiles.push_back(m_newTiles[i]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void PlayedRack::getAllTiles(vector<Tile> &oTiles) const
|
void PlayedRack::getAllTiles(vector<Tile> &oTiles) const
|
||||||
{
|
{
|
||||||
oTiles.clear();
|
oTiles.clear();
|
||||||
for (int i = 0; i < nOld(); i++)
|
oTiles = m_oldTiles;
|
||||||
oTiles.push_back(m_oldTiles[i]);
|
oTiles.insert(oTiles.end(), m_newTiles.begin(), m_newTiles.end());
|
||||||
for (int j = 0; j < nNew(); j++)
|
|
||||||
oTiles.push_back(m_newTiles[j]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -78,6 +75,7 @@ void PlayedRack::reset()
|
||||||
{
|
{
|
||||||
m_oldTiles.clear();
|
m_oldTiles.clear();
|
||||||
m_newTiles.clear();
|
m_newTiles.clear();
|
||||||
|
m_reject = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -122,72 +120,47 @@ void PlayedRack::getRack(Rack &oRack) const
|
||||||
|
|
||||||
void PlayedRack::setOld(const Rack &iRack)
|
void PlayedRack::setOld(const Rack &iRack)
|
||||||
{
|
{
|
||||||
list<Tile> l;
|
|
||||||
iRack.getTiles(l);
|
|
||||||
|
|
||||||
m_oldTiles.clear();
|
m_oldTiles.clear();
|
||||||
list<Tile>::const_iterator it;
|
iRack.getTiles(m_oldTiles);
|
||||||
for (it = l.begin(); it != l.end(); it++)
|
|
||||||
{
|
|
||||||
addOld(*it);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void PlayedRack::setNew(const Rack &iRack)
|
void PlayedRack::setNew(const Rack &iRack)
|
||||||
{
|
{
|
||||||
list<Tile> l;
|
|
||||||
iRack.getTiles(l);
|
|
||||||
|
|
||||||
m_newTiles.clear();
|
m_newTiles.clear();
|
||||||
list<Tile>::const_iterator it;
|
iRack.getTiles(m_newTiles);
|
||||||
for (it = l.begin(); it != l.end(); it++)
|
|
||||||
{
|
|
||||||
addNew(*it);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int PlayedRack::setManual(const wstring& iLetters)
|
|
||||||
|
void PlayedRack::setManual(const wstring& iLetters)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
|
||||||
reset();
|
reset();
|
||||||
|
|
||||||
if (iLetters.size() == 0)
|
// An empty rack is OK
|
||||||
{
|
if (iLetters.empty())
|
||||||
return 0; /* empty is ok */
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
|
unsigned int i;
|
||||||
for (i = 0; i < iLetters.size() && iLetters[i] != L'+'; i++)
|
for (i = 0; i < iLetters.size() && iLetters[i] != L'+'; i++)
|
||||||
{
|
{
|
||||||
Tile tile(iLetters[i]);
|
addOld(Tile(iLetters[i]));
|
||||||
if (tile.isEmpty())
|
|
||||||
{
|
|
||||||
return 1; /* */
|
|
||||||
}
|
|
||||||
addOld(tile);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (i < iLetters.size() && iLetters[i] == L'+')
|
if (i < iLetters.size() && iLetters[i] == L'+')
|
||||||
{
|
{
|
||||||
for (i++; i < iLetters.size(); i++)
|
for (i++; i < iLetters.size(); i++)
|
||||||
{
|
{
|
||||||
Tile tile(iLetters[i]);
|
addNew(Tile(iLetters[i]));
|
||||||
if (tile.isEmpty())
|
|
||||||
{
|
|
||||||
return 1; /* */
|
|
||||||
}
|
|
||||||
addNew(tile);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool PlayedRack::checkRack(int cMin, int vMin) const
|
|
||||||
|
bool PlayedRack::checkRack(unsigned int cMin, unsigned int vMin) const
|
||||||
{
|
{
|
||||||
vector<Tile>::const_iterator it;
|
vector<Tile>::const_iterator it;
|
||||||
int v = 0;
|
unsigned int v = 0;
|
||||||
int c = 0;
|
unsigned int c = 0;
|
||||||
|
|
||||||
for (it = m_oldTiles.begin(); it != m_oldTiles.end(); it++)
|
for (it = m_oldTiles.begin(); it != m_oldTiles.end(); it++)
|
||||||
{
|
{
|
||||||
|
@ -203,10 +176,9 @@ bool PlayedRack::checkRack(int cMin, int vMin) const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void PlayedRack::operator=(const PlayedRack &iOther)
|
void PlayedRack::shuffleNew()
|
||||||
{
|
{
|
||||||
m_oldTiles = iOther.m_oldTiles;
|
std::random_shuffle(m_newTiles.begin(), m_newTiles.end());
|
||||||
m_newTiles = iOther.m_newTiles;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -215,25 +187,23 @@ wstring PlayedRack::toString(display_mode mode) const
|
||||||
wstring s;
|
wstring s;
|
||||||
vector<Tile>::const_iterator it;
|
vector<Tile>::const_iterator it;
|
||||||
|
|
||||||
if (nOld() > 0)
|
if (mode >= RACK_EXTRA && m_reject)
|
||||||
|
{
|
||||||
|
s += L"-";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getNbOld() > 0)
|
||||||
{
|
{
|
||||||
for (it = m_oldTiles.begin(); it != m_oldTiles.end(); it++)
|
for (it = m_oldTiles.begin(); it != m_oldTiles.end(); it++)
|
||||||
s += it->toChar();
|
s += it->toChar();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mode > RACK_SIMPLE && nOld() > 0 && nNew() > 0)
|
if (mode > RACK_SIMPLE && getNbOld() > 0 && getNbNew() > 0)
|
||||||
{
|
{
|
||||||
s += L"+";
|
s += L"+";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mode > RACK_EXTRA && reject)
|
if (getNbNew() > 0)
|
||||||
{
|
|
||||||
s += L"-";
|
|
||||||
// new rack: reject
|
|
||||||
// not after a scrabble
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nNew() > 0)
|
|
||||||
{
|
{
|
||||||
for (it = m_newTiles.begin(); it != m_newTiles.end(); it++)
|
for (it = m_newTiles.begin(); it != m_newTiles.end(); it++)
|
||||||
s += it->toChar();
|
s += it->toChar();
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -47,7 +48,6 @@ class PlayedRack
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
PlayedRack();
|
PlayedRack();
|
||||||
virtual ~PlayedRack() {}
|
|
||||||
|
|
||||||
void reset();
|
void reset();
|
||||||
void resetNew();
|
void resetNew();
|
||||||
|
@ -58,11 +58,12 @@ public:
|
||||||
|
|
||||||
void setOld(const Rack &iRack);
|
void setOld(const Rack &iRack);
|
||||||
void setNew(const Rack &iRack);
|
void setNew(const Rack &iRack);
|
||||||
int setManual(const wstring& iLetters);
|
void setManual(const wstring& iLetters);
|
||||||
|
void setReject(bool iReject = true) { m_reject = iReject; }
|
||||||
|
|
||||||
int nTiles() const { return nNew() + nOld(); }
|
unsigned int getNbTiles() const { return getNbNew() + getNbOld(); }
|
||||||
int nNew() const { return m_newTiles.size(); }
|
unsigned int getNbNew() const { return m_newTiles.size(); }
|
||||||
int nOld() const { return m_oldTiles.size(); }
|
unsigned int getNbOld() const { return m_oldTiles.size(); }
|
||||||
|
|
||||||
void addNew(const Tile &t);
|
void addNew(const Tile &t);
|
||||||
void addOld(const Tile &t);
|
void addOld(const Tile &t);
|
||||||
|
@ -70,9 +71,10 @@ public:
|
||||||
void getOldTiles(vector<Tile> &oTiles) const;
|
void getOldTiles(vector<Tile> &oTiles) const;
|
||||||
void getAllTiles(vector<Tile> &oTiles) const;
|
void getAllTiles(vector<Tile> &oTiles) const;
|
||||||
|
|
||||||
bool checkRack(int cMin, int vMin) const;
|
bool checkRack(unsigned int cMin, unsigned int vMin) const;
|
||||||
|
|
||||||
void operator=(const PlayedRack &iOther);
|
/// Randomly change the order of the "new" tiles
|
||||||
|
void shuffleNew();
|
||||||
|
|
||||||
enum display_mode
|
enum display_mode
|
||||||
{
|
{
|
||||||
|
@ -83,7 +85,7 @@ public:
|
||||||
wstring toString(display_mode iShowExtraSigns = RACK_EXTRA) const;
|
wstring toString(display_mode iShowExtraSigns = RACK_EXTRA) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool reject;
|
bool m_reject;
|
||||||
vector<Tile> m_oldTiles;
|
vector<Tile> m_oldTiles;
|
||||||
vector<Tile> m_newTiles;
|
vector<Tile> m_newTiles;
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -22,23 +23,21 @@
|
||||||
* \file rack.cpp
|
* \file rack.cpp
|
||||||
* \brief Rack class : multiset of tiles
|
* \brief Rack class : multiset of tiles
|
||||||
* \author Antoine Fraboulet & Olivier Teuliere
|
* \author Antoine Fraboulet & Olivier Teuliere
|
||||||
* \date 2002 - 2005
|
* \date 2002 - 2007
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
|
#include "dic.h"
|
||||||
#include "encoding.h"
|
#include "encoding.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
// FIXME: should not be here (duplicated from tile.cpp)
|
|
||||||
#define TILES_NUMBER 28
|
|
||||||
#define MIN_CODE 1
|
|
||||||
|
|
||||||
|
|
||||||
Rack::Rack()
|
Rack::Rack()
|
||||||
: m_tiles(TILES_NUMBER, 0), m_ntiles(0)
|
: m_tiles(Dictionary::GetDic().getTileNumber() + 1, 0), m_ntiles(0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Rack::remove(const Tile &t)
|
void Rack::remove(const Tile &t)
|
||||||
{
|
{
|
||||||
ASSERT(in(t),
|
ASSERT(in(t),
|
||||||
|
@ -58,14 +57,13 @@ void Rack::clear()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Rack::getTiles(list<Tile> &oTiles) const
|
void Rack::getTiles(vector<Tile> &oTiles) const
|
||||||
{
|
{
|
||||||
for (unsigned int i = MIN_CODE; i < m_tiles.size(); i++)
|
oTiles.reserve(m_ntiles);
|
||||||
|
for (unsigned int i = 1; i < m_tiles.size(); i++)
|
||||||
{
|
{
|
||||||
for (unsigned int j = 0; j < m_tiles[i]; j++)
|
// Add m_tiles[i] copies of the tile at the end of the vector
|
||||||
{
|
oTiles.insert(oTiles.end(), m_tiles[i], Dictionary::GetDic().getTileFromCode(i));
|
||||||
oTiles.push_back(Tile::GetTileFromCode(i));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -73,12 +71,10 @@ void Rack::getTiles(list<Tile> &oTiles) const
|
||||||
wstring Rack::toString()
|
wstring Rack::toString()
|
||||||
{
|
{
|
||||||
wstring rs;
|
wstring rs;
|
||||||
for (unsigned int i = MIN_CODE; i < m_tiles.size(); i++)
|
for (unsigned int i = 1; i < m_tiles.size(); i++)
|
||||||
{
|
{
|
||||||
for (unsigned int j = 0; j < m_tiles[i]; j++)
|
// Append m_tiles[i] copies of the char
|
||||||
{
|
rs.append(m_tiles[i], Dictionary::GetDic().getTileFromCode(i).toChar());
|
||||||
rs += Tile::GetTileFromCode(i).toChar();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return rs;
|
return rs;
|
||||||
}
|
}
|
||||||
|
|
24
game/rack.h
24
game/rack.h
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2002-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -22,17 +23,17 @@
|
||||||
* \file rack.h
|
* \file rack.h
|
||||||
* \brief Rack class : multiset of tiles
|
* \brief Rack class : multiset of tiles
|
||||||
* \author Antoine Fraboulet & Olivier Teuliere
|
* \author Antoine Fraboulet & Olivier Teuliere
|
||||||
* \date 2002 - 2005
|
* \date 2002 - 2007
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _RACK_H_
|
#ifndef _RACK_H_
|
||||||
#define _RACK_H_
|
#define _RACK_H_
|
||||||
|
|
||||||
#include "tile.h"
|
#include <vector>
|
||||||
#include <set>
|
|
||||||
#include <list>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
#include "tile.h"
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
@ -44,23 +45,22 @@ class Rack
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Rack();
|
Rack();
|
||||||
virtual ~Rack() {}
|
|
||||||
|
|
||||||
int nTiles() const { return m_ntiles; }
|
unsigned int getNbTiles() const { return m_ntiles; }
|
||||||
bool isEmpty() const { return nTiles() == 0; }
|
bool isEmpty() const { return getNbTiles() == 0; }
|
||||||
|
|
||||||
unsigned int in(const Tile &t) const { return m_tiles[t.toCode()]; }
|
unsigned int in(const Tile &t) const { return m_tiles[t.toCode()]; }
|
||||||
void add(const Tile &t) { m_tiles[t.toCode()]++; m_ntiles++; }
|
void add(const Tile &t) { m_tiles[t.toCode()]++; m_ntiles++; }
|
||||||
void remove(const Tile &t);
|
void remove(const Tile &t);
|
||||||
void clear();
|
void clear();
|
||||||
void getTiles(list<Tile> &oTiles) const;
|
void getTiles(vector<Tile> &oTiles) const;
|
||||||
|
|
||||||
wstring toString();
|
wstring toString();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// Vector indexed by tile codes, containing the number of tiles
|
/// Vector indexed by tile codes, containing the number of tiles
|
||||||
vector<unsigned int> m_tiles;
|
vector<unsigned int> m_tiles;
|
||||||
int m_ntiles;
|
unsigned int m_ntiles;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -46,19 +47,19 @@ struct less_points : public binary_function<const Round&,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const Round & Results::get(int i) const
|
const Round & Results::get(unsigned int i) const
|
||||||
{
|
{
|
||||||
ASSERT(0 <= i && i < size(), "Results index out of bounds");
|
ASSERT(i < size(), "Results index out of bounds");
|
||||||
return m_rounds[i];
|
return m_rounds[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Results::search(const Dictionary &iDic, Board &iBoard,
|
void Results::search(const Dictionary &iDic, Board &iBoard,
|
||||||
const Rack &iRack, int iTurn)
|
const Rack &iRack, bool iFirstWord)
|
||||||
{
|
{
|
||||||
clear();
|
clear();
|
||||||
|
|
||||||
if (iTurn == 0)
|
if (iFirstWord)
|
||||||
{
|
{
|
||||||
iBoard.searchFirst(iDic, iRack, *this);
|
iBoard.searchFirst(iDic, iRack, *this);
|
||||||
}
|
}
|
||||||
|
@ -67,11 +68,11 @@ void Results::search(const Dictionary &iDic, Board &iBoard,
|
||||||
iBoard.search(iDic, iRack, *this);
|
iBoard.search(iDic, iRack, *this);
|
||||||
}
|
}
|
||||||
|
|
||||||
sort_by_points();
|
sortByPoints();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Results::sort_by_points()
|
void Results::sortByPoints()
|
||||||
{
|
{
|
||||||
less_points lp;
|
less_points lp;
|
||||||
std::sort(m_rounds.begin(), m_rounds.end(), lp);
|
std::sort(m_rounds.begin(), m_rounds.end(), lp);
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -33,9 +34,9 @@
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
class Dictionary;
|
||||||
class Board;
|
class Board;
|
||||||
class Rack;
|
class Rack;
|
||||||
typedef struct _Dictionary * Dictionary;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -47,24 +48,22 @@ typedef struct _Dictionary * Dictionary;
|
||||||
class Results
|
class Results
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Results() {}
|
unsigned int size() const { return m_rounds.size(); }
|
||||||
virtual ~Results() {}
|
|
||||||
|
|
||||||
int size() const { return m_rounds.size(); }
|
|
||||||
void clear() { m_rounds.clear(); }
|
void clear() { m_rounds.clear(); }
|
||||||
const Round & get(int) const;
|
const Round & get(unsigned int) const;
|
||||||
|
|
||||||
// Perform a search on the board
|
/// Perform a search on the board
|
||||||
void search(const Dictionary &iDic, Board &iBoard,
|
void search(const Dictionary &iDic, Board &iBoard,
|
||||||
const Rack &iRack, int iTurn);
|
const Rack &iRack, bool iFirstWord);
|
||||||
|
|
||||||
// FIXME: These methods are used to fill the container with the rounds,
|
// FIXME: This method is used to fill the container with the rounds,
|
||||||
// but they should not be part of the public interface
|
// but it should not be part of the public interface
|
||||||
void add(const Round &iRound) { m_rounds.push_back(iRound); }
|
void add(const Round &iRound) { m_rounds.push_back(iRound); }
|
||||||
|
|
||||||
void sort_by_points();
|
|
||||||
private:
|
private:
|
||||||
vector<Round> m_rounds;
|
vector<Round> m_rounds;
|
||||||
|
|
||||||
|
void sortByPoints();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -52,33 +53,27 @@ void Round::init()
|
||||||
|
|
||||||
void Round::setWord(const vector<Tile> &iTiles)
|
void Round::setWord(const vector<Tile> &iTiles)
|
||||||
{
|
{
|
||||||
m_word.clear();
|
m_word = iTiles;
|
||||||
|
// XXX: always from rack?
|
||||||
vector<Tile>::const_iterator it;
|
m_tileOrigin = vector<char>(iTiles.size(), FROMRACK);
|
||||||
for (it = iTiles.begin(); it != iTiles.end(); it++)
|
|
||||||
{
|
|
||||||
m_word.push_back(*it);
|
|
||||||
// XXX: always from rack?
|
|
||||||
m_tileOrigin.push_back(FROMRACK);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Round::setFromRack(int iIndex)
|
void Round::setFromRack(unsigned int iIndex)
|
||||||
{
|
{
|
||||||
m_tileOrigin[iIndex] &= ~FROMBOARD;
|
m_tileOrigin[iIndex] &= ~FROMBOARD;
|
||||||
m_tileOrigin[iIndex] |= FROMRACK;
|
m_tileOrigin[iIndex] |= FROMRACK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Round::setFromBoard(int iIndex)
|
void Round::setFromBoard(unsigned int iIndex)
|
||||||
{
|
{
|
||||||
m_tileOrigin[iIndex] &= ~FROMRACK;
|
m_tileOrigin[iIndex] &= ~FROMRACK;
|
||||||
m_tileOrigin[iIndex] |= FROMBOARD;
|
m_tileOrigin[iIndex] |= FROMBOARD;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Round::setJoker(int iIndex, bool value)
|
void Round::setJoker(unsigned int iIndex, bool value)
|
||||||
{
|
{
|
||||||
if (value)
|
if (value)
|
||||||
m_tileOrigin[iIndex] |= JOKER;
|
m_tileOrigin[iIndex] |= JOKER;
|
||||||
|
@ -87,25 +82,19 @@ void Round::setJoker(int iIndex, bool value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool Round::isJoker(int iIndex) const
|
bool Round::isJoker(unsigned int iIndex) const
|
||||||
{
|
{
|
||||||
return m_tileOrigin[iIndex] & JOKER;
|
return m_tileOrigin[iIndex] & JOKER;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const Tile& Round::getTile(int iIndex) const
|
const Tile& Round::getTile(unsigned int iIndex) const
|
||||||
{
|
{
|
||||||
return m_word[iIndex];
|
return m_word[iIndex];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Round::getWordLen() const
|
bool Round::isPlayedFromRack(unsigned int iIndex) const
|
||||||
{
|
|
||||||
return m_word.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool Round::isPlayedFromRack(int iIndex) const
|
|
||||||
{
|
{
|
||||||
return m_tileOrigin[iIndex] & FROMRACK;
|
return m_tileOrigin[iIndex] & FROMRACK;
|
||||||
}
|
}
|
||||||
|
@ -150,7 +139,7 @@ wstring Round::getWord() const
|
||||||
wchar_t c;
|
wchar_t c;
|
||||||
wstring s;
|
wstring s;
|
||||||
|
|
||||||
for (int i = 0; i < getWordLen(); i++)
|
for (unsigned int i = 0; i < getWordLen(); i++)
|
||||||
{
|
{
|
||||||
c = getTile(i).toChar();
|
c = getTile(i).toChar();
|
||||||
if (isJoker(i))
|
if (isJoker(i))
|
||||||
|
|
28
game/round.h
28
game/round.h
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -42,7 +43,6 @@ public:
|
||||||
*
|
*
|
||||||
*************************/
|
*************************/
|
||||||
Round();
|
Round();
|
||||||
virtual ~Round() {}
|
|
||||||
void init();
|
void init();
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
|
@ -58,23 +58,23 @@ public:
|
||||||
*************************/
|
*************************/
|
||||||
void setPoints(int iPoints) { m_points = iPoints; }
|
void setPoints(int iPoints) { m_points = iPoints; }
|
||||||
void setBonus(bool iBonus) { m_bonus = iBonus; }
|
void setBonus(bool iBonus) { m_bonus = iBonus; }
|
||||||
void setTile(int iIndex, const Tile &iTile) { m_word[iIndex] = iTile; }
|
void setTile(unsigned int iIndex, const Tile &iTile) { m_word[iIndex] = iTile; }
|
||||||
void setWord(const vector<Tile> &iTiles);
|
void setWord(const vector<Tile> &iTiles);
|
||||||
void setFromRack(int iIndex);
|
void setFromRack(unsigned int iIndex);
|
||||||
void setFromBoard(int iIndex);
|
void setFromBoard(unsigned int iIndex);
|
||||||
void setJoker(int iIndex, bool value = true);
|
void setJoker(unsigned int iIndex, bool value = true);
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
* General getters
|
* General getters
|
||||||
*************************/
|
*************************/
|
||||||
bool isJoker (int iIndex) const;
|
bool isJoker (unsigned int iIndex) const;
|
||||||
bool isPlayedFromRack(int iIndex) const;
|
bool isPlayedFromRack(unsigned int iIndex) const;
|
||||||
const Tile& getTile (int iIndex) const;
|
const Tile& getTile (unsigned int iIndex) const;
|
||||||
|
|
||||||
wstring getWord() const;
|
wstring getWord() const;
|
||||||
int getWordLen() const;
|
unsigned int getWordLen() const { return m_word.size(); }
|
||||||
int getPoints() const { return m_points; }
|
int getPoints() const { return m_points; }
|
||||||
int getBonus() const { return m_bonus; }
|
int getBonus() const { return m_bonus; }
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
* Coordinates
|
* Coordinates
|
||||||
|
|
134
game/settings.cpp
Normal file
134
game/settings.cpp
Normal file
|
@ -0,0 +1,134 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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 <cstdlib>
|
||||||
|
|
||||||
|
#include "settings.h"
|
||||||
|
|
||||||
|
|
||||||
|
Settings *Settings::m_instance = NULL;
|
||||||
|
|
||||||
|
|
||||||
|
Settings & Settings::Instance()
|
||||||
|
{
|
||||||
|
if (m_instance == NULL)
|
||||||
|
{
|
||||||
|
m_instance = new Settings;
|
||||||
|
}
|
||||||
|
return *m_instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Settings::Destroy()
|
||||||
|
{
|
||||||
|
delete m_instance;
|
||||||
|
m_instance = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Settings::Settings()
|
||||||
|
{
|
||||||
|
// ============== General options ==============
|
||||||
|
|
||||||
|
|
||||||
|
// ============== Training mode options ==============
|
||||||
|
|
||||||
|
|
||||||
|
// ============== Duplicate mode options ==============
|
||||||
|
|
||||||
|
// Minimum number of players in a duplicate game needed to apply a "solo" bonus
|
||||||
|
// (16 is the ODS value)
|
||||||
|
m_intHandler.addOption("duplicate-solo-players", 16);
|
||||||
|
// Number of points granted for a solo (10 is the ODS value)
|
||||||
|
m_intHandler.addOption("duplicate-solo-value", 10);
|
||||||
|
|
||||||
|
// If true, Eliot complains when the player does something illegal
|
||||||
|
// If false, the word is accepted (with a score of 0) and the player does
|
||||||
|
// not get a second chance
|
||||||
|
m_boolHandler.addOption("duplicate-reject-invalid", false);
|
||||||
|
|
||||||
|
|
||||||
|
// ============== Freegame mode options ==============
|
||||||
|
|
||||||
|
// If true, Eliot complains when the player does something illegal
|
||||||
|
// If false, the word is accepted (with a score of 0) and the player does
|
||||||
|
// not get a second chance.
|
||||||
|
// Trying to change letters or to pass the turn in an incorrect way will
|
||||||
|
// be rejected in any case.
|
||||||
|
m_boolHandler.addOption("freegame-reject-invalid", false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Settings::setBool(const string &iName, bool iValue)
|
||||||
|
{
|
||||||
|
m_boolHandler.setOption(iName, iValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool Settings::getBool(const string &iName) const
|
||||||
|
{
|
||||||
|
return m_boolHandler.getOption(iName);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Settings::setInt(const string &iName, int iValue)
|
||||||
|
{
|
||||||
|
m_intHandler.setOption(iName, iValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int Settings::getInt(const string &iName) const
|
||||||
|
{
|
||||||
|
return m_intHandler.getOption(iName);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void Settings::OptionsHandler<T>::addOption(const string &iName, const T &iValue)
|
||||||
|
{
|
||||||
|
m_options[iName] = iValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void Settings::OptionsHandler<T>::setOption(const string &iName, const T &iValue)
|
||||||
|
{
|
||||||
|
typename map<string, T>::iterator it = m_options.find(iName);
|
||||||
|
if (it == m_options.end())
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
it->second = iValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
const T& Settings::OptionsHandler<T>::getOption(const string &iName) const
|
||||||
|
{
|
||||||
|
typename map<string, T>::const_iterator it = m_options.find(iName);
|
||||||
|
if (it == m_options.end())
|
||||||
|
{
|
||||||
|
// FIXME: throw an exception object instead
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
return it->second;
|
||||||
|
}
|
||||||
|
|
109
game/settings.h
Normal file
109
game/settings.h
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* Eliot
|
||||||
|
* Copyright (C) 2007 Olivier Teulière
|
||||||
|
* Authors: 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
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef _SETTINGS_H_
|
||||||
|
#define _SETTINGS_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
using std::string;
|
||||||
|
using std::map;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class centralizes the various configuration options of Eliot.
|
||||||
|
* It implements the Singleton pattern.
|
||||||
|
*
|
||||||
|
* Currently, there are few settings, and their initial value is hard-coded.
|
||||||
|
* In a later phase, this class will be able to export/import settings
|
||||||
|
* to/from a configuration file, and it should be possible to override
|
||||||
|
* configuration settings with settings given on the command-line (TODO).
|
||||||
|
* The boost::program_options library could be useful for this.
|
||||||
|
*
|
||||||
|
* This class will also be helpful for the "Settings" dialog box of the GUI.
|
||||||
|
*/
|
||||||
|
class Settings
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// Access to the singleton
|
||||||
|
static Settings& Instance();
|
||||||
|
/// Destroy the singleton cleanly
|
||||||
|
static void Destroy();
|
||||||
|
|
||||||
|
void setBool(const string &iName, bool iValue);
|
||||||
|
bool getBool(const string &iName) const;
|
||||||
|
|
||||||
|
void setInt(const string &iName, int iValue);
|
||||||
|
int getInt(const string &iName) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This nested class is simply there to handle storage and retrieval
|
||||||
|
* for options of a particular type (and factorize code)
|
||||||
|
*/
|
||||||
|
template <typename T>
|
||||||
|
class OptionsHandler
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// Set the value of an option
|
||||||
|
/**
|
||||||
|
* If the option already exists, its value is replaced,
|
||||||
|
* otherwise the option is created
|
||||||
|
*/
|
||||||
|
void addOption(const string &iName, const T &iValue);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Change the value of an existing option.
|
||||||
|
* An exception is thrown if the option doesn't exist yet
|
||||||
|
*/
|
||||||
|
void setOption(const string &iName, const T &iValue);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Query the value of an option.
|
||||||
|
* An exception is thrown if the option doesn't exist
|
||||||
|
*/
|
||||||
|
const T& getOption(const string &iName) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
map<string, T> m_options;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/// Singleton instance
|
||||||
|
static Settings *m_instance;
|
||||||
|
Settings();
|
||||||
|
|
||||||
|
/// The settings can be of various types
|
||||||
|
OptionsHandler<bool> m_boolHandler;
|
||||||
|
OptionsHandler<int> m_intHandler;
|
||||||
|
// Add types as needed...
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/// Local Variables:
|
||||||
|
/// mode: c++
|
||||||
|
/// mode: hs-minor
|
||||||
|
/// c-basic-offset: 4
|
||||||
|
/// indent-tabs-mode: nil
|
||||||
|
/// End:
|
233
game/tile.cpp
233
game/tile.cpp
|
@ -1,233 +0,0 @@
|
||||||
/*****************************************************************************
|
|
||||||
* Copyright (C) 1999-2005 Eliot
|
|
||||||
* Authors: Olivier Teuliere <ipkiss@via.ecp.fr>
|
|
||||||
*
|
|
||||||
* 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 "tile.h"
|
|
||||||
#include <wctype.h>
|
|
||||||
|
|
||||||
|
|
||||||
/*************************
|
|
||||||
* French tiles
|
|
||||||
* Zero + 26 letters + joker
|
|
||||||
* tiles ares supposed to be contiguous and joker is separated
|
|
||||||
*************************/
|
|
||||||
|
|
||||||
#define TILE_START 'A'
|
|
||||||
#define TILE_END 'Z'
|
|
||||||
#define TILE_JOKER '?'
|
|
||||||
#define TILE_DUMMY '%'
|
|
||||||
|
|
||||||
#define TILE_IDX_DUMMY 0
|
|
||||||
#define TILE_IDX_START 1
|
|
||||||
#define TILE_IDX_END 26
|
|
||||||
#define TILE_IDX_JOKER 27
|
|
||||||
|
|
||||||
#define TILES_NUMBER 28
|
|
||||||
|
|
||||||
/* The jokers and the 'Y' can be considered both as vowels or consonants */
|
|
||||||
const unsigned int Tiles_vowels[TILES_NUMBER] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ? */
|
|
||||||
0,1,0,0,0, 1,0,0,0,1,0, 0,0,0,0,1,0,0,0,0,0,1,0, 0, 0, 1, 0,1
|
|
||||||
};
|
|
||||||
|
|
||||||
const unsigned int Tiles_consonants[TILES_NUMBER] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ? */
|
|
||||||
0,0,1,1,1, 0,1,1,1,0,1, 1,1,1,1,0,1,1,1,1,1,0,1, 1, 1, 1, 1,1
|
|
||||||
};
|
|
||||||
|
|
||||||
const unsigned int Tiles_numbers[TILES_NUMBER] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ? */
|
|
||||||
0,9,2,2,3,15,2,2,2,8,1, 1,5,3,6,6,2,1,6,6,6,6,2, 1, 1, 1, 1,2
|
|
||||||
};
|
|
||||||
|
|
||||||
const unsigned int Tiles_points[TILES_NUMBER] =
|
|
||||||
{
|
|
||||||
/* x A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ? */
|
|
||||||
0,1,3,3,2, 1,4,2,4,1,8,10,1,2,1,1,3,8,1,1,1,1,4,10,10,10,10,0
|
|
||||||
};
|
|
||||||
|
|
||||||
/***************************
|
|
||||||
***************************/
|
|
||||||
|
|
||||||
const Tile Tile::m_TheJoker(TILE_JOKER);
|
|
||||||
const Tile Tile::m_TheDummy(0);
|
|
||||||
list<Tile> Tile::m_tilesList;
|
|
||||||
vector<Tile> Tile::m_tilesVect(TILES_NUMBER, Tile::dummy());
|
|
||||||
bool Tile::m_vectInitialized(false);
|
|
||||||
|
|
||||||
|
|
||||||
Tile::Tile(wchar_t c)
|
|
||||||
{
|
|
||||||
if (c == TILE_JOKER)
|
|
||||||
{
|
|
||||||
m_joker = true;
|
|
||||||
m_dummy = false;
|
|
||||||
m_char = TILE_JOKER;
|
|
||||||
m_code = 27;
|
|
||||||
}
|
|
||||||
else if (isalpha(c))
|
|
||||||
{
|
|
||||||
m_joker = islower(c);
|
|
||||||
m_dummy = false;
|
|
||||||
m_char = towupper(c);
|
|
||||||
m_code = m_char - 'A' + 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
m_joker = false;
|
|
||||||
m_dummy = true;
|
|
||||||
m_char = 0;
|
|
||||||
m_code = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool Tile::isVowel() const
|
|
||||||
{
|
|
||||||
if (m_dummy)
|
|
||||||
return false;
|
|
||||||
if (m_joker)
|
|
||||||
return Tiles_vowels[TILE_IDX_JOKER];
|
|
||||||
return Tiles_vowels[TILE_IDX_START + m_char - TILE_START];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool Tile::isConsonant() const
|
|
||||||
{
|
|
||||||
if (m_dummy)
|
|
||||||
return false;
|
|
||||||
if (m_joker)
|
|
||||||
return Tiles_consonants[TILE_IDX_JOKER];
|
|
||||||
return Tiles_consonants[TILE_IDX_START + m_char - TILE_START];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
unsigned int Tile::maxNumber() const
|
|
||||||
{
|
|
||||||
if (m_dummy)
|
|
||||||
return false;
|
|
||||||
if (m_joker)
|
|
||||||
return Tiles_numbers[TILE_IDX_JOKER];
|
|
||||||
return Tiles_numbers[TILE_IDX_START + m_char - TILE_START];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
unsigned int Tile::getPoints() const
|
|
||||||
{
|
|
||||||
if (m_dummy)
|
|
||||||
return false;
|
|
||||||
if (m_joker)
|
|
||||||
return Tiles_points[TILE_IDX_JOKER];
|
|
||||||
return Tiles_points[TILE_IDX_START + m_char - TILE_START];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
const list<Tile>& Tile::getAllTiles()
|
|
||||||
{
|
|
||||||
if (Tile::m_tilesList.size() == 0)
|
|
||||||
{
|
|
||||||
// XXX: this should be filled from a "language file" instead
|
|
||||||
for (char i = TILE_START; i <= TILE_END; i++)
|
|
||||||
Tile::m_tilesList.push_back(Tile(i));
|
|
||||||
m_tilesList.push_back(Tile(TILE_JOKER));
|
|
||||||
}
|
|
||||||
return Tile::m_tilesList;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
const Tile& Tile::GetTileFromCode(unsigned int iCode)
|
|
||||||
{
|
|
||||||
if (!m_vectInitialized)
|
|
||||||
{
|
|
||||||
// XXX: this should be filled from a "language file" instead
|
|
||||||
for (char i = TILE_IDX_START; i <= TILE_IDX_END; i++)
|
|
||||||
Tile::m_tilesVect[i] = Tile(i + 'A' - TILE_IDX_START);
|
|
||||||
m_tilesVect[TILE_IDX_JOKER] = Tile::Joker();
|
|
||||||
m_vectInitialized = true;
|
|
||||||
}
|
|
||||||
return Tile::m_tilesVect[iCode];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
wchar_t Tile::toChar() const
|
|
||||||
{
|
|
||||||
if (m_dummy)
|
|
||||||
return TILE_DUMMY;
|
|
||||||
if (m_joker)
|
|
||||||
{
|
|
||||||
if (iswalpha(m_char))
|
|
||||||
return towlower(m_char);
|
|
||||||
else
|
|
||||||
return TILE_JOKER;
|
|
||||||
}
|
|
||||||
return m_char;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned int Tile::toCode() const
|
|
||||||
{
|
|
||||||
return m_code;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Tile::operator <(const Tile &iOther) const
|
|
||||||
{
|
|
||||||
if (iOther.m_dummy)
|
|
||||||
return false;
|
|
||||||
else if (m_dummy)
|
|
||||||
return true;
|
|
||||||
else if (m_joker)
|
|
||||||
return false;
|
|
||||||
else if (iOther.m_joker)
|
|
||||||
return true;
|
|
||||||
else
|
|
||||||
return m_char < iOther.m_char;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool Tile::operator ==(const Tile &iOther) const
|
|
||||||
{
|
|
||||||
if (m_dummy || iOther.m_dummy)
|
|
||||||
return m_dummy == iOther.m_dummy;
|
|
||||||
if (m_joker || iOther.m_joker)
|
|
||||||
{
|
|
||||||
if (m_joker != iOther.m_joker)
|
|
||||||
return false;
|
|
||||||
return m_char == iOther.m_char;
|
|
||||||
}
|
|
||||||
return m_char == iOther.m_char;
|
|
||||||
// return (m_joker && iOther.m_joker && m_char == iOther.m_char) ||
|
|
||||||
// (m_dummy && iOther.m_dummy) ||
|
|
||||||
// (!m_dummy && !iOther.m_dummy
|
|
||||||
// && !m_joker && !iOther.m_joker
|
|
||||||
// && m_char == iOther.m_char);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool Tile::operator !=(const Tile &iOther) const
|
|
||||||
{
|
|
||||||
return !(*this == iOther);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Local Variables:
|
|
||||||
/// mode: c++
|
|
||||||
/// mode: hs-minor
|
|
||||||
/// c-basic-offset: 4
|
|
||||||
/// indent-tabs-mode: nil
|
|
||||||
/// End:
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -18,10 +19,13 @@
|
||||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
#include "dic.h"
|
#include "dic.h"
|
||||||
#include "tile.h"
|
#include "tile.h"
|
||||||
#include "rack.h"
|
#include "rack.h"
|
||||||
#include "round.h"
|
#include "round.h"
|
||||||
|
#include "move.h"
|
||||||
#include "pldrack.h"
|
#include "pldrack.h"
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
#include "training.h"
|
#include "training.h"
|
||||||
|
@ -30,12 +34,8 @@
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
|
||||||
Training::Training(const Dictionary &iDic): Game(iDic)
|
Training::Training(const Dictionary &iDic)
|
||||||
{
|
: Game(iDic)
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Training::~Training()
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -46,11 +46,10 @@ int Training::setRackRandom(bool iCheck, set_rack_mode mode)
|
||||||
|
|
||||||
int res;
|
int res;
|
||||||
int try_number = 0;
|
int try_number = 0;
|
||||||
int p = m_currPlayer;
|
|
||||||
m_results.clear();
|
m_results.clear();
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
res = helperSetRackRandom(p, iCheck, mode);
|
res = helperSetRackRandomOld(m_currPlayer, iCheck, mode);
|
||||||
try_number ++;
|
try_number ++;
|
||||||
} while (res == 2 && try_number < MAX_RANDOM_TRY);
|
} while (res == 2 && try_number < MAX_RANDOM_TRY);
|
||||||
// 0 : ok
|
// 0 : ok
|
||||||
|
@ -59,28 +58,27 @@ int Training::setRackRandom(bool iCheck, set_rack_mode mode)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Training::setRackManual(bool iCheck, const wstring &iLetters)
|
int Training::setRackManual(bool iCheck, const wstring &iLetters)
|
||||||
{
|
{
|
||||||
int res;
|
// Letters can be lowercase or uppercase as they are
|
||||||
int p = m_currPlayer;
|
|
||||||
wstring::iterator it;
|
|
||||||
wstring uLetters; // uppercase letters
|
|
||||||
// letters can be lowercase or uppercase as they are
|
|
||||||
// coming from user input. We do not consider a lowercase
|
// coming from user input. We do not consider a lowercase
|
||||||
// letter to be a joker which has been assigned to a letter.
|
// letter to be a joker which has been assigned to a letter.
|
||||||
m_results.clear();
|
// As a result, we simply make all the letters uppercase
|
||||||
uLetters = iLetters;
|
wstring upperLetters = iLetters;
|
||||||
for (it = uLetters.begin(); it != uLetters.end(); it ++)
|
std::transform(upperLetters.begin(), upperLetters.end(),
|
||||||
{
|
upperLetters.begin(), towupper);
|
||||||
*it = towupper(*it);
|
int res = helperSetRackManual(m_currPlayer, iCheck, upperLetters);
|
||||||
}
|
// 0: ok
|
||||||
res = helperSetRackManual(p, iCheck, uLetters);
|
// 1: not enough tiles
|
||||||
// 0 : ok
|
// 2: check failed (number of vowels before round 15)
|
||||||
// 1 : not enough tiles
|
// 3: letters not in the dictionary
|
||||||
// 2 : check failed (number of voyels before round 15)
|
if (res == 0)
|
||||||
|
m_results.clear();
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Training::setRack(set_rack_mode iMode, bool iCheck, const wstring &iLetters)
|
int Training::setRack(set_rack_mode iMode, bool iCheck, const wstring &iLetters)
|
||||||
{
|
{
|
||||||
int res = 0;
|
int res = 0;
|
||||||
|
@ -99,9 +97,10 @@ int Training::setRack(set_rack_mode iMode, bool iCheck, const wstring &iLetters)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Training::play(const wstring &iCoord, const wstring &iWord)
|
int Training::play(const wstring &iCoord, const wstring &iWord)
|
||||||
{
|
{
|
||||||
/* Perform all the validity checks, and fill a round */
|
// Perform all the validity checks, and fill a round
|
||||||
Round round;
|
Round round;
|
||||||
|
|
||||||
int res = checkPlayedWord(iCoord, iWord, round);
|
int res = checkPlayedWord(iCoord, iWord, round);
|
||||||
|
@ -111,23 +110,21 @@ int Training::play(const wstring &iCoord, const wstring &iWord)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update the rack and the score of the current player */
|
|
||||||
debug("play: %s %s %d\n",
|
debug("play: %s %s %d\n",
|
||||||
convertToMb(round.getWord()).c_str(),
|
convertToMb(round.getWord()).c_str(),
|
||||||
convertToMb(round.getCoord().toString()).c_str(),
|
convertToMb(round.getCoord().toString()).c_str(),
|
||||||
round.getPoints());
|
round.getPoints());
|
||||||
|
|
||||||
m_players[m_currPlayer]->addPoints(round.getPoints());
|
Move move(round);
|
||||||
// see game.cpp::helperPlayRound():99 comment
|
// Update the rack and the score of the current player
|
||||||
m_players[m_currPlayer]->endTurn(round, m_history.getSize());
|
// Player::endTurn() must be called before Game::helperPlayMove().
|
||||||
|
// See the big comment in game.cpp, line 96
|
||||||
|
m_players[m_currPlayer]->endTurn(move, m_history.getSize());
|
||||||
|
|
||||||
/* Everything is OK, we can play the word */
|
// Everything is OK, we can play the word
|
||||||
if (helperPlayRound(round))
|
helperPlayMove(m_currPlayer, move);
|
||||||
{
|
|
||||||
debug("play: error during play\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Next turn */
|
// Next turn
|
||||||
endTurn();
|
endTurn();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -146,10 +143,9 @@ int Training::start()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Training::endTurn()
|
void Training::endTurn()
|
||||||
{
|
{
|
||||||
// Nothing to do?
|
// Nothing to do, but this method is kept for consistency with other modes
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -159,31 +155,27 @@ void Training::search()
|
||||||
Rack r;
|
Rack r;
|
||||||
m_players[m_currPlayer]->getCurrentRack().getRack(r);
|
m_players[m_currPlayer]->getCurrentRack().getRack(r);
|
||||||
debug("Training::search for %s\n", convertToMb(r.toString()).c_str());
|
debug("Training::search for %s\n", convertToMb(r.toString()).c_str());
|
||||||
m_results.search(*m_dic, m_board, r, m_history.getSize());
|
m_results.search(m_dic, m_board, r, m_history.beforeFirstRound());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int Training::playResult(int n)
|
int Training::playResult(unsigned int n)
|
||||||
{
|
{
|
||||||
Player *player = m_players[m_currPlayer];
|
|
||||||
if (n >= m_results.size())
|
if (n >= m_results.size())
|
||||||
return 2;
|
return 2;
|
||||||
const Round &round = m_results.get(n);
|
|
||||||
|
|
||||||
/* Update the rack and the score of the current player */
|
Move move(m_results.get(n));
|
||||||
player->addPoints(round.getPoints());
|
// Update the rack and the score of the current player
|
||||||
player->endTurn(round, m_history.getSize());
|
m_players[m_currPlayer]->endTurn(move, m_history.getSize());
|
||||||
|
|
||||||
int res = helperPlayRound(round);
|
// Update the game
|
||||||
|
helperPlayMove(m_currPlayer, move);
|
||||||
|
m_results.clear();
|
||||||
|
|
||||||
if (res == 0)
|
// Next turn
|
||||||
m_results.clear();
|
|
||||||
|
|
||||||
/* Next turn */
|
|
||||||
// XXX: Should it be done by the interface instead?
|
|
||||||
endTurn();
|
endTurn();
|
||||||
|
|
||||||
return res;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -201,9 +193,9 @@ void Training::addAIPlayer()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Training::testPlay(int num)
|
void Training::testPlay(unsigned int num)
|
||||||
{
|
{
|
||||||
ASSERT(0 <= num && num < m_results.size(), "Wrong result number");
|
ASSERT(num < m_results.size(), "Wrong result number");
|
||||||
m_testRound = m_results.get(num);
|
m_testRound = m_results.get(num);
|
||||||
m_board.testRound(m_results.get(num));
|
m_board.testRound(m_results.get(num));
|
||||||
}
|
}
|
||||||
|
@ -215,6 +207,7 @@ void Training::removeTestPlay()
|
||||||
m_testRound = Round();
|
m_testRound = Round();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
wstring Training::getTestPlayWord() const
|
wstring Training::getTestPlayWord() const
|
||||||
{
|
{
|
||||||
return m_testRound.getWord();
|
return m_testRound.getWord();
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Copyright (C) 1999-2005 Eliot
|
* Eliot
|
||||||
* Authors: Antoine Fraboulet <antoine.fraboulet@free.fr>
|
* Copyright (C) 1999-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
* Olivier Teuliere <ipkiss@via.ecp.fr>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
@ -24,6 +25,7 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#include "game.h"
|
#include "game.h"
|
||||||
|
#include "round.h"
|
||||||
#include "results.h"
|
#include "results.h"
|
||||||
|
|
||||||
using std::string;
|
using std::string;
|
||||||
|
@ -34,6 +36,7 @@ using std::wstring;
|
||||||
* This class handles the logic specific to a training game.
|
* This class handles the logic specific to a training game.
|
||||||
* As its name indicates, it is not a game in the literal meaning of the word,
|
* As its name indicates, it is not a game in the literal meaning of the word,
|
||||||
* in particular because the rack can be set at will.
|
* in particular because the rack can be set at will.
|
||||||
|
*
|
||||||
* Note: No player should be added to this game, a human player is added
|
* Note: No player should be added to this game, a human player is added
|
||||||
* automatically (in the start() method)
|
* automatically (in the start() method)
|
||||||
*/
|
*/
|
||||||
|
@ -48,10 +51,13 @@ public:
|
||||||
* Game handling
|
* Game handling
|
||||||
*************************/
|
*************************/
|
||||||
virtual int start();
|
virtual int start();
|
||||||
|
|
||||||
|
/// See description of Game::play()
|
||||||
virtual int play(const wstring &iCoord, const wstring &iWord);
|
virtual int play(const wstring &iCoord, const wstring &iWord);
|
||||||
virtual int endTurn();
|
|
||||||
void search();
|
void search();
|
||||||
int playResult(int);
|
const Results& getResults() const { return m_results; };
|
||||||
|
int playResult(unsigned int);
|
||||||
|
|
||||||
int setRackRandom(bool, set_rack_mode);
|
int setRackRandom(bool, set_rack_mode);
|
||||||
int setRackManual(bool iCheck, const wstring &iLetters);
|
int setRackManual(bool iCheck, const wstring &iLetters);
|
||||||
|
@ -59,7 +65,7 @@ public:
|
||||||
|
|
||||||
/*************************
|
/*************************
|
||||||
* Override the default behaviour of these methods, because in training
|
* Override the default behaviour of these methods, because in training
|
||||||
* we only want a human player
|
* mode we only want a human player
|
||||||
*************************/
|
*************************/
|
||||||
virtual void addHumanPlayer();
|
virtual void addHumanPlayer();
|
||||||
virtual void addAIPlayer();
|
virtual void addAIPlayer();
|
||||||
|
@ -68,23 +74,25 @@ public:
|
||||||
* Functions to access the current search results
|
* Functions to access the current search results
|
||||||
* The int parameter should be 0 <= int < getNResults
|
* The int parameter should be 0 <= int < getNResults
|
||||||
*************************/
|
*************************/
|
||||||
const Results& getResults() const { return m_results; };
|
|
||||||
|
|
||||||
/// Place a temporary word on the board for preview purpose
|
/// Place a temporary word on the board for preview purposes
|
||||||
void testPlay(int);
|
void testPlay(unsigned int);
|
||||||
/// Remove the temporary word(s)
|
/// Remove the temporary word
|
||||||
void removeTestPlay();
|
void removeTestPlay();
|
||||||
/// Get the temporary word
|
/// Get the temporary word
|
||||||
wstring getTestPlayWord() const;
|
wstring getTestPlayWord() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Private constructor and destructor to force using the GameFactory class
|
/// Private constructor and destructor to force using the GameFactory class
|
||||||
Training(const Dictionary &iDic);
|
Training(const Dictionary &iDic);
|
||||||
virtual ~Training();
|
|
||||||
|
|
||||||
// Search results, with all the possible rounds
|
void endTurn();
|
||||||
Round m_testRound;
|
|
||||||
|
/// Search results, with all the possible rounds
|
||||||
Results m_results;
|
Results m_results;
|
||||||
|
|
||||||
|
/// Round corresponding to the last test play (if any)
|
||||||
|
Round m_testRound;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* _TRAINING_H_ */
|
#endif /* _TRAINING_H_ */
|
||||||
|
|
|
@ -1,67 +1,57 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file turn.cpp
|
* \file turn.cpp
|
||||||
* \brief Game turn (= id + pldrack + round)
|
* \brief Game turn (= id + pldrack + move)
|
||||||
* \author Antoine Fraboulet
|
* \author Antoine Fraboulet
|
||||||
* \date 2005
|
* \date 2005
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include "pldrack.h"
|
|
||||||
#include "round.h"
|
|
||||||
#include "turn.h"
|
#include "turn.h"
|
||||||
|
|
||||||
|
|
||||||
|
// FIXME: move set to an invalid value. It would be better to get rid of this
|
||||||
|
// constructor completely
|
||||||
Turn::Turn()
|
Turn::Turn()
|
||||||
{
|
: m_num(0), m_playerId(0), m_move(L"", L"")
|
||||||
m_num = 0;
|
|
||||||
m_playerId = 0;
|
|
||||||
m_pldrack = PlayedRack();
|
|
||||||
m_round = Round();
|
|
||||||
}
|
|
||||||
|
|
||||||
Turn::Turn(int iNum, int iPlayerId,
|
|
||||||
const PlayedRack& iPldRack, const Round& iRound)
|
|
||||||
: m_num(iNum), m_playerId(iPlayerId), m_pldrack(iPldRack), m_round(iRound)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
#if 0
|
|
||||||
void Turn::operator=(const Turn &iOther)
|
Turn::Turn(unsigned int iNum, unsigned int iPlayerId,
|
||||||
|
const PlayedRack& iPldRack, const Move& iMove)
|
||||||
|
: m_num(iNum), m_playerId(iPlayerId), m_pldrack(iPldRack), m_move(iMove)
|
||||||
{
|
{
|
||||||
m_num = iOther.m_num;
|
|
||||||
m_playerId = iOther.m_playerId;
|
|
||||||
m_pldrack = iOther.m_pldrack;
|
|
||||||
m_round = iOther.m_round;
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
wstring Turn::toString(bool iShowExtraSigns) const
|
wstring Turn::toString(bool iShowExtraSigns) const
|
||||||
{
|
{
|
||||||
wstring rs = L"";
|
wstring rs;
|
||||||
if (iShowExtraSigns)
|
if (iShowExtraSigns)
|
||||||
{
|
{
|
||||||
// TODO
|
// TODO
|
||||||
}
|
}
|
||||||
rs = rs + m_pldrack.toString() + L" " + m_round.toString();
|
rs = rs + m_pldrack.toString() + L" " + m_move.toString();
|
||||||
return rs;
|
return rs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
84
game/turn.h
84
game/turn.h
|
@ -1,25 +1,27 @@
|
||||||
/* Eliot */
|
/*****************************************************************************
|
||||||
/* Copyright (C) 1999 Antoine Fraboulet */
|
* Eliot
|
||||||
/* */
|
* Copyright (C) 2005-2007 Antoine Fraboulet & Olivier Teulière
|
||||||
/* This file is part of Eliot. */
|
* Authors: Antoine Fraboulet <antoine.fraboulet @@ free.fr>
|
||||||
/* */
|
* Olivier Teulière <ipkiss @@ gmail.com>
|
||||||
/* Eliot is free software; you can redistribute it and/or modify */
|
*
|
||||||
/* it under the terms of the GNU General Public License as published by */
|
* This program is free software; you can redistribute it and/or modify
|
||||||
/* the Free Software Foundation; either version 2 of the License, or */
|
* it under the terms of the GNU General Public License as published by
|
||||||
/* (at your option) any later version. */
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
/* */
|
* (at your option) any later version.
|
||||||
/* Eliot is distributed in the hope that it will be useful, */
|
*
|
||||||
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
|
* This program is distributed in the hope that it will be useful,
|
||||||
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
/* GNU General Public License for more details. */
|
* 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 */
|
* You should have received a copy of the GNU General Public License
|
||||||
/* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file turn.h
|
* \file turn.h
|
||||||
* \brief Game turn (= id + pldrack + round)
|
* \brief Game turn (= id + pldrack + move)
|
||||||
* \author Antoine Fraboulet
|
* \author Antoine Fraboulet
|
||||||
* \date 2005
|
* \date 2005
|
||||||
*/
|
*/
|
||||||
|
@ -27,35 +29,45 @@
|
||||||
#ifndef _TURN_H
|
#ifndef _TURN_H
|
||||||
#define _TURN_H
|
#define _TURN_H
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include "pldrack.h"
|
||||||
|
#include "move.h"
|
||||||
|
|
||||||
|
using std::wstring;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A Turn is the information about one 'move' done by a player.
|
||||||
|
* It consists of the player who played, the rack, and the actual move.
|
||||||
|
* A turn also has an id (XXX: currently never read)
|
||||||
|
*
|
||||||
|
* This class has no logic, it is merely there to aggregate corresponding
|
||||||
|
* data.
|
||||||
|
*/
|
||||||
class Turn
|
class Turn
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Turn();
|
Turn();
|
||||||
Turn(int iNum, int iPlayerId,
|
Turn(unsigned int iNum, unsigned int iPlayerId,
|
||||||
const PlayedRack& iPldRack, const Round& iRound);
|
const PlayedRack& iPldRack, const Move& iMove);
|
||||||
virtual ~Turn() {};
|
|
||||||
|
|
||||||
void setNum(int iNum) { m_num = iNum; }
|
void setNum(unsigned int iNum) { m_num = iNum; }
|
||||||
void setPlayer(int iPlayerId) { m_playerId = iPlayerId; }
|
void setPlayer(unsigned int iPlayerId) { m_playerId = iPlayerId; }
|
||||||
void setPlayedRack(const PlayedRack& iPldRack) { m_pldrack = iPldRack; }
|
void setPlayedRack(const PlayedRack& iPldRack) { m_pldrack = iPldRack; }
|
||||||
void setRound(const Round& iRound) { m_round = iRound; }
|
void setMove(const Move& iMove) { m_move = iMove; }
|
||||||
|
|
||||||
int getNum() const { return m_num; }
|
unsigned int getNum() const { return m_num; }
|
||||||
int getPlayer() const { return m_playerId; }
|
unsigned int getPlayer() const { return m_playerId; }
|
||||||
const PlayedRack& getPlayedRack() const { return m_pldrack; }
|
const PlayedRack& getPlayedRack() const { return m_pldrack; }
|
||||||
const Round& getRound() const { return m_round; }
|
const Move& getMove() const { return m_move; }
|
||||||
|
|
||||||
#if 0
|
|
||||||
void operator=(const Turn &iOther);
|
|
||||||
#endif
|
|
||||||
wstring toString(bool iShowExtraSigns = false) const;
|
wstring toString(bool iShowExtraSigns = false) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int m_num;
|
unsigned int m_num;
|
||||||
int m_playerId;
|
unsigned int m_playerId;
|
||||||
PlayedRack m_pldrack;
|
PlayedRack m_pldrack;
|
||||||
Round m_round;
|
Move m_move;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,18 +1,30 @@
|
||||||
codeset.m4
|
codeset.m4
|
||||||
gettext.m4
|
gettext.m4
|
||||||
|
glibc2.m4
|
||||||
glibc21.m4
|
glibc21.m4
|
||||||
iconv.m4
|
iconv.m4
|
||||||
intdiv0.m4
|
intdiv0.m4
|
||||||
|
intl.m4
|
||||||
|
intldir.m4
|
||||||
|
intmax.m4
|
||||||
inttypes-pri.m4
|
inttypes-pri.m4
|
||||||
inttypes.m4
|
|
||||||
inttypes_h.m4
|
inttypes_h.m4
|
||||||
isc-posix.m4
|
|
||||||
lcmessage.m4
|
lcmessage.m4
|
||||||
lib-ld.m4
|
lib-ld.m4
|
||||||
lib-link.m4
|
lib-link.m4
|
||||||
lib-prefix.m4
|
lib-prefix.m4
|
||||||
|
lock.m4
|
||||||
|
longdouble.m4
|
||||||
|
longlong.m4
|
||||||
|
nls.m4
|
||||||
|
po.m4
|
||||||
|
printf-posix.m4
|
||||||
progtest.m4
|
progtest.m4
|
||||||
|
size_max.m4
|
||||||
stdint_h.m4
|
stdint_h.m4
|
||||||
uintmax_t.m4
|
uintmax_t.m4
|
||||||
ulonglong.m4
|
ulonglong.m4
|
||||||
|
visibility.m4
|
||||||
|
wchar_t.m4
|
||||||
|
wint_t.m4
|
||||||
|
xsize.m4
|
||||||
|
|
198
m4/ax_boost_base.m4
Normal file
198
m4/ax_boost_base.m4
Normal file
|
@ -0,0 +1,198 @@
|
||||||
|
##### http://autoconf-archive.cryp.to/ax_boost_base.html
|
||||||
|
#
|
||||||
|
# SYNOPSIS
|
||||||
|
#
|
||||||
|
# AX_BOOST_BASE([MINIMUM-VERSION])
|
||||||
|
#
|
||||||
|
# DESCRIPTION
|
||||||
|
#
|
||||||
|
# Test for the Boost C++ libraries of a particular version (or newer)
|
||||||
|
#
|
||||||
|
# If no path to the installed boost library is given the macro
|
||||||
|
# searchs under /usr, /usr/local, and /opt, and evaluates the
|
||||||
|
# $BOOST_ROOT environment variable. Further documentation is
|
||||||
|
# available at <http://randspringer.de/boost/index.html>.
|
||||||
|
#
|
||||||
|
# This macro calls:
|
||||||
|
#
|
||||||
|
# AC_SUBST(BOOST_CPPFLAGS) / AC_SUBST(BOOST_LDFLAGS)
|
||||||
|
#
|
||||||
|
# And sets:
|
||||||
|
#
|
||||||
|
# HAVE_BOOST
|
||||||
|
#
|
||||||
|
# LAST MODIFICATION
|
||||||
|
#
|
||||||
|
# 2007-03-15
|
||||||
|
#
|
||||||
|
# COPYLEFT
|
||||||
|
#
|
||||||
|
# Copyright (c) 2007 Thomas Porschberg <thomas@randspringer.de>
|
||||||
|
#
|
||||||
|
# Copying and distribution of this file, with or without
|
||||||
|
# modification, are permitted in any medium without royalty provided
|
||||||
|
# the copyright notice and this notice are preserved.
|
||||||
|
|
||||||
|
AC_DEFUN([AX_BOOST_BASE],
|
||||||
|
[
|
||||||
|
AC_ARG_WITH([boost],
|
||||||
|
AS_HELP_STRING([--with-boost@<:@=DIR@:>@], [use boost (default is yes) - it is possible to specify the root directory for boost (optional)]),
|
||||||
|
[
|
||||||
|
if test "$withval" = "no"; then
|
||||||
|
want_boost="no"
|
||||||
|
elif test "$withval" = "yes"; then
|
||||||
|
want_boost="yes"
|
||||||
|
ac_boost_path=""
|
||||||
|
else
|
||||||
|
want_boost="yes"
|
||||||
|
ac_boost_path="$withval"
|
||||||
|
fi
|
||||||
|
],
|
||||||
|
[want_boost="yes"])
|
||||||
|
|
||||||
|
if test "x$want_boost" = "xyes"; then
|
||||||
|
boost_lib_version_req=ifelse([$1], ,1.20.0,$1)
|
||||||
|
boost_lib_version_req_shorten=`expr $boost_lib_version_req : '\([[0-9]]*\.[[0-9]]*\)'`
|
||||||
|
boost_lib_version_req_major=`expr $boost_lib_version_req : '\([[0-9]]*\)'`
|
||||||
|
boost_lib_version_req_minor=`expr $boost_lib_version_req : '[[0-9]]*\.\([[0-9]]*\)'`
|
||||||
|
boost_lib_version_req_sub_minor=`expr $boost_lib_version_req : '[[0-9]]*\.[[0-9]]*\.\([[0-9]]*\)'`
|
||||||
|
if test "x$boost_lib_version_req_sub_minor" = "x" ; then
|
||||||
|
boost_lib_version_req_sub_minor="0"
|
||||||
|
fi
|
||||||
|
WANT_BOOST_VERSION=`expr $boost_lib_version_req_major \* 100000 \+ $boost_lib_version_req_minor \* 100 \+ $boost_lib_version_req_sub_minor`
|
||||||
|
AC_MSG_CHECKING(for boostlib >= $boost_lib_version_req)
|
||||||
|
succeeded=no
|
||||||
|
|
||||||
|
dnl first we check the system location for boost libraries
|
||||||
|
dnl this location ist chosen if boost libraries are installed with the --layout=system option
|
||||||
|
dnl or if you install boost with RPM
|
||||||
|
if test "$ac_boost_path" != ""; then
|
||||||
|
BOOST_LDFLAGS="-L$ac_boost_path/lib"
|
||||||
|
BOOST_CPPFLAGS="-I$ac_boost_path/include"
|
||||||
|
else
|
||||||
|
for ac_boost_path_tmp in /usr /usr/local /opt ; do
|
||||||
|
if test -d "$ac_boost_path_tmp/include/boost" && test -r "$ac_boost_path_tmp/include/boost"; then
|
||||||
|
BOOST_LDFLAGS="-L$ac_boost_path_tmp/lib"
|
||||||
|
BOOST_CPPFLAGS="-I$ac_boost_path_tmp/include"
|
||||||
|
break;
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
CPPFLAGS_SAVED="$CPPFLAGS"
|
||||||
|
CPPFLAGS="$CPPFLAGS $BOOST_CPPFLAGS"
|
||||||
|
export CPPFLAGS
|
||||||
|
|
||||||
|
LDFLAGS_SAVED="$LDFLAGS"
|
||||||
|
LDFLAGS="$LDFLAGS $BOOST_LDFLAGS"
|
||||||
|
export LDFLAGS
|
||||||
|
|
||||||
|
AC_LANG_PUSH(C++)
|
||||||
|
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
|
||||||
|
@%:@include <boost/version.hpp>
|
||||||
|
]], [[
|
||||||
|
#if BOOST_VERSION >= $WANT_BOOST_VERSION
|
||||||
|
// Everything is okay
|
||||||
|
#else
|
||||||
|
# error Boost version is too old
|
||||||
|
#endif
|
||||||
|
]])],[
|
||||||
|
AC_MSG_RESULT(yes)
|
||||||
|
succeeded=yes
|
||||||
|
found_system=yes
|
||||||
|
],[
|
||||||
|
])
|
||||||
|
AC_LANG_POP([C++])
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dnl if we found no boost with system layout we search for boost libraries
|
||||||
|
dnl built and installed without the --layout=system option or for a staged(not installed) version
|
||||||
|
if test "x$succeeded" != "xyes"; then
|
||||||
|
_version=0
|
||||||
|
if test "$ac_boost_path" != ""; then
|
||||||
|
BOOST_LDFLAGS="-L$ac_boost_path/lib"
|
||||||
|
if test -d "$ac_boost_path" && test -r "$ac_boost_path"; then
|
||||||
|
for i in `ls -d $ac_boost_path/include/boost-* 2>/dev/null`; do
|
||||||
|
_version_tmp=`echo $i | sed "s#$ac_boost_path##" | sed 's/\/include\/boost-//' | sed 's/_/./'`
|
||||||
|
V_CHECK=`expr $_version_tmp \> $_version`
|
||||||
|
if test "$V_CHECK" = "1" ; then
|
||||||
|
_version=$_version_tmp
|
||||||
|
fi
|
||||||
|
VERSION_UNDERSCORE=`echo $_version | sed 's/\./_/'`
|
||||||
|
BOOST_CPPFLAGS="-I$ac_boost_path/include/boost-$VERSION_UNDERSCORE"
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
for ac_boost_path in /usr /usr/local /opt ; do
|
||||||
|
if test -d "$ac_boost_path" && test -r "$ac_boost_path"; then
|
||||||
|
for i in `ls -d $ac_boost_path/include/boost-* 2>/dev/null`; do
|
||||||
|
_version_tmp=`echo $i | sed "s#$ac_boost_path##" | sed 's/\/include\/boost-//' | sed 's/_/./'`
|
||||||
|
V_CHECK=`expr $_version_tmp \> $_version`
|
||||||
|
if test "$V_CHECK" = "1" ; then
|
||||||
|
_version=$_version_tmp
|
||||||
|
best_path=$ac_boost_path
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
VERSION_UNDERSCORE=`echo $_version | sed 's/\./_/'`
|
||||||
|
BOOST_CPPFLAGS="-I$best_path/include/boost-$VERSION_UNDERSCORE"
|
||||||
|
BOOST_LDFLAGS="-L$best_path/lib"
|
||||||
|
|
||||||
|
if test "x$BOOST_ROOT" != "x"; then
|
||||||
|
if test -d "$BOOST_ROOT" && test -r "$BOOST_ROOT" && test -d "$BOOST_ROOT/stage/lib" && test -r "$BOOST_ROOT/stage/lib"; then
|
||||||
|
version_dir=`expr //$BOOST_ROOT : '.*/\(.*\)'`
|
||||||
|
stage_version=`echo $version_dir | sed 's/boost_//' | sed 's/_/./g'`
|
||||||
|
stage_version_shorten=`expr $stage_version : '\([[0-9]]*\.[[0-9]]*\)'`
|
||||||
|
V_CHECK=`expr $stage_version_shorten \>\= $_version`
|
||||||
|
if test "$V_CHECK" = "1" ; then
|
||||||
|
AC_MSG_NOTICE(We will use a staged boost library from $BOOST_ROOT)
|
||||||
|
BOOST_CPPFLAGS="-I$BOOST_ROOT"
|
||||||
|
BOOST_LDFLAGS="-L$BOOST_ROOT/stage/lib"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
CPPFLAGS="$CPPFLAGS $BOOST_CPPFLAGS"
|
||||||
|
export CPPFLAGS
|
||||||
|
LDFLAGS="$LDFLAGS $BOOST_LDFLAGS"
|
||||||
|
export LDFLAGS
|
||||||
|
|
||||||
|
AC_LANG_PUSH(C++)
|
||||||
|
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
|
||||||
|
@%:@include <boost/version.hpp>
|
||||||
|
]], [[
|
||||||
|
#if BOOST_VERSION >= $WANT_BOOST_VERSION
|
||||||
|
// Everything is okay
|
||||||
|
#else
|
||||||
|
# error Boost version is too old
|
||||||
|
#endif
|
||||||
|
]])],[
|
||||||
|
AC_MSG_RESULT(yes)
|
||||||
|
succeeded=yes
|
||||||
|
found_system=yes
|
||||||
|
],[
|
||||||
|
])
|
||||||
|
AC_LANG_POP([C++])
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test "$succeeded" != "yes" ; then
|
||||||
|
if test "$_version" = "0" ; then
|
||||||
|
AC_MSG_ERROR([[We could not detect the boost libraries (version $boost_lib_version_req_shorten or higher). If you have a staged boost library (still not installed) please specify \$BOOST_ROOT in your environment and do not give a PATH to --with-boost option. If you are sure you have boost installed, then check your version number looking in <boost/version.hpp>. See http://randspringer.de/boost for more documentation.]])
|
||||||
|
else
|
||||||
|
AC_MSG_NOTICE([Your boost libraries seems to old (version $_version).])
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
AC_SUBST(BOOST_CPPFLAGS)
|
||||||
|
AC_SUBST(BOOST_LDFLAGS)
|
||||||
|
AC_DEFINE(HAVE_BOOST,,[define if the Boost library is available])
|
||||||
|
fi
|
||||||
|
|
||||||
|
CPPFLAGS="$CPPFLAGS_SAVED"
|
||||||
|
LDFLAGS="$LDFLAGS_SAVED"
|
||||||
|
fi
|
||||||
|
|
||||||
|
])
|
|
@ -1,6 +1,7 @@
|
||||||
Makefile
|
Makefile
|
||||||
Makefile.in
|
Makefile.in
|
||||||
Makefile.in.in
|
Makefile.in.in
|
||||||
|
Makevars.template
|
||||||
POTFILES
|
POTFILES
|
||||||
*.mo
|
*.mo
|
||||||
*.gmo
|
*.gmo
|
||||||
|
@ -8,3 +9,9 @@ POTFILES
|
||||||
*.sin
|
*.sin
|
||||||
*.header
|
*.header
|
||||||
Rules-quot
|
Rules-quot
|
||||||
|
stamp-po
|
||||||
|
en@boldquot.insert-header
|
||||||
|
en@boldquot.po
|
||||||
|
en@quot.insert-header
|
||||||
|
en@quot.po
|
||||||
|
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
fr
|
fr en@quot en@boldquot
|
||||||
|
|
|
@ -1,16 +1,20 @@
|
||||||
./dic/automaton.c
|
./dic/automaton.cpp
|
||||||
./dic/automaton.h
|
./dic/automaton.h
|
||||||
./dic/compdic.c
|
./dic/header.cpp
|
||||||
./dic/dic.c
|
./dic/header.h
|
||||||
|
./dic/dic.cpp
|
||||||
./dic/dic.h
|
./dic/dic.h
|
||||||
./dic/dic_internals.h
|
./dic/dic_internals.h
|
||||||
./dic/dic_search.c
|
./dic/dic_search.cpp
|
||||||
./dic/dic_search.h
|
./dic/hashtable.cpp
|
||||||
./dic/hashtable.c
|
|
||||||
./dic/hashtable.h
|
./dic/hashtable.h
|
||||||
./dic/listdic.c
|
./dic/regexp.cpp
|
||||||
./dic/regexp.c
|
|
||||||
./dic/regexp.h
|
./dic/regexp.h
|
||||||
|
./dic/tile.cpp
|
||||||
|
./dic/tile.h
|
||||||
|
./dic/compdic.cpp
|
||||||
|
./dic/listdic.cpp
|
||||||
|
./dic/regexpmain.cpp
|
||||||
./game/bag.cpp
|
./game/bag.cpp
|
||||||
./game/bag.h
|
./game/bag.h
|
||||||
./game/board.cpp
|
./game/board.cpp
|
||||||
|
@ -39,11 +43,9 @@
|
||||||
./game/results.h
|
./game/results.h
|
||||||
./game/round.cpp
|
./game/round.cpp
|
||||||
./game/round.h
|
./game/round.h
|
||||||
./game/tile.cpp
|
|
||||||
./game/tile.h
|
|
||||||
./game/training.cpp
|
./game/training.cpp
|
||||||
./game/training.h
|
./game/training.h
|
||||||
./utils/eliottxt.cpp
|
#./utils/eliottxt.cpp
|
||||||
./utils/game_io.h
|
./utils/game_io.h
|
||||||
./utils/game_io.cpp
|
./utils/game_io.cpp
|
||||||
./utils/ncurses.cpp
|
./utils/ncurses.cpp
|
||||||
|
@ -68,4 +70,3 @@
|
||||||
./wxwin/printout.cc
|
./wxwin/printout.cc
|
||||||
./wxwin/searchpanel.h
|
./wxwin/searchpanel.h
|
||||||
./wxwin/searchpanel.cc
|
./wxwin/searchpanel.cc
|
||||||
./config.h
|
|
||||||
|
|
1068
po/eliot.pot
1068
po/eliot.pot
File diff suppressed because it is too large
Load diff
|
@ -1 +1,2 @@
|
||||||
*.run
|
*.run
|
||||||
|
load_saved_game.elt
|
||||||
|
|
20
test/driver
20
test/driver
|
@ -20,19 +20,25 @@ training_dict 0 # randseed unused
|
||||||
training_bag 0 # randseed unused
|
training_bag 0 # randseed unused
|
||||||
# Enter a rack, then display all the possibilities
|
# Enter a rack, then display all the possibilities
|
||||||
training_search 0 # randseed unused
|
training_search 0 # randseed unused
|
||||||
|
# Display the benjamins for several words
|
||||||
|
training_benj 0 # randseed unused
|
||||||
|
# Display the "raccords" for several words
|
||||||
|
training_racc 0 # randseed unused
|
||||||
|
# Test the 7 + 1 feature
|
||||||
|
training_7pl1 0 # randseed unused
|
||||||
# Several ways of getting a rack and playing a word
|
# Several ways of getting a rack and playing a word
|
||||||
training_play 4
|
training_play 4
|
||||||
# Training rack+search+play+back
|
# Training rack+search+play+back
|
||||||
training_back 5
|
training_back 5
|
||||||
# Joker problem on game search
|
# Joker problem on game search
|
||||||
training_rosace 0
|
training_rosace 0
|
||||||
|
|
||||||
# Board cross off by one score
|
# Board cross off by one score
|
||||||
training_cross 0
|
training_cross 0
|
||||||
# Board cross backward setAny
|
# Board cross backward setAny
|
||||||
training_cross2 0
|
training_cross2 0
|
||||||
# Board cross backward joker
|
# Board cross backward joker
|
||||||
training_cross3 0
|
training_cross3 0
|
||||||
|
|
||||||
# Joker problem on game load
|
# Joker problem on game load
|
||||||
training_joker 0
|
training_joker 0
|
||||||
|
@ -47,6 +53,8 @@ training_joker2 0
|
||||||
|
|
||||||
# 2 AI players
|
# 2 AI players
|
||||||
duplicate_2_ai 5
|
duplicate_2_ai 5
|
||||||
|
# 2 human players, one IA, with manual change of human player
|
||||||
|
duplicate_humans_ai 6
|
||||||
|
|
||||||
#################
|
#################
|
||||||
# Free game mode
|
# Free game mode
|
||||||
|
@ -60,7 +68,7 @@ freegame_change 3
|
||||||
freegame_3_ai 2
|
freegame_3_ai 2
|
||||||
|
|
||||||
##############
|
##############
|
||||||
# Load / Save
|
# Load / Save
|
||||||
##############
|
##############
|
||||||
|
|
||||||
# load a standard training game (fumee)
|
# load a standard training game (fumee)
|
||||||
|
|
|
@ -3,6 +3,7 @@ a S
|
||||||
a T
|
a T
|
||||||
a l
|
a l
|
||||||
a p
|
a p
|
||||||
|
a g
|
||||||
q
|
q
|
||||||
q
|
q
|
||||||
|
|
||||||
|
|
|
@ -3,14 +3,14 @@ commande> d 0 2
|
||||||
mode duplicate
|
mode duplicate
|
||||||
[?] pour l'aide
|
[?] pour l'aide
|
||||||
commande> a S
|
commande> a S
|
||||||
Joueur 0: 866
|
Joueur 0: 918
|
||||||
Joueur 1: 866
|
Joueur 1: 918
|
||||||
commande> a T
|
commande> a T
|
||||||
Joueur 0: RTTUW
|
Joueur 0: DQRU
|
||||||
Joueur 1: RTTUW
|
Joueur 1: DQRU
|
||||||
commande> a l
|
commande> a l
|
||||||
A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
A B C D E F G H I J K L M N O P Q R S T U V W X Y Z ?
|
||||||
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 2 1 0 1 0 0 0 0
|
0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 1 0 0 0 0 0 0
|
||||||
commande> a p
|
commande> a p
|
||||||
Eliot 1.5
|
Eliot 1.5
|
||||||
|
|
||||||
|
@ -20,30 +20,50 @@ Player 1: Computer
|
||||||
|
|
||||||
N | RACK | SOLUTION | REF | PTS | P | BONUS
|
N | RACK | SOLUTION | REF | PTS | P | BONUS
|
||||||
===|==========|=================|=====|=====|===|======
|
===|==========|=================|=====|=====|===|======
|
||||||
1 | EA?AEBF | FABAcEE | H4 | 80 | 0 | *
|
1 | ?EBAAEF | FABAcEE | H4 | 80 | 0 | *
|
||||||
2 | LMUAEYE | YEBLE | 6F | 38 | 0 |
|
2 | KEELIFE | KIEF | 10F | 36 | 0 |
|
||||||
3 | AMU+JEIG | MEJUGEAI | 9G | 78 | 0 | *
|
3 | EEEL+IJX | FIXEE | 4H | 34 | 0 |
|
||||||
4 | LEHNMGA | HALE | 8L | 46 | 0 |
|
4 | EJL+RANS | JEANS | 11D | 49 | 0 |
|
||||||
5 | GMN+NSEO | MENSONGE | O1 | 83 | 0 | *
|
5 | LR+OAHPU | LOPE | 9E | 29 | 0 |
|
||||||
6 | ARAURIU | RAIRA | 5B | 21 | 0 |
|
6 | AHRU+OAE | HOUE | 12A | 40 | 0 |
|
||||||
7 | UU+TSDEA | RUSTAUDE | B5 | 63 | 0 | *
|
7 | AAR+AVOC | HAVA | A12 | 30 | 0 |
|
||||||
8 | ONIAVPD | VIDA | A12 | 45 | 0 |
|
8 | ACOR+WDN | WAX | J2 | 41 | 0 |
|
||||||
9 | NOP+SONE | PENONS | K1 | 33 | 0 |
|
9 | -VNIALZO | OZONAI | B10 | 48 | 0 |
|
||||||
10 | O+OXTOLN | LUX | J8 | 32 | 0 |
|
10 | LV+NDULU | VIN | I3 | 21 | 0 |
|
||||||
11 | NOOOT+SZ | ZOOS | 11E | 38 | 0 |
|
11 | DLLUU+NO | DUNE | K1 | 22 | 0 |
|
||||||
12 | NOT+MIAI | DOMINAIT | 14A | 74 | 0 | *
|
12 | LLOU+DAM | DOUMA | 1K | 24 | 0 |
|
||||||
13 | CERPFEO | FEROCE | 15G | 47 | 0 |
|
13 | DLL+TOBI | MOLLIT | N1 | 18 | 0 |
|
||||||
14 | P+BSVQIU | PIQUAS | C1 | 32 | 0 |
|
14 | -ELTCUBE | CUBEBE | 6F | 31 | 0 |
|
||||||
15 | BV+ETLIE | LEVITE | A1 | 39 | 0 |
|
15 | -PE?STIE | SEPTImE | L6 | 85 | 0 | *
|
||||||
16 | B+RLD?UC | PUBLiC | 1C | 36 | 0 |
|
16 | LNRSISE | SERINS | J10 | 29 | 0 |
|
||||||
17 | DR+NTERR | DENREE | 2J | 22 | 0 |
|
17 | L+RGYAET | STYLER | 15J | 75 | 0 |
|
||||||
18 | RRT+TUKE | TREK | 13F | 36 | 0 |
|
18 | AG+CURGI | CARGUE | N10 | 30 | 0 |
|
||||||
19 | RTU+THWI | HAI | 7G | 23 | 0 |
|
19 | GI+EDRSS | DEGRISAS | 7B | 79 | 0 | *
|
||||||
|
20 | TOEHLRT | PHOT | 8L | 27 | 0 |
|
||||||
|
21 | ELRT+NME | REMELENT | C2 | 70 | 0 | *
|
||||||
|
22 | QDRMUTI | MITAN | 5E | 20 | 0 |
|
||||||
|
|
||||||
Total: 866
|
Total: 918
|
||||||
|
|
||||||
Rack 0: RTTUW
|
Rack 0: DQRU
|
||||||
Rack 1: RTTUW
|
Rack 1: DQRU
|
||||||
|
commande> a g
|
||||||
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
||||||
|
A - - - - - - - - - - - H A V A
|
||||||
|
B - - - - - - D - - O Z O N A I
|
||||||
|
C - R E M E L E N T - - U - - -
|
||||||
|
D - - - - - - G - - - J E - - -
|
||||||
|
E - - - - M - R - L - E - - - -
|
||||||
|
F - - - - I C I - O K A - - - -
|
||||||
|
G - - - - T U S - P I N - - - -
|
||||||
|
H - - - F A B A c E E S - - - -
|
||||||
|
I - - V I N E S - - F - - - - -
|
||||||
|
J - W A X - B - - - S E R I N S
|
||||||
|
K D U N E - E - - - - - - - - T
|
||||||
|
L O - - E - S E P T I m E - - Y
|
||||||
|
M U - - - - - - H - - - - - - L
|
||||||
|
N M O L L I T - O - C A R G U E
|
||||||
|
O A - - - - - - T - - - - - - R
|
||||||
commande> q
|
commande> q
|
||||||
fin du mode duplicate
|
fin du mode duplicate
|
||||||
commande> q
|
commande> q
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue