广告

买白酒,找南将

Skip to content

线条高级用法-曲线上的文字

Vue2 版本

lineAdv2.vue

javascript
<template>
  <div>
    <div ref="myPage" class="my-graph" :class="{
      'my-line-style-start': lineTextAnchor === 'start',
      'my-line-style-end': lineTextAnchor === 'end'
    }" style="height:calc(100vh);">
      <RelationGraph
        ref="graphRef"
        :options="graphOptions">
        <template #graph-plug>
          <div class="c-my-panel">
            <div class="c-option-name">
              连线形状:
            </div>
            <el-radio-group v-model="lineShape" size="small" @change="tabChange">
              <el-radio-button :label="1">直线</el-radio-button>
              <el-radio-button :label="2" >曲线2</el-radio-button>
              <el-radio-button :label="5" >曲线5</el-radio-button>
              <el-radio-button :label="6" >曲线6</el-radio-button>
              <el-radio-button :label="4" >折线</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">
              连接点:
            </div>
            <el-radio-group v-model="linePoint" size="small" @change="tabChange">
              <el-radio-button label="border">边缘</el-radio-button>
              <el-radio-button label="lr" >左右</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">
              连线文字自身对齐:
            </div>
            <el-radio-group v-model="lineTextAnchor" size="small" @change="tabChange">
              <el-radio-button label="start">start</el-radio-button>
              <el-radio-button label="middle">middle</el-radio-button>
              <el-radio-button label="end" >end</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">
              连线文字在连线上的位置:
            </div>
            <el-radio-group v-model="lineTextPlace" size="small" @change="tabChange">
              <el-radio-button label="">默认位置</el-radio-button>
              <el-radio-button label="start">靠近起点</el-radio-button>
              <el-radio-button label="end" >靠近终点</el-radio-button>
              <el-radio-button label="20%" >20%</el-radio-button>
              <el-radio-button label="40%" >40%</el-radio-button>
              <el-radio-button label="60%" >60%</el-radio-button>
              <el-radio-button label="80%" >80%</el-radio-button>
              <el-radio-button label="90%" >90%</el-radio-button>
            </el-radio-group>
<!--            <div class="c-option-name">连接文字x偏移量:{{defaultLineTextOffset_x}}</div>-->
<!--            <el-slider v-model="defaultLineTextOffset_x" :min="-250" :max="250" :show-tooltip="true" @change="tabChange"></el-slider>-->
<!--            <div class="c-option-name">连接文字y偏移量:{{defaultLineTextOffset_y}}</div>-->
<!--            <el-slider v-model="defaultLineTextOffset_y" :min="-50" :max="50" @change="tabChange"></el-slider>-->
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
const graphOptions = {
  allowSwitchLineShape: true,
  allowSwitchJunctionPoint: true,
  defaultLineColor: '#2E74B5',
  defaultNodeColor: '#2E74B5',
  defaultNodeBorderWidth: 0,
  defaultNodeBorderColor: '#2E74B5',
  defaultNodeFontColor: '#ffffff',
  defaultNodeWidth: 170,
  defaultNodeHeight: 40,
  toolBarDirection: 'h',
  toolBarPositionH: 'center',
  toolBarPositionV: 'bottom',
  defaultLineShape: 2,
  defaultJunctionPoint: 'lr',
  lineUseTextPath: true,
  defaultLineTextOffset_x: 2,
  defaultLineTextOffset_y: -3,
    lineTextMaxLength: 4,
  layout: {
    layoutName: 'tree',
    from: 'left',
    min_per_width: 410, // 根据节点的宽度设置,这个是让图谱看起来偏亮的关键
    min_per_height: 50
  }
  // 这里可以参考"Graph 图谱"中的参数进行设置
};
const __graph_json_data = {
  'rootId': 'a',
  'nodes': [
    { 'id': 'a', 'text': 'a' },
    { 'id': 'b', 'text': 'b' },
    { 'id': 'b1', 'text': 'b1' },
    { 'id': 'b1-1', 'text': 'b1-1' },
    { 'id': 'b1-2', 'text': 'b1-2' },
    { 'id': 'b1-3', 'text': 'b1-3' },
    { 'id': 'b1-4', 'text': 'b1-4' },
    { 'id': 'b1-5', 'text': 'b1-5' },
    { 'id': 'b1-6', 'text': 'b1-6' },
    { 'id': 'b2', 'text': 'b2' },
    { 'id': 'b2-1', 'text': 'b2-1' },
    { 'id': 'b2-2', 'text': 'b2-2' },
    { 'id': 'c', 'text': 'c' },
    { 'id': 'c1', 'text': 'c1' },
    { 'id': 'c2', 'text': 'c2' },
    { 'id': 'c3', 'text': 'c3' }],
  'lines': [
    { 'from': 'a', 'to': 'b', text: '这些关系文字都很长' },
    { 'from': 'b', 'to': 'b1', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-1', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-2', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-3', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-4', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-5', text: '这些关系文字都很长' },
    { 'from': 'b1', 'to': 'b1-6', text: '这些关系文字都很长' },
    { 'from': 'b', 'to': 'b2', text: '这些关系文字都很长' },
    { 'from': 'b2', 'to': 'b2-1', text: '这些关系文字都很长' },
    { 'from': 'b2', 'to': 'b2-2', text: '这些关系文字都很长' },
    { 'from': 'a', 'to': 'c', text: '这些关系文字都很长' },
    { 'from': 'c', 'to': 'c1', text: '这些关系文字都很长' },
    { 'from': 'c', 'to': 'c2', text: '这些关系文字都很长' },
    { 'from': 'c', 'to': 'c3', text: '这些关系文字都很长' }]
};
export default {
  name: 'Demo4AdvLineSlot',
  components: { },
  data() {
    return {
      defaultLineTextOffset_x: 2,
      defaultLineTextOffset_y: -3,
      lineShape: 2,
      linePoint: 'lr',
      lineTextAnchor: 'middle',
        lineTextPlace: '',
      graphOptions
    };
  },
  mounted() {
    this.showGraph();
  },
  methods: {
    showGraph() {
      this.$refs.graphRef.setJsonData(__graph_json_data, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码
      });
    },
    tabChange() {
      const graphInstance = this.$refs.graphRef.getInstance();
      const graphOptions = graphInstance.options;
      graphOptions.defaultLineShape = this.lineShape;
      graphOptions.defaultJunctionPoint = this.linePoint;
      graphOptions.defaultLineTextOffset_x = this.defaultLineTextOffset_x;
      graphOptions.defaultLineTextOffset_y = this.defaultLineTextOffset_y;
        const lines = graphInstance.getLinks().reduce((currentLines, link) => currentLines.concat(...link.relations), []);
        for (const line of lines) {
            line.placeText = this.lineTextPlace;
        }
    },
    async updateLayouterOptions() {
      const graphInstance = this.$refs.graphRef.getInstance();
      const graphOptions = graphInstance.options;
      graphOptions.defaultLineTextOffset_x = this.defaultLineTextOffset_x;
      graphOptions.defaultLineTextOffset_y = this.defaultLineTextOffset_y;
      // const graphInstance = this.$refs.graphRef.getInstance();
      // await graphInstance.setOptions(JSON.parse(JSON.stringify(this.graphOptions)));
      // await graphInstance.refresh();
    },
  }
};
</script>
<style lang="scss" scoped>

.c-my-panel {
  width: 350px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  padding:10px;
  background-color: rgba(239, 239, 239, 0.86);
  border: #eeeeee solid 1px;
  overflow: hidden;
  .c-option-name {
    color: #666666;
    font-size: 14px;
    line-height: 40px;
  }
}
::v-deep .my-line-style-start {
  .c-rg-line-text {
    textPath {
      text-anchor: start;
      startOffset: 0% !important;
    }
  }
}
::v-deep .my-line-style-end {
  .c-rg-line-text {
    textPath {
      text-anchor: end;
      startOffset: 100% !important;
    }
  }
}
</style>

Vue3 版本

lineAdv2.vue

javascript
<template>
  <div>
    <div
      ref="myPage" class="my-graph" :class="{
        'my-line-style-start': lineTextAnchor === 'start',
        'my-line-style-end': lineTextAnchor === 'end'
      }" style="height:calc(100vh);">
      <RelationGraph ref="graphRef" :options="graphOptions">
        <template #graph-plug>
          <div class="c-my-panel">
            <div class="c-option-name">
              Line Shape:
            </div>
            <el-radio-group v-model="lineShape" size="small" @change="tabChange">
              <el-radio-button :label="1">Straight</el-radio-button>
              <el-radio-button :label="2">Curve 2</el-radio-button>
              <el-radio-button :label="5">Curve 5</el-radio-button>
              <el-radio-button :label="6">Curve 6</el-radio-button>
              <el-radio-button :label="4">Polyline</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">
              Junction Point:
            </div>
            <el-radio-group v-model="linePoint" size="small" @change="tabChange">
              <el-radio-button label="border">Border</el-radio-button>
              <el-radio-button label="lr">Left-Right</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">
              Line Text Anchor:
            </div>
            <el-radio-group v-model="lineTextAnchor" size="small" @change="tabChange">
              <el-radio-button label="start">start</el-radio-button>
              <el-radio-button label="middle">middle</el-radio-button>
              <el-radio-button label="end">end</el-radio-button>
            </el-radio-group>
              <div class="c-option-name">
                  The Position Of The Text On The Line:
              </div>
              <el-radio-group v-model="lineTextPlace" size="small" @change="tabChange">
                  <el-radio-button label="">Default</el-radio-button>
                  <el-radio-button label="start">Close To Start</el-radio-button>
                  <el-radio-button label="end" >Close To End</el-radio-button>
                  <el-radio-button label="20%" >20%</el-radio-button>
                  <el-radio-button label="40%" >40%</el-radio-button>
                  <el-radio-button label="60%" >60%</el-radio-button>
                  <el-radio-button label="80%" >80%</el-radio-button>
                  <el-radio-button label="90%" >90%</el-radio-button>
              </el-radio-group>
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script lang="ts" setup>
import { ref, onMounted } from 'vue';
import RelationGraph, { RelationGraphComponent } from 'relation-graph-vue3';
import type { RGOptions, RGJsonData, RGNode, RGLine, RGLink, RGUserEvent } from 'relation-graph-vue3';

const graphOptions: RGOptions = {
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    defaultLineColor: '#2E74B5',
    defaultNodeColor: '#2E74B5',
    defaultNodeBorderWidth: 0,
    defaultNodeBorderColor: '#2E74B5',
    defaultNodeFontColor: '#ffffff',
    defaultNodeWidth: 170,
    defaultNodeHeight: 40,
    toolBarDirection: 'h',
    toolBarPositionH: 'center',
    toolBarPositionV: 'bottom',
    defaultLineShape: 2,
    defaultJunctionPoint: 'lr',
    lineUseTextPath: true,
    defaultLineTextOffset_x: 2,
    defaultLineTextOffset_y: -3,
    lineTextMaxLength: 5,
    layout: {
        layoutName: 'tree',
        from: 'left',
        'min_per_width': 410,
        'min_per_height': 50
    }
};

const __graph_json_data: RGJsonData = {
    rootId: 'a',
    nodes: [
        { id: 'a', text: 'a' },
        { id: 'b', text: 'b' },
        { id: 'b1', text: 'b1' },
        { id: 'b1-1', text: 'b1-1' },
        { id: 'b1-2', text: 'b1-2' },
        { id: 'b1-3', text: 'b1-3' },
        { id: 'b1-4', text: 'b1-4' },
        { id: 'b1-5', text: 'b1-5' },
        { id: 'b1-6', text: 'b1-6' },
        { id: 'b2', text: 'b2' },
        { id: 'b2-1', text: 'b2-1' },
        { id: 'b2-2', text: 'b2-2' },
        { id: 'c', text: 'c' },
        { id: 'c1', text: 'c1' },
        { id: 'c2', text: 'c2' },
        { id: 'c3', text: 'c3' }
    ],
    lines: [
        { from: 'a', to: 'b', text: 'These relationship texts are very long' },
        { from: 'b', to: 'b1', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-1', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-2', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-3', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-4', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-5', text: 'These relationship texts are very long' },
        { from: 'b1', to: 'b1-6', text: 'These relationship texts are very long' },
        { from: 'b', to: 'b2', text: 'These relationship texts are very long' },
        { from: 'b2', to: 'b2-1', text: 'These relationship texts are very long' },
        { from: 'b2', to: 'b2-2', text: 'These relationship texts are very long' },
        { from: 'a', to: 'c', text: 'These relationship texts are very long' },
        { from: 'c', to: 'c1', text: 'These relationship texts are very long' },
        { from: 'c', to: 'c2', text: 'These relationship texts are very long' },
        { from: 'c', to: 'c3', text: 'These relationship texts are very long' }
    ]
};
const lineShape = ref(2);
const linePoint = ref('lr');
const lineTextPlace = ref('');
const lineTextAnchor = ref('middle');
const defaultLineTextOffset_x = ref(0);
const defaultLineTextOffset_y = ref(0);

const graphRef = ref<RelationGraphComponent>();

const showGraph = async() => {
    const graphInstance = graphRef.value!.getInstance();
    await graphInstance.setJsonData(__graph_json_data);
};

const tabChange = () => {
    const graphInstance = graphRef.value!.getInstance();
    const graphOptions = graphInstance.options;
    graphOptions.defaultLineShape = lineShape.value;
    graphOptions.defaultJunctionPoint = linePoint.value;
    graphOptions.defaultLineTextOffset_x = defaultLineTextOffset_x.value;
    graphOptions.defaultLineTextOffset_y = defaultLineTextOffset_y.value;
    const lines = graphInstance.getLinks().reduce((currentLines, link) => currentLines.concat(...link.relations), []);
    for (const line of lines) {
        line.placeText = lineTextPlace.value;
    }
};

const updateLayouterOptions = async () => {
    const graphInstance = graphRef.value!.getInstance();
    const graphOptions = graphInstance.options;
    graphOptions.defaultLineTextOffset_x = defaultLineTextOffset_x.value;
    graphOptions.defaultLineTextOffset_y = defaultLineTextOffset_y.value;
};


onMounted(() => {
    showGraph();
});
</script>

<style lang="scss" scoped>
.c-my-panel {
  width: 350px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  padding: 10px;
  background-color: rgba(239, 239, 239, 0.86);
  border: #eeeeee solid 1px;
  overflow: hidden;
  .c-option-name {
    color: #666666;
    font-size: 14px;
    line-height: 40px;
  }
}
::v-deep(.my-line-style-start) {
  .relation-graph {
    .c-rg-line-text {
      textPath {
        text-anchor: start;
        startOffset: 0% !important;
      }
    }
  }
}
::v-deep(.my-line-style-end) {
  .relation-graph {
    .c-rg-line-text {
      textPath {
        text-anchor: end;
        startOffset: 100% !important;
      }
    }
  }
}
</style>

React 版本

lineAdv2.tsx

javascript
import React, {useEffect, useRef, useState} from 'react';
import RelationGraph, {
  RGOptions,
  RGJsonData,
  RGNode,
  RGLine,
  RGLink,
  RGUserEvent,
  RGNodeSlotProps,
  RelationGraphComponent, RGLineShape, RGJunctionPoint
} from 'relation-graph-react';
import {MySelector} from "./RGDemoComponents/MyUIComponents";
import './lineAdv2.scss';

const LineAdv2 = () => {
  const [lineShape, setLineShape] = useState(2);
  const [junctionPoint, setJunctionPoint] = useState('lr');
  const [lineTextPlace, setLineTextPlace] = useState('');
  const [lineTextAnchor, setLineTextAnchor] = useState('middle');
  const graphRef = useRef<RelationGraphComponent|null>(null);

  const graphOptions: RGOptions = {
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    defaultLineColor: '#2E74B5',
    defaultNodeColor: '#2E74B5',
    defaultNodeBorderWidth: 0,
    defaultNodeBorderColor: '#2E74B5',
    defaultNodeFontColor: '#ffffff',
    defaultNodeWidth: 170,
    defaultNodeHeight: 40,
    toolBarDirection: 'h',
    toolBarPositionH: 'center',
    toolBarPositionV: 'bottom',
    defaultLineShape: 2,
    defaultJunctionPoint: 'lr',
    lineUseTextPath: true,
    defaultLineTextOffset_x: 2,
    defaultLineTextOffset_y: -3,
    lineTextMaxLength: 5,
    layout: {
      layoutName: 'tree',
      from: 'left',
      'min_per_width': 410,
      'min_per_height': 50

    }
  };

  const __graph_json_data: RGJsonData = {
    rootId: 'a',
    nodes: [
      { id: 'a', text: 'a' },
      { id: 'b', text: 'b' },
      { id: 'b1', text: 'b1' },
      { id: 'b1-1', text: 'b1-1' },
      { id: 'b1-2', text: 'b1-2' },
      { id: 'b1-3', text: 'b1-3' },
      { id: 'b1-4', text: 'b1-4' },
      { id: 'b1-5', text: 'b1-5' },
      { id: 'b1-6', text: 'b1-6' },
      { id: 'b2', text: 'b2' },
      { id: 'b2-1', text: 'b2-1' },
      { id: 'b2-2', text: 'b2-2' },
      { id: 'c', text: 'c' },
      { id: 'c1', text: 'c1' },
      { id: 'c2', text: 'c2' },
      { id: 'c3', text: 'c3' }
    ],
    lines: [
      { from: 'a', to: 'b', text: 'These relationship texts are very long' },
      { from: 'b', to: 'b1', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-1', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-2', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-3', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-4', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-5', text: 'These relationship texts are very long' },
      { from: 'b1', to: 'b1-6', text: 'These relationship texts are very long' },
      { from: 'b', to: 'b2', text: 'These relationship texts are very long' },
      { from: 'b2', to: 'b2-1', text: 'These relationship texts are very long' },
      { from: 'b2', to: 'b2-2', text: 'These relationship texts are very long' },
      { from: 'a', to: 'c', text: 'These relationship texts are very long' },
      { from: 'c', to: 'c1', text: 'These relationship texts are very long' },
      { from: 'c', to: 'c2', text: 'These relationship texts are very long' },
      { from: 'c', to: 'c3', text: 'These relationship texts are very long' }
    ]
  };

  const MyGraphPlugSlot = () => {
    return (
      <div className="rounded-lg absolute left-20 top-20 z-20 p-4 bg-white border-solid border-2 border-black shadow-lg">
        <div className="c-option-name">
          Line Shape:
        </div>
        <MySelector
          data={[
            {value: 1, text: 'Straight' },
            {value: 2, text: 'Curve 2' },
            {value: 5, text: 'Curve 5' },
            {value: 6, text: 'Curve 6' },
            {value: 4, text: 'Polyline' },
          ]}
          currentValue={lineShape}
          onChange={(newValue: number, label) => {setLineShape(newValue);}}
        />
        <div className="c-option-name">
          Junction Point:
        </div>
        <MySelector
          data={[
            {value: 'border', text: 'Border' },
            {value: 'lr', text: 'Left-Right' },
          ]}
          currentValue={junctionPoint}
          onChange={(newValue: string, label) => {setJunctionPoint(newValue);}}
        />
        <div className="c-option-name">
          Line Text Anchor:
        </div>
        <MySelector
          data={[
            {value: 'start', text: 'start' },
            {value: 'middle', text: 'middle' },
            {value: 'end', text: 'end' },
          ]}
          currentValue={lineTextAnchor}
          onChange={(newValue: string, label) => {setLineTextAnchor(newValue);}}
        />
        <div className="c-option-name">
          The Position Of The Text On The Line:
        </div>
        <MySelector
          data={[
            {value: '', text: 'Default' },
            {value: 'start', text: 'Start' },
            {value: 'end', text: 'End' },
            {value: '20%', text: '20%' },
            {value: '40%', text: '40%' },
            {value: '60%', text: '60%' },
            {value: '80%', text: '80%' },
            {value: '90%', text: '90%' },
          ]}
          currentValue={lineTextPlace}
          onChange={(newValue: string, label) => {setLineTextPlace(newValue);}}
        />
      </div>
    );
  }
  const showGraph = async () => {
    const graphInstance = graphRef.current?.getInstance();
    if (graphInstance) {
      await graphInstance.setJsonData(__graph_json_data);
    }
  };
  const onTabChange = () => {
    const graphInstance = graphRef.current?.getInstance();
    if (graphInstance) {
      const graphOptions = graphInstance.options;
      graphOptions.defaultLineShape = lineShape as RGLineShape;
      graphOptions.defaultJunctionPoint = junctionPoint as RGJunctionPoint;
      // graphOptions.defaultLineTextOffset_x = defaultLineTextOffset_x.current;
      // graphOptions.defaultLineTextOffset_y = defaultLineTextOffset_y.current;
      const lines = graphInstance.getLinks().reduce((currentLines, link) => currentLines.concat(...link.relations), []);
      for (const line of lines) {
        line.placeText = lineTextPlace;
      }
      graphInstance.dataUpdated();
    }
  };
  useEffect(() => {
    showGraph();
  }, []);
  useEffect(() => {
    onTabChange();
  }, [lineShape, junctionPoint, lineTextPlace]);
  return (
    <div>
      <div
        className={`my-graph ${lineTextAnchor === 'start' ? 'my-line-style-start' : ''} ${lineTextAnchor === 'end' ? 'my-line-style-end' : ''}`}
        style={{ height: '100vh' }}
      >
        <RelationGraph
          ref={graphRef}
          options={graphOptions}
          graphPlugSlot={MyGraphPlugSlot}
        >
        </RelationGraph>
      </div>
    </div>
  );
};

export default LineAdv2;

lineAdv2.scss

scss
.c-my-panel {
  width: 350px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  padding: 10px;
  background-color: rgba(239, 239, 239, 0.86);
  border: #eeeeee solid 1px;
  overflow: hidden;
  .c-option-name {
    color: #666666;
    font-size: 14px;
    line-height: 40px;
  }
}

.my-line-style-start {
  .c-rg-line-text {
    textPath {
      text-anchor: start;
      startoffset: 0% !important;
    }
  }
}
.my-line-style-end {
  .c-rg-line-text {
    textPath {
      text-anchor: end;
      startoffset: 100% !important;
    }
  }
}

📂 RGDemoComponents

MyUIComponents.tsx

javascript
import React from "react";

export interface MySelectorProps {
  small?: boolean
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MySelector:React.FC<MySelectorProps> = ({small, data, onChange, currentValue}) => {
  return (
    <div className="flex flex-wrap justify-center rounded-lg border border-gray-900 overflow-hidden">
      {
        data.map(item =>
          <div key={item.value}
               className={`border-r w-auto text-xs cursor-pointer whitespace-nowrap ${currentValue === item.value && 'bg-blue-500 text-white'} ${small?' px-2 h-6 leading-6':'h-8 px-3 leading-8'}`}
               onClick={() => {onChange(item.value, item.text);}}
          >
          {item.text}
        </div>)
      }
    </div>
  );
};
export interface MySwitchProps {
  currentValue: boolean
  onChange: (newValue:boolean) => void
}
export const MySwitch:React.FC<MySwitchProps> = ({onChange, currentValue}) => {
  return (
    <div className={`w-14 flex rounded-full border p-0.5 ${currentValue ? 'justify-end border-blue-500' : 'justify-start border-gray-500'}`}>
      <div
        className={`w-8 h-5 leading-8 rounded-full w-auto px-3 text-xs cursor-pointer whitespace-nowrap ${currentValue ? 'bg-blue-500' : 'bg-gray-500'}`} onClick={() => {onChange(!currentValue);}}>
      </div>
    </div>
  );
};
export interface MySliderProps {
  min: number
  max: number
  step: number
  currentValue: number
  onChange: (newValue:number) => void
}
export const MySlider:React.FC<MySliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div>
      <input
        type="range"
        className="w-72"
        min={min}
        max={max}
        step={step}
        value={currentValue}
        onChange={(e) => { onChange(parseFloat(e.target.value))}}
      />
    </div>
  );
};

export interface MyRangeSliderProps {
  min: number
  max: number
  step: number
  currentValue: [number, number]
  onChange: (newValue:[number, number]) => void
}
export const MyRangeSlider:React.FC<MyRangeSliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div className="w-72">
      <div>Min:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[0]}
        onChange={(e) => { if (parseFloat(e.target.value) < currentValue[1]) onChange([parseFloat(e.target.value), currentValue[1]])}}
      />
      <div>Max:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[1]}
        onChange={(e) => { if (parseFloat(e.target.value) > currentValue[0]) onChange([currentValue[0], parseFloat(e.target.value)])}}
      />
    </div>
  );
};
export interface MyButtonProps {
  onClick: () => void
  disabled?: boolean
}
export const MyButton:React.FC<MyButtonProps> = ({children, onClick, disabled}) => {
  return (
    <button className={`mr-2 px-2 py-1 rounded ${disabled===true ? 'bg-gray-300 text-black cursor-not-allowed':'bg-blue-500 hover:bg-blue-700 text-white'}`}
            onClick={()=>{onClick();}}>{children}</button>
  );
};
export interface MyLinkButtonProps {
  onClick: () => void
}
export const MyLinkButton:React.FC<MyLinkButtonProps> = ({children, onClick}) => {
  return (
    <div className="text-blue-600 cursor-pointer underline decoration-1" onClick={()=>{onClick();}}>
      {children}
    </div>
  );
};
export interface MyCheckBoxProps {
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MyCheckBox:React.FC<MyCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onChange(thisItem.value, thisItem.text);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue === thisItem.value ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export interface CheckboxOption {
  value: string | number;
  text: string;
}

export interface MyMultiCheckBoxProps {
  currentValue: (string | number)[];
  checkboxOptions: CheckboxOption[];
  onChange: (newValue: (string | number)[]) => void;
}
export const MyMultiCheckBox:React.FC<MyMultiCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  const onClickItem = (item: string | number) => {
    const newValue = currentValue.includes(item)
      ? currentValue.filter((value) => value !== item)
      : [...currentValue, item];
    onChange(newValue);
  };
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onClickItem(thisItem.value);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue.includes(thisItem.value) ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export const ElMessage = (messageObject) => {
  console.warn(messageObject);
}
export const ElNotification = (messageObject) => {
  console.warn(messageObject);
}

本站搭建特别鸣谢【茂神大佬】