內容管理系統前端實現

一 創建相關文件

1 api

2 頁面組件

3 路由配置

  // 內容管理
  {
    path: '/ad',
    component: Layout,
    redirect: '/ad/list',
    name: 'Ad',
    meta: { title: '內容管理' },
    children: [
      {
        path: 'list',
        name: 'AdList',
        component: () => import('@/views/ad/list'),
        meta: { title: '廣告推薦' }
      },
      {
        path: 'create',
        name: 'AdCreate',
        component: () => import('@/views/ad/form'),
        meta: { title: '添加廣告推薦' },
        hidden: true
      },
      {
        path: 'edit/:id',
        name: 'AdEdit',
        component: () => import('@/views/ad/form'),
        meta: { title: '編輯廣告推薦' },
        hidden: true
      },


      {
        path: 'type-list',
        name: 'AdTypeList',
        component: () => import('@/views/adType/list'),
        meta: { title: '推薦位' }
      },
      {
        path: 'type-create',
        name: 'AdTypeCreate',
        component: () => import('@/views/adType/form'),
        meta: { title: '添加推薦位' },
        hidden: true
      },
      {
        path: 'type-edit/:id',
        name: 'AdTypeEdit',
        component: () => import('@/views/adType/form'),
        meta: { title: '編輯推薦位' },
        hidden: true
      }
    ]
  },

二 AdType模塊

1 api

// @ 符號在build/webpack.base.conf.js 中配置 表示 'src' 路徑
import request from '@/utils/request'


export default {


  list() {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: '/admin/cms/ad-type/list',
      method: 'get'
    })
  },


  pageList(page, limit) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad-type/list/${page}/${limit}`,
      method: 'get'
    })
  },


  removeById(id) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad-type/remove/${id}`,
      method: 'delete'
    })
  },


  save(adType) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: '/admin/cms/ad-type/save',
      method: 'post',
      data: adType
    })
  },


  getById(id) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad-type/get/${id}`,
      method: 'get'
    })
  },


  updateById(adType) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: '/admin/cms/ad-type/update',
      method: 'put',
      data: adType
    })
  }
}

2 views/AdType/list.vue

<template>
  <div class="app-container">


    <!-- 工具按鈕 -->
    <div style="margin-bottom: 10px">
      <router-link to="/ad/type-create">
        <el-button type="primary" size="mini" icon="el-icon-edit">添加推薦位</el-button>
      </router-link>
    </div>


    <!-- 表格 -->
    <el-table :data="list" border stripe>


      <el-table-column
        label="#"
        width="50">
        <template slot-scope="scope">
          {
  
  { (page - 1) * limit + scope.$index + 1 }}
        </template>
      </el-table-column>
      <el-table-column prop="title" label="推薦位名稱" />


      <el-table-column label="操作" width="200" align="center">
        <template slot-scope="scope">
          <router-link :to="'/ad/type-edit/'+scope.row.id">
            <el-button type="primary" size="mini" icon="el-icon-edit">修改</el-button>
          </router-link>
          <el-button type="danger" size="mini" icon="el-icon-delete" @click="removeById(scope.row.id)">刪除</el-button>
        </template>
      </el-table-column>
    </el-table>


    <!-- 分頁組件 -->
    <el-pagination
      :current-page="page"
      :total="total"
      :page-size="limit"
      :page-sizes="[5, 10, 20, 30, 40, 50, 100]"
      style="padding: 30px 0; text-align: center;"
      layout="total, sizes, prev, pager, next, jumper"
      @size-change="changePageSize"
      @current-change="changeCurrentPage"
    />
  </div>
</template>


<script>
import adTypeApi from '@/api/adType'


export default {
  // 定義數據模型
  data() {
    return {
      list: [], // 列表
      total: 0, // 總記錄數
      page: 1, // 頁碼
      limit: 10 // 每頁記錄數
    }
  },


  // 頁面渲染成功後獲取數據
  created() {
    this.fetchData()
  },


  // 定義方法
  methods: {
    fetchData() {
      // 調用api
      adTypeApi.pageList(this.page, this.limit).then(response => {
        this.list = response.data.rows
        this.total = response.data.total
      })
    },


    // 每頁記錄數改變,size:回調參數,表示當前選中的“每頁條數”
    changePageSize(size) {
      this.limit = size
      this.fetchData()
    },


    // 改變頁碼,page:回調參數,表示當前選中的“頁碼”
    changeCurrentPage(page) {
      this.page = page
      this.fetchData()
    },


    // 根據id刪除數據
    removeById(id) {
      this.$confirm('此操作將永久刪除該記錄, 是否繼續?', '提示', {
        confirmButtonText: '確定',
        cancelButtonText: '取消',
        type: 'warning'
      }).then(() => {
        return adTypeApi.removeById(id)
      }).then((response) => {
        this.fetchData()
        this.$message.success(response.message)
      }).catch(error => {
        // 當取消時會進入catch語句:error = 'cancel'
        // 當後端服務拋出異常時:error = 'error'
        if (error === 'cancel') {
          this.$message.info('取消刪除')
        }
      })
    }
  }
}
</script>

3 views/AdType/form.vue

<template>
  <div class="app-container">
    <!-- 輸入表單 -->
    <el-form label-width="120px">
      <el-form-item label="推薦位名稱">
        <el-input v-model="adType.title" />
      </el-form-item>


      <el-form-item>
        <el-button :disabled="saveBtnDisabled" type="primary" @click="saveOrUpdate()">保存</el-button>
      </el-form-item>
    </el-form>
  </div>
</template>


<script>
import adTypeApi from '@/api/adType'


export default {
  data() {
    return {
      adType: {},
      saveBtnDisabled: false // 保存按鈕是否禁用,防止表單重複提交
    }
  },


  // 頁面渲染成功
  created() {
    if (this.$route.params.id) {
      this.fetchDataById(this.$route.params.id)
    }
  },


  methods: {


    saveOrUpdate() {
      // 禁用保存按鈕
      this.saveBtnDisabled = true
      if (!this.adType.id) {
        this.saveData()
      } else {
        this.updateData()
      }
    },


    // 新增
    saveData() {
      // debugger
      adTypeApi.save(this.adType).then(response => {
        this.$message.success(response.message)
        this.$router.push({ path: '/ad/type-list' })
      })
    },


    // 根據id更新記錄
    updateData() {
      // teacher數據的獲取
      adTypeApi.updateById(this.adType).then(response => {
        this.$message.success(response.message)
        this.$router.push({ path: '/ad/type-list' })
      })
    },


    // 根據id查詢記錄
    fetchDataById(id) {
      adTypeApi.getById(id).then(response => {
        this.adType = response.data.item
      })
    }
  }
}
</script>

三 Ad模塊

1 api

// @ 符號在build/webpack.base.conf.js 中配置 表示 'src' 路徑
import request from '@/utils/request'


export default {


  pageList(page, limit) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad/list/${page}/${limit}`,
      method: 'get'
    })
  },


  removeById(id) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad/remove/${id}`,
      method: 'delete'
    })
  },


  save(ad) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: '/admin/cms/ad/save',
      method: 'post',
      data: ad
    })
  },


  getById(id) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: `/admin/cms/ad/get/${id}`,
      method: 'get'
    })
  },


  updateById(ad) {
    return request({
      baseURL: 'http://127.0.0.1:8140',
      url: '/admin/cms/ad/update',
      method: 'put',
      data: ad
    })
  }
}

2 views/Ad/list.vue

<template>
  <div class="app-container">


    <!-- 工具按鈕 -->
    <div style="margin-bottom: 10px">
      <router-link to="/ad/create">
        <el-button type="primary" size="mini" icon="el-icon-edit">添加廣告推薦</el-button>
      </router-link>
    </div>


    <!-- 表格 -->
    <el-table :data="list" border stripe>


      <el-table-column
        label="#"
        width="50">
        <template slot-scope="scope">
          {
  
  { (page - 1) * limit + scope.$index + 1 }}
        </template>
      </el-table-column>


      <el-table-column prop="type" label="推薦位名稱" />
      <el-table-column prop="title" label="廣告名稱" />
      <el-table-column prop="sort" label="排序" />


      <el-table-column label="操作" width="200" align="center">
        <template slot-scope="scope">
          <router-link :to="'/ad/edit/'+scope.row.id">
            <el-button type="primary" size="mini" icon="el-icon-edit">修改</el-button>
          </router-link>
          <el-button type="danger" size="mini" icon="el-icon-delete" @click="removeById(scope.row.id)">刪除</el-button>
        </template>
      </el-table-column>
    </el-table>


    <!-- 分頁組件 -->
    <el-pagination
      :current-page="page"
      :total="total"
      :page-size="limit"
      :page-sizes="[5, 10, 20, 30, 40, 50, 100]"
      style="padding: 30px 0; text-align: center;"
      layout="total, sizes, prev, pager, next, jumper"
      @size-change="changePageSize"
      @current-change="changeCurrentPage"
    />
  </div>
</template>


<script>
import adApi from '@/api/ad'


export default {
  // 定義數據模型
  data() {
    return {
      list: [], // 列表
      total: 0, // 總記錄數
      page: 1, // 頁碼
      limit: 10 // 每頁記錄數
    }
  },


  // 頁面渲染成功後獲取數據
  created() {
    this.fetchData()
  },


  // 定義方法
  methods: {
    fetchData() {
      // 調用api
      adApi.pageList(this.page, this.limit).then(response => {
        this.list = response.data.rows
        this.total = response.data.total
      })
    },


    // 每頁記錄數改變,size:回調參數,表示當前選中的“每頁條數”
    changePageSize(size) {
      this.limit = size
      this.fetchData()
    },


    // 改變頁碼,page:回調參數,表示當前選中的“頁碼”
    changeCurrentPage(page) {
      this.page = page
      this.fetchData()
    },


    // 根據id刪除數據
    removeById(id) {
      this.$confirm('此操作將永久刪除該記錄, 是否繼續?', '提示', {
        confirmButtonText: '確定',
        cancelButtonText: '取消',
        type: 'warning'
      }).then(() => {
        return adApi.removeById(id)
      }).then((response) => {
        this.fetchData()
        this.$message.success(response.message)
      }).catch(error => {
        // 當取消時會進入catch語句:error = 'cancel'
        // 當後端服務拋出異常時:error = 'error'
        if (error === 'cancel') {
          this.$message.info('取消刪除')
        }
      })
    }
  }
}
</script>

3 views/Ad/form.vue

<template>
  <div class="app-container">
    <!-- 輸入表單 -->
    <el-form label-width="120px">
      <el-form-item label="廣告推薦名稱">
        <el-input v-model="ad.title" />
      </el-form-item>
      <!-- 推薦位 -->
      <el-form-item label="推薦位">
        <el-select
          v-model="ad.typeId"
          placeholder="請選擇">
          <el-option
            v-for="adType in adTypeList"
            :key="adType.id"
            :label="adType.title"
            :value="adType.id"/>
        </el-select>
      </el-form-item>
      <el-form-item label="排序">
        <el-input-number v-model="ad.sort" :min="0"/>
      </el-form-item>
      <el-form-item label="廣告圖片">
        <el-upload
          :on-success="handleAvatarSuccess"
          :on-error="handleAvatarError"
          :on-exceed="handleUploadExceed"
          :before-upload="beforeAvatarUpload"
          :limit="1"
          :file-list="fileList"
          action="http://localhost:8120/admin/oss/file/upload?module=ad"
          list-type="picture">
          <el-button size="small" type="primary">點擊上傳</el-button>
        </el-upload>
      </el-form-item>
      <el-form-item label="背景顏色">
        <el-color-picker v-model="ad.color"/>
      </el-form-item>
      <el-form-item label="鏈接地址">
        <el-input v-model="ad.linkUrl" />
      </el-form-item>
      <el-form-item>
        <el-button :disabled="saveBtnDisabled" type="primary" @click="saveOrUpdate()">保存</el-button>
      </el-form-item>
    </el-form>
  </div>
</template>


<script>
import adApi from '@/api/ad'
import adTypeApi from '@/api/adType'


export default {
  data() {
    return {
      ad: {
        sort: 0
      },
      fileList: [], // 上傳文件列表
      adTypeList: [],
      saveBtnDisabled: false // 保存按鈕是否禁用,防止表單重複提交
    }
  },


  // 頁面渲染成功
  created() {
    if (this.$route.params.id) {
      this.fetchDataById(this.$route.params.id)
    }


    // 獲取推薦位列表
    this.initAdTypeList()
  },


  methods: {


    initAdTypeList() {
      adTypeApi.list().then(response => {
        this.adTypeList = response.data.items
      })
    },


    saveOrUpdate() {
      // 禁用保存按鈕
      this.saveBtnDisabled = true
      if (!this.ad.id) {
        this.saveData()
      } else {
        this.updateData()
      }
    },


    // 新增
    saveData() {
      // debugger
      adApi.save(this.ad).then(response => {
        this.$message.success(response.message)
        this.$router.push({ path: '/ad/list' })
      })
    },


    // 根據id更新記錄
    updateData() {
      // teacher數據的獲取
      adApi.updateById(this.ad).then(response => {
        this.$message.success(response.message)
        this.$router.push({ path: '/ad/list' })
      })
    },


    // 根據id查詢記錄
    fetchDataById(id) {
      adApi.getById(id).then(response => {
        this.ad = response.data.item
        this.fileList = [{ 'url': this.ad.imageUrl }]
      })
    },


    // 上傳多於一個文件
    handleUploadExceed(files, fileList) {
      this.$message.warning('想要重新上傳圖片,請先刪除已上傳的視頻')
    },


    // 上傳校驗
    beforeAvatarUpload(file) {
      const isJPG = file.type === 'image/jpeg'
      const isLt2M = file.size / 1024 / 1024 < 2


      if (!isJPG) {
        this.$message.error('上傳頭像圖片只能是 JPG 格式!')
      }
      if (!isLt2M) {
        this.$message.error('上傳頭像圖片大小不能超過 2MB!')
      }
      return isJPG && isLt2M
    },


    // 上傳成功回調
    handleAvatarSuccess(res, file) {
      console.log(res)
      if (res.success) {
        // console.log(res)
        this.ad.imageUrl = res.data.url
        // 強制重新渲染
        // this.$forceUpdate()
      } else {
        this.$message.error('上傳失敗1')
      }
    },


    // 錯誤處理
    handleAvatarError() {
      console.log('error')
      this.$message.error('上傳失敗2')
    }
  }
}
</script>

四 測試

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章