【问题标题】:Android - Preventing WebView reload on RotateAndroid - 防止 WebView 在旋转时重新加载
【发布时间】:2012-08-26 14:42:12
【问题描述】:

当我旋转屏幕时,WebView 会重新加载整个页面。我不能拥有这个,因为我的一些内容包含动态/随机材料。当前,当屏幕旋转时,会从 loadUrl() 方法重新加载原始 URL。

知道我的代码有什么问题吗?

MainActivity.java

package com.mark.myapp;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
import android.view.Menu;
import android.webkit.WebView;
import android.webkit.WebViewClient;

public class MainActivity extends Activity {

    WebView web;
    String webURL = "http://www.google.co.uk/";

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        if (savedInstanceState != null)
            ((WebView)findViewById(R.id.web)).restoreState(savedInstanceState);

        web = (WebView) findViewById(R.id.web);
        web.getSettings().setJavaScriptEnabled(true);
        web.loadUrl(webURL);
        web.setPadding(0, 0, 0, 0);
        web.getSettings().setLoadWithOverviewMode(true);
        web.getSettings().setUseWideViewPort(true);
        web.getSettings().setSupportZoom(true);
        web.getSettings().setBuiltInZoomControls(true);

        web.setWebViewClient(new HelloWebViewClient());
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }

    private class HelloWebViewClient extends WebViewClient {
        public boolean shouldOverrideUrlLoading(WebView web, String url) {
            web.loadUrl(url);
            return true;
        }
    }

    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if ((keyCode == KeyEvent.KEYCODE_BACK) && web.canGoBack()) {
            web.goBack();
            return true;
        }
        return super.onKeyDown(keyCode, event);
    }

}

AndroidManifest.xml

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.mark.myapp"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="15" />

    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"

            android:configChanges="orientation|keyboardHidden">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
    <uses-permission android:name="android.permission.INTERNET"/>
</manifest>

【问题讨论】:

标签: java android android-webview


【解决方案1】:

我认为主要问题是您调用 web.loadUrl(webURL);同样当 savedInstanceState != null

编辑

试试:

if (savedInstanceState == null)
{
  web.loadUrl(webURL);
}

EDIT2:您还需要 onSaveInstanceState 和 onRestoreInstanceState 覆盖。

@Override
protected void onSaveInstanceState(Bundle outState )
{
super.onSaveInstanceState(outState);
web.saveState(outState);
}

@Override
protected void onRestoreInstanceState(Bundle savedInstanceState)
{
super.onRestoreInstanceState(savedInstanceState);
web.restoreState(savedInstanceState);
}

注意:还请在您的 Activity 中添加您的 AndroidManifest.xml android:configChanges="方向|screenSize" 谢谢

【讨论】:

  • 是的!这对我有用(编辑后,蒂姆发现了!)!谢谢!您能否解释一下为什么以及如何工作以及为什么需要覆盖这些方法?
  • 最后一件事,我真的需要以下 if 语句吗: if (savedInstanceState != null) ((WebView)findViewById(R.id.web)).restoreState(savedInstanceState);跨度>
  • 这会保持相同的页面,但仍会在屏幕旋转时重新加载
  • 如果我将 android:configChanges 属性添加到 AndroidManifest 中的 中,则此方法有效,因此您的答案中的两件事加上那件事。这是一个更好的答案,包括所有三件事:stackoverflow.com/a/46849736/454780
【解决方案2】:

编辑: 找到了更合适的解决方案。

新:

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.Example.WebviewSample">
    <application
        <activity android:name=".WebViewActivity"
            <!-- ADD THIS -->
            android:configChanges="orientation|screenSize">
        </activity>
    </application>
</manifest>

WebViewActivity.java 文件中将此添加到onCreate() 方法

@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    wView = (WebView) findViewById(R.id.webView);
    if (savedInstanceState == null) {
        wView.loadUrl("https://google.com");
    }
}

同时更新onSaveInstanceState

@Override
protected void onSaveInstanceState(Bundle outState)
{
    super.onSaveInstanceState(outState);
    wView.saveState(outState);
}

onRestoreInstanceState

@Override
protected void onRestoreInstanceState(Bundle savedInstanceState)
{
    super.onRestoreInstanceState(savedInstanceState);
    wView.restoreState(savedInstanceState);
}

旧:

无需 Java 编码。 在清单文件中使用它。

 android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|uiMode|screenSize|smallestScreenSize"

喜欢:

<application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name="com.Example.WebviewSample.webviewsample"
        android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|uiMode|screenSize|smallestScreenSize"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>        
</application>

【讨论】:

  • 你在说什么变化?
  • 在@Mgamerz 评论之后.. 当事情轮换时.. 这是你能给出的最糟糕的建议
  • Nooooooooooooo。甚至不要考虑这样做。配置更改不仅限于轮换。此外,实例状态可以在从低内存恢复后恢复。这里的问题是执行此操作时配置不会正确更新。拜托,没有人这样做。
  • @EricCochran 如果你知道,你能写出正确的答案吗?接受的答案仍然会为我重新加载页面,它也会丢失会话信息。就我而言,这就像哇,你打开手机 - 你必须再次登录。
  • 这可能只是完整答案的一部分。看到这个更好的:stackoverflow.com/a/46849736/454780
【解决方案3】:

在标签中(清单)

android:configChanges="orientation|screenSize"

【讨论】:

  • 实际上没有理由重写onConfigurationChanged 方法,因为没有添加自定义功能。只需在清单文件中进行更改就可以了。
  • manifest 就足够了,只支持添加orientation|screenSize
  • 奇怪的是破坏行为是默认的。没有人期望旋转 Web 视图会重新启动 Web 应用程序。这只是......奇怪。
【解决方案4】:

在清单中添加 android:configChanges="orientation|screenSize" 对我有用

<activity
            android:name="com.example.HelloWorld.WebActivity"
            android:label="@string/title_activity_web"
            android:configChanges="orientation|screenSize" >
</activity>

【讨论】:

  • 你能发布你的 MainActivity 对 WebView 做了什么吗?
  • @trusktr 它不需要做任何事情。 android:configChanges="orientation|screenSize" 这样做是为了在设备旋转时不会重新启动活动。如果 Activity 永远不会重新启动,则无需处理它重新启动。
【解决方案5】:

我相信这不会再奏效了。在我的情况下,使用WebView.restore(Bundle savedInstanceState) 恢复状态仍然会触发重新加载 url。

查看docsrestoreState() 你会看到它说:

如果在此 WebView 有机会构建状态(加载页面、创建后退/前进列表等)之后调用它,则可能会产生不良副作用。

请注意,此方法不再恢复此 WebView 的显示数据。

感谢@e4c5 在他的answer 中为我指明正确的方向

当然,极端的做法是防止方向变化触发活动破坏/创建。如何做到这一点的文档是here

【讨论】:

    【解决方案6】:

    在清单中添加此代码

    <activity 
         ...
         android:configChanges="orientation|screenSize">
    

    【讨论】:

      【解决方案7】:

      将其放在 Manifest.xml 文件活动中:

      android:configChanges="orientation|screenSize"
      

      这是一个例子:

      <activity android:name=".MainActivity"
                android:label="@string/app_name"
                android:theme="@style/AppTheme.NoActionBar"
                android:configChanges="orientation|screenSize">
                <intent-filter>
                      <category android:name="android.intent.category.DEFAULT" />
                </intent-filter>
      </activity>
      

      【讨论】:

        【解决方案8】:

        重写onConfigChange 方法以避免在方向更改时重新加载数据

        关于您在AndroidMainfest 文件中的活动。

        android:configChanges="orientation|keyboardHidden" 
        

        并且在您的 WebView 设置中也有这些

        webview.getSettings().setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);
        webview.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
        webview.loadUrl("Your URL To Load");
        

        【讨论】:

          【解决方案9】:

          在你的清单文件中试试这个:

          android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|uiMode|screenSize|smallestScreenSize"
          

          【讨论】:

            【解决方案10】:

            引用here

            注意:从 Android 3.2(API 级别 13)开始,“屏幕尺寸” 当设备在纵向和横向之间切换时也会发生变化 方向。因此,如果您想防止运行时重新启动 为 API 级别 13 或更高级别(如 由 minSdkVersion 和 targetSdkVersion 属性声明),你 除了“方向”之外,还必须包含“screenSize”值 价值。也就是说,你必须 decalare android:configChanges="orientation|screenSize"。但是,如果您的 应用程序以 API 级别 12 或更低级别为目标,那么您的活动始终 自行处理此配置更改(此配置更改 不会重新启动您的活动,即使在 Android 3.2 或 更高的设备)。

            在您的活动上设置android:configChanges="orientation|screenSize" 将解决此问题。

            另外,请注意以下事项

            记住:当你声明你的活动来处理配置时 更改,您有责任重置您所针对的任何元素 提供替代方案。如果您声明您的活动来处理 方向改变,并有应该改变之间的图像 横向和纵向,您必须将每个资源重新分配给每个 onConfigurationChanged() 期间的元素。

            【讨论】:

              【解决方案11】:

              这个解决方案对我很有效:

              (1) 在 AndroidManifest.xml 添加这一行 android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|uiMode|screenSize|smallestScreenSize"

              像这样(和上面的答案一样)

              <application
                  android:allowBackup="true"
                  android:icon="@mipmap/ic_launcher"
                  android:label="@string/app_name"
                  android:roundIcon="@mipmap/ic_launcher_round"
                  android:supportsRtl="true"
                  android:theme="@style/AppTheme">
                  <activity
                      android:name=".MainActivity"
                      android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|uiMode|screenSize|smallestScreenSize"
                      android:label="@string/app_name"
                      android:theme="@style/AppTheme.NoActionBar">
                      <intent-filter>
                          <action android:name="android.intent.action.MAIN" />
              
                          <category android:name="android.intent.category.LAUNCHER" />
                      </intent-filter>
                  </activity>
              </application>
              

              (2) 然后在 MainActivity.java 中验证 savedInstanceState

              @Override
              protected void onCreate(Bundle savedInstanceState) {
                  super.onCreate(savedInstanceState);
                  setContentView(R.layout.activity_main);
                  mainContext = getApplicationContext();
                  ----
                  myWebView = (WebView) findViewById(R.id.webView);
                  prepareWebView();
                  myWebView.addJavascriptInterface(myJavaScriptInterface, "WEB2Android");
              
                  if (savedInstanceState == null) {
                      myWebView.post(new Runnable() {
                          @Override
                          public void run() {
                              myWebView.loadUrl("http://www.appbiz.ro/foto_konta");
                          }
                      });
                  }
                  ----
              }
              

              (3) 然后:

              @Override
              public void onConfigurationChanged(Configuration newConfig) {
                  super.onConfigurationChanged(newConfig);
              }
              
              @Override
              protected void onSaveInstanceState(Bundle outState )
              {
                  super.onSaveInstanceState(outState);
                  myWebView.saveState(outState);
              }
              
              @Override
              protected void onRestoreInstanceState(Bundle savedInstanceState)
              {
                  super.onRestoreInstanceState(savedInstanceState);
                  myWebView.restoreState(savedInstanceState);
              }
              

              【讨论】:

                【解决方案12】:

                在 Androidmanifest.xml 中添加这个:

                <activity android:name=".MyActivity"
                    android:configChanges="orientation|screenSize|screenLayout|keyboardHidden">
                

                现在,当这些配置之一发生更改时,MyActivity 不会重新启动。相反,MyActivity 会收到对 onConfigurationChanged() 的调用。

                添加这个:

                @Override
                public void onConfigurationChanged(Configuration newConfig) {
                    super.onConfigurationChanged(newConfig);
                
                    // Checks the orientation of the screen
                    if (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) {
                        Toast.makeText(this, "landscape", Toast.LENGTH_SHORT).show();
                    }
                    else if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
                        Toast.makeText(this, "portrait", Toast.LENGTH_SHORT).show();
                    }
                }
                

                【讨论】:

                  猜你喜欢
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 2014-04-16
                  • 2020-06-06
                  • 1970-01-01
                  • 1970-01-01
                  相关资源
                  最近更新 更多