網站首頁 編程語言 正文
既然是瀏覽器,按照國際按理先實現一個多窗口的功能
我打算用ViewPager+Fragment來實現,但仔細想想,這樣缺點是十分明顯因為要保證Fragment
不銷毀重建,當viewpager的fragment個數過多,會造成很明顯的卡頓,所以先用FragmentManager試試效果
返回棧
多窗口中,每一個窗口都有自己的返回棧
每個返回棧都有自己的 fragmentManager,因此這里使用一個無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) //點擊返回鍵,將當前棧的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 //創建窗口的第一個頁面,即首頁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>
搜索頁面暫時放一個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 * 多窗口中,具體展示的每一個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) //這里做一簡單的傳參顯示當前頁面 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") } }
再寫一個包含多個返回棧的父Fragment
MultiStackParentFragment
class MultiStackParentFragment: Fragment() { companion object { fun newInstance() = MultiStackParentFragment() } private lateinit var binding: FragmentMultiStackParentBinding private var windowNum: Int = 0 /** * 當前窗口的Index */ private var curWindowIndex: Int = 0 /** * 記錄創建的所有窗口對象 */ private val mStackList = ArrayList<NavHostFragment>() /** * 返回棧順序,存儲返回棧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() { } }
在里面實現添加窗口
childFragmentManager管理著以該Fragment作為父容器的所有子Fragment
private fun addWindow() { childFragmentManager.commitNow { //NavHostFragment代表一個窗口對象 val navHostFragment = NavHostFragment("窗口${++windowNum}", windowNum) curWindowIndex = windowNum mStackList.add(navHostFragment) add(R.id.content_fragment, navHostFragment) //添加窗口 } transWindowIndex(curWindowIndex) }
效果圖
原文鏈接:https://juejin.cn/post/7060141477778685965
相關推薦
- 2022-12-12 pycharm?console?打印中文為亂碼問題及解決_python
- 2022-10-28 React中使用react-file-viewer問題_React
- 2022-03-16 .net6環境下使用RestSharp請求GBK編碼網頁亂碼的解決方案_實用技巧
- 2022-05-13 類實例化 對象的內存模型 及 內存占用分析
- 2022-02-24 Kubernetes究竟是個容器應用程序還是集群操作系統,它這么復雜的原因出在哪?
- 2023-01-31 C#實現批量壓縮和解壓縮的示例代碼_C#教程
- 2022-05-27 一起來學習C語言的程序環境與預處理_C 語言
- 2022-06-08 SpringBoot jar包瘦身操作 -Dloader.path使用
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支