我正在使用swift中的CoreAudio,并且需要在用户更改系统音量时找到它。
我可以正确地获取音量,甚至可以添加一个属性侦听器来查找用户何时更改音量。但我需要在某些时候停止监听(如果用户更改默认输出设备)但我无法删除属性监听器。
我为playground创建了一个基本测试,并在命令行obj-c项目中进行了相同的测试。该测试在obj-c中运行良好,但在swift中不起作用
代码只是添加了侦听器然后将其删除,因此在运行代码后更改卷应该什么都不打印,但是在swift中它会继续打印
快速代码:
import CoreAudio
//first get default output device
var outputDeviceAOPA:AudioObjectPropertyAddress = AudioObjectPropertyAddress(
mSelector: kAudioHardwarePropertyDefaultOutputDevice,
mScope: kAudioObjectPropertyScopeGlobal,
mElement: kAudioObjectPropertyElementMaster)
var outputDeviceID = kAudioObjectUnknown
var propertySize = UInt32(sizeof(AudioDeviceID))
AudioObjectGetPropertyData(UInt32(kAudioObjectSystemObject), &outputDeviceAOPA,
0, nil, &propertySize, &outputDeviceID)
// get volume from device
var volumeAOPA:AudioObjectPropertyAddress = AudioObjectPropertyAddress(
mSelector: kAudioDevicePropertyVolumeScalar,
mScope: kAudioObjectPropertyScopeOutput,
mElement: kAudioObjectPropertyElementMaster
)
var volume:Float32 = 0.5
var volSize = UInt32(sizeof(Float32))
AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume)
print(volume)
var queue = dispatch_queue_create("testqueue", nil)
var listener:AudioObjectPropertyListenerBlock = {
_, _ in
AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume)
print(volume)
}
AudioObjectAddPropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener)
AudioObjectRemovePropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener)
while true{
//keep playground running
}
这是Objective-c代码:
//objective-c code working
// main.m
// objccatest
#import <Foundation/Foundation.h>
#import <CoreAudio/CoreAudio.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
//first get default output device
AudioObjectPropertyAddress outputDeviceAOPA;
outputDeviceAOPA.mSelector= kAudioHardwarePropertyDefaultOutputDevice;
outputDeviceAOPA.mScope= kAudioObjectPropertyScopeGlobal;
outputDeviceAOPA.mElement= kAudioObjectPropertyElementMaster;
AudioObjectID outputDeviceID = kAudioObjectUnknown;
UInt32 propertySize = sizeof(AudioDeviceID);
AudioObjectGetPropertyData(kAudioObjectSystemObject, &outputDeviceAOPA,
0, nil, &propertySize, &outputDeviceID);
// get volume from device
AudioObjectPropertyAddress volumeAOPA;
volumeAOPA.mSelector= kAudioDevicePropertyVolumeScalar;
volumeAOPA.mScope= kAudioObjectPropertyScopeOutput;
volumeAOPA.mElement= kAudioObjectPropertyElementMaster;
Float32 volume = 0.5;
UInt32 volSize = sizeof(Float32);
AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume);
NSLog(@"%f", volume);
dispatch_queue_t queue = dispatch_queue_create("testqueue", nil);
AudioObjectPropertyListenerBlock listener = ^(UInt32 a, const AudioObjectPropertyAddress* arst){
AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume);
NSLog(@"%f", volume);
};
AudioObjectAddPropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener);
AudioObjectRemovePropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener);
while (true){
//keep app running
}
}
return 0;
}
我认为这是Core Audio API中的一个错误,但是可能有一个变通方法或obj-c块与swift闭包不同。
答案 0 :(得分:2)
我遇到了同样的问题。
将Swift闭包传递给Objective-C API似乎是一个问题。 相同的闭包被块复制两次,每个副本具有不同的地址。因此,为注册器注册提供的块与注销期间使用的块不匹配。
我发现的解决方法是使用Objective-C helper注册和取消注册侦听器,这也将存储用于取消注册的块地址。
答案 1 :(得分:1)
是的,实际上它可能是一个错误,因为AudioObjectRemovePropertyListenerBlock
无法删除侦听器块。但是,我发现使用AudioObjectPropertyListenerProc
注册AudioObject
可以解决Swift问题。
//var queue = dispatch_queue_create("testqueue", nil)
//var listener:AudioObjectPropertyListenerBlock = {
// _, _ in
// AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume)
// print(volume)
//}
//
//AudioObjectAddPropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener)
//AudioObjectRemovePropertyListenerBlock(outputDeviceID, &volumeAOPA, queue, listener)
var data: UInt32 = 0
func listenerProc() -> AudioObjectPropertyListenerProc {
return { _, _, _, _ in
AudioObjectGetPropertyData(outputDeviceID, &volumeAOPA, 0, nil, &volSize, &volume)
print(volume)
return 0
}
}
AudioObjectAddPropertyListener(outputDeviceID, &volumeAOPA, listenerProc(), &data)
AudioObjectRemovePropertyListener(outputDeviceID, &volumeAOPA, listenerProc(), &data)