studio sistema propiedades example descargar codigo android android-webview

sistema - webview android studio 2018



¿Cómo escuchar un WebView terminando de cargar una URL? (13)

@ian esto no es 100% exacto. Si tiene varios iframes en una página, tendrá múltiples onPageFinished (y onPageStarted). Y si tienes varias redirecciones también puede fallar. Este enfoque resuelve (casi) todos los problemas:

boolean loadingFinished = true; boolean redirect = false; mWebView.setWebViewClient(new WebViewClient() { @Override public boolean shouldOverrideUrlLoading(WebView view, String urlNewString) { if (!loadingFinished) { redirect = true; } loadingFinished = false; webView.loadUrl(urlNewString); return true; } @Override public void onPageStarted(WebView view, String url) { loadingFinished = false; //SHOW LOADING IF IT ISNT ALREADY VISIBLE } @Override public void onPageFinished(WebView view, String url) { if (!redirect) { loadingFinished = true; } if (loadingFinished && !redirect) { //HIDE LOADING IT HAS FINISHED } else { redirect = false; } } });

ACTUALIZAR:

De acuerdo con la documentación: no se llamará a onPageStarted cuando cambie el contenido de un marco incrustado, es decir, al hacer clic en un enlace cuyo objetivo es un iframe.

Encontré un caso específico como ese en Twitter, donde solo se llamó a una página terminada y ensució un poco la lógica. Para resolver eso agregué una tarea programada para eliminar la carga después de X segundos. Esto no es necesario en todos los demás casos.

ACTUALIZACIÓN 2 :

Ahora con la implementación actual de Android WebView:

boolean loadingFinished = true; boolean redirect = false; mWebView.setWebViewClient(new WebViewClient() { @Override public boolean shouldOverrideUrlLoading( WebView view, WebResourceRequest request) { if (!loadingFinished) { redirect = true; } loadingFinished = false; webView.loadUrl(request.getUrl().toString()); return true; } @Override public void onPageStarted( WebView view, String url, Bitmap favicon) { super.onPageStarted(view, url, favicon); loadingFinished = false; //SHOW LOADING IF IT ISNT ALREADY VISIBLE } @Override public void onPageFinished(WebView view, String url) { if (!redirect) { loadingFinished = true; } if (loadingFinished && !redirect) { //HIDE LOADING IT HAS FINISHED } else { redirect = false; } } });

Tengo una vista WebView que está cargando una página de Internet. Quiero mostrar una ProgressBar hasta que se complete la carga.

¿Cómo escucho la finalización de la carga de una página web?


Aquí hay un método novedoso para detectar cuándo se ha cargado una URL utilizando la capacidad de Android para enlaces de JavaScript . Usando este patrón, aprovechamos el conocimiento de JavaScript del estado del documento para generar una llamada de método nativo dentro del tiempo de ejecución de Android. Estas llamadas accesibles a JavaScript se pueden hacer usando la anotación @JavaScriptInterface .

Esta implementación requiere que llamemos a setJavaScriptEnabled(true) en la configuración de WebView , por lo que podría no ser adecuado dependiendo de los requisitos de su aplicación, por ejemplo, problemas de seguridad.

src / io / github / cawfree / webviewcallback / MainActivity.java (Jelly Bean, nivel API 16)

package io.github.cawfree.webviewcallback; /** * Created by Alex Thomas (@Cawfree), 30/03/2017. **/ import android.net.http.SslError; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.webkit.JavascriptInterface; import android.webkit.SslErrorHandler; import android.webkit.WebView; import android.webkit.WebViewClient; import android.widget.Toast; /** An Activity demonstrating how to introduce a callback mechanism into Android''s WebView. */ public class MainActivity extends AppCompatActivity { /* Static Declarations. */ private static final String HOOK_JS = "Android"; private static final String URL_TEST = "http://www.zonal.co.uk/"; private static final String URL_PREPARE_WEBVIEW = ""; /* Member Variables. */ private WebView mWebView = null; /** Create the Activity. */ @Override protected final void onCreate(final Bundle pSavedInstanceState) { // Initialize the parent definition. super.onCreate(pSavedInstanceState); // Set the Content View. this.setContentView(R.layout.activity_main); // Fetch the WebView. this.mWebView = (WebView)this.findViewById(R.id.webView); // Enable JavaScript. this.getWebView().getSettings().setJavaScriptEnabled(true); // Define the custom WebClient. (Here I''m just suppressing security errors, since older Android devices struggle with TLS.) this.getWebView().setWebViewClient(new WebViewClient() { @Override public final void onReceivedSslError(final WebView pWebView, final SslErrorHandler pSslErrorHandler, final SslError pSslError) { pSslErrorHandler.proceed(); } }); // Define the WebView JavaScript hook. this.getWebView().addJavascriptInterface(this, MainActivity.HOOK_JS); // Make this initial call to prepare JavaScript execution. this.getWebView().loadUrl(MainActivity.URL_PREPARE_WEBVIEW); } /** When the Activity is Resumed. */ @Override protected final void onPostResume() { // Handle as usual. super.onPostResume(); // Load the URL as usual. this.getWebView().loadUrl(MainActivity.URL_TEST); // Use JavaScript to embed a hook to Android''s MainActivity. (The onExportPageLoaded() function implements the callback, whilst we add some tests for the state of the WebPage so as to infer when to export the event.) this.getWebView().loadUrl("javascript:" + "function onExportPageLoaded() { " + MainActivity.HOOK_JS + ".onPageLoaded(); }" + "if(document.readyState === ''complete'') { onExportPageLoaded(); } else { window.addEventListener(''onload'', function () { onExportPageLoaded(); }, false); }"); } /** Javascript-accessible callback for declaring when a page has loaded. */ @JavascriptInterface @SuppressWarnings("unused") public final void onPageLoaded() { // Display the Message. Toast.makeText(this, "Page has loaded!", Toast.LENGTH_SHORT).show(); } /* Getters. */ public final WebView getWebView() { return this.mWebView; } }

res / layout / activity_main.xml

<?xml version="1.0" encoding="utf-8"?> <WebView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/webView" android:layout_width="wrap_content" android:layout_height="wrap_content" />

Esencialmente, estamos agregando una función de JavaScript adicional que se utiliza para probar el estado del documento. Si está cargado, lanzamos un evento onPageLoaded() personalizado en onPageLoaded() de Android; de lo contrario, registramos un detector de eventos que actualiza Android una vez que la página está lista, usando window.addEventListener(''onload'', ...); .

Ya que estamos agregando este script después de que se haya realizado la llamada a this.getWebView().loadURL("") , es probable que no necesitemos ''escuchar'' los eventos en absoluto, ya que solo tenemos una oportunidad para anexar el loadURL JavaScript haciendo una llamada sucesiva a loadURL , una vez que la página ya se haya cargado.


Aquí hay un método que le permite registrar un Runnable para ser ejecutado una vez que una dirección web particular haya terminado de cargarse. Runnable cada Runnable con una String URL correspondiente en un Map , y usamos el método getOriginalUrl() WebView para elegir la devolución de llamada adecuada.

package io.github.cawfree.webviewcallback; /** * Created by Alex Thomas (@Cawfree), 30/03/2017. **/ import android.net.http.SslError; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.webkit.SslErrorHandler; import android.webkit.WebView; import android.webkit.WebViewClient; import java.util.HashMap; import java.util.Map; /** An Activity demonstrating how to introduce a callback mechanism into Android''s WebView. */ public class MainActivity extends AppCompatActivity { /* Member Variables. */ private WebView mWebView; private Map<String, Runnable> mCallbackMap; /** Create the Activity. */ @Override protected final void onCreate(final Bundle pSavedInstanceState) { // Initialize the parent definition. super.onCreate(pSavedInstanceState); // Set the Content View. this.setContentView(R.layout.activity_main); // Fetch the WebView. this.mWebView = (WebView)this.findViewById(R.id.webView); this.mCallbackMap = new HashMap<>(); // Define the custom WebClient. (Here I''m just suppressing security errors, since older Android devices struggle with TLS.) this.getWebView().setWebViewClient(new WebViewClient() { /** Handle when a request has been launched. */ @Override public final void onPageFinished(final WebView pWebView, final String pUrl) { // Determine whether we''re allowed to process the Runnable; if the page hadn''t been redirected, or if we''ve finished redirection. if(pUrl.equals(pWebView.getOriginalUrl())) { // Fetch the Runnable for the OriginalUrl. final Runnable lRunnable = getCallbackMap().get(pWebView.getOriginalUrl()); // Is it valid? if(lRunnable != null) { lRunnable.run(); } } // Handle as usual. super.onPageFinished(pWebView, pUrl); } /** Ensure we handle SSL state properly. */ @Override public final void onReceivedSslError(final WebView pWebView, final SslErrorHandler pSslErrorHandler, final SslError pSslError) { pSslErrorHandler.proceed(); } }); // Assert that we wish to visit Zonal''s website. this.getWebView().loadUrl("http://www.zonal.co.uk/"); // Align a Callback for Zonal; this will be serviced once the page has loaded. this.getCallbackMap().put("http://www.zonal.co.uk/", new Runnable() { @Override public void run() { /* Do something. */ } }); } /* Getters. */ public final WebView getWebView() { return this.mWebView; } private final Map<String, Runnable> getCallbackMap() { return this.mCallbackMap; } }


He simplificado el código de neteinstein para que sea así:

mWebView.setWebViewClient(new WebViewClient() { private int webViewPreviousState; private final int PAGE_STARTED = 0x1; private final int PAGE_REDIRECTED = 0x2; @Override public boolean shouldOverrideUrlLoading(WebView view, String urlNewString) { webViewPreviousState = PAGE_REDIRECTED; mWebView.loadUrl(urlNewString); return true; } @Override public void onPageStarted(WebView view, String url, Bitmap favicon) { super.onPageStarted(view, url, favicon); webViewPreviousState = PAGE_STARTED; if (dialog == null || !dialog.isShowing()) dialog = ProgressDialog.show(WebViewActivity.this, "", getString(R.string.loadingMessege), true, true, new OnCancelListener() { @Override public void onCancel(DialogInterface dialog) { // do something } }); } @Override public void onPageFinished(WebView view, String url) { if (webViewPreviousState == PAGE_STARTED) { dialog.dismiss(); dialog = null; } } });

Es fácil de entender, OnPageFinished si la devolución de llamada anterior está en onPageStarted, por lo que la página está completamente cargada.


Implemente WebViewClient y extienda onPageFinished () de la siguiente manera:

mWebView.setWebViewClient(new WebViewClient() { public void onPageFinished(WebView view, String url) { // do your stuff here } });


Puede rastrear el estado de progreso mediante el método getProgress en la clase WebView .

Inicializar el estado de progreso.

private int mProgressStatus = 0;

luego la AsyncTask para cargar de esta manera:

private class Task_News_ArticleView extends AsyncTask<Void, Void, Void> { private final ProgressDialog dialog = new ProgressDialog( your_class.this); // can use UI thread here protected void onPreExecute() { this.dialog.setMessage("Loading..."); this.dialog.setCancelable(false); this.dialog.show(); } @Override protected Void doInBackground(Void... params) { try { while (mProgressStatus < 100) { mProgressStatus = webview.getProgress(); } } catch (Exception e) { } return null; } protected void onPostExecute(Void result) { if (this.dialog.isShowing()) { this.dialog.dismiss(); } } }


Si desea mostrar una barra de progreso, debe escuchar un evento de cambio de progreso, no solo para completar la página:

mWebView.setWebChromeClient(new WebChromeClient(){ @Override public void onProgressChanged(WebView view, int newProgress) { //change your progress bar } });

Por cierto, si desea mostrar solo una barra de progreso indeterminada, anule el método onPageFinished es suficiente


Solo para mostrar la barra de progreso, los métodos "onPageStarted" y "onPageFinished" son suficientes; pero si desea tener un indicador "is_loading" (junto con redireccionamientos de página, ...), estos métodos pueden ejecutarse sin secuenciar, como "onPageStarted> onPageStarted> onPageFinished> onPageFinished" en la cola.

Pero con mi prueba corta (pruébelo usted mismo), la cola de valores del método "onProgressChanged" es "0-100> 0-100> 0-100> ..."

private boolean is_loading = false; webView.setWebChromeClient(new MyWebChromeClient(context)); private final class MyWebChromeClient extends WebChromeClient{ @Override public void onProgressChanged(WebView view, int newProgress) { if (newProgress == 0){ is_loading = true; } else if (newProgress == 100){ is_loading = false; } super.onProgressChanged(view, newProgress); } }

También establezca " is_loading = false " en el cierre de actividad, si es una variable estática porque la actividad se puede finalizar antes de que termine la página.


Soy bastante parcial con la solución @NeTeInStEiN (y @polen), pero la habría implementado con un contador en lugar de varios booleanos o observadores del estado (solo otro sabor pero pensé que podría compartir). Tiene un matiz de JS, pero siento que la lógica es un poco más fácil de entender.

private void setupWebViewClient() { webView.setWebViewClient(new WebViewClient() { private int running = 0; // Could be public if you want a timer to check. @Override public boolean shouldOverrideUrlLoading(WebView webView, String urlNewString) { running++; webView.loadUrl(urlNewString); return true; } @Override public void onPageStarted(WebView view, String url, Bitmap favicon) { running = Math.max(running, 1); // First request move it to 1. } @Override public void onPageFinished(WebView view, String url) { if(--running == 0) { // just "running--;" if you add a timer. // TODO: finished... if you want to fire a method. } } }); }


También encontré una solución elegante, pero no la he probado rigurosamente:

public void onPageFinished(WebView view, String url) { super.onPageFinished(view, url); if (m_webView.getProgress() == 100) { progressBar.setVisibility(View.GONE); m_webView.setVisibility(View.VISIBLE); } }



gracias por las respuestas Me ayudó, pero tuve que mejorarlo un poco para mis necesidades. Tuve varios inicios de página y acabados, así que agregué un temporizador que verifica si después de que la página finaliza se inicie un nuevo inicio de página. Está bien, mala explicación. Ver el codigo :)

myWebView.setWebViewClient(new WebViewClient() { boolean loadingFinished = true; boolean redirect = false; long last_page_start; long now; // Load the url public boolean shouldOverrideUrlLoading(WebView view, String url) { if (!loadingFinished) { redirect = true; } loadingFinished = false; view.loadUrl(url); return false; } @Override public void onPageStarted(WebView view, String url, Bitmap favicon) { Log.i("p","pagestart"); loadingFinished = false; last_page_start = System.nanoTime(); show_splash(); } // When finish loading page public void onPageFinished(WebView view, String url) { Log.i("p","pagefinish"); if(!redirect){ loadingFinished = true; } //call remove_splash in 500 miSec if(loadingFinished && !redirect){ now = System.nanoTime(); new android.os.Handler().postDelayed( new Runnable() { public void run() { remove_splash(); } }, 500); } else{ redirect = false; } } private void show_splash() { if(myWebView.getVisibility() == View.VISIBLE) { myWebView.setVisibility(View.GONE); myWebView_splash.setVisibility(View.VISIBLE); } } //if a new "page start" was fired dont remove splash screen private void remove_splash() { if (last_page_start < now) { myWebView.setVisibility(View.VISIBLE); myWebView_splash.setVisibility(View.GONE); } } });


Cargando url con SwipeRefreshLayout y ProgressBar :

UrlPageActivity.java:

WebView webView; SwipeRefreshLayout _swipe_procesbar; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_url_page); String url = "http://.com/"; _swipe_procesbar = (SwipeRefreshLayout)findViewById(R.id.url_path_swipe_procesbar); _swipe_procesbar.post(new Runnable() { @Override public void run() { _swipe_procesbar.setRefreshing(true); } } ); webView = (WebView) findViewById(R.id.url_page_web_view); webView.getSettings().setJavaScriptEnabled(true); webView.setWebViewClient(new WebViewClient() { public void onPageFinished(WebView view, String url) { _swipe_procesbar.setRefreshing(false); _swipe_procesbar.setEnabled(false); } }); webView.loadUrl(url); }

activity_url_page.xml:

<android.support.v4.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/url_path_swipe_procesbar" android:layout_width="match_parent" android:layout_height="wrap_content"> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.test.test1.UrlPageActivity"> <WebView android:layout_width="fill_parent" android:layout_height="fill_parent" android:id="@+id/url_page_web_view" /> </RelativeLayout> </android.support.v4.widget.SwipeRefreshLayout>