element-plus
Version:
A Component Library for Vue 3
301 lines (298 loc) • 11.5 kB
JavaScript
import { defineComponent, inject, ref, computed, markRaw, watch, openBlock, createElementBlock, unref, normalizeClass, normalizeStyle, Fragment, renderList, createVNode, withCtx, withDirectives, createBlock, resolveDynamicComponent, vShow, toDisplayString, createCommentVNode } from 'vue';
import { clamp } from 'lodash-unified';
import { ElIcon } from '../../icon/index.mjs';
import { rateProps, rateEmits } from './rate2.mjs';
import _export_sfc from '../../../_virtual/plugin-vue_export-helper.mjs';
import { formItemContextKey } from '../../form/src/constants.mjs';
import { useFormSize, useFormDisabled } from '../../form/src/hooks/use-form-common-props.mjs';
import { useNamespace } from '../../../hooks/use-namespace/index.mjs';
import { useFormItemInputId } from '../../form/src/hooks/use-form-item.mjs';
import { isArray, isObject, isString } from '@vue/shared';
import { UPDATE_MODEL_EVENT, CHANGE_EVENT } from '../../../constants/event.mjs';
import { getEventCode } from '../../../utils/dom/event.mjs';
import { EVENT_CODE } from '../../../constants/aria.mjs';
const _hoisted_1 = ["id", "aria-label", "aria-labelledby", "aria-valuenow", "aria-valuetext", "aria-valuemax"];
const _hoisted_2 = ["onMousemove", "onClick"];
const _sfc_main = defineComponent({
...{
name: "ElRate"
},
__name: "rate",
props: rateProps,
emits: rateEmits,
setup(__props, { expose: __expose, emit: __emit }) {
function getValueFromMap(value, map) {
const isExcludedObject = (val) => isObject(val);
const matchedKeys = Object.keys(map).map((key) => +key).filter((key) => {
const val = map[key];
const excluded = isExcludedObject(val) ? val.excluded : false;
return excluded ? value < key : value <= key;
}).sort((a, b) => a - b);
const matchedValue = map[matchedKeys[0]];
return isExcludedObject(matchedValue) && matchedValue.value || matchedValue;
}
const props = __props;
const emit = __emit;
const formItemContext = inject(formItemContextKey, void 0);
const rateSize = useFormSize();
const ns = useNamespace("rate");
const { inputId, isLabeledByFormItem } = useFormItemInputId(props, {
formItemContext
});
const currentValue = ref(props.modelValue);
const hoverIndex = ref(-1);
const pointerAtLeftHalf = ref(true);
const iconRefs = ref([]);
const iconClientWidths = computed(
() => iconRefs.value.map((icon) => icon.$el.clientWidth)
);
const rateClasses = computed(() => [ns.b(), ns.m(rateSize.value)]);
const rateDisabled = useFormDisabled();
const rateStyles = computed(() => {
return ns.cssVarBlock({
"void-color": props.voidColor,
"disabled-void-color": props.disabledVoidColor,
"fill-color": activeColor.value
});
});
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;
});
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(() => {
const color = getValueFromMap(currentValue.value, colorMap.value);
return isObject(color) ? "" : color;
});
const decimalStyle = computed(() => {
let width = "";
if (rateDisabled.value) {
width = `${valueDecimal.value}%`;
} else if (props.allowHalf) {
width = "50%";
}
return {
color: activeColor.value,
width
};
});
const componentMap = computed(() => {
let icons = isArray(props.icons) ? [...props.icons] : { ...props.icons };
icons = markRaw(icons);
return isArray(icons) ? {
[props.lowThreshold]: icons[0],
[props.highThreshold]: {
value: icons[1],
excluded: true
},
[props.max]: icons[2]
} : icons;
});
const decimalIconComponent = computed(
() => getValueFromMap(props.modelValue, componentMap.value)
);
const voidComponent = computed(
() => rateDisabled.value ? isString(props.disabledVoidIcon) ? props.disabledVoidIcon : markRaw(props.disabledVoidIcon) : isString(props.voidIcon) ? props.voidIcon : markRaw(props.voidIcon)
);
const activeComponent = computed(
() => getValueFromMap(currentValue.value, componentMap.value)
);
function showDecimalIcon(item) {
const showWhenDisabled = rateDisabled.value && valueDecimal.value > 0 && item - 1 < props.modelValue && item > props.modelValue;
const showWhenAllowHalf = props.allowHalf && pointerAtLeftHalf.value && item - 0.5 <= currentValue.value && item > currentValue.value;
return showWhenDisabled || showWhenAllowHalf;
}
function emitValue(value) {
if (props.clearable && value === props.modelValue) {
value = 0;
}
emit(UPDATE_MODEL_EVENT, value);
if (props.modelValue !== value) {
emit(CHANGE_EVENT, value);
}
}
function selectValue(value) {
if (rateDisabled.value) {
return;
}
if (props.allowHalf && pointerAtLeftHalf.value) {
emitValue(currentValue.value);
} else {
emitValue(value);
}
}
function handleKey(e) {
if (rateDisabled.value) {
return;
}
const code = getEventCode(e);
const step = props.allowHalf ? 0.5 : 1;
let _currentValue = currentValue.value;
switch (code) {
case EVENT_CODE.up:
case EVENT_CODE.right:
_currentValue += step;
break;
case EVENT_CODE.left:
case EVENT_CODE.down:
_currentValue -= step;
break;
}
_currentValue = clamp(_currentValue, 0, props.max);
if (_currentValue === currentValue.value) {
return;
}
e.stopPropagation();
e.preventDefault();
emit(UPDATE_MODEL_EVENT, _currentValue);
emit(CHANGE_EVENT, _currentValue);
return _currentValue;
}
function setCurrentValue(value, event) {
if (rateDisabled.value) {
return;
}
if (props.allowHalf && event) {
pointerAtLeftHalf.value = event.offsetX * 2 <= iconClientWidths.value[value - 1];
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;
}
watch(
() => props.modelValue,
(val) => {
currentValue.value = val;
pointerAtLeftHalf.value = props.modelValue !== Math.floor(props.modelValue);
}
);
if (!props.modelValue) {
emit(UPDATE_MODEL_EVENT, 0);
}
__expose({
setCurrentValue,
resetCurrentValue
});
return (_ctx, _cache) => {
var _a;
return openBlock(), createElementBlock("div", {
id: unref(inputId),
class: normalizeClass([rateClasses.value, unref(ns).is("disabled", unref(rateDisabled))]),
role: "slider",
"aria-label": !unref(isLabeledByFormItem) ? _ctx.ariaLabel || "rating" : void 0,
"aria-labelledby": unref(isLabeledByFormItem) ? (_a = unref(formItemContext)) == null ? void 0 : _a.labelId : void 0,
"aria-valuenow": currentValue.value,
"aria-valuetext": text.value || void 0,
"aria-valuemin": "0",
"aria-valuemax": _ctx.max,
tabindex: "0",
style: normalizeStyle(rateStyles.value),
onKeydown: handleKey
}, [
(openBlock(true), createElementBlock(
Fragment,
null,
renderList(_ctx.max, (item, key) => {
return openBlock(), createElementBlock("span", {
key,
class: normalizeClass(unref(ns).e("item")),
onMousemove: ($event) => setCurrentValue(item, $event),
onMouseleave: resetCurrentValue,
onClick: ($event) => selectValue(item)
}, [
createVNode(unref(ElIcon), {
ref_for: true,
ref_key: "iconRefs",
ref: iconRefs,
class: normalizeClass([
unref(ns).e("icon"),
{ hover: hoverIndex.value === item },
unref(ns).is("active", item <= currentValue.value),
unref(ns).is("focus-visible", item === Math.ceil(currentValue.value || 1))
])
}, {
default: withCtx(() => [
withDirectives((openBlock(), createBlock(
resolveDynamicComponent(activeComponent.value),
null,
null,
512
)), [
[vShow, !showDecimalIcon(item) && item <= currentValue.value]
]),
withDirectives((openBlock(), createBlock(
resolveDynamicComponent(voidComponent.value),
null,
null,
512
)), [
[vShow, !showDecimalIcon(item) && item > currentValue.value]
]),
withDirectives((openBlock(), createBlock(resolveDynamicComponent(voidComponent.value), {
class: normalizeClass([unref(ns).em("decimal", "box")])
}, null, 8, ["class"])), [
[vShow, showDecimalIcon(item)]
]),
withDirectives(createVNode(unref(ElIcon), {
style: normalizeStyle(decimalStyle.value),
class: normalizeClass([unref(ns).e("icon"), unref(ns).e("decimal")])
}, {
default: withCtx(() => [
(openBlock(), createBlock(resolveDynamicComponent(decimalIconComponent.value)))
]),
_: 1
}, 8, ["style", "class"]), [
[vShow, showDecimalIcon(item)]
])
]),
_: 2
}, 1032, ["class"])
], 42, _hoisted_2);
}),
128
)),
_ctx.showText || _ctx.showScore ? (openBlock(), createElementBlock(
"span",
{
key: 0,
class: normalizeClass(unref(ns).e("text")),
style: normalizeStyle({ color: _ctx.textColor })
},
toDisplayString(text.value),
7
)) : createCommentVNode("v-if", true)
], 46, _hoisted_1);
};
}
});
var Rate = /* @__PURE__ */ _export_sfc(_sfc_main, [["__file", "/home/runner/work/element-plus/element-plus/packages/components/rate/src/rate.vue"]]);
export { Rate as default };
//# sourceMappingURL=rate.mjs.map