鸿蒙开发:案例集合List:ListItem拖拽(交换位置,过渡动画)

🎯 案例集合List:ListItem拖拽(交换位置,过渡动画)

🌍 案例集合List

🚪 最近开启了学员班级点我欢迎加入(学习相关知识可获得定制礼盒)

🏷️ 效果图

🏷 onMove(拖拽排序)

🏷 onItemDragStart, onItemDragMove, onItemDrop

📖 参考

🧩 拆解

  • 原生(onMove)拖拽
javascript 复制代码
/**
 * 模拟数据
 */
const listData: string[] = ['购物', '体育', '服装', '军事', '电商', '娱乐', '科技']

@Component
export struct listDragAndDrop {
  build() {
    Column({ space: 10 }) {
      Text('可以左右拖拽滑动')
        .fontColor(20)
      List({ space: 10 }) {
        ForEach(listData, (item: string) => {
          ListItem() {
            Text(item)
              .width(80)
              .aspectRatio(1)
              .fontSize(20)
              .fontColor(Color.White)
              .textAlign(TextAlign.Center)
              .backgroundColor('#803c6ad0')
          }
        }, (item: string) => item)
          .onMove((from: number, to: number) => {
            const exchange = listData.splice(from, 1)
            listData.splice(to, 0, exchange[0])
          })
      }
      .height(80)
      .padding({ left: 15, right: 15 })
      .listDirection(Axis.Horizontal)
      .scrollBar(BarState.Off)
      .edgeEffect(EdgeEffect.None)
    }
    .width('100%')
    .height('100%')
    .justifyContent(FlexAlign.Center)
    .alignItems(HorizontalAlign.Center)
  }
}
  • 原生(onItemDragStart, onItemDragMove, onItemDrop)拖拽
javascript 复制代码
@Component
export struct listDragAndDrop {
  /**
   * 模拟数据
   */
  @State listData: string[] = ['购物', '体育', '服装', '军事', '电商', '娱乐', '科技']
  /**
   * 列表构造器
   */
  private listScroller: Scroller = new Scroller()
  /**
   * scale
   */
  @State itemScale: number = 0.75
  /**
   * 当前拖拽某个item的下标
   */
  @State curListItemDragIdx: number = -1
  /**
   * 可视区域的开头组件下标
   */
  @State scrollerStart: number = 0
  /**
   * 可视区域的结尾组件下标
   */
  @State scrollerEnd: number = 0

  /**
   * 拖拽样式
   */
  @Builder
  draggingBuilder(curListItemDragIdx: number) {
    Text(this.listData[curListItemDragIdx])
      .width(80)
      .aspectRatio(1)
      .fontSize(20)
      .fontColor(Color.White)
      .textAlign(TextAlign.Center)
      .backgroundColor('#803c6ad0')
  }

  /**
   * 拖拽结束
   */
  private dragAndDropEnds(idx_1: number, idx_2: number) {
    this.curListItemDragIdx = -1
    this.swapPositions(idx_1, idx_2)
  }

  /**
   * 交换位置
   * @param idx_1
   * @param idx_2
   */
  private swapPositions(idx_1: number, idx_2: number) {
    // 利用中间常量存交换值
    const swapItem = this.listData[idx_1]
    this.listData[idx_1] = this.listData[idx_2]
    this.listData[idx_2] = swapItem
  }

  build() {
    Column({ space: 10 }) {
      Text('可以左右拖拽滑动')
        .fontColor(20)
      List({ space: 10, scroller: this.listScroller }) {
        ForEach(this.listData, (item: string, idx: number) => {
          ListItem() {
            Text(item)
              .width(80)
              .aspectRatio(1)
              .fontSize(20)
              .fontColor(Color.White)
              .textAlign(TextAlign.Center)
              .backgroundColor('#803c6ad0')
              .scale({
                x: this.curListItemDragIdx === idx ? this.itemScale : 1,
                y: this.curListItemDragIdx === idx ? this.itemScale : 1
              })
              .animation({ curve: Curve.Smooth, duration: 200 })
          }
        })
      }
      .height(80)
      .padding({ left: 15, right: 15 })
      .listDirection(Axis.Horizontal)
      .scrollBar(BarState.Off)
      .edgeEffect(EdgeEffect.None)
      .onScrollIndex((start: number, end: number, center: number) => {
        this.scrollerStart = start
        this.scrollerEnd = end
        console.info('scrollerStart', start.toString() + '' + end.toString())

      })
      // 开始拖拽列表元素时触发
      .onItemDragStart((event: ItemDragInfo, curListItemDragIdx: number) => {
        // 拖拽中显示的内容
        return this.draggingBuilder(curListItemDragIdx)
      })
      .onItemDragMove((event: ItemDragInfo, itemIndex: number, insertIndex: number) => {
        // 更新当前拖拽到某个item的下标
        this.curListItemDragIdx = insertIndex

        // 向右滑动
        if (insertIndex === this.scrollerEnd && this.scrollerEnd <= this.listData.length - 1) {
          this.listScroller.scrollToIndex(insertIndex - 2, true)
        }

        // 向左滑动
        if (insertIndex === this.scrollerStart && this.scrollerStart !== 0) {
          this.listScroller.scrollToIndex(insertIndex - 1, true)
        }
      })
      // 拖拽结束
      .onItemDrop((event: ItemDragInfo, itemIndex: number, insertIndex: number, isSuccess: boolean) => {
        // 当前手势没释放 | 插入位置小于0 | 插入位置大于列表长度
        if (!isSuccess || insertIndex < 0 || insertIndex >= this.listData.length) {
          return
        }
        this.dragAndDropEnds(itemIndex, insertIndex)
      })
    }
    .width('100%')
    .height('100%')
    .justifyContent(FlexAlign.Center)
    .alignItems(HorizontalAlign.Center)
  }
}

🌸🌼🌺

相关推荐
王码码20351 小时前
Flutter for OpenHarmony 实战之基础组件:第三十一篇 Chip 系列组件 — 灵活的标签化交互
android·flutter·交互·harmonyos
坚果派·白晓明2 小时前
在鸿蒙设备上快速验证由lycium工具快速交叉编译的C/C++三方库
c语言·c++·harmonyos·鸿蒙·编程语言·openharmony·三方库
lbb 小魔仙2 小时前
【HarmonyOS实战】OpenHarmony + RN:自定义 useFormik 表单处理
react native·harmonyos
果粒蹬i3 小时前
【HarmonyOS】DAY7:鸿蒙跨平台 Tab 开发问题与列表操作难点深度复盘
华为·harmonyos
王码码20353 小时前
Flutter for OpenHarmony 实战之基础组件:第二十七篇 BottomSheet — 动态底部弹窗与底部栏菜单
android·flutter·harmonyos
熊猫钓鱼>_>3 小时前
移动端开发技术选型报告:三足鼎立时代的开发者指南(2026年2月)
android·人工智能·ios·app·鸿蒙·cpu·移动端
ITUnicorn3 小时前
【HarmonyOS6】ArkTS 自定义组件封装实战:动画水杯组件
华为·harmonyos·arkts·鸿蒙·harmonyos6
晚霞的不甘3 小时前
Flutter for OpenHarmony 构建简洁高效的待办事项应用 实战解析
flutter·ui·前端框架·交互·鸿蒙
全栈探索者4 小时前
@Component + struct = 你的新函数组件——React 开发者的鸿蒙入门指南(第 2 期)
react·harmonyos·arkts·前端开发·deveco studio·鸿蒙next·函数组件
廖松洋(Alina)4 小时前
【收尾以及复盘】flutter开发鸿蒙APP之成就徽章页面
flutter·华为·开源·harmonyos·鸿蒙