JNA Objective-C (Rococoa)日历回调



我想通过Rococoa访问Java中的Objective-C EKEventStore。API指定了一个回调,当用户接受日历访问提示时得到通知,这在纯Objective-C中工作得很好。

    Native.loadLibrary("EventKit", EventKitLibrary.class);
    EKEventStore store = EKEventStore.CLASS.alloc();
    store.init();
    //store = store.initWithAccessToEntityTypes(EKEntityType.EKEntityTypeEvent); // no notification
    EKEventStoreRequestAccessCompletionHandler handler = new EKEventStoreRequestAccessCompletionHandler() {
        @Override
        public void invoke(boolean granted, Pointer error) {
            System.out.println("Access: " + granted);
            NSArray calArray = store.calendarsForEntityType(EKEntityType.EKEntityTypeEvent);
            for (int i = 0; i < calArray.count(); i++) {
                NSObject calObject = calArray.objectAtIndex(i);
                EKCalendar osxcal = Rococoa.cast(calObject, EKCalendar.class);
                System.out.println(osxcal.title().toString());
            }
        }
    };
    ObjCObject object = Rococoa.proxy(handler); // get Objective C Callback Object to send
    store.requestAccessToEntityType_completion(EKEntityType.EKEntityTypeEvent, object.id());
    try {
        Thread.sleep(10000); // wait for the access prompt
    } catch (InterruptedException ex) {
    }
    // random object access to save instances from gc
    System.out.println(handler.toString());
    System.out.println(store.id());
    System.out.println(object.id());
图书馆

public interface EventKitLibrary extends Library {
    public static EventKitLibrary INSTANCE = (EventKitLibrary) Native.loadLibrary("EventKit", EventKitLibrary.class);
}

映射类

public abstract class EKEventStore extends NSObject {
    public static final _Class CLASS = Rococoa.createClass("EKEventStore", _Class.class);
    public interface _Class extends ObjCClass {
        public abstract EKEventStore alloc();
    }
    public static interface EKEntityType {
        public static final int EKEntityTypeEvent = 0;
        public static final int EKEntityTypeReminder = 1;
    };
    public static interface EKEntityMask {
        public static final int EKEntityMaskEvent = (1 << EKEntityType.EKEntityTypeEvent);
        public static final int EKEntityMaskReminder = (1 << EKEntityType.EKEntityTypeReminder);
    };
    public abstract EKEventStore initWithAccessToEntityTypes(int EKEntityMask);
    public abstract EKEventStore init();
    public abstract void requestAccessToEntityType_completion(int EKEntityType, ID handler);
    interface EKEventStoreRequestAccessCompletionHandler {
        void invoke(boolean granted, Pointer error);
    }
    public abstract NSArray calendarsForEntityType(int EKEntityType);
}

public abstract class EKCalendar extends NSObject {
    public static final _Class CLASS = Rococoa.createClass("EKCalendar", _Class.class);
    public static interface _Class extends ObjCClass {
        public NSObject alloc();
    }
    public abstract NSString title();
}

对于NSError参数的缺失类型转换,我只得到一个IllegalArgumentException。是我做错了什么,还是我应该实现一个TypeConverter?如果,我该怎么做呢?

编辑:

现在我使用指针而不是NSError作为回调函数的参数,我得到以下jvm崩溃。

EDIT2:

现在我使用Rococoa.proxy(handler)函数进行回调,就像在Rococoa库中一样。输入提示出现了,但是回调函数没有被调用。我认为我的回调初始化仍然是错误的

通常,TypeMapper是这样实现的,它将Pointer本地类型转换为其他Java类型:

class NSErrorTypeMapper extends DefaultTypeMapper {
    public NSErrorTypeMapper() {
        TypeConverter tc = new TypeConverter() {
            public Object toNative(Object value, ToNativeContext ctxt) {
                Pointer p = // convert your NSError "value" into a Pointer
                return p;
            }
            public Object fromNative(Object value, FromNativeContext ctxt) {
                Pointer p = (Pointer)value;
                Object object = // convert the pointer into an NSError object
                return object;    
            }
            public class nativeType() {
                return Pointer.class;
            }
        };
        addToNativeConverter(NSError.class, tc);
        addFromNativeConverter(NSError.class, tc);
    }
}

最新更新