iAd нельзя щелкнуть в программно созданном окне - баннер ViewActionShouldBegin не вызывается - PullRequest
3 голосов
/ 20 января 2011

В приложении iOS 4.2 я программно создал окно, контроллер представления, UIView и два подпредставления: EAGLView и ADBannerView.Мой код представляет собой комбинацию стандартного EAGLView и кода Apple BasicAdBanner.

Я могу видеть баннеры iAd, но не могу нажать на них - на устройстве или в симуляторе.Контроллер представления реализует ADBannerDelegate и получает сообщения bannerViewDidLoadAd и т. Д., Но никогда не получает сообщение bannerViewActionShouldBegin.

Поведение не изменяется путем замены представления EAGL на UITextView или только с ADBannerView, так чтоПредполагая, что моя проблема заключается в программном создании контроллера окна или представления.Буду признателен за любую помощь.

Кроме того, мои просмотры не меняются и не меняются автоматически, как в примере рекламного баннера, но это менее актуальная проблема.

Code Follows-

AppDelegate:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 
{
    // Create the window programatically:
    window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];

    // let's enable multitouch and user interaction
    window.userInteractionEnabled=YES;
    window.multipleTouchEnabled=YES;
    window.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
    window.autoresizesSubviews = YES;

    controller = [GLViewController alloc];
    window.rootViewController = controller;
    [window addSubview:controller.view];
    glView = [controller.glView retain];
    [window makeKeyAndVisible];

    return YES;
}

ViewController:

- (void)loadView 
{
    self.view = [[UIView alloc] initWithFrame:[UIScreen mainScreen].bounds];
}

-(void)viewDidLoad
{
    [super viewDidLoad];

    // I'd like to get device orientation notifications.
    //[[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications]; 
    //[[NSNotificationCenter defaultCenter] addObserver:self                                        
    //                                       selector:@selector(didRotate:) name:UIDeviceOrientationDidChangeNotification object:nil];

    // the GL VIEW I want.
    glView = [[EAGLView alloc] initWithFrame:[UIScreen mainScreen].bounds];
    glView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight | UIViewAutoresizingFlexibleBottomMargin;
    glView.userInteractionEnabled=NO;
    [self.view addSubview:glView];

    // testing with a textview
    //content = [[UITextView alloc] initWithFrame:[UIScreen mainScreen].bounds];
    //content.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight | UIViewAutoresizingFlexibleBottomMargin;
    //content.userInteractionEnabled=NO;
    //[self.view addSubview:content];

    if (self.banner==NULL)
        [self createADBannerView];

    [self layoutForCurrentOrientation:NO];
}

-(void)createADBannerView
{

    // --- WARNING ---
    // If you are planning on creating banner views at runtime in order to support iOS targets that don't support the iAd framework
    // then you will need to modify this method to do runtime checks for the symbols provided by the iAd framework
    // and you will need to weaklink iAd.framework in your project's target settings.
    // See the iPad Programming Guide, Creating a Universal Application for more information.
    // http://developer.apple.com/iphone/library/documentation/general/conceptual/iPadProgrammingGuide/Introduction/Introduction.html
    // --- WARNING ---

    // Depending on our orientation when this method is called, we set our initial content size.
    // If you only support portrait or landscape orientations, then you can remove this check and
    // select either ADBannerContentSizeIdentifierPortrait (if portrait only) or ADBannerContentSizeIdentifierLandscape (if landscape only).
    NSString *contentSize;
    if (&ADBannerContentSizeIdentifierPortrait != nil)
    {
        contentSize = UIInterfaceOrientationIsPortrait(self.interfaceOrientation) 
        ? ADBannerContentSizeIdentifierPortrait : ADBannerContentSizeIdentifierLandscape;
    }
    else
    {
        // user the older sizes 
        contentSize = UIInterfaceOrientationIsPortrait(self.interfaceOrientation) 
        ? ADBannerContentSizeIdentifier320x50 : ADBannerContentSizeIdentifier480x32;
    }

    // Calculate the intial location for the banner.
    // We want this banner to be at the bottom of the view controller, but placed
    // offscreen to ensure that the user won't see the banner until its ready.
    // We'll be informed when we have an ad to show because -bannerViewDidLoadAd: will be called.
    CGRect frame;
    frame.size = [ADBannerView sizeFromBannerContentSizeIdentifier:contentSize];
    frame.origin = CGPointMake(0.0f, 0.0f);//CGRectGetMaxY(self.view.bounds)); // this can't be called until the view exists!

    // Now to create and configure the banner view
    ADBannerView *bannerView = [[ADBannerView alloc] initWithFrame:frame];
    // Set the delegate to self, so that we are notified of ad responses.
    bannerView.delegate = self;
    bannerView.hidden = YES;
    // Set the autoresizing mask so that the banner is pinned to the bottom
    bannerView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight | UIViewAutoresizingFlexibleTopMargin;
    // Since we support all orientations in this view controller, support portrait and landscape content sizes.
    // If you only supported landscape or portrait, you could remove the other from this set.

    bannerView.requiredContentSizeIdentifiers = [NSSet setWithObjects:ADBannerContentSizeIdentifierPortrait, ADBannerContentSizeIdentifierLandscape, nil];
    //support for 4.0 and 4.1: //(&ADBannerContentSizeIdentifierPortrait != nil) ?[NSSet setWithObjects:ADBannerContentSizeIdentifierPortrait, ADBannerContentSizeIdentifierLandscape, nil]:[NSSet setWithObjects:ADBannerContentSizeIdentifier320x50, ADBannerContentSizeIdentifier480x32, nil];

    // At this point the ad banner is now be visible and looking for an ad.
    self.banner = bannerView;
    [self.view addSubview:bannerView];
    [bannerView release];
}


-(void)layoutForCurrentOrientation:(BOOL)animated
{

    //CGFloat animationDuration = animated ? 0.2f : 0.0f;
    // by default content consumes the entire view area
    CGRect contentFrame = self.view.bounds;
    // the banner still needs to be adjusted further, but this is a reasonable starting point
    // the y value will need to be adjusted by the banner height to get the final position
    CGPoint bannerOrigin = CGPointMake(CGRectGetMinX(contentFrame), CGRectGetMaxY(contentFrame));
    CGFloat bannerHeight = 0.0f;

    // First, setup the banner's content size and adjustment based on the current orientation
    if(UIInterfaceOrientationIsLandscape(self.interfaceOrientation))
//    if(UIDeviceOrientationIsLandscape([[UIDevice currentDevice] orientation]))
        banner.currentContentSizeIdentifier = ADBannerContentSizeIdentifierLandscape;//(&ADBannerContentSizeIdentifierLandscape != nil) ? ADBannerContentSizeIdentifierLandscape : ADBannerContentSizeIdentifier480x32;
    else
        banner.currentContentSizeIdentifier = ADBannerContentSizeIdentifierPortrait;//(&ADBannerContentSizeIdentifierPortrait != nil) ? ADBannerContentSizeIdentifierPortrait : ADBannerContentSizeIdentifier320x50; 
    bannerHeight = banner.bounds.size.height; 

    // Depending on if the banner has been loaded, we adjust the content frame and banner location
    // to accomodate the ad being on or off screen.
    // This layout is for an ad at the bottom of the view.
    if(banner.bannerLoaded)
    {
        // HMM this is going to be a problem for a GLView..
        banner.hidden = NO;
        contentFrame.size.height -= bannerHeight;
        bannerOrigin.y -= bannerHeight;
        NSLog(@"Banner will be %f x %f, content resized to %f x %f\n", banner.frame.size.width,bannerHeight,contentFrame.size.width,contentFrame.size.height);
    }
    else
    {
        banner.hidden = YES;
        //bannerOrigin.y += bannerHeight;
        NSLog(@"Banner will be offscreen, content resized to %f x %f\n", contentFrame.size.width,contentFrame.size.height);

    }

    //EDIT: contrary to my OP, this UIViewAnimateWithDuration code was not commented out!
    // And finally animate the changes, running layout for the content view if required.
    [UIView animateWithDuration:animationDuration
                     animations:^{
                         glView.frame = contentFrame;
                         [glView layoutIfNeeded];
                         banner.frame = CGRectMake(bannerOrigin.x, bannerOrigin.y, banner.frame.size.width, banner.frame.size.height);
                     }];
}

ADBannerViewDelegate:

-(void)bannerViewDidLoadAd:(ADBannerView *)banner
{
    [self layoutForCurrentOrientation:YES];
}

-(void)bannerView:(ADBannerView *)banner didFailToReceiveAdWithError:(NSError *)error
{
    NSLog(@"Failed to receive and ad: %s %s %s\n",error.localizedFailureReason.UTF8String, error.localizedDescription.UTF8String, error.localizedRecoverySuggestion.UTF8String);

    [self layoutForCurrentOrientation:YES];
}

-(BOOL)bannerViewActionShouldBegin:(ADBannerView *)banner willLeaveApplication:(BOOL)willLeave
{
    // this indicates that the banner has been clicked.. confirm?
    NSLog(@"banner clicked: will %s application\n",willLeave?"leave":"cover");

    return YES;
}

Ответы [ 2 ]

4 голосов
/ 28 июля 2012

Просто чтобы повторить то, что сказал forksandhope,

Если поместить ADView в UIView, а затем изменить размер кадра обоих, ADView станет неактивным.

Чтобы обойти это,создайте супер UIView и добавьте к этому представлению свой UIView и ADView.

Например:

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    // initialize the content UIView and/or UIViewController
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];

    // copy the content view to a class data member
    _contentView = self.view;

    // create the super UIView
    self.view = [[UIView alloc] initWithFrame:_contentView.frame];

    // add the content view to the super view
    [self.view addSubview:_contentView];

    if (self) {
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(willBeginBannerViewActionNotification:) name:BannerViewActionWillBegin object:nil];
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(didFinishBannerViewActionNotification:) name:BannerViewActionDidFinish object:nil];
    }

    return self;
}

Затем, когда объявление загружено, добавьте ADView в супер представление:

- (void)showBannerView:(ADBannerView *)bannerView animated:(BOOL)animated
{
     // ...

        // add the AdView to the super view
        [self.view addSubview:_adBanner.adBannerView];

     // ...

}

Затем раскройте вид по своему желанию

- (void)layoutAnimated:(BOOL)animated
{
    // ...

    [UIView animateWithDuration:animated ? 0.4 : 0.0 animations:^{
        _adBanner.adBannerView.frame = _bannerFrame; 
        [self.view layoutIfNeeded];
       // update the content view AND NOT the super view
       _contentView.frame = _contentFrame;
    }];

    // ...

}
1 голос
/ 20 января 2011

Пояснение: похоже, проблема связана с кодом, пытающимся установить или анимировать изменения фрейма контента в конце -layoutForCorrentOrientation (BOOL).если я только устанавливаю расположение рамки баннера, я могу щелкнуть баннер, но если я также попытаюсь установить рамку glView и воссоздать поверхность, баннер не будет реагировать на ввод.

//EDIT: contrary to my OP, this UIViewAnimateWithDuration code was not commented out!
// And finally animate the changes, running layout for the content view if required.
//[UIView animateWithDuration:animationDuration
//                 animations:^{
//                     glView.frame = contentFrame;
//                     [glView layoutIfNeeded];
                     banner.frame = CGRectMake(bannerOrigin.x, bannerOrigin.y, banner.frame.size.width, banner.frame.size.height);
//                 }];

APENDED: В конце концов, проблема заключалась в том, что у меня UIView верхнего уровня (controller.view), в котором glView и баннер отображаются в качестве вложенных видов.Оказывается, мне нужно было позвонить

[self.view setNeedsLayout]

, чтобы этот вид верхнего уровня обновил свои границы, чтобы разрешить нажатие кнопки на баннере!(кредит переходит к ответу Эфраима на пост на кнопках UIB, не отвечающих после поворотов

...