{"remainingRequest":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\vue-loader\\lib\\index.js??vue-loader-options!D:\\jenkins\\workspace\\dataShare-fvue\\src\\components\\mindMap\\index.vue?vue&type=script&lang=js","dependencies":[{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\src\\components\\mindMap\\index.vue","mtime":1739300167374},{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\@vue\\cli-plugin-babel\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\@vue\\cli-plugin-babel\\node_modules\\thread-loader\\dist\\cjs.js","mtime":499162500000},{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\babel-loader\\lib\\index.js","mtime":456789000000},{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\@vue\\cli-plugin-babel\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"D:\\jenkins\\workspace\\dataShare-fvue\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64://
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//

import domtoimage from 'dom-to-image';
import utils from '@/utils.js';
import 'jsmind/style/jsmind.css';
import 'jsmind/es6/jsmind.draggable-node.js';
import 'jsmind/es6/jsmind.screenshot.js';
import jsMind from 'jsmind';
import API from '@/api/business/arithmeticTask.js';
import { version } from 'codemirror';
export default {
  name: 'mindMap',
  data() {
    return {
      // 任务信息
      taskData: {
        id: '',
        name: '',
        code: '',
      },
      newVersion: '',
      // 配置信息
      showSideEditor: false,
      pagination: {
        // 分页初始化数据
        page: 1,
        total: 0,
        pageSize: 20,
      },
      searchTarget: {
        type: 1,
        name: '',
      },
      addTarget: {
        topic: '',
        code: '',
        type: 1,
        desc: '',
        source: 1,
      },
      activeName: 'first',
      tableList: [
        {
          id: '1.2',
          name: 'table01',
          nameCn: '表中文名01',
        },
        {
          id: '1.3',
          name: 'table02',
          nameCn: '表中文名02',
        },
        {
          id: '1.4',
          name: 'table03',
          nameCn: '表中文名03',
        },
      ],
      fieldList: [
        {
          id: '1.2',
          name: 'name',
          nameCn: '姓名',
          type: '普通字段',
        },
        {
          id: '1.3',
          name: 'sex',
          nameCn: '性别',
          type: '枚举字典',
          dictName: 'sexDict',
          dict: [
            {
              name: '男',
              id: '1',
            },
            {
              name: '女',
              id: '2',
            },
          ],
        },
        {
          id: '1.4',
          name: 'type',
          nameCn: '类型',
          type: '多维字典',
          dictName: 'sportsDict',
          dict: [
            {
              name: '运动',
              id: '1',
              children: [
                {
                  name: '水上运动水上运动水上运动水上运动水上运动水上运动水上运动水上运动水上运动水上运动水上运动水上运动',
                  id: '11',
                  children: [
                    {
                      name: '赛艇',
                      id: '111',
                    },
                    {
                      name: '翻板',
                      id: '112',
                    },
                    {
                      name: '皮划艇',
                      id: '113',
                    },
                  ],
                },
                {
                  name: '冬季运动',
                  id: '12',
                  children: [
                    {
                      name: '冰壶',
                      id: '121',
                    },
                    {
                      name: '花样滑冰',
                      id: '122',
                    },
                    {
                      name: '短道速滑',
                      id: '123',
                    },
                  ],
                },
                {
                  name: '球类运动',
                  id: '13',
                  children: [
                    {
                      name: '篮球',
                      id: '131',
                    },
                    {
                      name: '足球',
                      id: '132',
                    },
                    {
                      name: '曲棍球',
                      id: '133',
                    },
                  ],
                },
              ],
            },
          ],
        },
      ],
      defaultProps: {
        value: 'id',
        label: 'name',
      },
      versionData: [],
      mapType: 'multiToSingle', //multiToSingle or singleToMulti
      targetData: [],
      // 脑图信息
      // 脑图数据
      jm: null,
      editor: null,
      mindData: {
        meta: {
          name: '',
          author: 'artfess',
          version: '0.0.1',
        },
        format: 'node_array',
        data: [
          {
            isroot: true,
            id: utils.uuid(),
            topic: '',
            level: 0,
            'background-color': '#409eff',
            'font-size': 24, // 字体大小
          },
        ],
      },
      // 脑图配置
      options: {
        container: 'jsmind_container', // [必选] 容器的ID
        editable: true, // [可选] 是否启用编辑
        theme: 'primary', // [可选] 主题
        support_html: true,
        log_level: 'debug', // 设置日志级别为 debug
        view: {
          engine: 'canvas', // 思维导图各节点之间线条的绘制引擎
          enable_device_pixel_ratio: true, // 按设备像素比例绘制高清思维导图
          hmargin: 100, // 思维导图距容器外框的最小水平距离
          vmargin: 50, // 思维导图距容器外框的最小垂直距离
          line_width: 2, // 思维导图线条的粗细
          line_color: '#666', // 思维导图线条的颜色
          draggable: true, // 是否允许拖动画布
          hide_scrollbars_when_draggable: true,
          node_overflow: 'hidden',
          zoom: {
            // 缩放配置
            min: 0.1, // 最小缩放比例
            max: 2.1, // 最大缩放比例
            step: 0.1, // 缩放比例步长
          },
          expander_style: 'number',
          line_style: 'curved',
          custom_line_render: function ({ ctx, start_point, end_point }) {
            ctx.beginPath();
            ctx.moveTo(start_point.x, start_point.y);
            // ctx.lineTo(toX, toY);
            ctx.bezierCurveTo(start_point.x + ((end_point.x - start_point.x) * 2) / 3, start_point.y, start_point.x, end_point.y, end_point.x, end_point.y);
            ctx.strokeStyle = '#666';
            ctx.lineWidth = 2;
            ctx.stroke();
          },
          custom_node_render: function (jm, element, node) {
            element.innerHTML = `${node.topic}<div class='tip-icon${node.data.content ? ' showTip' : ''}' data-content='${
              node.data.content || ''
            }'><i class="el-icon-s-flag"></i></div>`;
            return true; // 表示已经渲染此节点
          },
        },
        layout: {
          hspace: 100, // 节点之间的水平间距
          vspace: 20, // 节点之间的垂直间距
        },
        shortcut: {
          // 禁用快捷键
          enable: false,
        },
      },
      predefineColors: ['#1e90ff', '#67c23a', '#e6a23c', '#f56c6c', '#ff4500', '#ff8c00', '#ffd700', '#90ee90', '#00ced1', '#409eff', '#c71585'], // 预定义颜色

      structure: {
        active: 'right',
      },

      createType: '', // 添加平级or子级
      currentNode: null, // 选中节点信息
      showMenu: false, // 是否显示右键菜单栏
      isroot: false,
      menuStyle: {
        top: '',
        bottom: '',
        left: '',
        right: '',
      },
      tipStyle: { top: '', bottom: '', left: '', right: '' },
      showPopover: false,
      currentMemo: '',
      showMap: true,
      firstAdd: false,
      curVersion: '',
    };
  },
  mounted() {
    this.getTaskData();
    const that = this;
    this.mapType = sessionStorage.getItem('mapType') || 'multiToSingle';
    this.options.container = `jsmind_container_${this.mapType}`;
    // if (this.mapType === 'singleToMulti') {
    //   console.log('singleToMulti');
    //   this.options.view.custom_line_render = function ({ ctx, start_point, end_point }) {
    //     that.drawCurveArrow(ctx, end_point, start_point, 0.9);
    //     // that.drawArrow(ctx, start_point, end_point);
    //   };
    // } else {
    //   console.log('multiToSingle');
    //   this.options.view.custom_line_render = function ({ ctx, start_point, end_point }) {
    //     that.drawCurveArrow(ctx, end_point, start_point, 0.5);
    //     // that.drawArrow(ctx, end_point, start_point);
    //   };
    // }
  },
  methods: {
    initMap() {
      this.$nextTick(() => {
        console.log(this.mindData, 'container');
        this.loadMind();
      });
    },
    //------------接口--------------
    // 获取任务信息
    async getTaskData() {
      const { id, name, code, version } = this.$route.query;
      this.taskData = { id, name, code, version };
      await this.getMapData(version);
      this.initMap();
    },

    // 获取脑图数据
    getMapData(versionId = '') {
      this.initMindData();

      let data = new FormData();
      // 有版本号的时候不传任务id
      data.append('taskId', versionId ? '' : this.taskData.id);
      data.append('versionId', versionId);

      return new Promise((resolve, reject) => {
        API.getBrainMapInfo(data)
          .then(async (res) => {
            if (res) {
              if (res.length === 0) {
                this.firstAdd = true;
                this.curVersion = await this.getNewVersion();
              } else {
                this.mindData.data = this.formatData(res);
                this.curVersion = this.getVersionCode(this.mindData.data[0]);
                this.firstAdd = false;
              }
              resolve();
            } else {
              reject(res.message);
            }
          })
          .catch((err) => {
            reject(err);
          });
      });
    },
    getVersionCode(data) {
      if (data.taskVersionName) {
        return data.taskVersionName.split('-')[1];
      } else {
        return '';
      }
    },
    initMindData() {
      this.mindData.meta.name = this.taskData.name;
      this.mindData.data[0].topic = this.taskData.name;
      this.mindData.data[0].id = utils.uuid();
    },

    formatData(data) {
      const newData = [];
      data.forEach((item) => {
        const { id, name: topic, parentId, level, relationContent, content, background, fontColor, fontSize, fontStyle, fontWetight, ...fields } = item;
        const newItem = {
          id,
          parentid: parentId,
          topic,
          level,
          relationContent,
          content,
          'background-color': background,
          'foreground-color': fontColor,
          'font-weight': fontWetight,
          'font-style': fontStyle,
          'font-size': fontSize,
          ...fields,
        };
        level === 0 ? (newItem.isroot = true) : null;
        newData.push(newItem);
      });
      return newData;
    },
    // 保存
    async handleSave(command) {
      this.newVersion = await this.getNewVersion();
      this.saveMapData(command);
    },
    saveMapData(command) {
      debugger;
      this.getMindData();
      const reasonList = this.mindData.data.map((item) => {
        return {
          id: item.id,
          name: item.topic,
          level: item.level,
          relationContent: item.relationContent,
          content: item.content,
          background: item['background-color'],
          fontColor: item['foreground-color'],
          fontSize: item['font-size'],
          fontWetight: item['font-weight'],
          fontStyle: item['font-style'],
          taskId: this.taskData.id,
          parentId: item.parentid,
          sn: item.sn,
        };
      });
      const newData = {
        reasonList,
        saveAS: command === 'saveAs' ? 1 : 0,
        version:
          this.firstAdd || command === 'saveAs'
            ? {
                type: this.mapType === 'multiToSingle' ? 2 : 1,
                versionCode: this.newVersion,
                versionName: this.taskData.code + '-' + this.newVersion,
                taskId: this.taskData.id,
                taskName: this.taskData.name,
                taskCode: this.taskData.code,
              }
            : undefined,
      };

      API.saveBrainMapInfo(newData)
        .then((res) => {
          if (res.state) {
            this.$message.success('保存成功');
          } else {
            this.$message.error('保存失败');
          }
        })
        .catch((err) => {
          console.log(err || '保存失败');
        });
    },
    // 获取新版本号
    getNewVersion() {
      const { id } = this.taskData;
      const data = new FormData();
      data.append('taskId', id);
      return new Promise((resolve, reject) => {
        API.getNewVersion(data)
          .then((res) => {
            if (res.value && res.state) {
              resolve(res.value);
            } else {
              reject(res.message);
            }
          })
          .catch((err) => {
            reject(err);
          });
      });
    },

    // 获取版本列表
    getVersionList() {
      const { id } = this.taskData;
      const query = {
        pageBean: {
          pageSize: -1,
        },
        querys: [
          {
            group: 'main',
            operation: 'EQUAL',
            property: 'taskId',
            relation: 'AND',
            value: id,
          },
        ],
      };
      API.getInferenceVersion(query)
        .then((res) => {
          this.versionData = res.rows || [];
        })
        .catch((error) => {
          this.$message.error(error || '获取版本信息失败！');
        });
    },
    // 获取指标列表
    getTargetList(page) {
      page ? (this.pagination.page = page) : null;
      const query = {
        pageBean: this.pagination,
        querys: [
          {
            group: 'main',
            operation: 'EQUAL',
            parentGroup: '',
            property: 'level',
            relation: 'AND',
            value: this.searchTarget.type,
          },
          {
            group: 'main',
            operation: 'LIKE',
            parentGroup: '',
            property: 'name',
            relation: 'AND',
            value: this.searchTarget.name,
          },
        ],
        sorter: [
          {
            direction: 'ASC',
            property: 'code',
          },
        ],
      };
      API.getTargetList(query)
        .then((res) => {
          this.targetData = res.rows || [];
          this.pagination.total = res.total || 0;
        })
        .catch((error) => {
          this.$message.error(error || '获取指标信息失败！');
        });
    },
    //--------------------------
    loadMind() {
      this.getMindData();
      this.toggleStucture(this.mapType);
      this.jm = null;
      this.jm = new jsMind(this.options);
      console.log(this.options.container, this.mindData, 'container');
      this.jm.show(this.mindData);
      this.editor = null;
      this.editor = this.jm.view.e_editor;
      this.init();
      this.mouseWheel();
      this.mouseDrag();
    },

    handleTableChange(val) {
      const target = this.tableList.filter((i) => i.id === val)[0];
      this.addTarget.tableNameCn = target?.nameCn || '';
    },
    handleFieldChange(val) {
      const target = this.fieldList.filter((i) => i.id === val)[0];
      this.addTarget.fieldNameCn = target?.nameCn || '';
      this.addTarget.fieldType = target?.type || '';
      this.addTarget.fieldDict = target?.dict || [];
      this.addTarget.fieldDictName = target?.dictName || [];
    },
    handleCurrentChange(val) {
      this.pagination.page = val;
      this.getTargetList();
    },
    // 类型切换
    changeMapType(type) {
      this.mapType = type;
      sessionStorage.setItem('mapType', type);
      location.reload();
    },
    handleChangeMapType(type) {
      if (this.mapType !== type) {
        this.$confirm('切换推理类型后需要刷新页面，请先保存数据，是否确认切换推理类型？')
          .then(() => {
            this.changeMapTypeFn(type);
          })
          .catch(() => {});
      } else {
        let typeName = this.mapType === 'singleToMulti' ? '一推多' : '多推一';
        this.$message.warning('当前推理类型已是 【' + typeName + '】，无需切换');
      }
    },
    changeMapTypeFn(type) {
      this.mapType = type;
      sessionStorage.setItem('mapType', type);
      location.reload();
    },

    drawArrow(ctx, start_point, end_point) {
      let fromX, fromY, toX, toY;
      fromX = end_point.x;
      fromY = end_point.y;
      toX = start_point.x;
      toY = start_point.y;

      var theta = 30,
        headlen = 12,
        width = 2,
        color = '#666';
      // 计算直线中点坐标

      var midX = (fromX + toX) / 2;
      var midY = (fromY + toY) / 2;

      // 计算箭头起点和终点的坐标
      var angle = (Math.atan2(fromY - toY, fromX - toX) * 180) / Math.PI;
      var angle1 = ((angle + theta) * Math.PI) / 180;
      var angle2 = ((angle - theta) * Math.PI) / 180;

      var topX = headlen * Math.cos(angle1);
      var topY = headlen * Math.sin(angle1);
      var botX = headlen * Math.cos(angle2);
      var botY = headlen * Math.sin(angle2);

      var arrowX, arrowY;

      ctx.beginPath();
      ctx.moveTo(fromX, fromY);
      ctx.lineTo(toX, toY);
      //ctx.bezierCurveTo(start_point.x + ((end_point.x - start_point.x) * 2) / 3, start_point.y, start_point.x, end_point.y, end_point.x, end_point.y);

      arrowX = midX + topX;
      arrowY = midY + topY;
      ctx.moveTo(arrowX, arrowY);
      ctx.lineTo(midX, midY);

      arrowX = midX + botX;
      arrowY = midY + botY;
      ctx.lineTo(arrowX, arrowY);
      ctx.strokeStyle = color;
      ctx.lineWidth = width;
      ctx.stroke();
    },

    drawCurveArrow(ctx, startPoint, endPoint, t) {
      const fromX = startPoint.x;
      const fromY = startPoint.y;
      const toX = endPoint.x;
      const toY = endPoint.y;

      const controlPoint1X = fromX + ((toX - fromX) * 2) / 3;
      const controlPoint1Y = fromY;
      const controlPoint2X = fromX;
      const controlPoint2Y = toY;

      const arrowTheta = 45; // 箭头角度（度）
      const arrowHeadLength = 12; // 箭头长度
      const lineWidth = 2; // 线条宽度
      const strokeColor = '#666'; // 线条颜色

      // 计算贝塞尔曲线的4/1处的点（t = 0.75）
      //   const t = 0.5;
      const curvePointX =
        Math.pow(1 - t, 3) * fromX + 3 * Math.pow(1 - t, 2) * t * controlPoint1X + 3 * (1 - t) * Math.pow(t, 2) * controlPoint2X + Math.pow(t, 3) * toX;
      const curvePointY =
        Math.pow(1 - t, 3) * fromY + 3 * Math.pow(1 - t, 2) * t * controlPoint1Y + 3 * (1 - t) * Math.pow(t, 2) * controlPoint2Y + Math.pow(t, 3) * toY;

      // 计算曲线在curvePoint处的方向（导数）
      const derivativeX =
        -3 * Math.pow(1 - t, 2) * fromX +
        3 * (3 * Math.pow(1 - t, 2) - 6 * (1 - t) * t) * controlPoint1X +
        3 * (-3 * (1 - t) * t + 2 * t * t) * controlPoint2X +
        3 * Math.pow(t, 2) * toX;
      const derivativeY =
        -3 * Math.pow(1 - t, 2) * fromY +
        3 * (3 * Math.pow(1 - t, 2) - 6 * (1 - t) * t) * controlPoint1Y +
        3 * (-3 * (1 - t) * t + 2 * t * t) * controlPoint2Y +
        3 * Math.pow(t, 2) * toY;

      // 计算箭头的角度
      const angle = Math.atan2(derivativeY, derivativeX);
      const angle1 = ((angle + arrowTheta) * Math.PI) / 180;
      const angle2 = ((angle - arrowTheta) * Math.PI) / 180;
      //   const angle1 = angle + (arrowTheta * Math.PI) / 180;
      //   const angle2 = angle - (arrowTheta * Math.PI) / 180;
      //   console.log(angle, angle1, angle2, 'angle');

      // 计算箭头两个分支的终点坐标
      const topX = toX + arrowHeadLength * Math.cos(angle1);
      const topY = toY + arrowHeadLength * Math.sin(angle1);
      const botX = toX + arrowHeadLength * Math.cos(angle2);
      const botY = toY + arrowHeadLength * Math.sin(angle2);

      ctx.beginPath();
      ctx.moveTo(fromX, fromY);
      ctx.bezierCurveTo(controlPoint1X, controlPoint1Y, controlPoint2X, controlPoint2Y, toX, toY);
      ctx.lineTo(topX, topY);
      ctx.moveTo(toX, toY);
      ctx.lineTo(botX, botY);
      ctx.strokeStyle = strokeColor;
      ctx.lineWidth = lineWidth;
      ctx.stroke();
    },
    drawCurveArrow(ctx, startPoint, endPoint) {
      const fromX = startPoint.x;
      const fromY = startPoint.y;
      const toX = endPoint.x;
      const toY = endPoint.y;

      const controlPoint1X = fromX + ((toX - fromX) * 2) / 3;
      const controlPoint1Y = fromY;
      const controlPoint2X = fromX;
      const controlPoint2Y = toY;

      const arrowTheta = 30; // 箭头角度（度）
      const arrowHeadLength = 12; // 箭头长度
      const lineWidth = 2; // 线条宽度
      const strokeColor = '#666'; // 线条颜色

      // 计算贝塞尔曲线的结束点（t = 1）
      const curvePointX = toX;
      const curvePointY = toY;

      // 计算曲线在结束点处的方向（导数，t = 1）
      const derivativeX = 2 * (toX - controlPoint2X);
      const derivativeY = 2 * (toY - controlPoint2Y);

      // 计算箭头的角度
      const angle = Math.atan2(derivativeY, derivativeX);
      const angle1 = angle + (arrowTheta * Math.PI) / 180;
      const angle2 = angle - (arrowTheta * Math.PI) / 180;

      // 计算箭头两个分支的终点坐标
      const topX = toX - arrowHeadLength * Math.cos(angle1);
      const topY = toY - arrowHeadLength * Math.sin(angle1);
      const botX = toX - arrowHeadLength * Math.cos(angle2);
      const botY = toY - arrowHeadLength * Math.sin(angle2);

      ctx.beginPath();
      ctx.moveTo(fromX, fromY);
      ctx.bezierCurveTo(controlPoint1X, controlPoint1Y, controlPoint2X, controlPoint2Y, toX, toY);
      ctx.lineTo(topX, topY);
      ctx.moveTo(toX, toY);
      ctx.lineTo(botX, botY);
      ctx.strokeStyle = strokeColor;
      ctx.lineWidth = lineWidth;
      ctx.stroke();
    },
    // tab切换
    handleClickTab(tab) {
      if (tab.name === 'fifth') {
        this.getVersionList();
      } else if (tab.name === 'second') {
        this.getTargetList(1);
      }
    },
    // 配置编辑显示隐藏
    toggleSideEditor(type) {
      this.showSideEditor = type === undefined ? !this.showSideEditor : type;
      setTimeout(() => {
        this.reset();
      }, 200);
    },
    // 缩小
    zoomOut() {
      this.jm.view.zoom_out();
    },
    // 放大
    zoomIn() {
      this.jm.view.zoom_in();
    },

    // 重置视图
    reset() {
      this.jm.resize();
    },

    exportImage() {
      let jm = jsMind.current;
      jm.shoot();
    },

    loopArrayData(list, callback) {
      list.forEach((item, i) => {
        callback(item, i);
      });
    },
    // 循环树结构
    loopTreeData(list, callback) {
      (function doOneFloor(list) {
        if (Array.isArray(list)) {
          for (let i = 0; i < list.length; i++) {
            const item = list[i];
            callback(item, i);
            if (item.children && item.children.length > 0) {
              doOneFloor(item.children);
            }
          }
        }
      })(list);
    },

    // 编辑节点

    // 禁用双击编辑
    stopDBclick() {
      const jmnodes = document.getElementsByTagName('jmnode');
      Object.values(jmnodes).forEach((jmnode) => {
        jmnode.ondblclick = function (e) {
          e.preventDefault();
          e.stopPropagation();
        };
      });
    },

    showTip(event) {
      if (this.showPopover) {
        return;
      }
      this.currentMemo = event?.fromElement?.children[0]?.dataset?.content || '';
      const el = document.querySelector('.common-tip');
      const width = parseFloat(window.getComputedStyle(el).width);
      const height = parseFloat(window.getComputedStyle(el).height) * 3 + 12;
      const windowHeight = window.innerHeight;
      const windowWidth = window.innerWidth;
      // 极限位置 避免越界
      if (event.clientY + height > windowHeight) {
        this.tipStyle.left = event.clientX + 'px';
        this.tipStyle.top = 'unset';
        this.tipStyle.bottom = 0;
      } else if (event.clientX + width > windowWidth) {
        this.tipStyle.top = event.clientY + 'px';
        this.tipStyle.left = 'unset';
        this.tipStyle.right = 0;
      } else {
        this.tipStyle.left = event.clientX + 'px';
        this.tipStyle.top = event.clientY + 'px';
        this.tipStyle.bottom = 'unset';
      }
      if (this.currentMemo) {
        this.showPopover = true;
      }
    },

    hideTip() {
      this.currentMemo = '';
      this.showPopover = false;
    },

    // 备注绑定点击事件
    tipsClick() {
      const tips = document.getElementsByClassName('tip-icon showTip');
      Object.values(tips).forEach((tip) => {
        tip.removeEventListener('mouseover', this.showTip);
        tip.addEventListener('mouseover', this.showTip);
      });
      Object.values(tips).forEach((tip) => {
        tip.removeEventListener('mouseleave', this.hideTip);
        tip.addEventListener('mouseleave', this.hideTip);
      });
    },

    // 右键点击新增
    rightClickMenu() {
      // 右键菜单
      this.jm.view.add_event(this.editor, 'contextmenu', (e) => {
        const selectedNode = this.jm.get_selected_node();

        if (selectedNode) {
          this.isroot = selectedNode.id === 'root';
          e.preventDefault();
          const el = document.querySelector('.context-menu .el-menu-item');
          let menu = this.$refs.context.$el;

          //   const width = parseFloat(window.getComputedStyle(el).width);
          //   const height = parseFloat(window.getComputedStyle(el).height) * 3 + 12;
          const width = parseFloat(menu.offsetWidth);
          const height = parseFloat(menu.offsetHeight) * 2 + 12;
          const windowHeight = window.innerHeight;
          const windowWidth = window.innerWidth;

          // 极限位置 避免越界
          if (e.clientY + height > windowHeight) {
            this.menuStyle.left = e.clientX + 'px';
            this.menuStyle.top = 'unset';
            this.menuStyle.bottom = 0;
          } else if (e.clientX + width > windowWidth) {
            this.menuStyle.top = e.clientY + 'px';
            this.menuStyle.left = 'unset';
            this.menuStyle.right = 0;
          } else {
            this.menuStyle.left = e.clientX + 20 + 'px';
            this.menuStyle.top = e.clientY + 20 + 'px';
            this.menuStyle.bottom = 'unset';
          }
          this.showMenu = true;
        } else {
          this.showMenu = false;
        }
      });
    },

    // 初始化配置
    init() {
      this.jm.expand_all();

      this.stopDBclick();

      this.tipsClick();

      this.rightClickMenu();
    },
    // 获取选中标签的 ID
    get_selected_nodeid() {
      const selectedNode = this.jm.get_selected_node();
      if (selectedNode) {
        return selectedNode.id;
      } else {
        return null;
      }
    },

    // 切换思维导图结构
    toggleStucture(type) {
      if (this.structure.active === type) return;
      this.structure.active = type;
      switch (type) {
        case 'side':
          // 两边分布
          this.loopArrayData(this.mindData.data, (item, i) => {
            item.direction = i % 2 ? 'left' : 'right';
          });
          break;

        case 'multiToSingle':
          // 向左分布
          this.loopArrayData(this.mindData.data, (item) => {
            item.direction = 'left';
          });
          break;

        case 'singleToMulti':
          // 向右分布
          this.loopArrayData(this.mindData.data, (item) => {
            item.direction = 'right';
          });
          break;

        default:
          break;
      }
    },

    // 拖拽
    handleDrop(draggingNode, dropNode) {
      // 前一个兄弟节点
      const prevNode = this.jm.find_node_before(dropNode);
      // 获取移动后的node
      const dragForm = {
        modelId: '',
        treeNum: !prevNode ? draggingNode : prevNode.id,
        thisTreeNum: dropNode,
      };
      console.log('dragForm', dragForm);

      // TODO 调接口
    },
    // 单击重置选中背景颜色
    nodeClick() {
      this.showMenu = false;
      const selectedNode = this.jm.get_selected_node();
      if (selectedNode) {
        console.log('selectedNode', selectedNode);

        this.currentNode = selectedNode;
        this.addTarget.topic = this.currentNode.topic || '';
        this.toggleSideEditor(true);
      }
    },
    // 阻止冒泡
    nodePreventClick() {},

    // 编辑节点 ---------------

    // 改变topic
    changeNodeTopic(val) {
      if (val) {
        this.jm.update_node(this.currentNode.id, val);
        this.tipsClick();
        this.stopDBclick();
      } else {
        this.$message.error('请输入节点标题');
      }
    },
    changeFontWeight() {
      const fontWeight = this.currentNode?.data?.['font-weight'] === 'bold' ? 'normal' : 'bold';
      this.jm.set_node_font_style(this.currentNode.id, null, fontWeight, null);
      let data = this.currentNode.data || {};
      data['font-weight'] = fontWeight;
      this.$set(this.currentNode, 'data', data);
      console.log(this.currentNode);
      this.$forceUpdate();
    },
    changeFontStyle() {
      const fontStyle = this.currentNode?.data?.['font-style'] === 'italic' ? 'normal' : 'italic';
      this.jm.set_node_font_style(this.currentNode.id, null, null, fontStyle);
      let data = this.currentNode.data || {};
      data['font-style'] = fontStyle;
      this.$set(this.currentNode, 'data', data);
      this.$forceUpdate();
    },
    changeTextSize(val) {
      if (val) {
        this.jm.set_node_font_style(this.currentNode.id, val, null, null);
      } else {
        this.$message.error('请选择字体大小');
      }
    },
    // 改变文字颜色
    changeTextColor(val) {
      if (val) {
        this.jm.set_node_color(this.currentNode.id, null, val);
      } else {
        this.$message.error('请选择文字颜色');
      }
    },
    // 改变背景颜色
    changeNodeColor(val) {
      if (val) {
        this.jm.set_node_color(this.currentNode.id, val, null);
      } else {
        this.$message.error('请选择背景颜色');
      }
    },

    // 编辑节点计算公式
    changeNodeMemo(val, id) {
      if (val) {
        this.loopArrayData(this.mindData.data, (item, i) => {
          if (item.id === id) {
            item.content = val;
          }
        });
        this.jm.show(this.mindData);
        this.stopDBclick();
        this.tipsClick();
      } else {
        this.$message.error('请输入节点计算公式');
      }
    },

    changeNodeRelation(val, id) {
      if (val) {
        this.loopArrayData(this.mindData.data, (item, i) => {
          if (item.id === id) {
            item.relationContent = val;
          }
        });
        this.jm.show(this.mindData);
        this.stopDBclick();
        this.tipsClick();
      } else {
        this.$message.error('请输入关系描述');
      }
    },

    // 导入子级
    importChild() {},

    // 插入子级
    addChild() {
      const selectedNode = this.jm.get_selected_node();
      if (selectedNode) {
        let id = utils.uuid();
        this.jm.add_node(selectedNode, id, '新增节点', {
          level: selectedNode.data.level + 1,
          'font-size': 16,
          'background-color':
            selectedNode.data.level + 1 > this.predefineColors.length ? this.predefineColors[1] : this.predefineColors[selectedNode.data.level + 1],
        });
        this.getMindData();
        this.jm.select_node(id);

        this.stopDBclick();
        this.nodeClick();
      } else {
        this.$message.error('请选择父节点');
      }

      this.showMenu = false;
    },

    addChildByTarget(row) {
      const selectedNode = this.jm.get_selected_node();
      if (selectedNode) {
        let id = utils.uuid();
        this.jm.add_node(selectedNode, id, row.name, {
          level: selectedNode.data.level + 1,
          code: row.code,
          indexType: row.level,
          'font-size': 16,
          'background-color':
            selectedNode.data.level + 1 > this.predefineColors.length ? this.predefineColors[1] : this.predefineColors[selectedNode.data.level + 1],
        });
        this.getMindData();
        this.jm.select_node(id);
        this.stopDBclick();
        this.nodeClick();
      } else {
        this.$message.error('请选择父节点');
      }
    },

    // 删除卡片
    delCard() {
      const selectedNode = this.jm.get_selected_node();
      if (selectedNode.data) {
        this.jm.remove_node(selectedNode.id);
        this.getMindData();
      } else {
        this.$message.error('请选择节点');
      }
      this.showMenu = false;
    },

    // 获取数据
    getMindData() {
      if (this.jm) {
        this.mindData = this.jm.get_data('node_array');
        console.log(this.mindData, 'this.dataUpdata');
      }
    },

    // 鼠标滚轮放大缩小
    mouseWheel() {
      if (document.addEventListener) {
        document.addEventListener('domMouseScroll', this.scrollFunc, false);
      }
      this.$refs.container.onmousewheel = this.scrollFunc;
    },

    // 滚轮缩放
    scrollFunc(e) {
      this.showMenu = false;
      e = e || window.event;
      if (e.wheelDelta) {
        if (e.wheelDelta > 0) {
          this.zoomIn();
        } else {
          this.zoomOut();
        }
      } else if (e.detail) {
        if (e.detail > 0) {
          this.zoomIn();
        } else {
          this.zoomOut();
        }
      }
      e.preventDefault();

      this.jm.resize();
    },

    // 鼠标拖拽
    mouseDrag() {
      // 里层
      const el = document.querySelector('.jsmind-inner');
      // 选中节点
      let selected;

      el.onmousedown = (ev) => {
        // 选中节点
        selected = this.jm.get_selected_node();
        // 标识 是否拖拽节点 避免冲突
        this.dragNodeFlag = !!selected;

        const disX = ev.clientX;
        const disY = ev.clientY;
        const originalScrollLeft = el.scrollLeft;
        const originalScrollTop = el.scrollTop;
        const originalScrollBehavior = el.style['scroll-behavior'];
        const originalPointerEvents = el.style['pointer-events'];
        // auto: 默认值，表示滚动框立即滚动到指定位置。
        el.style['scroll-behavior'] = 'auto';
        // 鼠标移动事件是监听的整个document，这样可以使鼠标能够在元素外部移动的时候也能实现拖动
        document.onmousemove = (ev) => {
          if (this.dragNodeFlag) return;
          this.drag = false;
          ev.preventDefault();
          // 计算拖拽的偏移距离
          const distanceX = ev.clientX - disX;
          const distanceY = ev.clientY - disY;

          el.scrollTo(originalScrollLeft - distanceX, originalScrollTop - distanceY);

          // 在鼠标拖动的时候将点击事件屏蔽掉
          el.style['pointer-events'] = 'none';
          el.style.cursor = 'grabbing';
        };
        document.onmouseup = () => {
          if (!this.dragNodeFlag) {
            el.style['scroll-behavior'] = originalScrollBehavior;
            el.style['pointer-events'] = originalPointerEvents;
            el.style.cursor = 'grab';
          }
          document.onmousemove = document.onmouseup = null;
        };
      };
    },

    save() {},
    handleVersionChange(row) {
      const { id, name, code } = this.taskData;
      // 打开新的页面 到 href 这个页面
      const href = this.$router.resolve({
        path: '/arithmeticManage/arithmeticTask/mindMapPage',
        query: { id, name, code, version: row.id },
      }).href;
      window.open(href, '_blank');
    },
    tableRowClassName({ row, rowIndex }) {
      if (row.versionCode === this.curVersion) {
        return 'current-row';
      }
      return '';
    },
    relationSearch() {
      // 选中节点
      let { id } = this.jm.get_selected_node();
      // 打开新的页面 到 href 这个页面
      const href = this.$router.resolve({
        path: '/arithmeticManage/arithmeticTask/targetRelation',
        query: { id },
      }).href;
      window.open(href, '_blank');
    },
  },

  beforeDestroy() {
    document.removeEventListener('domMouseScroll', this.scrollFunc, false);
  },
};
"},null]}