2016-08-04 11:51:12 +02:00
|
|
|
<?php
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2016-08-04 11:51:12 +02:00
|
|
|
class ShanaprojectBridge extends BridgeAbstract
|
|
|
|
{
|
2017-02-11 16:16:56 +01:00
|
|
|
const MAINTAINER = 'logmanoriginal';
|
|
|
|
const NAME = 'Shanaproject Bridge';
|
2019-07-02 20:45:31 +02:00
|
|
|
const URI = 'https://www.shanaproject.com';
|
2017-02-11 16:16:56 +01:00
|
|
|
const DESCRIPTION = 'Returns a list of anime from the current Season Anime List';
|
2019-07-02 21:16:23 +02:00
|
|
|
const PARAMETERS = [
|
2022-07-01 15:10:30 +02:00
|
|
|
[
|
2019-07-02 21:16:23 +02:00
|
|
|
'min_episodes' => [
|
|
|
|
'name' => 'Minimum Episodes',
|
|
|
|
'type' => 'number',
|
|
|
|
'title' => 'Minimum number of episodes before including in feed',
|
|
|
|
'defaultValue' => 0,
|
|
|
|
],
|
|
|
|
'min_total_episodes' => [
|
|
|
|
'name' => 'Minimum Total Episodes',
|
|
|
|
'type' => 'number',
|
|
|
|
'title' => 'Minimum total number of episodes before including in feed',
|
|
|
|
'defaultValue' => 0,
|
|
|
|
],
|
|
|
|
'require_banner' => [
|
|
|
|
'name' => 'Require Banner',
|
|
|
|
'type' => 'checkbox',
|
|
|
|
'title' => 'Only include anime with custom banner image',
|
|
|
|
'defaultValue' => false,
|
|
|
|
],
|
|
|
|
],
|
|
|
|
];
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:46:38 +02:00
|
|
|
private $uri;
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:46:38 +02:00
|
|
|
public function getURI()
|
|
|
|
{
|
|
|
|
return $this->uri ?? parent::getURI();
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
public function collectData()
|
|
|
|
{
|
|
|
|
$html = $this->loadSeasonAnimeList();
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
$animes = $html->find('div.header_display_box_info')
|
|
|
|
or returnServerError('Could not find anime headers!');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
$min_episodes = $this->getInput('min_episodes') ?: 0;
|
|
|
|
$min_total_episodes = $this->getInput('min_total_episodes') ?: 0;
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
foreach ($animes as $anime) {
|
Fix coding style missed by phpbcf (#2901)
$ composer require --dev friendsofphp/php-cs-fixer
$ echo >.php-cs-fixer.dist.php "<?php
$finder = PhpCsFixer\Finder::create()
->in(__DIR__);
$rules = [
'@PSR12' => true,
// '@PSR12:risky' => true,
'@PHP74Migration' => true,
// '@PHP74Migration:risky' => true,
// buggy, duplicates existing comment sometimes
'no_break_comment' => false,
'array_syntax' => true,
'lowercase_static_reference' => true,
'visibility_required' => false,
// Too much noise
'binary_operator_spaces' => false,
'heredoc_indentation' => false,
'trailing_comma_in_multiline' => false,
];
$config = new PhpCsFixer\Config();
return $config
->setRules($rules)
// ->setRiskyAllowed(true)
->setFinder($finder);
"
$ vendor/bin/php-cs-fixer --version
PHP CS Fixer 3.8.0 BerSzcz against war! by Fabien Potencier and Dariusz Ruminski.
PHP runtime: 8.1.7
$ vendor/bin/php-cs-fixer fix
$ rm .php-cs-fixer.cache
$ vendor/bin/php-cs-fixer fix
2022-07-08 13:00:52 +02:00
|
|
|
[
|
2019-07-02 21:16:23 +02:00
|
|
|
$episodes_released,
|
|
|
|
/* of */,
|
|
|
|
$episodes_total
|
|
|
|
] = explode(' ', $this->extractAnimeEpisodeInformation($anime));
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
// Skip if not enough episodes yet
|
|
|
|
if ($episodes_released < $min_episodes) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
// Skip if too many episodes in total
|
|
|
|
if ($episodes_total !== '?' && $episodes_total < $min_total_episodes) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
// Skip if https://static.shanaproject.com/no-art.jpg
|
|
|
|
if (
|
|
|
|
$this->getInput('require_banner')
|
|
|
|
&& strpos($this->extractAnimeBackgroundImage($anime), 'no-art') !== false
|
|
|
|
) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
$this->items[] = [
|
|
|
|
'title' => $this->extractAnimeTitle($anime),
|
2019-07-02 21:16:23 +02:00
|
|
|
'author' => $this->extractAnimeAuthor($anime),
|
2019-07-02 20:45:31 +02:00
|
|
|
'uri' => $this->extractAnimeUri($anime),
|
|
|
|
'timestamp' => $this->extractAnimeTimestamp($anime),
|
|
|
|
'content' => $this->buildAnimeContent($anime),
|
|
|
|
];
|
2022-07-01 15:10:30 +02:00
|
|
|
}
|
2019-07-02 20:45:31 +02:00
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Returns an html object for the Season Anime List (latest season)
|
|
|
|
private function loadSeasonAnimeList()
|
|
|
|
{
|
2022-01-02 14:36:09 +05:00
|
|
|
$html = getSimpleHTMLDOM(self::URI . '/seasons');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
$html = defaultLinkTo($html, self::URI . '/seasons');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
$season = $html->find('div.follows_menu > a', 1)
|
|
|
|
or returnServerError('Could not find \'Season Anime List\'!');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-01-02 14:36:09 +05:00
|
|
|
$html = getSimpleHTMLDOM($season->href);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:46:38 +02:00
|
|
|
$this->uri = $season->href;
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
$html = defaultLinkTo($html, $season->href);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
return $html;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the anime title
|
|
|
|
private function extractAnimeTitle($anime)
|
|
|
|
{
|
2019-07-02 20:45:31 +02:00
|
|
|
$title = $anime->find('a', 0)
|
|
|
|
or returnServerError('Could not find anime title!');
|
2017-02-11 16:16:56 +01:00
|
|
|
return trim($title->innertext);
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the anime URI
|
|
|
|
private function extractAnimeUri($anime)
|
|
|
|
{
|
2019-07-02 20:45:31 +02:00
|
|
|
$uri = $anime->find('a', 0)
|
|
|
|
or returnServerError('Could not find anime URI!');
|
|
|
|
return $uri->href;
|
2017-02-11 16:16:56 +01:00
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the anime release date (timestamp)
|
|
|
|
private function extractAnimeTimestamp($anime)
|
|
|
|
{
|
|
|
|
$timestamp = $anime->find('span.header_info_block', 1);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
if (!$timestamp) {
|
2017-04-10 13:38:02 +02:00
|
|
|
return null;
|
2019-07-02 20:45:31 +02:00
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
return strtotime($timestamp->innertext);
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the anime studio name (author)
|
|
|
|
private function extractAnimeAuthor($anime)
|
|
|
|
{
|
|
|
|
$author = $anime->find('span.header_info_block', 2);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
if (!$author) {
|
|
|
|
return null; // Sometimes the studio is unknown, so leave empty
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
return trim($author->innertext);
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the episode information (x of y released)
|
|
|
|
private function extractAnimeEpisodeInformation($anime)
|
|
|
|
{
|
2019-07-02 20:45:31 +02:00
|
|
|
$episode = $anime->find('div.header_info_episode', 0)
|
|
|
|
or returnServerError('Could not find anime episode information!');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
$retVal = preg_replace('/\r|\n/', ' ', $episode->plaintext);
|
|
|
|
$retVal = preg_replace('/\s+/', ' ', $retVal);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 21:16:23 +02:00
|
|
|
return $retVal;
|
2017-02-11 16:16:56 +01:00
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Extracts the background image
|
|
|
|
private function extractAnimeBackgroundImage($anime)
|
|
|
|
{
|
|
|
|
// Getting the picture is a little bit tricky as it is part of the style.
|
|
|
|
// Luckily the style is part of the parent div :)
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2019-07-02 20:45:31 +02:00
|
|
|
if (preg_match('/url\(\/\/([^\)]+)\)/i', $anime->parent->style, $matches)) {
|
2017-02-11 16:16:56 +01:00
|
|
|
return $matches[1];
|
2019-07-02 20:45:31 +02:00
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
returnServerError('Could not extract background image!');
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Builds an URI to search for a specific anime (subber is left empty)
|
|
|
|
private function buildAnimeSearchUri($anime)
|
|
|
|
{
|
2019-07-02 20:45:31 +02:00
|
|
|
return self::URI
|
2017-02-11 16:16:56 +01:00
|
|
|
. '/search/?title='
|
|
|
|
. urlencode($this->extractAnimeTitle($anime))
|
|
|
|
. '&subber=';
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2017-02-11 16:16:56 +01:00
|
|
|
// Builds the content string for a given anime
|
|
|
|
private function buildAnimeContent($anime)
|
|
|
|
{
|
|
|
|
// We'll use a template string to place our contents
|
|
|
|
return '<a href="'
|
|
|
|
. $this->extractAnimeUri($anime)
|
|
|
|
. '"><img src="http://'
|
|
|
|
. $this->extractAnimeBackgroundImage($anime)
|
|
|
|
. '" alt="'
|
|
|
|
. htmlspecialchars($this->extractAnimeTitle($anime))
|
|
|
|
. '" style="border: 1px solid black"></a><br><p>'
|
|
|
|
. $this->extractAnimeEpisodeInformation($anime)
|
|
|
|
. '</p><br><p><a href="'
|
|
|
|
. $this->buildAnimeSearchUri($anime)
|
|
|
|
. '">Search episodes</a></p>';
|
|
|
|
}
|
2016-08-04 11:51:12 +02:00
|
|
|
}
|