WebView에서 뒤로 버튼을 누르면 이전 페이지로 돌아가는 방법은 무엇입니까?


327

WebView웹 사이트를 표시 할 수있는 앱이 있습니다. 웹 페이지의 링크를 클릭하면 내 앱 내부의 웹 사이트에서 다음 페이지로 이동합니다. 그러나 전화의 뒤로 버튼을 클릭하면 앱으로 바로 연결됩니다. 대신 웹 사이트의 이전 페이지로 돌아가고 싶습니다. 어떻게해야합니까?

사용중인 코드 샘플은 다음과 같습니다.

public class Webdisplay extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);

        this.getWindow().requestFeature(Window.FEATURE_PROGRESS);
        setContentView(R.layout.webdisplay);

        getWindow().setFeatureInt(Window.FEATURE_PROGRESS,
                Window.PROGRESS_VISIBILITY_ON); 

        Toast loadingmess = Toast.makeText(this,
                "Cargando El Diario de Hoy", Toast.LENGTH_SHORT);
        loadingmess.show();

        WebView myWebView;

        myWebView = (WebView) findViewById(R.id.webview);
        myWebView.getSettings().setJavaScriptEnabled(true);
        myWebView.loadUrl("http://www.elsalvador.com");
        myWebView.setWebViewClient(new WebViewClient());
        myWebView.setInitialScale(1);
        myWebView.getSettings().setBuiltInZoomControls(true);
        myWebView.getSettings().setUseWideViewPort(true);

        final Activity MyActivity = this;
        myWebView.setWebChromeClient(new WebChromeClient() 
        {
            public void onProgressChanged(WebView view, int progress)   
            {
                MyActivity.setTitle("Loading...");
                MyActivity.setProgress(progress * 100); 

                if(progress == 100)
                    MyActivity.setTitle(R.string.app_name);
            }
        });
    }
}

답변:


520

WebViews 활동에서 다음과 같은 것을 사용합니다.

@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (event.getAction() == KeyEvent.ACTION_DOWN) {
        switch (keyCode) {
            case KeyEvent.KEYCODE_BACK:
                if (mWebView.canGoBack()) {
                    mWebView.goBack();
                } else {
                    finish();
                }
                return true;
        }

    }
    return super.onKeyDown(keyCode, event);
}

편집하다:

이 코드가 작동하려면 Activity포함하는 WebView에 필드를 추가해야합니다 .

private WebView mWebView;

에서 초기화 onCreate()방법으로 잘해야합니다.

mWebView = (WebView) findViewById(R.id.webView);

1
그 코드를 배치해야 했습니까? 내 모든 코드에서? 또는 builtinZoom처럼 loasUrl 바로 아래에 있습니까?
zvzej

3
액티비티 안에 있지만 onCreate () 메소드 밖에 있습니다.
FoamyGuy

코드를 배치 할 때 onWeb이 없기 때문에 mWebView 변수에 오류가 발생합니다. 변수를 어디에 배치해야합니까?
zvzej

4
대신 finish()넣어return super.onKeyDown(keyCode, event)
Bostone

6
또는 실제로 단순히 전체 else블록을 제거하십시오
Bostone

300

Android 2.2 이상 (현재 대부분의 장치)을 사용하는 경우 다음 코드는 원하는 것을 얻습니다.

@Override
public void onBackPressed() {
    if (webView.canGoBack()) {
        webView.goBack();
    } else {
        super.onBackPressed();
    }
}

5
좋은. 이 답변의 변형을 사용하지만 "else"블록은 사용하지 않습니다. 그렇지 않으면 사용자가 너무 많이 다시 누르면 빈 앱 "시작"화면이 나타나고 사용자가 앞으로 나아갈 수있는 옵션이 없습니다.
George Armhold

1
이것은 가장 적합한 답변처럼 보입니다. @CaffeineComa, 앱 "활동"기록에 표시되지 않도록 앱 시작 화면을 코딩 할 수 있습니다. 그냥 추가 android:noHistory="true"에 속성을 <activity>의 AndroidManifest.xml에, 당신이 원하는
LocalPCGuy

이럴까요 return;? 그것은 훈련 가이드에 있습니다.
Keith

대단히 감사합니다!
DmitryKanunnikoff

113

이것이 나의 해결책이다. 조각에서도 작동합니다.

webView.setOnKeyListener(new OnKeyListener()
{
    @Override
    public boolean onKey(View v, int keyCode, KeyEvent event)
    {
        if(event.getAction() == KeyEvent.ACTION_DOWN)
        {
            WebView webView = (WebView) v;

            switch(keyCode)
            {
                case KeyEvent.KEYCODE_BACK:
                    if(webView.canGoBack())
                    {
                        webView.goBack();
                        return true;
                    }
                    break;
            }
        }

        return false;
    }
});

16
이것이 나의 의견이다. 이것은 정답이어야한다.
Arvin

2
왜이 답변이 정답이어야합니까? 상위 2 개의 답변에 문제가 있습니까?
ca9163d9

12
@ dc7a9163d9 액티비티는 keyDown에 의존하는 여러 뷰 또는 객체를 원할 수 있습니다. 이 답변은 webview (더 나은 디자인)를 통해 활동을 키 다운하는 활동의 필요성을 추상화합니다. 활동에 반대하고, 다른 사람이 아닌이 대답 지원하는 사용 사례 것과 또한, 새로운 애플리케이션은 거의 독점적으로 조각을 사용
데이비드 T.

나는이 접근법을 좋아한다. 전체 활동 대신 WebView 만 들음
davejoem

이것이 더 낫다는 것에 동의하십시오. 따라서 주요 활동은 프래그먼트에서만 사용되는 WebView를 보유 할 필요가 없습니다.
Jannik

18

다음 버튼 및 진행률 표시 줄에 대한 전체 참조 : webview에서 뒤로 및 다음 버튼 놓기

휴대 전화의 뒤로 버튼을 클릭 할 때 뒤로 페이지로 이동하려면 다음을 사용하십시오.

@Override
public void onBackPressed() {
    if (webView.canGoBack()) {
        webView.goBack();
    } else {
        super.onBackPressed();
    }
} 

다음과 같이 사용자 정의 뒤로 버튼을 만들 수도 있습니다.

btnback.setOnClickListener(new View.OnClickListener() {

        @Override
        public void onClick(View v) {
            // TODO Auto-generated method stub

            if (wv.canGoBack()) {
                wv.goBack();
            }
        }
    }); 

16

onBackPressed에서도 초점을 확인해야합니다.

    @Override
    public void onBackPressed() {
        if (mWebview.isFocused() && mWebview.canGoBack()) {
            mWebview.goBack();       
        } else {
            super.onBackPressed();
            finish();
        }
    }

2
그럴듯하게 들리는데 이유를 설명해 주시겠습니까?
SharpC

14

왜 사용하지 onBackPressed()않습니까?

@Override
public void onBackPressed()
{
    // super.onBackPressed(); Do not call me!

    // Go to the previous web page.
}

sdk 2.2 이상에서만이 방법을 지원하기 때문에 .. sdk 1.6에서는 작동하지 않습니다
Maher Abuthraa

1
otoh, Honeycomb 및 최신 제품에는 하드웨어 백 키가 없으므로 onKeyDown이 호출되지 않습니다.
junkdog

10

다음은 확인 종료 코드입니다.

@Override
    public void onBackPressed()
    {
        if(webView.canGoBack()){
            webView.goBack();
        }else{
            new AlertDialog.Builder(this)
            .setIcon(android.R.drawable.ic_dialog_alert)
            .setTitle("Exit!")
            .setMessage("Are you sure you want to close?")
            .setPositiveButton("Yes", new DialogInterface.OnClickListener()
            {
                @Override
                public void onClick(DialogInterface dialog, int which) {
                    finish();    
                }

            })
            .setNegativeButton("No", null)
            .show();    
        }   
    }

6
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    // Check if the key event was the Back button and if there's history
    if ((keyCode == KeyEvent.KEYCODE_BACK) && myWebView.canGoBack()) {
        myWebView.goBack();
        return true;
    }
    // If it wasn't the Back key or there's no web page history, bubble up to the default
    // system behavior (probably exit the activity)
    return super.onKeyDown(keyCode, event);
}

3

kotlin에서 :

override fun onBackPressed() {
    when {
        webView.canGoBack() -> webView.goBack()
        else -> super.onBackPressed()
    }
}

webView-합성 참조를 사용하는 경우 xml의 webview 구성 요소 ID입니다.


2

FoamyGuy의 첫 번째 대답은 정확하지만 몇 가지 추가 사항이 있습니다. 평판이 낮 으면 댓글을 달 수 없습니다. 어떤 이유로 페이지가로드되지 않으면 실패를 기록하도록 플래그를 설정 한 후 onBackPressed 대체에서이를 확인하십시오. 그렇지 않으면 canGoBack () 실제 백 액티비티로 향하지 않고 영원히 실행됩니다 .

//flag with class wide access 
public boolean ploadFailFlag = false;

//your error handling override
@Override
public void onReceivedError(WebView view, WebResourceRequest req, WebResourceError rerr) {
    onReceivedError(view, rerr.getErrorCode(), rerr.getDescription().toString(), req.getUrl().toString());
    ploadFailFlag = true;       //note this change 
    .....
    .....
}

//finally to the answer to this question:
@Override
public void onBackPressed() {
    if(checkinWebView.canGoBack()){
        //if page load fails, go back for web view will not go back - no page to go to - yet overriding the super 
        if(ploadFailFlag){
            super.onBackPressed();
        }else {
            checkinWebView.goBack();
        }
    }else {
        Toast.makeText(getBaseContext(), "super:", Toast.LENGTH_LONG).show();
        super.onBackPressed();
    }
}

1

클래스의 다음 라이브러리가 onBackKeyPressed를 처리해야합니다. canGoBack ()은 웹뷰가 이전 페이지를 참조 할 수 있는지 확인합니다. 가능하면 goBack () 함수를 사용하여 이전 페이지를 참조하십시오 (뒤로).

 @Override
        public void onBackPressed() {
          if( mWebview.canGoBack()){
               mWebview.goBack(); 
           }else{
            //Do something else. like trigger pop up. Add rate app or see more app
           }
  }

1

Kotlin 솔루션은 다음과 같습니다.

override fun onKeyUp(keyCode: Int, event: KeyEvent?): Boolean {
    if (event?.action != ACTION_UP || event.keyCode != KEYCODE_BACK) {
        return super.onKeyUp(keyCode, event)
    }

    if (mWebView.canGoBack()) {
        mWebView.goBack()
    } else {
        finish()
    }
    return true
}

0

공식 코 틀린 웨이 :

override fun onKeyDown(keyCode: Int, event: KeyEvent?): Boolean {
    // Check if the key event was the Back button and if there's history
    if (keyCode == KeyEvent.KEYCODE_BACK && myWebView.canGoBack()) {
        myWebView.goBack()
        return true
    }
    // If it wasn't the Back key or there's no web page history, bubble up to the default
    // system behavior (probably exit the activity)
    return super.onKeyDown(keyCode, event)
}

https://developer.android.com/guide/webapps/webview.html#NavigatingHistory


0

누군가가 fragment 내부의 webView에 대해 backPressed를 처리 하려면 아래 코드를 사용할 수 있습니다.

  1. 아래 코드를 Activity클래스에 복사하십시오 (조각이 들어 있습니다 YourFragmmentName)

    @Override
    public void onBackPressed() {
    List<Fragment> fragmentList = getSupportFragmentManager().getFragments();
    
    boolean handled = false;
    for(Object f: fragmentList) {
        if(f instanceof YourFragmentName) {
            handled = ((YourFragmentName)f).onBackPressed();
            if(handled) {
                break;
            }
        }
    }
    
    if(!handled) {
        super.onBackPressed();
    }

    }

  2. 이 코드를 조각에 복사 YourFragmentName

    public boolean onBackPressed() {
       if (webView.canGoBack()) {
           webView.goBack();
           return true;
       } else {
           return false;
       }
    }

노트

  • Activity 사용중인 실제 Acitivity 클래스로 바꿔야합니다.
  • YourFragmentName 조각 이름으로 바꿔야합니다.
  • 선언 webView에서 YourFragmentName이 함수 내에서 액세스 할 수 있도록.

0

조각으로 webview에 이것을 시도 할 수 있습니다.

private lateinit var webView: WebView

override fun onCreateView(
    inflater: LayoutInflater,
    container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    val root = inflater.inflate(R.layout.fragment_name, container, false)
    webView = root!!.findViewById(R.id.home_web_view)
    var url: String = "http://yoururl.com"
    webView.settings.javaScriptEnabled = true
    webView.webViewClient = WebViewClient()
    webView.loadUrl(url)
    webView.canGoBack()
    webView.setOnKeyListener{ v, keyCode, event ->
        if(keyCode == KeyEvent.KEYCODE_BACK && event.action == MotionEvent.ACTION_UP
            && webView.canGoBack()){
            webView.goBack()
            return@setOnKeyListener true
        }
        false
    }
    return root
}

-2

이 코드를 사용하여 페이지로 돌아가고 마지막 페이지가 왔을 때 활동을 중단하십시오.

 @Override
    public void onBackPressed() {
        super.onBackPressed();
        Intent intent=new Intent(LiveImage.this,DashBoard.class);
        startActivity(intent);
    }

-5

Webview 활동, 코드 아래에서 나를 위해 일했습니다. 웹뷰에서 URL을로드 한 후 활동을 완료하십시오.

 webView = (WebView) findViewById(R.id.info_webView);
 webView.loadUrl(value);
 finish();
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.