设计一个FrameLayout(Kotlin)

本文介绍了如何设计一个自定义的FrameLayout,重点在于理解其特点、测量过程、布局逻辑以及源码分析。通过Kotlin实现,讨论了测量时的子View层叠、尺寸计算、MarginLayoutParams和layout_gravity的处理,以及布局过程中的边界设置。
摘要由CSDN通过智能技术生成

拆零件,然后再把零件拼装回去,来来回回对其结构也就熟悉了

FrameLayout的特点
  1. 子View按照添加顺序层叠显示
  2. FrameLayout的尺寸与其最大子View(可见的)的尺寸相等(加上padding值)
  3. 如果要让GONE的子View参与计算,则需要把setMeasureAllChildren(boolean) ,setConsiderGoneChildrenWhenMeasuring()设置为true
  4. 支持通过layout_gravity控制子View的布局

根据上述特点,我也要实现一个简单的FrameLayout,应该怎么开始呢?

这里自然考虑继承ViewGroup,然后重写onMeasure,onLayout方法,而且onLayout方法必须实现(因为这是一个抽象方法,子类必须实现)

测量

这里测量的过程是:遍历该ViewGroup,调用子View的measure方法进行测量,同时通过子View的LayoutParams获取到子View的Margin信息,然后综合View的测量尺寸与Margin值,算出该ViewGroup的尺寸,然后设置该尺寸为ViewGroup的测量尺寸。

/*
    * 尺寸测量
    * */
    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {

        var maxWidth: Int = 0
        var maxHeight: Int = 0

        //遍历子View进行测量
        for (i in 0 until childCount) {
            val child = getChildAt(i)
            if (child.visibility != View.GONE) {

                //调用ViewGroup的一个实例方法(本质上是调用View的measure方法),
                // 由于FrameLayout的特点,子View之间并不干扰尺寸大小,所以已经使用的空间为0
                //该计算过程会计算padding和margin
                measureChildWithMargins(child, widthMeasureSpec, 0, heightMeasureSpec, 0)

                var layoutParams = child.layoutParams as LayoutParams

                //得到最大宽高,并考虑子View的Margin
                maxWidth = Math.max(maxWidth, child.measuredWidth + layoutParams.leftMargin + layoutParams.rightMargin)
                maxHeight = Math.max(maxHeight, child.measuredHeight + layoutParams.topMargin + layoutParams.bottomMargin)

                //考虑FrameLayout本身的padding值
                maxWidth += paddingLeft + paddingRight
               
### 回答1: 当然可以为您提供关于使用 Kotlin 编写 Fragment 嵌套的信息。 首先,您需要在 Kotlin 代码中定义一个继承自 Fragment 的类。这个类将作为您 Fragment 的基础,并且您需要重写其中的一些方法,比如 onCreateView()。 接下来,您可以在 onCreateView() 方法中加载您的布局,可以使用 LayoutInflater 和 ViewGroup 对象来实现。如果您需要嵌套 Fragment,则可以在布局中添加一个 Fragment 占位符,然后在 onCreateView() 方法中创建一个子 FragmentManager,使用该占位符添加您的子 Fragment。 以下是一个简单的示例: ``` class ParentFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { val rootView = inflater.inflate(R.layout.fragment_parent, container, false) // 添加子 Fragment val childFragment = ChildFragment() childFragmentManager.beginTransaction() .replace(R.id.child_fragment_container, childFragment) .commit() return rootView } } class ChildFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { return inflater.inflate(R.layout.fragment_child, container, false) } } ``` 在上面的代码中,ParentFragment 加载了一个名为 fragment_parent 的布局,并在其中添加了一个 Fragment 占位符(使用 id 为 child_fragment_container 的 FrameLayout)。然后,ParentFragment 创建了一个 ChildFragment 对象,并使用子 FragmentManager 将其添加到了占位符中。 ChildFragment 则加载了一个名为 fragment_child 的布局,并在其中显示了一些内容。 希望这可以帮助您开始编写 Kotlin Fragment 的嵌套! ### 回答2: 在Kotlin中,我们可以使用Fragment和FragmentManager来实现嵌套Fragment。 首先,我们需要创建两个Fragment,一个作为外部Fragment,一个作为内部Fragment。在外部Fragment的布局文件中,我们可以通过添加一个FrameLayout来为内部Fragment留出空间。 ```kotlin class OuterFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { // Inflate the layout for this fragment val view = inflater.inflate(R.layout.fragment_outer, container, false) return view } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) // 创建内部Fragment的实例 val innerFragment = InnerFragment() childFragmentManager.beginTransaction() .replace(R.id.innerFragmentContainer, innerFragment) .commit() } } class InnerFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { // Inflate the layout for this fragment return inflater.inflate(R.layout.fragment_inner, container, false) } } ``` 在外部Fragment的布局文件fragment_outer.xml中,我们可以添加一个FrameLayout用于显示内部Fragment。 ```xml <FrameLayout android:id="@+id/innerFragmentContainer" android:layout_width="match_parent" android:layout_height="match_parent"/> ``` 接下来,我们可以在Activity中使用OuterFragment。 ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) supportFragmentManager.beginTransaction() .replace(R.id.fragmentContainer, OuterFragment()) .commit() } } ``` 在activity_main.xml中,我们添加一个FrameLayout用于显示OuterFragment。 ```xml <FrameLayout android:id="@+id/fragmentContainer" android:layout_width="match_parent" android:layout_height="match_parent"/> ``` 这样就实现了一个Kotlin编写的Fragment嵌套。外部Fragment中的FrameLayout用于显示内部Fragment,从而实现了Fragment的嵌套效果。 ### 回答3: 要使用Kotlin编写一个嵌套Fragment,我们需要以下步骤: 首先,在Kotlin中创建一个新的Fragment类。可以通过继承`androidx.fragment.app.Fragment`来创建: ```kotlin class OuterFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { val view = inflater.inflate(R.layout.fragment_outer, container, false) // 添加内部Fragment val innerFragment = InnerFragment() childFragmentManager.beginTransaction() .replace(R.id.inner_fragment_container, innerFragment) .commit() return view } } ``` 在`onCreateView`方法中,我们使用`LayoutInflater`来加载外部Fragment的布局文件,并将其作为根视图返回。 然后,我们创建一个内部Fragment类。内部Fragment的布局文件将在外部Fragment的布局文件中加载: ```kotlin class InnerFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { return inflater.inflate(R.layout.fragment_inner, container, false) } } ``` 在`onCreateView`方法中,我们使用`LayoutInflater`来加载内部Fragment的布局文件,并将其作为根视图返回。 接下来,我们需要在外部Fragment的布局文件中定义一个容器,用于放置内部Fragment。例如,我们在`fragment_outer.xml`中添加一个`FrameLayout`作为容器: ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" ...> <!-- 外部Fragment布局 --> <FrameLayout android:id="@+id/inner_fragment_container" android:layout_width="match_parent" android:layout_height="match_parent" /> </LinearLayout> ``` 最后,在使用嵌套Fragment的Activity中,我们可以使用`FragmentManager`动态加载外部Fragment到Activity的布局中: ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) val outerFragment = OuterFragment() supportFragmentManager.beginTransaction() .replace(R.id.fragment_container, outerFragment) .commit() } } ``` 在这个例子中,我们使用了一个`FrameLayout`作为容器,用于加载外部Fragment。 通过以上步骤,我们成功实现了使用Kotlin编写的Fragment嵌套。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值