Fix GitIgnore, Update from Sonarr

pull/6/head
Qstick 7 years ago
parent 77f1d2e64c
commit 22d9c5e666

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);

@ -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;
}

@ -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;

@ -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;

@ -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);
abortCurrentRequests[section] = abortRequest;
promise.done((data) => { 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);
abortCurrentRequests[section] = abortRequest;
promise.done((data) => { 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

@ -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…
Cancel
Save