js树结构所有叶子节点的值相加等于父节点的值(2)?

这是默认情况
image.png

麻烦根据我标的序号看下图,很重要
image.png

园区的count值也更新了,就是集团的没更新

=====

还有一个问题,就是当输入一个值的时候,所有变更过的地方都要显示一个红色的差值

image.png

需求是:输入框的值变动之后,所有值更新过的地方包括当前输入框都要与初始值计算差值,并且在下方显示+n或者-n,所以需要把初始化的数据存起来,因为每次都是跟初始化的数据对比,而不是上一次

然后我想就是初始化数据的时候把输入框的值都存起来,然后当输入的时候,计算之前输入框的差值,因为输入框+2的话其他地方也是+2,这个应该错不了

就是不知道怎么找到数据更新过的地方并且显示一个+n或者-n

======

这里是代码,麻烦帮忙修改一下


例子1
https://element-plus.run/#eyJ...

例子2
https://element-plus.run/#eyJ...

两个案例可以看看哪个好改一些,谢谢

阅读 1.7k
1 个回答

calc 的逻辑混得比较多,我做了一个拆分,把不同的逻辑细节拆出去了。

  • calcNode 专门用来计算某个节点内的计算数据(也就是几个 count)
  • calcParent 专门用来找到父节点并进行计算
  • calc 保留原来的接口,它会先计算当前节点再计算父节点

另外,getDiffVal 的本意应该是获取有变化的值。根据模板里的 type="danger",而且没加条件,那说明这里是没有判断值是否存在变化的。只要动了 input 就一定会显示,这也是个 BUG。改过之后的 getDiffVal 会判断值是否变化,如果变化返回原值,否则返回 "",这样 danger 才不会渲染出来。

最后是那个 map,这里我猜你是想保留原值,但是不知道为什么要在 change 里去计算它。原值保留着一直不变,只需要在 getDiffVal 的时候取出来用于比较就行。所以我重写了生成 map 的逻辑,并且改名为 originValues

其他细节不记得了,看 Demo 吧。

https://element-plus.run/#{"App.vue":"<template>\n  <ul>\n    <li>父节点的值是子节点值的和，不含孙子节点，这个目前是对的，以下三个有问题</li>\n    <li>count2 =num1 +num2</li>\n    <li>count3=num3+num4</li>\n    <li>count1 = count2+count3</li>\n  </ul>\n  <el-table :data=\"tableData.value\" default-expand-all row-key=\"id\">\n    <el-table-column prop=\"index\" label=\"序号\" type=\"index\" :index=\"indexMethod\" />\n    <el-table-column prop=\"title\" label=\"单位名称\" width=\"150\"></el-table-column>\n    <el-table-column prop=\"count1\" label=\"研究开发撞向count1\"></el-table-column>\n    <el-table-column prop=\"count2\" label=\"研究开发撞向count2\"></el-table-column>\n    <el-table-column prop=\"count3\" label=\"研究开发撞向count3\"></el-table-column>\n    <el-table-column label=\"总部管理成本num1\">\n      <template #default=\"scope\">\n        <p v-if=\"scope.row.children\">{{ scope.row.num1 }}</p>\n        <p v-else>\n          <el-input v-model.number=\"scope.row.num1\" clearable @input=\"change(scope, 'num1')\"></el-input>\n        </p>\n        <el-link type=\"danger\">{{ getDiffVal(scope.row.num1, scope.row, 'num1') }}</el-link>\n      </template>\n    </el-table-column>\n\n    <el-table-column label=\"总部管理成本num2\">\n      <template #default=\"scope\">\n        <p v-if=\"scope.row.children\">{{ scope.row.num2 }}</p>\n        <p v-else>\n          <el-input v-model.number=\"scope.row.num2\" clearable @input=\"change(scope, 'num2')\"></el-input>\n        </p>\n      </template>\n    </el-table-column>\n    <el-table-column label=\"总部管理成本num3\">\n      <template #default=\"scope\">\n        <p v-if=\"scope.row.children\">{{ scope.row.num3 }}</p>\n        <p v-else>\n          <el-input v-model.number=\"scope.row.num3\" clearable @input=\"change(scope, 'num3')\"></el-input>\n        </p>\n      </template>\n    </el-table-column>\n    <el-table-column label=\"总部管理成本num4\">\n      <template #default=\"scope\">\n        <p v-if=\"scope.row.children\">{{ scope.row.num4 }}</p>\n        <p v-else>\n          <el-input v-model.number=\"scope.row.num4\" clearable @input=\"change(scope, 'num4')\"></el-input>\n        </p>\n      </template>\n    </el-table-column>\n  </el-table>\n</template>\n\n<script lang=\"ts\" setup>\nimport { reactive } from 'vue'\nimport { data } from \"./data.js\";\n\nconst indexMethod = (index: number) => {\n  return index + 1\n}\n\nconst change = (scope: any, field: string) => {\n  const { row } = scope\n  calc(row, field)\n}\n\n// 遍历所有节点，先把旧值存下来\n// 实际只需要存 num，因为 num 变化一定会引起 count 变化\nconst originValues = (data => {\n    const map = {} as any;\n    const keys = Array.from({ length: 4 }, (_, i) => `num${i + 1}`);\n    const nodes = [...data.value] as { id: number, [key: string]: any }[];\n    while (nodes.length) {\n        const it = nodes.shift()!;\n        map[it.id] = Object.fromEntries(keys.map(key => [key, it[key]]));\n        if (it.children?.length) {\n            nodes.push(...it.children);\n        }\n    }\n    return map;\n})(data);\n\n// 现在如果值和原值相同也会是红的，\n// 逻辑应该是：比较，如果相同返回空字符串（不渲染）\n// 如果不同返回原值（由模板渲染成红色）\nconst getDiffVal = (v: number, row: any, field: string) => {\n    const value = originValues[row.id]?.[field];\n    return value === v ? \"\" : value;\n};\n\n// find 的目的是根据 ID 找到节点，所以，如果找到了返回该节点，\n// 如果没找到返回 undefined，不需要使用数组，也不需要输出参数\nconst findRow = (data: any[], id: number) => {\n    if (!Array.isArray(data)) { return; }\n    let found = data.find(it => it.id === id);\n\n    if (found) {\n        return found;\n    }\n\n    for (const it of data) {\n        if (!it.children?.length) { continue; }\n        found = findRow(it.children, id);\n        if (found) { return found; }\n    }\n    // return undefined;\n};\n\n/**\n * 计算某个节点的 count。这部分逻辑相对独立，可以独立成函数来处理\n * @param row\n */\nconst calcNode = (row: any) => {\n    row.count2 = row.num1 + row.num2;\n    row.count3 = row.num3 + row.num4;\n    row.count1 = row.count2 + row.count3;\n};\n\n/**\n * 递归修改父节点的值\n * @param row\n * @param field\n */\nconst calcParent = (row: any, field: string) => {\n    const { parentId } = row;\n    const parentRow = findRow(data.value, parentId);\n    if (!parentRow) { return; }\n\n    parentRow[field] = parentRow.children?.reduce((s: number, it: any) => s + it[field], 0);\n    calcNode(parentRow);\n    calcParent(parentRow, field);\n};\n\n// 在 calc 中把处理当前节点和处理父节点的逻辑独立开来\nconst calc = (row: any, field: string) => {\n    calcNode(row);\n    calcParent(row, field);\n};\n\nconst tableData = reactive(data);\n</script>\n","import_map.json":"{\n  \"imports\": {}\n}","data.js":"export const data = {\n  value: [\n    {\n      id: 1,\n      title: '园区1',\n      count1: 24,\n      count2: 12,\n      count3: 12,\n      num1: 6,\n      num2: 6,\n      num3: 6,\n      num4: 6,\n      children: [\n        {\n          id: 11,\n          parentId: 1,\n          title: '集团1',\n          count1: 12,\n          count2: 6,\n          count3: 6,\n          num1: 3,\n          num2: 3,\n          num3: 3,\n          num4: 3,\n          children: [\n            {\n              id: 111,\n              parentId: 11,\n              title: '公司111',\n              count1: 4,\n              count2: 2,\n              count3: 2,\n              num1: 1,\n              num2: 1,\n              num3: 1,\n              num4: 1,\n            },\n            {\n              id: 112,\n              parentId: 11,\n              title: '公司112',\n              count1: 8,\n              count2: 4,\n              count3: 4,\n              num1: 2,\n              num2: 2,\n              num3: 2,\n              num4: 2,\n            },\n          ],\n        },\n        {\n          id: 12,\n          parentId: 1,\n          title: '集团2',\n          count1: 12,\n          count2: 6,\n          count3: 6,\n          num1: 3,\n          num2: 3,\n          num3: 3,\n          num4: 3,\n          children: [\n            {\n              id: 121,\n              parentId: 12,\n              title: '公司1',\n              count1: 4,\n              count2: 2,\n              count3: 2,\n              num1: 1,\n              num2: 1,\n              num3: 1,\n              num4: 1,\n            },\n            {\n              id: 122,\n              parentId: 12,\n              title: '公司2',\n              count1: 8,\n              count2: 4,\n              count3: 4,\n              num1: 2,\n              num2: 2,\n              num3: 2,\n              num4: 2,\n            },\n          ],\n        },\n      ],\n    },\n  ],\n};","_o":{}}
推荐问题
logo
Microsoft
子站问答
访问
宣传栏