UNPKG

element-plus

Version:

A Component Library for Vue3.0

326 lines (319 loc) 12.1 kB
import { defineComponent, inject, ref, computed, watch, openBlock, createBlock, Fragment, renderList, createVNode, createCommentVNode, toDisplayString } from 'vue'; import { hasClass } from '../utils/dom'; import { EVENT_CODE } from '../utils/aria'; import { elFormKey } from '../el-form'; /** * Make a map and return a function for checking if a key * is in that map. * IMPORTANT: all calls of this function must be prefixed with * \/\*#\_\_PURE\_\_\*\/ * So that rollup can tree-shake them if necessary. */ const EMPTY_OBJ = (process.env.NODE_ENV !== 'production') ? Object.freeze({}) : {}; const EMPTY_ARR = (process.env.NODE_ENV !== 'production') ? Object.freeze([]) : []; const isArray = Array.isArray; const isObject = (val) => val !== null && typeof val === 'object'; var script = defineComponent({ name: 'ElRate', props: { modelValue: { type: Number, default: 0, }, lowThreshold: { type: Number, default: 2, }, highThreshold: { type: Number, default: 4, }, max: { type: Number, default: 5, }, colors: { type: [Array, Object], default: () => ['#F7BA2A', '#F7BA2A', '#F7BA2A'], }, voidColor: { type: String, default: '#C6D1DE', }, disabledVoidColor: { type: String, default: '#EFF2F7', }, iconClasses: { type: [Array, Object], default: () => ['el-icon-star-on', 'el-icon-star-on', 'el-icon-star-on'], }, voidIconClass: { type: String, default: 'el-icon-star-off', }, disabledVoidIconClass: { type: String, default: 'el-icon-star-on', }, disabled: { type: Boolean, default: false, }, allowHalf: { type: Boolean, default: false, }, showText: { type: Boolean, default: false, }, showScore: { type: Boolean, default: false, }, textColor: { type: String, default: '#1f2d3d', }, texts: { type: Array, default: () => ['Extremely bad', 'Disappointed', 'Fair', 'Satisfied', 'Surprise'], }, scoreTemplate: { type: String, default: '{value}', }, }, emits: ['update:modelValue', 'change'], setup(props, { emit }) { const elForm = inject(elFormKey, {}); const currentValue = ref(props.modelValue); const rateDisabled = computed(() => props.disabled || elForm.disabled); const text = computed(() => { let result = ''; if (props.showScore) { result = props.scoreTemplate.replace(/\{\s*value\s*\}/, rateDisabled.value ? `${props.modelValue}` : `${currentValue.value}`); } else if (props.showText) { result = props.texts[Math.ceil(currentValue.value) - 1]; } return result; }); function getValueFromMap(value, map) { const matchedKeys = Object.keys(map) .filter(key => { const val = map[key]; const excluded = isObject(val) ? val.excluded : false; return excluded ? value < key : value <= key; }) .sort((a, b) => a - b); const matchedValue = map[matchedKeys[0]]; return isObject(matchedValue) ? matchedValue.value : (matchedValue || ''); } const valueDecimal = computed(() => props.modelValue * 100 - Math.floor(props.modelValue) * 100); const colorMap = computed(() => isArray(props.colors) ? { [props.lowThreshold]: props.colors[0], [props.highThreshold]: { value: props.colors[1], excluded: true }, [props.max]: props.colors[2], } : props.colors); const activeColor = computed(() => getValueFromMap(currentValue.value, colorMap.value)); const decimalStyle = computed(() => { let width = ''; if (rateDisabled.value) { width = `${valueDecimal.value}%`; } else if (props.allowHalf) { width = '50%'; } return { color: activeColor.value, width, }; }); const classMap = computed(() => isArray(props.iconClasses) ? { [props.lowThreshold]: props.iconClasses[0], [props.highThreshold]: { value: props.iconClasses[1], excluded: true }, [props.max]: props.iconClasses[2], } : props.iconClasses); const decimalIconClass = computed(() => getValueFromMap(props.modelValue, classMap.value)); const voidClass = computed(() => rateDisabled.value ? props.disabledVoidIconClass : props.voidIconClass); const activeClass = computed(() => getValueFromMap(currentValue.value, classMap.value)); const classes = computed(() => { let result = Array(props.max); let threshold = currentValue.value; result.fill(activeClass.value, 0, threshold); result.fill(voidClass.value, threshold, props.max); return result; }); const pointerAtLeftHalf = ref(true); watch(() => props.modelValue, val => { currentValue.value = val; pointerAtLeftHalf.value = props.modelValue !== Math.floor(props.modelValue); }); function showDecimalIcon(item) { let showWhenDisabled = rateDisabled.value && valueDecimal.value > 0 && item - 1 < props.modelValue && item > props.modelValue; let showWhenAllowHalf = props.allowHalf && pointerAtLeftHalf.value && item - 0.5 <= currentValue.value && item > currentValue.value; return showWhenDisabled || showWhenAllowHalf; } function getIconStyle(item) { const voidColor = rateDisabled.value ? props.disabledVoidColor : props.voidColor; return { color: item <= currentValue.value ? activeColor.value : voidColor, }; } function selectValue(value) { if (rateDisabled.value) { return; } if (props.allowHalf && pointerAtLeftHalf.value) { emit('update:modelValue', currentValue.value); emit('change', this.currentValue); } else { emit('update:modelValue', value); emit('change', value); } } function handleKey(e) { if (rateDisabled.value) { return; } let _currentValue = currentValue.value; const code = e.code; if (code === EVENT_CODE.up || code === EVENT_CODE.right) { if (props.allowHalf) { _currentValue += 0.5; } else { _currentValue += 1; } e.stopPropagation(); e.preventDefault(); } else if (code === EVENT_CODE.left || code === EVENT_CODE.down) { if (props.allowHalf) { _currentValue -= 0.5; } else { _currentValue -= 1; } e.stopPropagation(); e.preventDefault(); } _currentValue = _currentValue < 0 ? 0 : _currentValue; _currentValue = _currentValue > props.max ? props.max : _currentValue; emit('update:modelValue', _currentValue); emit('change', _currentValue); return _currentValue; } const hoverIndex = ref(-1); function setCurrentValue(value, event) { if (rateDisabled.value) { return; } if (props.allowHalf) { let target = event.target; if (hasClass(target, 'el-rate__item')) { target = target.querySelector('.el-rate__icon'); } if (hasClass(target, 'el-rate__decimal')) { target = target.parentNode; } pointerAtLeftHalf.value = event.offsetX * 2 <= target.clientWidth; currentValue.value = pointerAtLeftHalf.value ? value - 0.5 : value; } else { currentValue.value = value; } hoverIndex.value = value; } function resetCurrentValue() { if (rateDisabled.value) { return; } if (props.allowHalf) { pointerAtLeftHalf.value = props.modelValue !== Math.floor(props.modelValue); } currentValue.value = props.modelValue; hoverIndex.value = -1; } if (!props.modelValue) { emit('update:modelValue', 0); } return { hoverIndex, currentValue, rateDisabled, text, decimalStyle, decimalIconClass, classes, showDecimalIcon, getIconStyle, selectValue, handleKey, setCurrentValue, resetCurrentValue, }; }, }); function render(_ctx, _cache, $props, $setup, $data, $options) { return (openBlock(), createBlock("div", { class: "el-rate", role: "slider", "aria-valuenow": _ctx.currentValue, "aria-valuetext": _ctx.text, "aria-valuemin": "0", "aria-valuemax": _ctx.max, tabindex: "0", onKeydown: _cache[2] || (_cache[2] = (...args) => (_ctx.handleKey && _ctx.handleKey(...args))) }, [ (openBlock(true), createBlock(Fragment, null, renderList(_ctx.max, (item, key) => { return (openBlock(), createBlock("span", { key: key, class: "el-rate__item", style: { cursor: _ctx.rateDisabled ? 'auto' : 'pointer' }, onMousemove: $event => (_ctx.setCurrentValue(item, $event)), onMouseleave: _cache[1] || (_cache[1] = (...args) => (_ctx.resetCurrentValue && _ctx.resetCurrentValue(...args))), onClick: $event => (_ctx.selectValue(item)) }, [ createVNode("i", { class: [[_ctx.classes[item - 1], { 'hover': _ctx.hoverIndex === item }], "el-rate__icon"], style: _ctx.getIconStyle(item) }, [ (_ctx.showDecimalIcon(item)) ? (openBlock(), createBlock("i", { key: 0, class: [_ctx.decimalIconClass, "el-rate__decimal"], style: _ctx.decimalStyle }, null, 6 /* CLASS, STYLE */)) : createCommentVNode("v-if", true) ], 6 /* CLASS, STYLE */) ], 44 /* STYLE, PROPS, HYDRATE_EVENTS */, ["onMousemove", "onClick"])) }), 128 /* KEYED_FRAGMENT */)), (_ctx.showText || _ctx.showScore) ? (openBlock(), createBlock("span", { key: 0, class: "el-rate__text", style: { color: _ctx.textColor } }, toDisplayString(_ctx.text), 5 /* TEXT, STYLE */)) : createCommentVNode("v-if", true) ], 40 /* PROPS, HYDRATE_EVENTS */, ["aria-valuenow", "aria-valuetext", "aria-valuemax"])) } script.render = render; script.__file = "packages/rate/src/index.vue"; script.install = (app) => { app.component(script.name, script); }; const _Rate = script; export default _Rate;