menu.mjs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. import { defineComponent, getCurrentInstance, inject, ref, computed, resolveComponent, openBlock, createBlock, normalizeClass, withCtx, createElementBlock, Fragment, renderList, createVNode, createTextVNode, toDisplayString, createCommentVNode } from 'vue';
  2. import { ElScrollbar } from '../../scrollbar/index.mjs';
  3. import '../../../hooks/index.mjs';
  4. import '../../../utils/index.mjs';
  5. import { Loading } from '@element-plus/icons-vue';
  6. import { ElIcon } from '../../icon/index.mjs';
  7. import ElCascaderNode from './node2.mjs';
  8. import { CASCADER_PANEL_INJECTION_KEY } from './types.mjs';
  9. import _export_sfc from '../../../_virtual/plugin-vue_export-helper.mjs';
  10. import { useNamespace } from '../../../hooks/use-namespace/index.mjs';
  11. import { useLocale } from '../../../hooks/use-locale/index.mjs';
  12. import { generateId } from '../../../utils/rand.mjs';
  13. const _sfc_main = defineComponent({
  14. name: "ElCascaderMenu",
  15. components: {
  16. Loading,
  17. ElIcon,
  18. ElScrollbar,
  19. ElCascaderNode
  20. },
  21. props: {
  22. nodes: {
  23. type: Array,
  24. required: true
  25. },
  26. index: {
  27. type: Number,
  28. required: true
  29. }
  30. },
  31. setup(props) {
  32. const instance = getCurrentInstance();
  33. const ns = useNamespace("cascader-menu");
  34. const { t } = useLocale();
  35. const id = generateId();
  36. let activeNode = null;
  37. let hoverTimer = null;
  38. const panel = inject(CASCADER_PANEL_INJECTION_KEY);
  39. const hoverZone = ref(null);
  40. const isEmpty = computed(() => !props.nodes.length);
  41. const isLoading = computed(() => !panel.initialLoaded);
  42. const menuId = computed(() => `cascader-menu-${id}-${props.index}`);
  43. const handleExpand = (e) => {
  44. activeNode = e.target;
  45. };
  46. const handleMouseMove = (e) => {
  47. if (!panel.isHoverMenu || !activeNode || !hoverZone.value)
  48. return;
  49. if (activeNode.contains(e.target)) {
  50. clearHoverTimer();
  51. const el = instance.vnode.el;
  52. const { left } = el.getBoundingClientRect();
  53. const { offsetWidth, offsetHeight } = el;
  54. const startX = e.clientX - left;
  55. const top = activeNode.offsetTop;
  56. const bottom = top + activeNode.offsetHeight;
  57. hoverZone.value.innerHTML = `
  58. <path style="pointer-events: auto;" fill="transparent" d="M${startX} ${top} L${offsetWidth} 0 V${top} Z" />
  59. <path style="pointer-events: auto;" fill="transparent" d="M${startX} ${bottom} L${offsetWidth} ${offsetHeight} V${bottom} Z" />
  60. `;
  61. } else if (!hoverTimer) {
  62. hoverTimer = window.setTimeout(clearHoverZone, panel.config.hoverThreshold);
  63. }
  64. };
  65. const clearHoverTimer = () => {
  66. if (!hoverTimer)
  67. return;
  68. clearTimeout(hoverTimer);
  69. hoverTimer = null;
  70. };
  71. const clearHoverZone = () => {
  72. if (!hoverZone.value)
  73. return;
  74. hoverZone.value.innerHTML = "";
  75. clearHoverTimer();
  76. };
  77. return {
  78. ns,
  79. panel,
  80. hoverZone,
  81. isEmpty,
  82. isLoading,
  83. menuId,
  84. t,
  85. handleExpand,
  86. handleMouseMove,
  87. clearHoverZone
  88. };
  89. }
  90. });
  91. function _sfc_render(_ctx, _cache, $props, $setup, $data, $options) {
  92. const _component_el_cascader_node = resolveComponent("el-cascader-node");
  93. const _component_loading = resolveComponent("loading");
  94. const _component_el_icon = resolveComponent("el-icon");
  95. const _component_el_scrollbar = resolveComponent("el-scrollbar");
  96. return openBlock(), createBlock(_component_el_scrollbar, {
  97. key: _ctx.menuId,
  98. tag: "ul",
  99. role: "menu",
  100. class: normalizeClass(_ctx.ns.b()),
  101. "wrap-class": _ctx.ns.e("wrap"),
  102. "view-class": [_ctx.ns.e("list"), _ctx.ns.is("empty", _ctx.isEmpty)],
  103. onMousemove: _ctx.handleMouseMove,
  104. onMouseleave: _ctx.clearHoverZone
  105. }, {
  106. default: withCtx(() => {
  107. var _a;
  108. return [
  109. (openBlock(true), createElementBlock(Fragment, null, renderList(_ctx.nodes, (node) => {
  110. return openBlock(), createBlock(_component_el_cascader_node, {
  111. key: node.uid,
  112. node,
  113. "menu-id": _ctx.menuId,
  114. onExpand: _ctx.handleExpand
  115. }, null, 8, ["node", "menu-id", "onExpand"]);
  116. }), 128)),
  117. _ctx.isLoading ? (openBlock(), createElementBlock("div", {
  118. key: 0,
  119. class: normalizeClass(_ctx.ns.e("empty-text"))
  120. }, [
  121. createVNode(_component_el_icon, {
  122. size: "14",
  123. class: normalizeClass(_ctx.ns.is("loading"))
  124. }, {
  125. default: withCtx(() => [
  126. createVNode(_component_loading)
  127. ]),
  128. _: 1
  129. }, 8, ["class"]),
  130. createTextVNode(" " + toDisplayString(_ctx.t("el.cascader.loading")), 1)
  131. ], 2)) : _ctx.isEmpty ? (openBlock(), createElementBlock("div", {
  132. key: 1,
  133. class: normalizeClass(_ctx.ns.e("empty-text"))
  134. }, toDisplayString(_ctx.t("el.cascader.noData")), 3)) : ((_a = _ctx.panel) == null ? void 0 : _a.isHoverMenu) ? (openBlock(), createElementBlock("svg", {
  135. key: 2,
  136. ref: "hoverZone",
  137. class: normalizeClass(_ctx.ns.e("hover-zone"))
  138. }, null, 2)) : createCommentVNode("v-if", true)
  139. ];
  140. }),
  141. _: 1
  142. }, 8, ["class", "wrap-class", "view-class", "onMousemove", "onMouseleave"]);
  143. }
  144. var ElCascaderMenu = /* @__PURE__ */ _export_sfc(_sfc_main, [["render", _sfc_render], ["__file", "/home/runner/work/element-plus/element-plus/packages/components/cascader-panel/src/menu.vue"]]);
  145. export { ElCascaderMenu as default };
  146. //# sourceMappingURL=menu.mjs.map