| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654 | <template>    <basic-container>        <avue-crud            :option="option"            :data="data"            ref="crud"            v-model="form"            :page.sync="page"            :permission="permissionList"            :before-open="beforeOpen"            :table-loading="loading"            @row-update="rowUpdate"            @row-save="rowSave"            @search-change="searchChange"            @search-reset="searchReset"            @selection-change="selectionChange"            @current-change="currentChange"            @size-change="sizeChange"            @refresh-change="refreshChange"            @on-load="onLoad"        >            <template slot="menuLeft">                <el-button                    type="primary"                    size="small"                    icon="el-icon-view"                    plain                    @click="handleDetail"                    v-if="selectionList.length === 1"                >                    查看详情                </el-button>            </template>            <template slot-scope="{row}" slot="categoryName">                <el-tag type="primary">{{ row.categoryName }}</el-tag>            </template>            <template slot-scope="{row}" slot="status">                <el-tag :type="row.status === 1 ? 'success' : 'danger'">                    {{ row.status === 1 ? '正常' : '禁用' }}                </el-tag>            </template>            <template slot-scope="{row}" slot="visibleRoles">                <el-tag>{{ getVisibleRolesText(row.visibleRoles) }}</el-tag>            </template>        </avue-crud>        <!-- 详情对话框 -->        <el-dialog title="公告详情" :visible.sync="detailVisible" width="60%" :close-on-click-modal="false">            <div class="detail-content" v-if="currentDetail.id">                <div class="detail-info">                    <p><strong>公告标题:</strong>{{ currentDetail.title }}</p>                    <p><strong>分类:</strong>{{ currentDetail.categoryName }}</p>                    <p><strong>组织:</strong>{{ currentDetail.orgName }}</p>                    <p><strong>创建时间:</strong>{{ currentDetail.createTime }}</p>                    <p><strong>可见角色:</strong>{{ getVisibleRolesText(currentDetail.visibleRoles) }}</p>                    <p><strong>状态:</strong>                        <el-tag :type="currentDetail.status === 1 ? 'success' : 'danger'">                            {{ currentDetail.status === 1 ? '正常' : '禁用' }}                        </el-tag>                    </p>                </div>                <div class="detail-body" v-html="currentDetail.content"></div>            </div>            <span slot="footer" class="dialog-footer">                <el-button @click="detailVisible = false">关 闭</el-button>            </span>        </el-dialog>    </basic-container></template><script>import { getList, update, add, getAnnouncement, getCategoryList } from "@/api/announcement";import { mapGetters } from "vuex";/** * 公告数据类型定义 * @typedef {Object} NoticeItem * @property {string} id - 公告ID * @property {string} title - 公告标题 * @property {string} content - 公告内容 * @property {string} categoryId - 分类ID * @property {string} categoryName - 分类名称 * @property {number} orgId - 组织ID * @property {string} orgCode - 组织编码 * @property {string} orgName - 组织名称 * @property {string} visibleRoles - 可见角色 * @property {Object|null} brandScope - 品牌范围 * @property {Object|null} customerBlacklist - 客户黑名单 * @property {string} remark - 备注 * @property {string} createTime - 创建时间 * @property {string} updateTime - 更新时间 * @property {number} status - 状态 * @property {number} isDeleted - 是否删除 *//** * 分类选项类型定义 * @typedef {Object} CategoryOption * @property {string} id - 分类ID * @property {string} name - 分类名称 * @property {string} value - 选项值 * @property {string} label - 选项标签 *//** * 分页信息类型定义 * @typedef {Object} PageInfo * @property {number} pageSize - 每页大小 * @property {number} currentPage - 当前页码 * @property {number} total - 总记录数 *//** * 查询参数类型定义 * @typedef {Object} QueryParams * @property {string} [title] - 公告标题 * @property {string} [categoryId] - 分类ID * @property {string} [content] - 公告内容 *//** * 公告管理组件 * @component NoticeIndex */export default {    name: 'NoticeIndex',    data() {        return {            /** @type {NoticeItem} 表单数据 */            form: {},            /** @type {QueryParams} 查询参数 */            query: {},            /** @type {boolean} 加载状态 */            loading: true,            /** @type {boolean} 详情对话框显示状态 */            detailVisible: false,            /** @type {NoticeItem} 当前查看的详情数据 */            currentDetail: {},            /** @type {PageInfo} 分页信息 */            page: {                pageSize: 10,                currentPage: 1,                total: 0            },            /** @type {NoticeItem[]} 选中的数据列表 */            selectionList: [],            /** @type {CategoryOption[]} 分类选项列表 */            categoryOptions: [],            /** @type {Object} 表格配置选项 */            option: {                height: 'auto',                calcHeight: 30,                dialogWidth: 1000,                labelWidth: 120,                tip: false,                searchShow: true,                searchMenuSpan: 6,                border: true,                index: true,                viewBtn: true,                selection: true,                excelBtn: false,                columnBtn: false,                delBtn: false, // 根据需求移除删除功能                dialogClickModal: false,                column: [                    {                        label: "公告标题",                        prop: "title",                        span: 12,                        search: true,                        overHidden: true,                        rules: [{                            required: true,                            message: "请输入公告标题",                            trigger: "blur"                        }]                    },                    {                        label: "分类",                        prop: "categoryId",                        type: "select",                        dicData: [],                        props: {                            label: "name",                            value: "id"                        },                        slot: true,                        search: true,                        span: 12,                        rules: [{                            required: true,                            message: "请选择分类",                            trigger: "change"                        }]                    },                    {                        label: "组织名称",                        prop: "orgName",                        span: 12,                        overHidden: true,                        rules: [{                            required: true,                            message: "请输入组织名称",                            trigger: "blur"                        }]                    },                    {                        label: "组织ID",                        prop: "orgId",                        span: 12,                        type: "number",                        rules: [{                            required: true,                            message: "请输入组织名称",                            trigger: "blur"                        }]                    },                    {                        label: "组织编码",                        prop: "orgCode",                        span: 12,                        rules: [{                            required: true,                            message: "请输入组织编码",                            trigger: "blur"                        }]                    },                    {                        label: "可见角色",                        prop: "visibleRoles",                        type: "select",                        dicData: [                            { label: "管理员", value: "1" },                            { label: "普通用户", value: "2" },                            { label: "访客", value: "3" },                            { label: "VIP用户", value: "4" }                        ],                        slot: true,                        span: 12,                        rules: [{                            required: true,                            message: "请选择可见角色",                            trigger: "change"                        }]                    },                    {                        label: "状态",                        prop: "status",                        type: "select",                        dicData: [                            { label: "正常", value: 1 },                            { label: "禁用", value: 0 }                        ],                        slot: true,                        addDisplay: false,                        editDisplay: false,                        width: 80                    },                    {                        label: "创建时间",                        prop: "createTime",                        type: "datetime",                        format: "yyyy-MM-dd HH:mm:ss",                        valueFormat: "yyyy-MM-dd HH:mm:ss",                        addDisplay: false,                        editDisplay: false,                        overHidden: true,                        width: 150                    },                    {                        label: "分类名称",                        prop: "categoryName",                        hide: true,                        addDisplay: false,                        editDisplay: false                    },                    {                        label: "备注",                        prop: "remark",                        type: "textarea",                        span: 24,                        minRows: 3,                        hide: true                    },                    {                        label: "公告内容",                        prop: "content",                        component: 'AvueUeditor',                        options: {                            action: '/api/blade-resource/oss/endpoint/put-file',                            props: {                                res: "data",                                url: "link",                            }                        },                        showColumn: false,                        hide: true,                        minRows: 6,                        span: 24,                        rules: [{                            required: true,                            message: "请输入公告内容",                            trigger: "blur"                        }]                    },                    {                        label: "品牌范围",                        prop: "brandScope",                        type: "json",                        hide: true,                        span: 24                    },                    {                        label: "客户黑名单",                        prop: "customerBlacklist",                        type: "json",                        hide: true,                        span: 24                    }                ]            },            /** @type {NoticeItem[]} 表格数据 */            data: []        };    },    computed: {        ...mapGetters(["permission", "userInfo"]),        /**         * 权限列表         * @returns {Object} 权限配置对象         */        permissionList() {            return {                // addBtn: this.vaildData(this.permission.announcement_add, false),                // viewBtn: this.vaildData(this.permission.announcement_view, false),                // delBtn: false,                // editBtn: this.vaildData(this.permission.announcement_edit, false)                addBtn: true,                viewBtn: true,                delBtn: false,                editBtn: true,            };        },        /**         * 选中的ID字符串         * @returns {string} 逗号分隔的ID字符串         */        ids() {            const ids = [];            this.selectionList.forEach(ele => {                ids.push(ele.id);            });            return ids.join(",");        }    },    created() {        this.loadCategoryOptions();    },    methods: {        /**         * 加载分类选项         * @async         * @returns {Promise<void>}         */        async loadCategoryOptions() {            try {                const res = await getCategoryList();                const categoryData = res.data.data || [];                this.categoryOptions = categoryData                    .filter(item => item.status === 1 && item.isDeleted === 0)                    .map(item => ({                        id: item.id,                        name: item.name,                        value: item.id,                        label: item.name,                        orgId: item.orgId,                        orgName: item.orgName,                        sortOrder: item.sortOrder || 0                    }))                    .sort((a, b) => a.sortOrder - b.sortOrder);                const categoryColumn = this.option.column.find(col => col.prop === 'categoryId');                if (categoryColumn) {                    categoryColumn.dicData = this.categoryOptions;                }            } catch (error) {                console.error('加载分类选项失败:', error);                // 使用默认分类                this.categoryOptions = [                    { id: '1', name: '系统公告', value: '1', label: '系统公告' },                    { id: '2', name: '部门公告', value: '2', label: '部门公告' }                ];                const categoryColumn = this.option.column.find(col => col.prop === 'categoryId');                if (categoryColumn) {                    categoryColumn.dicData = this.categoryOptions;                }            }        },        /**         * 获取可见角色文本         * @param {string} visibleRoles - 可见角色值         * @returns {string} 角色文本         */        getVisibleRolesText(visibleRoles) {            const roleMap = {                '1': '管理员',                '2': '普通用户',                '3': '访客',                '4': 'VIP用户'            };            return roleMap[visibleRoles] || '未知角色';        },        /**         * 查看详情         * @async         * @returns {Promise<void>}         */        async handleDetail() {            if (this.selectionList.length !== 1) {                this.$message.warning("请选择一条数据查看详情");                return;            }            try {                const res = await getAnnouncement(this.selectionList[0].id);                this.currentDetail = res.data.data;                this.detailVisible = true;            } catch (error) {                console.error('获取详情失败:', error);                this.$message.error('获取详情失败');            }        },        /**         * 保存行数据         * @async         * @param {NoticeItem} row - 行数据         * @param {Function} done - 完成回调         * @param {Function} loading - 加载回调         * @returns {Promise<void>}         */        async rowSave(row, done, loading) {            try {                // 设置默认值                const formData = {                    ...row,                    orgId: this.userInfo.orgId || 1,                    orgCode: this.userInfo.orgCode || 'ORG_0001',                    orgName: this.userInfo.orgName || '默认组织',                    brandScope: row.brandScope || {},                    customerBlacklist: row.customerBlacklist || {},                    remark: row.remark || ''                };                // 设置分类名称                const selectedCategory = this.categoryOptions.find(cat => cat.id === row.categoryId);                if (selectedCategory) {                    formData.categoryName = selectedCategory.name;                }                await add(formData);                this.onLoad(this.page);                this.$message({                    type: "success",                    message: "操作成功!"                });                done();            } catch (error) {                console.error('保存失败:', error);                this.$message.error('保存失败');                loading();            }        },        /**         * 更新行数据         * @async         * @param {NoticeItem} row - 行数据         * @param {number} index - 行索引         * @param {Function} done - 完成回调         * @param {Function} loading - 加载回调         * @returns {Promise<void>}         */        async rowUpdate(row, index, done, loading) {            try {                // 设置分类名称                const selectedCategory = this.categoryOptions.find(cat => cat.id === row.categoryId);                if (selectedCategory) {                    row.categoryName = selectedCategory.name;                }                // 确保必要字段存在                const formData = {                    ...row,                    brandScope: row.brandScope || {},                    customerBlacklist: row.customerBlacklist || {},                    remark: row.remark || ''                };                await update(formData);                this.onLoad(this.page);                this.$message({                    type: "success",                    message: "操作成功!"                });                done();            } catch (error) {                console.error('更新失败:', error);                this.$message.error('更新失败');                loading();            }        },        /**         * 重置搜索         * @returns {void}         */        searchReset() {            this.query = {};            this.onLoad(this.page);        },        /**         * 搜索变化         * @param {QueryParams} params - 搜索参数         * @param {Function} done - 完成回调         * @returns {void}         */        searchChange(params, done) {            this.query = params;            this.page.currentPage = 1;            this.onLoad(this.page, params);            done();        },        /**         * 选择变化         * @param {NoticeItem[]} list - 选中的数据列表         * @returns {void}         */        selectionChange(list) {            this.selectionList = list;        },        /**         * 清空选择         * @returns {void}         */        selectionClear() {            this.selectionList = [];            this.$refs.crud.toggleSelection();        },        /**         * 打开前回调         * @async         * @param {Function} done - 完成回调         * @param {string} type - 操作类型         * @returns {Promise<void>}         */        async beforeOpen(done, type) {            if (["edit", "view"].includes(type)) {                try {                    const res = await getAnnouncement(this.form.id);                    this.form = res.data.data;                } catch (error) {                    console.error('获取详情失败:', error);                }            } else if (type === "add") {                // 新增时设置默认值                this.form = {                    orgId: this.userInfo.orgId || 1,                    orgCode: this.userInfo.orgCode || 'ORG_0001',                    orgName: this.userInfo.orgName || '默认组织',                    visibleRoles: '2', // 默认普通用户                    brandScope: {},                    customerBlacklist: {},                    remark: ''                };            }            done();        },        /**         * 当前页变化         * @param {number} currentPage - 当前页码         * @returns {void}         */        currentChange(currentPage) {            this.page.currentPage = currentPage;        },        /**         * 页大小变化         * @param {number} pageSize - 页大小         * @returns {void}         */        sizeChange(pageSize) {            this.page.pageSize = pageSize;        },        /**         * 刷新变化         * @returns {void}         */        refreshChange() {            this.onLoad(this.page, this.query);        },        /**         * 加载数据         * @async         * @param {PageInfo} page - 分页信息         * @param {QueryParams} [params={}] - 查询参数         * @returns {Promise<void>}         */        async onLoad(page, params = {}) {            this.loading = true;            try {                const queryParams = {                    ...params,                    ...this.query                };                const res = await getList(page.currentPage, page.pageSize, queryParams);                const data = res.data.data;                this.page.total = data.total;                this.data = data.records;                this.loading = false;                this.selectionClear();            } catch (error) {                console.error('加载数据失败:', error);                this.$message.error('加载数据失败');                this.loading = false;            }        }    }};</script><style scoped>.detail-content {    padding: 20px;}.detail-info {    margin: 20px 0;    padding: 15px;    background-color: #f5f5f5;    border-radius: 4px;}.detail-info p {    margin: 8px 0;}.detail-body {    margin-top: 20px;    padding: 15px;    border: 1px solid #e4e7ed;    border-radius: 4px;    min-height: 200px;}</style>
 |