|
@@ -0,0 +1,452 @@
|
|
|
+<!--
|
|
|
+ * @Author: fhj
|
|
|
+ * @LastEditors: xianing
|
|
|
+ * @Description: f-表格组件
|
|
|
+-->
|
|
|
+<template>
|
|
|
+ <div class="f-table" :class="[
|
|
|
+ full ? 'max-area' : '',
|
|
|
+ classText
|
|
|
+ ]" @mouseup="mouseup" @mousedown="mousedown">
|
|
|
+ <el-table
|
|
|
+ :id="`fTable${tableId}`"
|
|
|
+ :key="tableKey"
|
|
|
+ ref="fTable"
|
|
|
+ v-bind="$attrs"
|
|
|
+ :data="dataList"
|
|
|
+ v-loading="loading"
|
|
|
+ :style="'width: 100%'"
|
|
|
+ :height="full ? '100%' : height || null"
|
|
|
+ :tree-props="treeProps"
|
|
|
+ :row-key="$attrs.rowKey || $attrs['row-key'] || ''"
|
|
|
+ v-on="listenersStatus ? $listeners : ''"
|
|
|
+ @select="$comSelect"
|
|
|
+ @select-all="$comSelectAll"
|
|
|
+ @current-change="$comRowChange"
|
|
|
+ @sort-change="sortChange"
|
|
|
+ >
|
|
|
+ <template slot="empty" class="flex-ju-al-center"> 暂无数据 </template>
|
|
|
+ <!-- checkbox -->
|
|
|
+ <el-table-column v-if="checkbox" type="selection" width="60" v-bind="{ ...selectionProps }"
|
|
|
+ :selectable="selectable" align="center" class-name="noTips table-checkbox" fixed />
|
|
|
+
|
|
|
+ <!-- 单选行样式 -->
|
|
|
+ <el-table-column v-if="highLightCurrentRow" width="60" align="center"
|
|
|
+ class-name="noTips table-checkbox" fixed>
|
|
|
+ <template slot-scope="scope">
|
|
|
+ <el-radio v-model="currentRow[selectKey]" :label="scope.row[selectKey]" class="noLabel"
|
|
|
+ @click.native.stop="radioClick($event, scope.row)" />
|
|
|
+ </template>
|
|
|
+ </el-table-column>
|
|
|
+
|
|
|
+ <!-- 树级样式 -->
|
|
|
+ <el-table-column v-if="$attrs.hasChildren" class-name="hasChildren" width="80px" />
|
|
|
+
|
|
|
+ <!-- 是否显示expand -->
|
|
|
+ <el-table-column v-if="showExpand" type="expand">
|
|
|
+ <template #default="scope">
|
|
|
+ <slot v-bind="scope" name="expand" />
|
|
|
+ </template>
|
|
|
+ </el-table-column>
|
|
|
+
|
|
|
+ <!-- 组件需求: 多选框列和内容列第一列固定 -->
|
|
|
+ <FTableColumn
|
|
|
+ v-for="(item, index) in currentConfig"
|
|
|
+ :key="'ftable' + index"
|
|
|
+ v-bind="{ ...item }"
|
|
|
+ :item="item"
|
|
|
+ :fixed="showFixed"
|
|
|
+ :prop="`${Object.keys(item)[0]}`"
|
|
|
+ :label="item[Object.keys(item)[0]]"
|
|
|
+ :width="item.width || ''"
|
|
|
+ :formatter="item.formatter || null"
|
|
|
+ show-overflow-tooltip :show-slots="showSlots"
|
|
|
+ :label-class-name="(item.labelClass || '') + (item.required ? ' required' : '')"
|
|
|
+ :class-name="item.columnClass"
|
|
|
+ :align="item.align || 'left' "
|
|
|
+ :slots-list="item.slots">
|
|
|
+ <!-- 表头收起 -->
|
|
|
+ <template v-if="item.hasClose" #header="scope">
|
|
|
+ <span>
|
|
|
+ {{ item[Object.keys(item)[0]] }}
|
|
|
+ <i :class="['table-header-icon', !item.isClose ? '' : 'table-header-icon-close']" @click="
|
|
|
+ (event) => {
|
|
|
+ handleHeaderClose(scope, event)
|
|
|
+ }
|
|
|
+ " />
|
|
|
+ </span>
|
|
|
+ </template>
|
|
|
+
|
|
|
+ <!-- 表头插槽传递 -->
|
|
|
+ <template v-else-if="item.hasSlotHeader" #header="scope">
|
|
|
+ <slot v-bind="scope" :name="item.slotsItem" />
|
|
|
+ </template>
|
|
|
+
|
|
|
+ <!-- 二级插槽传递 -->
|
|
|
+ <template v-for="slotsItem in showSlots" :slot="slotsItem" slot-scope="scope">
|
|
|
+ <slot v-bind="scope" :name="slotsItem" />
|
|
|
+ </template>
|
|
|
+ </FTableColumn>
|
|
|
+ <!-- 组件需求: 操作列固定 -->
|
|
|
+ <el-table-column v-if="showOperation" :fixed="showFixed ? 'right' : false" :label="operationLabel"
|
|
|
+ :width="operationWidth">
|
|
|
+ <template #default="scope">
|
|
|
+ <slot v-bind="scope" />
|
|
|
+ </template>
|
|
|
+ </el-table-column>
|
|
|
+ </el-table>
|
|
|
+ </div>
|
|
|
+</template>
|
|
|
+
|
|
|
+<script>
|
|
|
+import FTableColumn from './column.vue';
|
|
|
+import map from 'lodash/map';
|
|
|
+
|
|
|
+export default {
|
|
|
+ name: 'FTable',
|
|
|
+ components: { FTableColumn },
|
|
|
+ props: {
|
|
|
+ // table外层div自定义class
|
|
|
+ classText: {
|
|
|
+ type: String,
|
|
|
+ default: ''
|
|
|
+ },
|
|
|
+ // 表单数据
|
|
|
+ data: {
|
|
|
+ type: Array,
|
|
|
+ default: () => []
|
|
|
+ },
|
|
|
+ // 表单列内容
|
|
|
+ column: {
|
|
|
+ type: Array,
|
|
|
+ default: () => []
|
|
|
+ },
|
|
|
+ // 是否具备展开列
|
|
|
+ showExpand: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 是否固定首行
|
|
|
+ showFixed: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 遮罩层
|
|
|
+ loading: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 是否展示checkbox
|
|
|
+ checkbox: {
|
|
|
+ type: Boolean,
|
|
|
+ default: true
|
|
|
+ },
|
|
|
+ // selection操作列添加自定义属性
|
|
|
+ selectionProps: {
|
|
|
+ type: Object,
|
|
|
+ default: () => { }
|
|
|
+ },
|
|
|
+ // 复选框是否可选择
|
|
|
+ selectable: {
|
|
|
+ type: Function,
|
|
|
+ default: function (row) {
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ // 设置操作项宽度
|
|
|
+ operationWidth: {
|
|
|
+ type: String,
|
|
|
+ default: 'auto'
|
|
|
+ },
|
|
|
+ // 是否显示操作
|
|
|
+ showOperation: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 插槽数组
|
|
|
+ showSlots: {
|
|
|
+ type: Array,
|
|
|
+ default: () => []
|
|
|
+ },
|
|
|
+ // 高亮用选中的id-key 默认id
|
|
|
+ selectKey: {
|
|
|
+ type: String,
|
|
|
+ default: 'id'
|
|
|
+ },
|
|
|
+ // 是否撑满
|
|
|
+ full: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 高度
|
|
|
+ height: {
|
|
|
+ type: [String, Number],
|
|
|
+ default: ''
|
|
|
+ },
|
|
|
+ // 操作列文字
|
|
|
+ operationLabel: {
|
|
|
+ type: String,
|
|
|
+ default: '操作'
|
|
|
+ },
|
|
|
+ // 树数据配置
|
|
|
+ treeProps: {
|
|
|
+ type: Object,
|
|
|
+ default: () => { }
|
|
|
+ },
|
|
|
+ selectRows: {
|
|
|
+ type: Array,
|
|
|
+ default: () => []
|
|
|
+ },
|
|
|
+ // 是否为树形结构
|
|
|
+ isTreeStructure: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ },
|
|
|
+ // 是否单选行
|
|
|
+ highLightCurrentRow: {
|
|
|
+ type: Boolean,
|
|
|
+ default: false
|
|
|
+ }
|
|
|
+ },
|
|
|
+ data() {
|
|
|
+ return {
|
|
|
+ mouseTime: 0,
|
|
|
+ selectRow: [], // 选中行
|
|
|
+ indexList: [], // 需要高亮的行
|
|
|
+ // tableId
|
|
|
+ tableId: new Date().getTime(),
|
|
|
+ currentConfig: this.column.concat(),
|
|
|
+ isUpdateTable: true,
|
|
|
+ currentRow: {}, // 单选情况下选中的行
|
|
|
+ tableKey: 0, // table key
|
|
|
+ indexConfig: {}, // index配置
|
|
|
+ listenersStatus: true, // 是否绑定事件'
|
|
|
+ checkedKeys: false, // 树形结构是否已全选
|
|
|
+ copyText: '' // 复制的内容
|
|
|
+ };
|
|
|
+ },
|
|
|
+ computed: {
|
|
|
+ // 列表变化后重置select
|
|
|
+ dataList(data) {
|
|
|
+ this.indexList = [];
|
|
|
+ return this.data;
|
|
|
+ }
|
|
|
+ },
|
|
|
+
|
|
|
+ watch: {
|
|
|
+ dataList: {
|
|
|
+ handler(newValue) {
|
|
|
+ this.$nextTick(() => {
|
|
|
+ if (this.selectRows.length > 0) {
|
|
|
+ this.toggleSelect();
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ },
|
|
|
+ // 配置监听
|
|
|
+ column: {
|
|
|
+ deep: true,
|
|
|
+ immediate: true,
|
|
|
+ handler() {
|
|
|
+ const currentConfig = this.column.concat();
|
|
|
+ if (currentConfig.length) {
|
|
|
+ this.currentConfig = this.initCurrentConfig(currentConfig);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ },
|
|
|
+ updated() {
|
|
|
+ this.$nextTick(() => {
|
|
|
+ this.$refs['fTable'].doLayout();
|
|
|
+ });
|
|
|
+ },
|
|
|
+ mounted() {
|
|
|
+ this.$nextTick(() => {
|
|
|
+ if (this.selectRows.length > 0) {
|
|
|
+ this.toggleSelect();
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ // 需求:鼠标右键复制 选中后保存text 在用户右键复制时重新复制内容
|
|
|
+ const that = this;
|
|
|
+ document.oncopy = (e) => {
|
|
|
+ let isTableCopy = false;
|
|
|
+ const path = e.path || [];
|
|
|
+ // 通过e.path的div 追溯是否在table.cell下的元素复制 如果是才出发
|
|
|
+ path.map((item) => {
|
|
|
+ const className = item.className || '';
|
|
|
+ if (className.indexOf('f-table') !== -1) {
|
|
|
+ isTableCopy = true;
|
|
|
+ }
|
|
|
+ });
|
|
|
+ if (isTableCopy) {
|
|
|
+ setTimeout(() => {
|
|
|
+ that.copy();
|
|
|
+ }, 200);
|
|
|
+ }
|
|
|
+ };
|
|
|
+ },
|
|
|
+ methods: {
|
|
|
+ // 排序
|
|
|
+ sortChange({ column, prop, order }) {
|
|
|
+ this.$emit('sortChange', { column, prop, order });
|
|
|
+ },
|
|
|
+
|
|
|
+ // 把列表中已选中的在初始化时选中,但现在只能通过id来选择,如果有需要其他值来判断,把id改为外部传进来的值即可
|
|
|
+ toggleSelect() {
|
|
|
+ const selectIds = map(this.selectRows, this.selectKey);
|
|
|
+ const dataIds = map(this.dataList, this.selectKey);
|
|
|
+ const selectList = [];
|
|
|
+ selectIds.forEach((item) => {
|
|
|
+ const index = dataIds.indexOf(item || String(item));
|
|
|
+ if (index !== -1) {
|
|
|
+ selectList.push(this.dataList[index]);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ selectList.forEach((row) => {
|
|
|
+ this.$comRowChange(row, null);
|
|
|
+ this.$refs.fTable.toggleRowSelection(row, true);
|
|
|
+ });
|
|
|
+ },
|
|
|
+
|
|
|
+ // 初始化config
|
|
|
+ initCurrentConfig(list) {
|
|
|
+ const newList = list.map((item) => {
|
|
|
+ const newItem = { ...item };
|
|
|
+ if (newItem.tableList && newItem.tableList.length > 0) {
|
|
|
+ newItem.tableList = this.initCurrentConfig(newItem.tableList);
|
|
|
+ }
|
|
|
+ return newItem;
|
|
|
+ });
|
|
|
+ return newList;
|
|
|
+ },
|
|
|
+ // 选中事件触发
|
|
|
+ radioClick(e, row) {
|
|
|
+ if (e.pointerId === 1) {
|
|
|
+ // 防止多次触发
|
|
|
+ this.currentRow = row;
|
|
|
+ this.$emit('current-change', row);
|
|
|
+ }
|
|
|
+ },
|
|
|
+ $comRowChange(currentRow, oldCurrentRow) {
|
|
|
+ this.currentRow = currentRow || {};
|
|
|
+ this.$emit('current-change', currentRow, oldCurrentRow);
|
|
|
+ },
|
|
|
+ // 将选中的状态值抛出去接收
|
|
|
+ $comSelect(selectData, row) {
|
|
|
+ if (!this.checkbox) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ this.selectData = selectData;
|
|
|
+ this.$emit('onSelect', selectData);
|
|
|
+ // 如果外部传入id的key 则高亮选中行
|
|
|
+ if (this.selectKey) {
|
|
|
+ const { data, selectKey } = this;
|
|
|
+ const idList = selectData.map((item) => item[selectKey]);
|
|
|
+ const indexList = [];
|
|
|
+ idList.map((item) => {
|
|
|
+ for (let i = 0; i < data.length; i++) {
|
|
|
+ if (data[i][selectKey] === item) {
|
|
|
+ indexList.push(i);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ });
|
|
|
+ this.indexList = indexList;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ // 树形结构处理是否全选
|
|
|
+ splite(data, flag) {
|
|
|
+ data.forEach((row) => {
|
|
|
+ if (row.templateUrl) {
|
|
|
+ this.$refs.fTable.toggleRowSelection(row, flag);
|
|
|
+ } else {
|
|
|
+ this.$refs.fTable.toggleRowSelection(row, false);
|
|
|
+ }
|
|
|
+ if (row[this.treeProps.children] && row[this.treeProps.children][0]) {
|
|
|
+ this.splite(row[this.treeProps.children], flag);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ },
|
|
|
+ // 全选状态将选中的状态值抛出去接收
|
|
|
+ $comSelectAll(selectData, row) {
|
|
|
+ if (!this.checkbox) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ // 处理树形结构全选事件
|
|
|
+ if (this.isTreeStructure) {
|
|
|
+ this.checkedKeys = !this.checkedKeys;
|
|
|
+ this.splite(this.dataList, this.checkedKeys);
|
|
|
+ }
|
|
|
+ this.selectData = selectData;
|
|
|
+ this.$emit('onSelect', selectData);
|
|
|
+ this.$emit('select',selectData, row);
|
|
|
+ // 如果外部传入id的key 则高亮选中行
|
|
|
+ if (this.selectKey) {
|
|
|
+ const { data, selectKey } = this;
|
|
|
+ const idList = selectData.map((item) => item[selectKey]);
|
|
|
+ const indexList = [];
|
|
|
+ idList.map((item) => {
|
|
|
+ for (let i = 0; i < data.length; i++) {
|
|
|
+ if (data[i][selectKey] === item) {
|
|
|
+ indexList.push(i);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ });
|
|
|
+ this.indexList = indexList;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ isShowSlots(val) {
|
|
|
+ return this.showSlots.includes(val);
|
|
|
+ },
|
|
|
+
|
|
|
+ // 鼠标按下
|
|
|
+ mousedown() {
|
|
|
+ this.mouseTime = new Date().getTime();
|
|
|
+ },
|
|
|
+ // 鼠标弹起 用于选中复制
|
|
|
+ mouseup() {
|
|
|
+ if (new Date().getTime() - this.mouseTime < 200) {
|
|
|
+ this.listenersStatus = true;
|
|
|
+ this.mouseTime = new Date().getTime();
|
|
|
+ }
|
|
|
+ const txt = window.getSelection
|
|
|
+ ? window.getSelection()
|
|
|
+ : document.selection.createRange().text;
|
|
|
+ this.listenersStatus = txt.isCollapsed;
|
|
|
+ const text = txt?.anchorNode?.data;
|
|
|
+ if (text) {
|
|
|
+ localStorage.copyText = text;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ // 覆盖复制
|
|
|
+ copy() {
|
|
|
+ const copyText = localStorage.copyText;
|
|
|
+ if (copyText) {
|
|
|
+ const textarea = document.createElement('textarea');
|
|
|
+ // 将该 textarea 设为 readonly 防止 iOS 下自动唤起键盘,同时将 textarea 移出可视区域
|
|
|
+ textarea.readOnly = 'readonly';
|
|
|
+ textarea.style.position = 'absolute';
|
|
|
+ textarea.style.left = '-9999px';
|
|
|
+ // 将要 copy 的值赋给 textarea 标签的 value 属性
|
|
|
+ textarea.value = copyText;
|
|
|
+ // 重制copyText
|
|
|
+ localStorage.copyText = '';
|
|
|
+ // 将 textarea 插入到 body 中
|
|
|
+ document.body.appendChild(textarea);
|
|
|
+ // 选中值并复制
|
|
|
+ textarea.select();
|
|
|
+ document.execCommand('Copy');
|
|
|
+ document.body.removeChild(textarea);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+};
|
|
|
+</script>
|
|
|
+
|
|
|
+<style lang="scss" scoped>
|
|
|
+.noLabel ::v-deep.el-radio__label {
|
|
|
+ display: none;
|
|
|
+}
|
|
|
+</style>
|