Read Keyboard events in Android WebView

15,697

Solution 1

Caution! This solution supports only English letters.

Accomplished without any access to the HTML source code or JS events. Works for me on Android 5.0.2 for both soft and hardware keyboards.

public class MyWebView extends WebView {


    @Override
    public InputConnection onCreateInputConnection(EditorInfo outAttrs) {
        return new BaseInputConnection(this, false); //this is needed for #dispatchKeyEvent() to be notified.
    }

    @Override
    public boolean dispatchKeyEvent(KeyEvent event) {
        boolean dispatchFirst = super.dispatchKeyEvent(event);
        // Listening here for whatever key events you need
        if (event.getAction() == KeyEvent.ACTION_UP)
            switch (event.getKeyCode()) {
                case KeyEvent.KEYCODE_SPACE:
                case KeyEvent.KEYCODE_ENTER:
                    // e.g. get space and enter events here
                    break;
            }
        return dispatchFirst;
    }
}

The trick here is overriding the system-provided input implementation of InputConnection with a simplified one. Preventing developers from accessing the events by default was made by Googlers on purpose. Because the key event input isn't the only one anymore. There're gestures, voice and more is coming. Official recommendation is to "stop relying on legacy key events for text entry at all". Check out more details here: https://code.google.com/p/android/issues/detail?id=42904#c15

Solution 2

You can use onUnhandledKeyEvent() function overrid on WebViewClient class. Works perfectly for me.

onUnhandledKeyEvent

@Override
public void onUnhandledKeyEvent(WebView view, KeyEvent event) {
      return;
}

whenever a user presses any key on the keyboard, this event will get fire.
Please let me know if that's works for you?

Solution 3

I found a simple way to do this in Kotlin with a JavaScript callback.

    webview.apply {
        settings.javaScriptEnabled = true

        addJavascriptInterface(object {
            @JavascriptInterface
            fun keyInteraction(): Boolean {
                //<handle key interaction here>
                return true
            }
        },"Android")

        webViewClient = object : WebViewClient() {
            override fun onPageFinished(view: WebView?, url: String?) {
                loadingFinished()
                view?.loadUrl(
                    "javascript:(function() {" +
                    "    window.parent.addEventListener (" +
                    "        'keyup'," +
                    "        function(event) {" +
                    "            Android.keyInteraction();" +
                    "        });" +
                    "})()"
                )
                super.onPageFinished(view, url)
            }
        }
    }
Share:
15,697

Related videos on Youtube

PC.
Author by

PC.

Software Engineer. Languages used: Java, C++, C#, .Net Experienced in Enterprise Web App and Mobile App Development. A part of Firefox development community.

Updated on September 29, 2022

Comments

  • PC.
    PC. over 1 year

    I'm trying to listen to key events in android web view. Example when user is filling a form, I should receive the key events. This is my WebView code

    public class MyWebView extends WebView implements OnKeyListener
    {
        .....
    
        @Override
        public boolean onKeyDown(int keyCode, KeyEvent event)
        {
            Log.i("PcWebView", "onKeyDown keyCode=" + keyCode);
            return super.onKeyDown(keyCode, event);
        }
    
        @Override
        public boolean onKeyUp(int keyCode, KeyEvent event)
        {
            Log.i("PcWebView", "onKeyUp keyCode=" + keyCode);
            return super.onKeyUp(keyCode, event);
        }
    
        @Override // Listener is initialized in the constructor
        public boolean onKey(View v, int keyCode, KeyEvent event)
        {
            Log.i("PcWebView", "onKey keyCode=" + keyCode);
            return false;
        }
    
        .....
    
    }
    

    neither of the methods onKeyDown, onKeyUp and onKey are being called when user types into the textboxes or textareas. Is it possible to achieve this or has Android restricted this b'cos of a possible privacy breach?

    • PC.
      PC. over 10 years
      No, this type of action is considered to be insecure as passwords etc can be recorded. So it is not allowed in android.
  • riwnodennyk
    riwnodennyk about 9 years
    @PC. yes, sure. E.g. if the page loaded into the WebView is something like google.com.ua , you get notified on every letter input into the text box.
  • hmac
    hmac over 6 years
    This can cause problems. Because you ignore the EditorInfo (which carries InputType information) this can lead to the wrong keyboard being shown. eg. last used keyboard is numeric but user clicks into text field, the numeric keyboard appears (interestingly this was only happening on certain devices eg. HTC One M8). I was trying to hide keyboard on Enter, I had to achieve this a different way - I hid keyboard when URL is about to load (webViewClient.shouldOverrideUrlLoading()).
  • Wildcopper
    Wildcopper almost 5 years
    I needed to know if any virtual keyboard event arrived in a text field in my WebView for an inactivity timer, and this works perfectly. Thanks.
  • Arpit J.
    Arpit J. over 3 years
    Note: call requires API level 28.