带有SBJson框架的活动指示器

时间:2013-08-25 17:07:34

标签: objective-c json login connection activity-indicator

我目前正在开发一个使用基本登录页面检查用户是否有权访问该应用的应用。我在this tutorial的帮助下创建了登录部分,this frame work使用MBProgressHUD从简单的Web脚本中使用用户名和密码。

我希望其他人可能已经使用它或者可以帮助我处理我的问题。我不想显示活动指示器,我使用{{3}}作为活动指示器。

所以我已经对它进行了实验,但是当应用程序连接到URL时无法显示活动指示器。我已经使用登录过程模拟了一些错误的连接,但是当应用程序连接到URL时,活动指示器不会显示。它只显示错误,唯一显示成功加载的是登录按钮按下状态为“活动”(蓝色突出显示),直到加载完成。

所以这是我的代码,当用户输入用户名和密码并点击登录按钮时运行:

// Login button
- (IBAction)loginBtnClicked:(id)sender
{
    // Show the activity indicator
    [HUD showUIBlockingIndicatorWithText:@"Loggar in..."];

    @try {
        if([[userNameTxtField text] isEqualToString:@""] || [[passwordTxtField text] isEqualToString:@""] ) {

            // No username or password entered
            [self alertStatus:@"Du måste ange användarnamn och lösenord" :@"Något gick fel!"];

            // Hide activity indicator
            [HUD hideUIBlockingIndicator];

        } else {

            NSString *post =[[NSString alloc] initWithFormat:@"username=%@&password=%@",[userNameTxtField text],[passwordTxtField text]];
            NSLog(@"PostData: %@",post);

            NSURL *url=[NSURL URLWithString:@"http://www.nebulon.se/json/sendus/jsonlogin.php"];

            NSData *postData = [post dataUsingEncoding:NSASCIIStringEncoding allowLossyConversion:YES];

            NSString *postLength = [NSString stringWithFormat:@"%d", [postData length]];

            NSMutableURLRequest *request = [[NSMutableURLRequest alloc] init];
            [request setURL:url];
            [request setHTTPMethod:@"POST"];
            [request setValue:postLength forHTTPHeaderField:@"Content-Length"];
            [request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
            [request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
            [request setHTTPBody:postData];

            //[NSURLRequest setAllowsAnyHTTPSCertificate:YES forHost:[url host]];

            NSError *error = [[NSError alloc] init];
            NSHTTPURLResponse *response = nil;
            NSData *urlData=[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];

            NSLog(@"Response code: %d", [response statusCode]);
            if ([response statusCode] >=200 && [response statusCode] <300){

                NSString *responseData = [[NSString alloc]initWithData:urlData encoding:NSUTF8StringEncoding];
                NSLog(@"Response ==> %@", responseData);

                SBJsonParser *jsonParser = [SBJsonParser new];
                NSDictionary *jsonData = (NSDictionary *) [jsonParser objectWithString:responseData error:nil];
                NSLog(@"%@",jsonData);
                NSInteger success = [(NSNumber *) [jsonData objectForKey:@"success"] integerValue];
                NSLog(@"%d",success);

                if(success == 1){

                    // Login success, grant user access to app
                    NSLog(@"Login SUCCESS");

                    [self loginSuccess];

                    // Hide activity indicator
                    [HUD hideUIBlockingIndicator];

                    // Store username
                    NSString *userName = [userNameTxtField text];
                    NSUserDefaults *UserDefaults = [NSUserDefaults standardUserDefaults];
                    [UserDefaults setObject:userName forKey:@"userName"];
                    [UserDefaults synchronize];

                    [self dismissViewControllerAnimated:NO completion:nil];

                } else {

                    // Login error
                    NSString *error_msg = (NSString *) [jsonData objectForKey:@"error_message"];
                    [self alertStatus:error_msg :@"Inloggningen misslyckades"];
                    [self loginFailed];

                    // Hide activity indicator
                    [HUD hideUIBlockingIndicator];
                }

            } else {

                // Login error
                if (error) NSLog(@"Error: %@", error);
                [self alertStatus:@"Ingen nätverksuppkoppling hittades." :@"Ett fel har inträffat!"];
                [self loginFailed];

                // Hide activity indicator
                [HUD hideUIBlockingIndicator];
            }
        }
    }
    @catch (NSException * e) {

        // Login error
        NSLog(@"Exception: %@", e);
        [self alertStatus:@"Inloggningen misslyckades." :@"Ett fel har inträffat!"];
        [self loginFailed];

        // Hide activity indicator
        [HUD hideUIBlockingIndicator];
    }
}

1 个答案:

答案 0 :(得分:0)

我认为问题是由于使用了sendSynchronousRequest:returningResponse:error:

此方法将阻止主/ UI线程,因此在方法返回之前HUD实际上没有机会显示,此时代码继续并且HUD被隐藏。

我认为您应该考虑使用异步请求。 并实施NSURLConnection委托方法。

编辑:添加了代码示例。

假设您的目标是iOS 5及更高版本,您可以使用以下代码段,该代码段利用sendAsynchronousRequest:queue:completionHandler:和GCD的块。

NSOperationQueue *backgroundQueue = [[NSOperationQueue alloc] init];
[NSURLConnection sendAsynchronousRequest:request
                                   queue:backgroundQueue
                       completionHandler:^(NSURLResponse *response, NSData *data, NSError *error) {

                           // NOTE: This block is called on the background queue.

                           // Use GCD to get back onto the main thread
                           dispatch_async(dispatch_get_main_queue(), ^{
                               // This block will process the response and data on the main thread
                           });

                       }];

移植现有代码以使用此机制实际上是非常少的工作。如果您不知道块的工作方式,则应阅读the documentation,因为它们是一种非常强大的语言功能,并且正在越来越多的Apple和第三方框架中使用。

我还建议您暂时离开第三方网络库,直到您了解可能导致此类问题的细微差别为止。