13

我必须删除此栏作为此处的链接,但对于 iOS 7,此代码不起作用。

4

4 回答 4

26

我们使用一些 Objective C 运行时技巧来删除这个栏。

我们有一个类,它有一个方法:

@interface _SwizzleHelper : NSObject @end

@implementation _SwizzleHelper

    -(id)inputAccessoryView
    {
        return nil;
    }

@end

一旦我们有了一个想要从中删除栏的 Web 视图,我们就迭代它的滚动视图的子视图并获取UIWebDocumentView该类。然后,我们动态地将上面创建的类的超类设置为子视图的类(UIWebDocumentView - 但我们不能提前说,因为这是私有 API),并将子视图的类替换为我们的类。

#import "objc/runtime.h"    

-(void)__removeInputAccessoryView
{
    UIView* subview;

    for (UIView* view in self.scrollView.subviews) {
        if([[view.class description] hasPrefix:@"UIWeb"])
            subview = view;
    }

    if(subview == nil) return;

    NSString* name = [NSString stringWithFormat:@"%@_SwizzleHelper", subview.class.superclass];
    Class newClass = NSClassFromString(name);

    if(newClass == nil)
    {
        newClass = objc_allocateClassPair(subview.class, [name cStringUsingEncoding:NSASCIIStringEncoding], 0);
        if(!newClass) return;

        Method method = class_getInstanceMethod([_SwizzleHelper class], @selector(inputAccessoryView));
        class_addMethod(newClass, @selector(inputAccessoryView), method_getImplementation(method), method_getTypeEncoding(method));

        objc_registerClassPair(newClass);
    }

    object_setClass(subview, newClass);
}

相当于 Swift 3.0 中的上述内容:

import UIKit
import ObjectiveC

var swizzledClassMapping = [AnyClass]()

extension UIWebView {
    func noInputAccessoryView() -> UIView? {
        return nil
    }

    public func removeInputAccessoryView() {
        var subview: AnyObject?

        for (_, view) in scrollView.subviews.enumerated() {
            if NSStringFromClass(type(of: view)).hasPrefix("UIWeb") {
                subview = view
            }
        }

        guard subview != nil else {
            return
        }

        //Guard in case this method is called twice on the same webview.
        guard !(swizzledClassMapping as NSArray).contains(type(of: subview!)) else {
            return;
        }

        let className = "\type(of: subview!)_SwizzleHelper"
        var newClass : AnyClass? = NSClassFromString(className)

        if newClass == nil {
            newClass = objc_allocateClassPair(type(of: subview!), className, 0)

            guard newClass != nil else {
                return;
            }

            let method = class_getInstanceMethod(type(of: self), #selector(UIWebView.noInputAccessoryView))
            class_addMethod(newClass!, #selector(getter: UIResponder.inputAccessoryView), method_getImplementation(method), method_getTypeEncoding(method))

            objc_registerClassPair(newClass!)

            swizzledClassMapping += [newClass!]
        }

        object_setClass(subview!, newClass!)
    }
}
于 2013-09-27T03:33:53.987 回答
1

我根据@bjhomer的这篇博文制作了一个 cocoapod。

您可以替换 inputaccessoryview 而不仅仅是隐藏它。我希望这会帮助有同样问题的人。

https://github.com/lauracpierre/FA_InputAccessoryViewWebView

你可以在这里找到 cocoapod 页面。

于 2015-09-30T18:54:25.670 回答
0

我遇到了这个很棒的解决方案,但我还需要找回 inputAccessoryView。我添加了这个方法:

- (void)__addInputAccessoryView {

    UIView* subview;

    for (UIView* view in self.scrollView.subviews) {

        if([[view.class description] hasSuffix:@"SwizzleHelper"])
            subview = view;
    }

    if(subview == nil) return;

    Class newClass = subview.superclass;

    object_setClass(subview, newClass);
}

它似乎确实按预期工作,没有副作用,但我无法摆脱我的裤子着火的感觉。

如果您希望 Leo Natan 的解决方案使用 WKWebView 而不是 UIWebView,只需将前缀从“UIWeb”更改为“WKContent”。

于 2015-05-12T14:48:37.453 回答
0

我创建了一个要点来完成此操作: https ://gist.github.com/kgaidis/5f9a8c7063b687cc3946fad6379c1a66

这是一个UIWebView类别,您所做的就是更改customInputAccessoryView属性:

@interface UIWebView (CustomInputAccessoryView)

@property (strong, nonatomic) UIView *customInputAccessoryView;

@end

您可以将其设置为nil以将其删除,也可以在其上设置新视图以对其进行更改。

请记住,这也使用私有 API,因此使用风险自负,但似乎很多应用程序仍在做类似的事情。

于 2016-05-05T13:10:17.270 回答