Developing Apps for iOS

Recently several high-profile developers and teams shared their list of tools that helped them make apps for iPhone and iPad. I’m not as experienced with app development as they are but I though it might be useful for some new developers to see what I use to make my apps.

Obviously I use Xcode extensively for writing and compiling code. In a previous article I discussed how I turned on most of the compiler options so that my code is more robust. I’m still not fluent in ‘find and replace’ operations and Xcode crashes on me when I try to duplicate files, so I use BBEdit a lot for tweaking the code. I’ll take existing .m and .h, dup them in the finder and then use BBEdit to make global changes so that the new method name is changed everywhere in the file. I also use BBEdit to edit the .sql files that contain my data. Which brings me to the next most important program I use.

Most of the apps that I wrote are based on existing CDs or web pages. We used Filemaker Pro when we made the wordlists for the original CDs but we’ve switched to a web-based editing format instead. We have a bunch of PHP scripts that edit data in MySQL databases. It allows distributed editing of the content and we have a script that takes the data from MySQL and writes it out to an Xcode readable .sql file. We use a Firefox plug-in, SQLite Manager to verify the SQLite files that our scripts produce and occasionally we edit them as well.

Our graphic designer likes Flash so we use an older version to make the icons and graphics in the game. All of the photos were originally edited in Photoshop and we use it occasionally if we need to adjust the photos so they look better on devices. Because we have hundreds or thousands of images in our apps, we turned off automatic compression in the compiler and use ImageOptim to reduce the file size of the images and graphics. It makes a significant difference in the app size as I discussed in this article. I use Acorn to do quick edits to icons and to make placeholders. I had the free version but upgraded when they had a sale. I don’t do much in the way of photo editing, but if I did I’d use Graphic Converter. It has tons of filters and features for massaging photos and is only $40.

We do a lot of moving files back and forth from the server to the desktop and use Cyberduck for SFTP transfer. We used to have our own server but now we use a VPS hosted at Linode and are very happy with it—not to mention the fact that it costs $25/month (including backups) rather than $200.

We have a bunch of bash scripts for renaming sound files and Flash file exports and use Renamer 4 to rename files.

Since almost all of our apps rely extensively on sounds, we spent a lot of time in SoundStudio recording, cutting, and cleaning up the sounds. Audacity has matured enough that we use it on machines that we don’t have a SoundStudio license for or when we hire extra people to do massive amounts of sound cutting. We use a simple bash script to convert the files from .aiff to .m4a format.

From time to time we need to use formatted HTML in our apps. For example, we might want to highlight the target word in a sentence. We’ll export the sentences from the database and use Bean to color the target word red. Then export the text to HTML, clean up in BBEdit, and import to MySQL.

The import, export, and editing of the database is done with PHPMyAdmin and occasionally command line imports.

Unlike other teams, we don’t really need much communication. The little communication we do is by email and occasional in-person meetings.

I keep track of what needs to be done in a simple list in Notes.

That’s about it. I just bought a Mac Mini for around $1,000 since my Laptop won’t run Mountain Lion. Our total investment in software is $99 per year for the Xcode developer license and $25/month for the server. I’ve been using BBEdit forever so I don’t know how much I’ve paid for it, but a new copy is $49. You can get a free copy of TextWrangler from them that does most of what BBEdit does. Most of the rest of the programs that we use are either fee or cheap. We’ve donated a couple hundred dollars to Cyberduck and Audacity since they are so incredibly useful. We have old versions of Photoshop and Flash but if you were starting from scratch there are lots of replacements that are inexpensive. Acorn and Pixelmator work just as well as Photoshop for most or our graphics. If you are on a budget, you can easily get started on app development for less that $1,000. But even if you have money to burn, you don’t need to spend more than $2,000 to get all the tools you need.

Finding substrings in iOS

It one point I experimented with using titles on my answer buttons to keep track of which answer was selected. I switched to tags since they are integers and they are easier to use in this particular implementation. In any case, I though it interesting to show how to get a substring from a string. My titles are of the form, ‘Answer1’, ‘Answer2’, … ‘Answer16′. I really only need the last one or two digits. I pass an NSString into the delegate method from a button press.


- (void)checkBoxTapped:(UIButton *)buttonPressed {
    [[self delegate] quizAnswered:buttonPressed.currentTitle];
}

Then I grab either the last digit or the last two digits from the string.


NSString *answer = [selectedAnswer substringWithRange:NSMakeRange(6, 1)]; // wrongAnswer - wA
if (selectedAnswer.length == 8) wA = [selectedAnswer substringWithRange:NSMakeRange(6, 2)];

I was doing a string comparison to decide which question was answered,


if ([answer isEqualToString:@"1"] || 
    [answer isEqualToString:@"2"] || 
    [answer isEqualToString:@"3"] || 
    [answer isEqualToString:@"4"] ) {
    // do something
}

What I should have done is convert the string to an int and then do a simple comparison.


NSInteger answerAsInt = [answer intValue];
if ([answerAsInt < 5 ) {
    // do something
}

Null objects and NSArrays


- (void)displayWithParentView:(UIView *)parentView
                    withItems:(NSArray *)items
                 withOldItems:(NSArray *)oldItems  {
    
    NSString *pictR1C1Name = items[0];
    NSString *pictR1C2Name = items[1];
    NSString *pictR2C1Name = items[2];
    NSString *pictR2C2Name = items[3];
    
    UIButton *oldR1C1Button = (oldItems[0]  == (id)[NSNull null]) ? nil : oldItems[0];
    UIButton *oldR1C2Button = (oldItems[1]  == (id)[NSNull null]) ? nil : oldItems[1];
    UIButton *oldR2C1Button = (oldItems[2]  == (id)[NSNull null]) ? nil : oldItems[2];
    UIButton *oldR2C2Button = (oldItems[3]  == (id)[NSNull null]) ? nil : oldItems[3];

This is a snippet of the code I use to put pictures on the screen. The old pictures slide off the screen and the new ones slide on. When the screen is first initialized the oldItems array has not yet been initialized. So it is null. I can’t put a null object into a button though, so I test to see if there is an null object in the array and then set the button to nil. Otherwise, I set the button to the button that has been passed in from the array.

Likewise, I can’t put null objects into an array—recall that the way to define an array is


NSArray *newArray = [NSArray arrayWithObjects:object1, object2, nil];

The first nil object terminates the array. So to add nil objects to the array you need to first convert the nil to an object.

So you basically do the reverse of the code above to convert nil to an null object.


object1 = (object1  == nil) ? [NSNull null] : object1;
NSArray *newArray = [NSArray arrayWithObjects:object1, object2, nil];

Donating Platelets

I’ve been giving blood for a while now and it’s a pretty easy and painless thing to do. The blood bank used to be on the way to Taco Bell and the bank, so I’d stop by every eight weeks on my lunch hour. The giving blood part only takes about 15 minutes but the questionnaire and waiting after giving means that about an hour total is used up. I switched to doing an R2 donation a few years ago because it meant that I only had to go every 16 weeks and the amount of time actually donating is only about 45 minutes, so I saved a half hour of time. A few months ago they were desperately short of platelets and asked if I could give. I didn’t have anything else to do that day so I went home and got my laptop and donated.

It’s a little bit uncomfortable, but mostly because you have to stay in one place for 2 1/2 – 3 hours. When the blood leaves your body and goes to the machine, it feels just like when you are giving blood. In other words, no sensations at all. But when they put the red blood cells back in you get a weird sensation, almost like having a couple glasses of wine. Not unpleasant but weird. You can donate every two weeks, but that’s a bit too often for me, so I go once a month unless they have a critical shortage.

If you haven’t given blood before or haven’t done an R2, I’d recommend starting slow and working up. If you have given blood before, go for it. The platelets are used in patients undergoing chemotherapy or who have had organ transplants.

Here’s Flat Stanley and me donating.

Flat Stanley and me donating platelets

Things I can’t remember – Permissions

We recently migrated our sites to Linode. Since we switched distros from Gentoo to Ubuntu we had to add the users in by hand. That meant that the permissions on the old system didn’t match the permissions on the new system. In most cases, I want the person who owns the site to be the owner of the files and the group for the site be able to read and write the files. I usually put myself, artists, and system maintainers into the group. Everyone else, including Apache, gets read access. To do this manually is a real pain. Fortunately, there is an easy way.

To change the permissions on the directories go to the site you want to change and use:


sudo find . -type d -exec chmod 775 {} \;

To change the permissions on the files go to the site you want to change and use:


sudo find . -type f -exec chmod 664 {} \;

You need sudo since you probably aren’t the owner of the files. That’s it. The commands work recursively from wherever they are started.