我有一个应用程序,其中有一个我在其中显示一些网站的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);
            }
        });
    }
}


评论

对于xwalkview,请参阅如何使用“后退”按钮与CrossWalk的XWalkView一起返回,或将其禁用?

#1 楼

我在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);
}


编辑:

要使此代码起作用,您需要添加一个字段到包含WebView的Activity

private WebView mWebView;


onCreate()方法初始化它,您应该会满意。 >

评论


我应该放置该代码吗?在我所有的代码下?还是就像内置的Zoom一样在loasUrl下?

–zvzej
2011年5月20日20:54

您将其放在活动中,但在onCreate()方法之外。

–FoamyGuy
2011年5月20日在21:06

当我放置您的代码时,由于mWebView变量不在oncreate内,因此给我一个错误,我应该在哪里放置该变量?

–zvzej
2011年5月20日在21:12

代替finish()放回super.onKeyDown(keyCode,event)

– Bostone
2013年1月16日19:13

或者实际上只是删除整个else块

– Bostone
13年1月16日在19:23

#2 楼

如果使用的是Android 2.2及更高版本(现在是大多数设备),则以下代码将为您提供所需的信息。

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


评论


真好我使用此答案的变体,但没有“ else”块。否则,当用户多次按下时,它们将进入空白的应用程序“启动”屏幕,而用户没有前进的选项。

–乔治·阿姆霍尔德
2012年11月12日16:49

这看起来是最合适的答案。 @CaffeineComa,您可能只对应用程序开始屏幕进行了编码,以使其不显示在应用程序“活动”历史记录中。只需在所需的中的AndroidManifest.xml中添加android:noHistory =“ true”属性

– LocalPCGuy
2013年9月27日在21:48



这应该回来吗?它在培训指南中。

–基思
17年4月25日在5:29

#3 楼

这是我的解决方案。它也可以在Fragment中使用。

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


评论


我认为,这应该是正确的答案。

– Arvin
13年7月16日在2:23

为什么这个答案应该是正确的答案?前两个答案有什么问题吗?

–ca9163d9
2015年1月9日下午4:12

@ dc7a9163d9活动可能想要具有多个依赖于keyDown的视图或对象。这个答案通过webview(更好的设计)抽象出活动对keydown依赖的需求。而且,较新的应用几乎只使用片段而不是活动,并且此答案支持该用例,而不是其他情况。

– David T.
15年7月21日在4:02

我喜欢这种方法。而不是整个活动仅监听WebView

– davejoem
17年11月21日在8:55

同意这样做会更好,因此主要活动不需要保留仅由片段使用的WebView。

– Jannik
17年12月20日在23:19

#4 楼

下一个按钮和进度条的完整参考:在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();
            }
        }
    }); 


#5 楼

聚焦也应在onBackPressed中检查。

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


评论


听起来很合理,请您解释一下原因?

– SharpC
19年5月2日在20:51

#6 楼

为什么不使用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)
2012年1月25日10:46



otoh,Honeycomb及更高版本没有硬件后退键,导致永不调用onKeyDown。

–垃圾狗
2012年5月9日在3:57



#7 楼

这是带有确认退出的代码:

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


#8 楼

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


#9 楼

在kotlin中:

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


webView-xml中的Webview组件的ID(如果使用综合引用)。

#10 楼

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


#11 楼

您应该在您的类中的以下库处理onBackKeyPressed。
canGoBack()检查Web视图是否可以引用上一页。如果可能的话,请使用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
           }
  }


#12 楼

这是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
}


#13 楼

您可以在片段中尝试用于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
}


评论


为什么在setOnKeyListener之前调用webView.canGoBack()?

– CoolMind
20 Nov 18 '14:52

#14 楼

Kotlin官方方式:

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

#15 楼

如果有人想为片段中的webView处理backPressed,那么他可以使用下面的代码。

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


}


将代码复制到片段Activity

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



注意事项




YourFragmmentName应替换为您使用的实际Acitivity类。

YourFragmentName应替换与您的Fragment的名称相同。
Activity中声明YourFragmentName,以便可以从函数中对其进行访问。


#16 楼

使用此代码返回页面,当最后一页出现时,则退出活动

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


#17 楼

活动中的Webview,下面的代码对我有用,在webview.Ur中加载URL后,完成该活动回到上一个活动

 webView = (WebView) findViewById(R.id.info_webView);
 webView.loadUrl(value);
 finish();