Android Webview没有'有时不会加载HTML

Android Webview doesn't load HTML sometimes

本文关键字:加载 HTML Webview 没有 Android      更新时间:2023-09-26

我正在尝试在Web视图中加载本地HTML内容。但有时它无法加载内容,而是显示空白屏幕。每5次加载就发生一次。

注意我的HTML内容,我正在尝试加载的是官方2048源代码。

下面是我的活动源代码

public class GameActivity extends AppCompatActivity {
private WebView mWebView;
@SuppressWarnings("ConstantConditions")
@SuppressLint({ "SetJavaScriptEnabled", "NewApi", "ShowToast"})
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    getWindow().setFlags(WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED,
            WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED);
    setContentView(R.layout.activity_game);
    Toolbar toolbar = (Toolbar) findViewById(R.id.game_toolbar);
    if (toolbar != null) {
        setSupportActionBar(toolbar);
        ActionBar actionBar = getSupportActionBar();
        if (actionBar != null) {
            actionBar.setDisplayHomeAsUpEnabled(true);
        }
    }
    // Load webview with game
    mWebView = (WebView) findViewById(R.id.mainWebView);
    WebSettings settings = mWebView.getSettings();
    String packageName = getPackageName();
    settings.setJavaScriptEnabled(true);
    settings.setDomStorageEnabled(true);
    settings.setDatabaseEnabled(true);
    settings.setRenderPriority(WebSettings.RenderPriority.HIGH);
    if (Build.VERSION.SDK_INT >= 19) {
        // chromium, enable hardware acceleration
        mWebView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
    } else {
        // older android version, disable hardware acceleration
        mWebView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
    }
    // Since the database path is automatically handled by Chromium Webkit,
    // we should not mention the db path for greater than KitKat version
    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) {
        settings.setDatabasePath("/data/data/" + packageName + "/databases");
    }
    mWebView.addJavascriptInterface(new WebInterface2048(this), "Android");
    // If there is a previous instance restore it in the webview
    if (savedInstanceState != null) {
        mWebView.restoreState(savedInstanceState);
    } else {
        mWebView.loadUrl("file:///android_asset/2048/index.html");
    }
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case android.R.id.home:
            finish();
            break;
    }
    return super.onOptionsItemSelected(item);
}
public class WebInterface2048 {
    Context mContext;
    public WebInterface2048(Context context) {
        mContext = context;
    }
    @JavascriptInterface
    public void showToast(String toast) {
        Toast.makeText(mContext, toast, Toast.LENGTH_SHORT).show();
    }
}
}

到目前为止,我已经尝试了以下方法来解决问题:

  1. 将硬件加速放到Manifest文件的上
  2. 在"活动"中启用和禁用硬件加速
  3. 在shouldOverrideUrlLoading中重新加载了相同的URL
  4. 尝试在onStart()而不是onCreate()中加载URL

但似乎什么都不适合我。

我的日志:

D/OpenGLRenderer: endAllActiveAnimators on 0xb7d7e248 (RippleDrawable) with handle 0xb76b0cf0
I/AppCompatViewInflater: app:theme is now deprecated. Please move to using android:theme instead.
D/cr_Ime: [InputMethodManagerWrapper.java:30] Constructor
W/cr_AwContents: onDetachedFromWindow called when already detached. Ignoring
D/cr_Ime: [InputMethodManagerWrapper.java:59] isActive: false
I/cr_Ime: ImeThread is not enabled.
W/cr_BindingManager: Cannot call determinedVisibility() - never saw a connection for the pid: 18631
D/cr_Ime: [InputMethodManagerWrapper.java:59] isActive: true
D/cr_Ime: [InputMethodManagerWrapper.java:68] hideSoftInputFromWindow
D/OpenGLRenderer: endAllActiveAnimators on 0xb7a893f8 (RippleDrawable) with handle 0xb7ec8810
I/AppCompatViewInflater: app:theme is now deprecated. Please move to using android:theme instead.
D/cr_Ime: [InputMethodManagerWrapper.java:30] Constructor
W/cr_AwContents: onDetachedFromWindow called when already detached. Ignoring
D/cr_Ime: [InputMethodManagerWrapper.java:59] isActive: false
I/cr_Ime: ImeThread is not enabled.
W/cr_BindingManager: Cannot call determinedVisibility() - never saw a connection for the pid: 18631
D/cr_Ime: [InputMethodManagerWrapper.java:59] isActive: true
D/cr_Ime: [InputMethodManagerWrapper.java:68] hideSoftInputFromWindow

其他信息:我的设备是Moto G3(Android 6.0.1)

从未理解为什么会发生这种情况。但作为一种变通方法,我使用以下代码解决了这个问题:

mWebView.postDelayed(new Runnable() {
        @Override
        public void run() {
            mWebView.loadUrl("file:///android_asset/2048/index.html");
        }
    }, 500);

如果我在另一个线程中运行,它对我有效,而不是在主线程中加载URL

上面的解决方法对那些面临问题的人很有帮助,他们的Web视图在第一次加载时显示为空白,在重新加载时,它会加载内容。

这可能不是每个解决方案的问题,但它起了作用。如果有人正在寻找更好的解决方案,请发帖。

我通过在onCreate方法中手动将WebView添加到布局中解决了这个问题

        LinearLayout ll = findViewById(R.id.content);
        WebView help = new WebView(this);
        ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
        help.setLayoutParams(lp);
        ll.addView(help);

然后像往常一样加载html:

        try {
             String base64 = Base64.encodeToString(sampleHtml.getBytes("UTF-8"), Base64.DEFAULT);
             help.loadData(base64, "text/html; charset=utf-8", "base64");
         } catch (UnsupportedEncodingException e) {
             e.printStackTrace();
         } catch (Exception ex){
             ex.printStackTrace();
         }

尝试将其与代码的其余部分隔离开来。将其放入oncreate中进行测试。saveinstancestate并不总是null,因此不会加载。mWebView.loadUrl("file:///android_asset/2048/index.html");