Android焦点控制

本文探讨了Android系统中焦点控制的重要性,并详细介绍了如何查看和控制焦点。内容包括使用findFocus()、hasFocus()方法检查焦点状态,利用focusSearch()方法寻找方向上的下一个焦点,以及在XML布局中设置nextFocus*属性自定义焦点转移。此外,还提供了一个通过焦点控制实现激活码输入框自动切换的DEMO示例。

 

随着Android系统的应用越来越广泛,Android系统应用控制可以使用方向键、遥控器、轨迹球、触摸屏等方式。因此,了解Android焦点的控制是十分重要的,本文就此分享一些关于焦点控制的方法和经验。​

 

一、查看焦点​的相关方法

在View类中有成员函数findFocus(),可以通过该函数来获取当前View中的焦点。如果希望得到当前Activity的 视图 可以使用 :ActivityName.this.getWindow().getDecorView();      方法来获取当前窗口的顶层视图。

       

针对于某一个控件是否获取到了焦点,可以使用View的成员函数 hasFocus() 来查看当前控件是否获取到了焦点。​


当焦点在某个视图上时,我们希望找到该视图在某个方向上的下一个焦点,则我们可以用以下方法:

public View focusSearch(int direction)   该方法可以找到在当前聚焦视图的某个特定方向上最近的视图,需要注意的是该方法实际上并没有聚焦的作用。

方向(direction)包括View中的 FOCUS_BACKWARD FOCUS_FORWARD FOCUS_UP  FOCUS_DOWN FOCUS_LEFT FOCUS_RIGHT 。

 

二、焦点的控制

 让我们先来看几个重要的获取焦点的相关方法:

 

public final boolean isFocusable() 返回该视图是否能获取焦点
public final boolean isFocusableInTouchMode()​ 返回该视图在触摸模式下是否能获取焦点。​
public void setFocusable(boolean focusable) 设置该视图是否能获取焦点
public void setFocusableInTouchMode(boolean focusableInTouchMode)设置该视图在触摸模式下是否能获取焦点
public final boolean requestFocus()调用这个函数即试图把焦点集中在一个视图或者一个视图的子类上。返回值为该视图或子类是否获取到了焦点。

其中我们需要关注的是 requestFocus(),若调用该函数返回false,则很可能是isFocusable()或isFocusableInTouchMode()返回false,因此我们在调用 requestFocus()时,尽可能在之前调用setFocusable(true)和setFocusableInTouchMode(true), 这样可以尽可能的保证视图能获取到焦点。需要注意的是,即使在遥控器、方向键操控的产品中,设置setFocusableInTouchMode(true)也是有意义的,不然很可能会丢失焦点,导致界面无法操控。

    

当我们在可以用方向键操作的产品中,当我们按下方向键,系统会选取聚焦视图在该方向上最近的视图聚焦,但在很多情况下,我们需要控制方向键对焦点的改变,因此我们可以通过在XML布局文件中设置该View的属性来重设焦点的改变方式

nextFocusDown

nextFocusLeft

nextFocusRight

nextFocusUp

如下例子:

若当前焦点在按钮12上,在默认算法中,我们按下down时,焦点可能会聚焦到按钮11中,而开发者希望在按下down时,焦点能到按钮1中,则可在XML中进行以下设置:

 <Button
 android:text="12"
 android:id="@+id/button12"
 android:layout_alignParentTop="true"
 android:layout_centerHorizontal="true"
 android:nextFocusDown="@+id/button1">
 </Button>

则此时我们按下down时,焦点会聚焦在按钮1中。


三、一个类似激活码输入框的DEMO

该例子是通过焦点控制,当一个输入框输入满时,自动切换到下一个输入框继续输入。

让我们先来看下效果:



Java代码:

public class MainActivity extends Activity {

    EditText[] input = new EditText[4];

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        input[0] = (EditText)findViewById(R.id.inputZero);
        input[1] = (EditText)findViewById(R.id.inputOne);
        input[2] = (EditText)findViewById(R.id.inputTwo);
        input[3] = (EditText)findViewById(R.id.inputThree);

        for (int i=0;i<4;i++){
            input[i].addTextChangedListener(new ChangeFocusListener(i));
        }
    }

    class ChangeFocusListener implements TextWatcher{

        int id = 0;

        public ChangeFocusListener(int id){
            this.id = id;
        }

        @Override
        public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

        }

        @Override
        public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

        }

        @Override
        public void afterTextChanged(Editable editable) {
            if (editable.length()<4){
                return;
            }

            if (id == 3){
                Toast.makeText(MainActivity.this,"激活码输入完成", Toast.LENGTH_SHORT).show();
                return;
            }

            input[id+1].requestFocus();  //下一个输入框获得焦点
        }
    }
}

XML代码:

需要注意的是,获取焦点的控件记得要设置focusable和focusableInTouchMode为true,否则很可能会出问题,如不方便在XML里设置,在Java代码中设置也可。

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.liqk.focusdemo.MainActivity">

    <RelativeLayout
        android:background="#FF6100"
        android:layout_width="match_parent"
        android:layout_height="60dp">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerHorizontal="true"
            android:layout_centerVertical="true"
            android:text="请输入激活码"
            android:textSize="25dp"
            android:textColor="#ffffff"/>

    </RelativeLayout>

    <LinearLayout
        android:layout_centerVertical="true"
        android:layout_centerHorizontal="true"
        android:orientation="horizontal"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content">

        <EditText
            android:id="@+id/inputZero"
            android:maxLength="4"
            android:layout_width="70dp"
            android:layout_height="wrap_content"
            android:focusable="true"
            android:focusableInTouchMode="true" />

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text=" - " />

        <EditText
            android:id="@+id/inputOne"
            android:maxLength="4"
            android:layout_width="70dp"
            android:layout_height="wrap_content"
            android:focusable="true"
            android:focusableInTouchMode="true"/>

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text=" - " />

        <EditText
            android:id="@+id/inputTwo"
            android:maxLength="4"
            android:layout_width="70dp"
            android:layout_height="wrap_content"
            android:focusable="true"
            android:focusableInTouchMode="true"/>

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text=" - " />

        <EditText
            android:id="@+id/inputThree"
            android:maxLength="4"
            android:layout_width="70dp"
            android:layout_height="wrap_content"
            android:focusable="true"
            android:focusableInTouchMode="true"/>

    </LinearLayout>

</RelativeLayout>




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值