Connect to Twitter with OAuth

Project Setup

This tutorial will use a simple application called “TwitterRush” to demonstrate Twitter OAuth integration for the iPhone. By downloading the TwitterRush application, you will be able to precisely follow all steps in this tutorial. However, if you already have an iPhone project that you would like to connect with the Twitter API, you should still be able to follow along in your own code with only slight modifications.

In addition to TwitterRush or your own project, you will also need to download Ben Gottlieb’s Twitter-OAuth-iPhone project available on GitHub.

Step 1: Copy the Twitter+OAuth Folder

After downloading and unarchiving the Twitter-OAuth-iPhone library, drag the folder entitled “Twitter+OAuth” into the “Other Sources” folder of the Xcode Groups & Files pane. Be sure to check the “Copy items into destination group’s folder (if needed)” option and click “Add.”

Twitter API Guide 1

Trying to compile and run your application now will result in A LOT of errors (140 at the time of this writing with iOS SDK 4.1). Not to worry: we will easily fix all of them in Step 2.

Step 2: Add the libxml2 Library

Right click on the “Frameworks” folder in the Xcode Groups & Files pane and select Add -> Existing Frameworks. Scroll to near the bottom of the list presented and select “libxml2.dylib” then click “Add.”

Twitter API Guide 2

After adding the library to your project, you will need to modify the “header search paths” setting in your project’s build configuration. To do this, select the project file in the “Groups & Files” pane, right click, and select “Get Info.”

Twitter API Guide 3

Next, select the “Build” tab in the Project Info window and scroll down to the “Header Search Paths” setting. Double click this setting and then click the “+” button in the bottom left of the pop-up dialogue to add a new search path. Click the “recursive” check box, double click the “Path” field,  and add the following dynamic path:

  1. $(SDKROOT)/usr/include/libxml2 
$(SDKROOT)/usr/include/libxml2
Twitter API Guide 4

Press CMD + Y or select Build > Build and Debug from the Xcode menubar and you should now be able to build the application without any compile errors!

Step 3: Declare the NSXMLParserDelegate Protocol

While you are now able to compile and run the application without any errors, there are a number of warnings related to changes in the iOS4 SDK and the NSXMLParserDelegate protocol. You will need to explicitly declare that MGTwitterStatusesParser.h and MGTwitterXMLParser.h conform to this protocol in order to prevent these warnings from occurring.

To do so, open the MGTwitterStatusesParser.h file and modify the @interface declaration by declaring the NSXMLParserDelegate protocol like so:

  1. @interface MGTwitterStatusesParser : MGTwitterXMLParser <NSXMLParserDelegate> { 
@interface MGTwitterStatusesParser : MGTwitterXMLParser <NSXMLParserDelegate> {

Now do the same for MGTwitterXMLParser.h, modifying the @interface declaration to read:

  1. @interface MGTwitterXMLParser : NSObject <NSXMLParserDelegate> { 
@interface MGTwitterXMLParser : NSObject <NSXMLParserDelegate> {

You should now be able to smoothly compile the application without generating any errors or warnings! We’re now ready to begin integrating the Twitter-OAuth-iPhone library with our code.

Step 4: Import SA_OAuthTwitterController.h & Declare SA_OAuthTwitterEngine

We now need to begin importing the library classes that we will use to connect with the Twitter API. Open TwitterRushViewController.h and modify the code to read as follows:

  1. #import <UIKit/UIKit.h> 
  2. #import "SA_OAuthTwitterController.h" 
  3.  
  4. @class SA_OAuthTwitterEngine; 
  5.  
  6. @interface TwitterRushViewController : UIViewController <UITextFieldDelegate, SA_OAuthTwitterControllerDelegate> 
  7.     IBOutlet UITextField     *tweetTextField; 
  8.  
  9.     SA_OAuthTwitterEngine    *_engine; 
  10.  
  11. @property(nonatomic, retain) IBOutlet UITextField *tweetTextField; 
  12.  
  13. -(IBAction)updateTwitter:(id)sender;  
  14.  
  15. @end 
#import <UIKit/UIKit.h>
#import "SA_OAuthTwitterController.h"

@class SA_OAuthTwitterEngine;

@interface TwitterRushViewController : UIViewController <UITextFieldDelegate, SA_OAuthTwitterControllerDelegate>
{
IBOutlet UITextField *tweetTextField;

SA_OAuthTwitterEngine *_engine;
}

@property(nonatomic, retain) IBOutlet UITextField *tweetTextField;

-(IBAction)updateTwitter:(id)sender;

@end

On line 2, we import the SA_OAuthTwitterController class for use within our view controller. On line 4, we forward declare the SA_OAuthTwitterEngine class so we can declare an instance of that class in the @interface without actually importing the header file. On line 6 we declare the SA_OAuthTwitterControllerDelegate protocol -this will allow us to easily respond to Twitter API events later. Finally, on line 10 we declare the _engine object as an instance of the SA_OAuthTwitterEngine class.

Now switch to the TwitterRushViewController.m file. Import the SA_OAuthTwitterEngine class that we just forward declared in the class interface:

  1. #import "SA_OAuthTwitterEngine.h" 
#import "SA_OAuthTwitterEngine.h"

Because the SA_OAuthTwitterControllerDelegate only contains optional method declarations, at this point you should again be able to compile and run your application without any errors or warnings.

Step 5: Define Your Twitter API OAuth Credentials

In order to gain OAuth access to the Twitter API, you will need to first create a Consumer Key and a Secret Key for Twitter to be able to identify and authenticate your application. You can do this from the Twitter web site by logging into your account and navigating to the app registration form. When going through the registration process, be sure to specify “client” as the application type, check the “Yes, use Twitter for login” box, and select “Read & Write” as the default access type to enable your iPhone app to post tweets on behalf of your users.

After you have registered your application and Twitter has generated your application credentials, add the following to TwitterRushViewController.m above the class @implementation:

  1. #define kOAuthConsumerKey        @"Your consumer key here"         //REPLACE With Twitter App OAuth Key 
  2. #define kOAuthConsumerSecret    @"Your consumer secret here"     //REPLACE With Twitter App OAuth Secret 
#define kOAuthConsumerKey        @"Your consumer key here"         //REPLACE With Twitter App OAuth Key
#define kOAuthConsumerSecret @"Your consumer secret here" //REPLACE With Twitter App OAuth Secret

We will use these constants momentarily when we instantiate our _engine object.

Step 6: Launch the Twitter Login Screen

For our use-case, we want to initialize the _engine object when our ViewController is created and then display the Twitter OAuth login screen as soon as the view controller finishes loading. To initialize the _engine object, modify the viewDidAppear method to read as follows:

  1. - (void)viewDidAppear: (BOOL)animated { 
  2.  
  3.     if(!_engine){ 
  4.         _engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self]; 
  5.         _engine.consumerKey    = kOAuthConsumerKey; 
  6.         _engine.consumerSecret = kOAuthConsumerSecret; 
  7.     } 
  8.  
- (void)viewDidAppear: (BOOL)animated {

if(!_engine){
_engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self];
_engine.consumerKey = kOAuthConsumerKey;
_engine.consumerSecret = kOAuthConsumerSecret;
}

}

Now go ahead and release the _engine object in our view controller’s dealloc method:

  1. - (void)dealloc { 
  2.     [_engine release]; 
  3.     [tweetTextField release]; 
  4.     [super dealloc]; 
- (void)dealloc {
[_engine release];
[tweetTextField release];
[super dealloc];
}

After our view finishes loading, we want to immediately launch the Twitter login screen. To do so, you will need to again modify the viewDidAppear method like so:

  1. - (void)viewDidAppear: (BOOL)animated { 
  2.  
  3.     if(!_engine){ 
  4.         _engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self]; 
  5.         _engine.consumerKey    = kOAuthConsumerKey; 
  6.         _engine.consumerSecret = kOAuthConsumerSecret; 
  7.     } 
  8.  
  9.     UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self]; 
  10.  
  11.     if (controller){ 
  12.         [self presentModalViewController: controller animated: YES]; 
  13.     } 
  14.  
- (void)viewDidAppear: (BOOL)animated {

if(!_engine){
_engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self];
_engine.consumerKey = kOAuthConsumerKey;
_engine.consumerSecret = kOAuthConsumerSecret;
}

UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self];

if (controller){
[self presentModalViewController: controller animated: YES];
}

}

If you run the application now, you’ll see that we are successfully presenting the Twitter login screen whenever our custom view is displayed. However, there is one major problem with this setup: the login screen will always be displayed when the view appears, even if the user has already logged in. We need to add a conditional that will only display this control if the user has not yet been connected via OAuth.

To do this, add the following conditional before displaying the view:

  1. if(![_engine isAuthorized]){ 
  2.     UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self]; 
  3.  
  4.     if (controller){ 
  5.         [self presentModalViewController: controller animated: YES]; 
  6.     } 
	if(![_engine isAuthorized]){
UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self];

if (controller){
[self presentModalViewController: controller animated: YES];
}
}

The isAuthorized method will return a boolean value of TRUE if we have an OAuth authentication token. So, this conditional simply tests whether we do not have authorization, and then displays the Twitter login when needed.

For the isAuthorized method to work, we also need to add the following SA_OAuthTwitterEngineDelegate protocol methods responsible for storing our OAuth authentication token after the first login:

  1. //============================================================================================================================= 
  2. #pragma mark SA_OAuthTwitterEngineDelegate 
  3. - (void) storeCachedTwitterOAuthData: (NSString *) data forUsername: (NSString *) username { 
  4.     NSUserDefaults          *defaults = [NSUserDefaults standardUserDefaults]; 
  5.  
  6.     [defaults setObject: data forKey: @"authData"]; 
  7.     [defaults synchronize]; 
  8.  
  9. - (NSString *) cachedTwitterOAuthDataForUsername: (NSString *) username { 
  10.     return [[NSUserDefaults standardUserDefaults] objectForKey: @"authData"]; 
//=============================================================================================================================
#pragma mark SA_OAuthTwitterEngineDelegate
- (void) storeCachedTwitterOAuthData: (NSString *) data forUsername: (NSString *) username {
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];

[defaults setObject: data forKey: @"authData"];
[defaults synchronize];
}

- (NSString *) cachedTwitterOAuthDataForUsername: (NSString *) username {
return [[NSUserDefaults standardUserDefaults] objectForKey: @"authData"];
}

Step 7: Post Updates to Twitter

In what is perhaps the simplest step of the entire process, add the following single line of code to updateTwitter, our custom IBAction method, to actually post an update to Twitter:

  1. [_engine sendUpdate:tweetTextField.text]; 
[_engine sendUpdate:tweetTextField.text];

Voila! You should now be posting updates to your Twitter feed. However, we aren’t quite finished yet. What happens if our application fails to post the update? What if we wanted to present a confirmation view if the tweet is posted successfully? Thankfully, the TwitterEngineDelegate protocol has two methods defined for just this purpose.

Add the following code to TwitterRushViewController.m:

  1. //============================================================================================================================= 
  2. #pragma mark TwitterEngineDelegate 
  3. - (void) requestSucceeded: (NSString *) requestIdentifier { 
  4.     NSLog(@"Request %@ succeeded", requestIdentifier); 
  5.  
  6. - (void) requestFailed: (NSString *) requestIdentifier withError: (NSError *) error { 
  7.     NSLog(@"Request %@ failed with error: %@", requestIdentifier, error); 
//=============================================================================================================================
#pragma mark TwitterEngineDelegate
- (void) requestSucceeded: (NSString *) requestIdentifier {
NSLog(@"Request %@ succeeded", requestIdentifier);
}

- (void) requestFailed: (NSString *) requestIdentifier withError: (NSError *) error {
NSLog(@"Request %@ failed with error: %@", requestIdentifier, error);
}

You can see that the application will now log success and failure messages to the console depending on what happens after we click the “Tweet” button. This behavior can be easily modified to match the needs of your own apps.

Conclusion

If you have followed the step-by-step instructions above, you should now be able to post status updates to Twitter on behalf of your users!

The full TwitterRushViewController.h file should now look like this:

  1. #import <UIKit/UIKit.h> 
  2. #import "SA_OAuthTwitterController.h" 
  3.  
  4. @class SA_OAuthTwitterEngine; 
  5.  
  6. @interface TwitterRushViewController : UIViewController <UITextFieldDelegate, SA_OAuthTwitterControllerDelegate> 
  7. {  
  8.  
  9.     IBOutlet UITextField *tweetTextField; 
  10.  
  11.     SA_OAuthTwitterEngine *_engine;  
  12.  
  13.  
  14. @property(nonatomic, retain) IBOutlet UITextField *tweetTextField; 
  15.  
  16. -(IBAction)updateTwitter:(id)sender;  
  17.  
  18. @end 
#import <UIKit/UIKit.h>
#import "SA_OAuthTwitterController.h"

@class SA_OAuthTwitterEngine;

@interface TwitterRushViewController : UIViewController <UITextFieldDelegate, SA_OAuthTwitterControllerDelegate>
{

IBOutlet UITextField *tweetTextField;

SA_OAuthTwitterEngine *_engine;

}

@property(nonatomic, retain) IBOutlet UITextField *tweetTextField;

-(IBAction)updateTwitter:(id)sender;

@end

The full TwitterRushViewController.m file should read:

  1. #import "TwitterRushViewController.h" 
  2. #import "SA_OAuthTwitterEngine.h" 
  3.  
  4. /* Define the constants below with the Twitter
  5.    Key and Secret for your application. Create
  6.    Twitter OAuth credentials by registering your
  7.    application as an OAuth Client here: http://twitter.com/apps/new
  8. */ 
  9.  
  10. #define kOAuthConsumerKey               @"Your Key Here"        //REPLACE With Twitter App OAuth Key 
  11. #define kOAuthConsumerSecret            @"Your Secret Here"    //REPLACE With Twitter App OAuth Secret 
  12.  
  13. @implementation TwitterRushViewController 
  14.  
  15. @synthesize tweetTextField;  
  16.  
  17. #pragma mark Custom Methods 
  18.  
  19. -(IBAction)updateTwitter:(id)sender 
  20.     //Dismiss Keyboard 
  21.     [tweetTextField resignFirstResponder]; 
  22.  
  23.     //Twitter Integration Code Goes Here 
  24.     [_engine sendUpdate:tweetTextField.text]; 
  25.  
  26. #pragma mark ViewController Lifecycle 
  27.  
  28. - (void)viewDidAppear: (BOOL)animated { 
  29.  
  30.     // Twitter Initialization / Login Code Goes Here 
  31.     if(!_engine){ 
  32.         _engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self]; 
  33.         _engine.consumerKey    = kOAuthConsumerKey; 
  34.         _engine.consumerSecret = kOAuthConsumerSecret; 
  35.     } 
  36.  
  37.     if(![_engine isAuthorized]){ 
  38.         UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self]; 
  39.  
  40.         if (controller){ 
  41.             [self presentModalViewController: controller animated: YES]; 
  42.         } 
  43.     }    
  44.  
  45.  
  46. - (void)viewDidUnload { 
  47.     [tweetTextField release]; 
  48.     tweetTextField = nil; 
  49.  
  50. - (void)didReceiveMemoryWarning { 
  51.     [super didReceiveMemoryWarning]; 
  52.  
  53. - (void)dealloc { 
  54.     [_engine release]; 
  55.     [tweetTextField release]; 
  56.     [super dealloc]; 
  57.  
  58. //============================================================================================================================= 
  59. #pragma mark SA_OAuthTwitterEngineDelegate 
  60. - (void) storeCachedTwitterOAuthData: (NSString *) data forUsername: (NSString *) username { 
  61.     NSUserDefaults          *defaults = [NSUserDefaults standardUserDefaults]; 
  62.  
  63.     [defaults setObject: data forKey: @"authData"]; 
  64.     [defaults synchronize]; 
  65.  
  66. - (NSString *) cachedTwitterOAuthDataForUsername: (NSString *) username { 
  67.     return [[NSUserDefaults standardUserDefaults] objectForKey: @"authData"]; 
  68.  
  69. //============================================================================================================================= 
  70. #pragma mark TwitterEngineDelegate 
  71. - (void) requestSucceeded: (NSString *) requestIdentifier { 
  72.     NSLog(@"Request %@ succeeded", requestIdentifier); 
  73.  
  74. - (void) requestFailed: (NSString *) requestIdentifier withError: (NSError *) error { 
  75.     NSLog(@"Request %@ failed with error: %@", requestIdentifier, error); 
  76. @end 
#import "TwitterRushViewController.h"
#import "SA_OAuthTwitterEngine.h"

/* Define the constants below with the Twitter
Key and Secret for your application. Create
Twitter OAuth credentials by registering your
application as an OAuth Client here: http://twitter.com/apps/new
*/

#define kOAuthConsumerKey @"Your Key Here" //REPLACE With Twitter App OAuth Key
#define kOAuthConsumerSecret @"Your Secret Here" //REPLACE With Twitter App OAuth Secret

@implementation TwitterRushViewController

@synthesize tweetTextField;

#pragma mark Custom Methods

-(IBAction)updateTwitter:(id)sender
{
//Dismiss Keyboard
[tweetTextField resignFirstResponder];

//Twitter Integration Code Goes Here
[_engine sendUpdate:tweetTextField.text];
}

#pragma mark ViewController Lifecycle

- (void)viewDidAppear: (BOOL)animated {

// Twitter Initialization / Login Code Goes Here
if(!_engine){
_engine = [[SA_OAuthTwitterEngine alloc] initOAuthWithDelegate:self];
_engine.consumerKey = kOAuthConsumerKey;
_engine.consumerSecret = kOAuthConsumerSecret;
}

if(![_engine isAuthorized]){
UIViewController *controller = [SA_OAuthTwitterController controllerToEnterCredentialsWithTwitterEngine:_engine delegate:self];

if (controller){
[self presentModalViewController: controller animated: YES];
}
}

}

- (void)viewDidUnload {
[tweetTextField release];
tweetTextField = nil;
}

- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
}

- (void)dealloc {
[_engine release];
[tweetTextField release];
[super dealloc];
}

//=============================================================================================================================
#pragma mark SA_OAuthTwitterEngineDelegate
- (void) storeCachedTwitterOAuthData: (NSString *) data forUsername: (NSString *) username {
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];

[defaults setObject: data forKey: @"authData"];
[defaults synchronize];
}

- (NSString *) cachedTwitterOAuthDataForUsername: (NSString *) username {
return [[NSUserDefaults standardUserDefaults] objectForKey: @"authData"];
}

//=============================================================================================================================
#pragma mark TwitterEngineDelegate
- (void) requestSucceeded: (NSString *) requestIdentifier {
NSLog(@"Request %@ succeeded", requestIdentifier);
}

- (void) requestFailed: (NSString *) requestIdentifier withError: (NSError *) error {
NSLog(@"Request %@ failed with error: %@", requestIdentifier, error);
}
@end

Thanks for reading this tutorial on the Twitter-OAuth-iPhone library, and a very special thanks to Ben Gottlieb, Matt Gemmell, Jon Crosby, Chris Kimpton, and Isaiah Carew. Without their hard work, implementing the Twitter API with the iPhone SDK would take many, many more steps to achieve.

http://mobile.tutsplus.com/tutorials/iphone/twitter-api-iphone/

posted @ 2011-03-24 08:42  周宏伟  阅读(1254)  评论(0编辑  收藏  举报