Данные AG Grid сохраняются между компонентами, вызывая повторяющиеся ошибки узлов - PullRequest
0 голосов
/ 26 апреля 2019

Edit: я пытаюсь вручную использовать функцию grid api destroy в повторно используемом компоненте, а родительский компонент фактически вызывает ag grid (в шаблоне). Проблема с сохранением старых данных в ag-grid.

Я использую повторно используемый компонент для нашего приложения (у него 50 нечетных таблиц, поэтому мы завершили набор инструментов, чтобы его было проще обновлять / изменять). Я закодировал некоторые фильтры постоянства и постоянства строк (поэтому, если вы щелкнете по элементу, перейдете к деталям и вернетесь, он вернет вас к этому элементу и выберет его. Полезно для наших очень больших сеток)

Проблема в том, что когда я перехожу из одного компонента, в котором я не установил обратный вызов getRowNodeId, к тому, где я это делаю, это пугает, что есть дубликаты идентификаторов узлов (b / c, они все еще ссылаются на данные из старой сетки). Свойство в этом наборе данных не существует, поэтому все равно нулю).

Я пытался уничтожить экземпляр aggrid onDestroy вручную, но, похоже, это ничего не дало. Я попытался установить данные строки в пустой массив при инициализации, но это, кажется, нарушило логику инициализации (у меня есть некоторая логика для первого рисования данных), и в целом логика начальной загрузки для сетки ag, кажется, решает, что это сделано немедленно , Логика постоянства больше не работает, если я делаю это Я также пытался отложить настройку этого обратного вызова до первого извлечения данных, но затем он не использует обратный вызов.

Есть какие-нибудь мысли по поводу того, как сохранить getRowNodeId от spaszing, пока сетка ag разрешает новые данные сетки? (или очистите предыдущий экземпляр сетки ag. Кажется, он где-то сохраняет ссылку на него.)

Как настраиваются сетки

                    <grid-tools pagination = "true" IdField = "RouteId" agGrid = "true" persistFilters = "true" [gridEnum]="gridEnum" [agDataGrid]="transMatrixAgGrid" standardTools = "true"  [filterOptions]="filterOptions"></grid-tools>

                </div>
            </div>
            <div class="row add-top add-bottom">
                <div class="col-xs-12" style="width:100%;height:500px;">

                    <mat-progress-bar mode="indeterminate" *ngIf="loading"></mat-progress-bar>

                    <ag-grid-angular #transMatrixAgGrid class="ag-theme-material agGrid"
                        [rowData]="routes" 
                        [gridOptions] = "gridOptions"
                        (rowClicked) = "rowClick($event)" 
                        (selectionChanged)="updateSelectedItems($event)"                                                    
                        [frameworkComponents] = "frameworkComponents"                             
                        [columnDefs]="agColumns">
                    </ag-grid-angular>

                </div>

В компоненте сохраните ссылку на переменную шаблона.

@ViewChild("transMatrixAgGrid") transMatrixAgGrid;

Повторно используемый компонент. Мы переходим от devextreme, поэтому есть некоторая условная логика для сценариев ag grid. Я не хотел пытаться удалить часть этого кода на случай, если кто-то попытается вставить этот код в IDE. Это текущее рабочее состояние.

import { Component, OnInit, Input, OnDestroy } from "@angular/core";
import { MatBottomSheet, MatBottomSheetRef } from "@angular/material";
import { ToolsOptionsMenu } from "./ToolsOptionsMenu/ToolsOptionsMenu.component";
import { DxDataGridComponent } from "devextreme-angular";
import * as _ from "lodash";
import FilterOption from "@classes/FilterOptions.class";
import { utc } from 'moment';
import { FormControl } from '@angular/forms';

// Careful about modifications. Large grids can become slow if this isn't efficient
export function compareDates(filterLocalDateAtMidnight, cellValue) {
    var dateAsString = cellValue;
    if (dateAsString == null) return 0;

    // In the example application, dates are stored as dd/mm/yyyy
    // We create a Date object for comparison against the filter date
    var dateParts = dateAsString.split("T")[0].split("-");
    var day = Number(dateParts[2]);
    var month = Number(dateParts[1]) - 1;
    var year = Number(dateParts[0]);
    var cellDate = new Date(year, month, day);

    // Now that both parameters are Date objects, we can compare
    if (cellDate < filterLocalDateAtMidnight) {
        return -1;
    } else if (cellDate > filterLocalDateAtMidnight) {
        return 1;
    } else {
        return 0;
    }
}

//Ag-grid date formatter.
export function dateFormatter(dateIn) {

    return dateIn.value ? utc(dateIn.value).format("MM/DD/YYYY") : "";
}

//Default grid options
export var gridOptions = {
    rowSelection: "multiple",
    pagination: true,
    rowMultiSelectWithClick: true,
    animateRows: true,
    floatingFilter: true,
    rowBuffer: 20
}

//Default checkbox column options
export var checkboxColumnOptions = {
    field: 'RowSelect',
    width: 50,
    headerName: ' ',
    headerCheckboxSelection: true,
    headerCheckboxSelectionFilteredOnly: true,
    checkboxSelection: true,
    suppressMenu: true,
    sortable: false,
}

@Component({
    selector: "grid-tools",
    templateUrl: "./grid-tools.component.html",
    styleUrls: ["./grid-tools.component.scss"]
})
export class GridToolsComponent implements OnInit, OnDestroy {

    maxSelection = 30000;
    _filterOptions: Array<FilterOption> = []
    currentlySelected = [];
    currentPage = 1;
    pageControl: FormControl;
    storageKey = "";
    selectionStorageKey = "";

    ALL_FILTER = "ALL";
    currentFilter = this.ALL_FILTER;

    //!!!Using filterOptions.push will not trigger this!!!!
    @Input() set filterOptions(value: Array<FilterOption>) {

        // value.splice(0, 0, new FilterOption());
        value.push(new FilterOption());
        this._filterOptions = value;

    };

    get filterOptions() {
        return this._filterOptions;
    };

    @Input() dataGrid: DxDataGridComponent;
    @Input() agDataGrid;
    @Input() agGrid: boolean = false;
    @Input() standardTools: boolean = false;
    @Input() persistGridViewChild?;
    @Input() hideToolsButton = true; //This is until we make the change completely

    @Input() hideExport = false;
    @Input() hideColumnCustomization = false;
    @Input() hideClearFilters = false;
    @Input() hideResetGrid = false;
    @Input() persistFilters = false;
    @Input() pagination = false;
    @Input() gridEnum = null;
    @Input() IdField = null; //Required for navigating to the last selected row   

    constructor(private bottomSheet: MatBottomSheet) {

        this.filterOptions = [];
    }

    ngOnDestroy() {
        // console.log("Destroying component");
        // if (this.agDataGrid) {

        //     this.agDataGrid.api.destroy();
        // }
    }

    ngOnInit() {

        this.pageControl = new FormControl(this.currentPage);
        this.storageKey = "agGrid-" + this.gridEnum;
        this.selectionStorageKey = "agGrid-" + this.gridEnum + "-selection";

        if (this.dataGrid) {

            this.dataGrid.onContentReady.subscribe(result => {
                this.quickFilterSearch();
            });

            this.dataGrid.filterValueChange.subscribe(filterValue => {
                this.quickFilterSearch();
            });


            this.dataGrid.selectedRowKeysChange.subscribe(
                (selections) => {
                    this.currentlySelected = selections;
                }
            )
        }

        if (this.agDataGrid) {

            if (this.IdField) {

                this.agDataGrid.getRowNodeId = (data) => {
                    return data ? data[this.IdField] : data["Id"];
                };
            }

            this.agDataGrid.gridReady.subscribe(
                () => {

                }
            )

            if (this.pagination) {

                this.pageControl.valueChanges
                    .subscribe(newValue => {
                        if (newValue == undefined) {
                            newValue = 1;
                        }
                        this.agDataGrid.api.paginationGoToPage(newValue - 1);
                    });

                this.agDataGrid.paginationChanged.subscribe(($event) => {
                    this.onPaginationChanged($event);
                })

            }



            this.agDataGrid.selectionChanged.subscribe(
                (event) => {

                    this.currentlySelected = this.agDataGrid.api.getSelectedRows();
                }
            );

            this.agDataGrid.rowClicked.subscribe(
                (event) => {

                    if (this.persistFilters) {

                        this.storeSelectedItem(event.node);

                    }
                }
            )

            this.agDataGrid.rowSelected.subscribe(
                (event) => {

                    if (this.persistFilters) {
                        if (event.node.isSelected()) {

                            this.storeSelectedItem(event.node);
                        }
                        else {
                            if (this.agDataGrid.api.getSelectedRows().length == 0) {
                                localStorage.setItem(this.selectionStorageKey, null);
                            }
                        }
                    }
                }
            )

            this.agDataGrid.filterChanged.subscribe(
                (event) => {
                    let currentFilter = this.agDataGrid.api.getFilterModel();
                    if (Object.keys(currentFilter).length == 0) {
                        this.currentFilter = 'ALL';
                    }

                    if (this.persistFilters) {
                        this.setPersistedFilterState(currentFilter);
                    }

                    this.quickFilterSearch();
                }
            )

            this.agDataGrid.firstDataRendered.subscribe(
                (event) => {



                    if (this.persistFilters) {
                        this.restoreFilters();
                        this.restoreSelection();
                    }
                }
            )
        }
    }

    storeSelectedItem(node) {
        let selectedItem = {
            Id: node.id,
            data: node.data
        }
        localStorage.setItem(this.selectionStorageKey, JSON.stringify(selectedItem));
    }

    onPaginationChanged($event) {
        this.currentPage = this.agDataGrid.api.paginationGetCurrentPage()
        this.pageControl.patchValue(this.currentPage + 1);
        this.pageControl.updateValueAndValidity({ emitEvent: false, onlySelf: true });
    }

    restoreSelection() {
        try {

            let selection: any = localStorage.getItem(this.selectionStorageKey);

            if (selection) {

                selection = JSON.parse(selection);

                let node = this.agDataGrid.api.getRowNode(selection.Id);
                node.setSelected(true);
                this.agDataGrid.api.ensureNodeVisible(node, "middle");

            }
        }
        catch (error) {
            console.log("Something wrong with getting " + this.selectionStorageKey + " local storage");
        }

    }

    restoreFilters() {

        let filterModel = localStorage.getItem(this.storageKey);

        if (filterModel) {
            filterModel = JSON.parse(filterModel);

            //TODO: Manually compare to incoming filter options and see if something matches.
            this.currentFilter = '';
        }

        this.agDataGrid.api.setFilterModel(filterModel);
    }

    setPersistedFilterState = (filterValue: any): void => {

        const stringifiedState: string = JSON.stringify(filterValue);
        localStorage.setItem(this.storageKey, stringifiedState);
    };

    resetColumns() {
        if (this.persistGridViewChild) {

            this.persistGridViewChild.resetColumnDefaults();
        }

        if (this.agDataGrid) {
            this.agDataGrid.columnApi.resetColumnState();
        }
    }

    customGrid() {
        this.dataGrid.instance.showColumnChooser();
    }

    export() {
        if (this.currentlySelected.length < this.maxSelection) {

            let selectionOnly = this.currentlySelected.length == 0 ? false : true;

            if (this.agDataGrid) {
                this.agDataGrid.api.exportDataAsCsv({
                    onlySelected: selectionOnly
                });
            }

            if (this.dataGrid) {

                this.dataGrid.instance.exportToExcel(selectionOnly);
            }
        }
    }

    clearAgSelections() {
        this.agDataGrid.api.clearSelections();
    }

    clearSelections() {
        if (this.agDataGrid) {
            this.agDataGrid.api.deselectAll();
        }
        else {

            this.dataGrid.selectedRowKeys = [];
        }
    }

    quickFilterSearch() {
        let state: any = {};

        if (this.agDataGrid) {
            state = _.cloneDeep(this.agDataGrid.api.getFilterModel());
        }
        else {
            state = _.cloneDeep(this.dataGrid.instance.state());
        }


        if (this.agDataGrid) {
            //TODO
        }
        else {
            this.currentFilter = ""; //If a custom filter is created by the user, we don't want any of the chips to be highlighted.

            if (state.filterValue == null) {
                this.currentFilter = this.ALL_FILTER;
            } else {
                _.map(this._filterOptions, (option: any) => {
                    let isEqual = _.isEqual(option.filter, state.filterValue);

                    if (isEqual) {
                        this.currentFilter = option.label;
                    }
                });
            }
        }
    }

    isFilterActive(incomingFilter) {
        return this.currentFilter == incomingFilter;
    }

    showToolsOptions() {
        this.bottomSheet.open(ToolsOptionsMenu, {
            data: {
                grid: this.dataGrid,
                persistGrid: this.persistGridViewChild
            }
        });
    }

    public filterGrid(filterOptions = new FilterOption()) {
        if (this.dataGrid) {

            this.currentFilter = filterOptions.label;

            const state: any = _.cloneDeep(this.dataGrid.instance.state());
            state.filterValue = filterOptions.filter; // New filterValue to be applied to grid.
            this.dataGrid.instance.state(state);

            //The state mechanism seems to not work if persistance is not active on the grid. 
            //This grid is stupid.
            if (!this.persistGridViewChild) {

                this.dataGrid.instance.clearFilter();
            }
        }

        if (this.agDataGrid) {
            this.currentFilter = filterOptions.label;
            this.agDataGrid.api.setFilterModel(filterOptions.filter);

            if (this.persistFilters) {

                this.setPersistedFilterState(filterOptions.filter);
            }
        }
    }
}

Редактировать: я перестал использовать пользовательский обратный вызов, и он все еще делает это. Функция firstDataRendered фактически запускает старые данные. Не идеально: (

1 Ответ

0 голосов
/ 26 апреля 2019

Я не уверен, является ли это ошибкой ag-grid или нет, но в итоге это были мои попытки повторно использовать gridOptions из компонента.

Сетка инструментов Компонент

//Default grid options
export var gridOptions = {
    rowSelection: "multiple",
    pagination: true,
    rowMultiSelectWithClick: true,
    animateRows: true,
    floatingFilter: true,
    rowBuffer: 20
}

Компонент с использованием инструментов сетки

import { gridOptions } from '@app/grid-tools/grid-tools.component';

Присвоение значения

this.gridOptions = gridOptions;

Template

<ag-grid-angular #transMatrixAgGrid class="ag-theme-material agGrid"
                            [rowData]="routes" 
                            [gridOptions] = "gridOptions"                            
                            [columnDefs]="agColumns">
                        </ag-grid-angular>

Каким-то образом это заставляло ag-grid сохранять данные из предыдущих экземпляров сетки. Я исправил это, просто используя

this.gridOptions = Object.assign ({}, gridOptions)

Я предполагаю, что ag-grid изменял ссылку, и в ней содержалась информация о данных.

...