Version 5.1 - All the GraphQL #32
@ -1,19 +1,21 @@
|
|||||||
[anime_list]
|
[anime_list]
|
||||||
route_prefix = "/anime"
|
route_prefix = ""
|
||||||
[anime_list.items]
|
[anime_list.items]
|
||||||
watching = '/watching'
|
watch_history = '/history/anime'
|
||||||
plan_to_watch = '/plan_to_watch'
|
watching = '/anime/watching'
|
||||||
on_hold = '/on_hold'
|
plan_to_watch = '/anime/plan_to_watch'
|
||||||
dropped = '/dropped'
|
on_hold = '/anime/on_hold'
|
||||||
completed = '/completed'
|
dropped = '/anime/dropped'
|
||||||
all = '/all'
|
completed = '/anime/completed'
|
||||||
|
all = '/anime/all'
|
||||||
|
|
||||||
[manga_list]
|
[manga_list]
|
||||||
route_prefix = "/manga"
|
route_prefix = ""
|
||||||
[manga_list.items]
|
[manga_list.items]
|
||||||
reading = '/reading'
|
reading_history = '/history/manga'
|
||||||
plan_to_read = '/plan_to_read'
|
reading = '/manga/reading'
|
||||||
on_hold = '/on_hold'
|
plan_to_read = '/manga/plan_to_read'
|
||||||
dropped = '/dropped'
|
on_hold = '/manga/on_hold'
|
||||||
completed = '/completed'
|
dropped = '/manga/dropped'
|
||||||
all = '/all'
|
completed = '/manga/completed'
|
||||||
|
all = '/manga/all'
|
@ -193,16 +193,6 @@ $routes = [
|
|||||||
'username' => '.*?'
|
'username' => '.*?'
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
'anime_history' => [
|
|
||||||
'controller' => 'history',
|
|
||||||
'path' => '/history/anime',
|
|
||||||
'action' => 'anime',
|
|
||||||
],
|
|
||||||
'manga_history' => [
|
|
||||||
'controller' => 'history',
|
|
||||||
'path' => '/history/manga',
|
|
||||||
'action' => 'manga',
|
|
||||||
],
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Default / Shared routes
|
// Default / Shared routes
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
@ -289,6 +279,13 @@ $routes = [
|
|||||||
'view' => ALPHA_SLUG_PATTERN,
|
'view' => ALPHA_SLUG_PATTERN,
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
'history' => [
|
||||||
|
'controller' => 'history',
|
||||||
|
'path' => '/history/{type}',
|
||||||
|
'tokens' => [
|
||||||
|
'type' => SLUG_PATTERN
|
||||||
|
]
|
||||||
|
],
|
||||||
'index_redirect' => [
|
'index_redirect' => [
|
||||||
'path' => '/',
|
'path' => '/',
|
||||||
'action' => 'redirectToDefaultRoute',
|
'action' => 'redirectToDefaultRoute',
|
||||||
|
45
app/views/history.php
Normal file
45
app/views/history.php
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<main class="details fixed">
|
||||||
|
<?php if (empty($items)): ?>
|
||||||
|
<h3>No recent history.</h3>
|
||||||
|
<?php else: ?>
|
||||||
|
<section>
|
||||||
|
<?php foreach ($items as $name => $item): ?>
|
||||||
|
<article class="flex flex-no-wrap flex-justify-start">
|
||||||
|
<section class="flex-self-center history-img"><?= $helper->picture(
|
||||||
|
$item['coverImg'],
|
||||||
|
'jpg',
|
||||||
|
['width' => '110px', 'height' => '156px'],
|
||||||
|
['width' => '110px', 'height' => '156px']
|
||||||
|
) ?></section>
|
||||||
|
<section class="flex-self-center">
|
||||||
|
<?= $item['title'] ?>
|
||||||
|
<br />
|
||||||
|
<br />
|
||||||
|
<?= $item['action'] ?>
|
||||||
|
<br />
|
||||||
|
<small>
|
||||||
|
<?php if ( ! empty($item['dateRange'])):
|
||||||
|
[$startDate, $endDate] = array_map(
|
||||||
|
fn ($date) => $date->format('l, F d'),
|
||||||
|
$item['dateRange']
|
||||||
|
);
|
||||||
|
[$startTime, $endTime] = array_map(
|
||||||
|
fn ($date) => $date->format('h:i:s A'),
|
||||||
|
$item['dateRange']
|
||||||
|
);
|
||||||
|
?>
|
||||||
|
<?php if ($startDate === $endDate): ?>
|
||||||
|
<?= "{$startDate}, {$startTime} – {$endTime}" ?>
|
||||||
|
<?php else: ?>
|
||||||
|
<?= "{$startDate} {$startTime} – {$endDate} {$endTime}" ?>
|
||||||
|
<?php endif ?>
|
||||||
|
<?php else: ?>
|
||||||
|
<?= $item['updated']->format('l, F d h:i:s A') ?>
|
||||||
|
<?php endif ?>
|
||||||
|
</small>
|
||||||
|
</section>
|
||||||
|
</article>
|
||||||
|
<?php endforeach ?>
|
||||||
|
</section>
|
||||||
|
<?php endif ?>
|
||||||
|
</main>
|
@ -1,19 +0,0 @@
|
|||||||
<main class="details fixed">
|
|
||||||
<?php if (empty($items)): ?>
|
|
||||||
<h3>No recent watch history.</h3>
|
|
||||||
<?php else: ?>
|
|
||||||
<section>
|
|
||||||
<?php foreach ($items as $name => $item): ?>
|
|
||||||
<article class="flex flex-no-wrap flex-justify-start">
|
|
||||||
<section class="flex-self-center history-img"><?= $helper->picture(
|
|
||||||
$item['coverImg'],
|
|
||||||
'jpg',
|
|
||||||
['width' => '110px', 'height' => '156px'],
|
|
||||||
['width' => '110px', 'height' => '156px']
|
|
||||||
) ?></section>
|
|
||||||
<section class="flex-self-center"><?= $item['action'] ?></section>
|
|
||||||
</article>
|
|
||||||
<?php endforeach ?>
|
|
||||||
</section>
|
|
||||||
<?php endif ?>
|
|
||||||
</main>
|
|
@ -5,8 +5,8 @@ namespace Aviat\AnimeClient;
|
|||||||
$whose = $config->get('whose_list') . "'s ";
|
$whose = $config->get('whose_list') . "'s ";
|
||||||
$lastSegment = $urlGenerator->lastSegment();
|
$lastSegment = $urlGenerator->lastSegment();
|
||||||
$extraSegment = $lastSegment === 'list' ? '/list' : '';
|
$extraSegment = $lastSegment === 'list' ? '/list' : '';
|
||||||
$hasAnime = stripos($_SERVER['REQUEST_URI'], 'anime') === 1;
|
$hasAnime = stripos($_SERVER['REQUEST_URI'], 'anime') !== FALSE;
|
||||||
$hasManga = stripos($_SERVER['REQUEST_URI'], 'manga') === 1;
|
$hasManga = stripos($_SERVER['REQUEST_URI'], 'manga') !== FALSE;
|
||||||
|
|
||||||
?>
|
?>
|
||||||
<div id="main-nav" class="flex flex-align-end flex-wrap">
|
<div id="main-nav" class="flex flex-align-end flex-wrap">
|
||||||
@ -79,10 +79,12 @@ $hasManga = stripos($_SERVER['REQUEST_URI'], 'manga') === 1;
|
|||||||
<nav>
|
<nav>
|
||||||
<?php if ($container->get('util')->isViewPage() && ($hasAnime || $hasManga)): ?>
|
<?php if ($container->get('util')->isViewPage() && ($hasAnime || $hasManga)): ?>
|
||||||
<?= $helper->menu($menu_name) ?>
|
<?= $helper->menu($menu_name) ?>
|
||||||
|
<?php if (stripos($_SERVER['REQUEST_URI'], 'history') === FALSE): ?>
|
||||||
<br />
|
<br />
|
||||||
<ul>
|
<ul>
|
||||||
<li class="<?= Util::isNotSelected('list', $lastSegment) ?>"><a href="<?= $urlGenerator->url($route_path) ?>">Cover View</a></li>
|
<li class="<?= Util::isNotSelected('list', $lastSegment) ?>"><a href="<?= $urlGenerator->url($route_path) ?>">Cover View</a></li>
|
||||||
<li class="<?= Util::isSelected('list', $lastSegment) ?>"><a href="<?= $urlGenerator->url("{$route_path}/list") ?>">List View</a></li>
|
<li class="<?= Util::isSelected('list', $lastSegment) ?>"><a href="<?= $urlGenerator->url("{$route_path}/list") ?>">List View</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
<?php endif ?>
|
<?php endif ?>
|
||||||
|
<?php endif ?>
|
||||||
</nav>
|
</nav>
|
||||||
|
27
index.php
27
index.php
@ -24,17 +24,10 @@ use function Aviat\Ion\_dir;
|
|||||||
|
|
||||||
setlocale(LC_CTYPE, 'en_US');
|
setlocale(LC_CTYPE, 'en_US');
|
||||||
|
|
||||||
// Work around the silly timezone error
|
|
||||||
$timezone = ini_get('date.timezone');
|
|
||||||
if ($timezone === '' || $timezone === FALSE)
|
|
||||||
{
|
|
||||||
ini_set('date.timezone', 'GMT');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load composer autoloader
|
// Load composer autoloader
|
||||||
require_once __DIR__ . '/vendor/autoload.php';
|
require_once __DIR__ . '/vendor/autoload.php';
|
||||||
|
|
||||||
// if (array_key_exists('ENV', $_ENV) && $_ENV['ENV'] === 'development')
|
if (array_key_exists('ENV', $_SERVER) && $_SERVER['ENV'] === 'development')
|
||||||
{
|
{
|
||||||
$whoops = new Run;
|
$whoops = new Run;
|
||||||
$whoops->pushHandler(new PrettyPageHandler);
|
$whoops->pushHandler(new PrettyPageHandler);
|
||||||
@ -62,6 +55,24 @@ $overrideConfig = file_exists($overrideFile)
|
|||||||
$configArray = array_replace_recursive($baseConfig, $config, $overrideConfig);
|
$configArray = array_replace_recursive($baseConfig, $config, $overrideConfig);
|
||||||
|
|
||||||
$checkedConfig = ConfigType::check($configArray);
|
$checkedConfig = ConfigType::check($configArray);
|
||||||
|
|
||||||
|
// Set the timezone for date display
|
||||||
|
// First look in app config, then PHP config, and at last
|
||||||
|
// resort, just set to UTC.
|
||||||
|
$timezone = ini_get('date.timezone');
|
||||||
|
if (array_key_exists('timezone', $checkedConfig) && ! empty($checkedConfig['timezone']))
|
||||||
|
{
|
||||||
|
date_default_timezone_set($checkedConfig['timezone']);
|
||||||
|
}
|
||||||
|
else if ($timezone !== '')
|
||||||
|
{
|
||||||
|
date_default_timezone_set($timezone);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
date_default_timezone_set('UTC');
|
||||||
|
}
|
||||||
|
|
||||||
$container = $di($checkedConfig);
|
$container = $di($checkedConfig);
|
||||||
|
|
||||||
// Unset 'constants'
|
// Unset 'constants'
|
||||||
|
@ -34,6 +34,7 @@ use Aviat\AnimeClient\API\Kitsu\Transformer\{
|
|||||||
AnimeHistoryTransformer,
|
AnimeHistoryTransformer,
|
||||||
AnimeTransformer,
|
AnimeTransformer,
|
||||||
AnimeListTransformer,
|
AnimeListTransformer,
|
||||||
|
MangaHistoryTransformer,
|
||||||
MangaTransformer,
|
MangaTransformer,
|
||||||
MangaListTransformer
|
MangaListTransformer
|
||||||
};
|
};
|
||||||
@ -184,7 +185,7 @@ final class Model {
|
|||||||
public function getAnimeHistory(): array
|
public function getAnimeHistory(): array
|
||||||
{
|
{
|
||||||
$raw = $this->getRawHistoryList('anime');
|
$raw = $this->getRawHistoryList('anime');
|
||||||
$organized = (array)JsonAPI::organizeData($raw);
|
$organized = JsonAPI::organizeData($raw);
|
||||||
|
|
||||||
$organized = array_filter($organized, fn ($item) => array_key_exists('relationships', $item));
|
$organized = array_filter($organized, fn ($item) => array_key_exists('relationships', $item));
|
||||||
|
|
||||||
@ -204,11 +205,14 @@ final class Model {
|
|||||||
public function getMangaHistory(): array
|
public function getMangaHistory(): array
|
||||||
{
|
{
|
||||||
$raw = $this->getRawHistoryList('manga');
|
$raw = $this->getRawHistoryList('manga');
|
||||||
$organized = (array)JsonAPI::organizeData($raw);
|
$organized = JsonAPI::organizeData($raw);
|
||||||
|
|
||||||
$organized = array_filter($organized, fn ($item) => array_key_exists('relationships', $item));
|
$organized = array_filter($organized, fn ($item) => array_key_exists('relationships', $item));
|
||||||
|
|
||||||
return $organized;
|
$transformer = new MangaHistoryTransformer();
|
||||||
|
$transformer->setContainer($this->getContainer());
|
||||||
|
|
||||||
|
return $transformer->transform($organized);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -989,7 +993,7 @@ final class Model {
|
|||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
* @throws Throwable
|
* @throws Throwable
|
||||||
*/
|
*/
|
||||||
protected function getRawHistoryList(string $type = 'anime', int $entries = 60): array
|
protected function getRawHistoryList(string $type = 'anime', int $entries = 120): array
|
||||||
{
|
{
|
||||||
$size = 20;
|
$size = 20;
|
||||||
$pages = ceil($entries / $size);
|
$pages = ceil($entries / $size);
|
||||||
|
@ -19,6 +19,9 @@ namespace Aviat\AnimeClient\API\Kitsu\Transformer;
|
|||||||
use Aviat\AnimeClient\API\Mapping\AnimeWatchingStatus;
|
use Aviat\AnimeClient\API\Mapping\AnimeWatchingStatus;
|
||||||
use Aviat\AnimeClient\Types\HistoryItem;
|
use Aviat\AnimeClient\Types\HistoryItem;
|
||||||
use Aviat\Ion\Di\ContainerAware;
|
use Aviat\Ion\Di\ContainerAware;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
use DateTimeInterface;
|
||||||
|
use DateTimeZone;
|
||||||
|
|
||||||
class AnimeHistoryTransformer {
|
class AnimeHistoryTransformer {
|
||||||
use ContainerAware;
|
use ContainerAware;
|
||||||
@ -49,7 +52,7 @@ class AnimeHistoryTransformer {
|
|||||||
|
|
||||||
$kind = $entry['attributes']['kind'];
|
$kind = $entry['attributes']['kind'];
|
||||||
|
|
||||||
if ($kind === 'progressed')
|
if ($kind === 'progressed' && ! empty($entry['attributes']['changedData']['progress']))
|
||||||
{
|
{
|
||||||
$output[] = $this->transformProgress($entry);
|
$output[] = $this->transformProgress($entry);
|
||||||
}
|
}
|
||||||
@ -100,25 +103,30 @@ class AnimeHistoryTransformer {
|
|||||||
if (count($entries) > 1)
|
if (count($entries) > 1)
|
||||||
{
|
{
|
||||||
$episodes = [];
|
$episodes = [];
|
||||||
|
$updated = [];
|
||||||
|
|
||||||
foreach ($entries as $e)
|
foreach ($entries as $e)
|
||||||
{
|
{
|
||||||
$episodes[] = max($e['original']['attributes']['changedData']['progress']);
|
$episodes[] = max($e['original']['attributes']['changedData']['progress']);
|
||||||
|
$updated[] = $e['updated'];
|
||||||
}
|
}
|
||||||
$firstEpisode = min($episodes);
|
$firstEpisode = min($episodes);
|
||||||
$lastEpisode = max($episodes);
|
$lastEpisode = max($episodes);
|
||||||
|
$firstUpdate = min($updated);
|
||||||
|
$lastUpdate = max($updated);
|
||||||
|
|
||||||
$title = $entries[0]['title'];
|
$title = $entries[0]['title'];
|
||||||
|
|
||||||
$action = (count($entries) > 3)
|
$action = (count($entries) > 3)
|
||||||
? "Marathoned episodes {$firstEpisode}-{$lastEpisode} of {$title}"
|
? "Marathoned episodes {$firstEpisode}-{$lastEpisode}"
|
||||||
: "Watched episodes {$firstEpisode}-{$lastEpisode} of {$title}";
|
: "Watched episodes {$firstEpisode}-{$lastEpisode}";
|
||||||
|
|
||||||
$output[] = HistoryItem::check([
|
$output[] = HistoryItem::from([
|
||||||
'title' => $title,
|
|
||||||
'action' => $action,
|
'action' => $action,
|
||||||
'coverImg' => $entries[0]['coverImg'],
|
'coverImg' => $entries[0]['coverImg'],
|
||||||
|
'dateRange' => [$firstUpdate, $lastUpdate],
|
||||||
'isAggregate' => true,
|
'isAggregate' => true,
|
||||||
|
'title' => $title,
|
||||||
'updated' => $entries[0]['updated'],
|
'updated' => $entries[0]['updated'],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@ -126,16 +134,14 @@ class AnimeHistoryTransformer {
|
|||||||
$i += count($entries) - 1;
|
$i += count($entries) - 1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
$output[] = $entry;
|
$output[] = $entry;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return $output;
|
return $output;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function transformProgress ($entry): array
|
protected function transformProgress ($entry): HistoryItem
|
||||||
{
|
{
|
||||||
$animeId = array_keys($entry['relationships']['anime'])[0];
|
$animeId = array_keys($entry['relationships']['anime'])[0];
|
||||||
$animeData = $entry['relationships']['anime'][$animeId]['attributes'];
|
$animeData = $entry['relationships']['anime'][$animeId]['attributes'];
|
||||||
@ -143,17 +149,17 @@ class AnimeHistoryTransformer {
|
|||||||
$imgUrl = 'images/anime/' . $animeId . '.webp';
|
$imgUrl = 'images/anime/' . $animeId . '.webp';
|
||||||
$episode = max($entry['attributes']['changedData']['progress']);
|
$episode = max($entry['attributes']['changedData']['progress']);
|
||||||
|
|
||||||
return HistoryItem::check([
|
return HistoryItem::from([
|
||||||
'action' => "Watched episode {$episode} of {$title}",
|
'action' => "Watched episode {$episode}",
|
||||||
'coverImg' => $imgUrl,
|
'coverImg' => $imgUrl,
|
||||||
'kind' => 'progressed',
|
'kind' => 'progressed',
|
||||||
'original' => $entry,
|
'original' => $entry,
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'updated' => $entry['attributes']['updatedAt'],
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function transformUpdated($entry): array
|
protected function transformUpdated($entry): HistoryItem
|
||||||
{
|
{
|
||||||
$animeId = array_keys($entry['relationships']['anime'])[0];
|
$animeId = array_keys($entry['relationships']['anime'])[0];
|
||||||
$animeData = $entry['relationships']['anime'][$animeId]['attributes'];
|
$animeData = $entry['relationships']['anime'][$animeId]['attributes'];
|
||||||
@ -169,23 +175,23 @@ class AnimeHistoryTransformer {
|
|||||||
|
|
||||||
if ($statusName === 'Completed')
|
if ($statusName === 'Completed')
|
||||||
{
|
{
|
||||||
return HistoryItem::check([
|
return HistoryItem::from([
|
||||||
'action' => "Completed {$title}",
|
'action' => 'Completed',
|
||||||
'coverImg' => $imgUrl,
|
'coverImg' => $imgUrl,
|
||||||
'kind' => 'updated',
|
'kind' => 'updated',
|
||||||
'original' => $entry,
|
'original' => $entry,
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'updated' => $entry['attributes']['updatedAt'],
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return HistoryItem::check([
|
return HistoryItem::from([
|
||||||
'action' => "Set status of {$title} to {$statusName}",
|
'action' => "Set status to {$statusName}",
|
||||||
'coverImg' => $imgUrl,
|
'coverImg' => $imgUrl,
|
||||||
'kind' => 'updated',
|
'kind' => 'updated',
|
||||||
'original' => $entry,
|
'original' => $entry,
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'updated' => $entry['attributes']['updatedAt'],
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -199,4 +205,14 @@ class AnimeHistoryTransformer {
|
|||||||
$helper = $this->getContainer()->get('html-helper');
|
$helper = $this->getContainer()->get('html-helper');
|
||||||
return $helper->a($url, $animeData['canonicalTitle'], ['id' => $animeData['slug']]);
|
return $helper->a($url, $animeData['canonicalTitle'], ['id' => $animeData['slug']]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function parseDate (string $date): DateTimeImmutable
|
||||||
|
{
|
||||||
|
$dateTime = DateTimeImmutable::createFromFormat(
|
||||||
|
DateTimeInterface::RFC3339_EXTENDED,
|
||||||
|
$date
|
||||||
|
);
|
||||||
|
|
||||||
|
return $dateTime->setTimezone(new DateTimeZone(date_default_timezone_get()));
|
||||||
|
}
|
||||||
}
|
}
|
@ -0,0 +1,218 @@
|
|||||||
|
<?php declare(strict_types=1);
|
||||||
|
/**
|
||||||
|
* Hummingbird Anime List Client
|
||||||
|
*
|
||||||
|
* An API client for Kitsu to manage anime and manga watch lists
|
||||||
|
*
|
||||||
|
* PHP version 7.4
|
||||||
|
*
|
||||||
|
* @package HummingbirdAnimeClient
|
||||||
|
* @author Timothy J. Warren <tim@timshomepage.net>
|
||||||
|
* @copyright 2015 - 2020 Timothy J. Warren
|
||||||
|
* @license http://www.opensource.org/licenses/mit-license.html MIT License
|
||||||
|
* @version 5
|
||||||
|
* @link https://git.timshomepage.net/timw4mail/HummingBirdAnimeClient
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Aviat\AnimeClient\API\Kitsu\Transformer;
|
||||||
|
|
||||||
|
use Aviat\AnimeClient\API\Mapping\MangaReadingStatus;
|
||||||
|
use Aviat\AnimeClient\Types\HistoryItem;
|
||||||
|
use Aviat\Ion\Di\ContainerAware;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
use DateTimeInterface;
|
||||||
|
use DateTimeZone;
|
||||||
|
|
||||||
|
class MangaHistoryTransformer {
|
||||||
|
use ContainerAware;
|
||||||
|
|
||||||
|
protected array $skipList = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert raw history
|
||||||
|
*
|
||||||
|
* @param array $data
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function transform(array $data): array
|
||||||
|
{
|
||||||
|
$output = [];
|
||||||
|
|
||||||
|
foreach ($data as $id => $entry)
|
||||||
|
{
|
||||||
|
if ( ! isset($entry['relationships']['manga']))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (in_array($id, $this->skipList, FALSE))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$kind = $entry['attributes']['kind'];
|
||||||
|
|
||||||
|
if ($kind === 'progressed' && ! empty($entry['attributes']['changedData']['progress']))
|
||||||
|
{
|
||||||
|
$output[] = $this->transformProgress($entry);
|
||||||
|
}
|
||||||
|
else if ($kind === 'updated')
|
||||||
|
{
|
||||||
|
$output[] = $this->transformUpdated($entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->aggregate($output);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Combine consecutive 'progressed' events
|
||||||
|
*
|
||||||
|
* @param array $singles
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
protected function aggregate (array $singles): array
|
||||||
|
{
|
||||||
|
$output = [];
|
||||||
|
|
||||||
|
$count = count($singles);
|
||||||
|
for ($i = 0; $i < $count; $i++)
|
||||||
|
{
|
||||||
|
$entries = [];
|
||||||
|
$entry = $singles[$i];
|
||||||
|
$prevTitle = $entry['title'];
|
||||||
|
$nextId = $i;
|
||||||
|
$next = $singles[$nextId];
|
||||||
|
while (
|
||||||
|
$next['kind'] === 'progressed' &&
|
||||||
|
$next['title'] === $prevTitle
|
||||||
|
) {
|
||||||
|
$entries[] = $next;
|
||||||
|
$prevTitle = $next['title'];
|
||||||
|
|
||||||
|
if ($nextId + 1 < $count)
|
||||||
|
{
|
||||||
|
$nextId++;
|
||||||
|
$next = $singles[$nextId];
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($entries) > 1)
|
||||||
|
{
|
||||||
|
$chapters = [];
|
||||||
|
$updated = [];
|
||||||
|
|
||||||
|
foreach ($entries as $e)
|
||||||
|
{
|
||||||
|
$chapters[] = max($e['original']['attributes']['changedData']['progress']);
|
||||||
|
$updated[] = $e['updated'];
|
||||||
|
}
|
||||||
|
$firstChapter = min($chapters);
|
||||||
|
$lastChapter = max($chapters);
|
||||||
|
$firstUpdate = min($updated);
|
||||||
|
$lastUpdate = max($updated);
|
||||||
|
|
||||||
|
$title = $entries[0]['title'];
|
||||||
|
|
||||||
|
$action = (count($entries) > 3)
|
||||||
|
? "Marathoned chapters {$firstChapter}-{$lastChapter}"
|
||||||
|
: "Watched chapters {$firstChapter}-{$lastChapter}";
|
||||||
|
|
||||||
|
$output[] = HistoryItem::from([
|
||||||
|
'action' => $action,
|
||||||
|
'coverImg' => $entries[0]['coverImg'],
|
||||||
|
'dateRange' => [$firstUpdate, $lastUpdate],
|
||||||
|
'isAggregate' => true,
|
||||||
|
'title' => $title,
|
||||||
|
'updated' => $entries[0]['updated'],
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Skip the rest of the aggregate in the main loop
|
||||||
|
$i += count($entries) - 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$output[] = $entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $output;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function transformProgress ($entry): HistoryItem
|
||||||
|
{
|
||||||
|
$mangaId = array_keys($entry['relationships']['manga'])[0];
|
||||||
|
$mangaData = $entry['relationships']['manga'][$mangaId]['attributes'];
|
||||||
|
$title = $this->linkTitle($mangaData);
|
||||||
|
$imgUrl = 'images/manga/' . $mangaId . '.webp';
|
||||||
|
$chapter = max($entry['attributes']['changedData']['progress']);
|
||||||
|
|
||||||
|
return HistoryItem::from([
|
||||||
|
'action' => "Watched chapter {$chapter}",
|
||||||
|
'coverImg' => $imgUrl,
|
||||||
|
'kind' => 'progressed',
|
||||||
|
'original' => $entry,
|
||||||
|
'title' => $title,
|
||||||
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function transformUpdated($entry): HistoryItem
|
||||||
|
{
|
||||||
|
$mangaId = array_keys($entry['relationships']['manga'])[0];
|
||||||
|
$mangaData = $entry['relationships']['manga'][$mangaId]['attributes'];
|
||||||
|
$title = $this->linkTitle($mangaData);
|
||||||
|
$imgUrl = 'images/manga/' . $mangaId . '.webp';
|
||||||
|
|
||||||
|
$kind = array_key_first($entry['attributes']['changedData']);
|
||||||
|
|
||||||
|
if ($kind === 'status')
|
||||||
|
{
|
||||||
|
$status = array_pop($entry['attributes']['changedData']['status']);
|
||||||
|
$statusName = MangaReadingStatus::KITSU_TO_TITLE[$status];
|
||||||
|
|
||||||
|
if ($statusName === 'Completed')
|
||||||
|
{
|
||||||
|
return HistoryItem::from([
|
||||||
|
'action' => 'Completed',
|
||||||
|
'coverImg' => $imgUrl,
|
||||||
|
'kind' => 'updated',
|
||||||
|
'original' => $entry,
|
||||||
|
'title' => $title,
|
||||||
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return HistoryItem::from([
|
||||||
|
'action' => "Set status to {$statusName}",
|
||||||
|
'coverImg' => $imgUrl,
|
||||||
|
'kind' => 'updated',
|
||||||
|
'original' => $entry,
|
||||||
|
'title' => $title,
|
||||||
|
'updated' => $this->parseDate($entry['attributes']['updatedAt']),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function linkTitle (array $mangaData): string
|
||||||
|
{
|
||||||
|
$url = '/manga/details/' . $mangaData['slug'];
|
||||||
|
|
||||||
|
$helper = $this->getContainer()->get('html-helper');
|
||||||
|
return $helper->a($url, $mangaData['canonicalTitle'], ['id' => $mangaData['slug']]);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseDate (string $date): DateTimeImmutable
|
||||||
|
{
|
||||||
|
$dateTime = DateTimeImmutable::createFromFormat(
|
||||||
|
DateTimeInterface::RFC3339_EXTENDED,
|
||||||
|
$date
|
||||||
|
);
|
||||||
|
|
||||||
|
return $dateTime->setTimezone(new DateTimeZone(date_default_timezone_get()));
|
||||||
|
}
|
||||||
|
}
|
@ -54,11 +54,33 @@ final class History extends BaseController {
|
|||||||
$this->mangaModel = $container->get('manga-model');
|
$this->mangaModel = $container->get('manga-model');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function anime(): void
|
public function index(string $type = 'anime'): void
|
||||||
{
|
{
|
||||||
|
if (method_exists($this, $type))
|
||||||
|
{
|
||||||
|
$this->$type();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->notFound(
|
||||||
|
$this->config->get('whose_list') .
|
||||||
|
"'s List · History · " .
|
||||||
|
'History Not Found',
|
||||||
|
'History Not Found'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function anime(): void
|
||||||
|
{
|
||||||
|
$this->baseData = array_merge($this->baseData, [
|
||||||
|
'menu_name' => 'anime_list',
|
||||||
|
'other_type' => 'manga',
|
||||||
|
'url_type' => 'anime',
|
||||||
|
]);
|
||||||
|
|
||||||
// $this->outputJSON($this->animeModel->getHistory());
|
// $this->outputJSON($this->animeModel->getHistory());
|
||||||
// return;
|
// return;
|
||||||
$this->outputHTML('history/anime', [
|
$this->outputHTML('history', [
|
||||||
'title' => $this->formatTitle(
|
'title' => $this->formatTitle(
|
||||||
$this->config->get('whose_list') . "'s Anime List",
|
$this->config->get('whose_list') . "'s Anime List",
|
||||||
'Anime',
|
'Anime',
|
||||||
@ -68,11 +90,17 @@ final class History extends BaseController {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function manga(): void
|
private function manga(): void
|
||||||
{
|
{
|
||||||
$this->outputJSON($this->mangaModel->getHistory());
|
$this->baseData = array_merge($this->baseData, [
|
||||||
return;
|
'menu_name' => 'manga_list',
|
||||||
$this->outputHTML('history/manga', [
|
'other_type' => 'anime',
|
||||||
|
'url_type' => 'manga',
|
||||||
|
]);
|
||||||
|
|
||||||
|
// $this->outputJSON($this->mangaModel->getHistory());
|
||||||
|
// return;
|
||||||
|
$this->outputHTML('history', [
|
||||||
'title' => $this->formatTitle(
|
'title' => $this->formatTitle(
|
||||||
$this->config->get('whose_list') . "'s Manga List",
|
$this->config->get('whose_list') . "'s Manga List",
|
||||||
'Manga',
|
'Manga',
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
namespace Aviat\AnimeClient\Model;
|
namespace Aviat\AnimeClient\Model;
|
||||||
|
|
||||||
|
use function is_array;
|
||||||
use const Aviat\AnimeClient\SETTINGS_MAP;
|
use const Aviat\AnimeClient\SETTINGS_MAP;
|
||||||
|
|
||||||
use function Aviat\AnimeClient\arrayToToml;
|
use function Aviat\AnimeClient\arrayToToml;
|
||||||
@ -124,14 +125,14 @@ final class Settings {
|
|||||||
|
|
||||||
public function validateSettings(array $settings): array
|
public function validateSettings(array $settings): array
|
||||||
{
|
{
|
||||||
$config = (new Config($settings))->toArray();
|
$cfg = Config::check($settings);
|
||||||
|
|
||||||
$looseConfig = [];
|
$looseConfig = [];
|
||||||
$keyedConfig = [];
|
$keyedConfig = [];
|
||||||
|
|
||||||
// Convert 'boolean' values to true and false
|
// Convert 'boolean' values to true and false
|
||||||
// Also order keys so they can be saved properly
|
// Also order keys so they can be saved properly
|
||||||
foreach ($config as $key => $val)
|
foreach ($cfg as $key => $val)
|
||||||
{
|
{
|
||||||
if (is_scalar($val))
|
if (is_scalar($val))
|
||||||
{
|
{
|
||||||
@ -148,7 +149,7 @@ final class Settings {
|
|||||||
$looseConfig[$key] = $val;
|
$looseConfig[$key] = $val;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
elseif (\is_array($val) && ! empty($val))
|
elseif (is_array($val) && ! empty($val))
|
||||||
{
|
{
|
||||||
foreach($val as $k => $v)
|
foreach($val as $k => $v)
|
||||||
{
|
{
|
||||||
|
@ -52,6 +52,11 @@ class Config extends AbstractType {
|
|||||||
*/
|
*/
|
||||||
public $dark_theme; /* Deprecated */
|
public $dark_theme; /* Deprecated */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string The PHP timezone
|
||||||
|
*/
|
||||||
|
public string $timezone = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default Anime list status page, values are listed in
|
* Default Anime list status page, values are listed in
|
||||||
* Aviat\AnimeClient\API\Enum\AnimeWatchingStatus\Title
|
* Aviat\AnimeClient\API\Enum\AnimeWatchingStatus\Title
|
||||||
|
@ -16,6 +16,8 @@
|
|||||||
|
|
||||||
namespace Aviat\AnimeClient\Types;
|
namespace Aviat\AnimeClient\Types;
|
||||||
|
|
||||||
|
use DateTimeImmutable;
|
||||||
|
|
||||||
class HistoryItem extends AbstractType {
|
class HistoryItem extends AbstractType {
|
||||||
/**
|
/**
|
||||||
* @var string Title of the anime/manga
|
* @var string Title of the anime/manga
|
||||||
@ -43,9 +45,17 @@ class HistoryItem extends AbstractType {
|
|||||||
public string $kind = '';
|
public string $kind = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var string When the item was last updated
|
* @var DateTimeImmutable When the item was last updated
|
||||||
*/
|
*/
|
||||||
public string $updated = '';
|
public ?DateTimeImmutable $updated = NULL;
|
||||||
|
|
||||||
public $original;
|
/**
|
||||||
|
* @var array Range of updated times for the aggregated item
|
||||||
|
*/
|
||||||
|
public array $dateRange = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array The item before transformation
|
||||||
|
*/
|
||||||
|
public array $original = [];
|
||||||
}
|
}
|
@ -147,6 +147,12 @@ const SETTINGS_MAP = [
|
|||||||
'default' => 'Somebody',
|
'default' => 'Somebody',
|
||||||
'description' => 'Name of the owner of the list data.',
|
'description' => 'Name of the owner of the list data.',
|
||||||
],
|
],
|
||||||
|
'timezone' => [
|
||||||
|
'type' => 'string',
|
||||||
|
'title' => 'Timezone',
|
||||||
|
'default' => 'America/Detroit',
|
||||||
|
'description' => 'See https://www.php.net/manual/en/timezones.php for options'
|
||||||
|
],
|
||||||
'theme' => [
|
'theme' => [
|
||||||
'type' => 'select',
|
'type' => 'select',
|
||||||
'title' => 'Theme',
|
'title' => 'Theme',
|
||||||
|
Loading…
Reference in New Issue
Block a user