我们发布啦

This commit is contained in:
张乐
2020-08-13 16:12:57 +08:00
parent ec2ddb4e10
commit acd9b0cb1a
1884 changed files with 344865 additions and 2 deletions

View File

@@ -0,0 +1,148 @@
<template>
<el-dialog
title="用户等级"
:visible.sync="dialogVisible"
width="500px"
:before-close="handleClose">
<el-form :model="formValidate" :rules="rules" ref="formValidate" label-width="100px" class="demo-ruleForm" v-loading="loading">
<el-form-item label="等级名称" prop="name">
<el-input v-model="formValidate.name" placeholder="请输入等级名称"></el-input>
</el-form-item>
<el-form-item label="等级" prop="grade">
<el-input v-model.number="formValidate.grade" placeholder="请输入等级"></el-input>
</el-form-item>
<el-form-item label="享受折扣" prop="discount">
<el-input v-model="formValidate.discount" placeholder="请输入享受折扣"></el-input>
</el-form-item>
<el-form-item label="经验" prop="experience">
<el-input v-model.number="formValidate.experience" placeholder="请输入经验"></el-input>
</el-form-item>
<el-form-item label="图标:" prop="icon">
<div class="upLoadPicBox" @click="modalPicTap('1', 'icon')">
<div v-if="formValidate.icon" class="pictrue"><img :src="formValidate.icon"></div>
<div v-else class="upLoad">
<i class="el-icon-camera cameraIconfont" />
</div>
</div>
</el-form-item>
<el-form-item label="用户背景:" required prop="image">
<div class="upLoadPicBox" @click="modalPicTap('1', 'image')">
<div v-if="formValidate.image" class="pictrue"><img :src="formValidate.image"></div>
<div v-else class="upLoad">
<i class="el-icon-camera cameraIconfont" />
</div>
</div>
</el-form-item>
<el-form-item label="是否显示" required>
<el-radio-group v-model="formValidate.isShow">
<el-radio :label="true" class="radio">显示</el-radio>
<el-radio :label="false">隐藏</el-radio>
</el-radio-group>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="resetForm('formValidate')"> </el-button>
<el-button type="primary" @click="submitForm('formValidate')"> </el-button>
</span>
</el-dialog>
</template>
<script>
import { levelSaveApi, levelInfoApi, levelUpdateApi } from '@/api/user'
const obj = {
name:'',
grade: 1,
discount: '',
icon: '',
image: '',
isShow: true,
id: null
}
export default {
name: "CreatGrade",
data() {
return {
dialogVisible: false,
formValidate: Object.assign({},obj),
loading: false,
rules:{
name: [
{ required: true, message: '请输入等级名称', trigger: 'blur' }
],
grade: [
{ required: true, message: '请输入等级', trigger: 'blur' },
{ type: 'number', message: '等级必须为数字值'}
],
discount: [
{ required: true, message: '请输入折扣', trigger: 'blur'},
],
experience: [
{ required: true, message: '请输入经验', trigger: 'blur'},
{ type: 'number', message: '经验必须为数字值'}
],
icon: [
{ required: true, message: '请上传图标', trigger: 'change' }
],
image: [
{ required: true, message: '请上传用户背景', trigger: 'change' }
]
}
}
},
methods:{
// 点击商品图
modalPicTap (tit, num) {
const _this = this
this.$modalUpload(function(img) {
tit==='1'&& num === 'icon' ? _this.formValidate.icon = img[0].sattDir : _this.formValidate.image = img[0].sattDir
},tit , 'user')
},
info(id) {
this.loading = true
levelInfoApi({id: id}).then(res => {
this.formValidate = res
this.loading = false
}).catch(() => {
this.loading = false
})
},
handleClose() {
this.dialogVisible = false
this.$refs['formValidate'].resetFields();
},
submitForm(formName) {
this.$refs[formName].validate((valid) => {
if (valid) {
this.loading = true
this.formValidate.id ? levelUpdateApi({id:this.formValidate.id}, this.formValidate).then(res => {
this.$message.success('编辑成功')
this.loading = false
this.handleClose()
this.formValidate = Object.assign({},obj)
this.$parent.getList()
}).catch(() => {
this.loading = false
}): levelSaveApi(this.formValidate).then(res => {
this.$message.success('添加成功')
this.loading = false
this.handleClose()
this.$parent.getList()
}).catch(() => {
this.loading = false
})
} else {
return false;
}
});
},
resetForm(formName) {
this.dialogVisible = false
this.$refs[formName].resetFields();
}
}
}
</script>
<style scoped>
</style>

View File

@@ -0,0 +1,185 @@
<template>
<div class="divBox">
<el-card class="box-card">
<div slot="header" class="clearfix">
<div class="container">
<el-form :inline="true" size="small">
<el-form-item label="用户状态:" class="mr10">
<el-select v-model="tableFrom.isShow" clearable placeholder="请选择用户状态" @change="seachList" class="selWidth">
<el-option label="全部" value=""></el-option>
<el-option label="显示" value="1"></el-option>
<el-option label="隐藏" value="0"></el-option>
</el-select>
</el-form-item>
<el-form-item label="等级名称:">
<el-input v-model="tableFrom.name" placeholder="请输入等级名称" class="selWidth">
<el-button slot="append" icon="el-icon-search" @click="seachList" size="small"/>
</el-input>
</el-form-item>
</el-form>
</div>
<el-button type="primary" class="mr10" @click="add" size="small">添加用户等级</el-button>
</div>
<el-table
v-loading="listLoading"
:data="tableData.data"
style="width: 100%"
size="mini"
>
<el-table-column
prop="id"
label="ID"
min-width="50"
/>
<el-table-column label="等级图标" min-width="80">
<template slot-scope="scope">
<div class="demo-image__preview">
<el-image
style="width: 36px; height: 36px"
:src="scope.row.icon"
:preview-src-list="[scope.row.icon]"
/>
</div>
</template>
</el-table-column>
<el-table-column
prop="name"
label="等级名称"
min-width="150"
/>
<el-table-column
prop="grade"
label="等级"
min-width="100"
/>
<el-table-column
prop="discount"
label="享受折扣"
min-width="100"
/>
<el-table-column
label="是否显示"
min-width="150"
>
<template slot-scope="scope">
<el-switch
v-model="scope.row.isShow"
:active-value="true"
:inactive-value="false"
active-text="显示"
inactive-text="隐藏"
@change="onchangeIsShow(scope.row)"
/>
</template>
</el-table-column>
<el-table-column label="操作" min-width="120" fixed="right" align="center">
<template slot-scope="scope">
<el-button type="text" @click="edit(scope.row.id)" class="mr10">编辑</el-button>
<el-button type="text" @click="handleDelete(scope.row.id, scope.$index)">删除</el-button>
</template>
</el-table-column>
</el-table>
<div class="block">
<el-pagination
:page-sizes="[20, 40, 60, 80]"
:page-size="tableFrom.limit"
:current-page="tableFrom.page"
layout="total, sizes, prev, pager, next, jumper"
:total="tableData.total"
@size-change="handleSizeChange"
@current-change="pageChange"
/>
</div>
</el-card>
<creat-grade ref="grades"></creat-grade>
</div>
</template>
<script>
import { userListApi, groupListApi, levelListApi, levelUseApi, levelDeleteApi } from '@/api/user'
import creatGrade from './creatGrade'
export default {
name: 'Grade',
filters: {
typeFilter(status) {
const statusMap = {
'wechat': '微信用户',
'routine': '小程序你用户',
'h5': 'H5用户'
}
return statusMap[status]
}
},
components: {creatGrade},
data() {
return {
listLoading: true,
tableData: {
data: [],
total: 0
},
tableFrom: {
isShow: '',
name: '',
page: 1,
limit: 20,
isDel: 0
}
}
},
mounted() {
this.getList()
},
methods: {
seachList() {
this.tableFrom.page = 1
this.getList()
},
add() {
this.$refs.grades.dialogVisible = true
},
edit(id) {
this.$refs.grades.dialogVisible = true
this.$refs.grades.info(id)
},
// 列表
getList() {
this.listLoading = true
levelListApi(this.tableFrom).then(res => {
this.tableData.data = res.list
this.tableData.total = res.total
this.listLoading = false
}).catch(() => {
this.listLoading = false
})
},
pageChange(page) {
this.tableFrom.page = page
this.getList()
},
handleSizeChange(val) {
this.tableFrom.limit = val
this.getList()
},
// 删除
handleDelete(id, idx) {
this.$modalSure().then(() => {
levelDeleteApi({id:id}).then(() => {
this.$message.success('删除成功')
this.tableData.data.splice(idx, 1)
})
})
},
onchangeIsShow(row) {
levelUseApi({id: row.id, value:row.isShow}).then(() => {
this.$message.success('修改成功')
this.getList()
})
}
}
}
</script>
<style scoped lang="scss">
</style>

View File

@@ -0,0 +1,151 @@
<template>
<div class="divBox">
<el-card class="box-card">
<div slot="header" class="clearfix">
<el-button size="small" type="primary" @click="onAdd(null)">{{$route.path.indexOf('group') !== -1?'添加用户分组':'添加用户标签'}}</el-button>
</div>
<el-table
v-loading="listLoading"
:data="tableData.data"
style="width: 100%"
size="small"
>
<el-table-column
label="ID"
min-width="80"
prop="id"
/>
<el-table-column
:label="$route.path.indexOf('group') !== -1 ? '分组名称' : '标签名称'"
min-width="180"
>
<template slot-scope="{row}">
<span v-text="$route.path.indexOf('group') !== -1?row.groupName:row.name"></span>
</template>
</el-table-column>
<!--<el-table-column-->
<!--prop="create_time"-->
<!--label="创建时间"-->
<!--min-width="150"-->
<!--/>-->
<el-table-column label="操作" min-width="120" fixed="right" align="center">
<template slot-scope="scope">
<el-button class="mr10" type="text" size="small" @click="onAdd(scope.row)">编辑</el-button>
<el-button type="text" size="small" @click="handleDelete(scope.row.id, scope.$index)">删除</el-button>
</template>
</el-table-column>
</el-table>
<div class="block">
<el-pagination
:page-sizes="[20, 40, 60, 80]"
:page-size="tableFrom.limit"
:current-page="tableFrom.page"
layout="total, sizes, prev, pager, next, jumper"
:total="tableData.total"
@size-change="handleSizeChange"
@current-change="pageChange"
/>
</div>
</el-card>
</div>
</template>
<script>
import { tagListApi, tagDeleteApi, tagSaveApi, tagInfoApi, tagUpdateApi, groupListApi, groupDeleteApi, groupSaveApi, groupUpdateApi } from '@/api/user'
export default {
name: 'UserGroup',
data() {
return {
tableFrom: {
page: 1,
limit: 20
},
tableData: {
data: [],
total: 0
},
listLoading: true
}
},
mounted() {
this.getList()
},
methods: {
info(){
},
onAdd(row) {
this.$prompt(this.$route.path.indexOf('group') !== -1?'分组名称':'标签名称', {
confirmButtonText: '确定',
cancelButtonText: '取消',
inputErrorMessage: this.$route.path.indexOf('group') !== -1?'请输入分组名称':'请输入标签名称',
inputType: 'text',
inputValue: row ? (this.$route.path.indexOf('group') !== -1?row.groupName:row.name): '' ,
inputPlaceholder: this.$route.path.indexOf('group') !== -1?'请输入分组名称':'请输入标签名称',
inputValidator: (value) => { if(!value) return '输入不能为空'}
}).then(({value}) => {
if(this.$route.path.indexOf('group') !== -1){
row?groupUpdateApi({id: row.id},{groupName: value}).then(() => {
this.$message.success('编辑成功')
this.getList();
}):groupSaveApi({ groupName: value}).then(() => {
this.$message.success('新增成功')
this.getList();
})
}else{
row?tagUpdateApi({id: row.id}, {name: value}).then(() => {
this.$message.success('编辑成功')
this.getList();
}):tagSaveApi({ name: value}).then(() => {
this.$message.success('新增成功')
this.getList();
})
}
}).catch(() => {
this.$message.info('取消输入')
})
},
// 列表
getList() {
this.listLoading = true
this.$route.path.indexOf('group') !== -1 ? groupListApi(this.tableFrom).then(res => {
this.tableData.data = res.list
this.tableData.total = res.total
this.listLoading = false
}).catch(res => {
this.listLoading = false
}) : tagListApi(this.tableFrom).then(res => {
this.tableData.data = res.list
this.tableData.total = res.total
this.listLoading = false
}).catch(res => {
this.listLoading = false
})
},
pageChange(page) {
this.tableFrom.page = page
this.getList()
},
handleSizeChange(val) {
this.tableFrom.limit = val
this.getList()
},
// 删除
handleDelete(id, idx) {
this.$modalSure().then(() => {
this.$route.path.indexOf('group') !== -1 ? groupDeleteApi({id:id}).then(() => {
this.$message.success('删除成功')
this.tableData.data.splice(idx, 1)
}) : tagDeleteApi({id:id}).then(() => {
this.$message.success('删除成功')
this.tableData.data.splice(idx, 1)
})
})
}
}
}
</script>
<style scoped lang="stylus">
</style>

View File

@@ -0,0 +1,171 @@
<template>
<el-form :model="ruleForm" :rules="rules" ref="ruleForm" label-width="100px" class="demo-ruleForm">
<el-form-item label="用户编号:">
<el-input v-model="ruleForm.id" disabled class="selWidth"></el-input>
</el-form-item>
<el-form-item label="真实姓名:">
<el-input v-model="ruleForm.realName" class="selWidth"></el-input>
</el-form-item>
<el-form-item label="手机号码:" prop="phone">
<el-input v-model.number="ruleForm.phone" class="selWidth"></el-input>
</el-form-item>
<el-form-item label="生日:">
<el-date-picker
v-model="ruleForm.birthday"
type="date"
class="selWidth"
placeholder="选择日期"
format="yyyy 年 MM 月 dd 日"
value-format="yyyy-MM-dd">
</el-date-picker>
</el-form-item>
<el-form-item label="身份证号:">
<el-input v-model="ruleForm.cardId" class="selWidth"></el-input>
</el-form-item>
<el-form-item label="用户地址:">
<el-input v-model="ruleForm.addres" class="selWidth"></el-input>
</el-form-item>
<el-form-item label="用户等级:">
<el-select v-model="ruleForm.level" placeholder="请选择" class="selWidth" clearable filterable>
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in levelList" :key="index" :label="item.name"></el-option>
</el-select>
</el-form-item>
<el-form-item label="用户分组:">
<el-select v-model="ruleForm.groupId" placeholder="请选择" class="selWidth" clearable filterable>
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in groupList" :key="index" :label="item.groupName"></el-option>
</el-select>
</el-form-item>
<el-form-item label="用户标签:">
<el-select v-model="labelData" placeholder="请选择" class="selWidth" clearable filterable multiple >
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in labelLists" :key="index" :label="item.name"></el-option>
</el-select>
</el-form-item>
<el-form-item label="推广员">
<el-radio-group v-model="ruleForm.isPromoter">
<el-radio :label="true">开启</el-radio>
<el-radio :label="false">关闭</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label="状态">
<el-radio-group v-model="ruleForm.status">
<el-radio :label="true">开启</el-radio>
<el-radio :label="false">关闭</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="submitForm('ruleForm')">提交</el-button>
<el-button @click="resetForm('ruleForm')">取消</el-button>
</el-form-item>
</el-form>
</template>
<script>
import { groupListApi, levelListApi, tagListApi, userInfoApi, userUpdateApi } from '@/api/user'
const defaultObj = {
birthday: '',
cardId: '',
id : null,
mark: '',
phone: '',
realName: '',
addres:'',
groupId: 0,
level: 0,
isPromoter: false,
status: false,
}
export default {
name: "UserEdit",
props:{
uid: {
type: Number,
default: null
}
},
data() {
return {
ruleForm: Object.assign({}, defaultObj),
groupData: [],
labelData: [],
labelLists: [],
levelList: [],
groupList: [],
rules: {}
}
},
mounted() {
if(this.uid) this.userInfo()
this.groupLists()
this.levelLists()
this.getTagList()
},
methods: {
// 详情
userInfo () {
userInfoApi({ id: this.uid}).then(async res => {
this.ruleForm = {
birthday: res.birthday,
cardId: res.cardId,
id : res.uid,
mark: res.mark,
phone: res.phone,
realName: res.realName,
status: res.status,
addres: res.addres,
groupId: Number(res.groupId),
level: res.level,
isPromoter: res.isPromoter,
tagId: res.tagId
}
this.labelData = res.tagId.split(',').map(Number)
console.log(this.labelData)
})
},
// 分组列表
groupLists () {
groupListApi({ page: 1, limit: 9999}).then(async res => {
this.groupList = res.list
})
},
//标签列表
getTagList () {
tagListApi({ page: 1, limit: 9999}).then(res => {
this.labelLists = res.list
})
},
// 等级列表
levelLists () {
levelListApi({ page: 1, limit: 9999}).then(async res => {
this.levelList = res.list
})
},
submitForm(formName) {
this.$refs[formName].validate((valid) => {
if (valid) {
this.ruleForm.tagId=this.labelData.join(',')
userUpdateApi({id: this.ruleForm.id},this.ruleForm).then(async res => {
this.$message.success('编辑成功')
this.$parent.visible = false
this.$parent.getList()
})
} else {
return false;
}
});
},
resetForm(formName) {
this.$refs[formName].resetFields()
this.$parent.visible = false
}
}
}
</script>
<style scoped>
.selWidth{
width: 90%;
}
</style>

View File

@@ -0,0 +1,829 @@
<template>
<div class="divBox">
<el-card class="box-card">
<div slot="header" class="clearfix">
<el-tabs v-model="loginType" @tab-click="getList">
<el-tab-pane :label="item.name" :name="item.type.toString()" v-for="(item,index) in headeNum" :key="index"/>
</el-tabs>
<div class="container">
<el-form inline size="small" :label-position="labelPosition" label-width="100px">
<el-row>
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="用户搜索:">
<el-input v-model="userFrom.keywords" placeholder="请输入" clearable class="selWidth"/>
</el-form-item>
</el-col>
<!--<el-col :span="24">-->
<!--<el-col v-bind="grid">-->
<!--<el-form-item label="会员搜索:">-->
<!--<el-input v-model="userFrom.keywords" placeholder="请输入" clearable class="selWidth"/>-->
<!--</el-form-item>-->
<!--</el-col>-->
<!--</el-col>-->
</el-col>
<template v-if="collapse">
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="用户等级:">
<el-select v-model="levelData" placeholder="请选择" class="selWidth" clearable filterable multiple>
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in levelList" :key="index" :label="item.name"></el-option>
</el-select>
</el-form-item>
</el-col>
<el-col v-bind="grid">
<el-form-item label="用户分组:">
<el-select v-model="groupData" placeholder="请选择" class="selWidth" clearable filterable multiple>
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in groupList" :key="index" :label="item.groupName"></el-option>
</el-select>
</el-form-item>
</el-col>
<el-col v-bind="grid">
<el-form-item label="用户标签:">
<el-select v-model="labelData" placeholder="请选择" class="selWidth" clearable filterable multiple>
<el-option value="">全部</el-option>
<el-option :value="item.id" v-for="(item, index) in labelLists" :key="index" :label="item.name"></el-option>
</el-select>
</el-form-item>
</el-col>
</el-col>
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="国家:">
<el-select v-model="userFrom.country" placeholder="请选择" class="selWidth" clearable @on-change="changeCountry">
<el-option value="" label="全部"></el-option>
<el-option value="domestic" label="中国"></el-option>
<el-option value="abroad" label="外国"></el-option>
</el-select>
</el-form-item>
</el-col>
<el-col v-bind="grid" v-if="userFrom.country ==='domestic'">
<el-form-item label="省份:">
<el-cascader :options="addresData" :props="propsCity" v-model="address" @change="handleChange" class="selWidth"></el-cascader>
</el-form-item>
</el-col>
</el-col>
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="性别:">
<el-radio-group v-model="userFrom.sex" type="button" class="selWidth">
<el-radio-button label="">
<span>全部</span>
</el-radio-button>
<el-radio-button label="1">
<span></span>
</el-radio-button>
<el-radio-button label="2">
<span></span>
</el-radio-button>
<el-radio-button label="0">
<span>保密</span>
</el-radio-button>
</el-radio-group>
</el-form-item>
</el-col>
<el-col v-bind="grid">
<el-form-item label="身份:">
<el-radio-group v-model="userFrom.isPromoter" type="button" class="selWidth">
<el-radio-button label="">
<span>全部</span>
</el-radio-button>
<el-radio-button label="1">
<span>推广员</span>
</el-radio-button>
<el-radio-button label="0">
<span>普通会员</span>
</el-radio-button>
</el-radio-group>
</el-form-item>
</el-col>
</el-col>
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="访问情况:">
<el-select v-model="userFrom.accessType" placeholder="请选择" class="selWidth" clearable>
<el-option value="">全部</el-option>
<el-option value="visitno">时间段未访问</el-option>
<el-option value="visit">时间段访问过</el-option>
<el-option value="add_time">首次访问</el-option>
</el-select>
</el-form-item>
</el-col>
<el-col v-bind="grid">
<el-form-item label="消费情况:">
<el-select v-model="userFrom.payCount" placeholder="请选择" class="selWidth" clearable>
<el-option value="">全部</el-option>
<el-option value="-1">0</el-option>
<el-option value="0">1+</el-option>
<el-option value="1">2+</el-option>
<el-option value="2">3+</el-option>
<el-option value="3">4+</el-option>
<el-option value="4">5+</el-option>
</el-select>
</el-form-item>
</el-col>
</el-col>
<el-col :xs="24" :sm="24" :md="24" :lg="18" :xl="18">
<el-col v-bind="grid">
<el-form-item label="时间选择:" class="timeBox">
<el-date-picker
v-model="timeVal"
align="right"
unlink-panels
value-format="yyyy-MM-dd"
format="yyyy-MM-dd"
size="small"
type="daterange"
placement="bottom-end"
placeholder="自定义时间"
class="selWidth"
:picker-options="pickerOptions"
@change="onchangeTime"
/>
</el-form-item>
</el-col>
</el-col>
</template>
<el-col :xs="24" :sm="24" :md="24" :lg="6" :xl="6" class="text-right userFrom">
<el-form-item>
<el-button type="primary" icon="ios-search" label="default" class="mr15" size="small" @click="userSearchs">搜索</el-button>
<el-button class="ResetSearch mr10" @click="reset('userFrom')" size="small">重置</el-button>
<a class="ivu-ml-8" @click="collapse = !collapse">
<template v-if="!collapse">
展开 <i class="el-icon-arrow-down"></i>
</template>
<template v-else>
收起 <i class="el-icon-arrow-up"></i>
</template>
</a>
</el-form-item>
</el-col>
</el-row>
</el-form>
</div>
<el-button class="mr10" size="small" @click="onSend">发送优惠券</el-button>
<el-button v-show="loginType === 'wechat'" size="mini" class="mr10" @click="sendNews">发送文章</el-button>
<el-button class="mr10" size="small" @click="setBatch('group')">批量设置分组</el-button>
<el-button class="mr10" size="small" @click="setBatch('label')">批量设置标签</el-button>
</div>
<el-table
v-loading="listLoading"
:data="tableData.data"
style="width: 100%"
size="mini"
@selection-change="onSelectTab"
highlight-current-row
>
<el-table-column type="expand">
<template slot-scope="props">
<el-form label-position="left" inline class="demo-table-expand">
<el-form-item label="首次访问:">
<span>{{ props.row.createTime | filterEmpty }}</span>
</el-form-item>
<el-form-item label="近次访问:">
<span>{{ props.row.lastLoginTime | filterEmpty }}</span>
</el-form-item>
<el-form-item label="身份证号:">
<span>{{ props.row.cardId | filterEmpty }}</span>
</el-form-item>
<el-form-item label="手机号:">
<span>{{ props.row.phone | filterEmpty }}</span>
</el-form-item>
<el-form-item label="真实姓名:">
<span>{{ props.row.realName | filterEmpty }}</span>
</el-form-item>
<el-form-item label="标签:">
<span>{{ props.row.tagName | filterEmpty }}</span>
</el-form-item>
<el-form-item label="生日:">
<span>{{ props.row.birthday | filterEmpty }}</span>
</el-form-item>
<el-form-item label="地址:">
<span>{{ props.row.addres | filterEmpty }}</span>
</el-form-item>
<el-form-item label="备注:">
<span>{{ props.row.mark | filterEmpty}}</span>
</el-form-item>
</el-form>
</template>
</el-table-column>
<el-table-column
type="selection"
width="55">
</el-table-column>
<el-table-column
prop="uid"
label="ID"
min-width="50"
/>
<el-table-column label="头像" min-width="80">
<template slot-scope="scope">
<div class="demo-image__preview">
<el-image
style="width: 36px; height: 36px"
:src="scope.row.avatar"
:preview-src-list="[scope.row.avatar]"
/>
</div>
</template>
</el-table-column>
<el-table-column
prop="nickname"
label="姓名"
min-width="150"
/>
<el-table-column
label="用户等级"
min-width="100"
>
<template slot-scope="scope">
<span>{{scope.row.level | levelFilter | filterEmpty}}</span>
</template>
</el-table-column>
<el-table-column
prop="groupName"
label="分组"
min-width="100"
/>
<el-table-column
prop="spreadNickname"
label="推荐人"
min-width="130"
/>
<el-table-column
label="用户类型"
min-width="100"
>
<template slot-scope="scope">
<span>{{scope.row.userType | typeFilter}}</span>
</template>
</el-table-column>
<el-table-column
prop="nowMoney"
label="余额"
min-width="100"
/>
<el-table-column label="操作" min-width="200" fixed="right" align="center">
<template slot-scope="scope">
<el-button type="text" size="small" @click="onDetails(scope.row.uid)">账户详情</el-button>
<el-button type="text" @click="editUser(scope.row.uid)" size="small">编辑</el-button>
<el-button type="text" @click="editPoint(scope.row.uid)" size="small">积分余额</el-button>
</template>
</el-table-column>
</el-table>
<div class="block">
<el-pagination
:page-sizes="[20, 40, 60, 80]"
:page-size="userFrom.limit"
:current-page="userFrom.page"
layout="total, sizes, prev, pager, next, jumper"
:total="tableData.total"
@size-change="handleSizeChange"
@current-change="pageChange"
/>
</div>
</el-card>
<!--批量设置-->
<el-dialog
title="设置"
:visible.sync="dialogVisible"
width="500px"
:before-close="handleClose">
<el-form :model="dynamicValidateForm" ref="dynamicValidateForm" label-width="100px" class="demo-dynamic" v-loading="loading">
<el-form-item
prop="groupId"
label="用户分组"
:rules="[{ required: true, message: '请选择用户分组', trigger: 'change' }]"
v-if="batchName ==='group'"
>
<el-select v-model="dynamicValidateForm.groupId" placeholder="请选择分组" style="width: 80%" filterable multiple>
<el-option :value="item.id" v-for="(item, index) in groupList" :key="index" :label="item.groupName"></el-option>
</el-select>
</el-form-item>
<el-form-item
prop="groupId"
label="用户标签"
:rules="[{ required: true, message: '请选择用户标签', trigger: 'change' }]"
v-else
>
<el-select v-model="dynamicValidateForm.groupId" placeholder="请选择标签" style="width: 80%" filterable multiple>
<el-option :value="item.id" v-for="(item, index) in labelLists" :key="index" :label="item.name"></el-option>
</el-select>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="handleClose"> </el-button>
<el-button type="primary" @click="submitForm('dynamicValidateForm')"> </el-button>
</span>
</el-dialog>
<!--编辑-->
<el-dialog
title="编辑"
:visible.sync="visible"
width="600px"
>
<edit-from v-if="visible" :uid="uid"></edit-from>
</el-dialog>
<!--积分余额-->
<el-dialog
title="积分余额"
:visible.sync="VisiblePoint"
width="500px"
:before-close="handlePointClose">
<el-form :model="PointValidateForm" ref="PointValidateForm" label-width="100px" class="demo-dynamic" v-loading="loadingPoint">
<el-form-item
label="修改余额"
required
>
<el-radio-group v-model="PointValidateForm.moneyType">
<el-radio :label="1">增加</el-radio>
<el-radio :label="2">减少</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item
label="余额"
required
>
<el-input type="text" v-model.number="PointValidateForm.moneyValue"></el-input>
</el-form-item>
<el-form-item
label="修改积分"
required
>
<el-radio-group v-model="PointValidateForm.integralType">
<el-radio :label="1">增加</el-radio>
<el-radio :label="2">减少</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item
label="积分"
required
>
<el-input type="text" v-model.number="PointValidateForm.integralValue"></el-input>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="handlePointClose"> </el-button>
<el-button type="primary" :loading="loadingBtn" @click="submitPointForm('PointValidateForm')"> </el-button>
</span>
</el-dialog>
<!--账户详情-->
<el-dialog
title="用户详情"
:visible.sync="Visible"
width="1000px"
v-if="uid"
:before-close="Close">
<user-details ref="userDetails" :uid="uid" v-if="Visible"></user-details>
</el-dialog>
</div>
</template>
<script>
import { userListApi, groupListApi, levelListApi, tagListApi, groupPiApi, tagPiApi, foundsApi } from '@/api/user'
import editFrom from './edit'
import userDetails from './userDetails'
import * as logistics from '@/api/logistics.js'
import Cookies from 'js-cookie'
export default {
name: 'UserIndex',
components:{ editFrom, userDetails },
// filters: {
// typeFilter(status) {
// const statusMap = {
// 'wechat': '微信用户',
// 'routine': '小程序用户',
// 'h5': 'H5用户'
// }
// return statusMap[status]
// }
// },
data() {
return {
pickerOptions: {
shortcuts: [
{
text: '今天',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(new Date(new Date().getFullYear(), new Date().getMonth(), new Date().getDate()))
picker.$emit('pick', [start, end])
}
},
{
text: '昨天',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(start.setTime(new Date(new Date().getFullYear(), new Date().getMonth(), new Date().getDate() - 1)))
end.setTime(end.setTime(new Date(new Date().getFullYear(), new Date().getMonth(), new Date().getDate())))
picker.$emit('pick', [start, end])
}
},
{
text: '最近7天',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(start.getTime() - 3600 * 1000 * 24 * 7)
picker.$emit('pick', [start, end])
}
},
{
text: '最近30天',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(start.getTime() - 3600 * 1000 * 24 * 30)
picker.$emit('pick', [start, end])
}
},
{
text: '本月',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(start.setTime(new Date(new Date().getFullYear(), new Date().getMonth(), 1)))
picker.$emit('pick', [start, end])
}
},
{
text: '本年',
onClick(picker) {
const end = new Date()
const start = new Date()
start.setTime(start.setTime(new Date(new Date().getFullYear(), 0, 1)))
picker.$emit('pick', [start, end])
}
}
]
},
loadingBtn: false,
PointValidateForm: {
integralType: 2,
integralValue: 0,
moneyType: 2,
moneyValue: 0,
uid: ''
},
loadingPoint: false,
VisiblePoint: false,
visible: false,
userIds: '',
dialogVisible: false,
levelData: [],
groupData: [],
labelData: [],
selData:[],
labelPosition:'right',
collapse: false,
props: {
children: 'child',
label: 'name',
value: 'id',
emitPath: false
},
propsCity: {
children: 'child',
label: 'name',
value: 'cityId'
},
headeNum: [
{ 'type': '', 'name': '全部用户' },
{ 'type': 'wechat', 'name': '微信公众号' },
{ 'type': 'routine', 'name': '微信小程序' },
{ 'type': 'h5', 'name': 'H5' }
],
listLoading: true,
tableData: {
data: [],
total: 0
},
loginType: '',
userFrom: {
labelId: '',
loginType: '',
status: true,
sex: '',
isPromoter: '',
country: '',
payCount: '',
accessType: '',
dateLimit: '',
keywords: '',
province: '',
city: '',
page: 1,
limit: 15,
level: '',
groupId: ''
},
grid: {
xl: 8,
lg: 12,
md: 12,
sm: 24,
xs: 24
},
grid2: {
xl: 18,
lg: 16,
md: 12,
sm: 24,
xs: 24
},
grid3: {
xl: 8,
lg: 12,
md: 12,
sm: 24,
xs: 24
},
levelList: [],
labelLists: [],
groupList: [],
selectedData: [],
timeVal: [],
addresData: [],
dynamicValidateForm:{
groupId: []
},
loading: false,
groupIdFrom: [],
selectionList: [],
batchName: '',
uid: 0,
Visible: false,
keyNum: 0,
address: []
}
},
mounted() {
this.getList()
this.groupLists()
this.levelLists()
this.getTagList()
this.getCityList()
},
methods: {
// 列表
getCityList() {
logistics.cityListTree().then(res => {
res.forEach((el, index) => {
el.child.forEach((cel, j) => {
delete cel.child
})
})
this.addresData = res
})
},
// 发送文章
sendNews() {
if (this.selectionList.length === 0) return this.$message.warning('请先选择用户')
const _this = this
this.$modalArticle(function(row) {
console.log(row)
},'send')
},
// 发送优惠劵
onSend(){
if (this.selectionList.length === 0) return this.$message.warning('请选择要设置的用户');
const _this = this
this.$modalCoupon('send', this.keyNum += 1, [],function(row) {
_this.formValidate.give_coupon_ids = []
_this.couponData = []
row.map((item) => {
_this.formValidate.give_coupon_ids.push(item.coupon_id)
_this.couponData.push(item.title)
})
},this.userIds)
},
Close() {
this.Visible = false
},
// 账户详情
onDetails(id){
this.uid = id
this.Visible = true
},
// 积分余额
editPoint(id) {
this.uid = id
this.VisiblePoint = true
},
// 积分余额
submitPointForm(formName){
this.$refs[formName].validate((valid) => {
if (valid) {
this.PointValidateForm.uid = this.uid
this.loadingBtn = true
foundsApi(this.PointValidateForm).then(res => {
this.$message.success('设置成功')
this.loadingBtn = false
this.handlePointClose()
}).catch(() => {
this.loadingBtn = false
})
} else {
return false
}
})
},
// 积分余额
handlePointClose() {
this.VisiblePoint = false
this.PointValidateForm = {
integralType: 2,
integralValue: 0,
moneyType: 2,
moneyValue: 0,
uid: ''
}
},
editUser(id) {
this.visible = true
this.uid = id
},
submitForm(formName) {
this.$refs[formName].validate((valid) => {
if (valid) {
this.loading = true
this.batchName ==='group' ? groupPiApi({groupId: this.dynamicValidateForm.groupId.join(','), id: this.userIds}).then(res => {
this.$message.success('设置成功')
this.loading = false
this.handleClose()
this.getList()
}).catch(() => {
this.loading = false
}) : tagPiApi({tagId: this.dynamicValidateForm.groupId.join(','), id: this.userIds}).then(res => {
this.$message.success('设置成功')
this.loading = false
this.handleClose()
this.getList()
}).catch(() => {
this.loading = false
})
} else {
return false;
}
});
},
setBatch(name){
this.batchName = name
if (this.selectionList.length === 0) return this.$message.warning('请选择要设置的用户')
this.dialogVisible = true
},
handleClose(){
this.dialogVisible = false
this.$refs['dynamicValidateForm'].resetFields();
},
// 选择时间
selectChange (tab) {
this.timeVal = [];
this.getList();
},
// 全选
onSelectTab (selection) {
this.selectionList = selection;
let data = [];
this.selectionList.map((item) => {
data.push(item.uid)
});
this.userIds = data.join(',');
},
// 搜索
userSearchs () {
this.userFrom.page = 1;
this.getList();
},
// 选择国家
changeCountry () {
if (this.userFrom.country === 'abroad' || !this.userFrom.country) {
this.selectedData = [];
this.userFrom.province = '';
this.userFrom.city = '';
this.address = [];
}
},
// 选择地址
handleChange (value) {
this.userFrom.province = value[0];
this.userFrom.city = value[1];
},
// 具体日期
onchangeTime (e) {
this.timeVal = e;
console.log(e)
this.userFrom.dateLimit = e.join(',');
},
// 分组列表
groupLists () {
groupListApi({ page: 1, limit: 9999}).then(async res => {
this.groupList = res.list
})
},
//标签列表
getTagList () {
tagListApi({ page: 1, limit: 9999}).then(res => {
this.labelLists = res.list
})
},
// 等级列表
levelLists () {
levelListApi({ page: 1, limit: 9999}).then(async res => {
this.levelList = res.list
localStorage.setItem('levelKey', JSON.stringify(res.list))
})
},
// 列表
getList() {
this.listLoading = true
if(this.loginType == 0) this.loginType =''
this.userFrom.loginType = this.loginType
this.userFrom.level = this.levelData.join(',')
this.userFrom.groupId = this.groupData.join(',')
this.userFrom.labelId = this.labelData.join(',')
userListApi(this.userFrom).then(res => {
this.tableData.data = res.list
this.tableData.total = res.total
this.listLoading = false
}).catch(() => {
this.listLoading = false
})
},
pageChange(page) {
this.userFrom.page = page
this.getList()
},
handleSizeChange(val) {
this.userFrom.limit = val
this.getList()
},
// 删除
handleDelete(id, idx) {
this.$modalSure().then(() => {
productDeleteApi(id).then(() => {
this.$message.success('删除成功')
this.getList()
})
})
},
onchangeIsShow(row) {
row.isShow
? putOnShellApi( row.id ).then(() => {
this.$message.success('上架成功')
this.getList()
}) : offShellApi(row.id).then(() => {
this.$message.success('下架成功')
this.getList()
})
}
}
}
</script>
<style scoped lang="scss">
/*.timeBox{*/
/*width: 100%;*/
/*/deep/.el-form-item__content{*/
/*width: 87% !important;*/
/*}*/
/*}*/
.text-right{
text-align: right;
}
.demo-table-expand {
font-size: 0;
}
.demo-table-expand label {
width: 90px;
color: #99a9bf;
}
.demo-table-expand .el-form-item {
margin-right: 0;
margin-bottom: 0;
width: 33.33%;
}
.selWidth{
width: 100% !important;
}
.seachTiele{
line-height: 30px;
}
.container{
min-width: 821px;
/deep/.el-form-item{
width: 100%;
}
/deep/.el-form-item__content{
width: 72%;
}
}
.ivu-ml-8{
font-size: 12px;
color: #1682e6;
}
</style>

View File

@@ -0,0 +1,366 @@
<template>
<div>
<div class="acea-row row-middle" v-if="psInfo">
<div class="avatar mr15"><img :src="psInfo.user.avatar"></div>
<div class="dashboard-workplace-header-tip">
<p class="dashboard-workplace-header-tip-title" v-text="psInfo.user.nickname || '-'"></p>
<div class="dashboard-workplace-header-tip-desc">
<span class="dashboard-workplace-header-tip-desc-sp">余额: {{ psInfo.balance }}</span>
<span class="dashboard-workplace-header-tip-desc-sp">总计订单: {{ psInfo.allOrderCount }}</span>
<span class="dashboard-workplace-header-tip-desc-sp">总消费金额: {{ psInfo.allConsumeCount }}</span>
<span class="dashboard-workplace-header-tip-desc-sp">积分: {{ psInfo.integralCount }}</span>
<span class="dashboard-workplace-header-tip-desc-sp">本月订单: {{ psInfo.mothOrderCount }}</span>
<span class="dashboard-workplace-header-tip-desc-sp">本月消费金额: {{ psInfo.mothConsumeCount }}</span>
</div>
</div>
</div>
<el-row align="middle" :gutter="10" class="ivu-mt mt20">
<el-col :span="4">
<el-menu
default-active="0"
class="el-menu-vertical-demo"
@select="changeType"
>
<el-menu-item :name="item.val" v-for="(item, index) in list" :key="index" :index="item.val">
<span slot="title">{{item.label}}</span>
</el-menu-item >
</el-menu>
</el-col>
<el-col :span="20">
<el-table :data="tableData.data" class="tabNumWidth" size="mini" v-loading="loading">
<el-table-column
:prop="item.key"
:label="item.title"
width="item.minWidth"
v-for="(item, index) in columns" :key="index"
/>
</el-table>
<div class="block">
<el-pagination
:page-sizes="[20, 40, 60, 80]"
:page-size="tableFrom.limit"
:current-page="tableFrom.page"
layout="total, sizes, prev, pager, next, jumper"
:total="tableData.total"
@size-change="handleSizeChange"
@current-change="pageChange"
/>
</div>
</el-col>
</el-row>
</div>
</template>
<script>
import { infobyconditionApi, topdetailApi } from '@/api/user'
export default {
name: "UserDetails",
props:{
uid: {
type: Number,
default: null
}
},
data() {
return {
loading: false,
columns: [],
Visible: false,
list: [
{ val: '0', label: '消费记录' },
{ val: '1', label: '积分明细' },
{ val: '2', label: '签到记录' },
{ val: '3', label: '持有优惠券' },
{ val: '4', label: '余额变动' },
{ val: '5', label: '好友关系' }
],
tableData: {
data: [],
total: 0
},
tableFrom: {
page: 1,
limit: 6,
type : '0',
userId : ''
},
psInfo: null
}
},
mounted(){
if(this.uid){
this.getHeader()
this.getInfo()
}
},
methods: {
changeType(key) {
this.tableFrom.type = key
this.getInfo()
},
getInfo() {
this.tableFrom.userId = this.uid
this.loading = true
infobyconditionApi(this.tableFrom).then(res => {
this.tableData.data = res.list
this.tableData.total = res.total
switch (this.tableFrom.type) {
case '0':
this.columns = [
{
title: '订单ID',
key: 'orderId',
minWidth: 250
},
{
title: '收货人',
key: 'realName',
minWidth: 90
},
{
title: '商品数量',
key: 'totalNum',
minWidth: 80
},
{
title: '商品总价',
key: 'totalPrice',
minWidth: 90
},
{
title: '实付金额',
key: 'payPrice',
minWidth: 90
},
{
title: '交易完成时间',
key: 'payTime',
minWidth: 160
}
]
break;
case '1':
this.columns = [
{
title: '来源/用途',
key: 'title',
minWidth: 120
},
{
title: '积分变化',
key: 'number',
minWidth: 120
},
{
title: '变化后积分',
key: 'balance',
minWidth: 120
},
{
title: '日期',
key: 'add_time',
minWidth: 120
},
{
title: '备注',
key: 'mark',
minWidth: 120
}
]
break;
case '2':
this.columns = [
{
title: '动作',
key: 'title',
minWidth: 120
},
{
title: '获得积分',
key: 'number',
minWidth: 120
},
{
title: '签到时间',
key: 'createDay',
minWidth: 120
},
{
title: '备注',
key: 'title',
minWidth: 120
}
]
break;
case '3':
this.columns = [
{
title: '优惠券名称',
key: 'name',
minWidth: 120
},
{
title: '面值',
key: 'money',
minWidth: 120
},
{
title: '有效期',
key: 'endTime',
minWidth: 120
},
{
title: '最低消费额',
key: 'minPrice',
minWidth: 120
},
{
title: '兑换时间',
key: 'useTime',
minWidth: 120
}
]
break;
case '4':
this.columns = [
{
title: '变动金额',
key: 'number',
minWidth: 120
},
{
title: '变动后',
key: 'balance',
minWidth: 120
},
{
title: '类型',
key: 'type',
minWidth: 120
},
{
title: '创建时间',
key: 'add_time',
minWidth: 120
},
{
title: '备注',
key: 'mark',
minWidth: 120
}
]
break;
default:
this.columns = [
{
title: 'ID',
key: 'uid',
minWidth: 120
},
{
title: '昵称',
key: 'nickname',
minWidth: 120
},
{
title: '等级',
key: 'level',
minWidth: 120
},
{
title: '加入时间',
key: 'createTime',
minWidth: 120
}
]
}
this.loading = false
}).catch(() => {
this.loading = false
})
},
pageChange(page) {
this.tableFrom.page = page
this.getInfo()
},
handleSizeChange(val) {
this.tableFrom.limit = val
this.getInfo()
},
getHeader() {
topdetailApi({userId : this.uid}).then(res => {
this.psInfo = res
})
}
}
}
</script>
<style scoped lang="scss">
.avatar{
width: 60px;
height: 60px;
border-radius: 50%;
overflow: hidden;
margin-left: 18px;
img{
width: 100%;
height: 100%;
}
}
.dashboard-workplace {
&-header {
&-avatar {
width: 64px;
height: 64px;
border-radius: 50%;
margin-right: 16px;
font-weight: 600;
}
&-tip {
width: 82%;
display: inline-block;
vertical-align: middle;
margin-top: -12px;
&-title {
font-size: 13px;
color: #000000;
margin-bottom: 12px;
}
&-desc {
&-sp {
width: 32%;
color: #17233D;
font-size: 13px;
display: inline-block;
}
}
}
&-extra {
.ivu-col {
p {
text-align: right;
}
p:first-child {
span:first-child {
margin-right: 4px;
}
span:last-child {
color: #808695;
}
}
p:last-child {
font-size: 22px;
}
}
}
}
}
</style>

View File

@@ -0,0 +1,15 @@
<template>
<div>
用户通知
</div>
</template>
<script>
export default {
}
</script>
<style lang="sass" scoped>
</style>>