/* HBJob.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 "HBJob.h" #import "HBJob+Private.h" #import "HBTitle+Private.h" #import "HBAudioDefaults.h" #import "HBSubtitlesDefaults.h" #import "HBMutablePreset.h" #import "HBCodingUtilities.h" #import "HBLocalizationUtilities.h" #import "HBUtilities.h" #import "HBSecurityAccessToken.h" #include "hb.h" NSString *HBContainerChangedNotification = @"HBContainerChangedNotification"; NSString *HBChaptersChangedNotification = @"HBChaptersChangedNotification"; @interface HBJob () @property (nonatomic, readonly) NSString *name; /** Store the security scoped bookmarks, so we don't regenerate it each time */ @property (nonatomic, readonly) NSData *fileURLBookmark; @property (nonatomic, readwrite) NSData *outputURLFolderBookmark; /** Keep track of security scoped resources status. */ @property (nonatomic, readwrite) HBSecurityAccessToken *fileURLToken; @property (nonatomic, readwrite) HBSecurityAccessToken *outputURLToken; @property (nonatomic, readwrite) HBSecurityAccessToken *subtitlesToken; @property (nonatomic, readwrite) NSInteger *accessCount; @end @implementation HBJob - (instancetype)initWithTitle:(HBTitle *)title andPreset:(HBPreset *)preset { self = [super init]; if (self) { NSParameterAssert(title); NSParameterAssert(preset); _title = title; _titleIdx = title.index; _name = [title.name copy]; _fileURL = title.url; _container = HB_MUX_MP4; _angle = 1; _range = [[HBRange alloc] initWithTitle:title]; _video = [[HBVideo alloc] initWithJob:self]; _picture = [[HBPicture alloc] initWithTitle:title]; _filters = [[HBFilters alloc] init]; _audio = [[HBAudio alloc] initWithJob:self]; _subtitles = [[HBSubtitles alloc] initWithJob:self]; _chapterTitles = [title.chapters copy]; _presetName = @""; [self applyPreset:preset]; } return self; } #pragma mark - HBPresetCoding - (void)applyPreset:(HBPreset *)preset { NSAssert(self.title, @"HBJob: calling applyPreset: without a valid title loaded"); self.presetName = preset.name; NSDictionary *jobSettings = [self.title jobSettingsWithPreset:preset]; self.container = hb_container_get_from_name([preset[@"FileFormat"] UTF8String]); // MP4 specifics options. self.mp4HttpOptimize = [preset[@"Mp4HttpOptimize"] boolValue]; self.mp4iPodCompatible = [preset[@"Mp4iPodCompatible"] boolValue]; self.alignAVStart = [preset[@"AlignAVStart"] boolValue]; // Chapter Markers self.chaptersEnabled = [preset[@"ChapterMarkers"] boolValue]; [self.audio applyPreset:preset jobSettings:jobSettings]; [self.subtitles applyPreset:preset jobSettings:jobSettings]; [self.video applyPreset:preset jobSettings:jobSettings]; [self.picture applyPreset:preset jobSettings:jobSettings]; [self.filters applyPreset:preset jobSettings:jobSettings]; } - (void)writeToPreset:(HBMutablePreset *)preset { preset.name = self.presetName; preset[@"FileFormat"] = @(hb_container_get_short_name(self.container)); preset[@"ChapterMarkers"] = @(self.chaptersEnabled); // MP4 specifics options. preset[@"Mp4HttpOptimize"] = @(self.mp4HttpOptimize); preset[@"Mp4iPodCompatible"] = @(self.mp4iPodCompatible); preset[@"AlignAVStart"] = @(self.alignAVStart); [@[self.video, self.filters, self.picture, self.audio, self.subtitles] makeObjectsPerformSelector:@selector(writeToPreset:) withObject:preset]; } - (void)setUndo:(NSUndoManager *)undo { _undo = undo; [@[self.video, self.range, self.filters, self.picture, self.audio, self.subtitles] makeObjectsPerformSelector:@selector(setUndo:) withObject:_undo]; [self.chapterTitles makeObjectsPerformSelector:@selector(setUndo:) withObject:_undo]; } - (void)setPresetName:(NSString *)presetName { if (![presetName isEqualToString:_presetName]) { [[self.undo prepareWithInvocationTarget:self] setPresetName:_presetName]; } _presetName = [presetName copy]; } - (void)setOutputURL:(NSURL *)outputURL { if (![outputURL isEqualTo:_outputURL]) { [[self.undo prepareWithInvocationTarget:self] setOutputURL:_outputURL]; } _outputURL = [outputURL copy]; #ifdef __SANDBOX_ENABLED__ // Clear the bookmark to regenerate it later self.outputURLFolderBookmark = nil; #endif } - (void)setOutputFileName:(NSString *)outputFileName { if (![outputFileName isEqualTo:_outputFileName]) { [[self.undo prepareWithInvocationTarget:self] setOutputFileName:_outputFileName]; } _outputFileName = [outputFileName copy]; } - (BOOL)validateOutputFileName:(id *)ioValue error:(NSError * __autoreleasing *)outError { BOOL retval = YES; if (nil != *ioValue) { NSString *value = *ioValue; if ([value rangeOfString:@"/"].location != NSNotFound) { if (outError) { *outError = [NSError errorWithDomain:@"HBError" code:0 userInfo:@{NSLocalizedDescriptionKey: HBKitLocalizedString(@"Invalid name", @"HBJob -> invalid name error description"), NSLocalizedRecoverySuggestionErrorKey: HBKitLocalizedString(@"The file name can't contain the / character.", @"HBJob -> invalid name error recovery suggestion")}]; } return NO; } if (value.length == 0) { if (outError) { *outError = [NSError errorWithDomain:@"HBError" code:0 userInfo:@{NSLocalizedDescriptionKey: HBKitLocalizedString(@"Invalid name", @"HBJob -> invalid name error description"), NSLocalizedRecoverySuggestionErrorKey: HBKitLocalizedString(@"The file name can't be empty.", @"HBJob -> invalid name error recovery suggestion")}]; } return NO; } } if (*ioValue == nil) { if (outError) { *outError = [NSError errorWithDomain:@"HBError" code:0 userInfo:@{NSLocalizedDescriptionKey: HBKitLocalizedString(@"Invalid name", @"HBJob -> invalid name error description"), NSLocalizedRecoverySuggestionErrorKey: HBKitLocalizedString(@"The file name can't be empty.", @"HBJob -> invalid name error recovery suggestion")}]; } return NO; } return retval; } - (NSURL *)completeOutputURL { return [self.outputURL URLByAppendingPathComponent:self.outputFileName]; } - (void)setContainer:(int)container { if (container != _container) { [[self.undo prepareWithInvocationTarget:self] setContainer:_container]; } _container = container; [self.audio setContainer:container]; [self.subtitles setContainer:container]; [self.video containerChanged]; // post a notification for any interested observers to indicate that our video container has changed [[NSNotificationCenter defaultCenter] postNotificationName:HBContainerChangedNotification object:self]; } - (void)setAngle:(int)angle { if (angle != _angle) { [[self.undo prepareWithInvocationTarget:self] setAngle:_angle]; } _angle = angle; } - (void)setTitle:(HBTitle *)title { _title = title; self.range.title = title; } - (void)setMp4HttpOptimize:(BOOL)mp4HttpOptimize { if (mp4HttpOptimize != _mp4HttpOptimize) { [[self.undo prepareWithInvocationTarget:self] setMp4HttpOptimize:_mp4HttpOptimize]; } _mp4HttpOptimize = mp4HttpOptimize; [[NSNotificationCenter defaultCenter] postNotificationName:HBContainerChangedNotification object:self]; } - (void)setAlignAVStart:(BOOL)alignAVStart { if (alignAVStart != _alignAVStart) { [[self.undo prepareWithInvocationTarget:self] setAlignAVStart:_alignAVStart]; } _alignAVStart = alignAVStart; [[NSNotificationCenter defaultCenter] postNotificationName:HBContainerChangedNotification object:self]; } - (void)setMp4iPodCompatible:(BOOL)mp4iPodCompatible { if (mp4iPodCompatible != _mp4iPodCompatible) { [[self.undo prepareWithInvocationTarget:self] setMp4iPodCompatible:_mp4iPodCompatible]; } _mp4iPodCompatible = mp4iPodCompatible; [[NSNotificationCenter defaultCenter] postNotificationName:HBContainerChangedNotification object:self]; } - (void)setChaptersEnabled:(BOOL)chaptersEnabled { if (chaptersEnabled != _chaptersEnabled) { [[self.undo prepareWithInvocationTarget:self] setChaptersEnabled:_chaptersEnabled]; } _chaptersEnabled = chaptersEnabled; [[NSNotificationCenter defaultCenter] postNotificationName:HBChaptersChangedNotification object:self]; } - (NSString *)description { return self.name; } - (BOOL)startAccessingSecurityScopedResource { #ifdef __SANDBOX_ENABLED__ if (self.accessCount == 0) { self.fileURLToken = [HBSecurityAccessToken tokenWithObject:self.fileURL]; self.outputURLToken = [HBSecurityAccessToken tokenWithObject:self.outputURL]; self.subtitlesToken = [HBSecurityAccessToken tokenWithObject:self.subtitles]; } self.accessCount += 1; return YES; #else return NO; #endif } - (void)stopAccessingSecurityScopedResource { #ifdef __SANDBOX_ENABLED__ self.accessCount -= 1; NSAssert(self.accessCount >= 0, @"[HBJob stopAccessingSecurityScopedResource:] unbalanced call"); if (self.accessCount == 0) { self.fileURLToken = nil; self.outputURLToken = nil; self.subtitlesToken = nil; } #endif } #pragma mark - NSCopying - (instancetype)copyWithZone:(NSZone *)zone { HBJob *copy = [[[self class] alloc] init]; if (copy) { copy->_name = [_name copy]; copy->_presetName = [_presetName copy]; copy->_titleIdx = _titleIdx; copy->_fileURLBookmark = [_fileURLBookmark copy]; copy->_outputURLFolderBookmark = [_outputURLFolderBookmark copy]; copy->_fileURL = [_fileURL copy]; copy->_outputURL = [_outputURL copy]; copy->_outputFileName = [_outputFileName copy]; copy->_container = _container; copy->_angle = _angle; copy->_mp4HttpOptimize = _mp4HttpOptimize; copy->_mp4iPodCompatible = _mp4iPodCompatible; copy->_alignAVStart = _alignAVStart; copy->_range = [_range copy]; copy->_video = [_video copy]; copy->_picture = [_picture copy]; copy->_filters = [_filters copy]; copy->_video.job = copy; copy->_audio = [_audio copy]; copy->_audio.job = copy; copy->_subtitles = [_subtitles copy]; copy->_subtitles.job = copy; copy->_chaptersEnabled = _chaptersEnabled; copy->_chapterTitles = [[NSArray alloc] initWithArray:_chapterTitles copyItems:YES]; } return copy; } #pragma mark - NSCoding + (BOOL)supportsSecureCoding { return YES; } - (void)encodeWithCoder:(NSCoder *)coder { [coder encodeInt:4 forKey:@"HBJobVersion"]; encodeObject(_name); encodeObject(_presetName); encodeInt(_titleIdx); #ifdef __SANDBOX_ENABLED__ if (!_fileURLBookmark) { _fileURLBookmark = [HBUtilities bookmarkFromURL:_fileURL options:NSURLBookmarkCreationWithSecurityScope | NSURLBookmarkCreationSecurityScopeAllowOnlyReadAccess]; } encodeObject(_fileURLBookmark); if (!_outputURLFolderBookmark) { __attribute__((unused)) HBSecurityAccessToken *token = [HBSecurityAccessToken tokenWithObject:_outputURL]; _outputURLFolderBookmark = [HBUtilities bookmarkFromURL:_outputURL]; token = nil; } encodeObject(_outputURLFolderBookmark); #endif encodeObject(_fileURL); encodeObject(_outputURL); encodeObject(_outputFileName); encodeInt(_container); encodeInt(_angle); encodeBool(_mp4HttpOptimize); encodeBool(_mp4iPodCompatible); encodeBool(_alignAVStart); encodeObject(_range); encodeObject(_video); encodeObject(_picture); encodeObject(_filters); encodeObject(_audio); encodeObject(_subtitles); encodeBool(_chaptersEnabled); encodeObject(_chapterTitles); } - (instancetype)initWithCoder:(NSCoder *)decoder { int version = [decoder decodeIntForKey:@"HBJobVersion"]; if (version == 4 && (self = [super init])) { decodeObjectOrFail(_name, NSString); decodeObjectOrFail(_presetName, NSString); decodeInt(_titleIdx); if (_titleIdx < 0) { goto fail; } #ifdef __SANDBOX_ENABLED__ decodeObject(_fileURLBookmark, NSData) if (_fileURLBookmark) { _fileURL = [HBUtilities URLFromBookmark:_fileURLBookmark]; } if (!_fileURL) { decodeObjectOrFail(_fileURL, NSURL); } decodeObject(_outputURLFolderBookmark, NSData) if (_outputURLFolderBookmark) { _outputURL = [HBUtilities URLFromBookmark:_outputURLFolderBookmark]; } if (!_outputURL) { decodeObject(_outputURL, NSURL); } #else decodeObjectOrFail(_fileURL, NSURL); decodeObject(_outputURL, NSURL); #endif decodeObject(_outputFileName, NSString); decodeInt(_container); if (_container != HB_MUX_MP4 && _container != HB_MUX_MKV && _container != HB_MUX_WEBM) { goto fail; } decodeInt(_angle); if (_angle < 0) { goto fail; } decodeBool(_mp4HttpOptimize); decodeBool(_mp4iPodCompatible); decodeBool(_alignAVStart); decodeObjectOrFail(_range, HBRange); decodeObjectOrFail(_video, HBVideo); decodeObjectOrFail(_picture, HBPicture); decodeObjectOrFail(_filters, HBFilters); _video.job = self; decodeObjectOrFail(_audio, HBAudio); decodeObjectOrFail(_subtitles, HBSubtitles); _audio.job = self; _video.job = self; decodeBool(_chaptersEnabled); decodeCollectionOfObjects(_chapterTitles, NSArray, HBChapter); return self; } fail: return nil; } @end