YTPlayerView.m 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943
  1. // Copyright 2014 Google Inc. All rights reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #import "YTPlayerView.h"
  15. // These are instances of NSString because we get them from parsing a URL. It would be silly to
  16. // convert these into an integer just to have to convert the URL query string value into an integer
  17. // as well for the sake of doing a value comparison. A full list of response error codes can be
  18. // found here:
  19. // https://developers.google.com/youtube/iframe_api_reference
  20. NSString static *const kYTPlayerStateUnstartedCode = @"-1";
  21. NSString static *const kYTPlayerStateEndedCode = @"0";
  22. NSString static *const kYTPlayerStatePlayingCode = @"1";
  23. NSString static *const kYTPlayerStatePausedCode = @"2";
  24. NSString static *const kYTPlayerStateBufferingCode = @"3";
  25. NSString static *const kYTPlayerStateCuedCode = @"5";
  26. NSString static *const kYTPlayerStateUnknownCode = @"unknown";
  27. // Constants representing playback quality.
  28. NSString static *const kYTPlaybackQualitySmallQuality = @"small";
  29. NSString static *const kYTPlaybackQualityMediumQuality = @"medium";
  30. NSString static *const kYTPlaybackQualityLargeQuality = @"large";
  31. NSString static *const kYTPlaybackQualityHD720Quality = @"hd720";
  32. NSString static *const kYTPlaybackQualityHD1080Quality = @"hd1080";
  33. NSString static *const kYTPlaybackQualityHighResQuality = @"highres";
  34. NSString static *const kYTPlaybackQualityAutoQuality = @"auto";
  35. NSString static *const kYTPlaybackQualityDefaultQuality = @"default";
  36. NSString static *const kYTPlaybackQualityUnknownQuality = @"unknown";
  37. // Constants representing YouTube player errors.
  38. NSString static *const kYTPlayerErrorInvalidParamErrorCode = @"2";
  39. NSString static *const kYTPlayerErrorHTML5ErrorCode = @"5";
  40. NSString static *const kYTPlayerErrorVideoNotFoundErrorCode = @"100";
  41. NSString static *const kYTPlayerErrorNotEmbeddableErrorCode = @"101";
  42. NSString static *const kYTPlayerErrorCannotFindVideoErrorCode = @"105";
  43. NSString static *const kYTPlayerErrorSameAsNotEmbeddableErrorCode = @"150";
  44. // Constants representing player callbacks.
  45. NSString static *const kYTPlayerCallbackOnReady = @"onReady";
  46. NSString static *const kYTPlayerCallbackOnStateChange = @"onStateChange";
  47. NSString static *const kYTPlayerCallbackOnPlaybackQualityChange = @"onPlaybackQualityChange";
  48. NSString static *const kYTPlayerCallbackOnError = @"onError";
  49. NSString static *const kYTPlayerCallbackOnPlayTime = @"onPlayTime";
  50. NSString static *const kYTPlayerCallbackOnYouTubeIframeAPIReady = @"onYouTubeIframeAPIReady";
  51. NSString static *const kYTPlayerCallbackOnYouTubeIframeAPIFailedToLoad = @"onYouTubeIframeAPIFailedToLoad";
  52. NSString static *const kYTPlayerEmbedUrlRegexPattern = @"^http(s)://(www.)youtube.com/embed/(.*)$";
  53. NSString static *const kYTPlayerAdUrlRegexPattern = @"^http(s)://pubads.g.doubleclick.net/pagead/conversion/";
  54. NSString static *const kYTPlayerOAuthRegexPattern = @"^http(s)://accounts.google.com/o/oauth2/(.*)$";
  55. NSString static *const kYTPlayerStaticProxyRegexPattern = @"^https://content.googleapis.com/static/proxy.html(.*)$";
  56. NSString static *const kYTPlayerSyndicationRegexPattern = @"^https://tpc.googlesyndication.com/sodar/(.*).html$";
  57. @interface YTPlayerView() <WKNavigationDelegate, WKUIDelegate>
  58. @property (nonatomic) NSURL *originURL;
  59. @property (nonatomic, weak) UIView *initialLoadingView;
  60. @end
  61. @implementation YTPlayerView
  62. - (BOOL)loadWithVideoId:(NSString *)videoId {
  63. return [self loadWithVideoId:videoId playerVars:nil];
  64. }
  65. - (BOOL)loadWithPlaylistId:(NSString *)playlistId {
  66. return [self loadWithPlaylistId:playlistId playerVars:nil];
  67. }
  68. - (BOOL)loadWithVideoId:(NSString *)videoId playerVars:(NSDictionary *)playerVars {
  69. if (!playerVars) {
  70. playerVars = @{};
  71. }
  72. NSDictionary *playerParams = @{ @"videoId" : videoId, @"playerVars" : playerVars };
  73. return [self loadWithPlayerParams:playerParams];
  74. }
  75. - (BOOL)loadWithPlaylistId:(NSString *)playlistId playerVars:(NSDictionary *)playerVars {
  76. // Mutable copy because we may have been passed an immutable config dictionary.
  77. NSMutableDictionary *tempPlayerVars = [[NSMutableDictionary alloc] init];
  78. [tempPlayerVars setValue:@"playlist" forKey:@"listType"];
  79. [tempPlayerVars setValue:playlistId forKey:@"list"];
  80. if (playerVars) {
  81. [tempPlayerVars addEntriesFromDictionary:playerVars];
  82. }
  83. NSDictionary *playerParams = @{ @"playerVars" : tempPlayerVars };
  84. return [self loadWithPlayerParams:playerParams];
  85. }
  86. #pragma mark - Player methods
  87. - (void)playVideo {
  88. [self evaluateJavaScript:@"player.playVideo();"];
  89. }
  90. - (void)pauseVideo {
  91. [self notifyDelegateOfYouTubeCallbackUrl:[NSURL URLWithString:[NSString stringWithFormat:@"ytplayer://onStateChange?data=%@", kYTPlayerStatePausedCode]]];
  92. [self evaluateJavaScript:@"player.pauseVideo();"];
  93. }
  94. - (void)stopVideo {
  95. [self evaluateJavaScript:@"player.stopVideo();"];
  96. }
  97. - (void)seekToSeconds:(float)seekToSeconds allowSeekAhead:(BOOL)allowSeekAhead {
  98. NSNumber *secondsValue = [NSNumber numberWithFloat:seekToSeconds];
  99. NSString *allowSeekAheadValue = [self stringForJSBoolean:allowSeekAhead];
  100. NSString *command = [NSString stringWithFormat:@"player.seekTo(%@, %@);", secondsValue, allowSeekAheadValue];
  101. [self evaluateJavaScript:command];
  102. }
  103. #pragma mark - Cueing methods
  104. - (void)cueVideoById:(NSString *)videoId
  105. startSeconds:(float)startSeconds {
  106. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  107. NSString *command = [NSString stringWithFormat:@"player.cueVideoById('%@', %@);",
  108. videoId, startSecondsValue];
  109. [self evaluateJavaScript:command];
  110. }
  111. - (void)cueVideoById:(NSString *)videoId
  112. startSeconds:(float)startSeconds
  113. endSeconds:(float)endSeconds {
  114. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  115. NSNumber *endSecondsValue = [NSNumber numberWithFloat:endSeconds];
  116. NSString *command = [NSString stringWithFormat:@"player.cueVideoById({'videoId': '%@',"
  117. "'startSeconds': %@, 'endSeconds': %@});",
  118. videoId, startSecondsValue, endSecondsValue];
  119. [self evaluateJavaScript:command];
  120. }
  121. - (void)loadVideoById:(NSString *)videoId
  122. startSeconds:(float)startSeconds {
  123. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  124. NSString *command = [NSString stringWithFormat:@"player.loadVideoById('%@', %@);",
  125. videoId, startSecondsValue];
  126. [self evaluateJavaScript:command];
  127. }
  128. - (void)loadVideoById:(NSString *)videoId
  129. startSeconds:(float)startSeconds
  130. endSeconds:(float)endSeconds {
  131. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  132. NSNumber *endSecondsValue = [NSNumber numberWithFloat:endSeconds];
  133. NSString *command = [NSString stringWithFormat:@"player.loadVideoById({'videoId': '%@',"
  134. "'startSeconds': %@, 'endSeconds': %@});",
  135. videoId, startSecondsValue, endSecondsValue];
  136. [self evaluateJavaScript:command];
  137. }
  138. - (void)cueVideoByURL:(NSString *)videoURL
  139. startSeconds:(float)startSeconds {
  140. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  141. NSString *command = [NSString stringWithFormat:@"player.cueVideoByUrl('%@', %@);",
  142. videoURL, startSecondsValue];
  143. [self evaluateJavaScript:command];
  144. }
  145. - (void)cueVideoByURL:(NSString *)videoURL
  146. startSeconds:(float)startSeconds
  147. endSeconds:(float)endSeconds {
  148. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  149. NSNumber *endSecondsValue = [NSNumber numberWithFloat:endSeconds];
  150. NSString *command = [NSString stringWithFormat:@"player.cueVideoByUrl('%@', %@, %@);",
  151. videoURL, startSecondsValue, endSecondsValue];
  152. [self evaluateJavaScript:command];
  153. }
  154. - (void)loadVideoByURL:(NSString *)videoURL
  155. startSeconds:(float)startSeconds {
  156. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  157. NSString *command = [NSString stringWithFormat:@"player.loadVideoByUrl('%@', %@);",
  158. videoURL, startSecondsValue];
  159. [self evaluateJavaScript:command];
  160. }
  161. - (void)loadVideoByURL:(NSString *)videoURL
  162. startSeconds:(float)startSeconds
  163. endSeconds:(float)endSeconds {
  164. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  165. NSNumber *endSecondsValue = [NSNumber numberWithFloat:endSeconds];
  166. NSString *command = [NSString stringWithFormat:@"player.loadVideoByUrl('%@', %@, %@);",
  167. videoURL, startSecondsValue, endSecondsValue];
  168. [self evaluateJavaScript:command];
  169. }
  170. #pragma mark - Cueing methods for lists
  171. - (void)cuePlaylistByPlaylistId:(NSString *)playlistId
  172. index:(int)index
  173. startSeconds:(float)startSeconds {
  174. NSString *playlistIdString = [NSString stringWithFormat:@"'%@'", playlistId];
  175. [self cuePlaylist:playlistIdString
  176. index:index
  177. startSeconds:startSeconds];
  178. }
  179. - (void)cuePlaylistByVideos:(NSArray *)videoIds
  180. index:(int)index
  181. startSeconds:(float)startSeconds {
  182. [self cuePlaylist:[self stringFromVideoIdArray:videoIds]
  183. index:index
  184. startSeconds:startSeconds];
  185. }
  186. - (void)loadPlaylistByPlaylistId:(NSString *)playlistId
  187. index:(int)index
  188. startSeconds:(float)startSeconds {
  189. NSString *playlistIdString = [NSString stringWithFormat:@"'%@'", playlistId];
  190. [self loadPlaylist:playlistIdString
  191. index:index
  192. startSeconds:startSeconds];
  193. }
  194. - (void)loadPlaylistByVideos:(NSArray *)videoIds
  195. index:(int)index
  196. startSeconds:(float)startSeconds {
  197. [self loadPlaylist:[self stringFromVideoIdArray:videoIds]
  198. index:index
  199. startSeconds:startSeconds];
  200. }
  201. #pragma mark - Setting the playback rate
  202. - (void)playbackRate:(_Nullable YTFloatCompletionHandler)completionHandler {
  203. [self evaluateJavaScript:@"player.getPlaybackRate();"
  204. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  205. if (!completionHandler) {
  206. return;
  207. }
  208. if (error) {
  209. completionHandler(-1, error);
  210. return;
  211. }
  212. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  213. completionHandler(0, nil);
  214. return;
  215. }
  216. completionHandler([result floatValue], nil);
  217. }];
  218. }
  219. - (void)setPlaybackRate:(float)suggestedRate {
  220. NSString *command = [NSString stringWithFormat:@"player.setPlaybackRate(%f);", suggestedRate];
  221. [self evaluateJavaScript:command];
  222. }
  223. - (void)availablePlaybackRates:(_Nullable YTArrayCompletionHandler)completionHandler {
  224. [self evaluateJavaScript:@"player.getAvailablePlaybackRates();"
  225. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  226. if (!completionHandler) {
  227. return;
  228. }
  229. if (error) {
  230. completionHandler(nil, error);
  231. return;
  232. }
  233. if (!result || ![result isKindOfClass:[NSArray class]]) {
  234. completionHandler(nil, nil);
  235. return;
  236. }
  237. completionHandler(result, nil);
  238. }];
  239. }
  240. #pragma mark - Setting playback behavior for playlists
  241. - (void)setLoop:(BOOL)loop {
  242. NSString *loopPlayListValue = [self stringForJSBoolean:loop];
  243. NSString *command = [NSString stringWithFormat:@"player.setLoop(%@);", loopPlayListValue];
  244. [self evaluateJavaScript:command];
  245. }
  246. - (void)setShuffle:(BOOL)shuffle {
  247. NSString *shufflePlayListValue = [self stringForJSBoolean:shuffle];
  248. NSString *command = [NSString stringWithFormat:@"player.setShuffle(%@);", shufflePlayListValue];
  249. [self evaluateJavaScript:command];
  250. }
  251. #pragma mark - Playback status
  252. - (void)videoLoadedFraction:(_Nullable YTFloatCompletionHandler)completionHandler {
  253. [self evaluateJavaScript:@"player.getVideoLoadedFraction();"
  254. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  255. if (!completionHandler) {
  256. return;
  257. }
  258. if (error) {
  259. completionHandler(-1, error);
  260. return;
  261. }
  262. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  263. completionHandler(0, nil);
  264. return;
  265. }
  266. completionHandler([result floatValue], nil);
  267. }];
  268. }
  269. - (void)playerState:(_Nullable YTPlayerStateCompletionHandler)completionHandler {
  270. [self evaluateJavaScript:@"player.getPlayerState();"
  271. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  272. if (!completionHandler) {
  273. return;
  274. }
  275. if (error) {
  276. completionHandler(kYTPlayerStateUnknown, error);
  277. return;
  278. }
  279. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  280. completionHandler(kYTPlayerStateUnknown, error);
  281. return;
  282. }
  283. YTPlayerState state = [YTPlayerView playerStateForString:[result stringValue]];
  284. completionHandler(state, nil);
  285. }];
  286. }
  287. - (void)currentTime:(_Nullable YTFloatCompletionHandler)completionHandler {
  288. [self evaluateJavaScript:@"player.getCurrentTime();"
  289. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  290. if (!completionHandler) {
  291. return;
  292. }
  293. if (error) {
  294. completionHandler(-1, error);
  295. return;
  296. }
  297. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  298. completionHandler(0, nil);
  299. return;
  300. }
  301. completionHandler([result floatValue], nil);
  302. }];
  303. }
  304. #pragma mark - Video information methods
  305. - (void)duration:(_Nullable YTDoubleCompletionHandler)completionHandler {
  306. [self evaluateJavaScript:@"player.getDuration();"
  307. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  308. if (!completionHandler) {
  309. return;
  310. }
  311. if (error) {
  312. completionHandler(-1, error);
  313. return;
  314. }
  315. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  316. completionHandler(0, nil);
  317. return;
  318. }
  319. completionHandler([result doubleValue], nil);
  320. }];
  321. }
  322. - (void)videoUrl:(_Nullable YTURLCompletionHandler)completionHandler {
  323. [self evaluateJavaScript:@"player.getVideoUrl();"
  324. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  325. if (!completionHandler) {
  326. return;
  327. }
  328. if (error) {
  329. completionHandler(nil, error);
  330. return;
  331. }
  332. if (!result || ![result isKindOfClass:[NSString class]]) {
  333. completionHandler(nil, nil);
  334. return;
  335. }
  336. completionHandler([NSURL URLWithString:result], nil);
  337. }];
  338. }
  339. - (void)videoEmbedCode:(_Nullable YTStringCompletionHandler)completionHandler {
  340. [self evaluateJavaScript:@"player.getVideoEmbedCode();"
  341. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  342. if (!completionHandler) {
  343. return;
  344. }
  345. if (error) {
  346. completionHandler(nil, error);
  347. return;
  348. }
  349. if (!result || ![result isKindOfClass:[NSString class]]) {
  350. completionHandler(nil, nil);
  351. return;
  352. }
  353. completionHandler(result, nil);
  354. }];
  355. }
  356. #pragma mark - Playlist methods
  357. - (void)playlist:(_Nullable YTArrayCompletionHandler)completionHandler {
  358. [self evaluateJavaScript:@"player.getPlaylist();"
  359. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  360. if (!completionHandler) {
  361. return;
  362. }
  363. if (error) {
  364. completionHandler(nil, error);
  365. }
  366. if (!result || ![result isKindOfClass:[NSArray class]]) {
  367. completionHandler(nil, nil);
  368. return;
  369. }
  370. completionHandler(result, nil);
  371. }];
  372. }
  373. - (void)playlistIndex:(_Nullable YTIntCompletionHandler)completionHandler {
  374. [self evaluateJavaScript:@"player.getPlaylistIndex();"
  375. completionHandler:^(id _Nullable result, NSError * _Nullable error) {
  376. if (!completionHandler) {
  377. return;
  378. }
  379. if (error) {
  380. completionHandler(-1, error);
  381. return;
  382. }
  383. if (!result || ![result isKindOfClass:[NSNumber class]]) {
  384. completionHandler(0, nil);
  385. return;
  386. }
  387. completionHandler([result intValue], nil);
  388. }];
  389. }
  390. #pragma mark - Playing a video in a playlist
  391. - (void)nextVideo {
  392. [self evaluateJavaScript:@"player.nextVideo();"];
  393. }
  394. - (void)previousVideo {
  395. [self evaluateJavaScript:@"player.previousVideo();"];
  396. }
  397. - (void)playVideoAt:(int)index {
  398. NSString *command =
  399. [NSString stringWithFormat:@"player.playVideoAt(%@);", [NSNumber numberWithInt:index]];
  400. [self evaluateJavaScript:command];
  401. }
  402. #pragma mark - Helper methods
  403. /**
  404. * Convert a quality value from NSString to the typed enum value.
  405. *
  406. * @param qualityString A string representing playback quality. Ex: "small", "medium", "hd1080".
  407. * @return An enum value representing the playback quality.
  408. */
  409. + (YTPlaybackQuality)playbackQualityForString:(NSString *)qualityString {
  410. YTPlaybackQuality quality = kYTPlaybackQualityUnknown;
  411. if ([qualityString isEqualToString:kYTPlaybackQualitySmallQuality]) {
  412. quality = kYTPlaybackQualitySmall;
  413. } else if ([qualityString isEqualToString:kYTPlaybackQualityMediumQuality]) {
  414. quality = kYTPlaybackQualityMedium;
  415. } else if ([qualityString isEqualToString:kYTPlaybackQualityLargeQuality]) {
  416. quality = kYTPlaybackQualityLarge;
  417. } else if ([qualityString isEqualToString:kYTPlaybackQualityHD720Quality]) {
  418. quality = kYTPlaybackQualityHD720;
  419. } else if ([qualityString isEqualToString:kYTPlaybackQualityHD1080Quality]) {
  420. quality = kYTPlaybackQualityHD1080;
  421. } else if ([qualityString isEqualToString:kYTPlaybackQualityHighResQuality]) {
  422. quality = kYTPlaybackQualityHighRes;
  423. } else if ([qualityString isEqualToString:kYTPlaybackQualityAutoQuality]) {
  424. quality = kYTPlaybackQualityAuto;
  425. }
  426. return quality;
  427. }
  428. /**
  429. * Convert a state value from NSString to the typed enum value.
  430. *
  431. * @param stateString A string representing player state. Ex: "-1", "0", "1".
  432. * @return An enum value representing the player state.
  433. */
  434. + (YTPlayerState)playerStateForString:(NSString *)stateString {
  435. YTPlayerState state = kYTPlayerStateUnknown;
  436. if ([stateString isEqualToString:kYTPlayerStateUnstartedCode]) {
  437. state = kYTPlayerStateUnstarted;
  438. } else if ([stateString isEqualToString:kYTPlayerStateEndedCode]) {
  439. state = kYTPlayerStateEnded;
  440. } else if ([stateString isEqualToString:kYTPlayerStatePlayingCode]) {
  441. state = kYTPlayerStatePlaying;
  442. } else if ([stateString isEqualToString:kYTPlayerStatePausedCode]) {
  443. state = kYTPlayerStatePaused;
  444. } else if ([stateString isEqualToString:kYTPlayerStateBufferingCode]) {
  445. state = kYTPlayerStateBuffering;
  446. } else if ([stateString isEqualToString:kYTPlayerStateCuedCode]) {
  447. state = kYTPlayerStateCued;
  448. }
  449. return state;
  450. }
  451. #pragma mark - WKNavigationDelegate
  452. - (void)webView:(WKWebView *)webView
  453. decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction
  454. decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
  455. NSURLRequest *request = navigationAction.request;
  456. if ([request.URL.scheme isEqual:@"ytplayer"]) {
  457. [self notifyDelegateOfYouTubeCallbackUrl:request.URL];
  458. decisionHandler(WKNavigationActionPolicyCancel);
  459. return;
  460. } else if ([request.URL.scheme isEqual: @"http"] || [request.URL.scheme isEqual:@"https"]) {
  461. if ([self handleHttpNavigationToUrl:request.URL]) {
  462. decisionHandler(WKNavigationActionPolicyAllow);
  463. } else {
  464. decisionHandler(WKNavigationActionPolicyCancel);
  465. }
  466. return;
  467. }
  468. decisionHandler(WKNavigationActionPolicyAllow);
  469. }
  470. - (void)webView:(WKWebView *)webView didFailNavigation:(WKNavigation *)navigation
  471. withError:(NSError *)error {
  472. if (self.initialLoadingView) {
  473. [self.initialLoadingView removeFromSuperview];
  474. }
  475. }
  476. #pragma mark - WKUIDelegate
  477. - (WKWebView *)webView:(WKWebView *)webView
  478. createWebViewWithConfiguration:(WKWebViewConfiguration *)configuration
  479. forNavigationAction:(WKNavigationAction *)navigationAction
  480. windowFeatures:(WKWindowFeatures *)windowFeatures {
  481. // Handle navigation actions initiated by Javascript.
  482. [[UIApplication sharedApplication] openURL:navigationAction.request.URL
  483. options:@{}
  484. completionHandler:nil];
  485. // Returning nil results in canceling the navigation, which has already been handled above.
  486. return nil;
  487. }
  488. #pragma mark - Private methods
  489. - (NSURL *)originURL {
  490. if (!_originURL) {
  491. NSString *bundleId = [[NSBundle mainBundle] bundleIdentifier];
  492. NSString *stringURL = [[NSString stringWithFormat:@"http://%@", bundleId] lowercaseString];
  493. _originURL = [NSURL URLWithString:stringURL];
  494. }
  495. return _originURL;
  496. }
  497. /**
  498. * Private method to handle "navigation" to a callback URL of the format
  499. * ytplayer://action?data=someData
  500. * This is how the webview communicates with the containing Objective-C code.
  501. * Side effects of this method are that it calls methods on this class's delegate.
  502. *
  503. * @param url A URL of the format ytplayer://action?data=value.
  504. */
  505. - (void)notifyDelegateOfYouTubeCallbackUrl:(NSURL *) url {
  506. NSString *action = url.host;
  507. // We know the query can only be of the format ytplayer://action?data=SOMEVALUE,
  508. // so we parse out the value.
  509. NSString *query = url.query;
  510. NSString *data;
  511. if (query) {
  512. data = [query componentsSeparatedByString:@"="][1];
  513. }
  514. if ([action isEqual:kYTPlayerCallbackOnReady]) {
  515. if (self.initialLoadingView) {
  516. [self.initialLoadingView removeFromSuperview];
  517. }
  518. if ([self.delegate respondsToSelector:@selector(playerViewDidBecomeReady:)]) {
  519. [self.delegate playerViewDidBecomeReady:self];
  520. }
  521. } else if ([action isEqual:kYTPlayerCallbackOnStateChange]) {
  522. if ([self.delegate respondsToSelector:@selector(playerView:didChangeToState:)]) {
  523. YTPlayerState state = [YTPlayerView playerStateForString:data];
  524. [self.delegate playerView:self didChangeToState:state];
  525. }
  526. } else if ([action isEqual:kYTPlayerCallbackOnPlaybackQualityChange]) {
  527. if ([self.delegate respondsToSelector:@selector(playerView:didChangeToQuality:)]) {
  528. YTPlaybackQuality quality = [YTPlayerView playbackQualityForString:data];
  529. [self.delegate playerView:self didChangeToQuality:quality];
  530. }
  531. } else if ([action isEqual:kYTPlayerCallbackOnError]) {
  532. if ([self.delegate respondsToSelector:@selector(playerView:receivedError:)]) {
  533. YTPlayerError error = kYTPlayerErrorUnknown;
  534. if ([data isEqual:kYTPlayerErrorInvalidParamErrorCode]) {
  535. error = kYTPlayerErrorInvalidParam;
  536. } else if ([data isEqual:kYTPlayerErrorHTML5ErrorCode]) {
  537. error = kYTPlayerErrorHTML5Error;
  538. } else if ([data isEqual:kYTPlayerErrorNotEmbeddableErrorCode] ||
  539. [data isEqual:kYTPlayerErrorSameAsNotEmbeddableErrorCode]) {
  540. error = kYTPlayerErrorNotEmbeddable;
  541. } else if ([data isEqual:kYTPlayerErrorVideoNotFoundErrorCode] ||
  542. [data isEqual:kYTPlayerErrorCannotFindVideoErrorCode]) {
  543. error = kYTPlayerErrorVideoNotFound;
  544. }
  545. [self.delegate playerView:self receivedError:error];
  546. }
  547. } else if ([action isEqualToString:kYTPlayerCallbackOnPlayTime]) {
  548. if ([self.delegate respondsToSelector:@selector(playerView:didPlayTime:)]) {
  549. float time = [data floatValue];
  550. [self.delegate playerView:self didPlayTime:time];
  551. }
  552. } else if ([action isEqualToString:kYTPlayerCallbackOnYouTubeIframeAPIFailedToLoad]) {
  553. if (self.initialLoadingView) {
  554. [self.initialLoadingView removeFromSuperview];
  555. }
  556. }
  557. }
  558. - (BOOL)handleHttpNavigationToUrl:(NSURL *)url {
  559. // When loading the webView for the first time, webView tries loading the originURL
  560. // since it is set as the webView.baseURL.
  561. // In that case we want to let it load itself in the webView instead of trying
  562. // to load it in a browser.
  563. if ([[url.host lowercaseString] isEqualToString:[self.originURL.host lowercaseString]]) {
  564. return YES;
  565. }
  566. // Usually this means the user has clicked on the YouTube logo or an error message in the
  567. // player. Most URLs should open in the browser. The only http(s) URL that should open in this
  568. // webview is the URL for the embed, which is of the format:
  569. // http(s)://www.youtube.com/embed/[VIDEO ID]?[PARAMETERS]
  570. NSError *error = NULL;
  571. NSRegularExpression *ytRegex =
  572. [NSRegularExpression regularExpressionWithPattern:kYTPlayerEmbedUrlRegexPattern
  573. options:NSRegularExpressionCaseInsensitive
  574. error:&error];
  575. NSTextCheckingResult *ytMatch =
  576. [ytRegex firstMatchInString:url.absoluteString
  577. options:0
  578. range:NSMakeRange(0, [url.absoluteString length])];
  579. NSRegularExpression *adRegex =
  580. [NSRegularExpression regularExpressionWithPattern:kYTPlayerAdUrlRegexPattern
  581. options:NSRegularExpressionCaseInsensitive
  582. error:&error];
  583. NSTextCheckingResult *adMatch =
  584. [adRegex firstMatchInString:url.absoluteString
  585. options:0
  586. range:NSMakeRange(0, [url.absoluteString length])];
  587. NSRegularExpression *syndicationRegex =
  588. [NSRegularExpression regularExpressionWithPattern:kYTPlayerSyndicationRegexPattern
  589. options:NSRegularExpressionCaseInsensitive
  590. error:&error];
  591. NSTextCheckingResult *syndicationMatch =
  592. [syndicationRegex firstMatchInString:url.absoluteString
  593. options:0
  594. range:NSMakeRange(0, [url.absoluteString length])];
  595. NSRegularExpression *oauthRegex =
  596. [NSRegularExpression regularExpressionWithPattern:kYTPlayerOAuthRegexPattern
  597. options:NSRegularExpressionCaseInsensitive
  598. error:&error];
  599. NSTextCheckingResult *oauthMatch =
  600. [oauthRegex firstMatchInString:url.absoluteString
  601. options:0
  602. range:NSMakeRange(0, [url.absoluteString length])];
  603. NSRegularExpression *staticProxyRegex =
  604. [NSRegularExpression regularExpressionWithPattern:kYTPlayerStaticProxyRegexPattern
  605. options:NSRegularExpressionCaseInsensitive
  606. error:&error];
  607. NSTextCheckingResult *staticProxyMatch =
  608. [staticProxyRegex firstMatchInString:url.absoluteString
  609. options:0
  610. range:NSMakeRange(0, [url.absoluteString length])];
  611. if (ytMatch || adMatch || oauthMatch || staticProxyMatch || syndicationMatch) {
  612. return YES;
  613. } else {
  614. [[UIApplication sharedApplication] openURL:url
  615. options:@{UIApplicationOpenURLOptionUniversalLinksOnly: @NO}
  616. completionHandler:nil];
  617. return NO;
  618. }
  619. }
  620. /**
  621. * Private helper method to load an iframe player with the given player parameters.
  622. *
  623. * @param additionalPlayerParams An NSDictionary of parameters in addition to required parameters
  624. * to instantiate the HTML5 player with. This differs depending on
  625. * whether a single video or playlist is being loaded.
  626. * @return YES if successful, NO if not.
  627. */
  628. - (BOOL)loadWithPlayerParams:(NSDictionary *)additionalPlayerParams {
  629. NSDictionary *playerCallbacks = @{
  630. @"onReady" : @"onReady",
  631. @"onStateChange" : @"onStateChange",
  632. @"onPlaybackQualityChange" : @"onPlaybackQualityChange",
  633. @"onError" : @"onPlayerError"
  634. };
  635. NSMutableDictionary *playerParams = [[NSMutableDictionary alloc] init];
  636. if (additionalPlayerParams) {
  637. [playerParams addEntriesFromDictionary:additionalPlayerParams];
  638. }
  639. if (![playerParams objectForKey:@"height"]) {
  640. [playerParams setValue:@"100%" forKey:@"height"];
  641. }
  642. if (![playerParams objectForKey:@"width"]) {
  643. [playerParams setValue:@"100%" forKey:@"width"];
  644. }
  645. [playerParams setValue:playerCallbacks forKey:@"events"];
  646. NSMutableDictionary *playerVars = [[playerParams objectForKey:@"playerVars"] mutableCopy];
  647. if (!playerVars) {
  648. // playerVars must not be empty so we can render a '{}' in the output JSON
  649. playerVars = [NSMutableDictionary dictionary];
  650. }
  651. // We always want to ovewrite the origin to self.originURL, not just for
  652. // the webView.baseURL
  653. [playerVars setObject:self.originURL.absoluteString forKey:@"origin"];
  654. [playerParams setValue:playerVars forKey:@"playerVars"];
  655. // Remove the existing webview to reset any state
  656. [self.webView removeFromSuperview];
  657. _webView = [self createNewWebView];
  658. [self addSubview:self.webView];
  659. NSError *error = nil;
  660. NSString *path = [[NSBundle bundleForClass:[YTPlayerView class]] pathForResource:@"YTPlayerView-iframe-player"
  661. ofType:@"html"
  662. inDirectory:@"Assets"];
  663. // in case of using Swift and embedded frameworks, resources included not in main bundle,
  664. // but in framework bundle
  665. if (!path) {
  666. path = [[[self class] frameworkBundle] pathForResource:@"YTPlayerView-iframe-player"
  667. ofType:@"html"
  668. inDirectory:@"Assets"];
  669. }
  670. NSString *embedHTMLTemplate =
  671. [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:&error];
  672. if (error) {
  673. NSLog(@"Received error rendering template: %@", error);
  674. return NO;
  675. }
  676. // Render the playerVars as a JSON dictionary.
  677. NSError *jsonRenderingError = nil;
  678. NSData *jsonData = [NSJSONSerialization dataWithJSONObject:playerParams
  679. options:NSJSONWritingPrettyPrinted
  680. error:&jsonRenderingError];
  681. if (jsonRenderingError) {
  682. NSLog(@"Attempted configuration of player with invalid playerVars: %@ \tError: %@",
  683. playerParams,
  684. jsonRenderingError);
  685. return NO;
  686. }
  687. NSString *playerVarsJsonString =
  688. [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
  689. NSString *embedHTML = [NSString stringWithFormat:embedHTMLTemplate, playerVarsJsonString];
  690. [self.webView loadHTMLString:embedHTML baseURL: self.originURL];
  691. self.webView.navigationDelegate = self;
  692. self.webView.UIDelegate = self;
  693. if ([self.delegate respondsToSelector:@selector(playerViewPreferredInitialLoadingView:)]) {
  694. UIView *initialLoadingView = [self.delegate playerViewPreferredInitialLoadingView:self];
  695. if (initialLoadingView) {
  696. initialLoadingView.frame = self.bounds;
  697. initialLoadingView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
  698. [self addSubview:initialLoadingView];
  699. self.initialLoadingView = initialLoadingView;
  700. }
  701. }
  702. return YES;
  703. }
  704. /**
  705. * Private method for cueing both cases of playlist ID and array of video IDs. Cueing
  706. * a playlist does not start playback.
  707. *
  708. * @param cueingString A JavaScript string representing an array, playlist ID or list of
  709. * video IDs to play with the playlist player.
  710. * @param index 0-index position of video to start playback on.
  711. * @param startSeconds Seconds after start of video to begin playback.
  712. */
  713. - (void)cuePlaylist:(NSString *)cueingString
  714. index:(int)index
  715. startSeconds:(float)startSeconds {
  716. NSNumber *indexValue = [NSNumber numberWithInt:index];
  717. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  718. NSString *command = [NSString stringWithFormat:@"player.cuePlaylist(%@, %@, %@);",
  719. cueingString, indexValue, startSecondsValue];
  720. [self evaluateJavaScript:command];
  721. }
  722. /**
  723. * Private method for loading both cases of playlist ID and array of video IDs. Loading
  724. * a playlist automatically starts playback.
  725. *
  726. * @param cueingString A JavaScript string representing an array, playlist ID or list of
  727. * video IDs to play with the playlist player.
  728. * @param index 0-index position of video to start playback on.
  729. * @param startSeconds Seconds after start of video to begin playback.
  730. */
  731. - (void)loadPlaylist:(NSString *)cueingString
  732. index:(int)index
  733. startSeconds:(float)startSeconds {
  734. NSNumber *indexValue = [NSNumber numberWithInt:index];
  735. NSNumber *startSecondsValue = [NSNumber numberWithFloat:startSeconds];
  736. NSString *command = [NSString stringWithFormat:@"player.loadPlaylist(%@, %@, %@);",
  737. cueingString, indexValue, startSecondsValue];
  738. [self evaluateJavaScript:command];
  739. }
  740. /**
  741. * Private helper method for converting an NSArray of video IDs into its JavaScript equivalent.
  742. *
  743. * @param videoIds An array of video ID strings to convert into JavaScript format.
  744. * @return A JavaScript array in String format containing video IDs.
  745. */
  746. - (NSString *)stringFromVideoIdArray:(NSArray *)videoIds {
  747. NSMutableArray *formattedVideoIds = [[NSMutableArray alloc] init];
  748. for (id unformattedId in videoIds) {
  749. [formattedVideoIds addObject:[NSString stringWithFormat:@"'%@'", unformattedId]];
  750. }
  751. return [NSString stringWithFormat:@"[%@]", [formattedVideoIds componentsJoinedByString:@", "]];
  752. }
  753. /**
  754. * Private method for evaluating JavaScript in the webview.
  755. *
  756. * @param jsToExecute The JavaScript code in string format that we want to execute.
  757. */
  758. - (void)evaluateJavaScript:(NSString *)jsToExecute {
  759. [self evaluateJavaScript:jsToExecute completionHandler:nil];
  760. }
  761. /**
  762. * Private method for evaluating JavaScript in the webview.
  763. *
  764. * @param jsToExecute The JavaScript code in string format that we want to execute.
  765. * @param completionHandler A block to invoke when script evaluation completes or fails.
  766. */
  767. - (void)evaluateJavaScript:(NSString *)jsToExecute
  768. completionHandler:(void(^)(id _Nullable result, NSError *_Nullable error))completionHandler {
  769. [_webView evaluateJavaScript:jsToExecute
  770. completionHandler:^(id _Nullable result, NSError *_Nullable error) {
  771. if (!completionHandler) {
  772. return;
  773. }
  774. if (error) {
  775. completionHandler(nil, error);
  776. return;
  777. }
  778. if (!result || [result isKindOfClass:[NSNull class]]) {
  779. // we can consider this an empty result
  780. completionHandler(nil, nil);
  781. return;
  782. }
  783. completionHandler(result, nil);
  784. }];
  785. }
  786. /**
  787. * Private method to convert a Objective-C BOOL value to JS boolean value.
  788. *
  789. * @param boolValue Objective-C BOOL value.
  790. * @return JavaScript Boolean value, i.e. "true" or "false".
  791. */
  792. - (NSString *)stringForJSBoolean:(BOOL)boolValue {
  793. return boolValue ? @"true" : @"false";
  794. }
  795. #pragma mark - Exposed for Testing
  796. - (void)setWebView:(WKWebView *)webView {
  797. _webView = webView;
  798. }
  799. - (WKWebView *)createNewWebView {
  800. WKWebViewConfiguration *webViewConfiguration = [[WKWebViewConfiguration alloc] init];
  801. webViewConfiguration.allowsInlineMediaPlayback = YES;
  802. webViewConfiguration.mediaTypesRequiringUserActionForPlayback = WKAudiovisualMediaTypeNone;
  803. WKWebView *webView = [[WKWebView alloc] initWithFrame:self.bounds
  804. configuration:webViewConfiguration];
  805. webView.autoresizingMask = (UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight);
  806. webView.scrollView.scrollEnabled = NO;
  807. webView.scrollView.bounces = NO;
  808. if ([self.delegate respondsToSelector:@selector(playerViewPreferredWebViewBackgroundColor:)]) {
  809. webView.backgroundColor = [self.delegate playerViewPreferredWebViewBackgroundColor:self];
  810. if (webView.backgroundColor == [UIColor clearColor]) {
  811. webView.opaque = NO;
  812. }
  813. }
  814. return webView;
  815. }
  816. - (void)removeWebView {
  817. [self.webView removeFromSuperview];
  818. self.webView = nil;
  819. }
  820. + (NSBundle *)frameworkBundle {
  821. static NSBundle* frameworkBundle = nil;
  822. static dispatch_once_t predicate;
  823. dispatch_once(&predicate, ^{
  824. NSString* mainBundlePath = [[NSBundle bundleForClass:[self class]] resourcePath];
  825. NSString* frameworkBundlePath = [mainBundlePath stringByAppendingPathComponent:@"Assets.bundle"];
  826. frameworkBundle = [NSBundle bundleWithPath:frameworkBundlePath];
  827. });
  828. return frameworkBundle;
  829. }
  830. @end