android中调用requestFocus()的详细过程

view中
        
public boolean requestFocus(int direction, Rect previouslyFocusedRect) {
        return requestFocusNoSearch(direction, previouslyFocusedRect);
        }

private boolean requestFocusNoSearch(int direction, Rect previouslyFocusedRect) {
        // need to be focusable  
        if ((mViewFlags & FOCUSABLE_MASK) != FOCUSABLE ||
        (mViewFlags & VISIBILITY_MASK) != VISIBLE) {
        return false;
        }

        // need to be focusable in touch mode if in touch mode  
        if (isInTouchMode() &&
        (FOCUSABLE_IN_TOUCH_MODE != (mViewFlags & FOCUSABLE_IN_TOUCH_MODE))) {
        return false;
        }

        // need to not have any parents blocking us  
        if (hasAncestorThatBlocksDescendantFocus()) {//判断父视图是否阻止子视图获得焦点  
        return false;
        }

        handleFocusGainInternal(direction, previouslyFocusedRect);//进行具体的焦点获取  
        return true;
        }
        
        void handleFocusGainInternal(int direction, Rect previouslyFocusedRect) {
        if (DBG) {
        System.out.println(this + " requestFocus()");
        }

        if ((mPrivateFlags & FOCUSED) == 0) {
        mPrivateFlags |= FOCUSED;

        if (mParent != null) {
        mParent.requestChildFocus(this, this);//第一个参数是child视图,第二个是focused视图,该函数内部进行递归调用  
        }
        //注意时态,xxxed()和xxx()的区别是,前者是执行完之后回调,后者是在执行前回调  
        onFocusChanged(true, direction, previouslyFocusedRect);
        refreshDrawableState();

        if (AccessibilityManager.getInstance(mContext).isEnabled()) {
        notifyAccessibilityStateChanged();
        }
        }
        }
        一般来说,父视图是ViewGroup,requestChildFocus在ViewGroup中的实现
        
public void requestChildFocus(View child, View focused) {
        if (DBG) {
        System.out.println(this + " requestChildFocus()");
        }
        if (getDescendantFocusability() == FOCUS_BLOCK_DESCENDANTS) {
        return;
        }

        // Unfocus us, if necessary  
        super.unFocus();

        // We had a previous notion of who had focus. Clear it.  
        if (mFocused != child) {
        if (mFocused != null) {
        mFocused.unFocus();
        }

        mFocused = child;
        }
        if (mParent != null) {
        mParent.requestChildFocus(this, focused);//最终会递归到ViewRoot中的equestChildFocus  
        }
        }
        ViewRootImpl中的requestChildFocus
        
public void requestChildFocus(View child, View focused) {
        checkThread();//确保在UI线程中  
        if (mFocusedView != focused) {//其实ViewGroup中已经检查过,mFocusedView一定不是目标焦点视图  
        mAttachInfo.mTreeObserver.dispatchOnGlobalFocusChange(mFocusedView, focused);
        scheduleTraversals();//发起View遍历请求  
        }
        mFocusedView = mRealFocusedView = focused;
        if (DEBUG_INPUT_RESIZE) Log.v(TAG, "Request child focus: focus now "
        + mFocusedView);  view中
        
public boolean requestFocus(int direction, Rect previouslyFocusedRect) {
        return requestFocusNoSearch(direction, previouslyFocusedRect);
        }

private boolean requestFocusNoSearch(int direction, Rect previouslyFocusedRect) {
        // need to be focusable  
        if ((mViewFlags & FOCUSABLE_MASK) != FOCUSABLE ||
        (mViewFlags & VISIBILITY_MASK) != VISIBLE) {
        return false;
        }

        // need to be focusable in touch mode if in touch mode  
        if (isInTouchMode() &&
        (FOCUSABLE_IN_TOUCH_MODE != (mViewFlags & FOCUSABLE_IN_TOUCH_MODE))) {
        return false;
        }

        // need to not have any parents blocking us  
        if (hasAncestorThatBlocksDescendantFocus()) {//判断父视图是否阻止子视图获得焦点  
        return false;
        }

        handleFocusGainInternal(direction, previouslyFocusedRect);//进行具体的焦点获取  
        return true;
        }
        
        void handleFocusGainInternal(int direction, Rect previouslyFocusedRect) {
        if (DBG) {
        System.out.println(this + " requestFocus()");
        }

        if ((mPrivateFlags & FOCUSED) == 0) {
        mPrivateFlags |= FOCUSED;

        if (mParent != null) {
        mParent.requestChildFocus(this, this);//第一个参数是child视图,第二个是focused视图,该函数内部进行递归调用  
        }
        //注意时态,xxxed()和xxx()的区别是,前者是执行完之后回调,后者是在执行前回调  
        onFocusChanged(true, direction, previouslyFocusedRect);
        refreshDrawableState();

        if (AccessibilityManager.getInstance(mContext).isEnabled()) {
        notifyAccessibilityStateChanged();
        }
        }
        }
        一般来说,父视图是ViewGroup,requestChildFocus在ViewGroup中的实现
        
public void requestChildFocus(View child, View focused) {
        if (DBG) {
        System.out.println(this + " requestChildFocus()");
        }
        if (getDescendantFocusability() == FOCUS_BLOCK_DESCENDANTS) {
        return;
        }

        // Unfocus us, if necessary  
        super.unFocus();

        // We had a previous notion of who had focus. Clear it.  
        if (mFocused != child) {
        if (mFocused != null) {
        mFocused.unFocus();
        }

        mFocused = child;
        }
        if (mParent != null) {
        mParent.requestChildFocus(this, focused);//最终会递归到ViewRoot中的equestChildFocus  
        }
        }
        ViewRootImpl中的requestChildFocus
        
public void requestChildFocus(View child, View focused) {
        checkThread();//确保在UI线程中  
        if (mFocusedView != focused) {//其实ViewGroup中已经检查过,mFocusedView一定不是目标焦点视图  
        mAttachInfo.mTreeObserver.dispatchOnGlobalFocusChange(mFocusedView, focused);
        scheduleTraversals();//发起View遍历请求  
        }
        mFocusedView = mRealFocusedView = focused;
        if (DEBUG_INPUT_RESIZE) Log.v(TAG, "Request child focus: focus now " + mFocusedView);  

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 实现 TextView 文字滚动可以使用以下两种方式: 一、使用 Marquee(跑马灯)属性 在布局文件的 TextView 添加以下属性: ``` <TextView android:id="@+id/text_view" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="这是一段需要滚动的文字这是一段需要滚动的文字这是一段需要滚动的文字" android:singleLine="true" android:ellipsize="marquee" android:focusable="true" android:focusableInTouchMode="true" android:marqueeRepeatLimit="marquee_forever" android:scrollHorizontally="true" android:padding="5dp" android:textColor="#000000" android:textSize="20sp" /> ``` 其,关键属性为: - android:ellipsize="marquee":当文字超出 TextView 的宽度时,显示省略号并开启跑马灯效果。 - android:focusable="true" 和 android:focusableInTouchMode="true":设置为可获得焦点,让 TextView 能够滚动。 - android:marqueeRepeatLimit="marquee_forever":设置跑马灯无限循环。 - android:scrollHorizontally="true":可水平滚动。 在 Java 代码调用 setHorizontallyScrolling() 方法也可以实现水平滚动,示例代码如下: ``` TextView textView = findViewById(R.id.text_view); textView.setHorizontallyScrolling(true); ``` 二、使用代码实现 在 Java 代码使用 TextView 的 setEllipsize() 方法和 setMarqueeRepeatLimit() 方法可以实现 TextView 的文字滚动效果。示例代码如下: ``` TextView textView = findViewById(R.id.text_view); textView.setText("这是一段需要滚动的文字这是一段需要滚动的文字这是一段需要滚动的文字"); textView.setSingleLine(true); textView.setEllipsize(TextUtils.TruncateAt.MARQUEE); textView.setMarqueeRepeatLimit(-1); textView.setFocusable(true); textView.setFocusableInTouchMode(true); textView.requestFocus(); ``` 需要注意的是,如果在代码设置了跑马灯效果,还需要在布局文件设置以下属性: ``` android:singleLine="true" android:scrollHorizontally="true" ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值