Compare commits

...

16 Commits

Author SHA1 Message Date
563ba7ed0a rechangement de format du dictionnaire 2025-12-03 17:26:36 +01:00
974e2c203c truc inutile 2025-12-03 11:26:55 +01:00
00547c2c00 ajout définition dentiste 2025-11-29 03:38:48 +01:00
95d21a22c9 Ajout Desproges 2025-10-19 15:46:29 +02:00
822d353775 Nouvelles définitions
Signed-off-by: adrien <adrien@malingrey.fr>
2025-10-11 11:49:36 +02:00
e754e6e44b ommission des lettres isolées sur la dernière ligne 2025-08-22 13:09:00 +02:00
8287699a4b méthodes magiques 2025-08-22 11:04:42 +02:00
d3b11df015 méthodes magiques 2025-08-22 10:56:06 +02:00
6635adf26a la navigation au clavier saute les cases noires 2025-08-19 17:57:02 +02:00
c2e4b8ae53 plus rapide 2025-08-19 16:54:06 +02:00
5aa5da8c53 correction formattage définitions 2025-08-19 16:53:49 +02:00
d862dc572d corrections 2025-08-19 16:48:52 +02:00
c3986e70a3 ajustements 2025-08-01 23:57:29 +02:00
e19bb9d2dc ajout d'une définition 2025-07-23 18:10:59 +02:00
468a6a9231 Merge remote-tracking branch 'origin/HEAD' 2025-07-23 18:01:53 +02:00
c62afa80cc 7 2025-05-23 20:32:54 +02:00
8 changed files with 4159 additions and 2926 deletions

View File

@@ -2,16 +2,13 @@
include_once "dico.php"; include_once "dico.php";
const ECART_TYPE_ALEA = 5; const ECART_TYPE = 5;
$randmax = mt_getrandmax() + 1; $randmax = mt_getrandmax() + 1;
function gaussienne($moyenne = 0, $ecartType = 1.0): float { function gaussienne($moyenne = 0, $ecartType = 1.0): float {
global $randmax; global $randmax;
$u = 0;
$v = 0;
$u = (mt_rand() + 1) / $randmax; $u = (mt_rand() + 1) / $randmax;
$v = (mt_rand() + 1) / $randmax; $v = (mt_rand() + 1) / $randmax;
@@ -86,7 +83,11 @@ class Grille implements ArrayAccess
foreach($mots as $mot) { foreach($mots as $mot) {
$definitions = $this->dico[strlen($mot)][$mot]; $definitions = $this->dico[strlen($mot)][$mot];
if (count($definitions)) { if (count($definitions)) {
$this->definitions["horizontales"][$y][] = $definitions[mt_rand(0, count($definitions) - 1)]; $definition = $definitions[mt_rand(0, count($definitions) - 1)];
if (isset($definitions["nb_mots"])) {
$definition["nb_mots"] = $definitions["nb_mots"];
}
$this->definitions["horizontales"][$y][] = $definition;
} }
} }
} }
@@ -95,7 +96,11 @@ class Grille implements ArrayAccess
foreach($mots as $mot) { foreach($mots as $mot) {
$definitions = $this->dico[strlen($mot)][$mot]; $definitions = $this->dico[strlen($mot)][$mot];
if (count($definitions)) { if (count($definitions)) {
$this->definitions["verticales"][$x][] = $definitions[mt_rand(0, count($definitions) - 1)]; $definition = $definitions[mt_rand(0, count($definitions) - 1)];
if (isset($definitions["nb_mots"])) {
$definition["nb_mots"] = $definitions["nb_mots"];
}
$this->definitions["verticales"][$x][] = $definition;
} }
} }
} }
@@ -126,7 +131,7 @@ class Grille implements ArrayAccess
$lettres_colonne->branches $lettres_colonne->branches
); );
foreach ($lettres_communes as $lettre => $_) { foreach ($lettres_communes as $lettre => $_) {
$lettres_communes[$lettre] = count($lettres_ligne->branches[$lettre]) * count($lettres_colonne->branches[$lettre]) * gaussienne(1, ECART_TYPE_ALEA); $lettres_communes[$lettre] = count($lettres_ligne->branches[$lettre]) * count($lettres_colonne->branches[$lettre]) * gaussienne(1, ECART_TYPE);
} }
uksort($lettres_communes, function($a, $b) use ($lettres_communes) { uksort($lettres_communes, function($a, $b) use ($lettres_communes) {
return $lettres_communes[$b] <=> $lettres_communes[$a]; return $lettres_communes[$b] <=> $lettres_communes[$a];
@@ -137,13 +142,19 @@ class Grille implements ArrayAccess
$this->grille[$y][$x] = $lettre; $this->grille[$y][$x] = $lettre;
// Omission des lettres isolées // Omission des lettres isolées
if ($lettre == CASE_NOIRE if ($lettre == CASE_NOIRE) {
&& ($y - 2 < 0 || $this->grille[$y - 2][$x] == CASE_NOIRE) if (($y < 2 || $this->grille[$y - 2][$x] == CASE_NOIRE)
&& ($y - 1 < 0 || $x - 1 < 0 || $this->grille[$y - 1][$x - 1] == CASE_NOIRE) && ($y < 1 || $x == 0 || $this->grille[$y - 1][$x - 1] == CASE_NOIRE)
&& ($y - 1 < 0 || $x + 1 >= $this->largeur || $this->grille[$y - 1][$x + 1] == CASE_NOIRE) && ($y < 1 || $x + 1 >= $this->largeur || $this->grille[$y - 1][$x + 1] == CASE_NOIRE)
) { ) continue;
continue; if ($y == $this->hauteur - 1
} && ($x < 2 || $this[$y][$x - 2] == CASE_NOIRE)
&& ($x < 1 || $this[$y - 1][$x - 1] == CASE_NOIRE)
) continue;
} else if ($x == $this->largeur - 1 && $y == $this->hauteur - 1
&& $this[$y][$x - 1] == CASE_NOIRE
&& $this[$y-1][$x] == CASE_NOIRE
) continue;
// Omission des doublons // Omission des doublons
$mots = []; $mots = [];

View File

@@ -52,10 +52,40 @@ class Trie implements ArrayAccess, IteratorAggregate, Countable {
} }
} }
public function arrayIterator() { // ArrayAccess
public function offsetSet($array, $valeur): void {
if (is_string($array)) {
$array = str_split($array);
}
$this->arraySet($array, $valeur);
}
public function offsetExists($array): bool {
if (is_string($array)) {
$array = str_split($array);
}
return $this->arrayExists($array);
}
public function &offsetGet($array): mixed {
if (is_string($array)) {
$array = str_split($array);
}
return $this->arrayGet($array);
}
public function offsetUnset($array): void {
if (is_string($array)) {
$array = str_split($array);
}
$this->arrayUnset($array);
}
// IteratorAggregate
public function getIterator(): Traversable {
foreach ($this->branches as $cle => $branche) { foreach ($this->branches as $cle => $branche) {
if ($branche instanceof Trie) { if ($branche instanceof Trie) {
foreach($branche->arrayIterator() as $sous_cles => $feuille) { foreach($branche->getIterator() as $sous_cles => $feuille) {
yield array_merge([$cle], $sous_cles) => $feuille; yield array_merge([$cle], $sous_cles) => $feuille;
} }
} else { } else {
@@ -64,30 +94,6 @@ class Trie implements ArrayAccess, IteratorAggregate, Countable {
} }
} }
// ArrayAccess
public function offsetSet($string, $valeur): void {
$this->arraySet(str_split($string), $valeur);
}
public function offsetExists($string): bool {
return $this->arrayExists(str_split($string));
}
public function &offsetGet($string): mixed {
return $this->arrayGet(str_split($string));
}
public function offsetUnset($string): void {
$this->arrayUnset(str_split($string));
}
// IteratorAggregate
public function getIterator(): Traversable {
foreach($this->arrayIterator() as $array => $valeur) {
yield implode("", $array) => $valeur;
}
}
// Countable // Countable
public function count(): int { public function count(): int {
return $this->nb_branches; return $this->nb_branches;

2818
dico.csv

File diff suppressed because it is too large Load Diff

View File

@@ -1,18 +1,22 @@
<?php <?php
include_once "Trie.php"; include_once "Trie.php";
const CASE_NOIRE = " "; const CASE_NOIRE = " ";
const DEFINITION = 0;
const AUTEUR = 1;
const NB_MOTS = 2;
function dico($longueur_max) { function dico($longueur_max) {
$transliterator = Transliterator::createFromRules(':: Any-Latin; :: Latin-ASCII; :: NFD; :: [:Nonspacing Mark:] Remove; :: Upper(); :: NFC;', Transliterator::FORWARD); $transliterator = Transliterator::createFromRules(':: Any-Latin; :: Latin-ASCII; :: NFD; :: [:Nonspacing Mark:] Remove; :: Upper(); :: NFC;', Transliterator::FORWARD);
$dico = [[new Trie()]]; $dico = [];
for ($longueur = 0; $longueur <= $longueur_max; $longueur++) { for ($longueur = 0; $longueur <= $longueur_max; $longueur++) {
$dico[] = new Trie(); $dico[] = new Trie();
} }
if (($lecteur = fopen("dico.csv", "r")) !== FALSE) { if (($lecteur = fopen("dico.tsv", "r")) !== FALSE) {
$entete = fgetcsv($lecteur, 0, "\t"); $entete = fgetcsv($lecteur, 0, "\t");
while (($ligne = fgetcsv($lecteur, 0, "\t")) !== FALSE) { while (($ligne = fgetcsv($lecteur, 0, "\t")) !== FALSE) {
if ( if (
@@ -22,7 +26,7 @@ function dico($longueur_max) {
) continue; ) continue;
$mot = $ligne[0]; $mot = $ligne[0];
$definitions = array_slice($ligne, 1); $definition = array_slice($ligne, 1);
$mot = str_replace("-", CASE_NOIRE, $mot); $mot = str_replace("-", CASE_NOIRE, $mot);
$mot = $transliterator->transliterate($mot); $mot = $transliterator->transliterate($mot);
@@ -30,12 +34,16 @@ function dico($longueur_max) {
$mots = explode(CASE_NOIRE, $mot); $mots = explode(CASE_NOIRE, $mot);
$nb_mots = count($mots); $nb_mots = count($mots);
$mot = implode("", $mots); $mot = implode("", $mots);
foreach($definitions as $i => $definition) { } else {
$definitions[$i] = "$definition#$nb_mots"; $nb_mots = 1;
}
} }
$dico[strlen($mot)][$mot] = $definitions; if (array_key_exists($mot, $dico)) {
$dico[strlen($mot)][$mot][] = $definition;
} else {
$dico[strlen($mot)][$mot] = [$definition];
if ($nb_mots > 1) $dico[strlen($mot)][$mot]["nb_mots"] = $nb_mots;
}
} }
fclose($lecteur); fclose($lecteur);
} }
@@ -45,12 +53,12 @@ function dico($longueur_max) {
function mots_espaces($longueur_max) { function mots_espaces($longueur_max) {
$dico = dico($longueur_max); $dico = dico($longueur_max);
for ($longueur = 1; $longueur <= $longueur_max; $longueur++) { for ($longueur = $longueur_max; $longueur >= 2; $longueur--) {
for ($position_espace = 1; $position_espace + 1 < $longueur; $position_espace++) { for ($position_espace = $longueur - 2; $position_espace >= 1; $position_espace--) {
$mots_suivants = $dico[$longueur - $position_espace - 1]; $mots_suivants = $dico[$longueur - $position_espace - 1];
foreach ($dico[$position_espace]->arrayIterator() as $premier_mot => $definition) { foreach ($dico[$position_espace] as $premier_mot => $definition) {
$premier_mot[] = CASE_NOIRE; $premier_mot[] = CASE_NOIRE;
$dico[$longueur]->arraySet($premier_mot, $mots_suivants); $dico[$longueur][$premier_mot] = $mots_suivants;
} }
} }
} }

4009
dico.tsv Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,11 +1,16 @@
<?php <?php
ini_set('display_errors', '1');
ini_set('display_startup_errors', '1');
error_reporting(E_ALL);
include_once "Grille.php"; include_once "Grille.php";
const HAUTEUR_DEFAUT = 8; const HAUTEUR_DEFAUT = 7;
const HAUTEUR_MIN = 2; const HAUTEUR_MIN = 2;
const HAUTEUR_MAX = 10; const HAUTEUR_MAX = 10;
const LARGEUR_DEFAUT = 8; const LARGEUR_DEFAUT = 7;
const LARGEUR_MIN = 2; const LARGEUR_MIN = 2;
const LARGEUR_MAX = 10; const LARGEUR_MAX = 10;
@@ -43,19 +48,19 @@ if (!isset($_GET["grille"]) || $_GET["grille"] == "") {
} }
function formatter_definition($definition) { function formatter_definition($definition) {
if (strpos($definition, "#") !== false) { if (isset($definition["nb_mots"]) && $definition["nb_mots"] > 1){
[$definition, $nb_mots] = explode("#", $definition); $nb_mots = $definition["nb_mots"];
$nb_mots = " <small>($nb_mots mots)</small>"; $nb_mots = " <small>($nb_mots mots)</small>";
} else { } else {
$nb_mots = ""; $nb_mots = "";
} }
if (strpos($definition, "@") !== false) { if (array_key_exists(AUTEUR, $definition)) {
[$definition, $auteur] = explode("@", $definition); $auteur = $definition[AUTEUR];
$auteur = " <small><em>$auteur</em></small>"; $auteur = " <small><em>$auteur</em></small>";
} else { } else {
$auteur = ""; $auteur = "";
} }
return $definition; return $definition[DEFINITION] . $nb_mots . $auteur;
} }
?> ?>
<!DOCTYPE HTML> <!DOCTYPE HTML>
@@ -85,6 +90,7 @@ function formatter_definition($definition) {
<body> <body>
<form id="grilleForm" method="get" location="."> <form id="grilleForm" method="get" location=".">
<h1 class="large width"> <h1 class="large width">
<a href=".">
<table> <table>
<tbody> <tbody>
<tr> <tr>
@@ -110,8 +116,9 @@ function formatter_definition($definition) {
</tr> </tr>
</tbody> </tbody>
</table> </table>
</a>
</h1> </h1>
<h1 class="small width">Mots■croisés</h1> <h1 class="small width"><a href=".">Mots■croisés</a></h1>
<div class="grille-et-definitions"> <div class="grille-et-definitions">
<?php if ($grille_valide): ?> <?php if ($grille_valide): ?>
<div class="grille"> <div class="grille">
@@ -133,7 +140,7 @@ function formatter_definition($definition) {
</td> </td>
<?php else: ?> <?php else: ?>
<td class="case blanche"> <td class="case blanche">
<input id="<?= chr($x + 65) . ($y + 1) ?>" type="text" maxlength="1" size="1" pattern="[A-Z]" placeholder="<?= $grille[$y][$x] ?>" <input id="<?= chr($x + 65) . ($y + 1) ?>" type="text" maxlength="1" size="1" pattern="[A-Z]"
title="<?= strip_tags("→ " . implode("\n→ ", array_map("formatter_definition", $grille->definitions["horizontales"][$y] ?? [])) . "\n↓ " . implode("\n↓ ", array_map("formatter_definition", $grille->definitions["verticales"][$x] ?? []))) ?>" /> title="<?= strip_tags("→ " . implode("\n→ ", array_map("formatter_definition", $grille->definitions["horizontales"][$y] ?? [])) . "\n↓ " . implode("\n↓ ", array_map("formatter_definition", $grille->definitions["verticales"][$x] ?? []))) ?>" />
</td> </td>
<?php endif; ?> <?php endif; ?>

View File

@@ -35,26 +35,28 @@ for (let input of inputs) {
}; };
input.onkeydown = function (event) { input.onkeydown = function (event) {
next_input = null; next_input = input;
do {
switch (event.key) { switch (event.key) {
case "ArrowUp": case "ArrowUp":
next_input = inputs[(input.index - largeur + nb_cases) % nb_cases]; next_input = inputs[(next_input.index - largeur + nb_cases) % nb_cases];
break; break;
case "ArrowDown": case "ArrowDown":
next_input = inputs[(input.index + largeur) % nb_cases]; next_input = inputs[(next_input.index + largeur) % nb_cases];
break; break;
case "ArrowLeft": case "ArrowLeft":
next_input = inputs[(input.index - 1 + nb_cases) % nb_cases]; next_input = inputs[(next_input.index - 1 + nb_cases) % nb_cases];
break; break;
case "ArrowRight": case "ArrowRight":
next_input = inputs[(input.index + 1) % nb_cases]; next_input = inputs[(next_input.index + 1) % nb_cases];
break; break;
default:
return;
} }
if (next_input) { } while (next_input.disabled)
event.preventDefault();
next_input.focus(); next_input.focus();
next_input.select(); next_input.select();
event.preventDefault();
}
}; };
input.oninput = function (event) { input.oninput = function (event) {

View File

@@ -18,6 +18,12 @@ h1 {
letter-spacing: 0.2em; letter-spacing: 0.2em;
} }
h1 a {
width: 100%;
text-decoration: none;
color: inherit;
}
h1.large.width { h1.large.width {
display: inherit; display: inherit;
padding-top: 1rem; padding-top: 1rem;
@@ -199,6 +205,7 @@ h2 {
text-decoration: underline; text-decoration: underline;
} }
h1 a:hover,
.nouvelle-grille button:hover { .nouvelle-grille button:hover {
color: #2a6496; color: #2a6496;
} }
@@ -315,6 +322,7 @@ h2 {
background-color: #edeeee; background-color: #edeeee;
} }
h1 a:hover,
.nouvelle-grille button:hover { .nouvelle-grille button:hover {
color: #479fec; color: #479fec;
} }