Рендеринг JSON объекта в разные столбцы в Angular? - PullRequest
1 голос
/ 07 января 2020

Сценарий для моего mat-table :

<table mat-table....>

  ......

  <ng-container matColumnDef="tag">
    <th mat-header-cell *matHeaderCellDef>Tag</th>
    <td mat-cell *matCellDef="let element">{{element.tags}}</td>
  </ng-container>

  <ng-container matColumnDef="tag_occurence">
    <th mat-header-cell *matHeaderCellDef>Tag Occurence</th>
    <td mat-cell *matCellDef="let element"></td>
  </ng-container>

  <tr mat-header-row *matHeaderRowDef="solutionInfoColumns; sticky:true"></tr>
  <tr mat-row *matRowDef="let row; columns: solutionInfoColumns;"></tr>

</table>

Мой сервис имеет следующий скрипт:

getSolutionInfo(){
    let url = this.baseurl + .... + "/solutionInfo";

    return this.http.get<any>(url).subscribe(res => {
      res.forEach(_entry => {
        _entry.tags = JSON.stringify(_entry.tags);
      });
      this.datasetSubject.next(res);
    });

  }

Теперь мой таблица выглядит так:

My_Table

Но я хочу, чтобы таблица выглядела так:

Desired_Table

Я упомянул это , но не смог найти решение. Может ли кто-нибудь помочь мне с этим?

Данные, которые должны быть предоставлены:

[
    {
        "solution_name": "TW",
        "total_images": 360,
        "tags": {
            "TR": 1399,
            "NOTCH": 355,
            "DR": 348,
            "DC": 349,
            "DL": 349,
            "TW": 1396,
            "TL": 1399,
            "DT": 348
        },
        "tagged_images": 0
    },
    {
        "solution_name": "TestSolution",
        "total_images": 0,
        "tags": {},
        "tagged_images": 0
    }
]

Ответы [ 2 ]

1 голос
/ 07 января 2020

Попробуйте код ниже

В файле AppComponent.Ts

import { Component } from '@angular/core';

@Component({
  selector: 'my-app',
  templateUrl: './app.component.html',
  styleUrls: ['./app.component.css']
})
export class AppComponent {

  dataSource = [];
  displayedColumns = ['solution_name', 'total_images', 'tagged_images', 'tags','tag_occurence'];
  spans = [];
  tempRowId = null;
  tempRowCount = null;

   orgD = [];

   myData = [
    {
        "solution_name": "TW",
        "total_images": 360,
        "tags": {
            "TR": 1399,
            "NOTCH": 355,
            "DR": 348,
            "DC": 349,
            "DL": 349,
            "TW": 1396,
            "TL": 1399,
            "DT": 348
        },
        "tagged_images": 0
    },
    {
        "solution_name": "TestSolution",
        "total_images": 0,
        "tags": {
          "NOTCH": 355,
            "DR": 348,
        },
        "tagged_images": 1
    }
    ];



 constructor() {
     this.orgD =  this.makeData();
      this.dataSource = this.orgD.reduce((current, next) => {
          for(var d = 0 ; d < next.tags.length ; d++){
              current.push({ solution_name: next.solution_name, total_images: next.total_images, tagged_images: next.tagged_images, tags: next.tags[d],tag_occurence:next.tag_occurence[d] })
          }
  return current;
}, []);
     this.cacheSpan('solution_name', d => d.solution_name);
    this.cacheSpan('total_images', d => d.total_images);
    this.cacheSpan('tagged_images', d => d.tagged_images);
}

makeData = () => {
  let finalData = [];
    this.myData.forEach(d => {
      let a ={};
      a['solution_name'] = d.solution_name; 
      a['total_images'] = d.total_images; 
      a['tagged_images'] = d.tagged_images; 
      if(!!d.tags){
        let k = [];
        let val = [];
          Object.keys(d.tags).forEach(function(key) {
              console.log(key, d.tags[key]);
              k.push(key);
              val.push(d.tags[key]);
          });
          a['tags'] = k;
          a['tag_occurence'] = val;
      }else{
          a['tags'] = [];
          a['tag_occurence'] = [];
      }
   finalData.push(a);
    });
    return finalData;
}

  cacheSpan(key, accessor) {
    for (let i = 0; i < this.dataSource.length;) {
      let currentValue = accessor(this.dataSource[i]);
      let count = 1;
      for (let j = i + 1; j < this.dataSource.length; j++) {
        if (currentValue != accessor(this.dataSource[j])) {
          break;
        }
        count++;
      }
      if (!this.spans[i]) {
        this.spans[i] = {};
      }
      this.spans[i][key] = count;
      i += count;
    }
  }
  getRowSpan(col, index) {    
    return this.spans[index] && this.spans[index][col];
  }
}

В Html Файл

<table mat-table [dataSource]="dataSource" class="mat-elevation-z8" multiTemplateDataRows>

    <ng-container matColumnDef="solution_name">
        <th mat-header-cell *matHeaderCellDef> solution_name </th>
        <td mat-cell *matCellDef="let data;let i = dataIndex" [attr.rowspan]="getRowSpan('solution_name',i)" [style.display]="getRowSpan('solution_name', i) ? '' : 'none'">
         {{ data.solution_name }} </td>
    </ng-container>

    <ng-container matColumnDef="total_images">
        <th mat-header-cell *matHeaderCellDef> total_images </th>
        <td mat-cell *matCellDef="let data;let i = dataIndex" [attr.rowspan]="getRowSpan('total_images',i)" [style.display]="getRowSpan('total_images', i) ? '' : 'none'">
         {{ data.total_images }} </td>
    </ng-container>

    <ng-container matColumnDef="tagged_images">
        <th mat-header-cell *matHeaderCellDef> tagged_images </th>
        <td mat-cell *matCellDef="let data;let i = dataIndex" [attr.rowspan]="getRowSpan('tagged_images',i)" [style.display]="getRowSpan('tagged_images', i) ? '' : 'none'">
         {{ data.tagged_images }} </td>
    </ng-container>

    <ng-container matColumnDef="tags">
        <th mat-header-cell *matHeaderCellDef> tags </th>
        <td mat-cell *matCellDef="let data"> {{ data.tags }} </td>
    </ng-container>

    <ng-container matColumnDef="tag_occurence">
        <th mat-header-cell *matHeaderCellDef> tag_occurence </th>
        <td mat-cell *matCellDef="let data"> {{ data.tag_occurence }} </td>
    </ng-container>

    <tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
    <tr mat-row *matRowDef="let row; columns: displayedColumns;"></tr>



</table>

Согласно вашему изображению и данным, которые я понимаю и имею сделал изменения в соответствии с ним.

Надеюсь, это поможет вам

Дайте мне знать, если я что-то делаю не так.

Выходное изображение

enter image description here

ОБНОВЛЕННАЯ ФУНКЦИЯ КАК ЗА Требование

makeData = () => {
  let finalData = [];
    this.myData.forEach(d => {
      let a ={};
      a['solution_name'] = d.solution_name; 
      a['total_images'] = d.total_images; 
      a['tagged_images'] = d.tagged_images; 
      if(!!d.tags && Object.keys(d.tags).length >= 1){
        let k = [];
        let val = [];
          Object.keys(d.tags).forEach(function(key) {
              console.log(key, d.tags[key]);
              k.push(key);
              val.push(d.tags[key]);
          });
          a['tags'] = k;
          a['tag_occurence'] = val;
      }else{
        let k = [];
        let val = [];
          k.push("");
          val.push("");
          a['tags'] = k;
          a['tag_occurence'] = val;
      }
   finalData.push(a);
    });
    return finalData;
}

Обновленный файл TS (9-1-2020 )

import { Component } from "@angular/core";
@Component({
  selector: "table-basic-example",
  styleUrls: ["table-basic-example.css"],
  templateUrl: "table-basic-example.html"
})
export class TableBasicExample {
  dataSource = [];
  displayedColumns = [
    "solution_name",
    "total_images",
    "tagged_images",
    "tags",
    "tag_occurence"
  ];
  spans = [];
  tempRowId = null;
  tempRowCount = null;
  orgD = [];
  myData = [
    {
      solution_name: "TW",
      total_images: 360,
      tags: {
        TR: 1399,
        NOTCH: 355,
        DR: 348,
        DC: 349,
        DL: 349,
        TW: 1396,
        TL: 1399,
        DT: 348
      },
      tagged_images: 0
    },
    {
      solution_name: "TestSolution3",
      total_images: 0,
      tags: {
        NOTCH: 355,
        DR: 348
      },
      tagged_images: 0
    },
    {
      solution_name: "TestSolution3",
      total_images: 0,
      tags: {
        NOTCH: 355,
        DR: 348
      },
      tagged_images: 0
    },
    {
      solution_name: "TestSolution3",
      total_images: 0,
      tags: {
        NOTCH: 355,
        DR: 348
      },
      tagged_images: 3
    },
    {
      solution_name: "TestSolution3",
      total_images: 4,
      tags: {},
      tagged_images: 0
    }
  ];
  constructor() {
    this.orgD = this.makeData();
    this.dataSource = this.orgD.reduce((current, next) => {
      for (var d = 0; d < next.tags.length; d++) {
        current.push({
          solution_name: next.solution_name,
          total_images: next.total_images,
          tagged_images: next.tagged_images,
          tags: next.tags[d],
          tag_occurence: next.tag_occurence[d]
        });
      }
      return current;
    }, []);
    this.makeSpan();
  }

  makeSpan = () => {
    this.spans = [];
    let index = 0;
    this.myData.forEach(a => {
      let d = {};
      d["solution_name"] =
        Object.keys(a.tags).length >= 1 ? Object.keys(a.tags).length : 1;
      d["total_images"] =
        Object.keys(a.tags).length >= 1 ? Object.keys(a.tags).length : 1;
      d["tagged_images"] =
        Object.keys(a.tags).length >= 1 ? Object.keys(a.tags).length : 1;
      this.spans[index] = d;
      index += d["solution_name"];
    });
  };

  makeData = () => {
    let finalData = [];
    this.myData.forEach(d => {
      let a = {};
      a["solution_name"] = d.solution_name;
      a["total_images"] = d.total_images;
      a["tagged_images"] = d.tagged_images;
      if (!!d.tags && Object.keys(d.tags).length >= 1) {
        let k = [];
        let val = [];
        Object.keys(d.tags).forEach(function(key) {
          k.push(key);
          val.push(d.tags[key]);
        });
        a["tags"] = k;
        a["tag_occurence"] = val;
      } else {
        let k = [];
        let val = [];
        k.push("");
        val.push("");
        a["tags"] = k;
        a["tag_occurence"] = val;
      }
      finalData.push(a);
    });
    return finalData;
  };
  getRowSpan(col, index) {
    return this.spans[index] && this.spans[index][col];
  }
}

Замените приведенный выше код на файл ts, я пробовал несколько данных и внес изменения в соответствии с данными, чтобы сделать его более читабельным и обобщенным c.

Позвольте мне знаю, если это не сработает.

Спасибо

1 голос
/ 07 января 2020
I think this solution can help you out:

Ts file:

    displayedColumns = ['solution','images', 'tag', 'tag_occurence'];
      dataSource = [
      { 
        solution: 'TW', 
        images:'1500',
        array: [
          { 
            tag: 'TW',
            tag_occurence: 300
          },
          { 
            tag: 'TC',
            tag_occurence: 500
          },
          { 
            tag: 'node',
            tag_occurence: 250
          },
        ] 
      },
      { 
        solution: 'TL', 
        images:'1600',
        array: [
          { 
            tag: 'DC',
            tag_occurence: 300
          },
          { 
            tag: 'DQ',
            tag_occurence: 500
          },
          { 
            tag: 'DL',
            tag_occurence: 250
          },
        ] 
      }
    ];


HTML:



    <div class="example-container mat-elevation-z8">
      <mat-table #table [dataSource]="dataSource">

        <mat-header-row *matHeaderRowDef="displayedColumns"></mat-header-row>
        <mat-row *matRowDef="let row; columns: displayedColumns;"></mat-row>

        <ng-container matColumnDef="solution">
          <mat-header-cell *matHeaderCellDef> Solution </mat-header-cell>
          <mat-cell *matCellDef="let obj"> {{obj.solution}} </mat-cell>
        </ng-container>

        <ng-container matColumnDef="images">
          <mat-header-cell *matHeaderCellDef> Images </mat-header-cell>
          <mat-cell *matCellDef="let obj"> {{obj.images}} </mat-cell>
        </ng-container>

        <ng-container matColumnDef="tag">
          <mat-header-cell *matHeaderCellDef> Tag </mat-header-cell>
          <mat-cell *matCellDef="let obj">
            <mat-cell *ngFor="let book of obj.array">
               {{book.tag}} 
            </mat-cell>
          </mat-cell>
        </ng-container>

        <ng-container matColumnDef="tag_occurence">
          <mat-header-cell *matHeaderCellDef> Tag Occurences </mat-header-cell>
          <mat-cell *matCellDef="let obj">
            <mat-cell *ngFor="let book of obj.array">
               {{book.tag_occurence}} 
            </mat-cell>
          </mat-cell>
        </ng-container>

      </mat-table>
    </div>

CSS:


.mat-table {
 border: 1px solid black;
}
.mat-table .mat-cell {
    display: flex;
    text-align: center;
    flex-direction: column;
    justify-content: center;
}

.mat-table .mat-cell {
  min-height: 60px;
  width: 100%;
  border-bottom: 0.5px solid grey;
}

.mat-table .mat-cell:last-child {
  border-bottom: none;
}
...