Доступность AVAudioSession: наличие в Swift 4.2

После перехода на Swift 4.2 я получаю несколько ошибок, один из которых странный. Это похоже на ошибку в Xcode 10, но есть ли доступ к обходу?

do {
    try AVAudioSession.sharedInstance().setCategory(AVAudioSession.Category.playAndRecord, with: options)
} catch {
    NSLog("Could not set audio session category")
}

**** 'setCategory (_: with :)' недоступен в Swift

Ответ 1

iOS 10+

Если вы ориентируетесь на iOS 10+, просто перейдите на новый API и используйте:

try AVAudioSession.sharedInstance().setCategory(.playAndRecord, mode: .default, options: [])

Старые версии iOS

Когда вы попробуете это для приложения, ориентированного на более старую версию iOS (например, iOS 9), вы получите setCategory(_:mode:options:)' is only available on iOS 10.0 or newer версии Error.

Это было сообщено как ошибка в Apple API и исправлено в Xcode 10.2. Для более старых версий XCode (например, XCode 10.1) я нашел обходной путь. При создании помощника Objective -C, как описано, вы все равно можете получить доступ к старому API, потому что он все еще доступен для Objective -C.

Обходной путь 1: метод .perform()

Если вам нужно быстрое встроенное исправление без обработки ошибок, вы можете вызвать API Obj. -C с помощью .perform():

if #available(iOS 10.0, *) {
  try AVAudioSession.sharedInstance().setCategory(.playback, mode: .default, options: [])
} else { 
  // Set category with options (iOS 9+) setCategory(_:options:)       
  AVAudioSession.sharedInstance().perform(NSSelectorFromString("setCategory:withOptions:error:"), with: AVAudioSession.Category.playback, with:  [])

  // Set category without options (<= iOS 9) setCategory(_:)
  AVAudioSession.sharedInstance().perform(NSSelectorFromString("setCategory:error:"), with: AVAudioSession.Category.playback)
}

Обходной путь 2: метод класса Helper

Вот шаги, как сделать это прямо сейчас, если вы хотите больше контроля над ошибками

  1. Создайте новый файл Objective-C в моем случае AudioSessionHelper.m. Когда будет предложено создать файл заголовка моста, нажмите " Да" (если у вас его еще нет в вашем проекте)
  2. Создайте новый Header файл AudioSessionHelper.h
  3. Код вставки
AudioSessionHelper.h
#ifndef AudioSessionHelper_h
#define AudioSessionHelper_h
#import <AVFoundation/AVFoundation.h>

@interface AudioSessionHelper: NSObject
+ (BOOL) setAudioSessionWithError:(NSError **) error;
@end

#endif /* AudioSessionHelper_h */
AudioSessionHelper.m
#import "AudioSessionHelper.h"
#import <Foundation/Foundation.h>

@implementation AudioSessionHelper: NSObject

+ (BOOL) setAudioSessionWithError:(NSError **) error {
    BOOL success = [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback error:error];
    if (!success && error) {
        return false;
    } else {
        return true;
    }
}
@end
  1. Вставьте свой вспомогательный класс в файл заголовка моста
[ПРОЕКТ] -Bridging-Header.h
#import "AudioSessionHelper.h"
  1. Используйте это в своем проекте Swift
if #available(iOS 10.0, *) {
    try AVAudioSession.sharedInstance().setCategory(.playback, mode: .default, options: [])
} else {
    try AudioSessionHelper.setAudioSession()
}

Это не красиво и добавляет много ненужного кода и файлов в ваш проект, поэтому используйте его, если вы срочно хотите или должны использовать Swift 4.2 на Xcode 10.1 прямо сейчас. Во всех остальных случаях вам лучше использовать Xcode 10.2.

Ответ 2

Это проблема с AVFoundation в SDK Xcode 10. Вы можете обойти это, написав функцию Objective-C, которая вызывает старый API, поскольку они все еще доступны в Objective-C. Но если вы ориентируетесь только на iOS 10 или более позднюю версию, вы можете писать быстро

do{
    if #available(iOS 10.0, *) {
        try AVAudioSession.sharedInstance().setCategory(.playAndRecord, mode: .default, options: options)
    } else {
        //the following line is now "unavailable", hence there is no way to support iOS <10
        //try AVAudioSession.sharedInstance().setCategory(.playAndRecord, with: options)
    }
} catch let error {
    print("Could not set audio session category: \(error.localizedDescription)")
}

Источник: Swift Forum

Ответ 3

То, что я делаю, это call setCategory(_:mode:options:). Параметры options: параметр может быть опущен, и если у вас нет режима, вы можете использовать режим .default.

Ответ 4

Проблема была решена в Xcode 10.2.


Для Xcode 10.1 с iOS 9 (и старше) единственное другое предлагаемое решение от Benno и должно полагаться либо на Objective-C, либо на perform(NSSelectorFromString("setCategory:error:")). К сожалению, данный код не позволяет красиво обрабатывать ошибки, поэтому я хотел бы представить решение, основанное на простом использовании Swift 4.1 в отдельной среде при использовании Xcode 10.1 (и оно будет работать, даже если ваш проект использует Свифт 4.2).

Хитрость заключается в том, чтобы просто создать отдельный фреймворк (или lib), чтобы обернуть вызов setCategory(_:) используя более старую версию Swift (например, 4.1). В этих рамках вам нужно всего лишь иметь:

import AVFoundation

extension AVAudioSession {
    @available (iOS 3.0, tvOS 9.0, watchOS 2.0, *)
    @objc open func setCategorySwift(_ category: String) throws {
        try setCategory(category)
    }
}

(также доступно через pod 'AVAudioSessionSetCategorySwift')

Затем, вернувшись к своему проекту Swift 4.2 (и, в конечном итоге, к pod install), вы можете использовать его так же просто, как setCategorySwift(AVAudioSession.Category.playback.rawValue). Более длинный пример с try-catch может быть:

import AVAudioSessionSetCategorySwift

let session = AVAudioSession.sharedInstance()
do {
    if #available(iOS 10.0, *) {
        try session.setCategory(AVAudioSession.Category.playback, mode: .default, options: [])
    } else {
        // Swift 4.2 workaround: https://github.com/coeur/AVAudioSessionSetCategorySwift
        try session.setCategorySwift(AVAudioSession.Category.playback.rawValue)
    }
    try session.setActive(true)
} catch {
    print("AVAudioSession.setCategory error: \(error)")
}