vft 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +54 -0
- package/src/app/index.ts +3 -0
- package/src/app/page-loading/assets/spin.gif +0 -0
- package/src/app/page-loading/index.less +10 -0
- package/src/app/page-loading/index.ts +3 -0
- package/src/app/page-loading/index.vue +38 -0
- package/src/app/table/assets/sort_triangle.png +0 -0
- package/src/app/table/demos/basic.vue +117 -0
- package/src/app/table/demos/complex.vue +2543 -0
- package/src/app/table/demos/index.vue +453 -0
- package/src/app/table/demos/rightAngle.png +0 -0
- package/src/app/table/header.vue +203 -0
- package/src/app/table/index.less +119 -0
- package/src/app/table/index.ts +5 -0
- package/src/app/table/index.vue +478 -0
- package/src/app/table/md/api.md +23 -0
- package/src/app/table/md/demo.md +3 -0
- package/src/app/table/types.ts +45 -0
- package/src/common/badge/index.scss +78 -0
- package/src/common/badge/index.ts +4 -0
- package/src/common/badge/index.vue +111 -0
- package/src/common/badge/types.ts +1 -0
- package/src/common/clamp/clamp-toggle.vue +91 -0
- package/src/common/clamp/index.ts +4 -0
- package/src/common/clamp/index.vue +247 -0
- package/src/common/code/index.less +321 -0
- package/src/common/code/index.ts +3 -0
- package/src/common/code/index.vue +60 -0
- package/src/common/config-provider/index.ts +4 -0
- package/src/common/config-provider/index.vue +94 -0
- package/src/common/config-provider/types.ts +29 -0
- package/src/common/icon/iconfont/iconfont.css +334 -0
- package/src/common/icon/index.scss +31 -0
- package/src/common/icon/index.ts +4 -0
- package/src/common/icon/index.vue +74 -0
- package/src/common/icon/types.ts +16 -0
- package/src/common/index.ts +6 -0
- package/src/common/message/index.ts +5 -0
- package/src/common/message/instance.ts +29 -0
- package/src/common/message/message.scss +108 -0
- package/src/common/message/message.ts +80 -0
- package/src/common/message/message.vue +162 -0
- package/src/common/message/method.ts +172 -0
- package/src/common/overlay/index.scss +14 -0
- package/src/common/overlay/index.ts +3 -0
- package/src/common/overlay/index.vue +117 -0
- package/src/constants/comp.ts +1 -0
- package/src/constants/index.ts +1 -0
- package/src/index.ts +3 -0
- package/src/page/index.ts +1 -0
- package/src/page/page-wrapper/index.scss +31 -0
- package/src/page/page-wrapper/index.ts +5 -0
- package/src/page/page-wrapper/index.vue +31 -0
- package/src/styles/vars.scss +500 -0
- package/src/use/index.ts +14 -0
- package/src/use/onPopupReopen.ts +15 -0
- package/src/use/use-delayed-toggle/index.ts +30 -0
- package/src/use/use-floating/index.ts +121 -0
- package/src/use/use-forward-ref.ts +35 -0
- package/src/use/use-global-config/index.ts +81 -0
- package/src/use/use-id/index.ts +43 -0
- package/src/use/use-model-toggle/index.ts +151 -0
- package/src/use/use-namespace/index.ts +91 -0
- package/src/use/use-ordered-children/index.ts +43 -0
- package/src/use/use-popper-container/index.ts +43 -0
- package/src/use/use-timeout/index.ts +18 -0
- package/src/use/use-z-index/index.ts +20 -0
- package/src/use/useGlobalZIndex.ts +24 -0
- package/src/use/useLazyRender.ts +17 -0
- package/src/use/useLockScroll.ts +66 -0
- package/src/utils/aria.ts +126 -0
- package/src/utils/arrays.ts +13 -0
- package/src/utils/error.ts +23 -0
- package/src/utils/event.ts +15 -0
- package/src/utils/helper.ts +7 -0
- package/src/utils/index.ts +8 -0
- package/src/utils/interceptor.ts +39 -0
- package/src/utils/mount-component.ts +65 -0
- package/src/utils/popper.ts +6 -0
- package/src/utils/rand.ts +12 -0
- package/src/utils/scroll.ts +101 -0
- package/src/utils/vnode.ts +169 -0
- package/src/web/avatar/index.scss +50 -0
- package/src/web/avatar/index.ts +3 -0
- package/src/web/avatar/index.vue +83 -0
- package/src/web/back-top/index.scss +35 -0
- package/src/web/back-top/index.ts +3 -0
- package/src/web/back-top/index.vue +72 -0
- package/src/web/back-top/types.ts +14 -0
- package/src/web/back-top/use-back-top.ts +65 -0
- package/src/web/cascader/index.scss +214 -0
- package/src/web/cascader/index.ts +5 -0
- package/src/web/cascader/index.vue +767 -0
- package/src/web/cascader-panel/config.ts +44 -0
- package/src/web/cascader-panel/index.scss +134 -0
- package/src/web/cascader-panel/index.ts +5 -0
- package/src/web/cascader-panel/index.vue +319 -0
- package/src/web/cascader-panel/menu.vue +135 -0
- package/src/web/cascader-panel/node-content.ts +23 -0
- package/src/web/cascader-panel/node.ts +218 -0
- package/src/web/cascader-panel/node.vue +197 -0
- package/src/web/cascader-panel/store.ts +83 -0
- package/src/web/cascader-panel/types.ts +56 -0
- package/src/web/cascader-panel/utils.ts +40 -0
- package/src/web/cascader-select/README.md +31 -0
- package/src/web/cascader-select/index.scss +54 -0
- package/src/web/cascader-select/index.ts +5 -0
- package/src/web/cascader-select/index.vue +132 -0
- package/src/web/cascader-select/types.ts +9 -0
- package/src/web/context-menu/createContextMenu.ts +72 -0
- package/src/web/context-menu/index.ts +4 -0
- package/src/web/context-menu/index.vue +89 -0
- package/src/web/context-menu/types.ts +27 -0
- package/src/web/context-menu/useContextMenu.ts +14 -0
- package/src/web/descriptions/description-item.vue +34 -0
- package/src/web/descriptions/description.vue +124 -0
- package/src/web/descriptions/descriptions-cell.ts +95 -0
- package/src/web/descriptions/descriptions-item.scss +68 -0
- package/src/web/descriptions/descriptions-row.vue +49 -0
- package/src/web/descriptions/descriptions.scss +153 -0
- package/src/web/descriptions/descriptions.type.ts +19 -0
- package/src/web/descriptions/index.ts +4 -0
- package/src/web/descriptions/token.ts +4 -0
- package/src/web/divider/index.scss +53 -0
- package/src/web/divider/index.ts +5 -0
- package/src/web/divider/index.vue +60 -0
- package/src/web/divider/types.ts +2 -0
- package/src/web/empty/assets/no-collect.png +0 -0
- package/src/web/empty/assets/no-data.png +0 -0
- package/src/web/empty/assets/no-filter.png +0 -0
- package/src/web/empty/assets/no-page-data.png +0 -0
- package/src/web/empty/assets/no-search.png +0 -0
- package/src/web/empty/constants.ts +12 -0
- package/src/web/empty/index.scss +57 -0
- package/src/web/empty/index.ts +5 -0
- package/src/web/empty/index.vue +96 -0
- package/src/web/exception/exception.png +0 -0
- package/src/web/exception/index.ts +3 -0
- package/src/web/exception/index.vue +44 -0
- package/src/web/filter/README.md +25 -0
- package/src/web/filter/index.scss +14 -0
- package/src/web/filter/index.ts +5 -0
- package/src/web/filter/index.vue +60 -0
- package/src/web/filter/type.ts +13 -0
- package/src/web/focus-trap/index.ts +6 -0
- package/src/web/focus-trap/index.vue +328 -0
- package/src/web/focus-trap/tokens.ts +23 -0
- package/src/web/focus-trap/utils.ts +178 -0
- package/src/web/full-screen/index.scss +22 -0
- package/src/web/full-screen/index.ts +3 -0
- package/src/web/full-screen/index.vue +24 -0
- package/src/web/icon-text/index.ts +3 -0
- package/src/web/icon-text/index.vue +77 -0
- package/src/web/image/index.scss +46 -0
- package/src/web/image/index.ts +5 -0
- package/src/web/image/index.vue +251 -0
- package/src/web/image/types.ts +1 -0
- package/src/web/index.ts +33 -0
- package/src/web/input/index.scss +473 -0
- package/src/web/input/index.ts +3 -0
- package/src/web/input/index.vue +533 -0
- package/src/web/input/utils.ts +102 -0
- package/src/web/layouts/blank.vue +4 -0
- package/src/web/layouts/footer/index.scss +31 -0
- package/src/web/layouts/footer/index.ts +3 -0
- package/src/web/layouts/footer/index.vue +38 -0
- package/src/web/layouts/header/index.scss +35 -0
- package/src/web/layouts/header/index.ts +3 -0
- package/src/web/layouts/header/index.vue +47 -0
- package/src/web/layouts/iframe/index.scss +18 -0
- package/src/web/layouts/iframe/index.vue +36 -0
- package/src/web/layouts/iframe/page.vue +30 -0
- package/src/web/layouts/index.ts +8 -0
- package/src/web/layouts/router-view-content/index.vue +70 -0
- package/src/web/link/index.scss +95 -0
- package/src/web/link/index.ts +3 -0
- package/src/web/link/index.vue +68 -0
- package/src/web/loading/directive.ts +104 -0
- package/src/web/loading/index.ts +6 -0
- package/src/web/loading/loading.scss +108 -0
- package/src/web/loading/loading.ts +156 -0
- package/src/web/loading/service.ts +145 -0
- package/src/web/loading/types.ts +29 -0
- package/src/web/logo/index.scss +31 -0
- package/src/web/logo/index.ts +5 -0
- package/src/web/logo/index.vue +45 -0
- package/src/web/logo/types.ts +6 -0
- package/src/web/menu/index.scss +336 -0
- package/src/web/menu/index.ts +8 -0
- package/src/web/menu/menu-collapse-transition.vue +62 -0
- package/src/web/menu/menu-item-group.vue +27 -0
- package/src/web/menu/menu-item.vue +126 -0
- package/src/web/menu/menu.vue +459 -0
- package/src/web/menu/sub-menu.vue +440 -0
- package/src/web/menu/types.ts +66 -0
- package/src/web/menu/use-menu-css-var.ts +11 -0
- package/src/web/menu/use-menu.ts +60 -0
- package/src/web/menu/utils/menu-bar.ts +19 -0
- package/src/web/menu/utils/menu-item.ts +55 -0
- package/src/web/menu/utils/submenu.ts +66 -0
- package/src/web/multiple-select-flat/index.ts +5 -0
- package/src/web/multiple-select-flat/index.vue +53 -0
- package/src/web/multiple-select-flat/types.ts +5 -0
- package/src/web/multiple-tabs/index.scss +16 -0
- package/src/web/multiple-tabs/index.ts +5 -0
- package/src/web/multiple-tabs/index.vue +193 -0
- package/src/web/multiple-tabs/tab-content.vue +40 -0
- package/src/web/multiple-tabs/types.ts +3 -0
- package/src/web/multiple-tabs/use/index.ts +2 -0
- package/src/web/multiple-tabs/use/use-multiple-tabs.ts +86 -0
- package/src/web/multiple-tabs/use/use-tab-dropdown.ts +101 -0
- package/src/web/nodata/README.md +42 -0
- package/src/web/nodata/fail.vue +13 -0
- package/src/web/nodata/img/100.png +0 -0
- package/src/web/nodata/img/101.png +0 -0
- package/src/web/nodata/img/102.png +0 -0
- package/src/web/nodata/img/103.png +0 -0
- package/src/web/nodata/img/104.png +0 -0
- package/src/web/nodata/img/105.png +0 -0
- package/src/web/nodata/img/106.png +0 -0
- package/src/web/nodata/img/107.png +0 -0
- package/src/web/nodata/img/200.png +0 -0
- package/src/web/nodata/img/201.png +0 -0
- package/src/web/nodata/img/202.png +0 -0
- package/src/web/nodata/img/203.png +0 -0
- package/src/web/nodata/index.scss +37 -0
- package/src/web/nodata/index.ts +6 -0
- package/src/web/nodata/index.vue +46 -0
- package/src/web/nodata/types.ts +17 -0
- package/src/web/only-child/index.tsx +69 -0
- package/src/web/pagination/components/jumper.vue +49 -0
- package/src/web/pagination/components/next.vue +40 -0
- package/src/web/pagination/components/pager.vue +215 -0
- package/src/web/pagination/components/prev.vue +35 -0
- package/src/web/pagination/components/sizes.vue +76 -0
- package/src/web/pagination/components/total.vue +21 -0
- package/src/web/pagination/index.scss +231 -0
- package/src/web/pagination/index.ts +5 -0
- package/src/web/pagination/pagination.ts +363 -0
- package/src/web/pagination/usePagination.ts +13 -0
- package/src/web/popover/directive.ts +21 -0
- package/src/web/popover/index.scss +58 -0
- package/src/web/popover/index.ts +3 -0
- package/src/web/popover/index.vue +161 -0
- package/src/web/popover/types.ts +26 -0
- package/src/web/popper/arrow.vue +45 -0
- package/src/web/popper/content.vue +311 -0
- package/src/web/popper/index.scss +108 -0
- package/src/web/popper/index.ts +11 -0
- package/src/web/popper/popper.vue +57 -0
- package/src/web/popper/tokens.ts +28 -0
- package/src/web/popper/trigger.vue +166 -0
- package/src/web/popper/types.ts +49 -0
- package/src/web/popper/utils.ts +81 -0
- package/src/web/qrcode/drawCanvas.ts +32 -0
- package/src/web/qrcode/drawLogo.ts +82 -0
- package/src/web/qrcode/index.ts +5 -0
- package/src/web/qrcode/index.vue +107 -0
- package/src/web/qrcode/qrcodePlus.ts +4 -0
- package/src/web/qrcode/toCanvas.ts +11 -0
- package/src/web/qrcode/types.ts +38 -0
- package/src/web/result/index.scss +69 -0
- package/src/web/result/index.ts +3 -0
- package/src/web/result/index.vue +63 -0
- package/src/web/scrollbar/bar.vue +48 -0
- package/src/web/scrollbar/index.scss +91 -0
- package/src/web/scrollbar/index.ts +5 -0
- package/src/web/scrollbar/index.vue +236 -0
- package/src/web/scrollbar/thumb.vue +183 -0
- package/src/web/scrollbar/tokens.ts +10 -0
- package/src/web/scrollbar/types.ts +7 -0
- package/src/web/scrollbar/util.ts +38 -0
- package/src/web/select/constants.ts +13 -0
- package/src/web/select/index.ts +11 -0
- package/src/web/select/index.vue +555 -0
- package/src/web/select/option-group.scss +49 -0
- package/src/web/select/option-group.vue +97 -0
- package/src/web/select/option-item.scss +66 -0
- package/src/web/select/option.scss +32 -0
- package/src/web/select/option.vue +110 -0
- package/src/web/select/select-dropdown.scss +86 -0
- package/src/web/select/select-dropdown.vue +51 -0
- package/src/web/select/select.scss +213 -0
- package/src/web/select/token.ts +56 -0
- package/src/web/select/useOption.ts +146 -0
- package/src/web/select/useSelect.ts +942 -0
- package/src/web/select/utils.ts +5 -0
- package/src/web/side-menu/index.scss +66 -0
- package/src/web/side-menu/index.ts +4 -0
- package/src/web/side-menu/index.vue +228 -0
- package/src/web/side-menu/types.ts +20 -0
- package/src/web/single-select/index.scss +60 -0
- package/src/web/single-select/index.ts +5 -0
- package/src/web/single-select/index.vue +70 -0
- package/src/web/single-select/select@2x.png +0 -0
- package/src/web/single-select/types.ts +5 -0
- package/src/web/svg/index.ts +3 -0
- package/src/web/svg/index.vue +22 -0
- package/src/web/tabs/index.scss +579 -0
- package/src/web/tabs/index.ts +6 -0
- package/src/web/tabs/index.vue +236 -0
- package/src/web/tabs/tab-bar.vue +90 -0
- package/src/web/tabs/tab-nav.vue +403 -0
- package/src/web/tabs/tab-pane.vue +90 -0
- package/src/web/tabs/types.ts +66 -0
- package/src/web/tag/index.scss +182 -0
- package/src/web/tag/index.ts +5 -0
- package/src/web/tag/index.vue +78 -0
- package/src/web/tag/types.ts +2 -0
- package/src/web/tooltip/content.vue +239 -0
- package/src/web/tooltip/index.ts +4 -0
- package/src/web/tooltip/tokens.ts +21 -0
- package/src/web/tooltip/tooltip.vue +270 -0
- package/src/web/tooltip/trigger.vue +119 -0
- package/src/web/tooltip/types.ts +56 -0
- package/src/web/tooltip/utils.ts +20 -0
- package/src/web/transition/collapse-transition.vue +73 -0
- package/src/web/transition/index.ts +5 -0
- package/tsconfig.json +8 -0
- package/types/component.ts +1 -0
- package/types/index.d.ts +286 -0
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
<template>
|
|
2
|
+
<el-dropdown ref="CascaderSelectDropdown" trigger="click" placement="bottom-start" :hide-on-click="false" @visible-change="onVisibleChange">
|
|
3
|
+
<div :class="ns.b()">
|
|
4
|
+
<span :class="[ns.e('title'), ns.is('show', showDropdown || realCheckList.length)]">{{ title }}</span>
|
|
5
|
+
<vri-svg :class="[ns.e('icon-arrow-down'), ns.is('rotate', showDropdown)]" name="arrow-down" :color="showDropdown || realCheckList.length ? '#409eff' : '#606266'"/>
|
|
6
|
+
</div>
|
|
7
|
+
<template #dropdown>
|
|
8
|
+
<div :class="ns.e('dropdown')">
|
|
9
|
+
<cascader-panel ref="cascadePanel" v-model="checkList" :options="options" :props="props" :border="false">
|
|
10
|
+
<template #bottom>
|
|
11
|
+
<dropdown-bottom @reset="reset" @confirm="submit" />
|
|
12
|
+
</template>
|
|
13
|
+
</cascader-panel>
|
|
14
|
+
</div>
|
|
15
|
+
</template>
|
|
16
|
+
</el-dropdown>
|
|
17
|
+
</template>
|
|
18
|
+
|
|
19
|
+
<script setup lang="ts">
|
|
20
|
+
import CascaderPanel from '../cascader-panel/index.vue';
|
|
21
|
+
import VriSvg from '../svg/index.vue';
|
|
22
|
+
import DropdownBottom from '../dropdown-bottom/index.vue';
|
|
23
|
+
import { ref } from 'vue';
|
|
24
|
+
import { type CascaderProps } from 'element-plus';
|
|
25
|
+
import { useNamespace } from '../../use';
|
|
26
|
+
import type { IResult, ICheckedNodes } from './types';
|
|
27
|
+
|
|
28
|
+
const ns = /* hoist-static*/ useNamespace('cascader-select');
|
|
29
|
+
defineOptions({ name: ns.b() });
|
|
30
|
+
|
|
31
|
+
interface CascaderSelcetProps {
|
|
32
|
+
title: string;
|
|
33
|
+
options: Array<any>;
|
|
34
|
+
type?: string;
|
|
35
|
+
labelKey?: string;
|
|
36
|
+
valueKey?: string;
|
|
37
|
+
childrenKey?: string;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const { title, options, type = '', labelKey = 'name', valueKey = 'code', childrenKey = 'values' } = defineProps<CascaderSelcetProps>();
|
|
41
|
+
|
|
42
|
+
const props: CascaderProps = {
|
|
43
|
+
multiple: true,
|
|
44
|
+
label: labelKey,
|
|
45
|
+
value: valueKey,
|
|
46
|
+
children: childrenKey
|
|
47
|
+
};
|
|
48
|
+
const emit = defineEmits<{
|
|
49
|
+
(e: 'submit', val: IResult): void;
|
|
50
|
+
}>();
|
|
51
|
+
|
|
52
|
+
const CascaderSelectDropdown = ref();
|
|
53
|
+
|
|
54
|
+
const checkList = ref<any>([]);
|
|
55
|
+
const realCheckList = ref<any>([]);
|
|
56
|
+
|
|
57
|
+
const showDropdown = ref(false);
|
|
58
|
+
|
|
59
|
+
/** 监听下拉菜单关闭时实际值和选择值是否一致,若不一致则将显示值修改为和实际值一致 */
|
|
60
|
+
const onVisibleChange = (visible: boolean) => {
|
|
61
|
+
showDropdown.value = visible;
|
|
62
|
+
if (!visible && JSON.stringify(realCheckList.value.sort()) !== JSON.stringify(checkList.value.sort())) {
|
|
63
|
+
checkList.value = JSON.parse(JSON.stringify(realCheckList.value));
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
const reset = () => {
|
|
68
|
+
checkList.value = [];
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
const cascadePanel = ref();
|
|
72
|
+
const submit = () => {
|
|
73
|
+
realCheckList.value = JSON.parse(JSON.stringify(checkList.value));
|
|
74
|
+
let result: IResult = {};
|
|
75
|
+
|
|
76
|
+
/** 地区特殊处理 */
|
|
77
|
+
if (type === 'area') {
|
|
78
|
+
result = handleAreaChecked(cascadePanel.value.getCheckedNodes());
|
|
79
|
+
} else {
|
|
80
|
+
checkList.value.forEach((leaf: Array<string>) => {
|
|
81
|
+
leaf.forEach((val: string, index: number) => {
|
|
82
|
+
const lv = `lv${index + 1}`;
|
|
83
|
+
if (!result[lv]) {
|
|
84
|
+
result[lv] = [];
|
|
85
|
+
}
|
|
86
|
+
if (!result[lv].includes(val)) {
|
|
87
|
+
result[lv].push(val);
|
|
88
|
+
}
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
emit('submit', result);
|
|
93
|
+
CascaderSelectDropdown.value?.handleClose();
|
|
94
|
+
};
|
|
95
|
+
|
|
96
|
+
const handleAreaChecked = (nodes: Array<ICheckedNodes>) => {
|
|
97
|
+
let result: IResult = {
|
|
98
|
+
lv1: [],
|
|
99
|
+
lv2: [],
|
|
100
|
+
lv3: []
|
|
101
|
+
};
|
|
102
|
+
nodes.forEach((node) => {
|
|
103
|
+
if (node.level === 1) {
|
|
104
|
+
result.lv1.push(node.value);
|
|
105
|
+
}
|
|
106
|
+
if (node.level === 2) {
|
|
107
|
+
/** 非直辖市 */
|
|
108
|
+
if (nodes.findIndex((n) => n.value === node?.parent?.value) > -1) {
|
|
109
|
+
} else {
|
|
110
|
+
if (node.children.length) {
|
|
111
|
+
/** 非直辖市 */
|
|
112
|
+
result.lv2.push(node.value);
|
|
113
|
+
} else {
|
|
114
|
+
/** 直辖市 */
|
|
115
|
+
result.lv3.push(node.value);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
if (node.level === 3) {
|
|
120
|
+
if (nodes.findIndex((no) => no.value === node?.parent?.value) > -1) {
|
|
121
|
+
} else {
|
|
122
|
+
result.lv3.push(node.value);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
return result;
|
|
127
|
+
};
|
|
128
|
+
</script>
|
|
129
|
+
|
|
130
|
+
<style lang="scss" scoped>
|
|
131
|
+
@import './index.scss';
|
|
132
|
+
</style>
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { isClient } from '@vri/utils';
|
|
2
|
+
import { createVNode, render } from 'vue';
|
|
3
|
+
import contextMenuVue from './index.vue';
|
|
4
|
+
import type { ContextMenuProps, CreateContextOptions } from './types';
|
|
5
|
+
|
|
6
|
+
const menuManager: {
|
|
7
|
+
domList: Element[];
|
|
8
|
+
resolve: Fn;
|
|
9
|
+
} = {
|
|
10
|
+
domList: [],
|
|
11
|
+
resolve: () => {}
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
export const createContextMenu = function (options: CreateContextOptions) {
|
|
15
|
+
const { event } = options || {};
|
|
16
|
+
|
|
17
|
+
// event?.preventDefault() 阻止浏览器的默认右键菜单事件
|
|
18
|
+
event && event?.preventDefault();
|
|
19
|
+
|
|
20
|
+
if (!isClient) {
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
return new Promise((resolve) => {
|
|
24
|
+
const body = document.body;
|
|
25
|
+
|
|
26
|
+
const container = document.createElement('div');
|
|
27
|
+
const propsData: Partial<ContextMenuProps> = {
|
|
28
|
+
...options
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
if (options.event) {
|
|
32
|
+
propsData.axis = { x: event.clientX, y: event.clientY };
|
|
33
|
+
// @ts-ignore
|
|
34
|
+
delete propsData.event;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
const vm = createVNode(contextMenuVue, propsData);
|
|
38
|
+
render(vm, container);
|
|
39
|
+
|
|
40
|
+
const handleClick = function () {
|
|
41
|
+
menuManager.resolve('');
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
menuManager.domList.push(container);
|
|
45
|
+
|
|
46
|
+
const remove = function () {
|
|
47
|
+
menuManager.domList.forEach((dom: Element) => {
|
|
48
|
+
try {
|
|
49
|
+
dom && body.removeChild(dom);
|
|
50
|
+
} catch (error) {}
|
|
51
|
+
});
|
|
52
|
+
body.removeEventListener('click', handleClick);
|
|
53
|
+
body.removeEventListener('scroll', handleClick);
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
menuManager.resolve = function (arg) {
|
|
57
|
+
remove();
|
|
58
|
+
resolve(arg);
|
|
59
|
+
};
|
|
60
|
+
remove();
|
|
61
|
+
body.appendChild(container);
|
|
62
|
+
// body.addEventListener('click', handleClick);
|
|
63
|
+
// body.addEventListener('scroll', handleClick);
|
|
64
|
+
});
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
export const destroyContextMenu = function () {
|
|
68
|
+
if (menuManager) {
|
|
69
|
+
menuManager.resolve('');
|
|
70
|
+
menuManager.domList = [];
|
|
71
|
+
}
|
|
72
|
+
};
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
<script lang="ts" setup>
|
|
2
|
+
import { useClickAway, useEventListener, useTimeoutFn } from '@vri/use';
|
|
3
|
+
import { computed, ref, type CSSProperties } from 'vue';
|
|
4
|
+
import type { Axis, ContextMenuItem } from './types';
|
|
5
|
+
import { MenuItem } from '../menu';
|
|
6
|
+
|
|
7
|
+
interface Props {
|
|
8
|
+
axis?: Axis;
|
|
9
|
+
styles?: CSSProperties;
|
|
10
|
+
width?: number;
|
|
11
|
+
items?: ContextMenuItem[];
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
const {
|
|
15
|
+
axis, styles, width, items
|
|
16
|
+
} = defineProps<Props>();
|
|
17
|
+
|
|
18
|
+
let contextMenuRef = ref();
|
|
19
|
+
|
|
20
|
+
const getStyle = computed((): CSSProperties => {
|
|
21
|
+
const { x, y } = axis || { x: 0, y: 0 };
|
|
22
|
+
const menuHeight = (items || []).length * 40;
|
|
23
|
+
const menuWidth = width || 0;
|
|
24
|
+
const body = document.body;
|
|
25
|
+
|
|
26
|
+
const left = body.clientWidth < x + menuWidth ? x - menuWidth : x;
|
|
27
|
+
const top = body.clientHeight < y + menuHeight ? y - menuHeight : y;
|
|
28
|
+
return {
|
|
29
|
+
zIndex: 3000,
|
|
30
|
+
...styles,
|
|
31
|
+
position: 'fixed',
|
|
32
|
+
width: `${width}px`,
|
|
33
|
+
left: `${left + 1}px`,
|
|
34
|
+
top: `${top + 1}px`
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
let show = ref(false);
|
|
39
|
+
|
|
40
|
+
useTimeoutFn(() => {
|
|
41
|
+
show.value = true;
|
|
42
|
+
}, 0);
|
|
43
|
+
|
|
44
|
+
useClickAway(contextMenuRef, () => {
|
|
45
|
+
show.value = false;
|
|
46
|
+
}, {
|
|
47
|
+
listenerOptions: {
|
|
48
|
+
capture: true
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
useEventListener(window, 'scroll', () => {
|
|
53
|
+
show.value = false;
|
|
54
|
+
}, {
|
|
55
|
+
capture: true
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
useEventListener(window, 'drag', () => {
|
|
59
|
+
show.value = false;
|
|
60
|
+
}, {
|
|
61
|
+
capture: true
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
function handleClick (item: ContextMenuItem) {
|
|
65
|
+
if (!item?.disabled) {
|
|
66
|
+
item.handler?.();
|
|
67
|
+
show.value = false;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
</script>
|
|
71
|
+
|
|
72
|
+
<template>
|
|
73
|
+
<transition name="vri-zoom-in-top">
|
|
74
|
+
<div ref="contextMenuRef" v-show="show" :style="getStyle" @click.stop class="vri-menu">
|
|
75
|
+
<div class="vri-sub-menu">
|
|
76
|
+
<menu-item v-for="(item, index) in items"
|
|
77
|
+
@click="handleClick(item)"
|
|
78
|
+
:index="index + item.text"
|
|
79
|
+
:title="item.text"
|
|
80
|
+
:divider="item.divider"
|
|
81
|
+
:disabled="item.disabled"
|
|
82
|
+
:icon="item.iconCfg"
|
|
83
|
+
isAloneUse
|
|
84
|
+
:key="index" />
|
|
85
|
+
</div>
|
|
86
|
+
</div>
|
|
87
|
+
</transition>
|
|
88
|
+
</template>
|
|
89
|
+
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type CSSProperties } from 'vue';
|
|
2
|
+
import { type IconProps } from '../../common';
|
|
3
|
+
|
|
4
|
+
export interface Axis {
|
|
5
|
+
x: number;
|
|
6
|
+
y: number;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export interface ContextMenuItem {
|
|
10
|
+
text: string;
|
|
11
|
+
iconCfg?: IconProps;
|
|
12
|
+
disabled?: boolean;
|
|
13
|
+
handler?: Fn;
|
|
14
|
+
divider?: boolean;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export interface CreateContextOptions {
|
|
18
|
+
event: MouseEvent;
|
|
19
|
+
styles?: CSSProperties | Record<string, string>;
|
|
20
|
+
items?: ContextMenuItem[];
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export interface ContextMenuProps {
|
|
24
|
+
axis: Axis;
|
|
25
|
+
styles: CSSProperties;
|
|
26
|
+
items: ContextMenuItem[];
|
|
27
|
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { onUnmounted, getCurrentInstance } from 'vue';
|
|
2
|
+
import { createContextMenu, destroyContextMenu } from './createContextMenu';
|
|
3
|
+
import type { ContextMenuItem } from './types';
|
|
4
|
+
|
|
5
|
+
export type { ContextMenuItem };
|
|
6
|
+
|
|
7
|
+
export function useContextMenu(authRemove = true) {
|
|
8
|
+
if (getCurrentInstance() && authRemove) {
|
|
9
|
+
onUnmounted(() => {
|
|
10
|
+
destroyContextMenu();
|
|
11
|
+
});
|
|
12
|
+
}
|
|
13
|
+
return [createContextMenu, destroyContextMenu];
|
|
14
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
<template>
|
|
2
|
+
<slot/>
|
|
3
|
+
</template>
|
|
4
|
+
|
|
5
|
+
<script lang="ts" setup>
|
|
6
|
+
/* eslint-disable @typescript-eslint/no-unused-vars */
|
|
7
|
+
import { useNamespace } from '../../use';
|
|
8
|
+
|
|
9
|
+
interface Props {
|
|
10
|
+
label?: string
|
|
11
|
+
span?: number
|
|
12
|
+
width?: string | number
|
|
13
|
+
minWidth?: string | number
|
|
14
|
+
align?: string
|
|
15
|
+
labelAlign?: string
|
|
16
|
+
className?: string
|
|
17
|
+
labelClassName?: string
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const {
|
|
21
|
+
span = 1,
|
|
22
|
+
align = 'left'
|
|
23
|
+
} = defineProps<Props>();
|
|
24
|
+
|
|
25
|
+
const ns = /* hoist-static*/ useNamespace('descriptions-item');
|
|
26
|
+
|
|
27
|
+
defineOptions({
|
|
28
|
+
name: ns.b()
|
|
29
|
+
});
|
|
30
|
+
</script>
|
|
31
|
+
|
|
32
|
+
<style lang="scss">
|
|
33
|
+
@import "./descriptions-item.scss";
|
|
34
|
+
</style>
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
<template>
|
|
2
|
+
<div :class="descriptionKls">
|
|
3
|
+
<div
|
|
4
|
+
v-if="title || extra || $slots.title || $slots.extra"
|
|
5
|
+
:class="ns.e('header')"
|
|
6
|
+
>
|
|
7
|
+
<div :class="ns.e('title')">
|
|
8
|
+
<slot name="title">{{ title }}</slot>
|
|
9
|
+
</div>
|
|
10
|
+
<div :class="ns.e('extra')">
|
|
11
|
+
<slot name="extra">{{ extra }}</slot>
|
|
12
|
+
</div>
|
|
13
|
+
</div>
|
|
14
|
+
|
|
15
|
+
<div :class="ns.e('body')">
|
|
16
|
+
<table :class="[ns.e('table'), ns.is('bordered', border)]">
|
|
17
|
+
<tbody>
|
|
18
|
+
<template v-for="(row, index) in getRows()" :key="index">
|
|
19
|
+
<descriptions-row :row="row" />
|
|
20
|
+
</template>
|
|
21
|
+
</tbody>
|
|
22
|
+
</table>
|
|
23
|
+
</div>
|
|
24
|
+
</div>
|
|
25
|
+
</template>
|
|
26
|
+
|
|
27
|
+
<script lang="ts" setup>
|
|
28
|
+
import { computed, provide, useSlots, getCurrentInstance } from 'vue';
|
|
29
|
+
import { flattedChildren } from '../../utils';
|
|
30
|
+
import { useSize } from 'element-plus';
|
|
31
|
+
import { useNamespace } from '../../use';
|
|
32
|
+
import DescriptionsRow from './descriptions-row.vue';
|
|
33
|
+
import { descriptionsKey } from './token';
|
|
34
|
+
|
|
35
|
+
interface Props {
|
|
36
|
+
border?: boolean
|
|
37
|
+
column?: number
|
|
38
|
+
direction?: 'horizontal' | 'vertical'
|
|
39
|
+
size?: string
|
|
40
|
+
title?: string
|
|
41
|
+
extra?: string
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
const {
|
|
45
|
+
border = false,
|
|
46
|
+
column = 3,
|
|
47
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
48
|
+
direction = 'horizontal',
|
|
49
|
+
title,
|
|
50
|
+
extra
|
|
51
|
+
} = defineProps<Props>();
|
|
52
|
+
|
|
53
|
+
const ns = /* hoist-static*/ useNamespace('descriptions');
|
|
54
|
+
|
|
55
|
+
defineOptions({
|
|
56
|
+
name: ns.b()
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
const descriptionsSize = useSize();
|
|
60
|
+
|
|
61
|
+
const slots = useSlots();
|
|
62
|
+
|
|
63
|
+
const instance = getCurrentInstance()!;
|
|
64
|
+
|
|
65
|
+
provide(descriptionsKey, instance.props as any);
|
|
66
|
+
|
|
67
|
+
const descriptionKls = computed(() => [ns.b(), ns.m(descriptionsSize.value)]);
|
|
68
|
+
|
|
69
|
+
const filledNode = (node, span, count, isLast = false) => {
|
|
70
|
+
if (!node.props) {
|
|
71
|
+
node.props = {};
|
|
72
|
+
}
|
|
73
|
+
if (span > count) {
|
|
74
|
+
node.props.span = count;
|
|
75
|
+
}
|
|
76
|
+
if (isLast) {
|
|
77
|
+
// set the last span
|
|
78
|
+
node.props.span = span;
|
|
79
|
+
}
|
|
80
|
+
return node;
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
const getRows = () => {
|
|
84
|
+
const children = flattedChildren(slots.default?.()).filter(
|
|
85
|
+
(node) => node?.type?.name === 'vri-descriptions-item'
|
|
86
|
+
);
|
|
87
|
+
const rows: number[] = [];
|
|
88
|
+
let temp: number[] = [];
|
|
89
|
+
let count = column;
|
|
90
|
+
let totalSpan = 0; // all spans number of item
|
|
91
|
+
|
|
92
|
+
children.forEach((node, index) => {
|
|
93
|
+
const span = node.props?.span || 1;
|
|
94
|
+
|
|
95
|
+
if (index < children.length - 1) {
|
|
96
|
+
totalSpan += span > count ? count : span;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
if (index === children.length - 1) {
|
|
100
|
+
// calculate the last item span
|
|
101
|
+
const lastSpan = column - (totalSpan % column);
|
|
102
|
+
temp.push(filledNode(node, lastSpan, count, true));
|
|
103
|
+
rows.push(temp);
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
if (span < count) {
|
|
108
|
+
count -= span;
|
|
109
|
+
temp.push(node);
|
|
110
|
+
} else {
|
|
111
|
+
temp.push(filledNode(node, span, count));
|
|
112
|
+
rows.push(temp);
|
|
113
|
+
count = column;
|
|
114
|
+
temp = [];
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
return rows;
|
|
119
|
+
};
|
|
120
|
+
</script>
|
|
121
|
+
|
|
122
|
+
<style lang="scss">
|
|
123
|
+
@import "./descriptions.scss";
|
|
124
|
+
</style>
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { defineComponent, h, inject } from 'vue';
|
|
2
|
+
import { addUnit } from '@vri/utils';
|
|
3
|
+
import { useNamespace } from '../../use';
|
|
4
|
+
import { getNormalizedProps } from '../../utils';
|
|
5
|
+
import { descriptionsKey } from './token';
|
|
6
|
+
|
|
7
|
+
import type { VNode } from 'vue';
|
|
8
|
+
import type { IDescriptionsInject, IDescriptionsItemInject } from './descriptions.type';
|
|
9
|
+
|
|
10
|
+
export default defineComponent({
|
|
11
|
+
name: 'VriDescriptionsCell',
|
|
12
|
+
props: {
|
|
13
|
+
cell: {
|
|
14
|
+
type: Object
|
|
15
|
+
},
|
|
16
|
+
tag: {
|
|
17
|
+
type: String
|
|
18
|
+
},
|
|
19
|
+
type: {
|
|
20
|
+
type: String
|
|
21
|
+
}
|
|
22
|
+
},
|
|
23
|
+
setup() {
|
|
24
|
+
const descriptions = inject(descriptionsKey, {} as IDescriptionsInject);
|
|
25
|
+
|
|
26
|
+
return {
|
|
27
|
+
descriptions
|
|
28
|
+
};
|
|
29
|
+
},
|
|
30
|
+
render() {
|
|
31
|
+
const item = getNormalizedProps(this.cell as VNode) as IDescriptionsItemInject;
|
|
32
|
+
|
|
33
|
+
const { border, direction } = this.descriptions;
|
|
34
|
+
const isVertical = direction === 'vertical';
|
|
35
|
+
const label = this.cell?.children?.label?.() || item.label;
|
|
36
|
+
const content = this.cell?.children?.default?.();
|
|
37
|
+
const span = item.span;
|
|
38
|
+
const align = item.align ? `is-${item.align}` : '';
|
|
39
|
+
const labelAlign = item.labelAlign ? `is-${item.labelAlign}` : '' || align;
|
|
40
|
+
const className = item.className;
|
|
41
|
+
const labelClassName = item.labelClassName;
|
|
42
|
+
const style = {
|
|
43
|
+
width: addUnit(item.width),
|
|
44
|
+
minWidth: addUnit(item.minWidth)
|
|
45
|
+
};
|
|
46
|
+
const ns = useNamespace('descriptions');
|
|
47
|
+
|
|
48
|
+
switch (this.type) {
|
|
49
|
+
case 'label':
|
|
50
|
+
return h(
|
|
51
|
+
this.tag,
|
|
52
|
+
{
|
|
53
|
+
style,
|
|
54
|
+
class: [ns.e('cell'), ns.e('label'), ns.is('bordered-label', border), ns.is('vertical-label', isVertical), labelAlign, labelClassName],
|
|
55
|
+
colSpan: isVertical ? span : 1
|
|
56
|
+
},
|
|
57
|
+
label
|
|
58
|
+
);
|
|
59
|
+
case 'content':
|
|
60
|
+
return h(
|
|
61
|
+
this.tag,
|
|
62
|
+
{
|
|
63
|
+
class: [ns.e('cell'), ns.e('content'), ns.is('bordered-content', border), ns.is('vertical-content', isVertical), align, className],
|
|
64
|
+
colSpan: isVertical ? span : span * 2 - 1
|
|
65
|
+
},
|
|
66
|
+
content
|
|
67
|
+
);
|
|
68
|
+
default:
|
|
69
|
+
return h(
|
|
70
|
+
'td',
|
|
71
|
+
{
|
|
72
|
+
style,
|
|
73
|
+
class: [ns.e('cell'), align],
|
|
74
|
+
colSpan: span
|
|
75
|
+
},
|
|
76
|
+
[
|
|
77
|
+
h(
|
|
78
|
+
'span',
|
|
79
|
+
{
|
|
80
|
+
class: [ns.e('label'), labelClassName]
|
|
81
|
+
},
|
|
82
|
+
label
|
|
83
|
+
),
|
|
84
|
+
h(
|
|
85
|
+
'span',
|
|
86
|
+
{
|
|
87
|
+
class: [ns.e('content'), className]
|
|
88
|
+
},
|
|
89
|
+
content
|
|
90
|
+
)
|
|
91
|
+
]
|
|
92
|
+
);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
});
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
$descriptions-item-label-margin-right: () !default;
|
|
2
|
+
$descriptions-item-label-margin-right: map-merge(
|
|
3
|
+
(
|
|
4
|
+
'large': 16px,
|
|
5
|
+
'default': 16px,
|
|
6
|
+
'small': 12px,
|
|
7
|
+
),
|
|
8
|
+
$descriptions-item-label-margin-right
|
|
9
|
+
);
|
|
10
|
+
|
|
11
|
+
$descriptions-item-vertical-label-padding-bottom: () !default;
|
|
12
|
+
$descriptions-item-vertical-label-padding-bottom: map-merge(
|
|
13
|
+
(
|
|
14
|
+
'large': 8px,
|
|
15
|
+
'default': 6px,
|
|
16
|
+
'small': 4px,
|
|
17
|
+
),
|
|
18
|
+
$descriptions-item-vertical-label-padding-bottom
|
|
19
|
+
);
|
|
20
|
+
|
|
21
|
+
@include b(descriptions) {
|
|
22
|
+
@include e(label) {
|
|
23
|
+
&.#{$namespace}-descriptions__cell.is-bordered-label {
|
|
24
|
+
font-weight: bold;
|
|
25
|
+
color: getCssVar('text-color', 'regular');
|
|
26
|
+
background: getCssVar('descriptions-item-bordered-label-background');
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
&:not(.is-bordered-label) {
|
|
30
|
+
color: getCssVar('text-color', 'primary');
|
|
31
|
+
margin-right: map-get($descriptions-item-label-margin-right, 'default');
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
&.#{$namespace}-descriptions__cell:not(.is-bordered-label).is-vertical-label {
|
|
35
|
+
padding-bottom: map-get(
|
|
36
|
+
$descriptions-item-vertical-label-padding-bottom,
|
|
37
|
+
'default'
|
|
38
|
+
);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
@include e(content) {
|
|
43
|
+
&.#{$namespace}-descriptions__cell.is-bordered-content {
|
|
44
|
+
color: getCssVar('text-color', 'primary');
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
&:not(.is-bordered-label) {
|
|
48
|
+
color: getCssVar('text-color', 'regular');
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
@each $size in (large, small) {
|
|
53
|
+
@include m($size) {
|
|
54
|
+
@include e(label) {
|
|
55
|
+
&:not(.is-bordered-label) {
|
|
56
|
+
margin-right: map-get($descriptions-item-label-margin-right, $size);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
&.#{$namespace}-descriptions__cell:not(.is-bordered-label).is-vertical-label {
|
|
60
|
+
padding-bottom: map-get(
|
|
61
|
+
$descriptions-item-vertical-label-padding-bottom,
|
|
62
|
+
$size
|
|
63
|
+
);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|