VLCMediaFileDiscoverer.m 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. /*****************************************************************************
  2. * VLCMediaFileDiscoverer.m
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright (c) 2013 VideoLAN. All rights reserved.
  6. * $Id$
  7. *
  8. * Authors: Gleb Pinigin <gpinigin # gmail.com>
  9. *
  10. * Refer to the COPYING file of the official project for license.
  11. *****************************************************************************/
  12. #import "VLCMediaFileDiscoverer.h"
  13. #import "NSString+SupportedMedia.h"
  14. const float MediaTimerInterval = 2.f;
  15. @interface VLCMediaFileDiscoverer () {
  16. NSMutableArray *_observers;
  17. dispatch_source_t _directorySource;
  18. NSString *_directoryPath;
  19. NSArray *_directoryFiles;
  20. NSMutableDictionary *_addedFilesMapping;
  21. NSTimer *_addMediaTimer;
  22. }
  23. @end
  24. @implementation VLCMediaFileDiscoverer
  25. - (id)init
  26. {
  27. self = [super init];
  28. if (self) {
  29. _observers = [NSMutableArray array];
  30. _addedFilesMapping = [NSMutableDictionary dictionary];
  31. }
  32. return self;
  33. }
  34. + (instancetype)sharedInstance
  35. {
  36. static dispatch_once_t onceToken;
  37. static VLCMediaFileDiscoverer *instance;
  38. dispatch_once(&onceToken, ^{
  39. instance = [VLCMediaFileDiscoverer new];
  40. });
  41. return instance;
  42. }
  43. #pragma mark - observation
  44. - (void)addObserver:(id<VLCMediaFileDiscovererDelegate>)delegate
  45. {
  46. [_observers addObject:delegate];
  47. }
  48. - (void)removeObserver:(id<VLCMediaFileDiscovererDelegate>)delegate
  49. {
  50. [_observers removeObject:delegate];
  51. }
  52. - (void)notifyFileDeleted:(NSString *)fileName
  53. {
  54. if (![fileName isSupportedMediaFormat] && ![fileName isSupportedAudioMediaFormat])
  55. return;
  56. for (id<VLCMediaFileDiscovererDelegate> delegate in _observers) {
  57. if ([delegate respondsToSelector:@selector(mediaFileDeleted:)]) {
  58. [delegate mediaFileDeleted:[self filePath:fileName]];
  59. }
  60. }
  61. }
  62. - (void)notifyFileAdded:(NSString *)fileName loading:(BOOL)isLoading
  63. {
  64. for (id<VLCMediaFileDiscovererDelegate> delegate in _observers) {
  65. if ([delegate respondsToSelector:@selector(mediaFileAdded:loading:)]) {
  66. [delegate mediaFileAdded:[self filePath:fileName] loading:isLoading];
  67. }
  68. }
  69. }
  70. - (void)notifySizeChanged:(NSString *)fileName size:(unsigned long long)size
  71. {
  72. for (id<VLCMediaFileDiscovererDelegate> delegate in _observers) {
  73. if ([delegate respondsToSelector:@selector(mediaFileChanged:size:)]) {
  74. [delegate mediaFileChanged:[self filePath:fileName] size:size];
  75. }
  76. }
  77. }
  78. #pragma mark - discovering
  79. - (void)startDiscovering:(NSString *)directoryPath
  80. {
  81. _directoryPath = directoryPath;
  82. _directoryFiles = [self directoryFiles];
  83. int const folderDescriptor = open([directoryPath fileSystemRepresentation], O_EVTONLY);
  84. _directorySource = dispatch_source_create(DISPATCH_SOURCE_TYPE_VNODE, folderDescriptor,
  85. DISPATCH_VNODE_WRITE, DISPATCH_TARGET_QUEUE_DEFAULT);
  86. dispatch_source_set_event_handler(_directorySource, ^(){
  87. unsigned long const data = dispatch_source_get_data(_directorySource);
  88. if (data & DISPATCH_VNODE_WRITE) {
  89. // Do all the work on the main thread,
  90. // including timer scheduling, notifications delivering
  91. dispatch_async(dispatch_get_main_queue(), ^{
  92. [self directoryDidChange];
  93. });
  94. }
  95. });
  96. dispatch_source_set_cancel_handler(_directorySource, ^(){
  97. close(folderDescriptor);
  98. });
  99. dispatch_resume(_directorySource);
  100. }
  101. - (void)stopDiscovering
  102. {
  103. dispatch_source_cancel(_directorySource);
  104. [self invalidateTimer];
  105. }
  106. #pragma mark -
  107. - (NSArray *)directoryFiles
  108. {
  109. NSArray *foundFiles = [[NSFileManager defaultManager] contentsOfDirectoryAtPath:_directoryPath
  110. error:nil];
  111. return foundFiles;
  112. }
  113. - (NSString *)filePath:(NSString *)fileName
  114. {
  115. return [_directoryPath stringByAppendingPathComponent:fileName];
  116. }
  117. #pragma mark - directory watcher delegate
  118. - (void)directoryDidChange
  119. {
  120. NSArray *foundFiles = [self directoryFiles];
  121. if (_directoryFiles.count > foundFiles.count) { // File was deleted
  122. NSPredicate *filterPredicate = [NSPredicate predicateWithFormat:@"not (self in %@)", foundFiles];
  123. NSArray *deletedFiles = [_directoryFiles filteredArrayUsingPredicate:filterPredicate];
  124. for (NSString *fileName in deletedFiles) {
  125. [self notifyFileDeleted:fileName];
  126. }
  127. } else if (_directoryFiles.count < foundFiles.count) { // File was added
  128. NSPredicate *filterPredicate = [NSPredicate predicateWithFormat:@"not (self in %@)", _directoryFiles];
  129. NSArray *addedFiles = [foundFiles filteredArrayUsingPredicate:filterPredicate];
  130. for (NSString *fileName in addedFiles) {
  131. if ([fileName isSupportedMediaFormat] || [fileName isSupportedAudioMediaFormat]) {
  132. [_addedFilesMapping setObject:@(0) forKey:fileName];
  133. [self notifyFileAdded:fileName loading:YES];
  134. }
  135. }
  136. if (![_addMediaTimer isValid]) {
  137. _addMediaTimer = [NSTimer scheduledTimerWithTimeInterval:MediaTimerInterval
  138. target:self selector:@selector(addFileTimerFired)
  139. userInfo:nil repeats:YES];
  140. }
  141. }
  142. _directoryFiles = foundFiles;
  143. }
  144. #pragma mark - media timer
  145. - (void)addFileTimerFired
  146. {
  147. NSArray *allKeys = [_addedFilesMapping allKeys];
  148. NSFileManager *fileManager = [NSFileManager defaultManager];
  149. for (NSString *fileName in allKeys) {
  150. NSString *filePath = [self filePath:fileName];
  151. if (![fileManager fileExistsAtPath:filePath]) {
  152. [_addedFilesMapping removeObjectForKey:fileName];
  153. continue;
  154. }
  155. NSNumber *prevFetchedSize = [_addedFilesMapping objectForKey:fileName];
  156. NSDictionary *attribs = [fileManager attributesOfItemAtPath:filePath error:nil];
  157. NSNumber *updatedSize = [attribs objectForKey:NSFileSize];
  158. if (!updatedSize)
  159. continue;
  160. [self notifySizeChanged:fileName size:[updatedSize unsignedLongLongValue]];
  161. if ([prevFetchedSize compare:updatedSize] == NSOrderedSame) {
  162. [_addedFilesMapping removeObjectForKey:fileName];
  163. [self notifyFileAdded:fileName loading:NO];
  164. } else
  165. [_addedFilesMapping setObject:updatedSize forKey:fileName];
  166. }
  167. if (_addedFilesMapping.count == 0)
  168. [self invalidateTimer];
  169. }
  170. - (void)invalidateTimer
  171. {
  172. [_addMediaTimer invalidate];
  173. _addMediaTimer = nil;
  174. }
  175. @end