refactor: ♻️ 用户代码优化
This commit is contained in:
@@ -248,6 +248,8 @@ export interface UserPageQuery extends PageQuery {
|
|||||||
|
|
||||||
/** 用户分页对象 */
|
/** 用户分页对象 */
|
||||||
export interface UserPageVO {
|
export interface UserPageVO {
|
||||||
|
/** 用户ID */
|
||||||
|
id: number;
|
||||||
/** 用户头像URL */
|
/** 用户头像URL */
|
||||||
avatar?: string;
|
avatar?: string;
|
||||||
/** 创建时间 */
|
/** 创建时间 */
|
||||||
@@ -258,8 +260,6 @@ export interface UserPageVO {
|
|||||||
email?: string;
|
email?: string;
|
||||||
/** 性别 */
|
/** 性别 */
|
||||||
gender?: number;
|
gender?: number;
|
||||||
/** 用户ID */
|
|
||||||
id?: number;
|
|
||||||
/** 手机号 */
|
/** 手机号 */
|
||||||
mobile?: string;
|
mobile?: string;
|
||||||
/** 用户昵称 */
|
/** 用户昵称 */
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<el-dialog
|
<el-dialog
|
||||||
v-model="dialogVisible"
|
v-model="visible"
|
||||||
:align-center="true"
|
:align-center="true"
|
||||||
title="导入数据"
|
title="导入数据"
|
||||||
width="600px"
|
width="600px"
|
||||||
@@ -52,7 +52,7 @@
|
|||||||
<el-button
|
<el-button
|
||||||
type="primary"
|
type="primary"
|
||||||
:disabled="importFormData.files.length === 0"
|
:disabled="importFormData.files.length === 0"
|
||||||
@click="handleSubmit"
|
@click="handleUpload"
|
||||||
>
|
>
|
||||||
确 定
|
确 定
|
||||||
</el-button>
|
</el-button>
|
||||||
@@ -63,24 +63,14 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { type UploadUserFile } from "element-plus";
|
import { ElMessage, type UploadUserFile } from "element-plus";
|
||||||
|
|
||||||
import UserAPI from "@/api/system/user";
|
import UserAPI from "@/api/system/user";
|
||||||
|
|
||||||
const props = defineProps({
|
const emit = defineEmits(["import-success"]);
|
||||||
visible: {
|
const visible = defineModel("modelValue", {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
|
required: true,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const emit = defineEmits(["update:visible", "import-success"]);
|
|
||||||
|
|
||||||
const dialogVisible = computed({
|
|
||||||
get: () => props.visible,
|
|
||||||
set: (val) => {
|
|
||||||
emit("update:visible", val);
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const importFormRef = ref(null);
|
const importFormRef = ref(null);
|
||||||
@@ -96,41 +86,16 @@ const importFormRules = {
|
|||||||
files: [{ required: true, message: "文件不能为空", trigger: "blur" }],
|
files: [{ required: true, message: "文件不能为空", trigger: "blur" }],
|
||||||
};
|
};
|
||||||
|
|
||||||
const emptyFileList = () => {
|
// 文件超出个数限制
|
||||||
importFormData.files.length = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = () => {
|
|
||||||
emptyFileList();
|
|
||||||
dialogVisible.value = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSubmit = async () => {
|
|
||||||
if (!importFormData.files.length) {
|
|
||||||
ElMessage.warning("请选择文件");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await UserAPI.import(1, importFormData.files[0].raw as File);
|
|
||||||
ElMessage.success("上传成功");
|
|
||||||
emit("import-success");
|
|
||||||
handleClose();
|
|
||||||
} catch (error) {
|
|
||||||
ElMessage.error("上传失败");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleFileExceed = () => {
|
const handleFileExceed = () => {
|
||||||
ElMessage.warning("只能上传一个文件");
|
ElMessage.warning("只能上传一个文件");
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// 下载导入模板
|
||||||
const handleDownloadTemplate = () => {
|
const handleDownloadTemplate = () => {
|
||||||
UserAPI.downloadTemplate().then((response: any) => {
|
UserAPI.downloadTemplate().then((response: any) => {
|
||||||
const fileData = response.data;
|
const fileData = response.data;
|
||||||
const fileName = decodeURI(
|
const fileName = decodeURI(response.headers["content-disposition"].split(";")[1].split("=")[1]);
|
||||||
response.headers["content-disposition"].split(";")[1].split("=")[1]
|
|
||||||
);
|
|
||||||
const fileType =
|
const fileType =
|
||||||
"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8";
|
"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8";
|
||||||
|
|
||||||
@@ -148,4 +113,27 @@ const handleDownloadTemplate = () => {
|
|||||||
window.URL.revokeObjectURL(downloadUrl);
|
window.URL.revokeObjectURL(downloadUrl);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// 上传文件
|
||||||
|
const handleUpload = async () => {
|
||||||
|
if (!importFormData.files.length) {
|
||||||
|
ElMessage.warning("请选择文件");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await UserAPI.import(1, importFormData.files[0].raw as File);
|
||||||
|
ElMessage.success("上传成功");
|
||||||
|
emit("import-success");
|
||||||
|
handleClose();
|
||||||
|
} catch (error) {
|
||||||
|
ElMessage.error("上传失败");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 关闭弹窗
|
||||||
|
const handleClose = () => {
|
||||||
|
importFormData.files.length = 0;
|
||||||
|
visible.value = false;
|
||||||
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -47,14 +47,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<el-button type="primary" @click="handleQuery">
|
<el-button type="primary" icon="search" @click="handleQuery">搜索</el-button>
|
||||||
<template #icon><Search /></template>
|
<el-button icon="refresh" @click="handleResetQuery">重置</el-button>
|
||||||
搜索
|
|
||||||
</el-button>
|
|
||||||
<el-button @click="handleResetQuery">
|
|
||||||
<template #icon><Refresh /></template>
|
|
||||||
重置
|
|
||||||
</el-button>
|
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-form>
|
</el-form>
|
||||||
</div>
|
</div>
|
||||||
@@ -65,18 +59,18 @@
|
|||||||
<el-button
|
<el-button
|
||||||
v-hasPerm="['sys:user:add']"
|
v-hasPerm="['sys:user:add']"
|
||||||
type="success"
|
type="success"
|
||||||
|
icon="plus"
|
||||||
@click="handleOpenDialog()"
|
@click="handleOpenDialog()"
|
||||||
>
|
>
|
||||||
<template #icon><Plus /></template>
|
|
||||||
新增
|
新增
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
v-hasPerm="['sys:user:delete']"
|
v-hasPerm="['sys:user:delete']"
|
||||||
type="danger"
|
type="danger"
|
||||||
:disabled="removeIds.length === 0"
|
icon="delete"
|
||||||
|
:disabled="selectIds.length === 0"
|
||||||
@click="handleDelete()"
|
@click="handleDelete()"
|
||||||
>
|
>
|
||||||
<template #icon><Delete /></template>
|
|
||||||
删除
|
删除
|
||||||
</el-button>
|
</el-button>
|
||||||
</div>
|
</div>
|
||||||
@@ -101,88 +95,56 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<el-table
|
<el-table v-loading="loading" :data="pageData" @selection-change="handleSelectionChange">
|
||||||
v-loading="loading"
|
|
||||||
:data="pageData"
|
|
||||||
@selection-change="handleSelectionChange"
|
|
||||||
>
|
|
||||||
<el-table-column type="selection" width="50" align="center" />
|
<el-table-column type="selection" width="50" align="center" />
|
||||||
<el-table-column label="编号" align="center" prop="id" width="60" />
|
<el-table-column label="编号" align="center" prop="id" width="60" />
|
||||||
<el-table-column label="用户名" align="center" prop="username" />
|
<el-table-column label="用户名" align="center" prop="username" />
|
||||||
<el-table-column
|
<el-table-column label="昵称" width="100" align="center" prop="nickname" />
|
||||||
label="昵称"
|
|
||||||
width="100"
|
|
||||||
align="center"
|
|
||||||
prop="nickname"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<el-table-column label="性别" width="100" align="center">
|
<el-table-column label="性别" width="100" align="center">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<DictLabel v-model="scope.row.gender" code="gender" />
|
<DictLabel v-model="scope.row.gender" code="gender" />
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
<el-table-column label="部门" width="120" align="center" prop="deptName" />
|
||||||
<el-table-column
|
<el-table-column label="手机号码" align="center" prop="mobile" width="120" />
|
||||||
label="部门"
|
<el-table-column label="状态" align="center" prop="status" width="80">
|
||||||
width="120"
|
|
||||||
align="center"
|
|
||||||
prop="deptName"
|
|
||||||
/>
|
|
||||||
<el-table-column
|
|
||||||
label="手机号码"
|
|
||||||
align="center"
|
|
||||||
prop="mobile"
|
|
||||||
width="120"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<el-table-column
|
|
||||||
label="状态"
|
|
||||||
align="center"
|
|
||||||
prop="status"
|
|
||||||
width="80"
|
|
||||||
>
|
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<el-tag :type="scope.row.status == 1 ? 'success' : 'info'">
|
<el-tag :type="scope.row.status == 1 ? 'success' : 'info'">
|
||||||
{{ scope.row.status == 1 ? "正常" : "禁用" }}
|
{{ scope.row.status == 1 ? "正常" : "禁用" }}
|
||||||
</el-tag>
|
</el-tag>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column
|
<el-table-column label="创建时间" align="center" prop="createTime" width="120" />
|
||||||
label="创建时间"
|
|
||||||
align="center"
|
|
||||||
prop="createTime"
|
|
||||||
width="120"
|
|
||||||
/>
|
|
||||||
<el-table-column label="操作" fixed="right" width="220">
|
<el-table-column label="操作" fixed="right" width="220">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<el-button
|
<el-button
|
||||||
v-hasPerm="['sys:user:password:reset']"
|
v-hasPerm="['sys:user:password:reset']"
|
||||||
type="primary"
|
type="primary"
|
||||||
|
icon="RefreshLeft"
|
||||||
size="small"
|
size="small"
|
||||||
link
|
link
|
||||||
@click="hancleResetPassword(scope.row)"
|
@click="hancleResetPassword(scope.row)"
|
||||||
>
|
>
|
||||||
<template #icon><RefreshLeft /></template>
|
|
||||||
重置密码
|
重置密码
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
v-hasPerm="['sys:user:edit']"
|
v-hasPerm="['sys:user:edit']"
|
||||||
type="primary"
|
type="primary"
|
||||||
|
icon="edit"
|
||||||
link
|
link
|
||||||
size="small"
|
size="small"
|
||||||
@click="handleOpenDialog(scope.row.id)"
|
@click="handleOpenDialog(scope.row.id)"
|
||||||
>
|
>
|
||||||
<template #icon><Edit /></template>
|
|
||||||
编辑
|
编辑
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
v-hasPerm="['sys:user:delete']"
|
v-hasPerm="['sys:user:delete']"
|
||||||
type="danger"
|
type="danger"
|
||||||
|
icon="delete"
|
||||||
link
|
link
|
||||||
size="small"
|
size="small"
|
||||||
@click="handleDelete(scope.row.id)"
|
@click="handleDelete(scope.row.id)"
|
||||||
>
|
>
|
||||||
<template #icon><Delete /></template>
|
|
||||||
删除
|
删除
|
||||||
</el-button>
|
</el-button>
|
||||||
</template>
|
</template>
|
||||||
@@ -207,12 +169,7 @@
|
|||||||
append-to-body
|
append-to-body
|
||||||
@close="handleCloseDialog"
|
@close="handleCloseDialog"
|
||||||
>
|
>
|
||||||
<el-form
|
<el-form ref="userFormRef" :model="formData" :rules="rules" label-width="80px">
|
||||||
ref="userFormRef"
|
|
||||||
:model="formData"
|
|
||||||
:rules="rules"
|
|
||||||
label-width="80px"
|
|
||||||
>
|
|
||||||
<el-form-item label="用户名" prop="username">
|
<el-form-item label="用户名" prop="username">
|
||||||
<el-input
|
<el-input
|
||||||
v-model="formData.username"
|
v-model="formData.username"
|
||||||
@@ -252,19 +209,11 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="手机号码" prop="mobile">
|
<el-form-item label="手机号码" prop="mobile">
|
||||||
<el-input
|
<el-input v-model="formData.mobile" placeholder="请输入手机号码" maxlength="11" />
|
||||||
v-model="formData.mobile"
|
|
||||||
placeholder="请输入手机号码"
|
|
||||||
maxlength="11"
|
|
||||||
/>
|
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="邮箱" prop="email">
|
<el-form-item label="邮箱" prop="email">
|
||||||
<el-input
|
<el-input v-model="formData.email" placeholder="请输入邮箱" maxlength="50" />
|
||||||
v-model="formData.email"
|
|
||||||
placeholder="请输入邮箱"
|
|
||||||
maxlength="50"
|
|
||||||
/>
|
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="状态" prop="status">
|
<el-form-item label="状态" prop="status">
|
||||||
@@ -287,25 +236,13 @@
|
|||||||
</template>
|
</template>
|
||||||
</el-drawer>
|
</el-drawer>
|
||||||
|
|
||||||
<!-- 用户导入弹窗 -->
|
<!-- 用户导入 -->
|
||||||
<UserImport
|
<UserImport v-model="importDialogVisible" @import-success="handleQuery()" />
|
||||||
v-model:visible="importDialogVisible"
|
|
||||||
@import-success="handleUserImportSuccess()"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
defineOptions({
|
import UserAPI, { UserForm, UserPageQuery, UserPageVO } from "@/api/system/user";
|
||||||
name: "User",
|
|
||||||
inheritAttrs: false,
|
|
||||||
});
|
|
||||||
|
|
||||||
import UserAPI, {
|
|
||||||
UserForm,
|
|
||||||
UserPageQuery,
|
|
||||||
UserPageVO,
|
|
||||||
} from "@/api/system/user";
|
|
||||||
|
|
||||||
import DeptAPI from "@/api/system/dept";
|
import DeptAPI from "@/api/system/dept";
|
||||||
import RoleAPI from "@/api/system/role";
|
import RoleAPI from "@/api/system/role";
|
||||||
@@ -313,38 +250,32 @@ import RoleAPI from "@/api/system/role";
|
|||||||
import DeptTree from "./components/DeptTree.vue";
|
import DeptTree from "./components/DeptTree.vue";
|
||||||
import UserImport from "./components/UserImport.vue";
|
import UserImport from "./components/UserImport.vue";
|
||||||
|
|
||||||
|
defineOptions({
|
||||||
|
name: "User",
|
||||||
|
inheritAttrs: false,
|
||||||
|
});
|
||||||
|
|
||||||
const queryFormRef = ref(ElForm);
|
const queryFormRef = ref(ElForm);
|
||||||
const userFormRef = ref(ElForm);
|
const userFormRef = ref(ElForm);
|
||||||
|
|
||||||
const loading = ref(false);
|
|
||||||
const removeIds = ref([]);
|
|
||||||
const total = ref(0);
|
|
||||||
const pageData = ref<UserPageVO[]>();
|
|
||||||
// 部门下拉数据源
|
|
||||||
const deptOptions = ref<OptionType[]>();
|
|
||||||
// 角色下拉数据源
|
|
||||||
const roleOptions = ref<OptionType[]>();
|
|
||||||
// 用户查询参数
|
|
||||||
const queryParams = reactive<UserPageQuery>({
|
const queryParams = reactive<UserPageQuery>({
|
||||||
pageNum: 1,
|
pageNum: 1,
|
||||||
pageSize: 10,
|
pageSize: 10,
|
||||||
});
|
});
|
||||||
|
|
||||||
// 用户弹窗
|
const pageData = ref<UserPageVO[]>();
|
||||||
|
const total = ref(0);
|
||||||
|
const loading = ref(false);
|
||||||
|
|
||||||
const dialog = reactive({
|
const dialog = reactive({
|
||||||
visible: false,
|
visible: false,
|
||||||
title: "",
|
title: "新增用户",
|
||||||
});
|
});
|
||||||
|
|
||||||
// 导入弹窗显示状态
|
|
||||||
const importDialogVisible = ref(false);
|
|
||||||
|
|
||||||
// 用户表单数据
|
|
||||||
const formData = reactive<UserForm>({
|
const formData = reactive<UserForm>({
|
||||||
status: 1,
|
status: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
// 用户表单校验规则
|
|
||||||
const rules = reactive({
|
const rules = reactive({
|
||||||
username: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
|
username: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
|
||||||
nickname: [{ required: true, message: "用户昵称不能为空", trigger: "blur" }],
|
nickname: [{ required: true, message: "用户昵称不能为空", trigger: "blur" }],
|
||||||
@@ -366,6 +297,15 @@ const rules = reactive({
|
|||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 选中的用户ID
|
||||||
|
const selectIds = ref<number[]>([]);
|
||||||
|
// 部门下拉数据源
|
||||||
|
const deptOptions = ref<OptionType[]>();
|
||||||
|
// 角色下拉数据源
|
||||||
|
const roleOptions = ref<OptionType[]>();
|
||||||
|
// 导入弹窗显示状态
|
||||||
|
const importDialogVisible = ref(false);
|
||||||
|
|
||||||
// 查询
|
// 查询
|
||||||
function handleQuery() {
|
function handleQuery() {
|
||||||
loading.value = true;
|
loading.value = true;
|
||||||
@@ -388,24 +328,19 @@ function handleResetQuery() {
|
|||||||
handleQuery();
|
handleQuery();
|
||||||
}
|
}
|
||||||
|
|
||||||
// 行复选框选中记录选中ID集合
|
// 选中项发生变化
|
||||||
function handleSelectionChange(selection: any) {
|
function handleSelectionChange(selection: any[]) {
|
||||||
removeIds.value = selection.map((item: any) => item.id);
|
selectIds.value = selection.map((item) => item.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 重置密码
|
// 重置密码
|
||||||
function hancleResetPassword(row: { [key: string]: any }) {
|
function hancleResetPassword(row: UserPageVO) {
|
||||||
ElMessageBox.prompt(
|
ElMessageBox.prompt("请输入用户【" + row.username + "】的新密码", "重置密码", {
|
||||||
"请输入用户「" + row.username + "」的新密码",
|
|
||||||
"重置密码",
|
|
||||||
{
|
|
||||||
confirmButtonText: "确定",
|
confirmButtonText: "确定",
|
||||||
cancelButtonText: "取消",
|
cancelButtonText: "取消",
|
||||||
}
|
}).then(
|
||||||
).then(
|
|
||||||
({ value }) => {
|
({ value }) => {
|
||||||
if (!value || value.length < 6) {
|
if (!value || value.length < 6) {
|
||||||
// 检查密码是否为空或少于6位
|
|
||||||
ElMessage.warning("密码至少需要6位字符,请重新输入");
|
ElMessage.warning("密码至少需要6位字符,请重新输入");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -451,7 +386,7 @@ function handleCloseDialog() {
|
|||||||
formData.status = 1;
|
formData.status = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 表单提交
|
// 提交用户表单(防抖处理)
|
||||||
const handleSubmit = useThrottleFn(() => {
|
const handleSubmit = useThrottleFn(() => {
|
||||||
userFormRef.value.validate((valid: any) => {
|
userFormRef.value.validate((valid: any) => {
|
||||||
if (valid) {
|
if (valid) {
|
||||||
@@ -478,9 +413,13 @@ const handleSubmit = useThrottleFn(() => {
|
|||||||
});
|
});
|
||||||
}, 3000);
|
}, 3000);
|
||||||
|
|
||||||
// 删除用户
|
/**
|
||||||
|
* 删除用户
|
||||||
|
*
|
||||||
|
* @param id 用户ID
|
||||||
|
*/
|
||||||
function handleDelete(id?: number) {
|
function handleDelete(id?: number) {
|
||||||
const userIds = [id || removeIds.value].join(",");
|
const userIds = [id || selectIds.value].join(",");
|
||||||
if (!userIds) {
|
if (!userIds) {
|
||||||
ElMessage.warning("请勾选删除项");
|
ElMessage.warning("请勾选删除项");
|
||||||
return;
|
return;
|
||||||
@@ -505,23 +444,17 @@ function handleDelete(id?: number) {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
// 打开导入弹窗 *
|
|
||||||
|
// 打开导入弹窗
|
||||||
function handleOpenImportDialog() {
|
function handleOpenImportDialog() {
|
||||||
importDialogVisible.value = true;
|
importDialogVisible.value = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 导入用户成功
|
|
||||||
function handleUserImportSuccess() {
|
|
||||||
handleQuery();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 导出用户
|
// 导出用户
|
||||||
function handleExport() {
|
function handleExport() {
|
||||||
UserAPI.export(queryParams).then((response: any) => {
|
UserAPI.export(queryParams).then((response: any) => {
|
||||||
const fileData = response.data;
|
const fileData = response.data;
|
||||||
const fileName = decodeURI(
|
const fileName = decodeURI(response.headers["content-disposition"].split(";")[1].split("=")[1]);
|
||||||
response.headers["content-disposition"].split(";")[1].split("=")[1]
|
|
||||||
);
|
|
||||||
const fileType =
|
const fileType =
|
||||||
"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8";
|
"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8";
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user