在 UIWebView 中禁用用户选择

Posted

技术标签:

【中文标题】在 UIWebView 中禁用用户选择【英文标题】:Disabling user selection in UIWebView 【发布时间】:2011-08-25 02:39:39 【问题描述】:

我有一个应用程序,我在其中将内容加载到 UIWebView 并呈现。我不能完全禁用用户交互,因为我希望用户能够单击链接。我只需要禁用用户选择。我在 Internet 的某个地方找到了您可以使用的地方:

document.body.style.webkitUserSelect='none';

我尝试将其插入为

[self.contentView stringByEvaluatingjavascriptFromString:@"document.body.style.webkitUserSelect='none';"]; 

webViewDidFinishLoad:

但是,它不起作用。我仍然可以在 WebView 中选择和复制文本。

任何想法可能出了什么问题?

更新:这似乎只从 ios 4.3 开始发生

【问题讨论】:

【参考方案1】:

我可以确认以下代码适用于 iOS 5.0 - 8.0。

- (void)webViewDidFinishLoad:(UIWebView *)webView 
    // Disable user selection
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitUserSelect='none';"];
    // Disable callout
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitTouchCallout='none';"];

也适用于 iOS 9 及更高版本。这是快速代码:

func webViewDidFinishLoad(webView: UIWebView) 
    // Disable user selection
    webView.stringByEvaluatingJavaScriptFromString("document.documentElement.style.webkitUserSelect='none'")!
    // Disable callout
    webView.stringByEvaluatingJavaScriptFromString("document.documentElement.style.webkitTouchCallout='none'")!

【讨论】:

在 iOS 7 上确认! 注意:可以通过连接两个字符串将两个调用组合为一个。 在 iOS 9.x 上,长按后屏幕顶部会出现一个空的呼出,即使在上面使用时也是如此。 如何使用上面的代码..我只知道JS;不知道如何编辑目标 c 代码或更改 phonegap 插件..如果有人可以提供帮助 与 iOS 9 & 10 完美的人一起工作!【参考方案2】:
    let longPress:UILongPressGestureRecognizer = UILongPressGestureRecognizer(target: nil, action: nil)
    longPress.minimumPressDuration = 0.2
    webView.addGestureRecognizer(longPress)

只需将此代码添加到您的 viewDidLoad()。用户可以点击链接,但不能复制内容。

【讨论】:

【参考方案3】:

对我来说,我打算通过LongPressGestureUIWebView 获取图像的NSData

但是放大镜和复制/粘贴/剪切总是发生在我的函数执行之前。

我发现了这个:

这意味着,放大镜和复制/粘贴/剪切需要0.5s才能执行,所以如果你的func可以在0.49s内执行,DONE!

self.longPressPan.minimumPressDuration = 0.3

【讨论】:

这适用于我在 iOS 9 + Xcode 7 GM 中,我尝试的其他方法在 iOS 7&8 中才有用【参考方案4】:

使用网页视图交互功能

   webView.userInteractionEnabled = false

对我有用

PS:当您希望用户可以再次与 webview 交互时,请记住启用交互

【讨论】:

欢迎来到***。答案真的为这个老问题增加了价值吗?请查看How to Answer。 考虑扩展您的答案(提供一些小代码示例)以增加您的答案的价值。此外,包含“它对我有用”的答案并不能真正激发信心 - 最好按原样提供答案,然后在有人要求澄清您的答案时进行故障排除。【参考方案5】:

一周的出色工作的结果!如果您想在许多页面上保存鼠标事件和用户输入,则所有其他答案都不正确。

1) Swizzle 方法(通过rentzsch/jrswizzle 库):

[NSClassFromString(@"UIWebDocumentView") jr_swizzleMethod:@selector(canPerformAction:withSender:) withMethod:@selector(myCanPerformAction:withSender:) error:nil];

NSObject+myCanPerformAction.h:

@interface NSObject (myCanPerformAction)

- (BOOL)myCanPerformAction:(SEL)action withSender:(id)sender;

@end

NSObject+myCanPerformAction.m:

#import "NSObject+myCanPerformAction.h"

@implementation NSObject (myCanPerformAction)

- (BOOL)myCanPerformAction:(SEL)action withSender:(id)sender 
    if (action == @selector(copy:)) 
        return [self myCanPerformAction:action withSender:sender];
    
    if (action == @selector(paste:)) 
        return [self myCanPerformAction:action withSender:sender];
    
    return NO;


@end

2) 将 UIWebView 放在 UIView 上并添加代码:

    UITapGestureRecognizer* singleTap = [[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(handleSingleTap:)] autorelease];
    singleTap.numberOfTapsRequired = 2;
    singleTap.numberOfTouchesRequired = 1;
    singleTap.delegate = self;
    [self.view addGestureRecognizer:singleTap];

还有这个:

- (void)handleSingleTap:(UIGestureRecognizer*)gestureRecognizer 
    return;


- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer 
    if ([otherGestureRecognizer isKindOfClass:[UITapGestureRecognizer class]]) 
        UITapGestureRecognizer *gesture = (UITapGestureRecognizer *)otherGestureRecognizer;
        if (gesture.numberOfTapsRequired == 2) 
            [otherGestureRecognizer.view removeGestureRecognizer:otherGestureRecognizer];
        
    
    return YES;

【讨论】:

对这应该做什么的一些解释会很好。特别是,我很困惑为什么你的手势被称为 singleTap 但需要两次点击。【参考方案6】:

我可以确认这绝对适合你。

<style type="text/css">
  *:not(input):not(textarea) 
   -webkit-user-select: none; /* disable selection/Copy of UIWebView */
   -webkit-touch-callout: none; /* disable the IOS popup when long-press on a link */
          
</style>

如果你想只禁用锚按钮标签,请使用这个。

    a -webkit-user-select: none; /* disable selection/Copy of UIWebView */
   -webkit-touch-callout: none; /* disable the IOS popup when long-press on a link */
     

【讨论】:

【参考方案7】:

TPoschel 的答案是正确的,但在我看来,顺序很重要。

// this works - locks selection and callout
- (void)webViewDidFinishLoad:(UIWebView *)webView 
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitUserSelect='none';"];
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitTouchCallout='none';"];


// this doesn't work - locks only callout
- (void)webViewDidFinishLoad:(UIWebView *)webView 
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitTouchCallout='none';"];
    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitUserSelect='none';"];

【讨论】:

【参考方案8】:

给出的第一个解决方案对我来说非常有效......直到我将 .pdf 加载到我的 UIWebView 中。

加载 .doc 文件效果很好,但加载 .pdf 导致以下代码行不再具有所需的效果,并且在用户长按时再次弹出复制/定义菜单。

    [webView stringByEvaluatingJavaScriptFromString:@"document.documentElement.style.webkitUserSelect='none';"];

在又一次拉扯头发之后,我在 Johnny Rockex 上找到了这个答案,它就像一个冠军。 UIWebView without Copy/Paste when displaying PDF files

非常感谢他提供了这个易于实施的天才解决方案!

【讨论】:

【参考方案9】:

以下是一些禁用选择的方法:

将以下内容添加到您的移动网络文档中

<style type="text/css">
* 
    -webkit-touch-callout: none;
    -webkit-user-select: none; /* Disable selection/copy in UIWebView */

</style>

以编程方式加载以下 Javascript 代码:

NSString * jsCallBack = @"window.getSelection().removeAllRanges();";    
[webView stringByEvaluatingJavaScriptFromString:jsCallBack];

禁用复制/粘贴用户菜单:

- (BOOL)canPerformAction:(SEL)action withSender:(id)sender 
    
    if (action == @selector(copy:) ||
        action == @selector(paste:)||
        action == @selector(cut:)) 
    
        return _copyCutAndPasteEnabled;
    
    return [super canPerformAction:action withSender:sender];

【讨论】:

使用类别扩展UIWebView Engin,第一个在 Mobile Safari,iOS 4.3.1 中工作。但是 WrightCS 忘记添加选择器了。要将其应用于所有元素,请使用星号,例如:* /*css go here*/ 添加样式非常适合删除在我的 PhoneGap 应用程序内点击链接时出现的菜单。 canPerformAction 不会禁用剪切、复制、粘贴菜单。并选择,全选菜单。如何解决这个问题? -webkit-touch-calloutremoveAllRanges 中断用户输入。不能使用所有提供的方法:( 将这些无选择样式放在* 全局范围内的问题在于它会阻止用户在 Web 表单中输入。我发现通过放置在 body 标记内得到了更好的结果。【参考方案10】:

我在 android / iPhone 的网络应用程序(与 Trigger.IO 一起打包)中使用此技术,发现它仅适用于 :not() 伪类的链接语法:

*:not(input):not(textarea) 
-webkit-user-select: none; /* disable selection/Copy of UIWebView */
    -webkit-touch-callout: none; /* disable the IOS popup when long-press on a link */


【讨论】:

我正在使用 jQuery Mobile 和 Phonegap Build,这对我有用。我第一次尝试,*:not(input,textarea) -webkit-touch-callout: none; -webkit-user-select: none;,但这对我没有用。谢谢!【参考方案11】:

我喜欢 WrightsCS 解决方案,但我会使用它,以便用户仍然可以对输入使用复制、粘贴和选择操作

<style type="text/css">
*:not(input,textarea) 
    -webkit-touch-callout: none;
    -webkit-user-select: none; /* Disable selection/Copy of UIWebView */

</style>

【讨论】:

如果你要这样做,别忘了 textarea!【参考方案12】:

我不确定设置是如何完成的,但是为什么在调用 viewWillDisappear 时不清除 pasteBoard。也许在你的 appDelegate.m 中类似:

[UIPasteboard generalPasteboard].string = nil;

这将确保用户复制的任何数据都无法粘贴到应用程序之外。

另外,就像 Engin 所说,您可以在包含 uiwebview 的控制器类中覆盖 canPerformSelector 方法。

【讨论】:

这是最好的解决方案之一。这可以在 - (void)applicationDidEnterBackground:(UIApplication *)application 事件处理程序中设置。这可以确保没有数据从应用程序中流出。 @Krishan,您是否 100% 确定也会停止截屏? 我喜欢这个解决方案,但是由于它在应用程序处于活动状态时停留在粘贴板上,因此在后台线程中运行的另一个(恶意)应用程序可以访问通用粘贴板,而其中有数据。跨度> 这是否也会从应用程序中删除数据?也许在出现时获取数据并将其设置回消失时会更好

以上是关于在 UIWebView 中禁用用户选择的主要内容,如果未能解决你的问题,请参考以下文章

如何禁用 UIWebview 中的放大镜?

从后台获取加载 UIWebView

将 UIWebView 上的 ScreenEdgePanGesture 传递给父 PageViewController

禁用放大 UIWebView 但保留其他用户交互

禁用UIWebView中的链接?

UIWebView - 在网页上启用右键单击