Jetpack Compose | control (5) of article, Spacer, LazyRow, LazyColumn & let the Column slide
In the last article, we finished with Box, Row, and Column and left a question: “What if the container isn’t big enough to hold the content?” Let’s learn this part together.
The code in this article is based on version 1.0.1
Unless otherwise specified
Compose
Both refer toJetpack compose
All the code in this article can be obtained in WorkShop. This code is concentrated in post29 & Post30 package
A complete series of directory: making Pages | Denver | CSDN
A simple comparison with Android
In Android, the SDK provides things like: ScrollView NestedScrollView ListView GridView RecyclerView, and other controls applicable to all kinds of scenarios, based on sliding gestures to adjust the content display area, in order to achieve the purpose of displaying more content.
Further exploration can find that View itself contains the semi-finished implementation of Scroll mechanism. Of course, in this paper, we will not go into the content of Android, but draw a point with the help of Android knowledge we have mastered:
Based on Scroll mechanism, use small containers to display the essence of large content: on the basis of view measurement, combined with sliding gesture processing, adjust content layout and display after drawing.
In an earlier post, a blogger mentioned that the content for Compose is ScrollRow, ScrollColumn/LazyRow, and LazyColumn.
ScrollRow, ScrollColumn, etc., briefly existed in early previews, and seem to have been removed
Compose is also designed along these lines, and we’ll explore them in more detail in a future article. In this article, we’ll just learn how to use them.
Before we really get into this section, we’ll add a simple control, Spacer, that can easily create a space space that won’t be available in subsequent articles.
Spacer
In the previous article, we looked at Modifier, which includes some layout-related apis, such as: Padding, offset, but no margin, etc. According to the industry practice, if there is already a widely accepted noun, the new word is generally not used, at least the root of the word is the same. In Compose, Spacer is used, eliminating some application scenarios of margin.
Note: Calculations are always lossy, don’t abuse Spacer, and many scenarios have specific ways of handling spacing, as you’ll learn over time
How to use
@Composable
fun Spacer(modifier: Modifier)
Copy the code
Generally, you only need to specify its width and height, for example:
Spacer(modifier = Modifier.size(3.dp))
Copy the code
LazyColumn
In the last article, we learned that Row and Column differ only in direction and are very similar in implementation. The same goes for LazyRow and LazyColumn.
Doc says:
The vertically scrolling list that only composes and lays out the currently visible items. The content block defines a DSL which allows you to emit items of different types. For example you can use LazyListScope.item to add a single item and LazyListScope.items to add a list of items.
Combine only the calculation and layout of a vertical slidable list of currently visible elements. The content block defines a DSL that allows the creation of different types of elements.
For example, use lazyListScope.item to add a single element and lazyListScope.items to add a list of elements.
Note: “A content block defines a DSL that allows the creation of different types of elements”, which is different from the Android concept:RecyclerView#Adapter
Has to map data to the same typeViewHolder
Or different typesViewHolder
Ability. It means “when you add an element, it can beA single element
, or theList of elements
This is a different type.Literal translation can be misleading in Chinese context.
Obviously, it is similar to the Android ListView and RecyclerView, with an emphasis on Lazy, which does not compute and layout elements all at once.
Therefore, it does not register ScrollView, and in its usage scenario, the need for slippability is very common, so it is implemented by default!
We mentioned earlier:
In an earlier post, a blogger mentioned that the content for Compose is ScrollRow, ScrollColumn/LazyRow, and LazyColumn
There is nothing wrong with this, but it is definitely not misinterpreted: “Row and Column do not provide sliding capabilities, instead use LazyRow and LazyColumn.”
But the mood is there, so let’s finish that and then learn how Row and Column provide slippability.
How to use
@Composable
fun LazyColumn(
modifier: Modifier = Modifier,
state: LazyListState = rememberLazyListState(),
contentPadding: PaddingValues = PaddingValues(0.dp),
reverseLayout: Boolean = false,
verticalArrangement: Arrangement.Vertical =
if(! reverseLayout) Arrangement.Topelse Arrangement.Bottom,
horizontalAlignment: Alignment.Horizontal = Alignment.Start,
flingBehavior: FlingBehavior = ScrollableDefaults.flingBehavior(),
content: LazyListScope.() -> Unit
)
Copy the code
Meanings of each parameter:
- Modifier: modifier
- State: Used to control or observe list status
- ContentPadding: a Padding around the entire content,Note: white space around the content, take the vertical list as an example, you can’t see the white space at the tail when the tail is not displayedThis cannot be achieved through the Modifier,Note: The Modifier can only be used to achieve fixed white space for list containers. You can use it to leave white space before and after the first element. If you need space between elements, you can use verticalArrangement
- ReverseLayout: Whether to reverse the list
- VerticalArrangement: Vertical range of child controls.Can be used to add spacing between child controls, and how to arrange content when it is not enough to fill the minimum size of the list
- HorizontalAlignment: horizontalAlignment of child controls
- FlingBehavior: Processing logic of the fling behavior
- Content: Declares how to provide a DSL for child controls. Are there two ways
@LazyScopeMarker
interface LazyListScope {
fun item(key: Any? = null, content: @Composable LazyItemScope. () - >Unit)
fun items(
count: Int,
key: ((index: Int) - >Any)? = null,
itemContent: @Composable LazyItemScope. (index: Int) - >Unit
)
@ExperimentalFoundationApi
fun stickyHeader(key: Any? = null, content: @Composable LazyItemScope. () - >Unit)
}
Copy the code
By the way, in the last project I participated in, RecycleView was frequently used for content presentation. In order to conveniently deal with “spacing between items”, “white space between the beginning and the end”, and “spacing between specific items should not be used”, a set of components was written in the project, which could be removed and shared with everyone later.
Based on the lazyListScope.item method
This usage already appears under the Post29 package in the WorkShop content corresponding to the previous article.
Such as:
private fun LazyListScope.rowDemo(a) {
item {
CodeSample(code = "row sample 1:")
Row {
// ignore
}
}
item {
CodeSample(code = "Row sample 2: Vertically centered")
// ignore
}
// ignore
}
Copy the code
Based on the lazyListScope.items method
In addition to using the API directly, the SDK also provides some inline functions to eliminate code redundancy when dealing with data structures:
inline fun <T> LazyListScope.items(
items: List<T>,
noinline key: ((item: T) - >Any)? = null.crossinline itemContent: @Composable LazyItemScope. (item: T) - >Unit
)
inline fun <T> LazyListScope.itemsIndexed(
items: List<T>,
noinline key: ((index: Int.item: T) - >Any)? = null.crossinline itemContent: @Composable LazyItemScope. (index: Int.item: T) - >Unit
)
inline fun <T> LazyListScope.items(
items: Array<T>,
noinline key: ((item: T) - >Any)? = null.crossinline itemContent: @Composable LazyItemScope. (item: T) - >Unit
)
inline fun <T> LazyListScope.itemsIndexed(
items: Array<T>,
noinline key: ((index: Int.item: T) - >Any)? = null.crossinline itemContent: @Composable LazyItemScope. (index: Int.item: T) - >Unit
)
Copy the code
Based on past Android development experience, it’s easy to write code like this:
// The entry page in WorkShop enumerates the activities corresponding to each example
@Composable
fun TestList(activity: Activity, cases: List<Pair<String, Class<out Activity>>>) {
LazyColumn(contentPadding = PaddingValues(horizontal = 16.dp, vertical = 8.dp)) {
itemsIndexed(items = cases) { _, item ->
Column(
horizontalAlignment = Alignment.CenterHorizontally,
) {
Spacer(modifier = Modifier.size(3.dp))
Box(
modifier = Modifier
.height(48.dp)
.fillMaxWidth()
.background(
color = Color.LightGray,
shape = RoundedCornerShape(CornerSize(6.dp))
)
.clickable {
activity.startActivity(Intent(activity, item.second))
},
contentAlignment = Alignment.Center
) {
Text(text = item.first, color = MainTxt, textAlign = TextAlign.Center)
}
Spacer(modifier = Modifier.size(3.dp))
}
}
}
}
TestList(
activity = this@MainActivity, cases = arrayListOf(
"Layout samples" to P21LayoutSample::class.java,
"Draw samples" to P21DrawSample::class.java,
"Text samples" to P26TextSample::class.java,
"TextField samples" to P26TextFieldSample::class.java,
"Button samples" to P26ButtonSample::class.java,
"Icon samples" to P27IconSample::class.java,
"Image samples" to P27ImageSample::class.java,
"Switch,Checkbox,RadioButton samples" to P28SwitchRbCbSample::class.java,
"Box,Row,Column samples" to P29BoxRowColumnSample::class.java,
)
)
Copy the code
From an Android point of view, this code is equivalent to creating the ItemView of the ViewHolder and the implementation of onBindViewHolder
Column(
horizontalAlignment = Alignment.CenterHorizontally,
) {
Spacer(modifier = Modifier.size(3.dp))
Box(
modifier = Modifier
.height(48.dp)
.fillMaxWidth()
.background(
color = Color.LightGray,
shape = RoundedCornerShape(CornerSize(6.dp))
)
.clickable {
activity.startActivity(Intent(activity, item.second))
},
contentAlignment = Alignment.Center
) {
Text(text = item.first, color = MainTxt, textAlign = TextAlign.Center)
}
Spacer(modifier = Modifier.size(3.dp))
}
Copy the code
That is, we take advantage of the “white space” inherent in the ItemView to handle the spacing between items, which is obviously not a best practice!
More elegant handling of spacing and alignment
As mentioned above:
- contentPadding
- verticalArrangement
- horizontalAlignment
Based on this, we modified the code to reduce unnecessary nesting
@Composable
fun TestList(activity: Activity, cases: List<Pair<String, Class<out Activity>>>) {
LazyColumn(
contentPadding = PaddingValues(horizontal = 16.dp, vertical = 8.dp),
verticalArrangement = spacedBy(6.dp, Alignment.Top),
horizontalAlignment = Alignment.CenterHorizontally,
) {
itemsIndexed(items = cases) { _, item ->
Box(
modifier = Modifier
.height(48.dp)
.fillMaxWidth()
.background(
color = Color.LightGray,
shape = RoundedCornerShape(CornerSize(6.dp))
)
.clickable {
activity.startActivity(Intent(activity, item.second))
},
contentAlignment = Alignment.Center
) {
Text(
text = item.first,
color = MainTxt,
textAlign = TextAlign.Center
)
}
}
}
}
Copy the code
Consistent results can be obtained:
Make Column slidable
Reference makes Row slidable
Started in Modifier and familiar face in the process, we mentioned the androidx.com pose. The foundation pack, and contains a package: androidx.com pose. Foundation. Gestures, just as its name implies, the latter and gestures to handle relevant.
- androidx.compose.foundation.gestures.ScrollableKt#scrollable
- androidx.compose.foundation.ScrollKt#verticalScroll
- androidx.compose.foundation.ScrollKt#horizontalScroll
After this stage of learning, we can make a conclusion:
Compose contains a number of basic functions, as well as “embellished” advanced functions on top of the basic functions in a real-world usage scenario
From the naming, we can easily know that scrollable is a relatively basic function, verticalScroll and horizontalScroll are advanced functions decorated based on scrollable.
Reading the source code does confirm our speculation.
So if I take the verticalScroll for example, the horizontalScroll is not expanded yet
The Doc reads as follows:
Modify element to allow to scroll vertically when height of the content is bigger than max constraints allow. In order to use this modifier, you need to create and own [ScrollState] @see [rememberScrollState]
Decorates a layout element to allow scrolling vertically when its content height exceeds the maximum allowed limit.
Note: The maximum height of the content should take into account the height of the container, padding, offset, etc
To use it, you need to create and hold an instance of ScrollState, see rememberScrollState
Method prototype:
fun Modifier.verticalScroll(
state: ScrollState,
enabled: Boolean = true,
flingBehavior: FlingBehavior? = null,
reverseScrolling: Boolean = false
)
Copy the code
- State: ScrollState, instance of ScrollState
- Enabled: Whether sliding is allowed when a touch event occurs
- FlingBehavior: Fling processing logic
- ReverseScrolling: Whether to reverse slide
Demo
Column(
modifier = Modifier
.fillMaxWidth()
.height(600.dp)
.verticalScroll(
state = rememberScrollState()
)
) {
Box(
Modifier
.fillMaxWidth()
.height(400.dp)
.background(Color.Green)
)
Spacer(modifier = Modifier.height(50.dp))
Box(
Modifier
.fillMaxWidth()
.height(400.dp)
.background(Color.Blue)
)
}
Copy the code
Clearly, the content height has exceeded the maximum limit!
The effect
conclusion
At this point, we have learned the list in Compose. After a brief exploration of the related contents of the Modifier sliding, we have mastered the solution to make Column respond to sliding when the content exceeds the display limit. Unfortunately, we don’t know enough to continue our quest.
Of course, in future articles, we’ll continue to learn the basics of Compose and finally explore the deeper aspects of Compose together
Another: recently ancestral code changed a bit of schizophrenia, the article update efficiency or not to come.
Readers, if you find my sharing helpful, please like to let me know and give me the motivation to continue; If the content is not good, please leave a message to tell me what you want to read, so that we can communicate together in the follow-up article!