|
To explore the tech world, though mostly focusing on iOS and Drupal development
Sometimes it isn’t the perfect that gets in the way of the good, but the revolutionary gets in the way of the evolutionary.
|
|
|
|
|
Submitted by gregh on Wed, 10/26/2011 - 20:38
|
In one of my projects, while I waited for some art work, I wanted to get ahead with my Game Center integration. This tutorial is about how to get Achievements to be displayed in the Game when the achievement is sent to Game Center. I will further explore how to get the achievement images also included. A few simple frameworks will be used.
Also, I am taking it for granted that Game Center is up and running on your app. I was able to get it up and running in a few hours using the methods found in Learning Cocos2D or in numerous tutorials around, here or here.
Once you have a user authenticating, and you have setup a few iTunes Connect achievements, you might want to display an 'Achievement Attained' message when a new achievement has been accomplished. Again, this is different than just showing the Achievements window, that shows all the achievements a player has completed. We want a little notification message displayed.
In iOS 5, Apple finally gave us a prepackaged API call that will show an achievement banner when an achievement is complete (per your code giving a 100.0% to an achievement). Here is the simplest way:
- (void)sendAchievement:(GKAchievement *)achievement {
achievement.percentComplete = 100.0; //Indicates the achievement is done
achievement.showsCompletionBanner = YES; //Indicate that a banner should be shown
[achievement reportAchievementWithCompletionHandler:
^(NSError *error) {
dispatch_async(dispatch_get_main_queue(), ^(void)
{
if (error == NULL) {
NSLog(@"Successfully sent archievement!");
} else {
NSLog(@"Achievement failed to send... will try again \
later. Reason: %@", error.localizedDescription);
}
});
}];
}
This code creates the achievement object according to your identifier (whatever you setup in iTunes), you set the completion to 100%, set the 'showsCompletionBanner' property to YES, and shoot it off to Apple. The new iOS property is 'showsCompletionBanner' which defaults to NO, but if updated to YES, will display a cute little banner with the title and description of the achievement, as shown below (yes I am using the http://www.neuroshimahex.com/ game as inspiration, so placed as my dev background. Check it out, very fun strategy game).

But this approach has a few limitations. First off, it uses the generic Game Center icon, instead of the image that I uploaded for my game achievement. Worst off, if the player earns two achievements at the same time, only one of them will be displayed, which I think is a bummer. And lastly, this is an iOS 5 only API calls, so will crash any device not running iOS5, oops!
So to repair all of these issues, we will not be using the 'showsCompletionBanner' property, but will be implementing our own notification, or more correctly using and modifying some code of people who have gone before us.
To get started, I used the great code that Type One Error demonstrated in http://www.typeoneerror.com/articles/post/game-center-achievement-notifi.... Read that article for some background if you want, and then grab the code at https://github.com/typeoneerror/GKAchievementNotification and install into your project. We will be using the GKAchievementNotification and GKAchievementHandler classes, with some updates and modifications. First off, if you are using ARC for your game, do a quick scan of the code and remove the few release, retain and autoreleases you find in that code. If you don't want to scan, just try to build after you place the files in your project and fix wherever the compiler squawks at.
The Type One Error classes will display a notification that is similar to the one that the iOS 5 call gives, but your code needs to know what the achievement title and description is. To do this you need to populated a'GKAchievementDescription' object.
One great thing about GKAchievementDescription objects is that they are already localized according to the language setting of the user (if you support that language) and thus don't have to worry about any localization issue if you use this method.
The bummer is that you can't load just one Achievement description, you have to load them all. I believe the best time to do this is when a user has authenticated Game Center on your app, you should do an async call to get them. Galdly Apple does give an API call for this, which I place in the CompletionHandler of the user authentication call.
If you are using the code by Ray Wenderlich, then you have a method like this, which I add one line to, and a new method. Also add an NSMutableDictionary * self.achievementsDescDictionary to whatever class is processing your Game Center code, which will store the achievement data for the rest of the active session.
- (void)authenticateLocalUser {
if (!gameCenterAvailable) return;
NSLog(@"Authenticating local user...");
if ([GKLocalPlayer localPlayer].authenticated == NO) {
[[GKLocalPlayer localPlayer]
authenticateWithCompletionHandler:^(NSError *error) {
if([GKLocalPlayer localPlayer].isAuthenticated){
[self retrieveAchievmentMetadata]; //Here is the new code
}
}];
}
}
//Here is the new method.
- (void) retrieveAchievmentMetadata
{
self.achievementsDescDictionary = [[NSMutableDictionary alloc] initWithCapacity:2];
[GKAchievementDescription loadAchievementDescriptionsWithCompletionHandler:
^(NSArray *descriptions, NSError *error) {
if (error != nil) {
NSLog(@"Error %@", error);
} else {
if (descriptions != nil){
for (GKAchievementDescription* a in descriptions) {
[achievementsDescDictionary setObject: a forKey: a.identifier];
}
}
}
}];
}
The 'retrieveAchievmentMetadata' method initializes the dictionary then calls to get all the Achievement descriptions for your game, and cycles through and adds them to the dictionary. This is an async call, so it shouldn't slow down the start of your game or project.
Now that we have all the titles and descriptions for all your achievements, we can modify our original code to create an iOS 4/5 friendly notification, that will also show all achievements in succession using the Type One Error code, again inside of Ray Wenderlich's code (resource links at the top and bottom of this post).
- (void)reportAchievement:(NSString *)identifier
percentComplete:(double)percentComplete {
GKAchievement* achievement = [[GKAchievement alloc]
initWithIdentifier:identifier];
achievement.percentComplete = percentComplete;
if (percentComplete == 100.0) {
//Show banners manually
GKAchievementDescription *desc = [achievementsDescDictionary objectForKey:identifier]; //Update pull achievement description for dictionary
[[GKAchievementHandler defaultHandler] notifyAchievement:desc]; //Display to user
}
[achievementsToReport addObject:achievement]; //Queue up the achievement to be sent
[self save];
if (!gameCenterAvailable || !userAuthenticated) return;
[self sendAchievement:achievement]; //Try to send achievement
}
- (void)sendAchievement:(GKAchievement *)achievement {
[achievement reportAchievementWithCompletionHandler:
^(NSError *error) {
dispatch_async(dispatch_get_main_queue(), ^(void)
{
if (error == NULL) {
NSLog(@"Successfully sent archievement!");
[achievementsToReport removeObject:achievement]; //Remove Achievement from queue.
} else {
NSLog(@"Achievement failed to send... will try again \
later. Reason: %@", error.localizedDescription);
}
});
}];
}
The additional checks to see if the achievement is 100% complete, and if so, grabs the correct achievement description object and displays it to the user. Then it continues on with telling Apple about the new achievement, and if you have a landscape game, it will look something like this.

But, I think it would be even better to also have the achievement image displayed, instead of the default image. So to do that, update the code with the notification part to this:
if (percentComplete == 100.0) {
//Show banners manually
GKAchievementDescription *desc = [achievementsDescDictionary objectForKey:identifier];
[desc loadImageWithCompletionHandler:^(UIImage *image, NSError *error) {
if (error == nil)
{
[[GKAchievementHandler defaultHandler] setImage:desc.image]; //If image found, updates the image to the achievement image.
}
[[GKAchievementHandler defaultHandler] notifyAchievement:desc];
}];
}
Since that code is inside a completion handler block, it will only execute once the handler returns. If it returns an image, then the notification is updated with that image, and then it displays to the user.

Yeah, yeah, I have temporary art in there. I told you the art was falling behind. Anyway, using this method out of the box you will notice that if you have a landscape game, that the notifications are off to the side and sideways (the Type One Error guys made it for portrait only). To fix this for landscape you need to do two things, adjust the frame and the rotation.
Find the 'notifyAchievement' method in the GKAchievementHandler class and update to:
- (void)notifyAchievement:(GKAchievementDescription *)achievement
{
GKAchievementNotification *notification = [[GKAchievementNotification alloc] initWithAchievementDescription:achievement];
notification.frame = kGKAchievementFrameStart;
notification.handlerDelegate = self;
//Adjusting rotation.
if ([[UIApplication sharedApplication] statusBarOrientation] == UIInterfaceOrientationLandscapeLeft) {
notification.transform = CGAffineTransformRotate(notification.transform, degreesToRadian(-90));
} else {
notification.transform = CGAffineTransformRotate(notification.transform, degreesToRadian(90));
}
notification.frame = kGKAchievementFrameStartLandscape; //Update the frame, you need to create this definition.
[_queue addObject:notification];
if ([_queue count] == 1)
{
[self displayNotification:notification];
}
}
Also adjust the 'animateIn' and 'animateOut' frames as needed. I found these frames to be useful:
#define kGKAchievementFrameStartLandscape CGRectMake(-53.0f, 350.0f, 104.0f, 284.0f);
#define kGKAchievementFrameEndLandscape CGRectMake(20.0f, 350.0f, 104.0f, 284.0f);
So, by using the Game Center code from Ray Wenderlich's
tutorial as a starting point, along with the Type One Error notification code, we then added adjustments so our final solution does the following:
1) Display one or more achievement notifications (the Type One Error code queues the display of many notifications)
2) Displays the notifications with images and full title and descriptions.
3) In a manner that is iOS version independent.
You determine a player had completed an achievement, sprinkle in:
[GCHelper sharedInstance] reportAchievement:identifier
percentComplete:percentComplete];
Or whatever way you access your Game Center achievements
Lastly, if you believe pulling all the achievement data early in the authentication call is a performance or data issue, you can wait and place the 'retrieveAchievmentMetadata' call until the player earns an achievement, and place in its completion handler, add the code to display the achievement notification. But there is a little extra legwork needed to save the achievement until you have retrieved all the description data. I leave it to the reader to finish off that particular use case.
Resources:
http://developer.apple.com/library/ios/#documentation/NetworkingInternet...
http://www.typeoneerror.com/articles/post/game-center-achievement-notifi...
https://github.com/typeoneerror/GKAchievementNotification
http://www.raywenderlich.com/3276/how-to-make-a-simple-multiplayer-game-...
Comment below, or with Twitter,
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Fri, 10/21/2011 - 11:50
|
It has been a week and half since the last update. Trying not to fall behind on the development updates, but the time can really fly with so many projects up in the air.
This week I have really been hitting the wall when it comes to creativity, or my lack there of. I think of myself as a creative problem solver and a good coder. But I have no sense when it comes to both music and graphic art. Thus far I have been stubbing out the layout of controls, and what the animations might need to be. But as my development task list keeps shrinking, the list of art and music assets needed keeps growing.
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Wed, 10/12/2011 - 11:06
|
I was toying with the idea of writing a post about why I went independent, but I figure that wasn't appropriate for the wider audience of iDevBlogADay.com. Check my blog for those posts. So I decided to post a more technical piece regarding cocos2d menus and how to pass arguments through the Cocos2d menu system. But I was facing two issues. One I hated rewriting similar code over and over again. Second to make menu items more reusable, I needed to be able to pass arguments to the selector of the menu, which wasn't readily apparent how to do this. Also, checking the cocos2d forums and other places I saw that many others have had this issue.
So my goal is to create a reusable button factory which allows for the passing of arguments to the selector (read check this tutorial to get the basics for Button and Text Menu Items in Cocos2d).
For starters, here is a very typical pattern for adding a text menu item:
-(void) addMenu {
//Create the label for the menu
CCLabelTTF *label = [CCLabelTTF labelWithString:@"Button Text fontName:@"Arial" fontSize:12];
//Add label to a menu item.
CCMenuItemLabel *menuItem = [CCMenuItemLabel itemWithLabel:label target:self
selector:(menuItemPressed:)];
//Add menu item to a menu.
CCMenu *menu = [CCMenu menuWithItems:menuItem, nil];
menu.position = ccp(0,0);
//Add menu to a layer.
[self addChild:menu];
}
//Method to receive button pressed action
-(void) menuItemPressed: (id)sender {
[self doSomething];
}
As you might have noticed, the argument for the menu select is automatically supplied as a CCMenuItem of some sort or another, but generically it is just an id object. The question is how do you forward some information to the menuItemPressed method, from the menu item you created, and to no have to go through the effort of creating this sequence of code over and over again.
Example, this is ugly.
//Method to receive button1 pressed action
-(void) menuItemPressed1: (id)sender {
[self doSomething: 1];
}
//Method to receive button2 pressed action
-(void) menuItemPressed2: (id)sender {
[self doSomething: 2];
}
//Method to receive button3 pressed action
-(void) menuItemPressed3: (id)sender {
[self doSomething: 3];
}
I wanted something like this.
-(void) menuItemPressed: (id)sender {
[self doSomething: sender.passedVariable];
}
Gladly, all the CCMenuItem classes and subclasses all have a userData property that you can use to assign data to like this:
menuItem.userData = someObj;
To use the userData, and to make sure there weren't warnings or error in XCode, I explicitly cast the object to what is should, and assign it to a temp variable.
Update the code a bit and it becomes:
-(void) addMenu {
//Create the label for the menu
CCLabelTTF *label = [CCLabelTTF labelWithString:@"Button Text fontName:@"Arial" fontSize:12];
//Add label to a menu item.
CCMenuItemLabel *menuItem = [CCMenuItemLabel itemWithLabel:label target:self
selector:(menuItemPressed:)];
//Add menu item to a menu.
CCMenu *menu = [CCMenu menuWithItems:menuItem, nil];
menuItem.userData = someObj; //Of type SomeObj.
menu.position = ccp(0,0);
//Add menu to a layer.
[self addChild:menu];
}
//Method to receive button pressed action
-(void) menuItemPressed: (id)sender {
(SomeObj*) tempObj = (SomeObj*) sender.userData;
[self doSomething: sender.userData];
}
So now, we have a passed variable being used in a receiver of a pressed menu item. Great. Now write this code over and over again. OR, create a button factory that does the heavy lifting for you. Since I wanted to pass the position, and have a general layer to use, this method returns a CCLayer, but you could just return a CCMenu instead.
Here is my final button factory code, and an example usage (subject to improvements in the future, I will try to keep this article up to date).
-(CCLayer*) actionTrayButtonFactory: (NSString*) text withPosition: (CGPoint) pos withUserData: (id) userData selector: (SEL) selector {
CCLayer *layer = [CCLayer node];
CCLabelTTF *label = [CCLabelTTF labelWithString:text fontName:@"Arial" fontSize:13];
label.color = ccc3(255, 255, 255); //Set your own color.
CCMenuItemLabel *menuItem = [CCMenuItemLabel itemWithLabel:label target:self
selector:selector];
menuItem.userData = [userData retain];
CCMenu *menu = [CCMenu menuWithItems:menuItem, nil];
menu.position = pos;
[layer addChild:menu];
return layer;
}
Usage:
CCLayer *layer = [self actionTrayButtonFactory: @"Button Name" withPosition:ccp(100, 200) withUserData:[NSNumber numberWithInt:1] selector:@selector(menuItemPressed:)];
[self addChild: layer];
I would use this inside a loop where I am incrementing the button name, position height and the userData passed in.
Next, I will be making a similar button factory that takes in images and will create CCMenuItemImages instead. The same format will work. Also, a further extension would be to add in another argument for button type incase you wanted to adjust the size/shape/color of the manufactured menu item.
Also, if you are making a localized app, the button factory method can update the 'labelWithString:text' portion to 'labelWithString:NSLocalizedString(text, nil)', which is what I am doing in my game.
I am not sure if I needed to use objects in the userData property, instead of just integers or floats, but that is what was need when I got it to start working. I also at times needed to place a 'retain' on the object to maintain the data. Not exactly sure why. Any help from my readers on why that is the case will help clean up this code long term.
This code will show up in the Skejo Studios Outbreak project. Check out the game development diaries. This post is a part of the iDevBlogADay series. Check out some other recent posts including Who is an Indie Dev and What's my motivation.
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Tue, 10/11/2011 - 11:28
|
Should I get this playtest?
That is a question I have heard from people before, and I even ask myself often. At times I constantly have a list of new tasks, features, or controls that need to be completed. Might as well keep my head down w/o rolling a build and actually playing the game. Also my vanity doesn't want to show to anyone an unpolished app or game.
But let's all agree. Don't do this!
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Sun, 10/09/2011 - 11:02
|
I recently resigned from my startup engineering roles to start my own thing at mentioned here. I wanted to pursue my own projects and goal, work on stuff I was really excited about, learn a few new things and I currently have the financial safety to just go for it.
I started reading www.idevblogaday.com, and followed one of the authors to their blog and came across the Indie Budget, by Doug Davies. Since I haven't produced an indie game yet (but a few New and Noteworthy Travel apps) I won't comment on budgeting, but I think I can give a few opinions regarding his opening on who an indie dev is. I will return to Doug's definition in a moment, but I wanted to visit a few others first.

|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Sat, 10/08/2011 - 16:49
|
So on iDevBlogADay.com, I came across a few posts stating the lack of motivation, or taking a few weeks off because they just didn't feel like working on their iOS project. Being only six weeks into the full time indie dev situation, I guess I haven't hit that wall yet. Or since I am not doing this nights and weekends, but full time I have a bigger downside if I lose focus since I don't have a regular salary backing me up.
I choose to go full time, leaving behind a pretty good salary. I did this so I would have the freedom and time to pursue some personal and profession goals I want. But I also have to make money long term, so this isn't just a hobby excursion.
My goals as an independent developer, and founder of new iOS studio, Skejo Studios, are two-fold: create a lasting iOS company; work on games and projects that will use myself.
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Fri, 09/30/2011 - 15:44
|
I wish I could say I spent the whole week working on this project, but past contact was extended, and I worked on my other non-game project, the Dog Park Finder. We are updating the logo branding and adding thousands more dog friendly locations with new filters and map icons. But this is a dev diary about our Outbreak project (view all posts here). Also visit Skejo Studios to see all we are doing.
For this update, instead of working on a ton of under the hood methods, I hours I did spend on I used on some UI elements. Or more precisely, stubbing out the UI.
Below is one of out UI sketches we made. Partially we were just brainstorming on what a player would need to, prioritizing frequent interaction and highlighting them above less frequent actions. Though we in no way consider the sketch as perfect, we wanted to get something on paper, and then get it in rough for into the game. This way we are getting closer to a fully playable demo, as well as interactive with device and testing user and action flow.
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Wed, 09/28/2011 - 09:25
|
When I was making the transition into iOS development from Internet engineering, I grabbed two 500 page iPhone app development books, read and worked the examples over a Christmas vacation, coded nights and weekends for three months and produced an app that Apple highlighted as New and Noteworthy, the Dog Park Finder. So, now when I decided to make the leap from iOS development to iOS game development 18 months later, I wanted to grab a great intro book and go from there.
First I had to determine the tools I would use, and landed upon Cocos2D (this isn't a game platform review, so I am not going into the pros/cons of Cocos2D or other platforms, that is a different post). Following that decision, I grabbed Learning Cocos2D by Rod Strougo and Ray Wenderlich. The other obvious choice was Learn iPhone and iPad Cocos2D Game Development. Honestly, I came across Learning Cocos2D first, which is why I bought it.
Learning Cocos2D is a great book to use as an on-ramp to the Cocos2D platform, as well a giving good best practice insight into general game development, and design pattern to follow. If are a long term game developer, well versed in texture map optimization, OpenGL programming or physics modeling, then you probably should just read the Cocos2D API documentation. But if not, Learning Cocos2D is a hands on book for getting up to speed quickly (if you are altogether new to Objective-C and iOS development, the a read through an basic iPhone programming book before working through this book).
The book uses working example of building a Space Viking game (free at the app store). Through the examples you will be shown how to make a splash screen, menu pages (including credits and options pages), how to move and animate sprites, spawn enemies, scroll the landscape, create gravity puzzles and joy ride in a mine cart across an alien landscape. Among other things. All the while, Rod and Ray skillful introduce support tools and mix in programming best practices for optimizing memory and speed up the game.
Like many of books of this nature, all the sample code viewed in the book, is also available for download. It is a great help see and interact with an entire XCode project when a concept is eluding you. Also, the authors have given full permission to use the code in any manner to help accelerate your projects.
The book's 17 chapters are broken into five sections. Instead of highlights and review each chapter's content, I will do so at the section level.
The four chapters in the 'Getting Started' section give a steps the reader from a installing Cocos2D into XCode, to simple 'Hello World' program, to making your first game scene with a hero, a villain, some animations and basic collision detection. These chapters define the basic Cocos2D vocabulary of scenes, layers and sprites, along with batching sprites for performance, definition animation sequences in plists instead of in the code, and spells out some of the workflow and time saving API calls for handling sprite movements and scene updates.
The next short section involves chapters 5 and 6 where some decent amount of computer science concepts of class hierarchy, interfaces and other concepts are written along side instructions about sprite and layer affects, generating text labels, and other font usage. The authors also help guide you to a quick debugging layer for better testing of your games.
'From Level to Game', is the third section of the book. Chapter 7 deals with Cocos2D's menu items, which can be used to create dynamic menus, directing to new scenes for viewing credits, as well as creating options for defining game settings. Chapter 8 is dedicated to all things audio. Including different tools for importing sound and asynchronous loading of sound/music for a better user experience. Lastly Chapter 9 deals with larger levels than one screen can hold. Both scrolling, and parallax scrolling are covered as well as the efficient way to create and reuse time maps.
The most cerebral of sections, is 'Physics Engines' sections. I spent a little less time working the examples of the Physics chapters 10 through 13, but Rod and Ray extensively introduce both the Box2D and Chipmunk Physics engines. Starting a bit easy with gravity, mass and simple collisions to setting up a fully actuated Viking to throw and rag-doll around, these chapters lay a solid foundations for you to explore physics in the Cocos2D world on your own.
Lastly, a ragtag section is used to wrap up the basic particle system in Cocos, while giving pointers for toll in creating your own particle effects. A chapter is also dedicated to Game Center integration, including login/sign up, creating leader boards, setting achievements, and how to save updates when an Internet connection is not available. Besides the conclusion, the last chapter deals with general performance tips, debugging and profiling guides and some quick tips to loading your sprites faster.
Too sum up, Learning Cocos2D is not just a book on how to use Cocos2D, but really a how-to guide to making your first iOS Cocos2D game with good technical architecture, with many best practices and how-to's thrown in. At over 500 pages, it isn't a quick read, but I would recommend it to anyone looking for good book for iOS Cocos2D development
Before signing off, I thought I should also introduce myself, as this is my first post for iDevBlogADay.com. My name is Greg Holsclaw, and I am the founder of Skejo Studios, a new iOS shop. We have already produced two apps for ourselves (one is the Dog Park Finderwith over 200K downloads), and two more for clients. Since completing a BS in computational mathematics, I have been creating websites and backend engineering for 7 years, and creating iOS apps for the last two years.
My hobbies sites and weekend work has blossomed enough that I recently was able to resign my engineering lead role at a successful internet startup to create Skejo Studios, and there is no looking back. Look for more posts at www.tech-wanderings.com concerning the intersection of Drupal (a web CMS) and iOS apps, as well as development diaries for the games we are developing. Until next time. -Greg
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Fri, 09/23/2011 - 13:35
|
Today, I am kicking off the development diary for the project that inspired Skejo Studios to get started. We may have a second project in parallel (to fill any down time this project might have), but most of our energy will spent on this project.
I plan on making weekly project updates, at least on any week that I spend work on a particular project. This first update reflects about two weeks of elapsed time, but less than a week's worth of dedicated time. To set expectations, some updates will be technical in nature, where others might focus more on the design, UI or business side of things.
Currently, the in-house name is Outbreak, but that is of course subject to change at any point right up to launch. We are aiming to create an iPad evolution of a table top strategy game concept. A co-operative type game where there are two to four players who are trying to beat back four regional outbreaks. Most likely we will play a zombie theme for these outbreaks, but for now we are just working on the game mechanics. This project is being built on the Cocos2d game platform.
|
|
|
|
|
|
|
|
|
|
Submitted by gregh on Thu, 09/22/2011 - 09:12
|
From my years in industry, I know that re-inventing the wheel, just so I can say it is my wheel is a huge form of waste.
|
|
|
|
|
|