android的ViewpPger中,如何遍历得到下面布局问加你中的所有控件,方便对空间进行统一管理

android的ViewpPger中,如何遍历得到下面布局问加你中的所有控件,方便对空间进行统一管理,第1张

你可创建一个集合来保存你的几个布局:private ArrayList<View> pageViews;

pageViews = new ArrayList<View>();

pageViewsadd(v1);

pageViewsadd(v2);

pageViewsadd(v3);

pagerAdapter = new ViewPagerAdapter(pageViews);

这里创建Adapter的时候把这个集合传进去。

然后:uiControl = new UIControlForIntegrate(handler, OnAppPressedListener,

this);

uiControlsetResource(pageViews);//控制ui的地方也把这个集合传进去。这样它们指向的都是一个对象。你可以很方便的控制UI。或者你没有控制UI的类。直接在你的主类里面也可直接用这个集合里的View

我们知道,在 Android 中,View 绘制主要包含 3 大流程:

Android 中,主要有两种视图: View 和 ViewGroup ,其中:

虽然 ViewGroup 继承于 View ,但是在 View 绘制三大流程中,某些流程需要区分 View 和 ViewGroup ,它们之间的 *** 作并不完全相同,比如:

对 View 进行测量,主要包含两个步骤:

对于第一个步骤,即求取 View 的 MeasureSpec ,首先我们来看下 MeasureSpec 的源码定义:

MeasureSpec 是 View 的一个公有静态内部类,它是一个 32 位的 int 值,高 2 位表示 SpecMode(测量模式),低 30 位表示 SpecSize(测量尺寸/测量大小)。

MeasureSpec 将两个数据打包到一个 int 值上,可以减少对象内存分配,并且其提供了相应的工具方法可以很方便地让我们从一个 int 值中抽取出 View 的 SpecMode 和 SpecSize。

一个 MeasureSpec 表达的是:该 View 在该种测量模式(SpecMode)下对应的测量尺寸(SpecSize)。其中,SpecMode 有三种类型:

对 View 进行测量,最关键的一步就是计算得到 View 的 MeasureSpec ,子View 在创建时,可以指定不同的 LayoutParams (布局参数), LayoutParams 的源码主要内容如下所示:

其中:

LayoutParams 会受到父容器的 MeasureSpec 的影响,测量过程会依据两者之间的相互约束最终生成子View 的 MeasureSpec ,完成 View 的测量规格。

简而言之,View 的 MeasureSpec 受自身的 LayoutParams 和父容器的 MeasureSpec 共同决定( DecorView 的 MeasureSpec 是由自身的 LayoutParams 和屏幕尺寸共同决定,参考后文)。也因此,如果要求取子View 的 MeasureSpec ,那么首先就需要知道父容器的 MeasureSpec ,层层逆推而上,即最终就是需要知道顶层View(即 DecorView )的 MeasureSpec ,这样才能一层层传递下来,这整个过程需要结合 Activity 的启动过程进行分析。

我们知道,在 Android 中, Activity 是作为视图组件存在,主要就是在手机上显示视图界面,可以供用户 *** 作, Activity 就是 Andorid 中与用户直接交互最多的系统组件。

Activity 的基本视图层次结构如下所示:

Activity 中,实际承载视图的组件是 Window (更具体来说为 PhoneWindow ),顶层View 是 DecorView ,它是一个 FrameLayout , DecorView 内部是一个 LinearLayout ,该 LinearLayout 由两部分组成(不同 Android 版本或主题稍有差异): TitleView 和 ContentView ,其中, TitleView 就是标题栏,也就是我们常说的 TitleBar 或 ActionBar , ContentView 就是内容栏,它也是一个 FrameLayout ,主要用于承载我们的自定义根布局,即当我们调用 setContentView() 时,其实就是把我们自定义的布局设置到该 ContentView 中。

当 Activity 启动完成后,最终就会渲染出上述层次结构的视图。

因此,如果我们要求取得到子View 的 MeasureSpec ,那么第一步就是求取得到顶层View(即 DecorView )的 MeasureSpec 。大致过程如下所示:

经过上述步骤求取得到 View 的 MeasureSpec 后,接下来就可以真正对 View 进行测量,求取 View 的最终测量宽/高:

Android 内部对视图进行测量的过程是由 View#measure(int, int) 方法负责的,但是对于 View 和 ViewGroup ,其具体测量过程有所差异。

因此,对于测量过程,我们分别对 View 和 ViewGroup 进行分析:

综上,无论是对 View 的测量还是 ViewGroup 的测量,都是由 View#measure(int widthMeasureSpec, int heightMeasureSpec) 方法负责,然后真正执行 View 测量的是 View 的 onMeasure(int widthMeasureSpec, int heightMeasureSpec) 方法。

具体来说, View 直接在 onMeasure() 中测量并设置自己的最终测量宽/高。在默认测量情况下, View 的测量宽/高由其父容器的 MeasureSpec 和自身的 LayoutParams 共同决定,当 View 自身的测量模式为 LayoutParamsUNSPECIFIED 时,其测量宽/高为 android:minWidth / android:minHeight 和其背景宽/高之间的较大值,其余情况皆为自身 MeasureSpec 指定的测量尺寸。

而对于 ViewGroup 来说,由于布局特性的丰富性,只能自己手动覆写 onMeasure() 方法,实现自定义测量过程,但是总的思想都是先测量 子View 大小,最终才能确定自己的测量大小。

当确定了 View 的测量大小后,接下来就可以来确定 View 的布局位置了,也即将 View 放置到屏幕具体哪个位置。

View 的布局过程由 View#layout() 负责,其源码如下:

View#layout() 主要就做了两件事:

ViewGroup 的布局流程由 ViewGroup#layout() 负责,其源码如下:

可以看到, ViewGroup#layout() 最终也是通过 View#layout() 完成自身的布局过程,一个注意的点是, ViewGroup#layout() 是一个 final 方法,因此子类无法覆写该方法,主要是 ViewGroup#layout() 方法内部对子视图动画效果进行了相关设置。

由于 ViewGroup#layout() 内部最终调用的还是 View#layout() ,因此, ViewGroup#onLayout() 就会得到回调,用于处理 子View 的布局放置,其源码如下:

由于不同的 ViewGroup ,其布局特性不同,因此 ViewGroup#onLayout() 是一个抽象方法,交由 ViewGroup 子类依据自己的布局特性,摆放其 子View 的位置。

当 View 的测量大小,布局位置都确定后,就可以最终将该 View 绘制到屏幕上了。

View 的绘制过程由 View#draw() 方法负责,其源码如下:

其实注释已经写的很清楚了, View#draw() 主要做了以下 6 件事:

我们知道,在 Activity 启动过程中,会调用到 ActivityThreadhandleResumeActivity() ,该方法就是 View 视图绘制的起始之处:

可以看到, ActivityThreadhandleResumeActivity() 主要就是获取到当前 Activity 绑定的 ViewManager ,最后调用 ViewManageraddView() 方法将 DecorView 设置到 PhoneWindow 上,也即设置到当前 Activity 上。 ViewManager 是一个接口, WindowManager 继承 ViewManager ,而 WindowManagerImpl 实现了接口 WindowManager ,此处的 ViewManageraddView() 实际上调用的是 WindowManagerImpladdView() ,源码如下所示:

WindowManagerImpladdView() 内部转发到 WindowManagerGlobaladdView() :

在 WindowManagerGlobaladdView() 内部,会创建一个 ViewRootImpl 实例,然后调用 ViewRootImplsetView() 将 ViewRootImpl 与 DecorView 关联到一起:

ViewRootImplsetView() 内部首先关联了传递过来的 DecorView (通过属性 mView 指向 DecorView 即可建立关联),然后最终调用 requestLayout() ,而 requestLayout() 内部又会调用方法 scheduleTraversals() :

ViewRootImplscheduleTraversals() 内部主要做了两件事:

ChoreographerpostCallback() 会申请一次 VSYNC 中断信号,当 VSYNC 信号到达时,便会回调 ChoreographerdoFrame() 方法,内部会触发已经添加的回调任务, Choreographer 的回调任务有以下四种类型:

因此, ViewRootImplscheduleTraversals() 内部通过 mChoreographerpostCallback(ChoreographerCALLBACK_TRAVERSAL, mTraversalRunnable, null) 发送的异步视图渲染消息就会得到回调,即回调 mTra

将布局文件转换成视图,再从中获取那个TextView,再设置监听,

LayoutInflater inflater = LayoutInflaterfrom(this);

View view = inflaterinflate(Rlayoutxxxxx, null);

TextView tv = (TextView)viewfindViewById(Ridtv_xx);

tvsetOnXXXXXXX

先定位到相应的页卡,比如你有3个页卡分别是View1,View2,View3。

那么你要走在View1中添加就这样写

RelativeLayout insertLayout = (RelativeLayout)View1findViewById(Ridscreen)

View2中就是

RelativeLayout insertLayout = (RelativeLayout)View2findViewById(Ridscreen)

以此类推

以上就是关于android的ViewpPger中,如何遍历得到下面布局问加你中的所有控件,方便对空间进行统一管理全部的内容,包括:android的ViewpPger中,如何遍历得到下面布局问加你中的所有控件,方便对空间进行统一管理、Android - View 绘制流程、Android开发中,在一个activity中可以通过find view by ID获取其他布局文等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/9446615.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-28
下一篇 2023-04-28

发表评论

登录后才能评论

评论列表(0条)

保存