| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 |
- <!-- 自定义TimelineItem组件 -->
- <script setup>
- import { ref, computed, watch, inject } from 'vue'
- import { formatCls, isTrue, addUnit, Int } from './util.js'
- const props = defineProps({
- type: { type: String, default: '' },
- // 时间戳
- timestamp: { type: String, default: '' },
- // 隐藏时间戳
- hideTimestamp: { type: [Boolean, String], default: false },
- // 时间戳位置(top / bottom)
- placement: { type: String, default: 'bottom' },
- // 节点颜色
- color: String,
- // 节点图标
- icon: String,
- // 节点尺寸
- size: { type: [Number, String], default: 12 },
- // 分割线类型(solid / dashed / dotted)
- lineType: { type: String, default: 'solid' },
- // 分割线颜色
- lineColor: String
- })
- const emit = defineEmits(['click'])
- const preClass = formatCls(props, 've__timeline-item--', ['type'])
- const lineStyle = computed(() => {
- return `
- border-left-style: ${props.lineType};
- border-left-color: ${props.lineColor};
- `
- })
- const nodeStyle = computed(() => {
- return `
- background: ${props.color};
- height: ${addUnit(Int(props.size))};
- width: ${addUnit(Int(props.size))};
- left: ${Int(props.size) > 12 ? -(Int(props.size) - 12) / 2 + 'px' : (12 - Int(props.size)) / 2 + 'px'};
- `
- })
- // 事件
- const handleClick = (e) => {
- emit('click', e)
- }
- </script>
- <template>
- <li class="ve__timeline-item" :class="[preClass]">
- <!-- 时间线 -->
- <div class="ve__timeline-item__line" :style="lineStyle"></div>
- <!-- 时间线节点 -->
- <div v-if="!$slots.dot" class="ve__timeline-item__node" :style="nodeStyle">
- <i v-if="icon" :class="icon"></i>
- </div>
- <!-- 自定义时间线圆点dot -->
- <div v-if="$slots.dot" class="ve__timeline-item__dot" :style="{color}">
- <slot name="dot" />
- </div>
- <!-- 内容区 -->
- <div class="ve__timeline-item__body" :style="{'top': addUnit(Int(size) / 2 - 9)}">
- <div v-if="!isTrue(hideTimestamp) && placement == 'top'" class="ve__timeline-item__timestamp is-top">{{timestamp}}</div>
- <div class="ve__timeline-item__content">
- <slot />
- </div>
- <div v-if="!isTrue(hideTimestamp) && placement == 'bottom'" class="ve__timeline-item__timestamp is-bottom">{{timestamp}}</div>
- </div>
- </li>
- </template>
- <style lang="scss" scoped>
- $color-primary: #409eff;
- $color-success: #31c48d;
- $color-warning: #e6a23c;
- $color-danger: #f56c6c;
- $color-info: #909399;
- $color-white: #fff;
- $color-black: #000;
-
- /*Hover/Active Color*/
- $primary-outline-color: #d9ecff;
- /*Font Color*/
- $color-text-primary: #303133;
- $color-text-regular: #606266;
- $color-text-secondary: #909399;
- $color-text-placeholder: #c0c4cc;
- /*Border Color*/
- $border-color-base: #dcdfe6;
- $border-color-light: #e4e7ed;
- $border-color-lighter: #ebeef5;
- $border-color-extra-light: #f2f6fc;
- /*Background*/
- $background-color-base: #f5f7fa;
- /*Box-shadow*/
- $box-shadow: 0 3px 9px rgba(0,0,0,.08), 0 3px 3px rgba(0,0,0,.04);
- $box-shadow-base: 0 3px 6px -4px rgba(0, 0, 0, 0.12), 0 6px 16px 0 rgba(0, 0, 0, 0.08), 0 9px 28px 8px rgba(0, 0, 0, 0.05);
- $box-shadow-light: 0 4px 12px -8px rgba(0, 0, 0, 0.1), 0 6px 18px 0 rgba(0, 0, 0, 0.06), 0 8px 24px 16px rgba(0, 0, 0, 0.03);
- $box-shadow-lighter: 0 6px 16px -8px rgba(0, 0, 0, 0.08), 0 9px 28px 0 rgba(0, 0, 0, 0.04), 0 12px 48px 16px rgba(0, 0, 0, 0.03);
- $border-width-base: 1px;
- $border-style-base: solid;
- $border-color-hover: $color-text-placeholder;
- $border-base: $border-width-base $border-style-base $border-color-base;
- $border-radius-base: 4px;
- $border-radius-small: 2px;
- $border-radius-circle: 100%;
- $border-radius-zero: 0;
- $size-base: 14px;
- $font-size-extra-large: 18px;
- $font-size-large: 16px;
- $font-size-base: 14px;
- $font-size-small: 12px;
- $font-weight-primary: 500;
- $font-weight-secondary: 100;
- $font-line-height-primary: 24px;
- $font-line-height-secondary: 16px;
- $font-color-disabled-base: #bbb;
- $disabled-fill-base: $background-color-base;
- $disabled-color-base: $color-text-placeholder;
- $disabled-border-base: $border-color-light;
- $height-base: 32px;
- $large-height-base: 40px;
- $small-height-base: 24px;
-
- /* Timeline
- --------------------------------*/
- $timeline-font-size: $font-size-base;
- $timeline-node-color: $border-color-light;
- $timeline-node-size: $font-size-small;
- .ve__timeline {
- font-size: $timeline-font-size;
- margin: 0;
- padding: 0;
- list-style: none;
- }
- .ve__timeline-item {padding-bottom: 20px; position: relative;}
- .ve__timeline-item:last-child {padding-bottom: 0;}
- .ve__timeline-item__line {
- border-left: 2px solid $timeline-node-color;
- height: 100%;
- position: absolute;
- left: calc(($timeline-node-size - 2px) / 2);
- }
- .ve__timeline-item:last-child .ve__timeline-item__line {display: none;}
- .ve__timeline-item__node {
- display: inline-flex;
- align-items: center;
- justify-content: center;
- background-color: $timeline-node-color;
- border-radius: 50%;
- height: $timeline-node-size;
- width: $timeline-node-size;
- position: absolute;
- }
- .ve__timeline-item__node [class*=ve-icon-] {color: #fff; font-size: inherit;}
- .ve__timeline-item__dot {
- display: inline-flex;
- align-items: center;
- justify-content: center;
- position: absolute;
- }
- .ve__timeline-item__body {padding-left: calc($timeline-node-size + 18px); position: relative; top: calc($timeline-node-size / 2 - 9px);}
- .ve__timeline-item__content {color: $color-text-primary; font-size: inherit; font-family: system-ui; word-break: break-all;}
- .ve__timeline-item__timestamp {
- color: $color-text-secondary;
- font-size: 12px;
- line-height: 1;
- }
- .ve__timeline-item__timestamp.is-top {margin-bottom: 7px; padding-top: 3px;}
- .ve__timeline-item__timestamp.is-bottom {margin-top: 7px;}
- /*type*/
- .ve__timeline-item--primary .ve__timeline-item__node {background-color: $color-primary;}
- .ve__timeline-item--primary .ve__timeline-item__dot {color: $color-primary;}
- .ve__timeline-item--success .ve__timeline-item__node {background-color: $color-success;}
- .ve__timeline-item--success .ve__timeline-item__dot {color: $color-success;}
- .ve__timeline-item--warning .ve__timeline-item__node {background-color: $color-warning;}
- .ve__timeline-item--warning .ve__timeline-item__dot {color: $color-warning;}
- .ve__timeline-item--danger .ve__timeline-item__node {background-color: $color-danger;}
- .ve__timeline-item--danger .ve__timeline-item__dot {color: $color-danger;}
- .ve__timeline-item--info .ve__timeline-item__node {background-color: $color-info;}
- .ve__timeline-item--info .ve__timeline-item__dot {color: $color-info;}
- /*卡片card*/
- .ve__timeline--card .ve__timeline-item__content {
- border-radius: 4px;
- box-shadow: 0 1px 4px $border-color-base;
- padding: 20px;
- }
- </style>
|