feat(form): groupList支持复制单条记录,支持移动单条记得到指定位置,table支持复制单条记录

This commit is contained in:
roymondchen 2025-01-03 19:47:25 +08:00
parent 833dbcd8b4
commit a831413151
6 changed files with 146 additions and 30 deletions

View File

@ -20,6 +20,7 @@
:disabled="disabled" :disabled="disabled"
:group-model="model[name]" :group-model="model[name]"
@remove-item="removeHandler" @remove-item="removeHandler"
@copy-item="copyHandler"
@swap-item="swapHandler" @swap-item="swapHandler"
@change="changeHandler" @change="changeHandler"
@addDiffCount="onAddDiffCount()" @addDiffCount="onAddDiffCount()"
@ -41,6 +42,7 @@
<script setup lang="ts"> <script setup lang="ts">
import { computed, inject } from 'vue'; import { computed, inject } from 'vue';
import { Grid } from '@element-plus/icons-vue'; import { Grid } from '@element-plus/icons-vue';
import { cloneDeep } from 'lodash-es';
import { TMagicButton } from '@tmagic/design'; import { TMagicButton } from '@tmagic/design';
@ -131,11 +133,17 @@ const removeHandler = (index: number) => {
emit('change', props.model[props.name]); emit('change', props.model[props.name]);
}; };
const copyHandler = (index: number) => {
props.model[props.name].push(cloneDeep(props.model[props.name][index]));
};
const swapHandler = (idx1: number, idx2: number) => { const swapHandler = (idx1: number, idx2: number) => {
if (!props.name) return false; if (!props.name) return false;
const { length } = props.model[props.name];
const [currRow] = props.model[props.name].splice(idx1, 1); const [currRow] = props.model[props.name].splice(idx1, 1);
props.model[props.name].splice(idx2, 0, currRow); props.model[props.name].splice(Math.min(Math.max(idx2, 0), length - 1), 0, currRow);
emit('change', props.model[props.name]); emit('change', props.model[props.name]);
}; };

View File

@ -6,23 +6,83 @@
</TMagicButton> </TMagicButton>
<TMagicButton <TMagicButton
v-show="showDelete(parseInt(String(index)))" v-show="showDelete"
style="color: #f56c6c" type="danger"
size="small"
link link
:icon="Delete" :icon="Delete"
:disabled="disabled" :disabled="disabled"
@click="removeHandler" @click="removeHandler"
></TMagicButton> ></TMagicButton>
<template v-if="movable()"> <TMagicButton
<TMagicButton v-show="index !== 0" link :disabled="disabled" size="small" @click="changeOrder(-1)" v-if="copyable"
>上移<TMagicIcon><CaretTop /></TMagicIcon link
></TMagicButton> size="small"
<TMagicButton v-show="index !== length - 1" :disabled="disabled" link size="small" @click="changeOrder(1)" type="primary"
>下移<TMagicIcon><CaretBottom /></TMagicIcon :icon="DocumentCopy"
></TMagicButton> :disabled="disabled"
@click="copyHandler"
>复制</TMagicButton
>
<template v-if="movable">
<TMagicButton
v-show="index !== 0"
link
size="small"
:disabled="disabled"
:icon="CaretTop"
@click="changeOrder(-1)"
>上移</TMagicButton
>
<TMagicButton
v-show="index !== length - 1"
link
size="small"
:disabled="disabled"
:icon="CaretBottom"
@click="changeOrder(1)"
>下移</TMagicButton
>
</template> </template>
<TMagicPopover
v-if="config.moveSpecifyLocation"
trigger="click"
placement="top"
width="200"
:visible="moveSpecifyLocationVisible"
>
<template #reference>
<TMagicButton
link
size="small"
type="primary"
:icon="Position"
:disabled="disabled"
@click="moveSpecifyLocationVisible = true"
>移动至</TMagicButton
>
</template>
<div>
<div>
<TMagicInputNumber
style="margin: 0 5px"
v-model="moveSpecifyLocationIndex"
size="small"
:min="1"
:disabled="disabled"
></TMagicInputNumber
>
</div>
<div style="text-align: right; margin-top: 20px">
<TMagicButton size="small" text @click="moveSpecifyLocationVisible = false">取消</TMagicButton>
<TMagicButton size="small" type="primary" @click="moveSpecifyLocationHandler">确认</TMagicButton>
</div>
</div>
</TMagicPopover>
<span v-if="itemExtra" v-html="itemExtra" class="m-form-tip"></span> <span v-if="itemExtra" v-html="itemExtra" class="m-form-tip"></span>
</div> </div>
@ -44,9 +104,9 @@
<script setup lang="ts"> <script setup lang="ts">
import { computed, inject, ref } from 'vue'; import { computed, inject, ref } from 'vue';
import { CaretBottom, CaretRight, CaretTop, Delete } from '@element-plus/icons-vue'; import { CaretBottom, CaretRight, CaretTop, Delete, DocumentCopy, Position } from '@element-plus/icons-vue';
import { TMagicButton, TMagicIcon } from '@tmagic/design'; import { TMagicButton, TMagicIcon, TMagicInputNumber, TMagicPopover } from '@tmagic/design';
import type { ContainerChangeEventData, FormState, GroupListConfig } from '../schema'; import type { ContainerChangeEventData, FormState, GroupListConfig } from '../schema';
import { filterFunction } from '../utils/form'; import { filterFunction } from '../utils/form';
@ -70,7 +130,7 @@ const props = defineProps<{
disabled?: boolean; disabled?: boolean;
}>(); }>();
const emit = defineEmits(['swap-item', 'remove-item', 'change', 'addDiffCount']); const emit = defineEmits(['swap-item', 'remove-item', 'change', 'addDiffCount', 'copy-item']);
const mForm = inject<FormState | undefined>('mForm'); const mForm = inject<FormState | undefined>('mForm');
const expand = ref(props.config.expandAll || !props.index); const expand = ref(props.config.expandAll || !props.index);
@ -112,18 +172,18 @@ const expandHandler = () => {
}; };
// //
const showDelete = (index: number) => { const showDelete = computed(() => {
const deleteFunc = props.config.delete; const deleteFunc = props.config.delete;
if (deleteFunc && typeof deleteFunc === 'function') { if (deleteFunc && typeof deleteFunc === 'function') {
return deleteFunc(props.model, index, mForm?.values); return deleteFunc(props.model, props.index, mForm?.values);
} }
return true; return true;
}; });
// //
const changeOrder = (offset = 0) => emit('swap-item', props.index, props.index + offset); const changeOrder = (offset = 0) => emit('swap-item', props.index, props.index + offset);
const movable = () => { const movable = computed(() => {
const { movable } = props.config; const { movable } = props.config;
// //
@ -132,6 +192,20 @@ const movable = () => {
return movable(mForm, props.index || 0, props.model, props.groupModel); return movable(mForm, props.index || 0, props.model, props.groupModel);
} }
return movable; return movable;
}; });
const copyable = computed(() => filterFunction<boolean>(mForm, props.config.copyable, props));
const onAddDiffCount = () => emit('addDiffCount'); const onAddDiffCount = () => emit('addDiffCount');
const copyHandler = () => {
emit('copy-item', props.index);
};
const moveSpecifyLocationVisible = ref(false);
const moveSpecifyLocationIndex = ref(1);
const moveSpecifyLocationHandler = () => {
moveSpecifyLocationVisible.value = false;
emit('swap-item', props.index, moveSpecifyLocationIndex.value - 1);
};
</script> </script>

View File

@ -26,18 +26,32 @@
<TMagicTableColumn <TMagicTableColumn
label="操作" label="操作"
:width="config.operateColWidth || 55" :width="config.operateColWidth || 100"
align="center" align="center"
:fixed="config.fixed === false ? undefined : 'left'" :fixed="config.fixed === false ? undefined : 'left'"
> >
<template v-slot="scope"> <template v-slot="scope">
<slot name="operateCol" :scope="scope"></slot> <slot name="operateCol" :scope="scope"></slot>
<TMagicIcon <TMagicButton
v-show="showDelete(scope.$index + 1 + pagecontext * pagesize - 1)" v-show="showDelete(scope.$index + 1 + pagecontext * pagesize - 1)"
class="m-table-delete-icon" size="small"
type="danger"
link
title="删除"
:icon="Delete"
@click="removeHandler(scope.$index + 1 + pagecontext * pagesize - 1)" @click="removeHandler(scope.$index + 1 + pagecontext * pagesize - 1)"
><Delete ></TMagicButton>
/></TMagicIcon>
<TMagicButton
v-if="copyable(scope.$index + 1 + pagecontext * pagesize - 1)"
link
size="small"
type="primary"
title="复制"
:icon="DocumentCopy"
:disabled="disabled"
@click="copyHandler(scope.$index + 1 + pagecontext * pagesize - 1)"
></TMagicButton>
</template> </template>
</TMagicTableColumn> </TMagicTableColumn>
@ -189,13 +203,12 @@
<script setup lang="ts"> <script setup lang="ts">
import { computed, inject, onMounted, ref, toRefs, watchEffect } from 'vue'; import { computed, inject, onMounted, ref, toRefs, watchEffect } from 'vue';
import { ArrowDown, ArrowUp, Delete, FullScreen, Grid } from '@element-plus/icons-vue'; import { ArrowDown, ArrowUp, Delete, DocumentCopy, FullScreen, Grid } from '@element-plus/icons-vue';
import { cloneDeep } from 'lodash-es'; import { cloneDeep } from 'lodash-es';
import Sortable, { SortableEvent } from 'sortablejs'; import Sortable, { SortableEvent } from 'sortablejs';
import { import {
TMagicButton, TMagicButton,
TMagicIcon,
tMagicMessage, tMagicMessage,
TMagicPagination, TMagicPagination,
TMagicTable, TMagicTable,
@ -550,6 +563,22 @@ const showDelete = (index: number) => {
return true; return true;
}; };
const copyable = (index: number) => {
const copyableFunc = props.config.copyable;
if (copyableFunc && typeof copyableFunc === 'function') {
return copyableFunc(mForm, {
values: mForm?.initValues || {},
model: props.model,
parent: mForm?.parentValues || {},
formValue: mForm?.values || props.model,
prop: props.prop,
config: props.config,
index,
});
}
return true;
};
const clearHandler = () => { const clearHandler = () => {
const len = props.model[modelName.value].length; const len = props.model[modelName.value].length;
props.model[modelName.value].splice(0, len); props.model[modelName.value].splice(0, len);
@ -594,6 +623,10 @@ const handleCurrentChange = (val: number) => {
pagecontext.value = val - 1; pagecontext.value = val - 1;
}; };
const copyHandler = (index: number) => {
props.model[modelName.value].push(cloneDeep(props.model[modelName.value][index]));
};
const toggleMode = () => { const toggleMode = () => {
const calcLabelWidth = (label: string) => { const calcLabelWidth = (label: string) => {
if (!label) return '0px'; if (!label) return '0px';

View File

@ -684,6 +684,7 @@ export interface TableConfig extends FormItem {
addable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean; addable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean;
/** 是否显示删除按钮 */ /** 是否显示删除按钮 */
delete?: (model: any, index: number, values: any) => boolean | boolean; delete?: (model: any, index: number, values: any) => boolean | boolean;
copyable?: (model: any, data: any) => boolean | boolean;
/** 是否显示导入按钮 */ /** 是否显示导入按钮 */
importable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean; importable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean;
/** 是否显示checkbox */ /** 是否显示checkbox */
@ -718,12 +719,14 @@ export interface GroupListConfig extends FormItem {
addable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean; addable?: (mForm: FormState | undefined, data: any) => boolean | 'undefined' | boolean;
defaultAdd?: (mForm: FormState | undefined, data: any) => any; defaultAdd?: (mForm: FormState | undefined, data: any) => any;
delete?: (model: any, index: number | string | symbol, values: any) => boolean | boolean; delete?: (model: any, index: number | string | symbol, values: any) => boolean | boolean;
copyable?: FilterFunction<boolean>;
movable?: ( movable?: (
mForm: FormState | undefined, mForm: FormState | undefined,
index: number | string | symbol, index: number | string | symbol,
model: any, model: any,
groupModel: any, groupModel: any,
) => boolean | boolean; ) => boolean | boolean;
moveSpecifyLocation?: boolean;
[key: string]: any; [key: string]: any;
} }

View File

@ -62,9 +62,4 @@
.el-form-item { .el-form-item {
margin-bottom: 0; margin-bottom: 0;
} }
.m-table-delete-icon {
color: #f56c6c;
cursor: pointer;
}
} }

View File

@ -211,6 +211,7 @@ export default createForm([
{ {
type: 'table', type: 'table',
name: 'table', name: 'table',
copyable: true,
defautSort: { prop: 'name', order: 'descending' }, defautSort: { prop: 'name', order: 'descending' },
extra: 'extra', extra: 'extra',
itemExtra: (vm: any, { model }: any): any => `${model.text}itemExtra`, itemExtra: (vm: any, { model }: any): any => `${model.text}itemExtra`,
@ -232,6 +233,8 @@ export default createForm([
type: 'groupList', type: 'groupList',
name: 'groupList', name: 'groupList',
extra: '分组xxxxxxxxxxxx', extra: '分组xxxxxxxxxxxx',
copyable: true,
moveSpecifyLocation: true,
itemExtra: (vm: any, { model }: any) => `${model.name}extra`, itemExtra: (vm: any, { model }: any) => `${model.name}extra`,
items: [ items: [
{ {