Pick a random color in iOS

In one of my methods I wanted to put randomly colored text on the screen. I put this class method in my Utilities.m class.


#define ARC4RANDOM_MAX      0x100000000
+ (UIColor *) randomColor {
    CGFloat red =  (CGFloat)arc4random()/ARC4RANDOM_MAX;
    CGFloat blue = (CGFloat)arc4random()/ARC4RANDOM_MAX;
    CGFloat green = (CGFloat)arc4random()/ARC4RANDOM_MAX;
    return [UIColor colorWithRed:red green:green blue:blue alpha:1.0];
}

To set the color, just call


label.textColor = [Utilities randomColor];

Lazy Instantiation in iOS

I had a bunch of code like this in the init for my first view controller. Paul Hegerty had mentioned lazy instantiation in his Stanford CS193 courses. And the boilerplate code in the AppDelegate for creating Managed Object Contexts and Persistent Store Coordinator uses it a log. So I understood the concept, but I hadn’t used it for my own globals. After listening to his most recent class, I decided to convert all of my init code to lazy instantiation.

One benefit of lazy instantiation is that you don’t allocate resources until you need to use the object. In my case, a better reason is that I’m not cluttering up my view controller with code that initializes global variables. In the Model, View, Controller design pattern, initialization code really doesn’t belong in the controller. But even more important, since I check for initialization in the class that creates the variable, I can’t forget to initialize the variable.

Old Code


    if (![Globals sharedInstance].showmePict ) {
        [Globals sharedInstance].showmePict = @"Either";
    }
    
    if (![Globals sharedInstance].targetSoundDelay ) {
        [[Globals sharedInstance] resetTargetDelay:TARGET_SOUND_DELAY];
    }
    

    if ( !([Globals sharedInstance].targetSound) ){
#ifdef SHOWME_TARGET_SOUND
        [[Globals sharedInstance] resetTargetSound:SHOWME_TARGET_SOUND];
#endif
    
#ifndef SHOWME_TARGET_SOUND
        [[Globals sharedInstance] resetTargetSound:@""];
#endif
    }

New Code

In the singleton for globals.


- (NSString *)showmePict {
    
    if (!_showmePict ) _showmePict = @"Either";
    return _showmePict;
}

- (NSUInteger)targetSoundDelay {
    
    if ( !_targetSoundDelay ) [self resetTargetDelay:TARGET_SOUND_DELAY];
    return _targetSoundDelay;
}

- (NSString *)targetSound {
    
    if ( !_targetSound ){
    #ifdef SHOWME_TARGET_SOUND
        [self resetTargetSound:SHOWME_TARGET_SOUND];
    #endif
        
    #ifndef SHOWME_TARGET_SOUND
        [self resetTargetSound:@""];
    #endif
    }
    return _targetSound;
}

NSNotification

I wrote most of this post for StackOverflow since I’m getting multiple UIKeyboardDidShowNotification and UIKeyboardDidHideNotification notifications when I rotate the device and I can’t figure out why. But it has a good example of how to do notifications in iOS so I thought I’d post it here.

I have an app that lets you edit the text for each picture. (it’s in a UITextView.) Most of the time I need to slide the text up so you can see it above the keyboard, then I slide it down when you are done editing. Then I do a database update to save the new text. I use notifications to tell me when the keyboard is displayed and when it goes away. It works fine on iPad when the user taps the keyboard close icon on the keyboard. It also works fine if the user swipes to the next page and iOS closes the keyboard. Since iPhones and iPods don’t have a keyboard close key, I wrote a method to close the keyboard when the picture or background is tapped.

In my class that displays the pictures I start notifications when it is initialized.


- (id)initWithParentView:(UIView *)parentview  {
    
    self = [super init];
    if (self) {
        _parentView = parentview;
        if (ALLOW_DATABASE_EDITING) [self startNotifications];
    }
    
    return self;
}

- (void)startNotifications {
    
    // Listen for keyboard appearances and disappearances
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(keyboardDidShow:)
                                                 name:UIKeyboardDidShowNotification
                                               object:nil];
    
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(keyboardDidHide:)
                                                 name:UIKeyboardDidHideNotification
                                               object:nil];
}

Note that the @selector() is the name of the method that is called when the notification center gets a notification.
The View Controller calls the hideKeyboard method in the View when the user taps on the picture.


- (void)dismissKeyboard {
    
    if (self.showArtic.keyBoardIsShowing) {
        [self.showArtic hideTheKeyboard];
    }
}
<code>

resignFirstResponder sends a notification that closes the keyboard. When it hides a notification is sent and the keyBoardDidHide method is called.
<code class='smaller'>
- (void)hideTheKeyboard {
    
    id <ShowArticDelegate> SA_delegate = _delegate;
    // Don't update the database when there is no text.
    if ( ![self.editableTextView.text isEqualToString:@""] ) {
        [SA_delegate updateTextInDatabase:self.editableTextView.text];
    }
    [self.editableTextView resignFirstResponder];
}

These methods respond to the notifications. I use block animations to move the text up and back. I also shade the background so it stands out a bit, but the picture is still visible under the text block. The self.keyBoardIsShowing flag is used in the view controller to decide whether to tell the keyboard to hide when the picture is tapped.


- (void)keyboardDidShow:(NSNotification *)notification {
    NSLog(@"keyboardDidShow called. Keyboard showing flag is %@.", self.keyBoardIsShowing ? @"YES" : @"NO");
    self.keyBoardIsShowing = YES;
    if (self.textShiftAmount > 0) {
        self.editableTextView.backgroundColor = [UIColor colorWithRed:240/255.0f green:240/255.0f blue:240/255.0f alpha:.5f];
        [self.parentView bringSubviewToFront:self.editableTextView];
        [UIView animateWithDuration:.7
                         animations:^{
                             
                             CGRect frame = self.editableTextView.frame;
                             frame.origin.y = self.pictButton.frame.origin.y + self.pictButton.frame.size.height - self.textShiftAmount;
                             self.editableTextView.frame = frame;
                             
                         }
                         completion:^(BOOL finished){
                         }];
    }
}

- (void)keyboardDidHide:(NSNotification *)notification {
    NSLog(@"keyboardDidHide called. Keyboard showing flag is %@.", self.keyBoardIsShowing ? @"YES" : @"NO");
    self.keyBoardIsShowing = NO;

    id <ShowArticDelegate> SA_delegate = _delegate;
    // Don't update the database when there is no text.
    if ( ![self.editableTextView.text isEqualToString:@""] ) {
        [SA_delegate updateTextInDatabase:self.editableTextView.text];
    }
    if (self.textShiftAmount > 0) {
        self.editableTextView.backgroundColor = [UIColor whiteColor];

        [UIView animateWithDuration:.7
                         animations:^{
                             
                             CGRect frame = self.editableTextView.frame;
                             frame.origin.y = self.pictButton.frame.origin.y + self.pictButton.frame.size.height;
                             self.editableTextView.frame = frame;
                             
                         }
                         completion:^(BOOL finished){
                             self.editableTextView.backgroundColor = [UIColor clearColor];
                         }];
    }
    
}

Programmatically move a segmented control.

I have a .xib file that I use for options. One of the options is a sound delay before I play a synthesized sound. Since synthesized sound is not available before iOS7, I figured that I’d hide the option. But that leaves a gap in the screen where the control used to be. This code moves the next segmented control up to where the original one was. Make sure you link the text label to the Files Owner by control clicking and dragging from the Files Owner to the label in Interface Builder section of Xcode.


if ( SYSTEM_VERSION_LESS_THAN_OR_EQUAL_TO(@"7.0") ) {

    // Hide the delay control
    self.chooseDelaySegmentedControl.hidden = YES;
    self.targetSoundDelayText.text = @"";

    // Move the segmented control up
    CGRect scoringTypeFrame = self.chooseScoringTypeSegmentedControl.frame;
    scoringTypeFrame.origin.y = scoringTypeFrame.origin.y - 70.0f;
    self.chooseScoringTypeSegmentedControl.frame = scoringTypeFrame;
    
    CGRect scoringTypeTextFrame = self.scoringChoiceText.frame;
    scoringTypeTextFrame.origin.y = scoringTypeTextFrame.origin.y - 70.0f;
    self.scoringChoiceText.frame = scoringTypeTextFrame;
    
} 

Oddly enough, you can’t just change the x or y value, you need to change the whole frame.

Synthesize By Default

In 2012 Apple introduced Synthesize By Default in Xcode. You no longer have to synthesize instance variables. Just declare them as you normally would in the .h file and the compiler will automatically synthesize them for you. I learned how to use Xcode before the transition so I just learned about this by accident. If you haven’t done so, watch these two videos from WWDC 2012: Session 405 – Modern Objective-C and Session 413 – Migrating to Modern Objective-C.

Synthesize By Default is not available for NSManagedObjects so if your header file contains something like this:


@interface WordList : NSManagedObject

You’ll have to synthesize.

You can still manually create setters or getters and the compiler will automatically create the other accessor.

One caveat. I stopped supporting iOS4.3 and still have code that is compatible with iOS4. Since doesn’t have the weak indicator, you need to change assign to weak, then delete the ‘__unsafe_unretained id delegate;’ line. Then you can delete the @synthesize. If you still want to support iOS4, you’ll need to synthesize these variables.

Before


@implementation PracticeSightWords
@synthesize delegate = _delegate;


@interface PracticeSightWords : UIViewController {
    
    __unsafe_unretained id <PracticeSightWordsDelegate> delegate;
}

@property (nonatomic, assign) id <PracticeSightWordsDelegate> delegate;

After


@interface PracticeSightWords : UIViewController 

@property (nonatomic, weak) id <PracticeSightWordsDelegate> delegate;

I have my compiler warnings Turned up to 11 so I needed to turn off the warnings for “Implicit Synthesized Properties” in the ‘Warnings-Objective C’ section of the build settings.