Android-회전시 WebView 다시로드 방지


90

화면을 회전하면 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>

1
나 자신을 사용하는 것이 좋습니다 수있는 단순화 된 솔루션은 여기에 있습니다 : twigstechtips.blogspot.com/2013/08/...
안드리 Baruckis

답변:


103

주된 문제는 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);
}

참고 : 또한 활동에 AndroidManifest.xml을 추가하십시오 android : configChanges = "orientation | screenSize"감사합니다.


1
예! 이것은 나를 위해 작동합니다 (편집 후 Tim이 발견했습니다!)! 감사합니다! 이것이 작동하는 이유와 방법 및 이러한 방법이 재정의가 필요한 이유를 설명해 주시겠습니까?
마크


마지막으로 다음 if 문이 정말로 필요합니까? if (savedInstanceState! = null) ((WebView) findViewById (R.id.web)). restoreState (savedInstanceState);
마크

21
이 같은 페이지를 유지하지만 여전히 화면 회전에 다시로드
움켜 잡

2
이것은 AndroidManifest의 <activity>에 android : configChanges 속성을 추가하면 작동하므로 답변의 두 가지와 그 하나를 더한 것입니다. 다음은 세 가지를 모두 포함하는 더 나은 답변입니다. stackoverflow.com/a/46849736/454780
trusktr

63

자바 코딩이 필요하지 않습니다. 매니페스트 파일에서 이것을 사용하십시오.

 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>

2
무슨 변화에 대해 이야기하고 있습니까?
Pratik Poddar 2014 년

문제가 회전 할 때 @Mgamerz 주석 다음 .. .. 이것은 당신이 줄 수있는 최악의 제안입니다
길 헤르 메 올리베이라

54
안돼. 이 일에 대해 생각조차하지 마십시오. 구성 변경은 회전에만 국한되지 않습니다. 또한 메모리 부족에서 돌아온 후 인스턴스 상태를 복원 할 수 있습니다. 여기서 문제는이 작업을 수행 할 때 구성이 올바르게 업데이트되지 않는다는 것입니다. 제발, 아무도 이러지마.
에릭 코크란

6
@EricCochran 당신이 알고 있다면 정답을 쓸 수 있습니까? 수락 된 답변은 여전히 ​​나를 위해 페이지를 다시로드하고 세션 정보를 잃습니다. 제 경우에는 와우, 전화를 켜십시오. 다시 로그인해야합니다.
Atomosk

이것은 완전한 답변의 일부일 수 있습니다. 이 더 나은 일을 참조하십시오 stackoverflow.com/a/46849736/454780
trusktr

19

태그 (매니페스트)

android:configChanges="orientation|screenSize"

8
onConfigurationChanged추가 된 사용자 지정 기능이 없기 때문에 실제로 메서드 를 재정의 할 이유 가 없습니다. 매니페스트 파일을 변경하는 것만으로도 트릭이되었습니다.
i2097i

매니페스트 충분했다, 단지 추가하는 upvote에orientation|screenSize
하기 Varun Garg를을

14

android:configChanges="orientation|screenSize"매니페스트 추가 는 나를 위해 작동합니다.

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

WebView와 관련하여 MainActivity가 수행하는 작업을 게시 할 수 있습니까?
trusktr

8

나는 이것이 더 이상 작동하지 않을 것이라고 믿습니다. 제 경우에는 WebView.restore(Bundle savedInstanceState)여전히 사용하여 상태를 복원 하면 URL이 다시로드됩니다.

문서 를 보면 다음과 같이 restoreState()표시됩니다.

이 WebView가 상태 (페이지로드, 뒤로 / 앞으로 목록 만들기 등)를 빌드 할 기회가있는 후에 호출되면 바람직하지 않은 부작용이있을 수 있습니다.

이 방법은 더 이상이 WebView에 대한 디스플레이 데이터를 복원하지 않습니다.

그의 대답 에서 올바른 방향으로 나를 가리키는 @ e4c5 소품

물론 극단적 인 조치는 방향 변경이 활동 파괴 / 생성을 유발하지 않도록하는 것입니다. 이를 수행하는 방법에 대한 문서는 여기에 있습니다.


6

매니페스트에이 코드 추가

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

4

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");

4

매니페스트 파일에서 다음을 시도하십시오.

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

4

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>

4

여기 에 인용 된 바와 같이 ,

주의 : Android 3.2 (API 레벨 13)부터는 기기가 세로 방향과 가로 방향 사이를 전환 할 때 '화면 크기'도 변경됩니다. 따라서 API 레벨 13 이상 (minSdkVersion 및 targetSdkVersion 속성에 의해 선언 됨) 용으로 개발할 때 방향 변경으로 인해 런타임이 다시 시작되지 않도록하려면 "orientation"값과 함께 "screenSize"값을 포함해야합니다. 즉, android : configChanges = "orientation | screenSize"를 표시해야합니다. 그러나 애플리케이션이 API 레벨 12 이하를 대상으로하는 경우 활동은 항상이 구성 변경을 자체적으로 처리합니다 (이 구성 변경은 Android 3.2 이상 기기에서 실행되는 경우에도 활동을 다시 시작하지 않음).

android:configChanges="orientation|screenSize"활동을 설정 하면이 문제가 해결됩니다.

또한 다음 사항에 유의하십시오.

알아두기 : 구성 변경을 처리하기 위해 활동을 선언 할 때 대안을 제공하는 모든 요소를 ​​재설정해야합니다. 방향 변경을 처리하기 위해 활동을 선언하고 가로와 세로간에 변경해야하는 이미지가있는 경우 onConfigurationChanged () 중에 각 리소스를 각 요소에 다시 할당해야합니다.


3

이 솔루션은 저에게 잘 작동합니다.

(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);
}

0

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();
    }
}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.