将委托分配给视图控制器以外的其他内容时未调用的SFSafariViewControllerDelegate方法

时间:2016-03-28 23:50:51

标签: ios objective-c sfsafariviewcontroller

我有一个简单的视图控制器,当点击一个链接按钮时,它会显示一个SFSafariViewController。

#import "ViewController.h"
#import "SafariDelegate.h"

@interface ViewController ()

@property(nonatomic, strong) NSString *url;
@property(nonatomic, weak) SafariDelegate *safariDelegate;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.url = @"http://www.google.com";
}

- (IBAction)linkTapped:(id)sender {
    SFSafariViewController *vc = [[SFSafariViewController alloc] initWithURL:[NSURL URLWithString:self.url]];
    SafariDelegate *safariDelegate = [[SafariDelegate alloc] init];
    vc.delegate = safariDelegate;
    [self presentViewController:vc animated:YES completion:nil];
}

@end

SafariDelegate符合SFSafariViewControllerDelegate,只需在委托方法触发时记录。

SafariDelegate.h

#import <Foundation/Foundation.h>
#import <SafariServices/SafariServices.h>

@interface SafariDelegate : NSObject <SFSafariViewControllerDelegate>

@end

SafariDelegate.m

#import "SafariDelegate.h"

@implementation SafariDelegate

- (NSArray<UIActivity *> *)safariViewController:(SFSafariViewController *)controller activityItemsForURL:(NSURL *)URL     title:(nullable NSString *)title
{
    NSLog(@"safariViewController activityItemsForURL");

    return @[];
}

- (void)safariViewControllerDidFinish:(SFSafariViewController *)controller
{
    NSLog(@"safariViewControllerDidFinish");
}


- (void)safariViewController:(SFSafariViewController *)controller didCompleteInitialLoad:(BOOL)didLoadSuccessfully
{
    NSLog(@"safariViewController didCompleteInitialLoad");
}

@end

当我点击按钮时,safariviewcontroller会正确加载页面,但从不调用委托方法(包括我点击'完成'时)。我在显示'vc'的行处设置了一个断点,并确认该委托在该点正确设置。我错过了什么?

2 个答案:

答案 0 :(得分:5)

SafariViewController没有保留其委托,因此在linkTapped:方法返回后会立即取消分配,因为没有其他任何内容可以强制引用它。

你需要在某处保持对它的强烈引用。您可能只需将当前视图控制器中的属性更改为strong并将其存储在那里。您还可以将其保存在safari视图控制器上的关联对象中。

答案 1 :(得分:2)

我也发生了同样的问题。 我已经创建了“ SafariViewController函数内部”。因此,SafariViewController的内存已清除。因此,我无法从SafariViewControllerDelegate获取触发器。

最后我找到了解决这个问题的方法。

只要您在类中创建 SFSafariViewController 。您必须将其声明为全局变量

class Foo: NSObject {
    var safariVC: SFSafariViewController!
}

然后您可以分配您的 SafariViewController

func config(_ loadUrl: URL) {

   if #available(iOS 11.0, *) {
       let config = SFSafariViewController.Configuration()
       config.barCollapsingEnabled = true
       config.entersReaderIfAvailable = true
       safariVC = SFSafariViewController(url: loadUrl, configuration: config)
   } else {
      // Fallback on earlier versions
      safariVC = SFSafariViewController(url: loadUrl)
   }
}

然后设置一个代表

self.safariVC.delegate = self

最终出现了safariViewController。

 UIApplication.shared.keyWindow?.rootViewController?.present(safariVC, animated: true, completion: nil)

现在肯定会触发safariViewController委托。

// MARK: - SFSafariViewControllerDelegate
extension Foo: SFSafariViewControllerDelegate {

    func safariViewControllerDidFinish(_ controller: SFSafariViewController) {
        print("SafariVC is dismissed")
    }
}