Запретить WebView отображать «веб-страница недоступна» - PullRequest
81 голосов
/ 01 июля 2011

У меня есть приложение, которое широко использует WebView. Когда пользователь этого приложения не имеет подключения к Интернету, появляется страница с надписью «веб-страница недоступна» и другой другой текст. Есть ли способ не отображать этот общий текст в моем WebView? Я хотел бы предоставить свою собственную обработку ошибок.

private final Activity activity = this;

private class MyWebViewClient extends WebViewClient
 public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
  // I need to do something like this:
  activity.webView.wipeOutThePage();
  activity.myCustomErrorHandling();
  Toast.makeText(activity, description, Toast.LENGTH_LONG).show();
 }
}

Я обнаружил, WebView-> clearView на самом деле не очищает вид.

Ответы [ 16 ]

90 голосов
/ 23 сентября 2011

Сначала создайте свою собственную страницу ошибки в HTML и поместите ее в папку ресурсов, назовем ее myerrorpage.html Затем с onReceivedError:

mWebView.setWebViewClient(new WebViewClient() {
    public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
        mWebView.loadUrl("file:///android_asset/myerrorpage.html");

    }
});
28 голосов
/ 16 июля 2015

Лучшее решение, которое я нашел, - это загрузить пустую страницу в событии OnReceivedError следующим образом:

@Override
public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
    super.onReceivedError(view, errorCode, description, failingUrl);

    view.loadUrl("about:blank");
}
9 голосов
/ 08 апреля 2014

Наконец-то я решил это.(Работает до сих пор ..)

Мое решение похоже на это ...

  1. Подготовьте макет, чтобы он отображался при возникновении ошибки вместо веб-страницы (грязный)«страница не найдена») В макете есть одна кнопка, «RELOAD» с некоторыми руководящими сообщениями.

  2. Если произошла ошибка, запомните логическое значение и покажите готовый макет.

  3. Если пользователь нажимает кнопку «RELOAD», установите для mbErrorOccured значение false.И установите mbReloadPressed в true.
  4. , если mbErrorOccured имеет значение false, а mbReloadPressed - true, это означает, что загруженная веб-страница успешно загружена.Потому что, если ошибка возникла снова, mbErrorOccured будет установлено значение true для onReceivedError (...)

Вот мой полный источник.Проверьте это.

public class MyWebViewActivity extends ActionBarActivity implements OnClickListener {

    private final String TAG = MyWebViewActivity.class.getSimpleName();
    private WebView mWebView = null;
    private final String URL = "http://www.google.com";
    private LinearLayout mlLayoutRequestError = null;
    private Handler mhErrorLayoutHide = null;

    private boolean mbErrorOccured = false;
    private boolean mbReloadPressed = false;

    @SuppressLint("SetJavaScriptEnabled")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_webview);

        ((Button) findViewById(R.id.btnRetry)).setOnClickListener(this);
        mlLayoutRequestError = (LinearLayout) findViewById(R.id.lLayoutRequestError);
        mhErrorLayoutHide = getErrorLayoutHideHandler();

        mWebView = (WebView) findViewById(R.id.webviewMain);
        mWebView.setWebViewClient(new MyWebViewClient());
        WebSettings settings = mWebView.getSettings();
        settings.setJavaScriptEnabled(true);
        mWebView.setWebChromeClient(getChromeClient());
        mWebView.loadUrl(URL);
    }

    @Override
    public boolean onSupportNavigateUp() {
        return super.onSupportNavigateUp();
    }

    @Override
    public void onClick(View v) {
        int id = v.getId();

        if (id == R.id.btnRetry) {
            if (!mbErrorOccured) {
                return;
            }

            mbReloadPressed = true;
            mWebView.reload();
            mbErrorOccured = false;
        }
    }

    @Override
    public void onBackPressed() {
        if (mWebView.canGoBack()) {
            mWebView.goBack();
            return;
        }
        else {
            finish();
        }

        super.onBackPressed();
    }

    class MyWebViewClient extends WebViewClient {
        @Override
        public boolean shouldOverrideUrlLoading(WebView view, String url) {
            return super.shouldOverrideUrlLoading(view, url);
        }

        @Override
        public void onPageStarted(WebView view, String url, Bitmap favicon) {
            super.onPageStarted(view, url, favicon);
        }

        @Override
        public void onLoadResource(WebView view, String url) {
            super.onLoadResource(view, url);
        }

        @Override
        public void onPageFinished(WebView view, String url) {
            if (mbErrorOccured == false && mbReloadPressed) {
                hideErrorLayout();
                mbReloadPressed = false;
            }

            super.onPageFinished(view, url);
        }

        @Override
        public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
            mbErrorOccured = true;
            showErrorLayout();
            super.onReceivedError(view, errorCode, description, failingUrl);
        }
    }

    private WebChromeClient getChromeClient() {
        final ProgressDialog progressDialog = new ProgressDialog(MyWebViewActivity.this);
        progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
        progressDialog.setCancelable(false);

        return new WebChromeClient() {
            @Override
            public void onProgressChanged(WebView view, int newProgress) {
                super.onProgressChanged(view, newProgress);
            }
        };
    }

    private void showErrorLayout() {
        mlLayoutRequestError.setVisibility(View.VISIBLE);
    }

    private void hideErrorLayout() {
        mhErrorLayoutHide.sendEmptyMessageDelayed(10000, 200);
    }

    private Handler getErrorLayoutHideHandler() {
        return new Handler() {
            @Override
            public void handleMessage(Message msg) {
                mlLayoutRequestError.setVisibility(View.GONE);
                super.handleMessage(msg);
            }
        };
    }
}

Дополнение: Вот макет ....

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/rLayoutWithWebView"
android:layout_width="match_parent"
android:layout_height="match_parent" >

<WebView
    android:id="@+id/webviewMain"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

<LinearLayout
    android:id="@+id/lLayoutRequestError"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_centerInParent="true"
    android:background="@color/white"
    android:gravity="center"
    android:orientation="vertical"
    android:visibility="gone" >

    <Button
        android:id="@+id/btnRetry"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:minWidth="120dp"
        android:text="RELOAD"
        android:textSize="20dp"
        android:textStyle="bold" />
</LinearLayout>

6 голосов
/ 01 июля 2011

Ознакомьтесь с обсуждением на Android WebView onReceivedError () . Это довольно долго, но, похоже, все согласны с тем, что а) вы не можете остановить появление страницы «веб-страница недоступна», но б) вы всегда можете загрузить пустую страницу после того, как получите onReceivedError

5 голосов
/ 28 ноября 2013

Когда веб-просмотр встроен в какое-то пользовательское представление, так что пользователь почти полагает, что он видит собственное представление, а не веб-представление, в таком сценарии показывать ошибку «страница не может быть загружена» нелепо. Что я обычно делаю в такой ситуации, так это загружаю пустую страницу и показываю тост-сообщение, как показано ниже

webView.setWebViewClient(new WebViewClient() {

            @Override
            public void onReceivedError(WebView view, int errorCode,
                    String description, String failingUrl) {
                Log.e(TAG," Error occured while loading the web page at Url"+ failingUrl+"." +description);     
                view.loadUrl("about:blank");
                Toast.makeText(App.getContext(), "Error occured, please check newtwork connectivity", Toast.LENGTH_SHORT).show();
                super.onReceivedError(view, errorCode, description, failingUrl);
            }
        });
2 голосов
/ 12 июля 2011

Возможно, я неправильно понял вопрос, но звучит так, будто вы говорите, что получили ответный вызов с ошибкой, и просто спрашиваете, как лучше всего не показывать ошибку?Почему бы вам просто не удалить веб-представление с экрана и / или не показать другое представление поверх него?

2 голосов
/ 11 июля 2011

Вы можете использовать запрос GET, чтобы получить содержимое страницы, а затем отобразить эти данные, используя Webview, таким образом, вы не используете множественные серверные вызовы.В качестве альтернативы вы можете использовать Javascript для проверки правильности объекта DOM.

1 голос
/ 11 июля 2017

Здесь я нашел самое простое решение.проверить это ..

   @Override
        public void onReceivedError(WebView view, int errorCode,
                                    String description, String failingUrl) {
//                view.loadUrl("about:blank");
            mWebView.stopLoading();
            if (!mUtils.isInterentConnection()) {
                Toast.makeText(ReportingActivity.this, "Please Check Internet Connection!", Toast.LENGTH_SHORT).show();
            }
            super.onReceivedError(view, errorCode, description, failingUrl);
        }

А вот метод isInterentConnection () ...

public boolean isInterentConnection() {
    ConnectivityManager manager = (ConnectivityManager) mContext
            .getSystemService(Context.CONNECTIVITY_SERVICE);
    if (manager != null) {
        NetworkInfo info[] = manager.getAllNetworkInfo();
        if (info != null) {
            for (int i = 0; i < info.length; i++) {
                if (info[i].getState() == NetworkInfo.State.CONNECTED) {
                    return true;
                }
            }
        }
    }
    return false;
}
1 голос
/ 08 июля 2011

Полагаю, что если вы настаиваете на этом, вы можете просто проверить, есть ли ресурс, перед вызовом функции loadURL. Просто переопределите функции и выполните проверку перед вызовом super ()

ЗАМЕЧАНИЕ (возможно, не по теме): в http есть метод HEAD, который описывается следующим образом:

Метод HEAD идентичен GET, за исключением того, что сервер НЕ ДОЛЖЕН возвращать тело ответа в ответе

Этот метод может быть удобен. В любом случае, как бы вы это ни реализовали ... проверьте этот код:

import java.util.Map;

import android.content.Context;
import android.webkit.WebView;

public class WebViewPreLoad extends WebView{

public WebViewPreLoad(Context context) {
super(context);
}
public void loadUrl(String str){
if(//Check if exists)
super.loadUrl(str);
else
//handle error
}
public void loadUrl(String url, Map<String,String> extraHeaders){
if(//Check if exists)
super.loadUrl(url, extraHeaders);
else
//handle error
}
}

Вы можете попробовать эту проверку, используя

if(url.openConnection().getContentLength() > 0)
0 голосов
/ 02 октября 2017

Мне приходилось сталкиваться с этой проблемой, а также пытался решить ее с разных точек зрения.Наконец, я нашел решение, используя один флаг, чтобы проверить, произошла ли ошибка.

... extends WebViewClient {

    boolean error;

    @Override
    public void onPageStarted(WebView view, String url, Bitmap favicon) {

        showLoading(true);
        super.onPageStarted(view, url, favicon);

        error  = false; // IMPORTANT
    }

    @Override
    public void onPageFinished(WebView view, String url) {
        super.onPageFinished(view, url);

        if(!error) {
            Observable.timer(100, TimeUnit.MICROSECONDS, AndroidSchedulers.mainThread())
                    .subscribe((data) -> view.setVisibility(View.VISIBLE) );
        }

        showLoading(false);
    }


    @Override
    public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {

        view.stopLoading();

        view.setVisibility(View.INVISIBLE) 
        error  = true;  

        // Handle the error
    }


     @Override
    @TargetApi(android.os.Build.VERSION_CODES.M)
    public void onReceivedError(WebView view,
                                WebResourceRequest request,
                                WebResourceError error) {

        this.onReceivedError(view, error.getErrorCode(),
                error.getDescription().toString(),
                request.getUrl().toString());
    }
 }

Таким образом, я скрываю страницу каждый раз, когда появляется ошибка, и показываю ее, когда страница снова загрузилась правильно.

Также добавлена ​​небольшая задержка в случае.

Я избежал решения о загрузке пустой страницы, так как это не позволяет вам позже выполнить webview.reload (), поскольку она добавляет эту новую страницу в историю переходов.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...