feat: 表格支持多个编辑项

This commit is contained in:
RubyLiu 2023-09-05 17:56:36 +08:00 committed by 刘瑞斌
parent 16463a4bf0
commit bdcc5854a2
10 changed files with 55 additions and 31 deletions

View File

@ -9,7 +9,7 @@
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { computed, onBeforeMount, onMounted } from 'vue'; import { computed, onBeforeMount, onMounted, watchEffect } from 'vue';
import { useRouter } from 'vue-router'; import { useRouter } from 'vue-router';
import enUS from '@arco-design/web-vue/es/locale/lang/en-us'; import enUS from '@arco-design/web-vue/es/locale/lang/en-us';
import zhCN from '@arco-design/web-vue/es/locale/lang/zh-cn'; import zhCN from '@arco-design/web-vue/es/locale/lang/zh-cn';
@ -70,7 +70,7 @@
}); });
const checkIsLogin = async () => { const checkIsLogin = async () => {
const isLogin = await userStore.isLogin(); const isLogin = await userStore.isLogin();
const isLoginPage = window.location.hash === '#/login'; const isLoginPage = window.location.hash.indexOf('#/login') > -1;
if (isLoginPage && isLogin) { if (isLoginPage && isLogin) {
// //
router.push(WorkbenchRouteEnum.WORKBENCH); router.push(WorkbenchRouteEnum.WORKBENCH);

View File

@ -12,7 +12,8 @@ export default function checkStatus(status: number, msg: string, errorMessageMod
errMessage = `${msg}`; errMessage = `${msg}`;
break; break;
case 401: { case 401: {
if (msg.length > 100) { if (msg.length === 216) {
// 216是salt的长度
setSalt(msg); setSalt(msg);
} else { } else {
errMessage = msg || t('api.errMsg401'); errMessage = msg || t('api.errMsg401');

View File

@ -86,7 +86,11 @@
<template v-else-if="item.showTooltip"> <template v-else-if="item.showTooltip">
<a-tooltip placement="top" :content="String(record[item.dataIndex as string])"> <a-tooltip placement="top" :content="String(record[item.dataIndex as string])">
<a-input <a-input
v-if="editActiveKey === rowIndex && item.dataIndex === editKey" v-if="
editActiveKey === `${item.dataIndex}${rowIndex}` &&
item.editable &&
item.editType === ColumnEditTypeEnum.INPUT
"
ref="currentInputRef" ref="currentInputRef"
v-model="record[item.dataIndex as string]" v-model="record[item.dataIndex as string]"
@blur="handleEditInputBlur()" @blur="handleEditInputBlur()"
@ -99,18 +103,22 @@
</slot> </slot>
</div> </div>
<MsIcon <MsIcon
v-if="item.editable && item.dataIndex === editKey && !record.deleted" v-if="item.editable && !record.deleted"
class="ml-2 cursor-pointer" class="ml-2 cursor-pointer"
:class="{ 'ms-table-edit-active': editActiveKey === rowIndex }" :class="{ 'ms-table-edit-active': editActiveKey === rowIndex }"
type="icon-icon_edit_outlined" type="icon-icon_edit_outlined"
@click="handleEdit(rowIndex)" @click="handleEdit(item.dataIndex as string, rowIndex)"
/> />
</template> </template>
</a-tooltip> </a-tooltip>
</template> </template>
<template v-else> <template v-else>
<a-input <a-input
v-if="editActiveKey === rowIndex && item.dataIndex === editKey" v-if="
editActiveKey === `${item.dataIndex}${rowIndex}` &&
item.editable &&
item.editType === ColumnEditTypeEnum.INPUT
"
ref="currentInputRef" ref="currentInputRef"
v-model="record[item.dataIndex as string]" v-model="record[item.dataIndex as string]"
@blur="handleEditInputBlur()" @blur="handleEditInputBlur()"
@ -125,7 +133,7 @@
class="ml-2 cursor-pointer" class="ml-2 cursor-pointer"
:class="{ 'ms-table-edit-active': editActiveKey === rowIndex }" :class="{ 'ms-table-edit-active': editActiveKey === rowIndex }"
type="icon-icon_edit_outlined" type="icon-icon_edit_outlined"
@click="handleEdit(rowIndex)" @click="handleEdit(item.dataIndex as string, rowIndex)"
/> />
</template> </template>
</template> </template>
@ -170,7 +178,7 @@
import { useAttrs, computed, ref, onMounted, nextTick } from 'vue'; import { useAttrs, computed, ref, onMounted, nextTick } from 'vue';
import { useAppStore, useTableStore } from '@/store'; import { useAppStore, useTableStore } from '@/store';
import selectAll from './select-all.vue'; import selectAll from './select-all.vue';
import { SpecialColumnEnum, SelectAllEnum } from '@/enums/tableEnum'; import { SpecialColumnEnum, SelectAllEnum, ColumnEditTypeEnum } from '@/enums/tableEnum';
import BatchAction from './batchAction.vue'; import BatchAction from './batchAction.vue';
import MsPagination from '@/components/pure/ms-pagination/index'; import MsPagination from '@/components/pure/ms-pagination/index';
import ColumnSelector from './columnSelector.vue'; import ColumnSelector from './columnSelector.vue';
@ -208,7 +216,7 @@
// - // -
const selectTotal = ref(0); const selectTotal = ref(0);
// Active // Active
const editActiveKey = ref(-1); const editActiveKey = ref<string>('');
// inputRef // inputRef
const currentInputRef = ref(); const currentInputRef = ref();
const { rowKey, editKey }: Partial<MsTableProps<any>> = attrs; const { rowKey, editKey }: Partial<MsTableProps<any>> = attrs;
@ -308,13 +316,13 @@
}); });
const handleEditInputEnter = (record: TableData) => { const handleEditInputEnter = (record: TableData) => {
editActiveKey.value = -1; editActiveKey.value = '';
emit('rowNameChange', record); emit('rowNameChange', record);
}; };
const handleEditInputBlur = () => { const handleEditInputBlur = () => {
currentInputRef.value = null; currentInputRef.value = null;
editActiveKey.value = -1; editActiveKey.value = '';
}; };
// change // change
@ -336,8 +344,8 @@
}; };
// input // input
const handleEdit = (rowIndex: number) => { const handleEdit = (dataIndex: string, rowIndex: number) => {
editActiveKey.value = rowIndex; editActiveKey.value = dataIndex + rowIndex;
if (currentInputRef.value) { if (currentInputRef.value) {
currentInputRef.value[0].focus(); currentInputRef.value[0].focus();
} else { } else {

View File

@ -35,18 +35,18 @@
<div class="flex-col"> <div class="flex-col">
<div v-for="(item, idx) in nonSortColumn" :key="item.dataIndex" class="column-item"> <div v-for="(item, idx) in nonSortColumn" :key="item.dataIndex" class="column-item">
<div>{{ t(item.title as string) }}</div> <div>{{ t(item.title as string) }}</div>
<a-switch size="small" :model-value="item.showInTable" @change="handleFirstColumnChange(idx)" /> <a-switch :model-value="item.showInTable" size="small" @change="handleFirstColumnChange(idx)" />
</div> </div>
</div> </div>
<a-divider orientation="center" class="non-sort">{{ t('msTable.columnSetting.nonSort') }}</a-divider> <a-divider orientation="center" class="non-sort">{{ t('msTable.columnSetting.nonSort') }}</a-divider>
<Draggable tag="div" :list="couldSortColumn" class="list-group" handle=".handle" item-key="dateIndex"> <Draggable tag="div" :list="couldSortColumn" class="list-group" item-key="dateIndex">
<template #item="{ element, index }"> <template #item="{ element, index }">
<div class="column-drag-item"> <div class="column-drag-item">
<div class="flex items-center"> <div class="flex items-center">
<icon-drag-dot-vertical class="handle" /> <icon-drag-dot-vertical />
<div class="ml-[8px]">{{ t(element.title as string) }}</div> <div class="ml-[8px]">{{ t(element.title as string) }}</div>
</div> </div>
<a-switch size="small" :model-value="element.showInTable" @change="handleSecondColumnChange(index)" /> <a-switch :model-value="element.showInTable" size="small" @change="handleSecondColumnChange(index)" />
</div> </div>
</template> </template>
</Draggable> </Draggable>
@ -112,12 +112,14 @@
const handleFirstColumnChange = (idx: number) => { const handleFirstColumnChange = (idx: number) => {
const item = nonSortColumn.value[idx]; const item = nonSortColumn.value[idx];
item.showInTable = !item.showInTable; item.showInTable = !item.showInTable;
nonSortColumn.value[idx] = item;
hasChange.value = true; hasChange.value = true;
}; };
const handleSecondColumnChange = (idx: number) => { const handleSecondColumnChange = (idx: number) => {
const item = couldSortColumn.value[idx]; const item = couldSortColumn.value[idx];
item.showInTable = !item.showInTable; item.showInTable = !item.showInTable;
couldSortColumn.value[idx] = item;
hasChange.value = true; hasChange.value = true;
}; };

View File

@ -1,3 +1,4 @@
import { ColumnEditTypeEnum } from '@/enums/tableEnum';
import { TableColumnData, TableData, TableDraggable, TableChangeExtra } from '@arco-design/web-vue'; import { TableColumnData, TableData, TableDraggable, TableChangeExtra } from '@arco-design/web-vue';
export interface MsPaginationI { export interface MsPaginationI {
@ -26,6 +27,8 @@ export interface MsTableColumnData extends TableColumnData {
disableTitle?: string; disableTitle?: string;
// 当展示tooltip时是否是Tag // 当展示tooltip时是否是Tag
isTag?: boolean; isTag?: boolean;
// editType
editType?: ColumnEditTypeEnum;
} }
export type MsTableErrorStatus = boolean | 'error' | 'empty'; export type MsTableErrorStatus = boolean | 'error' | 'empty';
@ -77,8 +80,6 @@ export interface MsTableProps<T> {
showSetting?: boolean; showSetting?: boolean;
// 分页是否是简单模式 // 分页是否是简单模式
pageSimple?: boolean; pageSimple?: boolean;
// 编辑的key默认为name
editKey?: string;
// 是否展示禁用的行 // 是否展示禁用的行
noDisable?: boolean; noDisable?: boolean;
// 表格的错误状态默认为false // 表格的错误状态默认为false

View File

@ -36,7 +36,7 @@ export default function useTableProps<T>(
bordered: true, bordered: true,
showPagination: true, showPagination: true,
size: 'small', size: 'small',
scroll: { maxHeight: '800px', minWidth: '1600px', x: '1400px', y: '600px' }, scroll: { maxHeight: '800px', minWidth: '1600px', y: '600px' },
checkable: true, checkable: true,
loading: false, loading: false,
data: [], data: [],

View File

@ -37,3 +37,10 @@ export enum SelectAllEnum {
CURRENT = 'current', CURRENT = 'current',
NONE = 'none', NONE = 'none',
} }
// 列编辑类型
export enum ColumnEditTypeEnum {
INPUT = 'input',
SELECT = 'select',
SEARCH_SELECT = 'search_select',
}

View File

@ -26,7 +26,7 @@ export default function useUser() {
}; };
const isLoginPage = () => { const isLoginPage = () => {
return router.currentRoute.value.name === 'login'; return window.location.hash.indexOf('login') > -1;
}; };
return { return {

View File

@ -68,7 +68,7 @@
revokeDeleteProjectByOrg, revokeDeleteProjectByOrg,
createOrUpdateProjectByOrg, createOrUpdateProjectByOrg,
} from '@/api/modules/setting/organizationAndProject'; } from '@/api/modules/setting/organizationAndProject';
import { TableKeyEnum } from '@/enums/tableEnum'; import { ColumnEditTypeEnum, TableKeyEnum } from '@/enums/tableEnum';
import { MsTableColumn } from '@/components/pure/ms-table/type'; import { MsTableColumn } from '@/components/pure/ms-table/type';
import MsTableMoreAction from '@/components/pure/ms-table-more-action/index.vue'; import MsTableMoreAction from '@/components/pure/ms-table-more-action/index.vue';
import MsButton from '@/components/pure/ms-button/index.vue'; import MsButton from '@/components/pure/ms-button/index.vue';
@ -101,42 +101,51 @@
dataIndex: 'num', dataIndex: 'num',
width: 100, width: 100,
showTooltip: true, showTooltip: true,
editable: true,
editType: ColumnEditTypeEnum.INPUT,
}, },
{ {
title: 'system.organization.name', title: 'system.organization.name',
slotName: 'name', slotName: 'name',
editable: true, editable: true,
editType: ColumnEditTypeEnum.INPUT,
dataIndex: 'name', dataIndex: 'name',
showTooltip: true, showTooltip: true,
}, },
{ {
title: 'system.organization.member', title: 'system.organization.member',
slotName: 'memberCount', slotName: 'memberCount',
showDrag: true,
}, },
{ {
title: 'system.organization.status', title: 'system.organization.status',
dataIndex: 'enable', dataIndex: 'enable',
disableTitle: 'common.end', disableTitle: 'common.end',
showDrag: true,
}, },
{ {
title: 'system.organization.description', title: 'system.organization.description',
dataIndex: 'description', dataIndex: 'description',
ellipsis: true, ellipsis: true,
tooltip: true, tooltip: true,
showDrag: true,
}, },
{ {
title: 'system.organization.subordinateOrg', title: 'system.organization.subordinateOrg',
dataIndex: 'organizationName', dataIndex: 'organizationName',
showDrag: true,
}, },
{ {
title: 'system.organization.creator', title: 'system.organization.creator',
slotName: 'creator', slotName: 'creator',
width: 180, width: 180,
showDrag: true,
}, },
{ {
title: 'system.organization.createTime', title: 'system.organization.createTime',
dataIndex: 'createTime', dataIndex: 'createTime',
width: 180, width: 180,
showDrag: true,
}, },
{ {
title: 'system.organization.operation', title: 'system.organization.operation',

View File

@ -43,12 +43,7 @@
@submit="fetchData" @submit="fetchData"
@cancel="handleAddUserModalCancel" @cancel="handleAddUserModalCancel"
/> />
<UserDrawer <UserDrawer v-bind="currentUserDrawer" @request-fetch-data="fetchData" @cancel="handleUserDrawerCancel" />
:project-id="currentProjectId"
v-bind="currentUserDrawer"
@request-fetch-data="fetchData"
@cancel="handleUserDrawerCancel"
/>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
@ -181,7 +176,7 @@
const currentUserDrawer = reactive({ const currentUserDrawer = reactive({
visible: false, visible: false,
organizationId: '', projectId: '',
}); });
const tableActions: ActionsItem[] = [ const tableActions: ActionsItem[] = [
@ -260,11 +255,12 @@
const showUserDrawer = (record: TableData) => { const showUserDrawer = (record: TableData) => {
currentUserDrawer.visible = true; currentUserDrawer.visible = true;
currentUserDrawer.organizationId = record.id; currentUserDrawer.projectId = record.id;
}; };
const handleUserDrawerCancel = () => { const handleUserDrawerCancel = () => {
currentUserDrawer.visible = false; currentUserDrawer.visible = false;
currentUserDrawer.projectId = '';
}; };
const handleAddUserModalCancel = () => { const handleAddUserModalCancel = () => {