我正在尝试编写一个快速控制iTunes的应用程序。但是在初始化应用程序时会返回类型为AnyObject
的对象,但必须是iTunesApplication。
此对象不响应iTunes的方法和变量。谁知道如何使它发挥作用?
var iTunes = SBApplication.applicationWithBundleIdentifier("com.apple.iTunes")
iTunes.h标题还包含我需要访问但不能访问的类。 这些类会导致编译错误,就好像它们不在声明的iTunes.h中一样。
为什么这件事发生在我身上还不清楚。
通过@class声明为iTunes.h的类的完整列表:
@class iTunesPrintSettings,iTunesApplication,iTunesItem,iTunesAirPlayDevice,iTunesArtwork,iTunesEncoder,iTunesEQPreset,iTunesPlaylist,iTunesAudioCDPlaylist,iTunesLibraryPlaylist,iTunesRadioTunerPlaylist,iTunesSource,iTunesTrack,iTunesAudioCDTrack,iTunesFileTrack,iTunesSharedTrack,iTunesURLTrack,iTunesUserPlaylist,iTunesFolderPlaylist,iTunesVisual,iTunesWindow,iTunesBrowserWindow,iTunesEQWindow ,iTunesPlaylistWindow;
例如,在Objective - c中,您可以使用类似这样的内容来获取当前曲目
iTunesApplication *iTunes = [SBApplication applicationWithBundleIdentifier:@"com.apple.iTunes"];
NSLog(@"Current song is %@", [[iTunes currentTrack] name]);
但我无法迅速获得同等效力。
答案 0 :(得分:7)
在我的swift项目中,我遇到了使用生成的iTunes.h文件中定义的类型(链接错误等)的问题。
markhunte的答案解释说,您可以获得对应用程序对象的引用。但除此之外,我在尝试从该应用程序对象获取实例时遇到了编译/链接器错误。
在我的快速项目中,我最终创建了一个客观的C包装类,它将iTunes类型公开为基本目标C类型(数组和字典),并且也适应方法。
我的swift类使用此包装器而不是iTunes类型。
因此,目标C包装器看起来像这样(redux):
#import "ITunesBridgex.h"
#import "iTunes.h"
@interface ITunesBridgex(){
iTunesApplication *_iTunesApplication;
iTunesSource* _iTunesLibrary;
}
@end
@implementation ITunesBridgex
-(id)init {
self = [super init];
if (self) {
_iTunesApplication = [SBApplication applicationWithBundleIdentifier:@"com.apple.iTunes"];
NSArray *sources = [_iTunesApplication sources];
for (iTunesSource *source in sources) {
if ([source kind] == iTunesESrcLibrary) {
_iTunesLibrary = source;
break;
}
}
}
return self;
}
- (NSDictionary*) currentTrack {
iTunesTrack* track = _iTunesApplication.currentTrack;
if (!track)
return nil;
NSDictionary* dict = [NSDictionary dictionaryWithObjectsAndKeys: track.name, @"title", nil];
return dict;
}
@end
和调用swift代码:
import Foundation
import Cocoa
class ITunesBridgeSimple {
var iTunesBridgex: ITunesBridgex
init(){
iTunesBridgex = ITunesBridgex()
self.updateFromCurrentTrack()
}
func updateFromCurrentTrack() {
if let track = self.currentTrack {
if let title : AnyObject = track.objectForKey("title"){
println("Current track: \(title)")
}
}
}
}
答案 1 :(得分:3)
我怀疑问题是没有导入iTunes.h文件。因此,它的方法是不被捡起来。
所以我创建了一个-Bridging-Header.h文件。
我的项目名称为 swiftItunesTest 。所以-Bridging-Header.h文件命名为:
<强> swiftItunesTest-桥接-Header.h 强>
在此内部我放置了#import "iTunes.h"
行
在 AppDelegate.swift 文件中
import Cocoa
import Appkit
import ScriptingBridge
class AppDelegate: NSObject, NSApplicationDelegate {
@IBOutlet var window: NSWindow
func applicationDidFinishLaunching(aNotification: NSNotification?) {
var iTunes : AnyObject = SBApplication.applicationWithBundleIdentifier("com.apple.iTunes")
iTunes.playpause()
}
func applicationWillTerminate(aNotification: NSNotification?) {
// Insert code here to tear down your application
}
}
iTunesApplication(iTunes。)现在开始学习方法/功能
这是一个略微更新的例子。
func applicationDidFinishLaunching(aNotification: NSNotification) {
let iTunes : AnyObject = SBApplication(bundleIdentifier: "com.apple.iTunes")!
iTunes.playpause()
guard let currentTrack: AnyObject = iTunes.currentTrack!.name else {
print("No Tracks Playing")
return
}
print("\(currentTrack)")
}
func applicationWillTerminate(aNotification: NSNotification) {
}
答案 2 :(得分:2)
我编写了一个Python脚本来生成Scripting Bridge标头,然后自动生成一个原生的Swift版本。这样你就不必处理编写完整的包装器甚至使用桥接头。此外,不必担心链接器错误,因为它完全在Swift中。 java doc
答案 3 :(得分:1)
解决此问题的一种不太优雅的方法是将iTunes
声明为SBApplication
,然后,当您从iTunesApplication
调用函数时,向下转发iTunes
AnyObject
:
(iTunes as AnyObject).play()
请注意,如果您执行此操作,则没有类型安全:您可以调用任何Objective-C标头中声明的任何函数:它不能保证在特定类中实现,因此会使程序崩溃。