# transition-delay

CSS 的transition-delay属性规定了在过渡效果开始作用之前需要等待的时间。

# uni-app x 兼容性

Web Android iOS HarmonyOS HarmonyOS(Vapor)
4.0 3.9 4.11 4.61 5.0

# 语法

transition-delay: <time>#;

# 值限制

  • time

# App平台

  • 不支持指定多个时长
  • HBuilderX4.0以下版本,属性值可以不设置单位,不设置单位时当做 ms(毫秒) 处理。从 HBuilderX4.0 起,统一为web的策略,必须设置单位,无单位当做非法值。也就是会造成动画不生效。

# Web规范

属性值必须设置单位,无单位时当做非法值处理

# 默认值

0s

# 示例

示例为hello uni-app x alpha分支,与最新HBuilderX Alpha版同步。与最新正式版同步的master分支示例另见

扫码体验(手机浏览器跳转到App直达页)

示例

<template>
  <!-- #ifdef APP -->
  <scroll-view style="flex: 1">
  <!-- #endif -->
    <view>
      <text class="uni-title-text">view 组件 transition-delay:1s</text>
      <view :class="classValue"></view>
      <view class="button-container">
        <button class="button-item" @click="start">start</button>
        <button class="button-item" @click="reset">reset</button>
      </view>

      <text class="uni-title-text uni-common-mt">text 组件 transition-delay:2s</text>
      <text :class="textClassValue">transition-delay: 2s</text>
      <view class="button-container">
        <button class="button-item" @click="textStart">text start</button>
        <button class="button-item" @click="textReset">text reset</button>
      </view>

      <text class="uni-title-text uni-common-mt">image 组件 transition-delay:1s</text>
      <image :class="imageClassValue" src="/static/test-image/logo.png"></image>
      <view class="button-container">
        <button class="button-item" @click="imageStart">image start</button>
        <button class="button-item" @click="imageReset">image reset</button>
      </view>

      <view class="uni-common-mt">
        <text class="uni-title-text">setProperty 设置与 getPropertyValue 获取 transition-delay </text>
      </view>

      <view class="test-container">
        <view class="test-item">
          <text class="uni-subtitle-text">view 组件</text>
          <text class="uni-info">设置值: {{transitionDelayDynamic}}</text>
          <text class="uni-info">获取值: {{transitionDelayActual}}</text>
          <view class="test-box">
            <view ref="viewRefDynamic" class="common-image test-view" :style="{ transitionDelay: transitionDelayDynamic, transitionProperty: 'width', transitionDuration: '1s' }" @click="triggerTransitionDynamic">
              <text style="font-size: 12px;">点击view</text>
            </view>
          </view>
        </view>

        <view class="test-item">
          <text class="uni-subtitle-text">text 组件</text>
          <text class="uni-info">设置值: {{transitionDelayDynamic}}</text>
          <text class="uni-info">获取值: {{transitionDelayActualText}}</text>
          <view class="test-box">
            <text ref="textRefDynamic" class="common-text test-text" :style="{ transitionDelay: transitionDelayDynamic, transitionProperty: 'width', transitionDuration: '1s' }" @click="triggerTransitionTextDynamic">点击text</text>
          </view>
        </view>

        <view class="test-item">
          <text class="uni-subtitle-text">image 组件</text>
          <text class="uni-info">设置值: {{transitionDelayDynamic}}</text>
          <text class="uni-info">获取值: {{transitionDelayActualImage}}</text>
          <view class="test-box">
            <image ref="imageRefDynamic" class="common-image test-image" :style="{ transitionDelay: transitionDelayDynamic, transitionProperty: 'width', transitionDuration: '1s' }" @click="triggerTransitionImageDynamic" src="/static/test-image/logo.png"></image>
          </view>
        </view>
      </view>

      <view class="uni-common-mt uni-common-mb">
        <text class="uni-tips">第一个枚举值,'' (空字符串) - 空值情况</text>
        <enum-data :items="transitionDelayEnum" title="transition-delay 枚举值" @change="radioChangeTransitionDelay" :compact="true"></enum-data>
        <input-data :defaultValue="transitionDelayDynamic" title="transition-delay 自定义值" type="text" @confirm="inputChangeTransitionDelay"></input-data>
      </view>
    </view>
  <!-- #ifdef APP -->
  </scroll-view>
  <!-- #endif -->
</template>

<script setup lang="uts">

  const classValue = ref('box')
  const textClassValue = ref('text-box')
  const imageClassValue = ref('image-box')

  const start = () => {
    classValue.value = 'box ani'
  }

  const reset = () => {
    classValue.value = 'box'
  }

  const textStart = () => {
    textClassValue.value = 'text-box text-ani'
  }

  const textReset = () => {
    textClassValue.value = 'text-box'
  }

  const imageStart = () => {
    imageClassValue.value = 'image-box image-ani'
  }

  const imageReset = () => {
    imageClassValue.value = 'image-box'
  }

  const jest_start = () => {
    start()
  }

  const jest_reset = () => {
    reset()
  }

  import { ItemType } from '@/components/enum-data/enum-data-types'

  const transitionDelayDynamic = ref('1s')
  const transitionDelayActual = ref('')
  const transitionDelayActualText = ref('')
  const transitionDelayActualImage = ref('')
  const viewRefDynamic = ref(null as UniElement | null)
  const textRefDynamic = ref(null as UniTextElement | null)
  const imageRefDynamic = ref(null as UniImageElement | null)
  const isExpandedDynamic = ref(false)
  const isExpandedDynamicText = ref(false)
  const isExpandedDynamicImage = ref(false)

  const transitionDelayEnum: ItemType[] = [
    { value: 0, name: '' },
    { value: 1, name: '0s' },
    { value: 2, name: '0.5s' },
    { value: 3, name: '1s' },
    { value: 4, name: '2s' },
  ]

  const getPropertyValues = () => {
    transitionDelayActual.value = viewRefDynamic.value?.style.getPropertyValue('transition-delay') ?? ''
    transitionDelayActualText.value = textRefDynamic.value?.style.getPropertyValue('transition-delay') ?? ''
    transitionDelayActualImage.value = imageRefDynamic.value?.style.getPropertyValue('transition-delay') ?? ''
  }

  const changeTransitionDelayDynamic = (value: string) => {
    transitionDelayDynamic.value = value
    viewRefDynamic.value?.style.setProperty('transition-delay', value)
    textRefDynamic.value?.style.setProperty('transition-delay', value)
    imageRefDynamic.value?.style.setProperty('transition-delay', value)
    // 使用 nextTick 确保样式已应用后再获取值
    nextTick(() => {
      getPropertyValues()
    })
  }

  const radioChangeTransitionDelay = (index: number) => {
    const selectedItem = transitionDelayEnum.find((item): boolean => item.value === index)
    if (selectedItem != null) {
      changeTransitionDelayDynamic(selectedItem.name)
    }
  }

  const inputChangeTransitionDelay = (value: string) => {
    changeTransitionDelayDynamic(value)
  }

  const triggerTransitionDynamic = () => {
    isExpandedDynamic.value = !isExpandedDynamic.value
    const width = isExpandedDynamic.value ? '100px' : '50px'
    viewRefDynamic.value?.style.setProperty('width', width)
  }

  const triggerTransitionTextDynamic = () => {
    isExpandedDynamicText.value = !isExpandedDynamicText.value
    const width = isExpandedDynamicText.value ? '100px' : '50px'
    textRefDynamic.value?.style.setProperty('width', width)
  }

  const triggerTransitionImageDynamic = () => {
    isExpandedDynamicImage.value = !isExpandedDynamicImage.value
    const width = isExpandedDynamicImage.value ? '100px' : '50px'
    imageRefDynamic.value?.style.setProperty('width', width)
  }

  onReady(() => {
    getPropertyValues()
  })

  defineExpose({
    jest_start,
    jest_reset,
    radioChangeTransitionDelay
  })

</script>

<style>
  .box {
    width: 200px;
    height: 50px;
    background-color: blue;
  }

  .ani {
    transition-property: width;
    transition-duration: 2s;
    transition-delay: 1s;
    width: 300px;
  }

  .text-box {
    width: 200px;
    height: 60px;
    padding: 10px;
    font-size: 16px;
    background-color: lightblue;
    text-align: center;
  }

  .text-ani {
    transition-property: width;
    transition-duration: 2s;
    transition-delay: 2s;
    width: 300px;
  }

  .image-box {
    width: 100px;
    height: 100px;
  }

  .image-ani {
    transition-property: width, height;
    transition-duration: 2s;
    transition-delay: 1s;
    width: 150px;
    height: 150px;
  }

  .common-text {
    width: 50px;
    height: 50px;
    background-color: green;
    font-size: 12px;
    color: white;
  }

  .common-image {
    width: 50px;
    height: 50px;
    background-color: green;
  }

  .test-container {
    flex-direction: row;
    justify-content: space-between;
    margin-top: 10px;
  }

  .test-item {
    flex: 1;
    margin: 0 5px;
  }

  .test-box {
    width: 100%;
    height: 80px;
    background-color: gray;
  }


  .button-container {
    flex-direction: row;
    margin-top: 10px;
    padding-bottom: 15px;
    border-bottom:#d3d3d3 solid 1px;
  }

  .button-item {
    margin-right: 10px;
  }
</style>

# 参见