581Resizing an UIImage

Nice and simple Category to resize an UIImage

@interface UIImage (Resize)

- (UIImage*)scaleToSize:(CGSize)size;

@end
#import "UIImageResize.h"
#import 

@implementation UIImage (Resize)

- (UIImage*)scaleToSize:(CGSize)size {

    UIGraphicsBeginImageContext(size);

    [self drawInRect:CGRectMake(0, 0, size.width, size.height)];

    UIImage* scaledImage = UIGraphicsGetImageFromCurrentImageContext();

    UIGraphicsEndImageContext();

    return scaledImage;
}

@end

576NSDictionary and NSArray plist examples

NSDictionary





    arrayKey
    
        string1
        string2
        string3
        string4
    
    dicKey
    
        key1
        object1
        key2
        object2
        key3
        object3
    
    key2
    object2
    key3
    object3



NSArray





    string1
    string2
    string3
    
        key1
        object1
        key2
        object2
        key3
        object3
    
    string5



At the end, after </plist>, there's another CR.

569Layering one UIImage onto of another UIImage

Combining two images, especially useful, if the overlay image has an alpha value:

//
//  UIImage+Category.h
//  ImageOverlay
//
//  Created by Georg Tremmel on 29/04/2010.
//

#import 

@interface UIImage (combine)

- (UIImage*)overlayWith:(UIImage*)overlayImage;

@end

And the implementation file.

//
//  UIImage+Category.m
//  ImageOverlay
//
//  Created by Georg Tremmel on 29/04/2010.
//

#import "UIImage+Category.h"

@implementation UIImage (combine)

- (UIImage*)overlayWith:(UIImage*)overlayImage {

    // size is taken from the background image
    UIGraphicsBeginImageContext(self.size);

    [self drawAtPoint:CGPointZero];
    [overlayImage drawAtPoint:CGPointZero];

    /*
    // If Image Artifacts appear, replace the "overlayImage drawAtPoint" , method with the following
    // Yes, it's a workaround, yes I filed a bug report
    CGRect imageRect = CGRectMake(0, 0, self.size.width, self.size.height);
    [overlayImage drawInRect:imageRect blendMode:kCGBlendModeOverlay alpha:0.999999999];
    */

    UIImage *combinedImage = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();

    return combinedImage;
}

@end

An update to 334 Combining Images with UIImage & CGContext – (Offscreen drawing)

(Did I say, how much I love Categories...?)

Update I came across some strange behaviour when layering a PNG image with transparency over another image. Did not show up in the Simulator, only in iPhone 3GS (and probably also on other devices.) The base image draws fine, but the overlay image appears to be truncated and the last pixels shifted, producing some bright green artifacts. Changing

[overlayImage drawAtPoint:CGPointZero];

to

CGRect imageRect = CGRectMake(0, 0, self.size.width, self.size.height);
[overlayImage drawInRect:imageRect blendMode:kCGBlendModeOverlay alpha:1.0];

did not really help; the green artifacts remainded. It was strange though, that they did not appear in the other blendmodes. Using

CGContextDrawImage(c, imageRect, [overlayImage CGImage]);

would also work, but then the images turn up upside down. Not what I really needed. (Yes, I know, there might not be a hard fix for that, but really - it should be that complicated.)

After playing a bit more with the values, I found, that setting alpha lower than 1.0 gets rid of the display artifact:

[overlayImage drawInRect:imageRect blendMode:kCGBlendModeOverlay alpha:0.9999999];

Bug filed at Apple's Bug Report, let's see. Or maybe am I missing something here?

Anyway here they the files are, zipped and ready for download.

Post Scriptum

Test Project, showing the visual artifact in action. Only appears on the device, NOT IN THE SIMULATOR.

566Writing and Reading Preferences

Writing

CFStringRef textColorKey = CFSTR("defaultTextColor");
CFStringRef colorBLUE = CFSTR("BLUE");
CFPreferencesSetAppValue(textColorKey, colorBLUE, kCFPreferencesCurrentApplication);
CFPreferencesAppSynchronize(kCFPreferencesCurrentApplication);

Reading

CFStringRef textColorKey2 = CFSTR("defaultTextColor");
CFStringRef textColor;
textColor = (CFStringRef)CFPreferencesCopyAppValue(textColorKey2,kCFPreferencesCurrentApplication);

textColor must still be releasesed with CFRelease(textColor);

Side note As we should know by now, CFStringRef is a toll-free bridge to NSString, meaning it's possible to use a CFStringRef like a NSString.

NSLog(@"my CFStringRef %@", textColor);

More at Preferences Programming Topics for Core Foundation

565Stripping Characters from an NSString

NSString *stripped = [unstripped stringByTrimmingCharactersInSet:[NSCharacterSet characterSetWithCharactersInString:@"\n\t "]];

Well, Cocoa, a bit of too much syntactic nutrasweet here. Take a look at PHP and wheep:

$stripped = trim($unstripped);

(Yes, I am aware that's somewhat of an unfair comparion, but still...)

564NSData to NSString and vice versa

From NSData to NSString:

NSString *s = [[NSString alloc] initWithData:d encoding:NSUTF8StringEncoding];

And the other way round:

NSData *d = [s dataUsingEncoding:NSUTF8StringEncoding]:

560#import vs #include in Objective C – A quick reminder

#import is identical to #include, except that it makes sure that the same file is never included more than once. It’s therefore preferred and is used in place of #include in code examples throughout Objective-C–based documentation.

http://developer.apple.com/ mac/library/documentation/ cocoa/conceptual/ObjectiveC /Articles/ ocDefiningClasses.html

553CAAnimation and the Snapback Problem

Having previously done Core Animation only on the iPhone, I am still a beginner when it comes dealing with in on OSX.

The first thing that surprised me, is that layers have implicit animations; meaning whenever you change a value, the layer will animate to it using certain defaults. (duration = 0.25, etc.) While it is nice, I am still looking for a way to control it.

When you want to know, when an animation is finished, you need to make a CAAnimation, set the delegate and let it call animationDidStop:

CAAnimationGroup *group = [CAAnimationGroup animation];
group.delegate = self;
group.duration = 0.25;
group.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseOut];

CABasicAnimation *position = [CABasicAnimation animationWithKeyPath:@"position"];
position.fromValue = fromPointValue;
position.toValue = toPointValue;

CABasicAnimation *opacity = [CABasicAnimation animationWithKeyPath:@"opacity"];
opacity.fromValue = [NSNumber numberWithFloat:1.0];
opacity.toValue = [NSNumber numberWithFloat:0.0];

group.animations = [NSArray arrayWithObjects:position, opacity, nil];

[removeTarget addAnimation:group forKey:@"group"];

While this animates nicely, it still is not quite right. Because the layer basically does not know, that it values are changed in the animation, and therefore, after the animation stopped, the values are reset to the ones BEFORE the animation.

That's the annoying snapback.

To get rid of it, tell the layer the new values, right after you add the animation.

removeTarget.opacity = 0.0;
removeTarget.position = toPointValue;

%nbsp;

Interesting to see, that others seem to go throught the same learning experience.

532OSC to and from the iPhone with VVOSC

Update (October 2011) Opening and compiling the (original 3.1) OSCDemo project in Xcode 4.1 asks to 'modernize' it. Go ahead, modernize it. For sending from the iOS device, also make sure to set the sendiungToIP address, at line 40 of OSCDemoViewController.m. For testing, also don't forget to set the IP address in OSC Senderin the Quartz Composer patch. Here's the updated, Xcode 4.1 version of OSCDemo. Update (April 2011) Some users reported a problem where the transmission/reception speeds on newer hardware (iPhone 4, iPad) became unstable and "flaky". It seems that Bluetooth is causing some interference with OSC. Solution: Make sure Bluetooth is off when you're using this code. Thanks to Hans Tutschku for finding this solution.

Sending data to and from the iPhone via OSC to applications on other machines should not be that difficult. Unfortunatly it's not as simple as it should be, but fortunalty Mr Ray (of VIDVOX fame) is so kind to share his code and his implementation of OSC for Objective-C. There are some ancient libraries around, but let's better keep them buried where they are.

First a bit of an excursion on how to deal with external code on the iPhone.

Frameworks, Static Libraries or Inclusion? Frameworks are the preferred method of distributing shared code on OSX. The idea behind being, that only one copy of the code can be shared amongst any number of applications. But at the iPhone true multi-tasking is not possible, so shared frameworks are not possible. Heck, they are even forbidden by Apple. But of course their own UIKit Frameworks are fine.

You could convert code for reuse to a Static Library, but with iPhone Dev you might deal with the simulator and the device, so you would need to either pre-compile the library for all plattforms or create a specific project, that compile the static library on demand. More on that at 519. Frameworks are also nice, in that you only have to import them to your project and then they just work. Static Libraries need some special care, with header search paths, compiler flags, and of course putting them in a special place in ~/Library. That's fine if you develop on your own, but if you work with other people and share the code via a source code repository (like SVN), it might be difficult (or annoying) to maintain the same machine state across all the development machines.

So I resorted to including the classes directly into my project. That way they get compiled every time and I can poke around the code directly. If you really want a central source of your external classes, just don't add them to your project by copy but just by reference...

Sending OSC From the iPhone Sending OSC data from the iPhone was always quite straightforward:

- (id) init {
    NSLog(@"OSC init");

    manager = [[OSCManager alloc] init];
    [manager setDelegate:self];

    // sending
    NSString *sendingToIP = @"192.168.20.117";
    int sendingToPort = 50000;
    outPort = [manager createNewOutputToAddress:sendingToIP atPort:sendingToPort];

    // receiving
    int receivingOnPort = 51234;
    //inPort = [manager createNewInput]; // default at port 1234
    inPort = [manager createNewInputForPort:receivingOnPort];

    return self;
}

Receiving OSC At the iPhone The receiving part seems to catch many people out. Basically it's nothing more than set the right delegate, as specified in VVOSC:

// called by delegate on message
- (void) receivedOSCMessage:(OSCMessage *)m {
    NSString *address = [m address];
    OSCValue *value = [m value];
    NSString *message;

    if ([address isEqualToString:@"/mouseX"]) {
        message = [NSString stringWithFormat:@"mouseX: %i", [value intValue]];
        NSString *txt = [NSString stringWithFormat:@"%i", [value intValue]];
        [receiveLabel performSelectorOnMainThread:@selector(setText:) withObject:txt waitUntilDone:NO];
    } else if ([address isEqualToString:@"/mouseY"]) {
        message = [NSString stringWithFormat:@"mouseY: %i", [value intValue]];
    } else if ([address isEqualToString:@"/floatArray"]) {
        message = [NSString stringWithFormat:@"floatArray: %f", [value floatValue]];
    }
}

Change Notes As VVOSC is targeted at both Cocoa and Cocoa Touch, there is a compiler directive that takes care of the switching. Or at least it should take care. I was unable to get the original code to run on the iPhone, because the compiler directive IPHONE was not recognized. I am sure I am missing something there... To solve that, I change IPHONE (and !IPHONE) with TARGET_OS_IPHONE (and !TARGET_OS_IPHONE), and that did the trick.. And as this is going to be used for the iPhone only, I allowed myself the luxury of removing any code, that was not meant for execution on the iPhone OS.

Here's the original OSC demo Xcode 3.1-compatible project, hope it helps. Here's the newer, Xcode 3.2-compatible version (working with Xcode 4.1 on OSX 10.7). Drop me a line, if you have questions, suggestions or improvements.

536Adding a custom delegate

MyClassWithDelegate.h

#import 

@protocol MyClassWithDelegaet 
@optional
- (void)myDelegateMethod:(NSString*)tsst;
@end

@interface MyClassWithDelegate : NSView {
    id  delegate;
}

@property (nonatomic, assign) id  delegate;

@end

 

MyClassWithDelegate.m

#import "MyClassWithDelegate.h"

@implementation MainView

@synthesize delegate;

- (void)anyEvent:(NSEvent *)e { // an event or something like that
    // send delegate
    [[self delegate] myDelegateMethod:@"sss"];
}

@end

Thanks to Jon Sterling for his digest of Apple's 'How Delegation Works'