Swift和scriptingbridge对象初始化

时间:2014-06-06 13:08:51

标签: xcode itunes swift applescript-objc

我正在尝试编写一个快速控制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]);

但我无法迅速获得同等效力。

4 个答案:

答案 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标头中声明的任何函数:它不能保证在特定类中实现,因此会使程序崩溃。