mirror of
https://github.com/Sonarr/Sonarr.git
synced 2024-12-16 11:37:58 +02:00
No more wizard, now only show when advanced settings are off
This commit is contained in:
parent
d9b7dd257e
commit
5659a3c496
@ -10,5 +10,12 @@ public class NamingConfigResource : RestResource
|
|||||||
public string StandardEpisodeFormat { get; set; }
|
public string StandardEpisodeFormat { get; set; }
|
||||||
public string DailyEpisodeFormat { get; set; }
|
public string DailyEpisodeFormat { get; set; }
|
||||||
public string SeasonFolderFormat { get; set; }
|
public string SeasonFolderFormat { get; set; }
|
||||||
|
|
||||||
|
public bool IncludeSeriesTitle { get; set; }
|
||||||
|
public bool IncludeEpisodeTitle { get; set; }
|
||||||
|
public bool IncludeQuality { get; set; }
|
||||||
|
public bool ReplaceSpaces { get; set; }
|
||||||
|
public string Separator { get; set; }
|
||||||
|
public string NumberStyle { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -8,6 +8,7 @@
|
|||||||
using Nancy.ModelBinding;
|
using Nancy.ModelBinding;
|
||||||
using NzbDrone.Api.Mapping;
|
using NzbDrone.Api.Mapping;
|
||||||
using NzbDrone.Api.Extensions;
|
using NzbDrone.Api.Extensions;
|
||||||
|
using Omu.ValueInjecter;
|
||||||
|
|
||||||
namespace NzbDrone.Api.Config
|
namespace NzbDrone.Api.Config
|
||||||
{
|
{
|
||||||
@ -16,15 +17,18 @@ public class NamingModule : NzbDroneRestModule<NamingConfigResource>
|
|||||||
private readonly INamingConfigService _namingConfigService;
|
private readonly INamingConfigService _namingConfigService;
|
||||||
private readonly IFilenameSampleService _filenameSampleService;
|
private readonly IFilenameSampleService _filenameSampleService;
|
||||||
private readonly IFilenameValidationService _filenameValidationService;
|
private readonly IFilenameValidationService _filenameValidationService;
|
||||||
|
private readonly IBuildFileNames _filenameBuilder;
|
||||||
|
|
||||||
public NamingModule(INamingConfigService namingConfigService,
|
public NamingModule(INamingConfigService namingConfigService,
|
||||||
IFilenameSampleService filenameSampleService,
|
IFilenameSampleService filenameSampleService,
|
||||||
IFilenameValidationService filenameValidationService)
|
IFilenameValidationService filenameValidationService,
|
||||||
|
IBuildFileNames filenameBuilder)
|
||||||
: base("config/naming")
|
: base("config/naming")
|
||||||
{
|
{
|
||||||
_namingConfigService = namingConfigService;
|
_namingConfigService = namingConfigService;
|
||||||
_filenameSampleService = filenameSampleService;
|
_filenameSampleService = filenameSampleService;
|
||||||
_filenameValidationService = filenameValidationService;
|
_filenameValidationService = filenameValidationService;
|
||||||
|
_filenameBuilder = filenameBuilder;
|
||||||
GetResourceSingle = GetNamingConfig;
|
GetResourceSingle = GetNamingConfig;
|
||||||
GetResourceById = GetNamingConfig;
|
GetResourceById = GetNamingConfig;
|
||||||
UpdateResource = UpdateNamingConfig;
|
UpdateResource = UpdateNamingConfig;
|
||||||
@ -50,7 +54,12 @@ private void UpdateNamingConfig(NamingConfigResource resource)
|
|||||||
|
|
||||||
private NamingConfigResource GetNamingConfig()
|
private NamingConfigResource GetNamingConfig()
|
||||||
{
|
{
|
||||||
return _namingConfigService.GetConfig().InjectTo<NamingConfigResource>();
|
var nameSpec = _namingConfigService.GetConfig();
|
||||||
|
var resource = nameSpec.InjectTo<NamingConfigResource>();
|
||||||
|
var basicConfig = _filenameBuilder.GetBasicNamingConfig(nameSpec);
|
||||||
|
resource.InjectFrom(basicConfig);
|
||||||
|
|
||||||
|
return resource;
|
||||||
}
|
}
|
||||||
|
|
||||||
private NamingConfigResource GetNamingConfig(int id)
|
private NamingConfigResource GetNamingConfig(int id)
|
||||||
|
@ -324,6 +324,7 @@
|
|||||||
<Compile Include="Notifications\Xbmc\Model\VersionResult.cs" />
|
<Compile Include="Notifications\Xbmc\Model\VersionResult.cs" />
|
||||||
<Compile Include="Notifications\Xbmc\Model\XbmcJsonResult.cs" />
|
<Compile Include="Notifications\Xbmc\Model\XbmcJsonResult.cs" />
|
||||||
<Compile Include="Notifications\Xbmc\Model\XbmcVersion.cs" />
|
<Compile Include="Notifications\Xbmc\Model\XbmcVersion.cs" />
|
||||||
|
<Compile Include="Organizer\BasicNamingConfig.cs" />
|
||||||
<Compile Include="Organizer\FilenameValidationService.cs" />
|
<Compile Include="Organizer\FilenameValidationService.cs" />
|
||||||
<Compile Include="Organizer\EpisodeFormat.cs" />
|
<Compile Include="Organizer\EpisodeFormat.cs" />
|
||||||
<Compile Include="Organizer\Exception.cs" />
|
<Compile Include="Organizer\Exception.cs" />
|
||||||
|
17
src/NzbDrone.Core/Organizer/BasicNamingConfig.cs
Normal file
17
src/NzbDrone.Core/Organizer/BasicNamingConfig.cs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Organizer
|
||||||
|
{
|
||||||
|
public class BasicNamingConfig
|
||||||
|
{
|
||||||
|
public bool IncludeSeriesTitle { get; set; }
|
||||||
|
public bool IncludeEpisodeTitle { get; set; }
|
||||||
|
public bool IncludeQuality { get; set; }
|
||||||
|
public bool ReplaceSpaces { get; set; }
|
||||||
|
public string Separator { get; set; }
|
||||||
|
public string NumberStyle { get; set; }
|
||||||
|
}
|
||||||
|
}
|
@ -16,6 +16,7 @@ public interface IBuildFileNames
|
|||||||
string BuildFilename(IList<Episode> episodes, Series series, EpisodeFile episodeFile);
|
string BuildFilename(IList<Episode> episodes, Series series, EpisodeFile episodeFile);
|
||||||
string BuildFilename(IList<Episode> episodes, Series series, EpisodeFile episodeFile, NamingConfig namingConfig);
|
string BuildFilename(IList<Episode> episodes, Series series, EpisodeFile episodeFile, NamingConfig namingConfig);
|
||||||
string BuildFilePath(Series series, int seasonNumber, string fileName, string extension);
|
string BuildFilePath(Series series, int seasonNumber, string fileName, string extension);
|
||||||
|
BasicNamingConfig GetBasicNamingConfig(NamingConfig nameSpec);
|
||||||
}
|
}
|
||||||
|
|
||||||
public class FileNameBuilder : IBuildFileNames
|
public class FileNameBuilder : IBuildFileNames
|
||||||
@ -25,7 +26,7 @@ public class FileNameBuilder : IBuildFileNames
|
|||||||
private readonly ICached<EpisodeFormat> _patternCache;
|
private readonly ICached<EpisodeFormat> _patternCache;
|
||||||
private readonly Logger _logger;
|
private readonly Logger _logger;
|
||||||
|
|
||||||
private static readonly Regex TitleRegex = new Regex(@"(?<token>\{(?:\w+)(?<separator>\s|\W|_)\w+\})",
|
private static readonly Regex TitleRegex = new Regex(@"(?<token>\{(?:\w+)(?<separator>\s|\.|-|_)\w+\})",
|
||||||
RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
private static readonly Regex EpisodeRegex = new Regex(@"(?<episode>\{episode(?:\:0+)?})",
|
private static readonly Regex EpisodeRegex = new Regex(@"(?<episode>\{episode(?:\:0+)?})",
|
||||||
@ -176,6 +177,47 @@ public string BuildFilePath(Series series, int seasonNumber, string fileName, st
|
|||||||
return Path.Combine(path, fileName + extension);
|
return Path.Combine(path, fileName + extension);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public BasicNamingConfig GetBasicNamingConfig(NamingConfig nameSpec)
|
||||||
|
{
|
||||||
|
var episodeFormat = GetEpisodeFormat(nameSpec.StandardEpisodeFormat);
|
||||||
|
|
||||||
|
var basicNamingConfig = new BasicNamingConfig
|
||||||
|
{
|
||||||
|
Separator = episodeFormat.Separator,
|
||||||
|
NumberStyle = episodeFormat.SeasonEpisodePattern
|
||||||
|
};
|
||||||
|
|
||||||
|
var titleTokens = TitleRegex.Matches(nameSpec.StandardEpisodeFormat);
|
||||||
|
|
||||||
|
foreach (Match match in titleTokens)
|
||||||
|
{
|
||||||
|
var separator = match.Groups["separator"].Value;
|
||||||
|
var token = match.Groups["token"].Value;
|
||||||
|
|
||||||
|
if (!separator.Equals(" "))
|
||||||
|
{
|
||||||
|
basicNamingConfig.ReplaceSpaces = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.StartsWith("{Series", StringComparison.InvariantCultureIgnoreCase))
|
||||||
|
{
|
||||||
|
basicNamingConfig.IncludeSeriesTitle = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.StartsWith("{Episode", StringComparison.InvariantCultureIgnoreCase))
|
||||||
|
{
|
||||||
|
basicNamingConfig.IncludeEpisodeTitle = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.StartsWith("{Quality", StringComparison.InvariantCultureIgnoreCase))
|
||||||
|
{
|
||||||
|
basicNamingConfig.IncludeQuality = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return basicNamingConfig;
|
||||||
|
}
|
||||||
|
|
||||||
public static string CleanFilename(string name)
|
public static string CleanFilename(string name)
|
||||||
{
|
{
|
||||||
string result = name;
|
string result = name;
|
||||||
|
102
src/UI/Settings/MediaManagement/Naming/Basic/BasicNamingView.js
Normal file
102
src/UI/Settings/MediaManagement/Naming/Basic/BasicNamingView.js
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
'use strict';
|
||||||
|
define(
|
||||||
|
[
|
||||||
|
'underscore',
|
||||||
|
'vent',
|
||||||
|
'marionette',
|
||||||
|
'Settings/MediaManagement/Naming/NamingSampleModel',
|
||||||
|
'Settings/MediaManagement/Naming/Wizard/NamingWizardModel',
|
||||||
|
'Mixins/AsModelBoundView'
|
||||||
|
], function (_, vent, Marionette, NamingSampleModel, NamingWizardModel, AsModelBoundView) {
|
||||||
|
|
||||||
|
var view = Marionette.ItemView.extend({
|
||||||
|
template: 'Settings/MediaManagement/Naming/Basic/BasicNamingViewTemplate',
|
||||||
|
|
||||||
|
ui: {
|
||||||
|
namingOptions : '.x-naming-options',
|
||||||
|
singleEpisodeExample : '.x-single-episode-example',
|
||||||
|
multiEpisodeExample : '.x-multi-episode-example',
|
||||||
|
dailyEpisodeExample : '.x-daily-episode-example'
|
||||||
|
},
|
||||||
|
|
||||||
|
onRender: function () {
|
||||||
|
this.listenTo(this.model, 'change', this._buildFormat);
|
||||||
|
this._buildFormat();
|
||||||
|
},
|
||||||
|
|
||||||
|
_updateSamples: function () {
|
||||||
|
var data = {
|
||||||
|
renameEpisodes: true,
|
||||||
|
standardEpisodeFormat: this.standardEpisodeFormat,
|
||||||
|
dailyEpisodeFormat: this.dailyEpisodeFormat,
|
||||||
|
multiEpisodeStyle: this.model.get('multiEpisodeStyle')
|
||||||
|
};
|
||||||
|
|
||||||
|
this.namingSampleModel.fetch({data: data});
|
||||||
|
},
|
||||||
|
|
||||||
|
_buildFormat: function () {
|
||||||
|
if (_.has(this.model.changed, 'standardEpisodeFormat') || _.has(this.model.changed, 'dailyEpisodeFormat')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var standardEpisodeFormat = '';
|
||||||
|
var dailyEpisodeFormat = '';
|
||||||
|
|
||||||
|
if (this.model.get('includeSeriesTitle')) {
|
||||||
|
if (this.model.get('replaceSpaces')) {
|
||||||
|
standardEpisodeFormat += '{Series.Title}';
|
||||||
|
dailyEpisodeFormat += '{Series.Title}';
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
standardEpisodeFormat += '{Series Title}';
|
||||||
|
dailyEpisodeFormat += '{Series Title}';
|
||||||
|
}
|
||||||
|
|
||||||
|
standardEpisodeFormat += this.model.get('separator');
|
||||||
|
dailyEpisodeFormat += this.model.get('separator');
|
||||||
|
}
|
||||||
|
|
||||||
|
standardEpisodeFormat += this.model.get('numberStyle');
|
||||||
|
dailyEpisodeFormat += '{Air-Date}';
|
||||||
|
|
||||||
|
if (this.model.get('includeEpisodeTitle')) {
|
||||||
|
standardEpisodeFormat += this.model.get('separator');
|
||||||
|
dailyEpisodeFormat += this.model.get('separator');
|
||||||
|
|
||||||
|
if (this.model.get('replaceSpaces')) {
|
||||||
|
standardEpisodeFormat += '{Episode.Title}';
|
||||||
|
dailyEpisodeFormat += '{Episode.Title}';
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
standardEpisodeFormat += '{Episode Title}';
|
||||||
|
dailyEpisodeFormat += '{Episode Title}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.model.get('includeQuality')) {
|
||||||
|
if (this.model.get('replaceSpaces')) {
|
||||||
|
standardEpisodeFormat += ' {Quality.Title}';
|
||||||
|
dailyEpisodeFormat += ' {Quality.Title}';
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
standardEpisodeFormat += ' {Quality Title}';
|
||||||
|
dailyEpisodeFormat += ' {Quality Title}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.model.get('replaceSpaces')) {
|
||||||
|
standardEpisodeFormat = standardEpisodeFormat.replace(/\s/g, '.');
|
||||||
|
dailyEpisodeFormat = dailyEpisodeFormat.replace(/\s/g, '.');
|
||||||
|
}
|
||||||
|
|
||||||
|
this.model.set('standardEpisodeFormat', standardEpisodeFormat);
|
||||||
|
this.model.set('dailyEpisodeFormat', dailyEpisodeFormat);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return AsModelBoundView.call(view);
|
||||||
|
});
|
@ -0,0 +1,92 @@
|
|||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Include Series Title</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<label class="checkbox toggle well">
|
||||||
|
<input type="checkbox" name="includeSeriesTitle"/>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<span>Yes</span>
|
||||||
|
<span>No</span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="btn btn-primary slide-button"/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Include Episode Title</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<label class="checkbox toggle well">
|
||||||
|
<input type="checkbox" name="includeEpisodeTitle"/>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<span>Yes</span>
|
||||||
|
<span>No</span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="btn btn-primary slide-button"/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Include Quality</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<label class="checkbox toggle well">
|
||||||
|
<input type="checkbox" name="includeQuality"/>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<span>Yes</span>
|
||||||
|
<span>No</span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="btn btn-primary slide-button"/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Replace Spaces</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<label class="checkbox toggle well">
|
||||||
|
<input type="checkbox" name="replaceSpaces"/>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<span>Yes</span>
|
||||||
|
<span>No</span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="btn btn-primary slide-button"/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Separator</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<select class="inputClass" name="separator">
|
||||||
|
<option value=" - ">Dash</option>
|
||||||
|
<option value=" ">Space</option>
|
||||||
|
<option value=".">Period</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label">Numbering Style</label>
|
||||||
|
|
||||||
|
<div class="controls">
|
||||||
|
<select class="inputClass" name="numberStyle">
|
||||||
|
<option value="{season}x{episode:00}">1x05</option>
|
||||||
|
<option value="{season:00}x{episode:00}">01x05</option>
|
||||||
|
<option value="S{season:00}E{episode:00}">S01E05</option>
|
||||||
|
<option value="s{season:00}e{episode:00}">s01e05</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
@ -1,15 +1,16 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
define(
|
define(
|
||||||
[
|
[
|
||||||
'vent',
|
'underscore',
|
||||||
'marionette',
|
'marionette',
|
||||||
|
'Config',
|
||||||
'Settings/MediaManagement/Naming/NamingSampleModel',
|
'Settings/MediaManagement/Naming/NamingSampleModel',
|
||||||
'Settings/MediaManagement/Naming/Wizard/NamingWizardView',
|
'Settings/MediaManagement/Naming/Basic/BasicNamingView',
|
||||||
'Mixins/AsModelBoundView',
|
'Mixins/AsModelBoundView',
|
||||||
'Mixins/AsValidatedView'
|
'Mixins/AsValidatedView'
|
||||||
], function (vent, Marionette, NamingSampleModel, NamingWizardView, AsModelBoundView, AsValidatedView) {
|
], function (_, Marionette, Config, NamingSampleModel, BasicNamingView, AsModelBoundView, AsValidatedView) {
|
||||||
|
|
||||||
var view = Marionette.ItemView.extend({
|
var view = Marionette.Layout.extend({
|
||||||
template: 'Settings/MediaManagement/Naming/NamingViewTemplate',
|
template: 'Settings/MediaManagement/Naming/NamingViewTemplate',
|
||||||
|
|
||||||
ui: {
|
ui: {
|
||||||
@ -27,11 +28,17 @@ define(
|
|||||||
'click .x-naming-token-helper a' : '_addToken'
|
'click .x-naming-token-helper a' : '_addToken'
|
||||||
},
|
},
|
||||||
|
|
||||||
|
regions: {
|
||||||
|
basicNamingRegion: '.x-basic-naming'
|
||||||
|
},
|
||||||
|
|
||||||
onRender: function () {
|
onRender: function () {
|
||||||
if (!this.model.has('renameEpisodes')) {
|
if (!this.model.get('renameEpisodes')) {
|
||||||
this.ui.namingOptions.hide();
|
this.ui.namingOptions.hide();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.basicNamingView = new BasicNamingView({ model: this.model });
|
||||||
|
this.basicNamingRegion.show(this.basicNamingView);
|
||||||
this.namingSampleModel = new NamingSampleModel();
|
this.namingSampleModel = new NamingSampleModel();
|
||||||
|
|
||||||
this.listenTo(this.model, 'change', this._updateSamples);
|
this.listenTo(this.model, 'change', this._updateSamples);
|
||||||
@ -51,6 +58,10 @@ define(
|
|||||||
},
|
},
|
||||||
|
|
||||||
_updateSamples: function () {
|
_updateSamples: function () {
|
||||||
|
if (!_.has(this.model.changed, 'standardEpisodeFormat') && !_.has(this.model.changed, 'dailyEpisodeFormat')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.namingSampleModel.fetch({ data: this.model.toJSON() });
|
this.namingSampleModel.fetch({ data: this.model.toJSON() });
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -60,14 +71,6 @@ define(
|
|||||||
this.ui.dailyEpisodeExample.html(this.namingSampleModel.get('dailyEpisodeExample'));
|
this.ui.dailyEpisodeExample.html(this.namingSampleModel.get('dailyEpisodeExample'));
|
||||||
},
|
},
|
||||||
|
|
||||||
_showWizard: function () {
|
|
||||||
var modalView = new NamingWizardView();
|
|
||||||
vent.trigger(vent.Commands.OpenModalCommand, modalView);
|
|
||||||
this.listenTo(modalView, modalView.formatsUpdated, this._updateFormats);
|
|
||||||
|
|
||||||
vent.trigger(vent.Commands.ShowNamingWizard, { model: this.model });
|
|
||||||
},
|
|
||||||
|
|
||||||
_addToken: function (e) {
|
_addToken: function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
@ -88,12 +91,6 @@ define(
|
|||||||
|
|
||||||
this.ui.namingTokenHelper.removeClass('open');
|
this.ui.namingTokenHelper.removeClass('open');
|
||||||
input.focus();
|
input.focus();
|
||||||
},
|
|
||||||
|
|
||||||
_updateFormats: function (options) {
|
|
||||||
this.model.set('standardEpisodeFormat', options.standardEpisodeFormat);
|
|
||||||
this.model.set('dailyEpisodeFormat', options.dailyEpisodeFormat);
|
|
||||||
this.model.set('multiEpisodeStyle', options.multiEpisodeStyle);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -23,17 +23,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="x-naming-options">
|
<div class="x-naming-options">
|
||||||
<div class="control-group">
|
<div class="basic-setting x-basic-naming"></div>
|
||||||
<label class="control-label">Wizard</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<button class="btn x-show-wizard">Show Wizard</button>
|
|
||||||
|
|
||||||
<span class="help-inline">
|
|
||||||
<i class="icon-nd-form-info" title="Wizard to setup preferred naming style"/>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group advanced-setting">
|
<div class="control-group advanced-setting">
|
||||||
<label class="control-label">Standard Episode Format</label>
|
<label class="control-label">Standard Episode Format</label>
|
||||||
@ -90,7 +80,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="control-group advanced-setting">
|
<div class="control-group">
|
||||||
<label class="control-label">Multi-Episode Style</label>
|
<label class="control-label">Multi-Episode Style</label>
|
||||||
|
|
||||||
<div class="controls">
|
<div class="controls">
|
||||||
@ -108,10 +98,19 @@
|
|||||||
<label class="control-label">Season Folder Format</label>
|
<label class="control-label">Season Folder Format</label>
|
||||||
|
|
||||||
<div class="controls">
|
<div class="controls">
|
||||||
<input type="text" placeholder="Season {season}" name="seasonFolderFormat"/>
|
<div class="input-append x-helper-input">
|
||||||
<span class="help-inline">
|
<input type="text" class="naming-format" name="seasonFolderFormat"/>
|
||||||
<i class="icon-question-sign" title="How should season folders be named? Please see the wiki for customization options"/>
|
<div class="btn-group x-naming-token-helper">
|
||||||
</span>
|
<button class="btn btn-icon-only dropdown-toggle" data-toggle="dropdown">
|
||||||
|
<i class="icon-plus"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="dropdown-menu">
|
||||||
|
{{> SeriesTitleNamingPartial}}
|
||||||
|
{{> SeasonNamingPartial}}
|
||||||
|
{{> SeparatorNamingPartial}}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -1,17 +0,0 @@
|
|||||||
'use strict';
|
|
||||||
define(
|
|
||||||
[
|
|
||||||
'backbone'
|
|
||||||
], function (Backbone) {
|
|
||||||
return Backbone.Model.extend({
|
|
||||||
defaults: {
|
|
||||||
includeSeriesTitle : true,
|
|
||||||
includeEpisodeTitle: true,
|
|
||||||
includeQuality : true,
|
|
||||||
replaceSpaces : false,
|
|
||||||
separator : ' - ',
|
|
||||||
numberStyle : 'S{season:00}E{episode:00}',
|
|
||||||
multiEpisodeStyle : 0
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
@ -1,127 +0,0 @@
|
|||||||
'use strict';
|
|
||||||
define(
|
|
||||||
[
|
|
||||||
'vent',
|
|
||||||
'marionette',
|
|
||||||
'Settings/MediaManagement/Naming/NamingSampleModel',
|
|
||||||
'Settings/MediaManagement/Naming/Wizard/NamingWizardModel',
|
|
||||||
'Mixins/AsModelBoundView'
|
|
||||||
], function (vent, Marionette, NamingSampleModel, NamingWizardModel, AsModelBoundView) {
|
|
||||||
|
|
||||||
var view = Marionette.ItemView.extend({
|
|
||||||
template: 'Settings/MediaManagement/Naming/Wizard/NamingWizardViewTemplate',
|
|
||||||
|
|
||||||
ui: {
|
|
||||||
namingOptions : '.x-naming-options',
|
|
||||||
singleEpisodeExample : '.x-single-episode-example',
|
|
||||||
multiEpisodeExample : '.x-multi-episode-example',
|
|
||||||
dailyEpisodeExample : '.x-daily-episode-example'
|
|
||||||
},
|
|
||||||
|
|
||||||
events: {
|
|
||||||
'click .x-apply': '_applyNaming'
|
|
||||||
},
|
|
||||||
|
|
||||||
formatsUpdated: 'formatsUpdated',
|
|
||||||
|
|
||||||
initialize: function () {
|
|
||||||
this.model = new NamingWizardModel();
|
|
||||||
this.namingSampleModel = new NamingSampleModel();
|
|
||||||
},
|
|
||||||
|
|
||||||
onRender: function () {
|
|
||||||
this.listenTo(this.model, 'change', this._buildFormat);
|
|
||||||
this.listenTo(this.namingSampleModel, 'sync', this._showSamples);
|
|
||||||
this._buildFormat();
|
|
||||||
},
|
|
||||||
|
|
||||||
_updateSamples: function () {
|
|
||||||
var data = {
|
|
||||||
renameEpisodes: true,
|
|
||||||
standardEpisodeFormat: this.standardEpisodeFormat,
|
|
||||||
dailyEpisodeFormat: this.dailyEpisodeFormat,
|
|
||||||
multiEpisodeStyle: this.model.get('multiEpisodeStyle')
|
|
||||||
};
|
|
||||||
|
|
||||||
this.namingSampleModel.fetch({data: data});
|
|
||||||
},
|
|
||||||
|
|
||||||
_showSamples: function () {
|
|
||||||
this.ui.singleEpisodeExample.html(this.namingSampleModel.get('singleEpisodeExample'));
|
|
||||||
this.ui.multiEpisodeExample.html(this.namingSampleModel.get('multiEpisodeExample'));
|
|
||||||
this.ui.dailyEpisodeExample.html(this.namingSampleModel.get('dailyEpisodeExample'));
|
|
||||||
},
|
|
||||||
|
|
||||||
_applyNaming: function () {
|
|
||||||
var options = {
|
|
||||||
standardEpisodeFormat: this.standardEpisodeFormat,
|
|
||||||
dailyEpisodeFormat: this.dailyEpisodeFormat,
|
|
||||||
multiEpisodeStyle: this.model.get('multiEpisodeStyle')
|
|
||||||
};
|
|
||||||
|
|
||||||
this.trigger(this.formatsUpdated, options);
|
|
||||||
|
|
||||||
|
|
||||||
vent.trigger(vent.Commands.CloseModalCommand);
|
|
||||||
},
|
|
||||||
|
|
||||||
_buildFormat: function () {
|
|
||||||
this.standardEpisodeFormat = '';
|
|
||||||
this.dailyEpisodeFormat = '';
|
|
||||||
|
|
||||||
if (this.model.get('includeSeriesTitle')) {
|
|
||||||
if (this.model.get('replaceSpaces')) {
|
|
||||||
this.standardEpisodeFormat += '{Series.Title}';
|
|
||||||
this.dailyEpisodeFormat += '{Series.Title}';
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
this.standardEpisodeFormat += '{Series Title}';
|
|
||||||
this.dailyEpisodeFormat += '{Series Title}';
|
|
||||||
}
|
|
||||||
|
|
||||||
this.standardEpisodeFormat += this.model.get('separator');
|
|
||||||
this.dailyEpisodeFormat += this.model.get('separator');
|
|
||||||
}
|
|
||||||
|
|
||||||
this.standardEpisodeFormat += this.model.get('numberStyle');
|
|
||||||
this.dailyEpisodeFormat += '{Air-Date}';
|
|
||||||
|
|
||||||
if (this.model.get('includeEpisodeTitle')) {
|
|
||||||
this.standardEpisodeFormat += this.model.get('separator');
|
|
||||||
this.dailyEpisodeFormat += this.model.get('separator');
|
|
||||||
|
|
||||||
if (this.model.get('replaceSpaces')) {
|
|
||||||
this.standardEpisodeFormat += '{Episode.Title}';
|
|
||||||
this.dailyEpisodeFormat += '{Episode.Title}';
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
this.standardEpisodeFormat += '{Episode Title}';
|
|
||||||
this.dailyEpisodeFormat += '{Episode Title}';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.model.get('includeQuality')) {
|
|
||||||
if (this.model.get('replaceSpaces')) {
|
|
||||||
this.standardEpisodeFormat += ' {Quality.Title}';
|
|
||||||
this.dailyEpisodeFormat += ' {Quality.Title}';
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
this.standardEpisodeFormat += ' {Quality Title}';
|
|
||||||
this.dailyEpisodeFormat += ' {Quality Title}';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.model.get('replaceSpaces')) {
|
|
||||||
this.standardEpisodeFormat = this.standardEpisodeFormat.replace(/\s/g, '.');
|
|
||||||
this.dailyEpisodeFormat = this.dailyEpisodeFormat.replace(/\s/g, '.');
|
|
||||||
}
|
|
||||||
|
|
||||||
this._updateSamples();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return AsModelBoundView.call(view);
|
|
||||||
});
|
|
@ -1,141 +0,0 @@
|
|||||||
<div class="modal-header">
|
|
||||||
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button>
|
|
||||||
<h3>Naming Wizard</h3>
|
|
||||||
</div>
|
|
||||||
<div class="modal-body">
|
|
||||||
<div class="form-horizontal">
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Include Series Title</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<label class="checkbox toggle well">
|
|
||||||
<input type="checkbox" name="includeSeriesTitle"/>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<span>Yes</span>
|
|
||||||
<span>No</span>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div class="btn btn-primary slide-button"/>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Include Episode Title</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<label class="checkbox toggle well">
|
|
||||||
<input type="checkbox" name="includeEpisodeTitle"/>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<span>Yes</span>
|
|
||||||
<span>No</span>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div class="btn btn-primary slide-button"/>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Include Quality</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<label class="checkbox toggle well">
|
|
||||||
<input type="checkbox" name="includeQuality"/>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<span>Yes</span>
|
|
||||||
<span>No</span>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div class="btn btn-primary slide-button"/>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Replace Spaces</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<label class="checkbox toggle well">
|
|
||||||
<input type="checkbox" name="replaceSpaces"/>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<span>Yes</span>
|
|
||||||
<span>No</span>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div class="btn btn-primary slide-button"/>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Separator</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<select class="inputClass" name="separator">
|
|
||||||
<option value=" - ">Dash</option>
|
|
||||||
<option value=" ">Space</option>
|
|
||||||
<option value=".">Period</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Numbering Style</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<select class="inputClass" name="numberStyle">
|
|
||||||
<option value="{season}x{episode:00}">1x05</option>
|
|
||||||
<option value="{season:00}x{episode:00}">01x05</option>
|
|
||||||
<option value="S{season:00}E{episode:00}">S01E05</option>
|
|
||||||
<option value="s{season:00}e{episode:00}">s01e05</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Multi-Episode Style</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<select class="inputClass" name="multiEpisodeStyle">
|
|
||||||
<option value="0">Extend</option>
|
|
||||||
<option value="1">Duplicate</option>
|
|
||||||
<option value="2">Repeat</option>
|
|
||||||
<option value="3">Scene</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Single Episode Example</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<span class="x-single-episode-example naming-example"></span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Multi-Episode Example</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<span class="x-multi-episode-example naming-example"></span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="control-group">
|
|
||||||
<label class="control-label">Daily-Episode Example</label>
|
|
||||||
|
|
||||||
<div class="controls">
|
|
||||||
<span class="x-daily-episode-example naming-example"></span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="modal-footer">
|
|
||||||
<button class="btn" data-dismiss="modal">cancel</button>
|
|
||||||
<button class="btn btn-primary x-apply">apply</button>
|
|
||||||
</div>
|
|
@ -6,7 +6,7 @@ define(
|
|||||||
'backbone',
|
'backbone',
|
||||||
'Settings/SettingsModel',
|
'Settings/SettingsModel',
|
||||||
'Settings/General/GeneralSettingsModel',
|
'Settings/General/GeneralSettingsModel',
|
||||||
'Settings/MediaManagement/Naming/Model',
|
'Settings/MediaManagement/Naming/NamingModel',
|
||||||
'Settings/MediaManagement/MediaManagementLayout',
|
'Settings/MediaManagement/MediaManagementLayout',
|
||||||
'Settings/Quality/QualityLayout',
|
'Settings/Quality/QualityLayout',
|
||||||
'Settings/Indexers/IndexerLayout',
|
'Settings/Indexers/IndexerLayout',
|
||||||
|
@ -77,8 +77,16 @@ li.save-and-add:hover {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.basic-setting {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
.show-advanced-settings {
|
.show-advanced-settings {
|
||||||
.advanced-setting {
|
.advanced-setting {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.basic-setting {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user