// --------------------------------------------------------------------------------------------------------------------
//
// This file is part of the HandBrake source code - It may be used under the terms of the GNU General Public License.
//
//
// The Subtitles View Model
//
// --------------------------------------------------------------------------------------------------------------------
namespace HandBrakeWPF.ViewModels
{
using System.Collections.Generic;
using System.ComponentModel;
using System.IO;
using System.Linq;
using HandBrake.ApplicationServices.Utilities;
using HandBrakeWPF.Model.Subtitles;
using HandBrakeWPF.Properties;
using HandBrakeWPF.Services.Presets.Model;
using HandBrakeWPF.Services.Scan.Model;
using HandBrakeWPF.Utilities;
using HandBrakeWPF.ViewModels.Interfaces;
using Microsoft.Win32;
using EncodeTask = HandBrakeWPF.Services.Encode.Model.EncodeTask;
using OutputFormat = HandBrakeWPF.Services.Encode.Model.Models.OutputFormat;
using SubtitleTrack = HandBrakeWPF.Services.Encode.Model.Models.SubtitleTrack;
using SubtitleType = HandBrakeWPF.Services.Encode.Model.Models.SubtitleType;
///
/// The Subtitles View Model
///
public class SubtitlesViewModel : ViewModelBase, ISubtitlesViewModel
{
#region Constants and Fields
///
/// The Foreign Audio Search Track
///
private readonly Subtitle ForeignAudioSearchTrack;
///
/// Backing field for the source subtitle tracks.
///
private IList sourceTracks;
///
/// The show defaults panel.
///
private bool showDefaultsPanel;
///
/// The audio behaviours.
///
private SubtitleBehaviours subtitleBehaviours;
#endregion
#region Constructors and Destructors
///
/// Initializes a new instance of the class.
///
public SubtitlesViewModel()
{
this.SubtitleDefaultsViewModel = new SubtitlesDefaultsViewModel();
this.Task = new EncodeTask();
this.Langauges = LanguageUtilities.MapLanguages().Keys;
this.CharacterCodes = CharCodesUtilities.GetCharacterCodes();
this.ForeignAudioSearchTrack = new Subtitle { SubtitleType = SubtitleType.ForeignAudioSearch, Language = "Foreign Audio Search (Bitmap)" };
this.SourceTracks = new List { this.ForeignAudioSearchTrack };
}
#endregion
#region Properties
///
/// Gets or sets the audio defaults view model.
///
public ISubtitlesDefaultsViewModel SubtitleDefaultsViewModel { get; set; }
///
/// Gets or sets CharacterCodes.
///
public IEnumerable CharacterCodes { get; set; }
///
/// Gets or sets Langauges.
///
public IEnumerable Langauges { get; set; }
///
/// Gets or sets SourceTracks.
///
public IList SourceTracks
{
get
{
return this.sourceTracks;
}
set
{
this.sourceTracks = value;
this.NotifyOfPropertyChange(() => this.SourceTracks);
}
}
///
/// Gets or sets Task.
///
public EncodeTask Task { get; set; }
///
/// Gets or sets a value indicating whether show defaults panel.
///
public bool ShowDefaultsPanel
{
get
{
return this.showDefaultsPanel;
}
set
{
if (value.Equals(this.showDefaultsPanel))
{
return;
}
this.showDefaultsPanel = value;
this.NotifyOfPropertyChange(() => this.ShowDefaultsPanel);
this.NotifyOfPropertyChange(() => this.PanelTitle);
this.NotifyOfPropertyChange(() => this.SwitchDisplayTitle);
}
}
///
/// Gets the panel title.
///
public string PanelTitle
{
get
{
return this.ShowDefaultsPanel ? Resources.SubtitlesViewModel_SubDefaults : Resources.SubtitlesViewModel_SubTracks;
}
}
///
/// Gets the switch display title.
///
public string SwitchDisplayTitle
{
get
{
return this.ShowDefaultsPanel ? Resources.SubtitlesViewModel_SwitchToTracks : Resources.SubtitlesViewModel_ConfigureDefaults;
}
}
///
/// Gets the default audio behaviours.
///
public SubtitleBehaviours SubtitleBehaviours
{
get
{
return this.SubtitleDefaultsViewModel.SubtitleBehaviours;
}
}
#endregion
#region Public Methods
///
/// Add a new Track
///
public void Add()
{
this.Add(null);
}
///
/// Add all closed captions not already on the list.
///
public void AddAllClosedCaptions()
{
foreach (Subtitle subtitle in this.SourceTitlesSubset(null).Where(s => s.SubtitleType == SubtitleType.CC))
{
this.Add(subtitle);
}
}
///
/// Add all the remaining subtitle tracks.
///
public void AddAllRemaining()
{
foreach (Subtitle subtitle in this.SourceTitlesSubset(null))
{
this.Add(subtitle);
}
}
///
/// Add all remaining tracks for the users preferred and selected languages
///
public void AddAllRemainingForSelectedLanguages()
{
// Translate to Iso Codes
List iso6392Codes = this.SubtitleBehaviours.SelectedLangauges.Contains(Constants.Any)
? LanguageUtilities.GetIsoCodes()
: LanguageUtilities.GetLanguageCodes(
this.SubtitleBehaviours.SelectedLangauges.ToArray());
List availableTracks =
this.SourceTracks.Where(subtitle => iso6392Codes.Contains(subtitle.LanguageCodeClean)).ToList();
foreach (Subtitle subtitle in this.SourceTitlesSubset(availableTracks))
{
this.Add(subtitle);
}
}
///
/// The add first for selected languages.
///
private void AddFirstForSelectedLanguages()
{
foreach (Subtitle sourceTrack in this.GetSelectedLanguagesTracks())
{
// Step 2: Check if the track list already contrains this track
bool found = this.Task.SubtitleTracks.Any(track => Equals(track.SourceTrack, sourceTrack));
if (!found)
{
// Check if we are already using this language
bool foundLanguage = false;
Subtitle track = sourceTrack;
foreach (var item in this.Task.SubtitleTracks)
{
if (item.SourceTrack != null && item.SourceTrack.LanguageCode != null && track.LanguageCode.Contains(item.SourceTrack.LanguageCode))
{
foundLanguage = true;
}
}
if (foundLanguage)
{
continue;
}
// If it doesn't, add it.
this.Add(sourceTrack);
}
}
}
///
/// Import an SRT File.
///
public void Import()
{
OpenFileDialog dialog = new OpenFileDialog
{
Filter = "SRT files (*.srt)|*.srt",
CheckFileExists = true,
Multiselect = true
};
dialog.ShowDialog();
foreach (var srtFile in dialog.FileNames)
{
SubtitleTrack track = new SubtitleTrack
{
SrtFileName = Path.GetFileNameWithoutExtension(srtFile),
SrtOffset = 0,
SrtCharCode = "UTF-8",
SrtLang = "English",
SubtitleType = SubtitleType.SRT,
SrtPath = srtFile
};
this.Task.SubtitleTracks.Add(track);
}
}
///
/// Remove a Track
///
///
/// The track.
///
public void Remove(SubtitleTrack track)
{
this.Task.SubtitleTracks.Remove(track);
}
///
/// Clear all Tracks
///
public void Clear()
{
this.Task.SubtitleTracks.Clear();
}
///
/// Select the default subtitle track.
///
///
/// The subtitle.
///
public void SelectDefaultTrack(SubtitleTrack subtitle)
{
foreach (SubtitleTrack track in this.Task.SubtitleTracks)
{
if (track == subtitle)
{
continue; // Skip the track the user selected.
}
track.Default = false;
}
this.NotifyOfPropertyChange(() => this.Task);
}
///
/// Select the burned in track.
///
///
/// The subtitle.
///
public void SetBurnedToFalseForAllExcept(SubtitleTrack subtitle)
{
foreach (SubtitleTrack track in this.Task.SubtitleTracks)
{
if (track == subtitle)
{
continue; // Skip the track the user selected.
}
track.Burned = false;
}
this.NotifyOfPropertyChange(() => this.Task);
}
///
/// Automatic Subtitle Selection based on user preferences.
///
public void AutomaticSubtitleSelection()
{
this.Task.SubtitleTracks.Clear();
// Add Foreign Audio Scan
if (this.SubtitleBehaviours.AddForeignAudioScanTrack)
{
this.Add(ForeignAudioSearchTrack);
}
// Add Track Behaviours
switch (this.SubtitleBehaviours.SelectedBehaviour)
{
case SubtitleBehaviourModes.FirstMatch: // Adding all remaining tracks
this.AddFirstForSelectedLanguages();
break;
case SubtitleBehaviourModes.AllMatching: // Add Langauges tracks for the additional languages selected, in-order.
this.AddAllRemainingForSelectedLanguages();
break;
}
// Burn In Behaviour
if (this.Task.SubtitleTracks.Count >= 1)
{
bool burnInSet = false;
switch (this.SubtitleBehaviours.SelectedBurnInBehaviour)
{
case SubtitleBurnInBehaviourModes.None:
// Do Nothing. Only tracks where the container requires it will be burned in.
break;
case SubtitleBurnInBehaviourModes.ForeignAudio:
foreach (var track in this.Task.SubtitleTracks)
{
// Set the Foreign Audio Track to burned-in
if (track.SourceTrack.SubtitleType == SubtitleType.ForeignAudioSearch)
{
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
break;
}
}
break;
case SubtitleBurnInBehaviourModes.FirstTrack:
foreach (var track in this.Task.SubtitleTracks)
{
// Set the first track.
if (!burnInSet && track.SourceTrack.SubtitleType != SubtitleType.ForeignAudioSearch)
{
burnInSet = true;
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
}
}
break;
case SubtitleBurnInBehaviourModes.ForeignAudioPreferred:
foreach (var track in this.Task.SubtitleTracks)
{
// Set the first track.
if (!burnInSet)
{
burnInSet = true;
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
}
// But if there is a foreign audio track, prefer this to the first.
if (track.SourceTrack.SubtitleType == SubtitleType.ForeignAudioSearch)
{
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
break;
}
}
break;
}
}
// Add all closed captions if enabled.
if (this.SubtitleBehaviours.AddClosedCaptions)
{
this.AddAllClosedCaptions();
}
}
///
/// Open the options screen to the Audio and Subtitles tab.
///
public void SetDefaultBehaviour()
{
this.ShowDefaultsPanel = true;
}
///
/// The show audio defaults.
///
public void ShowSubtitleDefaultsPanel()
{
this.ShowDefaultsPanel = !this.ShowDefaultsPanel;
}
///
/// Reload the audio tracks based on the defaults.
///
public void ReloadDefaults()
{
this.AutomaticSubtitleSelection();
}
#endregion
#region Implemented Interfaces
///
/// Setup this tab for the specified preset.
///
///
/// The preset.
///
///
/// The task.
///
public void SetPreset(Preset preset, EncodeTask task)
{
// Note, We don't support Subtitles in presets yet.
this.Task = task;
this.NotifyOfPropertyChange(() => this.Task);
this.SubtitleDefaultsViewModel.SetupLanguages(preset);
this.AutomaticSubtitleSelection();
}
///
/// Update all the UI controls based on the encode task passed in.
///
///
/// The task.
///
public void UpdateTask(EncodeTask task)
{
this.Task = task;
this.NotifyOfPropertyChange(() => this.Task.SubtitleTracks);
this.NotifyOfPropertyChange(() => this.Task);
}
///
/// Setup this window for a new source
///
///
/// The source.
///
///
/// The title.
///
///
/// The preset.
///
///
/// The task.
///
public void SetSource(Source source, Title title, Preset preset, EncodeTask task)
{
this.SourceTracks.Clear();
this.SourceTracks.Add(ForeignAudioSearchTrack);
foreach (Subtitle subtitle in title.Subtitles)
{
this.SourceTracks.Add(subtitle);
}
this.Task = task;
this.NotifyOfPropertyChange(() => this.Task);
this.AutomaticSubtitleSelection();
}
#endregion
#region Methods
///
/// Add a subtitle track.
/// The Source track is set based on the following order. If null, it will skip to the next option.
/// 1. Passed in Subitle param
/// 2. First preferred Subtitle from source
/// 3. First subtitle from source.
/// Will not add a subtitle if the source has none.
///
///
/// The subtitle. Use null to add preferred, or first from source (based on user preference)
///
private void Add(Subtitle subtitle)
{
Subtitle source = subtitle
?? ((this.SourceTracks != null)
? (this.SourceTracks.FirstOrDefault(l => l.Language == this.GetPreferredSubtitleTrackLanguage())
?? this.SourceTracks.FirstOrDefault(
s => s.SubtitleType != SubtitleType.ForeignAudioSearch))
: null);
if (source == null)
{
source = ForeignAudioSearchTrack;
}
SubtitleTrack track = new SubtitleTrack
{
SubtitleType = SubtitleType.VobSub,
SourceTrack = source,
};
// Burn-in Behaviours
if (this.SubtitleBehaviours.SelectedBurnInBehaviour == SubtitleBurnInBehaviourModes.ForeignAudio
|| this.SubtitleBehaviours.SelectedBurnInBehaviour == SubtitleBurnInBehaviourModes.ForeignAudioPreferred)
{
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
}
// For MP4, PGS Subtitles must be burned in.
if (!track.Burned && (source.SubtitleType == SubtitleType.PGS) && this.Task != null && this.Task.OutputFormat == OutputFormat.Mp4)
{
if (track.CanBeBurned)
{
track.Burned = true;
this.SetBurnedToFalseForAllExcept(track);
}
}
var encodeTask = this.Task;
if (encodeTask != null)
{
encodeTask.SubtitleTracks.Add(track);
}
}
///
/// Gets a list of source tracks for the users selected languages.
///
///
/// A list of source subtitle tracks.
///
private IEnumerable GetSelectedLanguagesTracks()
{
List trackList = new List();
List isoCodes = this.SubtitleBehaviours.SelectedLangauges.Contains(Constants.Any)
? LanguageUtilities.GetIsoCodes()
: LanguageUtilities.GetLanguageCodes(
this.SubtitleBehaviours.SelectedLangauges.ToArray());
foreach (string code in isoCodes)
{
trackList.AddRange(this.SourceTracks.Where(source => source.LanguageCode == code));
}
return trackList;
}
///
/// The get preferred subtitle track, or the first if none available.
///
///
/// The users preferred language, or the first if none available.
///
private string GetPreferredSubtitleTrackLanguage()
{
return this.SubtitleBehaviours.SelectedLangauges.FirstOrDefault(w => w != Constants.Any);
}
///
/// Gets a list of Source subtitle tracks that are not currently used.
///
///
/// The subtitles. (Optional). If null, works on the full source subtitle collection
///
///
/// An IEnumerable collection of subtitles
///
private IEnumerable SourceTitlesSubset(IEnumerable subtitles)
{
return subtitles != null
? subtitles.Where(subtitle => !this.Task.SubtitleTracks.Any(track => Equals(track.SourceTrack, subtitle))).ToList()
: this.SourceTracks.Where(subtitle => !this.Task.SubtitleTracks.Any(track => Equals(track.SourceTrack, subtitle))).ToList();
}
#endregion
}
}