UNPKG

element-plus

Version:

A Component Library for Vue3.0

594 lines (581 loc) 24.5 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); var vue = require('vue'); var ElCascaderPanel = require('../el-cascader-panel'); var ElInput = require('../el-input'); var ElPopper = require('../el-popper'); var ElScrollbar = require('../el-scrollbar'); var ElTag = require('../el-tag'); var directives = require('../directives'); var locale = require('../locale'); var debounce = require('lodash/debounce'); var aria = require('../utils/aria'); var constants = require('../utils/constants'); var isServer = require('../utils/isServer'); var util = require('../utils/util'); var resizeEvent = require('../utils/resize-event'); var validators = require('../utils/validators'); var form = require('../el-form'); function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } var ElCascaderPanel__default = /*#__PURE__*/_interopDefaultLegacy(ElCascaderPanel); var ElInput__default = /*#__PURE__*/_interopDefaultLegacy(ElInput); var ElPopper__default = /*#__PURE__*/_interopDefaultLegacy(ElPopper); var ElScrollbar__default = /*#__PURE__*/_interopDefaultLegacy(ElScrollbar); var ElTag__default = /*#__PURE__*/_interopDefaultLegacy(ElTag); var debounce__default = /*#__PURE__*/_interopDefaultLegacy(debounce); var isServer__default = /*#__PURE__*/_interopDefaultLegacy(isServer); /** * 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 isFunction = (val) => typeof val === 'function'; const isObject = (val) => val !== null && typeof val === 'object'; const isPromise = (val) => { return isObject(val) && isFunction(val.then) && isFunction(val.catch); }; const DEFAULT_INPUT_HEIGHT = 40; const INPUT_HEIGHT_MAP = { medium: 36, small: 32, mini: 28, }; const popperOptions = { modifiers: [ { name: 'arrowPosition', enabled: true, phase: 'main', fn: ({ state }) => { const { modifiersData, elements } = state; const { reference, arrow } = elements; modifiersData.arrow.x = modifiersData.arrow.x - (reference.clientWidth - arrow.clientWidth) / 2 + 35; }, requires: ['arrow'], }, ], }; var script = vue.defineComponent({ name: 'ElCascader', components: { ElCascaderPanel: ElCascaderPanel__default['default'], ElInput: ElInput__default['default'], ElPopper: ElPopper__default['default'], ElScrollbar: ElScrollbar__default['default'], ElTag: ElTag__default['default'], }, directives: { Clickoutside: directives.ClickOutside, }, props: Object.assign(Object.assign({}, ElCascaderPanel.CommonProps), { size: { type: String, validator: validators.isValidComponentSize, }, placeholder: { type: String, default: () => locale.t('el.cascader.placeholder'), }, disabled: Boolean, clearable: Boolean, filterable: Boolean, filterMethod: { type: Function, default: (node, keyword) => node.text.includes(keyword), }, separator: { type: String, default: ' / ', }, showAllLevels: { type: Boolean, default: true, }, collapseTags: Boolean, debounce: { type: Number, default: 300, }, beforeFilter: { type: Function, default: () => true, }, popperClass: { type: String, default: '', } }), emits: [ constants.UPDATE_MODEL_EVENT, constants.CHANGE_EVENT, 'focus', 'blur', 'visible-change', 'expand-change', 'remove-tag', ], setup(props, { emit }) { let inputInitialHeight = 0; let pressDeleteCount = 0; const $ELEMENT = util.useGlobalConfig(); const elForm = vue.inject(form.elFormKey, {}); const elFormItem = vue.inject(form.elFormItemKey, {}); const popper = vue.ref(null); const input = vue.ref(null); const tagWrapper = vue.ref(null); const panel = vue.ref(null); const suggestionPanel = vue.ref(null); const popperVisible = vue.ref(false); const inputHover = vue.ref(false); const filtering = vue.ref(false); const inputValue = vue.ref(''); const searchInputValue = vue.ref(''); const presentTags = vue.ref([]); const suggestions = vue.ref([]); const isDisabled = vue.computed(() => props.disabled || elForm.disabled); const realSize = vue.computed(() => props.size || elFormItem.size || $ELEMENT.size); const tagSize = vue.computed(() => ['small', 'mini'].includes(realSize.value) ? 'mini' : 'small'); const multiple = vue.computed(() => !!props.props.multiple); const readonly = vue.computed(() => !props.filterable || multiple.value); const searchKeyword = vue.computed(() => multiple.value ? searchInputValue.value : inputValue.value); const checkedNodes = vue.computed(() => { var _a; return ((_a = panel.value) === null || _a === void 0 ? void 0 : _a.checkedNodes) || []; }); const clearBtnVisible = vue.computed(() => { if (!props.clearable || isDisabled.value || filtering.value || !inputHover.value) return false; return !!checkedNodes.value.length; }); const presentText = vue.computed(() => { const { showAllLevels, separator } = props; const nodes = checkedNodes.value; return nodes.length ? multiple.value ? ' ' : nodes[0].calcText(showAllLevels, separator) : ''; }); const checkedValue = vue.computed({ get() { return props.modelValue; }, set(val) { var _a; emit(constants.UPDATE_MODEL_EVENT, val); emit(constants.CHANGE_EVENT, val); (_a = elFormItem.formItemMitt) === null || _a === void 0 ? void 0 : _a.emit('el.form.change', [val]); }, }); const popperPaneRef = vue.computed(() => { var _a; return (_a = popper.value) === null || _a === void 0 ? void 0 : _a.popperRef; }); const togglePopperVisible = (visible) => { if (isDisabled.value) return; visible = visible !== null && visible !== void 0 ? visible : !popperVisible.value; if (visible !== popperVisible.value) { popperVisible.value = visible; input.value.input.setAttribute('aria-expanded', visible); if (visible) { updatePopperPosition(); vue.nextTick(panel.value.scrollToExpandingNode); } else if (props.filterable) { const { value } = presentText; inputValue.value = value; searchInputValue.value = value; } emit('visible-change', visible); } }; const updatePopperPosition = () => { vue.nextTick(popper.value.update); }; const hideSuggestionPanel = () => { filtering.value = false; }; const genTag = (node) => { const { showAllLevels, separator } = props; return { node, key: node.uid, text: node.calcText(showAllLevels, separator), hitState: false, closable: !isDisabled.value && !node.isDisabled, }; }; const deleteTag = (tag) => { const { node } = tag; node.doCheck(false); panel.value.calculateCheckedValue(); emit('remove-tag', node.valueByOption); }; const calculatePresentTags = () => { if (!multiple.value) return; const nodes = checkedNodes.value; const tags = []; if (nodes.length) { const [first, ...rest] = nodes; const restCount = rest.length; tags.push(genTag(first)); if (restCount) { if (props.collapseTags) { tags.push({ key: -1, text: `+ ${restCount}`, closable: false, }); } else { rest.forEach(node => tags.push(genTag(node))); } } } presentTags.value = tags; }; const calculateSuggestions = () => { const { filterMethod, showAllLevels, separator } = props; const res = panel.value.getFlattedNodes(!props.props.checkStrictly) .filter(node => { if (node.isDisabled) return false; node.calcText(showAllLevels, separator); return filterMethod(node, searchKeyword.value); }); if (multiple.value) { presentTags.value.forEach(tag => { tag.hitState = false; }); } filtering.value = true; suggestions.value = res; updatePopperPosition(); }; const focusFirstNode = () => { var _a; let firstNode = null; if (filtering.value && suggestionPanel.value) { firstNode = suggestionPanel.value.$el.querySelector('.el-cascader__suggestion-item'); } else { firstNode = (_a = panel.value) === null || _a === void 0 ? void 0 : _a.$el.querySelector('.el-cascader-node[tabindex="-1"]'); } if (firstNode) { firstNode.focus(); !filtering.value && firstNode.click(); } }; const updateStyle = () => { var _a; const inputInner = input.value.input; const tagWrapperEl = tagWrapper.value; const suggestionPanelEl = (_a = suggestionPanel.value) === null || _a === void 0 ? void 0 : _a.$el; if (isServer__default['default'] || !inputInner) return; if (suggestionPanelEl) { const suggestionList = suggestionPanelEl.querySelector('.el-cascader__suggestion-list'); suggestionList.style.minWidth = inputInner.offsetWidth + 'px'; } if (tagWrapperEl) { const { offsetHeight } = tagWrapperEl; const height = Math.max(offsetHeight + 6, inputInitialHeight) + 'px'; inputInner.style.height = height; updatePopperPosition(); } }; const getCheckedNodes = (leafOnly) => { return panel.value.getCheckedNodes(leafOnly); }; const handleExpandChange = (value) => { updatePopperPosition(); emit('expand-change', value); }; const handleKeyDown = (e) => { switch (e.code) { case aria.EVENT_CODE.enter: togglePopperVisible(); break; case aria.EVENT_CODE.down: togglePopperVisible(true); vue.nextTick(focusFirstNode); event.preventDefault(); break; case aria.EVENT_CODE.esc: case aria.EVENT_CODE.tab: togglePopperVisible(false); break; } }; const handleClear = () => { panel.value.clearCheckedNodes(); togglePopperVisible(false); }; const handleSuggestionClick = (node) => { const { checked } = node; if (multiple.value) { panel.value.handleCheckChange(node, !checked, false); } else { !checked && panel.value.handleCheckChange(node, true, false); togglePopperVisible(false); } }; const handleDelete = () => { const tags = presentTags.value; const lastTag = tags[tags.length - 1]; pressDeleteCount = searchInputValue.value ? 0 : pressDeleteCount + 1; if (!lastTag || !pressDeleteCount) return; if (lastTag.hitState) { deleteTag(lastTag); } else { lastTag.hitState = true; } }; const handleFilter = debounce__default['default'](() => { const { value } = searchKeyword; if (!value) return; const passed = props.beforeFilter(value); if (isPromise(passed)) { passed.then(calculateSuggestions); } else if (passed !== false) { calculateSuggestions(); } else { hideSuggestionPanel(); } }, props.debounce); const handleInput = (val, e) => { !popperVisible.value && togglePopperVisible(true); if (e === null || e === void 0 ? void 0 : e.isComposing) return; val ? handleFilter() : hideSuggestionPanel(); }; vue.watch(filtering, updatePopperPosition); vue.watch([checkedNodes, isDisabled], calculatePresentTags); vue.watch(presentTags, () => vue.nextTick(updateStyle)); vue.watch(presentText, val => inputValue.value = val, { immediate: true }); vue.onMounted(() => { const inputEl = input.value.$el; inputInitialHeight = (inputEl === null || inputEl === void 0 ? void 0 : inputEl.offsetHeight) || INPUT_HEIGHT_MAP[realSize.value] || DEFAULT_INPUT_HEIGHT; resizeEvent.addResizeListener(inputEl, updateStyle); }); vue.onBeforeUnmount(() => { resizeEvent.removeResizeListener(input.value.$el, updateStyle); }); return { popperOptions, popper, popperPaneRef, input, tagWrapper, panel, suggestionPanel, popperVisible, inputHover, filtering, presentText, checkedValue, inputValue, searchInputValue, presentTags, suggestions, isDisabled, realSize, tagSize, multiple, readonly, clearBtnVisible, t: locale.t, togglePopperVisible, hideSuggestionPanel, deleteTag, focusFirstNode, getCheckedNodes, handleExpandChange, handleKeyDown, handleClear, handleSuggestionClick, handleDelete, handleInput, }; }, }); const _hoisted_1 = { key: 0, ref: "tagWrapper", class: "el-cascader__tags" }; const _hoisted_2 = { key: 0, class: "el-icon-check" }; const _hoisted_3 = { class: "el-cascader__empty-text" }; function render(_ctx, _cache, $props, $setup, $data, $options) { const _component_el_input = vue.resolveComponent("el-input"); const _component_el_tag = vue.resolveComponent("el-tag"); const _component_el_cascader_panel = vue.resolveComponent("el-cascader-panel"); const _component_el_scrollbar = vue.resolveComponent("el-scrollbar"); const _component_el_popper = vue.resolveComponent("el-popper"); const _directive_clickoutside = vue.resolveDirective("clickoutside"); return (vue.openBlock(), vue.createBlock(_component_el_popper, { ref: "popper", visible: _ctx.popperVisible, "onUpdate:visible": _cache[16] || (_cache[16] = $event => (_ctx.popperVisible = $event)), "manual-mode": "", placement: "bottom-start", "popper-class": `el-cascader__dropdown ${_ctx.popperClass}`, "popper-options": _ctx.popperOptions, "stop-popper-mouse-event": false, transition: "el-zoom-in-top", "gpu-acceleration": false, effect: "light", pure: "", onAfterLeave: _ctx.hideSuggestionPanel }, { trigger: vue.withCtx(() => [ vue.withDirectives(vue.createVNode("div", { class: [ 'el-cascader', _ctx.realSize && `el-cascader--${_ctx.realSize}`, { 'is-disabled': _ctx.isDisabled } ], onClick: _cache[10] || (_cache[10] = () => _ctx.togglePopperVisible(_ctx.readonly ? undefined : true)), onKeydown: _cache[11] || (_cache[11] = (...args) => (_ctx.handleKeyDown && _ctx.handleKeyDown(...args))), onMouseenter: _cache[12] || (_cache[12] = $event => (_ctx.inputHover = true)), onMouseleave: _cache[13] || (_cache[13] = $event => (_ctx.inputHover = false)) }, [ vue.createVNode(_component_el_input, { ref: "input", modelValue: _ctx.inputValue, "onUpdate:modelValue": _cache[3] || (_cache[3] = $event => (_ctx.inputValue = $event)), modelModifiers: { trim: true }, placeholder: _ctx.placeholder, readonly: _ctx.readonly, disabled: _ctx.isDisabled, "validate-event": false, size: _ctx.realSize, class: { 'is-focus': _ctx.popperVisible }, onFocus: _cache[4] || (_cache[4] = e => _ctx.$emit('focus', e)), onBlur: _cache[5] || (_cache[5] = e => _ctx.$emit('blur', e)), onInput: _ctx.handleInput }, { suffix: vue.withCtx(() => [ (_ctx.clearBtnVisible) ? (vue.openBlock(), vue.createBlock("i", { key: "clear", class: "el-input__icon el-icon-circle-close", onClick: _cache[1] || (_cache[1] = vue.withModifiers((...args) => (_ctx.handleClear && _ctx.handleClear(...args)), ["stop"])) })) : (vue.openBlock(), vue.createBlock("i", { key: "arrow-down", class: [ 'el-input__icon', 'el-icon-arrow-down', _ctx.popperVisible && 'is-reverse' ], onClick: _cache[2] || (_cache[2] = vue.withModifiers($event => (_ctx.togglePopperVisible()), ["stop"])) }, null, 2 /* CLASS */)) ]), _: 1 /* STABLE */ }, 8 /* PROPS */, ["modelValue", "placeholder", "readonly", "disabled", "size", "class", "onInput"]), (_ctx.multiple) ? (vue.openBlock(), vue.createBlock("div", _hoisted_1, [ (vue.openBlock(true), vue.createBlock(vue.Fragment, null, vue.renderList(_ctx.presentTags, (tag) => { return (vue.openBlock(), vue.createBlock(_component_el_tag, { key: tag.key, type: "info", size: _ctx.tagSize, hit: tag.hitState, closable: tag.closable, "disable-transitions": "", onClose: $event => (_ctx.deleteTag(tag)) }, { default: vue.withCtx(() => [ vue.createVNode("span", null, vue.toDisplayString(tag.text), 1 /* TEXT */) ]), _: 2 /* DYNAMIC */ }, 1032 /* PROPS, DYNAMIC_SLOTS */, ["size", "hit", "closable", "onClose"])) }), 128 /* KEYED_FRAGMENT */)), (_ctx.filterable && !_ctx.isDisabled) ? vue.withDirectives((vue.openBlock(), vue.createBlock("input", { key: 0, "onUpdate:modelValue": _cache[6] || (_cache[6] = $event => (_ctx.searchInputValue = $event)), type: "text", class: "el-cascader__search-input", placeholder: _ctx.presentText ? '' : _ctx.placeholder, onInput: _cache[7] || (_cache[7] = e => _ctx.handleInput(_ctx.searchInputValue, e)), onClick: _cache[8] || (_cache[8] = vue.withModifiers($event => (_ctx.togglePopperVisible(true)), ["stop"])), onKeydown: _cache[9] || (_cache[9] = vue.withKeys((...args) => (_ctx.handleDelete && _ctx.handleDelete(...args)), ["delete"])) }, null, 40 /* PROPS, HYDRATE_EVENTS */, ["placeholder"])), [ [ vue.vModelText, _ctx.searchInputValue, void 0, { trim: true } ] ]) : vue.createCommentVNode("v-if", true) ], 512 /* NEED_PATCH */)) : vue.createCommentVNode("v-if", true) ], 34 /* CLASS, HYDRATE_EVENTS */), [ [_directive_clickoutside, () => _ctx.togglePopperVisible(false), _ctx.popperPaneRef] ]) ]), default: vue.withCtx(() => [ vue.withDirectives(vue.createVNode(_component_el_cascader_panel, { ref: "panel", modelValue: _ctx.checkedValue, "onUpdate:modelValue": _cache[14] || (_cache[14] = $event => (_ctx.checkedValue = $event)), options: _ctx.options, props: _ctx.props, border: false, "render-label": _ctx.$slots.default, onExpandChange: _ctx.handleExpandChange, onClose: _cache[15] || (_cache[15] = $event => (_ctx.togglePopperVisible(false))) }, null, 8 /* PROPS */, ["modelValue", "options", "props", "render-label", "onExpandChange"]), [ [vue.vShow, !_ctx.filtering] ]), (_ctx.filterable) ? vue.withDirectives((vue.openBlock(), vue.createBlock(_component_el_scrollbar, { key: 0, ref: "suggestionPanel", tag: "ul", class: "el-cascader__suggestion-panel", "view-class": "el-cascader__suggestion-list" }, { default: vue.withCtx(() => [ (_ctx.suggestions.length) ? (vue.openBlock(true), vue.createBlock(vue.Fragment, { key: 0 }, vue.renderList(_ctx.suggestions, (item) => { return (vue.openBlock(), vue.createBlock("li", { key: item.uid, class: [ 'el-cascader__suggestion-item', item.checked && 'is-checked' ], tabindex: -1, onClick: $event => (_ctx.handleSuggestionClick(item)) }, [ vue.createVNode("span", null, vue.toDisplayString(item.text), 1 /* TEXT */), (item.checked) ? (vue.openBlock(), vue.createBlock("i", _hoisted_2)) : vue.createCommentVNode("v-if", true) ], 10 /* CLASS, PROPS */, ["onClick"])) }), 128 /* KEYED_FRAGMENT */)) : vue.renderSlot(_ctx.$slots, "empty", { key: 1 }, () => [ vue.createVNode("li", _hoisted_3, vue.toDisplayString(_ctx.t('el.cascader.noMatch')), 1 /* TEXT */) ]) ]), _: 1 /* STABLE */ }, 512 /* NEED_PATCH */)), [ [vue.vShow, _ctx.filtering] ]) : vue.createCommentVNode("v-if", true) ]), _: 1 /* STABLE */ }, 8 /* PROPS */, ["visible", "popper-class", "popper-options", "onAfterLeave"])) } script.render = render; script.__file = "packages/cascader/src/index.vue"; script.install = (app) => { app.component(script.name, script); }; const _Cascader = script; exports.default = _Cascader;