vxe-pc-ui
Version:
A vue based PC component library
585 lines (584 loc) • 18.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _vue = require("vue");
var _xeUtils = _interopRequireDefault(require("xe-utils"));
var _ui = require("../../ui");
var _dom = require("../../ui/src/dom");
var _utils = require("../../ui/src/utils");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var _default = exports.default = (0, _vue.defineComponent)({
name: 'VxeIconPicker',
props: {
modelValue: String,
placeholder: String,
clearable: Boolean,
size: {
type: String,
default: () => (0, _ui.getConfig)().iconPicker.size || (0, _ui.getConfig)().size
},
className: [String, Function],
popupClassName: [String, Function],
showIconTitle: {
type: Boolean,
default: () => (0, _ui.getConfig)().iconPicker.showIconTitle
},
readonly: {
type: Boolean,
default: null
},
disabled: {
type: Boolean,
default: null
},
icons: Array,
placement: String,
transfer: {
type: Boolean,
default: null
}
},
emits: ['update:modelValue', 'change', 'clear', 'click'],
setup(props, context) {
const {
emit
} = context;
const $xeModal = (0, _vue.inject)('$xeModal', null);
const $xeDrawer = (0, _vue.inject)('$xeDrawer', null);
const $xeTable = (0, _vue.inject)('$xeTable', null);
const $xeForm = (0, _vue.inject)('$xeForm', null);
const formItemInfo = (0, _vue.inject)('xeFormItemInfo', null);
const xID = _xeUtils.default.uniqueId();
const {
computeSize
} = (0, _ui.useSize)(props);
const reactData = (0, _vue.reactive)({
initialized: false,
selectIcon: `${props.modelValue || ''}`,
panelIndex: 0,
panelStyle: {},
panelPlacement: null,
visiblePanel: false,
isAniVisible: false,
isActivated: false
});
const internalData = {
// hpTimeout: undefined
};
const refElem = (0, _vue.ref)();
const refInput = (0, _vue.ref)();
const refOptionPanel = (0, _vue.ref)();
const refMaps = {
refElem
};
const $xeIconPicker = {
xID,
props,
context,
reactData,
getRefMaps: () => refMaps
};
let iconPickerMethods = {};
const computeFormReadonly = (0, _vue.computed)(() => {
const {
readonly
} = props;
if (readonly === null) {
if ($xeForm) {
return $xeForm.props.readonly;
}
return false;
}
return readonly;
});
const computeIsDisabled = (0, _vue.computed)(() => {
const {
disabled
} = props;
if (disabled === null) {
if ($xeForm) {
return $xeForm.props.disabled;
}
return false;
}
return disabled;
});
const computeBtnTransfer = (0, _vue.computed)(() => {
const {
transfer
} = props;
if (transfer === null) {
const globalTransfer = (0, _ui.getConfig)().iconPicker.transfer;
if (_xeUtils.default.isBoolean(globalTransfer)) {
return globalTransfer;
}
if ($xeTable || $xeModal || $xeDrawer || $xeForm) {
return true;
}
}
return transfer;
});
const computeInpPlaceholder = (0, _vue.computed)(() => {
const {
placeholder
} = props;
if (placeholder) {
return (0, _utils.getFuncText)(placeholder);
}
const globalPlaceholder = (0, _ui.getConfig)().select.placeholder;
if (globalPlaceholder) {
return (0, _utils.getFuncText)(globalPlaceholder);
}
return (0, _ui.getI18n)('vxe.base.pleaseSelect');
});
const computeIconList = (0, _vue.computed)(() => {
let {
icons
} = props;
if (!icons || !icons.length) {
icons = (0, _ui.getConfig)().iconPicker.icons || [];
}
return icons.map(item => {
if (_xeUtils.default.isString(item)) {
return {
title: item,
icon: `vxe-icon-${`${item || ''}`.replace(/^vxe-icon-/, '')}`
};
}
return {
title: `${item.title || ''}`,
icon: item.icon || ''
};
});
});
const computeIconGroupList = (0, _vue.computed)(() => {
const iconList = computeIconList.value;
return _xeUtils.default.chunk(iconList, 4);
});
const updateZindex = () => {
if (reactData.panelIndex < (0, _utils.getLastZIndex)()) {
reactData.panelIndex = (0, _utils.nextZIndex)();
}
};
const updatePlacement = () => {
return (0, _vue.nextTick)().then(() => {
const {
placement
} = props;
const {
panelIndex
} = reactData;
const el = refElem.value;
const panelElem = refOptionPanel.value;
const btnTransfer = computeBtnTransfer.value;
if (panelElem && el) {
const targetHeight = el.offsetHeight;
const targetWidth = el.offsetWidth;
const panelHeight = panelElem.offsetHeight;
const panelWidth = panelElem.offsetWidth;
const marginSize = 5;
const panelStyle = {
zIndex: panelIndex
};
const {
boundingTop,
boundingLeft,
visibleHeight,
visibleWidth
} = (0, _dom.getAbsolutePos)(el);
let panelPlacement = 'bottom';
if (btnTransfer) {
let left = boundingLeft;
let top = boundingTop + targetHeight;
if (placement === 'top') {
panelPlacement = 'top';
top = boundingTop - panelHeight;
} else if (!placement) {
// 如果下面不够放,则向上
if (top + panelHeight + marginSize > visibleHeight) {
panelPlacement = 'top';
top = boundingTop - panelHeight;
}
// 如果上面不够放,则向下(优先)
if (top < marginSize) {
panelPlacement = 'bottom';
top = boundingTop + targetHeight;
}
}
// 如果溢出右边
if (left + panelWidth + marginSize > visibleWidth) {
left -= left + panelWidth + marginSize - visibleWidth;
}
// 如果溢出左边
if (left < marginSize) {
left = marginSize;
}
Object.assign(panelStyle, {
left: `${left}px`,
top: `${top}px`,
minWidth: `${targetWidth}px`
});
} else {
if (placement === 'top') {
panelPlacement = 'top';
panelStyle.bottom = `${targetHeight}px`;
} else if (!placement) {
// 如果下面不够放,则向上
if (boundingTop + targetHeight + panelHeight > visibleHeight) {
// 如果上面不够放,则向下(优先)
if (boundingTop - targetHeight - panelHeight > marginSize) {
panelPlacement = 'top';
panelStyle.bottom = `${targetHeight}px`;
}
}
}
}
reactData.panelStyle = panelStyle;
reactData.panelPlacement = panelPlacement;
return (0, _vue.nextTick)();
}
});
};
const showOptionPanel = () => {
const {
hpTimeout
} = internalData;
const isDisabled = computeIsDisabled.value;
if (!isDisabled) {
if (hpTimeout) {
clearTimeout(hpTimeout);
internalData.hpTimeout = undefined;
}
if (!reactData.initialized) {
reactData.initialized = true;
}
reactData.isActivated = true;
reactData.isAniVisible = true;
setTimeout(() => {
reactData.visiblePanel = true;
}, 10);
updateZindex();
updatePlacement();
}
};
const hideOptionPanel = () => {
reactData.visiblePanel = false;
internalData.hpTimeout = setTimeout(() => {
reactData.isAniVisible = false;
}, 350);
};
const changeEvent = (evnt, selectValue) => {
reactData.selectIcon = selectValue;
if (selectValue !== props.modelValue) {
emit('update:modelValue', selectValue);
dispatchEvent('change', {
value: selectValue
}, evnt);
// 自动更新校验状态
if ($xeForm && formItemInfo) {
$xeForm.triggerItemEvent(evnt, formItemInfo.itemConfig.field, selectValue);
}
}
};
const focusEvent = () => {
const isDisabled = computeIsDisabled.value;
if (!isDisabled) {
if (!reactData.visiblePanel) {
showOptionPanel();
}
}
};
const blurEvent = () => {
reactData.isActivated = false;
};
const clearValueEvent = (evnt, selectValue) => {
changeEvent(evnt, selectValue);
dispatchEvent('clear', {
value: selectValue
}, evnt);
};
const clearEvent = (params, evnt) => {
clearValueEvent(evnt, null);
hideOptionPanel();
};
const togglePanelEvent = evnt => {
evnt.preventDefault();
if (reactData.visiblePanel) {
hideOptionPanel();
} else {
showOptionPanel();
}
};
const clickEvent = evnt => {
togglePanelEvent(evnt);
dispatchEvent('click', {}, evnt);
};
const handleGlobalMousewheelEvent = evnt => {
const {
visiblePanel
} = reactData;
const isDisabled = computeIsDisabled.value;
if (!isDisabled) {
if (visiblePanel) {
const panelElem = refOptionPanel.value;
if ((0, _dom.getEventTargetNode)(evnt, panelElem).flag) {
updatePlacement();
} else {
hideOptionPanel();
}
}
}
};
const handleGlobalMousedownEvent = evnt => {
const {
visiblePanel
} = reactData;
const isDisabled = computeIsDisabled.value;
if (!isDisabled) {
const el = refElem.value;
const panelElem = refOptionPanel.value;
reactData.isActivated = (0, _dom.getEventTargetNode)(evnt, el).flag || (0, _dom.getEventTargetNode)(evnt, panelElem).flag;
if (visiblePanel && !reactData.isActivated) {
hideOptionPanel();
}
}
};
const handleGlobalKeydownEvent = evnt => {
const {
clearable
} = props;
const {
visiblePanel
} = reactData;
const isDisabled = computeIsDisabled.value;
if (!isDisabled) {
const isTab = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.TAB);
const isEnter = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.ENTER);
const isEsc = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.ESCAPE);
const isUpArrow = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.ARROW_UP);
const isDwArrow = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.ARROW_DOWN);
const isDel = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.DELETE);
const isSpacebar = _ui.globalEvents.hasKey(evnt, _ui.GLOBAL_EVENT_KEYS.SPACEBAR);
if (isTab) {
reactData.isActivated = false;
}
if (visiblePanel) {
if (isEsc || isTab) {
hideOptionPanel();
} else if (isEnter) {
evnt.preventDefault();
evnt.stopPropagation();
// changeOptionEvent(evnt, currentValue, currentOption)
} else if (isUpArrow || isDwArrow) {
evnt.preventDefault();
// let { firstOption, offsetOption } = findOffsetOption(currentValue, isUpArrow)
// if (!offsetOption && !findVisibleOption(currentValue)) {
// offsetOption = firstOption
// }
// setCurrentOption(offsetOption)
// scrollToOption(offsetOption, isDwArrow)
} else if (isSpacebar) {
evnt.preventDefault();
}
} else if ((isUpArrow || isDwArrow || isEnter || isSpacebar) && reactData.isActivated) {
evnt.preventDefault();
showOptionPanel();
}
if (reactData.isActivated) {
if (isDel && clearable) {
clearValueEvent(evnt, null);
}
}
}
};
const handleGlobalBlurEvent = () => {
hideOptionPanel();
};
const dispatchEvent = (type, params, evnt) => {
emit(type, (0, _ui.createEvent)(evnt, {
$iconPicker: $xeIconPicker
}, params));
};
iconPickerMethods = {
dispatchEvent,
isPanelVisible() {
return reactData.visiblePanel;
},
togglePanel() {
if (reactData.visiblePanel) {
hideOptionPanel();
} else {
showOptionPanel();
}
return (0, _vue.nextTick)();
},
hidePanel() {
if (reactData.visiblePanel) {
hideOptionPanel();
}
return (0, _vue.nextTick)();
},
showPanel() {
if (!reactData.visiblePanel) {
showOptionPanel();
}
return (0, _vue.nextTick)();
},
focus() {
const $input = refInput.value;
reactData.isActivated = true;
if ($input) {
$input.blur();
}
return (0, _vue.nextTick)();
},
blur() {
const $input = refInput.value;
if ($input) {
$input.blur();
}
reactData.isActivated = false;
return (0, _vue.nextTick)();
}
};
const handleClickIconEvent = (evnt, item) => {
const value = item.icon;
changeEvent(evnt, value);
hideOptionPanel();
};
Object.assign($xeIconPicker, iconPickerMethods);
const renderIconWrapper = () => {
const {
showIconTitle
} = props;
const iconGroupList = computeIconGroupList.value;
const isDisabled = computeIsDisabled.value;
return (0, _vue.h)('div', {
class: 'vxe-ico-picker--list-wrapper'
}, iconGroupList.map(list => {
return (0, _vue.h)('div', {
class: 'vxe-ico-picker--list'
}, list.map(item => {
return (0, _vue.h)('div', {
class: 'vxe-ico-picker--item',
onClick(evnt) {
if (!isDisabled) {
handleClickIconEvent(evnt, item);
}
}
}, [(0, _vue.h)('div', {
class: 'vxe-ico-picker--item-icon'
}, [(0, _vue.h)('i', {
class: item.icon || ''
})]), showIconTitle ? (0, _vue.h)('div', {
class: 'vxe-ico-picker--item-title'
}, `${item.title || ''}`) : (0, _vue.createCommentVNode)()]);
}));
}));
};
const renderVN = () => {
const {
className,
popupClassName,
clearable
} = props;
const {
initialized,
isActivated,
isAniVisible,
visiblePanel,
selectIcon
} = reactData;
const vSize = computeSize.value;
const isDisabled = computeIsDisabled.value;
const btnTransfer = computeBtnTransfer.value;
const formReadonly = computeFormReadonly.value;
const inpPlaceholder = computeInpPlaceholder.value;
if (formReadonly) {
return (0, _vue.h)('div', {
ref: refElem,
class: ['vxe-ico-picker--readonly', className]
}, [(0, _vue.h)('i', {
class: selectIcon
})]);
}
return (0, _vue.h)('div', {
ref: refElem,
class: ['vxe-ico-picker', className ? _xeUtils.default.isFunction(className) ? className({
$iconPicker: $xeIconPicker
}) : className : '', {
[`size--${vSize}`]: vSize,
'show--clear': clearable && !isDisabled && !!selectIcon,
'is--visible': visiblePanel,
'is--disabled': isDisabled,
'is--active': isActivated
}]
}, [(0, _vue.h)('div', {
class: 'vxe-ico-picker--inner',
onClick: clickEvent
}, [(0, _vue.h)('input', {
ref: refInput,
class: 'vxe-ico-picker--input',
onFocus: focusEvent,
onBlur: blurEvent
}), selectIcon ? (0, _vue.h)('div', {
class: 'vxe-ico-picker--icon'
}, [(0, _vue.h)('i', {
class: selectIcon
})]) : (0, _vue.h)('div', {
class: 'vxe-ico-picker--placeholder'
}, inpPlaceholder), (0, _vue.h)('div', {
class: 'vxe-ico-picker--suffix'
}, [(0, _vue.h)('div', {
class: 'vxe-ico-picker--clear-icon',
onClick: clearEvent
}, [(0, _vue.h)('i', {
class: (0, _ui.getIcon)().INPUT_CLEAR
})]), (0, _vue.h)('div', {
class: 'vxe-ico-picker--suffix-icon'
}, [(0, _vue.h)('i', {
class: visiblePanel ? (0, _ui.getIcon)().ICON_PICKER_OPEN : (0, _ui.getIcon)().ICON_PICKER_CLOSE
})])])]), (0, _vue.h)(_vue.Teleport, {
to: 'body',
disabled: btnTransfer ? !initialized : true
}, [(0, _vue.h)('div', {
ref: refOptionPanel,
class: ['vxe-table--ignore-clear vxe-ico-picker--panel', popupClassName ? _xeUtils.default.isFunction(popupClassName) ? popupClassName({
$iconPicker: $xeIconPicker
}) : popupClassName : '', {
[`size--${vSize}`]: vSize,
'is--transfer': btnTransfer,
'ani--leave': isAniVisible,
'ani--enter': visiblePanel
}],
placement: reactData.panelPlacement,
style: reactData.panelStyle
}, [initialized && (visiblePanel || isAniVisible) ? (0, _vue.h)('div', {
class: 'vxe-ico-picker--panel-wrapper'
}, [renderIconWrapper()]) : (0, _vue.createCommentVNode)()])])]);
};
(0, _vue.watch)(() => props.modelValue, val => {
reactData.selectIcon = `${val || ''}`;
});
(0, _vue.onMounted)(() => {
_ui.globalEvents.on($xeIconPicker, 'mousewheel', handleGlobalMousewheelEvent);
_ui.globalEvents.on($xeIconPicker, 'mousedown', handleGlobalMousedownEvent);
_ui.globalEvents.on($xeIconPicker, 'keydown', handleGlobalKeydownEvent);
_ui.globalEvents.on($xeIconPicker, 'blur', handleGlobalBlurEvent);
});
(0, _vue.onUnmounted)(() => {
_ui.globalEvents.off($xeIconPicker, 'mousewheel');
_ui.globalEvents.off($xeIconPicker, 'mousedown');
_ui.globalEvents.off($xeIconPicker, 'keydown');
_ui.globalEvents.off($xeIconPicker, 'blur');
});
(0, _vue.provide)('$xeIconPicker', $xeIconPicker);
$xeIconPicker.renderVN = renderVN;
return $xeIconPicker;
},
render() {
return this.renderVN();
}
});