代码编织梦想

简介

最近博主接到需求,画一个鱼骨图,网上找了下,没有现成的,于是博主自己写了个,可以给大家提供一个思路

效果图

在这里插入图片描述
博主是根据客户那边的要求进行一步一步调整的,样式比较丑,用的时候可以根据自己的需要修改样式,博主更多的是提供一个思路

首先新建一个fishboneDiagram.vue文件,然后将下面的代码复制进去

html

<template>
  <div class="fishboneDiagram">
    <div class="fishbone">
      <div class="content" ref="content">
        <!-- 大骨 -->
        <el-row type="flex" class="top-bone" align="bottom">
          <div
            class="item-bone"
            v-for="(item, index) in fishboneData.childrenList"
            :key="index"
          >
            <div class="item-bone-children" v-if="index % 2 == 0">
              <!-- 中骨 -->
              <div class="item_bone_children_item left_item_bone_children">
                <div
                  v-for="(metatarsus, i) in item.childrenList"
                  class="children-item"
                  :key="i"
                >
                  <div class="metatarsus left_metatarsus" v-if="i % 2 == 0">
                    <!-- 小骨 -->
                    <div class="metatarsus_content left_content top_content">
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text"
                          v-if="ossindex % 2 == 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                    <!-- 中骨 -->
                    <div class="metatarsus_main">
                      <div class="metatarsus_title">
                        {{ metatarsus.content }}
                      </div>
                      <div class="metatarsus_line"></div>
                    </div>
                    <!-- 小骨 -->
                    <div class="metatarsus_content left_content bottom_content">
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text"
                          v-if="ossindex % 2 != 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
              <div class="item_bone_children_bigBone">
                <div class="item_bone_children_bigBone_title">
                  {{ item.content }}
                </div>
                <div class="item_bone_children_bigBone_line"></div>
              </div>
              <!-- 中骨 -->
              <div class="item_bone_children_item right_item_bone_children">
                <div
                  v-for="(metatarsus, i) in item.childrenList"
                  class="children-item right_children_item"
                  :key="i"
                >
                  <div class="metatarsus" v-if="i % 2 != 0">
                    <!-- 小骨 -->
                    <div class="metatarsus_content right_content top_content">
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text"
                          v-if="ossindex % 2 == 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                    <!-- 中骨 -->
                    <div class="metatarsus_main">
                      <div class="metatarsus_line"></div>
                      <div class="metatarsus_title">
                        {{ metatarsus.content }}
                      </div>
                    </div>
                    <!-- 小骨 -->
                    <div
                      class="metatarsus_content right_content bottom_content"
                    >
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text"
                          v-if="ossindex % 2 != 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
          </div>
        </el-row>
        <!-- 主骨 -->
        <div class="center_cardinalBone">
          <div class="center_cardinalBone_line"></div>
          <div class="center_cardinalBone_title">
            {{ fishboneData.content }}
          </div>
        </div>
        <!-- 大骨 -->
        <el-row type="flex" class="bottom-bone">
          <div
            class="item-bone"
            v-for="(item, index) in fishboneData.childrenList"
            :key="index"
          >
            <div class="item-bone-children" v-if="index % 2 != 0">
              <!-- 中骨 -->
              <div class="item_bone_children_item left_item_bone_children">
                <div
                  v-for="(metatarsus, i) in item.childrenList"
                  class="children-item"
                  :key="i"
                >
                  <div class="metatarsus left_metatarsus" v-if="i % 2 == 0">
                    <div
                      class="metatarsus_content left_content bottom_left_content top_content"
                    >
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text bottom_text"
                          v-if="ossindex % 2 == 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                    <div class="metatarsus_main">
                      <div class="metatarsus_title bottom_text">
                        {{ metatarsus.content }}
                      </div>
                      <div class="metatarsus_line"></div>
                    </div>
                    <div
                      class="metatarsus_content left_content bottom_left_content bottom_content"
                    >
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text bottom_text"
                          v-if="ossindex % 2 != 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
              <div class="item_bone_children_bigBone">
                <div class="item_bone_children_bigBone_line"></div>
                <div class="item_bone_children_bigBone_title bottom_text">
                  {{ item.content }}
                </div>
              </div>
              <!-- 中骨 -->
              <div class="item_bone_children_item right_item_bone_children">
                <div
                  v-for="(metatarsus, i) in item.childrenList"
                  class="children-item right_children_item"
                  :key="i"
                >
                  <div class="metatarsus" v-if="i % 2 != 0">
                    <!-- 小骨 -->
                    <div
                      class="metatarsus_content right_content bottom_right_content top_content"
                    >
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text bottom_text"
                          v-if="ossindex % 2 == 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                    <!-- 中骨 -->
                    <div class="metatarsus_main">
                      <div class="metatarsus_line"></div>
                      <div class="metatarsus_title bottom_text">
                        {{ metatarsus.content }}
                      </div>
                    </div>
                    <!-- 小骨 -->
                    <div
                      class="metatarsus_content right_content bottom_content"
                    >
                      <div
                        v-for="(ossicle, ossindex) in metatarsus.childrenList"
                        :key="ossindex"
                        class="metatarsus_content_item"
                      >
                        <div
                          class="metatarsus_content_item_text bottom_text"
                          v-if="ossindex % 2 != 0"
                        >
                          {{ ossicle.content }}
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
          </div>
        </el-row>
      </div>
    </div>
  </div>
</template>

js部分

fishboneData是父组件传的数据,树结构的数据,子节点为childrenList,显示的文字参数为content

<script>
export default {
  name: "fishboneDiagram",
  props: {
    fishboneData: {
      type: Object,
      default: () => {},
    },
  },
  data() {
    return {};
  },
  created() {},
  mounted() {
  	// 动态计算
    this.$nextTick(() => {
      // 小骨与中骨位置间隔
      const metatarsusTitle =
        document.getElementsByClassName("metatarsus_title");
      const topContent = document.getElementsByClassName("top_content");
      const bottomContent = document.getElementsByClassName("bottom_content");
      for (let index = 0; index < metatarsusTitle.length; index++) {
        const topHeight = metatarsusTitle[index].clientHeight * -0.5;
        topContent[index].style.marginBottom = `${topHeight}px`;
        bottomContent[index].style.marginTop = `${topHeight}px`;
      }
      // 中骨与大骨位置间隔
      const itemBoneChildrenBigBoneTitle = document.getElementsByClassName(
        "item_bone_children_bigBone_title"
      );
      const leftItemBoneChildren = document.getElementsByClassName(
        "left_item_bone_children"
      );
      const rightItemBoneChildren = document.getElementsByClassName(
        "right_item_bone_children"
      );
      for (
        let index = 0;
        index < itemBoneChildrenBigBoneTitle.length;
        index++
      ) {
        const topHeight =
          itemBoneChildrenBigBoneTitle[index].clientWidth * -0.5;
        leftItemBoneChildren[index].style.marginRight = `${topHeight}px`;
        rightItemBoneChildren[index].style.marginLeft = `${topHeight}px`;
      }
      // 大骨与主骨位置间隔
      const centerCardinalBoneTitle = document.getElementsByClassName(
        "center_cardinalBone_title"
      );
      const topBone = document.getElementsByClassName("top-bone");
      const bottomBone = document.getElementsByClassName("bottom-bone");
      for (let index = 0; index < centerCardinalBoneTitle.length; index++) {
        const topHeight = centerCardinalBoneTitle[index].clientHeight * -0.5;
        topBone[index].style.marginBottom = `${topHeight}px`;
        bottomBone[index].style.marginTop = `${topHeight}px`;
      }
      // 中骨
      const metatarsusContent =
        document.getElementsByClassName("metatarsus_content");
      let metatarsusLine = document.getElementsByClassName("metatarsus_line");
      for (let index = 0; index < metatarsusLine.length; index++) {
        if (
          metatarsusContent[index * 2].clientWidth >
          metatarsusContent[index * 2 + 1].clientWidth
        ) {
          metatarsusLine[index].style.width = `${
            metatarsusContent[index * 2].clientWidth * 1.2
          }px`;
        } else {
          metatarsusLine[index].style.width = `${
            metatarsusContent[index * 2 + 1].clientWidth * 1.2
          }px`;
        }
      }
      // 大骨
      const elementItem = document.getElementsByClassName(
        "item_bone_children_item"
      );
      let element1 = document.getElementsByClassName(
        "item_bone_children_bigBone_line"
      );
      for (let index = 0; index < element1.length; index++) {
        // element1[index].style.height = `${element[index].clientHeight * 1}px`;
        if (
          elementItem[index * 2].clientHeight >
          elementItem[index * 2 + 1].clientHeight
        ) {
          element1[index].style.height = `${
            elementItem[index * 2].clientHeight * 1.2
          }px`;
        } else {
          element1[index].style.height = `${
            elementItem[index * 2 + 1].clientHeight * 1.2
          }px`;
        }
      }
    });
  },
  methods: {},
};
</script>

css部分

<style lang="less" scoped>
@bnoe-color: #002766;

.children-item {
  display: flex;
  justify-content: flex-end;
  align-items: center;
}

.right_children_item {
  justify-content: flex-start !important;
}
.fishboneDiagram {
  width: 100%;
  user-select: none;
}

.fishbone {
  min-height: 300px;
  height: 100%;
  position: relative;

  rgb(5, 10, 17) .content {
    width: 100%;
    height: 100%;
    position: absolute;
    left: 50%;
    top: 50%;
    transform: translate(-50%, -50%);
  }

  .center_cardinalBone {
    width: 100%;
    display: flex;
    justify-content: center;
    align-items: center;
    .center_cardinalBone_line {
      flex: 1;
      height: 2px;
      background-color: @bnoe-color;
    }

    .center_cardinalBone_title {
      width: fit-content;
      font-size: 16px;
      font-weight: bold;
      max-width: 100px;
    }
  }

  .top-bone {
    margin-bottom: -20px;
  }

  .bottom-bone {
    margin-top: -20px;
  }

  .top-bone,
  .bottom-bone {
    width: 100%;
    padding-right: 100px;

    .item-bone {
      float: left;
      margin: 0 10px;
    }

    .item-bone-children {
      // transform: skewX(45deg);
      margin: 0 20px;
      display: flex;
      justify-content: center;
      align-items: flex-end;
    }
  }

  .bottom-bone {
    bottom: 0;

    .item-bone-children {
      align-items: flex-start !important;
    }
  }

  .item_bone_children_bigBone {
    height: 100%;
    display: flex;
    align-items: center;
    flex-direction: column;
    .item_bone_children_bigBone_line {
      width: 2px;
      background-color: @bnoe-color;
    }

    .item_bone_children_bigBone_title {
      font-size: 14px;
      font-weight: bold;
      max-height: 100px;
      writing-mode: vertical-rl;
      text-orientation: upright;
      overflow-wrap: break-word;
    }
  }
}

::v-deep .el-row--flex {
  justify-content: center;
}

.metatarsus {
  width: fit-content;
  display: flex;
  align-items: flex-start;
  flex-direction: column;
  justify-content: space-between;
  margin: 10px 0;
  .metatarsus_content {
    width: fit-content;
    display: flex;
    align-items: center;
    justify-content: flex-end;
    padding: 5px 0;
  }
  .metatarsus_main {
    width: 100%;
    display: flex;
    align-items: center;
    justify-content: space-between;
    .metatarsus_line {
      flex: 1;
      height: 2px;
      background-color: @bnoe-color;
    }

    .metatarsus_title {
      width: fit-content;
      font-size: 14px;
      font-weight: bold;
      max-width: 100px;
    }
  }
}

.left_metatarsus {
  align-items: flex-end !important;
}

.metatarsus_content_item {
  width: 100%;
  .metatarsus_content_item_text {
    margin: 0 5px;
    font-weight: bold;
    border-left: 2px dashed @bnoe-color;
    max-height: 100px;
    writing-mode: vertical-rl;
    text-orientation: upright;
    overflow-wrap: break-word;
  }
}

.right_content {
  margin-left: 0 !important;
}

.left_content {
  margin-right: 10px !important;
}

.top_content {
  align-items: flex-end !important;
}

.bottom_content {
  align-items: flex-start !important;
}
</style>
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_44745401/article/details/135089818

【前端图表框架echarts】echarts小案例实现_sunalwaysonline的博客-爱代码爱编程

ECharts,是一个使用 JavaScript 实现的开源可视化库,可以流畅的运行在 PC 和移动设备上,兼容当前绝大部分浏览器(IE8/9/10/11,Chrome,Firefox,Safari等),底层依赖轻量级的矢量图形库 ZRender,提供直观,交互丰富,可高度个性化定制的数据可视化图表. 例如,现在需要比较2个同学的各学期绩点信息,使用ec

vue、jtopo绘制鱼骨图-爱代码爱编程

效果如下图所示: 一、文件目录结构: 二、绘制png的鱼头、鱼尾图片 fish_head.png:      fish_tail.png:      三、下载js文件 jquery、jtopo请到官网下载。 四、编写鱼骨图核心文件:MakFishBone.js let MakFishBone = (function (window)

使用 auto-drawing 画一个鱼骨图_xiaofei0627的博客-爱代码爱编程

使用 auto-drawing 画一个鱼骨图 auto-drawing 官方文档 基于 vue 环境 安装 yarn add auto-drawing 代码 fishbone.vue <templat

鱼骨图技能详解+13张精选模板,让你3分钟快速掌握鱼骨图分析法!-爱代码爱编程

鱼骨图看似很简单,其实有很多操作要点,它是一个非定量的工具,可以帮助我们找出引起问题的根本原因,使我们问自己:问题为什么会发生?促使人把目光聚焦于问题的原因,而非问题的症状。 接下来从鱼骨图的定义、鱼骨图的用途、鱼骨图的3种类型、鱼骨图的绘制方法和精选案例模板5个方面去展开阐述。 一、鱼骨图的定义 1953年,日本管理大师石川馨先生提

vue纯css实现动态鱼骨图组件,附图片示例-爱代码爱编程

先看示例图片吧 可以指定鱼骨根数和鱼翅的数量,具体看代码就行,默认4根鱼骨5个鱼翅   <template> <div class="fishbone"> <div class="content"> <el-row type="flex" class="top-bone">

医院不良事件报告系统源码:基于php+vue2+element+laravel8技术开发_医院安全(不良)事件报告系统源码 支持二次开发、支持源码交付-爱代码爱编程

医院不良事件报告系统源码 文末获取联系! 技术架构:前后端分离,仓储模式, 开发语言:PHP 开发工具:vscode 前端框架:vue2+element 后端框架:laravel8 数 据 库:mysql5.7 系统概述: 医院安全(不良)事件管理系统采用无责的、自愿的,能够以匿名、实名方式填报不良事件,系统包括十余种不良事

探索fishbone.js:优雅地绘制组织结构图-爱代码爱编程

探索Fishbone.js:优雅地绘制组织结构图 项目地址:https://gitcode.com/aemkei/fishbone.js 在软件开发中,可视化工具对于理解和交流复杂系统结构至关重要。今天,我们要向您推