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

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

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

参考:

http://blog.csdn.net/DrakeBlue/article/details/39049495

http://blog.csdn.NET/hclydao/article/details/44240799

http://blog.csdn.Net/jdsjlzx/article/details/39495319

物理键盘映射过程:

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

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

frameworks/…/res/values/attrs.xml

-主要部分:android.jar

一、问题描述:

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

理想效果:

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

二、思路

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

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

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

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

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

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

三、部分代码

Android版本源码

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();    }

注意 注意 注意,重要的说三遍;

加入以上你都修改了,以为这样都OK了,那你就OUT了

看看下面的你或许根本不知道,慢慢研究吧!骚年

四、以下操作全在ubuntu系统下完成

可以参考:http://blog.csdn.net/y150481863/article/details/8023422

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

1.下载好ubuntu系统之后。

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

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

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

有问题:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,591评论 18 139
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,351评论 0 17
  • 最近面试聊到很多启动过程的事情: Android系统的启动过程中就已经解析了系统中安装应用的androidMani...
    乐之飞于阅读 1,239评论 0 1
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,567评论 18 399
  • 今日体验,今天给奥拓做保养,需要更换半轴,同事在拆卸的过程中发现半轴很不好撬下来,于是过去实际感受了一下确实不好下...
    京心达宁威阅读 102评论 0 0