vant/src/progress/Progress.tsx
2021-08-25 17:09:59 +08:00

82 lines
2.0 KiB
TypeScript

import { computed, defineComponent, ExtractPropTypes } from 'vue';
import { truthProp, createNamespace, addUnit } from '../utils';
const [name, bem] = createNamespace('progress');
const transition = 'all 0.3s cubic-bezier(0.46, 0.03, 0.52, 0.96)';
const props = {
color: String,
inactive: Boolean,
pivotText: String,
textColor: String,
showPivot: truthProp,
pivotColor: String,
trackColor: String,
strokeWidth: [Number, String],
percentage: {
type: [Number, String],
validator: (value: number | string) => value >= 0 && value <= 100,
},
};
export type ProgressProps = ExtractPropTypes<typeof props>;
export default defineComponent({
name,
props,
setup(props) {
const background = computed(() =>
props.inactive ? '#cacaca' : props.color
);
const scaleX = computed(() => +props.percentage! / 100);
const renderPivot = () => {
const { textColor, pivotText, pivotColor, percentage } = props;
const text = pivotText ?? `${percentage}%`;
const show = props.showPivot && text;
if (show) {
const style = {
color: textColor,
left: `${+percentage!}%`,
transform: `translate(-${+percentage!}%,-50%)`,
background: pivotColor || background.value,
transition,
};
return (
<span style={style} class={bem('pivot')}>
{text}
</span>
);
}
};
return () => {
const { trackColor, strokeWidth } = props;
const rootStyle = {
background: trackColor,
height: addUnit(strokeWidth),
};
const portionStyle = {
background: background.value,
width: '100%',
transform: `scaleX(${scaleX.value})`,
'transform-origin': 0,
transition,
};
return (
<div class={bem()} style={rootStyle}>
<span class={bem('portion')} style={portionStyle}></span>
{renderPivot()}
</div>
);
};
},
});