Tích hợp tính năng Truyền vào ứng dụng iOS

Hướng dẫn dành cho nhà phát triển này mô tả cách thêm tính năng hỗ trợ Google Cast vào iOS của bạn bằng cách sử dụng iOS Sender SDK.

Thiết bị di động hoặc máy tính xách tay là người gửi điều khiển quá trình phát và Thiết bị Google Cast là bộ thu hiển thị nội dung trên TV.

Khung trình gửi đề cập đến tệp nhị phân của thư viện lớp Cast và tệp liên kết có trong thời gian chạy trên trình gửi. Ứng dụng người gửi hoặc Ứng dụng truyền đề cập đến một ứng dụng cũng chạy trên trình gửi. Ứng dụng Trình nhận web đề cập đến ứng dụng HTML chạy trên Web receiver.

Khung người gửi sử dụng thiết kế lệnh gọi lại không đồng bộ để thông báo cho người gửi ứng dụng của các sự kiện và để chuyển đổi giữa các trạng thái khác nhau của thời gian hoạt động của ứng dụng Truyền chu kỳ.

Luồng ứng dụng

Các bước sau đây mô tả quy trình thực thi cấp cao điển hình cho một người gửi Ứng dụng iOS:

  • Khung Truyền bắt đầu GCKDiscoveryManager dựa trên các cơ sở lưu trú được cung cấp tại GCKCastOptions đến bắt đầu quét tìm thiết bị.
  • Khi người dùng nhấp vào nút Truyền, khung sẽ trình bày giao diện Truyền hộp thoại với danh sách các Thiết bị truyền được phát hiện.
  • Khi người dùng chọn một Thiết bị truyền, khung sẽ cố chạy Ứng dụng Trình nhận web trên Thiết bị truyền.
  • Khung này sẽ gọi các lệnh gọi lại trong ứng dụng gửi để xác nhận rằng Phát hành ứng dụng Trình nhận web.
  • Khung này tạo ra một kênh liên lạc giữa người gửi và Các ứng dụng Web nhận.
  • Khung này sử dụng kênh liên lạc để tải và điều khiển nội dung nghe nhìn phát trên Web receiver.
  • Khung này sẽ đồng bộ hoá trạng thái phát nội dung nghe nhìn giữa người gửi và Web receiver: khi người dùng thực hiện thao tác trên giao diện người dùng của người gửi, khung này sẽ chuyển các yêu cầu điều khiển nội dung nghe nhìn đó đến Web receiver cũng như khi Web receiver gửi thông tin cập nhật trạng thái nội dung nghe nhìn, khung sẽ cập nhật trạng thái của giao diện người dùng gửi.
  • Khi người dùng nhấp vào nút Truyền để ngắt kết nối khỏi Thiết bị truyền, khung này sẽ ngắt kết nối ứng dụng của người gửi khỏi Web receiver.

Để khắc phục sự cố người gửi, bạn cần bật tính năng ghi nhật ký.

Để xem danh sách đầy đủ tất cả các lớp, phương thức và sự kiện trong Google Cast khung iOS, hãy xem API Google Cast iOS Tệp đối chiếu. Các phần sau đây trình bày các bước để tích hợp tính năng Truyền vào ứng dụng iOS.

Phương thức gọi từ luồng chính

Khởi chạy ngữ cảnh Truyền

Khung Cast có một đối tượng singleton toàn cầu, GCKCastContext là điều phối tất cả hoạt động của khung này. Bạn phải khởi động đối tượng này ở giai đoạn đầu trong vòng đời của ứng dụng, thường là trong -[application:didFinishLaunchingWithOptions:] của ứng dụng uỷ quyền, vì vậy việc tự động tiếp tục phiên khi khởi động lại ứng dụng của người gửi có thể kích hoạt đúng cách.

GCKCastOptions bạn phải cung cấp đối tượng khi khởi tạo GCKCastContext. Lớp này chứa các tuỳ chọn ảnh hưởng đến hành vi của khung. Nhiều nhất quan trọng trong số này là ID ứng dụng Trình nhận web, được dùng để lọc kết quả khám phá và khởi chạy ứng dụng Web receiver khi một phiên Truyền đầu.

Phương thức -[application:didFinishLaunchingWithOptions:] cũng là một phương thức hay để thiết lập uỷ quyền ghi nhật ký để nhận thông báo ghi nhật ký từ khung. Những thông tin này có thể hữu ích cho việc gỡ lỗi và khắc phục sự cố.

Swift
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate, GCKLoggerDelegate {
  let kReceiverAppID = kGCKDefaultMediaReceiverApplicationID
  let kDebugLoggingEnabled = true

  var window: UIWindow?

  func applicationDidFinishLaunching(_ application: UIApplication) {
    let criteria = GCKDiscoveryCriteria(applicationID: kReceiverAppID)
    let options = GCKCastOptions(discoveryCriteria: criteria)
    GCKCastContext.setSharedInstanceWith(options)

    // Enable logger.
    GCKLogger.sharedInstance().delegate = self

    ...
  }

  // MARK: - GCKLoggerDelegate

  func logMessage(_ message: String,
                  at level: GCKLoggerLevel,
                  fromFunction function: String,
                  location: String) {
    if (kDebugLoggingEnabled) {
      print(function + " - " + message)
    }
  }
}
Mục tiêu-C

AppDelegate.h

@interface AppDelegate () <GCKLoggerDelegate>
@end

AppDelegate.m

@implementation AppDelegate

static NSString *const kReceiverAppID = @"AABBCCDD";
static const BOOL kDebugLoggingEnabled = YES;

- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  GCKDiscoveryCriteria *criteria = [[GCKDiscoveryCriteria alloc]
                                    initWithApplicationID:kReceiverAppID];
  GCKCastOptions *options = [[GCKCastOptions alloc] initWithDiscoveryCriteria:criteria];
  [GCKCastContext setSharedInstanceWithOptions:options];

  // Enable logger.
  [GCKLogger sharedInstance].delegate = self;

  ...

  return YES;
}

...

#pragma mark - GCKLoggerDelegate

- (void)logMessage:(NSString *)message
           atLevel:(GCKLoggerLevel)level
      fromFunction:(NSString *)function
          location:(NSString *)location {
  if (kDebugLoggingEnabled) {
    NSLog(@"%@ - %@, %@", function, message, location);
  }
}

@end

Tiện ích Trải nghiệm người dùng của Cast

SDK Truyền dành cho iOS cung cấp các tiện ích này tuân thủ Thiết kế truyền Danh sách kiểm tra:

  • Lớp phủ giới thiệu: Lớp GCKCastContext có một phương thức presentCastInstructionsViewControllerOnceWithCastButton! có thể dùng để làm nổi bật nút Truyền vào lần đầu tiên Web nhận có sẵn. Ứng dụng dành cho người gửi có thể tuỳ chỉnh văn bản, vị trí của tiêu đề và nút Đóng.

  • Nút truyền: Kể từ SDK gửi cho iOS phiên bản 4.6.0, nút truyền sẽ luôn hiển thị khi thiết bị của người gửi được kết nối với Wi-Fi. Lần đầu tiên người dùng nhấn trên nút Truyền sau khi khởi động ứng dụng ban đầu, một hộp thoại quyền sẽ xuất hiện để người dùng có thể cấp cho ứng dụng quyền truy cập mạng cục bộ vào thiết bị trên mạng. Sau đó, khi người dùng nhấn vào nút truyền, màn hình sẽ truyền hộp thoại được hiển thị liệt kê các thiết bị được phát hiện. Khi người dùng nhấn trên nút truyền trong khi thiết bị được kết nối, nút này hiển thị siêu dữ liệu đa phương tiện (chẳng hạn như tiêu đề, tên phòng thu và hình thu nhỏ hình ảnh) hoặc cho phép người dùng ngắt kết nối khỏi thiết bị truyền. Khi người dùng nhấn vào nút truyền khi không có thiết bị, một màn hình sẽ hiển thị và cung cấp cho người dùng thông tin về lý do không tìm thấy thiết bị và cách khắc phục sự cố.

  • Tay điều khiển thu nhỏ: Khi người dùng đang truyền nội dung và đã rời khỏi trang hiện tại trang nội dung hoặc bộ điều khiển mở rộng sang một màn hình khác trong ứng dụng dành cho người gửi, bộ điều khiển mini hiển thị ở cuối màn hình để cho phép người dùng xem siêu dữ liệu nội dung nghe nhìn đang truyền và điều khiển việc phát.

  • Bộ điều khiển mở rộng: Khi người dùng đang truyền nội dung, nếu họ nhấp vào thông báo về nội dung nghe nhìn hoặc thu nhỏ, tay điều khiển mở rộng khởi chạy, hiển thị hiện đang phát siêu dữ liệu phương tiện và cung cấp một số nút để điều khiển phát nội dung nghe nhìn.

Thêm nút Truyền

Khung này cung cấp một thành phần nút Truyền dưới dạng một lớp con UIButton. Chiến dịch này có thể được thêm vào thanh tiêu đề của ứng dụng bằng cách gói ứng dụng đó trong một UIBarButtonItem. Một giá trị điển hình Lớp con UIViewController có thể cài đặt nút Truyền như sau:

Swift
let castButton = GCKUICastButton(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
castButton.tintColor = UIColor.gray
navigationItem.rightBarButtonItem = UIBarButtonItem(customView: castButton)
Mục tiêu-C
GCKUICastButton *castButton = [[GCKUICastButton alloc] initWithFrame:CGRectMake(0, 0, 24, 24)];
castButton.tintColor = [UIColor grayColor];
self.navigationItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:castButton];

Theo mặc định, nhấn vào nút này sẽ mở hộp thoại Truyền được cung cấp bởi khung.

GCKUICastButton cũng có thể được thêm trực tiếp vào bảng phân cảnh.

Định cấu hình chế độ khám phá thiết bị

Trong khung này, quá trình khám phá thiết bị sẽ diễn ra tự động. Không cần bắt đầu hoặc dừng quá trình khám phá một cách rõ ràng trừ phi bạn triển khai giao diện người dùng tuỳ chỉnh.

Chế độ Khám phá trong khung do lớp quản lý GCKDiscoveryManager! Đây là một thuộc tính của GCKCastContext. Chiến lược phát hành đĩa đơn khung cung cấp thành phần hộp thoại Truyền mặc định để lựa chọn thiết bị và kiểm soát. Danh sách thiết bị được sắp xếp từ vựng theo tên thân thiện với thiết bị.

Cách hoạt động của tính năng quản lý phiên

SDK truyền giới thiệu khái niệm về phiên Truyền, việc thiết lập kết hợp các bước kết nối với một thiết bị, khởi chạy (hoặc tham gia) một Web ứng dụng receiver, kết nối với ứng dụng đó và khởi tạo một kênh điều khiển nội dung nghe nhìn. Xem Trình nhận web Hướng dẫn về vòng đời của ứng dụng để biết thêm thông tin về các phiên Truyền và vòng đời của Trình thu phát web.

Các phiên hoạt động do lớp học quản lý GCKSessionManager! Đây là một thuộc tính của GCKCastContext. Các phiên riêng lẻ được biểu thị bằng các lớp con của lớp GCKSession: chẳng hạn như GCKCastSession biểu thị các phiên có Thiết bị truyền. Bạn có thể truy cập vào Cast đang hoạt động phiên (nếu có), dưới dạng thuộc tính currentCastSession của GCKSessionManager.

Chiến lược phát hành đĩa đơn GCKSessionManagerListener có thể dùng để theo dõi các sự kiện trong phiên, chẳng hạn như tạo phiên, tạm ngưng, tiếp tục và chấm dứt. Khung này sẽ tự động tạm ngưng các phiên hoạt động khi ứng dụng của người gửi chuyển sang chạy ở chế độ nền và cố gắng tiếp tục chúng khi ứng dụng trở về nền trước (hoặc được chạy lại sau khi chấm dứt ứng dụng bất thường/đột ngột trong khi một phiên đang hoạt động).

Nếu hộp thoại Truyền đang được sử dụng thì các phiên sẽ được tạo và thu nhỏ tự động để phản hồi cử chỉ của người dùng. Nếu không, ứng dụng có thể bắt đầu và kết thúc một cách rõ ràng thông qua các phương thức trên GCKSessionManager.

Nếu ứng dụng cần thực hiện quy trình xử lý đặc biệt để phản hồi vòng đời phiên các sự kiện này, mã này có thể đăng ký một hoặc nhiều thực thể GCKSessionManagerListener bằng GCKSessionManager. GCKSessionManagerListener là một giao thức xác định lệnh gọi lại cho các sự kiện như bắt đầu phiên, kết thúc phiên, v.v.

Chuyển sự kiện phát trực tiếp

Việc bảo toàn trạng thái phiên là cơ sở của quá trình chuyển luồng, trong đó người dùng có thể di chuyển luồng âm thanh và video hiện có giữa các thiết bị bằng lệnh thoại, Google Home Ứng dụng hoặc màn hình thông minh. Phương tiện ngừng phát trên một thiết bị (nguồn) và tiếp tục phát trên một thiết bị khác (nguồn) đích). Bất kỳ thiết bị Truyền nào có chương trình cơ sở mới nhất đều có thể đóng vai trò là nguồn hoặc đích trong một truyền trực tuyến.

Để có được thiết bị đích mới trong quá trình truyền, hãy sử dụng GCKCastSession#device trong khoảng thời gian [sessionManager:didResumeCastSession:] .

Xem Chuyển sự kiện phát trực tiếp trên Web receiver để biết thêm thông tin.

Tự động kết nối lại

Khung Cast bổ sung logic kết nối lại để tự động xử lý việc kết nối lại trong nhiều trường hợp khó thấy, chẳng hạn như:

  • Khôi phục sau khi mất Wi-Fi tạm thời
  • Khôi phục từ chế độ ngủ của thiết bị
  • Khôi phục sau khi chạy ứng dụng ở chế độ nền
  • Khôi phục nếu ứng dụng gặp sự cố

Cách hoạt động của tính năng điều khiển nội dung nghe nhìn

Nếu phiên Truyền được thiết lập bằng ứng dụng Bộ thu trên web có hỗ trợ nội dung nghe nhìn không gian tên, một bản sao của GCKRemoteMediaClient sẽ do khung này tạo tự động; có thể truy cập dưới dạng thuộc tính remoteMediaClient của GCKCastSession thực thể.

Tất cả các phương thức trên GCKRemoteMediaClient gửi yêu cầu đến Trình nhận web sẽ trả về a GCKRequest đối tượng có thể được sử dụng để theo dõi yêu cầu đó. Đáp GCKRequestDelegate có thể được chỉ định cho đối tượng này để nhận thông báo về kết quả của hoạt động.

Dự kiến thực thể của GCKRemoteMediaClient có thể được chia sẻ giữa nhiều phần của ứng dụng và thực sự là một số thành phần nội bộ của khung này, chẳng hạn như hộp thoại Truyền và các nút điều khiển nội dung nghe nhìn nhỏ sẽ chia sẻ thực thể. Để đạt được mục tiêu đó, GCKRemoteMediaClient hỗ trợ đăng ký nhiều GCKRemoteMediaClientListener.

Thiết lập siêu dữ liệu đa phương tiện

Chiến lược phát hành đĩa đơn GCKMediaMetadata biểu thị thông tin về một mục nội dung đa phương tiện bạn muốn truyền. Nội dung sau đây Ví dụ sẽ tạo một phiên bản GCKMediaMetadata mới của một bộ phim và đặt tiêu đề, phụ đề, tên phòng thu và hai hình ảnh.

Swift
let metadata = GCKMediaMetadata()
metadata.setString("Big Buck Bunny (2008)", forKey: kGCKMetadataKeyTitle)
metadata.setString("Big Buck Bunny tells the story of a giant rabbit with a heart bigger than " +
  "himself. When one sunny day three rodents rudely harass him, something " +
  "snaps... and the rabbit ain't no bunny anymore! In the typical cartoon " +
  "tradition he prepares the nasty rodents a comical revenge.",
                   forKey: kGCKMetadataKeySubtitle)
metadata.addImage(GCKImage(url: URL(string: "https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/images/BigBuckBunny.jpg")!,
                           width: 480,
                           height: 360))
Mục tiêu-C
GCKMediaMetadata *metadata = [[GCKMediaMetadata alloc]
                                initWithMetadataType:GCKMediaMetadataTypeMovie];
[metadata setString:@"Big Buck Bunny (2008)" forKey:kGCKMetadataKeyTitle];
[metadata setString:@"Big Buck Bunny tells the story of a giant rabbit with a heart bigger than "
 "himself. When one sunny day three rodents rudely harass him, something "
 "snaps... and the rabbit ain't no bunny anymore! In the typical cartoon "
 "tradition he prepares the nasty rodents a comical revenge."
             forKey:kGCKMetadataKeySubtitle];
[metadata addImage:[[GCKImage alloc]
                    initWithURL:[[NSURL alloc] initWithString:@"https://commondatastorage.googleapis.com/"
                                 "gtv-videos-bucket/sample/images/BigBuckBunny.jpg"]
                    width:480
                    height:360]];

Hãy xem phần Lựa chọn hình ảnh và Lưu vào bộ nhớ đệm về việc sử dụng hình ảnh cùng với siêu dữ liệu phương tiện.

Tải nội dung nghe nhìn

Để tải một mục nội dung nghe nhìn, hãy tạo một GCKMediaInformation bằng siêu dữ liệu của phương tiện. Sau đó, lấy giá trị hiện tại GCKCastSession và sử dụng GCKRemoteMediaClient để tải nội dung đa phương tiện trên ứng dụng nhận. Sau đó, bạn có thể sử dụng GCKRemoteMediaClient để điều khiển ứng dụng trình phát nội dung đa phương tiện chạy trên bộ thu, chẳng hạn như để phát, tạm dừng rồi dừng lại.

Swift
let url = URL.init(string: "https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4")
guard let mediaURL = url else {
  print("invalid mediaURL")
  return
}

let mediaInfoBuilder = GCKMediaInformationBuilder.init(contentURL: mediaURL)
mediaInfoBuilder.streamType = GCKMediaStreamType.none;
mediaInfoBuilder.contentType = "video/mp4"
mediaInfoBuilder.metadata = metadata;
mediaInformation = mediaInfoBuilder.build()

guard let mediaInfo = mediaInformation else {
  print("invalid mediaInformation")
  return
}

if let request = sessionManager.currentSession?.remoteMediaClient?.loadMedia(mediaInfo) {
  request.delegate = self
}
Mục tiêu-C
GCKMediaInformationBuilder *mediaInfoBuilder =
  [[GCKMediaInformationBuilder alloc] initWithContentURL:
   [NSURL URLWithString:@"https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4"]];
mediaInfoBuilder.streamType = GCKMediaStreamTypeNone;
mediaInfoBuilder.contentType = @"video/mp4";
mediaInfoBuilder.metadata = metadata;
self.mediaInformation = [mediaInfoBuilder build];

GCKRequest *request = [self.sessionManager.currentSession.remoteMediaClient loadMedia:self.mediaInformation];
if (request != nil) {
  request.delegate = self;
}

Ngoài ra, hãy xem phần này trên bằng các bản nhạc nội dung nghe nhìn.

Định dạng video 4K

Để xác định định dạng video cho nội dung nghe nhìn của bạn, hãy sử dụng thuộc tính videoInfo của GCKMediaStatus để lấy thực thể hiện tại của GCKVideoInfo. Phiên bản này chứa loại định dạng TV HDR cùng với chiều cao và chiều rộng trong điểm ảnh. Biến thể của định dạng 4K được biểu thị trong thuộc tính hdrType bằng enum giá trị GCKVideoInfoHDRType.

Thêm tay điều khiển mini

Theo Cast Design Danh sách kiểm tra, ứng dụng của người gửi sẽ cung cấp chế độ kiểm soát liên tục được gọi là cơ chế tay điều khiển sẽ xuất hiện khi người dùng điều hướng khỏi trang nội dung hiện tại. Bộ điều khiển mini cho phép truy cập tức thì và hiển thị lời nhắc cho người dùng phiên Truyền hiện tại.

Khung Truyền cung cấp một thanh điều khiển, GCKUIMiniMediaControlsViewController! Bạn có thể thêm nút này vào cảnh bạn muốn trình điều khiển thu nhỏ xuất hiện.

Khi ứng dụng của người gửi đang phát video hoặc âm thanh trực tiếp, SDK tự động hiển thị nút phát/dừng ở vị trí cho nút phát/tạm dừng trên trình điều khiển mini.

Xem Tuỳ chỉnh giao diện người gửi trên iOS để biết cách ứng dụng gửi có thể định cấu hình giao diện của tiện ích Truyền.

Có hai cách để thêm bộ điều khiển mini vào ứng dụng của người gửi:

  • Cho phép khung Truyền quản lý bố cục của trình điều khiển nhỏ bằng cách gói bộ điều khiển chế độ xem hiện tại của bạn với bộ điều khiển chế độ xem riêng.
  • Tự quản lý bố cục của tiện ích tay điều khiển nhỏ bằng cách thêm tiện ích đó vào hiện có bộ điều khiển khung hiển thị bằng cách cung cấp một khung hiển thị phụ trong bảng phân cảnh.

Gói bằng GCKUICastContainerViewController

Cách đầu tiên là sử dụng GCKUICastContainerViewController bao bọc một bộ điều khiển chế độ xem khác và thêm GCKUIMiniMediaControlsViewController ở dưới cùng. Phương pháp này bị hạn chế ở chỗ bạn không thể tuỳ chỉnh ảnh động và không thể định cấu hình hành vi của bộ điều khiển chế độ xem vùng chứa.

Cách này đầu tiên thường được thực hiện trong Phương thức -[application:didFinishLaunchingWithOptions:] của ứng dụng uỷ quyền:

Swift
func applicationDidFinishLaunching(_ application: UIApplication) {
  ...

  // Wrap main view in the GCKUICastContainerViewController and display the mini controller.
  let appStoryboard = UIStoryboard(name: "Main", bundle: nil)
  let navigationController = appStoryboard.instantiateViewController(withIdentifier: "MainNavigation")
  let castContainerVC =
          GCKCastContext.sharedInstance().createCastContainerController(for: navigationController)
  castContainerVC.miniMediaControlsItemEnabled = true
  window = UIWindow(frame: UIScreen.main.bounds)
  window!.rootViewController = castContainerVC
  window!.makeKeyAndVisible()

  ...
}
Mục tiêu-C
- (BOOL)application:(UIApplication *)application
        didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  ...

  // Wrap main view in the GCKUICastContainerViewController and display the mini controller.
  UIStoryboard *appStoryboard = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
  UINavigationController *navigationController =
          [appStoryboard instantiateViewControllerWithIdentifier:@"MainNavigation"];
  GCKUICastContainerViewController *castContainerVC =
          [[GCKCastContext sharedInstance] createCastContainerControllerForViewController:navigationController];
  castContainerVC.miniMediaControlsItemEnabled = YES;
  self.window = [[UIWindow alloc] initWithFrame:UIScreen.mainScreen.bounds];
  self.window.rootViewController = castContainerVC;
  [self.window makeKeyAndVisible];
  ...

}
Swift
var castControlBarsEnabled: Bool {
  set(enabled) {
    if let castContainerVC = self.window?.rootViewController as? GCKUICastContainerViewController {
      castContainerVC.miniMediaControlsItemEnabled = enabled
    } else {
      print("GCKUICastContainerViewController is not correctly configured")
    }
  }
  get {
    if let castContainerVC = self.window?.rootViewController as? GCKUICastContainerViewController {
      return castContainerVC.miniMediaControlsItemEnabled
    } else {
      print("GCKUICastContainerViewController is not correctly configured")
      return false
    }
  }
}
Mục tiêu-C

AppDelegate.h

@interface AppDelegate : UIResponder <UIApplicationDelegate>

@property (nonatomic, strong) UIWindow *window;
@property (nonatomic, assign) BOOL castControlBarsEnabled;

@end

AppDelegate.m

@implementation AppDelegate

...

- (void)setCastControlBarsEnabled:(BOOL)notificationsEnabled {
  GCKUICastContainerViewController *castContainerVC;
  castContainerVC =
      (GCKUICastContainerViewController *)self.window.rootViewController;
  castContainerVC.miniMediaControlsItemEnabled = notificationsEnabled;
}

- (BOOL)castControlBarsEnabled {
  GCKUICastContainerViewController *castContainerVC;
  castContainerVC =
      (GCKUICastContainerViewController *)self.window.rootViewController;
  return castContainerVC.miniMediaControlsItemEnabled;
}

...

@end

Nhúng vào bộ điều khiển chế độ xem hiện có

Cách thứ hai là thêm trực tiếp bộ điều khiển thu nhỏ vào chế độ xem hiện có bằng cách sử dụng createMiniMediaControlsViewController để tạo một GCKUIMiniMediaControlsViewController rồi thêm thực thể đó vào trình kiểm soát chế độ xem vùng chứa dưới dạng chế độ xem phụ.

Thiết lập bộ điều khiển chế độ xem trong tính năng uỷ quyền ứng dụng:

Swift
func application(_ application: UIApplication,
                 didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
  ...

  GCKCastContext.sharedInstance().useDefaultExpandedMediaControls = true
  window?.clipsToBounds = true

  let rootContainerVC = (window?.rootViewController as? RootContainerViewController)
  rootContainerVC?.miniMediaControlsViewEnabled = true

  ...

  return true
}
Mục tiêu-C
- (BOOL)application:(UIApplication *)application
    didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  ...

  [GCKCastContext sharedInstance].useDefaultExpandedMediaControls = YES;

  self.window.clipsToBounds = YES;

  RootContainerViewController *rootContainerVC;
  rootContainerVC =
      (RootContainerViewController *)self.window.rootViewController;
  rootContainerVC.miniMediaControlsViewEnabled = YES;

  ...

  return YES;
}

Trong bộ điều khiển chế độ xem gốc, hãy tạo một GCKUIMiniMediaControlsViewController và thêm thực thể đó vào trình kiểm soát chế độ xem vùng chứa dưới dạng chế độ xem phụ:

Swift
let kCastControlBarsAnimationDuration: TimeInterval = 0.20

@objc(RootContainerViewController)
class RootContainerViewController: UIViewController, GCKUIMiniMediaControlsViewControllerDelegate {
  @IBOutlet weak private var _miniMediaControlsContainerView: UIView!
  @IBOutlet weak private var _miniMediaControlsHeightConstraint: NSLayoutConstraint!
  private var miniMediaControlsViewController: GCKUIMiniMediaControlsViewController!
  var miniMediaControlsViewEnabled = false {
    didSet {
      if self.isViewLoaded {
        self.updateControlBarsVisibility()
      }
    }
  }

  var overriddenNavigationController: UINavigationController?

  override var navigationController: UINavigationController? {

    get {
      return overriddenNavigationController
    }

    set {
      overriddenNavigationController = newValue
    }
  }
  var miniMediaControlsItemEnabled = false

  override func viewDidLoad() {
    super.viewDidLoad()
    let castContext = GCKCastContext.sharedInstance()
    self.miniMediaControlsViewController = castContext.createMiniMediaControlsViewController()
    self.miniMediaControlsViewController.delegate = self
    self.updateControlBarsVisibility()
    self.installViewController(self.miniMediaControlsViewController,
                               inContainerView: self._miniMediaControlsContainerView)
  }

  func updateControlBarsVisibility() {
    if self.miniMediaControlsViewEnabled && self.miniMediaControlsViewController.active {
      self._miniMediaControlsHeightConstraint.constant = self.miniMediaControlsViewController.minHeight
      self.view.bringSubview(toFront: self._miniMediaControlsContainerView)
    } else {
      self._miniMediaControlsHeightConstraint.constant = 0
    }
    UIView.animate(withDuration: kCastControlBarsAnimationDuration, animations: {() -> Void in
      self.view.layoutIfNeeded()
    })
    self.view.setNeedsLayout()
  }

  func installViewController(_ viewController: UIViewController?, inContainerView containerView: UIView) {
    if let viewController = viewController {
      self.addChildViewController(viewController)
      viewController.view.frame = containerView.bounds
      containerView.addSubview(viewController.view)
      viewController.didMove(toParentViewController: self)
    }
  }

  func uninstallViewController(_ viewController: UIViewController) {
    viewController.willMove(toParentViewController: nil)
    viewController.view.removeFromSuperview()
    viewController.removeFromParentViewController()
  }

  override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "NavigationVCEmbedSegue" {
      self.navigationController = (segue.destination as? UINavigationController)
    }
  }

...
Mục tiêu-C

RootContainerViewController.h

static const NSTimeInterval kCastControlBarsAnimationDuration = 0.20;

@interface RootContainerViewController () <GCKUIMiniMediaControlsViewControllerDelegate> {
  __weak IBOutlet UIView *_miniMediaControlsContainerView;
  __weak IBOutlet NSLayoutConstraint *_miniMediaControlsHeightConstraint;
  GCKUIMiniMediaControlsViewController *_miniMediaControlsViewController;
}

@property(nonatomic, weak, readwrite) UINavigationController *navigationController;

@property(nonatomic, assign, readwrite) BOOL miniMediaControlsViewEnabled;
@property(nonatomic, assign, readwrite) BOOL miniMediaControlsItemEnabled;

@end

RootContainerViewController.m

@implementation RootContainerViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  GCKCastContext *castContext = [GCKCastContext sharedInstance];
  _miniMediaControlsViewController =
      [castContext createMiniMediaControlsViewController];
  _miniMediaControlsViewController.delegate = self;

  [self updateControlBarsVisibility];
  [self installViewController:_miniMediaControlsViewController
              inContainerView:_miniMediaControlsContainerView];
}

- (void)setMiniMediaControlsViewEnabled:(BOOL)miniMediaControlsViewEnabled {
  _miniMediaControlsViewEnabled = miniMediaControlsViewEnabled;
  if (self.isViewLoaded) {
    [self updateControlBarsVisibility];
  }
}

- (void)updateControlBarsVisibility {
  if (self.miniMediaControlsViewEnabled &&
      _miniMediaControlsViewController.active) {
    _miniMediaControlsHeightConstraint.constant =
        _miniMediaControlsViewController.minHeight;
    [self.view bringSubviewToFront:_miniMediaControlsContainerView];
  } else {
    _miniMediaControlsHeightConstraint.constant = 0;
  }
  [UIView animateWithDuration:kCastControlBarsAnimationDuration
                   animations:^{
                     [self.view layoutIfNeeded];
                   }];
  [self.view setNeedsLayout];
}

- (void)installViewController:(UIViewController *)viewController
              inContainerView:(UIView *)containerView {
  if (viewController) {
    [self addChildViewController:viewController];
    viewController.view.frame = containerView.bounds;
    [containerView addSubview:viewController.view];
    [viewController didMoveToParentViewController:self];
  }
}

- (void)uninstallViewController:(UIViewController *)viewController {
  [viewController willMoveToParentViewController:nil];
  [viewController.view removeFromSuperview];
  [viewController removeFromParentViewController];
}

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
  if ([segue.identifier isEqualToString:@"NavigationVCEmbedSegue"]) {
    self.navigationController =
        (UINavigationController *)segue.destinationViewController;
  }
}

...

@end

Chiến lược phát hành đĩa đơn GCKUIMiniMediaControlsViewControllerDelegate cho trình điều khiển chế độ xem của máy chủ biết khi nào nên hiển thị tay điều khiển nhỏ:

Swift
  func miniMediaControlsViewController(_: GCKUIMiniMediaControlsViewController,
                                       shouldAppear _: Bool) {
    updateControlBarsVisibility()
  }
Mục tiêu-C
- (void)miniMediaControlsViewController:
            (GCKUIMiniMediaControlsViewController *)miniMediaControlsViewController
                           shouldAppear:(BOOL)shouldAppear {
  [self updateControlBarsVisibility];
}

Thêm bộ điều khiển mở rộng

Danh sách kiểm tra thiết kế của Google Cast yêu cầu ứng dụng gửi phải cung cấp bản mở rộng tay điều khiển cho nội dung nghe nhìn đang được truyền. Bộ điều khiển mở rộng là phiên bản toàn màn hình của trình điều khiển mini.

Bộ điều khiển mở rộng là chế độ xem toàn màn hình, cho phép toàn quyền kiểm soát phát nội dung nghe nhìn từ xa. Chế độ xem này sẽ cho phép ứng dụng truyền quản lý mọi khung hình dễ quản lý của phiên truyền, ngoại trừ âm lượng của Bộ thu trên web vòng đời điều khiển và phiên (kết nối/dừng truyền). Nền tảng này cũng cung cấp tất cả thông tin trạng thái về phiên phát nội dung đa phương tiện (hình minh hoạ, tiêu đề, phụ đề, v.v. ).

Chức năng của chế độ xem này được triển khai bởi GCKUIExpandedMediaControlsViewController .

Việc đầu tiên bạn phải làm là bật bộ điều khiển mở rộng mặc định trong truyền bối cảnh. Sửa đổi tính năng uỷ quyền ứng dụng để bật bộ điều khiển mở rộng mặc định:

Swift
func applicationDidFinishLaunching(_ application: UIApplication) {
  ..

  GCKCastContext.sharedInstance().useDefaultExpandedMediaControls = true

  ...
}
Mục tiêu-C
- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  ...

  [GCKCastContext sharedInstance].useDefaultExpandedMediaControls = YES;

  ..
}

Thêm mã sau vào bộ điều khiển chế độ xem để tải bộ điều khiển mở rộng khi người dùng bắt đầu truyền video:

Swift
func playSelectedItemRemotely() {
  GCKCastContext.sharedInstance().presentDefaultExpandedMediaControls()

  ...

  // Load your media
  sessionManager.currentSession?.remoteMediaClient?.loadMedia(mediaInformation)
}
Mục tiêu-C
- (void)playSelectedItemRemotely {
  [[GCKCastContext sharedInstance] presentDefaultExpandedMediaControls];

  ...

  // Load your media
  [self.sessionManager.currentSession.remoteMediaClient loadMedia:mediaInformation];
}

Bộ điều khiển mở rộng cũng sẽ tự động chạy khi người dùng nhấn vào trình điều khiển mini.

Khi ứng dụng của người gửi đang phát video hoặc âm thanh trực tiếp, SDK tự động hiển thị nút phát/dừng ở vị trí cho nút phát/tạm dừng trong bộ điều khiển mở rộng.

Xem Áp dụng kiểu tuỳ chỉnh cho iOS của bạn Ứng dụng để biết cách ứng dụng gửi của bạn có thể định cấu hình giao diện của tiện ích Truyền.

Điều chỉnh âm lượng

Khung Cast tự động quản lý âm lượng cho ứng dụng gửi. Chiến lược phát hành đĩa đơn khung này tự động đồng bộ hoá với âm lượng Web receiver cho các tiện ích giao diện người dùng đã cung cấp. Để đồng bộ hoá thanh trượt do ứng dụng cung cấp, hãy sử dụng GCKUIDeviceVolumeController.

Kiểm soát âm lượng bằng nút vật lý

Bạn có thể sử dụng các nút âm lượng vật lý trên thiết bị gửi để thay đổi âm lượng của phiên Truyền trên Web receiver bằng cách sử dụng Cờ physicalVolumeButtonsWillControlDeviceVolume trên GCKCastOptions thân mến! được đặt trên GCKCastContext.

Swift
let criteria = GCKDiscoveryCriteria(applicationID: kReceiverAppID)
let options = GCKCastOptions(discoveryCriteria: criteria)
options.physicalVolumeButtonsWillControlDeviceVolume = true
GCKCastContext.setSharedInstanceWith(options)
Mục tiêu-C
GCKDiscoveryCriteria *criteria = [[GCKDiscoveryCriteria alloc]
                                          initWithApplicationID:kReceiverAppID];
GCKCastOptions *options = [[GCKCastOptions alloc]
                                          initWithDiscoveryCriteria :criteria];
options.physicalVolumeButtonsWillControlDeviceVolume = YES;
[GCKCastContext setSharedInstanceWithOptions:options];

Xử lý lỗi

Điều quan trọng là ứng dụng của người gửi phải xử lý tất cả các lệnh gọi lại báo lỗi và quyết định phản hồi tốt nhất cho từng giai đoạn trong vòng đời của tính năng Truyền. Ứng dụng có thể hiển thị hộp thoại lỗi cho người dùng hoặc có thể quyết định kết thúc phiên Truyền.

Ghi nhật ký

GCKLogger là một singleton dùng để ghi nhật ký theo khung. Sử dụng GCKLoggerDelegate để tuỳ chỉnh cách bạn xử lý thông điệp nhật ký.

Khi sử dụng GCKLogger, SDK sẽ tạo ra kết quả ghi nhật ký ở dạng gỡ lỗi thông báo, lỗi và cảnh báo. Những thông điệp nhật ký này hỗ trợ việc gỡ lỗi và hữu ích để khắc phục sự cố và xác định sự cố. Theo mặc định, đầu ra nhật ký là bị chặn, nhưng bằng cách chỉ định GCKLoggerDelegate, ứng dụng của người gửi có thể nhận những thông báo này từ SDK rồi ghi vào bảng điều khiển hệ thống.

Swift
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate, GCKLoggerDelegate {
  let kReceiverAppID = kGCKDefaultMediaReceiverApplicationID
  let kDebugLoggingEnabled = true

  var window: UIWindow?

  func applicationDidFinishLaunching(_ application: UIApplication) {
    ...

    // Enable logger.
    GCKLogger.sharedInstance().delegate = self

    ...
  }

  // MARK: - GCKLoggerDelegate

  func logMessage(_ message: String,
                  at level: GCKLoggerLevel,
                  fromFunction function: String,
                  location: String) {
    if (kDebugLoggingEnabled) {
      print(function + " - " + message)
    }
  }
}
Mục tiêu-C

AppDelegate.h

@interface AppDelegate () <GCKLoggerDelegate>
@end

AppDelegate.m

@implementation AppDelegate

static NSString *const kReceiverAppID = @"AABBCCDD";
static const BOOL kDebugLoggingEnabled = YES;

- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  ...

  // Enable logger.
  [GCKLogger sharedInstance].delegate = self;

  ...

  return YES;
}

...

#pragma mark - GCKLoggerDelegate

- (void)logMessage:(NSString *)message
           atLevel:(GCKLoggerLevel)level
      fromFunction:(NSString *)function
          location:(NSString *)location {
  if (kDebugLoggingEnabled) {
    NSLog(@"%@ - %@, %@", function, message, location);
  }
}

@end

Để bật các thông báo gỡ lỗi và thông báo chi tiết, hãy thêm dòng này vào đoạn mã sau thiết lập đại biểu (hiển thị trước đó):

Swift
let filter = GCKLoggerFilter.init()
filter.minimumLevel = GCKLoggerLevel.verbose
GCKLogger.sharedInstance().filter = filter
Mục tiêu-C
GCKLoggerFilter *filter = [[GCKLoggerFilter alloc] init];
[filter setMinimumLevel:GCKLoggerLevelVerbose];
[GCKLogger sharedInstance].filter = filter;

Bạn cũng có thể lọc thông điệp nhật ký do GCKLogger. Đặt cấp độ ghi nhật ký tối thiểu cho mỗi lớp, ví dụ:

Swift
let filter = GCKLoggerFilter.init()
filter.setLoggingLevel(GCKLoggerLevel.verbose, forClasses: ["GCKUICastButton",
                                                            "GCKUIImageCache",
                                                            "NSMutableDictionary"])
GCKLogger.sharedInstance().filter = filter
Mục tiêu-C
GCKLoggerFilter *filter = [[GCKLoggerFilter alloc] init];
[filter setLoggingLevel:GCKLoggerLevelVerbose
             forClasses:@[@"GCKUICastButton",
                          @"GCKUIImageCache",
                          @"NSMutableDictionary"
                          ]];
[GCKLogger sharedInstance].filter = filter;

Tên lớp có thể là tên cố định hoặc mẫu hình cầu, ví dụ: GCKUI\*GCK\*Session.