面向 29 的 WebView 不显示内容

sna*_*msm 10 android webview android-10.0

我有一个带有此最新声明的应用程序:

compileSdkVersion 29
buildToolsVersion '29.0.2'
useLibrary 'org.apache.http.legacy'

defaultConfig {
    minSdkVersion 19
    targetSdkVersion 29
    ...
Run Code Online (Sandbox Code Playgroud)

而且我WebView在应用程序中几乎没有s - 全尺寸ActivityFragment版本以及固定尺寸的广告(比如 300x300dp,水平居中)View放置在带有本机小部件的列表中。目标更新后,最后一个不再加载(只有这个,所有其他都在工作),但是当我在线更改(恢复)时:

    targetSdkVersion 28
Run Code Online (Sandbox Code Playgroud)

它和更新前一样工作。WebView此广告的整个设置如下所示:

webView = findViewById(R.id.ad_banner_webview);
webView.setLayerType(Build.VERSION.SDK_INT <= 19 ?
            WebView.LAYER_TYPE_SOFTWARE : // on older devices non-fullactivity webview is blinking/glitching
            WebView.LAYER_TYPE_HARDWARE, null);
webView.setBackgroundColor(Color.TRANSPARENT);
final String userAgent = AppInfo.getInstance(webView.getContext()).getUserAgent();
webView.getSettings().setUserAgentString(userAgent);
webView.getSettings().setJavaScriptEnabled(true);
webView.getSettings().setLoadWithOverviewMode(true);
webView.getSettings().setDomStorageEnabled(true);
webView.getSettings().setTextZoom(100);
webView.getSettings().setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);
webView.setVerticalScrollBarEnabled(false);
webView.setHorizontalScrollBarEnabled(false);
webView.setScrollBarStyle(View.SCROLLBARS_INSIDE_OVERLAY);
webView.setWebChromeClient(new WebChromeClient());
webView.setWebViewClient(new CustomWebViewClient() {
    @Override
    public void onPageFinished(WebView view, String url) {
        super.onPageFinished(view, url);
        if (!isDummy)
            webViewContainer.setTag(System.currentTimeMillis());
    }
});

//disabling long touch - text not selectable
webView.setOnLongClickListener(v -> true);
webView.setLongClickable(false);
webView.setHapticFeedbackEnabled(false);
Run Code Online (Sandbox Code Playgroud)

请注意,API 19(最低支持)webViewsetLayerType(WebView.LAYER_TYPE_SOFTWARE)调用并始终工作,无论目标如何......

问题很简单:这种行为的原因是什么以及如何解决这个问题?

我在DOC中查看目标更新后可能会改变我的应用程序行为的内容,但没有关于WebView. 此外,我还发现了一些关于渲染变化的隐藏信息WebView-所谓的 Trichrome,但仍然不知道它是否相关 - 我没有任何特殊的日志/输出和任何线索,并且问题存在于 5.0 到10,在这些我正在使用LAYER_TYPE_HARDWARE

sna*_*msm 33

原来这种加载内容的方式很重要……我们有loadUrl,loadData和 中的loadDataWithBaseURL方法WebView,所有三个都以非常不同的方式处理给定的输入数据

在我的情况下,问题在于loadData与此类似的代码:

<body style="margin: 0; padding: 0"><div class='banner-wrap'>        
<iframe class='html5-iframe' src='https://example.com/html5/123.html#https://example-ads.com/deliver/ad.php?param=123__destination=https%3A%2F%2Fwww.anothersite.com%2F' width='300' height='300'>
</iframe>    
<a href='https://example.com/html5/123.html#https://example-ads.com/deliver/ad.php?param=123__destination=https%3A%2F%2Fwww.anothersite.com%2F' target='_blank'>
</a>    
</div></body>
Run Code Online (Sandbox Code Playgroud)

请注意,url 是 html 转义的(存在 %),但有一个例外:url 中的#。URL iframe 中存在此字符将不会加载内容...并且在定位 <=28 时正在加载。当使用loadUrlwith-#-url 给出这个时,无论目标如何,一切正常。但我正在适应已经存在的广告分发系统,并且不会为应用程序更改它。我也认为操纵给定的 HTML 代码来提取srcurl 不是一种正确的方法(系统可能有一天会以某种方式升级)......

我现在的解决方案是使用base64并设置相同的编码WebView

String base64version = Base64.encodeToString(htmlData.getBytes(), Base64.DEFAULT);
webView.loadData(base64version, "text/html; charset=UTF-8", "base64");
Run Code Online (Sandbox Code Playgroud)