前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >猿实战06——不一样的地址管理

猿实战06——不一样的地址管理

作者头像
山旮旯的胖子
发布2020-09-06 19:31:21
6340
发布2020-09-06 19:31:21
举报
文章被收录于专栏:猿人工厂

上一章节,猿人君教会了你一个新鲜的东西——猿实战05——手把手教你拥有自己的代码生成器。大家掌握原理,知道怎么去抽象你的代码就好了,莫要过于纠结。今天我们来学习新的东西——地址管理。

也许你会好奇,电商系统里怎么会有地址管理这个概念呢?嗯,地址其实是电商系统的通用数据,网购过的朋友都知道,下单的时候一定会叫你填写收货地址。地址数据在此时尤为重要,在构建地址数据库时,尽量的让它标准化统一化,方便整个系统的使用,我们先看看,要做的功能。

数据库设计

其实一个用于运营的电商系统中,地址的数据是相当庞大而且繁复的。你在填写用户地址的时,往往要求填写的是四级地址而非三级地址。为什么会是四级?国家地址库一般不是三级吗?这个四级地址是哪里来的呢?

嗯,这个四级地址其实是很宝贵的资源。都是配送人员在实际的工作中收集下来,最后经过筛选,合并,形成一套标准地址。这些地址,往往表达的是某某街道,某某片区,某村几社,这个级别了,这是长期工作的积累,到目前为止,数据量其实是在3亿条以上,各个公司的数据都会不同。猿人君也没有办法搞到4级地址,所以我们在功能上只做了三级。

不过考虑到未来的扩展,那么四级地址的维护任务会比较繁重,所以在设计上和传统的设计会有一些区别。相信大家见过太多的地址表,往往是父子结构的,一张表搞定。不过我们为了未来的四级地址考虑,还是拆开来设计,分别维护,省、市、区、三级地址,以后要做四级地址时,再扩展一张表(考虑数据量够吗?)就好了。

代码生成初体验

既然我们已经自己写过代码生成器了,准备牛刀小试一下吧。三张表,要是自己手写,还是比较麻烦的。不过今天也就是体验下昨天的成果,对于新手同学而言,仅此一次吧,以后还是老老实实去码,太早用这类东西,对你的发展不是很好。

我们打开我们编写的代码生成器,然后开始做修改一些基本的配置,如下图:

打开CgTest类,三张表,我们定义好表名,以及PoJo名就好了,如下图。

然后运行程序,代码就好了。

我们将生成的代码copy到对应目录,并完成相应的配置。

Controller

考虑到每个页面的功能都比较类似,都是新增,修改以及列表的功能,不过数据还是分开维护的,我们编写三个Controller,去应对这些功能就可以了。

代码语言:javascript
复制
 
/**
 * Copyright(c) 2004-2020 pangzi
 * com.pz.basic.mall.controller.sys.MallProvinceController.java
 */
package com.pz.basic.mall.controller.sys;
 
import com.pz.basic.mall.domain.base.Result;
import com.pz.basic.mall.domain.sys.MallProvince;
import com.pz.basic.mall.domain.sys.query.QueryMallProvince;
import com.pz.basic.mall.service.sys.MallProvinceService;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
import java.util.List;
 
 
/**
 *
 * @author pangzi
 * @date 2020-06-22 20:47:27
 *
 *
 */
@RestController
@RequestMapping("/provinceManage")
public class MallProvinceController {
 


    private MallProvinceService mallProvinceService;
 


public void setMallProvinceService(MallProvinceService mallProvinceService) {
this.mallProvinceService = mallProvinceService;
}
 
    /**
     * 新增省
     * @param mallProvince
     * @return
     */
    @RequestMapping("/addMallProvince")
    public Result<MallProvince>  addMallProvince(@RequestBody MallProvince mallProvince){
        try{
 
            return   mallProvinceService.addMallProvince(mallProvince);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
 
 
    /**
     * 修改省
     * @param mallProvince
     * @return
     */
    @RequestMapping("/updateMallProvince")
    public Result updateMallProvince(@RequestBody MallProvince mallProvince){
        try{
            return  mallProvinceService.updateMallProvinceById(mallProvince);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
 
    /**
     * 分页返回省列表
     * @param queryMallProvince
     * @return
     */
    @RequestMapping("/findByPage")
    public  Result<List<MallProvince>> findByPage(@RequestBody  QueryMallProvince queryMallProvince){
        return mallProvinceService.getMallProvincesByPage(queryMallProvince);
    }


}

代码语言:javascript
复制

/**
 * Copyright(c) 2004-2020 pangzi
 * com.pz.basic.mall.controller.sys.MallCityController.java
 */
package com.pz.basic.mall.controller.sys;
 
import com.pz.basic.mall.domain.base.Result;
import com.pz.basic.mall.domain.sys.MallCity;
import com.pz.basic.mall.domain.sys.query.QueryMallCity;
import com.pz.basic.mall.service.sys.MallCityService;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
import java.util.List;
 
 
/**
 *
 * @author pangzi
 * @date 2020-06-22 20:47:27
 *
 *
 */
@RestController
@RequestMapping("/cityManage")
public class MallCityController {
 
 
    private MallCityService mallCityService;
 
    public void setMallCityService(MallCityService mallCityService) {
        this.mallCityService = mallCityService;
    }
 
    /**
     * 新增城市
     * @param mallCity
     * @return
     */
    @RequestMapping("/addMallCity")
    public Result<MallCity>  addMallCity(@RequestBody MallCity mallCity){
        try{
            return mallCityService.addMallCity(mallCity);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
    /**
     * 修改城市
     * @param area
     * @return
     */
    @RequestMapping("/updateMallCity")
    public Result updateMallCity(@RequestBody MallCity area){
        try{
            return  mallCityService.updateMallCityById(area);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
 
    /**
     * 分页返回城市列表
     * @param queryMallCity
     * @return
     */
    @RequestMapping("/findByPage")
    public  Result<List<MallCity>> findByPage(@RequestBody QueryMallCity queryMallCity){
        return mallCityService.getMallCitysByPage(queryMallCity);
    }
 
}

代码语言:javascript
复制
/**
 * Copyright(c) 2004-2020 pangzi
 * com.pz.basic.mall.controller.sys.MallAreaController.java
 */
package com.pz.basic.mall.controller.sys;
 
import com.pz.basic.mall.domain.base.Result;
import com.pz.basic.mall.domain.sys.MallArea;
import com.pz.basic.mall.domain.sys.query.QueryMallArea;
import com.pz.basic.mall.service.sys.MallAreaService;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
import java.util.List;
 
 
/**
 *
 * @author pangzi
 * @date 2020-06-22 20:47:27
 *
 *
 */
@RestController
@RequestMapping("/areaManage")
public class MallAreaController {
 


    private MallAreaService mallAreaService;
 
public void setMallAreaService(MallAreaService mallAreaService) {
this.mallAreaService = mallAreaService;
}
 
    /**
     * 新增地区
     * @param area
     * @return
     */
    @RequestMapping("/addMallArea")
    public Result<MallArea>  addMallArea(@RequestBody MallArea area){
        try{
            return mallAreaService.addMallArea(area);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
 
    /**
     * 修改地区
     * @param area
     * @return
     */
    @RequestMapping("/updateMallArea")
    public Result updateMallArea(@RequestBody MallArea area){
        try{
          return  mallAreaService.updateMallAreaById(area);
        }catch(Exception e){
            e.printStackTrace();
            return new Result(false);
        }
    }
 
 
    /**
     * 分页返回地区列表
     * @param queryMallArea
     * @return
     */
    @RequestMapping("/findByPage")
    public  Result<List<MallArea>> findByPage(@RequestBody QueryMallArea queryMallArea){
        return mallAreaService.getMallAreasByPage(queryMallArea);
    }
 
 
 
 
}

前端页面

考虑到在每个页面点击查看下级按钮,将进入到每一个地址的下级地址维护页面,你暂时也没有使用过自定义组件,在这里,我们可以考虑使用一个view来处理,然后在这个view中,使用自定义的三个组件(分别是省、市、区地址维护),代码如下。

代码语言:javascript
复制

<template>
  <div id="addressBaseDiv">
    <div v-if="provinceShow">
      <provinceSearch ref="provinceSearch" @lookSubordinate="lookOneSubordinate" />
    </div>
    <div v-if="cityShow">
      <citySearch ref="citySearch" :pid="provinceId" @lookSubordinate="lookTwoSubordinate" @returnBack="returnTwoBack" />
    </div>
    <div v-if="areaShow">
      <areaSearch ref="areaThreeSearch" :pid="cityId" @returnBack="returnThreeBack" />
    </div>
  </div>
</template>
 
<script>
import provinceSearch from '@/components/basedataManage/province'
import citySearch from '@/components/basedataManage/city'
import areaSearch from '@/components/basedataManage/area'
 
export default {
  components: {
    provinceSearch,
    citySearch,
    areaSearch
  },
  data() {
    return {
      // 一级类目
      provinceShow: false,
      // 二级类目
      cityShow: false,
      // 三级类目
      areaShow: false,
      provinceId: '',
      cityId: ''
    }
  },
  created() {
    // 显示一级地址
    this.provinceShow = true
  },
  methods: {
    // 二级回退
    returnTwoBack() {
      // 一级二级三级类目显示设置
      this.provinceShow = true
      this.cityShow = false
      this.areaShow = false
    },
    // 三级回退
    returnThreeBack() {
      // 一级二级三级类目显示设置
      this.provinceShow = false
      this.cityShow = true
      this.areaShow = false
    },
    // 一级查看下级类目
    lookOneSubordinate(row) {
      this.provinceId = row.provinceId
      console.log(row)
      // 一级二级三级类目显示设置
      this.provinceShow = false
      this.cityShow = true
      this.areaShow = false
    },
    // 二级查看下级类目
    lookTwoSubordinate(row) {
      this.cityId = row.cityId
      console.log(row)
      // 一级二级三级类目显示设置
      this.provinceShow = false
      this.cityShow = false
      this.areaShow = true
    }
  }
 
}
</script>
 
<style scoped>
</style>

我们看看组件的使用。

你需要注意的是,在查看二级地址,三级地址时,将上级地址的id传入下级组件。

自定义组件

关于自定义组件的使用,我们着重讲二级地址的使用,因为省、市、区的维护,从功能上讲,都差不多,而二级地址,有返回上级和查看下级的功能比较有代表性。

代码语言:javascript
复制

<template>
  <div id="citySearchDiv">
    <div style="float:right">
      <span @click="returnBack">< <span style="font-size:15px;margin-top:50px;line-height: 30px;">返回上一级</span></span>
      <el-button type="primary" icon="el-icon-edit" style="margin-bottom:20px;float: left;margin-right: 40px;" @click="addDate()">新增二级地址</el-button>
    </div>
    <div>
      <el-table
        ref="table"
        v-loading="listLoading"
        :data="list"
        style="width: 100%"
        border
      >
        <el-table-column label="二级地址ID">
          <template slot-scope="scope">{{ scope.row.cityId }}</template>
        </el-table-column>
        <el-table-column label="二级地址名">
          <template slot-scope="scope">{{ scope.row.cityName }}</template>
        </el-table-column>
        <el-table-column label="上级地址ID">
          <template slot-scope="scope">{{ scope.row.provinceId }}</template>
        </el-table-column>
        <el-table-column label="地址级别">
          二级地址
        </el-table-column>
        <el-table-column label="操作" width="200">
          <template slot-scope="scope">
            <el-button
              type="primary"
              size="mini"
              @click="handleSubordinate(scope.row)"
            >查看下级
            </el-button>
            <el-button
              type="primary"
              size="mini"
              @click="handleUpdate(scope.row)"
            >修改
            </el-button>
          </template>
        </el-table-column>
      </el-table>
    </div>
    <pagination v-show="total>0" :total="total" :page.sync="listQuery.page" :limit.sync="listQuery.pageSize" @pagination="getList" />
    <!-- 新增/编辑弹框 -->
    <el-dialog :title="textMap[dialogStatus]" :visible.sync="dialogFormVisible">
      <el-form ref="dataForm" :rules="rules" :model="temp" label-position="right" label-width="120px" style="width: 320px; margin-left:50px;">
        <!--注意prop 属性为需要验证的属性 -->
        <el-form-item label="一级地址编码:" prop="provinceId">
          <el-input v-model="temp.provinceId" :value="temp.provinceId" placeholder="请输入一级地址编码" :readonly="true" />
        </el-form-item>
        <el-form-item label="二级地址编码:" prop="cityId">
          <el-input v-model="temp.cityId" placeholder="请输入二级地址编码" />
        </el-form-item>
        <el-form-item label="二级地址名称:" prop="cityName">
          <el-input v-model="temp.cityName" placeholder="请输入二级地址名" />
        </el-form-item>
      </el-form>
      <div slot="footer">
        <el-button @click="dialogFormVisible = false">
          取消
        </el-button>
        <el-button type="primary" @click="dialogStatus==='create'?createData():updateData()">
          确定
        </el-button>
      </div>
    </el-dialog>
  </div>
</template>
 
<script>
import Pagination from '@/components/Pagination' // secondary package based on el-pagination
import { fetchCityList, createCity, updateCity } from '@/api/basedataManage/basedataManage'
export default {
  components: { Pagination },
  props: ['pid'],
  data() {
    return {
      dialogStatus: '',
      // 弹框是否显示
      dialogFormVisible: false,
      // 弹框校验规则
      rules: {
        cityName: [{ required: true, message: '二级地址名称必须填写', trigger: 'change' }],
        cityId: [{ required: true, message: '二级地址编码必须填写', trigger: 'change' }]
      },
      temp: {
        id: undefined,
        // 一级地址名称:
        cityName: '',
        cityId: ''
      },
      // 状态
      valueList: [{
        value: '是',
        label: '是'
      }, {
        value: '否',
        label: '否'
      }],
      textMap: {
        update: '二级地址修改',
        create: '二级地址新增'
      },
      // table集合
      list: null,
      multipleSelection: [],
      // 分页
      total: 0,
      // loading
      listLoading: true,
      listQuery: {
        page: 1,
        pageSize: 10,
        provinceId: this.pid
      }
    }
  },
  watch: {
    pid(value, old) {
      if (value) {
        console.log(value)
        this.temp.provinceId = this.pid
      }
    }
  },
  created() {
    // 列表查询
    this.getList(this.listQuery)
    this.temp.provinceId = this.pid
  },
  methods: {
    /**
     * 回退
     */
    returnBack() {
      this.$emit('returnBack')
    },
    // 查看下级
    handleSubordinate(row) {
      this.$emit('lookSubordinate', row)
    },
    // 编辑
    handleUpdate(row) {
      this.temp = Object.assign({}, row) // copy obj
      this.dialogStatus = 'update'
      this.dialogFormVisible = true
      this.$nextTick(() => {
        this.$refs['dataForm'].clearValidate()
      })
    },
    // 重置
    resetTemp() {
      this.temp = {
        id: undefined,
        // 一级地址名称:
        cityName: '',
        cityId: '',
        provinceId: this.pid
      }
    },
    // 新增一级类目
    addDate() {
      this.resetTemp()
      this.dialogStatus = 'create'
      this.dialogFormVisible = true
      this.$nextTick(() => {
        this.$refs['dataForm'].clearValidate()
      })
    },
    // 更新保存方法
    updateData() {
      this.$refs['dataForm'].validate((valid) => {
        if (valid) {
          const tempData = Object.assign({}, this.temp)
          updateCity(tempData).then(() => {
            const index = this.list.findIndex(v => v.id === this.temp.id)
            this.list.splice(index, 1, this.temp)
            this.dialogFormVisible = false
            this.$notify({
              title: 'Success',
              message: 'Update Successfully',
              type: 'success',
              duration: 2000
            })
          })
        }
      })
    },
    // 创建保存方法
    createData() {
      this.$refs['dataForm'].validate((valid) => {
        console.log(valid)
        if (valid) {
          createCity(this.temp).then((res) => {
            this.temp.id = res.model.id
            this.list.unshift(this.temp)
            this.dialogFormVisible = false
            this.$notify({
              title: 'Success',
              message: 'Created Successfully',
              type: 'success',
              duration: 2000
            })
          })
        }
      })
    },
    // 列表查询
    getList() {
      this.listLoading = true
      fetchCityList(this.listQuery).then(response => {
        this.list = response.model
        this.total = response.totalItem
 
        // Just to simulate the time of the request
        setTimeout(() => {
          this.listLoading = false
        }, 1.5 * 1000)
      })
    }
  }
}
</script>
 
<style scoped>
 
</style>

你需要特别注意的是,上级参数的接收。

由于二级地址维护的是上一级地址下的二级地址,你在做查询时,需要带上上一级地址的id,接收父组件的参数后,在查询时使用(this.变量名)。

回退功能的实现,你也需要注意下,this.$emit的使用.

子组件直接通过this.$emit("自定义事件"),然后父组件在组件中添加@自定义事件=“event”。 this.$emit('returnBack')触发了,父组上绑定的returnBack事件,从而调用了绑定的returnTwoBack函数,完成回退功能。

至于API的封装,以及新增修改功能的实现,大家请自行参考,品牌管理的实现。不动手自己搞点儿什么东西,是始终学不会的。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-08-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 猿人工厂 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档