Maison >interface Web >Voir.js >Comment implémenter des tables copiables avec Vue3
L'encapsulation de table la plus basique permet aux utilisateurs de se concentrer uniquement sur les données de ligne et de colonne, sans prêter attention à la structure DOM
. Nous pouvons nous référer à . AntDesign
, columns
dataSource
Ces deux attributs sont essentiels Le code est le suivant : DOM
结构是怎样的,我们可以参考 AntDesign
,columns
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('copy') } // 将复制方法暴露出去以供父组件可以直接调用 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('copy') // 别忘了把状态重置回来 copying.value = false }) } expose({ copy }) return (() => { return ( // ... ) }) as unknown as { copy: typeof copy } } })Ce qui nécessite une attention particulière, c'est
ColumnslotName
dans /code>, qui doit pouvoir personnaliser le contenu de la colonne qui doit être restituée (dans AntDesign
, c'est via TableColumn code > Implémenté par le composant, ici <code>slotName
est utilisé directement pour plus de commodité). Implémentez la fonction de copieTout d'abord, nous pouvons sélectionner manuellement la table à copier et l'essayer. Nous avons constaté que la table prend en charge la sélection et la copie, l'idée d'implémentation est donc très simple. puis exécutez la commande de copie. Le code est le suivant :
<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 'vue' import { Table as CTable } from '../components' import arrowUpIcon from '../assets/arrow-up.svg' const columns = [ { title: '序号', dataIndex: 'serial' }, { title: '班级', dataIndex: 'class' }, { title: '姓名', dataIndex: 'name' }, { title: '状态', dataIndex: 'status', slotName: 'status', slotNameOnCopy: 'statusOnCopy' } ] const dataSource = [ { serial: 1, class: '三年级1班', name: '张三' }, { serial: 2, class: '三年级2班', name: '李四' }, { serial: 3, class: '三年级3班', name: '王五' }, { serial: 4, class: '三年级4班', name: '赵六' }, { serial: 5, class: '三年级5班', name: '宋江' }, { serial: 6, class: '三年级6班', name: '卢俊义' }, { serial: 7, class: '三年级7班', name: '吴用' }, { serial: 8, class: '三年级8班', name: '公孙胜' }, ] const table = ref<InstanceType<typeof CTable> | null>(null) const handleCopy = () => { table.value?.copy() } </script> <style scoped> .status-icon { width: 20px; height: 20px; } </style>De cette façon, la fonction de copie est terminée. Il n'est pas nécessaire de faire attention à la façon de copier en externe. Il vous suffit d'appeler le
copy<.> méthode exposée par le composant. <h3></h3>Gestion des éléments non copiables dans le tableau<p></p>Bien que la fonction de copie soit très simple, il s'agit uniquement de copier du texte. S'il y a des éléments non copiables (tels que des images) dans le tableau, ceux-ci doivent être remplacés par ceux correspondants. symboles de texte lors de la copie, comment y parvenir ? <p></p>La solution consiste à définir un état de copie à l'intérieur du composant, à définir l'état sur copie lors de l'appel de la méthode de copie et à restituer un contenu différent en fonction de cet état (rendre les images dans l'état de non-copie et restituer les symboles de texte correspondants dans l'état de copie). état de copie), code Comme suit : 🎜<pre class="brush:js;">import { defineComponent, ref, nextTick } from &#39;vue&#39;
import type { PropType } from &#39;vue&#39;
interface Column {
title: string;
dataIndex: string;
slotName?: string;
slotNameOnCopy?: 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, expose }) {
const tableRef = ref<HTMLTableElement | null>(null)
const copying = ref(false)
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,
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(() => {
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 (
<table ref={tableRef}>
<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, slotNameOnCopy } = column
const text = record[dataIndex]
return (
<td key={dataIndex}>
{getTdContent(text, record, i, slotName, slotNameOnCopy)}
</td>
)
})}
</tr>
)
})}
</table>
)
}) as unknown as { copy: typeof copy }
}
})</pre>🎜Test🎜🎜Enfin nous pouvons écrire une démo pour tester si la fonction est normale Le code est le suivant : 🎜rrreee🎜Ci-joint le code complet : 🎜rrreee.</.>
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!