这里Himi给出对于开发iOS的朋友们整理一个指南集合,其中主要包括申请IDP需要注意的地方、有了开发者证书如何真机调试、在自己的游戏应用中如何接入GameCenter以及如何在游戏接入OpenFeint;
-----------申请企业级IDP,或者个人IDP
通过Himi的申请经验,直接打苹果在中国的客服,按照步骤一步一步详细的让客服进行指导,可能很多童鞋说我这句跟没说一样,呵呵,如果真的你是第一次申请IDP那么如果你不打客服,N多细节都会造成你1~15天耐心等待,Himi申请过程中由于一个名称和一个勾选错误整整耽误一个月的时间;最后仍是不停的跟客服交涉终于Ok顺利申请到;
这里Himi给出苹果在中国的客服电话:4006701855 (建议拨打客服之前大致的先百度google下申请IDP的流程,网上一大堆,这里Himi不赘述了)
---------------申请到IDP后如何真机调试
1.制作证书的过程Himi这里不多赘述,百度、google下N多文章呢;制作证书连接(前提是申请IDP成功):http://developer.apple.com/membercenter/index.action
2.正确制作证书后,有个这样的文件:
双击此文件,弹出Organizer-Devices界面,连接你的真机iphone、ipad或者touch,然后左侧可以看到如下图:
右侧的绿色小灯表示可以正常使用,这个小灯如果是×××,那就说明你的证书有问题,可能是此手机的UDID没有在证书内等原因;
3.确保真机正常后,点击你的项目,右侧点击PROJECT点击Build Settings页面,然后Code Signing下设置Code Signing Identity为你的证书,如下图:
4.点击你的项目,右侧点击TARGETS,点击Info页面下的设置Bundle identifier,这个Bundle identifier在你制作证书的过程中就会了解到,如下图
OK,可以编译运行你的项目到真机中了;
---------------游戏接入GameCenter 指南
1. iTunes Connect 设置
首先,申请一个应用程序,不必提交.目地是为了得到Bundle ID.
然后设置一下工程中Info.plist的Bundle identifier使之与iTunes Connect中的Bundle ID相同,否则当你尝试登录GameCenter的时候,会提示一个不支持GameCenter的错误.
申请完毕,打开你刚申请的application,点击Manage Game Center选项.
进入后点击Enable Game Center使你的Game Center生效.
接下来就可以设置自己的Leaderboard和Achievements.
2. Leaderboard设置
Leaderboard纵观图如下所示.
1.sort Order: Leaderboard中的内容是以升序还是降序排列.
2.Score Format Type:分数的类型.
3.*Categories:Leaderboard的一个分数榜,这个可以创建多个,比如游戏可以分为Easy,Normal,Hard三个难度,每个难度一个榜.
*设置完成后保存,完成了一个 Leaderboard的设置.我们可以根据需要添加多个 leaderboard.
4.**Score Format Location: leaderboard支持的语言.
**可以支持多种语言,每支持一种语言,需要完成一个上述操作.
这个时候右下角会出现save change按钮,点击完成leaderboard的设置.
你可以根据需要随时更改你的leaderboard,操作与上述内容类似.
3. Achievements设置
Achievements界面内容比较少,点击左上角的Add New Achievement,打开如下图所示的Achievements创建界面.
Hidden:表示该成就为解锁前玩家是否可见.
Achievement ID:程序通过这个属性来识别成就.
*Achievement Localization:该成就支持的语言.
*Achievement Localization设置如下图所示.
*其中,成就的Image必须是512X512,72DPI的.
一切设置完成后,点击save change按钮即完成一个成就的设置.
4.总体功能
在使用各个功能前,你需要了解一下块函数。传送门: https://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/Blocks/Articles/00_Introduction.html
4.1 对Game Center支持判断
- - (BOOL) isGameCenterAvailable
- {
- Class gcClass = (NSClassFromString(@"GKLocalPlayer"));
- NSString *reqSysVer = @"4.1";
- NSString *currSysVer = [[UIDevice currentDevice] systemVersion];
- BOOL osVersionSupported = ([currSysVer compare:reqSysVer options:NSNumericSearch] != NSOrderedAscending);
- return (gcClass && osVersionSupported);
- }
4.2用户登录
- - (void) authenticateLocalPlayer
- {
- [[GKLocalPlayer localPlayer] authenticateWithCompletionHandler:^(NSError *error){
- if (error == nil) {
- //成功处理
- NSLog(@"成功");
- NSLog(@"1--alias--.%@",[GKLocalPlayer localPlayer].alias);
- NSLog(@"2--authenticated--.%d",[GKLocalPlayer localPlayer].authenticated);
- NSLog(@"3--isFriend--.%d",[GKLocalPlayer localPlayer].isFriend);
- NSLog(@"4--playerID--.%@",[GKLocalPlayer localPlayer].playerID);
- NSLog(@"5--underage--.%d",[GKLocalPlayer localPlayer].underage);
- }else {
- //错误处理
- NSLog(@"失败 %@",error);
- }
- }];
- }
对于开发者来说,Game Center必须经过测试才能上线,没有上线的程序在测试环境中登录时会出现sandbox提示.如图.
4.3用户变更检测
由于4.0以后系统支持多任务,玩家的机器有可能被不同的玩家接触,导致Game Center上的用户发生变化,当发生变化的时候,程序必须在游戏中通知玩家.
- - (void) registerForAuthenticationNotification
- {
- NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
- [nc addObserver:self
- selector:@selector(authenticationChanged)
- name:GKPlayerAuthenticationDidChangeNotificationName
- object:nil];
- }
- - (void) authenticationChanged
- {
- if ([GKLocalPlayer localPlayer].isAuthenticated)
- {
- ;// Insert code here to handle a successful authentication.
- }
- else
- {
- ;// Insert code here to clean up any outstanding Game Center-related classes.
- }
- }
5.对Leaderboard进行操作
5.1上传一个分数
- - (void) reportScore: (int64_t) score forCategory: (NSString*) category
- {
- GKScore *scoreReporter = [[[GKScore alloc] initWithCategory:category] autorelease];
- scoreReporter.value = score;
- [scoreReporter reportScoreWithCompletionHandler:^(NSError *error) {
- if (error != nil)
- {
- // handle the reporting error
- NSLog(@"上传分数出错.");
- //If your application receives a network error, you should not discard the score.
- //Instead, store the score object and attempt to report the player’s process at
- //a later time.
- }else {
- NSLog(@"上传分数成功");
- }
- }];
- }
当上传分数出错的时候,要将上传的分数存储起来,比如将SKScore存入一个NSArray中.等可以上传的时候再次尝试.
5.2下载一个分数
- //GKScore objects provide the data your application needs to create a custom view.
- //Your application can use the score object’s playerID to load the player’s alias.
- //The value property holds the actual value you reported to Game Center. the formattedValue
- //property provides a string with the score value formatted according to the parameters
- //you provided in iTunes Connect.
- - (void) retrieveTopTenScores
- {
- GKLeaderboard *leaderboardRequest = [[GKLeaderboard alloc] init];
- if (leaderboardRequest != nil)
- {
- leaderboardRequest.playerScope = GKLeaderboardPlayerScopeGlobal;
- leaderboardRequest.timeScope = GKLeaderboardTimeScopeAllTime;
- leaderboardRequest.range = NSMakeRange(1,10);
- leaderboardRequest.category = @"TS_LB";
- [leaderboardRequest loadScoresWithCompletionHandler: ^(NSArray *scores, NSError *error) {
- if (error != nil){
- // handle the error.
- NSLog(@"下载失败");
- }
- if (scores != nil){
- // process the score information.
- NSLog(@"下载成功....");
- NSArray *tempScore = [NSArray arrayWithArray:leaderboardRequest.scores];
- for (GKScore *obj in tempScore) {
- NSLog(@" playerID : %@",obj.playerID);
- NSLog(@" category : %@",obj.category);
- NSLog(@" date : %@",obj.date);
- NSLog(@" formattedValue : %@",obj.formattedValue);
- NSLog(@" value : %d",obj.value);
- NSLog(@" rank : %d",obj.rank);
- NSLog(@"**************************************");
- }
- }
- }];
- }
- }
说明:
1) playerScope:表示检索玩家分数范围.
2) timeScope:表示某一段时间内的分数
3) range:表示分数排名的范围
4) category:表示你的Leaderboard的ID.
5.3玩家信息交互
Game Center最重要的一个功能就是玩家交互.所以,必须检索已经登录玩家的好友信息.根据自己的需要做出设置,比如,可以与好友比较分数,或者好友排行榜等.
//检索已登录用户好友列表
- - (void) retrieveFriends
- {
- GKLocalPlayer *lp = [GKLocalPlayer localPlayer];
- if (lp.authenticated)
- {
- [lp loadFriendsWithCompletionHandler:^(NSArray *friends, NSError *error) {
- if (error == nil)
- {
- [self loadPlayerData:friends];
- }
- else
- {
- ;// report an error to the user.
- }
- }];
- }
- }
上面的friends得到的只是一个身份列表,里面存储的是NSString,想要转换成好友ID,必须调用- (void) loadPlayerData: (NSArray *) identifiers方法,该方法得到的array里面存储的才是GKPlayer对象.如下
- /*
- Whether you received player identifiers by loading the identifiers for the local player’s
- friends, or from another Game Center class, you must retrieve the details about that player
- from Game Center.
- */
- - (void) loadPlayerData: (NSArray *) identifiers
- {
- [GKPlayer loadPlayersForIdentifiers:identifiers withCompletionHandler:^(NSArray *players, NSError *error) {
- if (error != nil)
- {
- // Handle the error.
- }
- if (players != nil)
- {
- NSLog(@"得到好友的alias成功");
- GKPlayer *friend1 = [players objectAtIndex:0];
- NSLog(@"friedns---alias---%@",friend1.alias);
- NSLog(@"friedns---isFriend---%d",friend1.isFriend);
- NSLog(@"friedns---playerID---%@",friend1.playerID);
- }
- }];
- }
至此,leaderboard功能介绍完毕
6.对Achievement进行操作
这一部分内容比较多,而且有的地方有点重复的感觉.
6.1汇报一个成就的进度
对于一个玩家可见的成就,你需要尽可能的报告给玩家解锁的进度;对于一个一部完成的成就,则不需要,当玩家的进度达到100%的时候,会自动解锁该成就.
- - (void) reportAchievementIdentifier: (NSString*) identifier percentComplete: (float) percent
- {
- GKAchievement *achievement = [[[GKAchievement alloc] initWithIdentifier: identifier] autorelease];
- if (achievement)
- {
- achievement.percentComplete = percent;
- [achievement reportAchievementWithCompletionHandler:^(NSError *error)
- {
- if (error != nil)
- {
- //The proper way for your application to handle network errors is retain
- //the achievement object (possibly adding it to an array). Then, periodically
- //attempt to report the progress until it is successfully reported.
- //The GKAchievement class supports the NSCoding protocol to allow your
- //application to archive an achie
- NSLog(@"报告成就进度失败 ,错误信息为: \n %@",error);
- }else {
- //对用户提示,已经完成XX%进度
- NSLog(@"报告成就进度---->成功!");
- NSLog(@" completed:%d",achievement.completed);
- NSLog(@" hidden:%d",achievement.hidden);
- NSLog(@" lastReportedDate:%@",achievement.lastReportedDate);
- NSLog(@" percentComplete:%f",achievement.percentComplete);
- NSLog(@" identifier:%@",achievement.identifier);
- }
- }];
- }
- }
其中该函数的参数中identifier是你成就的ID, percent是该成就完成的百分比
6.2读取一个成就
方法一:得到所有的成就
- - (void) loadAchievements
- {
- NSMutableDictionary *achievementDictionary = [[NSMutableDictionary alloc] init];
- [GKAchievement loadAchievementsWithCompletionHandler:^(NSArray *achievements,NSError *error)
- {
- if (error == nil) {
- NSArray *tempArray = [NSArray arrayWithArray:achievements];
- for (GKAchievement *tempAchievement in tempArray) {
- [achievementDictionary setObject:tempAchievement forKey:tempAchievement.identifier];
- NSLog(@" completed:%d",tempAchievement.completed);
- NSLog(@" hidden:%d",tempAchievement.hidden);
- NSLog(@" lastReportedDate:%@",tempAchievement.lastReportedDate);
- NSLog(@" percentComplete:%f",tempAchievement.percentComplete);
- NSLog(@" identifier:%@",tempAchievement.identifier);
- }
- }
- }];
- }
函数中NSArray返回的是你的所有成就ID.
方法二:根据ID获取成就
- - (GKAchievement*) getAchievementForIdentifier: (NSString*) identifier
- {
- NSMutableDictionary *achievementDictionary = [[NSMutableDictionary alloc] init];
- GKAchievement *achievement = [achievementDictionary objectForKey:identifier];
- if (achievement == nil)
- {
- achievement = [[[GKAchievement alloc] initWithIdentifier:identifier] autorelease];
- [achievementDictionary setObject:achievement forKey:achievement.identifier];
- }
- return [[achievement retain] autorelease];
- }
6.3获取成就描述和图片
在自定义界面中,玩家需要一个成就描述,以及该成就的图片,Game Center提供了该功能.当然,你也可以自己在程序中完成,毕竟玩家不可能时刻处于在线状态.
- - (NSArray*)retrieveAchievmentMetadata
- {
- //读取成就的描述
- [GKAchievementDescription loadAchievementDescriptionsWithCompletionHandler:
- ^(NSArray *descriptions, NSError *error) {
- if (error != nil)
- {
- // process the errors
- NSLog(@"读取成就说明出错");
- }
- if (descriptions != nil)
- {
- // use the achievement descriptions.
- for (GKAchievementDescription *achDescription in descriptions) {
- NSLog(@"1..identifier..%@",achDescription.identifier);
- NSLog(@"2..achievedDescription..%@",achDescription.achievedDescription);
- NSLog(@"3..title..%@",achDescription.title);
- NSLog(@"4..unachievedDescription..%@",achDescription.unachievedDescription);
- NSLog(@"5............%@",achDescription.p_w_picpath);
- //获取成就图片,如果成就未解锁,返回一个大文号
- /*
- [achDescription loadImageWithCompletionHandler:^(UIImage *p_w_picpath, NSError *error) {
- if (error == nil)
- {
- // use the loaded p_w_picpath. The p_w_picpath property is also populated with the same p_w_picpath.
- NSLog(@"成功取得成就的图片");
- UIImage *aImage = p_w_picpath;
- UIImageView *aView = [[UIImageView alloc] initWithImage:aImage];
- aView.frame = CGRectMake(50, 50, 200, 200);
- aView.backgroundColor = [UIColor clearColor];
- [[[CCDirector sharedDirector] openGLView] addSubview:aView];
- }else {
- NSLog(@"获得成就图片失败");
- }
- }];
- */
- }
- }
- }];
- return nil;
- }
如果你不主动使用注释中的方法,那么你得到的description中不会有图片,这样可以减少网络的使用,尽量少下载东西.当使用注释中的代码时,如果成就已经解锁,则返回该成就的图标,如果没有解锁,则返回一个大问号,至于未解锁图标是否可以自定义,我找寻的结果好像是不可以.