浏览代码

更新提示指令,删除@/components/CtTooltip/index.js旧代码

lifanagju_citu 8 月之前
父节点
当前提交
1400b056b1

+ 0 - 151
src/components/CtTooltip/ToolTip.vue

@@ -1,151 +0,0 @@
-<template>
-  <!-- 指示 -->
-  <transition name="el-fade-in-linear">
-    <div v-show="tooltipShow" :style="tooltipStyle" class="wq-tooltip"
-    >
-      <span class="wq-tooltip-text" v-text="text"></span>
-      <!-- 箭头 -->
-      <!-- <div :class="[
-        {'left':placements === 'left'},
-        {'bottom':placements==='bottom'},
-        {'right':placements==='right'},
-        {'top':placements==='top'}]" 
-        class="wq-tooltip-arrow"
-      >
-      </div> -->
-    </div>
-  </transition>
-</template>
-
-<script>
-import {ref, computed } from 'vue'
-
-export default {
-  setup() {
-    // 显示弹框
-    const tooltipShow = ref(false);
-
-    // 提示内容
-    const text = ref()
-
-    // 方向
-    const placements = ref('left')
-
-    // 显示
-    function showTip() {
-      tooltipShow.value = true
-    }
-    //设置提示内容
-    function setContent(content) {
-      text.value = content
-    }
-    //隐藏
-    function hiddenTip() {
-      tooltipShow.value = false
-    }
-
-    // 位置
-    const tooltipPosition = ref({
-      x: 0,
-      y: 0
-    })
-    
-    // const tooltipContainer = document.getElementsByClassName('wq-tooltip')
-    
-    const tooltipStyle = computed(() => {
-      return {
-        transform: `translate3d(${tooltipPosition.value.x}px,${tooltipPosition.value.y}px,0)`
-      }
-    })
-    return {
-      tooltipShow,
-      showTip,
-      hiddenTip,
-      setContent,
-      tooltipPosition,
-      tooltipStyle,
-      text,
-      placements,
-    }
-  }
-}
-</script>
-
-<style lang="scss" scoped>
-// tooltip
-.wq-tooltip {
-  padding: 5px 16px;
-  font-size: .875rem;
-  line-height: 1.6;
-  min-width: 10px;
-  word-wrap: break-word;
-  position: fixed;
-  left: 0;
-  top: 0;
-  pointer-events: none;
-  background: rgb(66, 66, 66);
-  color: rgb(238, 238, 238);
-  border-radius: 4px;
-  font-size: .875rem;
-  line-height: 1.6;
-  display: block;
-  z-index: 2000;
-  opacity: 1;
-  width: auto;
-  transition-property: opacity,width;
-}
-
-// 小箭头
-.wq-tooltip-arrow {
-  position: absolute;
-  width: 0;
-  height: 0;
-  border-width: 8px;
-  border-style: solid;
-}
-
-// 小箭头如果在左侧
-.wq-tooltip-arrow.left {
-  border-color: transparent transparent transparent #303133;
-  right: -15px;
-  top: 50%;
-  transform: translate3d(0, -50%, 0);
-}
-
-// 小箭头如果在下侧
-.wq-tooltip-arrow.bottom {
-  top: -15px;
-  border-color: transparent transparent #303133 transparent;
-  left: 50%;
-  transform: translate3d(-50%, 0, 0);
-}
-
-// 小箭头如果在右侧
-.wq-tooltip-arrow.right {
-  left: -15px;
-  top: 50%;
-  transform: translate3d(0, -50%, 0);
-  border-color: transparent #303133 transparent transparent;
-}
-
-// 小箭头如果在上侧
-.wq-tooltip-arrow.top {
-  bottom: -15px;
-  border-color: #303133 transparent transparent transparent;
-  left: 50%;
-  transform: translate3d(-50%, 0, 0);
-}
-
-/* 动画 */
-.tooltip-enter-from,
-.tooltip-leave-to {
-  opacity: 0;
-  transition: opacity .3s ease;
-}
-
-.tooltip-leave-from,
-.tooltip-enter-to {
-  transition: opacity .1s ease;
-}
-</style>
-

+ 0 - 130
src/components/CtTooltip/index.js

@@ -1,130 +0,0 @@
-// 引入组件
-import {createApp, nextTick} from "vue";
-import MyToolTip from './ToolTip.vue'
-
-/**
- * v-ellipse-tooltip.top
-*/
-
-// 位置定位
-function calculationLocation(el, target, placements) {
-  if (!el || !target) return;
-
-  el.tooltipPosition.y = 0;
-  el.tooltipPosition.x = 0;
-  let el_dom = el.$el?.nextElementSibling?.getBoundingClientRect() || target?.w_tipInstance?.$el?.nextElementSibling?.getBoundingClientRect() || null
-  let target_dom = target.getBoundingClientRect()
-  if(!el_dom) {
-    console.log('el_dom:不存在', el_dom)
-    return
-  }
-
-  if (placements === "left") {
-    el.tooltipPosition.x = target_dom.x - el_dom.width - 10
-    el.tooltipPosition.y = target_dom.y - el_dom.height / 2 + target_dom.height / 2
-  } else if (placements === "bottom") {
-    el.tooltipPosition.x = target_dom.x + target_dom.width / 2 - el_dom.width / 2
-    el.tooltipPosition.y = target_dom.y + el_dom.height + 10
-  } else if (placements === "right") {
-    el.tooltipPosition.x = target_dom.x + target_dom.width + 10
-    el.tooltipPosition.y = target_dom.y - el_dom.height / 2 + target_dom.height / 2
-  } else if (placements === "top") {
-    el.tooltipPosition.x = target_dom.x + target_dom.width / 2 - el_dom.width / 2
-    el.tooltipPosition.y = target_dom.y - el_dom.height - 10
-  }
-}
-
-// 方向
-const allPlacements = ['left', 'bottom', 'right', 'top']
-
-
-// function getElStyleAttr(element, attr) {
-//   const styles = window.getComputedStyle(element)
-//   return styles[attr]
-// }
-
-// const positionXY = getPosition(el)
-// const getPosition = (target) => {
-//   const x = target.offsetLeft + target.offsetWidth/2
-//   const y = target.offsetTop + target.offsetHeight/2
-//   return { x, y}
-// }
-
-const isOverflow = (target) => {
-  const scrollWidth = target.scrollWidth
-  const offsetWidth = target.offsetWidth
-  const range = document.createRange()
-  range.setStart(target, 0)
-  range.setEnd(target, target.childNodes.length)
-  // const rangeWidth = range.getBoundingClientRect().width
-  // const padding = (parseInt(getElStyleAttr(target, 'paddingLeft'), 10) || 0) + (parseInt(getElStyleAttr(target, 'paddingRight'), 10) || 0)
-  // return (rangeWidth + padding > target.offsetWidth) || scrollWidth > offsetWidth
-  return scrollWidth > offsetWidth
-}
-
-export const ellipsisTooltip = localStorage.getItem('useEllipseTooltip') ? {
-  mounted(el, binding) {
-    // console.log('mounted-> el:', el)
-    // console.log('mounted-> binding:', binding)
-    //获取指令的参数
-    const {
-        value: {
-            placement, content, destroyOnLeave
-        } = {}
-    } = binding;
-    // 加上超出...样式
-    el.style.overflow = "hidden";
-    el.style.textOverflow = "ellipsis";
-    el.style.whiteSpace = "nowrap";
-    //鼠标移开时 清除元素
-    const onMouseLeave = () => {
-      if (el.w_tipInstance) {
-        el.w_tipInstance.hiddenTip()
-        el.w_tooltip.remove()
-        el.w_tipInstance = null
-        el.w_tooltip = null
-      }
-    };
-    const onMouseEnter = () => {
-      // 判断内容长度 需要展示
-      if (isOverflow(el)) {
-        const directiveList = allPlacements.filter(placement => binding.modifiers[placement])
-        const placements = directiveList.length ? directiveList : allPlacements
-        // if (!el.w_tooltip) {}
-        // 创建tooltip实例
-        const vm = createApp(MyToolTip)
-        // 创建根元素
-        el.w_tooltip = document.createElement('div')
-        // 挂载到页面
-        document.body.appendChild(el.w_tooltip)
-        el.w_tooltip.id = `tooltip_${Math.floor(Math.random() * 10000)}`
-        el.w_tipInstance = vm.mount(el.w_tooltip)
-        // 设置 tooltip 显示方向
-        el.w_tipInstance.placements = placement || placements[0] || 'top'
-        // 设置显示内容
-        el.w_tipInstance.setContent(content || el.innerText)
-        // 使 tooltip 显示
-        el.w_tipInstance.showTip()
-        nextTick(() => {
-          // 计算 tooltip 在页面中的位置
-          calculationLocation(el.w_tipInstance, el, placements[0])
-        })
-        el._scrollHandler = () => {
-          // 重新定位位置
-          if (el.w_tipInstance && el.w_tipInstance.tooltipShow) calculationLocation(el.w_tipInstance, el, placements[0])
-        }
-        window.addEventListener('scroll', el._scrollHandler)
-        const _destroyOnLeave = destroyOnLeave || true
-        if (_destroyOnLeave) el.addEventListener("mouseleave", onMouseLeave);
-      }
-    };
-    el.addEventListener("mouseenter", onMouseEnter);
-  },
-  unmounted(el) {
-    if (el.w_tooltip) {
-      document.body.removeChild(el.w_tooltip)
-    }
-    window.removeEventListener('scroll', el._scrollHandler)
-  }
-} : {}
-

+ 0 - 182
src/components/Ellipsis/index.vue

@@ -1,182 +0,0 @@
-<!-- vue-amazing-ui组件源码地址 https://themusecatcher.github.io/vue-amazing-ui/guide/components/ellipsis.html -->
-
-<script setup>
-import { ref, computed, watch, onMounted } from 'vue'
-import Tooltip from '@/components/Tooltip'
-import { useResizeObserver } from '@/utils/ellipsis.js'
-
-const props = defineProps({
-  maxWidth: { // 文本最大宽度,单位 px
-    type: [String, Number],
-    default: '100%'
-  },
-  tooltipMaxWidth: { // 弹出提示最大宽度,单位 px,默认为 文本宽度 + 24
-    type: [String, Number],
-    default: undefined
-  },
-  line: { // 最大行数
-    type: Number,
-    default: undefined
-  },
-  expand: { // 是否启用点击文本展开全部
-    type: Boolean,
-    default: false
-  },
-  tooltip: { // 是否启用文本提示框,可自定义设置弹出提示内容 boolean | slot
-    type: Boolean,
-    default: true
-  },
-})
-const tooltipRef = ref() // tooltip 组件引用
-const observeScroll = ref() // tooltip 组件暴露的 observeScroll 函数
-const showTooltip = ref(false) // 是否显示提示框
-const showExpand = ref(false) // 是否可以启用点击展开
-const expanded = ref(false) // 启用点击展开时,是否展开
-const ellipsisRef = ref() // 文本 DOM 引用
-const computedTooltipMaxWidth = ref() // 计算后的弹出提示最大宽度
-const ellipsisLine = ref() // 行数
-const stopObservation = ref(false)
-const emit = defineEmits(['expandChange'])
-const textMaxWidth = computed(() => {
-  if (typeof props.maxWidth === 'number') {
-    return `${props.maxWidth}px`
-  }
-  return props.maxWidth
-})
-watch(
-  () => props.line,
-  (to) => {
-    if (to !== undefined) {
-      ellipsisLine.value = to
-    } else {
-      ellipsisLine.value = 'none'
-    }
-  },
-  {
-    immediate: true
-  }
-)
-watch(
-  () => [props.maxWidth, props.line, props.tooltip],
-  () => {
-    updateTooltipShow()
-  },
-  {
-    deep: true,
-    flush: 'post'
-  }
-)
-useResizeObserver(ellipsisRef, () => {
-  if (stopObservation.value) {
-    setTimeout(() => {
-      stopObservation.value = false
-    })
-  } else {
-    updateTooltipShow()
-  }
-})
-onMounted(() => {
-  updateTooltipShow()
-  observeScroll.value = tooltipRef.value.observeScroll
-})
-function updateTooltipShow() {
-  const scrollWidth = ellipsisRef.value.scrollWidth
-  const scrollHeight = ellipsisRef.value.scrollHeight
-  const clientWidth = ellipsisRef.value.clientWidth
-  const clientHeight = ellipsisRef.value.clientHeight
-  const offsetWidth = ellipsisRef.value.offsetWidth
-  computedTooltipMaxWidth.value = `${offsetWidth + 24}px`
-  // computedTooltipMaxWidth.value = `100vw`
-  if (scrollWidth > clientWidth || scrollHeight > clientHeight) {
-    if (props.expand) {
-      showExpand.value = true
-    }
-    if (props.tooltip) {
-      showTooltip.value = true
-    }
-  } else {
-    if (props.expand) {
-      showExpand.value = false
-    }
-    if (props.tooltip) {
-      showTooltip.value = false
-    }
-  }
-}
-function onExpand() {
-  stopObservation.value = true
-  if (ellipsisLine.value !== 'none') {
-    ellipsisLine.value = 'none'
-    if (props.tooltip && showTooltip.value) {
-      expanded.value = true
-      tooltipRef.value.hide()
-    }
-    emit('expandChange', true)
-  } else {
-    ellipsisLine.value = props.line ?? 'none'
-    if (props.tooltip && !showTooltip.value) {
-      expanded.value = false
-      showTooltip.value = true
-      tooltipRef.value.show()
-    }
-    emit('expandChange', false)
-  }
-}
-function onAnimationEnd() {
-  if (expanded.value) {
-    showTooltip.value = false
-  }
-}
-defineExpose({
-  observeScroll
-})
-</script>
-<template>
-  <Tooltip
-    ref="tooltipRef"
-    style="cursor: pointer;"
-    :style="`max-width: ${textMaxWidth}`"
-    :max-width="computedTooltipMaxWidth"
-    :content-style="{ maxWidth: textMaxWidth }"
-    :tooltip-style="{ padding: '8px 12px' }"
-    :transition-duration="200"
-    @animationend="onAnimationEnd"
-    v-bind="$attrs"
-  >
-    <template #tooltip>
-      <slot v-if="showTooltip" name="tooltip">
-        <slot></slot>
-      </slot>
-    </template>
-    <div
-      ref="ellipsisRef"
-      class="m-ellipsis"
-      :class="[line ? 'ellipsis-line' : 'not-ellipsis-line', { 'ellipsis-cursor-pointer': showExpand }]"
-      :style="`--ellipsis-max-width: ${textMaxWidth}; --ellipsis-line: ${ellipsisLine};`"
-      @click="showExpand ? onExpand() : () => false"
-    >
-      <slot></slot>
-    </div>
-  </Tooltip>
-</template>
-<style lang="scss" scoped>
-.m-ellipsis {
-  overflow: hidden;
-  cursor: text;
-  max-width: var(--ellipsis-max-width);
-}
-.ellipsis-line {
-  display: -webkit-inline-box;
-  -webkit-box-orient: vertical;
-  -webkit-line-clamp: var(--ellipsis-line);
-}
-.not-ellipsis-line {
-  display: inline-block;
-  vertical-align: bottom;
-  white-space: nowrap;
-  text-overflow: ellipsis;
-}
-.ellipsis-cursor-pointer {
-  cursor: pointer;
-}
-</style>

文件差异内容过多而无法显示
+ 0 - 1
src/components/Enterprise/hotPromoted.vue


+ 0 - 686
src/components/Tooltip/index.vue

@@ -1,686 +0,0 @@
-<!-- vue-amazing-ui组件源码地址 https://github.com/themusecatcher/vue-amazing-ui/blob/main/components/tooltip/Tooltip.vue -->
-
-<script setup>
-import { ref, computed, watch, nextTick, onMounted, onBeforeUnmount } from 'vue'
-// import type { CSSProperties } from 'vue'
-import {
-  useSlotsExist,
-  useMutationObserver,
-  useEventListener,
-  useResizeObserver,
-  rafTimeout,
-  cancelRaf
-} from '@/utils/ellipsis.js'
-
-const props = defineProps({
-  maxWidth: { // 文本最大宽度,单位 px
-    type: [String, Number],
-    default: 240
-  },
-  content: { // 展示的内容 string | slot
-    type: String,
-    default: undefined
-  },
-  contentClass: { // 设置展示内容的类名
-    type: String,
-    default: undefined
-  },
-  contentStyle: { // 设置展示内容的样式
-    type: Object,
-    default: () => {}
-  },
-  tooltip: { // 文字提示内容
-    type: String,
-    default: undefined
-  },
-  tooltipClass: { // 设置文字提示的类名
-    type: String,
-    default: undefined
-  },
-  tooltipStyle: { // 设置文字提示的样式
-    type: Object,
-    default: () => {}
-  },
-  bgColor: { // 文字提示框背景颜色
-    type: String,
-    default: 'rgba(0, 0, 0, 0.85)'
-  },
-  arrow: { // 是否显示箭头
-    type: Boolean,
-    default: false
-  },
-  placement: { // 文字提示位置
-    type: String,
-    default: 'top'
-  },
-  flip: { // 文字提示被浏览器窗口或最近可滚动父元素遮挡时自动调整弹出位置
-    type: Boolean,
-    default: false
-  },
-  trigger: { // 文字提示触发方式 hover/click
-    type: String,
-    default: 'click'
-  },
-  keyboard: { // 是否支持按键操作 (enter 显示;esc 关闭),仅当 trigger: 'click' 时生效
-    type: Boolean,
-    default: false
-  },
-  transitionDuration: { // 文字提示动画的过渡持续时间,单位 ms
-    type: Number,
-    default: 0
-  },
-  showDelay: { // 文字提示显示的延迟时间,单位 ms
-    type: Number,
-    default: 100
-  },
-  hideDelay: { // 文字提示隐藏的延迟时间,单位 ms
-    type: Number,
-    default: 100
-  },
-  show: { // (v-model) 文字提示是否显示
-    type: Boolean,
-    default: false
-  },
-  showControl: { // 只使用 show 属性控制显示隐藏,仅当 trigger: hover 时生效,此时移入移出将不会触发显示隐藏,全部由 show 属性控制
-    type: Boolean,
-    default: false
-  },
-})
-// const tooltip  = ref('')
-const tooltipShow = ref(false) // tooltip 显示隐藏标识
-const tooltipTimer = ref() // tooltip 延迟显示隐藏的定时器标识符
-const scrollTarget = ref(null) // 最近的可滚动父元素
-const top = ref(0) // 提示框 top 定位
-const left = ref(0) // 提示框 left 定位
-const tooltipPlace = ref('top') // 文字提示位置
-const contentRef = ref() // 声明一个同名的模板引用
-const contentWidth = ref(0) // 展示内容宽度
-const contentHeight = ref(0) // 展示内容高度
-const tooltipRef = ref() // tooltip 模板引用
-const tooltipCardRef = ref() // tooltip-card 模板引用
-const tooltipCardWidth = ref(0) // 文字提示内容 tooltip-card 宽度
-const tooltipCardHeight = ref(0) // 文字提示内容 tooltip-card 高度
-const viewportWidth = ref(document.documentElement.clientWidth) // 视口宽度(不包括滚动条)
-const viewportHeight = ref(document.documentElement.clientHeight) // 视口高度(不包括滚动条)
-const emits = defineEmits(['update:show', 'openChange', 'animationend'])
-const slotsExist = useSlotsExist(['tooltip'])
-const tooltipMaxWidth = computed(() => {
-  if (typeof props.maxWidth === 'number') {
-    return `${props.maxWidth}px`
-  }
-  return props.maxWidth
-})
-const showTooltip = computed(() => {
-  return slotsExist.tooltip || props.tooltip
-})
-const tooltipPlacement = computed(() => {
-  switch (tooltipPlace.value) {
-    case 'top':
-      return {
-        transformOrigin: `50% ${top.value}px`,
-        top: `${-top.value}px`,
-        left: `${-left.value}px`
-      }
-    case 'bottom':
-      return {
-        transformOrigin: `50% ${props.arrow ? -4 : -6}px`,
-        bottom: `${-top.value}px`,
-        left: `${-left.value}px`
-      }
-    case 'left':
-      return {
-        transformOrigin: `${left.value}px 50%`,
-        top: `${-top.value}px`,
-        left: `${-left.value}px`
-      }
-    case 'right':
-      return {
-        transformOrigin: `${props.arrow ? -4 : -6}px 50%`,
-        top: `${-top.value}px`,
-        right: `${-left.value}px`
-      }
-    default:
-      return {
-        transformOrigin: `50% ${top.value}px`,
-        top: `${-top.value}px`,
-        left: `${-left.value}px`
-      }
-  }
-})
-watch(
-  () => [props.placement, props.arrow, props.flip],
-  () => {
-    updatePosition()
-  },
-  {
-    deep: true
-  }
-)
-watch(
-  () => props.show,
-  (to) => {
-    if (to && !tooltipShow.value) {
-      onShow()
-    }
-    if (!to && tooltipShow.value) {
-      onHide()
-    }
-  },
-  {
-    immediate: true
-  }
-)
-onMounted(() => {
-  observeScroll()
-})
-onBeforeUnmount(() => {
-  cleanup()
-})
-const mutationObserver = useMutationObserver(
-  scrollTarget,
-  () => {
-    updatePosition()
-  },
-  { subtree: true, childList: true, attributes: true, characterData: true }
-)
-useEventListener(window, 'resize', getViewportSize)
-// 监听 tooltip-card 和 content 的尺寸变化,更新文字提示位置
-useResizeObserver([tooltipCardRef, contentRef], (entries) => {
-  // 排除 tooltip-card 显示过渡动画时的尺寸变化
-  if (entries.length === 1 && entries[0].target.className === 'tooltip-card') {
-    const { blockSize, inlineSize } = entries[0].borderBoxSize[0]
-    if (blockSize === tooltipCardHeight.value && inlineSize === tooltipCardWidth.value) {
-      return
-    }
-  }
-  updatePosition()
-})
-function getViewportSize() {
-  viewportWidth.value = document.documentElement.clientWidth
-  viewportHeight.value = document.documentElement.clientHeight
-  observeScroll() // 窗口尺寸变化时,重新查询并监听最近可滚动父元素
-  updatePosition()
-}
-// 查询并监听最近可滚动父元素
-function observeScroll() {
-  cleanup()
-  scrollTarget.value = getScrollParent(contentRef.value?.parentElement ?? null)
-  scrollTarget.value && scrollTarget.value.addEventListener('scroll', updatePosition)
-  if (scrollTarget.value === document.documentElement) {
-    mutationObserver.start()
-  }
-}
-function cleanup() {
-  scrollTarget.value && scrollTarget.value.removeEventListener('scroll', updatePosition)
-  scrollTarget.value = null
-  mutationObserver.stop()
-}
-// 查询最近的可滚动父元素
-function getScrollParent(el) {
-  const isScrollable = (el) => {
-    const style = window.getComputedStyle(el)
-    if (
-      (el.scrollWidth > el.clientWidth && ['scroll', 'auto'].includes(style.overflowX)) ||
-      (el.scrollHeight > el.clientHeight && ['scroll', 'auto'].includes(style.overflowY)) ||
-      ((el.scrollWidth > el.clientWidth || el.scrollHeight > el.clientHeight) && el === document.documentElement)
-    ) {
-      return true
-    }
-    return false
-  }
-  if (el) {
-    return isScrollable(el) ? el : getScrollParent(el.parentElement ?? null)
-  }
-  return null
-}
-// 更新文字提示位置
-function updatePosition() {
-  tooltipShow.value && getPosition()
-}
-
-// 计算文字提示位置
-async function getPosition() {
-  await nextTick()
-  contentWidth.value = contentRef.value.offsetWidth
-  contentHeight.value = contentRef.value.offsetHeight
-  tooltipCardWidth.value = tooltipCardRef.value.offsetWidth
-  tooltipCardHeight.value = tooltipCardRef.value.offsetHeight
-  // debugger
-  // offsetTop
-  if (props.flip) {
-    tooltipPlace.value = getPlacement()
-  }
-  if (['top', 'bottom'].includes(tooltipPlace.value)) {
-    top.value = tooltipCardHeight.value + (props.arrow ? 4 + 12 : 6)
-    // left.value = ((tooltipCardWidth.value - contentWidth.value) / 2) + contentWidth.value.offsetLeft
-    left.value = (tooltipCardWidth.value - contentWidth.value) / 2
-  } else {
-    top.value = (tooltipCardHeight.value - contentHeight.value) / 2
-    left.value = tooltipCardWidth.value + (props.arrow ? 4 + 12 : 6)
-  }
-}
-// 获取可滚动父元素或视口的矩形信息
-function getShelterRect() {
-  if (scrollTarget.value && scrollTarget.value !== document.documentElement) {
-    return scrollTarget.value.getBoundingClientRect()
-  }
-  return {
-    top: 0,
-    left: 0,
-    bottom: viewportHeight.value,
-    right: viewportWidth.value
-  }
-}
-// 文字提示被浏览器窗口或最近可滚动父元素遮挡时自动调整弹出位置
-function getPlacement() {
-  const { top, bottom, left, right } = contentRef.value.getBoundingClientRect() // 内容元素各边缘相对于浏览器视口的位置(不包括滚动条)
-  const { top: targetTop, bottom: targetBottom, left: targetLeft, right: targetRight } = getShelterRect() // 滚动元素或视口各边缘相对于浏览器视口的位置(不包括滚动条)
-  const topDistance = top - targetTop - (props.arrow ? 12 : 0) // 内容元素上边缘距离滚动元素上边缘的距离
-  const bottomDistance = targetBottom - bottom - (props.arrow ? 12 : 0) // 内容元素下边缘距离动元素下边缘的距离
-  const leftDistance = left - targetLeft - (props.arrow ? 12 : 0) // 内容元素左边缘距离滚动元素左边缘的距离
-  const rightDistance = targetRight - right - (props.arrow ? 12 : 0) // 内容元素右边缘距离滚动元素右边缘的距离
-  const horizontalDistance = (tooltipCardWidth.value - contentWidth.value) / 2 // 水平方向容纳文字提示需要的最小宽度
-  const verticalDistance = (tooltipCardHeight.value - contentHeight.value) / 2 // 垂直方向容纳文字提示需要的最小高度
-  return findPlace(props.placement, [])
-  // 查询满足条件的 place,如果没有,则返回默认值
-  function findPlace(place, disabledPlaces) {
-    if (place === 'top') {
-      if (!disabledPlaces.includes('top')) {
-        if (topDistance < tooltipCardHeight.value + (props.arrow ? 4 : 6) && disabledPlaces.length !== 3) {
-          return findPlace('bottom', [...disabledPlaces, 'top'])
-        } else {
-          if (leftDistance >= horizontalDistance && rightDistance >= horizontalDistance) {
-            return 'top'
-          } else {
-            if (disabledPlaces.length !== 3) {
-              if (leftDistance >= horizontalDistance) {
-                return findPlace('left', ['top', 'bottom', 'right'])
-              } else if (rightDistance >= horizontalDistance) {
-                return findPlace('right', ['top', 'bottom', 'left'])
-              }
-            }
-          }
-        }
-      } else {
-        if (!disabledPlaces.includes('bottom')) {
-          return findPlace('bottom', disabledPlaces)
-        } else if (!disabledPlaces.includes('left')) {
-          return findPlace('left', disabledPlaces)
-        } else {
-          return findPlace('right', disabledPlaces)
-        }
-      }
-    } else if (place === 'bottom') {
-      if (!disabledPlaces.includes('bottom')) {
-        if (bottomDistance < tooltipCardHeight.value + (props.arrow ? 4 : 6) && disabledPlaces.length !== 3) {
-          return findPlace('top', [...disabledPlaces, 'bottom'])
-        } else {
-          if (leftDistance >= horizontalDistance && rightDistance >= horizontalDistance) {
-            return 'bottom'
-          } else {
-            if (disabledPlaces.length !== 3) {
-              if (leftDistance >= horizontalDistance) {
-                return findPlace('left', ['top', 'bottom', 'right'])
-              } else if (rightDistance >= horizontalDistance) {
-                return findPlace('right', ['top', 'bottom', 'left'])
-              }
-            }
-          }
-        }
-      } else {
-        if (!disabledPlaces.includes('top')) {
-          return findPlace('top', disabledPlaces)
-        } else if (!disabledPlaces.includes('left')) {
-          return findPlace('left', disabledPlaces)
-        } else {
-          return findPlace('right', disabledPlaces)
-        }
-      }
-    } else if (place === 'left') {
-      if (!disabledPlaces.includes('left')) {
-        if (leftDistance < tooltipCardWidth.value + (props.arrow ? 4 : 6) && disabledPlaces.length !== 3) {
-          return findPlace('right', [...disabledPlaces, 'left'])
-        } else {
-          if (topDistance >= verticalDistance && bottomDistance >= verticalDistance) {
-            return 'left'
-          } else {
-            if (disabledPlaces.length !== 3) {
-              if (topDistance >= verticalDistance) {
-                return findPlace('top', ['left', 'right', 'bottom'])
-              } else if (bottomDistance >= verticalDistance) {
-                return findPlace('bottom', ['left', 'right', 'top'])
-              }
-            }
-          }
-        }
-      } else {
-        if (!disabledPlaces.includes('right')) {
-          return findPlace('right', disabledPlaces)
-        } else if (!disabledPlaces.includes('top')) {
-          return findPlace('top', disabledPlaces)
-        } else {
-          return findPlace('bottom', disabledPlaces)
-        }
-      }
-    } else if (place === 'right') {
-      if (!disabledPlaces.includes('right')) {
-        if (rightDistance < tooltipCardWidth.value + (props.arrow ? 4 : 6) && disabledPlaces.length !== 3) {
-          return findPlace('left', [...disabledPlaces, 'right'])
-        } else {
-          if (topDistance >= verticalDistance && bottomDistance >= verticalDistance) {
-            return 'right'
-          } else {
-            if (disabledPlaces.length !== 3) {
-              if (topDistance >= verticalDistance) {
-                return findPlace('top', ['left', 'right', 'bottom'])
-              } else if (bottomDistance >= verticalDistance) {
-                return findPlace('bottom', ['left', 'right', 'top'])
-              }
-            }
-          }
-        }
-      } else {
-        if (!disabledPlaces.includes('left')) {
-          return findPlace('left', disabledPlaces)
-        } else if (!disabledPlaces.includes('top')) {
-          return findPlace('top', disabledPlaces)
-        } else {
-          return findPlace('bottom', disabledPlaces)
-        }
-      }
-    }
-    return props.placement
-  }
-}
-function onShow() {
-  tooltipTimer.value && cancelRaf(tooltipTimer.value)
-  if (!tooltipShow.value) {
-    tooltipTimer.value = rafTimeout(() => {
-      tooltipShow.value = true
-      getPosition()
-      emits('update:show', true)
-      emits('openChange', true)
-      if (showTooltip.value && props.trigger === 'click') {
-        document.addEventListener('click', handleClick)
-      }
-    }, props.showDelay)
-  }
-}
-function onHide() {
-  tooltipTimer.value && cancelRaf(tooltipTimer.value)
-  if (tooltipShow.value) {
-    tooltipTimer.value = rafTimeout(() => {
-      tooltipShow.value = false
-      emits('update:show', false)
-      emits('openChange', false)
-      if (showTooltip.value && props.trigger === 'click') {
-        document.removeEventListener('click', handleClick)
-      }
-    }, props.hideDelay)
-  }
-}
-function toggleVisible() {
-  if (!tooltipShow.value) {
-    onShow()
-  } else {
-    onHide()
-  }
-}
-function handleClick(e) {
-  if (!tooltipRef.value.contains(e.target)) {
-    onHide()
-  }
-}
-function onEnterWrap() {
-  if (showTooltip.value && props.trigger === 'hover' && !props.showControl) {
-    onShow()
-  }
-}
-function onLeaveWrap() {
-  if (showTooltip.value && props.trigger === 'hover' && !props.showControl) {
-    onHide()
-  }
-}
-function onAnimationEnd() {
-  emits('animationend', tooltipShow.value)
-}
-function onEnterTooltip() {
-  if (props.trigger === 'hover' && !props.showControl) {
-    onShow()
-  }
-}
-function onLeaveTooltip() {
-  if (props.trigger === 'hover' && !props.showControl) {
-    onHide()
-  }
-}
-defineExpose({
-  show: onShow,
-  hide: onHide,
-  observeScroll
-})
-</script>
-<template>
-  <div
-    class="m-tooltip-wrap"
-    :style="`--tooltip-max-width: ${tooltipMaxWidth}; --tooltip-background-color: ${props.bgColor}; --transition-duration: ${props.transitionDuration}ms;`"
-    @mouseenter="onEnterWrap"
-    @mouseleave="onLeaveWrap"
-  >
-    <Transition
-      name="zoom"
-      enter-from-class="zoom-enter"
-      enter-active-class="zoom-enter"
-      enter-to-class="zoom-enter zoom-enter-active"
-      leave-from-class="zoom-leave"
-      leave-active-class="zoom-leave zoom-leave-active"
-      leave-to-class="zoom-leave zoom-leave-active"
-      @animationend="onAnimationEnd"
-    >
-      <div
-        v-show="showTooltip && tooltipShow"
-        ref="tooltipRef"
-        class="m-tooltip-card"
-        :class="{ [`tooltip-${tooltipPlace}-padding`]: props.arrow }"
-        :style="tooltipPlacement"
-        @mouseenter="onEnterTooltip"
-        @mouseleave="onLeaveTooltip"
-        @keydown.esc="props.trigger === 'click' && props.keyboard && tooltipShow ? onHide() : () => false"
-      >
-        <div ref="tooltipCardRef" class="tooltip-card" :class="props.tooltipClass" :style="props.tooltipStyle">
-          <slot name="tooltip">{{ props.tooltip }}</slot>
-        </div>
-        <div v-if="props.arrow" class="tooltip-arrow" :class="`arrow-${tooltipPlace || 'top'}`"></div>
-      </div>
-    </Transition>
-    <span
-      ref="contentRef"
-      class="tooltip-content"
-      :class="props.contentClass"
-      :style="props.contentStyle"
-      @click="showTooltip && props.trigger === 'click' && !tooltipShow ? onShow() : () => false"
-      @keydown.enter="showTooltip && props.trigger === 'click' && props.keyboard ? toggleVisible() : () => false"
-      @keydown.esc="showTooltip && props.trigger === 'click' && props.keyboard && tooltipShow ? onHide() : () => false"
-    >
-      <slot>{{ props.content }}</slot>
-    </span>
-  </div>
-</template>
-<style lang="scss" scoped>
-.zoom-enter {
-  transform: none;
-  opacity: 0;
-  animation-duration: var(--transition-duration);
-  animation-fill-mode: both;
-  animation-timing-function: cubic-bezier(0.08, 0.82, 0.17, 1);
-  animation-play-state: paused;
-}
-.zoom-enter-active {
-  animation-name: zoomIn;
-  animation-play-state: running;
-  @keyframes zoomIn {
-    0% {
-      transform: scale(0.8);
-      opacity: 0;
-    }
-    100% {
-      transform: scale(1);
-      opacity: 1;
-    }
-  }
-}
-.zoom-leave {
-  animation-duration: var(--transition-duration);
-  animation-fill-mode: both;
-  animation-play-state: paused;
-  animation-timing-function: cubic-bezier(0.78, 0.14, 0.15, 0.86);
-}
-.zoom-leave-active {
-  animation-name: zoomOut;
-  animation-play-state: running;
-  pointer-events: none;
-  @keyframes zoomOut {
-    0% {
-      transform: scale(1);
-      opacity: 1;
-    }
-    100% {
-      transform: scale(0.8);
-      opacity: 0;
-    }
-  }
-}
-.m-tooltip-wrap {
-  position: relative;
-  display: inline-block;
-  .m-tooltip-card {
-    position: absolute;
-    z-index: 999;
-    width: max-content;
-    outline: none;
-    .tooltip-card {
-      min-width: 32px;
-      max-width: var(--tooltip-max-width);
-      min-height: 32px;
-      padding: 6px 8px;
-      font-size: 14px;
-      color: #fff;
-      line-height: 1.5714285714285714;
-      text-align: justify;
-      text-decoration: none;
-      word-break: break-all;
-      background-color: var(--tooltip-background-color);
-      border-radius: 6px;
-      box-shadow:
-        0 6px 16px 0 rgba(0, 0, 0, 0.08),
-        0 3px 6px -4px rgba(0, 0, 0, 0.12),
-        0 9px 28px 8px rgba(0, 0, 0, 0.05);
-      :deep(svg) {
-        fill: currentColor;
-      }
-    }
-    .tooltip-arrow {
-      position: absolute;
-      z-index: 9;
-      display: block;
-      pointer-events: none;
-      width: 16px;
-      height: 16px;
-      overflow: hidden;
-      &::before {
-        position: absolute;
-        width: 16px;
-        height: 8px;
-        background-color: var(--tooltip-background-color);
-        clip-path: path(
-          'M 0 8 A 4 4 0 0 0 2.82842712474619 6.82842712474619 L 6.585786437626905 3.0710678118654755 A 2 2 0 0 1 9.414213562373096 3.0710678118654755 L 13.17157287525381 6.82842712474619 A 4 4 0 0 0 16 8 Z'
-        );
-        content: '';
-      }
-      &::after {
-        position: absolute;
-        width: 8.970562748477143px;
-        height: 8.970562748477143px;
-        margin: auto;
-        border-radius: 0 0 2px 0;
-        transform: translateY(50%) rotate(-135deg);
-        box-shadow: 3px 3px 7px rgba(0, 0, 0, 0.1);
-        z-index: 0;
-        background: transparent;
-        content: '';
-      }
-    }
-    .arrow-top {
-      left: 50%;
-      bottom: 12px;
-      transform: translateX(-50%) translateY(100%) rotate(180deg);
-      &::before {
-        bottom: 0;
-        left: 0;
-      }
-      &::after {
-        bottom: 0;
-        left: 0;
-        right: 0;
-      }
-    }
-    .arrow-bottom {
-      left: 50%;
-      top: 12px;
-      transform: translateX(-50%) translateY(-100%) rotate(0deg);
-      &::before {
-        bottom: 0;
-        left: 0;
-      }
-      &::after {
-        bottom: 0;
-        left: 0;
-        right: 0;
-      }
-    }
-    .arrow-left {
-      top: 50%;
-      right: 12px;
-      transform: translateX(100%) translateY(-50%) rotate(90deg);
-      &::before {
-        bottom: 0;
-        left: 0;
-      }
-      &::after {
-        bottom: 0;
-        left: 0;
-        right: 0;
-      }
-    }
-    .arrow-right {
-      top: 50%;
-      left: 12px;
-      transform: translateX(-100%) translateY(-50%) rotate(-90deg);
-      &::before {
-        bottom: 0;
-        left: 0;
-      }
-      &::after {
-        bottom: 0;
-        left: 0;
-        right: 0;
-      }
-    }
-  }
-  .tooltip-top-padding {
-    padding-bottom: 12px;
-  }
-  .tooltip-bottom-padding {
-    padding-top: 12px;
-  }
-  .tooltip-left-padding {
-    padding-right: 12px;
-  }
-  .tooltip-right-padding {
-    padding-left: 12px;
-  }
-  .tooltip-content {
-    display: inline-block;
-  }
-}
-</style>

+ 3 - 3
src/directives/ellipsisTooltip/index.js

@@ -29,8 +29,8 @@ function calculationLocation(el, target, placements) {
     let target_dom = target.getBoundingClientRect()
     if (placements === "left") {
         el.tooltipPosition.x = 0
-        el.tooltipPosition.y = target_dom.y - target_dom.height / 2
-        el.tooltipPosition.right = windowWidth - target_dom.left + target_dom.height / 2
+        el.tooltipPosition.y = target_dom.y - target_dom.height / 2 + 14
+        el.tooltipPosition.right = windowWidth - target_dom.left + target_dom.height / 2 - 22
     } else if (placements === "left-right") {
         el.tooltipPosition.x = 0
         el.tooltipPosition.y = target_dom.y - (el_dom?.height || 0) / 2 + target_dom.height / 2
@@ -59,7 +59,7 @@ function calculationLocation(el, target, placements) {
         el.arrowPosition.top = target_dom.height
     } else if (placements === "top") {
         el.tooltipPosition.x = target_dom.x + target_dom.width / 2 - (el_dom?.width || 0) / 2
-        el.tooltipPosition.y = target_dom.y - target_dom.height - 20
+        el.tooltipPosition.y = target_dom.y - target_dom.height - 11
         if (el.tooltipPosition.x > (windowWidth-el_dom.width) && target_dom.left + el_dom.width > windowWidth) {
             el.tooltipPosition.x = windowWidth - el_dom.width
             el.arrowPosition.left = (el_dom.width / 10) * 9

+ 13 - 10
src/directives/ellipsisTooltip/index.vue

@@ -4,7 +4,7 @@
     <div v-show="tooltipShow" :style="tooltipStyle" class="wq-tooltip"
     >
       <span class="wq-tooltip-text" v-text="text"></span>
-      <div :class="[placements]" :style="arrowStyle" class="wq-tooltip-arrow"></div>
+      <div v-show="showArrow" :class="[placements]" :style="arrowStyle" class="wq-tooltip-arrow"></div>
     </div>
   </transition>
 </template>
@@ -13,10 +13,12 @@ import {computed, ref} from 'vue'
 
 export default {
   setup() {
-
     // 显示弹框
     const tooltipShow = ref(false);
 
+    // 显示箭头
+    const showArrow = ref(false);
+
     // 提示内容
     const text = ref()
 
@@ -64,14 +66,15 @@ export default {
       }
     })
     const arrowStyle = computed(() => {
-      return {
+      return showArrow.value ? {
         left: arrowPosition.value.left?`${arrowPosition.value.left}px` : null,
         top: arrowPosition.value.top?`${arrowPosition.value.top}px` : null,
         right: arrowPosition.value.right?`${arrowPosition.value.right}px` : null,
         bottom: arrowPosition.value.bottom?`${arrowPosition.value.bottom}px` : null,
-      }
+      } : {}
     })
     return {
+      showArrow,
       tooltipShow,
       showTip,
       hiddenTip,
@@ -90,19 +93,19 @@ export default {
 <style lang="scss" scoped>
 // tooltip
 .wq-tooltip {
-  padding: 10px;
-  font-size: 12px;
-  line-height: 1.2;
+  padding: 5px 16px;
+  font-size: 14px;
+  line-height: 1.6;
   min-width: 30px;
   max-width: fit-content;
   word-wrap: break-word;
   position: fixed;
   top: 0;
-  background: #303133;
-  color: #fff;
+  background: rgb(66,66,66);
+  color: rgb(238,238,238);
   z-index: 1000;
   display: block;
-  border-radius: 8px;
+  border-radius: 4px;
   font-weight: 500;
   pointer-events: none;
 }

+ 0 - 1
src/main.js

@@ -31,7 +31,6 @@ import VueLuckyCanvas from '@lucky-canvas/vue'
 
 import router from './router'
 
-// import { ellipsisTooltip } from '@/components/CtTooltip/index.js'
 import { ellipsisTooltip } from '@/directives/ellipsisTooltip/index.js'
 // import imageDirective from '@/directives/previewImageDirective'
 

部分文件因为文件数量过多而无法显示