網(wǎng)站首頁 編程語言 正文
既然是瀏覽器,按照國(guó)際按理先實(shí)現(xiàn)一個(gè)多窗口的功能
我打算用ViewPager+Fragment來實(shí)現(xiàn),但仔細(xì)想想,這樣缺點(diǎn)是十分明顯因?yàn)橐WCFragment
不銷毀重建,當(dāng)viewpager的fragment個(gè)數(shù)過多,會(huì)造成很明顯的卡頓,所以先用FragmentManager試試效果
返回棧
多窗口中,每一個(gè)窗口都有自己的返回棧
每個(gè)返回棧都有自己的 fragmentManager,因此這里使用一個(gè)無ui的Fragment作為返回棧的載體
/** * @author huangweiliang */ class NavHostFragment(var name: String, var windowIndex: Int): Fragment() { private lateinit var binding: FragmentNavHostBinding private val TAG: String = "NavHostFragment" var curSelectFragment: Fragment? = null var curChildFragmentManager: FragmentManager? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) //點(diǎn)擊返回鍵,將當(dāng)前棧的Fragment作出棧處理 requireActivity().onBackPressedDispatcher.addCallback(this, object : OnBackPressedCallback(true) { override fun handleOnBackPressed() { isEnabled = childFragmentManager.backStackEntryCount > 0 if (isEnabled) childFragmentManager.popBackStackImmediate() // else requireActivity().onBackPressedDispatcher.onBackPressed() else requireActivity().finish() } }) } override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { val view = inflater.inflate(R.layout.fragment_nav_host, container, false) binding = FragmentNavHostBinding.bind(view) return view } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) var tag = name //創(chuàng)建窗口的第一個(gè)頁面,即首頁Fragment if (childFragmentManager.findFragmentByTag(tag) == null) { //這里使用的是childFragmentManager childFragmentManager.commitNow { val multiChildFragment = MultiChildFragment(name, 1, this@NavHostFragment) add(R.id.content, multiChildFragment, tag) //不能使用replace,否則每次返回都要重建 } } Log.i(TAG, "$name: onViewCreated") } override fun onStart() { super.onStart() Log.i(TAG, "$name: onStart") } override fun onResume() { super.onResume() Log.i(TAG, "$name: onResume") } override fun onPause() { super.onPause() Log.i(TAG, "$name: onPause") } override fun onDestroy() { super.onDestroy() Log.i(TAG, "$name: onDestroy") } }
FragmentNavHostBinding
<androidx.fragment.app.FragmentContainerView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/content" android:layout_width="match_parent" android:layout_height="match_parent"> </androidx.fragment.app.FragmentContainerView>
搜索頁面暫時(shí)放一個(gè)EditView
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent"> <EditText android:id="@+id/ed_search" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginStart="16dp" android:layout_marginTop="166dp" android:layout_marginEnd="16dp" android:padding="12dp" android:background="@drawable/gray_rounded_shape" android:drawableLeft="@drawable/ic_search_gray_24dp" app:layout_constraintTop_toTopOf="parent" /> </androidx.constraintlayout.widget.ConstraintLayout>
/** * @author huangweiliang * @date 2021/12/9 * 多窗口中,具體展示的每一個(gè)Fragment */ class MultiChildFragment(var name: String, var depth: Int, var hostFragment: Fragment) : Fragment() { private lateinit var binding: FragmentMultiChildBinding val TAG = "MultiChildFragment" override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { Log.i(TAG, "$name-$depth: onCreateView") val view = inflater.inflate(R.layout.fragment_multi_child, container, false) binding = FragmentMultiChildBinding.bind(view) //這里做一簡(jiǎn)單的傳參顯示當(dāng)前頁面 binding.edSearch.text.append("$name-$depth") init() return view } private fun init() { } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) Log.i(TAG, "$name-$depth: onViewCreated") } }
再寫一個(gè)包含多個(gè)返回棧的父Fragment
MultiStackParentFragment
class MultiStackParentFragment: Fragment() { companion object { fun newInstance() = MultiStackParentFragment() } private lateinit var binding: FragmentMultiStackParentBinding private var windowNum: Int = 0 /** * 當(dāng)前窗口的Index */ private var curWindowIndex: Int = 0 /** * 記錄創(chuàng)建的所有窗口對(duì)象 */ private val mStackList = ArrayList<NavHostFragment>() /** * 返回棧順序,存儲(chǔ)返回棧id */ private val mOrderStack = ArrayDeque<Int>() override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) } override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { val view = inflater.inflate(R.layout.fragment_multi_stack_parent, container, false) binding = FragmentMultiStackParentBinding.bind(view) init() return view } private fun init() { } }
在里面實(shí)現(xiàn)添加窗口
childFragmentManager管理著以該Fragment作為父容器的所有子Fragment
private fun addWindow() { childFragmentManager.commitNow { //NavHostFragment代表一個(gè)窗口對(duì)象 val navHostFragment = NavHostFragment("窗口${++windowNum}", windowNum) curWindowIndex = windowNum mStackList.add(navHostFragment) add(R.id.content_fragment, navHostFragment) //添加窗口 } transWindowIndex(curWindowIndex) }
效果圖
原文鏈接:https://juejin.cn/post/7060141477778685965
相關(guān)推薦
- 2022-04-19 運(yùn)行 npm run xxx 的時(shí)候都執(zhí)行了些什么
- 2022-09-28 C++List容器常用函數(shù)接口刨析_C 語言
- 2023-03-25 Golang實(shí)現(xiàn)優(yōu)雅的將struct轉(zhuǎn)換為map_Golang
- 2022-10-08 Pandas數(shù)據(jù)分析-pandas數(shù)據(jù)框的多層索引_python
- 2022-04-23 Android自定義View實(shí)現(xiàn)數(shù)字雨效果的全過程_Android
- 2023-04-11 一文帶你搞懂useCallback的使用方法_React
- 2022-12-22 python3中超級(jí)好用的日志模塊-loguru模塊使用詳解_python
- 2023-01-02 Python中的getter和setter的方法使用詳解_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支