mirror of
https://github.com/lidarr/lidarr.git
synced 2025-07-06 04:52:21 -07:00
Fix GitIgnore, Update from Sonarr
This commit is contained in:
parent
77f1d2e64c
commit
22d9c5e666
17 changed files with 603 additions and 51 deletions
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -120,8 +120,6 @@ _tests/
|
||||||
setup/Output/
|
setup/Output/
|
||||||
*.~is
|
*.~is
|
||||||
|
|
||||||
UI/
|
|
||||||
|
|
||||||
#VS outout folders
|
#VS outout folders
|
||||||
bin
|
bin
|
||||||
obj
|
obj
|
||||||
|
|
|
@ -20,7 +20,7 @@ import QueueConnector from 'Activity/Queue/QueueConnector';
|
||||||
import BlacklistConnector from 'Activity/Blacklist/BlacklistConnector';
|
import BlacklistConnector from 'Activity/Blacklist/BlacklistConnector';
|
||||||
import MissingConnector from 'Wanted/Missing/MissingConnector';
|
import MissingConnector from 'Wanted/Missing/MissingConnector';
|
||||||
import CutoffUnmetConnector from 'Wanted/CutoffUnmet/CutoffUnmetConnector';
|
import CutoffUnmetConnector from 'Wanted/CutoffUnmet/CutoffUnmetConnector';
|
||||||
import UISettingsConnector from 'Settings/UI/UISettingsConnector';
|
import Settings from 'Settings/Settings';
|
||||||
import MediaManagementConnector from 'Settings/MediaManagement/MediaManagementConnector';
|
import MediaManagementConnector from 'Settings/MediaManagement/MediaManagementConnector';
|
||||||
import Profiles from 'Settings/Profiles/Profiles';
|
import Profiles from 'Settings/Profiles/Profiles';
|
||||||
import Quality from 'Settings/Quality/Quality';
|
import Quality from 'Settings/Quality/Quality';
|
||||||
|
@ -29,6 +29,7 @@ import DownloadClientSettings from 'Settings/DownloadClients/DownloadClientSetti
|
||||||
import NotificationSettings from 'Settings/Notifications/NotificationSettings';
|
import NotificationSettings from 'Settings/Notifications/NotificationSettings';
|
||||||
import MetadataSettings from 'Settings/Metadata/MetadataSettings';
|
import MetadataSettings from 'Settings/Metadata/MetadataSettings';
|
||||||
import GeneralSettingsConnector from 'Settings/General/GeneralSettingsConnector';
|
import GeneralSettingsConnector from 'Settings/General/GeneralSettingsConnector';
|
||||||
|
import UISettingsConnector from 'Settings/UI/UISettingsConnector';
|
||||||
import Status from 'System/Status/Status';
|
import Status from 'System/Status/Status';
|
||||||
import TasksConnector from 'System/Tasks/TasksConnector';
|
import TasksConnector from 'System/Tasks/TasksConnector';
|
||||||
import BackupsConnector from 'System/Backup/BackupsConnector';
|
import BackupsConnector from 'System/Backup/BackupsConnector';
|
||||||
|
@ -142,8 +143,9 @@ function App({ store, history }) {
|
||||||
*/}
|
*/}
|
||||||
|
|
||||||
<Route
|
<Route
|
||||||
path="/settings/ui"
|
exact={true}
|
||||||
component={UISettingsConnector}
|
path="/settings"
|
||||||
|
component={Settings}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Route
|
<Route
|
||||||
|
@ -186,6 +188,11 @@ function App({ store, history }) {
|
||||||
component={GeneralSettingsConnector}
|
component={GeneralSettingsConnector}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Route
|
||||||
|
path="/settings/ui"
|
||||||
|
component={UISettingsConnector}
|
||||||
|
/>
|
||||||
|
|
||||||
{/*
|
{/*
|
||||||
System
|
System
|
||||||
*/}
|
*/}
|
||||||
|
|
|
@ -89,12 +89,8 @@ const links = [
|
||||||
{
|
{
|
||||||
iconName: icons.SETTINGS,
|
iconName: icons.SETTINGS,
|
||||||
title: 'Settings',
|
title: 'Settings',
|
||||||
to: '/settings/ui',
|
to: '/settings',
|
||||||
children: [
|
children: [
|
||||||
{
|
|
||||||
title: 'UI',
|
|
||||||
to: '/settings/ui'
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
title: 'Media Management',
|
title: 'Media Management',
|
||||||
to: '/settings/mediamanagement'
|
to: '/settings/mediamanagement'
|
||||||
|
@ -126,6 +122,10 @@ const links = [
|
||||||
{
|
{
|
||||||
title: 'General',
|
title: 'General',
|
||||||
to: '/settings/general'
|
to: '/settings/general'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'UI',
|
||||||
|
to: '/settings/ui'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
|
@ -2,11 +2,24 @@ import PropTypes from 'prop-types';
|
||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { clearPendingChanges } from 'Store/Actions/baseActions';
|
import { clearPendingChanges } from 'Store/Actions/baseActions';
|
||||||
|
import { cancelTestDownloadClient, cancelSaveDownloadClient } from 'Store/Actions/settingsActions';
|
||||||
import EditDownloadClientModal from './EditDownloadClientModal';
|
import EditDownloadClientModal from './EditDownloadClientModal';
|
||||||
|
|
||||||
const mapDispatchToProps = {
|
function createMapDispatchToProps(dispatch, props) {
|
||||||
clearPendingChanges
|
return {
|
||||||
};
|
dispatchClearPendingChanges() {
|
||||||
|
dispatch(clearPendingChanges);
|
||||||
|
},
|
||||||
|
|
||||||
|
dispatchCancelTestDownloadClient() {
|
||||||
|
dispatch(cancelTestDownloadClient);
|
||||||
|
},
|
||||||
|
|
||||||
|
dispatchCancelSaveDownloadClient() {
|
||||||
|
dispatch(cancelSaveDownloadClient);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
class EditDownloadClientModalConnector extends Component {
|
class EditDownloadClientModalConnector extends Component {
|
||||||
|
|
||||||
|
@ -14,7 +27,9 @@ class EditDownloadClientModalConnector extends Component {
|
||||||
// Listeners
|
// Listeners
|
||||||
|
|
||||||
onModalClose = () => {
|
onModalClose = () => {
|
||||||
this.props.clearPendingChanges({ section: 'downloadClients' });
|
this.props.dispatchClearPendingChanges({ section: 'downloadClients' });
|
||||||
|
this.props.dispatchCancelTestDownloadClient({ section: 'downloadClients' });
|
||||||
|
this.props.dispatchCancelSaveDownloadClient({ section: 'downloadClients' });
|
||||||
this.props.onModalClose();
|
this.props.onModalClose();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,9 +37,16 @@ class EditDownloadClientModalConnector extends Component {
|
||||||
// Render
|
// Render
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
const {
|
||||||
|
dispatchClearPendingChanges,
|
||||||
|
dispatchCancelTestDownloadClient,
|
||||||
|
dispatchCancelSaveDownloadClient,
|
||||||
|
...otherProps
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<EditDownloadClientModal
|
<EditDownloadClientModal
|
||||||
{...this.props}
|
{...otherProps}
|
||||||
onModalClose={this.onModalClose}
|
onModalClose={this.onModalClose}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -33,7 +55,9 @@ class EditDownloadClientModalConnector extends Component {
|
||||||
|
|
||||||
EditDownloadClientModalConnector.propTypes = {
|
EditDownloadClientModalConnector.propTypes = {
|
||||||
onModalClose: PropTypes.func.isRequired,
|
onModalClose: PropTypes.func.isRequired,
|
||||||
clearPendingChanges: PropTypes.func.isRequired
|
dispatchClearPendingChanges: PropTypes.func.isRequired,
|
||||||
|
dispatchCancelTestDownloadClient: PropTypes.func.isRequired,
|
||||||
|
dispatchCancelSaveDownloadClient: PropTypes.func.isRequired
|
||||||
};
|
};
|
||||||
|
|
||||||
export default connect(null, mapDispatchToProps)(EditDownloadClientModalConnector);
|
export default connect(null, createMapDispatchToProps)(EditDownloadClientModalConnector);
|
||||||
|
|
|
@ -2,11 +2,24 @@ import PropTypes from 'prop-types';
|
||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { clearPendingChanges } from 'Store/Actions/baseActions';
|
import { clearPendingChanges } from 'Store/Actions/baseActions';
|
||||||
|
import { cancelTestIndexer, cancelSaveIndexer } from 'Store/Actions/settingsActions';
|
||||||
import EditIndexerModal from './EditIndexerModal';
|
import EditIndexerModal from './EditIndexerModal';
|
||||||
|
|
||||||
const mapDispatchToProps = {
|
function createMapDispatchToProps(dispatch, props) {
|
||||||
clearPendingChanges
|
return {
|
||||||
};
|
dispatchClearPendingChanges() {
|
||||||
|
dispatch(clearPendingChanges);
|
||||||
|
},
|
||||||
|
|
||||||
|
dispatchCancelTestIndexer() {
|
||||||
|
dispatch(cancelTestIndexer);
|
||||||
|
},
|
||||||
|
|
||||||
|
dispatchCancelSaveIndexer() {
|
||||||
|
dispatch(cancelSaveIndexer);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
class EditIndexerModalConnector extends Component {
|
class EditIndexerModalConnector extends Component {
|
||||||
|
|
||||||
|
@ -14,7 +27,9 @@ class EditIndexerModalConnector extends Component {
|
||||||
// Listeners
|
// Listeners
|
||||||
|
|
||||||
onModalClose = () => {
|
onModalClose = () => {
|
||||||
this.props.clearPendingChanges({ section: 'indexers' });
|
this.props.dispatchClearPendingChanges({ section: 'indexers' });
|
||||||
|
this.props.dispatchCancelTestIndexer({ section: 'indexers' });
|
||||||
|
this.props.dispatchCancelSaveIndexer({ section: 'indexers' });
|
||||||
this.props.onModalClose();
|
this.props.onModalClose();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,9 +37,16 @@ class EditIndexerModalConnector extends Component {
|
||||||
// Render
|
// Render
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
const {
|
||||||
|
dispatchClearPendingChanges,
|
||||||
|
dispatchCancelTestIndexer,
|
||||||
|
dispatchCancelSaveIndexer,
|
||||||
|
...otherProps
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<EditIndexerModal
|
<EditIndexerModal
|
||||||
{...this.props}
|
{...otherProps}
|
||||||
onModalClose={this.onModalClose}
|
onModalClose={this.onModalClose}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -33,7 +55,9 @@ class EditIndexerModalConnector extends Component {
|
||||||
|
|
||||||
EditIndexerModalConnector.propTypes = {
|
EditIndexerModalConnector.propTypes = {
|
||||||
onModalClose: PropTypes.func.isRequired,
|
onModalClose: PropTypes.func.isRequired,
|
||||||
clearPendingChanges: PropTypes.func.isRequired
|
dispatchClearPendingChanges: PropTypes.func.isRequired,
|
||||||
|
dispatchCancelTestIndexer: PropTypes.func.isRequired,
|
||||||
|
dispatchCancelSaveIndexer: PropTypes.func.isRequired
|
||||||
};
|
};
|
||||||
|
|
||||||
export default connect(null, mapDispatchToProps)(EditIndexerModalConnector);
|
export default connect(null, createMapDispatchToProps)(EditIndexerModalConnector);
|
||||||
|
|
18
frontend/src/Settings/Settings.css
Normal file
18
frontend/src/Settings/Settings.css
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
.link {
|
||||||
|
composes: link from 'Components/Link/Link.css';
|
||||||
|
|
||||||
|
border-bottom: 1px solid #e5e5e5;
|
||||||
|
color: #3a3f51;
|
||||||
|
font-size: 21px;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: #616573;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.summary {
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
color: $dimColor;
|
||||||
|
}
|
122
frontend/src/Settings/Settings.js
Normal file
122
frontend/src/Settings/Settings.js
Normal file
|
@ -0,0 +1,122 @@
|
||||||
|
import React from 'react';
|
||||||
|
import Link from 'Components/Link/Link';
|
||||||
|
import PageContent from 'Components/Page/PageContent';
|
||||||
|
import PageContentBodyConnector from 'Components/Page/PageContentBodyConnector';
|
||||||
|
import SettingsToolbarConnector from './SettingsToolbarConnector';
|
||||||
|
import styles from './Settings.css';
|
||||||
|
|
||||||
|
function Settings() {
|
||||||
|
return (
|
||||||
|
<PageContent title="Settings">
|
||||||
|
<SettingsToolbarConnector
|
||||||
|
hasPendingChanges={false}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<PageContentBodyConnector>
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/mediamanagement"
|
||||||
|
>
|
||||||
|
Media Management
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Naming and file management settings
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/profiles"
|
||||||
|
>
|
||||||
|
Profiles
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Quality, Language and Delay profiles
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/quality"
|
||||||
|
>
|
||||||
|
Quality
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Quality sizes and naming
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/indexers"
|
||||||
|
>
|
||||||
|
Indexers
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Indexers and release restrictions
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/downloadclients"
|
||||||
|
>
|
||||||
|
Download Clients
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Download clients, download handling and remote path mappings
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/connect"
|
||||||
|
>
|
||||||
|
Connect
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Notifications, connections to media servers/players and custom scripts
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/metadata"
|
||||||
|
>
|
||||||
|
Metadata
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Create metadata files when episodes are imported or series are refreshed
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/general"
|
||||||
|
>
|
||||||
|
General
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Port, SSL, username/password, proxy, analytics and updates
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
className={styles.link}
|
||||||
|
to="/settings/ui"
|
||||||
|
>
|
||||||
|
UI
|
||||||
|
</Link>
|
||||||
|
|
||||||
|
<div className={styles.summary}>
|
||||||
|
Calendar, date and color impaired options
|
||||||
|
</div>
|
||||||
|
</PageContentBodyConnector>
|
||||||
|
</PageContent>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Settings.propTypes = {
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Settings;
|
197
frontend/src/Settings/UI/UISettings.js
Normal file
197
frontend/src/Settings/UI/UISettings.js
Normal file
|
@ -0,0 +1,197 @@
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
import { inputTypes } from 'Helpers/Props';
|
||||||
|
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
||||||
|
import FieldSet from 'Components/FieldSet';
|
||||||
|
import PageContent from 'Components/Page/PageContent';
|
||||||
|
import PageContentBodyConnector from 'Components/Page/PageContentBodyConnector';
|
||||||
|
import SettingsToolbarConnector from 'Settings/SettingsToolbarConnector';
|
||||||
|
import Form from 'Components/Form/Form';
|
||||||
|
import FormGroup from 'Components/Form/FormGroup';
|
||||||
|
import FormLabel from 'Components/Form/FormLabel';
|
||||||
|
import FormInputGroup from 'Components/Form/FormInputGroup';
|
||||||
|
|
||||||
|
class UISettings extends Component {
|
||||||
|
|
||||||
|
//
|
||||||
|
// Render
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const {
|
||||||
|
isFetching,
|
||||||
|
error,
|
||||||
|
settings,
|
||||||
|
hasSettings,
|
||||||
|
onInputChange,
|
||||||
|
onSavePress,
|
||||||
|
...otherProps
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
|
const firstDayOfWeekOptions = [
|
||||||
|
{ key: 0, value: 'Sunday' },
|
||||||
|
{ key: 1, value: 'Monday' }
|
||||||
|
];
|
||||||
|
|
||||||
|
const weekColumnOptions = [
|
||||||
|
{ key: 'ddd M/D', value: 'Tue 3/5' },
|
||||||
|
{ key: 'ddd MM/DD', value: 'Tue 03/05' },
|
||||||
|
{ key: 'ddd D/M', value: 'Tue 5/3' },
|
||||||
|
{ key: 'ddd DD/MM', value: 'Tue 05/03' }
|
||||||
|
];
|
||||||
|
|
||||||
|
const shortDateFormatOptions = [
|
||||||
|
{ key: 'MMM D YYYY', value: 'Mar 5 2014' },
|
||||||
|
{ key: 'DD MMM YYYY', value: '5 Mar 2014' },
|
||||||
|
{ key: 'MM/D/YYYY', value: '03/5/2014' },
|
||||||
|
{ key: 'MM/DD/YYYY', value: '03/05/2014' },
|
||||||
|
{ key: 'DD/MM/YYYY', value: '05/03/2014' },
|
||||||
|
{ key: 'YYYY-MM-DD', value: '2014-03-05' }
|
||||||
|
];
|
||||||
|
|
||||||
|
const longDateFormatOptions = [
|
||||||
|
{ key: 'dddd, MMMM D YYYY', value: 'Tuesday, March 5, 2014' },
|
||||||
|
{ key: 'dddd, D MMMM YYYY', value: 'Tuesday, 5 March, 2014' }
|
||||||
|
];
|
||||||
|
|
||||||
|
const timeFormatOptions = [
|
||||||
|
{ key: 'h(:mm)a', value: '5pm/5:30pm' },
|
||||||
|
{ key: 'HH:mm', value: '17:00/17:30' }
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PageContent title="UI Settings">
|
||||||
|
<SettingsToolbarConnector
|
||||||
|
{...otherProps}
|
||||||
|
onSavePress={onSavePress}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<PageContentBodyConnector>
|
||||||
|
{
|
||||||
|
isFetching &&
|
||||||
|
<LoadingIndicator />
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
!isFetching && error &&
|
||||||
|
<div>Unable to load UI settings</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
hasSettings && !isFetching && !error &&
|
||||||
|
<Form
|
||||||
|
id="uiSettings"
|
||||||
|
{...otherProps}
|
||||||
|
>
|
||||||
|
<FieldSet
|
||||||
|
legend="Calendar"
|
||||||
|
>
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>First Day of Week</FormLabel>
|
||||||
|
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.SELECT}
|
||||||
|
name="firstDayOfWeek"
|
||||||
|
values={firstDayOfWeekOptions}
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.firstDayOfWeek}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Week Column Header</FormLabel>
|
||||||
|
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.SELECT}
|
||||||
|
name="calendarWeekColumnHeader"
|
||||||
|
values={weekColumnOptions}
|
||||||
|
onChange={onInputChange}
|
||||||
|
helpText="Shown above each column when week is the active view"
|
||||||
|
{...settings.calendarWeekColumnHeader}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
</FieldSet>
|
||||||
|
|
||||||
|
<FieldSet
|
||||||
|
legend="Dates"
|
||||||
|
>
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Short Date Format</FormLabel>
|
||||||
|
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.SELECT}
|
||||||
|
name="shortDateFormat"
|
||||||
|
values={shortDateFormatOptions}
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.shortDateFormat}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Long Date Format</FormLabel>
|
||||||
|
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.SELECT}
|
||||||
|
name="longDateFormat"
|
||||||
|
values={longDateFormatOptions}
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.longDateFormat}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Time Format</FormLabel>
|
||||||
|
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.SELECT}
|
||||||
|
name="timeFormat"
|
||||||
|
values={timeFormatOptions}
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.timeFormat}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Show Relative Dates</FormLabel>
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.CHECK}
|
||||||
|
name="showRelativeDates"
|
||||||
|
helpText="Show relative (Today/Yesterday/etc) or absolute dates"
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.showRelativeDates}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
</FieldSet>
|
||||||
|
|
||||||
|
<FieldSet
|
||||||
|
legend="Style"
|
||||||
|
>
|
||||||
|
<FormGroup>
|
||||||
|
<FormLabel>Enable Color-Impaired mode</FormLabel>
|
||||||
|
<FormInputGroup
|
||||||
|
type={inputTypes.CHECK}
|
||||||
|
name="enableColorImpairedMode"
|
||||||
|
helpText="Altered style to allow color-impaired users to better distinguish color coded information"
|
||||||
|
onChange={onInputChange}
|
||||||
|
{...settings.enableColorImpairedMode}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
</FieldSet>
|
||||||
|
</Form>
|
||||||
|
}
|
||||||
|
</PageContentBodyConnector>
|
||||||
|
</PageContent>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
UISettings.propTypes = {
|
||||||
|
isFetching: PropTypes.bool.isRequired,
|
||||||
|
error: PropTypes.object,
|
||||||
|
settings: PropTypes.object.isRequired,
|
||||||
|
hasSettings: PropTypes.bool.isRequired,
|
||||||
|
onSavePress: PropTypes.func.isRequired,
|
||||||
|
onInputChange: PropTypes.func.isRequired
|
||||||
|
};
|
||||||
|
|
||||||
|
export default UISettings;
|
82
frontend/src/Settings/UI/UISettingsConnector.js
Normal file
82
frontend/src/Settings/UI/UISettingsConnector.js
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
import { createSelector } from 'reselect';
|
||||||
|
import createSettingsSectionSelector from 'Store/Selectors/createSettingsSectionSelector';
|
||||||
|
import { setUISettingsValue, saveUISettings, fetchUISettings } from 'Store/Actions/settingsActions';
|
||||||
|
import { clearPendingChanges } from 'Store/Actions/baseActions';
|
||||||
|
import connectSection from 'Store/connectSection';
|
||||||
|
import UISettings from './UISettings';
|
||||||
|
|
||||||
|
function createMapStateToProps() {
|
||||||
|
return createSelector(
|
||||||
|
(state) => state.settings.advancedSettings,
|
||||||
|
createSettingsSectionSelector(),
|
||||||
|
(advancedSettings, sectionSettings) => {
|
||||||
|
return {
|
||||||
|
advancedSettings,
|
||||||
|
...sectionSettings
|
||||||
|
};
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapDispatchToProps = {
|
||||||
|
setUISettingsValue,
|
||||||
|
saveUISettings,
|
||||||
|
fetchUISettings,
|
||||||
|
clearPendingChanges
|
||||||
|
};
|
||||||
|
|
||||||
|
class UISettingsConnector extends Component {
|
||||||
|
|
||||||
|
//
|
||||||
|
// Lifecycle
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.props.fetchUISettings();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this.props.clearPendingChanges({ section: this.props.section });
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Listeners
|
||||||
|
|
||||||
|
onInputChange = ({ name, value }) => {
|
||||||
|
this.props.setUISettingsValue({ name, value });
|
||||||
|
}
|
||||||
|
|
||||||
|
onSavePress = () => {
|
||||||
|
this.props.saveUISettings();
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Render
|
||||||
|
|
||||||
|
render() {
|
||||||
|
return (
|
||||||
|
<UISettings
|
||||||
|
onInputChange={this.onInputChange}
|
||||||
|
onSavePress={this.onSavePress}
|
||||||
|
{...this.props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
UISettingsConnector.propTypes = {
|
||||||
|
section: PropTypes.string.isRequired,
|
||||||
|
setUISettingsValue: PropTypes.func.isRequired,
|
||||||
|
saveUISettings: PropTypes.func.isRequired,
|
||||||
|
fetchUISettings: PropTypes.func.isRequired,
|
||||||
|
clearPendingChanges: PropTypes.func.isRequired
|
||||||
|
};
|
||||||
|
|
||||||
|
export default connectSection(
|
||||||
|
createMapStateToProps,
|
||||||
|
mapDispatchToProps,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
{ section: 'ui' }
|
||||||
|
)(UISettingsConnector);
|
|
@ -1,8 +1,19 @@
|
||||||
import $ from 'jquery';
|
|
||||||
import { batchActions } from 'redux-batched-actions';
|
import { batchActions } from 'redux-batched-actions';
|
||||||
|
import createAjaxRequest from 'Utilities/createAjaxRequest';
|
||||||
import getProviderState from 'Utilities/State/getProviderState';
|
import getProviderState from 'Utilities/State/getProviderState';
|
||||||
import { set, updateItem } from '../baseActions';
|
import { set, updateItem } from '../baseActions';
|
||||||
|
|
||||||
|
const abortCurrentRequests = {};
|
||||||
|
|
||||||
|
export function createCancelSaveProviderHandler(section) {
|
||||||
|
return function(payload) {
|
||||||
|
if (abortCurrentRequests[section]) {
|
||||||
|
abortCurrentRequests[section]();
|
||||||
|
abortCurrentRequests[section] = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
function createSaveProviderHandler(section, url, getFromState) {
|
function createSaveProviderHandler(section, url, getFromState) {
|
||||||
return function(payload) {
|
return function(payload) {
|
||||||
return function(dispatch, getState) {
|
return function(dispatch, getState) {
|
||||||
|
@ -24,9 +35,11 @@ function createSaveProviderHandler(section, url, getFromState) {
|
||||||
ajaxOptions.method = 'PUT';
|
ajaxOptions.method = 'PUT';
|
||||||
}
|
}
|
||||||
|
|
||||||
const promise = $.ajax(ajaxOptions);
|
const { request, abortRequest } = createAjaxRequest()(ajaxOptions);
|
||||||
|
|
||||||
promise.done((data) => {
|
abortCurrentRequests[section] = abortRequest;
|
||||||
|
|
||||||
|
request.done((data) => {
|
||||||
dispatch(batchActions([
|
dispatch(batchActions([
|
||||||
updateItem({ section, ...data }),
|
updateItem({ section, ...data }),
|
||||||
|
|
||||||
|
@ -39,11 +52,11 @@ function createSaveProviderHandler(section, url, getFromState) {
|
||||||
]));
|
]));
|
||||||
});
|
});
|
||||||
|
|
||||||
promise.fail((xhr) => {
|
request.fail((xhr) => {
|
||||||
dispatch(set({
|
dispatch(set({
|
||||||
section,
|
section,
|
||||||
isSaving: false,
|
isSaving: false,
|
||||||
saveError: xhr
|
saveError: xhr.aborted ? null : xhr
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,7 +1,18 @@
|
||||||
import $ from 'jquery';
|
import createAjaxRequest from 'Utilities/createAjaxRequest';
|
||||||
import getProviderState from 'Utilities/State/getProviderState';
|
import getProviderState from 'Utilities/State/getProviderState';
|
||||||
import { set } from '../baseActions';
|
import { set } from '../baseActions';
|
||||||
|
|
||||||
|
const abortCurrentRequests = {};
|
||||||
|
|
||||||
|
export function createCancelTestProviderHandler(section) {
|
||||||
|
return function(payload) {
|
||||||
|
if (abortCurrentRequests[section]) {
|
||||||
|
abortCurrentRequests[section]();
|
||||||
|
abortCurrentRequests[section] = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
function createTestProviderHandler(section, url, getFromState) {
|
function createTestProviderHandler(section, url, getFromState) {
|
||||||
return function(payload) {
|
return function(payload) {
|
||||||
return function(dispatch, getState) {
|
return function(dispatch, getState) {
|
||||||
|
@ -17,9 +28,11 @@ function createTestProviderHandler(section, url, getFromState) {
|
||||||
data: JSON.stringify(testData)
|
data: JSON.stringify(testData)
|
||||||
};
|
};
|
||||||
|
|
||||||
const promise = $.ajax(ajaxOptions);
|
const { request, abortRequest } = createAjaxRequest()(ajaxOptions);
|
||||||
|
|
||||||
promise.done((data) => {
|
abortCurrentRequests[section] = abortRequest;
|
||||||
|
|
||||||
|
request.done((data) => {
|
||||||
dispatch(set({
|
dispatch(set({
|
||||||
section,
|
section,
|
||||||
isTesting: false,
|
isTesting: false,
|
||||||
|
@ -27,11 +40,11 @@ function createTestProviderHandler(section, url, getFromState) {
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
promise.fail((xhr) => {
|
request.fail((xhr) => {
|
||||||
dispatch(set({
|
dispatch(set({
|
||||||
section,
|
section,
|
||||||
isTesting: false,
|
isTesting: false,
|
||||||
saveError: xhr
|
saveError: xhr.aborted ? null : xhr
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
|
@ -247,8 +247,10 @@ export const SELECT_INDEXER_SCHEMA = 'SELECT_INDEXER_SCHEMA';
|
||||||
export const SET_INDEXER_VALUE = 'SET_INDEXER_VALUE';
|
export const SET_INDEXER_VALUE = 'SET_INDEXER_VALUE';
|
||||||
export const SET_INDEXER_FIELD_VALUE = 'SET_INDEXER_FIELD_VALUE';
|
export const SET_INDEXER_FIELD_VALUE = 'SET_INDEXER_FIELD_VALUE';
|
||||||
export const SAVE_INDEXER = 'SAVE_INDEXER';
|
export const SAVE_INDEXER = 'SAVE_INDEXER';
|
||||||
|
export const CANCEL_SAVE_INDEXER = 'CANCEL_SAVE_INDEXER';
|
||||||
export const DELETE_INDEXER = 'DELETE_INDEXER';
|
export const DELETE_INDEXER = 'DELETE_INDEXER';
|
||||||
export const TEST_INDEXER = 'TEST_INDEXER';
|
export const TEST_INDEXER = 'TEST_INDEXER';
|
||||||
|
export const CANCEL_TEST_INDEXER = 'CANCEL_TEST_INDEXER';
|
||||||
|
|
||||||
export const FETCH_INDEXER_OPTIONS = 'FETCH_INDEXER_OPTIONS';
|
export const FETCH_INDEXER_OPTIONS = 'FETCH_INDEXER_OPTIONS';
|
||||||
export const SET_INDEXER_OPTIONS_VALUE = 'SET_INDEXER_OPTIONS_VALUE';
|
export const SET_INDEXER_OPTIONS_VALUE = 'SET_INDEXER_OPTIONS_VALUE';
|
||||||
|
@ -265,8 +267,10 @@ export const SELECT_DOWNLOAD_CLIENT_SCHEMA = 'SELECT_DOWNLOAD_CLIENT_SCHEMA';
|
||||||
export const SET_DOWNLOAD_CLIENT_VALUE = 'SET_DOWNLOAD_CLIENT_VALUE';
|
export const SET_DOWNLOAD_CLIENT_VALUE = 'SET_DOWNLOAD_CLIENT_VALUE';
|
||||||
export const SET_DOWNLOAD_CLIENT_FIELD_VALUE = 'SET_DOWNLOAD_CLIENT_FIELD_VALUE';
|
export const SET_DOWNLOAD_CLIENT_FIELD_VALUE = 'SET_DOWNLOAD_CLIENT_FIELD_VALUE';
|
||||||
export const SAVE_DOWNLOAD_CLIENT = 'SAVE_DOWNLOAD_CLIENT';
|
export const SAVE_DOWNLOAD_CLIENT = 'SAVE_DOWNLOAD_CLIENT';
|
||||||
|
export const CANCEL_SAVE_DOWNLOAD_CLIENT = 'CANCEL_SAVE_DOWNLOAD_CLIENT';
|
||||||
export const DELETE_DOWNLOAD_CLIENT = 'DELETE_DOWNLOAD_CLIENT';
|
export const DELETE_DOWNLOAD_CLIENT = 'DELETE_DOWNLOAD_CLIENT';
|
||||||
export const TEST_DOWNLOAD_CLIENT = 'TEST_DOWNLOAD_CLIENT';
|
export const TEST_DOWNLOAD_CLIENT = 'TEST_DOWNLOAD_CLIENT';
|
||||||
|
export const CANCEL_TEST_DOWNLOAD_CLIENT = 'CANCEL_TEST_DOWNLOAD_CLIENT';
|
||||||
|
|
||||||
export const FETCH_DOWNLOAD_CLIENT_OPTIONS = 'FETCH_DOWNLOAD_CLIENT_OPTIONS';
|
export const FETCH_DOWNLOAD_CLIENT_OPTIONS = 'FETCH_DOWNLOAD_CLIENT_OPTIONS';
|
||||||
export const SET_DOWNLOAD_CLIENT_OPTIONS_VALUE = 'SET_DOWNLOAD_CLIENT_OPTIONS_VALUE';
|
export const SET_DOWNLOAD_CLIENT_OPTIONS_VALUE = 'SET_DOWNLOAD_CLIENT_OPTIONS_VALUE';
|
||||||
|
@ -283,8 +287,9 @@ export const SELECT_NOTIFICATION_SCHEMA = 'SELECT_NOTIFICATION_SCHEMA';
|
||||||
export const SET_NOTIFICATION_VALUE = 'SET_NOTIFICATION_VALUE';
|
export const SET_NOTIFICATION_VALUE = 'SET_NOTIFICATION_VALUE';
|
||||||
export const SET_NOTIFICATION_FIELD_VALUE = 'SET_NOTIFICATION_FIELD_VALUE';
|
export const SET_NOTIFICATION_FIELD_VALUE = 'SET_NOTIFICATION_FIELD_VALUE';
|
||||||
export const SAVE_NOTIFICATION = 'SAVE_NOTIFICATION';
|
export const SAVE_NOTIFICATION = 'SAVE_NOTIFICATION';
|
||||||
|
export const CANCEL_SAVE_NOTIFICATION = 'CANCEL_SAVE_NOTIFICATION';
|
||||||
export const DELETE_NOTIFICATION = 'DELETE_NOTIFICATION';
|
export const DELETE_NOTIFICATION = 'DELETE_NOTIFICATION';
|
||||||
export const TEST_NOTIFICATION = 'TEST_NOTIFICATION';
|
export const CANCEL_TEST_NOTIFICATION = 'CANCEL_TEST_NOTIFICATION';
|
||||||
|
|
||||||
export const FETCH_METADATA = 'FETCH_METADATA';
|
export const FETCH_METADATA = 'FETCH_METADATA';
|
||||||
export const SET_METADATA_VALUE = 'SET_METADATA_VALUE';
|
export const SET_METADATA_VALUE = 'SET_METADATA_VALUE';
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
import $ from 'jquery';
|
import $ from 'jquery';
|
||||||
import { batchActions } from 'redux-batched-actions';
|
import { batchActions } from 'redux-batched-actions';
|
||||||
|
import createAjaxRequest from 'Utilities/createAjaxRequest';
|
||||||
import getNewSeries from 'Utilities/Series/getNewSeries';
|
import getNewSeries from 'Utilities/Series/getNewSeries';
|
||||||
import * as types from './actionTypes';
|
import * as types from './actionTypes';
|
||||||
import { set, update, updateItem } from './baseActions';
|
import { set, update, updateItem } from './baseActions';
|
||||||
|
|
||||||
let currentXHR = null;
|
let abortCurrentRequest = null;
|
||||||
let xhrCancelled = false;
|
|
||||||
const section = 'addArtist';
|
const section = 'addArtist';
|
||||||
|
|
||||||
const addArtistActionHandlers = {
|
const addArtistActionHandlers = {
|
||||||
|
@ -14,24 +14,20 @@ const addArtistActionHandlers = {
|
||||||
return function(dispatch, getState) {
|
return function(dispatch, getState) {
|
||||||
dispatch(set({ section, isFetching: true }));
|
dispatch(set({ section, isFetching: true }));
|
||||||
|
|
||||||
if (currentXHR) {
|
if (abortCurrentRequest) {
|
||||||
xhrCancelled = true;
|
abortCurrentRequest();
|
||||||
currentXHR.abort();
|
|
||||||
currentXHR = null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
currentXHR = new window.XMLHttpRequest();
|
const { request, abortRequest } = createAjaxRequest()({
|
||||||
xhrCancelled = false;
|
|
||||||
|
|
||||||
const promise = $.ajax({
|
|
||||||
url: '/artist/lookup',
|
url: '/artist/lookup',
|
||||||
xhr: () => currentXHR,
|
|
||||||
data: {
|
data: {
|
||||||
term: payload.term
|
term: payload.term
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
promise.done((data) => {
|
abortCurrentRequest = abortRequest;
|
||||||
|
|
||||||
|
request.done((data) => {
|
||||||
dispatch(batchActions([
|
dispatch(batchActions([
|
||||||
update({ section, data }),
|
update({ section, data }),
|
||||||
|
|
||||||
|
@ -44,12 +40,12 @@ const addArtistActionHandlers = {
|
||||||
]));
|
]));
|
||||||
});
|
});
|
||||||
|
|
||||||
promise.fail((xhr) => {
|
request.fail((xhr) => {
|
||||||
dispatch(set({
|
dispatch(set({
|
||||||
section,
|
section,
|
||||||
isFetching: false,
|
isFetching: false,
|
||||||
isPopulated: false,
|
isPopulated: false,
|
||||||
error: xhrCancelled ? null : xhr
|
error: xhr.aborted ? null : xhr
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
|
@ -5,9 +5,9 @@ import * as types from './actionTypes';
|
||||||
import createFetchHandler from './Creators/createFetchHandler';
|
import createFetchHandler from './Creators/createFetchHandler';
|
||||||
import createFetchSchemaHandler from './Creators/createFetchSchemaHandler';
|
import createFetchSchemaHandler from './Creators/createFetchSchemaHandler';
|
||||||
import createSaveHandler from './Creators/createSaveHandler';
|
import createSaveHandler from './Creators/createSaveHandler';
|
||||||
import createSaveProviderHandler from './Creators/createSaveProviderHandler';
|
import createSaveProviderHandler, { createCancelSaveProviderHandler } from './Creators/createSaveProviderHandler';
|
||||||
import createRemoveItemHandler from './Creators/createRemoveItemHandler';
|
import createRemoveItemHandler from './Creators/createRemoveItemHandler';
|
||||||
import createTestProviderHandler from './Creators/createTestProviderHandler';
|
import createTestProviderHandler, { createCancelTestProviderHandler } from './Creators/createTestProviderHandler';
|
||||||
import { set, update, clearPendingChanges } from './baseActions';
|
import { set, update, clearPendingChanges } from './baseActions';
|
||||||
|
|
||||||
const settingsActionHandlers = {
|
const settingsActionHandlers = {
|
||||||
|
@ -161,6 +161,8 @@ const settingsActionHandlers = {
|
||||||
'/indexer',
|
'/indexer',
|
||||||
(state) => state.settings.indexers),
|
(state) => state.settings.indexers),
|
||||||
|
|
||||||
|
[types.CANCEL_SAVE_INDEXER]: createCancelSaveProviderHandler('indexers'),
|
||||||
|
|
||||||
[types.DELETE_INDEXER]: createRemoveItemHandler('indexers',
|
[types.DELETE_INDEXER]: createRemoveItemHandler('indexers',
|
||||||
'/indexer',
|
'/indexer',
|
||||||
(state) => state.settings.indexers),
|
(state) => state.settings.indexers),
|
||||||
|
@ -169,6 +171,8 @@ const settingsActionHandlers = {
|
||||||
'/indexer',
|
'/indexer',
|
||||||
(state) => state.settings.indexers),
|
(state) => state.settings.indexers),
|
||||||
|
|
||||||
|
[types.CANCEL_TEST_INDEXER]: createCancelTestProviderHandler('indexers'),
|
||||||
|
|
||||||
[types.FETCH_INDEXER_OPTIONS]: createFetchHandler('indexerOptions', '/config/indexer'),
|
[types.FETCH_INDEXER_OPTIONS]: createFetchHandler('indexerOptions', '/config/indexer'),
|
||||||
[types.SAVE_INDEXER_OPTIONS]: createSaveHandler('indexerOptions', '/config/indexer', (state) => state.settings.indexerOptions),
|
[types.SAVE_INDEXER_OPTIONS]: createSaveHandler('indexerOptions', '/config/indexer', (state) => state.settings.indexerOptions),
|
||||||
|
|
||||||
|
@ -189,6 +193,8 @@ const settingsActionHandlers = {
|
||||||
'/downloadclient',
|
'/downloadclient',
|
||||||
(state) => state.settings.downloadClients),
|
(state) => state.settings.downloadClients),
|
||||||
|
|
||||||
|
[types.CANCEL_SAVE_DOWNLOAD_CLIENT]: createCancelSaveProviderHandler('downloadClients'),
|
||||||
|
|
||||||
[types.DELETE_DOWNLOAD_CLIENT]: createRemoveItemHandler('downloadClients',
|
[types.DELETE_DOWNLOAD_CLIENT]: createRemoveItemHandler('downloadClients',
|
||||||
'/downloadclient',
|
'/downloadclient',
|
||||||
(state) => state.settings.downloadClients),
|
(state) => state.settings.downloadClients),
|
||||||
|
@ -197,6 +203,8 @@ const settingsActionHandlers = {
|
||||||
'/downloadclient',
|
'/downloadclient',
|
||||||
(state) => state.settings.downloadClients),
|
(state) => state.settings.downloadClients),
|
||||||
|
|
||||||
|
[types.CANCEL_TEST_DOWNLOAD_CLIENT]: createCancelTestProviderHandler('downloadClients'),
|
||||||
|
|
||||||
[types.FETCH_DOWNLOAD_CLIENT_OPTIONS]: createFetchHandler('downloadClientOptions', '/config/downloadclient'),
|
[types.FETCH_DOWNLOAD_CLIENT_OPTIONS]: createFetchHandler('downloadClientOptions', '/config/downloadclient'),
|
||||||
[types.SAVE_DOWNLOAD_CLIENT_OPTIONS]: createSaveHandler('downloadClientOptions', '/config/downloadclient', (state) => state.settings.downloadClientOptions),
|
[types.SAVE_DOWNLOAD_CLIENT_OPTIONS]: createSaveHandler('downloadClientOptions', '/config/downloadclient', (state) => state.settings.downloadClientOptions),
|
||||||
|
|
||||||
|
@ -217,6 +225,8 @@ const settingsActionHandlers = {
|
||||||
'/notification',
|
'/notification',
|
||||||
(state) => state.settings.notifications),
|
(state) => state.settings.notifications),
|
||||||
|
|
||||||
|
[types.CANCEL_SAVE_NOTIFICATION]: createCancelSaveProviderHandler('notifications'),
|
||||||
|
|
||||||
[types.DELETE_NOTIFICATION]: createRemoveItemHandler('notifications',
|
[types.DELETE_NOTIFICATION]: createRemoveItemHandler('notifications',
|
||||||
'/notification',
|
'/notification',
|
||||||
(state) => state.settings.notifications),
|
(state) => state.settings.notifications),
|
||||||
|
@ -225,6 +235,8 @@ const settingsActionHandlers = {
|
||||||
'/notification',
|
'/notification',
|
||||||
(state) => state.settings.notifications),
|
(state) => state.settings.notifications),
|
||||||
|
|
||||||
|
[types.CANCEL_TEST_NOTIFICATION]: createCancelTestProviderHandler('notifications'),
|
||||||
|
|
||||||
[types.FETCH_METADATA]: createFetchHandler('metadata', '/metadata'),
|
[types.FETCH_METADATA]: createFetchHandler('metadata', '/metadata'),
|
||||||
|
|
||||||
[types.SAVE_METADATA]: createSaveProviderHandler('metadata',
|
[types.SAVE_METADATA]: createSaveProviderHandler('metadata',
|
||||||
|
|
|
@ -79,8 +79,10 @@ export const fetchIndexerSchema = settingsActionHandlers[types.FETCH_INDEXER_SCH
|
||||||
export const selectIndexerSchema = createAction(types.SELECT_INDEXER_SCHEMA);
|
export const selectIndexerSchema = createAction(types.SELECT_INDEXER_SCHEMA);
|
||||||
|
|
||||||
export const saveIndexer = settingsActionHandlers[types.SAVE_INDEXER];
|
export const saveIndexer = settingsActionHandlers[types.SAVE_INDEXER];
|
||||||
|
export const cancelSaveIndexer = settingsActionHandlers[types.CANCEL_SAVE_INDEXER];
|
||||||
export const deleteIndexer = settingsActionHandlers[types.DELETE_INDEXER];
|
export const deleteIndexer = settingsActionHandlers[types.DELETE_INDEXER];
|
||||||
export const testIndexer = settingsActionHandlers[types.TEST_INDEXER];
|
export const testIndexer = settingsActionHandlers[types.TEST_INDEXER];
|
||||||
|
export const cancelTestIndexer = settingsActionHandlers[types.CANCEL_TEST_INDEXER];
|
||||||
|
|
||||||
export const setIndexerValue = createAction(types.SET_INDEXER_VALUE, (payload) => {
|
export const setIndexerValue = createAction(types.SET_INDEXER_VALUE, (payload) => {
|
||||||
return {
|
return {
|
||||||
|
@ -121,8 +123,10 @@ export const fetchDownloadClientSchema = settingsActionHandlers[types.FETCH_DOWN
|
||||||
export const selectDownloadClientSchema = createAction(types.SELECT_DOWNLOAD_CLIENT_SCHEMA);
|
export const selectDownloadClientSchema = createAction(types.SELECT_DOWNLOAD_CLIENT_SCHEMA);
|
||||||
|
|
||||||
export const saveDownloadClient = settingsActionHandlers[types.SAVE_DOWNLOAD_CLIENT];
|
export const saveDownloadClient = settingsActionHandlers[types.SAVE_DOWNLOAD_CLIENT];
|
||||||
|
export const cancelSaveDownloadClient = settingsActionHandlers[types.CANCEL_SAVE_DOWNLOAD_CLIENT];
|
||||||
export const deleteDownloadClient = settingsActionHandlers[types.DELETE_DOWNLOAD_CLIENT];
|
export const deleteDownloadClient = settingsActionHandlers[types.DELETE_DOWNLOAD_CLIENT];
|
||||||
export const testDownloadClient = settingsActionHandlers[types.TEST_DOWNLOAD_CLIENT];
|
export const testDownloadClient = settingsActionHandlers[types.TEST_DOWNLOAD_CLIENT];
|
||||||
|
export const cancelTestDownloadClient = settingsActionHandlers[types.CANCEL_TEST_DOWNLOAD_CLIENT];
|
||||||
|
|
||||||
export const setDownloadClientValue = createAction(types.SET_DOWNLOAD_CLIENT_VALUE, (payload) => {
|
export const setDownloadClientValue = createAction(types.SET_DOWNLOAD_CLIENT_VALUE, (payload) => {
|
||||||
return {
|
return {
|
||||||
|
@ -163,8 +167,10 @@ export const fetchNotificationSchema = settingsActionHandlers[types.FETCH_NOTIFI
|
||||||
export const selectNotificationSchema = createAction(types.SELECT_NOTIFICATION_SCHEMA);
|
export const selectNotificationSchema = createAction(types.SELECT_NOTIFICATION_SCHEMA);
|
||||||
|
|
||||||
export const saveNotification = settingsActionHandlers[types.SAVE_NOTIFICATION];
|
export const saveNotification = settingsActionHandlers[types.SAVE_NOTIFICATION];
|
||||||
|
export const cancelSaveNotification = settingsActionHandlers[types.CANCEL_SAVE_NOTIFICATION];
|
||||||
export const deleteNotification = settingsActionHandlers[types.DELETE_NOTIFICATION];
|
export const deleteNotification = settingsActionHandlers[types.DELETE_NOTIFICATION];
|
||||||
export const testNotification = settingsActionHandlers[types.TEST_NOTIFICATION];
|
export const testNotification = settingsActionHandlers[types.TEST_NOTIFICATION];
|
||||||
|
export const cancelTestNotification = settingsActionHandlers[types.CANCEL_TEST_NOTIFICATION];
|
||||||
|
|
||||||
export const setNotificationValue = createAction(types.SET_NOTIFICATION_VALUE, (payload) => {
|
export const setNotificationValue = createAction(types.SET_NOTIFICATION_VALUE, (payload) => {
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
module.exports = {
|
module.exports = {
|
||||||
defaultColor: '#333',
|
defaultColor: '#333',
|
||||||
disabledColor: '#999',
|
disabledColor: '#999',
|
||||||
|
dimColor: '#555',
|
||||||
black: '#000',
|
black: '#000',
|
||||||
white: '#fff',
|
white: '#fff',
|
||||||
primaryColor: '#0b8750',
|
primaryColor: '#0b8750',
|
||||||
|
@ -84,6 +85,7 @@ module.exports = {
|
||||||
dangerHoverBorderColor: '#ec2626;',
|
dangerHoverBorderColor: '#ec2626;',
|
||||||
|
|
||||||
iconButtonHoverColor: '#666',
|
iconButtonHoverColor: '#666',
|
||||||
|
iconButtonHoverLightColor: '#ccc',
|
||||||
|
|
||||||
//
|
//
|
||||||
// Modal
|
// Modal
|
||||||
|
|
33
frontend/src/Utilities/createAjaxRequest.js
Normal file
33
frontend/src/Utilities/createAjaxRequest.js
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
import $ from 'jquery';
|
||||||
|
|
||||||
|
export default function createAjaxRequest() {
|
||||||
|
return function(ajaxOptions) {
|
||||||
|
const requestXHR = new window.XMLHttpRequest();
|
||||||
|
let aborted = false;
|
||||||
|
let complete = false;
|
||||||
|
|
||||||
|
function abortRequest() {
|
||||||
|
if (!complete) {
|
||||||
|
aborted = true;
|
||||||
|
requestXHR.abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const request = $.ajax({
|
||||||
|
xhr: () => requestXHR,
|
||||||
|
...ajaxOptions
|
||||||
|
}).then(null, (xhr, textStatus, errorThrown) => {
|
||||||
|
xhr.aborted = aborted;
|
||||||
|
|
||||||
|
return $.Deferred().reject(xhr, textStatus, errorThrown).promise();
|
||||||
|
})
|
||||||
|
.always(() => {
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
request,
|
||||||
|
abortRequest
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue