WatchConnectivity - 从手表配套应用程序在 iOS 设备上启动 AVPlayer

WatchConnectivity - Start AVPlayer on iOS device from watch companion app

提问人:Vasyl Nadtochii 提问时间:8/18/2023 更新时间:8/20/2023 访问量:64

问:

假设我有 SwiftUI 应用程序,它开始播放列表中所选 url 的音频:

struct ContentView: View {
    
    @ObservedObject var viewModel: ContentViewModel
    
    var body: some View {
        List(PlayerLinks.links, id: \.self) { link in
            Button(link) {
                viewModel.play(from: link)
            }
        }
    }
}

class ContentViewModel: ObservableObject {
    
    let player = Player()
    
    init() {
        NotificationCenter.default.addObserver(
            self,
            selector: #selector(notificationAction),
            name: .newLinkChosen,
            object: nil
        )
    }
    
    func play(from url: String) {
        player.play(from: url)
    }
    
    @objc func notificationAction(_ notification: Notification) {
        if let userInfo = notification.userInfo,
           let newURL = userInfo["newLink"] as? String {
            self.play(from: newURL)
        }
    }
}

class Player {
    
    var player: AVPlayer?
    
    init() {
        do {
            try AVAudioSession.sharedInstance()
                .setCategory(
                    AVAudioSession.Category.playback,
                    mode: AVAudioSession.Mode.default,
                    options: []
                )
        } catch let error as NSError {
            print(error.localizedDescription)
        }
    }
    
    func play(from url: String) {
        let playerItem = AVPlayerItem(url: URL(string: url)!)
        player = AVPlayer(playerItem: playerItem)
        player?.play()
    }
}

这个应用程序有 watchOS 配套应用程序,它还有一个电台列表:

struct ContentWatchView: View {
    
    @ObservedObject var viewModel: ContentWatchViewModel
    
    var body: some View {
        List(PlayerLinks.links, id: \.self) { link in
            Button(link) {
                viewModel.onRowSelected(url: link)
            }
        }
    }
}

class ContentWatchViewModel: ObservableObject {
    
    var connectivityManager: WatchConnectivityManager
    
    init(connectivityManager: WatchConnectivityManager) {
        self.connectivityManager = connectivityManager
    }
    
    func onRowSelected(url: String) {
        connectivityManager.sendPlayerLinkToIOS(url)
    }
}

通过在watchOS应用程序中点击行,我需要在iOS设备上启动AVPlayer播放。

为此,我实现了处理 watchOS 和 iOS 应用程序之间的通信:WatchConnectivityManager

class WatchConnectivityManager: NSObject, ObservableObject, WCSessionDelegate {
    
    private let session: WCSession = WCSession.default
    var isReachable = false
    
    static var shared = WatchConnectivityManager()

    override init() {
        super.init()
        if WCSession.isSupported() {
            session.delegate = self
            session.activate()
        }
    }
    
    func session(
        _ session: WCSession,
        activationDidCompleteWith activationState: WCSessionActivationState,
        error: Error?
    ) {
        #if os(iOS)
        print("ACTIVATED ON IOS")
        #elseif os(watchOS)
        print("ACTIVATED ON WATCHOS")
        #endif
        DispatchQueue.main.async {
            self.isReachable = session.isReachable
        }
    }
    
    func sessionReachabilityDidChange(_ session: WCSession) {
        DispatchQueue.main.async {
            self.isReachable = session.isReachable
        }
    }
    
    #if os(iOS)
    func sessionDidDeactivate(_ session: WCSession) {
        session.activate()
    }

    func sessionDidBecomeInactive(_ session: WCSession) {
        print("Session did become inactive: \(session.activationState.rawValue)")
    }

    func sessionWatchStateDidChange(_ session: WCSession) {
        print("Session watch state did change: \(session.activationState.rawValue)")
    }
    #endif
    
    // MARK: MESSAGE RECEIVER
    func session(
        _ session: WCSession,
        didReceiveMessage message: [String : Any],
        replyHandler: @escaping ([String : Any]) -> Void
    ) {
        #if os(iOS)
        if let action = message["action"] as? String,
           action == "newPlayerLinkChosen",
           let link = message["link"] as? String {
            DispatchQueue.main.async {
                NotificationCenter.default.post(
                    name: .newLinkChosen,
                    object: nil,
                    userInfo: ["newLink": link]
                )
                replyHandler(["success": true])
            }
        } else {
            replyHandler(["success": false])
        }
        #endif
    }
    
    // MARK: MESSAGE SENDERS
    
    #if os(watchOS)
    func sendPlayerLinkToIOS(_ link: String) {
        let message = [
            "action": "newPlayerLinkChosen",
            "link": link
        ]

        session.sendMessage(message) { replyHandler in
            print(replyHandler)
        } errorHandler: { error in
            print(error.localizedDescription)
        }
    }
    #endif
}

extension Notification.Name {
    
    static let newLinkChosen = Notification.Name("NewLinkChosen")
}

sendPlayerLinkToIOSfunc 发送带有所选链接的消息,该链接由 method 接收,然后它向 .iOS 将收到此通知,播放器将启动。MESSAGE RECEIVERNotificationCenter.defaultContentViewModel

当 iOS 应用程序在前台时,一切正常,但是当我们转到主屏幕并再次从观看应用程序中选择一些 url 时,它不起作用(iOS 应用程序未终止)。

以下是日志,如果它们对某人有帮助:

ACTIVATED ON IOS

2023-08-18 16:23:38.651377+0300 RemotePlayer[20479:6162721] [plugin] AddInstanceForFactory: No factory registered for id <CFUUID 0x6000022fce00> F8BB1C28-BAE8-11D6-9C31-00039315CD46

2023-08-18 16:23:52.046769+0300 RemotePlayer[20479:6163115] [AMCP]   4611          HALC_ProxyIOContext.cpp:783   HALC_ProxyIOContext::_StartIO(): Client running as an adaptive unboosted daemon

2023-08-18 16:23:52.047123+0300 RemotePlayer[20479:6163115]             HALPlugIn.cpp:519    HALPlugIn::StartIOProc: got an error from the plug-in routine, Error: 1852797029 (nope)

2023-08-18 16:23:52.048409+0300 RemotePlayer[20479:6163115] [aqme]                AQMEIO.cpp:211   error 1852797029

2023-08-18 16:23:52.049634+0300 RemotePlayer[20479:6163115] [aqme]  MEDeviceStreamClient.cpp:431   AQME Default-InputOutput: client stopping after failed start: <CA_UISoundClientBase@0x149514830>; running count now 0

2023-08-18 16:23:52.050233+0300 RemotePlayer[20479:6163115]      CA_UISoundClient.cpp:285   CA_UISoundClientBase::StartPlaying: AddRunningClient failed (status = 1852797029).
2023-08-18 16:23:53.738535+0300 RemotePlayer[20479:6163570] [AMCP]  59139          HALC_ProxyIOContext.cpp:783   HALC_ProxyIOContext::_StartIO(): Client running as an adaptive unboosted daemon

2023-08-18 16:23:53.741183+0300 RemotePlayer[20479:6163570]             HALPlugIn.cpp:519    HALPlugIn::StartIOProc: got an error from the plug-in routine, Error: 1852797029 (nope)

2023-08-18 16:23:53.744653+0300 RemotePlayer[20479:6163570] [aqme]                AQMEIO.cpp:211   error 1852797029

2023-08-18 16:23:53.745684+0300 RemotePlayer[20479:6163570] [aqme]  MEDeviceStreamClient.cpp:431   AQME Default-InputOutput: client stopping after failed start: <AudioQueueObject@0x14a809000; Unknown figplayer; [20479]; play>; running count now 0
iOS Swift WatchKit 手表连接性

评论


答:

0赞 Vasyl Nadtochii 8/19/2023 #1

刚给自己买了 Apple Watch 并再次测试。当应用程序在后台以及应用程序根本没有打开时,播放器工作正常。出于某种原因,这在模拟器上不起作用......因此,如果您在模拟器上遇到类似的问题,请在真实设备上尝试一下,也许这根本不是问题。