最近准备要做一个项目,需要用到自定义小键盘来确保安全,而且还需要精确获得用户点击键盘时的落点位置、力度、指尖接触屏幕的面积等参数。
down下来的代码用到我的项目时,出现了各种问题:
EditText会在应用打开的使用获得焦点,导致直接弹出系统输入法,而不是自定义输入法。解决的办法是使EditText在应用打开时不获取焦点,于是我在activity对应的布局文件的全局Layout标签中加入
android:focusable="true"
android:focusableInTouchMode="true"
activity运行时,EditText就失去了焦点,系统输入法也不会弹出。然后对EditText加入触摸监听器,当点击EditText控件时,就响应弹出我们的键盘
et.setOnTouchListener(new View.OnTouchListener(){
@Override
public boolean onTouch(View v, MotionEvent event) {
int inputback = et.getInputType();
et.setInputType(InputType.TYPE_NULL);
AdvantageKeyboard kb = new AdvantageKeyboard(act,ctx,et);
kb.showKeyboard();
et.setInputType(inputback);
v.performClick();
return false;
}
});
点击时,键盘弹出的提示框里面没有字体。例如这样
theme的问题,因为我用的是android5.0.1的API,自动生成项目时,生成的activity是直接继承ActionBarActivity的,直接改theme会导致应该出现秒退现象。如果要改theme就要先改继承ActionBarActivity为Activity,然后再去改theme,但是我很不喜欢这样,因为这样没有通用性,后来在google了一下外国人的贴子,才了解到这个弹出的东西叫preview,我们可以修改它的布局的。
KeyboardView中加入android:keyPreviewLayout标签,加入后如下:
<pre name="code" class="html"> <android.inputmethodservice.KeyboardView
android:id="@+id/keyboard_view"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:focusable="true"
android:focusableInTouchMode="true"
android:background="@color/black"
android:keyBackground="@drawable/btn_keyboard_key"
android:keyPreviewLayout="@layout/key_preview_layout"
android:keyTextColor="@color/white"
android:visibility="gone" />
文件夹中,新建一个key_preview_layout.xml文件,里面这样写
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="40sp"
android:textColor="@android:color/white"
android:gravity="center"
android:background="#ff8888ff"/>
android:background就是我们的提示框的背景颜色,android:textColor是字体颜色,改成你喜欢的就OK了。
3.加入键盘的触摸响应,并记录点击的相关信息(力度,接触面积,落点坐标等)。
keyboardView加入触摸监听器,例如这样写构造方法。
public AdvantageKeyboard(Activity act, Context ctx, EditText edit) {
this.act = act;
this.ctx = ctx;
this.ed = edit;
kb_letter = new Keyboard(this.ctx, R.xml.qwerty);
kb_number = new Keyboard(this.ctx, R.xml.symbols);
keyboardView = (KeyboardView) this.act.findViewById(R.id.keyboard_view);
keyboardView.setKeyboard(kb_number);
is_nun = true;
keyboardView.setEnabled(true);
keyboardView.setPreviewEnabled(true);
keyboardView.setOnKeyboardActionListener(action_listener);
keyboardView.setOnTouchListener(touch_listener);//加入触摸监听器
}
new一个我们处理这些数据的监听器,需要我们“重载”(可能叫“覆盖”比较合适)OnTouch函数,把采集的数据显示到Locat上。需要注意的是,这个OnTouch函数假如返回值为true,就代表这个动作已被处理,为false就代表没被处理,由于后面还有其他函数需要获得此动作,所以这里返回false,否则,接下来的对按键动作的响应(字体输出等)将不会发生。
private OnTouchListener touch_listener = new OnTouchListener(){
@Override
public boolean onTouch(View v, android.view.MotionEvent event) {
float pressure, size, rx, ry, x, y, interval;
long time, down_time;
int action = event.getAction();
switch (action) {
case (MotionEvent.ACTION_DOWN):
pressure = event.getPressure();
size = event.getSize();
time = event.getEventTime();
down_time = event.getDownTime();
x = event.getX();
y = event.getY();
rx = event.getRawX();
ry = event.getRawY();
interval = (float) 0.0;
if(is_first_press){
interval = down_time - last_down_time;
}
is_first_press = true;
last_down_time = down_time;
Log.i("!!!!!!!!!","pressure:"+String.valueOf(pressure)+"\n"
+"size:"+String.valueOf(size)+"\n"
+"time:"+String.valueOf(time)+"\n"
+"downtime:"+String.valueOf(down_time)+"\n"
+"x:"+String.valueOf(x)+" y:"+String.valueOf(y)+"\n"
+"rx:"+String.valueOf(rx)+" ry:"+String.valueOf(ry)+"\n"
+"interval:"+String.valueOf(interval)+"\n");
break;
case (MotionEvent.ACTION_UP):
v.performClick();
break;
case (MotionEvent.ACTION_MOVE):
break;
}
return false;
}
};
然后搞定了...