Android在外接物理键盘时,如何强制调用系统软键盘?

第一次写,写的不好请见谅

参考:

物理键盘映射过程:

手机/system/usr/keylayout/*.kl :内核将keyCode映射成有含义的字符串

KeycodeLabels.h : framework 将字符串映射成keyEvent的keyCode

frameworks/…/res/values/attrs.xml

一、问题描述:

当平板连接上蓝牙扫描枪(外接物理键盘)时候,不能弹出软键盘输入,需要打开系统的输入法选择界面关闭硬件物理键盘后才能调用弹出系统软键盘;

理想效果:

在平板连接上蓝牙扫描枪后仍可以调用系统软键盘输入,将系统的物理键盘默认设置为关闭状态,或不需要开启关闭物理键盘,

1.首先:你要知道AndroidManifest.xml文件。这里这里有的信息对于理解程序又很大的意义。

2.学会使用grep命令。修改系统的源代码时候这个很重要,

3.当想修改一个程序时,先找到这个程序的位置,大部分只要修改framework 和package 两个文件夹下的内容

4.关键字,例如要修改statuBars。先使用hierarchyviewer查看statuBar属于那个部分。

5.根据图标,使用grep在framework中查找对应的位置。

6.然后就是修改程序了。

frameworks/base/services/Java/com/android/server/wm/WindowManagerService.java

关键代码:行6618 computeScreenConfigurationLocked()方法中

[java] view plaincopy在CODE上查看代码片派生到我的代码片

boolean hardKeyboardAvailable = config.keyboard!= Configuration.KEYBOARD_NOKEYS;if (hardKeyboardAvailable != mHardKeyboardAvailable) {                  mHardKeyboardAvailable = hardKeyboardAvailable;mHardKeyboardEnabled = !hardKeyboardAvailable;mH.removeMessages(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);mH.sendEmptyMessage(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);}              if (!mHardKeyboardEnabled) {                  config.keyboard= Configuration.KEYBOARD_NOKEYS;}

将mHardKeyboardEnabled直接改成false

这样改软键盘是能用但是物理键盘是用不了的

最后研究代码frameworks\base\services\core\java\com\android\server\wm\WindowManagerService.java

如果把updateShowImeWithHardKeyboard方法中的showImeWithHardKeyboard变量直接置为true,则可以实现软键盘与物理键盘的同时使用,

但此举修改影响范围很大,不推荐。

publicvoidupdateShowImeWithHardKeyboard() {//modified by Janning for enble the HardKeyboard startfinalbooleanshowImeWithHardKeyboard = Settings.Secure.getIntForUser(                mContext.getContentResolver(), Settings.Secure.SHOW_IME_WITH_HARD_KEYBOARD,0,                mCurrentUserId) ==1;//final boolean showImeWithHardKeyboard = true;//modified by Janning for enble the HardKeyboard endsynchronized(mWindowMap) {if(mShowImeWithHardKeyboard != showImeWithHardKeyboard) {                mShowImeWithHardKeyboard = showImeWithHardKeyboard;                mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);            }        }    }

后续继续研究代码发现在WindowManagerService.java的computeScreenConfigurationLocked方法中有通过判断当前物理键盘类型来控制是否同时启用软件盘的处理逻辑:

boolean computeScreenConfigurationLocked(Configuration config) {        if (!mDisplayReady) {            return false;}        // TODO(multidisplay): For now, apply Configuration to main screen only.        final DisplayContent displayContent = getDefaultDisplayContentLocked();// Use the effective"visual"dimensions based on current rotation        final boolean rotated = (mRotation == Surface.ROTATION_90                || mRotation == Surface.ROTATION_270);final int realdw = rotated ?                displayContent.mBaseDisplayHeight: displayContent.mBaseDisplayWidth;final int realdh = rotated ?                displayContent.mBaseDisplayWidth: displayContent.mBaseDisplayHeight;int dw = realdw;int dh = realdh;if (mAltOrientation) {            if (realdw > realdh) {                // Turn landscape into portrait.                int maxw = (int)(realdh/1.3f);if (maxw < realdw) {                    dw = maxw;}            } else {                // Turn portrait into landscape.                int maxh = (int)(realdw/1.3f);if (maxh < realdh) {                    dh = maxh;}            }        }        if (config != null) {            config.orientation= (dw <= dh) ? Configuration.ORIENTATION_PORTRAIT :                    Configuration.ORIENTATION_LANDSCAPE;}        // Update application display metrics.        final int appWidth = mPolicy.getNonDecorDisplayWidth(dw, dh, mRotation);final int appHeight = mPolicy.getNonDecorDisplayHeight(dw, dh, mRotation);final DisplayInfo displayInfo = displayContent.getDisplayInfo();synchronized(displayContent.mDisplaySizeLock) {            displayInfo.rotation= mRotation;displayInfo.logicalWidth= dw;displayInfo.logicalHeight= dh;displayInfo.logicalDensityDpi= displayContent.mBaseDisplayDensity;displayInfo.appWidth= appWidth;displayInfo.appHeight= appHeight;displayInfo.getLogicalMetrics(mRealDisplayMetrics,                    CompatibilityInfo.DEFAULT_COMPATIBILITY_INFO, null);displayInfo.getAppMetrics(mDisplayMetrics);mDisplayManagerInternal.setDisplayInfoOverrideFromWindowManager(                    displayContent.getDisplayId(), displayInfo);}        if (false) {            Slog.i(TAG,"Set app display size: "+ appWidth +" x "+ appHeight);}        final DisplayMetrics dm = mDisplayMetrics;mCompatibleScreenScale = CompatibilityInfo.computeCompatibleScaling(dm,                mCompatDisplayMetrics);if (config != null) {            config.screenWidthDp= (int)(mPolicy.getConfigDisplayWidth(dw, dh, mRotation)                    / dm.density);config.screenHeightDp= (int)(mPolicy.getConfigDisplayHeight(dw, dh, mRotation)                    / dm.density);computeSizeRangesAndScreenLayout(displayInfo, rotated, dw, dh, dm.density, config);config.compatScreenWidthDp= (int)(config.screenWidthDp/ mCompatibleScreenScale);config.compatScreenHeightDp= (int)(config.screenHeightDp/ mCompatibleScreenScale);config.compatSmallestScreenWidthDp= computeCompatSmallestWidth(rotated, dm, dw, dh);config.densityDpi= displayContent.mBaseDisplayDensity;// Update the configuration based on available input devices, lid switch,            //andplatform configuration.            config.touchscreen= Configuration.TOUCHSCREEN_NOTOUCH;config.keyboard= Configuration.KEYBOARD_NOKEYS;config.navigation= Configuration.NAVIGATION_NONAV;int keyboardPresence =0;int navigationPresence =0;final InputDevice[] devices = mInputManager.getInputDevices();final int len = devices.length;for (int i =0; i < len; i++) {InputDevice device = devices[i];if (!device.isVirtual()) {                    final int sources = device.getSources();final int presenceFlag = device.isExternal() ?                            WindowManagerPolicy.PRESENCE_EXTERNAL :                                    WindowManagerPolicy.PRESENCE_INTERNAL;if (mIsTouchDevice) {                        if ((sources & InputDevice.SOURCE_TOUCHSCREEN) ==                                InputDevice.SOURCE_TOUCHSCREEN) {                            config.touchscreen= Configuration.TOUCHSCREEN_FINGER;}                    } else {                        config.touchscreen= Configuration.TOUCHSCREEN_NOTOUCH;}                    if ((sources & InputDevice.SOURCE_TRACKBALL) == InputDevice.SOURCE_TRACKBALL) {                        config.navigation= Configuration.NAVIGATION_TRACKBALL;navigationPresence |= presenceFlag;} else if ((sources & InputDevice.SOURCE_DPAD) == InputDevice.SOURCE_DPAD                            && config.navigation== Configuration.NAVIGATION_NONAV) {                        config.navigation= Configuration.NAVIGATION_DPAD;navigationPresence |= presenceFlag;}                    // 判断该物理设备的类型, InputDevice.KEYBOARD_TYPE_ALPHABETIC 是表示物理键盘设备                    if (device.getKeyboardType() == InputDevice.KEYBOARD_TYPE_ALPHABETIC) {                        config.keyboard= Configuration.KEYBOARD_QWERTY;keyboardPresence |= presenceFlag;}                    // 获取物理设备名称,判断是否是指定的名称,如果是则把 config.keyboard// 的属性置为 Configuration.KEYBOARD_NOKEYS ,如此则可以同时兼容软键盘                    // 物理键盘与软键盘可以同时启用                    //Addby Janning start                    // for show IME with HardKeyboard                    if (device.getName().equals("XXX-vinput-keypad")) {                        Slog.w("SLCODE","the hard device name is: "+ device.getName());config.keyboard= Configuration.KEYBOARD_NOKEYS;}                    //Addby Janning end                }            }            if (config.navigation== Configuration.NAVIGATION_NONAV && mHasPermanentDpad) {                config.navigation= Configuration.NAVIGATION_DPAD;navigationPresence |= WindowManagerPolicy.PRESENCE_INTERNAL;}            // Determine whether a hard keyboard is availableandenabled.            boolean hardKeyboardAvailable = config.keyboard!= Configuration.KEYBOARD_NOKEYS;if (hardKeyboardAvailable != mHardKeyboardAvailable) {                mHardKeyboardAvailable = hardKeyboardAvailable;mH.removeMessages(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);mH.sendEmptyMessage(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);}            if (mShowImeWithHardKeyboard) {                config.keyboard= Configuration.KEYBOARD_NOKEYS;}            // Let the policy update hidden states.            config.keyboardHidden= Configuration.KEYBOARDHIDDEN_NO;config.hardKeyboardHidden= Configuration.HARDKEYBOARDHIDDEN_NO;config.navigationHidden= Configuration.NAVIGATIONHIDDEN_NO;mPolicy.adjustConfigurationLw(config, keyboardPresence, navigationPresence);}        return true;}    public boolean isHardKeyboardAvailable() {        synchronized (mWindowMap) {            return mHardKeyboardAvailable;}    }    public void updateShowImeWithHardKeyboard() {        // 此处修改也可以实现物理键盘与软键盘的同时启用,即把showImeWithHardKeyboard 直接置为 true,        // 但此方法影响太大,不推荐该方案,建议根据设备名称判断 修改config.keyboard属性值(代码见上文)        //changed by Janning start        //modified by Janning for enble the HardKeyboard start        final boolean showImeWithHardKeyboard = Settings.Secure.getIntForUser(                mContext.getContentResolver(), Settings.Secure.SHOW_IME_WITH_HARD_KEYBOARD,0,                mCurrentUserId) ==1;//final boolean showImeWithHardKeyboard = true;//modified by Janning for enble the HardKeyboard end        //changed by Janning end        synchronized (mWindowMap) {            if (mShowImeWithHardKeyboard != showImeWithHardKeyboard) {                mShowImeWithHardKeyboard = showImeWithHardKeyboard;mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);}        }    }

当插入物理键盘后通知栏会弹出相应的选择键盘布局通知,对于该通知可以选择隐藏:

根据字符串查找到是在 frameworks\base\services\core\java\com\android\server\input\InputManagerService.java 中调用显示该通知的,

进一步分析代码发现是在 deliverInputDevicesChanged 方法中控制通知的显示。

InputManagerService.java

privatevoiddeliverInputDevicesChanged(InputDevice[] oldInputDevices) {        。。。。。。。。。。。。。。。。if(missingLayoutForExternalKeyboard) {if(missingLayoutForExternalKeyboardAdded) {if(multipleMissingLayoutsForExternalKeyboardsAdded) {// We have more than one keyboard missing a layout, so drop the// user at the generic input methods page so they can pick which// one to set.showMissingKeyboardLayoutNotification(null);                    }else{// 如果只插入了一个物理键盘则判断该物理键盘的名称是否是指定的,如果是则不让其显示键盘布局的通知// Modify by Janning beginif(keyboardMissingLayout !=null&& !keyboardMissingLayout.getName().equals("XXXX-vinput-keypad")) {                            showMissingKeyboardLayoutNotification(keyboardMissingLayout);                        }// Modify by Janning end}                }            }elseif(mKeyboardLayoutNotificationShown) {                hideMissingKeyboardLayoutNotification();            }        }        mTempFullKeyboards.clear();    }

操作步骤(不具体说明):

1.下载好ubuntu系统之后。

2.在找到sdk位置(就是你的SDK 找到你用的那个系统,在到ubuntu修改)。

………………………..(做起来不是这么简单)

3.再用ub系统上生成.jar,再放在SDK里面

有问题:

android外接键盘打汉字,Android在外接物理键盘时,如何强制调用系统软键盘相关推荐

  1. Android在外接物理键盘时,如何强制调用系统软键盘

    Android在外接物理键盘时,如何强制调用系统软键盘? 可以加群:466059178,可以共同讨论,大神指导 第一次写,写的不好请见谅 参考: http://blog.csdn.net/DrakeB ...

  2. 系统软键盘Android在外接物理键盘时,如何强制调用系统软键盘?

    第一次写,写的不好请见谅 物理键盘映射过程: 手机/system/usr/keylayout/*.kl :内核将keyCode映射成有含义的字符串 KeycodeLabels.h : framewor ...

  3. Android调用系统软键盘删除键 以达到删除输入框里面emoji表情的功能

    想必大家在很多项目里都会涉及到聊天功能,为了在聊天里面加入表情,我们可能会去自定义表情列表布局,而一般的表情列表的右下角都有一个删除按钮,用于删除输入框里面的文字.自定义表情.emoji表情等,那么我 ...

  4. Visual studio之C# 调用系统软键盘(外部osk.exe)

    背景 App需要调用系统软键盘输入数据,自己去实现软键盘有些复杂,现在又是急需,没时间去研究实现软键盘,所以只能调用系统软键盘. 正文 需要添加命名空间, using System.Diagnosti ...

  5. 【MFC】MFC中调用系统软键盘的几种方法

    1.直接运行微软系统自带的虚拟键盘程序"osk.exe" 在普通MFC项目中可以调用ShellExecute或者WinExec方法来直接运行微软系统自带的虚拟键盘程序"o ...

  6. C# 调用系统软键盘帮助类(兼容.netframework2.0)

    前言 最近再做触屏系统的时候需要手动调用打开系统软键盘的需求,网上查找到的资料很多,在高版本的fx上也能用,但是刚好我这个现场程序是基于fx2.0开发的,只能在之前的基础上改造支持.目前已经调整到可以 ...

  7. python调用系统软键盘_python库 pywinio虚拟键盘使用

    NOTICE: You must have a ps/2 keyboard connected to your PC if you test on win10 or later, otherwise ...

  8. Android --- 进入页面时,不弹出软键盘。当 EditText 被点击时不弹出软键盘,而是调用其他选择器,比如时间选择器等

    功能需求: 1.进入页面时,不弹出软键盘 2.当 EditText 被点击时不弹出软键盘,而是调用其他选择器,比如时间选择器等 搜索百度测试: 1.百度上说用 editText.setInputTyp ...

  9. android仿微信、QQ等聊天界面,实现点击输入框弹出软键盘、点击其他区域收起软键盘,默认滑动至最低端

    如图所示,点击输入框及选择图片和发送按钮时软键盘显示且不消失,点击其他区域,则隐藏软键盘. 主要代码如下: override fun dispatchTouchEvent(ev: MotionEven ...

最新文章

  1. 生产指挥调度系统_市安全生产应急救援指挥中心将大型装载机械设备储备信息纳入应急指挥调度系统...
  2. JSONObject.fromObject--JSON与对象的转换
  3. Hello,Word宏!
  4. golang http 操作 简介
  5. ecs要按两次才有效_猫咪想要增肥有什么办法?吃是最简单有效的了,但要吃对了才行...
  6. 牛客题霸 [二叉树的镜像]C++题解/答案
  7. 数仓建模 项目_模型设计_数仓建模 PDF 下载
  8. NLayerAppV3--.net Core2实现的DDD分层架构
  9. maven 镜像_Maven(一)
  10. c语言vs2017链表出错,【C++单链表就地逆置】程序无error错误,求指点TAT。
  11. 设计模式速查手册-创建型
  12. NSX发布Guest Introspection虚拟机时,主机报错的解决方法
  13. Ubuntu上安装visual studio code
  14. mysql朋友圈设计_微信朋友圈技术实现设想
  15. 逻辑学中的思维规律:同一律,不矛盾律,排中律,充足理由律
  16. 物联网是什么意思?物联网概念是什么?
  17. Apache Airflow调度中心发布任务步骤
  18. 大部分人朋友圈仅三天可见的原因,竟然是……
  19. 怎样下载安装python编程软件3.85_【Python下载】Python安装 v3.8.1 官方版-七喜软件园...
  20. 2016-1-21高博活动

热门文章

  1. python三引号注释_python使用三引号来注释的具体原因?
  2. 3.MongoDB数据查询
  3. 元素(块、行内、行内块
  4. Jarvis:一个值得关注的多模态端到端人机对话框架,针对所有行业适配
  5. 斯坦福CS224n追剧计划【大结局】:NLP和深度学习的未来
  6. 拒绝无脑试错:写给萌新的“科学炼丹”入门手册
  7. 史上最萌最认真的机器学习/深度学习/模式识别入门指导手册(一)
  8. 微服务系列:Dubbo与SpringCloud的Ribbon、Hystrix、Feign的优劣势比较
  9. 论文浅尝 | 融合多层次领域知识的分子图对比学习
  10. 专家:人工智能开始对现实世界产生重大影响​​