) => 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 ? (
-
+ <>
@@ -245,7 +251,7 @@ export function PathInputInternal(props: PathInputInternalProps) {
onChange={onChange}
onModalClose={handleFileBrowserModalClose}
/>
-
+ >
) : null}
);
diff --git a/frontend/src/Helpers/Props/icons.ts b/frontend/src/Helpers/Props/icons.ts
index b43420bd07..35c6217ebc 100644
--- a/frontend/src/Helpers/Props/icons.ts
+++ b/frontend/src/Helpers/Props/icons.ts
@@ -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;
diff --git a/frontend/src/Movie/Edit/EditMovieModalContent.tsx b/frontend/src/Movie/Edit/EditMovieModalContent.tsx
index 4ee7a281b1..d0b347dfc6 100644
--- a/frontend/src/Movie/Edit/EditMovieModalContent.tsx
+++ b/frontend/src/Movie/Edit/EditMovieModalContent.tsx
@@ -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={[
+
+
+ ,
+ ]}
includeFiles={false}
onChange={handleInputChange}
/>
@@ -221,6 +259,14 @@ function EditMovieModalContent({
+
+
+
+
+ );
+}
+
+export default RootFolderModal;
diff --git a/frontend/src/Movie/Edit/RootFolder/RootFolderModalContent.tsx b/frontend/src/Movie/Edit/RootFolder/RootFolderModalContent.tsx
new file mode 100644
index 0000000000..f7de5870cd
--- /dev/null
+++ b/frontend/src/Movie/Edit/RootFolder/RootFolderModalContent.tsx
@@ -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({
+ url: `/movie/${movieId}/folder`,
+ });
+
+ const onInputChange = useCallback(({ value }: InputChanged) => {
+ setRootFolderPath(value);
+ }, []);
+
+ const handleSavePress = useCallback(() => {
+ const separator = isWindows ? '\\' : '/';
+
+ onSavePress({
+ path: `${rootFolderPath}${separator}${data?.folder}`,
+ rootFolderPath,
+ });
+ }, [rootFolderPath, isWindows, data, onSavePress]);
+
+ return (
+
+ {translate('UpdateMoviePath')}
+
+
+
+ {translate('RootFolder')}
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
+
+export default RootFolderModalContent;
diff --git a/frontend/src/Movie/Index/Select/Edit/EditMoviesModalContent.tsx b/frontend/src/Movie/Index/Select/Edit/EditMoviesModalContent.tsx
index 09a6414e7f..6736943209 100644
--- a/frontend/src/Movie/Index/Select/Edit/EditMoviesModalContent.tsx
+++ b/frontend/src/Movie/Index/Select/Edit/EditMoviesModalContent.tsx
@@ -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}
/>
diff --git a/frontend/src/Movie/Movie.ts b/frontend/src/Movie/Movie.ts
index d261f04eb0..6946fb41d4 100644
--- a/frontend/src/Movie/Movie.ts
+++ b/frontend/src/Movie/Movie.ts
@@ -68,6 +68,7 @@ interface Movie extends ModelBase {
physicalRelease?: string;
digitalRelease?: string;
releaseDate?: string;
+ rootFolderPath: string;
runtime: number;
minimumAvailability: string;
path: string;
diff --git a/src/NzbDrone.Core/Localization/Core/en.json b/src/NzbDrone.Core/Localization/Core/en.json
index 86460577d9..06594cf015 100644
--- a/src/NzbDrone.Core/Localization/Core/en.json
+++ b/src/NzbDrone.Core/Localization/Core/en.json
@@ -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",
diff --git a/src/NzbDrone.Core/RootFolders/RootFolderService.cs b/src/NzbDrone.Core/RootFolders/RootFolderService.cs
index fd6ccb9b14..3277a16071 100644
--- a/src/NzbDrone.Core/RootFolders/RootFolderService.cs
+++ b/src/NzbDrone.Core/RootFolders/RootFolderService.cs
@@ -236,10 +236,10 @@ private string GetBestRootFolderPathInternal(string path, List rootF
{
var osPath = new OsPath(path);
- return osPath.Directory.ToString().TrimEnd(osPath.IsUnixPath ? '/' : '\\');
+ return osPath.Directory.ToString().GetCleanPath();
}
- return possibleRootFolder.Path;
+ return possibleRootFolder.Path.GetCleanPath();
}
}
}
diff --git a/src/Radarr.Api.V3/Movies/MovieFolderController.cs b/src/Radarr.Api.V3/Movies/MovieFolderController.cs
new file mode 100644
index 0000000000..a1ddfd2109
--- /dev/null
+++ b/src/Radarr.Api.V3/Movies/MovieFolderController.cs
@@ -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
+ };
+ }
+}