Vue组件(33)表单控件 + 查询控件 + 数据列表控件 实现增删改查(中)

调整文件结构

一开始把表单控件、查询控件、列表控件等都放在了一个页面里面,然后发现代码有点乱,也不便于更换,只是数据交互方面方便了一点。

于是分成了多个组件的形式如下:

文件结构
  • index
    这个是左侧的菜单导航,调用子路由加载对应的组件

  • data-list
    数据列表组件,加上查询控件、操作按钮、分页、表单等控件。
    因为数据列表是核心,添加数据后要更新列表,查询后要更新列表、分页后也要更新列表,所以就把各个控件直接放在列表组件里面了。

<template>
  <!--查询功能-->
  <data-find/>
  <!--操作按钮-->
  <div style="background:#fff;">
    <el-space wrap>
      <el-button type="primary" @click="addNew"> 添 加 </el-button>
      <el-button type="primary" @click="update"> 修 改 </el-button>
      <el-button type="primary" @click="deleteData"> 删 除 </el-button>
      <el-button type="primary" @click="loadList"> 更 新 </el-button>
    </el-space>
  </div>
  <!--数据列表-->
  <nf-grid-list
    v-bind="meta"
    :data-list="dataList"
  />
  <!--分页功能-->
  <data-page/>
  <!--弹窗的表单-->
  {{formState}}
  <data-edit :moduleId="moduleId" >
  </data-edit>
</template>

操作按钮也可以封装成控件,现在还没忙过了封装。
这样结构是不是很清晰了,而且便于位置调整。
也可以做成不同风格的组件,然后进行替换。

  • data-edit
    实现数据的添加、修改和查看的功能,主要放置表单控件,加载表单控件需要的meta,协调数据等功能。
<template>
  <!--弹窗的表单-->
  <el-dialog
    :title="formState.editState + '客户信息'"
    :modal="true"
    v-model="formState.isOpen"
  >
    <nf-form
      ref="formControl"
      v-model="formModel"
      v-model:partModel="partModel"
      :meta="meta"
    >
    </nf-form>
    <el-button type="primary" @click="addNew"> 确 定 </el-button>
  </el-dialog>
</template>

用弹窗来实现表单,el-dialog 的用法还没研究透。

  • data-find
    实现查询功能,放置查询控件,加载查询控件需要的meta,提供查询条件。
<template>
  <nf-find
    ref="findControl"
    v-model="query"
    :meta="meta"
  />
</template>

放上查询控件。

  • data-page
    实现分页功能,这个没有封装,直接使用 el-pagination 来实现。

  • crud-manage.js
    存放共享数据和状态管理。


import { provide, inject, reactive } from 'vue'

// 管理类
export default function crudManage () {
  /**
   * 表单的状态
   */
  const formState = reactive({
    isOpen: false,
    editState: 'add' // add/update/show
  })

  const formStates = reactive({})

  /**
   * 数据源
   */
  const dataSource = reactive([])

  // 注入状态
  const register = (id) => {
    provide('dataSource', dataSource)
    provide('formState', formState)

    if (typeof formStates[id] === 'undefined') {
      formStates[id] = {
        id: id,
        isOpen: false,
        editState: 'add' // add/update/show
      }
      provide('formState_' + id, formStates[id])
    }
  }

  /**
   * 获取弹窗表单的状态
   * @returns 返回弹窗表单的状态
   */
  const getFormState = (id) => {
    if (typeof id === 'undefined') {
      if (typeof inject('formState') === 'undefined') {
        return formState
      } else {
        return inject('formState')
      }
    } else {
      if (typeof inject('formState_' + id) === 'undefined') {
        return formState
      } else {
        return inject('formState_' + id)
      }
    }
  }

  /**
   * 表单添加、修改数据,列表获取数据
   * @returns 返回共用的数据源
   */
  const getDataSource = () => {
    if (typeof inject('dataSource') === 'undefined') {
      return dataSource
    } else {
      return inject('dataSource')
    }
  }

  /**
   * 表单里面添加新数据
   */
  const formAddNewData = (obj) => {
    let arr = []
    if (typeof inject('dataSource') === 'undefined') {
      arr = dataSource
    } else {
      arr = inject('dataSource')
    }
    console.log('inject--', inject)
    console.log('dataSource', arr)
    arr.push(obj)
  }

  return {
    formState, // 弹窗表单的状态
    getFormState,
    getDataSource,
    formAddNewData,
    register // 注入
  }
}

可以用Vuex实现,只是感觉有点大材小用,另外也是尝试一下注入的用法。
然后就发现个问题,在子组件里面,直接取 inject('formState') 能取出来,但是放在函数里面就取不出来了,也不知道是哪里写的有问题。所以只好在外面先取出来存放在ds里面,然后再在函数里面push。不知道差哪了。

    // 注入
    const {
      getFormState,
      getDataSource
    } = crudManage()

    const ds = getDataSource() // 这里可以获取

    // 添加新数据
    const addNew = () => {
      const newData = {
        id: new Date().valueOf()
      }
      Object.assign(newData, formModel)
      inject('formState')  // 在这里取不到,
      ds.unshift(newData)
    }
  • 为啥要分得这么细致呢?
    因为我喜欢原子级的组件,这样统一组件的接口风格后,然后每个组件里的代码就可以变得很少,便于维护和更换。再把共用代码提取出来,便于复用。

因为做项目,需要各种风格的页面结构,变得是页面,js代码几乎没啥变化,既然vue3提出来composition API,那么就别浪费了,拿来用呗。

本来今天想把数据都联动起来的,结果卡住了。后面是查询、分页和列表的联动。这个应该加上webSQL了。

添加的表单

满满的调试风格,控件类型还没有改。。。

数据列表

随便加的数据。

菜单

明明用的UI库,为啥还是不好看呢?没有艺术细胞真愁人。。。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容