在HarmonyOS NEXT开发中如何实验验证码倒计时后点击再次发送,重新显示倒计时?

在HarmonyOS NEXT开发中如何实验验证码倒计时后点击再次发送,重新显示倒计时?输入验证码页面,倒计时结束后显示重新发送,点击重新发送无法控制倒计时控件再次显示 当前困难影响:体验不好

阅读 774
avatarAI BotBETA

解决方案示例(使用ArkTS):

@Entry
@Component
struct VerifyCodePage {
  @State countdownTime: number = 60
  @State isCounting: boolean = false
  timerID: number = 0

  // 启动倒计时
  startCountdown() {
    if (this.isCounting) return
    
    this.isCounting = true
    this.countdownTime = 60
    
    this.timerID = setInterval(() => {
      if (this.countdownTime > 0) {
        this.countdownTime--
      } else {
        this.stopCountdown()
      }
    }, 1000)
  }

  // 停止倒计时
  stopCountdown() {
    clearInterval(this.timerID)
    this.isCounting = false
  }

  // 发送验证码请求
  sendVerifyCode() {
    // 调用API发送验证码...
    // 成功后重置倒计时
    this.startCountdown()
  }

  build() {
    Column() {
      Button(this.isCounting ? `${this.countdownTime}s后重发` : '获取验证码')
        .enabled(!this.isCounting)
        .onClick(() => {
          this.sendVerifyCode()
        })
    }
  }

  // 生命周期:页面消失时清除定时器
  aboutToDisappear() {
    this.stopCountdown()
  }
}

关键实现点说明:

  1. 状态管理
  2. 使用@State装饰器管理倒计时时间(countdownTime)和计时状态(isCounting)
  3. 通过timerID保存定时器引用
  4. 倒计时控制
  5. startCountdown()方法初始化倒计时并启动定时器
  6. stopCountdown()方法清除定时器并重置状态
  7. 定时器每1秒更新倒计时显示
  8. UI交互
  9. 按钮文本根据状态动态显示倒计时或"获取验证码"
  10. enabled属性控制按钮点击状态
  11. 点击事件触发验证码发送和倒计时重启
  12. 注意事项
  13. 添加页面生命周期管理(aboutToDisappear)防止内存泄漏
  14. 实际开发需要补充网络请求失败处理
  15. 建议添加防止重复点击机制(示例中已通过isCounting状态控制)

常见问题排查:

  1. 如果倒计时不刷新:检查@State变量是否被正确修改
  2. 如果点击无响应:检查按钮enabled状态绑定是否正确
  3. 如果计时加速:检查是否创建了多个定时器(确保每次启动前清除旧定时器)
1 个回答

具体如下所示:

@Entry 
@Component 
struct Page { 
  @State elapsedTime: number = 0; 
  textTimerController: TextTimerController = new TextTimerController() 
 
  build() { 
    Column() { 
      Row() { 
        TextTimer({ isCountDown: true, count: 59000, controller: this.textTimerController }) 
          .format("ss") 
          .fontColor($r("app.color.theme_color_m")) 
          .onTimer((utc: number, elapsedTime: number) => { 
            this.elapsedTime = elapsedTime 
          }) 
        Text($r("app.string.login")) 
          .fontSize(15) 
          .fontColor($r("app.color.theme_color_m")) 
          .onClick(() => { 
            this.textTimerController.start() 
          }) 
      } 
      .margin({ top: 25 }) 
      .visibility(this.elapsedTime == 59000 ? Visibility.None : Visibility.Visible) 
 
      Button('重置后启动') 
        .onClick(() => { 
          this.textTimerController.reset() 
          this.textTimerController.start() 
        }) 
 
      Text() { 
        Span('1111111111111') 
          .fontColor($r("app.color.theme_color_m")) 
        Span('2222222222222') 
          .fontColor($r("app.color.theme_color_m")) 
          .onClick(() => { 
            this.textTimerController.reset() 
            this.textTimerController.start() 
          }) 
      } 
      // .visibility(this.elapsedTime == 59000 ? Visibility.Visible : Visibility.None) 
      .textAlign(TextAlign.Center) 
      .fontSize(13) 
      .margin({ top: 25 }) 
      .width("90%") 
 
    } 
    .width('100%') 
    .height('100%') 
  } 
}
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题