鸿蒙开发:案例集合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)
  }
}

🌸🌼🌺

相关推荐
奔跑的露西ly7 小时前
【HarmonyOS NEXT】沉浸式页面实现
华为·harmonyos
小Tomkk8 小时前
我开发的一款鸿蒙游戏《猜数字大师》 应用介绍 和技术架构
游戏·华为·harmonyos
鸿蒙开发工程师—阿辉8 小时前
HarmonyOS 5 数据持久化:状态持久化 (PersistentStorage)
华为·harmonyos
音浪豆豆_Rachel9 小时前
Flutter鸿蒙化之深入解析Pigeon可空返回与参数设计:nullable_returns.dart全解
flutter·harmonyos
音浪豆豆_Rachel9 小时前
Flutter鸿蒙跨平台测试策略解析:从基础Widget测试到平台集成验证
flutter·harmonyos
音浪豆豆_Rachel9 小时前
Flutter鸿蒙跨平台通信协议解析:Pigeon生成的Dart端桥接艺术
flutter·华为·harmonyos
鸿蒙开发工程师—阿辉9 小时前
HarmonyOS 5 数据持久化:首选项 (Preferences)
华为·harmonyos
子榆.10 小时前
Flutter 与开源鸿蒙(OpenHarmony)分布式能力实战:基于软总线实现跨设备协同
flutter·开源·harmonyos
鸿蒙开发工程师—阿辉10 小时前
HarmonyOS 5 上下文的使用:AbilityContext 的使用
华为·harmonyos