RuntimePlayer-自定义WKWebView上Menu的Item

以后每用Runtime解决一个问题,就会记录在这个文集里。

我将以真实的使用场景为大家讲述Runtime的各种用法,让Runtime真正的投入到生产中。

合辑demo Github地址

Update!!!

最新测试。iOS11上苹果明智的修复了这个问题。现在可以使用常规的方式来定义Menu的item,和UITextView一样简单。

需求:

控制WKWebView长按弹出的Menu的Item。

具体的情况是,需求希望只出现“复制”“定义”而屏蔽系统自带的“共享”。

RuntimePlayer-自定义WKWebView上Menu的Item_第1张图片
demo1.png

解决:

在UITextView上这个问题简单的一逼。只要继承UITextView,在子类实现canPerformAction:withSender:就可实现。

代码大概如下。

@interface WELTextView : UITextView
@end

@implementation WELTextView

- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {
    if ([NSStringFromSelector(action) isEqualToString:@"copy:"]) {
        return YES;
    }
    return NO;
}

@end

然而,在WKWebView上却不可以。

如果需要刨根问底,就要先讲明白弹出的Menu到底是什么,以及它是怎么工作的。由于Menu本身就有坑,我决定之后单开一篇文章说这个,这里我们先形而上的想象一下。

我们可以想象Menu有个delegate,在显示的时候调用delegate的canPerformAction:withSender:来判断哪些item的需要显示的。TextView把delegate设置为自己,所以可以通过重写canPerformAction:withSender:来屏蔽,而WKWebView弹出的Menu,它的delegate并不是自己WKWebView,所以我们即使重写了canPerformAction:withSender:也无法实现该效果。

为了验证,我们可以看到WKWebView并没有实现相关的方法(copy:等),而TextView是现实了相关方法的。

于是我猜测,WKWebView必定持有了某个对象,然后它实现了copy:等方法。

找到这个对象的方法有许多,比如以WKWebView为根,属性为子,遍历这颗树,看看谁实现了copy:。(记得遍历的时候去重)代码就不上了,因为我并不是通过这种方式找到的:)

最终,可以找到WKContentView这个类。钩一下这个类的canPerformAction:withSender:就可以实现需求。

核心代码如下:

        Method  m = class_getInstanceMethod(NSClassFromString(@"WKContentView"), NSSelectorFromString(@"canPerformAction:withSender:"));
        
        class_addMethod(NSClassFromString(@"WKContentView"), NSSelectorFromString(@"wel_canPerformAction:withSender:"), (IMP)wel_canPerformAction, method_getTypeEncoding(m));
        
        Method m1 = class_getInstanceMethod(NSClassFromString(@"WKContentView"),NSSelectorFromString(@"canPerformAction:withSender:"));
        Method m2 = class_getInstanceMethod(NSClassFromString(@"WKContentView"), NSSelectorFromString(@"wel_canPerformAction:withSender:"));
        method_exchangeImplementations(m1,m2);

代码就是常规的RuntimeHook,没什么值得说的。完整代码见github

你可能感兴趣的:(RuntimePlayer-自定义WKWebView上Menu的Item)