У меня есть приложение, которое имеет
BaseListActivity
, в котором размещено
BaseController
-> Действует как ViewPager
ListController
-> вкладка 1 изViewPager
FavoriteController
-> вкладка 2 в ViewPager
DetailActivity
-> Запускается при нажатии элемента на вкладке 1
DetailController
-> нажатие на просмотр в DetailActivity
Итак, код, чтобы дать вам представление.
активность.xml:
<LinearLayout android:id="@+id/mainLayout"
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<com.bluelinelabs.conductor.ChangeHandlerFrameLayout
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
BaseListActivity:
class CoinListActivity : AppCompatActivity() {
private lateinit var router: Router
/**
* Current implementation uses BlueLineLabs Router to navigate between views and handle animations
*/
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_coin_list)
router = Conductor.attachRouter(this, container, savedInstanceState)
router.setRoot(RouterTransaction.with(CoinBaseController()))
RoomSingleton.initDB(this)
.
.
.
BaseController:
class CoinBaseController : Controller() {
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup): View {
return inflater.inflate(R.layout.coin_base_controller, container, false).apply {
viewPager.adapter = TabAdapter()
tabs.setupWithViewPager(viewPager)
}
}
.
.
.
inner class TabAdapter : RouterPagerAdapter(this) {
override fun configureRouter(router: Router, position: Int) {
router.setPopsLastView(true)
when (position) {
0 -> router.pushController(RouterTransaction.with(CoinListController.newInstance()))
1 -> router.pushController(RouterTransaction.with(CoinFavoriteController()))
}
if (router.hasRootController().not())
router.setRoot(RouterTransaction.with(this@CoinBaseController))
}
override fun getCount(): Int = 2
override fun getPageTitle(position: Int): CharSequence? {
return if (position == 0)
"List"
else
"Favorites"
}
}
}
CoinListController:
class CoinListController : BaseMvvmController<CoinListViewModel, CoinListContract.State>(), CoinListAdapter
.CoinListListener {
var list: List<CoinListItem> = emptyList()
var recyclerView: RecyclerView? = null
lateinit var swipeLayout: SwipeRefreshLayout
private val adapter by lazy(LazyThreadSafetyMode.NONE) {
CoinListAdapter(applicationContext!!, this)
}
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup): View {
val view = inflater.inflate(R.layout.coin_list_controller, container, false)
swipeLayout = view.findViewById(R.id.swipeContainer)
recyclerView = view.findViewById(R.id.listRecycler)
recyclerView!!.layoutManager = LinearLayoutManager(activity)
recyclerView!!.addItemDecoration(DividerItemDecoration(activity,
DividerItemDecoration.VERTICAL))
recyclerView!!.adapter = adapter
recyclerView!!.isNestedScrollingEnabled = false
view.swipeContainer.apply {
this.setOnRefreshListener {
viewModel.getCoinList()
this.isRefreshing = false
}
}
return view
}
override fun onAttach(view: View) {
super.onAttach(view)
with(viewModel) {
if (list.isEmpty()) {
getCoinList()
setSearchListener(view)
} else {
view.listRecycler.adapter = adapter
adapter.notifyDataSetChanged()
}
}
}
override fun onDetach(view: View) {
super.onDetach(view)
view.listRecycler.adapter = null
println("Detached")
}
override fun onCoinClicked(coin: CoinListItem) {
viewModel.onCoinClicked(coin)
}
override fun onFavoriteClicked(coinFavoriteItem: CoinFavoriteItem) {
viewModel.onFavoriteClicked()
}
override val viewModel: CoinListViewModel = get().koin.get()
override fun onStateChange(state: Mvvm.State) {
when (state) {
is CoinListContract.State.CoinListReceived -> {
list = state.coins
adapter.addItems(list)
}
is CoinListContract.State.CoinItemClicked -> {
Snackbar.make(view!!, "Coin Item Clicked", Snackbar.LENGTH_SHORT)
//router.onActivityStarted()
startActivity(CoinDetailActivity.getLaunchIntent(activity!!, state.coin))
}
is CoinListContract.State.FavoriteClicked -> {
Snackbar.make(view!!, "Favorite Button Clicked", Snackbar.LENGTH_SHORT)
}
is CoinListContract.State.QueryRan -> {
adapter.addItems(state.searchList)
}
is CoinListContract.State.Error -> {
Log.d("OnStateChange", state.throwable.localizedMessage!!)
}
}
}
Так что проблема в CoinListController
, когда я нажимаю элемент списка, запускается DetailActivity
затем я возвращаюсь ... возвращаясь к CoinListController
.
Однако для каждого последующего щелчка элемента добавляется дополнительный слушатель (я думаю). Так что, если я нахожусь на 3-м щелчке элемента, CoinListController.onCoinClicked
вызывается 3 раза, создавая 3 экземпляра CoinDetailActivity
. Поэтому мне нужно нажать 3 раза назад, чтобы вернуться к CoinListController
.
Это поведение отсутствует в режиме отладки.
Я устанавливаю адаптеры в null при отсоединении. Что сохраняется в жизненном цикле контроллеров? Это слушатель, который я установил в моем адаптере или что-то еще?