详细介绍Android中的视图焦点Focus的使用

在非触摸屏设备中接收事件和处理响应的控件是具有焦点(Focused)的控件。一个窗口中一个时间内只能有一个具有焦点的控件。在早期具有滚轮设备的android系统中以及现在的智能TV电视应用中视图的焦点控制就非常重要了。而在触摸设备上通常默认情况下只有EditText控件才具有焦点,而我们通常会遇到的一个问题就是当进入一个具有EditText的界面时键盘就会自动弹出,而且有时候可能无法消失,但需求可能是进入时不弹出键盘。而这些所有的东西都是和视图的焦点有关,因此本文的重点就是介绍视图的焦点属性和方法,get到这些技术点后你就可以完全控制和使用这些特性了。

下面是几个关于焦点特性的描述:

  1. ViewGroup中有一个mFocued成员来保存子视图中哪个子视图是具有焦点的视图,并且这样一直会递归下去。比如某个视图层次下的根视图ROOT下有A,B,C三个子视图,而B下面又有B1,B2,B3三个子视图,而这时候B3是具有焦点的子视图,那么在B中的mFocued保存的是B3,而ROOT下的mFocued保存的是B。
  2. ViewGroup没有焦点并不代表其子视图也没有焦点,这里没有父子制约关系。
  3. 任何时候一个窗口内都只有一个视图具有焦点,或者所有视图都无焦点。
  4. 并不是所有视图都可以获取焦点。

我们要设置一个视图是否可以获取焦点可以通过如下方法来完成:

 //设置视图是否可以获得焦点
public void setFocusable(boolean focusable)
//获取视图是否可以获取焦点
public final boolean isFocusable() 

对于触摸设备来说我们可以设置一个视图在被触摸时是否可以成为焦点视图。我们可以通过如下方法:

//设置视图是否在触摸模式下可以获得焦点
 public void setFocusableInTouchMode(boolean focusableInTouchMode)
 //获取视图是否在触摸模式下获得焦点
 public final boolean isFocusableInTouchMode() 

因此在触摸设备下,一个视图要想获得焦点必须要setFocusable和setFocusableInTouchMode同时为true时才可以获取焦点。

下面两个方法用来判断某个视图是否是焦点视图以及是否获取了焦点:

//是否当前视图就是焦点视图
 public boolean isFocused()
//当前视图是否是焦点视图,或者子视图里面有焦点视图。
public boolean hasFocus() 

hasFocus和isFocused区别主要在ViewGroup上,前者只要自己或者儿子视图是焦点视图都返回true,而后者是一定要自己是焦点视图。

我们可以用如下方法来判断视图是否可见并且可以获得焦点,如果自己不可获得焦点则会递归调用子视图判断是否可以获得焦点。 从上可见has和is的区别是是否是只判断自身。

public boolean isFocusable(); //只判断自身
public boolean hasFocusable();  //除了判断自身外还判断子视图

如果我们要清除某个具有焦点视图的焦点属性就可以调用如下方法:

 public void clearFocus()

清除视图的焦点时,会激发视图的onFocusChanged的调用,并且往上遍历调用clearChildFocus 将mFocued的值置空,然后再从根视图中再次遍历将某个最佳的视图设置成为焦点视图。因为清除某个视图的焦点属性时,系统为了保证拥有一个具有焦点的视图,就会再次遍历整个视图树来重新设置具有焦点的视图。

下面的函数用来查找具有焦点的视图,如果是View则判断自己是否有焦点,如果是ViewGroup则自己就是焦点返回自己,否则返回儿子视图里面的焦点视图。如果都没有焦点视图时则返回null

public View findFocus() 

下面的方法是ViewGroup中的方法,获取直接的焦点子视图,也就是返回mFocued数据成员。

public View getFocusedChild() 

下面的方法中如果调用者是View并且自身可以获取焦点,那么就将自身加入到views数组里面去,如果自身是ViewGroup则将里面的可获取焦点的子视图加入到views里面去。

public void addFocusables(ArrayList<View> views, int direction)

下面的方法可以获取一个View或者ViewGroup下所有可获取焦点的子视图列表。如果调用的对象是View则可能返回自身,如果调用的对象是ViewGroup则返回自身和下面所有子视图中可获取焦点的子视图。

//这里的direction参数貌似没有什么作用。
 public ArrayList<View> getFocusables(int direction) 

可以看出addFocusables和getFocusables其实具有类似的功能,都是将自身或者容器视图里面的子视图中具有获取焦点能力的子视图返回到数组里面去。

public void setNextFocusDownId(int nextFocusDownId) 

上面函数和一些getXX函数用于设置或者获取某个视图的下一个焦点的ID,主要用于键盘模式来移动焦点的位置。

下面的方法用来请求成为当前焦点视图。这个方法是视图获得焦点的关键:

public final boolean requestFocus()

如果调用者是View且自己不可见(invisible or gone)或者不可获得焦点(isFocusable为false)或者父视图不允许自己获取焦点就会返回false表示成为焦点视图失败。如果能够成为焦点视图,那么就会调用onFocusChanged方法清除其他焦点视图。

如果是ViewGroup则根据setDescendantFocusability中的规则进行:如果是阻止子视图则自己进行焦点的获取,否则就按规则先子节点或者后子节点。

下面的方法用于当视图是ViewGroup时的焦点获取策略:

 public void setDescendantFocusability(int focusability)

focusability可设置的值如下:

FOCUS_BLOCK_DESCENDANTS: 阻止子视图成为焦点视图,这样即使子视图调用了requestFocus也不能成为焦点视图。

FOCUS_BEFORE_DESCENDANTS: 当ViewGroup调用requestFocus时总是优先让自己成为焦点视图。

FOCUS_AFTER_DESCENDANTS: 当ViewGroup调用requestFocus时优先让里面的子视图成为焦点,只有子视图无法成为焦点时才让自己成为焦点视图。这个特性也是默认特性。

通过setDescendantFocusability和requestFocus方法的配合就可以解决那种只有一个EditText且一进入就自动键盘弹出的问题。因为默认的EditText是一个可成为焦点的视图,这样根据规则当界面展示时就会成为一个焦点视图从而弹出键盘,这样即使对EditText调用clearFocus也因为规则导致他还是焦点视图。解决的方案是把EditText的一个祖先视图也设置为可获取焦点的视图(setFocusable(true)),并且将这个祖先视图的setDescendantFocusability设置为FOCUS_BEFORE_DESCENDANTS。这样当对EditText调用clearFocus或者对祖先视图调用reqeustFoucs时都会优先让祖先视图获得焦点。

视图树加载时的焦点视图的遍历

在窗口里的视图第一次被装载时系统会调用ViewRoot的doTraversal,这个函数内部会调用根视图的requestFocus方法:

if (!mView.hasFocus()) {
          mView.requestFocus(View.FOCUS_FORWARD);
}

这样就会让系统的最叶子的某个视图得到焦点。。得到的顺序是顺序为0的子视图先得到焦点。

这里一个特殊的例子就是TextView即使设置了FocuableInTochMode,也没有用,因为在构造函数中TextView自己的构造函数会在基类的基础上再次判断是否设置了Focuable属性,如果没有设置则即使上面设置FocuableInTochMode也没有用。但是Button的Style里面是包括一个Foucable属性的。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android应用中clearFocus方法调用无效的问题解决

    clearFocus 无效? EditText在focus与非focus的时候,显示效果是不同的:focus的时候光标是闪的,而且我们通常也会给它设置selector,focus的时候给它加上边框之类的. 通常当我们触摸EditText之外的View时,需要清除EditText的焦点.很自然的就会想到EditText.clearFocus(),然而常常并没有用.(EditText.isFocus()依然是true,光标也依然在跳跃...) clearFocus的实现 clearFocus的调用栈

  • Android在view.requestFocus(0)返回false的解决办法

    我们有时候想让listview的第一行自动获取到焦点,我们就会使用view.requestFocus(0)来操作,而有时候并不生效,debug后显示rerurn为false. 这是因为我们获取焦点太早,listview控件还为加载完毕.可以尝试一下方法解决: listview.post(new Runnable(){ @Override public void run(){ view.requestFocus(0); } }); 以上所述是小编给大家介绍的Android在view.request

  • android:descendantFocusability方法介绍

    此方法适用于所有母控件无法获取焦点的情况 开发中很常见的一个问题,项目中的listview不仅仅是简单的文字,常常需要自己定义listview,自己的Adapter去继承BaseAdapter,在adapter中按照需求进行编写,问题就出现了,可能会发生点击每一个item的时候没有反应,无法获取的焦点.原因多半是由于在你自己定义的Item中存在诸如ImageButton,Button,CheckBox等子控件(也可以说是Button或者Checkable的子类控件),此时这些子控件会将焦点获取到

  • Android开发中requestfocus()无效的原因及解决办法

    前言 最近做公司项目的时候,经常会遇到一个问题,就是我为某个控件如EditText设置requestfocus()的时候不管用,比如说登陆的时候,我判断下用户输入的密码,如果正确就登陆,错误就提示密码错误,并且输入框获取焦点,但是实际中确不起作用 package com.example.hfs.requestfocusdemo; import android.content.Intent; import android.support.v7.app.AppCompatActivity; impo

  • Android requestFocus详解及实例

    Android requestFocus详解及实例 requestFocus的使用 一句话概括: <requestFocus />: 标签用于指定屏幕内的焦点View. 布局资源文件的根节点可以使用容器控件(如LinearLayout.FrameLayout等),也可以使用非容器控件(如:EditText.TextView等).对于非容器控件,只能在非容器控件标签中放<requestFocus />标签,表示将当前控件设为焦点.如下代码: <LinearLayout xmln

  • 详细介绍Android中的视图焦点Focus的使用

    在非触摸屏设备中接收事件和处理响应的控件是具有焦点(Focused)的控件.一个窗口中一个时间内只能有一个具有焦点的控件.在早期具有滚轮设备的android系统中以及现在的智能TV电视应用中视图的焦点控制就非常重要了.而在触摸设备上通常默认情况下只有EditText控件才具有焦点,而我们通常会遇到的一个问题就是当进入一个具有EditText的界面时键盘就会自动弹出,而且有时候可能无法消失,但需求可能是进入时不弹出键盘.而这些所有的东西都是和视图的焦点有关,因此本文的重点就是介绍视图的焦点属性和方

  • 详细介绍Android中回调函数机制

    提示:在阅读本文章之前,请确保您对Touch事件的分发机制有一定的了解 在Android的学习过程中经常会听到或者见到"回调"这个词,那么什么是回调呢?所谓的回调函数就是:在A类中定义了一个方法,这个方法中用到了一个接口和该接口中的抽象方法,但是抽象方法没有具体的实现,需要B类去实现,B类实现该方法后,它本身不会去调用该方法,而是传递给A类,供A类去调用,这种机制就称为回调. 下面我们拿具体的Button的点击事件进行模拟分析: 首先,在View类中我们能找到setOnClickLis

  • 详细介绍Java中的各种锁

    一.一张图了解21种锁 二.乐观锁 应用 CAS 思想 一种乐观思想,假定当前环境是读多写少,遇到并发写的概率比较低,读数据时认为别的线程不会正在进行修改 实现 写数据时,判断当前 与期望值是否相同,如果相同则进行更新(更新期间加锁,保证是原子性的) 三.悲观锁 应用 synchronized.vector.hashtable 思想: 一种悲观思想 ** ,即认为写多读少,遇到并发写的可能性高 实现 每次读写数据都会认为其他线程会修改,所以每次读写数据时都会上锁 缺点 他线程想要读写这个数据时,

  • 详细介绍Python中的set集合

    目录 Python中的set集合 一.集合是什么? 二.set集合怎么用? 1.创建set集合 2.删除set集合 3.访问set集合元素 4.删除集合中的元素 5.向集合中添加元素 三.set集合的交并补 1.交集 2.并集 3.差集 四.set中的其他方法 五.frozenset 集合 Python中的set集合 一.集合是什么? 集合是什么呢?相信读者朋友们哪怕是没有用过集合这个数据类型.也一定在数学课堂上听过集合这个名词.数学中的集合是一个基本概念,说白了一堆不重复的数字可以组成一个集合

  • 详细分析Android中onTouch事件传递机制

    onTach介绍 ontach是Android系统中整个事件机制的基础.Android中的其他事件,如onClick.onLongClick等都是以onTach为基础的. onTach包括从手指按下到离开手机屏幕的整个过程,在微观形式上,具体表现为action_down.action_move和action_up等过程. onTach两种主要定义形式如下: 1.在自定义控件中,常见的有重写onTouchEvent(MotionEvent ev)方法.如在开发中经常可以看到重写的onTouchEv

  • 简单介绍Android中Activity的四种启动模式

    在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化操作.在Android中Activity的启动模式决定了Activity的启动运行方式. Activity有四种启动模式: 1. standard,默认的启动模式,只要激活Activity,就会创建一个新的实例,并放入任务栈中,这样任务栈中可能同时有一个Activity的多个实例. 2. singleTop,激活Activity时,如果栈顶是这个Activity,就不会创建新的实例:如果栈顶

  • 详细分析Android中实现Zygote的源码

    概述 在Android系统中,所有的应用程序进程,以及用来运行系统关键服务的System进程都是由zygote进程负责创建的.因此,我们将它称为进程孵化器.zygote进程是通过复制自身的方式来创建System进程和应用程序进程的.由于zygote进程在启动时会在内部创建一个虚拟机实例,因此,通过复制zygote进程而得到的System进程和应用程序进程可以快速地在内部获得一个虚拟机实例拷贝. zygote进程在启动完成之后,会马上将System进程启动起来,以便它可以将系统的关键服务启动起来.

  • 详细讲解Android中使用LoaderManager加载数据的方法

    Android的设计之中,任何耗时的操作都不能放在UI主线程之中.所以类似于网络操作等等耗时的操作都需要使用异步的实现.而在ContentProvider之中,也有可能存在耗时的操作(当查询的数据量很大的时候),这个时候我们也需要使用异步的调用来完成数据的查询. 当使用异步的query的时候,我们就需要使用LoaderManager了.使用LoaderManager就可以在不阻塞UI主线程的情况下完成数据的加载. (1)获取loaderManger:activity.getLoaderManag

  • 详细介绍mysql中limit与offset的用法

    目录 mysql limit与offset用法 附:Mysql limit offset用法举例 总结 有的时候我们在学习或者工作中会使用到SQL语句,那么介绍一下limit和offset的使用方法. mysql limit与offset用法 mysql里分页一般用limit来实现,例如: 1.select* from user limit 3 表示直接取前三条数据 2.select * from user limit 1,3; 表示取1后面的第2,3,4三条条数据 3.select * fro

  • 详细介绍.NET中的动态编译技术

    代码的动态编译并执行是一个.NET平台提供给我们的很强大的工具用以灵活扩展(当然是面对内部开发人员)复杂而无法估算的逻辑,并通过一些额外的代码来扩展我们已有 的应用程序.这在很大程度上给我们提供了另外一种扩展的方式(当然这并不能算是严格意义上的扩展,但至少为我们提供了一种思路). 动态代码执行可以应用在诸如模板生成,外加逻辑扩展等一些场合.一个简单的例子,为了网站那的响应速度,HTML静态页面往往是我们最好的选择,但基于数据驱动的网站往往又很难用静态页面实现,那么将动态页面生成html的工作或许

随机推荐