From 5a64a2f70b662d2f9fea7f8e8bee3aa79ede5971 Mon Sep 17 00:00:00 2001 From: Gibran Chevalley Date: Thu, 13 Jun 2024 14:26:19 +0200 Subject: [PATCH 1/2] Keep keyboard open on configuration changed --- .../lib/richhtmleditor/Extensions.kt | 11 +++++++ .../richhtmleditor/RichHtmlEditorWebView.kt | 32 ++++++++++++++++++- .../richhtmleditor/executor/KeyboardOpener.kt | 25 +++++++++++++-- 3 files changed, 65 insertions(+), 3 deletions(-) diff --git a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/Extensions.kt b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/Extensions.kt index b5ec284..1b8c23b 100644 --- a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/Extensions.kt +++ b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/Extensions.kt @@ -1,6 +1,9 @@ package com.infomaniak.lib.richhtmleditor import android.content.Context +import android.os.Build +import android.os.Bundle +import android.view.AbsSavedState import android.webkit.WebView import java.io.BufferedReader @@ -33,3 +36,11 @@ internal fun WebView.injectCss(css: String) { evaluateJavascript(addCssJs, null) } + +fun Bundle.getParcelableCompat(key: String, clazz: Class): AbsSavedState? { + return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { + getParcelable(key, clazz) + } else { + getParcelable(key) + } +} diff --git a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt index 847a222..2ad8244 100644 --- a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt +++ b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt @@ -3,10 +3,14 @@ package com.infomaniak.lib.richhtmleditor import android.annotation.SuppressLint import android.content.Context import android.graphics.Rect +import android.os.Bundle +import android.os.Parcelable import android.util.AttributeSet +import android.view.AbsSavedState import android.view.ViewGroup import android.webkit.WebView import android.webkit.WebViewClient +import androidx.core.os.bundleOf import androidx.core.view.updateLayoutParams import com.infomaniak.lib.richhtmleditor.executor.JsExecutableMethod import com.infomaniak.lib.richhtmleditor.executor.JsExecutor @@ -38,6 +42,8 @@ class RichHtmlEditorWebView @JvmOverloads constructor( defStyleAttr: Int = 0, ) : WebView(context, attrs, defStyleAttr) { + private var keepKeyboardOpenedOnConfigurationChanged: Boolean = false + private val documentInitializer = DocumentInitializer() private val jsExecutor = JsExecutor(this) private val scriptCssInjector = ScriptCssInjector(this) @@ -135,6 +141,7 @@ class RichHtmlEditorWebView @JvmOverloads constructor( } fun requestFocusAndOpenKeyboard() { + keepKeyboardOpenedOnConfigurationChanged = true keyboardOpener.executeWhenDomIsLoaded(Unit) } @@ -143,9 +150,27 @@ class RichHtmlEditorWebView @JvmOverloads constructor( jsExecutor.executeWhenDomIsLoaded(JsExecutableMethod("exportHtml")) } + override fun onSaveInstanceState(): Parcelable { + val superState = super.onSaveInstanceState() + return bundleOf( + KEYBOARD_SHOULD_REOPEN_KEY to keepKeyboardOpenedOnConfigurationChanged, + SUPER_STATE_KEY to superState, + ) + } + + override fun onRestoreInstanceState(state: Parcelable?) { + (state as Bundle?)?.getBoolean(KEYBOARD_SHOULD_REOPEN_KEY)?.let { keepKeyboardOpenedOnConfigurationChanged = it } + super.onRestoreInstanceState(state?.getParcelableCompat(SUPER_STATE_KEY, AbsSavedState::class.java)) + } + override fun onFocusChanged(focused: Boolean, direction: Int, previouslyFocusedRect: Rect?) { super.onFocusChanged(focused, direction, previouslyFocusedRect) - if (focused) jsExecutor.executeWhenDomIsLoaded(JsExecutableMethod("requestFocus")) + if (focused) { + jsExecutor.executeWhenDomIsLoaded(JsExecutableMethod("requestFocus")) + if (keepKeyboardOpenedOnConfigurationChanged) keyboardOpener.executeWhenDomIsLoaded(Unit) + } else { + keepKeyboardOpenedOnConfigurationChanged = false + } } override fun onDetachedFromWindow() { @@ -206,4 +231,9 @@ class RichHtmlEditorWebView @JvmOverloads constructor( private fun unsupported() { throw UnsupportedOperationException("Use setHtml() instead") } + + companion object { + private const val KEYBOARD_SHOULD_REOPEN_KEY = "keyboardShouldReopen" + private const val SUPER_STATE_KEY = "superState" + } } diff --git a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt index 7402f79..dc887da 100644 --- a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt +++ b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt @@ -2,13 +2,34 @@ package com.infomaniak.lib.richhtmleditor.executor import android.app.Activity import android.view.View +import android.view.ViewTreeObserver import android.view.inputmethod.InputMethodManager internal class KeyboardOpener(private val view: View) : JsLifecycleAwareExecutor() { override fun executeImmediately(value: Unit) { if (view.requestFocus()) { - val inputMethodManager = view.context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager - inputMethodManager.showSoftInput(view, InputMethodManager.SHOW_IMPLICIT) + if (view.hasWindowFocus()) { + openKeyboard() + } else { + // The window won't have the focus most of the time when the configuration changes and we want to reopen the + // keyboard right away. When this happen, we need to wait for the window to get the focus before opening the + // keyboard. + val listener = object : ViewTreeObserver.OnWindowFocusChangeListener { + override fun onWindowFocusChanged(hasFocus: Boolean) { + if (hasFocus) { + openKeyboard() + view.viewTreeObserver.removeOnWindowFocusChangeListener(this) + } + } + } + + view.viewTreeObserver.addOnWindowFocusChangeListener(listener) + } } } + + private fun openKeyboard() { + val inputMethodManager = view.context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager + inputMethodManager.showSoftInput(view, InputMethodManager.SHOW_IMPLICIT) + } } From da7a566b011372349cdbaa0c691bbbf2726a8199 Mon Sep 17 00:00:00 2001 From: Gibran Chevalley Date: Thu, 13 Jun 2024 15:02:58 +0200 Subject: [PATCH 2/2] Remove listeners that could have stayed waiting for focus if getting focus is too slow --- .../lib/richhtmleditor/RichHtmlEditorWebView.kt | 1 + .../lib/richhtmleditor/executor/KeyboardOpener.kt | 10 +++++++++- 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt index 2ad8244..6a73d5d 100644 --- a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt +++ b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/RichHtmlEditorWebView.kt @@ -175,6 +175,7 @@ class RichHtmlEditorWebView @JvmOverloads constructor( override fun onDetachedFromWindow() { super.onDetachedFromWindow() + keyboardOpener.removePendingListener() jsBridgeJob.cancel() } diff --git a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt index dc887da..774ffa7 100644 --- a/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt +++ b/rich-html-editor/src/main/java/com/infomaniak/lib/richhtmleditor/executor/KeyboardOpener.kt @@ -6,6 +6,9 @@ import android.view.ViewTreeObserver import android.view.inputmethod.InputMethodManager internal class KeyboardOpener(private val view: View) : JsLifecycleAwareExecutor() { + + private var listener: ViewTreeObserver.OnWindowFocusChangeListener? = null + override fun executeImmediately(value: Unit) { if (view.requestFocus()) { if (view.hasWindowFocus()) { @@ -14,7 +17,7 @@ internal class KeyboardOpener(private val view: View) : JsLifecycleAwareExecutor // The window won't have the focus most of the time when the configuration changes and we want to reopen the // keyboard right away. When this happen, we need to wait for the window to get the focus before opening the // keyboard. - val listener = object : ViewTreeObserver.OnWindowFocusChangeListener { + listener = object : ViewTreeObserver.OnWindowFocusChangeListener { override fun onWindowFocusChanged(hasFocus: Boolean) { if (hasFocus) { openKeyboard() @@ -28,6 +31,11 @@ internal class KeyboardOpener(private val view: View) : JsLifecycleAwareExecutor } } + fun removePendingListener() { + view.viewTreeObserver.removeOnWindowFocusChangeListener(listener) + listener = null + } + private fun openKeyboard() { val inputMethodManager = view.context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager inputMethodManager.showSoftInput(view, InputMethodManager.SHOW_IMPLICIT)