Блоки Swift не работают


Я пытался понять, как использовать JavaScriptCore в swift. Я сталкиваюсь с проблемами, однако, когда мне приходится иметь дело с блоками в качестве аргументов, кажется, что блок запускается немедленно, и аргументы получают возвращаемое значение блока. Что я делаю не так?

Рабочая цель C-код:

JSContext* context = [[JSContext alloc] initWithVirtualMachine:[[JSVirtualMachine alloc] init]];
context[@"test"] = ^(NSString *string) {
    //code
};

Что я пробовал:

1:

var ctx = JSContext(virtualMachine:JSVirtualMachine())
var ctx["test"] = {(string:NSString)->() in /*code*/ }

//Gives me "'JSContext' does not have a member named 'subscript'"

2:

var ctx = JSContext(virtualMachine:JSVirtualMachine())
let n: (string: String)->() = {string in /*code*/}

ctx.setObject(n, forKeyedSubscript:"test")

//Gives me "Type '(x: String) -> () does not conform to protocol 'AnyObject'"

3:

var ctx = JSContext(virtualMachine:JSVirtualMachine())
let n: (string: String)->() = {string in /*code*/}

ctx.setObject(n as AnyObject, forKeyedSubscript:"test")

//Gives me "Cannot downcast from '(string: String) -> () to non-@objc protocol type 'AnyObject'"

Я что-то упустил, или это просто ошибка Свифт?

Правка:

Теперь я также попробовал предложения из Cast closures / blocks

class Block<T> {
    let f : T
    init (_ f: T) { self.f = f }
}

И затем

ctx.setObject(Block<()->Void> {
        /*code*/
    }, forKeyedSubscript: "test")

Это решение позволяет мне компилировать, но я получаю ошибку времени выполнения:

Thread 1: EXC_BREAKPOINT (code=EXC_I386_BPT, subcode=0x0)
2 7

2 ответа:

Это как-то связано с тем, как быстро реализовать закрытие. Вы должны использовать @convention(block), чтобы отметить, что закрытие является блоком ObjC. Используйте unsafeBitCast, чтобы заставить бросить его

var block : @convention(block) (NSString!) -> Void = {
   (string : NSString!) -> Void in 
    println("test")
}

ctx.setObject(unsafeBitCast(block, AnyObject.self), forKeyedSubscript: "test")

От REPL

swift
Welcome to Swift!  Type :help for assistance.
  1> import Foundation
  2> var block : @convention(block)(NSString!) -> Void = {(string : NSString!) -> Void in println("test")}
block: @convention(block)(NSString!) -> Void =
  3> var obj: AnyObject = reinterpretCast(block) as AnyObject
obj: __NSMallocBlock__ = {} // familiar block type

У меня есть рабочая демо-версия по адресу:

А вот та часть, которая реализует регистрацию блоков:

typealias ID = AnyObject!
extension JSContext {
    func fetch(key:NSString)->JSValue {
        return getJSVinJSC(self, key)
    }
    func store(key:NSString, _ val:ID) {
        setJSVinJSC(self, key, val)
    }
    func store(key:NSString, _ blk:()->ID) {
        setB0JSVinJSC(self, key, blk)
    }
    func store(key:NSString, _ blk:(ID)->ID) {
        setB1JSVinJSC(self, key, blk)
    }
    func store(key:NSString, _ blk:(ID,ID)->ID) {
        setB2JSVinJSC(self, key, blk)
    }
}

Вам нужен очень маленький код objc и заголовок моста, чтобы заставить его работать. Подробности смотрите в хранилище.