Skip to content

Commit

Permalink
Merge pull request bumble-tech#307 from antonshilov/bottom-tab
Browse files Browse the repository at this point in the history
Bottom tab navigation sample
  • Loading branch information
antonshilov authored Dec 21, 2022
2 parents e766b8d + d3d42fd commit 4f6678e
Show file tree
Hide file tree
Showing 4 changed files with 200 additions and 75 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
- [#287](https://github.com/bumble-tech/appyx/pull/287)**Added**: `ImmutableList` has been added to avoid non-skippable compositions.
- [#289](https://github.com/bumble-tech/appyx/issues/289)**Added**: Introduced `interop-rx3` for RxJava 3 support. This has identical functionality to `interop-rx2`.
- [#298](https://github.com/bumble-tech/appyx/pulls/298)**Updated**: ChildView documentation. `TransitionDescriptor` generics has been renamed to `NavTarget` and `State`
- [#307](https://github.com/bumble-tech/appyx/pull/307) - **Added**: `Spotlight.current()` method to observe currently active `NavTarget.

---

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,5 +11,8 @@ fun <T : Any> Spotlight<T>.hasPrevious() =
fun <T : Any> Spotlight<T>.activeIndex() =
elements.map { value -> value.currentIndex }

fun <T : Any> Spotlight<T>.current() =
elements.map { value -> value.current?.key?.navTarget }

fun <T : Any> Spotlight<T>.elementsCount() =
elements.value.size
Original file line number Diff line number Diff line change
Expand Up @@ -3,20 +3,33 @@ package com.bumble.appyx.sandbox.client.spotlight
import android.os.Parcelable
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.padding
import androidx.compose.material3.Button
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.ArrowBack
import androidx.compose.material.icons.filled.ArrowForward
import androidx.compose.material.icons.filled.Favorite
import androidx.compose.material.icons.outlined.FavoriteBorder
import androidx.compose.material3.CircularProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.FabPosition
import androidx.compose.material3.FilledIconButton
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButtonDefaults
import androidx.compose.material3.NavigationBar
import androidx.compose.material3.NavigationBarItem
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.State
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.shadow
import androidx.compose.ui.unit.dp
import androidx.lifecycle.coroutineScope
import com.bumble.appyx.core.composable.Children
Expand All @@ -25,27 +38,27 @@ import com.bumble.appyx.core.node.Node
import com.bumble.appyx.core.node.ParentNode
import com.bumble.appyx.navmodel.spotlight.Spotlight
import com.bumble.appyx.navmodel.spotlight.backpresshandler.GoToPrevious
import com.bumble.appyx.navmodel.spotlight.current
import com.bumble.appyx.navmodel.spotlight.elementsCount
import com.bumble.appyx.navmodel.spotlight.hasNext
import com.bumble.appyx.navmodel.spotlight.hasPrevious
import com.bumble.appyx.navmodel.spotlight.operation.next
import com.bumble.appyx.navmodel.spotlight.operation.activate
import com.bumble.appyx.navmodel.spotlight.operation.next
import com.bumble.appyx.navmodel.spotlight.operation.previous
import com.bumble.appyx.navmodel.spotlight.operation.updateElements
import com.bumble.appyx.navmodel.spotlight.transitionhandler.rememberSpotlightSlider
import com.bumble.appyx.sandbox.client.child.ChildNode
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.Item.C1
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.Item.C2
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.Item.C3
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.NavTarget.Child1
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.NavTarget.Child2
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.NavTarget.Child3
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.State.Loaded
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.State.Loading
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.ScreenState.Loaded
import com.bumble.appyx.sandbox.client.spotlight.SpotlightExampleNode.ScreenState.Loading
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch
import kotlinx.parcelize.Parcelize

/**
* Shows how to use spotlight to create a UI with BottomTabBar
*/
class SpotlightExampleNode(
buildContext: BuildContext,
private val spotlight: Spotlight<NavTarget> = Spotlight(
Expand All @@ -58,20 +71,20 @@ class SpotlightExampleNode(
navModel = spotlight
) {

private val screenState = mutableStateOf<State?>(null)
private val screenState = mutableStateOf<ScreenState?>(null)

sealed class State {
object Loading : State()
object Loaded : State()
sealed class ScreenState {
object Loading : ScreenState()
object Loaded : ScreenState()
}

init {
// simulate loading tabs
if (spotlight.elementsCount() == 0) {
screenState.value = Loading
lifecycle.coroutineScope.launch {
delay(2000)
spotlight.updateElements(items = Item.getItemList())
delay(1000)
spotlight.updateElements(items = Tab.getTabList())
screenState.value = Loaded
}
} else {
Expand All @@ -92,13 +105,13 @@ class SpotlightExampleNode(
}

@Parcelize
private enum class Item(val navTarget: NavTarget) : Parcelable {
private enum class Tab(val navTarget: NavTarget) : Parcelable {
C1(Child1),
C2(Child2),
C3(Child3);

companion object {
fun getItemList() = values().map { it.navTarget }
fun getTabList() = values().map { it.navTarget }
}
}

Expand Down Expand Up @@ -126,82 +139,95 @@ class SpotlightExampleNode(
}
}

@OptIn(ExperimentalMaterial3Api::class)
@Suppress("LongMethod")
@Composable
private fun LoadedState(modifier: Modifier = Modifier) {
val hasPrevious = spotlight.hasPrevious().collectAsState(initial = false)
val hasNext = spotlight.hasNext().collectAsState(initial = false)
Column(
verticalArrangement = Arrangement.spacedBy(24.dp),
horizontalAlignment = Alignment.CenterHorizontally,
modifier = modifier,
) {
Row(
modifier = Modifier
.fillMaxWidth()
.padding(24.dp),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween
) {
TextButton(
text = "Previous",
enabled = hasPrevious.value
) {
spotlight.previous()
}
TextButton(
text = "Next",
enabled = hasNext.value
) {
spotlight.next()
}
val currentTab = spotlight.current().collectAsState(initial = null)
Scaffold(
modifier = modifier.fillMaxSize(),
floatingActionButtonPosition = FabPosition.Center,
floatingActionButton = { PageButtons(hasPrevious.value, hasNext.value) },
bottomBar = {
BottomTabs(currentTab)
}
Row(
modifier = Modifier
.fillMaxWidth()
.padding(24.dp),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween
) {
TextButton(
text = "C1",
enabled = true
) {
spotlight.activate(C1)
}
TextButton(
text = "C2",
enabled = true
) {
spotlight.activate(C2)
}
TextButton(
text = "C3",
enabled = true
) {
spotlight.activate(C3)
}
}

) {
Children(
modifier = Modifier
.padding(top = 12.dp, bottom = 12.dp)
.fillMaxWidth(),
transitionHandler = rememberSpotlightSlider(clipToBounds = true),
.padding(it),
transitionHandler = rememberSpotlightFaderThrough(),
navModel = spotlight
)
}
}

@Composable
private fun BottomTabs(currentTab: State<NavTarget?>) {
NavigationBar {
Tab.values().forEach { tab ->
val selected = currentTab.value == tab.navTarget
NavigationBarItem(
icon = {
Icon(
if (selected)
Icons.Filled.Favorite
else
Icons.Outlined.FavoriteBorder,
contentDescription = tab.toString()
)
},
label = { Text(tab.toString()) },
selected = selected,
onClick = { spotlight.activate(tab) }
)
}
}
}

@Composable
private fun TextButton(text: String, enabled: Boolean = true, onClick: () -> Unit) {
Button(onClick = onClick, enabled = enabled, modifier = Modifier.padding(4.dp)) {
Text(text = text)
private fun PageButtons(
hasPrevious: Boolean,
hasNext: Boolean
) {
Row(
modifier = Modifier
.fillMaxWidth()
.padding(24.dp),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween
) {
FilledIconButton(
onClick = { spotlight.previous() },
modifier = if (hasPrevious) Modifier.shadow(
4.dp,
IconButtonDefaults.filledShape
) else Modifier,
enabled = hasPrevious,
) {
Icon(
Icons.Filled.ArrowBack,
contentDescription = "Previous"
)
}
FilledIconButton(
onClick = { spotlight.next() },
modifier = if (hasNext) Modifier.shadow(
4.dp,
IconButtonDefaults.filledShape
) else Modifier,
enabled = hasNext,
) {
Icon(
Icons.Filled.ArrowForward,
contentDescription = "Next"
)
}
}
}

private fun Spotlight<*>.activate(item: Item) {
private fun Spotlight<*>.activate(item: Tab) {
activate(item.ordinal)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
package com.bumble.appyx.sandbox.client.spotlight

import android.annotation.SuppressLint
import androidx.compose.animation.core.FastOutSlowInEasing
import androidx.compose.animation.core.Transition
import androidx.compose.animation.core.animateFloat
import androidx.compose.animation.core.tween
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Modifier
import androidx.compose.ui.composed
import androidx.compose.ui.draw.alpha
import androidx.compose.ui.draw.scale
import com.bumble.appyx.core.navigation.transition.ModifierTransitionHandler
import com.bumble.appyx.core.navigation.transition.TransitionDescriptor
import com.bumble.appyx.navmodel.spotlight.Spotlight

/**
* Fade through transition from material design
* [Specification](https://m2.material.io/design/motion/the-motion-system.html#fade-through)
*/
class SpotlightFaderThrough<T> : ModifierTransitionHandler<T, Spotlight.State>() {

@SuppressLint("ModifierFactoryExtensionFunction")
override fun createModifier(
modifier: Modifier,
transition: Transition<Spotlight.State>,
descriptor: TransitionDescriptor<T, Spotlight.State>
): Modifier = modifier.composed {
val alpha = transition.animateFloat(
transitionSpec = {
when (targetState) {
Spotlight.State.ACTIVE -> tween(
durationMillis = enterDuration,
delayMillis = exitDuration,
easing = FastOutSlowInEasing
)

Spotlight.State.INACTIVE_BEFORE,
Spotlight.State.INACTIVE_AFTER -> tween(
durationMillis = exitDuration,
easing = FastOutSlowInEasing
)
}

},
targetValueByState = {
when (it) {
Spotlight.State.ACTIVE -> 1f
else -> 0f
}
}, label = ""
)
val scale = transition.animateFloat(
transitionSpec = {
when (targetState) {
Spotlight.State.ACTIVE -> tween(
durationMillis = enterDuration,
delayMillis = exitDuration,
easing = FastOutSlowInEasing
)

Spotlight.State.INACTIVE_BEFORE,
Spotlight.State.INACTIVE_AFTER -> tween(
durationMillis = exitDuration,
easing = FastOutSlowInEasing
)
}

},
targetValueByState = {
when (it) {
Spotlight.State.ACTIVE -> 1f
else -> 0.92f
}
}, label = ""
)

if (transition.targetState == Spotlight.State.ACTIVE) {
scale(scale.value).alpha(alpha.value)
} else {
alpha(alpha.value)
}

}

companion object {
private const val enterDuration = 210
private const val exitDuration = 90
}
}

@Composable
fun <T> rememberSpotlightFaderThrough(): ModifierTransitionHandler<T, Spotlight.State> =
remember { SpotlightFaderThrough() }

0 comments on commit 4f6678e

Please sign in to comment.