wd-cell.vue 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. <template>
  2. <view
  3. :class="['wd-cell', isBorder ? 'is-border' : '', size ? 'is-' + size : '', center ? 'is-center' : '', customClass]"
  4. :style="customStyle"
  5. :hover-class="isLink || clickable ? 'is-hover' : 'none'"
  6. :hover-stay-time="70"
  7. @click="onClick"
  8. >
  9. <view :class="['wd-cell__wrapper', vertical ? 'is-vertical' : '']">
  10. <view v-if="showLeft" class="wd-cell__left" :style="titleWidth ? 'min-width:' + titleWidth + ';max-width:' + titleWidth + ';' : ''">
  11. <text v-if="isRequired && markerSide === 'before'" class="wd-cell__required wd-cell__required--left">*</text>
  12. <!--左侧icon部位-->
  13. <slot name="icon">
  14. <wd-icon v-if="icon" :name="icon" :size="iconSize" :custom-class="`wd-cell__icon ${customIconClass}`"></wd-icon>
  15. </slot>
  16. <view class="wd-cell__title">
  17. <!--title BEGIN-->
  18. <slot v-if="useTitleSlot && $slots.title" name="title"></slot>
  19. <text v-else-if="title" :class="customTitleClass">{{ title }}</text>
  20. <!--title END-->
  21. <!--label BEGIN-->
  22. <slot name="label">
  23. <view v-if="label" :class="`wd-cell__label ${customLabelClass}`">{{ label }}</view>
  24. </slot>
  25. <!--label END-->
  26. </view>
  27. <text v-if="isRequired && markerSide === 'after'" class="wd-cell__required">*</text>
  28. </view>
  29. <!--right content BEGIN-->
  30. <view class="wd-cell__right">
  31. <view class="wd-cell__body">
  32. <!--文案内容-->
  33. <view :class="`wd-cell__value ${customValueClass} wd-cell__value--${valueAlign} ${ellipsis ? 'wd-cell__value--ellipsis' : ''}`">
  34. <slot>{{ value }}</slot>
  35. </view>
  36. <!--箭头-->
  37. <wd-icon v-if="isLink" custom-class="wd-cell__arrow-right" name="arrow-right" />
  38. <slot v-else name="right-icon" />
  39. </view>
  40. <view v-if="errorMessage" class="wd-cell__error-message">{{ errorMessage }}</view>
  41. </view>
  42. <!--right content END-->
  43. </view>
  44. </view>
  45. </template>
  46. <script lang="ts">
  47. export default {
  48. name: 'wd-cell',
  49. options: {
  50. addGlobalClass: true,
  51. virtualHost: true,
  52. styleIsolation: 'shared'
  53. }
  54. }
  55. </script>
  56. <script lang="ts" setup>
  57. import wdIcon from '../wd-icon/wd-icon.vue'
  58. import { computed, useSlots } from 'vue'
  59. import { useCell } from '../composables/useCell'
  60. import { useParent } from '../composables/useParent'
  61. import { FORM_KEY } from '../wd-form/types'
  62. import { cellProps } from './types'
  63. import { isDef } from '../common/util'
  64. const props = defineProps(cellProps)
  65. const emit = defineEmits(['click'])
  66. // 获取插槽
  67. const slots = useSlots()
  68. const cell = useCell()
  69. const isBorder = computed(() => {
  70. return Boolean(isDef(props.border) ? props.border : cell.border.value)
  71. })
  72. const { parent: form } = useParent(FORM_KEY)
  73. const errorMessage = computed(() => {
  74. if (form && props.prop && form.errorMessages && form.errorMessages[props.prop]) {
  75. return form.errorMessages[props.prop]
  76. } else {
  77. return ''
  78. }
  79. })
  80. // 是否展示必填
  81. const isRequired = computed(() => {
  82. let formRequired = false
  83. if (form && form.props.rules) {
  84. const rules = form.props.rules
  85. for (const key in rules) {
  86. if (Object.prototype.hasOwnProperty.call(rules, key) && key === props.prop && Array.isArray(rules[key])) {
  87. formRequired = rules[key].some((rule) => rule.required)
  88. }
  89. }
  90. }
  91. return props.required || props.rules.some((rule) => rule.required) || formRequired
  92. })
  93. // 是否展示左侧部分
  94. const showLeft = computed(() => {
  95. // 插槽优先级高于props
  96. // 有icon插槽或icon属性
  97. const hasIcon = slots.icon || props.icon
  98. // 有title插槽或title属性
  99. const hasTitle = (slots.title && props.useTitleSlot) || props.title
  100. // 有label插槽或label属性
  101. const hasLabel = slots.label || props.label
  102. return hasIcon || hasTitle || hasLabel
  103. })
  104. /**
  105. * @description 点击cell的handle
  106. */
  107. function onClick() {
  108. const url = props.to
  109. if (props.clickable || props.isLink) {
  110. emit('click')
  111. }
  112. if (url && props.isLink) {
  113. if (props.replace) {
  114. uni.redirectTo({ url })
  115. } else {
  116. uni.navigateTo({ url })
  117. }
  118. }
  119. }
  120. </script>
  121. <style lang="scss" scoped>
  122. @import './index.scss';
  123. </style>