はい、RestKitで可能です。
この記事を読むために必要なすべてのドキュメントは公式ウィキから入手できます。ここには、必要なことを行うための説明があります。これがお役に立てば幸いです。
まず、エンティティごとに3つのオブジェクトが必要になります。1つはThingsList用、もう1つは1つのThingを表すため、もう1つは1つのThingAttributeを表すためです。最も基本的な要素であるThingAttributeから始めましょう。
SOAttribute.h
#import <Foundation/Foundation.h>
@interface SOAttribute : NSObject{
NSNumber *attributeId; //I use this nomenclature because the word 'id' is reserved, same for other objects.
NSString *name;
NSString *val;
}
@end
SOAttribute.m
@implementation SOAttribute
@end
これにより、JSONのこの部分を表すことができます。
{"id": "","name": "Content Level","val": "Introductory"}
次に、Thingを次のように定義します。
SOThing.h
#import <Foundation/Foundation.h>
@interface SOThing : NSObject{
NSArray *attributeList; //This represent the list of attributes defined before
BOOL active;
NSString *desc;
NSString *title;
NSString *end;
NSString *start;
NSString *num;
NSNumber *thingId;
}
@end
SOThing.m
#import "SOThing.h"
@implementation SOThing
@end
お気づきのように、attributeListをNSArrayとして定義する必要があります。RestKitは、そこで属性のリストを受け取る必要があることを自動的に認識します。したがって、次の構造を定義しました。
{
"attributeList": [...],
"active": true,
"desc": "This is a really cool thing",
"end": "16:00",
"id": "2011080146112",
"num": "1002A",
"start": "15:00",
"title": "The thing title"
}
最後に、Thingリストを次のように定義します。
SOThingList.h
#import <Foundation/Foundation.h>
@interface SOThingList : NSObject{
NSDate *lastModifiedDate;
NSArray *myList;
}
@end
SoThingList.m
#import "SOThingList.h"
@implementation SOThingList
@end
前と同じように、Thingsのリストが必要なので、myListをNSArrayとして定義します。
さて、それは簡単でした。ここに、各エンティティをマップする方法をRestKitに指示する魔法があります。App Delegateに、次のコードを配置します。
#import "SOThingList.h"
#import "SOThing.h"
#import "SOAttribute.h"
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{
RKObjectMappingProvider *mappingProvider = [RKObjectManager sharedManager].mappingProvider;
RKObjectMapping *mappingForAttribute = [RKObjectMapping mappingForClass:[SOAttribute class]];
//This can be mapped directly
[mappingForAttribute mapAttributes:@"name",@"val", nil];
//here you indicate the special case, id->attributeId on our object
[mappingForAttribute mapKeyPath:@"id" toAttribute:@"attributeId"];
//Set the new mapping into the mapping provider.
[mappingProvider addObjectMapping:mappingForAttribute];
RKObjectMapping *mappingForThing = [RKObjectMapping mappingForClass:[SOThing class]];
//Same as before, these attributes can be mapped directly
[mappingForThing mapAttributes:@"active",@"title",@"end",@"start",@"desc",@"num", nil];
[mappingForThing mapKeyPath:@"id" toAttribute:@"thingId"];
//Here I indicate that any object on NSArray on attributeList should be mapped using SOAttribute mapping defined above.
[mappingForThing mapKeyPath:@"attributeList" toRelationship:@"attributeList" withMapping:mappingForAttribute];
[mappingProvider addObjectMapping:mappingForThing];
RKObjectMapping *mappingForThingsList = [RKObjectMapping mappingForClass:[SOThingList class]];
[mappingForThingsList mapAttributes:@"lastModifiedDate", nil];
[mappingForThingsList mapKeyPath:@"myList" toRelationship:@"myList" withMapping:mappingForThing];
[mappingProvider addObjectMapping:mappingForThingsList];
}
上記のすべての後、ブロックを使用して次の方法でThingリストを取得できます。
SomeObject.m
- (void) get: (NSString *) resourcePath onLoad:(RKObjectLoaderDidLoadObjectBlock) loadBlock onError:(RKRequestDidFailLoadWithErrorBlock)failBlock{
RKObjectMapping *mapping = [[RKObjectManager sharedManager].mappingProvider objectMappingForClass:[SOThingList class]];
[[RKObjectManager sharedManager] loadObjectsAtResourcePath:resourcePath usingBlock:^(RKObjectLoader* loader) {
loader.objectMapping = mapping;
loader.delegate = self;
loader.onDidLoadObject = loadBlock;
loader.onDidFailWithError = ^(NSError * error){
NSLog(@"%@",error);
};
loader.onDidFailLoadWithError = failBlock;
loader.onDidLoadResponse = ^(RKResponse *response) {
//Do something
};
}];
}
そして、あなたはそれをこのように使用します...
[someObjectInstance get:@"http://mydomain.com/mywebservice/" onLoad:^(id object){
SOThingList *thingList = (SOThingList *) object;
//Do something with your shiny thingList
} onError:^(NSError *error) {
//Display an error message :)
}];