index.vue 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. <template>
  2. <div class="table-header-cell el-table-v2__header-cell-text">
  3. <template v-if="!filterable || filterStyle === 'arrow'">
  4. {{ label }}
  5. </template>
  6. <template v-if="filterable">
  7. <el-popover
  8. class="table-header-cell-popover"
  9. placement="bottom"
  10. trigger="click"
  11. @show="expand = true"
  12. @hide="expand = false"
  13. >
  14. <span
  15. v-if="filterStyle === 'underline'"
  16. slot="reference"
  17. :class="['btn-filter', { 'btn-filter-active': active }]"
  18. >{{ label }}</span
  19. >
  20. <i
  21. v-if="filterStyle === 'arrow'"
  22. slot="reference"
  23. :class="[
  24. 'filter-arrow',
  25. 'el-icon-arrow-down',
  26. { 'arrow-active': active, 'arrow-expand': expand },
  27. ]"
  28. />
  29. <el-form>
  30. <el-form-item :label="label">
  31. <el-select
  32. v-model="selections"
  33. size="small"
  34. placeholder="筛选"
  35. multiple
  36. filterable
  37. default-first-option
  38. collapse-tags
  39. clearable
  40. @change="
  41. newVal => {
  42. $emit('update:filter-values', newVal)
  43. }
  44. "
  45. >
  46. <el-option
  47. v-for="(option, index) in filterOptions"
  48. :key="option.value + index"
  49. :value="option.value"
  50. :label="option.label"
  51. />
  52. </el-select>
  53. </el-form-item>
  54. </el-form>
  55. </el-popover>
  56. </template>
  57. <template v-if="sortable">
  58. <span class="caret-wrapper" @click="sortChange">
  59. <i class="sort-caret ascending" />
  60. <i class="sort-caret descending" />
  61. </span>
  62. </template>
  63. </div>
  64. </template>
  65. <script setup lang="ts">
  66. import { PropType } from 'vue'
  67. const props = defineProps({
  68. label: {
  69. type: String,
  70. default: '',
  71. },
  72. filterStyle: {
  73. type: String as PropType<'arrow' | 'underline'>,
  74. default: 'arrow',
  75. },
  76. filterOptions: {
  77. type: Array as PropType<{ label: string, value: string }[]>,
  78. },
  79. filterValues: {
  80. type: Array<string>,
  81. },
  82. sortable: {
  83. type: Boolean,
  84. default: false,
  85. },
  86. sortRule: {
  87. type: String,
  88. default: '',
  89. },
  90. })
  91. const emit = defineEmits(['update:sortRule'])
  92. const selections = ref<string[]>([])
  93. const expand = ref(false)
  94. const active = computed(() => !!props.filterValues?.length)
  95. const filterable = computed(() => !!props.filterOptions)
  96. watchEffect(() => {
  97. if (props.filterValues) {
  98. selections.value = props.filterValues
  99. }
  100. })
  101. const sortChange = () => {
  102. const sortRule =
  103. props.sortRule === ''
  104. ? 'ascending'
  105. : props.sortRule === 'ascending'
  106. ? 'descending'
  107. : ''
  108. emit('update:sortRule', sortRule)
  109. }
  110. </script>
  111. <style lang="scss" scoped>
  112. .filter-arrow {
  113. cursor: pointer;
  114. transition: 0.3s transform;
  115. &.arrow-expand {
  116. transform: rotate(-180deg);
  117. }
  118. &.arrow-active {
  119. color: #2d7cff;
  120. font-weight: bold;
  121. }
  122. }
  123. .btn-filter {
  124. cursor: pointer;
  125. position: relative;
  126. &:hover {
  127. color: #2d7cff;
  128. }
  129. &::after {
  130. content: '';
  131. display: block;
  132. width: calc(100% + 4px);
  133. position: absolute;
  134. bottom: -4px;
  135. left: -2px;
  136. border-bottom: 1px solid #101116;
  137. }
  138. &.btn-filter-active,
  139. &:hover {
  140. &::after {
  141. border-bottom: 2px solid #2d7cff;
  142. }
  143. }
  144. }
  145. .el-select-dropdown__item.hover {
  146. background: #d2d6df;
  147. }
  148. </style>