Простой стек фрагментов в Android
Доброе время суток, уважаемые читатели! Хочу представить вашему вниманию статью (см. источник), основанную на моем опыте работы в Single Activity Architecture, в частности со стеком пользовательских представлений.
При первом знакомстве с Single Activity Architecture у меня возникало много вопросов: “Как можно управлять моментом добавления и удаления фрагментов?”, “Как фрагменту удерживать нажатие кнопки назад?”, “Возможно ли запускать фрагмент на результат?”, ”Как понять, когда пользователь вернулся на фрагмент?” и т.д.
Первый вопрос является почти тривиальным. Можно создать единый класс навигации, в который передавать менеджер фрагментов и использовать функции перехода на экран.
Второй вопрос тоже частично решается оповещением класса навигации о том, что произошло нажатие на кнопку "Назад". Но в этом случае навигатор начинает являться чем-то большим, чем просто хранителем путей, в нем появляется логика, которая, на мой взгляд, абсолютно не оправдана. Но ведь кто-то в системе должен обрабатывать движения вперед и назад?
С возвращением пользователя на фрагмент тоже есть некоторые сложности. Одним из самых критичных, на мой взгляд, является повторный вызов onCreateView. Как мы все знаем, там появляется пользовательское представление в виде View. Также думаю, ни для кого не секрет, что эта операция является довольно прожорливой.
По итогу получается класс с большим количеством логики переходов, созданием фрагментов различного рода, сомнительными вставками “очень полезной функциональности” в методы обработки перехода назад (если пользователь добавил что-то на предыдущем экране, нужно это добавить в список). По моему мнению, это не совсем то, что требуется от класса, который отвечает за навигацию внутри приложения. Разумное решение — это делегировать часть функционала другим частям системы. Таким образом, в моей программе появилась сущность стека фрагментов.
Требования к стеку фрагментов почти тривиальны: добавить фрагмент, перейти назад, перейти до, — за исключением некоторых нюансов. Для меня, как для проектировщика, основной проблемой стал жизненный цикл добавляемых/удаляемых фрагментов. Также некоей проблемой было завершение фрагмента с результатом и отправки результата его потребителю. Благо решение нашлось довольно быстро. Внутренней логической структурой я выбрал немного усовершенствованный стек: слоеный пирог. Идея заключается в том, что слои укладываются на корж. Коржом нашего абстрактного пирога можно считать точку входа в приложение (главный фрагмент, домашняя страница и т.д.). Слои же в свою очередь имеют следующие свойства:
- При добавлении слоя в первую очередь он создается. Затем бережно укладывается на корж или слой, который находится сверху.
- При снятии слоя нижний слой становится виден, а снимаемый слой выбрасывается. И правда, зачем нам слой, измазанный в креме?
Если отойти от сладкого примера, то добавление — это транзакция, состоящая из скрытия предыдущего фрагмента и добавления нового. Также в эту операцию я добавил оповещение скрываемого фрагмента о том, что пользователь с него ушел, и ограничение на размер стека. Операция удаления является более витиеватой, поэтому обо всем более подробно.
Логику, которая отвечает за отправку результата из фрагмента-поставщика к запрашивающему фрагменту, разумно вынести в отдельный класс. Например, воображаемый экран добавления записи в ежедневник пользователя мог бы возвращать добавленную запись для последующей ее обработки вызываемому блоку программы. Это некий аналог onActivityResult.
Если представить все вышесказанное на схеме, то она будет выглядеть следующим образом:
Результативные фрагменты
Для обеспечения результативности я создал отдельный класс ResultUtils и интерфейс ResultableFragment.
Потребителем может являться любой фрагмент, который расширяет интерфейс ResultableFragment. Данный интерфейс состоит из одной функции fun onFragmentResult(requestCode: Int, resultCode: Int, data: Bundle). Данная функция является аналогом onActivityResult.
interface ResultableFragment { fun onFragmentResult(requestCode: Int, resultCode: Int, data: Bundle) }
Реализация класса ResultUtils представляет из себя набор следующих методов:
fun addPromise(currentFragment: Fragment, targetFragment: Fragment, requestCode: Int)
— данный метод создает некие обязательства (по ключу requestCode) от целевого фрагмента к текущему. Тут текущим фрагментом является то, откуда пользователь переходит, а целевым — то, куда он хочет перейти. Система обязательств представляет из себя HashMap<Integer, Integer>, в которой ключ — это hash текущего фрагмента, а значение – requestCode.fun sendResultIfPossible(fromToFragmentPair: Pair<Fragment, Fragment>)
— служит для вызова метода onFragmentResult и передачи соответствующих параметров. Почему по возможности ("...IfPossible")? Потому что не каждый фрагмент хочет отправлять или получать результат. При успешной отправке результата обязательства можно считать выполненными, и они удаляются из структуры.fun setResult(fragment: Fragment, data: Bundle, resultCode: Int)
— предназначен для установки результата во фрагмент. Результат, как и ключ, хранится в аргументах данного фрагмента.fun onBackPressed(fromToFragmentPair: Pair<Fragment, Fragment>)
— используется для обработки кнопки назад, устанавливает результат “фрагмент закрыт” с пустыми данными.
Дополнительные методы жизненного цикла
Также мне потребовался интерфейс, который объединяет все фрагменты, подчиняющиеся новому жизненному циклу. Данный интерфейс я назвал LifeBoundFragment. Туда включены следующие методы:
onUserLeaveScreen
— вызывается, когда пользователь покидает экран;onUserBack
— вызывается, когда пользователь возвращается на экран.
interface LifeBoundFragment { fun onUserLeaveScreen() fun onUserBack() }
Стек
Прорабатывая внешний интерфейс стека, я выделил следующие основные функции:
fun pushEntryPoint(home: Fragment)
— данный метод предназначен для добавления точки входа в стек. В моем случае это домашний фрагмент (тот фрагмент, находясь на котором по нажатию кнопки назад пользователь покидает приложение).fun push(target: Fragment)
— добавление нового фрагмента в стек.fun push(target: T, requestCode: Int)
— добавление нового фрагмента в стек с запросом некоего результата.fun popToTarget(target: Class<T>)
— спускаться вниз до тех пор, пока не встретим запрашиваемый фрагмент. Если такой фрагмент не найден, то останавливаем спуск на нашей точке входа.fun pop()
— непосредственно переход назад.fun handleBackPressed(): Boolean
— данный метод передается из активити в стек по событию onBackPressed. Возвращает true, если стек может самостоятельно обработать нажатие назад. В противном случае false.onActivityPause
,onActivityResume
— методы жизненного цикла активити. Данные методы вызывают соответствующие методы LifeBoundFragment для оповещения, что пользователь покинул/вернулся на текущий экран.
Сам стек я организовал на структуре LinkedList. Наиболее интересными, с моей точки зрения, являются методы push(target: T, requestCode: Int)
, pop()
и popToTarget(target: Class<T>)
.
Метод push(target: T, requestCode: Int)
Как упоминалось ранее, данный метод добавляет новый фрагмент на экран, скрывает предыдущий и добавляет в новый ключи. Для того чтобы скрыть внутреннюю реализацию, я создал приватный метод pushFragment, который отвечал за всю логику добавления и удаления фрагмента. Метод pushFragment возвращает Pair<Fragment, Fragment>. Это по сути направление движения, где ключ – это фрагмент, с которого пользователь переходит, а значение – куда. По задумке при добавлении фрагмента мы должны оповестить фрагмент, который скрывается, о том, что пользователь с него уходит. Для этого достаточно убедиться, что скрываемый фрагмент расширяет интерфейс LifeBoundFragment, и отправить событие onUserLeaveScreen.
Также в этом методе стоит добавить обязательства через класс утилит ResultUtils, используя метод addPromise.
override fun push(target: Fragment, resultCode: Int) { val fromToPair = pushFragment(target) callPauseIfPossible(fromToPair.first) resultUtils.addPromise(fromToPair.first, fromToPair.second, resultCode) }
Наиболее интересным тут является метод pushFragment:
private fun pushFragment(navigationTargetFragment: Fragment): Pair<Fragment, Fragment> { val ransaction = fragmentManager.beginTransaction() if (stackLinkedList.size() >= STACK_SIZE) { val outOfStackFragment = stackLinkedList.remove(1) transaction.remove(outOfStackFragment) } val leaveFragment = stackLinkedList.getLast() transaction.hide(leaveFragment) stackLinkedList.add(navigationTargetFragment) transaction.add(R.id.fragmentContainer, navigationTargetFragment) transaction.commit() return Pair(leaveFragment, navigationTargetFragment) }
В данном методе происходит вся основная манипуляция со стеком, скрытие предыдущего фрагмента и ограничение на кол-во элементов стека.
Метод pop()
Метод pop() также является неким собирательным методом. Особенностью этого метода является вызов sendResultIfPossible класса ResultUtils.
override fun pop() { val fromToPair = popFragment() resultUtils.sendResultIfPossible(fromToPair) }
Основная логика метода popFragment вполне предсказуема. Так что особо на ней задерживаться смысла я не вижу.
private fun popFragment(): Pair<Fragment, Fragment> { val leaveFragment = stackLinkedList.removeLast() val targetFragment = stackLinkedList.getLast() val transaction = fragmentManager.beginTransaction() transaction.remove(leaveFragment) callResumeIfPossible(targetFragment) transaction.show(targetFragment) transaction.commit() return Pair(leaveFragment, targetFragment) }
Метод popToTarget
Данный метод, по моему мнению, является самым интересным. Он сочетает в себе практически все.
Когда я начал разрабатывать функционал класса ResultUtils, одним моим внутренним ограничением было то, что результат при переходе назад передается по цепочке. Исходя из этого ограничения, метод onFragmentResult будет вызываться по цепочке до тех пор, пока не наткнется на корневой вызов. Фрагменты, находящиеся посередине цепочки, я начал называть транзитными. Действительно, они получают вызов onFragmentResult, в котором могут установить результат для следующего фрагмента цепи.
fun <T : Fragment> popToTarget(target: Class<T>) { val transaction = fragmentManager.beginTransaction() val iterator = stackLinkedList.descendingIterator() val fromToPair: Pair<Fragment, Fragment>? = null while (iterator.hasNext()) { val targetFragment = iterator.next() if (targetFragment::class.java == target) break transaction.remove(targetFragment) iterator.remove() fromToPair = Pair(targetFragment, stackLinkedList.last()) if (stackLinkedList.last()::class.java != target) { resultUtils.sendResultIfPossible(fromToPair) } } val frontFragment = stackLinkedList.last() callResumeIfPossible(frontFragment) resultUtils.sendResultIfPossible(fromToPair) transaction.show(frontFragment) transaction.commit() }
В заключение
По моему мнению, получилась гибкая, простая и надежная система управления фрагментами. В данный момент мне удалось успешно применить этот подход в ряде проектов, в которых я участвовал. Из минусов, с которыми я столкнулся при использовании этого подхода, – это leanback (Android TV), но отчасти сама система не располагает к Single Activity Architecture. Далее я планирую придумать механизм хранения/восстановления истории, запуск приложения с заданной историей (будет полезно при push нотификации). Спасибо за внимание!
Источник: Простой Stack Fragment'ов