diff --git a/frontend/src/Author/Editor/AuthorEditor.js b/frontend/src/Author/Editor/AuthorEditor.js index d5dc7e24c..e8c4b4f43 100644 --- a/frontend/src/Author/Editor/AuthorEditor.js +++ b/frontend/src/Author/Editor/AuthorEditor.js @@ -6,10 +6,13 @@ import FilterMenu from 'Components/Menu/FilterMenu'; import PageContent from 'Components/Page/PageContent'; import PageContentBody from 'Components/Page/PageContentBody'; import PageToolbar from 'Components/Page/Toolbar/PageToolbar'; +import PageToolbarButton from 'Components/Page/Toolbar/PageToolbarButton'; import PageToolbarSection from 'Components/Page/Toolbar/PageToolbarSection'; +import PageToolbarSeparator from 'Components/Page/Toolbar/PageToolbarSeparator'; import Table from 'Components/Table/Table'; import TableBody from 'Components/Table/TableBody'; -import { align, sortDirections } from 'Helpers/Props'; +import TableOptionsModalWrapper from 'Components/Table/TableOptions/TableOptionsModalWrapper'; +import { align, icons, sortDirections } from 'Helpers/Props'; import getErrorMessage from 'Utilities/Object/getErrorMessage'; import getSelectedIds from 'Utilities/Table/getSelectedIds'; import selectAll from 'Utilities/Table/selectAll'; @@ -20,46 +23,6 @@ import AuthorEditorFooter from './AuthorEditorFooter'; import AuthorEditorRowConnector from './AuthorEditorRowConnector'; import OrganizeAuthorModal from './Organize/OrganizeAuthorModal'; -function getColumns(showMetadataProfile) { - return [ - { - name: 'status', - isSortable: true, - isVisible: true - }, - { - name: 'sortName', - label: 'Name', - isSortable: true, - isVisible: true - }, - { - name: 'qualityProfileId', - label: 'Quality Profile', - isSortable: true, - isVisible: true - }, - { - name: 'metadataProfileId', - label: 'Metadata Profile', - isSortable: true, - isVisible: showMetadataProfile - }, - { - name: 'path', - label: 'Path', - isSortable: true, - isVisible: true - }, - { - name: 'tags', - label: 'Tags', - isSortable: false, - isVisible: true - } - ]; -} - class AuthorEditor extends Component { // @@ -74,8 +37,7 @@ class AuthorEditor extends Component { lastToggled: null, selectedState: {}, isOrganizingAuthorModalOpen: false, - isRetaggingAuthorModalOpen: false, - columns: getColumns(props.showMetadataProfile) + isRetaggingAuthorModalOpen: false }; } @@ -155,6 +117,7 @@ class AuthorEditor extends Component { error, totalItems, items, + columns, selectedFilterKey, filters, customFilters, @@ -166,7 +129,7 @@ class AuthorEditor extends Component { deleteError, isOrganizingAuthor, isRetaggingAuthor, - showMetadataProfile, + onTableOptionChange, onSortPress, onFilterSelect } = this.props; @@ -174,8 +137,7 @@ class AuthorEditor extends Component { const { allSelected, allUnselected, - selectedState, - columns + selectedState } = this.state; const selectedAuthorIds = this.getSelectedIds(); @@ -185,6 +147,18 @@ class AuthorEditor extends Component { + + + + + + column.name === 'metadataProfileId').isVisible} onSaveSelected={this.onSaveSelected} onOrganizeAuthorPress={this.onOrganizeAuthorPress} onRetagAuthorPress={this.onRetagAuthorPress} @@ -283,6 +258,7 @@ AuthorEditor.propTypes = { error: PropTypes.object, totalItems: PropTypes.number.isRequired, items: PropTypes.arrayOf(PropTypes.object).isRequired, + columns: PropTypes.arrayOf(PropTypes.object).isRequired, sortKey: PropTypes.string, sortDirection: PropTypes.oneOf(sortDirections.all), selectedFilterKey: PropTypes.oneOfType([PropTypes.string, PropTypes.number]).isRequired, @@ -294,7 +270,7 @@ AuthorEditor.propTypes = { deleteError: PropTypes.object, isOrganizingAuthor: PropTypes.bool.isRequired, isRetaggingAuthor: PropTypes.bool.isRequired, - showMetadataProfile: PropTypes.bool.isRequired, + onTableOptionChange: PropTypes.func.isRequired, onSortPress: PropTypes.func.isRequired, onFilterSelect: PropTypes.func.isRequired, onSaveSelected: PropTypes.func.isRequired diff --git a/frontend/src/Author/Editor/AuthorEditorConnector.js b/frontend/src/Author/Editor/AuthorEditorConnector.js index 3ce0ab747..befa215d9 100644 --- a/frontend/src/Author/Editor/AuthorEditorConnector.js +++ b/frontend/src/Author/Editor/AuthorEditorConnector.js @@ -3,7 +3,7 @@ import React, { Component } from 'react'; import { connect } from 'react-redux'; import { createSelector } from 'reselect'; import * as commandNames from 'Commands/commandNames'; -import { saveAuthorEditor, setAuthorEditorFilter, setAuthorEditorSort } from 'Store/Actions/authorEditorActions'; +import { saveAuthorEditor, setAuthorEditorFilter, setAuthorEditorSort, setAuthorEditorTableOption } from 'Store/Actions/authorEditorActions'; import { executeCommand } from 'Store/Actions/commandActions'; import { fetchRootFolders } from 'Store/Actions/settingsActions'; import createClientSideCollectionSelector from 'Store/Selectors/createClientSideCollectionSelector'; @@ -12,15 +12,13 @@ import AuthorEditor from './AuthorEditor'; function createMapStateToProps() { return createSelector( - (state) => state.settings.metadataProfiles, createClientSideCollectionSelector('authors', 'authorEditor'), createCommandExecutingSelector(commandNames.RENAME_AUTHOR), createCommandExecutingSelector(commandNames.RETAG_AUTHOR), - (metadataProfiles, author, isOrganizingAuthor, isRetaggingAuthor) => { + (author, isOrganizingAuthor, isRetaggingAuthor) => { return { isOrganizingAuthor, isRetaggingAuthor, - showMetadataProfile: metadataProfiles.items.length > 1, ...author }; } @@ -30,6 +28,7 @@ function createMapStateToProps() { const mapDispatchToProps = { dispatchSetAuthorEditorSort: setAuthorEditorSort, dispatchSetAuthorEditorFilter: setAuthorEditorFilter, + dispatchSetAuthorEditorTableOption: setAuthorEditorTableOption, dispatchSaveAuthorEditor: saveAuthorEditor, dispatchFetchRootFolders: fetchRootFolders, dispatchExecuteCommand: executeCommand @@ -55,6 +54,10 @@ class AuthorEditorConnector extends Component { this.props.dispatchSetAuthorEditorFilter({ selectedFilterKey }); } + onTableOptionChange = (payload) => { + this.props.dispatchSetAuthorEditorTableOption(payload); + } + onSaveSelected = (payload) => { this.props.dispatchSaveAuthorEditor(payload); } @@ -76,6 +79,7 @@ class AuthorEditorConnector extends Component { onSortPress={this.onSortPress} onFilterSelect={this.onFilterSelect} onSaveSelected={this.onSaveSelected} + onTableOptionChange={this.onTableOptionChange} /> ); } @@ -84,6 +88,7 @@ class AuthorEditorConnector extends Component { AuthorEditorConnector.propTypes = { dispatchSetAuthorEditorSort: PropTypes.func.isRequired, dispatchSetAuthorEditorFilter: PropTypes.func.isRequired, + dispatchSetAuthorEditorTableOption: PropTypes.func.isRequired, dispatchSaveAuthorEditor: PropTypes.func.isRequired, dispatchFetchRootFolders: PropTypes.func.isRequired, dispatchExecuteCommand: PropTypes.func.isRequired diff --git a/frontend/src/Author/Editor/AuthorEditorFooter.js b/frontend/src/Author/Editor/AuthorEditorFooter.js index 144993782..1d8da9c50 100644 --- a/frontend/src/Author/Editor/AuthorEditorFooter.js +++ b/frontend/src/Author/Editor/AuthorEditorFooter.js @@ -138,7 +138,7 @@ class AuthorEditorFooter extends Component { isDeleting, isOrganizingAuthor, isRetaggingAuthor, - showMetadataProfile, + columns, onOrganizeAuthorPress, onRetagAuthorPress } = this.props; @@ -178,56 +178,88 @@ class AuthorEditorFooter extends Component { /> -
- - - -
- { - showMetadataProfile && -
- - - -
+ columns.map((column) => { + const { + name, + isVisible + } = column; + + if (!isVisible) { + return null; + } + + if (name === 'qualityProfileId') { + return ( +
+ + + +
+ ); + } + + if (name === 'metadataProfileId') { + return ( +
+ + + +
+ ); + } + + if (name === 'path') { + return ( +
+ + + +
+ ); + } + + return null; + }) } -
- - - -
-
- + { + columns.map((column) => { + const { + name, + isVisible + } = column; - - - + if (!isVisible) { + return null; + } - - {qualityProfile.name} - + if (name === 'status') { + return ( + + ); + } - { - _.find(columns, { name: 'metadataProfileId' }).isVisible && - - {metadataProfile.name} - - } + if (name === 'sortName') { + return ( + + + + ); + } + + if (name === 'qualityProfileId') { + return ( + + {qualityProfile.name} + + ); + } - - {path} - + if (name === 'metadataProfileId') { + return ( + + {metadataProfile.name} + + ); + } - - - + if (name === 'path') { + return ( + + {path} + + ); + } + + if (name === 'sizeOnDisk') { + return ( + + {formatBytes(statistics.sizeOnDisk)} + + ); + } + + if (name === 'tags') { + return ( + + + + ); + } + + return null; + }) + } ); } @@ -94,6 +142,7 @@ AuthorEditorRow.propTypes = { metadataProfile: PropTypes.object.isRequired, qualityProfile: PropTypes.object.isRequired, path: PropTypes.string.isRequired, + statistics: PropTypes.object.isRequired, tags: PropTypes.arrayOf(PropTypes.number).isRequired, columns: PropTypes.arrayOf(PropTypes.object).isRequired, isSelected: PropTypes.bool, diff --git a/frontend/src/Store/Actions/authorEditorActions.js b/frontend/src/Store/Actions/authorEditorActions.js index 9b3a04102..a7754c648 100644 --- a/frontend/src/Store/Actions/authorEditorActions.js +++ b/frontend/src/Store/Actions/authorEditorActions.js @@ -8,6 +8,7 @@ import { set, updateItem } from './baseActions'; import createHandleActions from './Creators/createHandleActions'; import createSetClientSideCollectionFilterReducer from './Creators/Reducers/createSetClientSideCollectionFilterReducer'; import createSetClientSideCollectionSortReducer from './Creators/Reducers/createSetClientSideCollectionSortReducer'; +import createSetTableOptionReducer from './Creators/Reducers/createSetTableOptionReducer'; // // Variables @@ -30,6 +31,52 @@ export const defaultState = { filters, filterPredicates, + columns: [ + { + name: 'status', + columnLabel: 'Status', + isSortable: true, + isVisible: true, + isModifiable: false + }, + { + name: 'sortName', + label: 'Name', + isSortable: true, + isVisible: true + }, + { + name: 'qualityProfileId', + label: 'Quality Profile', + isSortable: true, + isVisible: true + }, + { + name: 'metadataProfileId', + label: 'Metadata Profile', + isSortable: true, + isVisible: false + }, + { + name: 'path', + label: 'Path', + isSortable: true, + isVisible: true + }, + { + name: 'sizeOnDisk', + label: 'Size on Disk', + isSortable: true, + isVisible: false + }, + { + name: 'tags', + label: 'Tags', + isSortable: false, + isVisible: true + } + ], + filterBuilderProps: [ { name: 'monitored', @@ -65,6 +112,12 @@ export const defaultState = { label: 'Root Folder Path', type: filterBuilderTypes.EXACT }, + { + name: 'sizeOnDisk', + label: 'Size on Disk', + type: filterBuilderTypes.NUMBER, + valueType: filterBuilderValueTypes.BYTES + }, { name: 'tags', label: 'Tags', @@ -90,6 +143,7 @@ export const SET_AUTHOR_EDITOR_SORT = 'authorEditor/setAuthorEditorSort'; export const SET_AUTHOR_EDITOR_FILTER = 'authorEditor/setAuthorEditorFilter'; export const SAVE_AUTHOR_EDITOR = 'authorEditor/saveAuthorEditor'; export const BULK_DELETE_AUTHOR = 'authorEditor/bulkDeleteAuthor'; +export const SET_AUTHOR_EDITOR_TABLE_OPTION = 'authorEditor/setAuthorEditorTableOption'; // // Action Creators @@ -98,6 +152,7 @@ export const setAuthorEditorSort = createAction(SET_AUTHOR_EDITOR_SORT); export const setAuthorEditorFilter = createAction(SET_AUTHOR_EDITOR_FILTER); export const saveAuthorEditor = createThunk(SAVE_AUTHOR_EDITOR); export const bulkDeleteAuthor = createThunk(BULK_DELETE_AUTHOR); +export const setAuthorEditorTableOption = createAction(SET_AUTHOR_EDITOR_TABLE_OPTION); // // Action Handlers @@ -181,6 +236,7 @@ export const actionHandlers = handleThunks({ export const reducers = createHandleActions({ + [SET_AUTHOR_EDITOR_TABLE_OPTION]: createSetTableOptionReducer(section), [SET_AUTHOR_EDITOR_SORT]: createSetClientSideCollectionSortReducer(section), [SET_AUTHOR_EDITOR_FILTER]: createSetClientSideCollectionFilterReducer(section)