mirror of
https://github.com/nocobase/nocobase
synced 2024-11-16 06:48:46 +00:00
555 lines
10 KiB
Markdown
555 lines
10 KiB
Markdown
|
# Field
|
|||
|
|
|||
|
数据表字段管理类(抽象类)。同时是所有字段类型的基类,其他任意字段类型均通过继承该类来实现。
|
|||
|
|
|||
|
## 构造函数
|
|||
|
|
|||
|
通常不会直接由开发者调用,主要通过 `db.collection({ fields: [] })` 方法作为代理入口调用。
|
|||
|
|
|||
|
在扩展字段时主要通过继承 `Field` 抽象类,再注册到 Database 实例中来实现。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `constructor(options: FieldOptions, context: FieldContext)`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `options` | `FieldOptions` | - | 字段配置对象 |
|
|||
|
| `options.name` | `string` | - | 字段名称 |
|
|||
|
| `options.type` | `string` | - | 字段类型,对应在 db 中注册的字段类型名称 |
|
|||
|
| `context` | `FieldContext` | - | 字段上下文对象 |
|
|||
|
| `context.database` | `Database` | - | 数据库实例 |
|
|||
|
| `context.collection` | `Collection` | - | 数据表实例 |
|
|||
|
|
|||
|
## 实例成员
|
|||
|
|
|||
|
### `name`
|
|||
|
|
|||
|
字段名称。
|
|||
|
|
|||
|
### `type`
|
|||
|
|
|||
|
字段类型。
|
|||
|
|
|||
|
### `dataType`
|
|||
|
|
|||
|
字段数据库存储类型。
|
|||
|
|
|||
|
### `options`
|
|||
|
|
|||
|
字段初始化配置参数。
|
|||
|
|
|||
|
### `context`
|
|||
|
|
|||
|
字段上下文对象。
|
|||
|
|
|||
|
## 配置方法
|
|||
|
|
|||
|
### `on()`
|
|||
|
|
|||
|
基于数据表事件的快捷定义方式。相当于 `db.on(this.collection.name + '.' + eventName, listener)`。
|
|||
|
|
|||
|
继承时通常无需覆盖此方法。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `on(eventName: string, listener: (...args: any[]) => void)`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `eventName` | `string` | - | 事件名称 |
|
|||
|
| `listener` | `(...args: any[]) => void` | - | 事件监听器 |
|
|||
|
|
|||
|
### `off()`
|
|||
|
|
|||
|
基于数据表事件的快捷移除方式。相当于 `db.off(this.collection.name + '.' + eventName, listener)`。
|
|||
|
|
|||
|
继承时通常无需覆盖此方法。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `off(eventName: string, listener: (...args: any[]) => void)`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `eventName` | `string` | - | 事件名称 |
|
|||
|
| `listener` | `(...args: any[]) => void` | - | 事件监听器 |
|
|||
|
|
|||
|
### `bind()`
|
|||
|
|
|||
|
当字段被添加到数据表时触发的执行内容。通常用于添加数据表事件监听器和其他处理。
|
|||
|
|
|||
|
继承时需要先调用对应的 `super.bind()` 方法。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `bind()`
|
|||
|
|
|||
|
### `unbind()`
|
|||
|
|
|||
|
当字段从数据表中移除时触发的执行内容。通常用于移除数据表事件监听器和其他处理。
|
|||
|
|
|||
|
继承时需要先调用对应的 `super.unbind()` 方法。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `unbind()`
|
|||
|
|
|||
|
### `get()`
|
|||
|
|
|||
|
获取字段的配置项的值。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `get(key: string): any`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `key` | `string` | - | 配置项名称 |
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
const field = db.collection('users').getField('name');
|
|||
|
|
|||
|
// 获取字段名称配置项的值,返回 'name'
|
|||
|
console.log(field.get('name'));
|
|||
|
```
|
|||
|
|
|||
|
### `merge()`
|
|||
|
|
|||
|
合并字段的配置项的值。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `merge(options: { [key: string]: any }): void`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `options` | `{ [key: string]: any }` | - | 要合并的配置项对象 |
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
const field = db.collection('users').getField('name');
|
|||
|
|
|||
|
field.merge({
|
|||
|
// 添加一个索引配置
|
|||
|
index: true
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `remove()`
|
|||
|
|
|||
|
从数据表中移除字段(仅从内存中移除)。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
const books = db.getCollections('books');
|
|||
|
|
|||
|
books.getField('isbn').remove();
|
|||
|
|
|||
|
// really remove from db
|
|||
|
await books.sync();
|
|||
|
```
|
|||
|
|
|||
|
## 数据库方法
|
|||
|
|
|||
|
### `removeFromDb()`
|
|||
|
|
|||
|
从数据库中移除字段。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `removeFromDb(options?: Transactionable): Promise<void>`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `options.transaction?` | `Transaction` | - | 事务实例 |
|
|||
|
|
|||
|
### `existsInDb()`
|
|||
|
|
|||
|
判断字段是否存在于数据库中。
|
|||
|
|
|||
|
**签名**
|
|||
|
|
|||
|
* `existsInDb(options?: Transactionable): Promise<boolean>`
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `options.transaction?` | `Transaction` | - | 事务实例 |
|
|||
|
|
|||
|
## 内置字段类型列表
|
|||
|
|
|||
|
NocoBase 内置了一些常用的字段类型,可以直接在定义数据表的字段是使用对应的 type 名称来指定类型。不同类型的字段参数配置不同,具体可参考下面的列表。
|
|||
|
|
|||
|
所有字段类型的配置项除了以下额外介绍的以外,都会透传至 Sequelize,所以所有 Sequelize 支持的字段配置项都可以在这里使用(如 `allowNull`、`defaultValue` 等)。
|
|||
|
|
|||
|
另外 server 端的字段类型主要解决数据库存储和部分算法的问题,与前端的字段展示类型和使用组件基本无关。前端字段类型可以参考教程对应说明。
|
|||
|
|
|||
|
### `'boolean'`
|
|||
|
|
|||
|
逻辑值类型。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```js
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'boolean',
|
|||
|
name: 'published'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'integer'`
|
|||
|
|
|||
|
整型(32 位)。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'integer',
|
|||
|
name: 'pages'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'bigInt'`
|
|||
|
|
|||
|
长整型(64 位)。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'bigInt',
|
|||
|
name: 'words'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'double'`
|
|||
|
|
|||
|
双精度浮点型(64 位)。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'double',
|
|||
|
name: 'price'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'real'`
|
|||
|
|
|||
|
实数类型(仅 PG 适用)。
|
|||
|
|
|||
|
### `'decimal'`
|
|||
|
|
|||
|
十进制小数类型。
|
|||
|
|
|||
|
### `'string'`
|
|||
|
|
|||
|
字符串类型。相当于大部分数据库的 `VARCHAR` 类型。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'string',
|
|||
|
name: 'title'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'text'`
|
|||
|
|
|||
|
文本类型。相当于大部分数据库的 `TEXT` 类型。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'books',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'text',
|
|||
|
name: 'content'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'password'`
|
|||
|
|
|||
|
密码类型(NocoBase 扩展)。基于 Node.js 原生的 crypto 包的 `scrypt` 方法进行密码加密。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'users',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'password',
|
|||
|
name: 'password',
|
|||
|
length: 64, // 长度,默认 64
|
|||
|
randomBytesSize: 8 // 随机字节长度,默认 8
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
**参数**
|
|||
|
|
|||
|
| 参数名 | 类型 | 默认值 | 描述 |
|
|||
|
| --- | --- | --- | --- |
|
|||
|
| `length` | `number` | 64 | 字符长度 |
|
|||
|
| `randomBytesSize` | `number` | 8 | 随机字节大小 |
|
|||
|
|
|||
|
### `'date'`
|
|||
|
|
|||
|
日期类型。
|
|||
|
|
|||
|
### `'time'`
|
|||
|
|
|||
|
时间类型。
|
|||
|
|
|||
|
### `'array'`
|
|||
|
|
|||
|
数组类型(仅 PG 适用)。
|
|||
|
|
|||
|
### `'json'`
|
|||
|
|
|||
|
JSON 类型。
|
|||
|
|
|||
|
### `'jsonb'`
|
|||
|
|
|||
|
JSONB 类型(仅 PG 适用,其他会被兼容为 `'json'` 类型)。
|
|||
|
|
|||
|
### `'uuid'`
|
|||
|
|
|||
|
UUID 类型。
|
|||
|
|
|||
|
### `'uid'`
|
|||
|
|
|||
|
UID 类型(NocoBase 扩展)。短随机字符串标识符类型。
|
|||
|
|
|||
|
### `'formula'`
|
|||
|
|
|||
|
公式类型(NocoBase 扩展)。可配置基于 [mathjs](https://www.npmjs.com/package/mathjs) 的数学公式计算,公式中可以引用同一条记录中其他列的数值参与计算。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'orders',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'double',
|
|||
|
name: 'price'
|
|||
|
},
|
|||
|
{
|
|||
|
type: 'integer',
|
|||
|
name: 'quantity'
|
|||
|
},
|
|||
|
{
|
|||
|
type: 'formula',
|
|||
|
name: 'total',
|
|||
|
expression: 'price * quantity'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'radio'`
|
|||
|
|
|||
|
单选类型(NocoBase 扩展)。全表最多有一行数据的该字段值为 `true`,其他都为 `false` 或 `null`。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
整个系统只有一个被标记为 root 的用户,任意另一个用户的 root 值被改为 `true` 之后,其他所有 root 为 `true` 的记录均会被修改为 `false`:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'users',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'radio',
|
|||
|
name: 'root',
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'sort'`
|
|||
|
|
|||
|
排序类型(NocoBase 扩展)。基于整型数字进行排序,为新记录自动生成新序号,当移动数据时进行序号重排。
|
|||
|
|
|||
|
数据表如果定义了 `sortable` 选项,也会自动生成对应字段。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
文章基于所属用户可排序:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'posts',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'belongsTo',
|
|||
|
name: 'user',
|
|||
|
},
|
|||
|
{
|
|||
|
type: 'sort',
|
|||
|
name: 'priority',
|
|||
|
scopeKey: 'userId' // 以 userId 相同值分组的数据进行排序
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'virtual'`
|
|||
|
|
|||
|
虚拟类型。不实际储存数据,仅用于特殊 getter/setter 定义时使用。
|
|||
|
|
|||
|
### `'belongsTo'`
|
|||
|
|
|||
|
多对一关联类型。外键储存在自身表,与 hasOne/hasMany 相对。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
任意文章属于某个作者:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'posts',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'belongsTo',
|
|||
|
name: 'author',
|
|||
|
target: 'users', // 不配置默认为 name 复数名称的表名
|
|||
|
foreignKey: 'authorId', // 不配置默认为 <name> + Id 的格式
|
|||
|
sourceKey: 'id' // 不配置默认为 target 表的 id
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'hasOne'`
|
|||
|
|
|||
|
一对一关联类型。外键储存在关联表,与 belongsTo 相对。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
任意用户都有一份个人资料:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'users',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'hasOne',
|
|||
|
name: 'profile',
|
|||
|
target: 'profiles', // 可省略
|
|||
|
}
|
|||
|
]
|
|||
|
})
|
|||
|
```
|
|||
|
|
|||
|
### `'hasMany'`
|
|||
|
|
|||
|
一对多关联类型。外键储存在关联表,与 belongsTo 相对。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
任意用户可以拥有多篇文章:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'users',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'hasMany',
|
|||
|
name: 'posts',
|
|||
|
foreignKey: 'authorId',
|
|||
|
sourceKey: 'id'
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|
|||
|
|
|||
|
### `'belongsToMany'`
|
|||
|
|
|||
|
多对多关联类型。使用中间表储存双方外键,如不指定已存在的表为中间表的话,将会自动创建中间表。
|
|||
|
|
|||
|
**示例**
|
|||
|
|
|||
|
任意文章可以加任意多个标签,任意标签也可以被任意多篇文章添加:
|
|||
|
|
|||
|
```ts
|
|||
|
db.collection({
|
|||
|
name: 'posts',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'belongsToMany',
|
|||
|
name: 'tags',
|
|||
|
target: 'tags', // 同名可省略
|
|||
|
through: 'postsTags', // 中间表不配置将自动生成
|
|||
|
foreignKey: 'postId', // 自身表在中间表的外键
|
|||
|
sourceKey: 'id', // 自身表的主键
|
|||
|
otherKey: 'tagId' // 关联表在中间表的外键
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
|
|||
|
db.collection({
|
|||
|
name: 'tags',
|
|||
|
fields: [
|
|||
|
{
|
|||
|
type: 'belongsToMany',
|
|||
|
name: 'posts',
|
|||
|
through: 'postsTags', // 同一组关系指向同一张中间表
|
|||
|
}
|
|||
|
]
|
|||
|
});
|
|||
|
```
|