Skip to content

useCrud 组合式函数

通用 CRUD 操作 Hook,封装了列表查询、分页、加载状态管理等常用功能。

基础用法

vue
<script setup>
import { useCrud } from '@/composables/useCrud'
import { userApi } from '@/services/modules/user'

const {
  loading,
  tableData,
  total,
  queryParams,
  fetchList,
  handleQuery,
  resetQuery
} = useCrud({
  api: userApi
})
</script>

参数

UseCrudOptions

参数类型默认值说明
apiCrudApi<T>-CRUD API 对象(必填)
defaultQueryPartial<QueryParams>{}默认查询参数
immediatebooleantrue是否立即执行查询

CrudApi 接口

typescript
interface CrudApi<T> {
  getList: (params: QueryParams) => Promise<PageResult<T>>
  save?: (data: Partial<T>) => Promise<any>
  delete?: (ids: string[]) => Promise<any>
  getById?: (id: string) => Promise<T>
}

返回值

状态

属性类型说明
loadingRef<boolean>加载状态
tableDataRef<T[]>表格数据
totalRef<number>总条数
queryParamsReactive<QueryParams>查询参数
selectedIdsRef<string[]>选中的行ID
hasSelectionComputedRef<boolean>是否有选中项

方法

方法名说明参数
fetchList获取列表数据-
handleQuery查询(重置到第一页)-
resetQuery重置查询-
handlePageChange处理分页变化page: number
handleSizeChange处理每页条数变化size: number
handleSelectionChange处理选择变化selection: T[]
handleDelete删除单条数据row: T, message?: string
handleBatchDelete批量删除message?: string
handleSave保存数据data: Partial<T>, isEdit?: boolean
fetchDetail获取单条详情id: string

完整示例

vue
<template>
  <div class="page-container">
    <!-- 查询表单 -->
    <el-form :model="queryParams" inline>
      <el-form-item label="用户名">
        <el-input v-model="queryParams.username" placeholder="请输入用户名" clearable />
      </el-form-item>
      <el-form-item label="状态">
        <DictSelect v-model="queryParams.status" dictType="sys_user_status" />
      </el-form-item>
      <el-form-item>
        <el-button type="primary" @click="handleQuery">查询</el-button>
        <el-button @click="resetQuery">重置</el-button>
      </el-form-item>
    </el-form>

    <!-- 操作按钮 -->
    <div class="toolbar">
      <el-button type="primary" @click="handleAdd">新增</el-button>
      <el-button type="danger" :disabled="!hasSelection" @click="handleBatchDelete">批量删除</el-button>
    </div>

    <!-- 数据表格 -->
    <el-table
      v-loading="loading"
      :data="tableData"
      @selection-change="handleSelectionChange"
    >
      <el-table-column type="selection" width="55" />
      <el-table-column prop="username" label="用户名" />
      <el-table-column prop="nickname" label="昵称" />
      <el-table-column prop="status" label="状态">
        <template #default="{ row }">
          <DictTag :value="row.status" dictType="sys_user_status" />
        </template>
      </el-table-column>
      <el-table-column label="操作" width="200">
        <template #default="{ row }">
          <el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
          <el-button type="danger" link @click="handleDelete(row)">删除</el-button>
        </template>
      </el-table-column>
    </el-table>

    <!-- 分页 -->
    <el-pagination
      v-model:current-page="queryParams.pageNum"
      v-model:page-size="queryParams.pageSize"
      :total="total"
      @current-change="handlePageChange"
      @size-change="handleSizeChange"
    />
  </div>
</template>

<script setup>
import { ref } from 'vue'
import { useCrud } from '@/composables/useCrud'
import { userApi } from '@/services/modules/user'
import { DictSelect, DictTag } from '@/components/Dict'

// 使用 useCrud
const {
  loading,
  tableData,
  total,
  queryParams,
  hasSelection,
  fetchList,
  handleQuery,
  resetQuery,
  handlePageChange,
  handleSizeChange,
  handleSelectionChange,
  handleDelete,
  handleBatchDelete
} = useCrud({
  api: userApi,
  defaultQuery: {
    username: '',
    status: undefined
  }
})

// 新增
const handleAdd = () => {
  // 打开新增弹窗
}

// 编辑
const handleEdit = (row) => {
  // 打开编辑弹窗
}
</script>

注意事项

  1. API 必填api.getList 是必填的,其他方法可选
  2. 自动过滤:查询时会自动过滤掉空值参数
  3. 分页处理:删除后如果当前页无数据,会自动返回上一页
  4. 错误处理:内部已封装错误处理,无需额外 try-catch

基于 MIT 许可发布