mirror of
https://github.com/Respect/Validation.git
synced 2026-03-15 14:55:44 +01:00
281 lines
8.7 KiB
PHP
281 lines
8.7 KiB
PHP
<?php
|
|
|
|
/*
|
|
* Copyright (c) Alexandre Gomes Gaigalas <alganet@gmail.com>
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Respect\Validation\Message;
|
|
|
|
use Respect\Validation\Exceptions\ComponentException;
|
|
use Respect\Validation\Result;
|
|
|
|
use function array_filter;
|
|
use function array_key_exists;
|
|
use function array_map;
|
|
use function array_reduce;
|
|
use function array_values;
|
|
use function count;
|
|
use function current;
|
|
use function is_array;
|
|
use function is_string;
|
|
use function Respect\Stringifier\stringify;
|
|
use function rtrim;
|
|
use function sprintf;
|
|
use function str_repeat;
|
|
|
|
use const PHP_EOL;
|
|
|
|
final readonly class StandardFormatter implements Formatter
|
|
{
|
|
public function __construct(
|
|
private Renderer $renderer = new StandardRenderer(),
|
|
) {
|
|
}
|
|
|
|
/** @param array<string|int, mixed> $templates */
|
|
public function main(Result $result, array $templates, Translator $translator): string
|
|
{
|
|
$selectedTemplates = $this->selectTemplates($result, $templates);
|
|
if (!$this->isFinalTemplate($result, $selectedTemplates)) {
|
|
foreach ($this->extractDeduplicatedChildren($result) as $child) {
|
|
return $this->main($this->resultWithPath($result, $child), $selectedTemplates, $translator);
|
|
}
|
|
}
|
|
|
|
return $this->renderer->render($this->getTemplated($result, $selectedTemplates), $translator);
|
|
}
|
|
|
|
/** @param array<string|int, mixed> $templates */
|
|
public function full(
|
|
Result $result,
|
|
array $templates,
|
|
Translator $translator,
|
|
int $depth = 0,
|
|
Result ...$siblings,
|
|
): string {
|
|
$selectedTemplates = $this->selectTemplates($result, $templates);
|
|
$isFinalTemplate = $this->isFinalTemplate($result, $selectedTemplates);
|
|
|
|
$rendered = '';
|
|
if ($this->isAlwaysVisible($result, ...$siblings) || $isFinalTemplate) {
|
|
$indentation = str_repeat(' ', $depth * 2);
|
|
$rendered .= sprintf(
|
|
'%s- %s' . PHP_EOL,
|
|
$indentation,
|
|
$this->renderer->render(
|
|
$this->getTemplated($depth > 0 ? $result->withDeepestPath() : $result, $selectedTemplates),
|
|
$translator,
|
|
),
|
|
);
|
|
$depth++;
|
|
}
|
|
|
|
if (!$isFinalTemplate) {
|
|
$results = array_map(
|
|
fn(Result $child) => $this->resultWithPath($result, $child),
|
|
$this->extractDeduplicatedChildren($result),
|
|
);
|
|
foreach ($results as $child) {
|
|
$rendered .= $this->full(
|
|
$child,
|
|
$selectedTemplates,
|
|
$translator,
|
|
$depth,
|
|
...array_filter($results, static fn(Result $sibling) => $sibling !== $child),
|
|
);
|
|
$rendered .= PHP_EOL;
|
|
}
|
|
}
|
|
|
|
return rtrim($rendered, PHP_EOL);
|
|
}
|
|
|
|
/**
|
|
* @param array<string|int, mixed> $templates
|
|
*
|
|
* @return array<string|int, mixed>
|
|
*/
|
|
public function array(Result $result, array $templates, Translator $translator): array
|
|
{
|
|
$selectedTemplates = $this->selectTemplates($result, $templates);
|
|
$deduplicatedChildren = $this->extractDeduplicatedChildren($result);
|
|
if (count($deduplicatedChildren) === 0 || $this->isFinalTemplate($result, $selectedTemplates)) {
|
|
return [
|
|
$result->getDeepestPath() ?? $result->id => $this->renderer->render(
|
|
$this->getTemplated($result->withDeepestPath(), $selectedTemplates),
|
|
$translator,
|
|
),
|
|
];
|
|
}
|
|
|
|
$messages = [];
|
|
foreach ($deduplicatedChildren as $child) {
|
|
$key = $child->getDeepestPath() ?? $child->id ?? 0;
|
|
$messages[$key] = $this->array(
|
|
$this->resultWithPath($result, $child),
|
|
$this->selectTemplates($child, $selectedTemplates),
|
|
$translator,
|
|
);
|
|
if (count($messages[$key]) !== 1) {
|
|
continue;
|
|
}
|
|
|
|
$messages[$key] = current($messages[$key]);
|
|
}
|
|
|
|
if (count($messages) > 1) {
|
|
$self = [
|
|
'__root__' => $this->renderer->render(
|
|
$this->getTemplated($result->withDeepestPath(), $selectedTemplates),
|
|
$translator,
|
|
),
|
|
];
|
|
|
|
return $self + $messages;
|
|
}
|
|
|
|
return $messages;
|
|
}
|
|
|
|
public function resultWithPath(Result $parent, Result $child): Result
|
|
{
|
|
if ($parent->path !== null && $child->path !== null && $child->path !== $parent->path) {
|
|
return $child->withPath($parent->path);
|
|
}
|
|
|
|
if ($parent->path !== null && $child->path === null) {
|
|
return $child->withPath($parent->path);
|
|
}
|
|
|
|
return $child;
|
|
}
|
|
|
|
private function isAlwaysVisible(Result $result, Result ...$siblings): bool
|
|
{
|
|
if ($result->hasPassed) {
|
|
return false;
|
|
}
|
|
|
|
if ($result->hasCustomTemplate()) {
|
|
return true;
|
|
}
|
|
|
|
$childrenAlwaysVisible = array_filter(
|
|
$result->children,
|
|
fn(Result $child) => $this->isAlwaysVisible($child, ...array_filter(
|
|
$result->children,
|
|
static fn(Result $sibling) => $sibling !== $child,
|
|
)),
|
|
);
|
|
if (count($childrenAlwaysVisible) !== 1) {
|
|
return true;
|
|
}
|
|
|
|
if (count($siblings) === 0) {
|
|
return false;
|
|
}
|
|
|
|
return array_reduce(
|
|
$siblings,
|
|
fn(bool $carry, Result $currentSibling) => $carry || $this->isAlwaysVisible(
|
|
$currentSibling,
|
|
...array_filter($siblings, static fn(Result $sibling) => $sibling !== $currentSibling),
|
|
),
|
|
true,
|
|
);
|
|
}
|
|
|
|
/** @param array<string|int, mixed> $templates */
|
|
private function getTemplated(Result $result, array $templates): Result
|
|
{
|
|
if ($result->hasCustomTemplate()) {
|
|
return $result;
|
|
}
|
|
|
|
foreach ([$result->path, $result->name, $result->id, '__root__'] as $key) {
|
|
if ($key === null || !isset($templates[$key])) {
|
|
continue;
|
|
}
|
|
|
|
if (is_string($templates[$key])) {
|
|
return $result->withTemplate($templates[$key]);
|
|
}
|
|
|
|
throw new ComponentException(
|
|
sprintf('Template for "%s" must be a string, %s given', $key, stringify($templates[$key])),
|
|
);
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
/** @param array<string|int, mixed> $templates */
|
|
private function isFinalTemplate(Result $result, array $templates): bool
|
|
{
|
|
$keys = array_filter([$result->path, $result->name, $result->id], static fn($key) => $key !== null);
|
|
foreach ($keys as $key) {
|
|
if (isset($templates[$key]) && is_string($templates[$key])) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (count($templates) !== 1) {
|
|
return false;
|
|
}
|
|
|
|
foreach ($keys as $key) {
|
|
if (isset($templates[$key])) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return isset($templates['__root__']);
|
|
}
|
|
|
|
/**
|
|
* @param array<string|int, mixed> $templates
|
|
*
|
|
* @return array<string|int, mixed>
|
|
*/
|
|
private function selectTemplates(Result $result, array $templates): array
|
|
{
|
|
foreach ([$result->path, $result->name, $result->id] as $key) {
|
|
if ($key !== null && isset($templates[$key]) && is_array($templates[$key])) {
|
|
return $templates[$key];
|
|
}
|
|
}
|
|
|
|
return $templates;
|
|
}
|
|
|
|
/** @return array<Result> */
|
|
private function extractDeduplicatedChildren(Result $result): array
|
|
{
|
|
/** @var array<string, Result> $deduplicatedResults */
|
|
$deduplicatedResults = [];
|
|
$duplicateCounters = [];
|
|
foreach ($result->children as $child) {
|
|
$id = $child->getDeepestPath() ?? $child->id;
|
|
if (isset($duplicateCounters[$id])) {
|
|
$id .= '.' . ++$duplicateCounters[$id];
|
|
} elseif (array_key_exists($id, $deduplicatedResults)) {
|
|
$deduplicatedResults[$id . '.1'] = $deduplicatedResults[$id]?->withId($id . '.1');
|
|
unset($deduplicatedResults[$id]);
|
|
$duplicateCounters[$id] = 2;
|
|
$id .= '.2';
|
|
}
|
|
|
|
if ($child->path === null) {
|
|
$deduplicatedResults[$id] = $child->hasPassed ? null : $child->withId($id);
|
|
continue;
|
|
}
|
|
|
|
$deduplicatedResults[$id] = $child->hasPassed ? null : $child;
|
|
}
|
|
|
|
return array_values(array_filter($deduplicatedResults));
|
|
}
|
|
}
|