iOS hook webviewdelegate && 继承关系hook

主要有两个注意事项
1.判断方法是否被交换过,已经交换过就不再重复交换

     if ([delegate respondsToSelector:@selector(secre_webview_webView:decidePolicyForNavigationAction:decisionHandler:)]) {
            TLogD(@"WKNavigationDelegate class %@ has hooked yet.", [NSString stringWithUTF8String:object_getClassName(delegateClass)]);
            return;
        }

2.判断交换的对象为方法实现的类,不要在继承子类中进行方法交换。
原因:
2.1
父类superClass 中实现了methodA
子类selfClass 中没有实现methodA
2.2
如果selfClass中进入hook,那么实际交换的是selfClass中的新方法与superClass中的methodA; 这时selfClass已经hook,但是在superClass中找不过交换的新方法,会再次hook,最终进入死循环crash
2.3
所以,我们需要在selfClass中获取到实现methodA的superClass,直接交换superClass中的methodA,所有的子类方法都能调用到交换后的方法。

        
         //直接交换实现方法的类,避免继承之间循环交换

         //遍历父类找到实现方法的父类
        Class c = [self enumerateClasses:delegateClass];
        if (!c) {
           //如果没有父类方法实现需要交换的方法,就默认交换自己类的方法
            c = delegateClass;  
        }

        [SECREMethodSwizzling swizzleMethod:@selector(webView:decidePolicyForNavigationAction:decisionHandler:) ofClass:c withMethod:@selector(secre_webview_webView:decidePolicyForNavigationAction:decisionHandler:) defaultMethod:@selector(secre_webview_default_webView:decidePolicyForNavigationAction:decisionHandler:) fromClass:WKWebView.class];

以webview为例代码实现

1.方法交换

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        // swizzling method
        [SECREMethodSwizzling swizzleMethod:@selector(setNavigationDelegate:) withMethod:@selector(secre_webview_setNavigationDelegate:) ofClass:WKWebView.class];
        TLogD(@"has exchanged setNavigationDelegate: method.");
    });
}

2.交换方法自己与需要添加处理的方法

- (void)secre_webview_setNavigationDelegate:(id<WKNavigationDelegate>)delegate {
    if (delegate) {
        //    NSLog(@"Now is using %s as -[WKWebView setNavigationDelegate:]", __PRETTY_FUNCTION__);
        [self secre_webview_swizzleMethodOfWKNavigationDelegate:delegate];
        [self secre_webview_setNavigationDelegate:delegate];
    }
}
- (void)secre_webview_swizzleMethodOfWKNavigationDelegate:(id<WKNavigationDelegate>)delegate {
    Class delegateClass = delegate.class;
    @synchronized (sc_hookLock) {
        
         //直接交换实现方法的类,避免继承之间循环交换
        Class c = [self enumerateClasses:delegateClass];
        if (!c) {
            c = delegateClass;
        }
        
        if ([delegate respondsToSelector:@selector(secre_webview_webView:decidePolicyForNavigationAction:decisionHandler:)]) {
            TLogD(@"WKNavigationDelegate class %@ has hooked yet.", [NSString stringWithUTF8String:object_getClassName(delegateClass)]);
            return;
        }
        Class wkWebViewClass = WKWebView.class;
        // exchange implementation
        [SECREMethodSwizzling swizzleMethod:@selector(webView:decidePolicyForNavigationAction:decisionHandler:) ofClass:c withMethod:@selector(secre_webview_webView:decidePolicyForNavigationAction:decisionHandler:) defaultMethod:@selector(secre_webview_default_webView:decidePolicyForNavigationAction:decisionHandler:) fromClass:wkWebViewClass];
        // 将hook过的类添加进来
        // [hookedClassNames addObject:NSStringFromClass(delegateClass)];
        TLogD(@"WKNavigationDelegate class %@ has hooked.", [NSString stringWithUTF8String:object_getClassName(delegateClass)]);
    }
}

#pragma mark - helper
//判断页面是否实现了某个sel
- (Class)enumerateClasses:(Class)delegateClass {
    SEL sel = @selector(webView:decidePolicyForNavigationAction:decisionHandler:);
    if ([self isContainSel:sel inClass:[delegateClass class]]) {
        return delegateClass;
    }
    
    Class c = [delegateClass class];
    while (c) {
        c = class_getSuperclass(c);
        if ([self isContainSel:sel inClass:[c class]]) {
            return [c class];
        }
    }
    return nil;
}
- (BOOL)isContainSel:(SEL)sel inClass:(Class)class {
    unsigned int count;
    
    Method *methodList = class_copyMethodList(class,&count);
    for (int i = 0; i < count; i++) {
        Method method = methodList[i];
        NSString *tempMethodString = [NSString stringWithUTF8String:sel_getName(method_getName(method))];
        if ([tempMethodString isEqualToString:NSStringFromSelector(sel)]) {
            return YES;
        }
    }
    return NO;
}

3.默认方法与自己需要添加的实现

#pragma mark - default Methods
- (void)secre_webview_default_webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
    decisionHandler(WKNavigationActionPolicyAllow);
}

#pragma mark - swizzling Methods
- (void)secre_webview_webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
    TLogD(@"%s: will decide whether can visit url: %@, webview url: %@", __PRETTY_FUNCTION__, navigationAction.request.URL, webView.URL);
    
    NSLog(@"URL :%@", navigationAction.request.URL);
    // do something 
    [self secre_webview_webView:webView decidePolicyForNavigationAction:navigationAction decisionHandler:decisionHandler];
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • WebView·开车指南 2016-08-31BugDev 北京市东城区首席Bug布道师开山之作,一整月交通事故血...
    53c021c38a1d阅读 4,338评论 0 1
  • View 自定义View中在onDraw()方法中可以设置padding吗?答案是不能,设置padding后,Vi...
    ElvenShi阅读 5,920评论 0 0
  • 一、WebView 谷歌提供的系统组件,用来加载和展现html网页,其采用webkit内核驱动,来实现网页浏览功能...
    闲庭阅读 11,919评论 2 12
  • 一、深复制和浅复制的区别? 1、浅复制:只是复制了指向对象的指针,即两个指针指向同一块内存单元!而不复制指向对象的...
    iOS_Alex阅读 5,306评论 1 27
  • 当你达到一定境界的时候,你所看到的和你想的都是高于别人的,名家大师亦是如此,市井平民亦是如此,特别是咱们的国画精神...
    竹本良子阅读 3,430评论 0 6

友情链接更多精彩内容