mirror of
https://github.com/Readarr/Readarr.git
synced 2026-03-09 15:00:14 -04:00
Compare commits
57 Commits
v0.4.12.27
...
develop
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0b79d3000d | ||
|
|
0f3e716044 | ||
|
|
f53c4dc017 | ||
|
|
f48eac7e36 | ||
|
|
7cc02f95af | ||
|
|
1f92bf6679 | ||
|
|
a8d4aa6770 | ||
|
|
7661b5bc87 | ||
|
|
200ef600cd | ||
|
|
ad6228983b | ||
|
|
582ec9f7ce | ||
|
|
525e855038 | ||
|
|
7a629ed044 | ||
|
|
7f501322dd | ||
|
|
18bca0b228 | ||
|
|
9ddac60b47 | ||
|
|
bd8bc0b35b | ||
|
|
ae623f4481 | ||
|
|
e67d133bb6 | ||
|
|
772ea95ce4 | ||
|
|
5459a7bb7e | ||
|
|
614f98f9b4 | ||
|
|
55763dae43 | ||
|
|
a362dab503 | ||
|
|
dba9fbf254 | ||
|
|
59a7605385 | ||
|
|
f819e582cf | ||
|
|
0972d41bf8 | ||
|
|
05d2335bfe | ||
|
|
5b4f54a959 | ||
|
|
bb5ad605fd | ||
|
|
52c3a95e63 | ||
|
|
52c5460537 | ||
|
|
280cec3d0e | ||
|
|
f10c2c01d8 | ||
|
|
2b6a328dac | ||
|
|
4078525f67 | ||
|
|
214e4270ac | ||
|
|
5396dd3e8e | ||
|
|
d5d4996c40 | ||
|
|
8b2223a9c4 | ||
|
|
2a3e7f8dae | ||
|
|
6406ed6289 | ||
|
|
d5b0831b0f | ||
|
|
8d72d5dbab | ||
|
|
74d1ab84e2 | ||
|
|
7341d20c51 | ||
|
|
5abf4f2992 | ||
|
|
cc90050c77 | ||
|
|
7ba8f8baee | ||
|
|
70aec175ef | ||
|
|
080dd301f3 | ||
|
|
bab45481db | ||
|
|
3e1e03e0ce | ||
|
|
bb599d6dc9 | ||
|
|
c94685842f | ||
|
|
5966f6da51 |
20
README.md
20
README.md
@@ -1,3 +1,23 @@
|
||||
# Announcement: Retirement of Readarr
|
||||
|
||||
We would like to announce that the [Readarr project](<https://github.com/Readarr/Readarr>) has been retired. This difficult decision was made due to a combination of factors: the project's metadata has become unusable, we no longer have the time to remake or repair it, and the community effort to transition to using Open Library as the source has stalled without much progress.
|
||||
|
||||
Third-party metadata mirrors exist, but as we're not involved with them at all, we cannot provide support for them. Use of them is entirely at your own risk. The most popular mirror appears to be [rreading-glasses](<https://github.com/blampe/rreading-glasses>).
|
||||
|
||||
Without anyone to take over Readarr development, we expect it to wither away, so we still encourage you to seek alternatives to Readarr.
|
||||
|
||||
## Key Points:
|
||||
- **Effective Immediately**: The retirement takes effect immediately. Please stay tuned for any possible further communications.
|
||||
- **Support Window**: We will provide support during a brief transition period to help with troubleshooting non metadata related issues.
|
||||
- **Alternative Solutions**: Users are encouraged to explore and adopt any other possible solutions as alternatives to Readarr.
|
||||
- **Opportunities for Revival**: We are open to someone taking over and revitalizing the project. If you are interested, please get in touch.
|
||||
- **Gratitude**: We extend our deepest gratitude to all the contributors and community members who supported Readarr over the years.
|
||||
|
||||
Thank you for being part of the Readarr journey. For any inquiries or assistance during this transition, please contact our team.
|
||||
|
||||
Sincerely,
|
||||
The Servarr Team
|
||||
|
||||
# Readarr
|
||||
|
||||
[](https://dev.azure.com/Readarr/Readarr/_build/latest?definitionId=1&branchName=develop)
|
||||
|
||||
@@ -9,7 +9,7 @@ variables:
|
||||
testsFolder: './_tests'
|
||||
yarnCacheFolder: $(Pipeline.Workspace)/.yarn
|
||||
nugetCacheFolder: $(Pipeline.Workspace)/.nuget/packages
|
||||
majorVersion: '0.4.12'
|
||||
majorVersion: '0.4.19'
|
||||
minorVersion: $[counter('minorVersion', 1)]
|
||||
readarrVersion: '$(majorVersion).$(minorVersion)'
|
||||
buildName: '$(Build.SourceBranchName).$(readarrVersion)'
|
||||
@@ -19,7 +19,7 @@ variables:
|
||||
nodeVersion: '20.X'
|
||||
innoVersion: '6.2.0'
|
||||
windowsImage: 'windows-2022'
|
||||
linuxImage: 'ubuntu-20.04'
|
||||
linuxImage: 'ubuntu-22.04'
|
||||
macImage: 'macOS-13'
|
||||
|
||||
trigger:
|
||||
|
||||
@@ -165,7 +165,8 @@ function HistoryDetails(props) {
|
||||
|
||||
if (eventType === 'downloadFailed') {
|
||||
const {
|
||||
message
|
||||
message,
|
||||
indexer
|
||||
} = data;
|
||||
|
||||
return (
|
||||
@@ -177,11 +178,21 @@ function HistoryDetails(props) {
|
||||
/>
|
||||
|
||||
{
|
||||
!!message &&
|
||||
indexer ?
|
||||
<DescriptionListItem
|
||||
title={translate('Indexer')}
|
||||
data={indexer}
|
||||
/> :
|
||||
null
|
||||
}
|
||||
|
||||
{
|
||||
message ?
|
||||
<DescriptionListItem
|
||||
title={translate('Message')}
|
||||
data={message}
|
||||
/>
|
||||
/> :
|
||||
null
|
||||
}
|
||||
</DescriptionList>
|
||||
);
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
import ModelBase from 'App/ModelBase';
|
||||
|
||||
export type AuthorStatus = 'continuing' | 'ended';
|
||||
|
||||
interface Author extends ModelBase {
|
||||
added: string;
|
||||
genres: string[];
|
||||
@@ -10,6 +12,7 @@ interface Author extends ModelBase {
|
||||
metadataProfileId: number;
|
||||
rootFolderPath: string;
|
||||
sortName: string;
|
||||
status: AuthorStatus;
|
||||
tags: number[];
|
||||
authorName: string;
|
||||
isSaving?: boolean;
|
||||
|
||||
21
frontend/src/Author/AuthorStatus.ts
Normal file
21
frontend/src/Author/AuthorStatus.ts
Normal file
@@ -0,0 +1,21 @@
|
||||
import { AuthorStatus } from 'Author/Author';
|
||||
import { icons } from 'Helpers/Props';
|
||||
import translate from 'Utilities/String/translate';
|
||||
|
||||
export function getAuthorStatusDetails(status: AuthorStatus) {
|
||||
let statusDetails = {
|
||||
icon: icons.AUTHOR_CONTINUING,
|
||||
title: translate('StatusEndedContinuing'),
|
||||
message: translate('ContinuingMoreBooksAreExpected'),
|
||||
};
|
||||
|
||||
if (status === 'ended') {
|
||||
statusDetails = {
|
||||
icon: icons.AUTHOR_ENDED,
|
||||
title: translate('StatusEndedEnded'),
|
||||
message: translate('ContinuingNoAdditionalBooksAreExpected'),
|
||||
};
|
||||
}
|
||||
|
||||
return statusDetails;
|
||||
}
|
||||
@@ -7,6 +7,7 @@ import AuthorHistoryTable from 'Author/History/AuthorHistoryTable';
|
||||
import MonitoringOptionsModal from 'Author/MonitoringOptions/MonitoringOptionsModal';
|
||||
import BookEditorFooter from 'Book/Editor/BookEditorFooter';
|
||||
import BookFileEditorTable from 'BookFile/Editor/BookFileEditorTable';
|
||||
import Alert from 'Components/Alert';
|
||||
import IconButton from 'Components/Link/IconButton';
|
||||
import Link from 'Components/Link/Link';
|
||||
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
||||
@@ -17,7 +18,7 @@ import PageToolbarButton from 'Components/Page/Toolbar/PageToolbarButton';
|
||||
import PageToolbarSection from 'Components/Page/Toolbar/PageToolbarSection';
|
||||
import PageToolbarSeparator from 'Components/Page/Toolbar/PageToolbarSeparator';
|
||||
import SwipeHeaderConnector from 'Components/Swipe/SwipeHeaderConnector';
|
||||
import { align, icons } from 'Helpers/Props';
|
||||
import { align, icons, kinds } from 'Helpers/Props';
|
||||
import InteractiveSearchFilterMenuConnector from 'InteractiveSearch/InteractiveSearchFilterMenuConnector';
|
||||
import InteractiveSearchTable from 'InteractiveSearch/InteractiveSearchTable';
|
||||
import OrganizePreviewModalConnector from 'Organize/OrganizePreviewModalConnector';
|
||||
@@ -412,22 +413,25 @@ class AuthorDetails extends Component {
|
||||
|
||||
<div className={styles.contentContainer}>
|
||||
{
|
||||
!isPopulated && !booksError && !bookFilesError &&
|
||||
<LoadingIndicator />
|
||||
!isPopulated && !booksError && !bookFilesError ?
|
||||
<LoadingIndicator /> :
|
||||
null
|
||||
}
|
||||
|
||||
{
|
||||
!isFetching && booksError &&
|
||||
<div>
|
||||
!isFetching && booksError ?
|
||||
<Alert kind={kinds.DANGER}>
|
||||
{translate('LoadingBooksFailed')}
|
||||
</div>
|
||||
</Alert> :
|
||||
null
|
||||
}
|
||||
|
||||
{
|
||||
!isFetching && bookFilesError &&
|
||||
<div>
|
||||
!isFetching && bookFilesError ?
|
||||
<Alert kind={kinds.DANGER}>
|
||||
{translate('LoadingBookFilesFailed')}
|
||||
</div>
|
||||
</Alert> :
|
||||
null
|
||||
}
|
||||
|
||||
{
|
||||
|
||||
@@ -2,6 +2,7 @@ import PropTypes from 'prop-types';
|
||||
import React, { Component } from 'react';
|
||||
import TextTruncate from 'react-text-truncate';
|
||||
import AuthorPoster from 'Author/AuthorPoster';
|
||||
import { getAuthorStatusDetails } from 'Author/AuthorStatus';
|
||||
import HeartRating from 'Components/HeartRating';
|
||||
import Icon from 'Components/Icon';
|
||||
import Label from 'Components/Label';
|
||||
@@ -11,7 +12,7 @@ import MonitorToggleButton from 'Components/MonitorToggleButton';
|
||||
import Popover from 'Components/Tooltip/Popover';
|
||||
import Tooltip from 'Components/Tooltip/Tooltip';
|
||||
import { icons, kinds, sizes, tooltipPositions } from 'Helpers/Props';
|
||||
import QualityProfileNameConnector from 'Settings/Profiles/Quality/QualityProfileNameConnector';
|
||||
import QualityProfileName from 'Settings/Profiles/Quality/QualityProfileName';
|
||||
import fonts from 'Styles/Variables/fonts';
|
||||
import formatBytes from 'Utilities/Number/formatBytes';
|
||||
import stripHtml from 'Utilities/String/stripHtml';
|
||||
@@ -87,11 +88,11 @@ class AuthorDetailsHeader extends Component {
|
||||
titleWidth
|
||||
} = this.state;
|
||||
|
||||
const statusDetails = getAuthorStatusDetails(status);
|
||||
|
||||
const fanartUrl = getFanartUrl(images);
|
||||
const marqueeWidth = titleWidth - (isSmallScreen ? 85 : 160);
|
||||
|
||||
const continuing = status === 'continuing';
|
||||
|
||||
let bookFilesCountMessage = translate('BookFilesCountMessage');
|
||||
|
||||
if (bookFileCount === 1) {
|
||||
@@ -213,7 +214,7 @@ class AuthorDetailsHeader extends Component {
|
||||
|
||||
<span className={styles.qualityProfileName}>
|
||||
{
|
||||
<QualityProfileNameConnector
|
||||
<QualityProfileName
|
||||
qualityProfileId={qualityProfileId}
|
||||
/>
|
||||
}
|
||||
@@ -236,16 +237,16 @@ class AuthorDetailsHeader extends Component {
|
||||
|
||||
<Label
|
||||
className={styles.detailsLabel}
|
||||
title={continuing ? translate('ContinuingMoreBooksAreExpected') : translate('ContinuingNoAdditionalBooksAreExpected')}
|
||||
title={statusDetails.message}
|
||||
size={sizes.LARGE}
|
||||
>
|
||||
<Icon
|
||||
name={continuing ? icons.AUTHOR_CONTINUING : icons.AUTHOR_ENDED}
|
||||
name={statusDetails.icon}
|
||||
size={17}
|
||||
/>
|
||||
|
||||
<span className={styles.qualityProfileName}>
|
||||
{continuing ? 'Continuing' : 'Deceased'}
|
||||
{statusDetails.title}
|
||||
</span>
|
||||
</Label>
|
||||
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import PropTypes from 'prop-types';
|
||||
import React from 'react';
|
||||
import { getAuthorStatusDetails } from 'Author/AuthorStatus';
|
||||
import Icon from 'Components/Icon';
|
||||
import VirtualTableRowCell from 'Components/Table/Cells/TableRowCell';
|
||||
import { icons } from 'Helpers/Props';
|
||||
@@ -15,6 +16,8 @@ function AuthorStatusCell(props) {
|
||||
...otherProps
|
||||
} = props;
|
||||
|
||||
const statusDetails = getAuthorStatusDetails(status);
|
||||
|
||||
return (
|
||||
<Component
|
||||
className={className}
|
||||
@@ -28,8 +31,8 @@ function AuthorStatusCell(props) {
|
||||
|
||||
<Icon
|
||||
className={styles.statusIcon}
|
||||
name={status === 'ended' ? icons.AUTHOR_ENDED : icons.AUTHOR_CONTINUING}
|
||||
title={status === 'ended' ? translate('StatusEndedDeceased') : translate('StatusEndedContinuing')}
|
||||
name={statusDetails.icon}
|
||||
title={`${statusDetails.title}: ${statusDetails.message}`}
|
||||
/>
|
||||
</Component>
|
||||
);
|
||||
|
||||
@@ -2,6 +2,7 @@ import PropTypes from 'prop-types';
|
||||
import React from 'react';
|
||||
import Button from 'Components/Link/Button';
|
||||
import { kinds } from 'Helpers/Props';
|
||||
import translate from 'Utilities/String/translate';
|
||||
import styles from './NoAuthor.css';
|
||||
|
||||
function NoAuthor(props) {
|
||||
@@ -31,7 +32,7 @@ function NoAuthor(props) {
|
||||
to="/settings/mediamanagement"
|
||||
kind={kinds.PRIMARY}
|
||||
>
|
||||
Add Root Folder
|
||||
{translate('AddRootFolder')}
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
@@ -40,7 +41,7 @@ function NoAuthor(props) {
|
||||
to="/add/search"
|
||||
kind={kinds.PRIMARY}
|
||||
>
|
||||
Add New Author
|
||||
{translate('AddNewAuthor')}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,12 +1,11 @@
|
||||
import PropTypes from 'prop-types';
|
||||
import React, { Component } from 'react';
|
||||
import AuthorNameLink from 'Author/AuthorNameLink';
|
||||
import { getAuthorStatusDetails } from 'Author/AuthorStatus';
|
||||
import Icon from 'Components/Icon';
|
||||
import MonitorToggleButton from 'Components/MonitorToggleButton';
|
||||
import VirtualTableRowCell from 'Components/Table/Cells/VirtualTableRowCell';
|
||||
import VirtualTableSelectCell from 'Components/Table/Cells/VirtualTableSelectCell';
|
||||
import { icons } from 'Helpers/Props';
|
||||
import translate from 'Utilities/String/translate';
|
||||
import BookshelfBook from './BookshelfBook';
|
||||
import styles from './BookshelfRow.css';
|
||||
|
||||
@@ -30,6 +29,8 @@ class BookshelfRow extends Component {
|
||||
onBookMonitoredPress
|
||||
} = this.props;
|
||||
|
||||
const statusDetails = getAuthorStatusDetails(status);
|
||||
|
||||
return (
|
||||
<>
|
||||
<VirtualTableSelectCell
|
||||
@@ -52,8 +53,8 @@ class BookshelfRow extends Component {
|
||||
<VirtualTableRowCell className={styles.status}>
|
||||
<Icon
|
||||
className={styles.statusIcon}
|
||||
name={status === 'ended' ? icons.AUTHOR_ENDED : icons.AUTHOR_CONTINUING}
|
||||
title={status === 'ended' ? translate('StatusEndedEnded') : translate('StatusEndedContinuing')}
|
||||
name={statusDetails.icon}
|
||||
title={statusDetails.title}
|
||||
/>
|
||||
</VirtualTableRowCell>
|
||||
|
||||
|
||||
@@ -20,6 +20,8 @@ import HintedSelectInputSelectedValue from './HintedSelectInputSelectedValue';
|
||||
import TextInput from './TextInput';
|
||||
import styles from './EnhancedSelectInput.css';
|
||||
|
||||
const MINIMUM_DISTANCE_FROM_EDGE = 10;
|
||||
|
||||
function isArrowKey(keyCode) {
|
||||
return keyCode === keyCodes.UP_ARROW || keyCode === keyCodes.DOWN_ARROW;
|
||||
}
|
||||
@@ -137,18 +139,9 @@ class EnhancedSelectInput extends Component {
|
||||
// Listeners
|
||||
|
||||
onComputeMaxHeight = (data) => {
|
||||
const {
|
||||
top,
|
||||
bottom
|
||||
} = data.offsets.reference;
|
||||
|
||||
const windowHeight = window.innerHeight;
|
||||
|
||||
if ((/^botton/).test(data.placement)) {
|
||||
data.styles.maxHeight = windowHeight - bottom;
|
||||
} else {
|
||||
data.styles.maxHeight = top;
|
||||
}
|
||||
data.styles.maxHeight = windowHeight - MINIMUM_DISTANCE_FROM_EDGE;
|
||||
|
||||
return data;
|
||||
};
|
||||
@@ -457,6 +450,10 @@ class EnhancedSelectInput extends Component {
|
||||
order: 851,
|
||||
enabled: true,
|
||||
fn: this.onComputeMaxHeight
|
||||
},
|
||||
preventOverflow: {
|
||||
enabled: true,
|
||||
boundariesElement: 'viewport'
|
||||
}
|
||||
}}
|
||||
>
|
||||
|
||||
@@ -28,8 +28,7 @@ function createMapStateToProps() {
|
||||
if (includeNoChange) {
|
||||
values.unshift({
|
||||
key: 'noChange',
|
||||
value: '',
|
||||
name: translate('NoChange'),
|
||||
value: translate('NoChange'),
|
||||
isDisabled: includeNoChangeDisabled,
|
||||
isMissing: false
|
||||
});
|
||||
@@ -39,7 +38,6 @@ function createMapStateToProps() {
|
||||
values.push({
|
||||
key: '',
|
||||
value: '',
|
||||
name: '',
|
||||
isDisabled: true,
|
||||
isHidden: true
|
||||
});
|
||||
@@ -56,8 +54,7 @@ function createMapStateToProps() {
|
||||
|
||||
values.push({
|
||||
key: ADD_NEW_KEY,
|
||||
value: '',
|
||||
name: 'Add a new path'
|
||||
value: 'Add a new path'
|
||||
});
|
||||
|
||||
return {
|
||||
@@ -105,6 +102,27 @@ class RootFolderSelectInputConnector extends Component {
|
||||
}
|
||||
}
|
||||
|
||||
componentDidUpdate(prevProps) {
|
||||
const {
|
||||
name,
|
||||
value,
|
||||
values,
|
||||
onChange
|
||||
} = this.props;
|
||||
|
||||
if (prevProps.values === values) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!value && values.length && values.some((v) => !!v.key && v.key !== ADD_NEW_KEY)) {
|
||||
const defaultValue = values[0];
|
||||
|
||||
if (defaultValue.key !== ADD_NEW_KEY) {
|
||||
onChange({ name, value: defaultValue.key });
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// Render
|
||||
|
||||
|
||||
@@ -13,6 +13,15 @@
|
||||
}
|
||||
}
|
||||
|
||||
.value {
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.authorFolder {
|
||||
flex: 0 0 auto;
|
||||
color: var(--disabledColor);
|
||||
}
|
||||
|
||||
.freeSpace {
|
||||
margin-left: 15px;
|
||||
color: var(--darkGray);
|
||||
|
||||
@@ -1,10 +1,12 @@
|
||||
// This file is automatically generated.
|
||||
// Please do not change this file!
|
||||
interface CssExports {
|
||||
'authorFolder': string;
|
||||
'freeSpace': string;
|
||||
'isMissing': string;
|
||||
'isMobile': string;
|
||||
'optionText': string;
|
||||
'value': string;
|
||||
}
|
||||
export const cssExports: CssExports;
|
||||
export default cssExports;
|
||||
|
||||
@@ -7,18 +7,24 @@ import styles from './RootFolderSelectInputOption.css';
|
||||
|
||||
function RootFolderSelectInputOption(props) {
|
||||
const {
|
||||
id,
|
||||
value,
|
||||
name,
|
||||
freeSpace,
|
||||
authorFolder,
|
||||
isMissing,
|
||||
isMobile,
|
||||
isWindows,
|
||||
...otherProps
|
||||
} = props;
|
||||
|
||||
const text = value === '' ? name : `${name} [${value}]`;
|
||||
const slashCharacter = isWindows ? '\\' : '/';
|
||||
|
||||
const text = name === '' ? value : `[${name}] ${value}`;
|
||||
|
||||
return (
|
||||
<EnhancedSelectInputOption
|
||||
id={id}
|
||||
isMobile={isMobile}
|
||||
{...otherProps}
|
||||
>
|
||||
@@ -27,7 +33,18 @@ function RootFolderSelectInputOption(props) {
|
||||
isMobile && styles.isMobile
|
||||
)}
|
||||
>
|
||||
<div>{text}</div>
|
||||
<div className={styles.value}>
|
||||
{text}
|
||||
|
||||
{
|
||||
authorFolder && id !== 'addNew' ?
|
||||
<div className={styles.authorFolder}>
|
||||
{slashCharacter}
|
||||
{authorFolder}
|
||||
</div> :
|
||||
null
|
||||
}
|
||||
</div>
|
||||
|
||||
{
|
||||
freeSpace == null ?
|
||||
@@ -50,11 +67,18 @@ function RootFolderSelectInputOption(props) {
|
||||
}
|
||||
|
||||
RootFolderSelectInputOption.propTypes = {
|
||||
id: PropTypes.string.isRequired,
|
||||
name: PropTypes.string.isRequired,
|
||||
value: PropTypes.string.isRequired,
|
||||
freeSpace: PropTypes.number,
|
||||
authorFolder: PropTypes.string,
|
||||
isMissing: PropTypes.bool,
|
||||
isMobile: PropTypes.bool.isRequired
|
||||
isMobile: PropTypes.bool.isRequired,
|
||||
isWindows: PropTypes.bool
|
||||
};
|
||||
|
||||
RootFolderSelectInputOption.defaultProps = {
|
||||
name: ''
|
||||
};
|
||||
|
||||
export default RootFolderSelectInputOption;
|
||||
|
||||
@@ -7,12 +7,22 @@
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.path {
|
||||
.pathContainer {
|
||||
@add-mixin truncate;
|
||||
|
||||
display: flex;
|
||||
flex: 1 0 0;
|
||||
}
|
||||
|
||||
.path {
|
||||
flex: 0 1 auto;
|
||||
}
|
||||
|
||||
.authorFolder {
|
||||
@add-mixin truncate;
|
||||
flex: 0 1 auto;
|
||||
color: var(--disabledColor);
|
||||
}
|
||||
|
||||
.freeSpace {
|
||||
flex: 0 0 auto;
|
||||
margin-left: 15px;
|
||||
|
||||
@@ -1,8 +1,10 @@
|
||||
// This file is automatically generated.
|
||||
// Please do not change this file!
|
||||
interface CssExports {
|
||||
'authorFolder': string;
|
||||
'freeSpace': string;
|
||||
'path': string;
|
||||
'pathContainer': string;
|
||||
'selectedValue': string;
|
||||
}
|
||||
export const cssExports: CssExports;
|
||||
|
||||
@@ -9,19 +9,34 @@ function RootFolderSelectInputSelectedValue(props) {
|
||||
name,
|
||||
value,
|
||||
freeSpace,
|
||||
authorFolder,
|
||||
includeFreeSpace,
|
||||
isWindows,
|
||||
...otherProps
|
||||
} = props;
|
||||
|
||||
const text = value === '' ? name : `${name} [${value}]`;
|
||||
const slashCharacter = isWindows ? '\\' : '/';
|
||||
|
||||
const text = name === '' ? value : `[${name}] ${value}`;
|
||||
|
||||
return (
|
||||
<EnhancedSelectInputSelectedValue
|
||||
className={styles.selectedValue}
|
||||
{...otherProps}
|
||||
>
|
||||
<div className={styles.path}>
|
||||
{text}
|
||||
<div className={styles.pathContainer}>
|
||||
<div className={styles.path}>
|
||||
{text}
|
||||
</div>
|
||||
|
||||
{
|
||||
authorFolder ?
|
||||
<div className={styles.authorFolder}>
|
||||
{slashCharacter}
|
||||
{authorFolder}
|
||||
</div> :
|
||||
null
|
||||
}
|
||||
</div>
|
||||
|
||||
{
|
||||
@@ -38,10 +53,13 @@ RootFolderSelectInputSelectedValue.propTypes = {
|
||||
name: PropTypes.string,
|
||||
value: PropTypes.string,
|
||||
freeSpace: PropTypes.number,
|
||||
authorFolder: PropTypes.string,
|
||||
isWindows: PropTypes.bool,
|
||||
includeFreeSpace: PropTypes.bool.isRequired
|
||||
};
|
||||
|
||||
RootFolderSelectInputSelectedValue.defaultProps = {
|
||||
name: '',
|
||||
includeFreeSpace: true
|
||||
};
|
||||
|
||||
|
||||
@@ -52,6 +52,7 @@ class SelectInput extends Component {
|
||||
const {
|
||||
key,
|
||||
value: optionValue,
|
||||
isDisabled: optionIsDisabled = false,
|
||||
...otherOptionProps
|
||||
} = option;
|
||||
|
||||
@@ -59,6 +60,7 @@ class SelectInput extends Component {
|
||||
<option
|
||||
key={key}
|
||||
value={key}
|
||||
disabled={optionIsDisabled}
|
||||
{...otherOptionProps}
|
||||
>
|
||||
{typeof optionValue === 'function' ? optionValue() : optionValue}
|
||||
|
||||
@@ -83,13 +83,6 @@
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.modal.small,
|
||||
.modal.medium {
|
||||
width: 90%;
|
||||
}
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
.modalContainer {
|
||||
position: fixed;
|
||||
}
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
line-height: 1.52857143;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.cell {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.cell {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
@@ -10,7 +10,7 @@
|
||||
border-collapse: collapse;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.tableContainer {
|
||||
min-width: 100%;
|
||||
width: fit-content;
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
margin-left: 10px;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.headerCell {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
@@ -60,7 +60,7 @@
|
||||
height: 25px;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.pager {
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
margin-left: 10px;
|
||||
}
|
||||
|
||||
@media only screen and (max-width: $breakpointSmall) {
|
||||
@media only screen and (max-width: $breakpointMedium) {
|
||||
.headerCell {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
@@ -35,11 +35,12 @@
|
||||
.message {
|
||||
margin-top: 30px;
|
||||
text-align: center;
|
||||
font-weight: 300;
|
||||
font-size: $largeFontSize;
|
||||
}
|
||||
|
||||
.helpText {
|
||||
margin-bottom: 10px;
|
||||
font-weight: 300;
|
||||
font-size: 24px;
|
||||
}
|
||||
|
||||
|
||||
@@ -82,7 +82,8 @@ class AddNewItem extends Component {
|
||||
render() {
|
||||
const {
|
||||
error,
|
||||
items
|
||||
items,
|
||||
hasExistingAuthors
|
||||
} = this.props;
|
||||
|
||||
const term = this.state.term;
|
||||
@@ -186,7 +187,8 @@ class AddNewItem extends Component {
|
||||
}
|
||||
|
||||
{
|
||||
!term &&
|
||||
term ?
|
||||
null :
|
||||
<div className={styles.message}>
|
||||
<div className={styles.helpText}>
|
||||
{translate('ItsEasyToAddANewAuthorOrBookJustStartTypingTheNameOfTheItemYouWantToAdd')}
|
||||
@@ -199,6 +201,24 @@ class AddNewItem extends Component {
|
||||
</div>
|
||||
}
|
||||
|
||||
{
|
||||
!term && !hasExistingAuthors ?
|
||||
<div className={styles.message}>
|
||||
<div className={styles.noAuthorsText}>
|
||||
You haven't added any authors yet, do you want to add an existing library location (Root Folder) and update?
|
||||
</div>
|
||||
<div>
|
||||
<Button
|
||||
to="/settings/mediamanagement"
|
||||
kind={kinds.PRIMARY}
|
||||
>
|
||||
{translate('AddRootFolder')}
|
||||
</Button>
|
||||
</div>
|
||||
</div> :
|
||||
null
|
||||
}
|
||||
|
||||
<div />
|
||||
</PageContentBody>
|
||||
</PageContent>
|
||||
@@ -213,6 +233,7 @@ AddNewItem.propTypes = {
|
||||
isAdding: PropTypes.bool.isRequired,
|
||||
addError: PropTypes.object,
|
||||
items: PropTypes.arrayOf(PropTypes.object).isRequired,
|
||||
hasExistingAuthors: PropTypes.bool.isRequired,
|
||||
onSearchChange: PropTypes.func.isRequired,
|
||||
onClearSearch: PropTypes.func.isRequired
|
||||
};
|
||||
|
||||
@@ -10,13 +10,15 @@ import AddNewItem from './AddNewItem';
|
||||
function createMapStateToProps() {
|
||||
return createSelector(
|
||||
(state) => state.search,
|
||||
(state) => state.authors.items.length,
|
||||
(state) => state.router.location,
|
||||
(search, location) => {
|
||||
(search, existingAuthorsCount, location) => {
|
||||
const { params } = parseUrl(location.search);
|
||||
|
||||
return {
|
||||
...search,
|
||||
term: params.term,
|
||||
...search
|
||||
hasExistingAuthors: existingAuthorsCount > 0
|
||||
};
|
||||
}
|
||||
);
|
||||
|
||||
@@ -9,6 +9,7 @@ import ModalContent from 'Components/Modal/ModalContent';
|
||||
import ModalFooter from 'Components/Modal/ModalFooter';
|
||||
import ModalHeader from 'Components/Modal/ModalHeader';
|
||||
import { kinds } from 'Helpers/Props';
|
||||
import translate from 'Utilities/String/translate';
|
||||
import AddAuthorOptionsForm from '../Common/AddAuthorOptionsForm.js';
|
||||
import styles from './AddNewAuthorModalContent.css';
|
||||
|
||||
@@ -54,7 +55,7 @@ class AddNewAuthorModalContent extends Component {
|
||||
return (
|
||||
<ModalContent onModalClose={onModalClose}>
|
||||
<ModalHeader>
|
||||
Add new Author
|
||||
{translate('AddNewAuthor')}
|
||||
</ModalHeader>
|
||||
|
||||
<ModalBody>
|
||||
@@ -133,7 +134,7 @@ class AddNewAuthorModalContent extends Component {
|
||||
|
||||
AddNewAuthorModalContent.propTypes = {
|
||||
authorName: PropTypes.string.isRequired,
|
||||
disambiguation: PropTypes.string.isRequired,
|
||||
disambiguation: PropTypes.string,
|
||||
overview: PropTypes.string,
|
||||
images: PropTypes.arrayOf(PropTypes.object).isRequired,
|
||||
isAdding: PropTypes.bool.isRequired,
|
||||
|
||||
@@ -4,6 +4,7 @@ import { connect } from 'react-redux';
|
||||
import { createSelector } from 'reselect';
|
||||
import { addAuthor, setAuthorAddDefault } from 'Store/Actions/searchActions';
|
||||
import createDimensionsSelector from 'Store/Selectors/createDimensionsSelector';
|
||||
import createSystemStatusSelector from 'Store/Selectors/createSystemStatusSelector';
|
||||
import selectSettings from 'Store/Selectors/selectSettings';
|
||||
import AddNewAuthorModalContent from './AddNewAuthorModalContent';
|
||||
|
||||
@@ -12,7 +13,8 @@ function createMapStateToProps() {
|
||||
(state) => state.search,
|
||||
(state) => state.settings.metadataProfiles,
|
||||
createDimensionsSelector(),
|
||||
(searchState, metadataProfiles, dimensions) => {
|
||||
createSystemStatusSelector(),
|
||||
(searchState, metadataProfiles, dimensions, systemStatus) => {
|
||||
const {
|
||||
isAdding,
|
||||
addError,
|
||||
@@ -32,6 +34,7 @@ function createMapStateToProps() {
|
||||
isSmallScreen: dimensions.isSmallScreen,
|
||||
validationErrors,
|
||||
validationWarnings,
|
||||
isWindows: systemStatus.isWindows,
|
||||
...settings
|
||||
};
|
||||
}
|
||||
|
||||
@@ -78,6 +78,7 @@ class AddNewAuthorSearchResult extends Component {
|
||||
status,
|
||||
overview,
|
||||
ratings,
|
||||
folder,
|
||||
images,
|
||||
isExistingAuthor,
|
||||
isSmallScreen
|
||||
@@ -205,6 +206,7 @@ class AddNewAuthorSearchResult extends Component {
|
||||
disambiguation={disambiguation}
|
||||
year={year}
|
||||
overview={overview}
|
||||
folder={folder}
|
||||
images={images}
|
||||
onModalClose={this.onAddAuthorModalClose}
|
||||
/>
|
||||
@@ -222,6 +224,7 @@ AddNewAuthorSearchResult.propTypes = {
|
||||
status: PropTypes.string.isRequired,
|
||||
overview: PropTypes.string,
|
||||
ratings: PropTypes.object.isRequired,
|
||||
folder: PropTypes.string.isRequired,
|
||||
images: PropTypes.arrayOf(PropTypes.object).isRequired,
|
||||
isExistingAuthor: PropTypes.bool.isRequired,
|
||||
isSmallScreen: PropTypes.bool.isRequired
|
||||
|
||||
@@ -10,6 +10,7 @@ import ModalFooter from 'Components/Modal/ModalFooter';
|
||||
import ModalHeader from 'Components/Modal/ModalHeader';
|
||||
import { kinds } from 'Helpers/Props';
|
||||
import stripHtml from 'Utilities/String/stripHtml';
|
||||
import translate from 'Utilities/String/translate';
|
||||
import AddAuthorOptionsForm from '../Common/AddAuthorOptionsForm.js';
|
||||
import styles from './AddNewBookModalContent.css';
|
||||
|
||||
@@ -58,7 +59,7 @@ class AddNewBookModalContent extends Component {
|
||||
return (
|
||||
<ModalContent onModalClose={onModalClose}>
|
||||
<ModalHeader>
|
||||
Add new Book
|
||||
{translate('AddNewBook')}
|
||||
</ModalHeader>
|
||||
|
||||
<ModalBody>
|
||||
|
||||
@@ -4,6 +4,7 @@ import { connect } from 'react-redux';
|
||||
import { createSelector } from 'reselect';
|
||||
import { addBook, setBookAddDefault } from 'Store/Actions/searchActions';
|
||||
import createDimensionsSelector from 'Store/Selectors/createDimensionsSelector';
|
||||
import createSystemStatusSelector from 'Store/Selectors/createSystemStatusSelector';
|
||||
import selectSettings from 'Store/Selectors/selectSettings';
|
||||
import AddNewBookModalContent from './AddNewBookModalContent';
|
||||
|
||||
@@ -13,7 +14,8 @@ function createMapStateToProps() {
|
||||
(state) => state.search,
|
||||
(state) => state.settings.metadataProfiles,
|
||||
createDimensionsSelector(),
|
||||
(isExistingAuthor, searchState, metadataProfiles, dimensions) => {
|
||||
createSystemStatusSelector(),
|
||||
(isExistingAuthor, searchState, metadataProfiles, dimensions, systemStatus) => {
|
||||
const {
|
||||
isAdding,
|
||||
addError,
|
||||
@@ -33,6 +35,7 @@ function createMapStateToProps() {
|
||||
isSmallScreen: dimensions.isSmallScreen,
|
||||
validationErrors,
|
||||
validationWarnings,
|
||||
isWindows: systemStatus.isWindows,
|
||||
...settings
|
||||
};
|
||||
}
|
||||
|
||||
@@ -203,6 +203,7 @@ class AddNewBookSearchResult extends Component {
|
||||
disambiguation={disambiguation}
|
||||
authorName={author.authorName}
|
||||
overview={overview}
|
||||
folder={author.folder}
|
||||
images={images}
|
||||
onModalClose={this.onAddBookModalClose}
|
||||
/>
|
||||
|
||||
@@ -39,7 +39,9 @@ class AddAuthorOptionsForm extends Component {
|
||||
includeNoneMetadataProfile,
|
||||
includeSpecificBookMonitor,
|
||||
showMetadataProfile,
|
||||
folder,
|
||||
tags,
|
||||
isWindows,
|
||||
onInputChange,
|
||||
...otherProps
|
||||
} = this.props;
|
||||
@@ -54,6 +56,15 @@ class AddAuthorOptionsForm extends Component {
|
||||
<FormInputGroup
|
||||
type={inputTypes.ROOT_FOLDER_SELECT}
|
||||
name="rootFolderPath"
|
||||
valueOptions={{
|
||||
authorFolder: folder,
|
||||
isWindows
|
||||
}}
|
||||
selectedValueOptions={{
|
||||
authorFolder: folder,
|
||||
isWindows
|
||||
}}
|
||||
helpText={translate('AddNewAuthorRootFolderHelpText', { folder })}
|
||||
onChange={onInputChange}
|
||||
{...rootFolderPath}
|
||||
/>
|
||||
@@ -179,8 +190,14 @@ AddAuthorOptionsForm.propTypes = {
|
||||
showMetadataProfile: PropTypes.bool.isRequired,
|
||||
includeNoneMetadataProfile: PropTypes.bool.isRequired,
|
||||
includeSpecificBookMonitor: PropTypes.bool.isRequired,
|
||||
folder: PropTypes.string.isRequired,
|
||||
tags: PropTypes.object.isRequired,
|
||||
isWindows: PropTypes.bool.isRequired,
|
||||
onInputChange: PropTypes.func.isRequired
|
||||
};
|
||||
|
||||
AddAuthorOptionsForm.defaultProps = {
|
||||
includeSpecificBookMonitor: false
|
||||
};
|
||||
|
||||
export default AddAuthorOptionsForm;
|
||||
|
||||
@@ -10,11 +10,11 @@ import ModalBody from 'Components/Modal/ModalBody';
|
||||
import ModalContent from 'Components/Modal/ModalContent';
|
||||
import ModalFooter from 'Components/Modal/ModalFooter';
|
||||
import ModalHeader from 'Components/Modal/ModalHeader';
|
||||
import Column from 'Components/Table/Column';
|
||||
import Table from 'Components/Table/Table';
|
||||
import TableBody from 'Components/Table/TableBody';
|
||||
import useSelectState from 'Helpers/Hooks/useSelectState';
|
||||
import { kinds } from 'Helpers/Props';
|
||||
import SortDirection from 'Helpers/Props/SortDirection';
|
||||
import {
|
||||
bulkDeleteDownloadClients,
|
||||
bulkEditDownloadClients,
|
||||
@@ -35,7 +35,7 @@ type OnSelectedChangeCallback = React.ComponentProps<
|
||||
typeof ManageDownloadClientsModalRow
|
||||
>['onSelectedChange'];
|
||||
|
||||
const COLUMNS = [
|
||||
const COLUMNS: Column[] = [
|
||||
{
|
||||
name: 'name',
|
||||
label: () => translate('Name'),
|
||||
@@ -82,8 +82,6 @@ const COLUMNS = [
|
||||
|
||||
interface ManageDownloadClientsModalContentProps {
|
||||
onModalClose(): void;
|
||||
sortKey?: string;
|
||||
sortDirection?: SortDirection;
|
||||
}
|
||||
|
||||
function ManageDownloadClientsModalContent(
|
||||
|
||||
@@ -10,11 +10,11 @@ import ModalBody from 'Components/Modal/ModalBody';
|
||||
import ModalContent from 'Components/Modal/ModalContent';
|
||||
import ModalFooter from 'Components/Modal/ModalFooter';
|
||||
import ModalHeader from 'Components/Modal/ModalHeader';
|
||||
import Column from 'Components/Table/Column';
|
||||
import Table from 'Components/Table/Table';
|
||||
import TableBody from 'Components/Table/TableBody';
|
||||
import useSelectState from 'Helpers/Hooks/useSelectState';
|
||||
import { kinds } from 'Helpers/Props';
|
||||
import SortDirection from 'Helpers/Props/SortDirection';
|
||||
import {
|
||||
bulkDeleteIndexers,
|
||||
bulkEditIndexers,
|
||||
@@ -35,7 +35,7 @@ type OnSelectedChangeCallback = React.ComponentProps<
|
||||
typeof ManageIndexersModalRow
|
||||
>['onSelectedChange'];
|
||||
|
||||
const COLUMNS = [
|
||||
const COLUMNS: Column[] = [
|
||||
{
|
||||
name: 'name',
|
||||
label: () => translate('Name'),
|
||||
@@ -82,8 +82,6 @@ const COLUMNS = [
|
||||
|
||||
interface ManageIndexersModalContentProps {
|
||||
onModalClose(): void;
|
||||
sortKey?: string;
|
||||
sortDirection?: SortDirection;
|
||||
}
|
||||
|
||||
function ManageIndexersModalContent(props: ManageIndexersModalContentProps) {
|
||||
|
||||
@@ -0,0 +1,18 @@
|
||||
import React from 'react';
|
||||
import { useSelector } from 'react-redux';
|
||||
import { createQualityProfileSelectorForHook } from 'Store/Selectors/createQualityProfileSelector';
|
||||
import translate from 'Utilities/String/translate';
|
||||
|
||||
interface QualityProfileNameProps {
|
||||
qualityProfileId: number;
|
||||
}
|
||||
|
||||
function QualityProfileName({ qualityProfileId }: QualityProfileNameProps) {
|
||||
const qualityProfile = useSelector(
|
||||
createQualityProfileSelectorForHook(qualityProfileId)
|
||||
);
|
||||
|
||||
return <span>{qualityProfile?.name ?? translate('Unknown')}</span>;
|
||||
}
|
||||
|
||||
export default QualityProfileName;
|
||||
@@ -1,31 +0,0 @@
|
||||
import PropTypes from 'prop-types';
|
||||
import React from 'react';
|
||||
import { connect } from 'react-redux';
|
||||
import { createSelector } from 'reselect';
|
||||
import createQualityProfileSelector from 'Store/Selectors/createQualityProfileSelector';
|
||||
|
||||
function createMapStateToProps() {
|
||||
return createSelector(
|
||||
createQualityProfileSelector(),
|
||||
(qualityProfile) => {
|
||||
return {
|
||||
name: qualityProfile.name
|
||||
};
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
function QualityProfileNameConnector({ name, ...otherProps }) {
|
||||
return (
|
||||
<span>
|
||||
{name}
|
||||
</span>
|
||||
);
|
||||
}
|
||||
|
||||
QualityProfileNameConnector.propTypes = {
|
||||
qualityProfileId: PropTypes.number.isRequired,
|
||||
name: PropTypes.string.isRequired
|
||||
};
|
||||
|
||||
export default connect(createMapStateToProps)(QualityProfileNameConnector);
|
||||
@@ -27,6 +27,12 @@ export default function translate(
|
||||
key: string,
|
||||
tokens: Record<string, string | number | boolean> = {}
|
||||
) {
|
||||
const { isProduction = true } = window.Readarr;
|
||||
|
||||
if (!isProduction && !(key in translations)) {
|
||||
console.warn(`Missing translation for key: ${key}`);
|
||||
}
|
||||
|
||||
const translation = translations[key] || key;
|
||||
|
||||
tokens.appName = 'Readarr';
|
||||
|
||||
1
frontend/typings/Globals.d.ts
vendored
1
frontend/typings/Globals.d.ts
vendored
@@ -7,5 +7,6 @@ interface Window {
|
||||
theme: string;
|
||||
urlBase: string;
|
||||
version: string;
|
||||
isProduction: boolean;
|
||||
};
|
||||
}
|
||||
|
||||
@@ -37,7 +37,7 @@
|
||||
<PackageVersion Include="NLog.Extensions.Logging" Version="5.2.3" />
|
||||
<PackageVersion Include="NLog" Version="5.1.4" />
|
||||
<PackageVersion Include="NLog.Targets.Syslog" Version="7.0.0" />
|
||||
<PackageVersion Include="Npgsql" Version="7.0.9" />
|
||||
<PackageVersion Include="Npgsql" Version="7.0.10" />
|
||||
<PackageVersion Include="NUnit3TestAdapter" Version="4.2.1" />
|
||||
<PackageVersion Include="NUnit" Version="3.14.0" />
|
||||
<PackageVersion Include="NunitXml.TestLogger" Version="3.0.117" />
|
||||
@@ -45,7 +45,7 @@
|
||||
<PackageVersion Include="RestSharp.Serializers.SystemTextJson" Version="106.15.0" />
|
||||
<PackageVersion Include="RestSharp" Version="106.15.0" />
|
||||
<PackageVersion Include="Selenium.Support" Version="3.141.0" />
|
||||
<PackageVersion Include="Selenium.WebDriver.ChromeDriver" Version="91.0.4472.10100" />
|
||||
<PackageVersion Include="Selenium.WebDriver.ChromeDriver" Version="134.0.6998.16500" />
|
||||
<PackageVersion Include="Sentry" Version="4.0.2" />
|
||||
<PackageVersion Include="SharpZipLib" Version="1.4.2" />
|
||||
<PackageVersion Include="SixLabors.ImageSharp" Version="3.1.7" />
|
||||
@@ -58,7 +58,7 @@
|
||||
<PackageVersion Include="System.IO.Abstractions.TestingHelpers" Version="17.0.24" />
|
||||
<PackageVersion Include="System.IO.Abstractions" Version="17.0.24" />
|
||||
<PackageVersion Include="System.IO.FileSystem.AccessControl" Version="5.0.0" />
|
||||
<PackageVersion Include="System.Memory" Version="4.6.0" />
|
||||
<PackageVersion Include="System.Memory" Version="4.6.2" />
|
||||
<PackageVersion Include="System.Reflection.TypeExtensions" Version="4.7.0" />
|
||||
<PackageVersion Include="System.Resources.Extensions" Version="6.0.0" />
|
||||
<PackageVersion Include="System.Runtime.Loader" Version="4.3.0" />
|
||||
@@ -66,7 +66,7 @@
|
||||
<PackageVersion Include="System.ServiceProcess.ServiceController" Version="6.0.1" />
|
||||
<PackageVersion Include="System.Text.Encoding.CodePages" Version="6.0.0" />
|
||||
<PackageVersion Include="System.Text.Json" Version="6.0.10" />
|
||||
<PackageVersion Include="System.ValueTuple" Version="4.5.0" />
|
||||
<PackageVersion Include="System.ValueTuple" Version="4.6.1" />
|
||||
<PackageVersion Include="TagLibSharp-Lidarr" Version="2.2.0.19" />
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
</Project>
|
||||
|
||||
@@ -40,15 +40,16 @@ namespace NzbDrone.Automation.Test
|
||||
var service = ChromeDriverService.CreateDefaultService();
|
||||
|
||||
// Timeout as windows automation tests seem to take alot longer to get going
|
||||
driver = new ChromeDriver(service, options, new TimeSpan(0, 3, 0));
|
||||
driver = new ChromeDriver(service, options, TimeSpan.FromMinutes(3));
|
||||
|
||||
driver.Manage().Window.Size = new System.Drawing.Size(1920, 1080);
|
||||
driver.Manage().Window.FullScreen();
|
||||
|
||||
_runner = new NzbDroneRunner(LogManager.GetCurrentClassLogger(), null);
|
||||
_runner.KillAll();
|
||||
_runner.Start(true);
|
||||
|
||||
driver.Url = "http://localhost:8787";
|
||||
driver.Navigate().GoToUrl("http://localhost:8787");
|
||||
|
||||
var page = new PageBase(driver);
|
||||
page.WaitForNoSpinner();
|
||||
@@ -68,7 +69,7 @@ namespace NzbDrone.Automation.Test
|
||||
{
|
||||
try
|
||||
{
|
||||
var image = ((ITakesScreenshot)driver).GetScreenshot();
|
||||
var image = (driver as ITakesScreenshot).GetScreenshot();
|
||||
image.SaveAsFile($"./{name}_test_screenshot.png", ScreenshotImageFormat.Png);
|
||||
}
|
||||
catch (Exception ex)
|
||||
|
||||
@@ -1,19 +1,17 @@
|
||||
using System;
|
||||
using System.Threading;
|
||||
using OpenQA.Selenium;
|
||||
using OpenQA.Selenium.Remote;
|
||||
using OpenQA.Selenium.Support.UI;
|
||||
|
||||
namespace NzbDrone.Automation.Test.PageModel
|
||||
{
|
||||
public class PageBase
|
||||
{
|
||||
private readonly RemoteWebDriver _driver;
|
||||
private readonly IWebDriver _driver;
|
||||
|
||||
public PageBase(RemoteWebDriver driver)
|
||||
public PageBase(IWebDriver driver)
|
||||
{
|
||||
_driver = driver;
|
||||
driver.Manage().Window.Maximize();
|
||||
}
|
||||
|
||||
public IWebElement FindByClass(string className, int timeout = 5)
|
||||
|
||||
@@ -3,6 +3,7 @@ using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.IO.Abstractions;
|
||||
using System.Linq;
|
||||
using System.Threading;
|
||||
using NLog;
|
||||
using NzbDrone.Common.EnsureThat;
|
||||
using NzbDrone.Common.EnvironmentInfo;
|
||||
@@ -306,9 +307,26 @@ namespace NzbDrone.Common.Disk
|
||||
{
|
||||
Ensure.That(path, () => path).IsValidPath(PathValidationType.CurrentOs);
|
||||
|
||||
var files = GetFiles(path, recursive);
|
||||
var files = GetFiles(path, recursive).ToList();
|
||||
|
||||
files.ToList().ForEach(RemoveReadOnly);
|
||||
files.ForEach(RemoveReadOnly);
|
||||
|
||||
var attempts = 0;
|
||||
|
||||
while (attempts < 3 && files.Any())
|
||||
{
|
||||
EmptyFolder(path);
|
||||
|
||||
if (GetFiles(path, recursive).Any())
|
||||
{
|
||||
// Wait for IO operations to complete after emptying the folder since they aren't always
|
||||
// instantly removed and it can lead to false positives that files are still present.
|
||||
Thread.Sleep(3000);
|
||||
}
|
||||
|
||||
attempts++;
|
||||
files = GetFiles(path, recursive).ToList();
|
||||
}
|
||||
|
||||
_fileSystem.Directory.Delete(path, recursive);
|
||||
}
|
||||
|
||||
@@ -108,6 +108,15 @@ namespace NzbDrone.Common.Extensions
|
||||
return Directory.GetParent(cleanPath)?.FullName;
|
||||
}
|
||||
|
||||
public static string GetCleanPath(this string path)
|
||||
{
|
||||
var cleanPath = OsInfo.IsWindows
|
||||
? PARENT_PATH_END_SLASH_REGEX.Replace(path, "")
|
||||
: path.TrimEnd(Path.DirectorySeparatorChar);
|
||||
|
||||
return cleanPath;
|
||||
}
|
||||
|
||||
public static bool IsParentPath(this string parentPath, string childPath)
|
||||
{
|
||||
if (parentPath != "/" && !parentPath.EndsWith(":\\"))
|
||||
|
||||
@@ -4,27 +4,27 @@ namespace NzbDrone.Common.Instrumentation.Extensions
|
||||
{
|
||||
public static class LoggerExtensions
|
||||
{
|
||||
[MessageTemplateFormatMethod("message")]
|
||||
public static void ProgressInfo(this Logger logger, string message, params object[] args)
|
||||
{
|
||||
var formattedMessage = string.Format(message, args);
|
||||
LogProgressMessage(logger, LogLevel.Info, formattedMessage);
|
||||
LogProgressMessage(logger, LogLevel.Info, message, args);
|
||||
}
|
||||
|
||||
[MessageTemplateFormatMethod("message")]
|
||||
public static void ProgressDebug(this Logger logger, string message, params object[] args)
|
||||
{
|
||||
var formattedMessage = string.Format(message, args);
|
||||
LogProgressMessage(logger, LogLevel.Debug, formattedMessage);
|
||||
LogProgressMessage(logger, LogLevel.Debug, message, args);
|
||||
}
|
||||
|
||||
[MessageTemplateFormatMethod("message")]
|
||||
public static void ProgressTrace(this Logger logger, string message, params object[] args)
|
||||
{
|
||||
var formattedMessage = string.Format(message, args);
|
||||
LogProgressMessage(logger, LogLevel.Trace, formattedMessage);
|
||||
LogProgressMessage(logger, LogLevel.Trace, message, args);
|
||||
}
|
||||
|
||||
private static void LogProgressMessage(Logger logger, LogLevel level, string message)
|
||||
private static void LogProgressMessage(Logger logger, LogLevel level, string message, object[] parameters)
|
||||
{
|
||||
var logEvent = new LogEventInfo(level, logger.Name, message);
|
||||
var logEvent = new LogEventInfo(level, logger.Name, null, message, parameters);
|
||||
logEvent.Properties.Add("Status", "");
|
||||
|
||||
logger.Log(logEvent);
|
||||
|
||||
@@ -6,6 +6,7 @@ using System.ComponentModel;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using NLog;
|
||||
using NzbDrone.Common.EnvironmentInfo;
|
||||
using NzbDrone.Common.Model;
|
||||
@@ -117,7 +118,9 @@ namespace NzbDrone.Common.Processes
|
||||
UseShellExecute = false,
|
||||
RedirectStandardError = true,
|
||||
RedirectStandardOutput = true,
|
||||
RedirectStandardInput = true
|
||||
RedirectStandardInput = true,
|
||||
StandardOutputEncoding = Encoding.UTF8,
|
||||
StandardErrorEncoding = Encoding.UTF8
|
||||
};
|
||||
|
||||
if (environmentVariables != null)
|
||||
|
||||
@@ -17,7 +17,7 @@ namespace NzbDrone.Common.Reflection
|
||||
|
||||
public static List<Type> ImplementationsOf<T>(this Assembly assembly)
|
||||
{
|
||||
return assembly.GetTypes().Where(c => typeof(T).IsAssignableFrom(c)).ToList();
|
||||
return assembly.GetExportedTypes().Where(c => typeof(T).IsAssignableFrom(c)).ToList();
|
||||
}
|
||||
|
||||
public static bool IsSimpleType(this Type type)
|
||||
@@ -68,7 +68,7 @@ namespace NzbDrone.Common.Reflection
|
||||
|
||||
public static Type FindTypeByName(this Assembly assembly, string name)
|
||||
{
|
||||
return assembly.GetTypes().SingleOrDefault(c => c.Name.Equals(name, StringComparison.InvariantCultureIgnoreCase));
|
||||
return assembly.GetExportedTypes().SingleOrDefault(c => c.Name.Equals(name, StringComparison.InvariantCultureIgnoreCase));
|
||||
}
|
||||
|
||||
public static bool HasAttribute<TAttribute>(this Type type)
|
||||
|
||||
@@ -13,7 +13,7 @@ using NzbDrone.Core.Test.Framework;
|
||||
namespace NzbDrone.Core.Test.MetadataSource.Goodreads
|
||||
{
|
||||
[TestFixture]
|
||||
[Ignore("Waiting for metadata to be back again", Until = "2025-05-15 00:00:00Z")]
|
||||
[Ignore("Waiting for metadata to be back again", Until = "2026-01-15 00:00:00Z")]
|
||||
public class BookInfoProxyFixture : CoreTest<BookInfoProxy>
|
||||
{
|
||||
private MetadataProfile _metadataProfile;
|
||||
|
||||
@@ -15,7 +15,7 @@ using NzbDrone.Test.Common;
|
||||
namespace NzbDrone.Core.Test.MetadataSource.Goodreads
|
||||
{
|
||||
[TestFixture]
|
||||
[Ignore("Waiting for metadata to be back again", Until = "2025-05-15 00:00:00Z")]
|
||||
[Ignore("Waiting for metadata to be back again", Until = "2026-01-15 00:00:00Z")]
|
||||
public class BookInfoProxySearchFixture : CoreTest<BookInfoProxy>
|
||||
{
|
||||
[SetUp]
|
||||
|
||||
@@ -222,7 +222,7 @@ namespace NzbDrone.Core.Books
|
||||
|
||||
protected override void DeleteEntity(Author local, bool deleteFiles)
|
||||
{
|
||||
_authorService.DeleteAuthor(local.Id, true);
|
||||
_authorService.DeleteAuthor(local.Id, deleteFiles);
|
||||
}
|
||||
|
||||
protected override List<Book> GetRemoteChildren(Author local, Author remote)
|
||||
|
||||
@@ -239,7 +239,7 @@ namespace NzbDrone.Core.Books
|
||||
|
||||
protected override void DeleteEntity(Book local, bool deleteFiles)
|
||||
{
|
||||
_bookService.DeleteBook(local.Id, true);
|
||||
_bookService.DeleteBook(local.Id, deleteFiles);
|
||||
}
|
||||
|
||||
protected override List<Edition> GetRemoteChildren(Book local, Book remote)
|
||||
|
||||
@@ -126,7 +126,7 @@ namespace NzbDrone.Core.Books
|
||||
if (ShouldDelete(local))
|
||||
{
|
||||
_logger.Warn($"{typeof(TEntity).Name} {local} not found in metadata and is being deleted");
|
||||
DeleteEntity(local, true);
|
||||
DeleteEntity(local, false);
|
||||
return false;
|
||||
}
|
||||
else
|
||||
|
||||
@@ -264,7 +264,7 @@ namespace NzbDrone.Core.Datastore
|
||||
|
||||
protected void Delete(SqlBuilder builder)
|
||||
{
|
||||
var sql = builder.AddDeleteTemplate(typeof(TModel)).LogQuery();
|
||||
var sql = builder.AddDeleteTemplate(typeof(TModel));
|
||||
|
||||
using (var conn = _database.OpenConnection())
|
||||
{
|
||||
|
||||
@@ -40,6 +40,7 @@ namespace NzbDrone.Core.Datastore
|
||||
Environment.SetEnvironmentVariable("No_Expand", "true");
|
||||
Environment.SetEnvironmentVariable("No_SQLiteXmlConfigFile", "true");
|
||||
Environment.SetEnvironmentVariable("No_PreLoadSQLite", "true");
|
||||
Environment.SetEnvironmentVariable("No_SQLiteFunctions", "true");
|
||||
}
|
||||
|
||||
public DbFactory(IMigrationController migrationController,
|
||||
|
||||
@@ -10,6 +10,9 @@ namespace NzbDrone.Core.History
|
||||
{
|
||||
public const string DOWNLOAD_CLIENT = "downloadClient";
|
||||
public const string RELEASE_SOURCE = "releaseSource";
|
||||
public const string RELEASE_GROUP = "releaseGroup";
|
||||
public const string SIZE = "size";
|
||||
public const string INDEXER = "indexer";
|
||||
|
||||
public EntityHistory()
|
||||
{
|
||||
|
||||
@@ -116,6 +116,7 @@ namespace NzbDrone.Core.History
|
||||
{
|
||||
var builder = Builder()
|
||||
.Join<EntityHistory, Author>((h, a) => h.AuthorId == a.Id)
|
||||
.LeftJoin<EntityHistory, Book>((h, b) => h.BookId == b.Id)
|
||||
.Where<EntityHistory>(x => x.Date >= date);
|
||||
|
||||
if (eventType.HasValue)
|
||||
@@ -123,9 +124,10 @@ namespace NzbDrone.Core.History
|
||||
builder.Where<EntityHistory>(h => h.EventType == eventType);
|
||||
}
|
||||
|
||||
return _database.QueryJoined<EntityHistory, Author>(builder, (history, author) =>
|
||||
return _database.QueryJoined<EntityHistory, Author, Book>(builder, (history, author, book) =>
|
||||
{
|
||||
history.Author = author;
|
||||
history.Book = book;
|
||||
return history;
|
||||
}).OrderBy(h => h.Date).ToList();
|
||||
}
|
||||
|
||||
@@ -263,7 +263,9 @@ namespace NzbDrone.Core.History
|
||||
history.Data.Add("DownloadClient", message.DownloadClient);
|
||||
history.Data.Add("DownloadClientName", message.TrackedDownload?.DownloadItem.DownloadClientInfo.Name);
|
||||
history.Data.Add("Message", message.Message);
|
||||
history.Data.Add("Size", message.TrackedDownload?.DownloadItem.TotalSize.ToString());
|
||||
history.Data.Add("ReleaseGroup", message.TrackedDownload?.RemoteBook?.ParsedBookInfo?.ReleaseGroup ?? message.Data.GetValueOrDefault(EntityHistory.RELEASE_GROUP));
|
||||
history.Data.Add("Size", message.TrackedDownload?.DownloadItem.TotalSize.ToString() ?? message.Data.GetValueOrDefault(EntityHistory.SIZE));
|
||||
history.Data.Add("Indexer", message.TrackedDownload?.RemoteBook?.Release?.Indexer ?? message.Data.GetValueOrDefault(EntityHistory.INDEXER));
|
||||
|
||||
_historyRepository.Insert(history);
|
||||
}
|
||||
@@ -373,6 +375,7 @@ namespace NzbDrone.Core.History
|
||||
history.Data.Add("Message", message.Message);
|
||||
history.Data.Add("ReleaseGroup", message.TrackedDownload?.RemoteBook?.ParsedBookInfo?.ReleaseGroup);
|
||||
history.Data.Add("Size", message.TrackedDownload?.DownloadItem.TotalSize.ToString());
|
||||
history.Data.Add("Indexer", message.TrackedDownload?.RemoteBook?.Release?.Indexer);
|
||||
|
||||
historyToAdd.Add(history);
|
||||
}
|
||||
|
||||
@@ -653,5 +653,7 @@
|
||||
"UnmappedFiles": "المجلدات غير المعينة",
|
||||
"UpdateAppDirectlyLoadError": "تعذر تحديث {appName} مباشرة ،",
|
||||
"Clone": "قريب",
|
||||
"BuiltIn": "مدمج"
|
||||
"BuiltIn": "مدمج",
|
||||
"AddNewAuthorRootFolderHelpText": "سيتم إنشاء المجلد الفرعي \"{folder}\" تلقائيًا",
|
||||
"AddRootFolder": "إضافة مجلد جذر"
|
||||
}
|
||||
|
||||
@@ -695,5 +695,7 @@
|
||||
"AuthenticationRequired": "Изисква се удостоверяване",
|
||||
"AuthenticationRequiredPasswordHelpTextWarning": "Въведете нова парола",
|
||||
"ApplyChanges": "Прилагане на промените",
|
||||
"AuthenticationRequiredPasswordConfirmationHelpTextWarning": "Потвърдете новата парола"
|
||||
"AuthenticationRequiredPasswordConfirmationHelpTextWarning": "Потвърдете новата парола",
|
||||
"AddNewAuthorRootFolderHelpText": "Подпапката „{0}“ ще бъде създадена автоматично",
|
||||
"AddRootFolder": "Добавяне на коренна папка"
|
||||
}
|
||||
|
||||
@@ -242,7 +242,7 @@
|
||||
"PackageVersion": "Versió del paquet",
|
||||
"PageSize": "Mida de la pàgina",
|
||||
"PageSizeHelpText": "Nombre d'elements per mostrar a cada pàgina",
|
||||
"Proper": "Proper",
|
||||
"Proper": "Correcte",
|
||||
"ProxyBypassFilterHelpText": "Utilitzeu ',' com a separador i '*.' com a comodí per als subdominis",
|
||||
"ProxyCheckBadRequestMessage": "No s'ha pogut provar el servidor intermediari. Codi d'estat: {0}",
|
||||
"ProxyCheckFailedToTestMessage": "No s'ha pogut provar el servidor intermediari: {0}",
|
||||
@@ -316,7 +316,7 @@
|
||||
"URLBase": "Base URL",
|
||||
"Backups": "Còpies de seguretat",
|
||||
"Connections": "Connexions",
|
||||
"CopyUsingHardlinksHelpText": "Utilitzeu els enllaços durs quan intenteu copiar fitxers de torrents que encara s'estan sembrant",
|
||||
"CopyUsingHardlinksHelpText": "Els enllaços durs permeten a Readarr importar torrents de sembra a la carpeta de la sèrie sense prendre espai de disc extra o copiar tot el contingut del fitxer. Els enllaços durs només funcionaran si l'origen i la destinació estan en el mateix volum",
|
||||
"DeleteBackup": "Suprimeix la còpia de seguretat",
|
||||
"DeleteBackupMessageText": "Esteu segur que voleu suprimir la còpia de seguretat '{name}'?",
|
||||
"DeleteDownloadClient": "Suprimeix el client de descàrrega",
|
||||
@@ -336,7 +336,7 @@
|
||||
"IgnoredAddresses": "Adreces ignorades",
|
||||
"IgnoredPlaceHolder": "Afegeix una nova restricció",
|
||||
"ImportExtraFiles": "Importa fitxers addicionals",
|
||||
"ImportFailedInterp": "ImportFailedInterp",
|
||||
"ImportFailedInterp": "Importació fallida: {0}",
|
||||
"IncludeHealthWarningsHelpText": "Inclou advertències de salut",
|
||||
"NotificationTriggers": "Activadors de notificacions",
|
||||
"NoUpdatesAreAvailable": "No hi ha actualitzacions disponibles",
|
||||
@@ -428,52 +428,52 @@
|
||||
"BlocklistRelease": "Publicació de la llista de bloqueig",
|
||||
"HasPendingChangesNoChanges": "Sense Canvis",
|
||||
"ManualImportSelectEdition": "Importació manual - Seleccioneu la pel·lícula",
|
||||
"MissingFromDisk": "{appName} no ha pogut trobar el fitxer al disc, de manera que el fitxer es desenllaçarà de la pel·lícula a la base de dades",
|
||||
"MissingFromDisk": "Readarr no ha pogut trobar el fitxer al disc, de manera que el fitxer es desenllaçarà de la pel·lícula a la base de dades",
|
||||
"SupportsRssvalueRSSIsNotSupportedWithThisIndexer": "RSS no és compatible amb aquest indexador",
|
||||
"SupportsSearchvalueWillBeUsedWhenAutomaticSearchesArePerformedViaTheUIOrByReadarr": "S'utilitzarà quan es realitzin cerques automàtiques mitjançant la interfície d'usuari o per {appName}",
|
||||
"CutoffHelpText": "Un cop s'assoleixi aquesta qualitat, {appName} ja no baixarà pel·lícules",
|
||||
"SupportsSearchvalueWillBeUsedWhenAutomaticSearchesArePerformedViaTheUIOrByReadarr": "S'utilitzarà quan es realitzin cerques automàtiques mitjançant la interfície d'usuari o per Readarr",
|
||||
"CutoffHelpText": "Un cop s'assoleixi aquesta qualitat, Readarr ja no baixarà pel·lícules",
|
||||
"ResetAPIKeyMessageText": "Esteu segur que voleu restablir la clau API?",
|
||||
"PropersAndRepacks": "Propietats i Repacks",
|
||||
"RemotePathMappingCheckFolderPermissions": "{appName} pot veure però no accedir al directori de descàrregues {0}. Error de permisos probable.",
|
||||
"RemotePathMappingCheckFolderPermissions": "Readarr pot veure però no accedir al directori de descàrregues {1}. Error de permisos probable.",
|
||||
"RescanAuthorFolderAfterRefresh": "Torna a escanejar la carpeta de pel·lícules després de l'actualització",
|
||||
"RescanAfterRefreshHelpText": "Torneu a escanejar la carpeta de la pel·lícula després d'actualitzar la pel·lícula",
|
||||
"RestartReadarr": "Reinicia {appName}",
|
||||
"RestartReadarr": "Reinicia Readarr",
|
||||
"ShowRelativeDatesHelpText": "Mostra dates relatives (avui/ahir/etc) o absolutes",
|
||||
"ShowSearchActionHelpText": "Mostra el botó de cerca al passar el cursor",
|
||||
"TheAuthorFolderAndAllOfItsContentWillBeDeleted": "La carpeta de pel·lícules '{0}' i tot el seu contingut es suprimiran.",
|
||||
"UrlBaseHelpTextWarning": "Cal reiniciar perquè tingui efecte",
|
||||
"ApplicationURL": "URL de l'aplicació",
|
||||
"ApplicationUrlHelpText": "URL extern de l'aplicació, inclòs http(s)://, port i URL base",
|
||||
"BackupFolderHelpText": "Els camins relatius estaran sota el directori AppData del {appName}",
|
||||
"BackupFolderHelpText": "Els camins relatius estaran sota el directori AppData de Readarr",
|
||||
"CancelPendingTask": "Esteu segur que voleu cancel·lar aquesta tasca pendent?",
|
||||
"ChownGroupHelpTextWarning": "Això només funciona si l'usuari que executa {appName} és el propietari del fitxer. És millor assegurar-se que el client de descàrrega utilitza el mateix grup que {appName}.",
|
||||
"ChownGroupHelpTextWarning": "Això només funciona si l'usuari que executa Readarr és el propietari del fitxer. És millor assegurar-se que el client de descàrrega utilitza el mateix grup que Readarr.",
|
||||
"ConnectSettingsSummary": "Notificacions, connexions a servidors/reproductors multimèdia i scripts personalitzats",
|
||||
"DeleteEmptyFoldersHelpText": "Suprimeix les carpetes de pel·lícules buides durant l'exploració del disc i quan s'esborren els fitxers de pel·lícules",
|
||||
"DeleteImportListMessageText": "Esteu segur que voleu suprimir la llista '{name}'?",
|
||||
"DeleteMetadataProfileMessageText": "Esteu segur que voleu suprimir el perfil de qualitat {0}",
|
||||
"DeleteMetadataProfileMessageText": "Esteu segur que voleu suprimir el perfil de metadades ‘{name}’?",
|
||||
"ForMoreInformationOnTheIndividualDownloadClientsClickOnTheInfoButtons": "Per obtenir més informació sobre els clients de baixada individuals, feu clic als botons de més informació.",
|
||||
"ForMoreInformationOnTheIndividualIndexersClickOnTheInfoButtons": "Per obtenir més informació sobre els indexadors individuals, feu clic als botons d'informació.",
|
||||
"ForMoreInformationOnTheIndividualListsClickOnTheInfoButtons": "Per obtenir més informació sobre les llistes d'importació individuals, feu clic als botons d'informació.",
|
||||
"IndexerPriorityHelpText": "Prioritat de l'indexador d'1 (la més alta) a 50 (la més baixa). Per defecte: 25. S'utilitza quan s'agafa llançaments com a desempat per a versions iguals, {appName} encara utilitzarà tots els indexadors habilitats per a la sincronització i la cerca RSS",
|
||||
"IndexerRssHealthCheckNoIndexers": "No hi ha indexadors disponibles amb la sincronització RSS activada, {appName} no capturarà les noves versions automàticament",
|
||||
"IndexerSearchCheckNoAutomaticMessage": "No hi ha indexadors disponibles amb la cerca automàtica activada, {appName} no proporcionarà cap resultat de cerca automàtica",
|
||||
"IndexerSearchCheckNoInteractiveMessage": "No hi ha indexadors amb la cerca interactiva activada, {appName} no obtindrà cap resultat de cerca",
|
||||
"IndexerPriorityHelpText": "Prioritat de l'indexador d'1 (la més alta) a 50 (la més baixa). Per defecte: 25. Usada per a desempatar llançaments iguals quan es capturen, Readarr continuarà usant tots els indexadores habilitats per a Sincronització d'RSS i Cerca.",
|
||||
"IndexerRssHealthCheckNoIndexers": "No hi ha indexadors disponibles amb la sincronització RSS activada, Readarr no capturarà les noves versions automàticament",
|
||||
"IndexerSearchCheckNoAutomaticMessage": "No hi ha indexadors disponibles amb la cerca automàtica activada, Readarr no proporcionarà cap resultat de cerca automàtica",
|
||||
"IndexerSearchCheckNoInteractiveMessage": "No hi ha indexadors amb la cerca interactiva activada, Readarr no obtindrà cap resultat de cerca",
|
||||
"IsCutoffUpgradeUntilThisQualityIsMetOrExceeded": "Actualitzeu fins que s'assoleixi o superi aquesta qualitat",
|
||||
"IsTagUsedCannotBeDeletedWhileInUse": "No es pot suprimir mentre està en ús",
|
||||
"LaunchBrowserHelpText": " Obriu un navegador web i navegueu a la pàgina d'inici de {appName} a l'inici de l'aplicació.",
|
||||
"LaunchBrowserHelpText": " Obriu un navegador web i navegueu a la pàgina d'inici de Readarr a l'inici de l'aplicació.",
|
||||
"LoadingBookFilesFailed": "No s'han pogut carregar els fitxers de pel·lícules",
|
||||
"NoHistory": "Sense història",
|
||||
"NoHistory": "Sense història.",
|
||||
"OnBookFileDeleteForUpgradeHelpText": "Al suprimir el fitxer de pel·lícula per a l'actualització",
|
||||
"OnBookFileDeleteHelpText": "Al suprimir fitxer de pel·lícula",
|
||||
"ReleaseBranchCheckOfficialBranchMessage": "La branca {0} no és una branca de llançament de {appName} vàlida, no rebreu actualitzacions",
|
||||
"ReleaseBranchCheckOfficialBranchMessage": "La branca {0} no és una branca de llançament de Readarr vàlida, no rebreu actualitzacions",
|
||||
"ReleaseDate": "Dates de llançament",
|
||||
"RemotePathMappingCheckDownloadPermissions": "{appName} pot veure però no accedir a la pel·lícula baixada {0}. Error de permisos probable.",
|
||||
"RemotePathMappingCheckFilesGenericPermissions": "El client de baixada {0} ha informat de fitxers a {1} però {appName} no pot veure aquest directori. És possible que hàgiu d'ajustar els permisos de la carpeta.",
|
||||
"RemotePathMappingCheckGenericPermissions": "El client de baixada {0} col·loca les baixades a {1} però {appName} no pot veure aquest directori. És possible que hàgiu d'ajustar els permisos de la carpeta.",
|
||||
"ReplaceIllegalCharactersHelpText": "Substitueix caràcters il·legals. Si no es marca, {appName} els eliminarà",
|
||||
"RemotePathMappingCheckDownloadPermissions": "Readarr pot veure però no accedir a la pel·lícula baixada {0}. Error de permisos probable.",
|
||||
"RemotePathMappingCheckFilesGenericPermissions": "El client de baixada {0} ha informat de fitxers a {1} però Readarr no pot veure aquest directori. És possible que hàgiu d'ajustar els permisos de la carpeta.",
|
||||
"RemotePathMappingCheckGenericPermissions": "El client de baixada {0} col·loca les baixades a {1} però Readarr no pot veure aquest directori. És possible que hàgiu d'ajustar els permisos de la carpeta.",
|
||||
"ReplaceIllegalCharactersHelpText": "Substitueix caràcters il·legals. Si no es marca, Readarr els eliminarà",
|
||||
"RssSyncIntervalHelpText": "Interval en minuts. Establiu a zero per desactivar (això aturarà tota captura automàtica de llançaments)",
|
||||
"SelectedCountBooksSelectedInterp": "S'han seleccionat {0} pel·lícules",
|
||||
"SettingsRemotePathMappingLocalPathHelpText": "Camí que {appName} hauria d'utilitzar per accedir al camí remot localment",
|
||||
"SettingsRemotePathMappingLocalPathHelpText": "Camí que Readarr hauria d'utilitzar per accedir al camí remot localment",
|
||||
"ShortDateFormat": "Format de data curta",
|
||||
"ShowBookTitleHelpText": "Mostra el títol de la pel·lícula sota el cartell",
|
||||
"ShowRelativeDates": "Mostra les dates relatives",
|
||||
@@ -491,34 +491,34 @@
|
||||
"UserAgentProvidedByTheAppThatCalledTheAPI": "Agent d'usuari proporcionat per l'aplicació per fer peticions a l'API",
|
||||
"BranchUpdateMechanism": "Branca utilitzada pel mecanisme d'actualització extern",
|
||||
"WriteTagsNo": "Mai",
|
||||
"RestartReloadNote": "Nota: {appName} es reiniciarà i tornarà a carregar automàticament la interfície d'usuari durant el procés de restauració.",
|
||||
"RestartReloadNote": "Nota: Readarr es reiniciarà i tornarà a carregar automàticament la interfície d'usuari durant el procés de restauració.",
|
||||
"Series": "Sèries",
|
||||
"ShownAboveEachColumnWhenWeekIsTheActiveView": "Es mostra a sobre de cada columna quan la setmana és la visualització activa",
|
||||
"SorryThatAuthorCannotBeFound": "Ho sentim, aquesta pel·lícula no s'ha trobat.",
|
||||
"SorryThatBookCannotBeFound": "Ho sentim, aquesta pel·lícula no s'ha trobat.",
|
||||
"SupportsSearchvalueWillBeUsedWhenInteractiveSearchIsUsed": "S'utilitzarà quan s'utilitzi la cerca interactiva",
|
||||
"ThisWillApplyToAllIndexersPleaseFollowTheRulesSetForthByThem": "Això s'aplicarà a tots els indexadors, si us plau, seguiu les regles establertes per ells",
|
||||
"UnableToLoadHistory": "No es pot carregar l'historial",
|
||||
"UnableToLoadHistory": "No es pot carregar l'historial.",
|
||||
"IconTooltip": "Programat",
|
||||
"ReadarrTags": "Etiquetes de {appName}",
|
||||
"ReadarrTags": "Etiquetes de Readarr",
|
||||
"DownloadPropersAndRepacksHelpTexts1": "Si s'ha d'actualitzar automàticament o no a Propers/Repacks",
|
||||
"GrabReleaseMessageText": "{appName} no ha pogut determinar per a quina pel·lícula era aquest llançament. És possible que {appName} no pugui importar automàticament aquesta versió. Voleu capturar \"{0}\"?",
|
||||
"GrabReleaseMessageText": "Readarr no ha pogut determinar per a quina pel·lícula era aquest llançament. És possible que Readarr no pugui importar automàticament aquesta versió. Voleu capturar '{0}'?",
|
||||
"IsCutoffCutoff": "Requisit",
|
||||
"MountCheckMessage": "El muntatge que conté una ruta de pel·lícula es munta com a només de lectura: ",
|
||||
"RescanAfterRefreshHelpTextWarning": "{appName} no detectarà automàticament els canvis als fitxers quan no estigui configurat com a \"Sempre\"",
|
||||
"RescanAfterRefreshHelpTextWarning": "Readarr no detectarà automàticament els canvis als fitxers quan no estigui configurat com a \"Sempre\"",
|
||||
"ShowUnknownAuthorItems": "Mostra elements de pel·lícula desconeguda",
|
||||
"Size": " Mida",
|
||||
"SkipFreeSpaceCheckWhenImportingHelpText": "Utilitzeu-lo quan {appName} no pugui detectar espai lliure a la carpeta arrel de la pel·lícula",
|
||||
"SkipFreeSpaceCheckWhenImportingHelpText": "Utilitzeu-lo quan Readarr no pugui detectar espai lliure a la carpeta arrel de la pel·lícula",
|
||||
"StandardBookFormat": "Format de pel·lícula estàndard",
|
||||
"UnableToAddANewImportListExclusionPleaseTryAgain": "No es pot afegir una nova llista d'exclusió, torneu-ho a provar.",
|
||||
"UnableToLoadReleaseProfiles": "No es poden carregar els perfils de retard",
|
||||
"UnmonitoredHelpText": "Inclou pel·lícules no monitorades al canal iCal",
|
||||
"UpdateAll": "Actualitzar-ho tot",
|
||||
"AutoUnmonitorPreviouslyDownloadedBooksHelpText": "Les pel·lícules suprimides del disc no es descarten automàticament al {appName}",
|
||||
"AutoUnmonitorPreviouslyDownloadedBooksHelpText": "Les pel·lícules suprimides del disc no es descarten automàticament al Readarr",
|
||||
"ChownGroupHelpText": "Nom del grup o gid. Utilitzeu gid per a sistemes de fitxers remots.",
|
||||
"AuthorClickToChangeBook": "Feu clic per canviar la pel·lícula",
|
||||
"ChmodFolderHelpTextWarning": "Això només funciona si l'usuari que executa {appName} és el propietari del fitxer. És millor assegurar-se que el client de descàrrega estableixi correctament els permisos.",
|
||||
"CopyUsingHardlinksHelpTextWarning": "De tant en tant, els bloquejos de fitxers poden impedir reanomenar els fitxers que s'estan sembrant. Podeu desactivar temporalment la compartició i utilitzar la funció de reanomenar de {appName} com a solució.",
|
||||
"ChmodFolderHelpTextWarning": "Això només funciona si l'usuari que executa Readarr és el propietari del fitxer. És millor assegurar-se que el client de descàrrega estableixi correctament els permisos.",
|
||||
"CopyUsingHardlinksHelpTextWarning": "De tant en tant, els bloquejos de fitxers poden impedir reanomenar els fitxers que s'estan sembrant. Podeu desactivar temporalment la compartició i utilitzar la funció de reanomenar de Readarr com a solució.",
|
||||
"CouldntFindAnyResultsForTerm": "No s'ha pogut trobar cap resultat per a '{0}'",
|
||||
"CreateEmptyAuthorFolders": "Creeu carpetes buides per a les pel·lícules",
|
||||
"CreateEmptyAuthorFoldersHelpText": "Creeu carpetes de pel·lícules que falten durant l'exploració del disc",
|
||||
@@ -531,32 +531,32 @@
|
||||
"ImportExtraFilesHelpText": "Importeu fitxers addicionals coincidents (subtítols, nfo, etc.) després d'importar un fitxer de pel·lícula",
|
||||
"ImportListExclusions": "Suprimeix l'exclusió de la llista d'importació",
|
||||
"LongDateFormat": "Format de data llarga",
|
||||
"MaximumSizeHelpText": "Mida màxima per a una versió que es pot capturar en MB. Establiu a zero per establir-lo en il·limitat",
|
||||
"MaximumSizeHelpText": "Mida màxima per a una versió que es pot capturar en MB. Establiu a zero per establir-lo en il·limitat,",
|
||||
"MetadataProfile": "perfil de metadades",
|
||||
"MetadataProfiles": "perfil de metadades",
|
||||
"OnBookFileDelete": "Al suprimir fitxer de pel·lícula",
|
||||
"OnBookFileDeleteForUpgrade": "Al suprimir el fitxer de pel·lícula per a l'actualització",
|
||||
"ReadarrSupportsAnyDownloadClient": "{appName} admet molts clients de baixada de torrent i usenet populars.",
|
||||
"ReadarrSupportsAnyIndexerThatUsesTheNewznabStandardAsWellAsOtherIndexersListedBelow": "{appName} admet qualsevol indexador que utilitzi l'estàndard Newznab, així com altres indexadors que s'enumeren a continuació.",
|
||||
"ReadarrSupportsAnyDownloadClient": "Readarr admet molts clients de baixada de torrent i usenet populars.",
|
||||
"ReadarrSupportsAnyIndexerThatUsesTheNewznabStandardAsWellAsOtherIndexersListedBelow": "Readarr admet qualsevol indexador que utilitzi l'estàndard Newznab, així com altres indexadors que s'enumeren a continuació.",
|
||||
"RecycleBinHelpText": "Els fitxers de pel·lícula aniran aquí quan se suprimeixin en lloc de suprimir-se permanentment",
|
||||
"RenameBooksHelpText": "{appName} utilitzarà el nom del fitxer existent si el reanomenat està desactivat",
|
||||
"RequiredHelpText": "El llançament ha de contenir almenys un d'aquests termes (no distingeix entre majúscules i minúscules)",
|
||||
"UILanguageHelpText": "Idioma que utilitzarà {appName} per a la interfície d'usuari",
|
||||
"RenameBooksHelpText": "Readarr utilitzarà el nom del fitxer existent si el reanomenat està desactivat",
|
||||
"RequiredHelpText": "Aquesta condició {0} ha de coincidir amb el format personalitzat a aplicar. En cas contrari, una única coincidència {0} és suficient.",
|
||||
"UILanguageHelpText": "Idioma que utilitzarà Readarr per a la interfície d'usuari",
|
||||
"UnableToAddANewRootFolderPleaseTryAgain": "No es pot afegir un indexador nou, torneu-ho a provar.",
|
||||
"UnableToLoadMetadataProfiles": "No es poden carregar els perfils de qualitat",
|
||||
"UpdateMechanismHelpText": "Utilitzeu l'actualitzador integrat de {appName} o un script",
|
||||
"UpdateMechanismHelpText": "Utilitzeu l'actualitzador integrat de Readarr o un script",
|
||||
"UpdateSelected": "Actualització seleccionada",
|
||||
"Database": "Base de dades",
|
||||
"DeleteQualityProfileMessageText": "Esteu segur que voleu suprimir el perfil de qualitat '{name}'?",
|
||||
"DeleteReleaseProfile": "Suprimeix el perfil de llançament",
|
||||
"DeleteReleaseProfileMessageText": "Esteu segur que voleu suprimir aquest perfil de retard?",
|
||||
"DeleteRootFolderMessageText": "Esteu segur que voleu suprimir l'indexador '{0}'?",
|
||||
"DeleteRootFolderMessageText": "Esteu segur que voleu suprimir la carpeta arrel '{name}'?",
|
||||
"DeleteSelectedBookFiles": "Suprimeix els fitxers de pel·lícules seleccionats",
|
||||
"DeleteSelectedBookFilesMessageText": "Esteu segur que voleu suprimir els fitxers de pel·lícules seleccionats?",
|
||||
"IncludeUnknownAuthorItemsHelpText": "Mostra elements sense pel·lícula a la cua. Això podria incloure pel·lícules eliminades o qualsevol altra cosa de la categoria de {appName}",
|
||||
"IncludeUnknownAuthorItemsHelpText": "Mostra elements sense pel·lícula a la cua. Això podria incloure pel·lícules eliminades o qualsevol altra cosa de la categoria de Readarr",
|
||||
"LogLevelvalueTraceTraceLoggingShouldOnlyBeEnabledTemporarily": "El registre de traça només s'hauria d'habilitar temporalment",
|
||||
"PortHelpTextWarning": "Cal reiniciar perquè tingui efecte",
|
||||
"RemotePathMappingCheckImportFailed": "{appName} no ha pogut importar una pel·lícula. Comproveu els vostres registres per obtenir més informació.",
|
||||
"RemotePathMappingCheckImportFailed": "Readarr no ha pogut importar una pel·lícula. Comproveu els vostres registres per obtenir més informació.",
|
||||
"RemoveTagExistingTag": "Etiqueta existent",
|
||||
"RemoveTagRemovingTag": "S'està eliminant l'etiqueta",
|
||||
"SupportsSearchvalueSearchIsNotSupportedWithThisIndexer": "La cerca no és compatible amb aquest indexador",
|
||||
@@ -564,9 +564,9 @@
|
||||
"RequiredPlaceHolder": "Afegeix una nova restricció",
|
||||
"20MinutesTwenty": "20 minuts: {0}",
|
||||
"AlternateTitles": "Títols alternatius",
|
||||
"AnalyticsEnabledHelpText": "Envieu informació anònima d'ús i errors als servidors de {appName}. Això inclou informació sobre el vostre navegador, quines pàgines {appName} WebUI feu servir, informes d'errors, així com el sistema operatiu i la versió del temps d'execució. Utilitzarem aquesta informació per prioritzar les funcions i les correccions d'errors.",
|
||||
"AnalyticsEnabledHelpText": "Envieu informació anònima d'ús i errors als servidors de Readarr. Això inclou informació sobre el vostre navegador, quines pàgines de l'interfície de Readarr feu servir, informes d'errors, així com el sistema operatiu i la versió del temps d'execució. Utilitzarem aquesta informació per prioritzar les funcions i les correccions d'errors.",
|
||||
"AnalyticsEnabledHelpTextWarning": "Cal reiniciar perquè tingui efecte",
|
||||
"AuthenticationMethodHelpText": "Requereix nom d'usuari i contrasenya per accedir al radar",
|
||||
"AuthenticationMethodHelpText": "Requereix el nom d'usuari i la contrasenya per accedir a {appName}",
|
||||
"CalendarWeekColumnHeaderHelpText": "Es mostra a sobre de cada columna quan la setmana és la visualització activa",
|
||||
"45MinutesFourtyFive": "45 minuts: {0}",
|
||||
"60MinutesSixty": "60 minuts: {0}",
|
||||
@@ -584,13 +584,13 @@
|
||||
"BypassIfHighestQuality": "Bypass si és de màxima qualitat",
|
||||
"MinimumCustomFormatScore": "Puntuació mínima de format personalitzat",
|
||||
"CustomFormatScore": "Puntuació de format personalitzat",
|
||||
"EnableRssHelpText": "S'utilitzarà quan {appName} cerqui publicacions periòdicament mitjançant RSS Sync",
|
||||
"EnableRssHelpText": "S'utilitzarà quan Readarr cerqui publicacions periòdicament mitjançant RSS Sync",
|
||||
"ImportListMultipleMissingRoots": "Falten diverses carpetes arrel per a les llistes d'importació: {0}",
|
||||
"IndexerDownloadClientHelpText": "Especifiqueu quin client de baixada s'utilitza per a capturar des d'aquest indexador",
|
||||
"ThemeHelpText": "Canvieu el tema de la interfície d'usuari de l'aplicació, el tema \"Automàtic\" utilitzarà el tema del vostre sistema operatiu per configurar el mode clar o fosc. Inspirat en Theme.Park",
|
||||
"UnableToLoadCustomFormats": "No es poden carregar formats personalitzats",
|
||||
"DeleteCustomFormat": "Suprimeix el format personalitzat",
|
||||
"DeleteFormatMessageText": "Esteu segur que voleu suprimir l'etiqueta de format {0} ?",
|
||||
"DeleteFormatMessageText": "Esteu segur que voleu suprimir l'etiqueta de format '{0}'?",
|
||||
"ExportCustomFormat": "Exporta el format personalitzat",
|
||||
"Formats": "Formats",
|
||||
"IncludeCustomFormatWhenRenamingHelpText": "Inclou en {Custom Formats} el format de canvi de nom",
|
||||
@@ -607,8 +607,8 @@
|
||||
"CustomFormat": "Format personalitzat",
|
||||
"CustomFormatSettings": "Configuració de formats personalitzats",
|
||||
"CustomFormats": "Formats personalitzats",
|
||||
"CutoffFormatScoreHelpText": "Un cop s'arribi a aquesta puntuació de format personalitzat, {appName} ja no baixarà pel·lícules",
|
||||
"DeleteCustomFormatMessageText": "Esteu segur que voleu suprimir l'indexador '{0}'?",
|
||||
"CutoffFormatScoreHelpText": "Un cop s'arribi a aquesta puntuació de format personalitzat, Readarr ja no baixarà pel·lícules",
|
||||
"DeleteCustomFormatMessageText": "Esteu segur que voleu suprimir el format personalitzat ‘{name}’?",
|
||||
"ImportListMissingRoot": "Falta la carpeta arrel per a les llistes d'importació: {0}",
|
||||
"IndexerTagsHelpText": "Utilitzeu aquest indexador només per a pel·lícules amb almenys una etiqueta coincident. Deixeu-ho en blanc per utilitzar-ho amb totes les pel·lícules.",
|
||||
"ColonReplacement": "Substitució de dos punts",
|
||||
@@ -711,7 +711,7 @@
|
||||
"DownloadClientDelugeSettingsDirectoryCompleted": "Directori al qual es mou quan s'hagi completat",
|
||||
"DownloadClientDelugeSettingsDirectoryCompletedHelpText": "Ubicació opcional de les baixades completades, deixeu-lo en blanc per utilitzar la ubicació predeterminada de Deluge",
|
||||
"DownloadClientDelugeSettingsDirectoryHelpText": "Ubicació opcional de les baixades completades, deixeu-lo en blanc per utilitzar la ubicació predeterminada de Deluge",
|
||||
"WhatsNew": "Novetats",
|
||||
"WhatsNew": "Què hi ha de nou?",
|
||||
"SelectDropdown": "Seleccioneu...",
|
||||
"NoCutoffUnmetItems": "No hi ha elements de tall no assolits",
|
||||
"ApplyTagsHelpTextHowToApplyAuthors": "Com aplicar etiquetes a les pel·lícules seleccionades",
|
||||
@@ -723,13 +723,13 @@
|
||||
"ResetQualityDefinitions": "Restableix les definicions de qualitat",
|
||||
"Small": "Petita",
|
||||
"TotalSpace": "Espai total",
|
||||
"BlocklistReleaseHelpText": "Impedeix que {appName} torni a capturar aquesta versió automàticament",
|
||||
"BlocklistReleaseHelpText": "Impedeix que Readarr torni a capturar aquesta versió automàticament",
|
||||
"CatalogNumber": "número de catàleg",
|
||||
"LastWriteTime": "La darrera hora d'escriptura",
|
||||
"NextExecution": "Propera execució",
|
||||
"RemoveCompleted": "S'ha eliminat",
|
||||
"SelectReleaseGroup": "Seleccioneu grup de llançament",
|
||||
"CountDownloadClientsSelected": "{count} client(s) de baixada seleccionat(s)",
|
||||
"CountDownloadClientsSelected": "{selectedCount} client(s) de baixada seleccionat(s)",
|
||||
"Authors": "Autor",
|
||||
"FreeSpace": "Espai lliure",
|
||||
"ExtraFileExtensionsHelpText": "Llista separada per comes de fitxers addicionals per importar (.nfo s'importarà com a .nfo-orig)",
|
||||
@@ -740,7 +740,7 @@
|
||||
"RemoveFailed": "Ha fallat l'eliminació",
|
||||
"ImportLists": "llista d'importació",
|
||||
"RemovingTag": "S'està eliminant l'etiqueta",
|
||||
"ApiKeyValidationHealthCheckMessage": "Actualitzeu la vostra clau de l'API perquè tingui almenys {length} caràcters. Podeu fer-ho mitjançant la configuració o el fitxer de configuració",
|
||||
"ApiKeyValidationHealthCheckMessage": "Actualitzeu la vostra clau de l'API perquè tingui almenys {0} caràcters. Podeu fer-ho mitjançant la configuració o el fitxer de configuració",
|
||||
"ExtraFileExtensionsHelpTextsExamples": "Exemples: '.sub, .nfo' o 'sub,nfo'",
|
||||
"SourceTitle": "Títol de la font",
|
||||
"NoEventsFound": "No s'han trobat esdeveniments",
|
||||
@@ -750,9 +750,9 @@
|
||||
"RecentChanges": "Canvis recents",
|
||||
"Rejections": "Rebutjats",
|
||||
"StatusEndedContinuing": "Continua",
|
||||
"DeleteBookFileMessageText": "Esteu segur que voleu suprimir '{path}'?",
|
||||
"DeleteBookFileMessageText": "Esteu segur que voleu suprimir '{0}'?",
|
||||
"DownloadClientTagHelpText": "Utilitzeu aquest indexador només per a pel·lícules amb almenys una etiqueta coincident. Deixeu-ho en blanc per utilitzar-ho amb totes les pel·lícules.",
|
||||
"DownloadClientRemovesCompletedDownloadsHealthCheckMessage": "El client de baixada {downloadClientName} està configurat per eliminar les baixades completades. Això pot provocar que les baixades s'eliminin del vostre client abans que {1} pugui importar-les.",
|
||||
"DownloadClientRemovesCompletedDownloadsHealthCheckMessage": "El client de baixada {0} està configurat per eliminar les baixades completades. Això pot provocar que les baixades s'eliminin del vostre client abans que {1} pugui importar-les.",
|
||||
"AutoRedownloadFailedFromInteractiveSearchHelpText": "Cerqueu i intenteu baixar automàticament una versió diferent quan es trobi una versió fallida a la cerca interactiva",
|
||||
"FailedLoadingSearchResults": "No s'han pogut carregar els resultats de la cerca, torneu-ho a provar.",
|
||||
"IndexerFlags": "Indicadors de l'indexador",
|
||||
@@ -800,5 +800,327 @@
|
||||
"Install": "Instal·la",
|
||||
"PasswordConfirmation": "Confirmeu la contrasenya",
|
||||
"PreviouslyInstalled": "Instal·lat anteriorment",
|
||||
"AddNewItem": "Afegeix un nou element"
|
||||
"AddNewItem": "Afegeix un nou element",
|
||||
"ErrorLoadingContent": "S'ha produït un error en carregar aquest contingut",
|
||||
"DashOrSpaceDashDependingOnName": "Traç o guió d'espai depenent del nom",
|
||||
"EnabledHelpText": "Marqueu-ho per a habilitar el perfil de la versió",
|
||||
"IgnoreDownload": "Ignora la baixada",
|
||||
"IgnoreDownloadHint": "Atura {appName} de processar aquesta baixada més",
|
||||
"IgnoreDownloads": "Ignora les baixades",
|
||||
"IgnoreDownloadsHint": "Atura {appName} de processar aquestes baixades més",
|
||||
"IndexerIdHelpText": "Especifiqueu a quin indexador s'aplica el perfil",
|
||||
"LabelIsRequired": "L'etiqueta és necessària",
|
||||
"MediaManagementSettingsSummary": "Nomenat, configuració de la gestió de fitxers i carpetes arrel",
|
||||
"MetadataSource": "Font de les metadades",
|
||||
"NotificationsPlexSettingsAuthenticateWithPlexTv": "Autentica amb Plex.tv",
|
||||
"RemoveCompletedDownloads": "Elimina les baixades completes",
|
||||
"RemoveFailedDownloads": "Elimina les baixades fallides",
|
||||
"RemoveFromDownloadClientHint": "Elimina la baixada i el(s) fitxer(s) del client de baixada",
|
||||
"RemoveMultipleFromDownloadClientHint": "Elimina les baixades i els fitxers del client de baixada",
|
||||
"IndexerSettingsSeedTime": "Temps de la llavor",
|
||||
"IndexerSettingsSeedTimeHelpText": "El temps en què s'ha de sembrar un torrent abans d'aturar-lo, el buit utilitza el valor per defecte del client de baixada",
|
||||
"IndexerSettingsSeedRatio": "Ràtio de la llavor",
|
||||
"ResetQualityDefinitionsMessageText": "Esteu segur que voleu restablir les definicions de qualitat?",
|
||||
"SelectIndexerFlags": "Selecciona les banderes de l'indexador",
|
||||
"ShowBanners": "Mostra els bàners",
|
||||
"SkipRedownload": "Omet que es torni a descarregar",
|
||||
"SmartReplace": "Reemplaçament intel·ligent",
|
||||
"UseSSL": "Usa SSL",
|
||||
"WhySearchesCouldBeFailing": "Feu clic aquí per saber per què les cerques podrien estar fallant",
|
||||
"ThereWasAnErrorLoadingThisItem": "S'ha produït un error en carregar aquest element",
|
||||
"SearchMonitored": "Cerca monitorats",
|
||||
"DeleteSelected": "Suprimeix els seleccionats",
|
||||
"AllExpandedExpandAll": "Expandeix-ho tot",
|
||||
"CustomFormatsSettingsTriggerInfo": "Un format personalitzat s'aplicarà a un llançament o fitxer quan coincideixi almenys amb un de cada un dels diferents tipus de condició escollits.",
|
||||
"InstallMajorVersionUpdate": "Instal·la l'actualització",
|
||||
"InstallMajorVersionUpdateMessage": "Aquesta actualització instal·larà una nova versió principal i pot no ser compatible amb el vostre sistema. Esteu segur que voleu instal·lar aquesta actualització?",
|
||||
"NotificationsPlexSettingsAuthToken": "Testimoni d'autenticació",
|
||||
"Unknown": "Desconegut",
|
||||
"IsShowingMonitoredMonitorSelected": "Monitor seleccionat",
|
||||
"IsShowingMonitoredUnmonitorSelected": "Unmonitor seleccionat",
|
||||
"NotificationsSettingsUpdateMapPathsFromHelpText": "Camí de {appName}, utilitzat per modificar els camins de sèries quan {serviceName} veu la ubicació del camí de la biblioteca diferent de {appName} (requereix 'Biblioteca d'actualització')",
|
||||
"NotificationsSettingsUpdateMapPathsToHelpText": "Camí de {serviceName}, utilitzat per modificar els camins de sèries quan {serviceName} veu la ubicació del camí de la biblioteca diferent de {appName} (requereix 'Biblioteca d'actualització')",
|
||||
"NoMissingItems": "No falten elements",
|
||||
"InstallMajorVersionUpdateMessageLink": "Si us plau, comproveu [{domain}]({url}) per a més informació.",
|
||||
"RemoveQueueItemsRemovalMethodHelpTextWarning": "'Elimina del client de baixada' eliminarà les baixades i els fitxers del client de baixada.",
|
||||
"Monitoring": "Monitorant",
|
||||
"RemoveSelectedItemBlocklistMessageText": "Esteu segur que voleu eliminar els elements seleccionats de la llista de bloqueigs?",
|
||||
"NotificationsSettingsUseSslHelpText": "Connecta a {serviceName} a través d'HTTPS en lloc d'HTTP",
|
||||
"RemoveQueueItemRemovalMethod": "Mètode d'eliminació",
|
||||
"RemoveQueueItemRemovalMethodHelpTextWarning": "'Elimina des del client de baixada' eliminarà la baixada i el(s) fitxer(s) del client de baixada.",
|
||||
"LastSearched": "Darrera cerca",
|
||||
"NotificationsSettingsUpdateMapPathsFrom": "Mapear els camins des de",
|
||||
"NotificationsSettingsUpdateMapPathsTo": "Mapear els camins a",
|
||||
"RemoveQueueItem": "Elimina - {sourceTitle}",
|
||||
"SetIndexerFlags": "Estableix els indicadors de l'indexador",
|
||||
"NotificationsSettingsUpdateLibrary": "Actualitza la biblioteca",
|
||||
"IndexerSettingsSeedRatioHelpText": "Ràtio a la qual ha d'arribar un torrent abans d'aturar-se, buit utilitza el valor per defecte del client de baixada. La relació ha de ser com a mínim 1.0 i seguir les regles dels indexadors",
|
||||
"FailedToFetchSettings": "No s'ha pogut recuperar la configuració",
|
||||
"MonitoringOptions": "Opcions de monitoratge",
|
||||
"RootFolderPathHelpText": "Els elements de la llista del directori arrel s'afegiran a",
|
||||
"ThereWasAnErrorLoadingThisPage": "S'ha produït un error en carregar aquesta pàgina",
|
||||
"BookEditor": "Editor de llibres",
|
||||
"BookProgressBarText": "{bookCount} / {totalBookCount} (Files: {bookFileCount})",
|
||||
"ConsoleLogLevel": "Nivell de registre de la consola",
|
||||
"Country": "País",
|
||||
"DataFutureBooks": "Controla els llibres que encara no s'han publicat",
|
||||
"EmbedMetadataInBookFiles": "Incrusta les metadades als fitxers de llibre",
|
||||
"ForeignId": "ID estranger",
|
||||
"GoToAuthorListing": "Ves a la llista d'autors",
|
||||
"HasMonitoredBooksNoMonitoredBooksForThisAuthor": "No hi ha llibres supervisats per a aquest autor",
|
||||
"HideBooks": "Oculta els llibres",
|
||||
"ISBN": "ISBN",
|
||||
"IgnoreDeletedBooks": "Ignora els llibres eliminats",
|
||||
"IfYouDontAddAnImportListExclusionAndTheAuthorHasAMetadataProfileOtherThanNoneThenThisBookMayBeReaddedDuringTheNextAuthorRefresh": "Si no afegeixes una exclusió de llista d'importació i l'autor té un perfil de metadades distint de 'Cap', llavors aquest llibre pot ser de nou afegit durant el següent refresc d'autor.",
|
||||
"IgnoredMetaHelpText": "Els llibres s'ignoraran si contenen un o més termes (insensible a majúscules i minúscules)",
|
||||
"MassBookSearchWarning": "Esteu segur que voleu realitzar una cerca massiva de llibres per {0} llibres?",
|
||||
"InteractiveSearchModalHeaderBookAuthor": "Cerca interactiva - {bookTitle} per {authorName}",
|
||||
"IsCalibreLibraryHelpText": "Usa el servidor de contingut del Calibre per manipular la biblioteca",
|
||||
"IsExpandedHideBooks": "Oculta els llibres",
|
||||
"IsExpandedShowBooks": "Mostra els llibres",
|
||||
"IsInUseCantDeleteAQualityProfileThatIsAttachedToAnAuthorOrImportList": "No es pot suprimir un perfil de qualitat que està adjuntat a un autor o llista d'importació",
|
||||
"Iso639-3": "Codis de llengua ISO 639-3, o 'null', separats per comes",
|
||||
"LibraryHelpText": "Nom de la biblioteca del servidor de contingut del Calibre. Deixeu-ho en blanc per defecte.",
|
||||
"LogRotation": "Rotació del registre",
|
||||
"LogSqlHelpText": "Registra totes les consultes SQL des de Readarr",
|
||||
"MassBookSearch": "Cerca massiva de llibres",
|
||||
"MetadataConsumers": "Consumidors de metadades",
|
||||
"MetadataSourceHelpText": "Font alternativa de metadades (Deixa en blanc per defecte)",
|
||||
"BookNaming": "Nom del llibre",
|
||||
"CalibreLibrary": "Biblioteca del Calibre",
|
||||
"OnImportFailure": "En importar fallada",
|
||||
"OnAuthorAdded": "En afegir l'autor",
|
||||
"SendMetadataToCalibre": "Envia les metadades al Calibre",
|
||||
"SeriesNumber": "Número de sèrie",
|
||||
"SeriesTotal": "Sèries ({0})",
|
||||
"AllowFingerprintingHelpTextWarning": "Això requereix que el Readarr llegeixi parts del fitxer que alentiran els escanejos i poden causar una alta activitat de disc o de xarxa.",
|
||||
"AuthorProgressBarText": "{availableBookCount} / {bookCount} (Total: {totalBookCount}, Fitxers: {bookFileCount})",
|
||||
"AutomaticallySwitchEdition": "Canvia l'edició automàticament",
|
||||
"BackupIntervalHelpText": "Interval per a fer una còpia de seguretat de la base de dades de Readarr i de la configuració",
|
||||
"CalibreContentServer": "Servidor de contingut del Calibre",
|
||||
"CalibreContentServerText": "L'ús d'un Servidor de Contingut Calibre (no Calibre Web) permet a Readarr afegir llibres a la vostra biblioteca Calibre i activar les conversions entre formats",
|
||||
"CalibrePassword": "Contrasenya del Calibre",
|
||||
"CalibreUsername": "Nom d'usuari del Calibre",
|
||||
"CountIndexersSelected": "{selectedCount} indexador(s) seleccionat",
|
||||
"DataListMonitorAll": "Controla els autors i tots els llibres de cada autor inclosos a la llista d'importació",
|
||||
"DataMissingBooks": "Monitoritza els llibres que encara no tenen fitxers o que encara no s'han publicat",
|
||||
"DefaultMonitorOptionHelpText": "Quins llibres s'han de controlar en afegir-los inicialment per als autors detectats en aquesta carpeta",
|
||||
"DefaultQualityProfileIdHelpText": "Perfil de qualitat per defecte per als autors detectats en aquesta carpeta",
|
||||
"EditList": "Edita la llista",
|
||||
"EmbedMetadataHelpText": "Digues al Calibre que escrigui les metadades al fitxer de llibre real",
|
||||
"ExistingBooks": "Llibres existents",
|
||||
"ForeignIdHelpText": "L'identificador estranger de l'autor/llibre a excloure",
|
||||
"FutureDaysHelpText": "Dies per a l'alimentació iCal per mirar al futur",
|
||||
"ImportFailures": "Importa fallades",
|
||||
"ImportListSpecificSettings": "Importa la configuració específica de la llista",
|
||||
"ItsEasyToAddANewAuthorOrBookJustStartTypingTheNameOfTheItemYouWantToAdd": "És fàcil afegir un Autor nou o un Llibre simplement començar a escriure el nom de l'element que voleu afegir",
|
||||
"LatestBook": "Últim llibre",
|
||||
"MinPagesHelpText": "Ignora els llibres amb menys pàgines que això",
|
||||
"MinPopularityHelpText": "La popularitat és la qualificació mitjana * nombre de vots",
|
||||
"MinimumPopularity": "Popularitat mínima",
|
||||
"MissingBooks": "Llibres que falten",
|
||||
"MonitorAuthor": "Autor del monitor",
|
||||
"MonitorBook": "Llibre del monitor",
|
||||
"MonitorBookExistingOnlyWarning": "Aquest és un ajust ajustat de la configuració supervisada per a cada llibre. Utilitzeu l'opció d'Autor/Edita per controlar què passa amb els llibres acabats d'afegir",
|
||||
"MonitorNewBooks": "Monitora els llibres nous",
|
||||
"MonitorNewItemsHelpText": "Quins llibres nous s'han de controlar",
|
||||
"MonitoredAuthorIsMonitored": "L'autor està monitoritzat",
|
||||
"MonitoredAuthorIsUnmonitored": "L'autor no està monitoritzat",
|
||||
"MonitoringOptionsHelpText": "Quins llibres s'han de controlar després d'afegir l'autor (ajust d'un sol cop)",
|
||||
"MusicBrainzBookID": "ID del llibre MusicBrainz",
|
||||
"MusicBrainzReleaseID": "ID de llançament del MusicBrainz",
|
||||
"MusicBrainzTrackID": "ID de la pista MusicBrainz",
|
||||
"NameLastFirst": "Cognom Nom",
|
||||
"NewBooks": "Llibres nous",
|
||||
"NoName": "No mostris el nom",
|
||||
"OnAuthorAddedHelpText": "En afegir l'autor",
|
||||
"OnAuthorDelete": "En suprimir l'autor",
|
||||
"OnAuthorDeleteHelpText": "En suprimir l'autor",
|
||||
"OnBookDeleteHelpText": "En suprimir el llibre",
|
||||
"OnBookRetagHelpText": "En reetiquetar el llibre",
|
||||
"OnDownloadFailure": "A la fallada de baixada",
|
||||
"OnDownloadFailureHelpText": "A la fallada de baixada",
|
||||
"OnImportFailureHelpText": "En importar fallada",
|
||||
"OnReleaseImport": "En publicar la importació",
|
||||
"OnReleaseImportHelpText": "En publicar la importació",
|
||||
"OutputFormatHelpText": "Opcionalment, demaneu al Calibre que es converteixi a altres formats en importar. Llista separada per comes.",
|
||||
"PathHelpText": "Carpeta arrel que conté la biblioteca de llibres",
|
||||
"PathHelpTextWarning": "Això ha de ser diferent del directori on el vostre client de baixada posa fitxers",
|
||||
"PortHelpText": "Port del servidor de contingut del Calibre",
|
||||
"PreviewRetag": "Reetiqueta de la vista prèvia",
|
||||
"QualityProfileIdHelpText": "Els elements de la llista de perfils de qualitat s'han d'afegir amb",
|
||||
"ReadarrSupportsMultipleListsForImportingBooksAndAuthorsIntoTheDatabase": "Readarr admet múltiples llistes per importar llibres i autors a la base de dades.",
|
||||
"RecycleBinUnableToWriteHealthCheck": "No s'ha pogut escriure a la carpeta de contenidors de reciclatge configurada: {0}. Assegureu-vos que aquest camí existeix i que l'usuari que executa el Readarr pot escriure",
|
||||
"RefreshAuthor": "Actualitza l'autor",
|
||||
"RefreshBook": "Actualitza el llibre",
|
||||
"RefreshInformation": "Actualitza la informació",
|
||||
"RemotePathMappingsInfo": "Remote Path Mappings són molt rarament necessaris, si {app} i el vostre client de descàrrega estan en el mateix sistema, és millor que coincideixi amb els vostres camins. Per a més informació, vegeu el [wiki]({wikiLink}).",
|
||||
"SearchForMonitoredBooks": "Cerca llibres monitoritzats",
|
||||
"SearchForNewItems": "Cerca elements nous",
|
||||
"SelectBook": "Selecciona el llibre",
|
||||
"SelectEdition": "Selecciona l'edició",
|
||||
"ShouldMonitorExisting": "Controla els llibres existents",
|
||||
"ShouldSearchHelpText": "Cerca indexadors per als elements nous afegits. Utilitza amb precaució per a llistes grans.",
|
||||
"ShowBookCount": "Mostra el recompte de llibres",
|
||||
"ShowTitleHelpText": "Mostra el nom de l'autor sota el cartell",
|
||||
"StatusEndedDeceased": "Defunció",
|
||||
"TooManyBooks": "Falten o hi ha massa llibres? Modifica o crea un nou",
|
||||
"UnableToLoadMetadataProviderSettings": "No s'ha pogut carregar la configuració del proveïdor de metadades",
|
||||
"UrlBaseHelpText": "Afegeix un prefix a l'URL del Calibre, p. ex. http://[host]:[port]/[urlBase]",
|
||||
"ShouldMonitorExistingHelpText": "Controla automàticament els llibres d'aquesta llista que ja són a Readarr",
|
||||
"ShowBannersHelpText": "Mostra els bàners en lloc dels noms",
|
||||
"ShowLastBook": "Mostra l'últim llibre",
|
||||
"ShowName": "Mostra el nom",
|
||||
"SkipBooksWithNoISBNOrASIN": "Omet els llibres sense ISBN ni ASIN",
|
||||
"SkipPartBooksAndSets": "Omet els llibres de parts i els conjunts",
|
||||
"SkipSecondarySeriesBooks": "Omet els llibres de les sèries secundàries",
|
||||
"SpecificBook": "Llibre específic",
|
||||
"FileDetails": "Detalls del fitxer",
|
||||
"FilesTotal": "Fitxers ({0})",
|
||||
"FilterAnalyticsEvents": "Filtra els esdeveniments d'anàlisi",
|
||||
"FilterAuthor": "Filtra l'autor",
|
||||
"FilterSentryEventsHelpText": "Filtra els esdeveniments d'error d'usuari coneguts perquè s'enviïn com a Analytics",
|
||||
"IsExpandedShowFileInfo": "Mostra la informació del fitxer",
|
||||
"IsInUseCantDeleteAMetadataProfileThatIsAttachedToAnAuthorOrImportList": "No es pot suprimir un perfil de metadades que està adjuntat a un autor o a una llista d'importació",
|
||||
"PasswordHelpText": "Contrasenya del servidor de contingut del Calibre",
|
||||
"PastDays": "Dies passats",
|
||||
"WriteTagsAll": "Tots els fitxers; només importació inicial",
|
||||
"TagsHelpText": "Aplica als autors amb almenys una etiqueta coincident. Deixa en blanc per aplicar a tots els autors",
|
||||
"TotalBookCountBooksTotalBookFileCountBooksWithFilesInterp": "{0} llibres en total. {1} llibres amb fitxers.",
|
||||
"TrackNumber": "Número de pista",
|
||||
"TrackTitle": "Títol de la pista",
|
||||
"WatchLibraryForChangesHelpText": "Torna a explorar automàticament quan els fitxers canviïn en una carpeta arrel",
|
||||
"WatchRootFoldersForFileChanges": "Vigila les carpetes arrel per als canvis de fitxer",
|
||||
"WriteAudioTagsScrub": "Neteja les etiquetes existents",
|
||||
"WriteAudioTagsScrubHelp": "Elimina les etiquetes existents dels fitxers, deixant només les afegides pel Readarr.",
|
||||
"LogSQL": "Log SQL",
|
||||
"AnyEditionOkHelpText": "Readarr canviarà automàticament a l'edició que coincideixi millor amb els fitxers baixats",
|
||||
"AudioFileMetadata": "Escriu les metadades als fitxers d'àudio",
|
||||
"AuthorEditor": "Editor de l'autor",
|
||||
"AuthorFolderFormat": "Format de carpeta d'autor",
|
||||
"EntityName": "Nom de l'entitat",
|
||||
"FilterPlaceHolder": "Filtra el llibre",
|
||||
"UsernameHelpText": "Nom d'usuari del servidor de contingut del Calibre",
|
||||
"OnBookDelete": "En suprimir el llibre",
|
||||
"BypassIfHighestQualityHelpText": "Evita el retard quan el llançament tingui la qualitat més alta habilitada en el perfil de qualitat",
|
||||
"CountImportListsSelected": "{selectedCount} llista(es) d'importació seleccionada",
|
||||
"DataNone": "No es controlarà cap llibre",
|
||||
"IsExpandedHideFileInfo": "Amaga la informació del fitxer",
|
||||
"NoTagsHaveBeenAddedYet": "No s'han afegit etiquetes encara. Afegeix etiquetes per enllaçar autors amb perfils de retard, restriccions o notificacions. Feu clic a {0} per obtenir més informació sobre les etiquetes a Readarr.",
|
||||
"DeleteMetadataProfile": "Suprimeix el perfil de metadades",
|
||||
"FutureBooks": "Llibres futurs",
|
||||
"FutureDays": "Dies de futur",
|
||||
"MetadataProfileIdHelpText": "Els elements de la llista de perfils de metadades s'han d'afegir amb",
|
||||
"NameFirstLast": "Nom Cognom",
|
||||
"DownloadPropersAndRepacksHelpTexts2": "Usa 'No prefereixis' per ordenar per puntuació de paraules preferida sobre propers/repacks",
|
||||
"IndexerIdHelpTextWarning": "L'ús d'un indexador específic amb paraules preferides pot conduir a versions duplicades",
|
||||
"LogRotateHelpText": "Nombre màxim de fitxers de registre a mantenir desats a la carpeta de registres",
|
||||
"MetadataProviderSource": "Font del proveïdor de metadades",
|
||||
"MissingBooksAuthorMonitored": "Llibres que falten (controlat per l'autor)",
|
||||
"MissingBooksAuthorNotMonitored": "Llibres que falten (Autor no supervisat)",
|
||||
"MonitoredHelpText": "Readarr cercarà i descarregarà un llibre",
|
||||
"UseCalibreContentServer": "Usa el servidor de contingut del Calibre",
|
||||
"MonitorNewItems": "Monitora els llibres nous",
|
||||
"SearchForAllCutoffUnmetBooks": "Cerca tots els llibres retallats i no satisfets",
|
||||
"SearchForAllMissingBooks": "Cerca tots els llibres que falten",
|
||||
"TheFollowingFilesWillBeDeleted": "S'eliminaran els següents fitxers:",
|
||||
"TagsSettingsSummary": "Gestiona les etiquetes d'autor, perfil, restricció i notificació",
|
||||
"WriteAudioTags": "Etiqueta els fitxers d'àudio amb metadades",
|
||||
"AddImportListExclusionHelpText": "Evita que el llibre s'afegeixi al Readarr mitjançant la importació de llistes o l'actualització de l'autor",
|
||||
"BookStudio": "Book Studio",
|
||||
"BookTitle": "Títol del llibre",
|
||||
"Books": "Llibres",
|
||||
"DataListMonitorSpecificBook": "Autors de monitors, però només monitoritza els llibres inclosos explícitament a la llista",
|
||||
"OnBookTagUpdate": "En actualitzar l'etiqueta del llibre",
|
||||
"CalibreNotCalibreWeb": "Readarr pot interactuar amb el servidor de continguts de Calibre. No pot utilitzar Calibre-Web, que és programari no relacionat.",
|
||||
"ShouldMonitorHelpText": "Monitora els autors i llibres nous afegits d'aquesta llista",
|
||||
"MusicBrainzRecordingID": "ID d'enregistrament del MusicBrainz",
|
||||
"AllAuthorBooks": "Tots els autors",
|
||||
"EditionsHelpText": "Canvia l'edició d'aquest llibre",
|
||||
"BooksTotal": "Llibres ({0})",
|
||||
"CountAuthorsSelected": "{selectedCount} autors seleccionats",
|
||||
"DataNewAllBooks": "Monitora tots els llibres nous",
|
||||
"LoadingEditionsFailed": "Ha fallat la càrrega d'edicions",
|
||||
"MetadataSettingsSummary": "Crea fitxers de metadades quan s'importin llibres o s'actualitzi l'autor",
|
||||
"SearchBoxPlaceHolder": "P. ex. Guerra i pau, goodreads:656, isbn:067003469X, asin:B00JCDK5ME",
|
||||
"SkipRedownloadHelpText": "Evita que el Readarr intenti baixar versions alternatives per als elements eliminats",
|
||||
"CalibreMetadata": "Metadades del Calibre",
|
||||
"Bookshelf": "Prestatge",
|
||||
"DataListMonitorNone": "No vigilar autors ni llibres",
|
||||
"ImportListSettings": "Configuració general de la llista d'importació",
|
||||
"AuthorIndex": "Índex de l'autor",
|
||||
"AuthorNameHelpText": "El nom de l'autor/llibre a excloure (pot ser qualsevol cosa significativa)",
|
||||
"BookIndex": "Índex del llibre",
|
||||
"BookList": "Llista de llibres",
|
||||
"CalibreHost": "Amfitrió del Calibre",
|
||||
"CalibrePort": "Port del Calibre",
|
||||
"CalibreSettings": "Paràmetres del Calibre",
|
||||
"CalibreUrlBase": "Base d'url del Calibre",
|
||||
"UpdateCovers": "Actualitza les converses",
|
||||
"UpdateCoversHelpText": "Estableix les portades del llibre al Calibre perquè coincideixin amb les del Readarr",
|
||||
"UseSslHelpText": "Usa SSL per a connectar al servidor de contingut Calibre",
|
||||
"WriteBookTagsHelpTextWarning": "En seleccionar ‘Tots els fitxers’ s'alteraran els fitxers existents quan s'importin.",
|
||||
"WriteTagsNew": "Només per a baixades noves",
|
||||
"AddedAuthorSettings": "Configuració de l'autor afegit",
|
||||
"DefaultMetadataProfileIdHelpText": "Perfil predeterminat de metadades per als autors detectats en aquesta carpeta",
|
||||
"ContinuingMoreBooksAreExpected": "S'esperen més llibres",
|
||||
"DataNewBooks": "Supervisa els llibres nous publicats després del llibre més nou existent",
|
||||
"AllowAuthorChangeClickToChangeAuthor": "Feu clic per canviar l'autor",
|
||||
"AllowedLanguages": "Llengües permeses",
|
||||
"ContinuingNoAdditionalBooksAreExpected": "No s'espera cap llibre addicional",
|
||||
"DeleteBookFile": "Suprimeix el fitxer de llibre",
|
||||
"ConvertToFormat": "Converteix a format",
|
||||
"DataAllBooks": "Controla tots els llibres",
|
||||
"DataExistingBooks": "Controla els llibres que tenen fitxers o que encara no s'han publicat",
|
||||
"DeleteFormat": "Suprimeix el format",
|
||||
"DeleteFilesHelpText": "Suprimeix els fitxers del llibre i la carpeta de l'autor",
|
||||
"DiscNumber": "Número de disc",
|
||||
"EndedAllBooksDownloaded": "Ended (Tots els llibres baixats)",
|
||||
"FirstBook": "Primer llibre",
|
||||
"HostHelpText": "Servidor de contingut del Calibre",
|
||||
"MonitorExistingBooks": "Monitora els llibres existents",
|
||||
"MusicBrainzAuthorID": "ID de l'autor del MusicBrainz",
|
||||
"NameStyle": "Estil del nom de l'autor",
|
||||
"PastDaysHelpText": "Dies per a l'alimentació iCal per a mirar el passat",
|
||||
"DataFirstBook": "Fes el seguiment del primer llibre. Tots els altres llibres seran ignorats",
|
||||
"DataLatestBook": "Controla els últims llibres i futurs",
|
||||
"DataNewNone": "No vigila cap llibre nou",
|
||||
"DefaultReadarrTags": "Etiquetes del lector per defecte",
|
||||
"DefaultTagsHelpText": "Etiquetes de Readarr per defecte per als autors detectats en aquesta carpeta",
|
||||
"DiscCount": "Comptador de discs",
|
||||
"EditAuthor": "Edita l'autor",
|
||||
"EditBook": "Edita el llibre",
|
||||
"ExistingItems": "Elements existents",
|
||||
"ManualDownload": "Baixada manual",
|
||||
"MinimumPages": "Pàgines mínimes",
|
||||
"CollapseMultipleBooks": "Redueix diversos llibres",
|
||||
"CollapseMultipleBooksHelpText": "Redueix diversos llibres que es publiquen el mateix dia",
|
||||
"ContinuingAllBooksDownloaded": "Continuant (tots els llibres baixats)",
|
||||
"Development": "Desenvolupament",
|
||||
"EnableAutomaticAddHelpText": "Afegeix autor/llibres al Readarr quan es realitzin sincronitzacions a través de la IU o per Readarr",
|
||||
"RenameBooks": "Canvia el nom dels llibres",
|
||||
"SearchBook": "Cerca al llibre",
|
||||
"SelectedCountAuthorsSelectedInterp": "{0} Autor/s seleccionat/s",
|
||||
"SetReadarrTags": "Estableix les etiquetes del Readarr",
|
||||
"SkipBooksWithMissingReleaseDate": "Omet els llibres amb la data de publicació que manca",
|
||||
"TheBooksFilesWillBeDeleted": "S'eliminaran els fitxers del llibre.",
|
||||
"UpdatingIsDisabledInsideADockerContainerUpdateTheContainerImageInstead": "L'actualització està desactivada dins d'un contenidor d'acobladors. Actualitza la imatge del contenidor.",
|
||||
"WriteTagsSync": "Tots els fitxers; mantén la sincronització amb Goodreads",
|
||||
"ASIN": "ASIN",
|
||||
"BookAvailableButMissing": "Llibre disponible, però desaparegut",
|
||||
"CalibreOutputFormat": "Format de sortida del Calibre",
|
||||
"CalibreOutputProfile": "Perfil de sortida del Calibre",
|
||||
"BookFilesCountMessage": "No hi ha fitxers de llibre",
|
||||
"BookMonitoring": "Seguiment del llibre",
|
||||
"AllBooks": "Tots els llibres",
|
||||
"AllowFingerprinting": "Permet la impressió digital",
|
||||
"AllowFingerprintingHelpText": "Utilitza l'empremta digital per millorar la precisió de la coincidència de llibres",
|
||||
"ExistingTagsScrubbed": "Etiquetes existents rastrejades",
|
||||
"NETCore": ".NET Core",
|
||||
"WriteMetadataTags": "Escriu les etiquetes de les metadades",
|
||||
"AddNewAuthorRootFolderHelpText": "La subcarpeta '{folder}' es crearà automàticament",
|
||||
"AddRootFolder": "Afegeix una carpeta arrel",
|
||||
"Book": "Llibre",
|
||||
"AddNewBook": "Afegeix nou llibre",
|
||||
"AddNewAuthor": "Afegeix nou autor"
|
||||
}
|
||||
|
||||
@@ -755,5 +755,10 @@
|
||||
"CustomFormatsSpecificationRegularExpression": "Běžný výraz",
|
||||
"ChangeCategoryMultipleHint": "Změní stahování do kategorie „Post-Import“ z aplikace Download Client",
|
||||
"FailedToFetchSettings": "Nepodařilo se načíst nastavení",
|
||||
"NoCutoffUnmetItems": "Žádné neodpovídající nesplněné položky"
|
||||
"NoCutoffUnmetItems": "Žádné neodpovídající nesplněné položky",
|
||||
"DownloadClientDelugeSettingsDirectoryCompleted": "Adresář kam přesunout po dokončení",
|
||||
"DownloadClientDelugeSettingsDirectoryCompletedHelpText": "Nepovinné - umístění kam přesunout dokončená stahování, pokud ponecháte prázné, použije se výchozí umístění Deluge",
|
||||
"DownloadClientDelugeSettingsDirectoryHelpText": "Nepovinné - umístění stahovaných souborů, pokud ponecháte prázné, použije se výchozí umístění Deluge",
|
||||
"AddNewAuthorRootFolderHelpText": "Podsložka '{folder}' bude vytvořena automaticky",
|
||||
"AddRootFolder": "Přidat kořenový adresář"
|
||||
}
|
||||
|
||||
@@ -670,5 +670,8 @@
|
||||
"ExternalUpdater": "{appName} er konfigureret til at bruge en ekstern opdateringsmekanisme",
|
||||
"OnLatestVersion": "Den seneste version af {appName} er allerede installeret",
|
||||
"WouldYouLikeToRestoreBackup": "Vil du gendanne sikkerhedskopien »{name}«?",
|
||||
"MetadataProfile": "metadataprofil"
|
||||
"MetadataProfile": "metadataprofil",
|
||||
"Unknown": "Ukendt",
|
||||
"AddRootFolder": "Tilføj rodmappe",
|
||||
"AddNewAuthorRootFolderHelpText": "Undermappen '{0}' oprettes automatisk"
|
||||
}
|
||||
|
||||
@@ -1117,5 +1117,8 @@
|
||||
"OnAuthorAddedHelpText": "Beim Hinzufügen des Autors",
|
||||
"SelectBook": "Buch auswählen",
|
||||
"SelectEdition": "Wähle Edition",
|
||||
"LastSearched": "Letzte Suche"
|
||||
"LastSearched": "Letzte Suche",
|
||||
"Unknown": "Unbekannt",
|
||||
"AddRootFolder": "Stammverzeichnis hinzufügen",
|
||||
"AddNewAuthorRootFolderHelpText": "'{folder}' Unterordner wird automatisch erstellt werden"
|
||||
}
|
||||
|
||||
@@ -1020,5 +1020,7 @@
|
||||
"Script": "Γραφή",
|
||||
"UpdateAppDirectlyLoadError": "Δεν είναι δυνατή η απευθείας ενημέρωση του {appName},",
|
||||
"ExternalUpdater": "Το {appName} έχει ρυθμιστεί να χρησιμοποιεί έναν εξωτερικό μηχανισμό ενημέρωσης",
|
||||
"InstallLatest": "Εγκατάσταση πιο πρόσφατου"
|
||||
"InstallLatest": "Εγκατάσταση πιο πρόσφατου",
|
||||
"AddNewAuthorRootFolderHelpText": "Ο υποφάκελος \"{0}\" θα δημιουργηθεί αυτόματα",
|
||||
"AddRootFolder": "Προσθήκη φακέλου ρίζας"
|
||||
}
|
||||
|
||||
@@ -11,7 +11,11 @@
|
||||
"AddListExclusion": "Add List Exclusion",
|
||||
"AddMissing": "Add missing",
|
||||
"AddNew": "Add New",
|
||||
"AddNewBook": "Add New Book",
|
||||
"AddNewAuthor": "Add New Author",
|
||||
"AddNewAuthorRootFolderHelpText": "'{folder}' subfolder will be created automatically",
|
||||
"AddNewItem": "Add New Item",
|
||||
"AddRootFolder": "Add Root Folder",
|
||||
"AddedAuthorSettings": "Added Author Settings",
|
||||
"AddingTag": "Adding tag",
|
||||
"AgeWhenGrabbed": "Age (when grabbed)",
|
||||
@@ -1063,6 +1067,7 @@
|
||||
"UnableToLoadTheCalendar": "Unable to load the calendar",
|
||||
"UnableToLoadUISettings": "Unable to load UI settings",
|
||||
"Ungroup": "Ungroup",
|
||||
"Unknown": "Unknown",
|
||||
"UnmappedFiles": "Unmapped Files",
|
||||
"Unmonitored": "Unmonitored",
|
||||
"UnmonitoredHelpText": "Include unmonitored books in the iCal feed",
|
||||
|
||||
@@ -237,7 +237,7 @@
|
||||
"PosterSize": "Tamaño de póster",
|
||||
"PreviewRename": "Previsualizar renombrado",
|
||||
"Profiles": "Perfiles",
|
||||
"Proper": "Proper",
|
||||
"Proper": "Correcto",
|
||||
"PropersAndRepacks": "Propers y Repacks",
|
||||
"Protocol": "Protocolo",
|
||||
"ProtocolHelpText": "Elige qué protocolo(s) usar y cuál se prefiere cuando se elige entre lanzamientos equivalentes",
|
||||
@@ -423,7 +423,7 @@
|
||||
"UsenetDelay": "Retraso de usenet",
|
||||
"UsenetDelayHelpText": "Retraso en minutos a esperar antes de capturar un lanzamiento desde usenet",
|
||||
"Username": "Usuario",
|
||||
"BranchUpdate": "Rama a utilizar para actualizar Readarr",
|
||||
"BranchUpdate": "Rama a utilizar para actualizar {appName}",
|
||||
"BranchUpdateMechanism": "Rama usada por el mecanismo de actualización externo",
|
||||
"Version": "Versión",
|
||||
"WeekColumnHeader": "Cabecera de columna de semana",
|
||||
@@ -564,7 +564,7 @@
|
||||
"Label": "Etiqueta",
|
||||
"MissingFromDisk": "Readarr no pudo encontrar el archivo en el disco, por lo que el archivo fue desvinculado del libro en la base de datos",
|
||||
"RemotePathMappingCheckDownloadPermissions": "Readarr puede ver pero no acceder al libro descargado {0}. Probablemente sea un error de permisos.",
|
||||
"RemotePathMappingCheckFolderPermissions": "Readarr puede ver pero no acceder al directorio de descarga {0}. Probablemente sea un error de permisos.",
|
||||
"RemotePathMappingCheckFolderPermissions": "Readarr puede ver pero no acceder al directorio de descarga {1}. Probablemente sea un error de permisos.",
|
||||
"RemotePathMappingCheckFilesGenericPermissions": "El cliente de descarga {0} informó de la existencia de archivos en {1} pero Readarr no puede ver este directorio. Puede que tengas que ajustar los permisos de la carpeta.",
|
||||
"RemotePathMappingCheckGenericPermissions": "El cliente de descarga {0} coloca las descargas en {1} pero Readarr no puede ver este directorio. Puede que tengas que ajustar los permisos de la carpeta.",
|
||||
"RemotePathMappingCheckImportFailed": "Readarr falló al importar un libro. Comprueba tus registros para más detalles.",
|
||||
@@ -855,13 +855,13 @@
|
||||
"NotificationsSettingsUseSslHelpText": "Conectar a {serviceName} sobre HTTPS en vez de HTTP",
|
||||
"Rejections": "Rechazados",
|
||||
"SelectIndexerFlags": "Seleccionar indicadores del indexador",
|
||||
"RecycleBinUnableToWriteHealthCheck": "No se pudo escribir en la carpeta configurada de la papelera de reciclaje: {path}. Asegúrate de que esta ruta existe y es modificable por el usuario que ejecuta {appName}",
|
||||
"RecycleBinUnableToWriteHealthCheck": "No se pudo escribir en la carpeta configurada de la papelera de reciclaje: {0}. Asegúrate de que esta ruta existe y es modificable por el usuario que ejecuta Readarr",
|
||||
"SearchForAllMissingBooks": "Buscar todos los episodios perdidos",
|
||||
"IndexerIdHelpText": "Especifica a qué indexador se aplica el perfil",
|
||||
"ProfilesSettingsSummary": "Perfiles de calidad, de retraso de idioma y de lanzamiento",
|
||||
"DataExistingBooks": "Monitoriza libros que no tienen archivos o que no se han lanzado aún",
|
||||
"MonitoredAuthorIsMonitored": "El artista no está vigilado",
|
||||
"RemotePathMappingsInfo": "Los mapeos de ruta remota son muy raramente solicitados, si {appName} y tu cliente de descarga están en el mismo sistema es mejor coincidir sus rutas. Para más información consulta la [wiki]({wikiLink}).",
|
||||
"RemotePathMappingsInfo": "Los mapeos de ruta remota son muy raramente requeridos, si {app} y tu cliente de descarga están en el mismo sistema es mejor coincidir sus rutas. Para más información consulta la [wiki]({wikiLink}).",
|
||||
"ShowBannersHelpText": "Muestra carteles en lugar de nombres",
|
||||
"DefaultTagsHelpText": "Etiquetas predeterminadas de Readarr para los autores detectados en esta carpeta",
|
||||
"UseSSL": "Usar SSL",
|
||||
@@ -1117,5 +1117,10 @@
|
||||
"InstallMajorVersionUpdate": "Instalar actualización",
|
||||
"InstallMajorVersionUpdateMessage": "Esta actualización instalará una nueva versión principal y podría no ser compatible con tu sistema. ¿Estás seguro que quieres instalar esta actualización?",
|
||||
"InstallMajorVersionUpdateMessageLink": "Por favor revisa [{domain}]({url}) para más información.",
|
||||
"LastSearched": "Último buscado"
|
||||
"LastSearched": "Último buscado",
|
||||
"Unknown": "Desconocido",
|
||||
"AddNewBook": "Añadir Nuevo Libro",
|
||||
"AddNewAuthor": "Añadir Nuevo Autor",
|
||||
"AddNewAuthorRootFolderHelpText": "La subcarpeta '{folder}' será creada automáticamente",
|
||||
"AddRootFolder": "Añadir Carpeta Raíz"
|
||||
}
|
||||
|
||||
@@ -1117,5 +1117,10 @@
|
||||
"ConvertToFormat": "Muunna muotoon",
|
||||
"DownloadPropersAndRepacksHelpTexts2": "\"Älä suosi\" käyttää Proper-/Repack-julkaisujen sijaan haluttua sanapisteytystä.",
|
||||
"IgnoreDeletedBooks": "Ohita poistetut kirjat",
|
||||
"LastSearched": "Edellinen haku"
|
||||
"LastSearched": "Edellinen haku",
|
||||
"Unknown": "Tuntematon",
|
||||
"AddNewAuthorRootFolderHelpText": "Alikansio \"{folder}\" luodaan automaattisesti.",
|
||||
"AddRootFolder": "Lisää juurikansio",
|
||||
"AddNewBook": "Lisää uusi kirja",
|
||||
"AddNewAuthor": "Lisää uusi kirjailija"
|
||||
}
|
||||
|
||||
@@ -1117,5 +1117,8 @@
|
||||
"Script": "Script",
|
||||
"UpdateAppDirectlyLoadError": "Impossible de mettre à jour directement {appName},",
|
||||
"InstallMajorVersionUpdateMessage": "Cette mise à jour installera une nouvelle version majeure et pourrait ne pas être compatible avec votre système. Êtes-vous sûr de vouloir installer cette mise à jour ?",
|
||||
"LastSearched": "Dernière recherche"
|
||||
"LastSearched": "Dernière recherche",
|
||||
"Unknown": "Inconnu",
|
||||
"AddNewAuthorRootFolderHelpText": "Le sous-dossier « {folder} » sera créé automatiquement",
|
||||
"AddRootFolder": "Ajouter un dossier racine"
|
||||
}
|
||||
|
||||
@@ -687,5 +687,7 @@
|
||||
"Script": "תַסרִיט",
|
||||
"UnmappedFiles": "תיקיות לא ממופות",
|
||||
"UpdateAppDirectlyLoadError": "לא ניתן לעדכן את {appName} ישירות,",
|
||||
"Clone": "סגור"
|
||||
"Clone": "סגור",
|
||||
"AddNewAuthorRootFolderHelpText": "תיקיית המשנה '{0}' תיווצר באופן אוטומטי",
|
||||
"AddRootFolder": "הוסף תיקיית שורש"
|
||||
}
|
||||
|
||||
@@ -650,5 +650,7 @@
|
||||
"OnLatestVersion": "रेडर का नवीनतम संस्करण पहले से ही स्थापित है",
|
||||
"Script": "लिपि",
|
||||
"UpdateAppDirectlyLoadError": "सीधे {appName} अद्यतन करने में असमर्थ,",
|
||||
"Clone": "बंद करे"
|
||||
"Clone": "बंद करे",
|
||||
"AddNewAuthorRootFolderHelpText": "'{0}' सबफ़ोल्डर स्वचालित रूप से बनाया जाएगा",
|
||||
"AddRootFolder": "रूट फ़ोल्डर जोड़ें"
|
||||
}
|
||||
|
||||
@@ -229,5 +229,6 @@
|
||||
"DeleteSelectedBookFilesMessageText": "Jeste li sigurni da želite obrisati ovaj profil odgode?",
|
||||
"Clone": "Zatvori",
|
||||
"BuiltIn": "Ugrađeno",
|
||||
"AptUpdater": "Koristi apt kako bi instalirao ažuriranje"
|
||||
"AptUpdater": "Koristi apt kako bi instalirao ažuriranje",
|
||||
"AddRootFolder": "Dodaj Korijensku Mapu"
|
||||
}
|
||||
|
||||
@@ -1084,5 +1084,8 @@
|
||||
"OnLatestVersion": "A {appName} legújabb verziója már telepítva van",
|
||||
"PreviouslyInstalled": "Korábban telepítve",
|
||||
"Script": "Szkript",
|
||||
"UpdateAppDirectlyLoadError": "Nem lehetséges közvetlenül frissíteni a {appName}-t"
|
||||
"UpdateAppDirectlyLoadError": "Nem lehetséges közvetlenül frissíteni a {appName}-t",
|
||||
"Unknown": "Ismeretlen",
|
||||
"AddNewAuthorRootFolderHelpText": "A „{folder}” almappa automatikusan létrejön",
|
||||
"AddRootFolder": "Gyökérmappa hozzáadása"
|
||||
}
|
||||
|
||||
@@ -652,5 +652,7 @@
|
||||
"OnLatestVersion": "Nýjasta útgáfan af {appName} er þegar uppsett",
|
||||
"Script": "Handrit",
|
||||
"UnmappedFiles": "Ókortlagðar möppur",
|
||||
"UpdateAppDirectlyLoadError": "Ekki er hægt að uppfæra {appName} beint,"
|
||||
"UpdateAppDirectlyLoadError": "Ekki er hægt að uppfæra {appName} beint,",
|
||||
"AddNewAuthorRootFolderHelpText": "„{0}“ undirmöppan verður búin til sjálfkrafa",
|
||||
"AddRootFolder": "Bæta við rótarmöppu"
|
||||
}
|
||||
|
||||
@@ -725,7 +725,7 @@
|
||||
"AppUpdated": "{appName} Aggiornato",
|
||||
"AllResultsAreHiddenByTheAppliedFilter": "Tutti i risultati sono nascosti dal filtro applicato",
|
||||
"AutoRedownloadFailed": "Download fallito",
|
||||
"AddListExclusion": "Aggiungi elenco esclusioni",
|
||||
"AddListExclusion": "Aggiungi lista esclusioni",
|
||||
"Location": "Posizione",
|
||||
"ListsSettingsSummary": "Liste",
|
||||
"RecentChanges": "Cambiamenti Recenti",
|
||||
@@ -912,5 +912,10 @@
|
||||
"Script": "Script",
|
||||
"UpdateAppDirectlyLoadError": "Impossibile aggiornare {appName} direttamente,",
|
||||
"AutoRedownloadFailedFromInteractiveSearch": "Riesecuzione del download non riuscita dalla ricerca interattiva",
|
||||
"AutoRedownloadFailedFromInteractiveSearchHelpText": "Cerca automaticamente e tenta di scaricare una versione diversa quando il rilascio non riuscito è stato acquisito dalla ricerca interattiva"
|
||||
"AutoRedownloadFailedFromInteractiveSearchHelpText": "Cerca automaticamente e tenta di scaricare una versione diversa quando il rilascio non riuscito è stato acquisito dalla ricerca interattiva",
|
||||
"Unknown": "Sconosciuto",
|
||||
"AddNewAuthorRootFolderHelpText": "La sottocartella '{0}' verrà creata automaticamente",
|
||||
"AddRootFolder": "Aggiungi Cartella Radice",
|
||||
"AddNewBook": "Aggiungi nuovo libro",
|
||||
"AddNewAuthor": "Aggiungi nuovo autore"
|
||||
}
|
||||
|
||||
@@ -651,5 +651,7 @@
|
||||
"CurrentlyInstalled": "現在インストール中",
|
||||
"DockerUpdater": "Dockerコンテナを更新して、更新を受信します",
|
||||
"ExternalUpdater": "{appName}は、外部更新メカニズムを使用するように構成されています",
|
||||
"OnLatestVersion": "{appName}の最新バージョンはすでにインストールされています"
|
||||
"OnLatestVersion": "{appName}の最新バージョンはすでにインストールされています",
|
||||
"AddRootFolder": "ルートフォルダを追加する",
|
||||
"AddNewAuthorRootFolderHelpText": "'{0}'サブフォルダは自動的に作成されます"
|
||||
}
|
||||
|
||||
@@ -4,8 +4,8 @@
|
||||
"AlternateTitles": "대체 제목",
|
||||
"Analytics": "해석학",
|
||||
"DownloadPropersAndRepacksHelpTexts1": "Propers / Repacks로 자동 업그레이드할지 여부",
|
||||
"20MinutesTwenty": "120 분 : {0}",
|
||||
"45MinutesFourtyFive": "60 분 : {0}",
|
||||
"20MinutesTwenty": "20분 : {0}",
|
||||
"45MinutesFourtyFive": "45분 : {0}",
|
||||
"ChownGroupHelpText": "그룹 이름 또는 gid. 원격 파일 시스템에 gid를 사용하십시오.",
|
||||
"DeleteReleaseProfile": "지연 프로필 삭제",
|
||||
"DeleteIndexer": "인덱서 삭제",
|
||||
@@ -752,5 +752,35 @@
|
||||
"BypassIfAboveCustomFormatScore": "사용자 정의 형식 점수보다 높으면 무시",
|
||||
"ThemeHelpText": "애플리케이션 UI 테마 변경, '자동' 테마는 OS 테마를 사용하여 라이트 또는 다크 모드를 설정합니다. Theme.Park에서 영감을 받음",
|
||||
"ThereWasAnErrorLoadingThisPage": "이 페이지를 로드하는 중ㅇ 오류가 발생했습니다",
|
||||
"WhySearchesCouldBeFailing": "검색이 실패하는 이유를 알아보려면 여기를 클릭하세요"
|
||||
"WhySearchesCouldBeFailing": "검색이 실패하는 이유를 알아보려면 여기를 클릭하세요",
|
||||
"AllExpandedExpandAll": "모두 펼치기",
|
||||
"RootFolderPathHelpText": "루트 폴더 목록 항목이 다음에 추가됩니다:",
|
||||
"AllExpandedCollapseAll": "모두 접기",
|
||||
"IsShowingMonitoredMonitorSelected": "선택 모니터링 설정",
|
||||
"NotificationsSettingsUpdateMapPathsToHelpText": "{serviceName} 경로는 {serviceName}이 라이브러리 경로 위치를 {appName}와 다르게 볼 때 시리즈 경로를 수정하는 데 사용됨 (라이브러리 업데이트 필요)",
|
||||
"NotificationsSettingsUpdateMapPathsFromHelpText": "{appName} 경로는 {serviceName}이 라이브러리 경로 위치를 {appName}와 다르게 볼 때 시리즈 경로를 수정하는 데 사용됨 (라이브러리 업데이트 필요)",
|
||||
"InvalidUILanguage": "UI가 잘못된 언어로 설정되어 있습니다, 수정하고 설정을 저장하세요",
|
||||
"IndexerSettingsSeedRatio": "시드 비율",
|
||||
"EnabledHelpText": "출시 프로필을 활성화하려면 체크하세요",
|
||||
"ErrorLoadingContent": "이 콘텐트를 로드하는 중 오류가 발생했습니다",
|
||||
"NotificationsSettingsUpdateMapPathsFrom": "다음 위치부터 경로 매핑하기",
|
||||
"Unknown": "알 수 없음",
|
||||
"Negated": "부정",
|
||||
"NoDownloadClientsFound": "다운로드 클라이언트를 찾을 수 없음",
|
||||
"DownloadClientQbittorrentSettingsContentLayoutHelpText": "qBittorrent의 구성된 콘텐츠 레이아웃을 사용할지, 토런트의 원래 레이아웃을 사용할지, 항상 하위 폴더를 생성할지(qBittorrent 4.3.2+)",
|
||||
"FailedToFetchSettings": "설정을 가져오는데 실패함",
|
||||
"FailedToFetchUpdates": "업데이트를 가져오는데 실패함",
|
||||
"IndexerDownloadClientHelpText": "이 인덱서에서 가져온 것을 가져오는 데 사용되는 다운로드 클라이언트를 지정하세요",
|
||||
"IndexerSettingsSeedRatioHelpText": "토렌드가 멈추기 전에 도달해야 하는 비율, 비어 있을 경우 다운로드 클라이언트의 기본값을 사용합니다. 비율은 최소 1.0이어야 하며 인덱서 규칙을 따라야 합니다",
|
||||
"IndexerSettingsSeedTimeHelpText": "토렌드가 중지되기 전에 시드되어야 하는 시간, 비어 있을 경우 다운로드 클라이언트의 기본값을 사용합니다",
|
||||
"Monitoring": "모니터링 중",
|
||||
"NoEventsFound": "이벤트가 없음",
|
||||
"Other": "기타",
|
||||
"IgnoreDownloadHint": "{appName}가 이 다운로드를 더 이상 처리하지 못하도록 합니다",
|
||||
"NotificationsSettingsUpdateMapPathsTo": "지도 경로",
|
||||
"Series": "시리즈",
|
||||
"IgnoreDownloadsHint": "{appName}가 이러한 다운로드를 더 이상 처리하지 않도록 중지합니다",
|
||||
"InstallMajorVersionUpdateMessageLink": "상세 내용은 [{domain}]({url})을 확인하세요.",
|
||||
"AddNewAuthorRootFolderHelpText": "'{folder}' 하위 폴더가 자동으로 생성됩니다",
|
||||
"AddRootFolder": "루트 폴더 추가"
|
||||
}
|
||||
|
||||
@@ -206,5 +206,8 @@
|
||||
"FailedLoadingSearchResults": "Kunne ikke laste søkeresultat, vennligst prøv igjen.",
|
||||
"IgnoredPlaceHolder": "Legg til ny begrensning",
|
||||
"RequiredPlaceHolder": "Legg til ny begrensning",
|
||||
"UnableToAddANewRemotePathMappingPleaseTryAgain": "Kunne ikke legge til ny ekstern stimapping, vennligst prøv igjen."
|
||||
"UnableToAddANewRemotePathMappingPleaseTryAgain": "Kunne ikke legge til ny ekstern stimapping, vennligst prøv igjen.",
|
||||
"AddNewAuthorRootFolderHelpText": "Undermappa \"{folder}\" vil bli automatisk laget",
|
||||
"AddRootFolder": "Legg til rotmappe",
|
||||
"History": "Historikk"
|
||||
}
|
||||
|
||||
@@ -50,22 +50,22 @@
|
||||
"ChangeHasNotBeenSavedYet": "Wijziging is nog niet opgeslagen",
|
||||
"ChmodFolder": "chmod Map",
|
||||
"ChmodFolderHelpText": "Octaal, toegepast tijdens importeren/hernoemen op media mappen en bestanden (zonder uitvoeringsbits)",
|
||||
"ChmodFolderHelpTextWarning": "Dit werkt alleen als de gebruiker die {appName} draait de eigenaar is van het bestand. Het is beter om zeker te zijn dat de downloader de juiste rechten zet.",
|
||||
"ChownGroupHelpText": "Groep naam of gid. Gebruik gid voor externe bestandssystemen.",
|
||||
"ChownGroupHelpTextWarning": "Dit werkt alleen als de gebruiker die {appName} draait de eigenaar is van het bestand. Het is beter om zeker te zijn dat de downloader dezelfde groep gebruikt als {appName}.",
|
||||
"ChmodFolderHelpTextWarning": "Dit werkt alleen als de gebruiker die {appName} uitvoert de eigenaar is van het bestand. Het is beter om ervoor te zorgen dat de downloadclient de juiste rechten instelt.",
|
||||
"ChownGroupHelpText": "Groepsnaam of gid. Gebruik gid voor externe bestandssystemen.",
|
||||
"ChownGroupHelpTextWarning": "Dit werkt alleen als de gebruiker die {appName} uitvoert de eigenaar is van het bestand. Het is beter om ervoor te zorgen dat de downloadclient dezelfde groep gebruikt als {appName}.",
|
||||
"Clear": "Wis",
|
||||
"ClickToChangeQuality": "Klik om kwaliteit te wijzigen",
|
||||
"ClientPriority": "Client Prioriteit",
|
||||
"CloneIndexer": "Dupliceer Indexeerder",
|
||||
"CloneProfile": "Dupliceer Profiel",
|
||||
"Close": "Sluit",
|
||||
"ClientPriority": "Prioriteit Client",
|
||||
"CloneIndexer": "Indexeerder Dupliceren",
|
||||
"CloneProfile": "Profiel Dupliceren",
|
||||
"Close": "Sluiten",
|
||||
"Columns": "Kolommen",
|
||||
"CompletedDownloadHandling": "Voltooide Download Afhandeling",
|
||||
"ConnectSettings": "Connecties Instellingen",
|
||||
"Connections": "Connecties",
|
||||
"CopyUsingHardlinksHelpText": "Gebruik hardlinks bij het kopiëren van torrent bestanden die nog actief zijn",
|
||||
"CopyUsingHardlinksHelpTextWarning": "Sporadisch kan bestandsvergrendeling het hernoemen van in gebruik zijnde bestanden blokkeren. Als noodoplossing kunt u de hernoem functie van {appName} gebruiken na het opheffen van de bestandsvergrendeling.",
|
||||
"CreateEmptyAuthorFoldersHelpText": "Film mappen aanmaken voor ontbrekende films tijdens schijfscan",
|
||||
"CompletedDownloadHandling": "Afhandeling Voltooide Downloads",
|
||||
"ConnectSettings": "Instellingen Verbindingen",
|
||||
"Connections": "Verbindingen",
|
||||
"CopyUsingHardlinksHelpText": "Hardlinks staan {appName} toe om torrents te importeren die nog actief zijn zonder extra opslag in beslag te nemen of het volledige bestand te kopiëren. Hardlinks werken alleen maar als de bron en de bestemming zich op hetzelfde opslagvolume bevinden",
|
||||
"CopyUsingHardlinksHelpTextWarning": "Sporadisch kan bestandsvergrendeling het hernoemen van bestanden blokkeren die geseed worden. Als tijdelijke oplossing kan u seeden uitzetten en de hernoemfunctie van {appName} gebruiken.",
|
||||
"CreateEmptyAuthorFoldersHelpText": "Ontbrekende auteursmappen aanmaken tijdens schijfscan",
|
||||
"CreateGroup": "Groep aanmaken",
|
||||
"CutoffHelpText": "Wanneer deze kwaliteit is behaald, zal {appName} niet langer films downloaden",
|
||||
"CutoffUnmet": "Onbereikte Drempel",
|
||||
@@ -104,7 +104,7 @@
|
||||
"DetailedProgressBarHelpText": "Toon tekst op voortgangsbalk",
|
||||
"DiskSpace": "Schijfruimte",
|
||||
"Docker": "Docker",
|
||||
"DownloadClient": "Downloader",
|
||||
"DownloadClient": "Downloadclient",
|
||||
"DownloadClientSettings": "Downloader Instellingen",
|
||||
"DownloadClients": "Downloaders",
|
||||
"DownloadFailedCheckDownloadClientForMoreDetails": "Download mislukt: controleer de downloader voor meer details",
|
||||
@@ -435,7 +435,7 @@
|
||||
"ProxyPasswordHelpText": "Je moet alleen een gebruikersnaam en wachtwoord ingeven als dit vereist is, laat ze anders leeg.",
|
||||
"SslCertPathHelpTextWarning": "Herstarten vereist om in werking te treden",
|
||||
"UnableToLoadMetadataProfiles": "Vertragingsprofielen kunnen niet worden geladen",
|
||||
"DownloadClientCheckDownloadingToRoot": "Downloadclient {0} plaatst downloads in de hoofdmap {1}. U mag niet naar een hoofdmap downloaden.",
|
||||
"DownloadClientCheckDownloadingToRoot": "Downloadclient {0} plaatst downloads in de hoofdmap {1}. Het wordt afgeraden om naar een hoofdmap downloaden.",
|
||||
"MaintenanceRelease": "Onderhoudsuitgave",
|
||||
"Actions": "Acties",
|
||||
"Tomorrow": "Morgen",
|
||||
@@ -493,7 +493,7 @@
|
||||
"MissingFromDisk": "{appName} kon het bestand niet vinden op de schijf dus werd het verwijderd",
|
||||
"Disabled": "Uitgeschakeld",
|
||||
"IndexerPriorityHelpText": "Indexeerder Prioriteit van 1 (Hoogste) tot 50 (Laagste). Standaard: 25.",
|
||||
"ConnectSettingsSummary": "Notificaties, connecties met media servers/spelers en scripts",
|
||||
"ConnectSettingsSummary": "Meldingen, verbindingen met mediaservers/spelers en scripts",
|
||||
"DownloadClientCheckNoneAvailableMessage": "Er is geen downloader beschikbaar",
|
||||
"DownloadClientCheckUnableToCommunicateMessage": "Niet in staat om te communiceren met {0}.",
|
||||
"IndexersSettingsSummary": "Indexeerders en uitgave restricties",
|
||||
@@ -559,7 +559,7 @@
|
||||
"OnBookFileDeleteHelpText": "Op filmbestand verwijderen",
|
||||
"UpdateCheckStartupTranslocationMessage": "Kan de update niet installeren omdat de map '{0}' zich in een 'App Translocation' map bevindt.",
|
||||
"WriteTagsNo": "Nooit",
|
||||
"Connect": "Connecties",
|
||||
"Connect": "Verbindingen",
|
||||
"General": "Algemeen",
|
||||
"Lists": "Lijsten",
|
||||
"ProxyCheckBadRequestMessage": "Testen van proxy is mislukt. Statuscode: {0}",
|
||||
@@ -568,7 +568,7 @@
|
||||
"TimeLeft": "Resterende Tijd",
|
||||
"UpdateCheckStartupNotWritableMessage": "Kan de update niet installeren omdat de map '{0}' niet schrijfbaar is voor de gebruiker '{1}'.",
|
||||
"CouldntFindAnyResultsForTerm": "Kon geen resultaten vinden voor '{0}'",
|
||||
"CreateEmptyAuthorFolders": "Lege film mappen aanmaken",
|
||||
"CreateEmptyAuthorFolders": "Lege auteursmappen aanmaken",
|
||||
"FileWasDeletedByViaUI": "File werd verwijderd via de UI",
|
||||
"RestartRequiredHelpTextWarning": "Herstarten vereist om in werking te treden",
|
||||
"AddList": "Lijst Toevoegen",
|
||||
@@ -588,16 +588,16 @@
|
||||
"ApplicationUrlHelpText": "De externe URL van deze applicatie inclusief http(s)://,Port en URL base",
|
||||
"ImportListExclusions": "Verwijder van Uitzonderingenlijst",
|
||||
"MetadataProfile": "Metadata profiel toevoegen",
|
||||
"ChooseImportMethod": "Kies Importmodus",
|
||||
"ClickToChangeReleaseGroup": "Klik om de releasegroep te wijzigen",
|
||||
"ChooseImportMethod": "Kies Importmethode",
|
||||
"ClickToChangeReleaseGroup": "Klik om de uitgavegroep te wijzigen",
|
||||
"HardlinkCopyFiles": "Hardlink/Kopieer Bestanden",
|
||||
"OnApplicationUpdate": "Bij applicatie update",
|
||||
"MoveFiles": "Verplaats Bestanden",
|
||||
"OnApplicationUpdateHelpText": "Bij applicatie update",
|
||||
"BypassIfHighestQuality": "Omzeilen indien de hoogste kwaliteit",
|
||||
"CustomFormatScore": "Eigen Formaat Score",
|
||||
"CustomFormatScore": "Aangepaste Formaatscore",
|
||||
"MinimumCustomFormatScore": "Minimum Eigen Formaat Score",
|
||||
"Conditions": "Condities",
|
||||
"Conditions": "Voorwaarden",
|
||||
"DeleteCustomFormat": "Verwijder Eigen Formaat",
|
||||
"DeleteCustomFormatMessageText": "Bent u zeker dat u de indexeerder '{0}' wilt verwijderen?",
|
||||
"DeleteFormatMessageText": "Weet je zeker dat je formaat tag {0} wilt verwijderen?",
|
||||
@@ -607,12 +607,12 @@
|
||||
"NegateHelpText": "Indien aangevinkt, zal het eigen formaat niet worden toegepast indien deze {0} conditie overeenstemt.",
|
||||
"UnableToLoadCustomFormats": "Eigen formaten kunnen niet worden geladen",
|
||||
"UpgradesAllowed": "Upgrades toegestaan",
|
||||
"CloneCustomFormat": "Dupliceer Eigen Formaat",
|
||||
"CopyToClipboard": "Kopieer naar Klembord",
|
||||
"CustomFormat": "Eigen Formaat",
|
||||
"CustomFormatSettings": "Eigen Formaten Instellingen",
|
||||
"CustomFormats": "Eigen Formaten",
|
||||
"CutoffFormatScoreHelpText": "Wanneer deze eigen formaat score is behaald, zal {appName} niet langer films downloaden",
|
||||
"CloneCustomFormat": "Aangepast Formaat Dupliceren",
|
||||
"CopyToClipboard": "Kopiëren naar klembord",
|
||||
"CustomFormat": "Aangepast Formaat",
|
||||
"CustomFormatSettings": "Aangepaste Formaatinstellingen",
|
||||
"CustomFormats": "Aangepaste Formaten",
|
||||
"CutoffFormatScoreHelpText": "Wanneer deze aangepaste formaatscore is behaald, zal {appName} niet langer boeken downloaden",
|
||||
"IncludeCustomFormatWhenRenamingHelpText": "Voeg toe aan het {Eigen Formaten} hernoemingsformaat",
|
||||
"ImportListMissingRoot": "Ontbrekende hoofdmap voor importlijst(en): {0}",
|
||||
"ImportListMultipleMissingRoots": "Er ontbreken meerdere hoofdmappen voor importlijsten: {0}",
|
||||
@@ -635,7 +635,7 @@
|
||||
"ResetTitles": "Reset titels",
|
||||
"ApplyTagsHelpTextHowToApplyIndexers": "Hoe tags toepassen op de geselecteerde indexeerders",
|
||||
"AutomaticAdd": "Automatisch Toevoegen",
|
||||
"CloneCondition": "Kloon Conditie",
|
||||
"CloneCondition": "Voorwaarde dupliceren",
|
||||
"ApplyTagsHelpTextHowToApplyImportLists": "Hoe tags toepassen op de geselecteerde import lijsten",
|
||||
"ApplyChanges": "Pas Wijzigingen Toe",
|
||||
"ApplyTagsHelpTextAdd": "Toevoegen: Voeg de tags toe aan de bestaande tag lijst",
|
||||
@@ -662,7 +662,7 @@
|
||||
"No": "Nee",
|
||||
"NoChange": "Geen Wijziging",
|
||||
"ApplyTagsHelpTextHowToApplyAuthors": "Hoe tags toe te passen op de geselecteerd films",
|
||||
"CountDownloadClientsSelected": "{count} download client(s) geselecteerd",
|
||||
"CountDownloadClientsSelected": "{selectedCount} downloadclient(s) geselecteerd",
|
||||
"DeleteConditionMessageText": "Bent u zeker dat u de lijst '{name}' wilt verwijderen?",
|
||||
"FreeSpace": "Vrije Ruimte",
|
||||
"ImportLists": "importlijst",
|
||||
@@ -685,7 +685,7 @@
|
||||
"RemoveSelectedItem": "Verwijder geselecteerde item",
|
||||
"DeleteSelectedIndexers": "Verwijder Indexeerder",
|
||||
"ExistingTag": "Bestaande tag",
|
||||
"ConnectionLost": "Verbinding Onderbroken",
|
||||
"ConnectionLost": "Verbinding Verbroken",
|
||||
"LastDuration": "Laatste Looptijd",
|
||||
"Medium": "Gemiddeld",
|
||||
"NoEventsFound": "Geen gebeurtenissen gevonden",
|
||||
@@ -694,7 +694,7 @@
|
||||
"EnableRssHelpText": "Wordt gebruikt wanneer {appName} periodiek zoekt naar uitgaven via RSS synchronisatie",
|
||||
"Location": "Locatie",
|
||||
"RecentChanges": "Recente wijzigingen",
|
||||
"CustomFilter": "Aangepaste Filters",
|
||||
"CustomFilter": "Aangepaste Filter",
|
||||
"ErrorLoadingContent": "Er ging iets fout bij het laden van dit item",
|
||||
"SourceTitle": "Bron Titel",
|
||||
"FailedLoadingSearchResults": "Fout bij laden van zoek resultaten, probeer het opnieuw.",
|
||||
@@ -740,20 +740,20 @@
|
||||
"DisabledForLocalAddresses": "Uitgeschakeld voor lokale adressen",
|
||||
"Enabled": "Ingeschakeld",
|
||||
"ApiKey": "API-sleutel",
|
||||
"ClickToChangeIndexerFlags": "Klik om indexeringsvlaggen te wijzigen",
|
||||
"ClickToChangeIndexerFlags": "Klik om indexeerdersvlaggen te wijzigen",
|
||||
"CustomFormatsSpecificationFlag": "Vlag",
|
||||
"CustomFormatsSpecificationRegularExpression": "Reguliere expressie",
|
||||
"BlocklistOnlyHint": "Blokkeer lijst zonder te zoeken naar een vervanger",
|
||||
"BlocklistAndSearch": "Blokkeerlijst en zoeken",
|
||||
"BlocklistAndSearchHint": "Een vervanger zoeken na het blokkeren",
|
||||
"BlocklistAndSearchMultipleHint": "Zoekopdrachten voor vervangers starten na het blokkeren van de lijst",
|
||||
"CustomFormatsSettingsTriggerInfo": "Een Aangepast Formaat wordt toegepast op een uitgave of bestand als het overeenkomt met ten minste één van de verschillende condities die zijn gekozen.",
|
||||
"ConnectionSettingsUrlBaseHelpText": "Voegt een voorvoegsel toe aan de {connectionName} url, zoals {url}",
|
||||
"CustomFormatsSettingsTriggerInfo": "Een aangepast formaat wordt toegepast op een uitgave of bestand wanneer het overeenkomt met ten minste één van elk van de verschillende voorwaarden die zijn gekozen.",
|
||||
"ConnectionSettingsUrlBaseHelpText": "Voegt een voorvoegsel toe aan de {connectionName} URL, zoals {url}",
|
||||
"BlocklistMultipleOnlyHint": "Blocklist zonder te zoeken naar vervangers",
|
||||
"BlocklistOnly": "Alleen bloklijst",
|
||||
"ChangeCategoryHint": "Verandert download naar de 'Post-Import Categorie' van Downloadclient",
|
||||
"Clone": "Kloon",
|
||||
"CustomFormatsSpecificationRegularExpressionHelpText": "Aangepaste opmaak RegEx is hoofdletterongevoelig",
|
||||
"Clone": "Dupliceren",
|
||||
"CustomFormatsSpecificationRegularExpressionHelpText": "RegEx van aangepaste formaten is hoofdletterongevoelig",
|
||||
"ChangeCategoryMultipleHint": "Wijzigt downloads naar de 'Post-Import Categorie' van Downloadclient",
|
||||
"BypassIfAboveCustomFormatScore": "Omzeilen indien boven aangepaste opmaak score",
|
||||
"BypassIfAboveCustomFormatScoreHelpText": "Schakel omleiding in als de release een score heeft die hoger is dan de geconfigureerde minimale aangepaste formaatscore",
|
||||
@@ -767,5 +767,19 @@
|
||||
"Script": "Script",
|
||||
"UpdateAppDirectlyLoadError": "Kan {appName} niet rechtstreeks updaten,",
|
||||
"CurrentlyInstalled": "Momenteel Geïnstalleerd",
|
||||
"ExternalUpdater": "{appName} is geconfigureerd om een extern update mechanisme te gebruiken"
|
||||
"ExternalUpdater": "{appName} is geconfigureerd om een extern update mechanisme te gebruiken",
|
||||
"AddNewAuthorRootFolderHelpText": "'{0}' submap zal automatisch worden aangemaakt",
|
||||
"AddRootFolder": "Voeg hoofdmap toe",
|
||||
"CollapseMultipleBooks": "Meerdere Boeken Samenvouwen",
|
||||
"Country": "Land",
|
||||
"CollapseMultipleBooksHelpText": "Meerdere boeken die op dezelfde dag worden uitgegeven samenvouwen",
|
||||
"ConvertToFormat": "Converteren Naar Formaat",
|
||||
"ContinuingMoreBooksAreExpected": "Er worden meer boeken verwacht",
|
||||
"CountAuthorsSelected": "{selectedCount} auteur(s) geselecteerd",
|
||||
"ContinuingNoAdditionalBooksAreExpected": "Er worden geen nieuwe boeken verwacht",
|
||||
"CountImportListsSelected": "{selectedCount} importeerlijst(en) geselecteerd",
|
||||
"CountIndexersSelected": "{selectedCount} indexeerder(s) geselecteerd",
|
||||
"Continuing": "Doorgaan",
|
||||
"ContinuingAllBooksDownloaded": "Doorgaan (Alle boeken gedownload)",
|
||||
"StatusEndedContinuing": "Doorgaan"
|
||||
}
|
||||
|
||||
@@ -718,5 +718,7 @@
|
||||
"Script": "Scenariusz",
|
||||
"UnmappedFiles": "Niezmapowane foldery",
|
||||
"UpdateAppDirectlyLoadError": "Nie można bezpośrednio zaktualizować {appName},",
|
||||
"DockerUpdater": "zaktualizuj kontener Dockera, aby otrzymać aktualizację"
|
||||
"DockerUpdater": "zaktualizuj kontener Dockera, aby otrzymać aktualizację",
|
||||
"AddNewAuthorRootFolderHelpText": "Podfolder „{0}” zostanie utworzony automatycznie",
|
||||
"AddRootFolder": "Dodaj folder główny"
|
||||
}
|
||||
|
||||
@@ -963,5 +963,9 @@
|
||||
"DockerUpdater": "atualize o contentor do Docker para receber a atualização",
|
||||
"ExternalUpdater": "O {appName} está definido para usar um mecanismo de atualização externo",
|
||||
"Script": "Script",
|
||||
"UpdateAppDirectlyLoadError": "Não foi possível atualizar o {appName} diretamente,"
|
||||
"UpdateAppDirectlyLoadError": "Não foi possível atualizar o {appName} diretamente,",
|
||||
"AddNewAuthorRootFolderHelpText": "A subpasta \"{0}\" será criada automaticamente",
|
||||
"AddRootFolder": "Adicionar pasta raiz",
|
||||
"ThereWasAnErrorLoadingThisItem": "Houve um erro ao carregar este item",
|
||||
"ThereWasAnErrorLoadingThisPage": "Houve um erro ao carregar esta página"
|
||||
}
|
||||
|
||||
@@ -1117,5 +1117,10 @@
|
||||
"FailedToFetchUpdates": "Falha ao buscar atualizações",
|
||||
"InstallMajorVersionUpdateMessage": "Esta atualização instalará uma nova versão principal e pode não ser compatível com o seu sistema. Tem certeza de que deseja instalar esta atualização?",
|
||||
"InstallMajorVersionUpdateMessageLink": "Verifique [{domain}]({url}) para obter mais informações.",
|
||||
"LastSearched": "Última Pesquisa"
|
||||
"LastSearched": "Última Pesquisa",
|
||||
"Unknown": "Desconhecido",
|
||||
"AddRootFolder": "Adicionar pasta raiz",
|
||||
"AddNewAuthorRootFolderHelpText": "A subpasta \"{folder}\" será criada automaticamente",
|
||||
"AddNewBook": "Adicionar Novo Livro",
|
||||
"AddNewAuthor": "Adicionar Novo Autor"
|
||||
}
|
||||
|
||||
@@ -671,5 +671,8 @@
|
||||
"InstallLatest": "Instalați cel mai recent",
|
||||
"OnLatestVersion": "Cea mai recentă versiune a {appName} este deja instalată",
|
||||
"Script": "Script",
|
||||
"UpdateAppDirectlyLoadError": "Imposibil de actualizat direct {appName},"
|
||||
"UpdateAppDirectlyLoadError": "Imposibil de actualizat direct {appName},",
|
||||
"Unknown": "Necunoscut",
|
||||
"AddRootFolder": "Adăugați folderul rădăcină",
|
||||
"AddNewAuthorRootFolderHelpText": "Subfolderul „{0}” va fi creat automat"
|
||||
}
|
||||
|
||||
@@ -877,5 +877,12 @@
|
||||
"InstallMajorVersionUpdate": "Установить обновление",
|
||||
"InstallMajorVersionUpdateMessage": "Это обновление установит новую версию, которая может не поддерживаться вашей системой. Вы уверены, что хотите установить это обновление?",
|
||||
"InstallMajorVersionUpdateMessageLink": "Пожалуйста, проверьте [{domain}]({url}) для получения дополнительной информации.",
|
||||
"UpdateAppDirectlyLoadError": "Невозможно обновить {appName} напрямую,"
|
||||
"UpdateAppDirectlyLoadError": "Невозможно обновить {appName} напрямую,",
|
||||
"NoCutoffUnmetItems": "Нет элементов не достигших максимального качества",
|
||||
"DeleteSelected": "Удалить выбранные",
|
||||
"Unknown": "Неизвестный",
|
||||
"WhySearchesCouldBeFailing": "Нажмите здесь, чтобы найти причину ошибок писка",
|
||||
"LastSearched": "Искали недавно",
|
||||
"AddRootFolder": "Добавить корневой каталог",
|
||||
"AddNewAuthorRootFolderHelpText": "Подпапка \"{folder}\" будет создана автоматически"
|
||||
}
|
||||
|
||||
@@ -200,5 +200,9 @@
|
||||
"Reason": "Séria",
|
||||
"Clone": "Zatvoriť",
|
||||
"AptUpdater": "Použiť apt pre inštaláciu aktualizácie",
|
||||
"BuiltIn": "Vstavaný"
|
||||
"BuiltIn": "Vstavaný",
|
||||
"AddRootFolder": "Pridať koreňový priečinok",
|
||||
"AddNewAuthorRootFolderHelpText": "'{folder}' podpriečinok sa vytvorí automaticky",
|
||||
"IgnoredPlaceHolder": "Pridať nové obmedzenie",
|
||||
"RequiredPlaceHolder": "Pridať nové obmedzenie"
|
||||
}
|
||||
|
||||
@@ -882,5 +882,7 @@
|
||||
"InstallLatest": "Installera senaste",
|
||||
"OnLatestVersion": "Den senaste versionen av {appName} är redan installerad",
|
||||
"Script": "Skript",
|
||||
"UpdateAppDirectlyLoadError": "Det går inte att uppdatera {appName} direkt,"
|
||||
"UpdateAppDirectlyLoadError": "Det går inte att uppdatera {appName} direkt,",
|
||||
"AddNewAuthorRootFolderHelpText": "Mappen \"{folder}\" kommer skapas automatiskt",
|
||||
"AddRootFolder": "Lägg till rotmapp"
|
||||
}
|
||||
|
||||
@@ -653,5 +653,7 @@
|
||||
"UpdateAppDirectlyLoadError": "ไม่สามารถอัปเดต {appName} ได้โดยตรง",
|
||||
"UnmappedFiles": "โฟลเดอร์ที่ไม่ได้แมป",
|
||||
"AptUpdater": "ใช้ apt เพื่อติดตั้งการอัปเดต",
|
||||
"OnLatestVersion": "มีการติดตั้ง {appName} เวอร์ชันล่าสุดแล้ว"
|
||||
"OnLatestVersion": "มีการติดตั้ง {appName} เวอร์ชันล่าสุดแล้ว",
|
||||
"AddRootFolder": "เพิ่มโฟลเดอร์รูท",
|
||||
"AddNewAuthorRootFolderHelpText": "โฟลเดอร์ย่อย \"{0}\" จะถูกสร้างขึ้นโดยอัตโนมัติ"
|
||||
}
|
||||
|
||||
@@ -864,5 +864,8 @@
|
||||
"LogRotateHelpText": "Günlük klasöründe saklanacak maksimum günlük dosyası sayısı",
|
||||
"FilterAnalyticsEvents": "Analitik Olayları Filtrele",
|
||||
"ConsoleLogLevel": "Konsol Günlük Düzeyi",
|
||||
"LastSearched": "Son Aranan"
|
||||
"LastSearched": "Son Aranan",
|
||||
"Unknown": "Bilinmeyen",
|
||||
"AddNewAuthorRootFolderHelpText": "'{0}' alt klasörü otomatik olarak oluşturulacak",
|
||||
"AddRootFolder": "Kök Klasör Ekle"
|
||||
}
|
||||
|
||||
@@ -755,5 +755,110 @@
|
||||
"UpdateAppDirectlyLoadError": "Неможливо оновити {appName} безпосередньо,",
|
||||
"CurrentlyInstalled": "В даний час встановлено",
|
||||
"DockerUpdater": "Оновіть контейнер docker, щоб отримати оновлення",
|
||||
"OnLatestVersion": "Остання версія {appName} вже встановлена"
|
||||
"OnLatestVersion": "Остання версія {appName} вже встановлена",
|
||||
"CustomFormatsSpecificationRegularExpression": "Регулярний вираз",
|
||||
"ManageImportLists": "Керування списками імпорта",
|
||||
"ManageIndexers": "Керування індексаторами",
|
||||
"ErrorLoadingContent": "Сталася помилка при завантаженні цього вмісту",
|
||||
"DashOrSpaceDashDependingOnName": "Тире або пробіл залежно від імені",
|
||||
"EnabledHelpText": "Установіть прапорець, щоб увімкнути профіль релізу",
|
||||
"DeleteCondition": "Видалити умову",
|
||||
"NoCutoffUnmetItems": "Не має елементів що не досягли порогу",
|
||||
"ThereWasAnErrorLoadingThisItem": "Сталася помилка при завантаженні цього елемента",
|
||||
"SearchMonitored": "Шукати серіал",
|
||||
"SkipRedownload": "Пропустити повторне завантаження",
|
||||
"RemoveFailedDownloads": "Видалення невдалих завантажень",
|
||||
"SmartReplace": "Розумна заміна",
|
||||
"AllExpandedExpandAll": "Розгорнути все",
|
||||
"Author": "Автор",
|
||||
"DoNotBlocklistHint": "Видалити без внесення в чорний список",
|
||||
"External": "Зовнішній",
|
||||
"IgnoreDownloadHint": "Не дозволяє додатку {appName} продовжити обробку цього завантаження",
|
||||
"IgnoreDownloads": "Ігнорувати завантаження",
|
||||
"IndexerSettingsSeedTimeHelpText": "Час, протягом якого торрент має залишатися на роздачі перед зупинкою, якщо порожньо — використовується значення клієнта завантаження за замовчуванням",
|
||||
"IsShowingMonitoredMonitorSelected": "Відстеження вибрано",
|
||||
"IsShowingMonitoredUnmonitorSelected": "Не відстежувати вибрані",
|
||||
"LabelIsRequired": "Необхідна мітка",
|
||||
"ManageDownloadClients": "Керування клієнтами завантаження",
|
||||
"ManageLists": "Керування списками",
|
||||
"NotificationsPlexSettingsAuthToken": "Токен авторизації",
|
||||
"NotificationsSettingsUpdateMapPathsFrom": "Карта шляхів від",
|
||||
"NotificationsSettingsUseSslHelpText": "Підключайтеся до {serviceName} по протоколу HTTPS замість HTTP",
|
||||
"Other": "Інше",
|
||||
"Rejections": "Відмови",
|
||||
"EnableProfile": "Увімкнути профіль",
|
||||
"RemoveCompletedDownloads": "Видалити завершені завантаження",
|
||||
"RemoveFromDownloadClientHint": "Видаляє завантаження і файли з завантажувального клієнта",
|
||||
"RemoveMultipleFromDownloadClientHint": "Видаляє завантаження та файли з клієнта завантаження",
|
||||
"RemoveQueueItem": "Видалити - {sourceTitle}",
|
||||
"RemoveQueueItemRemovalMethod": "Метод видалення",
|
||||
"RemoveSelectedItemBlocklistMessageText": "Ви впевнені, що хочете видалити вибрані елементи з чорного списку?",
|
||||
"SelectIndexerFlags": "Вибрати прапорці індексатора",
|
||||
"ResetQualityDefinitionsMessageText": "Ви впевнені, що хочете скинути визначення якості??",
|
||||
"Unknown": "Невідомо",
|
||||
"AutoAdd": "Автоматичне додавання",
|
||||
"NotificationsSettingsUpdateMapPathsFromHelpText": "Шлях {appName}, який використовується для зміни шляхів до серіалів, коли {serviceName} бачить шлях до бібліотеки інакше, ніж {appName} (необхідно 'Оновити бібліотеку')",
|
||||
"ThereWasAnErrorLoadingThisPage": "Сталася помилка під час завантаження цієї сторінки",
|
||||
"DeleteRemotePathMappingMessageText": "Ви впевнені, що хочете видалити це зіставлення віддаленого шляху?",
|
||||
"IndexerSettingsSeedRatioHelpText": "Рейтинг, якого має досягти торрент перед зупинкою. Якщо порожньо — використовується значення за замовчуванням клієнта завантаження. Рейтинг має бути не менше 1,0 і відповідати правилам індексаторів",
|
||||
"MetadataSource": "Джерело метаданих",
|
||||
"RemoveQueueItemsRemovalMethodHelpTextWarning": "«Видалення з завантажувального клієнта» видалить завантаження та файли з завантажувального клієнта.",
|
||||
"ShowBanners": "Показувати банери",
|
||||
"WhySearchesCouldBeFailing": "Натисніть тут, щоб дізнатися, чому пошуки можуть не працювати",
|
||||
"UseSSL": "Використовувати SSL",
|
||||
"WouldYouLikeToRestoreBackup": "Бажаєте відновити резервну копію '{name}'?",
|
||||
"CustomFormatsSpecificationFlag": "Мітка",
|
||||
"IgnoreDownload": "Ігнорувати завантаження",
|
||||
"Implementation": "Реалізація",
|
||||
"ListRefreshInterval": "Інтервал оновлення списку",
|
||||
"NotificationsPlexSettingsAuthenticateWithPlexTv": "Аутентифікація через Plex.tv",
|
||||
"ManageClients": "Керування клієнтами",
|
||||
"Loading": "Завантаження",
|
||||
"EditSelectedIndexers": "Редагувати вибраний індексатор",
|
||||
"DoNotBlocklist": "Не додавати до чорного списку",
|
||||
"EditSelectedDownloadClients": "Редагувати вибрані клієнти завантаження",
|
||||
"EditSelectedImportLists": "Редагувати вибрані списки імпорту",
|
||||
"DownloadClientQbittorrentSettingsContentLayout": "Макет контента",
|
||||
"IndexerIdHelpText": "Вкажіть, до якого індексатору застосовується профіль",
|
||||
"InvalidUILanguage": "У вашому інтерфейсі встановлена недопустима мова. Виправте її та збережіть налаштування",
|
||||
"MediaManagementSettingsSummary": "Налаштування іменування, управління файлами та кореневі папки",
|
||||
"IndexerSettingsSeedRatio": "Коефіцієнт роздачі",
|
||||
"IndexerSettingsSeedTime": "Час сидіння",
|
||||
"NotificationsSettingsUpdateMapPathsToHelpText": "Шлях {serviceName}, що використовується для зміни шляхів до серіалів, коли {serviceName} бачить шлях до бібліотеки інакше, ніж {appName} (потрібно 'Оновити бібліотеку')",
|
||||
"NotificationsSettingsUpdateLibrary": "Оновити бібліотеку",
|
||||
"RemoveQueueItemRemovalMethodHelpTextWarning": "«Видалення з завантажувального клієнта» видалить завантаження і файли з завантажувального клієнта.",
|
||||
"RootFolderPathHelpText": "Елементи списку кореневих тек будуть додані в",
|
||||
"SelectDropdown": "Вибрати...",
|
||||
"DeleteSelected": "Видалити вибрані",
|
||||
"DownloadClientDelugeSettingsDirectory": "Тека завантаження",
|
||||
"DownloadClientDelugeSettingsDirectoryCompleted": "Перемістити теку після завершення",
|
||||
"FailedToFetchSettings": "Не вдалося отримати налаштування",
|
||||
"FailedToFetchUpdates": "Не вдалося завантажити оновлення",
|
||||
"IgnoreDownloadsHint": "Не дозволяє додатку {appName} обробляти ці завантаження",
|
||||
"Install": "Встановити",
|
||||
"InstallMajorVersionUpdate": "Встановити оновлення",
|
||||
"InstallMajorVersionUpdateMessage": "Це оновлення встановить нову основну версію і може бути несумісним з вашою системою. Ви впевнені, що хочете встановити це оновлення?",
|
||||
"InstallMajorVersionUpdateMessageLink": "Будь ласка, перевірте [{domain}]({url}) для отримання додаткової інформації.",
|
||||
"Label": "Мітка",
|
||||
"LastSearched": "Останній пошук",
|
||||
"Monitoring": "Відстежування",
|
||||
"MonitoringOptions": "Опції відстеження",
|
||||
"No": "Ні",
|
||||
"NoDownloadClientsFound": "Клієнти завантаження не знайдено",
|
||||
"NoImportListsFound": "Списки імпорта не знайдено",
|
||||
"NoIndexersFound": "Индексаторі не знайдено",
|
||||
"NoMissingItems": "Немає відсутніх елементів",
|
||||
"NotificationsSettingsUpdateMapPathsTo": "Карта шляхів до",
|
||||
"PasswordConfirmation": "Підтвердження пароля",
|
||||
"PreviouslyInstalled": "Раніше встановлений",
|
||||
"SetIndexerFlags": "Встановити прапорці індексатора",
|
||||
"AddRootFolder": "Додати корневий каталог",
|
||||
"AddNewAuthorRootFolderHelpText": "Вкладена папка \"{0}\" буде створена автоматично",
|
||||
"WriteBookTagsHelpTextWarning": "Вибір \"Усі файли\" змінить існуючі файли під час їх імпорту.",
|
||||
"ForeignId": "Зовнішній ідентифікатор",
|
||||
"AudioFileMetadata": "Записувати метадані до аудіофайлів",
|
||||
"OnDownloadFailureHelpText": "При помилці завантаження",
|
||||
"WriteAudioTags": "Тегувати аудіофайли метаданими",
|
||||
"SearchForNewItems": "Пошук нових елементів",
|
||||
"WriteAudioTagsScrub": "Очистити існуючі теги"
|
||||
}
|
||||
|
||||
@@ -657,5 +657,8 @@
|
||||
"UpdateAppDirectlyLoadError": "Không thể cập nhật {appName} trực tiếp,",
|
||||
"AptUpdater": "Sử dụng apt để cài đặt bản cập nhật",
|
||||
"ExternalUpdater": "{appName} được định cấu hình để sử dụng cơ chế cập nhật bên ngoài",
|
||||
"WhySearchesCouldBeFailing": "Nhấp vào đây để tìm hiểu lý do tại sao tìm kiếm thất bại"
|
||||
"WhySearchesCouldBeFailing": "Nhấp vào đây để tìm hiểu lý do tại sao tìm kiếm thất bại",
|
||||
"Unknown": "Không rõ",
|
||||
"AddNewAuthorRootFolderHelpText": "Thư mục con '{0}' sẽ được tạo tự động",
|
||||
"AddRootFolder": "Thêm thư mục gốc"
|
||||
}
|
||||
|
||||
@@ -1117,5 +1117,8 @@
|
||||
"PreviouslyInstalled": "上次安装",
|
||||
"UpdateAppDirectlyLoadError": "无法直接更新{appName},",
|
||||
"FailedToFetchSettings": "设置同步失败",
|
||||
"LastSearched": "最近搜索"
|
||||
"LastSearched": "最近搜索",
|
||||
"Unknown": "未知",
|
||||
"AddNewAuthorRootFolderHelpText": "将自动创建 '{folder}' 子文件夹",
|
||||
"AddRootFolder": "添加根目录"
|
||||
}
|
||||
|
||||
@@ -2,5 +2,6 @@
|
||||
"Analytics": "分析",
|
||||
"About": "关于",
|
||||
"Username": "用户名",
|
||||
"Activity": "111"
|
||||
"Activity": "111",
|
||||
"Actions": "Actions"
|
||||
}
|
||||
|
||||
@@ -171,5 +171,6 @@
|
||||
"IgnoredPlaceHolder": "加入新的限制",
|
||||
"RequiredPlaceHolder": "加入新的限制",
|
||||
"RedownloadFailed": "失敗時重新下載",
|
||||
"UnableToAddANewRemotePathMappingPleaseTryAgain": "無法加入新的遠程路徑對應,請重試。"
|
||||
"UnableToAddANewRemotePathMappingPleaseTryAgain": "無法加入新的遠程路徑對應,請重試。",
|
||||
"AddRootFolder": "加入根目錄資料夾"
|
||||
}
|
||||
|
||||
@@ -30,13 +30,11 @@ namespace NzbDrone.Core.MediaCover
|
||||
private string _url;
|
||||
public string Url
|
||||
{
|
||||
get
|
||||
{
|
||||
return _url;
|
||||
}
|
||||
get => _url;
|
||||
set
|
||||
{
|
||||
_url = value;
|
||||
|
||||
if (Extension.IsNullOrWhiteSpace())
|
||||
{
|
||||
Extension = Path.GetExtension(value);
|
||||
@@ -46,6 +44,7 @@ namespace NzbDrone.Core.MediaCover
|
||||
|
||||
public MediaCoverTypes CoverType { get; set; }
|
||||
public string Extension { get; private set; }
|
||||
public string RemoteUrl { get; set; }
|
||||
|
||||
public MediaCover()
|
||||
{
|
||||
|
||||
70
src/NzbDrone.Core/MediaCover/MediaCoverProxy.cs
Normal file
70
src/NzbDrone.Core/MediaCover/MediaCoverProxy.cs
Normal file
@@ -0,0 +1,70 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using NzbDrone.Common.Cache;
|
||||
using NzbDrone.Common.Extensions;
|
||||
using NzbDrone.Common.Http;
|
||||
using NzbDrone.Core.Configuration;
|
||||
|
||||
namespace NzbDrone.Core.MediaCover
|
||||
{
|
||||
public interface IMediaCoverProxy
|
||||
{
|
||||
string RegisterUrl(string url);
|
||||
|
||||
string GetUrl(string hash);
|
||||
byte[] GetImage(string hash);
|
||||
}
|
||||
|
||||
public class MediaCoverProxy : IMediaCoverProxy
|
||||
{
|
||||
private readonly IHttpClient _httpClient;
|
||||
private readonly IConfigFileProvider _configFileProvider;
|
||||
private readonly ICached<string> _cache;
|
||||
|
||||
public MediaCoverProxy(IHttpClient httpClient, IConfigFileProvider configFileProvider, ICacheManager cacheManager)
|
||||
{
|
||||
_httpClient = httpClient;
|
||||
_configFileProvider = configFileProvider;
|
||||
_cache = cacheManager.GetCache<string>(GetType());
|
||||
}
|
||||
|
||||
public string RegisterUrl(string url)
|
||||
{
|
||||
if (url.IsNullOrWhiteSpace())
|
||||
{
|
||||
return null;
|
||||
}
|
||||
|
||||
var hash = url.SHA256Hash();
|
||||
|
||||
_cache.Set(hash, url, TimeSpan.FromHours(24));
|
||||
|
||||
_cache.ClearExpired();
|
||||
|
||||
var fileName = Path.GetFileName(url);
|
||||
return _configFileProvider.UrlBase + @"/MediaCoverProxy/" + hash + "/" + fileName;
|
||||
}
|
||||
|
||||
public string GetUrl(string hash)
|
||||
{
|
||||
var result = _cache.Find(hash);
|
||||
|
||||
if (result == null)
|
||||
{
|
||||
throw new KeyNotFoundException("Url no longer in cache");
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public byte[] GetImage(string hash)
|
||||
{
|
||||
var url = GetUrl(hash);
|
||||
|
||||
var request = new HttpRequest(url);
|
||||
|
||||
return _httpClient.Get(request).ResponseData;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -31,6 +31,7 @@ namespace NzbDrone.Core.MediaCover
|
||||
{
|
||||
private const string USER_AGENT = "Dalvik/2.1.0 (Linux; U; Android 10; SM-G975U Build/QP1A.190711.020)";
|
||||
|
||||
private readonly IMediaCoverProxy _mediaCoverProxy;
|
||||
private readonly IImageResizer _resizer;
|
||||
private readonly IBookService _bookService;
|
||||
private readonly IHttpClient _httpClient;
|
||||
@@ -46,7 +47,8 @@ namespace NzbDrone.Core.MediaCover
|
||||
// So limit the number of concurrent resizing tasks
|
||||
private static SemaphoreSlim _semaphore = new SemaphoreSlim((int)Math.Ceiling(Environment.ProcessorCount / 2.0));
|
||||
|
||||
public MediaCoverService(IImageResizer resizer,
|
||||
public MediaCoverService(IMediaCoverProxy mediaCoverProxy,
|
||||
IImageResizer resizer,
|
||||
IBookService bookService,
|
||||
IHttpClient httpClient,
|
||||
IDiskProvider diskProvider,
|
||||
@@ -56,6 +58,7 @@ namespace NzbDrone.Core.MediaCover
|
||||
IEventAggregator eventAggregator,
|
||||
Logger logger)
|
||||
{
|
||||
_mediaCoverProxy = mediaCoverProxy;
|
||||
_resizer = resizer;
|
||||
_bookService = bookService;
|
||||
_httpClient = httpClient;
|
||||
@@ -82,28 +85,42 @@ namespace NzbDrone.Core.MediaCover
|
||||
|
||||
public void ConvertToLocalUrls(int entityId, MediaCoverEntity coverEntity, IEnumerable<MediaCover> covers)
|
||||
{
|
||||
foreach (var mediaCover in covers)
|
||||
if (entityId == 0)
|
||||
{
|
||||
if (mediaCover.CoverType == MediaCoverTypes.Unknown)
|
||||
// Author isn't in Readarr yet, map via a proxy to circument referrer issues
|
||||
foreach (var mediaCover in covers)
|
||||
{
|
||||
continue;
|
||||
mediaCover.RemoteUrl = mediaCover.Url;
|
||||
mediaCover.Url = _mediaCoverProxy.RegisterUrl(mediaCover.RemoteUrl);
|
||||
}
|
||||
|
||||
var filePath = GetCoverPath(entityId, coverEntity, mediaCover.CoverType, mediaCover.Extension, null);
|
||||
|
||||
if (coverEntity == MediaCoverEntity.Book)
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach (var mediaCover in covers)
|
||||
{
|
||||
mediaCover.Url = _configFileProvider.UrlBase + @"/MediaCover/Books/" + entityId + "/" + mediaCover.CoverType.ToString().ToLower() + GetExtension(mediaCover.CoverType, mediaCover.Extension);
|
||||
}
|
||||
else
|
||||
{
|
||||
mediaCover.Url = _configFileProvider.UrlBase + @"/MediaCover/" + entityId + "/" + mediaCover.CoverType.ToString().ToLower() + GetExtension(mediaCover.CoverType, mediaCover.Extension);
|
||||
}
|
||||
if (mediaCover.CoverType == MediaCoverTypes.Unknown)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
if (_diskProvider.FileExists(filePath))
|
||||
{
|
||||
var lastWrite = _diskProvider.FileGetLastWrite(filePath);
|
||||
mediaCover.Url += "?lastWrite=" + lastWrite.Ticks;
|
||||
var filePath = GetCoverPath(entityId, coverEntity, mediaCover.CoverType, mediaCover.Extension, null);
|
||||
|
||||
mediaCover.RemoteUrl = mediaCover.Url;
|
||||
|
||||
if (coverEntity == MediaCoverEntity.Book)
|
||||
{
|
||||
mediaCover.Url = _configFileProvider.UrlBase + @"/MediaCover/Books/" + entityId + "/" + mediaCover.CoverType.ToString().ToLower() + GetExtension(mediaCover.CoverType, mediaCover.Extension);
|
||||
}
|
||||
else
|
||||
{
|
||||
mediaCover.Url = _configFileProvider.UrlBase + @"/MediaCover/" + entityId + "/" + mediaCover.CoverType.ToString().ToLower() + GetExtension(mediaCover.CoverType, mediaCover.Extension);
|
||||
}
|
||||
|
||||
if (_diskProvider.FileExists(filePath))
|
||||
{
|
||||
var lastWrite = _diskProvider.FileGetLastWrite(filePath);
|
||||
mediaCover.Url += "?lastWrite=" + lastWrite.Ticks;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -97,7 +97,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
var iDecision = 1;
|
||||
foreach (var bookDecision in bookDecisions)
|
||||
{
|
||||
_logger.ProgressInfo($"Importing book {iDecision++}/{bookDecisions.Count} {bookDecision.First().Item.Book}");
|
||||
_logger.ProgressInfo("Importing book {0}/{1} {2}", iDecision++, bookDecisions.Count, bookDecision.First().Item.Book);
|
||||
|
||||
var decisionList = bookDecision.ToList();
|
||||
|
||||
@@ -130,7 +130,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
// RemoveExistingTrackFiles(author, book);
|
||||
// }
|
||||
|
||||
// make sure part numbers are populated for audio books
|
||||
// Make sure part numbers are populated for audiobooks
|
||||
// If all audio files and all part numbers are zero, set them by filename order
|
||||
if (decisionList.All(b => MediaFileExtensions.AudioExtensions.Contains(Path.GetExtension(b.Item.Path)) && b.Item.Part == 0))
|
||||
{
|
||||
@@ -147,7 +147,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
book.Editions = _editionService.SetMonitored(newRelease);
|
||||
|
||||
// Publish book edited event.
|
||||
// Deliberatly don't put in the old book since we don't want to trigger an AuthorScan.
|
||||
// Deliberately don't put in the old book since we don't want to trigger an AuthorScan.
|
||||
_eventAggregator.PublishEvent(new BookEditedEvent(book, book));
|
||||
}
|
||||
|
||||
@@ -158,8 +158,8 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
.SelectMany(c => c)
|
||||
.ToList();
|
||||
|
||||
_logger.ProgressInfo($"Importing {qualifiedImports.Count} files");
|
||||
_logger.Debug($"Importing {qualifiedImports.Count} files. replaceExisting: {replaceExisting}");
|
||||
_logger.ProgressInfo("Importing {0} files", qualifiedImports.Count);
|
||||
_logger.Debug("Importing {0} files. Replace existing: {1}", qualifiedImports.Count, replaceExisting);
|
||||
|
||||
var filesToAdd = new List<BookFile>(qualifiedImports.Count);
|
||||
var trackImportedEvents = new List<TrackImportedEvent>(qualifiedImports.Count);
|
||||
@@ -310,7 +310,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
var watch = new System.Diagnostics.Stopwatch();
|
||||
watch.Start();
|
||||
_mediaFileService.AddMany(filesToAdd);
|
||||
_logger.Debug($"Inserted new trackfiles in {watch.ElapsedMilliseconds}ms");
|
||||
_logger.Debug("Inserted new trackfiles in {0}ms", watch.ElapsedMilliseconds);
|
||||
|
||||
// now that trackfiles have been inserted and ids generated, publish the import events
|
||||
foreach (var trackImportedEvent in trackImportedEvents)
|
||||
@@ -354,7 +354,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
|
||||
if (booksToRefresh.Any())
|
||||
{
|
||||
_logger.Debug($"Refreshing info for {booksToRefresh.Count} new books");
|
||||
_logger.Debug("Refreshing info for {0} new books", booksToRefresh.Count);
|
||||
_commandQueueManager.Push(new BulkRefreshBookCommand(booksToRefresh.Select(x => x.Id).ToList()));
|
||||
}
|
||||
|
||||
@@ -371,7 +371,8 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
|
||||
if (dbAuthor == null)
|
||||
{
|
||||
_logger.Debug($"Adding remote author {author}");
|
||||
_logger.Debug("Adding remote author {0}", author);
|
||||
|
||||
var path = decisions.First().Item.Path;
|
||||
var rootFolder = _rootFolderService.GetBestRootFolder(path);
|
||||
|
||||
@@ -438,7 +439,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
|
||||
if (dbBook == null)
|
||||
{
|
||||
_logger.Debug($"Adding remote book {book}");
|
||||
_logger.Debug("Adding remote book {0}", book);
|
||||
|
||||
if (book.AuthorMetadataId == 0)
|
||||
{
|
||||
@@ -497,7 +498,8 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
|
||||
if (dbEdition == null)
|
||||
{
|
||||
_logger.Debug($"Adding remote edition {edition}");
|
||||
_logger.Debug("Adding remote edition {0}", edition);
|
||||
|
||||
try
|
||||
{
|
||||
edition.BookId = book.Id;
|
||||
@@ -540,7 +542,7 @@ namespace NzbDrone.Core.MediaFiles.BookImport
|
||||
var rootFolder = _diskProvider.GetParentFolder(author.Path);
|
||||
var previousFiles = _mediaFileService.GetFilesByBook(book.Id);
|
||||
|
||||
_logger.Debug($"Deleting {previousFiles.Count} existing files for {book}");
|
||||
_logger.Debug("Deleting {0} existing files for {1}", previousFiles.Count, book);
|
||||
|
||||
foreach (var previousFile in previousFiles)
|
||||
{
|
||||
|
||||
@@ -103,8 +103,8 @@ namespace NzbDrone.Core.MetadataSource.BookInfo
|
||||
}
|
||||
catch (BookInfoException e)
|
||||
{
|
||||
_logger.Warn(e, "Unexpected error getting author info");
|
||||
throw new AuthorNotFoundException(foreignAuthorId);
|
||||
_logger.Warn(e, "Unexpected error getting author info: {foreignAuthorId}", foreignAuthorId);
|
||||
throw;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -126,8 +126,8 @@ namespace NzbDrone.Core.MetadataSource.BookInfo
|
||||
}
|
||||
catch (BookInfoException e)
|
||||
{
|
||||
_logger.Warn(e, "Unexpected error getting book info");
|
||||
throw new BookNotFoundException(foreignBookId);
|
||||
_logger.Warn(e, "Unexpected error getting book info: {foreignBookId}", foreignBookId);
|
||||
throw;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -430,7 +430,7 @@ namespace NzbDrone.Core.MetadataSource.BookInfo
|
||||
{
|
||||
var author = PollAuthor(newId);
|
||||
|
||||
book = author.Books.Value.Where(b => b.Editions.Value.Any(e => e.ForeignEditionId == id.ToString())).FirstOrDefault();
|
||||
book = author.Books.Value.FirstOrDefault(b => b.Editions.Value.Any(e => e.ForeignEditionId == id.ToString()));
|
||||
authors = new List<AuthorMetadata> { author.Metadata.Value };
|
||||
}
|
||||
else if (type == "work")
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user