From 0bf3f3967322f09ec83051b93309866d2ef091c9 Mon Sep 17 00:00:00 2001
From: jiangping <jp@doumee.com>
Date: 星期五, 04 七月 2025 11:10:30 +0800
Subject: [PATCH] ss

---
 server/src/main/java/com/doumee/dao/business/model/Seo.java                       |   91 +++++++
 admin/src/components/business/OperaSeoWindow.vue                                  |   71 +++++
 server/src/main/java/com/doumee/dao/business/model/Information.java               |    5 
 server/src/main/java/com/doumee/service/business/impl/CarouselServiceImpl.java    |    4 
 admin/src/views/business/seo.vue                                                  |  100 ++++++++
 server/src/main/java/com/doumee/service/business/impl/InformationServiceImpl.java |    5 
 admin/src/api/business/seo.js                                                     |   32 ++
 server/src/main/java/com/doumee/dao/business/SeoMapper.java                       |   13 +
 server/src/main/java/com/doumee/service/business/impl/SeoServiceImpl.java         |  169 ++++++++++++++
 server/src/main/java/com/doumee/service/business/SeoService.java                  |   97 ++++++++
 server/src/main/java/com/doumee/api/business/SeoController.java                   |   85 +++++++
 11 files changed, 670 insertions(+), 2 deletions(-)

diff --git a/admin/src/api/business/seo.js b/admin/src/api/business/seo.js
new file mode 100644
index 0000000..49237fc
--- /dev/null
+++ b/admin/src/api/business/seo.js
@@ -0,0 +1,32 @@
+import request from '../../utils/request'
+
+// 鏌ヨ
+export function fetchList (data) {
+  return request.post('/business/seo/page', data, {
+    trim: true
+  })
+}
+
+// 鍒涘缓
+export function create (data) {
+  return request.post('/business/seo/create', data)
+}
+
+// 淇敼
+export function updateById (data) {
+  return request.post('/business/seo/updateById', data)
+}
+
+// 鍒犻櫎
+export function deleteById (id) {
+  return request.get(`/business/seo/delete/${id}`)
+}
+
+// 鎵归噺鍒犻櫎
+export function deleteByIdInBatch (ids) {
+  return request.get('/business/seo/delete/batch', {
+    params: {
+      ids
+    }
+  })
+}
diff --git a/admin/src/components/business/OperaSeoWindow.vue b/admin/src/components/business/OperaSeoWindow.vue
new file mode 100644
index 0000000..703ccc1
--- /dev/null
+++ b/admin/src/components/business/OperaSeoWindow.vue
@@ -0,0 +1,71 @@
+<template>
+  <GlobalWindow
+    :title="title"
+    :visible.sync="visible"
+    :confirm-working="isWorking"
+    @confirm="confirm"
+  >
+    <el-form :model="form" ref="form" :rules="rules">
+      <el-form-item label="椤甸潰鍚嶇О" prop="name">
+        <el-input v-model="form.name" placeholder="璇疯緭鍏ラ〉闈㈠悕绉�" v-trim/>
+      </el-form-item>
+      <el-form-item label="椤甸潰鍦板潃" prop="pageAddr">
+        <el-input v-model="form.pageAddr" placeholder="璇疯緭鍏ラ〉闈㈠湴鍧�锛�/default.html 琛ㄧず缃戠珯榛樿閰嶇疆椤癸級" v-trim/>
+      </el-form-item>
+      <el-form-item label="鏍囬" prop="title">
+        <el-input v-model="form.title" placeholder="璇疯緭鍏ユ爣棰�" v-trim/>
+      </el-form-item>
+      <el-form-item label="鍏抽敭瀛�" prop="keywords">
+        <el-input type="textarea" v-model="form.keywords" placeholder="璇疯緭鍏ュ叧閿瓧" v-trim/>
+      </el-form-item>
+      <el-form-item label="浣滆�呬俊鎭�" prop="author">
+        <el-input v-model="form.author" placeholder="璇疯緭鍏ヤ綔鑰呬俊鎭�" v-trim/>
+      </el-form-item>
+      <el-form-item label="鎺掑簭鐮�(鍗囧簭)" prop="sortnum">
+        <el-input v-model="form.sortnum" type="number" placeholder="璇疯緭鍏ユ帓搴忕爜" v-trim/>
+      </el-form-item>
+      <el-form-item label="鎻忚堪" prop="remark">
+        <el-input v-model="form.remark" type="textarea" placeholder="璇疯緭鍏ユ弿杩�" v-trim/>
+      </el-form-item>
+      <el-form-item label="鍥剧墖alt" prop="imgalt">
+        <el-input v-model="form.imgalt" placeholder="璇疯緭鍏ュ浘鐗嘺lt" v-trim/>
+      </el-form-item>
+    </el-form>
+  </GlobalWindow>
+</template>
+
+<script>
+import BaseOpera from '@/components/base/BaseOpera'
+import GlobalWindow from '@/components/common/GlobalWindow'
+export default {
+  name: 'OperaSeoWindow',
+  extends: BaseOpera,
+  components: { GlobalWindow },
+  data () {
+    return {
+      // 琛ㄥ崟鏁版嵁
+      form: {
+        id: null,
+        remark: '',
+        name: '',
+        title: '',
+        keywords: '',
+        author: '',
+        icon: '',
+        pageAddr: '',
+        sortnum: '',
+        imgalt: ''
+      },
+      // 楠岃瘉瑙勫垯
+      rules: {
+      }
+    }
+  },
+  created () {
+    this.config({
+      api: '/business/seo',
+      'field.id': 'id'
+    })
+  }
+}
+</script>
diff --git a/admin/src/views/business/seo.vue b/admin/src/views/business/seo.vue
new file mode 100644
index 0000000..a59aeb7
--- /dev/null
+++ b/admin/src/views/business/seo.vue
@@ -0,0 +1,100 @@
+<template>
+  <TableLayout :permissions="['business:seo:query']">
+    <!-- 鎼滅储琛ㄥ崟 -->
+    <el-form ref="searchForm" slot="search-form" :model="searchForm" label-width="100px" inline>
+      <el-form-item label="椤甸潰鍚嶇О" prop="name">
+        <el-input v-model="searchForm.name" placeholder="璇疯緭鍏ラ〉闈㈠悕绉�" @keypress.enter.native="search"></el-input>
+      </el-form-item>
+      <el-form-item label="鏍囬" prop="title">
+        <el-input v-model="searchForm.title" placeholder="璇疯緭鍏ユ爣棰�" @keypress.enter.native="search"></el-input>
+      </el-form-item>
+      <el-form-item label="鍏抽敭瀛�" prop="keywords">
+        <el-input v-model="searchForm.keywords" placeholder="璇疯緭鍏ュ叧閿瓧" @keypress.enter.native="search"></el-input>
+      </el-form-item>
+      <el-form-item label="椤甸潰鍦板潃" prop="pageAddr">
+        <el-input v-model="searchForm.pageAddr" placeholder="璇疯緭鍏ラ〉闈㈠湴鍧�" @keypress.enter.native="search"></el-input>
+      </el-form-item>
+      <section>
+        <el-button type="primary" @click="search">鎼滅储</el-button>
+        <el-button @click="reset">閲嶇疆</el-button>
+      </section>
+    </el-form>
+    <!-- 琛ㄦ牸鍜屽垎椤� -->
+    <template v-slot:table-wrap>
+      <ul class="toolbar" v-permissions="['business:seo:create', 'business:seo:delete']">
+        <li><el-button type="primary" @click="$refs.operaSeoWindow.open('鏂板缓缃戠珯SEO閰嶇疆')" icon="el-icon-plus" v-permissions="['business:seo:create']">鏂板缓</el-button></li>
+        <li><el-button @click="deleteByIdInBatch" icon="el-icon-delete" v-permissions="['business:seo:delete']">鍒犻櫎</el-button></li>
+      </ul>
+      <el-table
+        v-loading="isWorking.search"
+        :data="tableData.list"
+        stripe
+        @selection-change="handleSelectionChange"
+      >
+        <el-table-column type="selection" width="55"></el-table-column>
+        <el-table-column prop="name" label="椤甸潰鍚嶇О" min-width="100px"></el-table-column>
+        <el-table-column prop="title" label="鏍囬" min-width="100px"></el-table-column>
+        <el-table-column prop="keywords" label="鍏抽敭瀛�" min-width="150px"></el-table-column>
+        <el-table-column prop="author" label="浣滆�呬俊鎭�" min-width="100px"></el-table-column>
+        <el-table-column prop="pageAddr" label="椤甸潰鍦板潃" min-width="150px"></el-table-column>
+        <el-table-column prop="remark" label="鎻忚堪" min-width="100px"></el-table-column>
+        <el-table-column prop="imgalt" label="鍥剧墖alt" min-width="100px"></el-table-column>
+        <el-table-column prop="sortnum" label="鎺掑簭鐮�" min-width="100px"></el-table-column>
+        <el-table-column prop="editDate" label="鎿嶄綔鏃堕棿" min-width="150px"></el-table-column>
+        <el-table-column prop="editorName" label="鎿嶄綔浜�" min-width="100px"></el-table-column>
+        <el-table-column
+          v-if="containPermissions(['business:seo:update', 'business:seo:delete'])"
+          label="鎿嶄綔"
+          min-width="120"
+          fixed="right"
+        >
+          <template slot-scope="{row}">
+            <el-button type="text" @click="$refs.operaSeoWindow.open('缂栬緫缃戠珯SEO閰嶇疆', row)" icon="el-icon-edit" v-permissions="['business:seo:update']">缂栬緫</el-button>
+            <el-button type="text" @click="deleteById(row)" icon="el-icon-delete" v-permissions="['business:seo:delete']">鍒犻櫎</el-button>
+          </template>
+        </el-table-column>
+      </el-table>
+      <pagination
+        @size-change="handleSizeChange"
+        @current-change="handlePageChange"
+        :pagination="tableData.pagination"
+      >
+      </pagination>
+    </template>
+    <!-- 鏂板缓/淇敼 -->
+    <OperaSeoWindow ref="operaSeoWindow" @success="handlePageChange"/>
+  </TableLayout>
+</template>
+
+<script>
+import BaseTable from '@/components/base/BaseTable'
+import TableLayout from '@/layouts/TableLayout'
+import Pagination from '@/components/common/Pagination'
+import OperaSeoWindow from '@/components/business/OperaSeoWindow'
+export default {
+  name: 'Seo',
+  extends: BaseTable,
+  components: { TableLayout, Pagination, OperaSeoWindow },
+  data () {
+    return {
+      // 鎼滅储
+      searchForm: {
+        name: '',
+        title: '',
+        keywords: '',
+        author: '',
+        pageAddr: ''
+      }
+    }
+  },
+  created () {
+    this.config({
+      module: '缃戠珯SEO閰嶇疆',
+      api: '/business/seo',
+      'field.id': 'id',
+      'field.main': 'id'
+    })
+    this.search()
+  }
+}
+</script>
diff --git a/server/src/main/java/com/doumee/api/business/SeoController.java b/server/src/main/java/com/doumee/api/business/SeoController.java
new file mode 100644
index 0000000..957764d
--- /dev/null
+++ b/server/src/main/java/com/doumee/api/business/SeoController.java
@@ -0,0 +1,85 @@
+package com.doumee.api.business;
+
+import com.doumee.api.BaseController;
+import com.doumee.core.annotation.excel.ExcelExporter;
+import com.doumee.core.annotation.pr.PreventRepeat;
+import com.doumee.core.model.ApiResponse;
+import com.doumee.core.model.PageWrap;
+import com.doumee.core.model.PageData;
+import com.doumee.dao.business.model.Seo;
+import com.doumee.service.business.SeoService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import org.apache.shiro.authz.annotation.RequiresPermissions;    
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+import javax.servlet.http.HttpServletResponse;
+
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @author 姹熻箘韫�
+ * @since 2025/07/02 11:35
+ */
+@Api(tags = "缃戠珯SEO閰嶇疆")
+@RestController
+@RequestMapping("/business/seo")
+public class SeoController extends BaseController {
+
+    @Autowired
+    private SeoService seoService;
+
+    @PreventRepeat
+    @ApiOperation("鏂板缓")
+    @PostMapping("/create")
+    @RequiresPermissions("business:seo:create")
+    public ApiResponse create(@RequestBody Seo seo) {
+        return ApiResponse.success(seoService.create(seo));
+    }
+
+    @ApiOperation("鏍规嵁ID鍒犻櫎")
+    @GetMapping("/delete/{id}")
+    @RequiresPermissions("business:seo:delete")
+    public ApiResponse deleteById(@PathVariable Integer id) {
+        seoService.deleteById(id);
+        return ApiResponse.success(null);
+    }
+
+    @ApiOperation("鎵归噺鍒犻櫎")
+    @GetMapping("/delete/batch")
+    @RequiresPermissions("business:seo:delete")
+    public ApiResponse deleteByIdInBatch(@RequestParam String ids) {
+        seoService.deleteByIdInBatch(this.getIdList(ids));
+        return ApiResponse.success(null);
+    }
+
+    @ApiOperation("鏍规嵁ID淇敼")
+    @PostMapping("/updateById")
+    @RequiresPermissions("business:seo:update")
+    public ApiResponse updateById(@RequestBody Seo seo) {
+        seoService.updateById(seo);
+        return ApiResponse.success(null);
+    }
+
+    @ApiOperation("鍒嗛〉鏌ヨ")
+    @PostMapping("/page")
+    @RequiresPermissions("business:seo:query")
+    public ApiResponse<PageData<Seo>> findPage (@RequestBody PageWrap<Seo> pageWrap) {
+        return ApiResponse.success(seoService.findPage(pageWrap));
+    }
+
+    @ApiOperation("瀵煎嚭Excel")
+    @PostMapping("/exportExcel")
+    @RequiresPermissions("business:seo:exportExcel")
+    public void exportExcel (@RequestBody PageWrap<Seo> pageWrap, HttpServletResponse response) {
+        ExcelExporter.build(Seo.class).export(seoService.findPage(pageWrap).getRecords(), "缃戠珯SEO閰嶇疆", response);
+    }
+
+    @ApiOperation("鏍规嵁ID鏌ヨ")
+    @GetMapping("/{id}")
+    @RequiresPermissions("business:seo:query")
+    public ApiResponse findById(@PathVariable Integer id) {
+        return ApiResponse.success(seoService.findById(id));
+    }
+}
diff --git a/server/src/main/java/com/doumee/dao/business/SeoMapper.java b/server/src/main/java/com/doumee/dao/business/SeoMapper.java
new file mode 100644
index 0000000..3787fac
--- /dev/null
+++ b/server/src/main/java/com/doumee/dao/business/SeoMapper.java
@@ -0,0 +1,13 @@
+package com.doumee.dao.business;
+
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import com.doumee.dao.business.model.Seo;
+import com.github.yulichang.base.MPJBaseMapper;
+
+/**
+ * @author 姹熻箘韫�
+ * @since 2025/07/02 11:35
+ */
+public interface SeoMapper extends MPJBaseMapper<Seo> {
+
+}
diff --git a/server/src/main/java/com/doumee/dao/business/model/Information.java b/server/src/main/java/com/doumee/dao/business/model/Information.java
index 8567208..3d785b1 100644
--- a/server/src/main/java/com/doumee/dao/business/model/Information.java
+++ b/server/src/main/java/com/doumee/dao/business/model/Information.java
@@ -81,6 +81,11 @@
     @ApiModelProperty(value = "鐘舵�� 0姝e父 1绂佺敤", example = "1")
     @ExcelColumn(name="鐘舵�� 0姝e父 1绂佺敤")
     private Integer status;
+    @ApiModelProperty(value = "鎵�灞炴ā鍧� 0琛屼笟璧勮銆�1杩愯惀缁忛獙銆�2鍔熻兘浠嬬粛銆�3甯傚満淇℃伅", example = "1")
+    @ExcelColumn(name="module 0姝e父 1绂佺敤")
+    private Integer module;
+
+
 
     @ApiModelProperty(value = "灏侀潰鍥剧墖鍦板潃")
     @TableField(exist = false)
diff --git a/server/src/main/java/com/doumee/dao/business/model/Seo.java b/server/src/main/java/com/doumee/dao/business/model/Seo.java
new file mode 100644
index 0000000..66ceb69
--- /dev/null
+++ b/server/src/main/java/com/doumee/dao/business/model/Seo.java
@@ -0,0 +1,91 @@
+package com.doumee.dao.business.model;
+
+import com.baomidou.mybatisplus.annotation.TableField;
+import com.doumee.core.annotation.excel.ExcelColumn;
+import com.doumee.core.annotation.excel.ExcelColumn;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import com.baomidou.mybatisplus.annotation.IdType;
+import com.baomidou.mybatisplus.annotation.TableId;
+import com.baomidou.mybatisplus.annotation.TableName;
+import lombok.Data;
+import com.fasterxml.jackson.annotation.JsonFormat;
+import java.util.Date;
+
+/**
+ * 缃戠珯SEO閰嶇疆
+ * @author 姹熻箘韫�
+ * @since 2025/07/02 11:35
+ */
+@Data
+@ApiModel("缃戠珯SEO閰嶇疆")
+@TableName("`seo`")
+public class Seo {
+
+    @TableId(type = IdType.AUTO)
+    @ApiModelProperty(value = "涓婚敭", example = "1")
+    @ExcelColumn(name="涓婚敭")
+    private Integer id;
+
+    @ApiModelProperty(value = "鍒涘缓浜虹紪鐮�", example = "1")
+    @ExcelColumn(name="鍒涘缓浜虹紪鐮�")
+    private Integer creator;
+
+    @ApiModelProperty(value = "鍒涘缓鏃堕棿")
+    @ExcelColumn(name="鍒涘缓鏃堕棿")
+    @JsonFormat(pattern = "yyyy-MM-dd")
+    private Date createDate;
+
+    @ApiModelProperty(value = "鏇存柊浜虹紪鐮�", example = "1")
+    @ExcelColumn(name="鏇存柊浜虹紪鐮�")
+    private Integer editor;
+
+    @ApiModelProperty(value = "鏇存柊鏃堕棿")
+    @ExcelColumn(name="鏇存柊鏃堕棿")
+    @JsonFormat(pattern = "yyyy-MM-dd")
+    private Date editDate;
+
+    @ApiModelProperty(value = "鏄惁鍒犻櫎0鍚� 1鏄�", example = "1")
+    @ExcelColumn(name="鏄惁鍒犻櫎0鍚� 1鏄�")
+    private Integer isdeleted;
+
+    @ApiModelProperty(value = "澶囨敞")
+    @ExcelColumn(name="澶囨敞")
+    private String remark;
+
+    @ApiModelProperty(value = "椤甸潰鍚嶇О")
+    @ExcelColumn(name="椤甸潰鍚嶇О")
+    private String name;
+
+    @ApiModelProperty(value = "鏍囬")
+    @ExcelColumn(name="鏍囬")
+    private String title;
+
+    @ApiModelProperty(value = "鍏抽敭瀛�")
+    @ExcelColumn(name="鍏抽敭瀛�")
+    private String keywords;
+
+    @ApiModelProperty(value = "浣滆�呬俊鎭�")
+    @ExcelColumn(name="浣滆�呬俊鎭�")
+    private String author;
+
+    @ApiModelProperty(value = "鍥炬爣鍦板潃", example = "1")
+    @ExcelColumn(name="鍥炬爣鍦板潃")
+    private Integer icon;
+
+    @ApiModelProperty(value = "椤甸潰鍦板潃锛�/default.html 琛ㄧず缃戠珯榛樿閰嶇疆椤癸級")
+    @ExcelColumn(name="椤甸潰鍦板潃锛�/default.html 琛ㄧず缃戠珯榛樿閰嶇疆椤癸級")
+    private String pageAddr;
+    @ApiModelProperty(value = "鍥剧墖alt灞炴�у��")
+    @ExcelColumn(name="鍥剧墖alt灞炴�у��")
+    private String imgalt;
+
+    @ApiModelProperty(value = "鎺掑簭鐮�", example = "1")
+    @ExcelColumn(name="鎺掑簭鐮�")
+    private Integer sortnum;
+
+    @ApiModelProperty(value = "鏇存柊浜虹紪鐮�", example = "1")
+    @TableField(exist = false)
+    private String editorName;
+
+}
diff --git a/server/src/main/java/com/doumee/service/business/SeoService.java b/server/src/main/java/com/doumee/service/business/SeoService.java
new file mode 100644
index 0000000..bae1b16
--- /dev/null
+++ b/server/src/main/java/com/doumee/service/business/SeoService.java
@@ -0,0 +1,97 @@
+package com.doumee.service.business;
+
+import com.doumee.core.model.PageData;
+import com.doumee.core.model.PageWrap;
+import com.doumee.dao.business.model.Seo;
+import java.util.List;
+
+/**
+ * 缃戠珯SEO閰嶇疆Service瀹氫箟
+ * @author 姹熻箘韫�
+ * @since 2025/07/02 11:35
+ */
+public interface SeoService {
+
+    /**
+     * 鍒涘缓
+     * 
+     * @param seo 瀹炰綋瀵硅薄
+     * @return Integer
+     */
+    Integer create(Seo seo);
+
+    /**
+     * 涓婚敭鍒犻櫎
+     *
+     * @param id 涓婚敭
+     */
+    void deleteById(Integer id);
+
+    /**
+     * 鍒犻櫎
+     *
+     * @param seo 瀹炰綋瀵硅薄
+     */
+    void delete(Seo seo);
+
+    /**
+     * 鎵归噺涓婚敭鍒犻櫎
+     *
+     * @param ids 涓婚敭闆�
+     */
+    void deleteByIdInBatch(List<Integer> ids);
+
+    /**
+     * 涓婚敭鏇存柊
+     *
+     * @param seo 瀹炰綋瀵硅薄
+     */
+    void updateById(Seo seo);
+
+    /**
+     * 鎵归噺涓婚敭鏇存柊
+     *
+     * @param seos 瀹炰綋闆�
+     */
+    void updateByIdInBatch(List<Seo> seos);
+
+    /**
+     * 涓婚敭鏌ヨ
+     *
+     * @param id 涓婚敭
+     * @return Seo
+     */
+    Seo findById(Integer id);
+
+    /**
+     * 鏉′欢鏌ヨ鍗曟潯璁板綍
+     *
+     * @param seo 瀹炰綋瀵硅薄
+     * @return Seo
+     */
+    Seo findOne(Seo seo);
+
+    /**
+     * 鏉′欢鏌ヨ
+     *
+     * @param seo 瀹炰綋瀵硅薄
+     * @return List<Seo>
+     */
+    List<Seo> findList(Seo seo);
+  
+    /**
+     * 鍒嗛〉鏌ヨ
+     *
+     * @param pageWrap 鍒嗛〉瀵硅薄
+     * @return PageData<Seo>
+     */
+    PageData<Seo> findPage(PageWrap<Seo> pageWrap);
+
+    /**
+     * 鏉′欢缁熻
+     *
+     * @param seo 瀹炰綋瀵硅薄
+     * @return long
+     */
+    long count(Seo seo);
+}
diff --git a/server/src/main/java/com/doumee/service/business/impl/CarouselServiceImpl.java b/server/src/main/java/com/doumee/service/business/impl/CarouselServiceImpl.java
index d47455c..9c7f0a0 100644
--- a/server/src/main/java/com/doumee/service/business/impl/CarouselServiceImpl.java
+++ b/server/src/main/java/com/doumee/service/business/impl/CarouselServiceImpl.java
@@ -86,7 +86,9 @@
         if (CollectionUtils.isEmpty(ids)) {
             return;
         }
-        carouselMapper.deleteBatchIds(ids);
+        for(Integer id :ids){
+            this.deleteById(id);
+        }
     }
 
     @Override
diff --git a/server/src/main/java/com/doumee/service/business/impl/InformationServiceImpl.java b/server/src/main/java/com/doumee/service/business/impl/InformationServiceImpl.java
index b4ff068..3961dbe 100644
--- a/server/src/main/java/com/doumee/service/business/impl/InformationServiceImpl.java
+++ b/server/src/main/java/com/doumee/service/business/impl/InformationServiceImpl.java
@@ -77,7 +77,9 @@
         if (CollectionUtils.isEmpty(ids)) {
             return;
         }
-        informationMapper.deleteBatchIds(ids);
+        for (Integer id : ids) {
+            this.deleteById(id);
+        }
     }
 
     @Override
@@ -140,6 +142,7 @@
         queryWrapper.lambda()
                 .eq(Information::getIsdeleted,Constants.ZERO)
                 .eq(pageWrap.getModel().getId() != null, Information::getId, pageWrap.getModel().getId())
+                .eq(pageWrap.getModel().getModule() != null, Information::getModule, pageWrap.getModel().getModule())
                 .eq(pageWrap.getModel().getType() != null, Information::getType, pageWrap.getModel().getType())
                 .eq(pageWrap.getModel().getStatus() != null, Information::getStatus, pageWrap.getModel().getStatus())
                 .eq(pageWrap.getModel().getRemark() != null, Information::getRemark, pageWrap.getModel().getRemark())
diff --git a/server/src/main/java/com/doumee/service/business/impl/SeoServiceImpl.java b/server/src/main/java/com/doumee/service/business/impl/SeoServiceImpl.java
new file mode 100644
index 0000000..4a10c23
--- /dev/null
+++ b/server/src/main/java/com/doumee/service/business/impl/SeoServiceImpl.java
@@ -0,0 +1,169 @@
+package com.doumee.service.business.impl;
+
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.doumee.core.constants.Constants;
+import com.doumee.core.constants.ResponseStatus;
+import com.doumee.core.exception.BusinessException;
+import com.doumee.core.model.LoginUserInfo;
+import com.doumee.core.model.PageData;
+import com.doumee.core.model.PageWrap;
+import com.doumee.core.utils.Utils;
+import com.doumee.dao.business.SeoMapper;
+import com.doumee.dao.business.model.Carousel;
+import com.doumee.dao.business.model.Seo;
+import com.doumee.dao.system.model.SystemUser;
+import com.doumee.service.business.SeoService;
+import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
+import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper;
+import com.baomidou.mybatisplus.core.metadata.IPage;
+import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
+import com.github.yulichang.query.MPJQueryWrapper;
+import com.github.yulichang.wrapper.MPJLambdaWrapper;
+import org.apache.commons.lang3.StringUtils;
+import org.apache.shiro.SecurityUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+import org.springframework.util.CollectionUtils;
+
+import java.util.Date;
+import java.util.List;
+import java.util.Objects;
+
+/**
+ * 缃戠珯SEO閰嶇疆Service瀹炵幇
+ * @author 姹熻箘韫�
+ * @since 2025/07/02 11:35
+ */
+@Service
+public class SeoServiceImpl implements SeoService {
+
+    @Autowired
+    private SeoMapper seoMapper;
+
+    @Override
+    public Integer create(Seo seo) {
+        LoginUserInfo user = (LoginUserInfo) SecurityUtils.getSubject().getPrincipal();
+        if(Objects.isNull(seo)
+                || Objects.isNull(seo.getPageAddr())
+                || Objects.isNull(seo.getName())
+        ){
+            throw new BusinessException(ResponseStatus.BAD_REQUEST);
+        }
+        if(seoMapper.selectCount(new LambdaQueryWrapper<Seo>().eq(Seo::getIsdeleted,Constants.ZERO)
+                .eq(Seo::getPageAddr,seo.getPageAddr())) >0 ){
+            throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(),"璇ラ〉闈㈠湴鍧�宸查厤缃紝璇峰嬁閲嶅閰嶇疆锛�");
+        }
+        seo.setIsdeleted(Constants.ZERO);
+        seo.setCreateDate(new Date());
+        seo.setCreator(user.getId());
+        seo.setEditDate(new Date());
+        seo.setEditor(user.getId());
+        seoMapper.insert(seo);
+        return seo.getId();
+    }
+
+    @Override
+    public void deleteById(Integer id) {
+        LoginUserInfo user = (LoginUserInfo) SecurityUtils.getSubject().getPrincipal();
+        seoMapper.update(new UpdateWrapper<Seo>().lambda()
+                .set(Seo::getEditDate,new Date())
+                .set(Seo::getEditor,user.getId())
+                .set(Seo::getIsdeleted,Constants.ONE)
+                .eq(Seo::getId,id));
+    }
+
+    @Override
+    public void delete(Seo seo) {
+        UpdateWrapper<Seo> deleteWrapper = new UpdateWrapper<>(seo);
+        seoMapper.delete(deleteWrapper);
+    }
+
+    @Override
+    public void deleteByIdInBatch(List<Integer> ids) {
+        if (CollectionUtils.isEmpty(ids)) {
+            return;
+        }
+        for(Integer id :ids){
+            this.deleteById(id);
+        }
+    }
+
+    @Override
+    public void updateById(Seo seo) {
+        LoginUserInfo user = (LoginUserInfo) SecurityUtils.getSubject().getPrincipal();
+        if(StringUtils.isNotBlank(seo.getPageAddr()) &&
+                seoMapper.selectCount(new LambdaQueryWrapper<Seo>().eq(Seo::getIsdeleted,Constants.ZERO)
+                .eq(Seo::getPageAddr,seo.getPageAddr()).ne(Seo::getId,seo.getId())) >0 ){
+            throw new BusinessException(ResponseStatus.BAD_REQUEST.getCode(),"璇ラ〉闈㈠湴鍧�宸查厤缃紝璇峰嬁閲嶅閰嶇疆锛�");
+        }
+        seo.setIsdeleted(Constants.ZERO);
+        seo.setEditDate(new Date());
+        seo.setEditor(user.getId());
+        seoMapper.updateById(seo);
+    }
+
+    @Override
+    public void updateByIdInBatch(List<Seo> seos) {
+        if (CollectionUtils.isEmpty(seos)) {
+            return;
+        }
+        for (Seo seo: seos) {
+            this.updateById(seo);
+        }
+    }
+
+    @Override
+    public Seo findById(Integer id) {
+        return seoMapper.selectById(id);
+    }
+
+    @Override
+    public Seo findOne(Seo seo) {
+        QueryWrapper<Seo> wrapper = new QueryWrapper<>(seo);
+        return seoMapper.selectOne(wrapper);
+    }
+
+    @Override
+    public List<Seo> findList(Seo seo) {
+        seo.setIsdeleted(Constants.ZERO);
+        QueryWrapper<Seo> wrapper = new QueryWrapper<>(seo);
+        return seoMapper.selectList(wrapper);
+    }
+  
+    @Override
+    public PageData<Seo> findPage(PageWrap<Seo> pageWrap) {
+        IPage<Seo> page = new Page<>(pageWrap.getPage(), pageWrap.getCapacity());
+        MPJLambdaWrapper<Seo> queryWrapper = new MPJLambdaWrapper<Seo>();
+        Utils.MP.blankToNull(pageWrap.getModel());
+        pageWrap.getModel().setIsdeleted(Constants.ZERO);
+        queryWrapper.selectAll(Seo.class )
+                .selectAs(SystemUser::getUsername,Seo::getEditorName)
+                .leftJoin(SystemUser.class,SystemUser::getId,Seo::getEditor)
+                .eq(pageWrap.getModel().getId() != null, Seo::getId, pageWrap.getModel().getId())
+                .eq(pageWrap.getModel().getCreator() != null, Seo::getCreator, pageWrap.getModel().getCreator())
+                .ge(pageWrap.getModel().getCreateDate() != null, Seo::getCreateDate, Utils.Date.getStart(pageWrap.getModel().getCreateDate()))
+                .le(pageWrap.getModel().getCreateDate() != null, Seo::getCreateDate, Utils.Date.getEnd(pageWrap.getModel().getCreateDate()))
+                .eq(pageWrap.getModel().getEditor() != null, Seo::getEditor, pageWrap.getModel().getEditor())
+                .ge(pageWrap.getModel().getEditDate() != null, Seo::getEditDate, Utils.Date.getStart(pageWrap.getModel().getEditDate()))
+                .le(pageWrap.getModel().getEditDate() != null, Seo::getEditDate, Utils.Date.getEnd(pageWrap.getModel().getEditDate()))
+                .eq(pageWrap.getModel().getIsdeleted() != null, Seo::getIsdeleted, pageWrap.getModel().getIsdeleted())
+                .eq(pageWrap.getModel().getRemark() != null, Seo::getRemark, pageWrap.getModel().getRemark())
+                .like(pageWrap.getModel().getName() != null, Seo::getName, pageWrap.getModel().getName())
+                .like(pageWrap.getModel().getTitle() != null, Seo::getTitle, pageWrap.getModel().getTitle())
+                .like(pageWrap.getModel().getKeywords() != null, Seo::getKeywords, pageWrap.getModel().getKeywords())
+                .like(pageWrap.getModel().getAuthor() != null, Seo::getAuthor, pageWrap.getModel().getAuthor())
+                .eq(pageWrap.getModel().getIcon() != null, Seo::getIcon, pageWrap.getModel().getIcon())
+                .like(pageWrap.getModel().getImgalt() != null, Seo::getImgalt, pageWrap.getModel().getImgalt())
+                .like(pageWrap.getModel().getPageAddr() != null, Seo::getPageAddr, pageWrap.getModel().getPageAddr())
+                .eq(pageWrap.getModel().getSortnum() != null, Seo::getSortnum, pageWrap.getModel().getSortnum())
+                  .orderByAsc(Seo::getSortnum);
+
+        return PageData.from(seoMapper.selectPage(page, queryWrapper));
+    }
+
+    @Override
+    public long count(Seo seo) {
+        QueryWrapper<Seo> wrapper = new QueryWrapper<>(seo);
+        return seoMapper.selectCount(wrapper);
+    }
+}

--
Gitblit v1.9.3