From: Soren Stoutner <soren@stoutner.com> Date: Wed, 3 Jun 2020 19:19:59 +0000 (-0700) Subject: Fix problems with swipe to refresh. https://redmine.stoutner.com/issues/514 X-Git-Tag: v3.5~5 X-Git-Url: https://gitweb.stoutner.com/?a=commitdiff_plain;h=fd8556e4e7c0db9f58e1cf14a2e0c6590f91305e;p=PrivacyBrowserAndroid.git Fix problems with swipe to refresh. https://redmine.stoutner.com/issues/514 --- diff --git a/app/src/main/java/com/stoutner/privacybrowser/activities/MainWebViewActivity.java b/app/src/main/java/com/stoutner/privacybrowser/activities/MainWebViewActivity.java index becbd5ac..89a73f25 100644 --- a/app/src/main/java/com/stoutner/privacybrowser/activities/MainWebViewActivity.java +++ b/app/src/main/java/com/stoutner/privacybrowser/activities/MainWebViewActivity.java @@ -4050,23 +4050,29 @@ public class MainWebViewActivity extends AppCompatActivity implements CreateBook // Store the swipe to refresh status in the nested scroll WebView. nestedScrollWebView.setSwipeToRefresh(defaultSwipeToRefresh); - // Apply swipe to refresh according to the default. This can be removed once the minimum API >= 23 because it is continuously set by an on scroll change listener. - swipeRefreshLayout.setEnabled(defaultSwipeToRefresh); + // Update the swipe refresh layout. + if (defaultSwipeToRefresh) { // Swipe to refresh is enabled. + // Only enable the swipe refresh layout if the WebView is scrolled to the top. It is updated every time the scroll changes. + swipeRefreshLayout.setEnabled(currentWebView.getY() == 0); + } else { // Swipe to refresh is disabled. + // Disable the swipe refresh layout. + swipeRefreshLayout.setEnabled(false); + } break; case DomainsDatabaseHelper.ENABLED: // Store the swipe to refresh status in the nested scroll WebView. nestedScrollWebView.setSwipeToRefresh(true); - // Enable swipe to refresh. This can be removed once the minimum API >= 23 because it is continuously set by an on scroll change listener. - swipeRefreshLayout.setEnabled(true); + // Only enable the swipe refresh layout if the WebView is scrolled to the top. It is updated every time the scroll changes. + swipeRefreshLayout.setEnabled(currentWebView.getY() == 0); break; case DomainsDatabaseHelper.DISABLED: // Store the swipe to refresh status in the nested scroll WebView. nestedScrollWebView.setSwipeToRefresh(false); - // Disable swipe to refresh. This can be removed once the minimum API >= 23 because it is continuously set by an on scroll change listener. + // Disable swipe to refresh. swipeRefreshLayout.setEnabled(false); } @@ -4154,8 +4160,14 @@ public class MainWebViewActivity extends AppCompatActivity implements CreateBook // Store the swipe to refresh status in the nested scroll WebView. nestedScrollWebView.setSwipeToRefresh(defaultSwipeToRefresh); - // Apply swipe to refresh according to the default. - swipeRefreshLayout.setEnabled(defaultSwipeToRefresh); + // Update the swipe refresh layout. + if (defaultSwipeToRefresh) { // Swipe to refresh is enabled. + // Only enable the swipe refresh layout if the WebView is scrolled to the top. It is updated every time the scroll changes. + swipeRefreshLayout.setEnabled(currentWebView.getY() == 0); + } else { // Swipe to refresh is disabled. + // Disable the swipe refresh layout. + swipeRefreshLayout.setEnabled(false); + } // Reset the pinned variables. nestedScrollWebView.setDomainSettingsDatabaseId(-1); @@ -5176,26 +5188,55 @@ public class MainWebViewActivity extends AppCompatActivity implements CreateBook }); // Update the status of swipe to refresh based on the scroll position of the nested scroll WebView. Also reinforce full screen browsing mode. - // Once the minimum API >= 23 this can be replaced with `nestedScrollWebView.setOnScrollChangeListener()`. - nestedScrollWebView.getViewTreeObserver().addOnScrollChangedListener(() -> { - if (nestedScrollWebView.getSwipeToRefresh()) { - // Only enable swipe to refresh if the WebView is scrolled to the top. - swipeRefreshLayout.setEnabled(nestedScrollWebView.getScrollY() == 0); - } + // On API < 23, `getViewTreeObserver().addOnScrollChangedListener()` must be used, but it is a little bit buggy and appears to get garbage collected from time to time. + if (Build.VERSION.SDK_INT >= 23) { + nestedScrollWebView.setOnScrollChangeListener((view, i, i1, i2, i3) -> { + if (nestedScrollWebView.getSwipeToRefresh()) { + // Only enable swipe to refresh if the WebView is scrolled to the top. + swipeRefreshLayout.setEnabled(nestedScrollWebView.getScrollY() == 0); + } else { + // Disable swipe to refresh. + swipeRefreshLayout.setEnabled(false); + } - // Reinforce the system UI visibility flags if in full screen browsing mode. - // This hides the status and navigation bars, which are displayed if other elements are shown, like dialog boxes, the options menu, or the keyboard. - if (inFullScreenBrowsingMode) { - /* Hide the system bars. - * SYSTEM_UI_FLAG_FULLSCREEN hides the status bar at the top of the screen. - * SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN makes the root frame layout fill the area that is normally reserved for the status bar. - * SYSTEM_UI_FLAG_HIDE_NAVIGATION hides the navigation bar on the bottom or right of the screen. - * SYSTEM_UI_FLAG_IMMERSIVE_STICKY makes the status and navigation bars translucent and automatically re-hides them after they are shown. - */ - rootFrameLayout.setSystemUiVisibility(View.SYSTEM_UI_FLAG_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | - View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY); - } - }); + // Reinforce the system UI visibility flags if in full screen browsing mode. + // This hides the status and navigation bars, which are displayed if other elements are shown, like dialog boxes, the options menu, or the keyboard. + if (inFullScreenBrowsingMode) { + /* Hide the system bars. + * SYSTEM_UI_FLAG_FULLSCREEN hides the status bar at the top of the screen. + * SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN makes the root frame layout fill the area that is normally reserved for the status bar. + * SYSTEM_UI_FLAG_HIDE_NAVIGATION hides the navigation bar on the bottom or right of the screen. + * SYSTEM_UI_FLAG_IMMERSIVE_STICKY makes the status and navigation bars translucent and automatically re-hides them after they are shown. + */ + rootFrameLayout.setSystemUiVisibility(View.SYSTEM_UI_FLAG_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | + View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY); + } + }); + } else { + nestedScrollWebView.getViewTreeObserver().addOnScrollChangedListener(() -> { + if (nestedScrollWebView.getSwipeToRefresh()) { + // Only enable swipe to refresh if the WebView is scrolled to the top. + swipeRefreshLayout.setEnabled(nestedScrollWebView.getScrollY() == 0); + } else { + // Disable swipe to refresh. + swipeRefreshLayout.setEnabled(false); + } + + + // Reinforce the system UI visibility flags if in full screen browsing mode. + // This hides the status and navigation bars, which are displayed if other elements are shown, like dialog boxes, the options menu, or the keyboard. + if (inFullScreenBrowsingMode) { + /* Hide the system bars. + * SYSTEM_UI_FLAG_FULLSCREEN hides the status bar at the top of the screen. + * SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN makes the root frame layout fill the area that is normally reserved for the status bar. + * SYSTEM_UI_FLAG_HIDE_NAVIGATION hides the navigation bar on the bottom or right of the screen. + * SYSTEM_UI_FLAG_IMMERSIVE_STICKY makes the status and navigation bars translucent and automatically re-hides them after they are shown. + */ + rootFrameLayout.setSystemUiVisibility(View.SYSTEM_UI_FLAG_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | + View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY); + } + }); + } // Set the web chrome client. nestedScrollWebView.setWebChromeClient(new WebChromeClient() { diff --git a/app/src/main/res/values-de/strings.xml b/app/src/main/res/values-de/strings.xml index 4c5b84a4..5e282ead 100644 --- a/app/src/main/res/values-de/strings.xml +++ b/app/src/main/res/values-de/strings.xml @@ -492,7 +492,8 @@ </string-array> <string name="custom_user_agent">Eigener User Agent</string> <string name="incognito_mode">Inkognito-Modus</string> - <string name="incognito_mode_summary">Löscht den Verlauf und den Cache nach jedem fertigen Laden einer Webseite. Zurück und vorwärts funktioniert im Inkognito-Modus nicht.</string> + <string name="incognito_mode_summary">Löscht den Verlauf und den Cache nach jedem fertigen Laden einer Webseite. + Im Incognito-Modus schliesst \'Zurück\'-Button den aktive Tab (oder die App, wenn nur ein Tab geöffnet wurde).</string> <string name="do_not_track">Nicht verfolgen (Do not track)</string> <string name="do_not_track_summary">Einen "Do Not Track-Header" senden, der freundlich anfragt, dass Webserver diesen Browser nicht nachverfolgen sollen.</string> <string name="allow_screenshots">Screenshots zulassen</string> diff --git a/app/src/main/res/values-es/strings.xml b/app/src/main/res/values-es/strings.xml index 2928a6ed..1152e50c 100644 --- a/app/src/main/res/values-es/strings.xml +++ b/app/src/main/res/values-es/strings.xml @@ -489,7 +489,8 @@ </string-array> <string name="custom_user_agent">Agente de usuario personalizado</string> <string name="incognito_mode">Modo incógnito</string> - <string name="incognito_mode_summary">Borrar el historial y el caché después de que cada página web termine de cargar. Adelante y atrás no funcionan en el Modo incógnito.</string> + <string name="incognito_mode_summary">Borrar el historial y el caché después de que cada página web termine de cargar. + En el modo Incógnito, Atrás cierra la pestaña (o la app si hay solo una pestaña).</string> <string name="do_not_track">No rastrear</string> <string name="do_not_track_summary">Enviar la cabecera de no rastrear (DNT) que educadamente sugiere que los servidores web no rastreen este navegador.</string> <string name="allow_screenshots">Permitir capturas de pantalla</string> diff --git a/app/src/main/res/values-fr/strings.xml b/app/src/main/res/values-fr/strings.xml index afd9e9dc..a2923cee 100644 --- a/app/src/main/res/values-fr/strings.xml +++ b/app/src/main/res/values-fr/strings.xml @@ -489,7 +489,7 @@ </string-array> <string name="custom_user_agent">User agent personnalisé</string> <string name="incognito_mode">Mode Incognito</string> - <string name="incognito_mode_summary">Vider l\'historique et le cache après le chargement de chaque page. Précédent et Suivant ne fonctionnent pas en mode Incognito.</string> + <string name="incognito_mode_summary">Vider l\'historique et le cache après le chargement de chaque page. In Incognito Mode, back closes the tab (or the app if there is only one tab).</string> <string name="do_not_track">Ne pas me pister</string> <string name="do_not_track_summary">Envoyer aux sites web un signal «Ne pas me pister» indiquant poliement que vous ne souhaitez pas être pisté</string> <string name="allow_screenshots">Autoriser captures d\'écrans</string> diff --git a/app/src/main/res/values-it/strings.xml b/app/src/main/res/values-it/strings.xml index 051a38b1..874c6972 100644 --- a/app/src/main/res/values-it/strings.xml +++ b/app/src/main/res/values-it/strings.xml @@ -488,7 +488,8 @@ </string-array> <string name="custom_user_agent">User agent personalizzato</string> <string name="incognito_mode">Modalità Incognito</string> - <string name="incognito_mode_summary">Cancella la cronologia e la cache al termine del caricamento di ogni pagina. I pulsanti Avanti e Indietro non funzionano in modalità incognito.</string> + <string name="incognito_mode_summary">Cancella la cronologia e la cache al termine del caricamento di ogni pagina. + In Modalità Incognito, il tasto "Indietro" chiude la scheda (o l\'applicazione se è aperta solo una scheda).</string> <string name="do_not_track">Non tracciare</string> <string name="do_not_track_summary">Invia un\'intestazione di non tracciamento per chiedere al web server di non tracciare il browser.</string> <string name="allow_screenshots">Permetti gli screenshot</string> diff --git a/app/src/main/res/values-ru/strings.xml b/app/src/main/res/values-ru/strings.xml index 51fafbd9..4e32aa5c 100644 --- a/app/src/main/res/values-ru/strings.xml +++ b/app/src/main/res/values-ru/strings.xml @@ -483,7 +483,8 @@ </string-array> <string name="custom_user_agent">ÐаÑÑÑаиваемÑй user agent</string> <string name="incognito_mode">Режим инкогниÑо</string> - <string name="incognito_mode_summary">ÐÑиÑÑка жÑÑнала и кÑÑа поÑле завеÑÑÐµÐ½Ð¸Ñ Ð·Ð°Ð³ÑÑзки каждой веб-ÑÑÑаниÑÑ. Ðнопки \'ÐпеÑед\' и \'Ðазад\' не ÑабоÑаÑÑ Ð² ÑÑом Ñежиме.</string> + <string name="incognito_mode_summary">ÐÑиÑÑка жÑÑнала и кÑÑа по завеÑÑении загÑÑзки каждой веб-ÑÑÑаниÑÑ. + Ð Ñежиме инкогниÑо нажаÑие кнопки Ðазад пÑÐ¸Ð²ÐµÐ´ÐµÑ Ðº закÑÑÑÐ¸Ñ Ð²ÐºÐ»Ð°Ð´ÐºÐ¸ (или пÑиложениÑ, еÑли оÑкÑÑÑа ÑолÑко одна вкладка).</string> <string name="do_not_track">Ðе оÑÑлеживаÑÑ</string> <string name="do_not_track_summary">ÐÑпÑавлÑÑÑ Ð·Ð°Ð³Ð¾Ð»Ð¾Ð²Ð¾Ðº \'Ðе оÑÑлеживаÑÑ\', пÑедлагаÑÑий веб-ÑеÑвеÑÑ Ð½Ðµ оÑÑлеживаÑÑ ÑÑÐ¾Ñ Ð±ÑаÑзеÑ.</string> <string name="allow_screenshots">РазÑеÑиÑÑ ÑкÑинÑоÑÑ</string> diff --git a/app/src/main/res/values-tr/strings.xml b/app/src/main/res/values-tr/strings.xml index 87c8351c..5f0b590c 100644 --- a/app/src/main/res/values-tr/strings.xml +++ b/app/src/main/res/values-tr/strings.xml @@ -461,7 +461,7 @@ </string-array> <string name="custom_user_agent">Ãzel kullanıcı aracısı</string> <string name="incognito_mode">Gizli Mod</string> - <string name="incognito_mode_summary">Her web sayfasının yüklenmesi bittikten sonra geçmiÅi ve önbelleÄi temizler. Ä°leri ve geri seçenekleri Gizli Mod aktifken çalıÅmaz.</string> + <string name="incognito_mode_summary">Her web sayfasının yüklenmesi bittikten sonra geçmiÅi ve önbelleÄi temizler. In Incognito Mode, back closes the tab (or the app if there is only one tab).</string> <string name="do_not_track">Takip Etme</string> <string name="do_not_track_summary">Web Serverlarına bu tarayıcıyı takip etmemesi için Takip Etme BaÅlıÄı gönderir.</string> <string name="allow_screenshots">Ekran görüntülerine izin ver</string> diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index e6c35629..91df074d 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -525,7 +525,7 @@ <string name="custom_user_agent">Custom user agent</string> <string name="system_default_user_agent" translatable="false">System default user agent</string> <!-- This item is referenced in code. It is never displayed on the screen. --> <string name="incognito_mode">Incognito Mode</string> - <string name="incognito_mode_summary">Clear the history and cache after each webpage finishes loading. Forward and back do not work in Incognito Mode.</string> + <string name="incognito_mode_summary">Clear the history and cache after each webpage finishes loading. In Incognito Mode, back closes the tab (or the app if there is only one tab).</string> <string name="do_not_track">Do Not Track</string> <string name="do_not_track_summary">Send the Do Not Track header, which politely suggests that web servers not track this browser.</string> <string name="allow_screenshots">Allow screenshots</string>