New: Ability to change root folder when editing movie

(cherry picked from commit 417af2b91542e709e4b99aa5ca55b0501ba426ad)
This commit is contained in:
Mark McDowall 2024-11-23 20:21:24 -08:00 committed by Bogdan
parent b91517afd5
commit c52f9c5ec4
13 changed files with 227 additions and 8 deletions

View file

@ -14,13 +14,14 @@ function FormInputButton({
className = styles.button,
canSpin = false,
isLastButton = true,
kind = kinds.PRIMARY,
...otherProps
}: FormInputButtonProps) {
if (canSpin) {
return (
<SpinnerButton
className={classNames(className, !isLastButton && styles.middleButton)}
kind={kinds.PRIMARY}
kind={kind}
{...otherProps}
/>
);
@ -29,7 +30,7 @@ function FormInputButton({
return (
<Button
className={classNames(className, !isLastButton && styles.middleButton)}
kind={kinds.PRIMARY}
kind={kind}
{...otherProps}
/>
);

View file

@ -16,3 +16,7 @@
height: 35px;
}
.fileBrowserMiddleButton {
composes: middleButton from '~./FormInputButton.css';
}

View file

@ -2,6 +2,7 @@
// Please do not change this file!
interface CssExports {
'fileBrowserButton': string;
'fileBrowserMiddleButton': string;
'hasFileBrowser': string;
'inputWrapper': string;
'pathMatch': string;

View file

@ -1,3 +1,4 @@
import classNames from 'classnames';
import React, {
KeyboardEvent,
SyntheticEvent,
@ -29,6 +30,7 @@ export interface PathInputProps {
value?: string;
placeholder?: string;
includeFiles: boolean;
hasButton?: boolean;
hasFileBrowser?: boolean;
onChange: (change: InputChanged<string>) => void;
}
@ -96,6 +98,7 @@ export function PathInputInternal(props: PathInputInternalProps) {
value: inputValue = '',
paths,
includeFiles,
hasButton,
hasFileBrowser = true,
onChange,
onFetchPaths,
@ -229,9 +232,12 @@ export function PathInputInternal(props: PathInputInternalProps) {
/>
{hasFileBrowser ? (
<div>
<>
<FormInputButton
className={styles.fileBrowserButton}
className={classNames(
styles.fileBrowserButton,
hasButton && styles.fileBrowserMiddleButton
)}
onPress={handleFileBrowserOpenPress}
>
<Icon name={icons.FOLDER_OPEN} />
@ -245,7 +251,7 @@ export function PathInputInternal(props: PathInputInternalProps) {
onChange={onChange}
onModalClose={handleFileBrowserModalClose}
/>
</div>
</>
) : null}
</div>
);

View file

@ -71,6 +71,7 @@ import {
faFire as fasFire,
faFlag as fasFlag,
faFolderOpen as fasFolderOpen,
faFolderTree as farFolderTree,
faForward as fasForward,
faHeart as fasHeart,
faHistory as fasHistory,
@ -216,6 +217,7 @@ export const REMOVE = fasTimes;
export const RESTART = fasRedoAlt;
export const RESTORE = fasHistory;
export const REORDER = fasBars;
export const ROOT_FOLDER = farFolderTree;
export const RSS = fasRss;
export const SAVE = fasSave;
export const SCHEDULED = farClock;

View file

@ -4,6 +4,7 @@ import MovieMinimumAvailabilityPopoverContent from 'AddMovie/MovieMinimumAvailab
import AppState from 'App/State/AppState';
import Form from 'Components/Form/Form';
import FormGroup from 'Components/Form/FormGroup';
import FormInputButton from 'Components/Form/FormInputButton';
import FormInputGroup from 'Components/Form/FormInputGroup';
import FormLabel from 'Components/Form/FormLabel';
import Icon from 'Components/Icon';
@ -28,6 +29,8 @@ import { saveMovie, setMovieValue } from 'Store/Actions/movieActions';
import selectSettings from 'Store/Selectors/selectSettings';
import { InputChanged } from 'typings/inputs';
import translate from 'Utilities/String/translate';
import RootFolderModal from './RootFolder/RootFolderModal';
import { RootFolderUpdated } from './RootFolder/RootFolderModalContent';
import styles from './EditMovieModalContent.css';
export interface EditMovieModalContentProps {
@ -49,6 +52,7 @@ function EditMovieModalContent({
qualityProfileId,
path,
tags,
rootFolderPath: initialRootFolderPath,
} = useMovie(movieId)!;
const { isSaving, saveError, pendingChanges } = useSelector(
@ -57,6 +61,10 @@ function EditMovieModalContent({
const wasSaving = usePrevious(isSaving);
const [isRootFolderModalOpen, setIsRootFolderModalOpen] = useState(false);
const [rootFolderPath, setRootFolderPath] = useState(initialRootFolderPath);
const isPathChanging = pendingChanges.path && path !== pendingChanges.path;
const [isConfirmMoveModalOpen, setIsConfirmMoveModalOpen] = useState(false);
@ -91,6 +99,26 @@ function EditMovieModalContent({
[dispatch]
);
const handleRootFolderPress = useCallback(() => {
setIsRootFolderModalOpen(true);
}, []);
const handleRootFolderModalClose = useCallback(() => {
setIsRootFolderModalOpen(false);
}, []);
const handleRootFolderChange = useCallback(
({
path: newPath,
rootFolderPath: newRootFolderPath,
}: RootFolderUpdated) => {
setIsRootFolderModalOpen(false);
setRootFolderPath(newRootFolderPath);
handleInputChange({ name: 'path', value: newPath });
},
[handleInputChange]
);
const handleCancelPress = useCallback(() => {
setIsConfirmMoveModalOpen(false);
}, []);
@ -183,6 +211,16 @@ function EditMovieModalContent({
type={inputTypes.PATH}
name="path"
{...settings.path}
buttons={[
<FormInputButton
key="fileBrowser"
kind={kinds.DEFAULT}
title={translate('RootFolder')}
onPress={handleRootFolderPress}
>
<Icon name={icons.ROOT_FOLDER} />
</FormInputButton>,
]}
includeFiles={false}
onChange={handleInputChange}
/>
@ -221,6 +259,14 @@ function EditMovieModalContent({
</SpinnerErrorButton>
</ModalFooter>
<RootFolderModal
isOpen={isRootFolderModalOpen}
movieId={movieId}
rootFolderPath={rootFolderPath}
onSavePress={handleRootFolderChange}
onModalClose={handleRootFolderModalClose}
/>
<MoveMovieModal
originalPath={path}
destinationPath={pendingChanges.path}

View file

@ -0,0 +1,30 @@
import React from 'react';
import Modal from 'Components/Modal/Modal';
import RootFolderModalContent, {
RootFolderModalContentProps,
} from './RootFolderModalContent';
interface RootFolderModalProps extends RootFolderModalContentProps {
isOpen: boolean;
}
function RootFolderModal({
isOpen,
rootFolderPath,
movieId,
onSavePress,
onModalClose,
}: RootFolderModalProps) {
return (
<Modal isOpen={isOpen} onModalClose={onModalClose}>
<RootFolderModalContent
movieId={movieId}
rootFolderPath={rootFolderPath}
onSavePress={onSavePress}
onModalClose={onModalClose}
/>
</Modal>
);
}
export default RootFolderModal;

View file

@ -0,0 +1,93 @@
import React, { useCallback, useState } from 'react';
import { useSelector } from 'react-redux';
import FormGroup from 'Components/Form/FormGroup';
import FormInputGroup from 'Components/Form/FormInputGroup';
import FormLabel from 'Components/Form/FormLabel';
import Button from 'Components/Link/Button';
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 useApiQuery from 'Helpers/Hooks/useApiQuery';
import { inputTypes, sizes } from 'Helpers/Props';
import createSystemStatusSelector from 'Store/Selectors/createSystemStatusSelector';
import { InputChanged } from 'typings/inputs';
import translate from 'Utilities/String/translate';
export interface RootFolderUpdated {
path: string;
rootFolderPath: string;
}
export interface RootFolderModalContentProps {
movieId: number;
rootFolderPath: string;
onSavePress(change: RootFolderUpdated): void;
onModalClose(): void;
}
interface MovieFolder {
folder: string;
}
function RootFolderModalContent(props: RootFolderModalContentProps) {
const { movieId, onSavePress, onModalClose } = props;
const { isWindows } = useSelector(createSystemStatusSelector());
const [rootFolderPath, setRootFolderPath] = useState(props.rootFolderPath);
const { isLoading, data } = useApiQuery<MovieFolder>({
url: `/movie/${movieId}/folder`,
});
const onInputChange = useCallback(({ value }: InputChanged<string>) => {
setRootFolderPath(value);
}, []);
const handleSavePress = useCallback(() => {
const separator = isWindows ? '\\' : '/';
onSavePress({
path: `${rootFolderPath}${separator}${data?.folder}`,
rootFolderPath,
});
}, [rootFolderPath, isWindows, data, onSavePress]);
return (
<ModalContent onModalClose={onModalClose}>
<ModalHeader>{translate('UpdateMoviePath')}</ModalHeader>
<ModalBody>
<FormGroup size={sizes.MEDIUM}>
<FormLabel>{translate('RootFolder')}</FormLabel>
<FormInputGroup
type={inputTypes.ROOT_FOLDER_SELECT}
name="rootFolderPath"
value={rootFolderPath}
valueOptions={{
movieFolder: data?.folder,
isWindows,
}}
selectedValueOptions={{
movieFolder: data?.folder,
isWindows,
}}
helpText={translate('MovieEditRootFolderHelpText')}
onChange={onInputChange}
/>
</FormGroup>
</ModalBody>
<ModalFooter>
<Button onPress={onModalClose}>{translate('Cancel')}</Button>
<Button disabled={isLoading || !data?.folder} onPress={handleSavePress}>
{translate('UpdatePath')}
</Button>
</ModalFooter>
</ModalContent>
);
}
export default RootFolderModalContent;

View file

@ -203,7 +203,7 @@ function EditMoviesModalContent(props: EditMoviesModalContentProps) {
includeNoChange={true}
includeNoChangeDisabled={false}
selectedValueOptions={{ includeFreeSpace: false }}
helpText="Moving movies to the same root folder can be used to rename movie folders to match updated title or naming format"
helpText={translate('MovieEditRootFolderHelpText')}
onChange={onInputChange}
/>
</FormGroup>

View file

@ -68,6 +68,7 @@ interface Movie extends ModelBase {
physicalRelease?: string;
digitalRelease?: string;
releaseDate?: string;
rootFolderPath: string;
runtime: number;
minimumAvailability: string;
path: string;

View file

@ -1114,6 +1114,7 @@
"MovieCollectionRootFolderMissingRootHealthCheckMessage": "Missing root folder for movie collection: {rootFolderInfo}",
"MovieDetailsGoTo": "Go to {0}",
"MovieDownloaded": "Movie Downloaded",
"MovieEditRootFolderHelpText": "Moving movies to the same root folder can be used to rename movie folders to match updated title or naming format",
"MovieEditor": "Movie Editor",
"MovieExcludedFromAutomaticAdd": "Movie Excluded From Automatic Add",
"MovieFileDeleted": "Movie File Deleted",
@ -1949,6 +1950,8 @@
"UpdateCheckUINotWritableMessage": "Cannot install update because UI folder '{uiFolder}' is not writable by the user '{userName}'.",
"UpdateFiltered": "Update Filtered",
"UpdateMechanismHelpText": "Use {appName}'s built-in updater or a script",
"UpdateMoviePath": "Update Movie Path",
"UpdatePath": "Update Path",
"UpdateScriptPathHelpText": "Path to a custom script that takes an extracted update package and handle the remainder of the update process",
"UpdateSelected": "Update Selected",
"UpdaterLogFiles": "Updater Log Files",

View file

@ -236,10 +236,10 @@ namespace NzbDrone.Core.RootFolders
{
var osPath = new OsPath(path);
return osPath.Directory.ToString().TrimEnd(osPath.IsUnixPath ? '/' : '\\');
return osPath.Directory.ToString().GetCleanPath();
}
return possibleRootFolder.Path;
return possibleRootFolder.Path.GetCleanPath();
}
}
}

View file

@ -0,0 +1,32 @@
using Microsoft.AspNetCore.Mvc;
using NzbDrone.Core.Movies;
using NzbDrone.Core.Organizer;
using Radarr.Http;
namespace Radarr.Api.V3.Movies;
[V3ApiController("movie")]
public class MovieFolderController : Controller
{
private readonly IMovieService _movieService;
private readonly IBuildFileNames _fileNameBuilder;
public MovieFolderController(IMovieService movieService, IBuildFileNames fileNameBuilder)
{
_movieService = movieService;
_fileNameBuilder = fileNameBuilder;
}
[HttpGet("{id}/folder")]
[Produces("application/json")]
public object GetFolder([FromRoute] int id)
{
var series = _movieService.GetMovie(id);
var folder = _fileNameBuilder.GetMovieFolder(series);
return new
{
folder
};
}
}