nur-sery/nur_src/v/bs3/vc/CTable.php

640 lines
18 KiB
PHP

<?php
namespace nur\v\bs3\vc;
use Exception;
use Iterator;
use nur\A;
use nur\b\coll\BaseArray;
use nur\b\params\IParametrable;
use nur\b\params\Tparametrable1;
use nur\b\ValueException;
use nur\data\types\md_utils;
use nur\data\types\Metadata;
use nur\func;
use nur\iter;
use nur\sery\cl;
use nur\SL;
use nur\v\base\ComponentPrintable;
use nur\v\v;
use nur\v\vo;
/**
* Class CTable: affiche un tableau de données
*/
class CTable extends ComponentPrintable implements IParametrable {
use Tparametrable1;
const COLS = null;
const EXCLUDE_COLS = null;
const ADD_COLS = null;
const HEADERS = null;
const ADD_HEADERS = null;
/** @var array schéma des données à afficher */
const SCHEMA = null;
const TABLE_CLASS = "table-bordered";
const AUTOPRINT = null;
/** @var ?array schéma des informations supplémentaires */
const DATA_SCHEMA = null;
/**
* retourner la valeur de COLS construite dynamiquement. cette méthode n'est
* appelée que si c'est nécessaire
*/
protected function COLS(): ?array {
return null;
}
/**
* retourner la valeur de HEADERS construite dynamiquement. cette méthode
* n'est appelée que si c'est nécessaire
*/
protected function HEADERS(): ?array {
return null;
}
function __construct(?iterable $rows=null, ?array $params=null) {
self::set_parametrable_params_defaults($params, [
"cols" => static::COLS,
"exclude_cols" => static::EXCLUDE_COLS,
"add_cols" => static::ADD_COLS,
"headers" => static::HEADERS,
"add_headers" => static::ADD_HEADERS,
"schema" => static::SCHEMA,
"table_class" => static::TABLE_CLASS,
"autoprint" => static::AUTOPRINT,
"data_schema" => static::DATA_SCHEMA,
]);
if ($rows === null) $rows = $this->defaultRows();
A::set_nn($params, "rows", $rows);
[$params, $data] = $this->splitParametrableParams($params);
if ($data) A::merge($params["data"], $data);
$this->initParametrableParams($params);
if ($this->ppAutoprint) $this->print();
}
protected function defaultRows(): ?iterable {
return null;
}
const PARAMETRABLE_PARAMS_SCHEMA = [
"rows" => ["?iterable", null, "source des lignes à afficher"],
"filter_func" => ["?callable", null, "fonction permettant de filter les éléments à afficher"],
"map_func" => ["?callable", null, "fonction permettant de mapper les éléments"],
"map" => ["?array", null, "tableau permettant de sélectionner les éléments"],
"cols" => ["?array", null, "colonnes à extraire et à afficher"],
"exclude_cols" => ["?array", null, "colonnes à exclure de la liste calculée automatiquement"],
"add_cols" => ["?array", null, "colonnes à ajouter à la liste calculée automatiquement"],
"headers" => ["?array", null, "en-têtes correspondant aux colonnes"],
"add_headers" => ["?array", null, "en-têtes à ajouter à la liste calculée automatiquement"],
"schema" => ["?array", null, "schéma des données à afficher"],
"table_class" => ["?array", null, "classe CSS du tableau"],
"table_attrs" => ["?array", null, "autres attributs à placer sur le tableau"],
"before_table" => ["?content", null, "Contenu à afficher avant le tableau s'il y a des lignes à afficher"],
"after_table" => ["?content", null, "Contenu à afficher après le tableau s'il y a des lignes à afficher"],
"no_data" => ["?content", null, "Contenu à afficher s'il n'y a pas de lignes à afficher"],
"on_exception" => ["?callable", null, "fonction à appeler en cas d'exception"],
"autoprint" => ["bool", false, "faut-il afficher automatiquement le tableau"],
"data" => ["?array", null, "donnés supplémentaires utilisées pour l'affichage"],
"data_schema" => ["?array", null, "schéma des données supplémentaire"],
];
const COL_SCHEMA = [
"name" => ["?string", null, "Nom de la colonne"],
"pkey" => ["?string", null, "Clé permettant de calculer la valeur effective de la colonne"],
];
private static $col_md;
const HEADER_SCHEMA = [
"name" => ["string", null, "Nom de l'en-tête"],
"span" => ["?int", null, "nombre de colonnes que cet en-tête doit occuper"],
"label" => ["?string", null, "Libellé de l'en-tête",
"desc" => "vaut [name] par défaut",
],
];
private static $header_md;
/** @var Metadata schéma des données */
protected $md;
function pp_setSchema($schema): void {
$this->md = Metadata::with($schema);
}
/** @var ?iterable */
protected $ppRows;
/** @var array */
protected $filterCtx;
function pp_setFilterFunc(?callable $filterFunc): void {
if ($filterFunc === null) $this->filterCtx = null;
else $this->filterCtx = func::_prepare($filterFunc);
}
/** @var array */
protected $mapCtx;
function pp_setMapFunc(?callable $mapFunc): void {
if ($mapFunc === null) $this->mapCtx = null;
else $this->mapCtx = func::_prepare($mapFunc);
}
/** @var ?array */
protected $ppMap;
protected $ppCols;
protected $ppExcludeCols;
protected $ppAddCols;
protected $ppHeaders;
protected $ppAddHeaders;
/** @var array classe CSS du tableau */
protected $ppTableClass;
protected function getTableClass(): array {
return $this->ppTableClass;
}
/** @var array */
protected $ppTableAttrs;
/** @var array|string */
protected $ppBeforeTable;
/** @var array|string */
protected $ppAfterTable;
/** @var array|string */
protected $ppNoData;
/** @var callable */
protected $ppOnException;
/** @var bool */
protected $ppAutoprint;
/** @var Metadata */
protected $dataSchema;
function pp_setDataSchema($dataSchema): void {
$this->dataSchema = Metadata::with($dataSchema);
}
/** @var mixed données supplémentaires utilisées pour l'affichage */
protected $data;
function pp_setData(?array $data): void {
$this->data = $data;
}
protected function afterSetParametrableParams(array $modifiedKeys, ?Metadata $md=null): void {
if (self::was_parametrable_param_modified($modifiedKeys
, "cols", "exclude_cols", "add_cols"
, "headers", "add_headers")) {
$this->cols = null;
$this->headers = null;
}
if (self::was_parametrable_param_modified($modifiedKeys, "data", "data_schema")) {
if ($this->dataSchema !== null) {
$this->dataSchema->ensureSchema($this->data);
}
}
}
/** @var array liste des colonnes à extraire des lignes */
protected $cols;
/** @var array informations sur les colonnes à extraire des lignes */
protected $colInfos;
/** @var array en-têtes correspondant aux colonnes extraites */
protected $headers;
protected function updateColInfos(?array &$cols, ?array &$colInfos): void {
# extraire les données de colonnes pour que $cols soit juste une liste de
# colonnes
if ($cols === null) return;
if ($colInfos === null) $colInfos = [];
$colKeys = [];
$index = 0;
foreach ($cols as $key => $col) {
if ($key === $index) {
$index++;
if ($col === null) {
# colonne "nulle", à prendre telle quelle
$colKeys[] = null;
} elseif (!array_key_exists($col, $colInfos)) {
$colKeys[] = $col;
$colInfos[$col] = null;
}
} else {
if (is_array($col) && array_key_exists("name", $col)) {
$key = $col["name"];
}
if (!array_key_exists($key, $colInfos)) $colKeys[] = $key;
$colInfos[$key] = $col;
}
}
$cols = $colKeys;
}
protected function resolveColsHeaders($row): bool {
if ($this->cols !== null || $this->headers !== null) return false;
$schema = $this->md;
$cols = $this->ppCols;
$colsExcludes = $this->ppExcludeCols;
if ($cols === null) $cols = $this->COLS();
if ($cols === null && $schema !== null) {
$cols = A::xselect($schema->getKeys(), null, $colsExcludes);
}
$this->updateColInfos($cols, $colInfos);
$headers = $this->ppHeaders;
$sfields = null;
if ($headers === null) $headers = $this->HEADERS();
if ($headers === null) {
if ($schema !== null) {
# ici, $cols est forcément défini, à cause de la commande ci-dessus
$headers = [];
$sfields = $schema->getSfields();
foreach ($cols as $col) {
$sfield = A::get($sfields, $col);
$headers[] = $sfield !== null? $sfield["header"]: $col;
}
} elseif ($cols !== null) {
$headers = $cols;
}
}
$skipRow = false;
if ($cols === null || $headers === null) {
# si pas de colonnes ou de headers, prendre la première ligne si
# c'est une séquence ou ses clés si c'est un tableau associatif
if (A::is_seq($row)) {
$skipRow = true;
$keys = A::xselect($row, null, $colsExcludes);
if ($headers === null) $headers = $keys;
if ($cols === null) $cols = array_keys($keys);
} else {
$row = A::xselect_keys($row, null, $colsExcludes);
if ($headers === null) $headers = array_keys($row);
if ($cols === null) $cols = array_keys($row);
}
}
# ces cas ne devraient pas se produire puisqu'ils devraient être récupérés
# par la section no_contents, mais bon... au cas où
if ($headers === null) $headers = [];
if ($cols === null) $cols = [];
$addCols = $this->ppAddCols;
$this->updateColInfos($addCols, $colInfos);
$addHeaders = $this->ppAddHeaders;
if ($addCols !== null && $addHeaders !== null) {
# si on fournit les deux, les intégrer sans modification
A::merge($cols, $addCols);
A::merge($headers, $addHeaders);
} elseif ($addCols !== null) {
# si on ne donne que des colonnes supplémentaires, inférer les en-têtes
# à rajouter
foreach ($addCols as $col) {
$cols[] = $col;
if ($col === null) {
$headers[] = "";
} elseif ($sfields !== null) {
$sfield = A::get($sfields, $col);
$headers[] = $sfield !== null? $sfield["header"]: $col;
} else {
$headers[] = $col;
}
}
} elseif ($addHeaders !== null) {
# si on ne donne que des en-têtes supplémentaires, rajouter des colonnes
# nulles pour compenser
foreach ($addHeaders as $header) {
$cols[] = null;
$headers[] = $header;
}
}
md_utils::ensure_md(self::$col_md, self::COL_SCHEMA)
->eachEnsureSchema($colInfos);
md_utils::ensure_md(self::$header_md, self::HEADER_SCHEMA)
->eachEnsureSchema($headers);
$this->cols = $cols;
$this->colInfos = $colInfos;
$this->headers = $headers;
return $skipRow;
}
protected function _rewindRows(): bool {
return iter::rewind($this->ppRows);
}
protected function _validRow(): bool {
return iter::valid($this->ppRows);
}
protected function _currentRow(&$key=null) {
return iter::current($this->ppRows, $key);
}
protected function _nextRow(): void {
iter::next($this->ppRows);
}
protected function ensureArray($row): array {
if (!is_array($row)) {
if ($row instanceof BaseArray) $row = $row->array();
elseif ($row instanceof Iterator) $row = iterator_to_array($row);
else throw ValueException::unexpected_type("array", $row);
}
return $row;
}
protected function nextRow(): ?array {
$filterCtx = $this->filterCtx;
$mapCtx = $this->mapCtx;
$map = $this->ppMap;
$row = $this->rawRow;
if ($filterCtx !== null) {
if (!func::_call($filterCtx, [$row, $this->rowKey, $this->rowIndex])) return null;
}
if ($mapCtx !== null) {
# si la valeur est séquentielle, c'est une clé dans $row
# si la valeur est associative et que c'est une fonction, elle est appelée
# avec la valeur
$map = func::_call($mapCtx, [$row, $this->rowKey, $this->rowIndex]);
}
if ($map !== null) {
$row = $this->ensureArray($row);
$index = 0;
$mapped = [];
foreach ($map as $key => $value) {
if ($key === $index) {
$index++;
if ($value === null) $mapped[] = null;
else $mapped[$value] = A::get($row, $value);
} elseif (is_callable($value)) {
$mapped[$key] = func::call($value, A::get($row, $key), $key, $row, $this->rowKey, $this->rowIndex);
} else {
$mapped[$key] = $value;
}
}
$row = $mapped;
} else {
$row = $this->ensureArray($row);
}
return $row;
}
protected function getRowValues(array $row): array {
$havePkey = false;
if ($this->colInfos !== null) {
foreach ($this->colInfos as $colInfo) {
if ($colInfo["pkey"] !== null) {
$havePkey = true;
break;
}
}
}
if (!$havePkey) return $row;
$cooked = [];
foreach ($this->cols as $col) {
if ($col === null) {
$cooked[] = null;
} else {
$colInfo = A::get($this->colInfos, $col);
$pkey = $colInfo !== null? $colInfo["pkey"]: null;
if ($pkey !== null) $value = A::pget($row, $pkey);
else $value = $row[$col];
$cooked[$col] = $value;
}
}
# compléter avec les autres champs tels quels
foreach ($row as $key => $value) {
if (!array_key_exists($key, $cooked)) {
$cooked[$key] = $value;
}
}
return $cooked;
}
protected $results;
protected function ensureRowSchema(array &$row): void {
$md = $this->md;
if ($md !== null) {
$md->ensureSchema($row);
$md->verifix($row, $this->results, false, true);
#XXX il faut garder les champs "autres"!
#$row = $md->format($row);
}
}
protected function cookRow(array $row): array {
$cooked = $this->getRowValues($row);
$this->ensureRowSchema($cooked);
return $cooked;
}
function print(): void {
try {
$this->_rewindRows();
$this->rowIndex = 0;
$haveRows = false;
while ($this->_validRow()) {
$this->rawRow = $this->_currentRow($this->rowKey);;
try {
$this->origRow = $this->nextRow();
if ($this->origRow === null) continue;
$skipRow = false;
if (!$haveRows) $skipRow = $this->resolveColsHeaders($this->origRow);
if (!$skipRow) {
$this->row = $this->cookRow($this->origRow);
if (!$haveRows) {
$haveRows = true;
vo::print($this->beforeTable());
vo::stable(["class" => "table", $this->table()]);
vo::sthead($this->thead());
$this->printBeforeHeader();
$this->printHeader($this->headers);
$this->printAfterHeader();
vo::ethead();
vo::stbody($this->tbody());
}
$this->printBeforeRow();
$this->printRow($this->row);
$this->printAfterRow();
$this->rowIndex++;
}
} finally {
$this->_nextRow();
}
}
if ($haveRows) {
vo::etbody();
vo::etable();
vo::print($this->afterTable());
} else {
vo::print($this->noData());
}
} catch (Exception $e) {
$onException = $this->ppOnException;
if ($onException !== null) func::call($onException, $e);
else throw $e;
}
}
/** contenu à afficher avant le tableau s'il y a des lignes à afficher */
function beforeTable(): ?iterable {
return $this->ppBeforeTable;
}
/** contenu à afficher après le tableau s'il y a des lignes à afficher */
function afterTable(): ?iterable {
return $this->ppAfterTable;
}
/** contenu à afficher s'il n'y a pas de lignes à afficher */
function noData(): ?iterable {
return $this->ppNoData;
}
function table(): ?iterable {
return SL::merge(["class" => $this->getTableClass()], $this->ppTableAttrs);
}
function thead(): ?iterable {
return null;
}
/** contenu à afficher avant la ligne */
function beforeHeader(): ?iterable {
return null;
}
function printBeforeHeader(): void {
$beforeHeaderContents = $this->beforeHeader();
if ($beforeHeaderContents !== null) vo::tr($beforeHeaderContents);
}
/** contenu à afficher après la ligne */
function afterHeader(): ?iterable {
return null;
}
function printAfterHeader(): void {
$afterHeaderContents = $this->afterHeader();
if ($afterHeaderContents !== null) vo::tr($afterHeaderContents);
}
function printHeader(array $headers): void {
vo::print($this->headerTr($headers));
}
function headerTr(array $headers): array {
return v::tr($this->headerRow($headers));
}
function headerRow(array $headers): ?iterable {
$vs = [];
foreach ($headers as $header) {
$label = $header["label"];
if ($label === null) $label = $header["name"];
$colspan = $header["span"];
$vs[] = v::th(["colspan" => $colspan, $label]);
}
return $vs;
}
function tbody(): ?iterable {
return null;
}
/** @var int index de la ligne courante */
protected $rowIndex;
/** @var string|int clé de la ligne courante */
protected $rowKey;
/** @var array la ligne originale, avant le mapping */
protected $rawRow;
/** @var array la ligne mappée avant sa "cuisine" */
protected $origRow;
/** @var array la ligne cuisinée */
protected $row;
/** contenu à afficher avant la ligne */
function beforeRow(): ?iterable {
return null;
}
function printBeforeRow(): void {
$beforeRowContents = $this->beforeRow();
if ($beforeRowContents !== null) vo::tr($beforeRowContents);
}
/** contenu à afficher après la ligne */
function afterRow(): ?iterable {
return null;
}
function printAfterRow(): void {
$afterRowContents = $this->afterRow();
if ($afterRowContents !== null) vo::tr($afterRowContents);
}
function printRow(array $row): void {
vo::print($this->rowTr($row));
}
function rowTr(array $row): array {
return v::tr($this->row($row));
}
function row(array $row): ?iterable {
$vs = [];
$this->index = 0;
foreach ($this->cols as $this->col) {
if ($this->col === null) $value = null;
else $value = $row[$this->col];
$vs[] = $this->colTd($value);
$this->index++;
}
return $vs;
}
/** @var int index de la colonne courante */
protected $index;
/** @var string|int clé de la colonne courante */
protected $col;
function colTd($value): array {
$result = A::get($this->results, $this->col);
$valid = $result === null || $result["valid"];
return v::td([
"class" => ["danger" => !$valid],
$this->col($value),
]);
}
function col($value): ?iterable {
#XXX formatter selon le schéma, ou en utilisant des méthodes définies dans
# cette classe
return q($value);
}
}