我已经在
swift中复制了
iAdSuite with Storyboards,并且iAd在所有视图控制器上都可以正常工作,除了当调用委托方法bannerViewDidLoadAd(banner:ADBannerView!)时,横幅视图不会立即出现.我必须旋转模拟器或转到另一个标签,以便它出现.
我认为这是Objective-C中Apple示例代码的一部分,我无法“翻译”为swift?这可能导致iAd一旦加载就不会异步显示…
+ (BannerViewManager *)sharedInstance { static BannerViewManager *sharedInstance = nil; static dispatch_once_t oncetoken; dispatch_once(&oncetoken,^{ sharedInstance = [[BannerViewManager alloc] init]; }); return sharedInstance; }
这是我的BannerViewManager类,带有ADBannerViewDelegate方法.
class BannerViewManager: NSObject,ADBannerViewDelegate { // MARK: - Shared instance static let sharedInstance = BannerViewManager() var bannerView: ADBannerView! var bannerViewControllers: NSMutableSet! override init() { super.init() bannerView = ADBannerView(adType: .Banner) bannerViewControllers = NSMutableSet() bannerView.delegate = self } func addBannerViewController(controller: BannerViewController) { bannerViewControllers.addobject(controller) } func removeBannerViewController(controller: BannerViewController) { bannerViewControllers.removeObject(controller) } // MARK: - Ad banner view delegate methods func bannerViewDidLoadAd(banner: ADBannerView!) { NSLog("bannerViewDidLoadAd") for bannerViewController in bannerViewControllers { (bannerViewController as! BannerViewController).updateLayout() } } func bannerView(banner: ADBannerView!,didFailToReceiveAdWithError error: NSError!) { NSLog("didFailToReceiveAdWithError %@",error); for bannerViewController in bannerViewControllers { (bannerViewController as! BannerViewController).updateLayout() } } func bannerViewActionShouldBegin(banner: ADBannerView!,willLeaveApplication willLeave: Bool) -> Bool { NSNotificationCenter.defaultCenter().postNotificationName(BannerViewActionWillBegin,object: self) return true } func bannerViewActionDidFinish(banner: ADBannerView!) { NSNotificationCenter.defaultCenter().postNotificationName(BannerViewActionDidFinish,object: self) } }
编辑
我已设法使用this post on SO将上面的Objective-C代码转换为swift但我仍然遇到同样的问题. iAd一旦加载就不会出现.必须对视图进行某种更改才能实现,无论是旋转设备还是点击另一个选项卡等等……
所以现在我的sharedInstance看起来像这样:
static var sharedInstance: BannerViewManager { struct Static { static var oncetoken: dispatch_once_t = 0 static var instance: BannerViewManager? = nil } dispatch_once(&Static.oncetoken) { Static.instance = BannerViewManager() } return Static.instance! }
解决方法
我想我弄清楚问题是什么.它与我初始化BannerViewManager类的共享实例的方式无关.事实上,无论是使用dispatch_once还是使用新的swift 2.0方式,它都会产生完全相同的行为.
调用bannerViewDidLoadAd时导致iAd无法立即显示的事实是viewDidLayoutSubviews实际上从未被调用过!
我不得不在这里和那里添加一些打印语句.在我的代码和Apple示例代码中都得出了这个结论.
因此setNeedsLayout而不是layoutIfNeeded强制触发viewDidLayoutSubviews