Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 0430dba6a9 | |||
| beed7d6940 | |||
| 924472ed56 | |||
| 7dd61355b7 | |||
| 2a1f69edd4 |
@@ -41,6 +41,8 @@ export default class extends Controller {
|
||||
if (this.countValue === this.totalValue) {
|
||||
this.toggleIcon();
|
||||
this.countValue = 0;
|
||||
console.log('filtering')
|
||||
document.getElementById('filter').filterResults();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,14 +12,14 @@ export default class extends Controller {
|
||||
seasons = []
|
||||
|
||||
activeFilter = {
|
||||
"resolution": "",
|
||||
"codec": "",
|
||||
"language": "",
|
||||
"provider": "",
|
||||
"quality": "",
|
||||
"resolution": [],
|
||||
"codec": [],
|
||||
"language": [],
|
||||
"provider": [],
|
||||
"quality": [],
|
||||
}
|
||||
|
||||
defaultOptions = '<option value="">n/a</option><option value="-">-</option>';
|
||||
defaultOptions = '<option value="-">-</option>';
|
||||
|
||||
static outlets = ['tv-episode-list']
|
||||
static targets = ['resolution', 'codec', 'language', 'provider', 'season', 'quality', 'loadingIcon', 'selectAll', 'downloadSelected']
|
||||
@@ -31,13 +31,21 @@ export default class extends Controller {
|
||||
}
|
||||
|
||||
async connect() {
|
||||
await this.setInitialFilter();
|
||||
this.setTimerToStopLoadingIcon();
|
||||
this.element.filterResults = this.filter.bind(this);
|
||||
document.addEventListener('optionsLoaded', this.loadOptions.bind(this));
|
||||
}
|
||||
|
||||
async setInitialFilter() {
|
||||
const response = await fetch('/api/user/filters');
|
||||
const filters = await response.json();
|
||||
if (filters.length > 0) {
|
||||
this.activeFilter = filters[0];
|
||||
}
|
||||
if (this.mediaTypeValue === "tvshows") {
|
||||
this.activeFilter['season'] = 1;
|
||||
}
|
||||
this.filter();
|
||||
this.setTimerToStopLoadingIcon();
|
||||
|
||||
document.addEventListener('optionsLoaded', this.loadOptions.bind(this));
|
||||
}
|
||||
|
||||
setTimerToStopLoadingIcon() {
|
||||
@@ -45,13 +53,10 @@ export default class extends Controller {
|
||||
}
|
||||
|
||||
// Event is fired from movies/tvshows controllers to populate this data
|
||||
loadOptions({detail: { options }}) {
|
||||
options.forEach((option) => {
|
||||
this.addLanguages(option);
|
||||
this.addProviders(option);
|
||||
this.addQualities(option);
|
||||
async loadOptions({detail: { options }}) {
|
||||
await options.forEach((option) => {
|
||||
option.filter({detail: {activeFilter: this.activeFilter }});
|
||||
})
|
||||
this.filter();
|
||||
}
|
||||
|
||||
selectAllEpisodes() {
|
||||
@@ -66,41 +71,6 @@ export default class extends Controller {
|
||||
document.dispatchEvent(new CustomEvent('downloadSelectedEpisodes', {}));
|
||||
}
|
||||
|
||||
addLanguages(option) {
|
||||
option.languages.forEach((language) => {
|
||||
if (!this.languages.includes(language)) {
|
||||
this.languages.push(language);
|
||||
}
|
||||
});
|
||||
const preferred = JSON.parse(this.languageTarget.dataset.preferred) ?? [];
|
||||
this.languageTarget.innerHTML = this.#serializeSelectOptions(this.languages);
|
||||
this.languageTarget.tomselect.items = preferred;
|
||||
}
|
||||
|
||||
addProviders(option) {
|
||||
if (!this.providers.includes(option.provider)) {
|
||||
this.providers.push(option.provider);
|
||||
}
|
||||
const preferred = JSON.parse(this.providerTarget.dataset.preferred) ?? [];
|
||||
this.providerTarget.innerHTML = this.#serializeSelectOptions(this.providers);
|
||||
this.providerTarget.tomselect.items = preferred;
|
||||
|
||||
}
|
||||
|
||||
addQualities(option) {
|
||||
if (option.quality.toLowerCase() in this.reverseMappedQualitiesValue) {
|
||||
let quality = this.reverseMappedQualitiesValue[option.quality.toLowerCase()];
|
||||
// converts api returned quality with a value the system recognizes
|
||||
option.quality = quality;
|
||||
if (!this.qualities.includes(option.quality)) {
|
||||
this.qualities.push(quality);
|
||||
}
|
||||
}
|
||||
const preferred = JSON.parse(this.qualityTarget.dataset.preferred) ?? [];
|
||||
this.qualityTarget.innerHTML = this.#serializeSelectOptions(this.qualities);
|
||||
this.qualityTarget.tomselect.items = preferred;
|
||||
}
|
||||
|
||||
filter() {
|
||||
const downloadSeasonSpan = document.querySelector("#downloadSeasonModal");
|
||||
|
||||
|
||||
@@ -30,11 +30,11 @@ export default class extends Controller {
|
||||
option.querySelector('.download-btn').dataset['title'] = this.titleValue
|
||||
);
|
||||
this.element.options[0].querySelector('input[type="checkbox"]').checked = true;
|
||||
this.loadingIconOutlet.increaseCount();
|
||||
document.dispatchEvent(new CustomEvent('optionsLoaded', {detail: {options: this.element.options}}));
|
||||
} else {
|
||||
this.countTarget.innerText = 0;
|
||||
this.episodeSelectorTarget.disabled = true;
|
||||
}
|
||||
this.loadingIconOutlet.increaseCount();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -165,9 +165,9 @@ dialog[data-dialog-target="dialog"][closing] {
|
||||
padding: 0;
|
||||
|
||||
.ts-control {
|
||||
background: transparent !important;
|
||||
border: none !important;
|
||||
box-shadow: none !important;
|
||||
@apply bg-orange-500/60 backdrop-filter backdrop-blur-md;
|
||||
}
|
||||
|
||||
.item[data-ts-item] {
|
||||
@@ -175,10 +175,10 @@ dialog[data-dialog-target="dialog"][closing] {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
text-shadow: none;
|
||||
@apply bg-orange-500 rounded-ms font-bold;
|
||||
@apply bg-orange-500 rounded-ms font-bold text-black;
|
||||
}
|
||||
|
||||
@apply border-b-2 border-b-orange-600 bg-transparent;
|
||||
@apply border border-orange-500 bg-transparent rounded-ms;
|
||||
}
|
||||
|
||||
.ts-wrapper.plugin-remove_button:not(.rtl) .item .remove {
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
|
||||
namespace App\Base\Framework\Controller;
|
||||
|
||||
use App\Monitor\Action\Command\MonitorTvShowCommand;
|
||||
use App\Monitor\Action\Handler\MonitorTvShowHandler;
|
||||
use App\Tmdb\Tmdb;
|
||||
use App\User\Framework\Entity\User;
|
||||
@@ -48,4 +49,13 @@ final class IndexController extends AbstractController
|
||||
'message' => 'Email sent!'
|
||||
]);
|
||||
}
|
||||
|
||||
#[Route('/test')]
|
||||
public function monitorTvShow(): Response
|
||||
{
|
||||
$this->monitorTvShowHandler->handle(new MonitorTvShowCommand(96));
|
||||
return $this->json([
|
||||
'Success' => 'Monitor added'
|
||||
]);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -4,7 +4,7 @@ namespace App\Base\Service;
|
||||
|
||||
use Aimeos\Map;
|
||||
use App\Download\Framework\Entity\Download;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||
use Symfony\Component\Filesystem\Filesystem;
|
||||
|
||||
246
src/Base/Util/PTN.php
Normal file
246
src/Base/Util/PTN.php
Normal file
@@ -0,0 +1,246 @@
|
||||
<?php
|
||||
/**
|
||||
* Parse Torrent Name (PTN)
|
||||
*
|
||||
* PHP port of parse-torrent-name written in Python.
|
||||
*
|
||||
* Javascript version by jzjzjzj
|
||||
* https://github.com/jzjzjzj/parse-torrent-name
|
||||
*
|
||||
* Python version by divijbindlish
|
||||
* https://github.com/divijbindlish/parse-torrent-name
|
||||
*
|
||||
* Copyright (c) 2014 - 2018, British Columbia Institute of Technology
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*
|
||||
* @package PTN
|
||||
* @author Drew Smith
|
||||
* @copyright copyright (c) 2018, Nihilarr (https://www.nihilarr.com)
|
||||
* @license http://opensource.org/licenses/MIT MIT License
|
||||
* @link https://gitlab.com/nihilarr/parse-torrent-name
|
||||
* @version 0.0.1
|
||||
*/
|
||||
|
||||
namespace App\Base\Util;
|
||||
|
||||
class PTN {
|
||||
|
||||
public $torrent;
|
||||
public $excess_raw;
|
||||
public $group_raw;
|
||||
public $start;
|
||||
public $end;
|
||||
public $title_raw;
|
||||
public $parts;
|
||||
|
||||
public $patterns = array(
|
||||
array('season' => '(s?([0-9]{1,3}))[ex]'),
|
||||
array('episode' => '([ex]([0-9]{1,3})(?:[^0-9]|$))'),
|
||||
array('year' => '([\[\(]?((?:19[0-9]|20[01])[0-9])[\]\)]?)'),
|
||||
array('resolution' => '([0-9]{3,4}p)'),
|
||||
array('quality' => '((?:PPV\.)?[HP]DTV|(?:HD)?CAM|B[DR]Rip|(?:HD-?)?TS|(?:PPV )?WEB-?DL(?: DVDRip)?|HDRip|DVDRip|DVDRIP|CamRip|W[EB]BRip|BluRay|DvDScr|hdtv|telesync)'),
|
||||
array('codec' => '(xvid|[hx]\.?26[45])'),
|
||||
array('audio' => '(MP3|DD5\.?1|Dual[\- ]Audio|LiNE|DTS|AAC[.-]LC|AAC(?:\.?2\.0)?|AC3(?:\.5\.1)?)'),
|
||||
array('group' => '(- ?([^-]+(?:-={[^-]+-?$)?))$'),
|
||||
array('region' => 'R[0-9]'),
|
||||
array('extended' => '(EXTENDED(:?.CUT)?)'),
|
||||
array('hardcoded' => 'HC'),
|
||||
array('proper' => 'PROPER'),
|
||||
array('repack' => 'REPACK'),
|
||||
array('container' => '(MKV|AVI|MP4)'),
|
||||
array('widescreen' => 'WS'),
|
||||
array('website' => '^(\[ ?([^\]]+?) ?\])'),
|
||||
array('language' => '(rus\.eng|ita\.eng)'),
|
||||
array('sbs' => '(?:Half-)?SBS'),
|
||||
array('unrated' => 'UNRATED'),
|
||||
array('size' => '(\d+(?:\.\d+)?(?:GB|MB))'),
|
||||
array('3d' => '3D')
|
||||
);
|
||||
|
||||
public $types = array(
|
||||
'season' => 'integer',
|
||||
'episode' => 'integer',
|
||||
'year' => 'integer',
|
||||
'extended' => 'boolean',
|
||||
'hardcoded' => 'boolean',
|
||||
'proper' => 'boolean',
|
||||
'repack' => 'boolean',
|
||||
'widescreen' => 'boolean',
|
||||
'unrated' => 'boolean',
|
||||
'3d' => 'boolean'
|
||||
);
|
||||
|
||||
public function __construct() {}
|
||||
|
||||
public function parse($name) {
|
||||
$this->parts = array();
|
||||
$this->torrent = array('name' => $name);
|
||||
$this->excess_raw = $name;
|
||||
$this->group_raw = '';
|
||||
$this->start = 0;
|
||||
$this->end = null;
|
||||
$this->title_raw = null;
|
||||
|
||||
foreach($this->patterns as $patterns_single) {
|
||||
foreach($patterns_single as $key => $pattern) {
|
||||
if(!in_array($key, array('season', 'episode', 'website'))) {
|
||||
$pattern = "\b{$pattern}\b";
|
||||
}
|
||||
|
||||
$clean_name = str_replace('_', ' ', $this->torrent['name']);
|
||||
if(preg_match("/{$pattern}/i", $clean_name, $match) == 0) break;
|
||||
|
||||
$index = array();
|
||||
if(is_array($match)) {
|
||||
array_shift($match);
|
||||
}
|
||||
if(sizeof($match) == 0) break;
|
||||
if(sizeof($match) > 1) {
|
||||
$index['raw'] = 0;
|
||||
$index['clean'] = 1;
|
||||
}
|
||||
else {
|
||||
$index['raw'] = 0;
|
||||
$index['clean'] = 0;
|
||||
}
|
||||
|
||||
if(isset($this->types[$key]) && $this->types[$key] == 'boolean') {
|
||||
$clean = true;
|
||||
}
|
||||
else {
|
||||
$clean = $match[$index['clean']];
|
||||
if(isset($this->types[$key]) && $this->types[$key] == 'integer') {
|
||||
$clean = (int)$clean;
|
||||
}
|
||||
}
|
||||
|
||||
if($key == 'group') {
|
||||
if((isset($this->patterns[5][1]) && preg_match_all("/{$this->patterns[5][1]}/i", $clean)) ||
|
||||
(isset($this->patterns[4][1]) && preg_match_all("/{$this->patterns[4][1]}/", $clean))) {
|
||||
break;
|
||||
}
|
||||
if(preg_match('/[^ ]+ [^ ]+ .+/', $clean)) {
|
||||
$key = 'episodeName';
|
||||
}
|
||||
}
|
||||
if($key == 'episode') {
|
||||
$sub_pattern = $this->escape_regex($match[$index['raw']]);
|
||||
$this->torrent['map'] = preg_replace("/{$sub_pattern}/", '{episode}', $this->torrent['name']);
|
||||
}
|
||||
|
||||
$this->part($key, $match, $match[$index['raw']], $clean);
|
||||
}
|
||||
}
|
||||
|
||||
$raw = $this->torrent['name'];
|
||||
if(!is_null($this->end)) {
|
||||
$raw = explode('(', substr($raw, $this->start, $this->end - $this->start));
|
||||
$raw = $raw[0];
|
||||
}
|
||||
|
||||
$clean = preg_replace("/^ -/", '', $raw);
|
||||
if(strpos($clean, ' ') === false && strpos($clean, '.') !== false) {
|
||||
$clean = str_replace('.', ' ', $clean);
|
||||
}
|
||||
$clean = str_replace('_', ' ', $clean);
|
||||
$clean = trim(preg_replace("/([\[\(_]|- )$/", '', $clean));
|
||||
|
||||
$this->part('title', array(), $raw, $clean);
|
||||
|
||||
$clean = preg_replace("/(^[-\. ()]+)|([-\. ]+$)/", '', $this->excess_raw);
|
||||
$clean = preg_replace("/[\(\)\/]/", ' ', $clean);
|
||||
$match = preg_split("/\.\.+| +/", $clean);
|
||||
if(sizeof($match) > 0 && is_array($match[0])) {
|
||||
$match = $match[0];
|
||||
}
|
||||
|
||||
$clean = $match;
|
||||
$clean = array_filter($clean, function($var) {
|
||||
return $var != '-' ? true : false;
|
||||
});
|
||||
$clean = array_filter($clean, function($var) {
|
||||
return trim($var, '-');
|
||||
});
|
||||
$clean = array_values($clean);
|
||||
|
||||
if(sizeof($clean) > 0) {
|
||||
$group_pattern = $clean[sizeof($clean) - 1] . $this->group_raw;
|
||||
if(strpos($this->torrent['name'], $group_pattern) == strlen($this->torrent['name']) - strlen($group_pattern)) {
|
||||
$this->late('group', array_pop($clean) . $this->group_raw);
|
||||
}
|
||||
|
||||
if(isset($this->torrent['map']) && sizeof($clean) > 0) {
|
||||
$episode_name_pattern = '{episode}' . preg_replace("/_+$/", '', $clean[0]);
|
||||
|
||||
if(strpos($this->torrent['map'], $episode_name_pattern) != -1) {
|
||||
$this->late('episodeName', array_shift($clean));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if(sizeof($clean) != 0) {
|
||||
if(sizeof($clean) == 1) {
|
||||
$clean = $clean[0];
|
||||
}
|
||||
$this->part('excess', array(), $this->excess_raw, $clean);
|
||||
}
|
||||
return $this->parts;
|
||||
}
|
||||
|
||||
private function escape_regex($subject) {
|
||||
return preg_replace("/[\-\[\]{}()*+?.,\\\^$|#\s]/", "\\\\$&", $subject);
|
||||
}
|
||||
|
||||
private function part($name, $match, $raw, $clean) {
|
||||
# The main core instructuions
|
||||
$this->parts[$name] = $clean;
|
||||
|
||||
if(sizeof($match) > 0) {
|
||||
# The instructions for extracting title
|
||||
$index = strpos($this->torrent['name'], $match[0]);
|
||||
if($index == 0) {
|
||||
$this->start = strlen($match[0]);
|
||||
}
|
||||
elseif(is_null($this->end) || $index < $this->end) {
|
||||
$this->end = $index;
|
||||
}
|
||||
}
|
||||
if($name != 'excess') {
|
||||
if($name == 'group') {
|
||||
$this->group_raw = $raw;
|
||||
}
|
||||
|
||||
if(!is_null($raw)) {
|
||||
$this->excess_raw = str_replace($raw, '', $this->excess_raw);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private function late($name, $clean) {
|
||||
if($name == 'group') {
|
||||
$this->part($name, array(), null, $clean);
|
||||
}
|
||||
elseif($name == 'episodeName') {
|
||||
$clean = preg_replace("/[\._]/", ' ', $clean);
|
||||
$clean = preg_replace("/_+$/", '', $clean);
|
||||
$this->part($name, array(), null, trim($clean));
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -14,7 +14,7 @@ use App\Torrentio\Action\Command\GetTvShowOptionsCommand;
|
||||
use App\Torrentio\Action\Handler\GetTvShowOptionsHandler;
|
||||
use App\User\Dto\UserPreferencesFactory;
|
||||
use App\User\Framework\Repository\UserRepository;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use OneToMany\RichBundle\Contract\CommandInterface;
|
||||
use OneToMany\RichBundle\Contract\HandlerInterface;
|
||||
use OneToMany\RichBundle\Contract\ResultInterface;
|
||||
|
||||
@@ -3,92 +3,80 @@
|
||||
namespace App\Download;
|
||||
|
||||
use Aimeos\Map;
|
||||
use App\Monitor\Framework\Entity\Monitor;
|
||||
use App\Torrentio\Result\TorrentioResult;
|
||||
use App\User\Dto\UserPreferences;
|
||||
|
||||
class DownloadOptionEvaluator
|
||||
{
|
||||
/**
|
||||
* @param Monitor $monitor
|
||||
* @param TorrentioResult[] $results
|
||||
* @param UserPreferences $filter
|
||||
* @return TorrentioResult|null
|
||||
* @throws \Throwable
|
||||
*/
|
||||
public function evaluateOptions(array $results, UserPreferences $userPreferences): ?TorrentioResult
|
||||
public function evaluateOptions(array $results, UserPreferences $filter): ?TorrentioResult
|
||||
{
|
||||
$sizeLow = 000;
|
||||
$sizeHigh = 4096;
|
||||
|
||||
$bestMatches = [];
|
||||
$matches = [];
|
||||
|
||||
foreach ($results as $result) {
|
||||
if (!in_array($userPreferences->language, $result->languages)) {
|
||||
continue;
|
||||
$matches = Map::from($results)->filter(function ($result) use ($filter) {
|
||||
if (false === $this->validateFilterItems($result, $filter)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($result->resolution === $userPreferences->resolution
|
||||
&& $result->codec === $userPreferences->codec
|
||||
) {
|
||||
$bestMatches[] = $result;
|
||||
if (false === $this->validateSize($result, $filter)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($userPreferences->resolution === '2160p'
|
||||
&& $userPreferences->codec === $result->codec
|
||||
&& $result->resolution === '1080p'
|
||||
) {
|
||||
$matches[] = $result;
|
||||
}
|
||||
return true;
|
||||
});
|
||||
|
||||
if ($userPreferences->codec === 'h264'
|
||||
&& $userPreferences->resolution === $result->resolution
|
||||
&& $result->codec === 'h265'
|
||||
) {
|
||||
$matches[] = $result;
|
||||
}
|
||||
|
||||
if (($userPreferences->codec === null )
|
||||
&& ($userPreferences->resolution === null )) {
|
||||
$matches[] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
$sizeMatches = [];
|
||||
|
||||
foreach ($bestMatches as $result) {
|
||||
if (str_contains($result->size, 'GB')) {
|
||||
$size = (int) trim(str_replace('GB', '', $result->size)) * 1024;
|
||||
} else {
|
||||
$size = (int) trim(str_replace('MB', '', $result->size));
|
||||
}
|
||||
|
||||
if ($size > $sizeLow && $size < $sizeHigh) {
|
||||
$sizeMatches[] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
if (!empty($sizeMatches)) {
|
||||
return Map::from($sizeMatches)->usort(fn($a, $b) => $a->seeders <=> $b->seeders)->last();
|
||||
}
|
||||
|
||||
foreach ($matches as $result) {
|
||||
$size = 0;
|
||||
if (str_contains($result->size, 'GB')) {
|
||||
$size = (int) trim(str_replace('GB', '', $result->size)) * 1024;
|
||||
} else {
|
||||
$size = (int) trim(str_replace('MB', '', $result->size));
|
||||
}
|
||||
|
||||
if ($size > $sizeLow && $size < $sizeHigh) {
|
||||
$sizeMatches[] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
if (!empty($sizeMatches)) {
|
||||
return Map::from($sizeMatches)->usort(fn($a, $b) => $a->seeders <=> $b->seeders)->last();
|
||||
if ($matches->count() > 0) {
|
||||
return Map::from($matches)->usort(fn($a, $b) => $a->seeders <=> $b->seeders)->last();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private function validateFilterItems(TorrentioResult $result, UserPreferences $filter): bool
|
||||
{
|
||||
if (array_intersect($filter->language, $result->languages) === []) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$valid = true;
|
||||
|
||||
if (null !== $filter->resolution && !in_array($result->resolution, $filter->resolution)) {
|
||||
$valid = false;
|
||||
}
|
||||
|
||||
if (null !== $filter->codec && in_array($result->codec, $filter->codec)) {
|
||||
$valid = false;
|
||||
}
|
||||
|
||||
if (null !== $filter->quality && in_array($result->quality, $filter->quality)) {
|
||||
$valid = false;
|
||||
}
|
||||
|
||||
if (null !== $filter->provider && in_array($result->provider, $filter->provider)) {
|
||||
$valid = false;
|
||||
}
|
||||
|
||||
return $valid;
|
||||
}
|
||||
|
||||
private function validateSize(TorrentioResult $result, UserPreferences $filter): bool
|
||||
{
|
||||
$sizeLow = 000;
|
||||
$sizeHigh = 4096;
|
||||
|
||||
if (str_contains($result->size, 'GB')) {
|
||||
$size = (int) trim(str_replace('GB', '', $result->size)) * 1024;
|
||||
} else {
|
||||
$size = (int) trim(str_replace('MB', '', $result->size));
|
||||
}
|
||||
|
||||
if ($size > $sizeLow && $size < $sizeHigh) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -6,7 +6,7 @@ use App\Download\Framework\Repository\DownloadRepository;
|
||||
use App\User\Framework\Entity\User;
|
||||
use Doctrine\ORM\Mapping as ORM;
|
||||
use Gedmo\Timestampable\Traits\TimestampableEntity;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use Symfony\Component\Serializer\Attribute\Ignore;
|
||||
use Symfony\UX\Turbo\Attribute\Broadcast;
|
||||
|
||||
|
||||
@@ -7,7 +7,7 @@ use App\Download\Framework\Entity\Download;
|
||||
use App\User\Framework\Entity\User;
|
||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||
use Doctrine\Persistence\ManagerRegistry;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use Symfony\Component\Security\Core\User\UserInterface;
|
||||
|
||||
/**
|
||||
|
||||
@@ -6,7 +6,7 @@ use App\Base\Service\MediaFiles;
|
||||
use App\Library\Action\Command\LibrarySearchCommand;
|
||||
use App\Library\Action\Result\LibrarySearchResult;
|
||||
use App\Library\Dto\MediaFileDto;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use OneToMany\RichBundle\Contract\CommandInterface;
|
||||
use OneToMany\RichBundle\Contract\HandlerInterface;
|
||||
use OneToMany\RichBundle\Contract\ResultInterface;
|
||||
|
||||
@@ -12,7 +12,7 @@ use App\Monitor\Framework\Repository\MonitorRepository;
|
||||
use App\Tmdb\Tmdb;
|
||||
use DateTimeImmutable;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use OneToMany\RichBundle\Contract\CommandInterface;
|
||||
use OneToMany\RichBundle\Contract\HandlerInterface;
|
||||
use OneToMany\RichBundle\Contract\ResultInterface;
|
||||
|
||||
@@ -13,7 +13,7 @@ use App\Tmdb\Tmdb;
|
||||
use Carbon\Carbon;
|
||||
use DateTimeImmutable;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
use OneToMany\RichBundle\Contract\CommandInterface;
|
||||
use OneToMany\RichBundle\Contract\HandlerInterface;
|
||||
use OneToMany\RichBundle\Contract\ResultInterface;
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
namespace App\Torrentio\Result;
|
||||
|
||||
use App\User\Database\CountryLanguages;
|
||||
use Nihilarr\PTN;
|
||||
use App\Base\Util\PTN;
|
||||
|
||||
class ResultFactory
|
||||
{
|
||||
|
||||
@@ -80,7 +80,7 @@ class UtilExtension
|
||||
|
||||
// Capture season
|
||||
$seasonMatch = [];
|
||||
preg_match('/[sS]\d\d/', $episodeId, $seasonMatch);
|
||||
preg_match('/[sS]\d\d(\d)?(\d)?/', $episodeId, $seasonMatch);
|
||||
if (empty($seasonMatch)) {
|
||||
$season = "";
|
||||
} else {
|
||||
@@ -89,7 +89,7 @@ class UtilExtension
|
||||
|
||||
// Capture episode
|
||||
$episodeMatch = [];
|
||||
preg_match('/[eE]\d\d/', $episodeId, $episodeMatch);
|
||||
preg_match('/[eE]\d\d(\d)?(\d)?/', $episodeId, $episodeMatch);
|
||||
if (empty($episodeMatch)) {
|
||||
$episode = "";
|
||||
} else {
|
||||
|
||||
@@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace App\User\Framework\Controller\Api;
|
||||
|
||||
use App\User\Dto\UserPreferences;
|
||||
use App\User\Dto\UserPreferencesFactory;
|
||||
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
||||
use Symfony\Component\HttpFoundation\Response;
|
||||
use Symfony\Component\Routing\Attribute\Route;
|
||||
|
||||
#[Route('/api/user/filters')]
|
||||
class UserFilterApiController extends AbstractController
|
||||
{
|
||||
#[Route('', 'api.user.filters', methods: ['GET'])]
|
||||
public function getFilters(): Response
|
||||
{
|
||||
return $this->json([
|
||||
UserPreferencesFactory::createFromUser($this->getUser())
|
||||
]);
|
||||
}
|
||||
}
|
||||
@@ -53,9 +53,9 @@ class UserMediaPreferencesForm extends AbstractType
|
||||
'data-preferred' => \json_encode($this->userPreferences->$fieldName),
|
||||
],
|
||||
'row_attr' => [
|
||||
'class' => 'filter-label'
|
||||
'class' => 'filter-label text-white'
|
||||
],
|
||||
'label_attr' => ['class' => 'text-white block font-semibold mb-2'],
|
||||
'label_attr' => ['class' => 'block font-semibold mb-2'],
|
||||
'choices' => $this->addDefaultChoice($choices),
|
||||
'required' => false,
|
||||
'multiple' => true,
|
||||
@@ -68,13 +68,13 @@ class UserMediaPreferencesForm extends AbstractType
|
||||
$resolver->setDefaults([
|
||||
'action' => $this->urlGenerator->generate('app_user_media_preferences_submit'),
|
||||
'attr' => [
|
||||
'class' => 'filter-items w-full p-4 bg-black/20 border-2 border-orange-500 text-md text-gray-500 dark:text-gray-50 rounded-lg',
|
||||
'class' => 'filter-items w-full p-4 bg-black/20 border-2 border-orange-500 text-md dark:text-gray-50 rounded-lg',
|
||||
]
|
||||
]);
|
||||
}
|
||||
|
||||
private function addDefaultChoice(array $choices): iterable
|
||||
{
|
||||
return ['n/a' => ''] + $choices;
|
||||
return ['n/a' => 'n/a'] + $choices;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -6,9 +6,11 @@
|
||||
data-result-filter-tv-episode-list-outlet=".episode-list"
|
||||
data-action="change->result-filter#filter action-button:downloadSeason@window->result-filter#downloadSeason"
|
||||
>
|
||||
|
||||
{% set preferences_form = form %}
|
||||
{{ form_start(preferences_form) }}
|
||||
<div class="flex flex-col md:flex-row gap-2 justify-between">
|
||||
<h3 class="font-bold text-lg mb-2 md:mb-4">Apply a filter to your results</h3>
|
||||
<div class="flex flex-col md:flex-row gap-2 justify-between">
|
||||
{{ form_row(preferences_form.resolution) }}
|
||||
{{ form_row(preferences_form.codec) }}
|
||||
{{ form_row(preferences_form.language) }}
|
||||
@@ -20,7 +22,7 @@
|
||||
<label for="season">
|
||||
Season
|
||||
</label>
|
||||
<select id="season" name="season" value="1" data-result-filter-target="season" class="px-1 mb-4 py-1 md:py-2 text-center bg-orange-500 text-white rounded-ms"
|
||||
<select id="season" name="season" value="1" data-result-filter-target="season" class="px-1 mb-4 py-1 md:py-2 text-center bg-orange-500 rounded-ms text-black"
|
||||
{{ stimulus_action('result_filter', 'setSeason', 'change') }}
|
||||
{{ stimulus_action('result_filter', 'uncheckSelectAllBtn', 'change') }}
|
||||
>
|
||||
|
||||
@@ -16,7 +16,7 @@
|
||||
active: 'true',
|
||||
}) }}
|
||||
>
|
||||
<div class="p-4 md:p-6 flex flex-col gap-6 bg-orange-500 bg-clip-padding backdrop-filter backdrop-blur-md bg-opacity-60 rounded-md">
|
||||
<div class="p-4 md:p-6 flex flex-col gap-6 bg-orange-500/60 bg-clip-padding backdrop-filter backdrop-blur-md rounded-md">
|
||||
<div class="flex flex-col md:flex-row gap-4">
|
||||
{% if episode['poster'] != null %}
|
||||
<img class="w-full md:w-64 rounded-lg" src="{{ episode['poster'] }}" />
|
||||
|
||||
Reference in New Issue
Block a user