/* $Id: HBSubtitles.m This file is part of the HandBrake source code. Homepage: . It may be used under the terms of the GNU General Public License. */ #import "HBSubtitlesController.h" #import "HBSubtitlesDefaultsController.h" #import "HBSubtitlesDefaults.h" #import "Controller.h" #include "hb.h" #include "lang.h" NSString *keySubTrackSelectionIndex = @"keySubTrackSelectionIndex"; NSString *keySubTrackName = @"keySubTrackName"; NSString *keySubTrackIndex = @"keySubTrackIndex"; NSString *keySubTrackLanguage = @"keySubTrackLanguage"; NSString *keySubTrackLanguageIsoCode = @"keySubTrackLanguageIsoCode"; NSString *keySubTrackType = @"keySubTrackType"; NSString *keySubTrackForced = @"keySubTrackForced"; NSString *keySubTrackBurned = @"keySubTrackBurned"; NSString *keySubTrackDefault = @"keySubTrackDefault"; NSString *keySubTrackSrtOffset = @"keySubTrackSrtOffset"; NSString *keySubTrackSrtFilePath = @"keySubTrackSrtFilePath"; NSString *keySubTrackSrtCharCode = @"keySubTrackSrtCharCode"; NSString *keySubTrackSrtCharCodeIndex = @"keySubTrackSrtCharCodeIndex"; NSString *keySubTrackLanguageIndex = @"keySubTrackLanguageIndex"; #define CHAR_CODE_DEFAULT_INDEX 11 @interface HBSubtitlesController () // IBOutles @property (assign) IBOutlet NSTableView *fTableView; @property (assign) IBOutlet NSPopUpButton *trackPopUp; @property (assign) IBOutlet NSButton *configureDefaults; @property (assign) IBOutlet NSButton *reloadDefaults; @property (nonatomic, readwrite) BOOL enabled; // Subtitles arrays @property (nonatomic, readonly) NSMutableArray *subtitleArray; @property (nonatomic, readonly) NSMutableArray *subtitleSourceArray; @property (nonatomic, readwrite, retain) NSString *foreignAudioSearchTrackName; @property (nonatomic, readwrite) int container; // Defaults @property (nonatomic, readwrite, retain) HBSubtitlesDefaultsController *defaultsController; @property (nonatomic, readwrite, retain) HBSubtitlesDefaults *settings; // Table view cells models @property (nonatomic, readonly) NSArray *charCodeArray; @property (nonatomic, readwrite) BOOL foreignAudioSearchSelected; @property (nonatomic, readonly) NSArray *languagesArray; @property (nonatomic, readonly) NSInteger languagesArrayDefIndex; // Cached table view's cells @property (nonatomic, readonly) NSPopUpButtonCell *languagesCell; @property (nonatomic, readonly) NSPopUpButtonCell *encodingsCell; @end @implementation HBSubtitlesController - (instancetype)init { self = [super initWithNibName:@"Subtitles" bundle:nil]; if (self) { _subtitleSourceArray = [[NSMutableArray alloc] init]; _subtitleArray = [[NSMutableArray alloc] init]; _languagesArray = [[self populateLanguageArray] retain]; // populate the charCodeArray. _charCodeArray = [@[@"ANSI_X3.4-1968", @"ANSI_X3.4-1986", @"ANSI_X3.4", @"ANSI_X3.110-1983", @"ANSI_X3.110", @"ASCII", @"ECMA-114", @"ECMA-118", @"ECMA-128", @"ECMA-CYRILLIC", @"IEC_P27-1", @"ISO-8859-1", @"ISO-8859-2", @"ISO-8859-3", @"ISO-8859-4", @"ISO-8859-5", @"ISO-8859-6", @"ISO-8859-7", @"ISO-8859-8", @"ISO-8859-9", @"ISO-8859-9E", @"ISO-8859-10", @"ISO-8859-11", @"ISO-8859-13", @"ISO-8859-14", @"ISO-8859-15", @"ISO-8859-16", @"UTF-7", @"UTF-8", @"UTF-16", @"UTF-16LE", @"UTF-16BE", @"UTF-32", @"UTF-32LE", @"UTF-32BE"] retain]; // Register as observer for the HBController notifications. [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(containerChanged:) name: HBContainerChangedNotification object: nil]; [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(titleChanged:) name: HBTitleChangedNotification object: nil]; } return self; } - (void)setUIEnabled:(BOOL)flag { [self.trackPopUp setEnabled:flag]; [self.configureDefaults setEnabled:flag]; [self.reloadDefaults setEnabled:flag]; [self.fTableView setEnabled:flag]; self.enabled = flag; } - (void)titleChanged:(NSNotification *)aNotification { NSDictionary *notDict = [aNotification userInfo]; NSData *theData = notDict[keyTitleTag]; hb_title_t *title = NULL; [theData getBytes: &title length: sizeof(title)]; /* reset the subtitles arrays */ [self.subtitleArray removeAllObjects]; [self.subtitleSourceArray removeAllObjects]; if (title) { /* now populate the array with the source subtitle track info */ NSMutableArray *forcedSourceNamesArray = [[NSMutableArray alloc] init]; for (int i = 0; i < hb_list_count(title->list_subtitle); i++) { hb_subtitle_t *subtitle = (hb_subtitle_t *)hb_list_item(title->list_subtitle, i); /* Human-readable representation of subtitle->source */ NSString *bitmapOrText = subtitle->format == PICTURESUB ? @"Bitmap" : @"Text"; NSString *subSourceName = @(hb_subsource_name(subtitle->source)); /* if the subtitle track can be forced, add its source name to the array */ if (hb_subtitle_can_force(subtitle->source) && [forcedSourceNamesArray containsObject:subSourceName] == NO) { [forcedSourceNamesArray addObject:subSourceName]; } // Use the native language name if available iso639_lang_t *language = lang_for_code2(subtitle->iso639_2); NSString *nativeLanguage = strlen(language->native_name) ? @(language->native_name) : @(language->eng_name); /* create a dictionary of source subtitle information to store in our array */ [self.subtitleSourceArray addObject:@{keySubTrackName: [NSString stringWithFormat:@"%d: %@ (%@) (%@)", i, nativeLanguage, bitmapOrText, subSourceName], keySubTrackIndex: @(i), keySubTrackType: @(subtitle->source), keySubTrackLanguage: nativeLanguage, keySubTrackLanguageIsoCode: @(subtitle->iso639_2)}]; } /* now set the name of the Foreign Audio Search track */ if ([forcedSourceNamesArray count]) { [forcedSourceNamesArray sortUsingComparator:^(id obj1, id obj2) { return [((NSString *)obj1) compare:((NSString *)obj2)]; }]; NSString *tempList = @""; for (NSString *tempString in forcedSourceNamesArray) { if ([tempList length]) { tempList = [tempList stringByAppendingString:@", "]; } tempList = [tempList stringByAppendingString:tempString]; } self.foreignAudioSearchTrackName = [NSString stringWithFormat:@"Foreign Audio Search (Bitmap) (%@)", tempList]; } else { self.foreignAudioSearchTrackName = @"Foreign Audio Search (Bitmap)"; } [forcedSourceNamesArray release]; // Append an empty track at the end // to display a "None" row in the table view [self.subtitleArray addObject:[self createSubtitleTrack]]; } [self.fTableView reloadData]; } - (void)containerChanged:(NSNotification *)aNotification { NSDictionary *notDict = [aNotification userInfo]; self.container = [notDict[keyContainerTag] intValue]; [self validatePassthru]; [self.fTableView reloadData]; } - (NSArray *)subtitles { NSMutableArray *ret = [self.subtitleArray mutableCopy]; [ret removeLastObject]; return [ret autorelease]; } - (void)addTracksFromQueue:(NSArray *)queueSubtitleArray { /* Note: we need to look for external subtitles so it can be added to the source array track. * Remember the source container subs are already loaded with resetTitle which is already called * so any external sub sources need to be added to our source subs here */ for (id tempObject in queueSubtitleArray) { /* We have an srt track */ if ([tempObject[keySubTrackType] intValue] == SRTSUB) { NSString *filePath = tempObject[keySubTrackSrtFilePath]; /* create a dictionary of source subtitle information to store in our array */ [self.subtitleSourceArray addObject:@{keySubTrackIndex: @(self.subtitleSourceArray.count + 1), keySubTrackName: [filePath lastPathComponent], keySubTrackType: @(SRTSUB), keySubTrackSrtFilePath: filePath}]; } } // Set the subtitleArray to the newSubtitleArray [self.subtitleArray setArray:queueSubtitleArray]; [self.subtitleArray addObject:[self createSubtitleTrack]]; [self.fTableView reloadData]; } - (void)applySettingsFromPreset:(NSDictionary *)preset { self.settings = [[[HBSubtitlesDefaults alloc] init] autorelease]; [self.settings applySettingsFromPreset:preset]; [self addTracksFromDefaults:self]; } #pragma mark - Actions - (BOOL)validateUserInterfaceItem:(id < NSValidatedUserInterfaceItem >)anItem { return self.enabled; } /** * Add every subtitles track that still isn't in the subtitles array. */ - (IBAction)addAll:(id)sender { [self.subtitleArray removeAllObjects]; // Add the foreign audio search pass [self addTrack:[self trackFromSourceTrackIndex:-1]]; // Add the remainings tracks for (NSDictionary *track in self.subtitleSourceArray) { NSInteger sourceIndex = [track[keySubTrackIndex] integerValue]; [self addTrack:[self trackFromSourceTrackIndex:sourceIndex]]; } [self.subtitleArray addObject:[self createSubtitleTrack]]; [self validatePassthru]; [self.fTableView reloadData]; } /** * Remove all the subtitles tracks. */ - (IBAction)removeAll:(id)sender { [self.subtitleArray removeAllObjects]; [self.subtitleArray addObject:[self createSubtitleTrack]]; [self.fTableView reloadData]; } /** * Remove all the subtitles tracks and * add new ones based on the defaults settings */ - (IBAction)addTracksFromDefaults:(id)sender { // Keeps a set of the indexes of the added track // so we don't add the same track twice. NSMutableIndexSet *tracksAdded = [NSMutableIndexSet indexSet]; [self.subtitleArray removeAllObjects]; // Add the foreign audio search pass if (self.settings.addForeignAudioSearch) { [self addTrack:[self trackFromSourceTrackIndex:-1]]; } // Add the tracks for the selected languages if (self.settings.trackSelectionBehavior != HBSubtitleTrackSelectionBehaviorNone) { for (NSString *lang in self.settings.trackSelectionLanguages) { for (NSDictionary *track in self.subtitleSourceArray) { if ([lang isEqualToString:@"und"] || [track[keySubTrackLanguageIsoCode] isEqualToString:lang]) { NSInteger sourceIndex = [track[keySubTrackIndex] intValue]; if (![tracksAdded containsIndex:sourceIndex]) { [self addTrack:[self trackFromSourceTrackIndex:sourceIndex]]; } [tracksAdded addIndex:sourceIndex]; if (self.settings.trackSelectionBehavior == HBSubtitleTrackSelectionBehaviorFirst) { break; } } } } } // Add the closed captions track if there is one. if (self.settings.addCC) { for (NSDictionary *track in self.subtitleSourceArray) { if ([track[keySubTrackType] intValue] == CC608SUB) { NSInteger sourceIndex = [track[keySubTrackIndex] intValue]; if (![tracksAdded containsIndex:sourceIndex]) { [self addTrack:[self trackFromSourceTrackIndex:sourceIndex]]; } if (self.settings.trackSelectionBehavior == HBSubtitleTrackSelectionBehaviorFirst) { break; } } } } // Add an empty track [self.subtitleArray addObject:[self createSubtitleTrack]]; [self validatePassthru]; [self.fTableView reloadData]; } - (IBAction)showSettingsSheet:(id)sender { self.defaultsController = [[[HBSubtitlesDefaultsController alloc] initWithSettings:self.settings] autorelease]; [NSApp beginSheet:[self.defaultsController window] modalForWindow:[[self view] window] modalDelegate:self didEndSelector:@selector(sheetDidEnd) contextInfo:NULL]; } - (void)sheetDidEnd { self.defaultsController = nil; } #pragma mark - Subtitles tracks creation and validation /** * Convenience method to add a track to subtitlesArray. * It calculates the keySubTrackSelectionIndex. * * @param track the track to add. */ - (void)addTrack:(NSMutableDictionary *)newTrack { newTrack[keySubTrackSelectionIndex] = @([newTrack[keySubTrackIndex] integerValue] + 1 + (self.subtitleArray.count == 0)); [self.subtitleArray addObject:newTrack]; } /** * Creates a new subtitle track. */ - (NSMutableDictionary *)createSubtitleTrack { NSMutableDictionary *newSubtitleTrack = [[NSMutableDictionary alloc] init]; newSubtitleTrack[keySubTrackIndex] = @0; newSubtitleTrack[keySubTrackSelectionIndex] = @0; newSubtitleTrack[keySubTrackName] = @"None"; newSubtitleTrack[keySubTrackForced] = @0; newSubtitleTrack[keySubTrackBurned] = @0; newSubtitleTrack[keySubTrackDefault] = @0; return [newSubtitleTrack autorelease]; } /** * Creates a new track dictionary from a source track. * * @param index the index of the source track in the subtitlesSourceArray, * -1 means a Foreign Audio Search pass. * * @return a new mutable track dictionary. */ - (NSMutableDictionary *)trackFromSourceTrackIndex:(NSInteger)index { NSMutableDictionary *track = [self createSubtitleTrack]; if (index == -1) { /* * we are foreign lang search, which is inherently bitmap * * since it can be either VOBSUB or PGS and the latter can't be * passed through to MP4, we need to know whether there are any * PGS tracks in the source - otherwise we can just set the * source track type to VOBSUB */ int subtitleTrackType = VOBSUB; if ([self.foreignAudioSearchTrackName rangeOfString:@(hb_subsource_name(PGSSUB))].location != NSNotFound) { subtitleTrackType = PGSSUB; } // Use -1 to indicate the foreign lang search track[keySubTrackIndex] = @(-1); track[keySubTrackName] = self.foreignAudioSearchTrackName; track[keySubTrackType] = @(subtitleTrackType); // foreign lang search is most useful when combined w/Forced Only - make it default track[keySubTrackForced] = @1; } else { NSDictionary *sourceTrack = self.subtitleSourceArray[index]; track[keySubTrackIndex] = @(index); track[keySubTrackName] = sourceTrack[keySubTrackName]; /* check to see if we are an srt, in which case set our file path and source track type kvp's*/ if ([self.subtitleSourceArray[index][keySubTrackType] intValue] == SRTSUB) { track[keySubTrackType] = @(SRTSUB); track[keySubTrackSrtFilePath] = sourceTrack[keySubTrackSrtFilePath]; track[keySubTrackLanguageIndex] = @(self.languagesArrayDefIndex); track[keySubTrackLanguageIsoCode] = self.languagesArray[self.languagesArrayDefIndex][1]; track[keySubTrackSrtCharCodeIndex] = @(CHAR_CODE_DEFAULT_INDEX); track[keySubTrackSrtCharCode] = self.charCodeArray[CHAR_CODE_DEFAULT_INDEX]; } else { track[keySubTrackType] = sourceTrack[keySubTrackType]; } } if (!hb_subtitle_can_burn([track[keySubTrackType] intValue])) { /* the source track cannot be burned in, so uncheck the widget */ track[keySubTrackBurned] = @0; } if (!hb_subtitle_can_force([track[keySubTrackType] intValue])) { /* the source track does not support forced flags, so uncheck the widget */ track[keySubTrackForced] = @0; } return track; } /** * Checks whether any subtitles in the list cannot be passed through. * Set the first of any such subtitles to burned-in, remove the others. */ - (void)validatePassthru { int subtitleTrackType; BOOL convertToBurnInUsed = NO; NSMutableArray *tracksToDelete = [[NSMutableArray alloc] init]; // convert any non-None incompatible tracks to burn-in or remove them for (id tempObject in self.subtitleArray) { if (tempObject[keySubTrackType] == nil) { continue; } subtitleTrackType = [tempObject[keySubTrackType] intValue]; if (!hb_subtitle_can_pass(subtitleTrackType, self.container)) { if (convertToBurnInUsed == NO) { //we haven't set any track to burned-in yet, so we can tempObject[keySubTrackBurned] = @1; convertToBurnInUsed = YES; //remove any additional tracks } else { //we already have a burned-in track, we must remove others [tracksToDelete addObject:tempObject]; } } } //if we converted a track to burned-in, unset it for tracks that support passthru if (convertToBurnInUsed == YES) { for (id tempObject in self.subtitleArray) { if (tempObject[keySubTrackType] == nil) { continue; } subtitleTrackType = [tempObject[keySubTrackType] intValue]; if (hb_subtitle_can_pass(subtitleTrackType, self.container)) { tempObject[keySubTrackBurned] = @0; } } } if (tracksToDelete.count) { [self.subtitleArray removeObjectsInArray:tracksToDelete]; [self.fTableView reloadData]; } [tracksToDelete release]; } - (void)validateBurned:(NSInteger)index { [self.subtitleArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) { if (idx != index) { obj[keySubTrackBurned] = @0; } }]; [self validatePassthru]; } - (void)validateDefault:(NSInteger)index { [self.subtitleArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) { if (idx != index) { obj[keySubTrackDefault] = @0; } }]; } #pragma mark - #pragma mark Subtitle Table Data Source Methods - (NSInteger)numberOfRowsInTableView:(NSTableView *)aTableView { return self.subtitleArray.count; } - (id)tableView:(NSTableView *)aTableView objectValueForTableColumn:(NSTableColumn *)aTableColumn row:(NSInteger)rowIndex { NSDictionary *track = self.subtitleArray[rowIndex]; if ([[aTableColumn identifier] isEqualToString:@"track"]) { NSNumber *index = track[keySubTrackSelectionIndex]; if (index) return index; else return @0; } else if ([[aTableColumn identifier] isEqualToString:@"forced"]) { return track[keySubTrackForced]; } else if ([[aTableColumn identifier] isEqualToString:@"burned"]) { return track[keySubTrackBurned]; } else if ([[aTableColumn identifier] isEqualToString:@"default"]) { return track[keySubTrackDefault]; } /* These next three columns only apply to srt's. they are disabled for source subs */ else if ([[aTableColumn identifier] isEqualToString:@"srt_lang"]) { if ([track[keySubTrackType] intValue] == SRTSUB) { return track[keySubTrackLanguageIndex]; } else { return @0; } } else if ([[aTableColumn identifier] isEqualToString:@"srt_charcode"]) { if ([track[keySubTrackType] intValue] == SRTSUB) { return track[keySubTrackSrtCharCodeIndex]; } else { return @0; } } else if ([[aTableColumn identifier] isEqualToString:@"srt_offset"]) { if (track[keySubTrackSrtOffset]) { return [track[keySubTrackSrtOffset] stringValue]; } else { return @"0"; } } return nil; } /** * Called whenever a widget in the table is edited or changed, we use it to record the change in the controlling array * including removing and adding new tracks via the "None" ("track" index of 0) */ - (void)tableView:(NSTableView *)aTableView setObjectValue:(id)anObject forTableColumn:(NSTableColumn *)aTableColumn row:(NSInteger)rowIndex { if ([[aTableColumn identifier] isEqualToString:@"track"]) { /* Set the array to track if we are vobsub (picture sub) */ if ([anObject intValue] > 0) { NSMutableDictionary *newTrack = [self trackFromSourceTrackIndex:[anObject integerValue] - 1 - (rowIndex == 0)]; // Selection index calculation newTrack[keySubTrackSelectionIndex] = @([anObject integerValue]); self.subtitleArray[rowIndex] = newTrack; } } else if ([[aTableColumn identifier] isEqualToString:@"forced"]) { self.subtitleArray[rowIndex][keySubTrackForced] = @([anObject intValue]); } else if ([[aTableColumn identifier] isEqualToString:@"burned"]) { self.subtitleArray[rowIndex][keySubTrackBurned] = @([anObject intValue]); if([anObject intValue] == 1) { /* Burned In and Default are mutually exclusive */ self.subtitleArray[rowIndex][keySubTrackDefault] = @0; } /* now we need to make sure no other tracks are set to burned if we have set burned */ if ([anObject intValue] == 1) { [self validateBurned:rowIndex]; } } else if ([[aTableColumn identifier] isEqualToString:@"default"]) { self.subtitleArray[rowIndex][keySubTrackDefault] = @([anObject intValue]); if([anObject intValue] == 1) { /* Burned In and Default are mutually exclusive */ self.subtitleArray[rowIndex][keySubTrackBurned] = @0; } /* now we need to make sure no other tracks are set to default */ if ([anObject intValue] == 1) { [self validateDefault:rowIndex]; } } /* These next three columns only apply to srt's. they are disabled for source subs */ else if ([[aTableColumn identifier] isEqualToString:@"srt_lang"]) { self.subtitleArray[rowIndex][keySubTrackLanguageIndex] = @([anObject intValue]); self.subtitleArray[rowIndex][keySubTrackLanguageIsoCode] = self.languagesArray[[anObject intValue]][1]; } else if ([[aTableColumn identifier] isEqualToString:@"srt_charcode"]) { /* charCodeArray */ self.subtitleArray[rowIndex][keySubTrackSrtCharCodeIndex] = @([anObject intValue]); self.subtitleArray[rowIndex][keySubTrackSrtCharCode] = self.charCodeArray[[anObject intValue]]; } else if ([[aTableColumn identifier] isEqualToString:@"srt_offset"]) { self.subtitleArray[rowIndex][keySubTrackSrtOffset] = @([anObject integerValue]); } /* now lets do a bit of logic to add / remove tracks as necessary via the "None" track (index 0) */ if ([[aTableColumn identifier] isEqualToString:@"track"]) { /* Since currently no quicktime based playback devices support soft vobsubs in mp4, we make sure "burned in" is specified * by default to avoid massive confusion and anarchy. However we also want to guard against multiple burned in subtitle tracks * as libhb would ignore all but the first one anyway. Plus it would probably be stupid. */ if ((self.container & HB_MUX_MASK_MP4) && ([anObject intValue] != 0)) { if ([self.subtitleArray[rowIndex][keySubTrackType] intValue] == VOBSUB) { /* lets see if there are currently any burned in subs specified */ BOOL subtrackBurnedInFound = NO; for (id tempObject in self.subtitleArray) { if ([tempObject[keySubTrackBurned] intValue] == 1) { subtrackBurnedInFound = YES; } } /* if we have no current vobsub set to burn it in ... burn it in by default */ if (!subtrackBurnedInFound) { self.subtitleArray[rowIndex][keySubTrackBurned] = @1; /* Burned In and Default are mutually exclusive */ self.subtitleArray[rowIndex][keySubTrackDefault] = @0; } } } /* We use the track popup index number (presumes index 0 is "None" which is ignored and only used to remove tracks if need be) * to determine whether to 1 modify an existing track, 2. add a new empty "None" track or 3. remove an existing track. */ if ([anObject intValue] != 0 && rowIndex == [self.subtitleArray count] - 1) // if we have a last track which != "None" { /* add a new empty None track */ [self.subtitleArray addObject:[self createSubtitleTrack]]; } else if ([anObject intValue] == 0 && rowIndex != ([self.subtitleArray count] -1))// if this track is set to "None" and not the last track displayed { /* we know the user chose to remove this track by setting it to None, so remove it from the array */ /* However,if this is the first track we have to reset the selected index of the next track by + 1, since it will now become * the first track, which has to account for the extra "Foreign Language Search" index. */ if (rowIndex == 0 && [self.subtitleArray[1][keySubTrackSelectionIndex] intValue] != 0) { /* get the index of the selection in row one (which is track two) */ int trackOneSelectedIndex = [self.subtitleArray[1][keySubTrackSelectionIndex] intValue]; /* increment the index of the subtitle menu item by one, to account for Foreign Language Search which is unique to the first track */ self.subtitleArray[1][keySubTrackSelectionIndex] = @(trackOneSelectedIndex + 1); } /* now that we have made the adjustment for track one (index 0) go ahead and delete the track */ [self.subtitleArray removeObjectAtIndex: rowIndex]; } // Validate the current passthru tracks. [self validatePassthru]; } [aTableView reloadData]; } #pragma mark - #pragma mark Subtitle Table Delegate Methods - (NSCell *)tableView:(NSTableView *)tableView dataCellForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)rowIndex { if ([[tableColumn identifier] isEqualToString:@"track"]) { // 'track' is a popup of all available source subtitle tracks for the given title NSPopUpButtonCell *cellTrackPopup = [[NSPopUpButtonCell alloc] init]; [cellTrackPopup setControlSize:NSSmallControlSize]; [cellTrackPopup setFont:[NSFont systemFontOfSize:[NSFont systemFontSizeForControlSize:NSSmallControlSize]]]; // Add our initial "None" track which we use to add source tracks or remove tracks. // "None" is always index 0. [[cellTrackPopup menu] addItemWithTitle:@"None" action:NULL keyEquivalent:@""]; // Foreign Audio Search (index 1 in the popup) is only available for the first track if (rowIndex == 0) { [[cellTrackPopup menu] addItemWithTitle:self.foreignAudioSearchTrackName action:NULL keyEquivalent:@""]; } for (NSDictionary *track in self.subtitleSourceArray) { [[cellTrackPopup menu] addItemWithTitle:track[keySubTrackName] action:NULL keyEquivalent:@""]; } return [cellTrackPopup autorelease]; } else if ([[tableColumn identifier] isEqualToString:@"srt_lang"]) { return self.languagesCell; } else if ([[tableColumn identifier] isEqualToString:@"srt_charcode"]) { return self.encodingsCell; } return nil; } /** * Enables/Disables the table view cells. */ - (void)tableView:(NSTableView *)aTableView willDisplayCell:(id)aCell forTableColumn:(NSTableColumn *)aTableColumn row:(NSInteger)rowIndex { if ([[aTableColumn identifier] isEqualToString:@"track"]) { return; } // If the Track is None, we disable the other cells as None is an empty track if ([self.subtitleArray[rowIndex][keySubTrackSelectionIndex] intValue] == 0) { [aCell setEnabled:NO]; } else { // Since we have a valid track, we go ahead and enable the rest of the widgets and set them according to the controlling array */ [aCell setEnabled:YES]; } if ([[aTableColumn identifier] isEqualToString:@"forced"]) { // Disable the "Forced Only" checkbox if a) the track is "None" or b) the subtitle track doesn't support forced flags if (![self.subtitleArray[rowIndex][keySubTrackSelectionIndex] intValue] || !hb_subtitle_can_force([self.subtitleArray[rowIndex][keySubTrackType] intValue])) { [aCell setEnabled:NO]; } else { [aCell setEnabled:YES]; } } else if ([[aTableColumn identifier] isEqualToString:@"burned"]) { /* * Disable the "Burned In" checkbox if: * a) the track is "None" OR * b) the subtitle track can't be burned in OR * c) the subtitle track can't be passed through (e.g. PGS w/MP4) */ int subtitleTrackType = [self.subtitleArray[rowIndex][keySubTrackType] intValue]; if (![self.subtitleArray[rowIndex][keySubTrackSelectionIndex] intValue] || !hb_subtitle_can_burn(subtitleTrackType) || !hb_subtitle_can_pass(subtitleTrackType, self.container)) { [aCell setEnabled:NO]; } else { [aCell setEnabled:YES]; } } else if ([[aTableColumn identifier] isEqualToString:@"default"]) { /* * Disable the "Default" checkbox if: * a) the track is "None" OR * b) the subtitle track can't be passed through (e.g. PGS w/MP4) */ if (![self.subtitleArray[rowIndex][keySubTrackSelectionIndex] intValue] || !hb_subtitle_can_pass([self.subtitleArray[rowIndex][keySubTrackType] intValue], self.container)) { [aCell setEnabled:NO]; } else { [aCell setEnabled:YES]; } } /* These next three columns only apply to srt's. they are disabled for source subs */ else if ([[aTableColumn identifier] isEqualToString:@"srt_lang"]) { /* We have an srt file so set the track type (Source or SRT, and the srt file path ) kvp's*/ if ([self.subtitleArray[rowIndex][keySubTrackType] intValue] == SRTSUB) { [aCell setEnabled:YES]; } else { [aCell setEnabled:NO]; } } else if ([[aTableColumn identifier] isEqualToString:@"srt_charcode"]) { /* We have an srt file so set the track type (Source or SRT, and the srt file path ) kvp's*/ if ([self.subtitleArray[rowIndex][keySubTrackType] intValue] == SRTSUB) { [aCell setEnabled:YES]; } else { [aCell setEnabled:NO]; } } else if ([[aTableColumn identifier] isEqualToString:@"srt_offset"]) { if ([self.subtitleArray[rowIndex][keySubTrackType] intValue] == SRTSUB) { [aCell setEnabled:YES]; } else { [aCell setEnabled:NO]; } } } #pragma mark - Srt import /** * Imports a srt file. * * @param sender */ - (IBAction)browseImportSrtFile:(id)sender { NSOpenPanel *panel = [NSOpenPanel openPanel]; [panel setAllowsMultipleSelection:NO]; [panel setCanChooseFiles:YES]; [panel setCanChooseDirectories:NO]; NSURL *sourceDirectory; if ([[NSUserDefaults standardUserDefaults] URLForKey:@"LastSrtImportDirectoryURL"]) { sourceDirectory = [[NSUserDefaults standardUserDefaults] URLForKey:@"LastSrtImportDirectoryURL"]; } else { sourceDirectory = [[NSURL fileURLWithPath:NSHomeDirectory()] URLByAppendingPathComponent:@"Desktop"]; } /* we open up the browse srt sheet here and call for browseImportSrtFileDone after the sheet is closed */ NSArray *fileTypes = @[@"plist", @"srt"]; [panel setDirectoryURL:sourceDirectory]; [panel setAllowedFileTypes:fileTypes]; [panel beginSheetModalForWindow:[[self view] window] completionHandler:^(NSInteger result) { if (result == NSOKButton) { NSURL *importSrtFileURL = [panel URL]; NSURL *importSrtDirectory = [importSrtFileURL URLByDeletingLastPathComponent]; [[NSUserDefaults standardUserDefaults] setURL:importSrtDirectory forKey:@"LastSrtImportDirectoryURL"]; /* Create a new entry for the subtitle source array so it shows up in our subtitle source list */ NSString *displayname = [importSrtFileURL lastPathComponent];// grok an appropriate display name from the srt subtitle */ /* create a dictionary of source subtitle information to store in our array */ [self.subtitleSourceArray addObject:@{keySubTrackIndex: @(self.subtitleSourceArray.count), keySubTrackName: displayname, keySubTrackType: @(SRTSUB), keySubTrackSrtFilePath: importSrtFileURL.path}]; // Now create a new srt subtitle dictionary assuming the user wants to add it to their list NSMutableDictionary *newSubtitleSrtTrack = [self trackFromSourceTrackIndex:self.subtitleSourceArray.count - 1]; // Calculate the pop up selection index newSubtitleSrtTrack[keySubTrackSelectionIndex] = @(self.subtitleSourceArray.count + (self.subtitleArray.count == 1)); [self.subtitleArray insertObject:newSubtitleSrtTrack atIndex:self.subtitleArray.count - 1]; [self.fTableView reloadData]; } }]; } #pragma mark - UI cells - (NSArray *)populateLanguageArray { NSMutableArray *languages = [[[NSMutableArray alloc] init] autorelease]; for (const iso639_lang_t * lang = lang_get_next(NULL); lang != NULL; lang = lang_get_next(lang)) { [languages addObject:@[@(lang->eng_name), @(lang->iso639_2)]]; if (!strcasecmp(lang->eng_name, "English")) { _languagesArrayDefIndex = [languages count] - 1; } } return [[languages copy] autorelease]; } @synthesize languagesCell = _languagesCell; - (NSPopUpButtonCell *)languagesCell { if (!_languagesCell) { // 'srt_lang' is a popup of commonly used languages to be matched to the source srt file _languagesCell = [[NSPopUpButtonCell alloc] init]; // Set the Popups properties [_languagesCell setControlSize:NSSmallControlSize]; [_languagesCell setFont:[NSFont systemFontOfSize:[NSFont systemFontSizeForControlSize:NSSmallControlSize]]]; // list our languages as per the languagesArray for (NSArray *lang in self.languagesArray) { [[_languagesCell menu] addItemWithTitle:lang[0] action:NULL keyEquivalent:@""]; } } return _languagesCell; } @synthesize encodingsCell = _encodingsCell; - (NSPopUpButtonCell *)encodingsCell { if (!_encodingsCell) { // 'srt_charcode' is a popup of commonly used character codes to be matched to the source srt file _encodingsCell = [[NSPopUpButtonCell alloc] init]; // Set the Popups properties [_encodingsCell setControlSize:NSSmallControlSize]; [_encodingsCell setFont:[NSFont systemFontOfSize:[NSFont systemFontSizeForControlSize:NSSmallControlSize]]]; // list our character codes, as per charCodeArray for (NSString *charCode in self.charCodeArray) { [[_encodingsCell menu] addItemWithTitle:charCode action: NULL keyEquivalent: @""]; } } return _encodingsCell; } @end