在HarmonyOS NEXT开发中bindPopup遇到问题?

在HarmonyOS NEXT开发中bindPopup遇到问题?给每个ListItem长按后增加了一个Popup
问题1:在外层设置了一个@state 的状态,长按后会同时弹出好几个Pop怎样监听按哪一个弹哪一个?
问题2:点击Pop后有删除当前条目的操作,怎么监听POP的点击?点击了删除后,怎么让Pop消失?

阅读 1.2k
avatarAI BotBETA

问题解答

问题1: 在外层设置了一个@state 的状态,长按后会同时弹出好几个Popup,怎样监听按哪一个弹哪一个?

在 HarmonyOS NEXT 开发中,如果你希望在长按 ListItem 时只弹出对应的 Popup,而不是同时弹出多个,你需要确保每个 ListItem 都有独立的 Popup 状态。这通常通过为每个 ListItem 维护一个独立的状态变量来实现。

答案:

你可以为每个 ListItem 添加一个唯一的标识符(例如索引或唯一ID),并在 @State 中为每个 Popup 创建一个独立的状态变量。在长按事件中,只更新对应 ListItem 的 Popup 状态。

示例代码:

@Entry
@Component
struct ListComponent {
  @State items: Array<any> = [/* your items */];
  @State popups: Map<number, boolean> = new Map(); // Map to store popup states

  @Builder build() {
    Column() {
      ForEach(this.items, (item, index) => {
        ListItem({ onClickEvent: () => this.handleItemClick(index) }) {
          // List item content
        }.longPressEvent(() => this.showPopup(index))

        if (this.popups.get(index)) {
          Popup({ onClose: () => this.closePopup(index) }) {
            // Popup content
            Button('Delete')
              .onClick(() => {
                this.deleteItem(index);
                this.closePopup(index);
              });
          }
        }
      })
    }
  }

  private showPopup(index: number) {
    this.popups.set(index, true);
  }

  private closePopup(index: number) {
    this.popups.set(index, false);
  }

  private deleteItem(index: number) {
    // Logic to delete the item from the list
    this.items.splice(index, 1);
    // Update popups map to remove the deleted item's popup state
    this.popups.delete(index);
    // Adjust other indexes if necessary
  }
}

问题2: 点击Pop后有删除当前条目的操作,怎么监听POP的点击?点击了删除后,怎么让Pop消失?

答案:

在 Popup 内部添加一个按钮,并为该按钮设置点击事件监听器。在点击事件中,执行删除条目的逻辑,并调用关闭 Popup 的方法。

如上例所示,在 Popup 组件内添加一个 Button,并为其设置 onClick 事件。在 onClick 事件处理函数中,调用删除条目的方法 deleteItem,并随后调用 closePopup 方法来关闭 Popup。

确保在删除条目后,也更新 popups 状态映射,以移除已删除条目的 Popup 状态,并调整其他条目的索引(如果需要)。

Button('Delete')
  .onClick(() => {
    this.deleteItem(index);
    this.closePopup(index);
  });

这样,当你点击删除按钮时,条目将被删除,并且对应的 Popup 也会消失。

1 个回答

可参考以下demo:

import { router } from '@kit.ArkUI' 
import { BusinessError } from '@kit.BasicServicesKit' 
 
@Entry 
@Component 
export struct Index { 
  @State list: stockListData = new stockListData([]) 
 
  aboutToAppear(): void { 
    const list: ItemInfo[] = [] 
    for (let i = 0; i < 50; ++i) { 
      let stockName = "item" + i 
      let bgState = 0 
      list.push(new ItemInfo(stockName, bgState)) 
    } 
    this.list.modifyAllData(list) 
 
  } 
 
  build() { 
    Column() { 
      Text("开始").height('50') 
        .width('100%').onClick(() => { 
      }) 
      List({ space: 10 }) { 
        LazyForEach(this.list, (info: ItemInfo, index: number) => { 
          ListItem() { 
            ItemView({ 
              info: info, itemFun: (name) => { 
                let itemIndex = 0 
                for (let im of this.list.getAllData()) { 
                  if (im.stockName === name) { 
                    break 
                  } 
                  itemIndex++ 
                } 
                console.info("animate--ItemView--aboutToReuse---delete Name=" + name) 
                this.list.getAllData().splice(itemIndex, 1) 
                this.list.notifyDataDelete(itemIndex) 
              } 
            }) 
              .onClick(() => { 
              }) 
          }.height('80') 
        }, (info: ItemInfo, index: number) => info.stockName + "type" + index) 
      } 
      .height('100%') 
      .width('100%') 
    } 
    .height('100%') 
    .width('100%') 
  } 
} 
 
@Reusable 
@Component 
struct ItemView { 
  @Prop info: ItemInfo 
  private lastName: string = "" 
 
  aboutToAppear(): void { 
    this.lastName = this.info.stockName 
  } 
 
  aboutToReuse(params: Record<string, Object>): void { 
    console.info("animate--ItemView--aboutToReuse---name=" + this.info.stockName + ",lastName=" + this.lastName) 
 
    this.lastName = this.info.stockName 
  } 
 
  @State customPopup: boolean = false 
  itemFun: (name: string) => void = (name) => { 
  } 
 
  @Builder 
  popItemsView() { 
    Text("qipao pop").onClick(() => { 
      this.customPopup = !this.customPopup 
      this.itemFun(this.info.stockName) 
    }) 
  } 
 
  build() { 
    Column() { 
      Flex({ justifyContent: FlexAlign.Center, alignItems: ItemAlign.Center }) { 
        Column() { 
          Text(this.info.stockName) 
            .maxFontSize('16') 
            .minFontSize('8') 
            .maxLines(1) 
            .margin({ bottom: 2 }) 
        } 
        .width('36.27%') 
        .alignItems(HorizontalAlign.Start) 
        .padding({ left: '16', right: 8 }) 
      } 
      .backgroundColor("#00ffea") 
      .height('100%') 
    } 
    .bindPopup(this.customPopup, { 
      builder: this.popItemsView, // 气泡的内容 
      placement: Placement.Top, // 气泡的弹出位置 
      offset: { x: 0, y: 30 }, 
      radius: 12, 
      popupColor: Color.Pink, // 气泡的背景色 
      onStateChange: (e) => { 
        console.info('tag', JSON.stringify(e.isVisible)) 
        if (!e.isVisible) { 
          this.customPopup = false 
          // this.bgColor = this.info.topStatus == 1 ? $r('app.color.dialog_background') : $r('app.color.start_window_background') 
        } else { 
          this.customPopup = true 
          // this.bgColor = $r('app.color.dialog_button') 
        } 
      } 
    }) 
    .gesture( 
      LongPressGesture({ repeat: false })// 由于repeat设置为true,长按动作存在时会连续触发,触发间隔为duration(默认值500ms) 
        .onAction((event?: GestureEvent) => { 
          console.log("tag", "LongPressGesture start event--customPopup=" + this.customPopup) 
          if (event) { 
            this.customPopup = !this.customPopup 
          } 
          // if (event && event.repeat) { 
          // } 
        })// 长按动作一结束触发 
        .onActionEnd((event?: GestureEvent) => { 
          if (event) { 
          } 
        }) 
    ) 
  } 
} 
 
@Observed 
class ItemInfo { 
  @Track 
  stockName: string = "--" 
  @Track 
  bgState: number = 0 
 
  constructor(name: string, bg: number) { 
    this.stockName = name 
    this.bgState = bg 
  } 
} 
 
 
class BasicDataSource implements IDataSource { 
  private listeners: DataChangeListener[] = []; 
 
  public totalCount(): number { 
    return 0; 
  } 
 
  public getData(index: number): ItemInfo | undefined { 
    return undefined; 
  } 
 
  registerDataChangeListener(listener: DataChangeListener): void { 
    if (this.listeners.indexOf(listener) < 0) { 
      this.listeners.push(listener); 
    } 
  } 
 
  unregisterDataChangeListener(listener: DataChangeListener): void { 
    const position = this.listeners.indexOf(listener); 
    if (position >= 0) { 
      this.listeners.splice(position, 1); 
    } 
  } 
 
  notifyDataReload(): void { 
    this.listeners.forEach((listener: DataChangeListener) => { 
      listener.onDataReloaded(); 
    }) 
  } 
 
  notifyDataAdd(index: number): void { 
    this.listeners.forEach((listener: DataChangeListener) => { 
      listener.onDataAdd(index); 
    }) 
  } 
 
  notifyDataChange(index: number): void { 
    this.listeners.forEach((listener: DataChangeListener) => { 
      listener.onDataChange(index); 
    }) 
  } 
 
  notifyDataDelete(index: number): void { 
    this.listeners.forEach((listener: DataChangeListener) => { 
      listener.onDataDelete(index); 
    }) 
  } 
 
  notifyDataMove(from: number, to: number): void { 
    this.listeners.forEach((listener: DataChangeListener) => { 
      listener.onDataMove(from, to); 
    }) 
  } 
} 
 
export class stockListData extends BasicDataSource { 
  private listData: ItemInfo[] = []; 
 
  constructor(dataArray: ItemInfo[]) { 
    super() 
    this.listData = dataArray; 
  } 
 
  public totalCount(): number { 
    return this.listData.length; 
  } 
 
  public getAllData(): ItemInfo[] { 
    return this.listData; 
  } 
 
  public getData(index: number): ItemInfo { 
    return this.listData[index]; 
  } 
 
  public modifyAllData(data: ItemInfo[]): void { 
    this.listData = data 
    this.notifyDataReload() 
  } 
}
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进