OpenDataV低代码平台新增组件流程

OpenDataV计划采用子库的方式添加子组件,即每一个组件都当做一个子库,子库有自己的依赖,而项目本身的依赖只针对框架,因此每一个组件我们都当做一个子库来开发。下面我带着大家一步步详细的开发一个数字展示组件。

创建组件目录和文件

  • 进入组件库目录下

所有的可拖拽组件都存放在src/resource/components目录下

cd src/resource/components
  • 根据组件名称创建目录

默认组件目录是以组件的名称命名,当然也可以根据自己的要求命名,组件可以放在components目录下,也可以放在其子目录的目录下。

mkdir DigitalText
  • 创建组件所需的固定文件

每个组件必须的文件有vue文件xxx.vue、配置文件config.ts、导出文件index.ts,每个文件有自己的用处,vue文件不用说了是组件渲染的主体,导出文件名称固定为index.ts,主要是导出组件的信息给外部引用,配置文件主要是在编辑页面右侧展示的配置项,这个我们后面再详述。所以这里我们需要创建三个文件:DigitalText.vue、config.ts、index.ts

以上我们就创建好了组件所需的文件,下面就需要做组件的初始化了

初始化组件文件

因为我们的组件都是以子库的方式引入,所以需要进行包的初始化,执行以下命令

cd src/resource/components/Text/DigitalText

npm init

这里使用npm初始化包会让我们选择填写部分数据。

下面我们先初始化一下组件文件DigitalText.vue,先初始化最简单的组件数据

<template>
  <div>数字展示</div>
</template>

<script lang="ts" setup></script>

<style lang="less" scoped></style>

然后我们要初始化组件的配置文件config.ts

import { ComponentGroup, FormType } from '@/enum'
import type { PropsType } from '@/types/component'
import { BaseComponent } from '@/resource/models'

export const componentName = 'Digital'
class DigitalComponent extends BaseComponent {
  constructor(id?: string, name?: string, icon?: string) {
    super({
      component: componentName,
      group: ComponentGroup.TEXT,
      name: name ? name : '数字文本',
      id,
      width: 100,
      height: 30,
      icon
    })
  }
}

export default DigitalComponent

这里要说明的点:componentName是组件在项目中的注册名称,因此必须保证唯一,group是给组件分组,这里的分组主要是展示在组件拖拽页面,其类型的ComponentGroup是固定好的,可以自己增加,展示位置如下:

name是组件在拖拽页面上显示的名称,widthheight是组件拖拽到画布上显示的初始大小

配置完组件后数据后,就可以配置导出文件index.ts了,主要是导出组件名、组件对象和配置项。

import DigitalTextComponent, { componentName } from './config'

export default {
  componentName,
  component: () => import('./DigitalText.vue'),
  config: DigitalTextComponent
}

初始化到这一步,我们的组件已经可以在编辑页面正常使用了,看一下效果:

在右边的编辑页面我们看到有样式属性,所有的组件都包含基础样式位置大小,包括组件的上下边距和宽高,属性包含公共属性,其中组件组件ID无法修改,主要是展示来看的,名称可以修改,名称属性主要是展示在图层上,修改名称后图层上会响应的显示修改后的名称。

这是最基础的组件,只能展示固定数据,不能进行任何配置,下面我们要做组件的配置项。

组件配置项

样式配置

作为文字显示组件,最基础的字体相关属性配置应该要有,比如字体、字体颜色、字体大小、字体宽度,配置项依然是在配置文件中添加,继承自基础组件类的私有属性_style

class DigitalTextComponent extends BaseComponent {
  constructor(id?: string, name?: string, icon?: string) {......}
  _style: PropsType[] = [
    {
      label: '字体设置',
      prop: 'font',
      children: [
        {
          prop: 'color',
          label: '颜色',
          type: FormType.COLOR,
          componentOptions: {
            defaultValue: 'skyblue'
          }
        },
        {
          prop: 'fontSize',
          label: '字体大小',
          type: FormType.NUMBER,
          componentOptions: {
            defaultValue: 20
          }
        },
        {
          prop: 'fontWeight',
          label: '字体宽度',
          type: FormType.FONT_WEIGHT,
          componentOptions: {
            defaultValue: 200
          }
        },
        {
          prop: 'fontFamily',
          label: '字体',
          type: FormType.FONT_STYLE,
          componentOptions: {
            defaultValue: 'Arial'
          }
        }
      ]
    }
  ]
}

样式配置的格式已经定义好了,其中需要注意的是所有children下的子项中prop必须是html元素的css属性,具体的css属性名称并不是我们在style文件中填写的,而是在js中对应的名称,这个可以在网上搜索:css3 中文手册,类似如下:

我们再详述一下各配置项的意义:

  • label:配置显示的分组名
  • prop:唯一的属性区分,此属性要与同级别中的其他配置不同
  • children:此属性组下面的配置项
    • label:每个属性的名称
    • prop:css属性值
    • type:属性编辑时显示的组件类型,目前可显示的组件是固定的,类型都定义在FormType
    • componentOptions:属性对应的配置项,不同类型的组件有不同的配置项,具体可以在src/types/component.d.ts中查看定义,所有配置都有defaultValue配置,作为属性初始化时的默认值

配置完样式后,我们在编辑页面看一下效果:

搞清楚了样式的配置,下面我们再说说属性的配置,属性配置与样式配置格式一致,有一些小细节需要注意。

属性配置

属性配置是继承私有变量_prop,配置格式与样式相同,我们这里配置一个示例属性:

class DigitalTextComponent extends BaseComponent {
  constructor(id?: string, name?: string, icon?: string) {......}
  _prop: PropsType[] = [
      {
        label: '数据配置',
        prop: 'dataconfig',
        children: [
          {
            prop: 'data',
            label: '数据',
            type: FormType.NUMBER,
            componentOptions: {
              defaultValue: 100000,
              max: 99999999,
              min: 0
            }
          }
        ]
      }
    ]
}

格式这里就不解释了,这里我们用到了数值类型,因此可以配置最大最小值。

接下来就是要在vue文件中使用属性配置了,属性不像样式,样式是html元素本身就支持的,因此只要我们配置好,就可以生效了,但是属性是组件专有的,什么属性要产生什么效果全凭我们自己书写逻辑,因此配置好属性我们只会在编辑页面看到属性展示和配置,但是实际配置后是没有任何效果的,具体效果我们在vue中实现。

属性使用

首先我们要添加一个类型定义文件,因为ts最基础的优势就是类型提示,而我们封装的组件基类是通用的,因此需要在每个组件中使用自己的属性类型定义,定义如下:

// DigitalText/type.ts
export interface DigitalType {
  dataconfig: {
    data: number
  }
}

为了准确的进行提示,类型定义必须和属性配置一直,具体来讲就是children下面的prop作为属性值,children外面的prop作为属性键,可以对比一下type.ts中的配置和_prop的配置。

组件的配置信息是通过外部传入的,所以所有组件都必须接收外部数据,我们已经定义好了固定的格式

const props = defineProps<{
  component: DigitalTextComponent
}>()

组件相关的所有信息都将通过component传入进来,为了监听属性变化和类型提示,我们封装了一个hook,减少每个组件中通用的处理,useProp的使用如下:

const propChange = (prop: string, key: string, value: number) => {
  console.log(prop, key, value)
}

const { propValue } = useProp<DigitalType>(props.component, propChange)

useProp接收三个参数,一个是component,主要是为了添加类型提示,所以这里也传入了一个泛型定义,就是我们在type.ts中定义的类型,另外两个参数是属性变化回调函数和样式变化回调函数。一般情况下我们只需要处理属性变化回调,样式变化是自动生效的,所以基本上不用处理,如果有特殊需求才需要。属性变化回调函数中有三个参数,prop对应的是属性配置中外层的prop值,key对应的是属性配置中children中的prop值,而value就是属性变化的值。

最终我们的属性处理结果如下:

<template>
  <div>{{ data }}</div>
</template>

<script lang="ts" setup>
import { ref } from 'vue'
import DigitalTextComponent from './config'
import { useProp } from '@/resource/hooks'
import { DigitalType } from './type'

const props = defineProps<{
  component: DigitalTextComponent
}>()

const propChange = (prop: string, key: string, value: number) => {
  if (prop === 'dataconfig' && key === 'data') {
    data.value = value
  }
}
const { propValue } = useProp<DigitalType>(props.component, propChange)

const data = ref<number>(propValue.dataconfig.data)
</script>

<style lang="less" scoped></style>

看一下页面上的效果:

上面我们用了属性回调去处理值变化响应,实际上还有其他的方式可以处理,我们要明白属性回调的根本需求是什么?主要就是为了编辑了对应的属性后,我们在组件内能监测到变化反馈到显示上。相同这一点,可用的方法就多了。

  • 直接使用props传递的属性值在template中渲染数据
<template>
  <div>{{ propValue.dataconfig.data }}</div>
</template>

<script lang="ts" setup>
import DigitalTextComponent from './config'
import { useProp } from '@/resource/hooks'
import { DigitalType } from './type'

const props = defineProps<{
  component: DigitalTextComponent
}>()

const { propValue } = useProp<DigitalType>(props.component)

因为vue响应式的原因,props中的数据是可以响应变化的,那么我们直接在template中使用即可,不需要做任何监测。

  • 使用computed或者watch监听属性变化

这里和上面是一样的道理,vue会自动帮我们处理响应式数据,只要用vue的计算属性或者watch也可以监听到属性变化。

<template>
  <div>{{ data }}</div>
</template>

<script lang="ts" setup>
import { computed } from 'vue'
import DigitalTextComponent from './config'
import { useProp } from '@/resource/hooks'
import { DigitalType } from './type'

const props = defineProps<{
  component: DigitalTextComponent
}>()

const { propValue } = useProp<DigitalType>(props.component)

const data = computed<number>(() => {
  return propValue.dataconfig.data
})
</script>

<style lang="less" scoped></style>

  • 属性变化回调的另一种用法
<template>
  <div>{{ data }}</div>
</template>

<script lang="ts" setup>
import { ref } from 'vue'
import DigitalTextComponent from './config'
import { useProp } from '@/resource/hooks'
import { DigitalType } from './type'

const props = defineProps<{
  component: DigitalTextComponent
}>()

const propChange = () => {
  data.value = propValue.dataconfig.data
}

const { propValue } = useProp<DigitalType>(props.component, propChange)

const data = ref<number>(propValue.dataconfig.data)
</script>

<style lang="less" scoped></style>

属性变化回调接收任何参数,我们可以选择接收参数,也可以不接收参数,在一些配置项比较多的组件中,我们不想在属性回调中去一个一个判断变化的属性,那么就可以使用这种方式,在这种方式中我们只是把属性回调作为一个通知,即通知我们属性发生变化了,而我们不关心哪一个属性发生了变化,把所有的属性都修改一遍即可,虽然听起来比较麻烦,但是在一些复杂组件中确实很有作用。在这里我们要明白,只要属性发生了变化,那么prop中的数据也必定发生了变化,所以我们随时取prop中的数据它都是最新的。

总结

到这里,一个组件的整个添加过程就讲完了,根据目前的开发进度来看,基本上所有的部分都讲到了,如果有人在使用过程中发现了什么问题或者有哪些地方不够清楚的,可以在项目的issue中提,也可以通过其他方式反馈。

添加微信公众号了解更多信息:

本文作者:星星在线

本文链接:https://www.cnblogs.com/small-bud/p/16686407.html

版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。

posted @   星星在线  阅读(636)  评论(5编辑  收藏  举报
点击右上角即可分享
微信分享提示
💬
评论
📌
收藏
💗
关注
👍
推荐
🚀
回顶
收起
🔑
  1. 1 Victory REOL
Victory - REOL
00:00 / 00:00
An audio error has occurred.

作曲 : Reol

作词 : Reol

fade away...do over again...

fade away...do over again...

歌い始めの一文字目 いつも迷ってる

歌い始めの一文字目 いつも迷ってる

どうせとりとめのないことだけど

伝わらなきゃもっと意味がない

どうしたってこんなに複雑なのに

どうしたってこんなに複雑なのに

噛み砕いてやらなきゃ伝わらない

ほら結局歌詞なんかどうだっていい

僕の音楽なんかこの世になくたっていいんだよ

Everybody don't know why.

Everybody don't know why.

Everybody don't know much.

僕は気にしない 君は気付かない

何処にももういないいない

Everybody don't know why.

Everybody don't know why.

Everybody don't know much.

忘れていく 忘れられていく

We don't know,We don't know.

目の前 広がる現実世界がまた歪んだ

目の前 広がる現実世界がまた歪んだ

何度リセットしても

僕は僕以外の誰かには生まれ変われない

「そんなの知ってるよ」

気になるあの子の噂話も

シニカル標的は次の速報

麻痺しちゃってるこっからエスケープ

麻痺しちゃってるこっからエスケープ

遠く遠くまで行けるよ

安定なんてない 不安定な世界

安定なんてない 不安定な世界

安定なんてない きっと明日には忘れるよ

fade away...do over again...

fade away...do over again...

そうだ世界はどこかがいつも嘘くさい

そうだ世界はどこかがいつも嘘くさい

綺麗事だけじゃ大事な人たちすら守れない

くだらない 僕らみんなどこか狂ってるみたい

本当のことなんか全部神様も知らない

Everybody don't know why.

Everybody don't know why.

Everybody don't know much.

僕は気にしない 君は気付かない

何処にももういないいない

Everybody don't know why.

Everybody don't know why.

Everybody don't know much.

忘れていく 忘れられていく

We don't know,We don't know.