Merge branch 'develop' of https://github.com/tidusjar/ombi into develop

This commit is contained in:
TidusJar 2018-08-01 08:14:28 +01:00
commit 58af83959e
104 changed files with 2319 additions and 17403 deletions

View file

@ -1,10 +1,10 @@
#tool "nuget:?package=GitVersion.CommandLine" #tool "nuget:?package=GitVersion.CommandLine"
#addin "Cake.Gulp" #addin "Cake.Gulp"
#addin "nuget:?package=Cake.Npm&version=0.13.0"
#addin "SharpZipLib" #addin "SharpZipLib"
#addin nuget:?package=Cake.Compression&version=0.1.4 #addin nuget:?package=Cake.Compression&version=0.1.4
#addin "Cake.Incubator" #addin "Cake.Incubator"
#addin "Cake.Yarn"
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
// ARGUMENTS // ARGUMENTS
@ -122,36 +122,19 @@ Task("SetVersionInfo")
Task("NPM") Task("NPM")
.Does(() => { .Does(() => {
var settings = new NpmInstallSettings { Yarn.FromPath(webProjDir).Install();
LogLevel = NpmLogLevel.Silent,
WorkingDirectory = webProjDir,
Production = true
};
NpmInstall(settings);
}); });
Task("Gulp Publish") Task("Gulp Publish")
.IsDependentOn("NPM") .IsDependentOn("NPM")
.Does(() => { .Does(() => {
Yarn.FromPath(webProjDir).RunScript("publish");
var runScriptSettings = new NpmRunScriptSettings {
ScriptName="publish",
WorkingDirectory = webProjDir,
};
NpmRunScript(runScriptSettings);
}); });
Task("TSLint") Task("TSLint")
.Does(() => .Does(() =>
{ {
var settings = new NpmRunScriptSettings { Yarn.FromPath(webProjDir).RunScript("lint");
WorkingDirectory = webProjDir,
ScriptName = "lint"
};
NpmRunScript(settings);
}); });
Task("PrePublish") Task("PrePublish")

View file

@ -29,7 +29,10 @@ namespace Ombi.Core.Tests.Rule.Search
{ {
ProviderId = "123" ProviderId = "123"
}); });
var search = new SearchMovieViewModel(); var search = new SearchMovieViewModel()
{
TheMovieDbId = "123",
};
var result = await Rule.Execute(search); var result = await Rule.Execute(search);
Assert.True(result.Success); Assert.True(result.Success);

View file

@ -15,9 +15,7 @@ namespace Ombi.Core.Engine.Interfaces
Task<RequestEngineResult> DenyChildRequest(int requestId); Task<RequestEngineResult> DenyChildRequest(int requestId);
Task<RequestsViewModel<TvRequests>> GetRequestsLite(int count, int position, OrderFilterModel type); Task<RequestsViewModel<TvRequests>> GetRequestsLite(int count, int position, OrderFilterModel type);
Task<IEnumerable<TvRequests>> SearchTvRequest(string search); Task<IEnumerable<TvRequests>> SearchTvRequest(string search);
Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> SearchTvRequestTree(string search);
Task<TvRequests> UpdateTvRequest(TvRequests request); Task<TvRequests> UpdateTvRequest(TvRequests request);
Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> GetRequestsTreeNode(int count, int position);
Task<IEnumerable<ChildRequests>> GetAllChldren(int tvId); Task<IEnumerable<ChildRequests>> GetAllChldren(int tvId);
Task<ChildRequests> UpdateChildRequest(ChildRequests request); Task<ChildRequests> UpdateChildRequest(ChildRequests request);
Task RemoveTvChild(int requestId); Task RemoveTvChild(int requestId);

View file

@ -7,16 +7,10 @@ namespace Ombi.Core.Engine.Interfaces
public interface ITvSearchEngine public interface ITvSearchEngine
{ {
Task<IEnumerable<SearchTvShowViewModel>> Search(string searchTerm); Task<IEnumerable<SearchTvShowViewModel>> Search(string searchTerm);
Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> SearchTreeNode(string searchTerm);
Task<TreeNode<SearchTvShowViewModel>> GetShowInformationTreeNode(int tvdbid);
Task<SearchTvShowViewModel> GetShowInformation(int tvdbid); Task<SearchTvShowViewModel> GetShowInformation(int tvdbid);
Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> PopularTree();
Task<IEnumerable<SearchTvShowViewModel>> Popular(); Task<IEnumerable<SearchTvShowViewModel>> Popular();
Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> AnticipatedTree();
Task<IEnumerable<SearchTvShowViewModel>> Anticipated(); Task<IEnumerable<SearchTvShowViewModel>> Anticipated();
Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> MostWatchesTree();
Task<IEnumerable<SearchTvShowViewModel>> MostWatches(); Task<IEnumerable<SearchTvShowViewModel>> MostWatches();
Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> TrendingTree();
Task<IEnumerable<SearchTvShowViewModel>> Trending(); Task<IEnumerable<SearchTvShowViewModel>> Trending();
} }
} }

View file

@ -1,23 +0,0 @@
using System.Collections.Generic;
namespace Ombi.Core.Engine
{
public class TreeNode<T>
{
public string Label { get; set; }
public T Data { get; set; }
public List<TreeNode<T>> Children { get; set; }
public bool Leaf { get; set; }
public bool Expanded { get; set; }
}
public class TreeNode<T,U>
{
public string Label { get; set; }
public T Data { get; set; }
public List<TreeNode<U>> Children { get; set; }
public bool Leaf { get; set; }
public bool Expanded { get; set; }
}
}

View file

@ -202,38 +202,6 @@ namespace Ombi.Core.Engine
Collection = allRequests Collection = allRequests
}; };
} }
public async Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> GetRequestsTreeNode(int count, int position)
{
var shouldHide = await HideFromOtherUsers();
List<TvRequests> allRequests;
if (shouldHide.Hide)
{
allRequests = await TvRepository.Get(shouldHide.UserId)
.Include(x => x.ChildRequests)
.ThenInclude(x => x.SeasonRequests)
.ThenInclude(x => x.Episodes)
.Where(x => x.ChildRequests.Any())
.OrderByDescending(x => x.ChildRequests.Max(y => y.RequestedDate))
.Skip(position).Take(count).ToListAsync();
FilterChildren(allRequests, shouldHide);
}
else
{
allRequests = await TvRepository.Get()
.Include(x => x.ChildRequests)
.ThenInclude(x => x.SeasonRequests)
.ThenInclude(x => x.Episodes)
.Where(x => x.ChildRequests.Any())
.OrderByDescending(x => x.ChildRequests.Max(y => y.RequestedDate))
.Skip(position).Take(count).ToListAsync();
}
allRequests.ForEach(async r => { await CheckForSubscription(shouldHide, r); });
return ParseIntoTreeNode(allRequests);
}
public async Task<IEnumerable<TvRequests>> GetRequests() public async Task<IEnumerable<TvRequests>> GetRequests()
{ {
var shouldHide = await HideFromOtherUsers(); var shouldHide = await HideFromOtherUsers();
@ -360,23 +328,6 @@ namespace Ombi.Core.Engine
return results; return results;
} }
public async Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> SearchTvRequestTree(string search)
{
var shouldHide = await HideFromOtherUsers();
IQueryable<TvRequests> allRequests;
if (shouldHide.Hide)
{
allRequests = TvRepository.Get(shouldHide.UserId);
}
else
{
allRequests = TvRepository.Get();
}
var results = await allRequests.Where(x => x.Title.Contains(search, CompareOptions.IgnoreCase)).ToListAsync();
results.ForEach(async r => { await CheckForSubscription(shouldHide, r); });
return ParseIntoTreeNode(results);
}
public async Task<TvRequests> UpdateTvRequest(TvRequests request) public async Task<TvRequests> UpdateTvRequest(TvRequests request)
{ {
await Audit.Record(AuditType.Updated, AuditArea.TvRequest, $"Updated Request {request.Title}", Username); await Audit.Record(AuditType.Updated, AuditArea.TvRequest, $"Updated Request {request.Title}", Username);
@ -596,28 +547,6 @@ namespace Ombi.Core.Engine
return await AfterRequest(model.ChildRequests.FirstOrDefault()); return await AfterRequest(model.ChildRequests.FirstOrDefault());
} }
private static List<TreeNode<TvRequests, List<ChildRequests>>> ParseIntoTreeNode(IEnumerable<TvRequests> result)
{
var node = new List<TreeNode<TvRequests, List<ChildRequests>>>();
foreach (var value in result)
{
node.Add(new TreeNode<TvRequests, List<ChildRequests>>
{
Data = value,
Children = new List<TreeNode<List<ChildRequests>>>
{
new TreeNode<List<ChildRequests>>
{
Data = SortEpisodes(value.ChildRequests),
Leaf = true
}
}
});
}
return node;
}
private static List<ChildRequests> SortEpisodes(List<ChildRequests> items) private static List<ChildRequests> SortEpisodes(List<ChildRequests> items)
{ {
foreach (var value in items) foreach (var value in items)

View file

@ -59,11 +59,6 @@ namespace Ombi.Core.Engine
return null; return null;
} }
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> SearchTreeNode(string searchTerm)
{
var result = await Search(searchTerm);
return result.Select(ParseIntoTreeNode).ToList();
}
public async Task<SearchTvShowViewModel> GetShowInformation(int tvdbid) public async Task<SearchTvShowViewModel> GetShowInformation(int tvdbid)
{ {
var show = await TvMazeApi.ShowLookupByTheTvDbId(tvdbid); var show = await TvMazeApi.ShowLookupByTheTvDbId(tvdbid);
@ -116,19 +111,6 @@ namespace Ombi.Core.Engine
return await ProcessResult(mapped); return await ProcessResult(mapped);
} }
public async Task<TreeNode<SearchTvShowViewModel>> GetShowInformationTreeNode(int tvdbid)
{
var result = await GetShowInformation(tvdbid);
return ParseIntoTreeNode(result);
}
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> PopularTree()
{
var result = await Cache.GetOrAdd(CacheKeys.PopularTv, async () => await TraktApi.GetPopularShows(), DateTime.Now.AddHours(12));
var processed = await ProcessResults(result);
return processed.Select(ParseIntoTreeNode).ToList();
}
public async Task<IEnumerable<SearchTvShowViewModel>> Popular() public async Task<IEnumerable<SearchTvShowViewModel>> Popular()
{ {
var result = await Cache.GetOrAdd(CacheKeys.PopularTv, async () => await TraktApi.GetPopularShows(), DateTime.Now.AddHours(12)); var result = await Cache.GetOrAdd(CacheKeys.PopularTv, async () => await TraktApi.GetPopularShows(), DateTime.Now.AddHours(12));
@ -136,12 +118,6 @@ namespace Ombi.Core.Engine
return processed; return processed;
} }
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> AnticipatedTree()
{
var result = await Cache.GetOrAdd(CacheKeys.AnticipatedTv, async () => await TraktApi.GetAnticipatedShows(), DateTime.Now.AddHours(12));
var processed = await ProcessResults(result);
return processed.Select(ParseIntoTreeNode).ToList();
}
public async Task<IEnumerable<SearchTvShowViewModel>> Anticipated() public async Task<IEnumerable<SearchTvShowViewModel>> Anticipated()
{ {
@ -150,12 +126,6 @@ namespace Ombi.Core.Engine
return processed; return processed;
} }
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> MostWatchesTree()
{
var result = await Cache.GetOrAdd(CacheKeys.MostWatchesTv, async () => await TraktApi.GetMostWatchesShows(), DateTime.Now.AddHours(12));
var processed = await ProcessResults(result);
return processed.Select(ParseIntoTreeNode).ToList();
}
public async Task<IEnumerable<SearchTvShowViewModel>> MostWatches() public async Task<IEnumerable<SearchTvShowViewModel>> MostWatches()
{ {
var result = await Cache.GetOrAdd(CacheKeys.MostWatchesTv, async () => await TraktApi.GetMostWatchesShows(), DateTime.Now.AddHours(12)); var result = await Cache.GetOrAdd(CacheKeys.MostWatchesTv, async () => await TraktApi.GetMostWatchesShows(), DateTime.Now.AddHours(12));
@ -163,13 +133,6 @@ namespace Ombi.Core.Engine
return processed; return processed;
} }
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> TrendingTree()
{
var result = await Cache.GetOrAdd(CacheKeys.TrendingTv, async () => await TraktApi.GetTrendingShows(), DateTime.Now.AddHours(12));
var processed = await ProcessResults(result);
return processed.Select(ParseIntoTreeNode).ToList();
}
public async Task<IEnumerable<SearchTvShowViewModel>> Trending() public async Task<IEnumerable<SearchTvShowViewModel>> Trending()
{ {
var result = await Cache.GetOrAdd(CacheKeys.TrendingTv, async () => await TraktApi.GetTrendingShows(), DateTime.Now.AddHours(12)); var result = await Cache.GetOrAdd(CacheKeys.TrendingTv, async () => await TraktApi.GetTrendingShows(), DateTime.Now.AddHours(12));
@ -177,22 +140,6 @@ namespace Ombi.Core.Engine
return processed; return processed;
} }
private static TreeNode<SearchTvShowViewModel> ParseIntoTreeNode(SearchTvShowViewModel result)
{
return new TreeNode<SearchTvShowViewModel>
{
Data = result,
Children = new List<TreeNode<SearchTvShowViewModel>>
{
new TreeNode<SearchTvShowViewModel>
{
Data = result, Leaf = true
}
},
Leaf = false
};
}
private async Task<IEnumerable<SearchTvShowViewModel>> ProcessResults<T>(IEnumerable<T> items) private async Task<IEnumerable<SearchTvShowViewModel>> ProcessResults<T>(IEnumerable<T> items)
{ {
var retVal = new List<SearchTvShowViewModel>(); var retVal = new List<SearchTvShowViewModel>();

23
src/Ombi/.gitignore vendored
View file

@ -1,23 +1,10 @@
/wwwroot/css/** node_modules
/wwwroot/fonts/** bin
/wwwroot/lib/** obj
/wwwroot/maps/** wwwroot/dist
/wwwroot/dist/** *.log
/wwwroot/*.js.map
/wwwroot/*.js
# dependencies
/node_modules
/bower_components
# misc
/.sass-cache /.sass-cache
/connect.lock /connect.lock
/coverage/* /coverage/*
/libpeerconnection.log
npm-debug.log
testem.log
#/typings
/systemjs.config.js*
/Logs/** /Logs/**
**.db **.db

View file

@ -4,7 +4,7 @@
"version": "2.0.0", "version": "2.0.0",
"tasks": [ "tasks": [
{ {
"taskName": "restore", "label": "restore",
"command": "npm", "command": "npm",
"type": "shell", "type": "shell",
"args": [ "args": [
@ -14,7 +14,16 @@
"problemMatcher": [] "problemMatcher": []
}, },
{ {
"taskName": "build", "label": "clean",
"command": "dotnet",
"type": "shell",
"args": [
"clean"
],
"problemMatcher": "$msCompile"
},
{
"label": "build",
"command": "dotnet", "command": "dotnet",
"type": "shell", "type": "shell",
"args": [ "args": [
@ -27,7 +36,7 @@
"problemMatcher": "$msCompile" "problemMatcher": "$msCompile"
}, },
{ {
"taskName": "lint", "label": "lint",
"type": "shell", "type": "shell",
"command": "npm", "command": "npm",
"args": [ "args": [

View file

@ -1,7 +1,7 @@
import { animate, style, transition, trigger } from "@angular/animations"; import { animate, style, transition, trigger } from "@angular/animations";
import { AnimationEntryMetadata } from "@angular/core"; import { AnimationTriggerMetadata } from "@angular/animations";
export const fadeInOutAnimation: AnimationEntryMetadata = trigger("fadeInOut", [ export const fadeInOutAnimation: AnimationTriggerMetadata = trigger("fadeInOut", [
transition(":enter", [ // :enter is alias to 'void => *' transition(":enter", [ // :enter is alias to 'void => *'
style({ opacity: 0 }), style({ opacity: 0 }),
animate(1000, style({ opacity: 1 })), animate(1000, style({ opacity: 1 })),

View file

@ -40,7 +40,7 @@ export class AppComponent implements OnInit {
__webpack_public_path__ = base + "/dist/"; __webpack_public_path__ = base + "/dist/";
} }
this.translate.addLangs(["en", "de", "fr","da","es","it","nl","sv","no", "pl", "pt"]); this.translate.addLangs(["en", "de", "fr", "da", "es", "it", "nl", "sv", "no", "pl", "pt"]);
// this language will be used as a fallback when a translation isn't found in the current language // this language will be used as a fallback when a translation isn't found in the current language
this.translate.setDefaultLang("en"); this.translate.setDefaultLang("en");
@ -88,8 +88,8 @@ export class AppComponent implements OnInit {
public openMobileApp(event: any) { public openMobileApp(event: any) {
event.preventDefault(); event.preventDefault();
if(!this.customizationSettings.applicationUrl) { if (!this.customizationSettings.applicationUrl) {
this.notificationService.warning("Mobile","Please ask your admin to setup the Application URL!"); this.notificationService.warning("Mobile", "Please ask your admin to setup the Application URL!");
return; return;
} }

View file

@ -1,12 +1,12 @@
import {CommonModule, PlatformLocation} from "@angular/common"; import { CommonModule, PlatformLocation } from "@angular/common";
import {HttpClient, HttpClientModule} from "@angular/common/http"; import { HttpClient, HttpClientModule } from "@angular/common/http";
import {NgModule} from "@angular/core"; import { NgModule } from "@angular/core";
import {FormsModule, ReactiveFormsModule} from "@angular/forms"; import { FormsModule, ReactiveFormsModule } from "@angular/forms";
import {HttpModule} from "@angular/http"; import { HttpModule } from "@angular/http";
import {MatButtonModule, MatCardModule, MatInputModule, MatTabsModule} from "@angular/material"; import { MatButtonModule, MatCardModule, MatInputModule, MatTabsModule } from "@angular/material";
import {BrowserModule} from "@angular/platform-browser"; import { BrowserModule } from "@angular/platform-browser";
import {BrowserAnimationsModule} from "@angular/platform-browser/animations"; import { BrowserAnimationsModule } from "@angular/platform-browser/animations";
import {RouterModule, Routes} from "@angular/router"; import { RouterModule, Routes } from "@angular/router";
import { JwtModule } from "@auth0/angular-jwt"; import { JwtModule } from "@auth0/angular-jwt";
@ -15,7 +15,7 @@ import { TranslateLoader, TranslateModule } from "@ngx-translate/core";
import { TranslateHttpLoader } from "@ngx-translate/http-loader"; import { TranslateHttpLoader } from "@ngx-translate/http-loader";
import { CookieService } from "ng2-cookies"; import { CookieService } from "ng2-cookies";
import { GrowlModule } from "primeng/components/growl/growl"; import { GrowlModule } from "primeng/components/growl/growl";
import { ButtonModule, CaptchaModule, ConfirmationService, ConfirmDialogModule, DataTableModule,DialogModule, SharedModule, SidebarModule, TooltipModule } from "primeng/primeng"; import { ButtonModule, CaptchaModule, ConfirmationService, ConfirmDialogModule, DataTableModule, DialogModule, SharedModule, SidebarModule, TooltipModule } from "primeng/primeng";
// Components // Components
import { AppComponent } from "./app.component"; import { AppComponent } from "./app.component";
@ -67,6 +67,14 @@ export function HttpLoaderFactory(http: HttpClient, platformLocation: PlatformLo
return new TranslateHttpLoader(http, "/translations/", `.json?v=${version}`); return new TranslateHttpLoader(http, "/translations/", `.json?v=${version}`);
} }
export function JwtTokenGetter() {
const token = localStorage.getItem("id_token");
if (!token) {
return "";
}
return token;
}
@NgModule({ @NgModule({
imports: [ imports: [
RouterModule.forRoot(routes), RouterModule.forRoot(routes),
@ -92,13 +100,7 @@ export function HttpLoaderFactory(http: HttpClient, platformLocation: PlatformLo
CommonModule, CommonModule,
JwtModule.forRoot({ JwtModule.forRoot({
config: { config: {
tokenGetter: () => { tokenGetter: JwtTokenGetter,
const token = localStorage.getItem("id_token");
if (!token) {
return "";
}
return token;
},
}, },
}), }),
TranslateModule.forRoot({ TranslateModule.forRoot({

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { JwtHelperService } from "@auth0/angular-jwt"; import { JwtHelperService } from "@auth0/angular-jwt";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../services"; import { ServiceHelpers } from "../services";
import { ILocalUser, IUserLogin } from "./IUserLogin"; import { ILocalUser, IUserLogin } from "./IUserLogin";
@ -53,9 +53,9 @@ export class AuthService extends ServiceHelpers {
} else { } else {
u.roles.push(roles); u.roles.push(roles);
} }
return <ILocalUser>u; return <ILocalUser> u;
} }
return <ILocalUser>{}; return <ILocalUser> { };
} }
public hasRole(role: string): boolean { public hasRole(role: string): boolean {

View file

@ -11,7 +11,7 @@ export class CookieComponent implements OnInit {
public ngOnInit() { public ngOnInit() {
const cookie = this.cookieService.getAll(); const cookie = this.cookieService.getAll();
if(cookie.Auth) { if (cookie.Auth) {
const jwtVal = cookie.Auth; const jwtVal = cookie.Auth;
localStorage.setItem("id_token", jwtVal); localStorage.setItem("id_token", jwtVal);
this.router.navigate(["search"]); this.router.navigate(["search"]);

View file

@ -71,6 +71,10 @@ export interface ITvRequests {
status: string; status: string;
childRequests: IChildRequests[]; childRequests: IChildRequests[];
qualityOverride: number; qualityOverride: number;
background: any;
totalSeasons: number;
tvDbId: number;
open: boolean; // THIS IS FOR THE UI
// For UI display // For UI display
qualityOverrideTitle: string; qualityOverrideTitle: string;

View file

@ -28,8 +28,16 @@ export interface ISearchTvResult {
available: boolean; available: boolean;
plexUrl: string; plexUrl: string;
embyUrl: string; embyUrl: string;
quality: string;
firstSeason: boolean; firstSeason: boolean;
latestSeason: boolean; latestSeason: boolean;
theTvDbId: string;
subscribed: boolean;
showSubscribe: boolean;
fullyAvailable: boolean;
partlyAvailable: boolean;
background: any;
open: boolean; // THIS IS FOR THE UI
} }
export interface ITvRequestViewModel { export interface ITvRequestViewModel {

View file

@ -63,8 +63,8 @@ export class IssueDetailsComponent implements OnInit {
issueCategoryId: x.issueCategoryId, issueCategoryId: x.issueCategoryId,
subject: x.subject, subject: x.subject,
description: x.description, description: x.description,
status:x.status, status: x.status,
resolvedDate:x.resolvedDate, resolvedDate: x.resolvedDate,
title: x.title, title: x.title,
requestType: x.requestType, requestType: x.requestType,
requestId: x.requestId, requestId: x.requestId,
@ -117,7 +117,7 @@ export class IssueDetailsComponent implements OnInit {
} else { } else {
this.imageService.getTvBackground(Number(issue.providerId)).subscribe(x => { this.imageService.getTvBackground(Number(issue.providerId)).subscribe(x => {
if(x) { if (x) {
this.backgroundPath = this.sanitizer.bypassSecurityTrustStyle this.backgroundPath = this.sanitizer.bypassSecurityTrustStyle
("url(" + x + ")"); ("url(" + x + ")");
} }

View file

@ -61,5 +61,4 @@ export class IssuesComponent implements OnInit {
this.resolvedIssues = x; this.resolvedIssues = x;
}); });
} }
} }

View file

@ -32,7 +32,7 @@ export class LoginComponent implements OnDestroy, OnInit {
public loginWithOmbi: boolean; public loginWithOmbi: boolean;
public get appName(): string { public get appName(): string {
if(this.customizationSettings.applicationName) { if (this.customizationSettings.applicationName) {
return this.customizationSettings.applicationName; return this.customizationSettings.applicationName;
} else { } else {
return "Ombi"; return "Ombi";
@ -72,7 +72,7 @@ export class LoginComponent implements OnDestroy, OnInit {
} }
}); });
if(authService.loggedIn()) { if (authService.loggedIn()) {
this.router.navigate(["search"]); this.router.navigate(["search"]);
} }
} }
@ -103,9 +103,9 @@ export class LoginComponent implements OnDestroy, OnInit {
return; return;
} }
const value = form.value; const value = form.value;
const user = { password: value.password, username: value.username, rememberMe: value.rememberMe, usePlexOAuth: false, plexTvPin: { id: 0, code: ""} }; const user = { password: value.password, username: value.username, rememberMe: value.rememberMe, usePlexOAuth: false, plexTvPin: { id: 0, code: "" } };
this.authService.requiresPassword(user).subscribe(x => { this.authService.requiresPassword(user).subscribe(x => {
if(x && this.authenticationSettings.allowNoPassword) { if (x && this.authenticationSettings.allowNoPassword) {
// Looks like this user requires a password // Looks like this user requires a password
this.authenticationSettings.allowNoPassword = false; this.authenticationSettings.allowNoPassword = false;
return; return;
@ -125,9 +125,9 @@ export class LoginComponent implements OnDestroy, OnInit {
} }
public oauth() { public oauth() {
this.plexTv.GetPin(this.clientId, this.appName).subscribe(pin => { this.plexTv.GetPin(this.clientId, this.appName).subscribe((pin: any) => {
this.authService.login({usePlexOAuth: true, password:"",rememberMe:true,username:"", plexTvPin: pin}).subscribe(x => { this.authService.login({ usePlexOAuth: true, password: "", rememberMe: true, username: "", plexTvPin: pin }).subscribe(x => {
if (window.frameElement) { if (window.frameElement) {
// in frame // in frame
window.open(x.url, "_blank"); window.open(x.url, "_blank");

View file

@ -16,7 +16,6 @@ export class LoginOAuthComponent implements OnInit {
this.route.params this.route.params
.subscribe((params: any) => { .subscribe((params: any) => {
this.pin = params.pin; this.pin = params.pin;
}); });
} }
@ -26,7 +25,7 @@ export class LoginOAuthComponent implements OnInit {
public auth() { public auth() {
this.authService.oAuth(this.pin).subscribe(x => { this.authService.oAuth(this.pin).subscribe(x => {
if(x.access_token) { if (x.access_token) {
localStorage.setItem("id_token", x.access_token); localStorage.setItem("id_token", x.access_token);
if (this.authService.loggedIn()) { if (this.authService.loggedIn()) {
@ -34,13 +33,12 @@ export class LoginOAuthComponent implements OnInit {
return; return;
} }
} }
if(x.errorMessage) { if (x.errorMessage) {
this.error = x.errorMessage; this.error = x.errorMessage;
} }
}, err => { }, err => {
this.notify.error(err.statusText); this.notify.error(err.statusText);
this.router.navigate(["login"]); this.router.navigate(["login"]);
}); });
} }

View file

@ -4,7 +4,7 @@ import { FormBuilder, FormGroup, Validators } from "@angular/forms";
import { DomSanitizer } from "@angular/platform-browser"; import { DomSanitizer } from "@angular/platform-browser";
import { ICustomizationSettings } from "../interfaces"; import { ICustomizationSettings } from "../interfaces";
import { IdentityService, ImageService,NotificationService, SettingsService } from "../services"; import { IdentityService, ImageService, NotificationService, SettingsService } from "../services";
@Component({ @Component({
templateUrl: "./resetpassword.component.html", templateUrl: "./resetpassword.component.html",

View file

@ -1,5 +1,5 @@
import { Component, OnInit } from "@angular/core"; import { Component, OnInit } from "@angular/core";
import { NguCarousel } from "@ngu/carousel"; import { NguCarouselConfig } from "@ngu/carousel";
import { ImageService, RecentlyAddedService } from "../services"; import { ImageService, RecentlyAddedService } from "../services";
import { IRecentlyAddedMovies, IRecentlyAddedTvShows } from "./../interfaces"; import { IRecentlyAddedMovies, IRecentlyAddedTvShows } from "./../interfaces";
@ -43,7 +43,7 @@ export class RecentlyAddedComponent implements OnInit {
public groupTv: boolean = false; public groupTv: boolean = false;
// https://github.com/sheikalthaf/ngu-carousel // https://github.com/sheikalthaf/ngu-carousel
public carouselTile: NguCarousel; public carouselTile: NguCarouselConfig;
constructor(private recentlyAddedService: RecentlyAddedService, constructor(private recentlyAddedService: RecentlyAddedService,
private imageService: ImageService) {} private imageService: ImageService) {}
@ -67,10 +67,10 @@ export class RecentlyAddedComponent implements OnInit {
} }
public close() { public close() {
if(this.range.length < 2) { if (this.range.length < 2) {
return; return;
} }
if(!this.range[1]) { if (!this.range[1]) {
// If we do not have a second date then just set it to now // If we do not have a second date then just set it to now
this.range[1] = new Date(); this.range[1] = new Date();
} }
@ -82,13 +82,13 @@ export class RecentlyAddedComponent implements OnInit {
} }
private getShows() { private getShows() {
if(this.groupTv) { if (this.groupTv) {
this.recentlyAddedService.getRecentlyAddedTvGrouped().subscribe(x => { this.recentlyAddedService.getRecentlyAddedTvGrouped().subscribe(x => {
this.tv = x; this.tv = x;
this.tv.forEach((t) => { this.tv.forEach((t) => {
this.imageService.getTvPoster(t.tvDbId).subscribe(p => { this.imageService.getTvPoster(t.tvDbId).subscribe(p => {
if(p) { if (p) {
t.posterPath = p; t.posterPath = p;
} }
}); });
@ -100,7 +100,7 @@ export class RecentlyAddedComponent implements OnInit {
this.tv.forEach((t) => { this.tv.forEach((t) => {
this.imageService.getTvPoster(t.tvDbId).subscribe(p => { this.imageService.getTvPoster(t.tvDbId).subscribe(p => {
if(p) { if (p) {
t.posterPath = p; t.posterPath = p;
} }
}); });
@ -114,11 +114,11 @@ export class RecentlyAddedComponent implements OnInit {
this.movies = x; this.movies = x;
this.movies.forEach((movie) => { this.movies.forEach((movie) => {
if(movie.theMovieDbId) { if (movie.theMovieDbId) {
this.imageService.getMoviePoster(movie.theMovieDbId).subscribe(p => { this.imageService.getMoviePoster(movie.theMovieDbId).subscribe(p => {
movie.posterPath = p; movie.posterPath = p;
}); });
} else if(movie.imdbId) { } else if (movie.imdbId) {
this.imageService.getMoviePoster(movie.imdbId).subscribe(p => { this.imageService.getMoviePoster(movie.imdbId).subscribe(p => {
movie.posterPath = p; movie.posterPath = p;
}); });

View file

@ -45,8 +45,6 @@
<div> <div>
<div *ngFor="let request of movieRequests"> <div *ngFor="let request of movieRequests">
<div class="row"> <div class="row">
<div class="myBg backdrop" [style.background-image]="request.backgroundPath"></div> <div class="myBg backdrop" [style.background-image]="request.backgroundPath"></div>

View file

@ -1,15 +1,12 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Component, Input, OnInit } from "@angular/core"; import { Component, Input, OnInit } from "@angular/core";
import { DomSanitizer } from "@angular/platform-browser"; import { DomSanitizer } from "@angular/platform-browser";
import "rxjs/add/operator/debounceTime"; import { Subject } from "rxjs";
import "rxjs/add/operator/distinctUntilChanged"; import { debounceTime, distinctUntilChanged } from "rxjs/operators";
import "rxjs/add/operator/map";
import { Subject } from "rxjs/Subject";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { NotificationService, RadarrService, RequestService } from "../services";
import { FilterType, IFilter, IIssueCategory, IMovieRequests, IPagenator, IRadarrProfile, IRadarrRootFolder, OrderType } from "../interfaces"; import { FilterType, IFilter, IIssueCategory, IMovieRequests, IPagenator, IRadarrProfile, IRadarrRootFolder, OrderType } from "../interfaces";
import { NotificationService, RadarrService, RequestService } from "../services";
@Component({ @Component({
selector: "movie-requests", selector: "movie-requests",
@ -45,16 +42,17 @@ export class MovieRequestsComponent implements OnInit {
private currentlyLoaded: number; private currentlyLoaded: number;
private amountToLoad: number; private amountToLoad: number;
constructor(private requestService: RequestService, constructor(
private requestService: RequestService,
private auth: AuthService, private auth: AuthService,
private notificationService: NotificationService, private notificationService: NotificationService,
private radarrService: RadarrService, private radarrService: RadarrService,
private sanitizer: DomSanitizer, private sanitizer: DomSanitizer,
private readonly platformLocation: PlatformLocation) { private readonly platformLocation: PlatformLocation) {
this.searchChanged this.searchChanged.pipe(
.debounceTime(600) // Wait Xms after the last event before emitting last event debounceTime(600), // Wait Xms after the last event before emitting last event
.distinctUntilChanged() // only emit if value is different from previous value distinctUntilChanged(), // only emit if value is different from previous value
.subscribe(x => { ).subscribe(x => {
this.searchText = x as string; this.searchText = x as string;
if (this.searchText === "") { if (this.searchText === "") {
this.resetSearch(); this.resetSearch();
@ -102,7 +100,7 @@ export class MovieRequestsComponent implements OnInit {
public changeAvailability(request: IMovieRequests, available: boolean) { public changeAvailability(request: IMovieRequests, available: boolean) {
request.available = available; request.available = available;
if(available) { if (available) {
this.requestService.markMovieAvailable({ id: request.id }).subscribe(x => { this.requestService.markMovieAvailable({ id: request.id }).subscribe(x => {
if (x.result) { if (x.result) {
this.notificationService.success( this.notificationService.success(
@ -241,7 +239,7 @@ export class MovieRequestsComponent implements OnInit {
this.requestService.getMovieRequests(amountToLoad, currentlyLoaded, this.orderType, this.filter) this.requestService.getMovieRequests(amountToLoad, currentlyLoaded, this.orderType, this.filter)
.subscribe(x => { .subscribe(x => {
this.setOverrides(x.collection); this.setOverrides(x.collection);
if(!this.movieRequests) { if (!this.movieRequests) {
this.movieRequests = []; this.movieRequests = [];
} }
this.movieRequests = x.collection; this.movieRequests = x.collection;

View file

@ -4,7 +4,7 @@ import { IChildRequests } from "../interfaces";
import { NotificationService, RequestService } from "../services"; import { NotificationService, RequestService } from "../services";
@Component({ @Component({
selector:"tvrequests-children", selector: "tvrequests-children",
templateUrl: "./tvrequest-children.component.html", templateUrl: "./tvrequest-children.component.html",
}) })
export class TvRequestChildrenComponent { export class TvRequestChildrenComponent {
@ -26,12 +26,12 @@ export class TvRequestChildrenComponent {
public changeAvailability(request: IChildRequests, available: boolean) { public changeAvailability(request: IChildRequests, available: boolean) {
request.available = available; request.available = available;
request.seasonRequests.forEach((season)=> { request.seasonRequests.forEach((season) => {
season.episodes.forEach((ep)=> { season.episodes.forEach((ep) => {
ep.available = available; ep.available = available;
}); });
}); });
if(available) { if (available) {
this.requestService.markTvAvailable({ id: request.id }).subscribe(x => { this.requestService.markTvAvailable({ id: request.id }).subscribe(x => {
if (x.result) { if (x.result) {
this.notificationService.success( this.notificationService.success(

View file

@ -4,59 +4,41 @@
</div> </div>
</div> </div>
<br /> <br />
<!--TODO: I believe this +1 is causing off by one error skipping loading of tv shows
When removed and scrolling very slowly everything works as expected, however
if you scroll really quickly then you start getting duplicates of movies
since it's async and some subsequent results return first and then incrementer
is increased so you see movies which had already been gotten show up...
Removing infinte-scroll and setting max to 1000 till we work out some sort of fix <div>
<div *ngFor="let node of tvRequests.collection">
-->
<!--<div infinite-scroll
[infiniteScrollDistance]="1"
[infiniteScrollThrottle]="100"
(scrolled)="loadMore()">-->
<div>
<p-treeTable [value]="tvRequests">
<p-column>
<ng-template let-col let-node="rowData" pTemplate="header">
Results
</ng-template>
<ng-template let-col let-node="rowData" pTemplate="body">
<!--This is the section that holds the parent level results set--> <!--This is the section that holds the parent level results set-->
<div *ngIf="!node.leaf"> <div>
<div class="row"> <div class="row">
<div class="myBg backdrop" [style.background-image]="node?.data?.background"></div> <div class="myBg backdrop" [style.background-image]="node?.background"></div>
<div class="tint" style="background-image: linear-gradient(to bottom, rgba(0,0,0,0.6) 0%,rgba(0,0,0,0.6) 100%);"></div> <div class="tint" style="background-image: linear-gradient(to bottom, rgba(0,0,0,0.6) 0%,rgba(0,0,0,0.6) 100%);"></div>
<div class="col-sm-2 small-padding" > <div class="col-sm-2 small-padding">
<img class="img-responsive poster" src="{{node.data.posterPath || null}}" alt="poster"> <img class="img-responsive poster" src="{{node.posterPath || null}}" alt="poster">
</div> </div>
<div class="col-sm-5 small-padding"> <div class="col-sm-5 small-padding">
<div> <div>
<a href="http://www.imdb.com/title/{{node.data.imdbId}}/" target="_blank"> <a href="http://www.imdb.com/title/{{node.imdbId}}/" target="_blank">
<h4 class="request-title">{{node.data.title}} ({{node.data.releaseDate | date: 'yyyy'}})</h4> <h4 class="request-title">{{node.title}} ({{node.releaseDate | date: 'yyyy'}})</h4>
</a> </a>
</div> </div>
<br /> <br />
<div> <div>
<span>Status: </span> <span>Status: </span>
<span class="label label-success">{{node.data.status}}</span> <span class="label label-success">{{node.status}}</span>
</div> </div>
<div>Release Date: {{node.data.releaseDate | date}}</div> <div>Release Date: {{node.releaseDate | date}}</div>
<div *ngIf="isAdmin"> <div *ngIf="isAdmin">
<div *ngIf="node.data.qualityOverrideTitle" class="quality-override">{{ 'Requests.QualityOverride' | translate }} <div *ngIf="node.qualityOverrideTitle" class="quality-override">{{ 'Requests.QualityOverride' | translate }}
<span>{{node.data.qualityOverrideTitle}} </span> <span>{{node.qualityOverrideTitle}} </span>
</div> </div>
<div *ngIf="node.data.rootPathOverrideTitle" class="root-override">{{ 'Requests.RootFolderOverride' | translate }} <div *ngIf="node.rootPathOverrideTitle" class="root-override">{{ 'Requests.RootFolderOverride' | translate }}
<span>{{node.data.rootPathOverrideTitle}} </span> <span>{{node.rootPathOverrideTitle}} </span>
</div> </div>
</div> </div>
@ -64,7 +46,8 @@
</div> </div>
<div class="col-sm-3 col-sm-push-3 small-padding"> <div class="col-sm-3 col-sm-push-3 small-padding">
<button style="text-align: right" class="btn btn-sm btn-success-outline" (click)="openClosestTab($event)"><i class="fa fa-plus"></i> View</button> <button style="text-align: right" class="btn btn-sm btn-success-outline" (click)="openClosestTab(node,$event)">
<i class="fa fa-plus"></i> View</button>
<div *ngIf="isAdmin"> <div *ngIf="isAdmin">
<!--Sonarr Root Folder--> <!--Sonarr Root Folder-->
<div *ngIf="sonarrRootFolders" class="btn-group btn-split" id="rootFolderBtn"> <div *ngIf="sonarrRootFolders" class="btn-group btn-split" id="rootFolderBtn">
@ -77,7 +60,7 @@
</button> </button>
<ul class="dropdown-menu"> <ul class="dropdown-menu">
<li *ngFor="let folder of sonarrRootFolders"> <li *ngFor="let folder of sonarrRootFolders">
<a href="#" (click)="selectRootFolder(node.data, folder, $event)">{{folder.path}}</a> <a href="#" (click)="selectRootFolder(node, folder, $event)">{{folder.path}}</a>
</li> </li>
</ul> </ul>
</div> </div>
@ -93,35 +76,39 @@
</button> </button>
<ul class="dropdown-menu"> <ul class="dropdown-menu">
<li *ngFor="let profile of sonarrProfiles"> <li *ngFor="let profile of sonarrProfiles">
<a href="#" (click)="selectQualityProfile(node.data, profile, $event)">{{profile.name}}</a> <a href="#" (click)="selectQualityProfile(node, profile, $event)">{{profile.name}}</a>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>
<div class="dropdown" *ngIf="issueCategories && issuesEnabled" id="issueBtn"> <div class="dropdown" *ngIf="issueCategories && issuesEnabled" id="issueBtn">
<button class="btn btn-sm btn-primary-outline dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="true"> <button class="btn btn-sm btn-primary-outline dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true"
aria-expanded="true">
<i class="fa fa-plus"></i> {{ 'Requests.ReportIssue' | translate }} <i class="fa fa-plus"></i> {{ 'Requests.ReportIssue' | translate }}
<span class="caret"></span> <span class="caret"></span>
</button> </button>
<ul class="dropdown-menu" aria-labelledby="dropdownMenu1"> <ul class="dropdown-menu" aria-labelledby="dropdownMenu1">
<li *ngFor="let cat of issueCategories"><a [routerLink]="" (click)="reportIssue(cat, node.data)">{{cat.value}}</a></li> <li *ngFor="let cat of issueCategories">
<a [routerLink]="" (click)="reportIssue(cat, node)">{{cat.value}}</a>
</li>
</ul> </ul>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<!--This is the section that holds the child seasons if they want to specify specific episodes--> <!--This is the section that holds the child seasons if they want to specify specific episodes-->
<div *ngIf="node.leaf"> <div *ngIf="node.open">
<tvrequests-children [childRequests]="node.data" [isAdmin] ="isAdmin" <tvrequests-children [childRequests]="node.childRequests" [isAdmin]="isAdmin" (requestDeleted)="childRequestDeleted($event)"></tvrequests-children>
(requestDeleted)="childRequestDeleted($event)"></tvrequests-children> </div>
<br/>
<br/>
</div> </div>
</ng-template>
</p-column>
</p-treeTable>
<p-paginator [rows]="10" [totalRecords]="totalTv" (onPageChange)="paginate($event)"></p-paginator> <p-paginator [rows]="10" [totalRecords]="totalTv" (onPageChange)="paginate($event)"></p-paginator>
</div> </div>
<issue-report [movie]="false" [visible]="issuesBarVisible" [title]="issueRequest?.title" <issue-report [movie]="false" [visible]="issuesBarVisible" [title]="issueRequest?.title" [issueCategory]="issueCategorySelected"
[issueCategory]="issueCategorySelected" [id]="issueRequest?.id" [providerId]="issueProviderId" (visibleChange)="issuesBarVisible = $event;"></issue-report> [id]="issueRequest?.id" [providerId]="issueProviderId" (visibleChange)="issuesBarVisible = $event;"></issue-report>

View file

@ -1,21 +1,13 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Component, Input, OnInit } from "@angular/core"; import { Component, Input, OnInit } from "@angular/core";
import { DomSanitizer } from "@angular/platform-browser"; import { DomSanitizer } from "@angular/platform-browser";
import "rxjs/add/operator/debounceTime"; import { Subject } from "rxjs";
import "rxjs/add/operator/distinctUntilChanged"; import { debounceTime, distinctUntilChanged } from "rxjs/operators";
import "rxjs/add/operator/map";
import { Subject } from "rxjs/Subject";
import { ImageService } from "./../services/image.service";
import "rxjs/add/operator/debounceTime";
import "rxjs/add/operator/distinctUntilChanged";
import "rxjs/add/operator/map";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { FilterType, IIssueCategory, IPagenator, IRequestsViewModel, ISonarrProfile, ISonarrRootFolder, ITvRequests, OrderType } from "../interfaces";
import { NotificationService, RequestService, SonarrService } from "../services"; import { NotificationService, RequestService, SonarrService } from "../services";
import { ImageService } from "./../services/image.service";
import { TreeNode } from "primeng/primeng";
import { IIssueCategory, IPagenator, ISonarrProfile, ISonarrRootFolder, ITvRequests } from "../interfaces";
@Component({ @Component({
selector: "tv-requests", selector: "tv-requests",
@ -24,7 +16,7 @@ import { IIssueCategory, IPagenator, ISonarrProfile, ISonarrRootFolder, ITvRequ
}) })
export class TvRequestsComponent implements OnInit { export class TvRequestsComponent implements OnInit {
public tvRequests: TreeNode[]; public tvRequests: IRequestsViewModel<ITvRequests>;
public searchChanged = new Subject<string>(); public searchChanged = new Subject<string>();
public searchText: string; public searchText: string;
public isAdmin: boolean; public isAdmin: boolean;
@ -46,27 +38,28 @@ export class TvRequestsComponent implements OnInit {
private currentlyLoaded: number; private currentlyLoaded: number;
private amountToLoad: number; private amountToLoad: number;
constructor(private requestService: RequestService, constructor(
private requestService: RequestService,
private auth: AuthService, private auth: AuthService,
private sanitizer: DomSanitizer, private sanitizer: DomSanitizer,
private imageService: ImageService, private imageService: ImageService,
private sonarrService: SonarrService, private sonarrService: SonarrService,
private notificationService: NotificationService, private notificationService: NotificationService,
private readonly platformLocation: PlatformLocation) { private readonly platformLocation: PlatformLocation) {
this.searchChanged this.searchChanged.pipe(
.debounceTime(600) // Wait Xms after the last event before emitting last event debounceTime(600), // Wait Xms after the last event before emitting last event
.distinctUntilChanged() // only emit if value is different from previous value distinctUntilChanged(), // only emit if value is different from previous value
.subscribe(x => { ).subscribe(x => {
this.searchText = x as string; this.searchText = x as string;
if (this.searchText === "") { if (this.searchText === "") {
this.resetSearch(); this.resetSearch();
return; return;
} }
this.requestService.searchTvRequestsTree(this.searchText) this.requestService.searchTvRequests(this.searchText)
.subscribe(m => { .subscribe(m => {
this.tvRequests = m; this.tvRequests.collection = m;
this.tvRequests.forEach((val) => this.loadBackdrop(val)); this.tvRequests.collection.forEach((val) => this.loadBackdrop(val));
this.tvRequests.forEach((val) => this.setOverride(val.data)); this.tvRequests.collection.forEach((val) => this.setOverride(val));
}); });
}); });
this.defaultPoster = "../../../images/default_tv_poster.png"; this.defaultPoster = "../../../images/default_tv_poster.png";
@ -76,47 +69,15 @@ export class TvRequestsComponent implements OnInit {
} }
} }
public openClosestTab(el: any) { public openClosestTab(node: ITvRequests,el: any) {
const rowclass = "undefined ng-star-inserted"; el.preventDefault();
el = el.toElement || el.relatedTarget || el.target || el.srcElement; node.open = !node.open;
if (el.nodeName === "BUTTON") {
const isButtonAlreadyActive = el.parentElement.querySelector(".active");
// if a Button already has Class: .active
if (isButtonAlreadyActive) {
isButtonAlreadyActive.classList.remove("active");
} else {
el.className += " active";
}
}
while (el.className !== rowclass) {
// Increment the loop to the parent node until we find the row we need
el = el.parentNode;
}
// At this point, the while loop has stopped and `el` represents the element that has
// the class you specified
// Then we loop through the children to find the caret which we want to click
const caretright = "fa-caret-right";
const caretdown = "fa-caret-down";
for (const value of el.children) {
// the caret from the ui has 2 class selectors depending on if expanded or not
// we search for both since we want to still toggle the clicking
if (value.className.includes(caretright) || value.className.includes(caretdown)) {
// Then we tell JS to click the element even though we hid it from the UI
value.click();
//Break from loop since we no longer need to continue looking
break;
}
}
} }
public ngOnInit() { public ngOnInit() {
this.amountToLoad = 10; this.amountToLoad = 10;
this.currentlyLoaded = 10; this.currentlyLoaded = 10;
this.tvRequests = []; this.tvRequests = {collection:[], total:0};
this.isAdmin = this.auth.hasRole("admin") || this.auth.hasRole("poweruser"); this.isAdmin = this.auth.hasRole("admin") || this.auth.hasRole("poweruser");
this.loadInit(); this.loadInit();
@ -125,7 +86,7 @@ export class TvRequestsComponent implements OnInit {
public paginate(event: IPagenator) { public paginate(event: IPagenator) {
const skipAmount = event.first; const skipAmount = event.first;
this.requestService.getTvRequestsTree(this.amountToLoad, skipAmount) this.requestService.getTvRequests(this.amountToLoad, skipAmount, OrderType.RequestedDateDesc, FilterType.None, FilterType.None)
.subscribe(x => { .subscribe(x => {
this.tvRequests = x; this.tvRequests = x;
this.currentlyLoaded = this.currentlyLoaded + this.amountToLoad; this.currentlyLoaded = this.currentlyLoaded + this.amountToLoad;
@ -204,17 +165,17 @@ export class TvRequestsComponent implements OnInit {
private loadInit() { private loadInit() {
this.requestService.getTotalTv().subscribe(x => this.totalTv = x); this.requestService.getTotalTv().subscribe(x => this.totalTv = x);
this.requestService.getTvRequestsTree(this.amountToLoad, 0) this.requestService.getTvRequests(this.amountToLoad, 0, OrderType.RequestedDateDesc, FilterType.None, FilterType.None)
.subscribe(x => { .subscribe(x => {
this.tvRequests = x; this.tvRequests = x;
this.tvRequests.forEach((val, index) => { this.tvRequests.collection.forEach((val, index) => {
this.setDefaults(val); this.setDefaults(val);
this.loadBackdrop(val); this.loadBackdrop(val);
this.setOverride(val.data); this.setOverride(val);
}); });
}); });
if(this.isAdmin) { if (this.isAdmin) {
this.sonarrService.getQualityProfilesWithoutSettings() this.sonarrService.getQualityProfilesWithoutSettings()
.subscribe(x => this.sonarrProfiles = x); .subscribe(x => this.sonarrProfiles = x);
@ -228,20 +189,20 @@ export class TvRequestsComponent implements OnInit {
this.loadInit(); this.loadInit();
} }
private setDefaults(val: any) { private setDefaults(val: ITvRequests) {
if (val.data.posterPath === null) { if (val.posterPath === null) {
val.data.posterPath = this.defaultPoster; val.posterPath = this.defaultPoster;
} }
} }
private loadBackdrop(val: TreeNode): void { private loadBackdrop(val: ITvRequests): void {
if (val.data.background != null) { if (val.background != null) {
val.data.background = this.sanitizer.bypassSecurityTrustStyle val.background = this.sanitizer.bypassSecurityTrustStyle
("url(https://image.tmdb.org/t/p/w1280" + val.data.background + ")"); ("url(https://image.tmdb.org/t/p/w1280" + val.background + ")");
} else { } else {
this.imageService.getTvBanner(val.data.tvDbId).subscribe(x => { this.imageService.getTvBanner(val.tvDbId).subscribe(x => {
if(x) { if (x) {
val.data.background = this.sanitizer.bypassSecurityTrustStyle val.background = this.sanitizer.bypassSecurityTrustStyle
("url(" + x + ")"); ("url(" + x + ")");
} }
}); });

View file

@ -1,11 +1,9 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Component, Input, OnInit } from "@angular/core"; import { Component, Input, OnInit } from "@angular/core";
import { DomSanitizer } from "@angular/platform-browser"; import { DomSanitizer } from "@angular/platform-browser";
import { TranslateService } from "@ngx-translate/core"; import { TranslateService } from "@ngx-translate/core";
import "rxjs/add/operator/debounceTime"; import { Subject } from "rxjs";
import "rxjs/add/operator/distinctUntilChanged"; import { debounceTime, distinctUntilChanged } from "rxjs/operators";
import "rxjs/add/operator/map";
import { Subject } from "rxjs/Subject";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { IIssueCategory, IRequestEngineResult, ISearchMovieResult } from "../interfaces"; import { IIssueCategory, IRequestEngineResult, ISearchMovieResult } from "../interfaces";
@ -32,15 +30,16 @@ export class MovieSearchComponent implements OnInit {
public issueCategorySelected: IIssueCategory; public issueCategorySelected: IIssueCategory;
public defaultPoster: string; public defaultPoster: string;
constructor(private searchService: SearchService, private requestService: RequestService, constructor(
private searchService: SearchService, private requestService: RequestService,
private notificationService: NotificationService, private authService: AuthService, private notificationService: NotificationService, private authService: AuthService,
private readonly translate: TranslateService, private sanitizer: DomSanitizer, private readonly translate: TranslateService, private sanitizer: DomSanitizer,
private readonly platformLocation: PlatformLocation) { private readonly platformLocation: PlatformLocation) {
this.searchChanged this.searchChanged.pipe(
.debounceTime(600) // Wait Xms after the last event before emitting last event debounceTime(600), // Wait Xms after the last event before emitting last event
.distinctUntilChanged() // only emit if value is different from previous value distinctUntilChanged(), // only emit if value is different from previous value
.subscribe(x => { ).subscribe(x => {
this.searchText = x as string; this.searchText = x as string;
if (this.searchText === "") { if (this.searchText === "") {
this.clearResults(); this.clearResults();

View file

@ -1,13 +1,10 @@
import { Component, OnInit } from "@angular/core"; import { Component, OnInit } from "@angular/core";
import "rxjs/add/operator/debounceTime"; import { Subject } from "rxjs";
import "rxjs/add/operator/distinctUntilChanged"; import { debounceTime, distinctUntilChanged } from "rxjs/operators";
import "rxjs/add/operator/map";
import { Subject } from "rxjs/Subject";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { NotificationService, RequestService, SearchService } from "../services";
import { IRequestEngineResult, ISearchMovieResult, ISearchMovieResultContainer } from "../interfaces"; import { IRequestEngineResult, ISearchMovieResult, ISearchMovieResultContainer } from "../interfaces";
import { NotificationService, RequestService, SearchService } from "../services";
@Component({ @Component({
selector: "movie-search-grid", selector: "movie-search-grid",
@ -22,13 +19,14 @@ export class MovieSearchGridComponent implements OnInit {
public result: IRequestEngineResult; public result: IRequestEngineResult;
public searchApplied = false; public searchApplied = false;
constructor(private searchService: SearchService, private requestService: RequestService, constructor(
private searchService: SearchService, private requestService: RequestService,
private notificationService: NotificationService, private authService: AuthService) { private notificationService: NotificationService, private authService: AuthService) {
this.searchChanged this.searchChanged.pipe(
.debounceTime(600) // Wait Xms afterthe last event before emitting last event debounceTime(600), // Wait Xms afterthe last event before emitting last event
.distinctUntilChanged() // only emit if value is different from previous value distinctUntilChanged(), // only emit if value is different from previous value
.subscribe(x => { ).subscribe(x => {
this.searchText = x as string; this.searchText = x as string;
if (this.searchText === "") { if (this.searchText === "") {
this.clearResults(); this.clearResults();
@ -67,7 +65,7 @@ export class MovieSearchGridComponent implements OnInit {
} }
try { try {
this.requestService.requestMovie({ theMovieDbId : searchResult.id}) this.requestService.requestMovie({ theMovieDbId: searchResult.id })
.subscribe(x => { .subscribe(x => {
this.result = x; this.result = x;
@ -149,15 +147,14 @@ export class MovieSearchGridComponent implements OnInit {
} }
private processGrid(movies: ISearchMovieResult[]) { private processGrid(movies: ISearchMovieResult[]) {
let container = <ISearchMovieResultContainer>{ movies: [] }; let container = <ISearchMovieResultContainer> { movies: [] };
movies.forEach((movie, i) => { movies.forEach((movie, i) => {
i++; i++;
if((i % 4) === 0) { if ((i % 4) === 0) {
container.movies.push(movie); container.movies.push(movie);
this.movieResultGrid.push(container); this.movieResultGrid.push(container);
container = <ISearchMovieResultContainer>{ movies: [] }; container = <ISearchMovieResultContainer> { movies: [] };
} else { } else {
container.movies.push(movie); container.movies.push(movie);
} }
}); });

View file

@ -1,5 +1,4 @@
import { Component, Input, OnInit} from "@angular/core"; import { Component, Input, OnInit} from "@angular/core";
import "rxjs/add/operator/takeUntil";
import { NotificationService } from "../services"; import { NotificationService } from "../services";
import { RequestService } from "../services"; import { RequestService } from "../services";
@ -39,20 +38,20 @@ export class SeriesInformationComponent implements OnInit {
}); });
}); });
if(!selected) { if (!selected) {
this.notificationService.error("You need to select some episodes!"); this.notificationService.error("You need to select some episodes!");
return; return;
} }
this.series.requested = true; this.series.requested = true;
const viewModel = <ITvRequestViewModel>{ firstSeason: this.series.firstSeason, latestSeason: this.series.latestSeason, requestAll: this.series.requestAll, tvDbId: this.series.id}; const viewModel = <ITvRequestViewModel> { firstSeason: this.series.firstSeason, latestSeason: this.series.latestSeason, requestAll: this.series.requestAll, tvDbId: this.series.id};
viewModel.seasons = []; viewModel.seasons = [];
this.series.seasonRequests.forEach((season) => { this.series.seasonRequests.forEach((season) => {
const seasonsViewModel = <ISeasonsViewModel>{seasonNumber: season.seasonNumber, episodes: []}; const seasonsViewModel = <ISeasonsViewModel> {seasonNumber: season.seasonNumber, episodes: []};
season.episodes.forEach(ep => { season.episodes.forEach(ep => {
if(!this.series.latestSeason || !this.series.requestAll || !this.series.firstSeason) { if (!this.series.latestSeason || !this.series.requestAll || !this.series.firstSeason) {
if(ep.selected) { if (ep.selected) {
seasonsViewModel.episodes.push({episodeNumber: ep.episodeNumber}); seasonsViewModel.episodes.push({episodeNumber: ep.episodeNumber});
} }
} }

View file

@ -42,46 +42,46 @@
<i class='fa fa-film no-search-results-icon'></i> <i class='fa fa-film no-search-results-icon'></i>
<div class='no-search-results-text'>{{ 'Search.NoResults' | translate }}</div> <div class='no-search-results-text'>{{ 'Search.NoResults' | translate }}</div>
</div> </div>
<p-treeTable [value]="tvResults"> <div *ngIf="tvResults" >
<p-column> <div *ngFor="let node of tvResults">
<ng-template let-col let-node="rowData" pTemplate="header">
{{ 'Search.TvShows.Results' | translate }}
</ng-template>
<ng-template let-col let-node="rowData" pTemplate="body">
<!--This is the section that holds the parent level search results set--> <!--This is the section that holds the parent level search results set-->
<div *ngIf="!node.leaf"> <div *ngIf="node">
<div class="row" > <div class="row">
<div class="myBg backdrop" [style.background-image]="node?.data?.background"></div> <div class="myBg backdrop" [style.background-image]="node?.background"></div>
<div class="tint" style="background-image: linear-gradient(to bottom, rgba(0,0,0,0.6) 0%,rgba(0,0,0,0.6) 100%);"></div> <div class="tint" style="background-image: linear-gradient(to bottom, rgba(0,0,0,0.6) 0%,rgba(0,0,0,0.6) 100%);"></div>
<div class="col-sm-2 small-padding"> <div class="col-sm-2 small-padding">
<img *ngIf="node?.data?.banner" class="img-responsive poster" width="150" [src]="node.data.banner" alt="poster"> <img *ngIf="node.banner" class="img-responsive poster" width="150" [src]="node.banner" alt="poster">
</div> </div>
<div class="col-sm-8 small-padding"> <div class="col-sm-8 small-padding">
<div> <div>
<a *ngIf="node.data.imdbId" href="{{node.data.imdbId}}" target="_blank"> <a *ngIf="node.imdbId" href="{{node.imdbId}}" target="_blank">
<h4>{{node.data.title}} ({{node.data.firstAired | date: 'yyyy'}})</h4> <h4>{{node.title}} ({{node.firstAired | date: 'yyyy'}})</h4>
</a> </a>
<span class="tags"> <span class="tags">
<a *ngIf="node.data.homepage" id="homepageLabel" href="{{node.data.homepage}}" target="_blank"><span class="label label-info" >{{ 'Search.Movies.HomePage' | translate }}</span></a> <a *ngIf="node.homepage" id="homepageLabel" href="{{node.homepage}}" target="_blank">
<span class="label label-info">{{ 'Search.Movies.HomePage' | translate }}</span>
</a>
<a *ngIf="node.data.trailer" id="trailerLabel" href="{{node.data.trailer}}" target="_blank"><span class="label label-info">{{ 'Search.Movies.Trailer' | translate }}</span></a> <a *ngIf="node.trailer" id="trailerLabel" href="{{node.trailer}}" target="_blank">
<span class="label label-info">{{ 'Search.Movies.Trailer' | translate }}</span>
</a>
<span *ngIf="node.data.status" class="label label-primary" id="statusLabel" target="_blank">{{node.data.status}}</span> <span *ngIf="node.status" class="label label-primary" id="statusLabel" target="_blank">{{node.status}}</span>
<span *ngIf="node.data.firstAired" class="label label-info" target="_blank" id="airDateLabel">{{ 'Search.TvShows.AirDate' | translate }} {{node.data.firstAired | date: 'dd/MM/yyyy'}}</span> <span *ngIf="node.firstAired" class="label label-info" target="_blank" id="airDateLabel">{{ 'Search.TvShows.AirDate' | translate }} {{node.firstAired | date: 'dd/MM/yyyy'}}</span>
<span *ngIf="node.data.network" class="label label-info" id="networkLabel" target="_blank">{{node.data.network}}</span> <span *ngIf="node.network" class="label label-info" id="networkLabel" target="_blank">{{node.network}}</span>
<span *ngIf="node.data.available" class="label label-success" id="availableLabel">{{ 'Common.Available' | translate }}</span> <span *ngIf="node.available" class="label label-success" id="availableLabel">{{ 'Common.Available' | translate }}</span>
<span *ngIf="node.data.partlyAvailable" class="label label-warning" id="partiallyAvailableLabel">{{ 'Common.PartlyAvailable' | translate }}</span> <span *ngIf="node.partlyAvailable" class="label label-warning" id="partiallyAvailableLabel">{{ 'Common.PartlyAvailable' | translate }}</span>
@ -90,47 +90,45 @@
<br /> <br />
<br /> <br />
</div> </div>
<p class="tv-overview">{{node.data.overview}}</p> <p class="tv-overview">{{node.overview}}</p>
</div> </div>
<div class="col-sm-2 small-padding"> <div class="col-sm-2 small-padding">
<div *ngIf="!node.data.fullyAvailable" class="dropdown"> <div *ngIf="!node.fullyAvailable" class="dropdown">
<button class="btn btn-primary-outline dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="true"> <button class="btn btn-primary-outline dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="true">
<i class="fa fa-plus"></i> {{ 'Common.Request' | translate }} <i class="fa fa-plus"></i> {{ 'Common.Request' | translate }}
<span class="caret"></span> <span class="caret"></span>
</button> </button>
<ul class="dropdown-menu" aria-labelledby="dropdownMenu1"> <ul class="dropdown-menu" aria-labelledby="dropdownMenu1">
<li> <li>
<a href="#" (click)="allSeasons(node.data, $event)">{{ 'Search.TvShows.AllSeasons' | translate }}</a> <a href="#" (click)="allSeasons(node, $event)">{{ 'Search.TvShows.AllSeasons' | translate }}</a>
</li> </li>
<li> <li>
<a href="#" (click)="firstSeason(node.data, $event)">{{ 'Search.TvShows.FirstSeason' | translate }}</a> <a href="#" (click)="firstSeason(node, $event)">{{ 'Search.TvShows.FirstSeason' | translate }}</a>
</li> </li>
<li> <li>
<a href="#" (click)="latestSeason(node.data, $event)">{{ 'Search.TvShows.LatestSeason' | translate }}</a> <a href="#" (click)="latestSeason(node, $event)">{{ 'Search.TvShows.LatestSeason' | translate }}</a>
</li> </li>
<li> <li>
<a href="#" (click)="openClosestTab($event)">{{ 'Search.TvShows.Select' | translate }}</a> <a href="#" (click)="openClosestTab(node, $event)">{{ 'Search.TvShows.Select' | translate }}</a>
</li> </li>
</ul> </ul>
</div> </div>
<div *ngIf="node.data.fullyAvailable"> <div *ngIf="node.fullyAvailable">
<button style="text-align: right" class="btn btn-success-outline disabled" disabled> <button style="text-align: right" class="btn btn-success-outline disabled" disabled>
<i class="fa fa-check"></i> {{ 'Common.Available' | translate }} <i class="fa fa-check"></i> {{ 'Common.Available' | translate }}
</button> </button>
</div> </div>
<br /> <br />
<div *ngIf="node.data.plexUrl && node.data.available"> <div *ngIf="node.plexUrl && node.available">
<a style="text-align: right" class="btn btn-sm btn-success-outline" href="{{node.data.plexUrl}}" <a style="text-align: right" class="btn btn-sm btn-success-outline" href="{{node.plexUrl}}" target="_blank">
target="_blank">
<i class="fa fa-eye"></i> {{ 'Search.ViewOnPlex' | translate }} <i class="fa fa-eye"></i> {{ 'Search.ViewOnPlex' | translate }}
</a> </a>
</div> </div>
<div *ngIf="node.data.embyUrl && node.data.available"> <div *ngIf="node.embyUrl && node.available">
<a style="text-align: right" id="embybtn" class="btn btn-sm btn-success-outline" href="{{node.data.embyUrl}}" <a style="text-align: right" id="embybtn" class="btn btn-sm btn-success-outline" href="{{node.embyUrl}}" target="_blank">
target="_blank">
<i class="fa fa-eye"></i> {{ 'Search.ViewOnEmby' | translate }} <i class="fa fa-eye"></i> {{ 'Search.ViewOnEmby' | translate }}
</a> </a>
</div> </div>
@ -142,11 +140,11 @@
</button> </button>
<ul class="dropdown-menu" aria-labelledby="dropdownMenu1"> <ul class="dropdown-menu" aria-labelledby="dropdownMenu1">
<li *ngFor="let cat of issueCategories"> <li *ngFor="let cat of issueCategories">
<a [routerLink]="" (click)="reportIssue(cat, node.data)">{{cat.value}}</a> <a [routerLink]="" (click)="reportIssue(cat, node)">{{cat.value}}</a>
</li> </li>
</ul> </ul>
</div> </div>
<div *ngIf="!node.data.available"> <div *ngIf="!node.available">
<br /> <br />
<br /> <br />
</div> </div>
@ -156,15 +154,13 @@
</div> </div>
</div> </div>
<!--This is the section that holds the child seasons if they want to specify specific episodes--> <!--This is the section that holds the child seasons if they want to specify specific episodes-->
<div *ngIf="node.leaf"> <div *ngIf="node.open">
<seriesinformation [seriesId]="node.data.id"></seriesinformation> <seriesinformation [seriesId]="node.id"></seriesinformation>
</div> </div>
<br/> <br/>
<br/> <br/>
</ng-template> </div>
</p-column>
</p-treeTable>
</div> </div>
</div> </div>

View file

@ -1,15 +1,13 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Component, Input, OnInit } from "@angular/core"; import { Component, Input, OnInit } from "@angular/core";
import { DomSanitizer } from "@angular/platform-browser"; import { DomSanitizer } from "@angular/platform-browser";
import { Subject } from "rxjs/Subject"; import { Subject } from "rxjs";
import { debounceTime, distinctUntilChanged } from "rxjs/operators";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { IIssueCategory, IRequestEngineResult, ISearchTvResult, ISeasonsViewModel, ITvRequestViewModel } from "../interfaces";
import { ImageService, NotificationService, RequestService, SearchService } from "../services"; import { ImageService, NotificationService, RequestService, SearchService } from "../services";
import { TreeNode } from "primeng/primeng";
import { IRequestEngineResult } from "../interfaces";
import { IIssueCategory, ISearchTvResult, ISeasonsViewModel, ITvRequestViewModel } from "../interfaces";
@Component({ @Component({
selector: "tv-search", selector: "tv-search",
templateUrl: "./tvsearch.component.html", templateUrl: "./tvsearch.component.html",
@ -19,7 +17,7 @@ export class TvSearchComponent implements OnInit {
public searchText: string; public searchText: string;
public searchChanged = new Subject<string>(); public searchChanged = new Subject<string>();
public tvResults: TreeNode[]; public tvResults: ISearchTvResult[];
public result: IRequestEngineResult; public result: IRequestEngineResult;
public searchApplied = false; public searchApplied = false;
public defaultPoster: string; public defaultPoster: string;
@ -32,21 +30,22 @@ export class TvSearchComponent implements OnInit {
public issueProviderId: string; public issueProviderId: string;
public issueCategorySelected: IIssueCategory; public issueCategorySelected: IIssueCategory;
constructor(private searchService: SearchService, private requestService: RequestService, constructor(
private searchService: SearchService, private requestService: RequestService,
private notificationService: NotificationService, private authService: AuthService, private notificationService: NotificationService, private authService: AuthService,
private imageService: ImageService, private sanitizer: DomSanitizer, private imageService: ImageService, private sanitizer: DomSanitizer,
private readonly platformLocation: PlatformLocation) { private readonly platformLocation: PlatformLocation) {
this.searchChanged this.searchChanged.pipe(
.debounceTime(600) // Wait Xms after the last event before emitting last event debounceTime(600), // Wait Xms after the last event before emitting last event
.distinctUntilChanged() // only emit if value is different from previous value distinctUntilChanged(), // only emit if value is different from previous value
.subscribe(x => { ).subscribe(x => {
this.searchText = x as string; this.searchText = x as string;
if (this.searchText === "") { if (this.searchText === "") {
this.clearResults(); this.clearResults();
return; return;
} }
this.searchService.searchTvTreeNode(this.searchText) this.searchService.searchTv(this.searchText)
.subscribe(x => { .subscribe(x => {
this.tvResults = x; this.tvResults = x;
this.searchApplied = true; this.searchApplied = true;
@ -55,34 +54,13 @@ export class TvSearchComponent implements OnInit {
}); });
this.defaultPoster = "../../../images/default_tv_poster.png"; this.defaultPoster = "../../../images/default_tv_poster.png";
const base = this.platformLocation.getBaseHrefFromDOM(); const base = this.platformLocation.getBaseHrefFromDOM();
if(base) { if (base) {
this.defaultPoster = "../../.." + base + "/images/default_tv_poster.png"; this.defaultPoster = "../../.." + base + "/images/default_tv_poster.png";
} }
} }
public openClosestTab(el: any) { public openClosestTab(node: ISearchTvResult,el: any) {
el.preventDefault(); el.preventDefault();
const rowclass = "undefined ng-star-inserted"; node.open = !node.open;
el = el.toElement || el.relatedTarget || el.target;
while (el.className !== rowclass) {
// Increment the loop to the parent node until we find the row we need
el = el.parentNode;
}
// At this point, the while loop has stopped and `el` represents the element that has
// the class you specified
// Then we loop through the children to find the caret which we want to click
const caretright = "fa-caret-right";
const caretdown = "fa-caret-down";
for (const value of el.children) {
// the caret from the ui has 2 class selectors depending on if expanded or not
// we search for both since we want to still toggle the clicking
if (value.className.includes(caretright) || value.className.includes(caretdown)) {
// Then we tell JS to click the element even though we hid it from the UI
value.click();
//Break from loop since we no longer need to continue looking
break;
}
}
} }
public ngOnInit() { public ngOnInit() {
@ -91,7 +69,7 @@ export class TvSearchComponent implements OnInit {
this.result = { this.result = {
message: "", message: "",
result: false, result: false,
errorMessage:"", errorMessage: "",
}; };
this.popularShows(); this.popularShows();
} }
@ -138,16 +116,16 @@ export class TvSearchComponent implements OnInit {
public getExtraInfo() { public getExtraInfo() {
this.tvResults.forEach((val, index) => { this.tvResults.forEach((val, index) => {
this.imageService.getTvBanner(val.data.id).subscribe(x => { this.imageService.getTvBanner(val.id).subscribe(x => {
if(x) { if (x) {
val.data.background = this.sanitizer. val.background = this.sanitizer.
bypassSecurityTrustStyle bypassSecurityTrustStyle
("url(" + x + ")"); ("url(" + x + ")");
} }
}); });
this.searchService.getShowInformationTreeNode(val.data.id) this.searchService.getShowInformation(val.id)
.subscribe(x => { .subscribe(x => {
if (x.data) { if (x) {
this.setDefaults(x); this.setDefaults(x);
this.updateItem(val, x); this.updateItem(val, x);
} else { } else {
@ -166,14 +144,14 @@ export class TvSearchComponent implements OnInit {
searchResult.approved = true; searchResult.approved = true;
} }
const viewModel = <ITvRequestViewModel>{ firstSeason: searchResult.firstSeason, latestSeason: searchResult.latestSeason, requestAll: searchResult.requestAll, tvDbId: searchResult.id}; const viewModel = <ITvRequestViewModel> { firstSeason: searchResult.firstSeason, latestSeason: searchResult.latestSeason, requestAll: searchResult.requestAll, tvDbId: searchResult.id };
viewModel.seasons = []; viewModel.seasons = [];
searchResult.seasonRequests.forEach((season) => { searchResult.seasonRequests.forEach((season) => {
const seasonsViewModel = <ISeasonsViewModel>{seasonNumber: season.seasonNumber, episodes: []}; const seasonsViewModel = <ISeasonsViewModel> { seasonNumber: season.seasonNumber, episodes: [] };
season.episodes.forEach(ep => { season.episodes.forEach(ep => {
if(!searchResult.latestSeason || !searchResult.requestAll || !searchResult.firstSeason) { if (!searchResult.latestSeason || !searchResult.requestAll || !searchResult.firstSeason) {
if(ep.requested) { if (ep.requested) {
seasonsViewModel.episodes.push({episodeNumber: ep.episodeNumber}); seasonsViewModel.episodes.push({ episodeNumber: ep.episodeNumber });
} }
} }
}); });
@ -223,29 +201,29 @@ export class TvSearchComponent implements OnInit {
this.issueProviderId = req.id.toString(); this.issueProviderId = req.id.toString();
} }
private updateItem(key: TreeNode, updated: TreeNode) { private updateItem(key: ISearchTvResult, updated: ISearchTvResult) {
const index = this.tvResults.indexOf(key, 0); const index = this.tvResults.indexOf(key, 0);
if (index > -1) { if (index > -1) {
// Update certain properties, otherwise we will loose some data // Update certain properties, otherwise we will loose some data
this.tvResults[index].data.title = updated.data.title; this.tvResults[index].title = updated.title;
this.tvResults[index].data.banner = updated.data.banner; this.tvResults[index].banner = updated.banner;
this.tvResults[index].data.imdbId = updated.data.imdbId; this.tvResults[index].imdbId = updated.imdbId;
this.tvResults[index].data.seasonRequests = updated.data.seasonRequests; this.tvResults[index].seasonRequests = updated.seasonRequests;
this.tvResults[index].data.seriesId = updated.data.seriesId; this.tvResults[index].seriesId = updated.seriesId;
this.tvResults[index].data.fullyAvailable = updated.data.fullyAvailable; this.tvResults[index].fullyAvailable = updated.fullyAvailable;
this.tvResults[index].data.backdrop = updated.data.backdrop; this.tvResults[index].background = updated.banner;
} }
} }
private setDefaults(x: any) { private setDefaults(x: ISearchTvResult) {
if (x.data.banner === null) { if (x.banner === null) {
x.data.banner = this.defaultPoster; x.banner = this.defaultPoster;
} }
if (x.data.imdbId === null) { if (x.imdbId === null) {
x.data.imdbId = "https://www.tvmaze.com/shows/" + x.data.seriesId; x.imdbId = "https://www.tvmaze.com/shows/" + x.seriesId;
} else { } else {
x.data.imdbId = "http://www.imdb.com/title/" + x.data.imdbId + "/"; x.imdbId = "http://www.imdb.com/title/" + x.imdbId + "/";
} }
} }

View file

@ -1,7 +1,7 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../service.helpers"; import { ServiceHelpers } from "../service.helpers";

View file

@ -1,7 +1,7 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../service.helpers"; import { ServiceHelpers } from "../service.helpers";

View file

@ -1,12 +1,12 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../service.helpers"; import { ServiceHelpers } from "../service.helpers";
import { IPlexAuthentication, IPlexLibResponse, IPlexOAuthViewModel,IPlexServer, IPlexServerViewModel, IUsersModel } from "../../interfaces"; import { IPlexAuthentication, IPlexLibResponse, IPlexOAuthViewModel, IPlexServer, IPlexServerViewModel, IUsersModel } from "../../interfaces";
@Injectable() @Injectable()
export class PlexService extends ServiceHelpers { export class PlexService extends ServiceHelpers {

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../service.helpers"; import { ServiceHelpers } from "../service.helpers";

View file

@ -2,7 +2,7 @@
import { HttpClient, HttpHeaders } from "@angular/common/http"; import { HttpClient, HttpHeaders } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IPlexPin } from "../../interfaces"; import { IPlexPin } from "../../interfaces";
@ -10,11 +10,10 @@ import { IPlexPin } from "../../interfaces";
export class PlexTvService { export class PlexTvService {
constructor(private http: HttpClient, public platformLocation: PlatformLocation) { constructor(private http: HttpClient, public platformLocation: PlatformLocation) {
} }
public GetPin(clientId: string, applicationName: string): Observable<IPlexPin> { public GetPin(clientId: string, applicationName: string): Observable<IPlexPin> {
const headers = new HttpHeaders({"Content-Type":"application/json", const headers = new HttpHeaders({"Content-Type": "application/json",
"X-Plex-Client-Identifier": clientId, "X-Plex-Client-Identifier": clientId,
"X-Plex-Product": applicationName, "X-Plex-Product": applicationName,
"X-Plex-Version": "3", "X-Plex-Version": "3",

View file

@ -1,7 +1,7 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IRadarrProfile, IRadarrRootFolder } from "../../interfaces"; import { IRadarrProfile, IRadarrRootFolder } from "../../interfaces";
import { IRadarrSettings } from "../../interfaces"; import { IRadarrSettings } from "../../interfaces";

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ISonarrSettings } from "../../interfaces"; import { ISonarrSettings } from "../../interfaces";
import { ISonarrProfile, ISonarrRootFolder } from "../../interfaces"; import { ISonarrProfile, ISonarrRootFolder } from "../../interfaces";

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "../service.helpers"; import { ServiceHelpers } from "../service.helpers";
@ -37,6 +37,7 @@ export class TesterService extends ServiceHelpers {
public pushbulletTest(settings: IPushbulletNotificationSettings): Observable<boolean> { public pushbulletTest(settings: IPushbulletNotificationSettings): Observable<boolean> {
return this.http.post<boolean>(`${this.url}pushbullet`, JSON.stringify(settings), {headers: this.headers}); return this.http.post<boolean>(`${this.url}pushbullet`, JSON.stringify(settings), {headers: this.headers});
} }
public pushoverTest(settings: IPushoverNotificationSettings): Observable<boolean> { public pushoverTest(settings: IPushoverNotificationSettings): Observable<boolean> {
return this.http.post<boolean>(`${this.url}pushover`, JSON.stringify(settings), {headers: this.headers}); return this.http.post<boolean>(`${this.url}pushover`, JSON.stringify(settings), {headers: this.headers});
} }
@ -80,9 +81,11 @@ export class TesterService extends ServiceHelpers {
public sickrageTest(settings: ISickRageSettings): Observable<boolean> { public sickrageTest(settings: ISickRageSettings): Observable<boolean> {
return this.http.post<boolean>(`${this.url}sickrage`, JSON.stringify(settings), {headers: this.headers}); return this.http.post<boolean>(`${this.url}sickrage`, JSON.stringify(settings), {headers: this.headers});
} }
public newsletterTest(settings: INewsletterNotificationSettings): Observable<boolean> { public newsletterTest(settings: INewsletterNotificationSettings): Observable<boolean> {
return this.http.post<boolean>(`${this.url}newsletter`, JSON.stringify(settings), {headers: this.headers}); return this.http.post<boolean>(`${this.url}newsletter`, JSON.stringify(settings), {headers: this.headers});
} }
public mobileNotificationTest(settings: IMobileNotificationTestSettings): Observable<boolean> { public mobileNotificationTest(settings: IMobileNotificationTestSettings): Observable<boolean> {
return this.http.post<boolean>(`${this.url}mobile`, JSON.stringify(settings), {headers: this.headers}); return this.http.post<boolean>(`${this.url}mobile`, JSON.stringify(settings), {headers: this.headers});
} }

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ICheckbox, ICreateWizardUser, IIdentityResult, IResetPasswordToken, IUpdateLocalUser, IUser, IWizardUserResult } from "../interfaces"; import { ICheckbox, ICreateWizardUser, IIdentityResult, IResetPasswordToken, IUpdateLocalUser, IUser, IWizardUserResult } from "../interfaces";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";

View file

@ -1,6 +1,6 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";

View file

@ -1,10 +1,10 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IIssueCategory, IIssueComments,IIssueCount, IIssues, IIssuesChat, INewIssueComments, IssueStatus, IUpdateStatus } from "../interfaces"; import { IIssueCategory, IIssueComments, IIssueCount, IIssues, IIssuesChat, INewIssueComments, IssueStatus, IUpdateStatus } from "../interfaces";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";
@Injectable() @Injectable()

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";

View file

@ -1,6 +1,6 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IMobileUsersViewModel } from "./../interfaces"; import { IMobileUsersViewModel } from "./../interfaces";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IMassEmailModel } from "./../interfaces"; import { IMassEmailModel } from "./../interfaces";

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { IRecentlyAddedMovies, IRecentlyAddedTvShows } from "./../interfaces"; import { IRecentlyAddedMovies, IRecentlyAddedTvShows } from "./../interfaces";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";
@ -19,6 +19,7 @@ export class RecentlyAddedService extends ServiceHelpers {
public getRecentlyAddedTv(): Observable<IRecentlyAddedTvShows[]> { public getRecentlyAddedTv(): Observable<IRecentlyAddedTvShows[]> {
return this.http.get<IRecentlyAddedTvShows[]>(`${this. url}tv/`, {headers: this.headers}); return this.http.get<IRecentlyAddedTvShows[]>(`${this. url}tv/`, {headers: this.headers});
} }
public getRecentlyAddedTvGrouped(): Observable<IRecentlyAddedTvShows[]> { public getRecentlyAddedTvGrouped(): Observable<IRecentlyAddedTvShows[]> {
return this.http.get<IRecentlyAddedTvShows[]>(`${this.url}tv/grouped`, {headers: this.headers}); return this.http.get<IRecentlyAddedTvShows[]>(`${this.url}tv/grouped`, {headers: this.headers});
} }

View file

@ -1,12 +1,11 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { TreeNode } from "primeng/primeng"; import { TreeNode } from "primeng/primeng";
import { IRequestEngineResult } from "../interfaces"; import { FilterType, IChildRequests, IFilter, IMovieRequestModel, IMovieRequests, IMovieUpdateModel, IRequestEngineResult, IRequestsViewModel, ITvRequests, ITvUpdateModel, OrderType } from "../interfaces";
import { IChildRequests, IFilter, IMovieRequestModel, IMovieRequests, IMovieUpdateModel, IRequestsViewModel, ITvRequests,ITvUpdateModel, OrderType } from "../interfaces";
import { ITvRequestViewModel } from "../interfaces"; import { ITvRequestViewModel } from "../interfaces";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";
@ -64,8 +63,8 @@ export class RequestService extends ServiceHelpers {
return this.http.put<IMovieRequests>(`${this.url}movie/`, JSON.stringify(request), {headers: this.headers}); return this.http.put<IMovieRequests>(`${this.url}movie/`, JSON.stringify(request), {headers: this.headers});
} }
public getTvRequests(count: number, position: number): Observable<ITvRequests[]> { public getTvRequests(count: number, position: number, order: OrderType, status: FilterType, availability: FilterType): Observable<IRequestsViewModel<ITvRequests>> {
return this.http.get<ITvRequests[]>(`${this.url}tv/${count}/${position}`, {headers: this.headers}); return this.http.get<IRequestsViewModel<ITvRequests>>(`${this.url}tv/${count}/${position}/${order}/${status}/${availability}`, {headers: this.headers});
} }
public getTvRequestsTree(count: number, position: number): Observable<TreeNode[]> { public getTvRequestsTree(count: number, position: number): Observable<TreeNode[]> {

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { TreeNode } from "primeng/primeng"; import { TreeNode } from "primeng/primeng";
import { ISearchMovieResult } from "../interfaces"; import { ISearchMovieResult } from "../interfaces";
@ -56,16 +56,16 @@ export class SearchService extends ServiceHelpers {
return this.http.get<ISearchTvResult>(`${this.url}/Tv/info/${theTvDbId}`, {headers: this.headers}); return this.http.get<ISearchTvResult>(`${this.url}/Tv/info/${theTvDbId}`, {headers: this.headers});
} }
public popularTv(): Observable<TreeNode[]> { public popularTv(): Observable<ISearchTvResult[]> {
return this.http.get<TreeNode[]>(`${this.url}/Tv/popular/tree`, {headers: this.headers}); return this.http.get<ISearchTvResult[]>(`${this.url}/Tv/popular`, {headers: this.headers});
} }
public mostWatchedTv(): Observable<TreeNode[]> { public mostWatchedTv(): Observable<ISearchTvResult[]> {
return this.http.get<TreeNode[]>(`${this.url}/Tv/mostwatched/tree`, {headers: this.headers}); return this.http.get<ISearchTvResult[]>(`${this.url}/Tv/mostwatched`, {headers: this.headers});
} }
public anticipatedTv(): Observable<TreeNode[]> { public anticipatedTv(): Observable<ISearchTvResult[]> {
return this.http.get<TreeNode[]>(`${this.url}/Tv/anticipated/tree`, {headers: this.headers}); return this.http.get<ISearchTvResult[]>(`${this.url}/Tv/anticipated`, {headers: this.headers});
} }
public trendingTv(): Observable<TreeNode[]> { public trendingTv(): Observable<ISearchTvResult[]> {
return this.http.get<TreeNode[]>(`${this.url}/Tv/trending/tree`, {headers: this.headers}); return this.http.get<ISearchTvResult[]>(`${this.url}/Tv/trending`, {headers: this.headers});
} }
} }

View file

@ -1,6 +1,5 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient, HttpHeaders } from "@angular/common/http"; import { HttpClient, HttpHeaders } from "@angular/common/http";
import "rxjs/add/observable/throw";
export class ServiceHelpers { export class ServiceHelpers {

View file

@ -1,7 +1,7 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { import {
IAbout, IAbout,

View file

@ -1,8 +1,8 @@
import { PlatformLocation } from "@angular/common"; import { PlatformLocation } from "@angular/common";
import { Injectable } from "@angular/core"; import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http"; import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs/Rx"; import { Observable } from "rxjs";
import { ServiceHelpers } from "./service.helpers"; import { ServiceHelpers } from "./service.helpers";

View file

@ -25,13 +25,13 @@ export class CustomizationComponent implements OnInit {
return item.fullName === this.settings.presetThemeName; return item.fullName === this.settings.presetThemeName;
})[0]; })[0];
if(existingTheme) { if (existingTheme) {
const index = this.themes.indexOf(existingTheme, 0); const index = this.themes.indexOf(existingTheme, 0);
if (index > -1) { if (index > -1) {
this.themes.splice(index, 1); this.themes.splice(index, 1);
} }
} }
if(x.hasPresetTheme) { if (x.hasPresetTheme) {
this.themes.unshift({displayName: x.presetThemeDisplayName, fullName: x.presetThemeName, url: existingTheme.url, version: x.presetThemeVersion}); this.themes.unshift({displayName: x.presetThemeDisplayName, fullName: x.presetThemeName, url: existingTheme.url, version: x.presetThemeVersion});
this.themes.unshift({displayName: "None", fullName: "None", url: "", version: ""}); this.themes.unshift({displayName: "None", fullName: "None", url: "", version: ""});
} else { } else {
@ -45,8 +45,8 @@ export class CustomizationComponent implements OnInit {
public save() { public save() {
this.settingsService.verifyUrl(this.settings.applicationUrl).subscribe(x => { this.settingsService.verifyUrl(this.settings.applicationUrl).subscribe(x => {
if(this.settings.applicationUrl) { if (this.settings.applicationUrl) {
if(!x) { if (!x) {
this.notificationService.error(`The URL "${this.settings.applicationUrl}" is not valid. Please format it correctly e.g. http://www.google.com/`); this.notificationService.error(`The URL "${this.settings.applicationUrl}" is not valid. Please format it correctly e.g. http://www.google.com/`);
return; return;
} }
@ -64,16 +64,16 @@ export class CustomizationComponent implements OnInit {
} }
public dropDownChange(event: any): void { public dropDownChange(event: any): void {
const selectedThemeFullName = <string>event.target.value; const selectedThemeFullName = <string> event.target.value;
const selectedTheme = this.themes.filter((val) => { const selectedTheme = this.themes.filter((val) => {
return val.fullName === selectedThemeFullName; return val.fullName === selectedThemeFullName;
})[0]; })[0];
if(selectedTheme.fullName === this.settings.presetThemeName) { if (selectedTheme.fullName === this.settings.presetThemeName) {
return; return;
} }
if(selectedTheme.fullName === "None" || selectedTheme.fullName === "-1") { if (selectedTheme.fullName === "None" || selectedTheme.fullName === "-1") {
this.settings.presetThemeName = ""; this.settings.presetThemeName = "";
this.settings.presetThemeContent = ""; this.settings.presetThemeContent = "";
return; return;

View file

@ -39,7 +39,7 @@ export class DiscordComponent implements OnInit {
return; return;
} }
const settings = <IDiscordNotifcationSettings>form.value; const settings = <IDiscordNotifcationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.saveDiscordNotificationSettings(settings).subscribe(x => { this.settingsService.saveDiscordNotificationSettings(settings).subscribe(x => {

View file

@ -54,7 +54,7 @@ export class EmailNotificationComponent implements OnInit {
return; return;
} }
const settings = <IEmailNotificationSettings>form.value; const settings = <IEmailNotificationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.saveEmailNotificationSettings(settings).subscribe(x => { this.settingsService.saveEmailNotificationSettings(settings).subscribe(x => {

View file

@ -29,7 +29,7 @@ export class MattermostComponent implements OnInit {
username: [x.username], username: [x.username],
webhookUrl: [x.webhookUrl, [Validators.required]], webhookUrl: [x.webhookUrl, [Validators.required]],
channel: [x.channel], channel: [x.channel],
iconUrl:[x.iconUrl], iconUrl: [x.iconUrl],
}); });
}); });
@ -41,7 +41,7 @@ export class MattermostComponent implements OnInit {
return; return;
} }
const settings = <IMattermostNotifcationSettings>form.value; const settings = <IMattermostNotifcationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.saveMattermostNotificationSettings(settings).subscribe(x => { this.settingsService.saveMattermostNotificationSettings(settings).subscribe(x => {

View file

@ -32,9 +32,9 @@ export class MobileComponent implements OnInit {
}); });
this.mobileService.getUserDeviceList().subscribe(x => { this.mobileService.getUserDeviceList().subscribe(x => {
if(x.length <= 0) { if (x.length <= 0) {
this.userList = []; this.userList = [];
this.userList.push({username:"None",devices:0, userId:""}); this.userList.push({username: "None", devices: 0, userId: ""});
} else { } else {
this.userList = x; this.userList = x;
} }
@ -47,7 +47,7 @@ export class MobileComponent implements OnInit {
return; return;
} }
const settings = <IMobileNotifcationSettings>form.value; const settings = <IMobileNotifcationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.saveMobileNotificationSettings(settings).subscribe(x => { this.settingsService.saveMobileNotificationSettings(settings).subscribe(x => {
@ -65,8 +65,8 @@ export class MobileComponent implements OnInit {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
if(!this.testUserId) { if (!this.testUserId) {
this.notificationService.warning("Warning","Please select a user to send the test notification"); this.notificationService.warning("Warning", "Please select a user to send the test notification");
return; return;
} }

View file

@ -51,10 +51,10 @@ export class NewsletterComponent implements OnInit {
} }
public addEmail() { public addEmail() {
if(this.emailToAdd) { if (this.emailToAdd) {
const emailRegex = "[a-zA-Z0-9.-_]{1,}@[a-zA-Z.-]{2,}[.]{1}[a-zA-Z]{2,}"; const emailRegex = "[a-zA-Z0-9.-_]{1,}@[a-zA-Z.-]{2,}[.]{1}[a-zA-Z]{2,}";
const match = this.emailToAdd.match(emailRegex)!; const match = this.emailToAdd.match(emailRegex)!;
if(match && match.length > 0) { if (match && match.length > 0) {
this.settings.externalEmails.push(this.emailToAdd); this.settings.externalEmails.push(this.emailToAdd);
this.emailToAdd = ""; this.emailToAdd = "";
} else { } else {

View file

@ -37,7 +37,7 @@ export class PushbulletComponent implements OnInit {
return; return;
} }
const settings = <IPushbulletNotificationSettings>form.value; const settings = <IPushbulletNotificationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.savePushbulletNotificationSettings(settings).subscribe(x => { this.settingsService.savePushbulletNotificationSettings(settings).subscribe(x => {

View file

@ -37,7 +37,7 @@ export class PushoverComponent implements OnInit {
return; return;
} }
const settings = <IPushoverNotificationSettings>form.value; const settings = <IPushoverNotificationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.savePushoverNotificationSettings(settings).subscribe(x => { this.settingsService.savePushoverNotificationSettings(settings).subscribe(x => {

View file

@ -41,7 +41,7 @@ export class SlackComponent implements OnInit {
return; return;
} }
const settings = <ISlackNotificationSettings>form.value; const settings = <ISlackNotificationSettings> form.value;
if (settings.iconEmoji && settings.iconUrl) { if (settings.iconEmoji && settings.iconUrl) {
this.notificationService.error("You cannot have a Emoji icon and a URL icon"); this.notificationService.error("You cannot have a Emoji icon and a URL icon");
@ -65,7 +65,7 @@ export class SlackComponent implements OnInit {
return; return;
} }
const settings = <ISlackNotificationSettings>form.value; const settings = <ISlackNotificationSettings> form.value;
if (settings.iconEmoji && settings.iconUrl) { if (settings.iconEmoji && settings.iconUrl) {
this.notificationService.error("You cannot have a Emoji icon and a URL icon"); this.notificationService.error("You cannot have a Emoji icon and a URL icon");

View file

@ -40,7 +40,7 @@ export class TelegramComponent implements OnInit {
return; return;
} }
const settings = <ITelegramNotifcationSettings>form.value; const settings = <ITelegramNotifcationSettings> form.value;
settings.notificationTemplates = this.templates; settings.notificationTemplates = this.templates;
this.settingsService.saveTelegramNotificationSettings(settings).subscribe(x => { this.settingsService.saveTelegramNotificationSettings(settings).subscribe(x => {

View file

@ -41,9 +41,9 @@ export class OmbiComponent implements OnInit {
return; return;
} }
const result = <IOmbiSettings>form.value; const result = <IOmbiSettings> form.value;
if(result.baseUrl && result.baseUrl.length > 0) { if (result.baseUrl && result.baseUrl.length > 0) {
if(!result.baseUrl.startsWith("/")) { if (!result.baseUrl.startsWith("/")) {
this.notificationService.error("Please ensure your base url starts with a '/'"); this.notificationService.error("Please ensure your base url starts with a '/'");
return; return;
} }

View file

@ -1,10 +1,8 @@
import { Component, OnDestroy, OnInit } from "@angular/core"; import { Component, OnDestroy, OnInit } from "@angular/core";
import "rxjs/add/operator/takeUntil"; import { Subject } from "rxjs";
import { Subject } from "rxjs/Subject"; import { takeUntil } from "rxjs/operators";
import { IPlexServerResponse, IPlexServerViewModel } from "../../interfaces";
import { IPlexLibrariesSettings, IPlexServer, IPlexSettings } from "../../interfaces";
import { IPlexLibrariesSettings, IPlexServer, IPlexServerResponse, IPlexServerViewModel, IPlexSettings } from "../../interfaces";
import { JobService, NotificationService, PlexService, SettingsService, TesterService } from "../../services"; import { JobService, NotificationService, PlexService, SettingsService, TesterService } from "../../services";
@Component({ @Component({
@ -21,7 +19,8 @@ export class PlexComponent implements OnInit, OnDestroy {
private subscriptions = new Subject<void>(); private subscriptions = new Subject<void>();
constructor(private settingsService: SettingsService, constructor(
private settingsService: SettingsService,
private notificationService: NotificationService, private notificationService: NotificationService,
private plexService: PlexService, private plexService: PlexService,
private testerService: TesterService, private testerService: TesterService,
@ -34,9 +33,9 @@ export class PlexComponent implements OnInit, OnDestroy {
} }
public requestServers(server: IPlexServer) { public requestServers(server: IPlexServer) {
this.plexService.getServers(this.username, this.password) this.plexService.getServers(this.username, this.password).pipe(
.takeUntil(this.subscriptions) takeUntil(this.subscriptions),
.subscribe(x => { ).subscribe(x => {
if (x.success) { if (x.success) {
this.loadedServers = x; this.loadedServers = x;
this.serversButton = true; this.serversButton = true;
@ -72,7 +71,7 @@ export class PlexComponent implements OnInit, OnDestroy {
if (this.settings.servers == null) { if (this.settings.servers == null) {
this.settings.servers = []; this.settings.servers = [];
} }
this.settings.servers.push(<IPlexServer>{ name: "New*", id: Math.floor(Math.random() * (99999 - 0 + 1) + 1) }); this.settings.servers.push(<IPlexServer> { name: "New*", id: Math.floor(Math.random() * (99999 - 0 + 1) + 1) });
} }
@ -120,7 +119,7 @@ export class PlexComponent implements OnInit, OnDestroy {
public runCacher(): void { public runCacher(): void {
this.jobService.runPlexCacher().subscribe(x => { this.jobService.runPlexCacher().subscribe(x => {
if(x) { if (x) {
this.notificationService.success("Triggered the Plex Full Sync"); this.notificationService.success("Triggered the Plex Full Sync");
} }
}); });
@ -128,7 +127,7 @@ export class PlexComponent implements OnInit, OnDestroy {
public runRecentlyAddedCacher(): void { public runRecentlyAddedCacher(): void {
this.jobService.runPlexRecentlyAddedCacher().subscribe(x => { this.jobService.runPlexRecentlyAddedCacher().subscribe(x => {
if(x) { if (x) {
this.notificationService.success("Triggered the Plex Recently Added Sync"); this.notificationService.success("Triggered the Plex Recently Added Sync");
} }
}); });

View file

@ -93,7 +93,7 @@ export class RadarrComponent implements OnInit {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
const settings = <IRadarrSettings>form.value; const settings = <IRadarrSettings> form.value;
this.testerService.radarrTest(settings).subscribe(x => { this.testerService.radarrTest(settings).subscribe(x => {
if (x === true) { if (x === true) {
this.notificationService.success("Successfully connected to Radarr!"); this.notificationService.success("Successfully connected to Radarr!");
@ -108,12 +108,12 @@ public onSubmit(form: FormGroup) {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
if(form.controls.defaultQualityProfile.value === "-1" || form.controls.defaultRootPath.value === "Please Select") { if (form.controls.defaultQualityProfile.value === "-1" || form.controls.defaultRootPath.value === "Please Select") {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
const settings = <IRadarrSettings>form.value; const settings = <IRadarrSettings> form.value;
this.settingsService.saveRadarr(settings).subscribe(x => { this.settingsService.saveRadarr(settings).subscribe(x => {
if (x) { if (x) {
this.notificationService.success("Successfully saved Radarr settings"); this.notificationService.success("Successfully saved Radarr settings");

View file

@ -1,14 +1,16 @@
import { CommonModule } from "@angular/common"; import { CommonModule } from "@angular/common";
import { NgModule } from "@angular/core"; import { NgModule } from "@angular/core";
import { FormsModule, ReactiveFormsModule } from "@angular/forms"; import { FormsModule, ReactiveFormsModule } from "@angular/forms";
import { RouterModule, Routes } from "@angular/router"; import { RouterModule, Routes } from "@angular/router";
import { NgbAccordionModule, NgbModule } from "@ng-bootstrap/ng-bootstrap"; import { NgbAccordionModule, NgbModule } from "@ng-bootstrap/ng-bootstrap";
import { ClipboardModule } from "ngx-clipboard/dist"; import { ClipboardModule } from "ngx-clipboard";
import { AuthGuard } from "../auth/auth.guard"; import { AuthGuard } from "../auth/auth.guard";
import { AuthService } from "../auth/auth.service"; import { AuthService } from "../auth/auth.service";
import { CouchPotatoService, EmbyService, IssuesService, JobService, MobileService, NotificationMessageService, PlexService, RadarrService, import {
SonarrService, TesterService, ValidationService } from "../services"; CouchPotatoService, EmbyService, IssuesService, JobService, MobileService, NotificationMessageService, PlexService, RadarrService,
SonarrService, TesterService, ValidationService,
} from "../services";
import { PipeModule } from "../pipes/pipe.module"; import { PipeModule } from "../pipes/pipe.module";
import { AboutComponent } from "./about/about.component"; import { AboutComponent } from "./about/about.component";

View file

@ -40,7 +40,7 @@ export class SickRageComponent implements OnInit {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
const settings = <ISickRageSettings>form.value; const settings = <ISickRageSettings> form.value;
this.testerService.sickrageTest(settings).subscribe(x => { this.testerService.sickrageTest(settings).subscribe(x => {
if (x) { if (x) {
this.notificationService.success("Successfully connected to SickRage!"); this.notificationService.success("Successfully connected to SickRage!");

View file

@ -93,7 +93,7 @@ export class SonarrComponent implements OnInit {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
const settings = <ISonarrSettings>form.value; const settings = <ISonarrSettings> form.value;
this.testerService.sonarrTest(settings).subscribe(x => { this.testerService.sonarrTest(settings).subscribe(x => {
if (x) { if (x) {
this.notificationService.success("Successfully connected to Sonarr!"); this.notificationService.success("Successfully connected to Sonarr!");
@ -108,13 +108,13 @@ export class SonarrComponent implements OnInit {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
return; return;
} }
if(form.controls.defaultQualityProfile) { if (form.controls.defaultQualityProfile) {
if(form.controls.defaultQualityProfile.value === "-1") { if (form.controls.defaultQualityProfile.value === "-1") {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
} }
} }
if(form.controls.defaultRootPath) { if (form.controls.defaultRootPath) {
if(form.controls.defaultRootPath.value === "Please Select") { if (form.controls.defaultRootPath.value === "Please Select") {
this.notificationService.error("Please check your entered values"); this.notificationService.error("Please check your entered values");
} }
} }

View file

@ -19,7 +19,7 @@ export class UpdateDetailsComponent implements OnInit {
this.identityService.getUser().subscribe(x => { this.identityService.getUser().subscribe(x => {
const localUser = x as IUpdateLocalUser; const localUser = x as IUpdateLocalUser;
this.form = this.fb.group({ this.form = this.fb.group({
id:[localUser.id], id: [localUser.id],
username: [localUser.userName], username: [localUser.userName],
emailAddress: [localUser.emailAddress, [Validators.email]], emailAddress: [localUser.emailAddress, [Validators.email]],
confirmNewPassword: [localUser.confirmNewPassword], confirmNewPassword: [localUser.confirmNewPassword],

View file

@ -1,6 +1,6 @@
import { Component, OnInit } from "@angular/core"; import { Component, OnInit } from "@angular/core";
import { ICheckbox, ICustomizationSettings, IEmailNotificationSettings,IUser } from "../interfaces"; import { ICheckbox, ICustomizationSettings, IEmailNotificationSettings, IUser } from "../interfaces";
import { IdentityService, NotificationService, SettingsService } from "../services"; import { IdentityService, NotificationService, SettingsService } from "../services";
@Component({ @Component({
@ -21,9 +21,9 @@ export class UserManagementComponent implements OnInit {
public bulkMovieLimit?: number; public bulkMovieLimit?: number;
public bulkEpisodeLimit?: number; public bulkEpisodeLimit?: number;
constructor(private readonly identityService: IdentityService, constructor(private identityService: IdentityService,
private readonly settingsService: SettingsService, private settingsService: SettingsService,
private readonly notificationService: NotificationService) { } private notificationService: NotificationService) { }
public ngOnInit() { public ngOnInit() {
this.users = []; this.users = [];
@ -37,7 +37,7 @@ export class UserManagementComponent implements OnInit {
} }
public welcomeEmail(user: IUser) { public welcomeEmail(user: IUser) {
if(!user.emailAddress) { if (!user.emailAddress) {
this.notificationService.error("The user needs an email address."); this.notificationService.error("The user needs an email address.");
return; return;
} }
@ -45,7 +45,7 @@ export class UserManagementComponent implements OnInit {
this.notificationService.error("Email Notifications are not setup, cannot send welcome email"); this.notificationService.error("Email Notifications are not setup, cannot send welcome email");
return; return;
} }
if(!this.emailSettings.notificationTemplates.some(x => { if (!this.emailSettings.notificationTemplates.some(x => {
return x.enabled && x.notificationType === 8; return x.enabled && x.notificationType === 8;
})) { })) {
this.notificationService.error("The Welcome Email template is not enabled in the Email Setings"); this.notificationService.error("The Welcome Email template is not enabled in the Email Setings");
@ -74,20 +74,20 @@ export class UserManagementComponent implements OnInit {
}); });
this.users.forEach(x => { this.users.forEach(x => {
if(!x.checked) { if (!x.checked) {
return; return;
} }
if(anyRoles) { if (anyRoles) {
x.claims = this.availableClaims; x.claims = this.availableClaims;
} }
if(this.bulkEpisodeLimit) { if (this.bulkEpisodeLimit) {
x.episodeRequestLimit = this.bulkEpisodeLimit; x.episodeRequestLimit = this.bulkEpisodeLimit;
} }
if(this.bulkMovieLimit) { if (this.bulkMovieLimit) {
x.movieRequestLimit = this.bulkMovieLimit; x.movieRequestLimit = this.bulkMovieLimit;
} }
this.identityService.updateUser(x).subscribe(y => { this.identityService.updateUser(x).subscribe(y => {
if(!y.successful) { if (!y.successful) {
this.notificationService.error(`Could not update user ${x.userName}. Reason ${y.errors[0]}`); this.notificationService.error(`Could not update user ${x.userName}. Reason ${y.errors[0]}`);
} }
}); });

View file

@ -11,7 +11,7 @@ import { IEmbySettings } from "../../interfaces";
}) })
export class EmbyComponent implements OnInit { export class EmbyComponent implements OnInit {
private embySettings: IEmbySettings; public embySettings: IEmbySettings;
constructor(private embyService: EmbyService, constructor(private embyService: EmbyService,
private router: Router, private router: Router,
@ -21,7 +21,7 @@ export class EmbyComponent implements OnInit {
public ngOnInit() { public ngOnInit() {
this.embySettings = { this.embySettings = {
servers: [], servers: [],
id:0, id: 0,
enable: true, enable: true,
}; };
this.embySettings.servers.push({ this.embySettings.servers.push({

View file

@ -50,9 +50,9 @@ export class PlexComponent implements OnInit {
} }
public oauth() { public oauth() {
this.plexTv.GetPin(this.clientId, "Ombi").subscribe(pin => { this.plexTv.GetPin(this.clientId, "Ombi").subscribe((pin: any) => {
this.plexService.oAuth({wizard: true, pin}).subscribe(x => { this.plexService.oAuth({ wizard: true, pin }).subscribe(x => {
if(x.url) { if (x.url) {
window.location.href = x.url; window.location.href = x.url;
} }
}); });

View file

@ -24,7 +24,7 @@ export class PlexOAuthComponent implements OnInit {
public ngOnInit(): void { public ngOnInit(): void {
this.plexOauth.oAuth(this.pinId).subscribe(x => { this.plexOauth.oAuth(this.pinId).subscribe(x => {
if(!x.accessToken) { if (!x.accessToken) {
return; return;
// RETURN // RETURN
} }
@ -50,5 +50,4 @@ export class PlexOAuthComponent implements OnInit {
}); });
} }
} }

View file

@ -1,2 +0,0 @@
import "rxjs/add/operator/catch";
import "rxjs/add/operator/map";

View file

@ -1,5 +1,6 @@
import * as Pace from "pace-progress"; // Main
import * as Pace from "pace-progress";
Pace.start(); Pace.start();
import "bootstrap/dist/js/bootstrap"; import "bootstrap/dist/js/bootstrap";
@ -11,8 +12,6 @@ import "./polyfills";
import "hammerjs"; import "hammerjs";
import "./imports";
import { enableProdMode } from "@angular/core"; import { enableProdMode } from "@angular/core";
import { platformBrowserDynamic } from "@angular/platform-browser-dynamic"; import { platformBrowserDynamic } from "@angular/platform-browser-dynamic";
import { AppModule } from "./app/app.module"; import { AppModule } from "./app/app.module";
@ -29,7 +28,9 @@ if (module.hot) {
oldRootElem.parentNode.insertBefore(newRootElem, oldRootElem); oldRootElem.parentNode.insertBefore(newRootElem, oldRootElem);
oldRootElem.parentNode.removeChild(oldRootElem); oldRootElem.parentNode.removeChild(oldRootElem);
} }
if (modulePromise) {
modulePromise.then(appModule => appModule.destroy()); modulePromise.then(appModule => appModule.destroy());
}
}); });
} else { } else {
enableProdMode(); enableProdMode();

View file

@ -9,6 +9,7 @@ declare var module: any;
if (module.hot) { if (module.hot) {
Error.stackTraceLimit = Infinity; Error.stackTraceLimit = Infinity;
// tslint:disable:no-var-requires
// tslint:disable-next-line
require("zone.js/dist/long-stack-trace-zone"); require("zone.js/dist/long-stack-trace-zone");
} }

View file

@ -24,3 +24,7 @@ $bg-colour-disabled: #252424;
background: $primary-colour !important; background: $primary-colour !important;
color: white; color: white;
}*/ }*/
.label {
margin: 3px;
}

View file

@ -883,11 +883,6 @@ textarea {
display: none; display: none;
} }
.ui-treetable-toggler.fa.fa-fw.ui-clickable.fa-caret-right,
.ui-treetable-toggler.fa.fa-fw.ui-clickable.fa-caret-down {
display: none;
}
.ui-state-highlight { .ui-state-highlight {
background: $primary-colour; background: $primary-colour;
} }
@ -900,15 +895,6 @@ textarea {
border: 1px solid $form-color-lighter; border: 1px solid $form-color-lighter;
} }
.ui-treetable tfoot td, .ui-treetable th {
text-align: left;
}
.ui-treetable tbody td {
white-space: inherit;
overflow: visible;
}
table a:not(.btn) { table a:not(.btn) {
text-decoration: none; text-decoration: none;
} }

View file

@ -153,18 +153,6 @@ namespace Ombi.Controllers
return await MovieRequestEngine.DenyMovieById(model.Id); return await MovieRequestEngine.DenyMovieById(model.Id);
} }
/// <summary>
/// Gets the tv requests.
/// </summary>
/// <param name="count">The count of items you want to return.</param>
/// <param name="position">The position.</param>
/// <returns></returns>
[HttpGet("tv/{count:int}/{position:int}/tree")]
public async Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> GetTvRequestsTree(int count, int position)
{
return await TvRequestEngine.GetRequestsTreeNode(count, position);
}
/// <summary> /// <summary>
/// Gets the total amount of TV requests. /// Gets the total amount of TV requests.
/// </summary> /// </summary>
@ -267,17 +255,6 @@ namespace Ombi.Controllers
return await TvRequestEngine.SearchTvRequest(searchTerm); return await TvRequestEngine.SearchTvRequest(searchTerm);
} }
/// <summary>
/// Searches for a specific tv request
/// </summary>
/// <param name="searchTerm">The search term.</param>
/// <returns></returns>
[HttpGet("tv/search/{searchTerm}/tree")]
public async Task<IEnumerable<TreeNode<TvRequests, List<ChildRequests>>>> SearchTvTree(string searchTerm)
{
return await TvRequestEngine.SearchTvRequestTree(searchTerm);
}
/// <summary> /// <summary>
/// Deletes the a specific tv request /// Deletes the a specific tv request
/// </summary> /// </summary>

View file

@ -1,5 +1,4 @@
using System; using System.Collections.Generic;
using System.Collections.Generic;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Authorization;
@ -127,31 +126,6 @@ namespace Ombi.Controllers
return await TvEngine.Search(searchTerm); return await TvEngine.Search(searchTerm);
} }
/// <summary>
/// Searches for a Tv Show.
/// </summary>
/// <param name="searchTerm">The search term.</param>
/// <remarks>We use TvMaze as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/{searchTerm}/tree")]
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> SearchTvTreeNode(string searchTerm)
{
return await TvEngine.SearchTreeNode(searchTerm);
}
/// <summary>
/// Gets extra show information.
/// </summary>
/// <param name="tvdbId">The TVDB identifier.</param>
/// <remarks>We use TvMaze as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/info/{tvdbId}/tree")]
public async Task<TreeNode<SearchTvShowViewModel>> GetShowInfoTreeNode(int tvdbId)
{
if (tvdbId == 0) return new TreeNode<SearchTvShowViewModel>();
return await TvEngine.GetShowInformationTreeNode(tvdbId);
}
/// <summary> /// <summary>
/// Gets extra show information. /// Gets extra show information.
/// </summary> /// </summary>
@ -164,17 +138,6 @@ namespace Ombi.Controllers
return await TvEngine.GetShowInformation(tvdbId); return await TvEngine.GetShowInformation(tvdbId);
} }
/// <summary>
/// Returns Popular Tv Shows
/// </summary>
/// <remarks>We use Trakt.tv as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/popular/tree")]
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> PopularTvTree()
{
return await TvEngine.PopularTree();
}
/// <summary> /// <summary>
/// Returns Popular Tv Shows /// Returns Popular Tv Shows
/// </summary> /// </summary>
@ -186,18 +149,6 @@ namespace Ombi.Controllers
return await TvEngine.Popular(); return await TvEngine.Popular();
} }
/// <summary>
/// Returns most Anticiplateds tv shows.
/// </summary>
/// <remarks>We use Trakt.tv as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/anticipated/tree")]
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> AnticipatedTvTree()
{
return await TvEngine.AnticipatedTree();
}
/// <summary> /// <summary>
/// Returns most Anticiplateds tv shows. /// Returns most Anticiplateds tv shows.
/// </summary> /// </summary>
@ -209,16 +160,6 @@ namespace Ombi.Controllers
return await TvEngine.Anticipated(); return await TvEngine.Anticipated();
} }
/// <summary>
/// Returns Most watched shows.
/// </summary>
/// <remarks>We use Trakt.tv as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/mostwatched/tree")]
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> MostWatchedTree()
{
return await TvEngine.MostWatchesTree();
}
/// <summary> /// <summary>
/// Returns Most watched shows. /// Returns Most watched shows.
@ -231,17 +172,6 @@ namespace Ombi.Controllers
return await TvEngine.MostWatches(); return await TvEngine.MostWatches();
} }
/// <summary>
/// Returns trending shows
/// </summary>
/// <remarks>We use Trakt.tv as the Provider</remarks>
/// <returns></returns>
[HttpGet("tv/trending/tree")]
public async Task<IEnumerable<TreeNode<SearchTvShowViewModel>>> TrendingTree()
{
return await TvEngine.TrendingTree();
}
/// <summary> /// <summary>
/// Returns trending shows /// Returns trending shows
/// </summary> /// </summary>

View file

@ -1,5 +1,4 @@
<Project Sdk="Microsoft.NET.Sdk.Web"> <Project ToolsVersion="15.0" Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup> <PropertyGroup>
<TargetFramework>netcoreapp2.1</TargetFramework> <TargetFramework>netcoreapp2.1</TargetFramework>
<RuntimeIdentifiers>win10-x64;win10-x86;osx-x64;ubuntu-x64;debian.8-x64;centos.7-x64;linux-x64;linux-arm;linux-arm64;</RuntimeIdentifiers> <RuntimeIdentifiers>win10-x64;win10-x86;osx-x64;ubuntu-x64;debian.8-x64;centos.7-x64;linux-x64;linux-arm;linux-arm64;</RuntimeIdentifiers>
@ -9,6 +8,7 @@
<FileVersion>$(SemVer)</FileVersion> <FileVersion>$(SemVer)</FileVersion>
<Version>$(FullVer)</Version> <Version>$(FullVer)</Version>
<PackageVersion></PackageVersion> <PackageVersion></PackageVersion>
<TypeScriptToolsVersion>2.8</TypeScriptToolsVersion>
</PropertyGroup> </PropertyGroup>
<PropertyGroup> <PropertyGroup>
<ServerGarbageCollection>false</ServerGarbageCollection> <ServerGarbageCollection>false</ServerGarbageCollection>
@ -24,10 +24,6 @@
<DefineConstants>TRACE;RELEASE;NETCOREAPP2_0</DefineConstants> <DefineConstants>TRACE;RELEASE;NETCOREAPP2_0</DefineConstants>
</PropertyGroup> </PropertyGroup>
<Target Name="NpmCommandsDebug" Condition="'$(Configuration)'=='Debug'" AfterTargets="Build">
<Exec Command="npm run-script vendor" />
</Target>
<ItemGroup> <ItemGroup>
<!-- Files not to show in IDE --> <!-- Files not to show in IDE -->
<Compile Remove="Logs\**" /> <Compile Remove="Logs\**" />
@ -46,7 +42,17 @@
<None Remove="Styles\**" /> <None Remove="Styles\**" />
<None Remove="wwwroot\dist\**" /> <None Remove="wwwroot\dist\**" />
</ItemGroup> </ItemGroup>
<Target Name="NpmCommandsDebug" Condition="'$(Configuration)'=='Debug'" AfterTargets="Build">
<Exec Command="npm run vendor" />
</Target>
<!-- Disabled as run by CI once for many different runtime builds
<Target Name="NpmCommandsRelease" Condition="'$(Configuration)'=='Release'" AfterTargets="Build">
<Exec Command="npm run publish" />
</Target>
-->
<Target Name="NpmCommandsClean" AfterTargets="Clean">
<Exec Command="npm run clean" />
</Target>
<Target Name="RunWebpack" AfterTargets="ComputeFilesToPublish"> <Target Name="RunWebpack" AfterTargets="ComputeFilesToPublish">
<ItemGroup> <ItemGroup>
<DistFiles Include="wwwroot\dist\**" /> <DistFiles Include="wwwroot\dist\**" />
@ -56,8 +62,6 @@
</ResolvedFileToPublish> </ResolvedFileToPublish>
</ItemGroup> </ItemGroup>
</Target> </Target>
<ItemGroup> <ItemGroup>
<PackageReference Include="AutoMapper" Version="6.1.1" /> <PackageReference Include="AutoMapper" Version="6.1.1" />
<PackageReference Include="CommandLineParser" Version="2.1.1-beta" /> <PackageReference Include="CommandLineParser" Version="2.1.1-beta" />
@ -94,18 +98,4 @@
<ProjectReference Include="..\Ombi.TheMovieDbApi\Ombi.Api.TheMovieDb.csproj" /> <ProjectReference Include="..\Ombi.TheMovieDbApi\Ombi.Api.TheMovieDb.csproj" />
<ProjectReference Include="..\Ombi.Updater\Ombi.Updater.csproj" /> <ProjectReference Include="..\Ombi.Updater\Ombi.Updater.csproj" />
</ItemGroup> </ItemGroup>
<ItemGroup>
<None Include="wwwroot\loading.css" />
<None Include="wwwroot\translations\*.json" />
</ItemGroup>
<ItemGroup>
<None Remove="ClientApp\app\animations\fadeinout.ts" />
</ItemGroup>
<ItemGroup>
<TypeScriptCompile Include="ClientApp\app\animations\fadeinout.ts" />
</ItemGroup>
</Project> </Project>

View file

@ -156,7 +156,12 @@ namespace Ombi
app.UseWebpackDevMiddleware(new WebpackDevMiddlewareOptions app.UseWebpackDevMiddleware(new WebpackDevMiddlewareOptions
{ {
HotModuleReplacement = true, HotModuleReplacement = true,
ConfigFile = "webpack.dev.js" ConfigFile = "webpack.config.ts",
//EnvParam = new
//{
// aot = true // can't use AOT with HMR currently https://github.com/angular/angular-cli/issues/6347
//}
}); });
} }

View file

@ -71,8 +71,8 @@
<meta name="smartbanner:button-url-apple" content="https://itunes.apple.com/us/genre/ios/id36?mt=8"> <meta name="smartbanner:button-url-apple" content="https://itunes.apple.com/us/genre/ios/id36?mt=8">
<meta name="smartbanner:button-url-google" content="https://play.google.com/store"> <meta name="smartbanner:button-url-google" content="https://play.google.com/store">
<meta name="smartbanner:enabled-platforms" content="android,ios">e.com/store/apps/details?id=com.tidusjar.Ombi"> <meta name="smartbanner:enabled-platforms" content="android,ios">e.com/store/apps/details?id=com.tidusjar.Ombi">
<meta name="smartbanner:enabled-platforms" content="android,ios"> <meta name="smartbanner:enabled-platforms" content="android,ios">
<!-- End SmartBanner configuration -->*@ <!-- End SmartBanner configuration -->*@
<meta charset="utf-8" /> <meta charset="utf-8" />
<meta http-equiv="x-ua-compatible" content="ie=edge"> <meta http-equiv="x-ua-compatible" content="ie=edge">
<meta name="description" content="Ombi, media request tool"> <meta name="description" content="Ombi, media request tool">
@ -98,8 +98,10 @@
<link rel="stylesheet" href="~/loading.css" asp-append-version="true" /> <link rel="stylesheet" href="~/loading.css" asp-append-version="true" />
<link rel="stylesheet" href="~/dist/vendor.css" asp-append-version="true" /> <link rel="stylesheet" href="~/dist/vendor.css" asp-append-version="true" />
<environment include="Development">
<script src="~/dist/vendor.js" asp-append-version="true" defer></script> <script src="~/dist/vendor.js" asp-append-version="true" defer></script>
<script src="~/dist/main.js" asp-append-version="true" defer></script> </environment>
<script src="~/dist/app.js" asp-append-version="true" defer></script>
</head> </head>
<body> <body>
@{ @{

View file

@ -1,21 +1,25 @@
'use strict'; "use strict";
const gulp = require('gulp'); const gulp = require("gulp");
const run = require('gulp-run'); const run = require("gulp-run");
const runSequence = require('run-sequence'); const runSequence = require("run-sequence");
const del = require('del'); const del = require("del");
const path = require('path'); const path = require("path");
const fs = require('fs'); const fs = require("fs");
const outputDir = './wwwroot/dist'; const outputDir = "./wwwroot/dist";
global.aot = true;
function getEnvOptions() { function getEnvOptions() {
var options = []; const options = [];
if (global.prod) { if (global.prod) {
options.push('--env.prod'); options.push("--env.prod");
} }
if (global.analyse) { if (global.analyse) {
options.push('--env.analyse'); options.push("--env.analyse");
}
if (global.aot) {
options.push("--env.aot");
} }
if (options.length > 0) { if (options.length > 0) {
return " " + options.join(" "); return " " + options.join(" ");
@ -24,12 +28,12 @@ function getEnvOptions() {
} }
} }
function webpack(type) {
function webpack(vendor) { // 'webpack' instead of direct path can cause https://github.com/angular/angular-cli/issues/6417
return run(`webpack --config webpack.config${vendor ? '.vendor' : ''}.ts${getEnvOptions()}`).exec(); return run(`node node_modules\\webpack\\bin\\webpack.js --config webpack.config${type ? `.${type}` : ""}.ts${getEnvOptions()}`).exec();
} }
gulp.task('vendor', () => { gulp.task("vendor", () => {
let build = false; let build = false;
const vendorPath = path.join(outputDir, "vendor.js"); const vendorPath = path.join(outputDir, "vendor.js");
const vendorExists = fs.existsSync(vendorPath); const vendorExists = fs.existsSync(vendorPath);
@ -37,44 +41,47 @@ gulp.task('vendor', () => {
const vendorStat = fs.statSync(vendorPath); const vendorStat = fs.statSync(vendorPath);
const packageStat = fs.statSync("package.json"); const packageStat = fs.statSync("package.json");
const vendorConfigStat = fs.statSync("webpack.config.vendor.ts"); const vendorConfigStat = fs.statSync("webpack.config.vendor.ts");
const commonConfigStat = fs.statSync("webpack.config.common.ts");
if (packageStat.mtime > vendorStat.mtime) { if (packageStat.mtime > vendorStat.mtime) {
build = true; build = true;
} }
if (vendorConfigStat.mtime > vendorStat.mtime) { if (vendorConfigStat.mtime > vendorStat.mtime) {
build = true; build = true;
} }
if (commonConfigStat.mtime > vendorStat.mtime) {
build = true;
}
} else { } else {
build = true; build = true;
} }
if (build) { if (build) {
return webpack(true); return webpack("vendor");
} }
}); });
gulp.task("vendor_force", () => {
gulp.task('vendor_force', () => { return webpack("vendor");
return webpack(true);
}) })
gulp.task('main', () => { gulp.task("main", () => {
return webpack() return webpack()
}); });
gulp.task('prod_var', () => { gulp.task("prod_var", () => {
global.prod = true; global.prod = true;
}) })
gulp.task('analyse_var', () => { gulp.task("analyse_var", () => {
global.analyse = true; global.analyse = true;
}) })
gulp.task('clean', () => { gulp.task("clean", () => {
del.sync(outputDir, { force: true }); del.sync(outputDir, { force: true });
}); });
gulp.task("lint", () => run("npm run lint").exec());
gulp.task('lint', () => run("npm run lint").exec()); gulp.task("lint_fix", () => run("npm run lint -- --fix").exec());
gulp.task('build', callback => runSequence('vendor', 'main', callback)); gulp.task("build", callback => runSequence("vendor", "main", callback));
gulp.task('analyse', callback => runSequence('analyse_var', 'build')); gulp.task("analyse", callback => runSequence("analyse_var", "clean", "build", callback));
gulp.task('full', callback => runSequence('clean', 'build')); gulp.task("full", callback => runSequence("clean", "build", callback));
gulp.task('publish', callback => runSequence('prod_var', 'build')); gulp.task("publish", callback => runSequence("prod_var", "full", callback));

13399
src/Ombi/package-lock.json generated

File diff suppressed because it is too large Load diff

View file

@ -1,94 +1,93 @@
{ {
"name": "ombi", "name": "ombi",
"version": "2.6.0",
"private": true, "private": true,
"version": "1.0.0",
"scripts": { "scripts": {
"vendor": "gulp vendor", "vendor": "gulp vendor",
"main": "gulp main", "lint": "tslint -p .",
"publish": "gulp publish", "publish": "gulp publish",
"lint": "tslint ClientApp/**/*.ts", "restore": "dotnet restore && yarn install",
"restore": "dotnet restore && npm install" "clean": "gulp clean"
}, },
"dependencies": { "dependencies": {
"@angular/animations": "^5.2.5", "@angular/animations": "^6.0.7",
"@angular/cdk": "^5.2.1", "@angular/cdk": "^6.3.1",
"@angular/common": "^5.2.5", "@angular/common": "^6.0.7",
"@angular/compiler": "^5.2.5", "@angular/compiler": "^6.0.7",
"@angular/core": "^5.2.5", "@angular/compiler-cli": "^6.0.7",
"@angular/forms": "^5.2.5", "@angular/core": "^6.0.7",
"@angular/http": "^5.2.5", "@angular/forms": "^6.0.7",
"@angular/material": "^5.2.1", "@angular/http": "^6.0.7",
"@angular/platform-browser": "^5.2.5", "@angular/material": "^6.3.1",
"@angular/platform-browser-dynamic": "^5.2.5", "@angular/platform-browser": "^6.0.7",
"@angular/platform-server": "^5.2.5", "@angular/platform-browser-dynamic": "^6.0.7",
"@angular/router": "^5.2.5", "@angular/platform-server": "^6.0.7",
"@auth0/angular-jwt": "1.0.0-beta.9", "@angular/router": "^6.0.7",
"@ng-bootstrap/ng-bootstrap": "^1.0.0", "@auth0/angular-jwt": "^2.0.0",
"@ngu/carousel": "^1.4.8", "@ng-bootstrap/ng-bootstrap": "^2.2.0",
"@ngx-translate/core": "^8.0.0", "@ngtools/webpack": "^6.1.0-beta.2",
"@ngx-translate/http-loader": "^2.0.1", "@ngu/carousel": "^1.4.9-beta-2",
"@types/core-js": "^0.9.46", "@ngx-translate/core": "^10.0.2",
"@types/extract-text-webpack-plugin": "^3.0.1", "@ngx-translate/http-loader": "^3.0.1",
"@types/intro.js": "^2.4.3", "@types/core-js": "^2.5.0",
"@types/node": "^8.9.4", "@types/mini-css-extract-plugin": "^0.2.0",
"@types/webpack": "^3.8.7", "@types/node": "^10.5.1",
"angular-router-loader": "^0.8.2", "@types/webpack": "^4.4.4",
"angular2-moment": "^1.8.0", "@types/webpack-bundle-analyzer": "^2.9.2",
"@types/webpack-merge": "^4.1.3",
"angular-router-loader": "^0.8.5",
"angular2-template-loader": "^0.6.2", "angular2-template-loader": "^0.6.2",
"aspnet-webpack": "^2.0.3", "aspnet-webpack": "^3.0.0",
"awesome-typescript-loader": "^3.4.1", "awesome-typescript-loader": "^5.2.0",
"bootstrap": "3.3.7", "bootstrap": "3.3.7",
"bootswatch": "3.3.7", "bootswatch": "3.3.7",
"core-js": "^2.5.3", "copy-webpack-plugin": "^4.5.2",
"css": "^2.2.1", "core-js": "^2.5.7",
"css-loader": "^0.28.9", "css": "^2.2.3",
"css-loader": "^0.28.11",
"del": "^3.0.0", "del": "^3.0.0",
"event-source-polyfill": "^0.0.11", "event-source-polyfill": "^0.0.12",
"expose-loader": "^0.7.4", "expose-loader": "^0.7.5",
"extract-text-webpack-plugin": "^3.0.2", "file-loader": "^1.1.11",
"file-loader": "^1.1.6",
"font-awesome": "^4.7.0", "font-awesome": "^4.7.0",
"gulp": "^3.9.1", "gulp": "^3.9.1",
"gulp-run": "^1.7.1", "gulp-run": "^1.7.1",
"hammerjs": "^2.0.8", "hammerjs": "^2.0.8",
"html-loader": "0.5.1", "html-loader": "^0.5.5",
"jquery": "^3.3.1", "jquery": "^3.3.1",
"mini-css-extract-plugin": "^0.4.1",
"moment": "^2.22.2",
"ng2-cookies": "^1.0.12", "ng2-cookies": "^1.0.12",
"ngx-clipboard": "8.1.1", "ngx-clipboard": "^11.1.1",
"ngx-infinite-scroll": "^0.6.1", "ngx-infinite-scroll": "^6.0.1",
"ngx-moment": "^3.0.1",
"ngx-order-pipe": "^2.0.1", "ngx-order-pipe": "^2.0.1",
"node-sass": "^4.7.2", "node-sass": "^4.9.0",
"npm": "^5.6.0",
"pace-progress": "^1.0.2", "pace-progress": "^1.0.2",
"primeng": "5.0.2", "primeng": "^6.0.0",
"reflect-metadata": "0.1.10", "raw-loader": "^0.5.1",
"reflect-metadata": "^0.1.12",
"run-sequence": "^2.2.1", "run-sequence": "^2.2.1",
"rxjs": "5.5.2", "rxjs": "^6.2.1",
"sass-loader": "^6.0.6", "sass-loader": "^7.0.3",
"style-loader": "^0.19.1", "style-loader": "^0.21.0",
"to-string-loader": "^1.1.5", "to-string-loader": "^1.1.5",
"ts-node": "^3.3.0", "ts-node": "^7.0.0",
"tslint": "^5.9.1", "tslint": "^5.10.0",
"tslint-language-service": "^0.9.8", "tslint-language-service": "^0.9.9",
"typescript": "^2.7.1", "typescript": "2.7.2",
"uglify-es": "^3.3.10", "uglify-es": "^3.3.9",
"uglifyjs-webpack-plugin": "^1.1.8", "uglifyjs-webpack-plugin": "^1.2.7",
"url-loader": "^0.6.2", "url-loader": "^1.0.1",
"webpack": "^3.11.0", "webpack": "^4.14.0",
"webpack-bundle-analyzer": "^2.10.0", "webpack-bundle-analyzer": "^2.13.1",
"webpack-hot-middleware": "^2.21.0", "webpack-cli": "^3.0.8",
"zone.js": "^0.8.20" "webpack-dev-middleware": "^3.1.3",
"webpack-hot-middleware": "^2.22.2",
"webpack-merge": "^4.1.3",
"zone.js": "^0.8.26"
}, },
"devDependencies": { "resolutions": {
"@types/chai": "4.0.4", "@types/tapable": "1.0.0"
"@types/jasmine": "2.6.2",
"chai": "4.1.2",
"jasmine-core": "2.8.0",
"karma": "1.7.1",
"karma-chai": "0.1.0",
"karma-chrome-launcher": "2.2.0",
"karma-cli": "1.0.1",
"karma-jasmine": "1.1.0",
"karma-webpack": "2.0.5"
} }
} }

View file

@ -2,7 +2,7 @@
"compilerOptions": { "compilerOptions": {
"target": "es5", "target": "es5",
"lib": [ "lib": [
"es6", "es2017",
"dom" "dom"
], ],
"moduleResolution": "node", "moduleResolution": "node",
@ -27,11 +27,15 @@
} }
] ]
}, },
"exclude": [ "include": [
"bin", "ClientApp/**/*",
"node_modules" "typings/**/*",
"webpack.config.*"
], ],
"angularCompilerOptions": { "angularCompilerOptions": {
"preserveWhitespaces": false "preserveWhitespaces": false,
} "genDir": "./ClientApp/app/ngfactory",
"entryModule": "ClientApp/app/app.module#AppModule"
},
"compileOnSave": false
} }

View file

@ -8,7 +8,7 @@
], ],
"max-line-length": [ "max-line-length": [
true, true,
200 250
], ],
"arrow-parens": false, "arrow-parens": false,
"radix": false, "radix": false,
@ -16,22 +16,33 @@
"indent": [ "indent": [
false false
], ],
"whitespace": [
false
],
"no-unused-expression": [ "no-unused-expression": [
true, true,
"allow-new" "allow-new"
], ],
"no-trailing-whitespace": [
false
],
"max-classes-per-file": [ "max-classes-per-file": [
false false
], ],
"no-shadowed-variable": false, "no-shadowed-variable": false,
"comment-format": [ "comment-format": [
false false
],
"no-namespace": [
false
],
"no-internal-module": [
false
],
"whitespace": [
false,
"check-branch",
"check-decl",
"check-operator",
"check-separator",
"check-type"
],
"no-trailing-whitespace":[
false
] ]
} }
} }

4
src/Ombi/typings/globals.d.ts vendored Normal file
View file

@ -0,0 +1,4 @@
// Globals
declare module "pace-progress";
declare var __webpack_public_path__: any;

View file

@ -1,12 +0,0 @@
// Globals
declare var module: any;
declare var require: any;
declare var localStorage: any;
declare var introJs: any;
declare var __webpack_public_path__: any;
declare module "pace-progress";
declare module "webpack-bundle-analyzer";
declare module "uglifyjs-webpack-plugin";

View file

@ -1 +1 @@
/// <reference path="globals/globals.d.ts" /> /// <reference path="globals.d.ts" />

View file

@ -0,0 +1,83 @@
import { AngularCompilerPlugin } from "@ngtools/webpack";
import * as MiniCssExtractPlugin from "mini-css-extract-plugin";
import * as path from "path";
import { Configuration, ContextReplacementPlugin, ProvidePlugin } from "webpack";
import { BundleAnalyzerPlugin } from "webpack-bundle-analyzer";
export const outputDir = "./wwwroot/dist";
export function isProd(env: any) {
return env && env.prod as boolean;
}
export function isAOT(env: any) {
return env && env.aot as boolean;
}
export const WebpackCommonConfig = (env: any, type: string) => {
const prod = isProd(env);
const aot = isAOT(env);
const vendor = type === "vendor";
console.log(`${prod ? "Production" : "Dev"} ${type} build`);
console.log(`Output directory: ${outputDir}`);
console.log(`${aot ? "Using" : "Not using"} AOT compiler`);
const analyse = env && env.analyse as boolean;
if (analyse) { console.log("Analysing build"); }
const cssLoader = prod ? "css-loader?minimize" : "css-loader";
const bundleConfig: Configuration = {
mode: prod ? "production" : "development",
resolve: {
extensions: [".ts", ".js"],
alias: {
pace: "pace-progress",
},
},
output: {
path: path.resolve(outputDir),
filename: "[name].js",
chunkFilename: "[id].chunk.js",
publicPath: "/dist/",
},
module: {
rules: [
{ test: /\.ts$/, loader: aot ? "@ngtools/webpack" : ["awesome-typescript-loader?silent=true", "angular2-template-loader", "angular-router-loader"] },
{ test: /\.html$/, use: "html-loader?minimize=false" },
{ test: /\.css$/, use: [MiniCssExtractPlugin.loader, cssLoader] },
{ test: /\.scss$/, exclude: /ClientApp/, use: [MiniCssExtractPlugin.loader, cssLoader, "sass-loader"] },
{ test: /\.scss$/, include: /ClientApp(\\|\/)app/, use: ["to-string-loader", cssLoader, "sass-loader"] },
{ test: /\.scss$/, include: /ClientApp(\\|\/)styles/, use: ["style-loader", cssLoader, "sass-loader"] },
{ test: /\.(png|jpg|jpeg|gif|woff|woff2|eot|ttf|svg)(\?|$)/, use: "url-loader?limit=8192" },
{ test: /[\/\\]@angular[\/\\].+\.js$/, parser: { system: true } }, // ignore System.import warnings https://github.com/angular/angular/issues/21560
],
},
plugins: [
new MiniCssExtractPlugin({
filename: "[name].css",
}),
new ProvidePlugin({ $: "jquery", jQuery: "jquery", Hammer: "hammerjs/hammer" }), // Global identifiers
].concat(aot && !vendor ? [
new AngularCompilerPlugin({
mainPath: "./ClientApp/main.ts",
tsConfigPath: "./tsconfig.json",
skipCodeGeneration: false,
compilerOptions: {
noEmit: false,
},
}),
] : [
// AOT chunk splitting does not work while this is active but doesn't seem to be needed under AOT anyway https://github.com/angular/angular-cli/issues/4431
new ContextReplacementPlugin(/angular(\\|\/)core(\\|\/)/, path.join(__dirname, "./ClientApp")), // Workaround for https://github.com/angular/angular/issues/14898
]).concat(analyse ? [
new BundleAnalyzerPlugin({
analyzerMode: "static",
reportFilename: `${type}.html`,
openAnalyzer: false,
}),
] : []),
node: {
fs: "empty",
},
};
return bundleConfig;
};

View file

@ -1,59 +1,26 @@
import { CheckerPlugin } from "awesome-typescript-loader";
import * as path from "path"; import * as path from "path";
import * as UglifyJSPlugin from "uglifyjs-webpack-plugin"; import { Configuration, DllReferencePlugin } from "webpack";
import { BundleAnalyzerPlugin } from "webpack-bundle-analyzer"; import * as webpackMerge from "webpack-merge";
import * as webpack from "webpack"; import { isAOT, isProd, outputDir, WebpackCommonConfig } from "./webpack.config.common";
module.exports = (env: any) => { module.exports = (env: any) => {
// const prod = env && env.prod as boolean; const prod = isProd(env);
const prod = true; const aot = isAOT(env);
console.log(prod ? "Production" : "Dev" + " main build"); if (!prod && aot) { console.warn("Vendor dll bundle will not be used as AOT is enabled"); }
const analyse = env && env.analyse as boolean; const bundleConfig: Configuration = webpackMerge(WebpackCommonConfig(env, "main"), {
if (analyse) { console.log("Analysing build"); } entry: {
const cssLoader = prod ? "css-loader?-url&minimize" : "css-loader?-url"; app: "./ClientApp/main.ts",
const outputDir = "./wwwroot/dist"; },
const bundleConfig: webpack.Configuration = {
entry: { main: "./ClientApp/main.ts" },
stats: { modules: false },
context: __dirname,
resolve: { extensions: [".ts", ".js"] },
devtool: prod ? "source-map" : "eval-source-map", devtool: prod ? "source-map" : "eval-source-map",
output: { plugins: prod || aot ? [] : [
filename: "[name].js", // AOT chunk splitting does not work while this is active https://github.com/angular/angular-cli/issues/4565
chunkFilename: "[id].[chunkhash].js", new DllReferencePlugin({
publicPath: "/dist/",
path: path.join(__dirname, outputDir),
},
module: {
rules: [
{ test: /\.ts$/, include: /ClientApp/, use: ["awesome-typescript-loader?silent=true", "angular2-template-loader", "angular-router-loader"] },
{ test: /\.html$/, use: "html-loader?minimize=false" },
{ test: /\.css$/, use: ["to-string-loader", cssLoader] },
{ test: /\.scss$/, include: /ClientApp(\\|\/)app/, use: ["to-string-loader", cssLoader, "sass-loader"] },
{ test: /\.scss$/, include: /ClientApp(\\|\/)styles/, use: ["style-loader", cssLoader, "sass-loader"] },
{ test: /\.(png|jpg|jpeg|gif|svg)$/, use: "url-loader?limit=25000" },
],
},
plugins: [
new CheckerPlugin(),
new webpack.DllReferencePlugin({
context: __dirname, context: __dirname,
manifest: require(path.join(__dirname, outputDir, "vendor-manifest.json")), manifest: require(path.join(__dirname, outputDir, "vendor-manifest.json")),
}), }),
].concat(prod ? [ ],
// Plugins that apply in production builds only });
new UglifyJSPlugin({ sourceMap: true }),
] : [
// Plugins that apply in development builds only
]).concat(analyse ? [
new BundleAnalyzerPlugin({
analyzerMode: "static",
reportFilename: "main.html",
openAnalyzer: false,
}),
] : []),
};
return bundleConfig; return bundleConfig;
}; };

View file

@ -1,41 +1,25 @@
import * as ExtractTextPlugin from "extract-text-webpack-plugin";
import * as path from "path"; import * as path from "path";
import * as UglifyJSPlugin from "uglifyjs-webpack-plugin";
import * as webpack from "webpack"; import * as webpack from "webpack";
import { BundleAnalyzerPlugin } from "webpack-bundle-analyzer"; import * as webpackMerge from "webpack-merge";
import { isProd, outputDir, WebpackCommonConfig } from "./webpack.config.common";
module.exports = (env: any) => { module.exports = (env: any) => {
const extractCSS = new ExtractTextPlugin("vendor.css"); const prod = isProd(env);
const prod = env && env.prod as boolean; const bundleConfig = webpackMerge(WebpackCommonConfig(env, "vendor"), {
console.log(prod ? "Production" : "Dev" + " vendor build"); output: {
const analyse = env && env.analyse as boolean; library: "[name]_[hash]",
if (analyse) { console.log("Analysing build"); }
const outputDir = "./wwwroot/dist";
const bundleConfig = {
stats: { modules: false },
resolve: {
extensions: [".js"],
alias: {
pace: "pace-progress",
},
},
module: {
rules: [
{ test: /\.(png|woff|woff2|eot|ttf|svg|gif)(\?|$)/, use: "url-loader?limit=100000" },
{ test: /\.css(\?|$)/, use: extractCSS.extract({ use: prod ? "css-loader?minimize" : "css-loader" }) },
{ test: /\.scss(\?|$)/, use: extractCSS.extract({ use: [prod ? "css-loader?minimize" : "css-loader", "sass-loader"] }) },
],
}, },
entry: { entry: {
vendor: [ vendor: (<string[]>[ // add any vendor styles here e.g. bootstrap/dist/css/bootstrap.min.css
"pace-progress/themes/orange/pace-theme-flash.css", "pace-progress/themes/orange/pace-theme-flash.css",
"primeng/resources/primeng.min.css", "primeng/resources/primeng.min.css",
"@angular/material/prebuilt-themes/deeppurple-amber.css", "@angular/material/prebuilt-themes/deeppurple-amber.css",
"font-awesome/scss/font-awesome.scss", "font-awesome/scss/font-awesome.scss",
"bootswatch/superhero/bootstrap.min.css", "bootswatch/superhero/bootstrap.min.css",
]).concat(prod ? [] : [ // used to speed up dev launch time
"@angular/animations", "@angular/animations",
"@angular/common", "@angular/common",
"@angular/common/http",
"@angular/compiler", "@angular/compiler",
"@angular/core", "@angular/core",
"@angular/forms", "@angular/forms",
@ -67,41 +51,14 @@ module.exports = (env: any) => {
"@ngx-translate/core", "@ngx-translate/core",
"@ngx-translate/http-loader", "@ngx-translate/http-loader",
"ngx-order-pipe", "ngx-order-pipe",
//"smartbanner.js/dist/smartbanner.js", ]),
//"smartbanner.js/dist/smartbanner.css",
],
}, },
output: { plugins: prod ? [] : [
publicPath: "/dist/",
filename: "[name].js",
library: "[name]_[hash]",
path: path.join(__dirname, outputDir),
},
node: {
fs: "empty",
},
plugins: [
new webpack.ProvidePlugin({ $: "jquery", jQuery: "jquery", Hammer: "hammerjs/hammer" }), // Global identifiers
new webpack.ContextReplacementPlugin(/\@angular(\\|\/)core(\\|\/)esm5/, path.join(__dirname, "./client")), // Workaround for https://github.com/angular/angular/issues/20357
new webpack.ContextReplacementPlugin(/\@angular\b.*\b(bundles|linker)/, path.join(__dirname, "./ClientApp")), // Workaround for https://github.com/angular/angular/issues/11580
new webpack.ContextReplacementPlugin(/angular(\\|\/)core(\\|\/)@angular/, path.join(__dirname, "./ClientApp")), // Workaround for https://github.com/angular/angular/issues/14898
extractCSS,
new webpack.DllPlugin({ new webpack.DllPlugin({
path: path.join(__dirname, outputDir, "[name]-manifest.json"), path: path.join(__dirname, outputDir, "[name]-manifest.json"),
name: "[name]_[hash]", name: "[name]_[hash]",
}), }),
].concat(prod ? [ ],
// Plugins that apply in production builds only });
new UglifyJSPlugin(),
] : [
// Plugins that apply in development builds only
]).concat(analyse ? [
new BundleAnalyzerPlugin({
analyzerMode: "static",
reportFilename: "vendor.html",
openAnalyzer: false,
}),
] : []),
};
return bundleConfig; return bundleConfig;
}; };

View file

@ -1,4 +0,0 @@
// https://github.com/aspnet/JavaScriptServices/issues/1046
require('ts-node/register')
module.exports = require("./webpack.config.ts");

View file

@ -1,171 +1,171 @@
{ {
"Login": { "Login": {
"SignInButton": "Sign in", "SignInButton": "Zaloguj się",
"UsernamePlaceholder": "Username", "UsernamePlaceholder": "Nazwa użytkownika",
"PasswordPlaceholder": "Password", "PasswordPlaceholder": "Hasło",
"RememberMe": "Remember Me", "RememberMe": "Zapamiętaj mnie",
"ForgottenPassword": "Forgot your password?", "ForgottenPassword": "Nie pamiętasz hasła?",
"Errors": { "Errors": {
"IncorrectCredentials": "Incorrect username or password" "IncorrectCredentials": "Nieprawidłowa nazwa użytkownika lub hasło"
} }
}, },
"Common": { "Common": {
"ContinueButton": "Continue", "ContinueButton": "Dalej",
"Available": "Available", "Available": "Dostępny",
"NotAvailable": "Not Available", "NotAvailable": "Niedostępny",
"ProcessingRequest": "Processing Request", "ProcessingRequest": "Przetwarzanie zgłoszenia",
"PendingApproval": "Pending Approval", "PendingApproval": "Oczekujące na zatwierdzenie",
"RequestDenied": "Request Denied", "RequestDenied": "Zgłoszenie odrzucone",
"NotRequested": "Not Requested", "NotRequested": "Nie zgłoszone",
"Requested": "Requested", "Requested": "Zgłoszone",
"Request": "Request", "Request": "Zgłoszenie",
"Denied": "Denied", "Denied": "Odrzucone",
"Approve": "Approve", "Approve": "Zatwierdź",
"PartlyAvailable": "Partly Available", "PartlyAvailable": "Dostępne częściowo",
"Errors": { "Errors": {
"Validation": "Please check your entered values" "Validation": "Sprawdź wprowadzone dane"
} }
}, },
"PasswordReset": { "PasswordReset": {
"EmailAddressPlaceholder": "Email Address", "EmailAddressPlaceholder": "Adres e-mail",
"ResetPasswordButton": "Reset Password" "ResetPasswordButton": "Zresetuj hasło"
}, },
"LandingPage": { "LandingPage": {
"OnlineHeading": "Currently Online", "OnlineHeading": "Dostępny",
"OnlineParagraph": "The media server is currently online", "OnlineParagraph": "Serwer mediów jest dostępny",
"PartiallyOnlineHeading": "Partially Online", "PartiallyOnlineHeading": "Dostępny częściowo",
"PartiallyOnlineParagraph": "The media server is partially online.", "PartiallyOnlineParagraph": "Serwer mediów jest dostępny częściowo.",
"MultipleServersUnavailable": "There are {{serversUnavailable}} servers offline out of {{totalServers}}.", "MultipleServersUnavailable": "{{serversUnavailable}} serwerów spośród {{totalServers}} jest niedostępnych.",
"SingleServerUnavailable": "There is {{serversUnavailable}} server offline out of {{totalServers}}.", "SingleServerUnavailable": "Jeden serwer spośród {{totalServers}} jest niedostępny.",
"OfflineHeading": "Currently Offline", "OfflineHeading": "Niedostępny",
"OfflineParagraph": "The media server is currently offline.", "OfflineParagraph": "Serwer mediów jest niedostępny.",
"CheckPageForUpdates": "Check this page for continuous site updates." "CheckPageForUpdates": "Tutaj znajdziesz aktualizacje dotyczące tej strony."
}, },
"NavigationBar": { "NavigationBar": {
"Search": "Search", "Search": "Szukaj",
"Requests": "Requests", "Requests": "Zgłoszenia",
"UserManagement": "User Management", "UserManagement": "Zarządzanie użytkownikami",
"Issues": "Issues", "Issues": "Problemy",
"Donate": "Donate!", "Donate": "Wesprzyj!",
"DonateLibraryMaintainer": "Donate to Library Maintainer", "DonateLibraryMaintainer": "Wesprzyj właściciela biblioteki",
"DonateTooltip": "This is how I convince my wife to let me spend my spare time developing Ombi ;)", "DonateTooltip": "W ten sposób przekonuję moją żonę by spędzać mój wolny czas rozwijając Ombi ;)",
"UpdateAvailableTooltip": "Update Available!", "UpdateAvailableTooltip": "Dostępna aktualizacja!",
"Settings": "Settings", "Settings": "Ustawienia",
"Welcome": "Welcome {{username}}", "Welcome": "Witaj {{username}}",
"UpdateDetails": "Update Details", "UpdateDetails": "Podaj szczegóły",
"Logout": "Logout", "Logout": "Wyloguj",
"Language": { "Language": {
"English": "English", "English": "Angielski",
"French": "French", "French": "Francuski",
"Spanish": "Spanish", "Spanish": "Hiszpański",
"German": "German", "German": "Niemiecki",
"Italian": "Italian", "Italian": "Włoski",
"Danish": "Danish", "Danish": "Duński",
"Dutch": "Dutch", "Dutch": "Holenderski",
"Norwegian": "Norwegian", "Norwegian": "Norweski",
"BrazillianPortuguese": "Brazillian Portuguese", "BrazillianPortuguese": "Brazylijski portugalski",
"Polish": "Polish" "Polish": "Polski"
}, },
"OpenMobileApp": "Open Mobile App", "OpenMobileApp": "Otwórz aplikację mobilną",
"RecentlyAdded": "Recently Added" "RecentlyAdded": "Ostatnio dodane"
}, },
"Search": { "Search": {
"Title": "Search", "Title": "Szukaj",
"Paragraph": "Want to watch something that is not currently available? No problem, just search for it below and request it!", "Paragraph": "Chcesz obejrzeć coś, co nie jest obecnie dostępne? Żaden problem, po prostu wyszukaj poniżej i dodaj zgłoszenie!",
"MoviesTab": "Movies", "MoviesTab": "Filmy",
"TvTab": "TV Shows", "TvTab": "Seriale",
"Suggestions": "Suggestions", "Suggestions": "Sugestie",
"NoResults": "Sorry, we didn't find any results!", "NoResults": "Niestety nic nie znaleziono!",
"DigitalDate": "Digital Release: {{date}}", "DigitalDate": "Wydanie cyfrowe: {{date}}",
"TheatricalRelease": "Theatrical Release: {{date}}", "TheatricalRelease": "Premiera kinowa: {{date}}",
"ViewOnPlex": "View On Plex", "ViewOnPlex": "Obejrzyj w Plex",
"ViewOnEmby": "View On Emby", "ViewOnEmby": "Obejrzyj na Emby",
"RequestAdded": "Request for {{title}} has been added successfully", "RequestAdded": "Zgłoszenie dla {{title}} zostało dodane",
"Similar": "Similar", "Similar": "Podobne",
"Movies": { "Movies": {
"PopularMovies": "Popular Movies", "PopularMovies": "Popularne filmy",
"UpcomingMovies": "Upcoming Movies", "UpcomingMovies": "Wkrótce w kinach",
"TopRatedMovies": "Top Rated Movies", "TopRatedMovies": "Najwyżej oceniane filmy",
"NowPlayingMovies": "Now Playing Movies", "NowPlayingMovies": "W kinach",
"HomePage": "Home Page", "HomePage": "Strona główna",
"Trailer": "Trailer" "Trailer": "Zwiastun"
}, },
"TvShows": { "TvShows": {
"Popular": "Popular", "Popular": "Popularne",
"Trending": "Trending", "Trending": "Zyskujące popularność",
"MostWatched": "Most Watched", "MostWatched": "Najczęściej oglądane",
"MostAnticipated": "Most Anticipated", "MostAnticipated": "Najbardziej oczekiwane",
"Results": "Results", "Results": "Wyniki",
"AirDate": "Air Date:", "AirDate": "Data emisji:",
"AllSeasons": "All Seasons", "AllSeasons": "Wszystkie sezony",
"FirstSeason": "First Season", "FirstSeason": "Pierwszy sezon",
"LatestSeason": "Latest Season", "LatestSeason": "Najnowszy sezon",
"Select": "Select ...", "Select": "Wybierz…",
"SubmitRequest": "Submit Request", "SubmitRequest": "Dodaj zgłoszenie",
"Season": "Season: {{seasonNumber}}", "Season": "Sezon: {{seasonNumber}}",
"SelectAllInSeason": "Select All in Season {{seasonNumber}}" "SelectAllInSeason": "Wybierz wszystkie w sezonie {{seasonNumber}}"
} }
}, },
"Requests": { "Requests": {
"Title": "Requests", "Title": "Zgłoszenia",
"Paragraph": "Below you can see yours and all other requests, as well as their download and approval status.", "Paragraph": "Poniżej są twoje i wszystkie inne zgłoszenia, a także ich status akceptacji i pobierania.",
"MoviesTab": "Movies", "MoviesTab": "Filmy",
"TvTab": "TV Shows", "TvTab": "Seriale",
"RequestedBy": "Requested By:", "RequestedBy": "Zgłoszone przez:",
"Status": "Status:", "Status": "Stan:",
"RequestStatus": "Request status:", "RequestStatus": "Stan zgłoszenia:",
"Denied": " Denied:", "Denied": "Odrzucono:",
"TheatricalRelease": "Theatrical Release: {{date}}", "TheatricalRelease": "Premiera kinowa: {{date}}",
"TheatricalReleaseSort": "Theatrical Release", "TheatricalReleaseSort": "Premiera kinowa",
"DigitalRelease": "Digital Release: {{date}}", "DigitalRelease": "Wydanie cyfrowe: {{date}}",
"RequestDate": "Request Date:", "RequestDate": "Data zgłoszenia:",
"QualityOverride": "Quality Override:", "QualityOverride": "Wymuszenie jakości:",
"RootFolderOverride": "Root Folder Override:", "RootFolderOverride": "Wymuszenie folderu głównego:",
"ChangeRootFolder": "Root Folder", "ChangeRootFolder": "Folder główny",
"ChangeQualityProfile": "Quality Profile", "ChangeQualityProfile": "Wybór jakości",
"MarkUnavailable": "Mark Unavailable", "MarkUnavailable": "Oznacz jako niedostępne",
"MarkAvailable": "Mark Available", "MarkAvailable": "Oznacz jako dostępne",
"Remove": "Remove", "Remove": "Usuń",
"Deny": "Deny", "Deny": "Odrzuć",
"Season": "Season:", "Season": "Sezon:",
"GridTitle": "Title", "GridTitle": "Tytuł",
"AirDate": "AirDate", "AirDate": "Data emisji",
"GridStatus": "Status", "GridStatus": "Stan",
"ReportIssue": "Report Issue", "ReportIssue": "Zgłoś problem",
"Filter": "Filter", "Filter": "Filtr",
"Sort": "Sort", "Sort": "Sortowanie",
"SeasonNumberHeading": "Season: {seasonNumber}", "SeasonNumberHeading": "Sezon: {seasonNumber}",
"SortTitleAsc": "Title ▲", "SortTitleAsc": "Tytuł ▲",
"SortTitleDesc": "Title ▼", "SortTitleDesc": "Tytuł ▼",
"SortRequestDateAsc": "Request Date ▲", "SortRequestDateAsc": "Data zgłoszenia ▲",
"SortRequestDateDesc": "Request Date ▼", "SortRequestDateDesc": "Data zgłoszenia ▼",
"SortStatusAsc": "Status ▲", "SortStatusAsc": "Stan ▲",
"SortStatusDesc": "Status ▼" "SortStatusDesc": "Stan ▼"
}, },
"Issues": { "Issues": {
"Title": "Issues", "Title": "Problemy",
"PendingTitle": "Pending Issues", "PendingTitle": "Oczekujące problemy",
"InProgressTitle": "In Progress Issues", "InProgressTitle": "Problemy w trakcie",
"ResolvedTitle": "Resolved Issues", "ResolvedTitle": "Problemy rozwiązane",
"ColumnTitle": "Title", "ColumnTitle": "Tytuł",
"Category": "Category", "Category": "Kategoria",
"Status": "Status", "Status": "Stan",
"Details": "Details", "Details": "Szczegóły",
"Description": "Description", "Description": "Opis",
"NoComments": "No Comments!", "NoComments": "Brak komentarzy!",
"MarkInProgress": "Mark In Progress", "MarkInProgress": "Oznacz jako \"w trakcie\"",
"MarkResolved": "Mark Resolved", "MarkResolved": "Oznacz jako rozwiązane",
"SendMessageButton": "Send", "SendMessageButton": "Wyślij",
"Subject": "Subject", "Subject": "Temat",
"Comments": "Comments", "Comments": "Komentarze",
"WriteMessagePlaceholder": "Write your message here...", "WriteMessagePlaceholder": "Tutaj wpisz swoją wiadomość…",
"ReportedBy": "Reported By" "ReportedBy": "Zgłoszone przez"
}, },
"Filter": { "Filter": {
"ClearFilter": "Clear Filter", "ClearFilter": "Wyczyść fltr",
"FilterHeaderAvailability": "Availability", "FilterHeaderAvailability": "Dostępność",
"FilterHeaderRequestStatus": "Status", "FilterHeaderRequestStatus": "Stan",
"Approved": "Approved", "Approved": "Zatwierdzone",
"PendingApproval": "Pending Approval" "PendingApproval": "Oczekujące na zatwierdzenie"
} }
} }

View file

@ -1,171 +1,171 @@
{ {
"Login": { "Login": {
"SignInButton": "Sign in", "SignInButton": "Registrar",
"UsernamePlaceholder": "Username", "UsernamePlaceholder": "Nome de usuário",
"PasswordPlaceholder": "Password", "PasswordPlaceholder": "Senha",
"RememberMe": "Remember Me", "RememberMe": "Lembre-se de mim",
"ForgottenPassword": "Forgot your password?", "ForgottenPassword": "Esqueceu sua senha?",
"Errors": { "Errors": {
"IncorrectCredentials": "Incorrect username or password" "IncorrectCredentials": "Nome de usuário ou senha incorretos"
} }
}, },
"Common": { "Common": {
"ContinueButton": "Continue", "ContinueButton": "Continuar",
"Available": "Available", "Available": "Disponível",
"NotAvailable": "Not Available", "NotAvailable": "Inisponível",
"ProcessingRequest": "Processing Request", "ProcessingRequest": "Processando Solicitação",
"PendingApproval": "Pending Approval", "PendingApproval": "Aprovação Pendente",
"RequestDenied": "Request Denied", "RequestDenied": "Solicitação Negada",
"NotRequested": "Not Requested", "NotRequested": "Não Solicitado",
"Requested": "Requested", "Requested": "Solicitado",
"Request": "Request", "Request": "Solicitar",
"Denied": "Denied", "Denied": "Negado",
"Approve": "Approve", "Approve": "Aprovar",
"PartlyAvailable": "Partly Available", "PartlyAvailable": "Parcialmente Disponível",
"Errors": { "Errors": {
"Validation": "Please check your entered values" "Validation": "Por favor, verifique os dados inseridos"
} }
}, },
"PasswordReset": { "PasswordReset": {
"EmailAddressPlaceholder": "Email Address", "EmailAddressPlaceholder": "Endereço de e-mail",
"ResetPasswordButton": "Reset Password" "ResetPasswordButton": "Redefinir Senha"
}, },
"LandingPage": { "LandingPage": {
"OnlineHeading": "Currently Online", "OnlineHeading": "Online Agora",
"OnlineParagraph": "The media server is currently online", "OnlineParagraph": "O servidor de mídia está atualmente online",
"PartiallyOnlineHeading": "Partially Online", "PartiallyOnlineHeading": "Parcialmente Online",
"PartiallyOnlineParagraph": "The media server is partially online.", "PartiallyOnlineParagraph": "O servidor de mídia está parcialmente online.",
"MultipleServersUnavailable": "There are {{serversUnavailable}} servers offline out of {{totalServers}}.", "MultipleServersUnavailable": "Existem {{serversUnavailable}} servidores offline em um total de {{totalServers}}.",
"SingleServerUnavailable": "There is {{serversUnavailable}} server offline out of {{totalServers}}.", "SingleServerUnavailable": "Existe {{serversUnavailable}} servidor offline em um total de {{totalServers}}.",
"OfflineHeading": "Currently Offline", "OfflineHeading": "Offline Agora",
"OfflineParagraph": "The media server is currently offline.", "OfflineParagraph": "O servidor de mídia está atualmente offline.",
"CheckPageForUpdates": "Check this page for continuous site updates." "CheckPageForUpdates": "Verifique esta página para atualizações contínuas do site."
}, },
"NavigationBar": { "NavigationBar": {
"Search": "Search", "Search": "Pesquisar",
"Requests": "Requests", "Requests": "Solicitações",
"UserManagement": "User Management", "UserManagement": "Gerenciador de Usuário",
"Issues": "Issues", "Issues": "Problemas",
"Donate": "Donate!", "Donate": "Fazer uma doação!",
"DonateLibraryMaintainer": "Donate to Library Maintainer", "DonateLibraryMaintainer": "Doar para o Dono da Biblioteca",
"DonateTooltip": "This is how I convince my wife to let me spend my spare time developing Ombi ;)", "DonateTooltip": "É assim que eu convenço a minha mulher a deixar-me passar o meu tempo livre desenvolvendo Ombi;)",
"UpdateAvailableTooltip": "Update Available!", "UpdateAvailableTooltip": "Atualização Disponível!",
"Settings": "Settings", "Settings": "Configurações",
"Welcome": "Welcome {{username}}", "Welcome": "Bem-vindo, {{username}}",
"UpdateDetails": "Update Details", "UpdateDetails": "Detalhes da Atualização",
"Logout": "Logout", "Logout": "Sair",
"Language": { "Language": {
"English": "English", "English": "Inglês",
"French": "French", "French": "Francês",
"Spanish": "Spanish", "Spanish": "Espanhol",
"German": "German", "German": "Alemão",
"Italian": "Italian", "Italian": "Italiano",
"Danish": "Danish", "Danish": "Dinamarquês",
"Dutch": "Dutch", "Dutch": "Holandês",
"Norwegian": "Norwegian", "Norwegian": "Norueguês",
"BrazillianPortuguese": "Brazillian Portuguese", "BrazillianPortuguese": "Português (Brasil)",
"Polish": "Polish" "Polish": "Polonês"
}, },
"OpenMobileApp": "Open Mobile App", "OpenMobileApp": "Abrir aplicativo do celular",
"RecentlyAdded": "Recently Added" "RecentlyAdded": "Recentemente adicionado"
}, },
"Search": { "Search": {
"Title": "Search", "Title": "Pesquisar",
"Paragraph": "Want to watch something that is not currently available? No problem, just search for it below and request it!", "Paragraph": "Quer assistir a algo que não está disponível? Não há problema, basta procurar abaixo e solicitar!",
"MoviesTab": "Movies", "MoviesTab": "Filmes",
"TvTab": "TV Shows", "TvTab": "Séries",
"Suggestions": "Suggestions", "Suggestions": "Sugestões",
"NoResults": "Sorry, we didn't find any results!", "NoResults": "Desculpe, não encontramos nenhum resultado!",
"DigitalDate": "Digital Release: {{date}}", "DigitalDate": "Lançamento digital: {{date}}",
"TheatricalRelease": "Theatrical Release: {{date}}", "TheatricalRelease": "Lançamento nos Cinemas: {{date}}",
"ViewOnPlex": "View On Plex", "ViewOnPlex": "Assistir no Plex",
"ViewOnEmby": "View On Emby", "ViewOnEmby": "Assistir no Emby",
"RequestAdded": "Request for {{title}} has been added successfully", "RequestAdded": "Pedido de {{title}} foi adicionado com sucesso",
"Similar": "Similar", "Similar": "Semelhante",
"Movies": { "Movies": {
"PopularMovies": "Popular Movies", "PopularMovies": "Filmes populares",
"UpcomingMovies": "Upcoming Movies", "UpcomingMovies": "Próximos filmes",
"TopRatedMovies": "Top Rated Movies", "TopRatedMovies": "Filmes mais votados",
"NowPlayingMovies": "Now Playing Movies", "NowPlayingMovies": "Filmes em cartazes",
"HomePage": "Home Page", "HomePage": "Home Page",
"Trailer": "Trailer" "Trailer": "Trailer"
}, },
"TvShows": { "TvShows": {
"Popular": "Popular", "Popular": "Popular",
"Trending": "Trending", "Trending": "Mais populares",
"MostWatched": "Most Watched", "MostWatched": "Mais assistidos",
"MostAnticipated": "Most Anticipated", "MostAnticipated": "Mais esperados",
"Results": "Results", "Results": "Resultados",
"AirDate": "Air Date:", "AirDate": "Data de exibição:",
"AllSeasons": "All Seasons", "AllSeasons": "Todas as temporadas",
"FirstSeason": "First Season", "FirstSeason": "Primeira temporada",
"LatestSeason": "Latest Season", "LatestSeason": "Última temporada",
"Select": "Select ...", "Select": "Selecione...",
"SubmitRequest": "Submit Request", "SubmitRequest": "Enviar solicitação",
"Season": "Season: {{seasonNumber}}", "Season": "Temporada: {{seasonNumber}}",
"SelectAllInSeason": "Select All in Season {{seasonNumber}}" "SelectAllInSeason": "Selecione tudo na temporada {{seasonNumber}}"
} }
}, },
"Requests": { "Requests": {
"Title": "Requests", "Title": "Solicitações",
"Paragraph": "Below you can see yours and all other requests, as well as their download and approval status.", "Paragraph": "Abaixo, você pode ver o seu e todos os outros pedidos, bem como o seu download e status de aprovação.",
"MoviesTab": "Movies", "MoviesTab": "Filmes",
"TvTab": "TV Shows", "TvTab": "Séries",
"RequestedBy": "Requested By:", "RequestedBy": "Solicitado por:",
"Status": "Status:", "Status": "Status:",
"RequestStatus": "Request status:", "RequestStatus": "Status da solicitação:",
"Denied": " Denied:", "Denied": " Negados:",
"TheatricalRelease": "Theatrical Release: {{date}}", "TheatricalRelease": "Lançamento nos Cinemas: {{date}}",
"TheatricalReleaseSort": "Theatrical Release", "TheatricalReleaseSort": "Lançamento nos Cinemas",
"DigitalRelease": "Digital Release: {{date}}", "DigitalRelease": "Lançamento digital: {{date}}",
"RequestDate": "Request Date:", "RequestDate": "Data da Solicitação:",
"QualityOverride": "Quality Override:", "QualityOverride": "Substituição de qualidade:",
"RootFolderOverride": "Root Folder Override:", "RootFolderOverride": "Substituição da pasta raiz:",
"ChangeRootFolder": "Root Folder", "ChangeRootFolder": "Pasta raiz",
"ChangeQualityProfile": "Quality Profile", "ChangeQualityProfile": "Perfil de qualidade",
"MarkUnavailable": "Mark Unavailable", "MarkUnavailable": "Marcar como Indisponível",
"MarkAvailable": "Mark Available", "MarkAvailable": "Marcar como Disponível",
"Remove": "Remove", "Remove": "Remover",
"Deny": "Deny", "Deny": "Negar",
"Season": "Season:", "Season": "Temporada:",
"GridTitle": "Title", "GridTitle": "Título",
"AirDate": "AirDate", "AirDate": "Data de exibição",
"GridStatus": "Status", "GridStatus": "Status",
"ReportIssue": "Report Issue", "ReportIssue": "Relatar Problema",
"Filter": "Filter", "Filter": "Filtro",
"Sort": "Sort", "Sort": "Ordenar por",
"SeasonNumberHeading": "Season: {seasonNumber}", "SeasonNumberHeading": "Temporada: {seasonNumber}",
"SortTitleAsc": "Title ▲", "SortTitleAsc": "Título ▲",
"SortTitleDesc": "Title ▼", "SortTitleDesc": "Título ▼",
"SortRequestDateAsc": "Request Date ▲", "SortRequestDateAsc": "Data da Solicitação",
"SortRequestDateDesc": "Request Date ▼", "SortRequestDateDesc": "Data da Solicitação",
"SortStatusAsc": "Status ▲", "SortStatusAsc": "Status ▲",
"SortStatusDesc": "Status ▼" "SortStatusDesc": "Status ▼"
}, },
"Issues": { "Issues": {
"Title": "Issues", "Title": "Problemas",
"PendingTitle": "Pending Issues", "PendingTitle": "Problemas pendentes",
"InProgressTitle": "In Progress Issues", "InProgressTitle": "Resolvendo Problemas",
"ResolvedTitle": "Resolved Issues", "ResolvedTitle": "Problemas Resolvidos",
"ColumnTitle": "Title", "ColumnTitle": "Título",
"Category": "Category", "Category": "Categoria",
"Status": "Status", "Status": "Status",
"Details": "Details", "Details": "Detalhes",
"Description": "Description", "Description": "Descrição",
"NoComments": "No Comments!", "NoComments": "Sem Comentários!",
"MarkInProgress": "Mark In Progress", "MarkInProgress": "Marcar como em andamento",
"MarkResolved": "Mark Resolved", "MarkResolved": "Marcar como resolvido",
"SendMessageButton": "Send", "SendMessageButton": "Enviar",
"Subject": "Subject", "Subject": "Assunto",
"Comments": "Comments", "Comments": "Comentários",
"WriteMessagePlaceholder": "Write your message here...", "WriteMessagePlaceholder": "Escreva sua mensagem aqui...",
"ReportedBy": "Reported By" "ReportedBy": "Reportado por"
}, },
"Filter": { "Filter": {
"ClearFilter": "Clear Filter", "ClearFilter": "Limpar Filtro",
"FilterHeaderAvailability": "Availability", "FilterHeaderAvailability": "Disponibilidade",
"FilterHeaderRequestStatus": "Status", "FilterHeaderRequestStatus": "Status",
"Approved": "Approved", "Approved": "Aprovado",
"PendingApproval": "Pending Approval" "PendingApproval": "Aprovação Pendente"
} }
} }

File diff suppressed because it is too large Load diff