• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# Scroll
2
3可滚动的容器组件,当子组件的布局尺寸超过父组件的尺寸时,内容可以滚动。
4
5>  **说明:**
6>  - 该组件从API version 7开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。
7>  - 该组件嵌套List子组件滚动时,若List不设置宽高,则默认全部加载,在对性能有要求的场景下建议指定List的宽高。
8>  - 该组件滚动的前提是主轴方向大小小于内容大小。
9>  - 该组件回弹的前提是要有滚动。内容小于一屏时,没有回弹效果。
10
11
12## 子组件
13
14支持单个子组件。
15
16
17## 接口
18
19Scroll(scroller?: Scroller)
20
21## 属性
22
23除支持[通用属性](ts-universal-attributes-size.md)外,还支持以下属性:
24
25| 名称             | 参数类型                                     | 描述                                       |
26| -------------- | ---------------------------------------- | ---------------------------------------- |
27| scrollable     | [ScrollDirection](#scrolldirection枚举说明)  | 设置滚动方向。<br/>默认值:ScrollDirection.Vertical |
28| scrollBar      | [BarState](ts-appendix-enums.md#barstate) | 设置滚动条状态。<br/>默认值:BarState.Auto           |
29| scrollBarColor | string&nbsp;\|&nbsp;number&nbsp;\|&nbsp;[Color](ts-appendix-enums.md#color) | 设置滚动条的颜色。                                |
30| scrollBarWidth | string&nbsp;\|&nbsp;number               | 设置滚动条的宽度。                                |
31| edgeEffect     | [EdgeEffect](ts-appendix-enums.md#edgeeffect) | 设置滑动效果,目前支持的滑动效果参见EdgeEffect的枚举说明。<br/>默认值:EdgeEffect.None |
32
33## ScrollDirection枚举说明
34| 名称                           | 描述                                  |
35| ---------------------------- | ----------------------------------- |
36| Horizontal                   | 仅支持水平方向滚动。                          |
37| Vertical                     | 仅支持竖直方向滚动。                          |
38| None                         | 不可滚动。                               |
39| Free<sup>(deprecated) </sup> | 支持竖直或水平方向滚动<br/> 从API version 9开始废弃 |
40
41## 事件
42
43| 名称                                       | 功能描述                                     |
44| ---------------------------------------- | ---------------------------------------- |
45| onScrollBegin<sup>9+</sup>(event: (dx: number, dy: number) => { dxRemain: number, dyRemain: number }) | 滚动开始事件回调。<br>参数:<br>- dx:即将发生的水平方向滚动量。<br>- dy:即将发生的竖直方向滚动量。<br>返回值:<br>- dxRemain:水平方向滚动剩余量。<br>- dyRemain:竖直方向滚动剩余量。 |
46| onScroll(event: (xOffset: number, yOffset: number) => void) | 滚动事件回调,&nbsp;返回滚动时水平、竖直方向偏移量。            |
47| onScrollEdge(event: (side: Edge) => void) | 滚动到边缘事件回调。                               |
48| onScrollEnd(event: () => void)           | 滚动停止事件回调。                                |
49
50>  **说明:**
51>
52>  若通过onScrollBegin事件和scrollBy方法实现容器嵌套滚动,需设置子滚动节点的EdgeEffect为None。如Scroll嵌套List滚动时,List组件的edgeEffect属性需设置为EdgeEffect.None53
54## Scroller
55
56可滚动容器组件的控制器,可以将此组件绑定至容器组件,然后通过它控制容器组件的滚动,同一个控制器不可以控制多个容器组件,目前支持绑定到List、Scroll、ScrollBar上。
57
58
59### 导入对象
60
61```
62scroller: Scroller = new Scroller()
63```
64
65
66### scrollTo
67
68scrollTo(value: { xOffset: number | string, yOffset: number | string, animation?: { duration: number, curve: Curve } }): void
69
70
71滑动到指定位置。
72
73**参数:**
74
75| 参数名       | 参数类型                                     | 必填   | 参数描述                                     |
76| --------- | ---------------------------------------- | ---- | ---------------------------------------- |
77| xOffset   | Length                                   | 是    | 水平滑动偏移。                                  |
78| yOffset   | Length                                   | 是    | 竖直滑动偏移。                                  |
79| animation | {<br/>duration:&nbsp;number,<br/>curve:&nbsp;[Curve](ts-animatorproperty.md)<br/>} | 否    | 动画配置:<br/>-&nbsp;duration:&nbsp;滚动时长设置。<br/>-&nbsp;curve:&nbsp;滚动曲线设置。 |
80
81
82### scrollEdge
83
84scrollEdge(value: Edge): void
85
86
87滚动到容器边缘。
88
89**参数:**
90
91| 参数名   | 参数类型                              | 必填   | 参数描述      |
92| ----- | --------------------------------- | ---- | --------- |
93| value | [Edge](ts-appendix-enums.md#edge) | 是    | 滚动到的边缘位置。 |
94
95
96### scrollPage
97
98scrollPage(value: { next: boolean, direction?: Axis }): void
99
100滚动到下一页或者上一页。
101
102**参数:**
103
104| 参数名                               | 参数类型                              | 必填   | 参数描述                                    |
105| --------------------------------- | --------------------------------- | ---- | --------------------------------------- |
106| next                              | boolean                           | 是    | 是否向下翻页。true表示向下翻页,false表示向上翻页。          |
107| direction<sup>(deprecated) </sup> | [Axis](ts-appendix-enums.md#axis) | 否    | 设置滚动方向为水平或竖直方向。<br/> 从API version 9开始废弃 |
108
109
110### currentOffset
111
112currentOffset()
113
114
115返回当前的滚动偏移量。
116
117**返回值**
118
119| 类型                                       | 描述                                       |
120| ---------------------------------------- | ---------------------------------------- |
121| {<br/>xOffset:&nbsp;number,<br/>yOffset:&nbsp;number<br/>} | xOffset:&nbsp;水平滑动偏移;<br/>yOffset:&nbsp;竖直滑动偏移。 |
122
123
124### scrollToIndex
125
126scrollToIndex(value: number): void
127
128
129滑动到指定Index。
130
131
132>  **说明:**
133>
134>  仅支持list组件。
135
136**参数:**
137
138| 参数名   | 参数类型   | 必填   | 参数描述              |
139| ----- | ------ | ---- | ----------------- |
140| value | number | 是    | 要滑动到的列表项在列表中的索引值。 |
141
142
143### scrollBy<sup>9+</sup>
144
145scrollBy(dx: Length, dy: Length): void
146
147
148滑动指定距离。
149
150
151>  **说明:**
152>
153>  仅支持Scroll组件。
154
155**参数:**
156
157| 参数名  | 参数类型   | 必填   | 参数描述               |
158| ---- | ------ | ---- | ------------------ |
159| dx   | Length | 是    | 水平方向滚动距离,不支持百分比形式。 |
160| dy   | Length | 是    | 竖直方向滚动距离,不支持百分比形式。 |
161
162
163## 示例
164### 示例1
165
166```ts
167// xxx.ets
168@Entry
169@Component
170struct ScrollExample {
171  scroller: Scroller = new Scroller();
172  private arr: number[] = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
173
174  build() {
175    Stack({ alignContent: Alignment.TopStart }) {
176      Scroll(this.scroller) {
177        Column() {
178          ForEach(this.arr, (item) => {
179            Text(item.toString())
180              .width('90%')
181              .height(150)
182              .backgroundColor(0xFFFFFF)
183              .borderRadius(15)
184              .fontSize(16)
185              .textAlign(TextAlign.Center)
186              .margin({ top: 10 })
187          }, item => item)
188        }.width('100%')
189      }
190      .scrollable(ScrollDirection.Vertical)  // 滚动方向纵向
191      .scrollBar(BarState.On)  // 滚动条常驻显示
192      .scrollBarColor(Color.Gray)  // 滚动条颜色
193      .scrollBarWidth(30) // 滚动条宽度
194      .edgeEffect(EdgeEffect.None)
195      .onScroll((xOffset: number, yOffset: number) => {
196        console.info(xOffset + ' ' + yOffset);
197      })
198      .onScrollEdge((side: Edge) => {
199        console.info('To the edge');
200      })
201      .onScrollEnd(() => {
202        console.info('Scroll Stop');
203      })
204
205      Button('scroll 150')
206        .onClick(() => { // 点击后下滑指定距离150.0vp
207          this.scroller.scrollBy(0,150);
208        })
209        .margin({ top: 10, left: 20 })
210      Button('scroll 100')
211        .onClick(() => { // 点击后滑动到指定位置,即下滑100.0vp的距离
212          this.scroller.scrollTo({ xOffset: 0, yOffset: this.scroller.currentOffset().yOffset + 100 });
213        })
214        .margin({ top: 60, left: 20 })
215      Button('back top')
216        .onClick(() => { // 点击后回到顶部
217          this.scroller.scrollEdge(Edge.Top);
218        })
219        .margin({ top: 110, left: 20 })
220      Button('next page')
221        .onClick(() => { // 点击后滑到下一页
222          this.scroller.scrollPage({ next: true });
223        })
224        .margin({ top: 170, left: 20 })
225    }.width('100%').height('100%').backgroundColor(0xDCDCDC)
226  }
227}
228```
229
230![zh-cn_image_0000001174104386](figures/zh-cn_image_0000001174104386.gif)
231
232### 示例2
233```ts
234@Entry
235@Component
236struct NestedScroll {
237  @State listPosition: number = 0; // 0代表滚动到List顶部,1代表中间值,2代表滚动到List底部。
238  private arr: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
239  private scroller: Scroller = new Scroller();
240
241  build() {
242    Flex() {
243      Scroll(this.scroller) {
244        Column() {
245          Text("Scroll Area")
246            .width("100%").height("40%").backgroundColor(0X330000FF)
247            .fontSize(16).textAlign(TextAlign.Center)
248
249          List({ space: 20 }) {
250            ForEach(this.arr, (item) => {
251              ListItem() {
252                Text("ListItem" + item)
253                  .width("100%").height("100%").borderRadius(15)
254                  .fontSize(16).textAlign(TextAlign.Center).backgroundColor(Color.White)
255              }.width("100%").height(100)
256            }, item => item)
257          }
258          .width("100%").height("50%").edgeEffect(EdgeEffect.None)
259          .onReachStart(() => {
260            this.listPosition = 0;
261          })
262          .onReachEnd(() => {
263            this.listPosition = 2;
264          })
265          .onScrollBegin((dx: number, dy: number) => {
266            if ((this.listPosition == 0 && dy >= 0) || (this.listPosition == 2 && dy <= 0)) {
267              this.scroller.scrollBy(0, -dy);
268              return { dxRemain: dx, dyRemain: 0 };
269            }
270            this.listPosition = 1;
271            return { dxRemain: dx, dyRemain: dy };
272          })
273
274          Text("Scroll Area")
275            .width("100%").height("40%").backgroundColor(0X330000FF)
276            .fontSize(16).textAlign(TextAlign.Center)
277        }
278      }
279      .width("100%").height("100%")
280    }.width('100%').height('100%').backgroundColor(0xDCDCDC).padding(20)
281  }
282}
283```
284
285![NestedScroll](figures/NestedScroll.gif)