vue2+element 组件封装使用

1.图片上传组件

使用

  <el-descriptions-item label="附件图片" :span="3"><ImageUpload v-model="picUrl" :fileSize="0"  @getImg="getImg"></ImageUpload></el-descriptions-item>
    // 图片getImg(val){console.log(this.imgList,val)this.imgList = val},

 imageUpload组件封装

<template><div class="component-upload-image"><el-uploadmultiple:action="uploadImgUrl"list-type="picture-card":on-success="handleUploadSuccess":before-upload="handleBeforeUpload":limit="limit":on-error="handleUploadError":on-exceed="handleExceed"ref="imageUpload":on-remove="handleDelete":show-file-list="true":headers="headers":file-list="fileList":on-preview="handlePictureCardPreview":class="{hide: this.fileList.length >= this.limit}"><i class="el-icon-plus"></i></el-upload><!-- 上传提示 --><div class="el-upload__tip" slot="tip" v-if="showTip">请上传<template v-if="fileSize"> 大小不超过 <b style="color: #f56c6c">{{ fileSize }}MB</b> </template><template v-if="fileType"> 格式为 <b style="color: #f56c6c">{{ fileType.join("/") }}</b> </template>的文件</div><el-dialog:visible.sync="dialogVisible"title="预览"width="800"append-to-body><img:src="dialogImageUrl"style="display: block; max-width: 100%; margin: 0 auto"/></el-dialog></div>
</template><script>
import { getToken } from "@/utils/auth";
import { listByIds, delOss } from "@/api/system/oss";export default {props: {value: [String, Object, Array],// 图片数量限制limit: {type: Number,default: 5,},// 大小限制(MB)fileSize: {type: Number,default: 5,},// 文件类型, 例如['png', 'jpg', 'jpeg']fileType: {type: Array,default: () => ["png", "jpg", "jpeg"],},// 是否显示提示isShowTip: {type: Boolean,default: true}},data() {return {number: 0,uploadList: [],dialogImageUrl: "",dialogVisible: false,hideUpload: false,baseUrl: process.env.VUE_APP_BASE_API,uploadImgUrl: process.env.VUE_APP_BASE_API + "/system/oss/upload", // 上传的图片服务器地址headers: {Authorization: "Bearer " + getToken(),},fileList: []};},watch: {value: {async handler(val) {if (val) {// 首先将值转为数组let list;if (Array.isArray(val)) {list = val;} else {await listByIds(val).then(res => {list = res.data;this.$emit("getImg", list);})}// 然后将数组转为对象数组this.fileList = list.map(item => {// 此处name使用ossId 防止删除出现重名item = { name: item.ossId, url: item.url, ossId: item.ossId };return item;});} else {this.fileList = [];return [];}},deep: true,immediate: true}},computed: {// 是否显示提示showTip() {return this.isShowTip && (this.fileType || this.fileSize);},},methods: {// 上传前loading加载handleBeforeUpload(file) {let isImg = false;if (this.fileType.length) {let fileExtension = "";if (file.name.lastIndexOf(".") > -1) {fileExtension = file.name.slice(file.name.lastIndexOf(".") + 1);}isImg = this.fileType.some((type) => {if (file.type.indexOf(type) > -1) return true;if (fileExtension && fileExtension.indexOf(type) > -1) return true;return false;});} else {isImg = file.type.indexOf("image") > -1;}if (!isImg) {this.$modal.msgError(`文件格式不正确, 请上传${this.fileType.join("/")}图片格式文件!`);return false;}if (this.fileSize) {const isLt = file.size / 1024 / 1024 < this.fileSize;if (!isLt) {this.$modal.msgError(`上传头像图片大小不能超过 ${this.fileSize} MB!`);return false;}}this.$modal.loading("正在上传图片,请稍候...");this.number++;},// 文件个数超出handleExceed() {this.$modal.msgError(`上传文件数量不能超过 ${this.limit} 个!`);},// 上传成功回调handleUploadSuccess(res, file) {if (res.code === 200) {this.uploadList.push({ name: res.data.fileName, url: res.data.url, ossId: res.data.ossId });this.uploadedSuccessfully();} else {this.number--;this.$modal.closeLoading();this.$modal.msgError(res.msg);this.$refs.imageUpload.handleRemove(file);this.uploadedSuccessfully();}},// 删除图片handleDelete(file) {const findex = this.fileList.map(f => f.name).indexOf(file.name);if(findex > -1) {let ossId = this.fileList[findex].ossId;// delOss(ossId);this.fileList.splice(findex, 1);this.$emit("input", this.listToString(this.fileList));}},// 上传失败handleUploadError(res) {this.$modal.msgError("上传图片失败,请重试");this.$modal.closeLoading();},// 上传结束处理uploadedSuccessfully() {if (this.number > 0 && this.uploadList.length === this.number) {this.fileList = this.fileList.concat(this.uploadList);this.uploadList = [];this.number = 0;this.$emit("input", this.listToString(this.fileList));this.$modal.closeLoading();}},// 预览handlePictureCardPreview(file) {this.dialogImageUrl = file.url;this.dialogVisible = true;},// 对象转成指定字符串分隔listToString(list, separator) {let strs = "";separator = separator || ",";for (let i in list) {if (list[i].ossId) {strs += list[i].ossId + separator;}}return strs != "" ? strs.substr(0, strs.length - 1) : "";}}
};
</script>
<style scoped lang="scss">
// .el-upload--picture-card 控制加号部分
::v-deep.hide .el-upload--picture-card {display: none;
}
// 去掉动画效果
::v-deep .el-list-enter-active,
::v-deep .el-list-leave-active {transition: all 0s;
}::v-deep .el-list-enter, .el-list-leave-active {opacity: 0;transform: translateY(0);
}
</style>

2.图片回显组件

使用

 <ImagePreview:src="form.picUrl":width="150":height="150"></ImagePreview>

imagePreview组件封装

<template><div><el-image v-for="(item,index) in realSrcList" :key="index":src="`${item}`"fit="cover":style="`width:${realWidth};height:${realHeight};`":preview-src-list="realSrcList"><div slot="error" class="image-slot"><i class="el-icon-picture-outline"></i></div></el-image></div></template><script>export default {name: "ImagePreview",props: {src: {type: String,default: ""},width: {type: [Number, String],default: ""},height: {type: [Number, String],default: ""}},computed: {realSrc() {if (!this.src) {return;}let real_src = this.src.split(",")[0];return real_src;},realSrcList() {if (!this.src) {return;}let real_src_list = this.src.split(",");let srcList = [];real_src_list.forEach(item => {return srcList.push(item);});return srcList;},realWidth() {return typeof this.width == "string" ? this.width : `${this.width}px`;},realHeight() {return typeof this.height == "string" ? this.height : `${this.height}px`;}},
};
</script><style lang="scss" scoped>
.el-image {border-radius: 5px;background-color: #ebeef5;box-shadow: 0 0 5px 1px #ccc;::v-deep .el-image__inner {transition: all 0.3s;cursor: pointer;&:hover {transform: scale(1.2);}}::v-deep .image-slot {display: flex;justify-content: center;align-items: center;width: 100%;height: 100%;color: #909399;font-size: 30px;}
}
</style>

3.文件上传组件

使用

  <el-descriptions-item label="附件文件" :span="3"><FileUpload v-model="fileIds" :fileSize="0"  @getfile="getfile"></FileUpload></el-descriptions-item>
 // 文件列表getfile(val) {this.fileList = val;},

fileUpload组件的封装

<template><div class="upload-file"><el-uploadv-if="isUpload"multiple:action="uploadFileUrl":before-upload="handleBeforeUpload":file-list="fileList":limit="limit":on-error="handleUploadError":on-exceed="handleExceed":on-success="handleUploadSuccess":show-file-list="false":headers="headers"class="upload-file-uploader"ref="fileUpload"><!-- 上传按钮 --><el-button size="mini" type="primary">选取文件</el-button><!-- 上传提示 --><div class="el-upload__tip" slot="tip" v-if="showTip">请上传<template v-if="fileSize">大小不超过 <b style="color: #f56c6c">{{ fileSize }}MB</b></template><template v-if="fileType">格式为 <b style="color: #f56c6c">{{ fileType.join("/") }}</b></template>的文件</div></el-upload><!-- 文件列表 --><transition-groupclass="upload-file-list el-upload-list el-upload-list--text"name="el-fade-in-linear"tag="ul"v-if="isFileList"><li:key="file.url"class="el-upload-list__item ele-upload-list__item-content"style="padding: 0 4px"v-for="(file, index) in fileList"><el-link :href="`${file.url}`" :underline="false" target="_blank"><span class="el-icon-document"> {{ getFileName(file.name) }} </span></el-link><div class="ele-upload-list__item-content-action" v-if="isUpload"><el-link :underline="false" @click="handleDelete(index)" type="danger">删除</el-link></div></li></transition-group></div>
</template><script>
import { getToken } from "@/utils/auth";
import { listByIds, delOss } from "@/api/system/oss";export default {name: "FileUpload",props: {// 值value: [String, Object, Array],// 数量限制limit: {type: Number,default: 5,},// 大小限制(MB)fileSize: {type: Number,default: 5,},// 文件类型, 例如['png', 'jpg', 'jpeg']fileType: {type: Array | Boolean,// default: () => ["doc", "xls", "ppt", "txt", "pdf"],default: () => ["doc", "docx", "xls", "xlsx", "ppt", ".pptx", "pdf"],},// 是否显示提示isShowTip: {type: Boolean,default: true,},// 是否显示文件列表isFileList: {type: Boolean,default: true,},dataIndex: {type: Number | String,default: 0,},// 是否可以上传isUpload: {type: Boolean,default: true,},},data() {return {number: 0,uploadList: [],baseUrl: process.env.VUE_APP_BASE_API,uploadFileUrl: process.env.VUE_APP_BASE_API + "/system/oss/upload", // 上传文件服务器地址headers: {Authorization: "Bearer " + getToken(),},fileList: [],};},watch: {value: {async handler(val) {if (val) {let temp = 1;// 首先将值转为数组let list;if (Array.isArray(val)) {list = val;} else {console.log(val);await listByIds(val).then((res) => {list = res.data.map((oss) => {oss = {name: oss.originalName,url: oss.url,ossId: oss.ossId,};return oss;});this.$emit("getfile", list, this.dataIndex);});}// 然后将数组转为对象数组this.fileList = list.map((item) => {item = { name: item.name, url: item.url, ossId: item.ossId };item.uid = item.uid || new Date().getTime() + temp++;return item;});} else {this.fileList = [];return [];}},deep: true,immediate: true,},},computed: {// 是否显示提示showTip() {return this.isShowTip && (this.fileType || this.fileSize);},},methods: {// 上传前校检格式和大小handleBeforeUpload(file) {// 校检文件类型if (this.fileType) {const fileName = file.name.split(".");const fileExt = fileName[fileName.length - 1];const isTypeOk = this.fileType.indexOf(fileExt) >= 0;if (!isTypeOk) {this.$modal.msgError(`文件格式不正确, 请上传${this.fileType.join("/")}格式文件!`);return false;}}// 校检文件大小if (this.fileSize) {const isLt = file.size / 1024 / 1024 < this.fileSize;if (!isLt) {this.$modal.msgError(`上传文件大小不能超过 ${this.fileSize} MB!`);return false;}}this.$modal.loading("正在上传文件,请稍候...");this.number++;return true;},// 文件个数超出handleExceed() {this.$modal.msgError(`上传文件数量不能超过 ${this.limit} 个!`);},// 上传失败handleUploadError(err) {this.$modal.msgError("上传文件失败,请重试");this.$modal.closeLoading();},// 上传成功回调handleUploadSuccess(res, file) {if (res.code === 200) {this.uploadList.push({name: res.data.fileName,url: res.data.url,ossId: res.data.ossId,});this.uploadedSuccessfully();} else {this.number--;this.$modal.closeLoading();this.$modal.msgError(res.msg);this.$refs.fileUpload.handleRemove(file);this.uploadedSuccessfully();}},// 删除文件handleDelete(index) {let ossId = this.fileList[index].ossId;// delOss(ossId);this.fileList.splice(index, 1);this.$emit("input", this.listToString(this.fileList));},// 上传结束处理uploadedSuccessfully() {if (this.number > 0 && this.uploadList.length === this.number) {this.fileList = this.fileList.concat(this.uploadList);this.uploadList = [];this.number = 0;this.$emit("input", this.listToString(this.fileList));this.$modal.closeLoading();}},// 获取文件名称getFileName(name) {// 如果是url那么取最后的名字 如果不是直接返回if (name.lastIndexOf("/") > -1) {return name.slice(name.lastIndexOf("/") + 1);} else {return name;}},// 对象转成指定字符串分隔listToString(list, separator) {let strs = "";separator = separator || ",";for (let i in list) {strs += list[i].ossId + separator;}return strs != "" ? strs.substr(0, strs.length - 1) : "";},},
};
</script><style scoped lang="scss">
.upload-file-uploader {margin-bottom: 5px;
}
.upload-file-list .el-upload-list__item {border: 1px solid #e4e7ed;line-height: 2;margin-bottom: 10px;position: relative;
}
.upload-file-list .ele-upload-list__item-content {display: flex;justify-content: space-between;align-items: center;color: inherit;
}
.ele-upload-list__item-content-action .el-link {margin-right: 10px;
}
</style>

4.树形图组件

使用

 <orginTreev-model="queryParams.deptId"@change="changexx"></orginTree>
    changexx(value) {console.log(value.label);// this.$set(this.queryParams, "deptId", value);// console.log("this.queryParams", this.queryParams);},

树形组件封装

<template><div><a-tree-select v-model="orgId" style="width: 100%" size="large" :dropdown-style="{ maxHeight: '400px', overflow: 'auto',zIndex:3000 }"placeholder="请选择" allow-clear tree-default-expand-all :disabled="disabled" :tree-data="vorganTreeData":replaceFields="replaceFields" @change="onChange"></a-tree-select></div>
</template><script>
//注意!!!!!
//在modal弹窗组件中使用该组件需要在关闭弹窗方法里清空数据否则会报错
import { userdepList } from "@/api/user/user";
export default {name: 'vorganTree',props: {value: {type: [String, Number],default: undefined,},disabled: {type: Boolean,default: false,},deptId: {type: [String, Number],default: '',},replaceFields: {type: Object,default: () => {return {children: 'children',title: 'label',key: 'id',value: 'id',}},},},data () {return {orgId: this.value,vorganTreeData: [],}},watch: {value: {handler (newVal) {console.log(this.orgId)this.orgId = newVal},immediate: true,},deptId:{handler (newVal) {this.userdepList()},immediate: true,}},mounted () {this.userdepList()},methods: {userdepList () {let mData = { ancestors: this.deptId }console.log(mData)userdepList(mData).then(res => {this.vorganTreeData = res.dataconsole.log(this.vorganTreeData)})},selectClear () {this.orgId = undefined},onChange (value,label,node) {this.$emit('input', value)this.$emit('update:value', value)this.$emit('change',value,label,node.triggerNode)},},
}
</script><style scoped lang="less">
</style>

5.多选表格组件

使用

   <selectInput v-model="form.userIds" :selectRow="rowData"></selectInput>
<template><div><el-selectv-model="selectValue"style="width: 100%"readonlyref="modelRef"clearablemultiplefilterable@focus="modelClick()"@click.native="modelClick()"@clear="selectClear"@remove-tag="selectRemoveClear"allow-createplaceholder="请选择"><el-optionv-for="item in selectRow":key="item[userId]":label="item.nickName":value="item[userId]"></el-option></el-select><el-dialog:title="title"width="1000px":visible.sync="dialogFormVisible"append-to-body><el-row :gutter="24"><el-col :span="16"><el-form ref="form" :model="queryParam" label-width="50px"><el-row><el-col :span="8"><el-form-item label="昵称"><el-input v-model="queryParam.nickName"></el-input></el-form-item></el-col><el-col :span="8"><el-form-item label="账号"><el-input v-model="queryParam.userName"></el-input></el-form-item></el-col><el-col :span="8"><el-buttontype="primary"icon="el-icon-search"@click="searchQuery"style="margin-left: 10px">搜索</el-button><el-button icon="el-icon-refresh-right" @click="searchReset">重置</el-button></el-col></el-row></el-form><el-table:data="tableData"v-loading="loading"border@selection-change="handleSelectionChange"@select="handleSelectionChange"@row-click="handleRowSelectionChange"ref="multipleTable":row-key="userId":reserve-selection="true"><el-table-columntype="selection"width="40":reserve-selection="true"></el-table-column><el-table-column:property="item.property":label="item.label"v-for="(item, index) in column":key="index"></el-table-column></el-table><div style="float: right; margin-top: 10px"><el-paginationbackgroundlayout="prev, pager, next":total="total"@current-change="handleCurrentChange":current-page.sync="currentPage":page-size="5"></el-pagination></div></el-col><el-col :span="7" style="margin: 14px"><span style="font-size: 16px" class="select">已选 {{ selectRow.length }}</span><el-table :data="tableData1" border style="margin-top: 20px"><el-table-columnsize="mini":property="item.property":label="item.label"v-for="(item, index) in column1":key="index"></el-table-column><el-table-column prop="action" label="操作" width="50"><template slot-scope="scope"><el-buttontype="text"size="small"@click="deleteRecord(scope.row)">删除</el-button></template></el-table-column></el-table><div style="float: right; margin-top: 10px"><el-paginationbackgroundlayout="prev, pager, next":total="total1"@current-change="handleCurrentChange1":current-page.sync="currentPage1":page-size="5"></el-pagination></div></el-col></el-row><div slot="footer" class="dialog-footer"><el-button @click="dialogFormVisible = false">关闭</el-button></div></el-dialog></div>
</template><script>
import request from "@/utils/request";
export default {props: {title: {type: String,default: "",},value: {type: String,default: "",},// deptId: {//   type: String,//   default: "",// },},data() {return {dialogFormVisible: false,searchText: "",queryParam: {},column: [{property: "nickName",label: "昵称",},{property: "userName",label: "账号",},{property: "orgName",label: "组织",},],column1: [{property: "nickName",label: "昵称",},{property: "userName",label: "账号",},],options: [],loading: false,tableData: [],tableData1: [],currentPage: 1,currentPage1: 1,total: 0,total1: 0,url: "system/user/list", // 一直urlselectRowKey: [],selectRow: [],selectValue: [],userId: "userId", // 唯一键值deptId: "",};},watch: {value: {immediate: true,handler(val) {if (val) {console.log("11111", val);this.selectValue = val.split(",");this.ifValue();} else {this.selectValue = [];this.selectRow = [];}},},// deptId: {//   immediate: true,//   handler(val) {//     console.log("val", val);//     this.selectValue = [];//     this.selectRow = [];//     // if (val) {//     //   this.selectValue = val.split(",");//     //   this.ifValue();//     // } else {//     //   this.selectValue = [];//     //   this.selectRow = [];//     // }//   },// },selectValue: {deep: true,handler(val) {console.log("selectValue", val);let data = val.join(",");this.$emit("input", data);// this.$emit("getuserList", this.tableData1);},},selectRow: {deep: true,handler(val) {this.pageData();},},// tableData1: {//   deep: true,//   handler(val) {//     this.$bus.$emit("tableData1", this.tableData1);//     // this.$emit("getuserList", this.tableData1);//   },// },},created() {},mounted() {this.$bus.$on("deptId", (data) => {console.log("我是selectInput组件,收到了数据", data);this.deptId = data;});},methods: {// handleInput(value) {//   this.queryParam.userName = value;//   this.queryParam.nickName = value;// },selectRemoveClear(val) {let selectIndex = -1;this.selectRow.forEach((item, index) => {if (item[this.userId] == val) {selectIndex = index;}});this.currentPage1 = 1;this.selectRow.splice(selectIndex, 1);this.$refs.multipleTable.selection.splice(selectIndex, 1);},// 清空selectClear() {this.currentPage1 = 1;this.selectRow = [];this.selectValue = [];this.$refs.multipleTable.clearSelection();},deleteSelect(val) {let selectIndex = 0;this.selectValue.forEach((item, index) => {if (item == val[this.userId]) {selectIndex = index;}});this.selectValue.splice(selectIndex, 1);this.selectRow.splice(selectIndex, 1);this.$refs.multipleTable.selection.splice(selectIndex, 1);},// 用于将选择框中的选项添加到已选中的选项中。//在用户选择选项时调用。//将选择框中新选中的选项添加到 selectRow 和 selectValue 中,并更新页面数据。addSelection() {let mRow = JSON.parse(JSON.stringify(this.selectRow));this.$refs.multipleTable.selection.forEach((item, index) => {if (!mRow.some((items) => {return items[this.userId] == item[this.userId];})) {this.selectRow.push(item);this.selectValue.push(item[this.userId]);}});},//     用于从已选中的选项中删除选项。// 在用户取消选中选项时调用。// 从 selectRow 和 selectValue 中删除取消选中的选项,并更新页面数据。deleteSelection() {let selection = JSON.parse(JSON.stringify(this.$refs.multipleTable.selection));this.selectRow = selection;this.selectValue = selection.map((item) => item[this.userId]);},//     用于检查已选中的选项是否与选择框中的选项一致。// 在组件初始化和 value 数据变化时调用。// 如果已选中的选项与选择框中的选项不一致,则调用 getUser 方法。ifValue() {if (this.selectRow.length != this.selectValue.length) {this.getUser();} else if (this.selectRow.length != 0) {this.selectRow.forEach((item) => {if (!this.selectValue.some((items) => {return item[this.userId] == items;})) {this.getUser();}});}},//     用于根据已选中的选项获取用户信息。// 在 ifValue 方法中调用。// 发送网络请求获取已选中的选项对应的用户信息,并更新 selectRow 中的数据。getUser() {// this.getPage("`system/user/list?id=${this.deptId}`").then((res) => {this.getPage("system/user/list/ids", { ids: this.value }).then((res) => {console.log("this.deptId22222", this.deptId);console.log("有id时候res", res);if (res.code == 200) {let records = res.data.map((item) => {let mItem = {email: item.email,nickName: item.nickName,orgName: item.dept ? item.dept.deptName : "",phonenumber: item.phonenumber,userName: item.userName,};mItem[this.userId] = item[this.userId].toString();return mItem;});this.selectRow = records;}});},//     用于删除指定的记录。// 当用户点击已选中的选项的删除按钮时调用。// 调用 deleteSelect 方法删除指定的选项。deleteRecord(record) {this.deleteSelect(record);},toggleSelection(rows) {if (rows) {rows.forEach((row) => {this.$refs.multipleTable.toggleRowSelection(row);});} else {this.$refs.multipleTable.clearSelection();}},ifSelectValue(val) {let selectBoole = false;this.selectValue.forEach((item, index) => {if (item == val[this.userId]) {selectBoole = true;}});if (selectBoole) {this.deleteSelect(val);return false;} else {this.selectValue.push(val[this.userId]);this.selectRow.push(val);return true;}},handleRowSelectionChange(val, row) {if (this.ifSelectValue(val)) {this.$refs.multipleTable.selection.push(val);} else {}},handleSelectionChange(val, row) {this.multipleSelection = val;this.addSelection();if (val.length < this.selectRow.length) {this.deleteSelection();}},handleCurrentChange(val) {this.currentPage = val;this.getList();},handleCurrentChange1(val) {this.currentPage1 = val;this.pageData();},searchReset() {this.queryParam = {};this.currentPage = 1;this.getList();},searchQuery() {this.currentPage = 1;this.getList();},getList() {this.loading = true;// let mData = {//   pageNum: this.currentPage,//   pageSize: 5,// };// console.log("this.queryParam", this.queryParam);// let deptId = this.deptId;this.getPage(this.url, {deptId: this.deptId,pageNum: this.currentPage,pageSize: 5,...this.queryParam,})// this.getPage(this.url, Object.assign(deptId, mData, this.queryParam)).then((res) => {this.loading = false;if (res.code == 200) {// console.log('this.deptId',this.deptId);console.log(11111111, res);this.tableData = res.rows.map((item) => {let mItem = {email: item.email,nickName: item.nickName,orgName: item.dept ? item.dept.deptName : "",phonenumber: item.phonenumber,userName: item.userName,};mItem[this.userId] = item[this.userId].toString();return mItem;});this.total = res.total;}}).catch((err) => {this.loading = false;});},pageData() {let mm = JSON.parse(JSON.stringify(this.selectRow));this.total1 = mm.length;this.tableData1 = mm.slice((this.currentPage1 - 1) * 5,this.currentPage1 * 5);if (this.currentPage1 > 1 && this.tableData1.length == 0) {this.currentPage1 = this.currentPage1 - 1;this.pageData();}console.log("tableData1", this.tableData1);},getPage(url, query) {return request({url: url,method: "get",params: query,});},ifselection() {let mmblo = false;if (this.selectRow.length != this.$refs.multipleTable.selection.length) {mmblo = true;return mmblo;}this.selectRow.forEach((item) => {if (!this.$refs.multipleTable.selection.some((items) => {return item[this.userId] == items[this.userId];})) {mmblo = true;}});return mmblo;},modelClick() {this.dialogFormVisible = true;this.queryParam = {};setTimeout(() => {this.$refs.modelRef.blur();if (this.ifselection()) {let mms = JSON.parse(JSON.stringify(this.$refs.multipleTable.selection));mms.forEach((item) => {this.$refs.multipleTable.selection.pop();});this.selectRow.forEach((item) => {this.$refs.multipleTable.selection.push(item);});this.currentPage = 1;this.getList();}}, 1);this.getList();},},
};
</script>
<style>
/* .select {
} */
.tableData1 {padding-top: 10px;/* background-color: pink; */
}
</style>

6.时间选择器格式

在el中

 value-format="yyyy-MM-dd"
 <el-date-pickerv-model="queryParams.appraisalYear"type="year"placeholder="选择年"value-format="yyyy-MM-dd"></el-date-picker>

在antd中

valueFormat="YYYY-MM-DD"
 <a-form-model-itemlabel="计划时间:"prop="times":labelCol="labelCol":wrapperCol="wrapperCol"><a-range-picker v-model="model.times" valueFormat="YYYY-MM-DD" /></a-form-model-item>

7.自定义校验规则

prop="liaisonUserIds"

  <el-form-item label="联络员" prop="liaisonUserIds" ref="devType"><selectInputv-model="form.liaisonUserIds":selectRow="rowData"></selectInput></el-form-item>

 rules里面liaisonUserIds

 rules: {userId: [{ required: true, message: "请选则", trigger: "change" }],liaisonUserIds: [{required: true,validator: this.validatePass,message: "请选择",trigger: "change",},],},
 // 自定义校验规则validatePass(rule, value, callback) {console.log("rule", rule, "value", value);if (!this.form.liaisonUserIds) {return callback(new Error("联系人为空")); // 校验失败,传入错误信息} else {callback(); // 校验通过,不传入参数或传入 null}},

8.vue2数组不更新,强制更新

  this.$set(this.form, "deptId", selectedUser.deptId);

9.下载

使用

 downExport() {this.download("/epb/dutyPerformItem/export",{dutyPerformId: 1,type: 0,...this.queryParams,},`党支部履责情况列表.xlsx`);},

在mian.js中全局注册

import { download } from '@/utils/request'

在request

import axios from 'axios'
import { Notification, MessageBox, Message, Loading } from 'element-ui'
import store from '@/store'
import { getToken } from '@/utils/auth'
import errorCode from '@/utils/errorCode'
import { tansParams, blobValidate } from "@/utils/ruoyi";
import cache from '@/plugins/cache'
import { saveAs } from 'file-saver'let downloadLoadingInstance;
// 是否显示重新登录
export let isRelogin = { show: false };axios.defaults.headers['Content-Type'] = 'application/json;charset=utf-8'
// 对应国际化资源文件后缀
axios.defaults.headers['Content-Language'] = 'zh_CN'
// 创建axios实例
const service = axios.create({// axios中请求配置有baseURL选项,表示请求URL公共部分baseURL: process.env.VUE_APP_BASE_API,// 超时timeout: 10000
})// request拦截器
service.interceptors.request.use(config => {// 是否需要设置 tokenconst isToken = (config.headers || {}).isToken === false// 是否需要防止数据重复提交const isRepeatSubmit = (config.headers || {}).repeatSubmit === falseif (getToken() && !isToken) {config.headers['Authorization'] = 'Bearer ' + getToken() // 让每个请求携带自定义token 请根据实际情况自行修改}// get请求映射params参数if (config.method === 'get' && config.params) {let url = config.url + '?' + tansParams(config.params);url = url.slice(0, -1);config.params = {};config.url = url;}if (!isRepeatSubmit && (config.method === 'post' || config.method === 'put')) {const requestObj = {url: config.url,data: typeof config.data === 'object' ? JSON.stringify(config.data) : config.data,time: new Date().getTime()}const sessionObj = cache.session.getJSON('sessionObj')if (sessionObj === undefined || sessionObj === null || sessionObj === '') {cache.session.setJSON('sessionObj', requestObj)} else {const s_url = sessionObj.url;                  // 请求地址const s_data = sessionObj.data;                // 请求数据const s_time = sessionObj.time;                // 请求时间const interval = 1000;                         // 间隔时间(ms),小于此时间视为重复提交if (s_data === requestObj.data && requestObj.time - s_time < interval && s_url === requestObj.url) {const message = '数据正在处理,请勿重复提交';console.warn(`[${s_url}]: ` + message)return Promise.reject(new Error(message))} else {cache.session.setJSON('sessionObj', requestObj)}}}return config
}, error => {console.log(error)Promise.reject(error)
})// 响应拦截器
service.interceptors.response.use(res => {// 未设置状态码则默认成功状态const code = res.data.code || 200;// 获取错误信息const msg = errorCode[code] || res.data.msg || errorCode['default']// 二进制数据则直接返回if (res.request.responseType ===  'blob' || res.request.responseType ===  'arraybuffer') {return res.data}if (code === 401) {if (!isRelogin.show) {isRelogin.show = true;MessageBox.confirm('登录状态已过期,您可以继续留在该页面,或者重新登录', '系统提示', { confirmButtonText: '重新登录', cancelButtonText: '取消', type: 'warning' }).then(() => {isRelogin.show = false;store.dispatch('LogOut').then(() => {location.href = process.env.VUE_APP_CONTEXT_PATH + "index";})}).catch(() => {isRelogin.show = false;});}return Promise.reject('无效的会话,或者会话已过期,请重新登录。')} else if (code === 500) {Message({ message: msg, type: 'error' })return Promise.reject(new Error(msg))} else if (code === 601) {Message({ message: msg, type: 'warning' })return Promise.reject('error')} else if (code !== 200) {Notification.error({ title: msg })return Promise.reject('error')} else {return res.data}},error => {console.log('err' + error)let { message } = error;if (message == "Network Error") {message = "后端接口连接异常";} else if (message.includes("timeout")) {message = "系统接口请求超时";} else if (message.includes("Request failed with status code")) {message = "系统接口" + message.substr(message.length - 3) + "异常";}Message({ message: message, type: 'error', duration: 5 * 1000 })return Promise.reject(error)}
)// 通用下载方法
export function download(url, params, filename, config) {downloadLoadingInstance = Loading.service({ text: "正在下载数据,请稍候", spinner: "el-icon-loading", background: "rgba(0, 0, 0, 0.7)", })return service.post(url, params, {transformRequest: [(params) => { return tansParams(params) }],headers: { 'Content-Type': 'application/x-www-form-urlencoded' },responseType: 'blob',...config}).then(async (data) => {const isBlob = blobValidate(data);if (isBlob) {const blob = new Blob([data])saveAs(blob, filename)} else {const resText = await data.text();const rspObj = JSON.parse(resText);const errMsg = errorCode[rspObj.code] || rspObj.msg || errorCode['default']Message.error(errMsg);}downloadLoadingInstance.close();}).catch((r) => {console.error(r)Message.error('下载文件出现错误,请联系管理员!')downloadLoadingInstance.close();})
}export default service

10.导出

使用

和9是一样的

 handleExport(data) {console.log("data导出", data);this.download("/epb/dutyItem/export",{dutyAppraisalId: data.dutyAppraisalId,},`${data.appraisalYear + "年" + data.deptName + "责任制考核"}.xlsx`);},

11.导入

 <el-uploadclass="upload-demo":action="uploadFileUrl"multiple:show-file-list="false":headers="headers":on-success="handleUploadSuccess"><el-button>导入</el-button></el-upload>
 return {uploadFileUrl: `${process.env.VUE_APP_BASE_API}/epb/workItem/importData${this.form && this.form.id ? "?workId=" + this.form.id : ""}`,// uploadFileUrl: `${process.env.VUE_APP_BASE_API}/epb/workItem/importData?workId=${this.form.id}`, // 上传文件服务器地址// uploadFileUrl: process.env.VUE_APP_BASE_API + "/epb/workItem/importData", // 上传文件服务器地址headers: {Authorization: "Bearer " + getToken(),},};
import { getToken } from "@/utils/auth";
   // 导入文件handleUploadSuccess(res, file) {// console.log("res", res, "file", file);if (res.code == 200) {const tableData = [];let mArr = JSON.parse(JSON.stringify(tableData));// 页面操作  concat() 方法用于合并两个或多个数组。此方法不会更改现有数组,而是返回一个新数组。mArr = mArr.concat(res.data.workItemVos);mArr = mArr.map((item, index) => {return {...item,sort: item.sort + 1, //一开始就遍历加上序号markTime: new Date().getTime(),};});// console.log("mArr", mArr);this.getContentList(mArr);// this.workList = [];this.workList = this.workList.concat(res.data.workLabelVos);console.log("this.workList", this.workList);}},

12.处理分类数据

 // 处理数据handleData(sourceData) {// 分类let classifyWorkContentList = []; //类别数组[...sourceData].forEach((item) => {// .includes()方法返回一个布尔值(true或false)if (classifyWorkContentList.indexOf(item.workContent) === -1) {classifyWorkContentList.push(item.workContent);}});console.log("classifyWorkContentList", classifyWorkContentList);// 按照表格样式修改数据结构// 例如[{workContent: 'xxx', workNormList: [{workItemId: 1111,workId: 11111 ,workContent:'xxx',sort: 1,workNorm:'xxx',No: 1,isDone: 0}]const handleData = classifyWorkContentList.map((item) => {return {workContent: item,workNormList: sourceData.filter((item2) => item2.workContent === item),};});return handleData;},

13.文件回显

两种回显

一种

 <span v-for="(item, index) in form.fileList" :key="index"><a :href="item.fileUrl" target="_blank">{{ item.fileName }}</a>,</span>
 showDetailModal(data) {console.log("data", data);this.title = "详情";this.dialogVisible = true;const fileArr =data?.fileUrl?.split(",")?.map((item, index) => {return {fileUrl: item,fileName: data?.fileName?.split(",")[index],};}) || [];this.form = data;console.log("fileArr", fileArr);this.$set(this.form, "fileList", fileArr);this.getDetail(data.taskId);},

第二种

 <FileUploadv-model="form.fileId":isUpload="false":isShowTip="false":fileType="false"></FileUpload>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/854678.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在哪里可以查到一手的标讯信息?

标讯信息集招投标讯息的简称。在市场上&#xff0c;标讯是一种非常关键的信息&#xff0c;包括招标公告&#xff0c;文件&#xff0c;截止日期等关键内容&#xff0c;便于需求方和供应商进行业务合作。 对于企业来说&#xff0c;及时获取到最新的标讯信息是非常重要的&#xf…

MySQL8新特性:窗口函数

目录 一、 概念二、语法基本语法语法解析进阶语法命名窗口WINDOW AS框架FRAME_CLAUSE 三、窗口函数ROW_NUMBERRANKDENSE_RANKPERCENT_RANKCUME_DISTFIRST_VALUE与LAST_VALUENTH_VALUELAG与LEADNTILE 四、窗口函数使用事项五、窗口函数优化方法六、面试常问&#xff08;持续更新…

设计模式-创建型-04-建造者模式

1、盖房项目需求 1&#xff09;需要建房子&#xff1a;这一过程为打桩、砌墙、封顶2&#xff09;房子有各种各样的&#xff0c;比如普通房&#xff0c;高楼&#xff0c;别墅&#xff0c;各种房子的过程虽然一样&#xff0c;但是要求不要相同的3&#xff09;请编写程序&#xf…

【1990-2023】上市公司高新技术企业数据(Excel+stata)+do代码

数据简介&#xff1a;根据《上市公司资质认定信息文件》 数据进行整理。筛选“认定项目类型” 为“高新技术企业”&#xff1b;筛选“认定对象身份”为“上市公司本身”&#xff0c;根据“认定时间”和“有效期限”判断当年是否为高新技术企业。有效期限通常为3年&#xff0c;缺…

ultralytics版本及对应的更新

Ultralytics Ultralytics 是一家专注于计算机视觉和深度学习工具的公司&#xff0c;尤以其开源的 YOLO (You Only Look Once) 系列深受欢迎。目前&#xff0c;Ultralytics 主要管理和开发 YOLOv5 和 YOLOv8。以下是各个版本的概述及其主要更新&#xff1a; YOLOv5 YOLOv5 是…

力扣-2379. 得到 K 个黑块的最少涂色次数

文章目录 力扣题目工程代码C实现python实现 力扣题目 给你一个长度为 n 下标从 0 开始的字符串 blocks &#xff0c;blocks[i] 要么是 ‘W’ 要么是 ‘B’ &#xff0c;表示第 i 块的颜色。字符 ‘W’ 和 ‘B’ 分别表示白色和黑色。 给你一个整数 k &#xff0c;表示想要 连…

华为手环9省电设置

1、 手环开启熄屏显示续航约3天&#xff0c;原因为屏幕持续常亮显示&#xff1b;如不需要可通过手环“设置->表盘->熄屏显示”路径进行关闭&#xff1b; 2、 手环具备后台健康自动检测功能&#xff0c;您可根据需要选择是否使用或关闭&#xff1a; &#xff08;1&#x…

实用软件分享-----一款免费的人工智能替换face的神器

专栏介绍:本专栏主要分享一些实用的软件(Po Jie版); 声明1:软件不保证时效性;只能保证在写本文时,该软件是可用的;不保证后续时间该软件能一直正常运行;不保证没有bug;如果软件不可用了,我知道后会第一时间在题目上注明(已失效)。介意者请勿订阅。 声明2:本专栏的…

DoIP——step2:车辆发现

文章目录 前言一、IP地址配置1.1 AutoIP1.2 DHCP1.3 DoIP实体的IP地址配置流程二、车辆发现车辆声明报文内容如下:前言 完成诊断设备到车辆的物理连接并通过激活线使能诊断连接后边缘节点将会将连接状态传递至应用层,在开始车辆发现过程之前,需要先进行各自的IP地址配置,获…

海外媒体发稿渠道和方法有哪些?如何选择靠谱的国外媒体发稿服务商?

在选择海外媒体发稿服务商时&#xff0c;以下是一些关键点可以帮助您找到靠谱的服务商&#xff1a; 服务商的经验和口碑&#xff1a;查找该服务商在行业内的声誉和客户评价。拥有丰富经验和良好口碑的服务商通常更可靠。 媒体资源和覆盖范围&#xff1a;了解服务商所能提供的媒…

vue中的模板语法

第1部分&#xff1a;引言 1.1 介绍Vue框架 Vue.js&#xff0c;简称Vue&#xff0c;是一个用于构建用户界面的渐进式JavaScript框架。自从2014年首次发布以来&#xff0c;Vue因其简洁、易学和高效而迅速赢得了开发者的青睐。Vue的核心库只关注视图层&#xff0c;易于与其他库或…

Python并行编程技术与方法详解:线程池、进程池及优化策略

目录 一、引言 二、线程池 线程池的概念 Python中的线程池实现 线程池的优缺点 三、进程池 进程池的概念 Python中的进程池实现 进程池的优缺点 四、优化策略 合理设置线程池和进程池的大小 任务的拆分与合并 使用队列和锁等同步机制 选择合适的并行框架 五、异…

【机器学习300问】121、RNN是如何生成文本的?

当RNN模型训练好后&#xff0c;如何让他生成一个句子&#xff1f;其实就是一个RNN前向传播的过程。通常遵循以下的步骤。 &#xff08;1&#xff09;初始化 文本生成可以什么都不给&#xff0c;让他生成一首诗。首先&#xff0c;你需要确定采样的起始点。这可以是一个特殊的开…

MySQL-DDL(Data Definition Language)

078-对表结构进行增删改操作 增删改表结构DDL&#xff08;Data Definition Language&#xff09; 创建一个学生表 create table t_student( no bigint, name varchar(255), age int comment 年龄 );查看建表语句 show create table t_student;修改表名 alter table 表名 r…

关于scrapy模块中间件的简单理解

摘要 Scrapy爬虫模块是爬虫程序员使用最多的一个模块&#xff0c;它以快速&#xff0c;高层次等优势&#xff0c;深受爬虫工作者的喜爱&#xff0c;其中Scrapy的中间件功能也极其重要。中间件处于引擎和爬虫之间的钩子框架&#xff0c;允许开发者自定义处理请求和响应的过程。…

富格林:需知k线阻挠交易欺诈

富格林指出&#xff0c;尽管现货黄金的投资机会十分充足&#xff0c;但也并不代表能够轻松获得盈利。尤其是近段时间行情较多&#xff0c;做单的难度无疑也会有所提升&#xff0c;所以要想阻挠交易欺诈在黄金市场当中盈利&#xff0c;就得学会分析现货黄金K线图。下面富格林将总…

Linux网络 - HTTP协议

文章目录 前言一、HTTP协议1.urlurl特殊字符 requestrespond 总结 前言 上一章内容我们讲了在应用层制定了我们自己自定义的协议、序列化和反序列化。 协议的制定相对来讲还是比较麻烦的&#xff0c;不过既然应用层的协议制定是必要的&#xff0c;那么肯定已经有许多计算机大佬…

C++使用spdlog输出日志文件

参考博客&#xff1a; 日志记录库 spdlog 基础使用_spdlog 写日志-CSDN博客 GitHub - gabime/spdlog: Fast C logging library. 首先在github上下载spdlog源码&#xff0c;不想编译成库的话&#xff0c;可以直接使用源码&#xff0c;将include文件夹下的spdlog文件夹&#x…

网络:用2个IP地址描述一个连接

用2个IP地址描述一个连接。这是在阅读了《TCP/IP指南》后的感想&#xff0c;与工业标准不同&#xff0c;需注意区分。 如果一个IP地址有48位&#xff0c;则用96位描述一个连接 对于单播&#xff0c;是每个IP分别描述位置。位置包括&#xff1a;邮局编号主机编号&#xff0c;采用…

Qt 信号与槽机制概述

在 Qt 中&#xff0c;emit 是一个用于发射信号的关键字。它是 Qt 的信号与槽&#xff08;Signals and Slots&#xff09;机制的一部分&#xff0c;用于在某个条件满足时通知其他对象。发射信号是一种实现对象之间通信的方式&#xff0c;特别是在事件驱动编程模型中。 Qt 信号与…