>웹 프론트엔드 >View.js >Vue3으로 복사 가능한 테이블을 구현하는 방법

Vue3으로 복사 가능한 테이블을 구현하는 방법

WBOY
WBOY앞으로
2023-05-11 20:19:121658검색

가장 기본적인 테이블 캡슐화

가장 기본적인 테이블 캡슐화는 사용자가 DOM 구조에 신경쓰지 않고 행과 열 데이터에만 집중할 수 있도록 하는 것입니다. AntDesign, columns dataSource 이 두 가지 속성은 필수적입니다. 코드는 다음과 같습니다. DOM 结构是怎样的,我们可以参考 AntDesigncolumns dataSource 这两个属性是必不可少的,代码如下:

import { defineComponent } from 'vue'
import type { PropType } from 'vue'

interface Column {
  title: string;
  dataIndex: string;
  slotName?: string;
}
type TableRecord = Record<string, unknown>;

export const Table = defineComponent({
  props: {
    columns: {
      type: Array as PropType<Column[]>,
      required: true,
    },
    dataSource: {
      type: Array as PropType<TableRecord[]>,
      default: () => [],
    },
    rowKey: {
      type: Function as PropType<(record: TableRecord) => string>,
    }
  },
  setup(props, { slots }) {
    const getRowKey = (record: TableRecord, index: number) => {
      if (props.rowKey) {
        return props.rowKey(record)
      }
      return record.id ? String(record.id) : String(index)
    }
    const getTdContent = (
      text: any,
      record: TableRecord,
      index: number,
      slotName?: string
    ) => {
      if (slotName) {
        return slots[slotName]?.(text, record, index)
      }
      return text
    }

    return () => {
      return (
        <table>
          <tr>
            {props.columns.map(column => {
              const { title, dataIndex } = column
              return <th key={dataIndex}>{title}</th>
            })}
          </tr>
          {props.dataSource.map((record, index) => {
            return (
              <tr key={getRowKey(record, index)}>
                {props.columns.map((column, i) => {
                  const { dataIndex, slotName } = column
                  const text = record[dataIndex]

                  return (
                    <td key={dataIndex}>
                      {getTdContent(text, record, i, slotName)}
                    </td>
                  )
                })}
              </tr>
            )
          })}
        </table>
      )
    }
  }
})

需要关注一下的是 Column 中有一个 slotName 属性,这是为了能够自定义该列的所需要渲染的内容(在 AntDesign 中是通过 TableColumn 组件实现的,这里为了方便直接使用 slotName)。

实现复制功能

首先我们可以手动选中表格复制尝试一下,发现表格是支持选中复制的,那么实现思路也就很简单了,通过代码选中表格再执行复制命令就可以了,代码如下:

export const Table = defineComponent({
  props: {
      // ...
  },
  setup(props, { slots, expose }) {
    // 新增,存储table节点
    const tableRef = ref<HTMLTableElement | null>(null)

    // ...
    
    // 复制的核心方法
    const copy = () => {
      if (!tableRef.value) return

      const range = document.createRange()
      range.selectNode(tableRef.value)
      const selection = window.getSelection()
      if (!selection) return

      if (selection.rangeCount > 0) {
        selection.removeAllRanges()
      }
      selection.addRange(range)
      document.execCommand(&#39;copy&#39;)
    }
    
    // 将复制方法暴露出去以供父组件可以直接调用
    expose({ copy })

    return (() => {
      return (
        // ...
      )
    }) as unknown as { copy: typeof copy } // 这里是为了让ts能够通过类型校验,否则调用`copy`方法ts会报错
  }
})

这样复制功能就完成了,外部是完全不需要关注如何复制的,只需要调用组件暴露出去的 copy

export const Table = defineComponent({
  props: {
      // ...
  },
  setup(props, { slots, expose }) {
    const tableRef = ref<HTMLTableElement | null>(null)
    // 新增,定义复制状态
    const copying = ref(false)

    // ...
    const getTdContent = (
      text: any,
      record: TableRecord,
      index: number,
      slotName?: string,
      slotNameOnCopy?: string
    ) => {
      // 如果处于复制状态,则渲染复制状态下的内容
      if (copying.value && slotNameOnCopy) {
        return slots[slotNameOnCopy]?.(text, record, index)
      }

      if (slotName) {
        return slots[slotName]?.(text, record, index)
      }
      return text
    }
    
    const copy = () => {
      copying.value = true
      // 将复制行为放到 nextTick 保证复制到正确的内容
      nextTick(() => {
        if (!tableRef.value) return
  
        const range = document.createRange()
        range.selectNode(tableRef.value)
        const selection = window.getSelection()
        if (!selection) return
  
        if (selection.rangeCount > 0) {
          selection.removeAllRanges()
        }
        selection.addRange(range)
        document.execCommand(&#39;copy&#39;)
        
        // 别忘了把状态重置回来
        copying.value = false
      })
    }
    
    expose({ copy })

    return (() => {
      return (
        // ...
      )
    }) as unknown as { copy: typeof copy }
  }
})

주의가 필요한 것은 Column의 code>slotName 속성(AntDesign에서는 TableColumn을 통해 이루어짐) code > 구성요소에 의해 구현됩니다. 여기서는 편의상 <code>slotName을 직접 사용합니다.)

복사 기능 구현

우선 복사할 테이블을 수동으로 선택하여 사용해 볼 수 있습니다. 테이블이 선택 및 복사를 지원하므로 구현 아이디어는 매우 간단합니다. 그런 다음 복사 명령을 실행하면 다음과 같습니다.

<template>
  <button @click="handleCopy">点击按钮复制表格</button>
  <c-table
    :columns="columns"
    :data-source="dataSource"
    border="1"
    
    ref="table"
  >
    <template #status>
      <img class="status-icon" :src="arrowUpIcon" />
    </template>
    <template #statusOnCopy>
      →
    </template>
  </c-table>
</template>

<script setup lang="ts">
import { ref } from &#39;vue&#39;
import { Table as CTable } from &#39;../components&#39;
import arrowUpIcon from &#39;../assets/arrow-up.svg&#39;

const columns = [
  { title: &#39;序号&#39;, dataIndex: &#39;serial&#39; },
  { title: &#39;班级&#39;, dataIndex: &#39;class&#39; },
  { title: &#39;姓名&#39;, dataIndex: &#39;name&#39; },
  { title: &#39;状态&#39;, dataIndex: &#39;status&#39;, slotName: &#39;status&#39;, slotNameOnCopy: &#39;statusOnCopy&#39; }
]

const dataSource = [
  { serial: 1, class: &#39;三年级1班&#39;, name: &#39;张三&#39; },
  { serial: 2, class: &#39;三年级2班&#39;, name: &#39;李四&#39; },
  { serial: 3, class: &#39;三年级3班&#39;, name: &#39;王五&#39; },
  { serial: 4, class: &#39;三年级4班&#39;, name: &#39;赵六&#39; },
  { serial: 5, class: &#39;三年级5班&#39;, name: &#39;宋江&#39; },
  { serial: 6, class: &#39;三年级6班&#39;, name: &#39;卢俊义&#39; },
  { serial: 7, class: &#39;三年级7班&#39;, name: &#39;吴用&#39; },
  { serial: 8, class: &#39;三年级8班&#39;, name: &#39;公孙胜&#39; },
]

const table = ref<InstanceType<typeof CTable> | null>(null)
const handleCopy = () => {
  table.value?.copy()
}
</script>

<style scoped>
.status-icon {
  width: 20px;
  height: 20px;
}
</style>

이렇게 하면 복사 기능이 외부에서 어떻게 복사되는지 신경 쓸 필요 없이 copy 메소드는 구성요소에 의해 노출됩니다. <h3></h3>테이블에서 복사할 수 없는 요소를 처리하세요<p></p>복사 기능은 매우 간단하지만 텍스트만 복사하는 기능입니다. 테이블에 복사할 수 없는 요소(예: 그림)가 있는 경우 이를 해당 항목으로 바꿔야 합니다. 복사할 때 텍스트 기호를 사용하는 방법은 무엇입니까? <p></p>해결책은 구성 요소 내부에 복사 상태를 정의하고, 복사 메서드 호출 시 상태를 복사로 설정하고, 이 상태에 따라 다른 콘텐츠를 렌더링하는 것입니다(비복사 상태에서 이미지를 렌더링하고, 해당 텍스트 기호를 🎜<pre class="brush:js;">import { defineComponent, ref, nextTick } from &amp;#39;vue&amp;#39; import type { PropType } from &amp;#39;vue&amp;#39; interface Column { title: string; dataIndex: string; slotName?: string; slotNameOnCopy?: string; } type TableRecord = Record&lt;string, unknown&gt;; export const Table = defineComponent({ props: { columns: { type: Array as PropType&lt;Column[]&gt;, required: true, }, dataSource: { type: Array as PropType&lt;TableRecord[]&gt;, default: () =&gt; [], }, rowKey: { type: Function as PropType&lt;(record: TableRecord) =&gt; string&gt;, } }, setup(props, { slots, expose }) { const tableRef = ref&lt;HTMLTableElement | null&gt;(null) const copying = ref(false) const getRowKey = (record: TableRecord, index: number) =&gt; { if (props.rowKey) { return props.rowKey(record) } return record.id ? String(record.id) : String(index) } const getTdContent = ( text: any, record: TableRecord, index: number, slotName?: string, slotNameOnCopy?: string ) =&gt; { if (copying.value &amp;&amp; slotNameOnCopy) { return slots[slotNameOnCopy]?.(text, record, index) } if (slotName) { return slots[slotName]?.(text, record, index) } return text } const copy = () =&gt; { copying.value = true nextTick(() =&gt; { if (!tableRef.value) return const range = document.createRange() range.selectNode(tableRef.value) const selection = window.getSelection() if (!selection) return if (selection.rangeCount &gt; 0) { selection.removeAllRanges() } selection.addRange(range) document.execCommand(&amp;#39;copy&amp;#39;) copying.value = false }) } expose({ copy }) return (() =&gt; { return ( &lt;table ref={tableRef}&gt; &lt;tr&gt; {props.columns.map(column =&gt; { const { title, dataIndex } = column return &lt;th key={dataIndex}&gt;{title}&lt;/th&gt; })} &lt;/tr&gt; {props.dataSource.map((record, index) =&gt; { return ( &lt;tr key={getRowKey(record, index)}&gt; {props.columns.map((column, i) =&gt; { const { dataIndex, slotName, slotNameOnCopy } = column const text = record[dataIndex] return ( &lt;td key={dataIndex}&gt; {getTdContent(text, record, i, slotName, slotNameOnCopy)} &lt;/td&gt; ) })} &lt;/tr&gt; ) })} &lt;/table&gt; ) }) as unknown as { copy: typeof copy } } })</pre>🎜Test🎜🎜마지막으로 함수가 정상인지 테스트하는 데모를 작성할 수 있습니다. 코드는 다음과 같습니다. 🎜rrreee🎜첨부된 코드는 전체 코드입니다. 🎜rrreee

위 내용은 Vue3으로 복사 가능한 테이블을 구현하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 yisu.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제