fix(项目管理): 任务中心列表过滤

This commit is contained in:
xinxin.wu 2024-04-12 13:47:48 +08:00 committed by 刘瑞斌
parent 6533c45623
commit 7b4f30277c
11 changed files with 264 additions and 119 deletions

View File

@ -97,7 +97,11 @@
</div> </div>
<div v-else> <div v-else>
<div class="flex items-center"> <div class="flex items-center">
<div class="font-medium text-[var(--color-text-1)]">{{ t('ms.message.notice.title') }}</div> <a-badge :count="9" dot :offset="[6, -2]">
<div class="one-line-text max-w-[400px] font-medium text-[var(--color-text-1)]">{{
t('ms.message.notice.title')
}}</div>
</a-badge>
</div> </div>
<div class="flex items-center"> <div class="flex items-center">
<div class="font-medium text-[var(--color-text-2)]">{{ item.userName }}&nbsp;&nbsp;</div> <div class="font-medium text-[var(--color-text-2)]">{{ item.userName }}&nbsp;&nbsp;</div>
@ -375,7 +379,6 @@
box-sizing: border-box; box-sizing: border-box;
line-height: 1.8715; line-height: 1.8715;
} }
.case { .case {
/* 底部样式 */ /* 底部样式 */
position: fixed; position: fixed;
@ -383,7 +386,6 @@
padding: 20px; padding: 20px;
text-align: center; text-align: center;
} }
.clickable { .clickable {
cursor: pointer; cursor: pointer;
transition: background-color 0.3s ease; transition: background-color 0.3s ease;

View File

@ -8,7 +8,7 @@ import {
getUserByProjectByOrg, getUserByProjectByOrg,
} from '@/api/modules/setting/organizationAndProject'; } from '@/api/modules/setting/organizationAndProject';
import { getOrgUserGroupOption, getSystemUserGroupOption } from '@/api/modules/setting/usergroup'; import { getOrgUserGroupOption, getSystemUserGroupOption } from '@/api/modules/setting/usergroup';
import { getOrgOptions } from '@/api/modules/system';
// eslint-disable-next-line no-shadow // eslint-disable-next-line no-shadow
export enum UserRequestTypeEnum { export enum UserRequestTypeEnum {
SYSTEM_USER_GROUP = 'SYSTEM_USER_GROUP', SYSTEM_USER_GROUP = 'SYSTEM_USER_GROUP',
@ -24,6 +24,7 @@ export enum UserRequestTypeEnum {
SYSTEM_ORGANIZATION_MEMBER = 'SYSTEM_ORGANIZATION_MEMBER', SYSTEM_ORGANIZATION_MEMBER = 'SYSTEM_ORGANIZATION_MEMBER',
PROJECT_PERMISSION_MEMBER = 'PROJECT_PERMISSION_MEMBER', PROJECT_PERMISSION_MEMBER = 'PROJECT_PERMISSION_MEMBER',
PROJECT_USER_GROUP = 'PROJECT_USER_GROUP', PROJECT_USER_GROUP = 'PROJECT_USER_GROUP',
SYSTEM_ORGANIZATION_LIST = 'SYSTEM_ORGANIZATION_LIST',
} }
export default function initOptionsFunc(type: string, params: Record<string, any>) { export default function initOptionsFunc(type: string, params: Record<string, any>) {
if (type === UserRequestTypeEnum.SYSTEM_USER_GROUP) { if (type === UserRequestTypeEnum.SYSTEM_USER_GROUP) {
@ -66,4 +67,8 @@ export default function initOptionsFunc(type: string, params: Record<string, any
// 项目-用户组 // 项目-用户组
return getProjectUserGroupOptions(params.projectId, params.userRoleId, params.keyword); return getProjectUserGroupOptions(params.projectId, params.userRoleId, params.keyword);
} }
if (type === UserRequestTypeEnum.SYSTEM_ORGANIZATION_LIST) {
// 系统-组织
return getOrgOptions();
}
} }

View File

@ -100,7 +100,8 @@
function searchStep() { function searchStep() {
const splitLevel = props.keyWords.split('-'); const splitLevel = props.keyWords.split('-');
const stepTypeStatus = splitLevel[1]; const stepTypeStatus = splitLevel[1];
const stepType = splitLevel[0] !== 'REQUEST' ? ['API', 'API_CASE', 'CUSTOM_REQUEST'] : splitLevel[0]; const requestType = ['API', 'API_CASE', 'CUSTOM_REQUEST'];
const stepType = splitLevel[0] === 'CUSTOM_REQUEST' ? ['API', 'API_CASE', 'CUSTOM_REQUEST'] : splitLevel[0];
const search = (_data: ScenarioItemType[]) => { const search = (_data: ScenarioItemType[]) => {
const result: ScenarioItemType[] = []; const result: ScenarioItemType[] = [];
_data.forEach((item) => { _data.forEach((item) => {

View File

@ -91,18 +91,24 @@
</div> </div>
</template> </template>
<template #default="{ loading }"> <template #default="{ loading }">
<a-spin :loading="detailLoading" class="w-full"> <div
<div ref="wrapperRef" class="h-full bg-white">
<MsSplitBox
ref="wrapperRef" ref="wrapperRef"
class="wrapperRef bg-white"
:style="{
height: 'calc(100% - 86px)',
}"
>
<MsSplitBox
expand-direction="right" expand-direction="right"
:max="0.7" :max="0.7"
:min="0.7" :min="0.7"
:size="900" :size="900"
direction="horizontal"
:class="{ 'left-bug-detail': activeTab === 'comment' }" :class="{ 'left-bug-detail': activeTab === 'comment' }"
> >
<template #first> <template #first>
<div class="leftWrapper h-full"> <div class="leftWrapper h-full">
<a-spin :loading="detailLoading" class="w-full">
<div class="header h-[50px]"> <div class="header h-[50px]">
<MsTab <MsTab
v-model:active-key="activeTab" v-model:active-key="activeTab"
@ -110,6 +116,7 @@
:get-text-func="getTabBadge" :get-text-func="getTabBadge"
class="no-content relative mb-[8px]" class="no-content relative mb-[8px]"
/> />
</div>
<div class="tab-pane-container"> <div class="tab-pane-container">
<BugDetailTab <BugDetailTab
v-if="activeTab === 'detail'" v-if="activeTab === 'detail'"
@ -133,13 +140,13 @@
<BugHistoryTab v-else-if="activeTab === 'history'" :bug-id="detailInfo.id" /> <BugHistoryTab v-else-if="activeTab === 'history'" :bug-id="detailInfo.id" />
</div> </div>
</div> </a-spin>
</div> </div>
</template> </template>
<template #second> <template #second>
<a-spin :loading="rightLoading" class="w-full"> <a-spin :loading="rightLoading" class="w-full">
<!-- 所属平台一致, 详情展示 --> <!-- 所属平台一致, 详情展示 -->
<div v-if="props.currentPlatform === detailInfo.platform" class="rightWrapper p-[24px]"> <div v-if="props.currentPlatform === detailInfo.platform" class="rightWrapper h-full p-[24px]">
<!-- 自定义字段开始 --> <!-- 自定义字段开始 -->
<div class="inline-block w-full break-words"> <div class="inline-block w-full break-words">
<a-skeleton v-if="loading" class="w-full" :loading="loading" :animation="true"> <a-skeleton v-if="loading" class="w-full" :loading="loading" :animation="true">
@ -199,7 +206,6 @@
</template> </template>
</MsSplitBox> </MsSplitBox>
</div> </div>
</a-spin>
<CommentInput <CommentInput
v-if="activeTab === 'comment' && hasAnyPermission(['PROJECT_BUG:READ+COMMENT'])" v-if="activeTab === 'comment' && hasAnyPermission(['PROJECT_BUG:READ+COMMENT'])"
:content="commentContent" :content="commentContent"

View File

@ -1,5 +1,5 @@
<template> <template>
<div> <div class="p-4">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<a-button type="primary" :disabled="!hasAnyPermission(['PROJECT_BUG:READ+UPDATE'])" @click="handleSelect">{{ <a-button type="primary" :disabled="!hasAnyPermission(['PROJECT_BUG:READ+UPDATE'])" @click="handleSelect">{{
t('caseManagement.featureCase.linkCase') t('caseManagement.featureCase.linkCase')

View File

@ -9,7 +9,12 @@
<template #content> <template #content>
<div class="arco-table-filters-content"> <div class="arco-table-filters-content">
<div class="ml-[6px] flex w-full items-center justify-start overflow-hidden px-[6px] py-[2px]"> <div class="ml-[6px] flex w-full items-center justify-start overflow-hidden px-[6px] py-[2px]">
<a-checkbox-group v-model:model-value="innerStatusFilters" direction="vertical" size="small"> <a-checkbox-group
v-if="props.mode === 'static' && props.list?.length"
v-model:model-value="innerStatusFilters"
direction="vertical"
size="small"
>
<a-checkbox <a-checkbox
v-for="(item, index) of props.list" v-for="(item, index) of props.list"
:key="item[props.valueKey || 'value']" :key="item[props.valueKey || 'value']"
@ -23,7 +28,15 @@
</a-checkbox> </a-checkbox>
</a-checkbox-group> </a-checkbox-group>
</div> </div>
<div class="filter-button"> <div v-if="props.mode === 'remote'" class="min-h-[100px] w-[200px] p-4">
<MsUserSelector
v-model="innerStatusFilters"
:load-option-params="props.loadOptionParams"
:type="props.type"
:placeholder="props.placeholderText"
/>
</div>
<div class="flex items-center p-4" :class="[props.mode === 'static' ? 'justify-between' : 'justify-end']">
<a-button size="mini" class="mr-[8px]" @click="resetFilter"> <a-button size="mini" class="mr-[8px]" @click="resetFilter">
{{ t('common.reset') }} {{ t('common.reset') }}
</a-button> </a-button>
@ -39,6 +52,9 @@
<script setup lang="ts"> <script setup lang="ts">
import { useVModel } from '@vueuse/core'; import { useVModel } from '@vueuse/core';
import MsUserSelector from '@/components/business/ms-user-selector/index.vue';
import type { UserRequestTypeEnum } from '@/components/business/ms-user-selector/utils';
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
const { t } = useI18n(); const { t } = useI18n();
@ -47,14 +63,23 @@
[key: string]: any; [key: string]: any;
} }
const props = defineProps<{ const props = withDefaults(
defineProps<{
mode?: 'static' | 'remote';
visible: boolean; visible: boolean;
title: string; title: string;
statusFilters: string[]; statusFilters: string[];
list: FilterListItem[]; list?: FilterListItem[];
valueKey?: string; valueKey?: string;
labelKey?: string; labelKey?: string;
}>(); type?: UserRequestTypeEnum; //
loadOptionParams?: Record<string, any>; //
placeholderText?: string;
}>(),
{
mode: 'static',
}
);
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:visible', visible: boolean): void; (e: 'update:visible', visible: boolean): void;

View File

@ -102,6 +102,33 @@
</template> </template>
</TableFilter> </TableFilter>
</template> </template>
<template v-if="appStore.packageType === 'enterprise'" #orgFilterName="{ columnConfig }">
<TableFilter
v-model:visible="orgFilterVisible"
v-model:status-filters="orgFiltersMap[props.moduleType]"
:title="(columnConfig.title as string)"
mode="remote"
value-key="id"
label-key="name"
:type="UserRequestTypeEnum.SYSTEM_ORGANIZATION_LIST"
:placeholder-text="t('project.taskCenter.filterOrgPlaceholderText')"
@search="initData()"
>
</TableFilter>
</template>
<template #projectFilterName="{ columnConfig }">
<TableFilter
v-model:visible="projectFilterVisible"
v-model:status-filters="projectFiltersMap[props.moduleType]"
:title="(columnConfig.title as string)"
mode="remote"
:load-option-params="{ organizationId: appStore.currentOrgId }"
:placeholder-text="t('project.taskCenter.filterProPlaceholderText')"
:type="UserRequestTypeEnum.SYSTEM_ORGANIZATION_PROJECT"
@search="initData()"
>
</TableFilter>
</template>
<template #operationTime="{ record }"> <template #operationTime="{ record }">
<span>{{ dayjs(record.operationTime).format('YYYY-MM-DD HH:mm:ss') }}</span> <span>{{ dayjs(record.operationTime).format('YYYY-MM-DD HH:mm:ss') }}</span>
</template> </template>
@ -145,6 +172,7 @@
import MsBaseTable from '@/components/pure/ms-table/base-table.vue'; import MsBaseTable from '@/components/pure/ms-table/base-table.vue';
import type { BatchActionParams, BatchActionQueryParams, MsTableColumn } from '@/components/pure/ms-table/type'; import type { BatchActionParams, BatchActionQueryParams, MsTableColumn } from '@/components/pure/ms-table/type';
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import { UserRequestTypeEnum } from '@/components/business/ms-user-selector/utils';
import ExecutionStatus from './executionStatus.vue'; import ExecutionStatus from './executionStatus.vue';
import caseAndScenarioReportDrawer from '@/views/api-test/components/caseAndScenarioReportDrawer.vue'; import caseAndScenarioReportDrawer from '@/views/api-test/components/caseAndScenarioReportDrawer.vue';
import ReportDetailDrawer from '@/views/api-test/report/component/reportDetailDrawer.vue'; import ReportDetailDrawer from '@/views/api-test/report/component/reportDetailDrawer.vue';
@ -164,7 +192,7 @@
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
import useModal from '@/hooks/useModal'; import useModal from '@/hooks/useModal';
import useOpenNewPage from '@/hooks/useOpenNewPage'; import useOpenNewPage from '@/hooks/useOpenNewPage';
import { useTableStore } from '@/store'; import { useAppStore, useTableStore } from '@/store';
import { characterLimit } from '@/utils'; import { characterLimit } from '@/utils';
import { hasAnyPermission } from '@/utils/permission'; import { hasAnyPermission } from '@/utils/permission';
@ -177,7 +205,7 @@
const { openNewPage } = useOpenNewPage(); const { openNewPage } = useOpenNewPage();
const tableStore = useTableStore(); const tableStore = useTableStore();
const appStore = useAppStore();
const { openModal } = useModal(); const { openModal } = useModal();
const { t } = useI18n(); const { t } = useI18n();
@ -391,6 +419,23 @@
API_SCENARIO: statusFilterApiScenario.value, API_SCENARIO: statusFilterApiScenario.value,
}); });
const orgFilterVisible = ref<boolean>(false);
const projectFilterVisible = ref<boolean>(false);
const orgApiCaseFilter = ref([]);
const orgApiScenarioFilter = ref([]);
const orgFiltersMap = ref<Record<string, string[]>>({
API_CASE: orgApiCaseFilter.value,
API_SCENARIO: orgApiScenarioFilter.value,
});
const projectApiCaseFilter = ref([]);
const projectApiScenarioFilter = ref([]);
const projectFiltersMap = ref<Record<string, string[]>>({
API_CASE: projectApiCaseFilter.value,
API_SCENARIO: projectApiScenarioFilter.value,
});
function initData() { function initData() {
setLoadListParams({ setLoadListParams({
keyword: keyword.value, keyword: keyword.value,
@ -398,6 +443,8 @@
filter: { filter: {
status: statusFiltersMap.value[props.moduleType], status: statusFiltersMap.value[props.moduleType],
triggerMode: triggerModeFiltersMap.value[props.moduleType], triggerMode: triggerModeFiltersMap.value[props.moduleType],
organizationIds: orgFiltersMap.value[props.moduleType],
projectIds: projectFiltersMap.value[props.moduleType],
}, },
}); });
loadList(); loadList();
@ -502,7 +549,7 @@
function execution(record: any) {} function execution(record: any) {}
onBeforeMount(() => { onBeforeMount(async () => {
initData(); initData();
}); });

View File

@ -64,6 +64,33 @@
</a-option> </a-option>
</a-select> </a-select>
</template> </template>
<template v-if="appStore.packageType === 'enterprise'" #orgFilterName="{ columnConfig }">
<TableFilter
v-model:visible="orgFilterVisible"
v-model:status-filters="orgFiltersMap[props.moduleType]"
:title="(columnConfig.title as string)"
mode="remote"
value-key="id"
label-key="name"
:type="UserRequestTypeEnum.SYSTEM_ORGANIZATION_LIST"
:placeholder-text="t('project.taskCenter.filterOrgPlaceholderText')"
@search="initData()"
>
</TableFilter>
</template>
<template #projectFilterName="{ columnConfig }">
<TableFilter
v-model:visible="projectFilterVisible"
v-model:status-filters="projectFiltersMap[props.moduleType]"
:title="(columnConfig.title as string)"
mode="remote"
:load-option-params="{ organizationId: appStore.currentOrgId }"
:placeholder-text="t('project.taskCenter.filterProPlaceholderText')"
:type="UserRequestTypeEnum.SYSTEM_ORGANIZATION_PROJECT"
@search="initData()"
>
</TableFilter>
</template>
<template #operation="{ record }"> <template #operation="{ record }">
<a-switch <a-switch
v-model="record.enable" v-model="record.enable"
@ -99,6 +126,8 @@
import type { BatchActionParams, BatchActionQueryParams, MsTableColumn } from '@/components/pure/ms-table/type'; import type { BatchActionParams, BatchActionQueryParams, MsTableColumn } from '@/components/pure/ms-table/type';
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import type { ActionsItem } from '@/components/pure/ms-table-more-action/types'; import type { ActionsItem } from '@/components/pure/ms-table-more-action/types';
import { UserRequestTypeEnum } from '@/components/business/ms-user-selector/utils';
import TableFilter from '@/views/case-management/caseManagementFeature/components/tableFilter.vue';
import { import {
batchDisableScheduleOrgTask, batchDisableScheduleOrgTask,
@ -123,7 +152,7 @@
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
import useModal from '@/hooks/useModal'; import useModal from '@/hooks/useModal';
import useOpenNewPage from '@/hooks/useOpenNewPage'; import useOpenNewPage from '@/hooks/useOpenNewPage';
import { useTableStore } from '@/store'; import { useAppStore, useTableStore } from '@/store';
import { hasAnyPermission } from '@/utils/permission'; import { hasAnyPermission } from '@/utils/permission';
import { BatchApiParams } from '@/models/common'; import { BatchApiParams } from '@/models/common';
@ -134,6 +163,8 @@
import { ordAndProjectColumn, resourceTypeMap } from './utils'; import { ordAndProjectColumn, resourceTypeMap } from './utils';
const appStore = useAppStore();
const { openNewPage } = useOpenNewPage(); const { openNewPage } = useOpenNewPage();
const tableStore = useTableStore(); const tableStore = useTableStore();
@ -301,6 +332,23 @@
columns, columns,
}, },
}; };
const orgFilterVisible = ref<boolean>(false);
const projectFilterVisible = ref<boolean>(false);
const orgApiCaseFilter = ref([]);
const orgApiScenarioFilter = ref([]);
const orgFiltersMap = ref<Record<string, string[]>>({
API_IMPORT: orgApiCaseFilter.value,
API_SCENARIO: orgApiScenarioFilter.value,
});
const projectApiCaseFilter = ref([]);
const projectApiScenarioFilter = ref([]);
const projectFiltersMap = ref<Record<string, string[]>>({
API_CASE: projectApiCaseFilter.value,
API_SCENARIO: projectApiScenarioFilter.value,
});
const hasJumpPermission = computed(() => hasAnyPermission(permissionsMap[props.group][props.moduleType].jump)); const hasJumpPermission = computed(() => hasAnyPermission(permissionsMap[props.group][props.moduleType].jump));
const { propsRes, propsEvent, loadList, setLoadListParams, resetSelector } = useTable( const { propsRes, propsEvent, loadList, setLoadListParams, resetSelector } = useTable(
@ -327,6 +375,8 @@
setLoadListParams({ setLoadListParams({
keyword: keyword.value, keyword: keyword.value,
scheduleTagType: props.moduleType, scheduleTagType: props.moduleType,
organizationIds: orgFiltersMap.value[props.moduleType],
projectIds: projectFiltersMap.value[props.moduleType],
}); });
loadList(); loadList();
} }

View File

@ -1,7 +1,10 @@
import type { MsTableColumn } from '@/components/pure/ms-table/type'; import type { MsTableColumn } from '@/components/pure/ms-table/type';
import { useAppStore } from '@/store';
import { TaskCenterEnum } from '@/enums/taskCenter'; import { TaskCenterEnum } from '@/enums/taskCenter';
const appStore = useAppStore();
export const TaskStatus = { export const TaskStatus = {
[TaskCenterEnum.API_CASE]: { [TaskCenterEnum.API_CASE]: {
SUCCESS: { SUCCESS: {
@ -176,6 +179,7 @@ export const ordAndProjectColumn: MsTableColumn = [
title: 'project.belongOrganization', title: 'project.belongOrganization',
dataIndex: 'organizationName', dataIndex: 'organizationName',
slotName: 'organizationName', slotName: 'organizationName',
titleSlotName: 'orgFilterName',
showTooltip: true, showTooltip: true,
showDrag: true, showDrag: true,
width: 200, width: 200,
@ -185,6 +189,7 @@ export const ordAndProjectColumn: MsTableColumn = [
title: 'project.belongProject', title: 'project.belongProject',
dataIndex: 'projectName', dataIndex: 'projectName',
slotName: 'projectName', slotName: 'projectName',
titleSlotName: 'projectFilterName',
showTooltip: true, showTooltip: true,
showDrag: true, showDrag: true,
width: 200, width: 200,

View File

@ -59,4 +59,6 @@ export default {
'project.taskCenter.confirmDisable': 'Confirm disable', 'project.taskCenter.confirmDisable': 'Confirm disable',
'project.taskCenter.enableSuccess': 'Enable successfully', 'project.taskCenter.enableSuccess': 'Enable successfully',
'project.taskCenter.disableSuccess': 'Disable successfully', 'project.taskCenter.disableSuccess': 'Disable successfully',
'project.taskCenter.filterPlaceholderText': 'Please select a project',
'project.taskCenter.filterOrgPlaceholderText': 'Please select an organization',
}; };

View File

@ -57,4 +57,6 @@ export default {
'project.taskCenter.confirmDisable': '确认关闭', 'project.taskCenter.confirmDisable': '确认关闭',
'project.taskCenter.enableSuccess': '开启成功', 'project.taskCenter.enableSuccess': '开启成功',
'project.taskCenter.disableSuccess': '关闭成功', 'project.taskCenter.disableSuccess': '关闭成功',
'project.taskCenter.filterProPlaceholderText': '请选择所属项目',
'project.taskCenter.filterOrgPlaceholderText': '请选择所属组织',
}; };