2026-01-03 19:22:42 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="app-container">
|
|
|
|
|
|
<!-- 统计卡片 -->
|
|
|
|
|
|
<el-row :gutter="20" class="mb20">
|
|
|
|
|
|
<el-col :span="6">
|
|
|
|
|
|
<el-card shadow="hover">
|
|
|
|
|
|
<div class="stat-card">
|
|
|
|
|
|
<i class="el-icon-present stat-icon" style="color: #409EFF"></i>
|
|
|
|
|
|
<div class="stat-content">
|
|
|
|
|
|
<div class="stat-value">{{ statistics.total }}</div>
|
|
|
|
|
|
<div class="stat-label">礼物总数</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="6">
|
|
|
|
|
|
<el-card shadow="hover">
|
|
|
|
|
|
<div class="stat-card">
|
|
|
|
|
|
<i class="el-icon-check stat-icon" style="color: #67C23A"></i>
|
|
|
|
|
|
<div class="stat-content">
|
|
|
|
|
|
<div class="stat-value">{{ statistics.enabled }}</div>
|
|
|
|
|
|
<div class="stat-label">启用中</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="6">
|
|
|
|
|
|
<el-card shadow="hover">
|
|
|
|
|
|
<div class="stat-card">
|
|
|
|
|
|
<i class="el-icon-close stat-icon" style="color: #F56C6C"></i>
|
|
|
|
|
|
<div class="stat-content">
|
|
|
|
|
|
<div class="stat-value">{{ statistics.disabled }}</div>
|
|
|
|
|
|
<div class="stat-label">已禁用</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="6">
|
|
|
|
|
|
<el-card shadow="hover">
|
|
|
|
|
|
<div class="stat-card">
|
|
|
|
|
|
<i class="el-icon-coin stat-icon" style="color: #E6A23C"></i>
|
|
|
|
|
|
<div class="stat-content">
|
|
|
|
|
|
<div class="stat-value">{{ statistics.totalValue }}</div>
|
|
|
|
|
|
<div class="stat-label">总价值(钻石)</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 搜索和操作栏 -->
|
|
|
|
|
|
<el-card class="mb20">
|
|
|
|
|
|
<el-form :inline="true" :model="queryParams" class="demo-form-inline">
|
|
|
|
|
|
<el-form-item label="礼物名称">
|
|
|
|
|
|
<el-input v-model="queryParams.name" placeholder="请输入礼物名称" clearable @keyup.enter.native="handleQuery" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="状态">
|
|
|
|
|
|
<el-select v-model="queryParams.status" placeholder="请选择状态" clearable>
|
|
|
|
|
|
<el-option label="全部" value="" />
|
|
|
|
|
|
<el-option label="启用" :value="1" />
|
|
|
|
|
|
<el-option label="禁用" :value="0" />
|
|
|
|
|
|
</el-select>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item>
|
|
|
|
|
|
<el-button type="primary" icon="el-icon-search" @click="handleQuery">搜索</el-button>
|
|
|
|
|
|
<el-button icon="el-icon-refresh" @click="resetQuery">重置</el-button>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
<el-row>
|
|
|
|
|
|
<el-button type="primary" icon="el-icon-plus" @click="handleAdd">添加礼物</el-button>
|
|
|
|
|
|
<el-button type="danger" icon="el-icon-delete" :disabled="multiple" @click="handleDelete">批量删除</el-button>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 礼物列表 -->
|
|
|
|
|
|
<el-card>
|
|
|
|
|
|
<el-table v-loading="loading" :data="giftList" @selection-change="handleSelectionChange">
|
|
|
|
|
|
<el-table-column type="selection" width="55" align="center" />
|
|
|
|
|
|
<el-table-column label="ID" align="center" prop="id" width="80" />
|
|
|
|
|
|
<el-table-column label="礼物图标" align="center" width="100">
|
|
|
|
|
|
<template slot-scope="scope">
|
|
|
|
|
|
<el-image
|
|
|
|
|
|
:src="scope.row.image"
|
|
|
|
|
|
:preview-src-list="[scope.row.image]"
|
|
|
|
|
|
style="width: 50px; height: 50px"
|
|
|
|
|
|
fit="cover"
|
2026-01-04 19:00:02 +08:00
|
|
|
|
>
|
|
|
|
|
|
<div slot="error" class="image-slot">
|
|
|
|
|
|
<i class="el-icon-picture-outline"></i>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-image>
|
2026-01-03 19:22:42 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="礼物名称" align="center" prop="name" />
|
|
|
|
|
|
<el-table-column label="价格(钻石)" align="center" prop="diamondPrice" width="120">
|
|
|
|
|
|
<template slot-scope="scope">
|
|
|
|
|
|
<el-tag type="warning">{{ scope.row.diamondPrice }}</el-tag>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="亲密度" align="center" prop="intimacy" width="100" />
|
|
|
|
|
|
<el-table-column label="等级" align="center" prop="level" width="80">
|
|
|
|
|
|
<template slot-scope="scope">
|
|
|
|
|
|
<el-tag :type="['', 'success', 'info', 'warning', 'danger'][scope.row.level] || 'info'">
|
|
|
|
|
|
{{ scope.row.level }}级
|
|
|
|
|
|
</el-tag>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="排序" align="center" prop="sort" width="80" />
|
|
|
|
|
|
<el-table-column label="状态" align="center" width="100">
|
|
|
|
|
|
<template slot-scope="scope">
|
|
|
|
|
|
<el-switch
|
|
|
|
|
|
v-model="scope.row.status"
|
|
|
|
|
|
:active-value="1"
|
|
|
|
|
|
:inactive-value="0"
|
|
|
|
|
|
@change="handleStatusChange(scope.row)"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="创建时间" align="center" prop="createTime" width="180" />
|
|
|
|
|
|
<el-table-column label="操作" align="center" width="200" class-name="small-padding fixed-width">
|
|
|
|
|
|
<template slot-scope="scope">
|
|
|
|
|
|
<el-button size="mini" type="text" icon="el-icon-edit" @click="handleUpdate(scope.row)">编辑</el-button>
|
|
|
|
|
|
<el-button size="mini" type="text" icon="el-icon-delete" @click="handleDelete(scope.row)" style="color: #F56C6C">删除</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
</el-table>
|
|
|
|
|
|
|
|
|
|
|
|
<pagination
|
|
|
|
|
|
v-show="total > 0"
|
|
|
|
|
|
:total="total"
|
|
|
|
|
|
:page.sync="queryParams.page"
|
|
|
|
|
|
:limit.sync="queryParams.limit"
|
2026-01-04 19:00:02 +08:00
|
|
|
|
@size-change="getList"
|
|
|
|
|
|
@current-change="getList"
|
2026-01-03 19:22:42 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 添加或修改礼物对话框 -->
|
|
|
|
|
|
<el-dialog :title="title" :visible.sync="open" width="600px" append-to-body>
|
|
|
|
|
|
<el-form ref="form" :model="form" :rules="rules" label-width="100px">
|
|
|
|
|
|
<el-form-item label="礼物名称" prop="name">
|
|
|
|
|
|
<el-input v-model="form.name" placeholder="请输入礼物名称" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="礼物图标" prop="image">
|
|
|
|
|
|
<el-upload
|
|
|
|
|
|
class="avatar-uploader"
|
|
|
|
|
|
action="#"
|
|
|
|
|
|
:show-file-list="false"
|
|
|
|
|
|
:http-request="handleUpload"
|
|
|
|
|
|
:before-upload="beforeUpload"
|
|
|
|
|
|
>
|
|
|
|
|
|
<img v-if="form.image" :src="form.image" class="avatar">
|
|
|
|
|
|
<i v-else class="el-icon-plus avatar-uploader-icon"></i>
|
|
|
|
|
|
</el-upload>
|
|
|
|
|
|
<div class="el-upload__tip">建议上传200x200像素的PNG图片</div>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="价格(钻石)" prop="diamondPrice">
|
|
|
|
|
|
<el-input-number v-model="form.diamondPrice" :min="0.01" :max="999999" :precision="2" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="亲密度" prop="intimacy">
|
|
|
|
|
|
<el-input-number v-model="form.intimacy" :min="0" :max="999999" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="礼物等级" prop="level">
|
|
|
|
|
|
<el-select v-model="form.level" placeholder="请选择等级">
|
|
|
|
|
|
<el-option label="1级" :value="1" />
|
|
|
|
|
|
<el-option label="2级" :value="2" />
|
|
|
|
|
|
<el-option label="3级" :value="3" />
|
|
|
|
|
|
<el-option label="4级" :value="4" />
|
|
|
|
|
|
<el-option label="5级" :value="5" />
|
|
|
|
|
|
</el-select>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="是否心动礼物" prop="isHeartbeat">
|
|
|
|
|
|
<el-switch v-model="form.isHeartbeat" :active-value="1" :inactive-value="0" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="排序" prop="sort">
|
|
|
|
|
|
<el-input-number v-model="form.sort" :min="0" :max="9999" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="状态" prop="status">
|
|
|
|
|
|
<el-radio-group v-model="form.status">
|
|
|
|
|
|
<el-radio :label="1">启用</el-radio>
|
|
|
|
|
|
<el-radio :label="0">禁用</el-radio>
|
|
|
|
|
|
</el-radio-group>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="备注" prop="remark">
|
|
|
|
|
|
<el-input v-model="form.remark" type="textarea" placeholder="请输入备注" />
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
<div slot="footer" class="dialog-footer">
|
|
|
|
|
|
<el-button @click="cancel">取 消</el-button>
|
|
|
|
|
|
<el-button type="primary" @click="submitForm">确 定</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
|
import { giftListApi, giftAddApi, giftUpdateApi, giftDeleteApi, giftStatusApi } from '@/api/gift';
|
2026-01-04 19:00:02 +08:00
|
|
|
|
import { fileImageApi } from '@/api/systemSetting';
|
2026-01-03 19:22:42 +08:00
|
|
|
|
import DataPagination from '@/components/common/DataPagination';
|
|
|
|
|
|
|
|
|
|
|
|
export default {
|
|
|
|
|
|
name: 'GiftManage',
|
|
|
|
|
|
components: {
|
|
|
|
|
|
pagination: DataPagination
|
|
|
|
|
|
},
|
|
|
|
|
|
data() {
|
|
|
|
|
|
return {
|
|
|
|
|
|
loading: false,
|
|
|
|
|
|
multiple: true,
|
|
|
|
|
|
ids: [],
|
|
|
|
|
|
total: 0,
|
|
|
|
|
|
giftList: [],
|
|
|
|
|
|
title: '',
|
|
|
|
|
|
open: false,
|
|
|
|
|
|
statistics: {
|
|
|
|
|
|
total: 0,
|
|
|
|
|
|
enabled: 0,
|
|
|
|
|
|
disabled: 0,
|
|
|
|
|
|
totalValue: 0
|
|
|
|
|
|
},
|
|
|
|
|
|
queryParams: {
|
|
|
|
|
|
page: 1,
|
|
|
|
|
|
limit: 10,
|
|
|
|
|
|
name: '',
|
|
|
|
|
|
status: ''
|
|
|
|
|
|
},
|
|
|
|
|
|
form: {},
|
|
|
|
|
|
rules: {
|
|
|
|
|
|
name: [
|
|
|
|
|
|
{ required: true, message: '礼物名称不能为空', trigger: 'blur' }
|
|
|
|
|
|
],
|
|
|
|
|
|
image: [
|
|
|
|
|
|
{ required: true, message: '请上传礼物图标', trigger: 'change' }
|
|
|
|
|
|
],
|
|
|
|
|
|
diamondPrice: [
|
|
|
|
|
|
{ required: true, message: '价格不能为空', trigger: 'blur' }
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
},
|
|
|
|
|
|
created() {
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
this.getStatistics();
|
|
|
|
|
|
},
|
|
|
|
|
|
methods: {
|
|
|
|
|
|
getList() {
|
|
|
|
|
|
this.loading = true;
|
|
|
|
|
|
giftListApi(this.queryParams).then(response => {
|
2026-01-04 19:00:02 +08:00
|
|
|
|
console.log('礼物列表响应:', response);
|
|
|
|
|
|
|
|
|
|
|
|
// axios 拦截器已经返回了 res.data,所以 response 就是 CommonPage 对象
|
|
|
|
|
|
// response = { list: [], total: 0, page: 1, limit: 10, totalPage: 1 }
|
|
|
|
|
|
const list = response.list || [];
|
|
|
|
|
|
|
|
|
|
|
|
// 确保 status 字段是数字类型(1 或 0)
|
|
|
|
|
|
this.giftList = list.map(item => ({
|
|
|
|
|
|
...item,
|
|
|
|
|
|
status: Number(item.status) // 确保是数字类型
|
|
|
|
|
|
}));
|
|
|
|
|
|
this.total = response.total || 0;
|
|
|
|
|
|
|
|
|
|
|
|
console.log('礼物列表数据:', this.giftList);
|
|
|
|
|
|
console.log('总数:', this.total);
|
|
|
|
|
|
|
2026-01-03 19:22:42 +08:00
|
|
|
|
this.loading = false;
|
2026-01-04 19:00:02 +08:00
|
|
|
|
}).catch(error => {
|
|
|
|
|
|
console.error('获取礼物列表失败:', error);
|
|
|
|
|
|
this.$message.error('获取礼物列表失败');
|
2026-01-03 19:22:42 +08:00
|
|
|
|
this.loading = false;
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
getStatistics() {
|
|
|
|
|
|
// 计算统计数据
|
|
|
|
|
|
giftListApi({ page: 1, limit: 9999 }).then(response => {
|
2026-01-04 19:00:02 +08:00
|
|
|
|
const list = response.list || [];
|
2026-01-03 19:22:42 +08:00
|
|
|
|
this.statistics.total = list.length;
|
2026-01-04 19:00:02 +08:00
|
|
|
|
this.statistics.enabled = list.filter(item => item.status === 1 || item.status === true).length;
|
|
|
|
|
|
this.statistics.disabled = list.filter(item => item.status === 0 || item.status === false).length;
|
|
|
|
|
|
this.statistics.totalValue = list.reduce((sum, item) => sum + (parseFloat(item.diamondPrice) || 0), 0).toFixed(2);
|
2026-01-03 19:22:42 +08:00
|
|
|
|
}).catch(error => {
|
|
|
|
|
|
console.error('获取统计数据失败:', error);
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
handleQuery() {
|
|
|
|
|
|
this.queryParams.page = 1;
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
},
|
|
|
|
|
|
resetQuery() {
|
|
|
|
|
|
this.queryParams = {
|
|
|
|
|
|
page: 1,
|
|
|
|
|
|
limit: 10,
|
|
|
|
|
|
name: '',
|
|
|
|
|
|
status: ''
|
|
|
|
|
|
};
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
},
|
|
|
|
|
|
handleAdd() {
|
|
|
|
|
|
this.reset();
|
|
|
|
|
|
this.open = true;
|
|
|
|
|
|
this.title = '添加礼物';
|
|
|
|
|
|
},
|
|
|
|
|
|
handleUpdate(row) {
|
|
|
|
|
|
this.reset();
|
|
|
|
|
|
this.form = { ...row };
|
|
|
|
|
|
this.open = true;
|
|
|
|
|
|
this.title = '修改礼物';
|
|
|
|
|
|
},
|
|
|
|
|
|
handleDelete(row) {
|
|
|
|
|
|
const ids = row.id || this.ids;
|
|
|
|
|
|
this.$confirm('是否确认删除选中的礼物?', '警告', {
|
|
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
|
|
cancelButtonText: '取消',
|
|
|
|
|
|
type: 'warning'
|
|
|
|
|
|
}).then(() => {
|
|
|
|
|
|
return giftDeleteApi(ids);
|
|
|
|
|
|
}).then(() => {
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
this.getStatistics();
|
|
|
|
|
|
this.$message.success('删除成功');
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
handleStatusChange(row) {
|
|
|
|
|
|
const text = row.status === 1 ? '启用' : '禁用';
|
|
|
|
|
|
this.$confirm('确认要' + text + '该礼物吗?', '警告', {
|
|
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
|
|
cancelButtonText: '取消',
|
|
|
|
|
|
type: 'warning'
|
|
|
|
|
|
}).then(() => {
|
|
|
|
|
|
return giftStatusApi(row.id, row.status);
|
|
|
|
|
|
}).then(() => {
|
|
|
|
|
|
this.$message.success(text + '成功');
|
|
|
|
|
|
this.getStatistics();
|
|
|
|
|
|
}).catch(() => {
|
|
|
|
|
|
row.status = row.status === 0 ? 1 : 0;
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
handleSelectionChange(selection) {
|
|
|
|
|
|
this.ids = selection.map(item => item.id);
|
|
|
|
|
|
this.multiple = !selection.length;
|
|
|
|
|
|
},
|
|
|
|
|
|
submitForm() {
|
|
|
|
|
|
this.$refs['form'].validate(valid => {
|
|
|
|
|
|
if (valid) {
|
|
|
|
|
|
if (this.form.id) {
|
|
|
|
|
|
giftUpdateApi(this.form).then(() => {
|
|
|
|
|
|
this.$message.success('修改成功');
|
|
|
|
|
|
this.open = false;
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
this.getStatistics();
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
giftAddApi(this.form).then(() => {
|
|
|
|
|
|
this.$message.success('新增成功');
|
|
|
|
|
|
this.open = false;
|
|
|
|
|
|
this.getList();
|
|
|
|
|
|
this.getStatistics();
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
cancel() {
|
|
|
|
|
|
this.open = false;
|
|
|
|
|
|
this.reset();
|
|
|
|
|
|
},
|
|
|
|
|
|
reset() {
|
|
|
|
|
|
this.form = {
|
|
|
|
|
|
id: null,
|
|
|
|
|
|
name: '',
|
|
|
|
|
|
image: '',
|
|
|
|
|
|
diamondPrice: 1,
|
|
|
|
|
|
intimacy: 1,
|
|
|
|
|
|
level: 1,
|
|
|
|
|
|
isHeartbeat: 0,
|
|
|
|
|
|
sort: 0,
|
|
|
|
|
|
status: 1,
|
|
|
|
|
|
remark: '',
|
|
|
|
|
|
buyType: '钻石',
|
|
|
|
|
|
belong: '平台'
|
|
|
|
|
|
};
|
|
|
|
|
|
this.$nextTick(() => {
|
|
|
|
|
|
if (this.$refs['form']) {
|
|
|
|
|
|
this.$refs['form'].clearValidate();
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
handleUpload(param) {
|
|
|
|
|
|
const formData = new FormData();
|
2026-01-04 19:00:02 +08:00
|
|
|
|
formData.append('multipart', param.file); // 后端参数名是 multipart
|
|
|
|
|
|
|
|
|
|
|
|
// 显示上传中提示
|
|
|
|
|
|
const loading = this.$loading({
|
|
|
|
|
|
lock: true,
|
|
|
|
|
|
text: '上传中...',
|
|
|
|
|
|
spinner: 'el-icon-loading',
|
|
|
|
|
|
background: 'rgba(0, 0, 0, 0.7)'
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
fileImageApi(formData, { model: 'gift', pid: 0 })
|
|
|
|
|
|
.then(response => {
|
|
|
|
|
|
loading.close();
|
|
|
|
|
|
console.log('上传响应:', response);
|
|
|
|
|
|
|
|
|
|
|
|
// 根据响应格式设置图片地址
|
|
|
|
|
|
if (response) {
|
|
|
|
|
|
// response 可能是 FileResultVo 对象或直接是 URL
|
|
|
|
|
|
this.form.image = response.url || response.fileUrl || response;
|
|
|
|
|
|
this.$message.success('上传成功');
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.$message.error('上传失败,返回数据格式错误');
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
.catch(error => {
|
|
|
|
|
|
loading.close();
|
|
|
|
|
|
console.error('上传失败:', error);
|
|
|
|
|
|
this.$message.error('上传失败: ' + (error.message || '未知错误'));
|
|
|
|
|
|
});
|
2026-01-03 19:22:42 +08:00
|
|
|
|
},
|
|
|
|
|
|
beforeUpload(file) {
|
|
|
|
|
|
const isImage = file.type.indexOf('image/') === 0;
|
|
|
|
|
|
const isLt2M = file.size / 1024 / 1024 < 2;
|
|
|
|
|
|
if (!isImage) {
|
|
|
|
|
|
this.$message.error('只能上传图片文件!');
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!isLt2M) {
|
|
|
|
|
|
this.$message.error('上传图片大小不能超过 2MB!');
|
|
|
|
|
|
}
|
|
|
|
|
|
return isImage && isLt2M;
|
2026-01-04 19:00:02 +08:00
|
|
|
|
},
|
|
|
|
|
|
getImageUrl(url) {
|
|
|
|
|
|
if (!url) {
|
|
|
|
|
|
console.log('图片URL为空');
|
|
|
|
|
|
return '';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log('原始图片URL:', url);
|
|
|
|
|
|
console.log('URL类型:', typeof url);
|
|
|
|
|
|
|
|
|
|
|
|
// 转换为字符串(防止是对象)
|
|
|
|
|
|
let urlStr = String(url).trim();
|
|
|
|
|
|
console.log('转换后的URL:', urlStr);
|
|
|
|
|
|
|
|
|
|
|
|
// 修复:如果URL被重复拼接,提取正确的路径
|
|
|
|
|
|
// 例如:http://domain/http://domain/crmebimage/... -> crmebimage/...
|
|
|
|
|
|
if (urlStr.indexOf('http://') > 0 || urlStr.indexOf('https://') > 0) {
|
|
|
|
|
|
console.log('检测到重复拼接的URL,尝试修复');
|
|
|
|
|
|
// 找到第二个 http:// 或 https:// 的位置
|
|
|
|
|
|
const secondHttpIndex = urlStr.indexOf('http://', 1);
|
|
|
|
|
|
const secondHttpsIndex = urlStr.indexOf('https://', 1);
|
|
|
|
|
|
|
|
|
|
|
|
if (secondHttpIndex > 0) {
|
|
|
|
|
|
urlStr = urlStr.substring(secondHttpIndex);
|
|
|
|
|
|
console.log('修复后的URL:', urlStr);
|
|
|
|
|
|
} else if (secondHttpsIndex > 0) {
|
|
|
|
|
|
urlStr = urlStr.substring(secondHttpsIndex);
|
|
|
|
|
|
console.log('修复后的URL:', urlStr);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 如果已经是完整的 URL(http:// 或 https://),直接返回
|
|
|
|
|
|
if (urlStr.indexOf('http://') === 0 || urlStr.indexOf('https://') === 0) {
|
|
|
|
|
|
console.log('检测到完整URL,直接返回');
|
|
|
|
|
|
return urlStr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 如果是相对路径,拼接服务器地址
|
|
|
|
|
|
// 移除开头的 / 避免重复
|
|
|
|
|
|
let path = urlStr;
|
|
|
|
|
|
if (path.charAt(0) === '/') {
|
|
|
|
|
|
path = path.substring(1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 使用当前页面的协议和主机名
|
|
|
|
|
|
const baseUrl = window.location.origin;
|
|
|
|
|
|
const fullUrl = baseUrl + '/' + path;
|
|
|
|
|
|
|
|
|
|
|
|
console.log('拼接后的完整URL:', fullUrl);
|
|
|
|
|
|
|
|
|
|
|
|
return fullUrl;
|
2026-01-03 19:22:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.mb20 {
|
|
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-card {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-icon {
|
|
|
|
|
|
font-size: 48px;
|
|
|
|
|
|
margin-right: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-content {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-value {
|
|
|
|
|
|
font-size: 28px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: #303133;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-label {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #909399;
|
|
|
|
|
|
margin-top: 5px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.avatar-uploader {
|
|
|
|
|
|
border: 1px dashed #d9d9d9;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.avatar-uploader:hover {
|
|
|
|
|
|
border-color: #409EFF;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.avatar-uploader-icon {
|
|
|
|
|
|
font-size: 28px;
|
|
|
|
|
|
color: #8c939d;
|
|
|
|
|
|
width: 120px;
|
|
|
|
|
|
height: 120px;
|
|
|
|
|
|
line-height: 120px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.avatar {
|
|
|
|
|
|
width: 120px;
|
|
|
|
|
|
height: 120px;
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
}
|
2026-01-04 19:00:02 +08:00
|
|
|
|
|
|
|
|
|
|
.image-slot {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
background: #f5f7fa;
|
|
|
|
|
|
color: #909399;
|
|
|
|
|
|
font-size: 30px;
|
|
|
|
|
|
}
|
2026-01-03 19:22:42 +08:00
|
|
|
|
</style>
|