开发者问题收集

Angular:无法读取未定义的属性“名称”

2018-01-29
15824

在我的 angular 4 应用程序中,每次我的侧边栏组件呈现时,我都会收到这个奇怪的错误。我的代码中没有任何属性“名称”。很难弄清楚为什么会抛出错误。以下是控制台错误日志。任何帮助都非常感谢。

html 代码:

<div class="sidebar" >
        <app-sidebar></app-sidebar>
        <div class="sidebar-background" style="background-image: url(assets/img/sidebar-1.jpg)"></div>
    </div>

错误日志:

ERROR TypeError: Cannot read property 'name' of undefined
    at checkBindingNoChanges (core.js:9912)
    at checkNoChangesNodeInline (core.js:13961)
    at checkNoChangesNode (core.js:13935)
    at debugCheckNoChangesNode (core.js:14764)
    at debugCheckDirectivesFn (core.js:14666)
    at Object.eval [as updateDirectives] (AppComponent.html:2)
    at Object.debugUpdateDirectives [as updateDirectives] (core.js:14648)
    at checkNoChangesView (core.js:13773)
    at callViewAction (core.js:14126)
    at execComponentViewsAction (core.js:14078)
    at checkNoChangesView (core.js:13776)
    at callWithDebugContext (core.js:15049)
    at Object.debugCheckNoChangesView [as checkNoChangesView] (core.js:14593)
    at ViewRef_.webpackJsonp.../../../core/esm5/core.js.ViewRef_.checkNoChanges (core.js:11584)
    at core.js:5903
View_AppComponent_0 @ AppComponent.html:2
proxyClass @ compiler.js:14645
webpackJsonp.../../../core/esm5/core.js.DebugContext_.logError @ core.js:14989
webpackJsonp.../../../core/esm5/core.js.ErrorHandler.handleError @ core.js:1501
(anonymous) @ core.js:5908
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.invoke @ zone.js:365
webpackJsonp.../../../../zone.js/dist/zone.js.Zone.run @ zone.js:125
webpackJsonp.../../../core/esm5/core.js.NgZone.runOutsideAngular @ core.js:4681
webpackJsonp.../../../core/esm5/core.js.ApplicationRef.tick @ core.js:5908
(anonymous) @ core.js:5734
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.invoke @ zone.js:365
onInvoke @ core.js:4733
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.invoke @ zone.js:364
webpackJsonp.../../../../zone.js/dist/zone.js.Zone.run @ zone.js:125
webpackJsonp.../../../core/esm5/core.js.NgZone.run @ core.js:4550
next @ core.js:5734
schedulerFn @ core.js:4319
webpackJsonp.../../../../rxjs/Subscriber.js.SafeSubscriber.__tryOrUnsub @ Subscriber.js:239
webpackJsonp.../../../../rxjs/Subscriber.js.SafeSubscriber.next @ Subscriber.js:186
webpackJsonp.../../../../rxjs/Subscriber.js.Subscriber._next @ Subscriber.js:126
webpackJsonp.../../../../rxjs/Subscriber.js.Subscriber.next @ Subscriber.js:90
webpackJsonp.../../../../rxjs/Subject.js.Subject.next @ Subject.js:55
webpackJsonp.../../../core/esm5/core.js.EventEmitter.emit @ core.js:4299
checkStable @ core.js:4698
onHasTask @ core.js:4746
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.hasTask @ zone.js:418
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate._updateTaskCount @ zone.js:438
webpackJsonp.../../../../zone.js/dist/zone.js.Zone._updateTaskCount @ zone.js:262
webpackJsonp.../../../../zone.js/dist/zone.js.Zone.runTask @ zone.js:182
drainMicroTaskQueue @ zone.js:593
Promise resolved (async)
scheduleQueueDrain @ zone.js:552
scheduleMicroTask @ zone.js:560
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.scheduleTask @ zone.js:387
webpackJsonp.../../../../zone.js/dist/zone.js.Zone.scheduleTask @ zone.js:209
webpackJsonp.../../../../zone.js/dist/zone.js.Zone.scheduleMicroTask @ zone.js:229
scheduleResolveOrReject @ zone.js:758
webpackJsonp.../../../../zone.js/dist/zone.js.ZoneAwarePromise.then @ zone.js:847
webpackJsonp.../../../core/esm5/core.js.PlatformRef.bootstrapModule @ core.js:5562
../../../../../src/main.ts @ main.ts:11
__webpack_require__ @ bootstrap 655e16dd1fe7443b4474:54
0 @ main.bundle.js:1835
__webpack_require__ @ bootstrap 655e16dd1fe7443b4474:54
webpackJsonpCallback @ bootstrap 655e16dd1fe7443b4474:25
(anonymous) @ main.bundle.js:1
AppComponent.html:2 ERROR CONTEXT 

完整 TS:

import { Component, OnInit } from '@angular/core';
import { LabelsModel } from './../../core/models/labels';
import { LevelModel } from './../../core/models/level';
import { DataService } from './../../core/service/data.service';
import * as _ from 'underscore';
import { AppService } from './../../core/service/app.service';
import { CategoryModel } from './../../core/models/cat';
import { OnDestroy } from '@angular/core/src/metadata/lifecycle_hooks';
import { AttributeModel } from './../../core/models/att';
import { CountModel, CountChildModel } from './../../core/models/count';
import { SearchModel } from './../../core/models/search';


declare const $: any;

@Component({
    selector: 'app-sidebar',
    templateUrl: './sidebar.component.html',
    styleUrls: ['./sidebar.component.css']
})
export class SidebarComponent implements OnInit {
    menuItems: any[];

    highlightedDiv: number;
    productCategory: CategoryModel[];
    productCategoryM: CategoryModel[];
    productFiltered: AttributeModel[];
    searchOrder = [];
    productAttribute: AttributeModel[];
    productAttributeM: AttributeModel[];
    countModel: CountModel[];
    countChildModel: Array<CountChildModel> = [];
    countVal: any;
    data = [];
    searchArray = [];
    searchModel: SearchModel = {};
    childSearch: string[] = [];
    productSelected: LevelModel;
    searchCheck: SearchModel[];
    checkB: boolean;
    searchCheck1: SearchModel;
    // productName: string;
    filterSelected: boolean;
    labelList: LabelsModel[] = [];

    constructor(private _AppService: AppService, private _dataService: DataService) { }


    ngOnInit() {

        this._AppService.getData().subscribe(te => this.labelList = te);
        this._dataService.productSelectedName.subscribe(product => this.productChange(product));
        this._dataService.toggleValue.subscribe(toggleVal => this.changeToggle(toggleVal));
        this._AppService.getCat$().subscribe(categoryValues => this.processCategory(categoryValues));
        this._dataService.filterList.subscribe(filterValue => this.changeFilter(filterValue));
        this._dataService.setFilterSelected(false);
        this._dataService.filterSelect.subscribe(val => this.filterSelected = val);

    }

    isMobileMenu() {
        if ($(window).width() > 991) {
            return false;
        } else {
            return true;
        }
    };
    changeToggle(toggle): void {
        this.highlightedDiv = toggle;
    }

    clearFilters() {
        this._dataService.setFilterList([]);
        this._dataService.setFilterSelected(false);
        this._dataService.setDisplay(0);
    }

    productChange(product) {

        this.productSelected = {};
        if (product === undefined || product == null) {
            this.productSelected.level = 'level1';
            this.productSelected.product = 'root';

        } else {
            this.productSelected = product;
        }
        if (this.productCategoryM !== undefined) {
            this.processCategory(this.productCategoryM);
        }
    }
    processCategory(categoryValues): void {
        this.searchOrder = [];
        this.productCategory = _.where(categoryValues, { id: this.productSelected.product });
        this.productCategoryM = categoryValues;
        for (const search of this.productCategory[0].search) {
            this.searchOrder.push(search);
        }
        this._AppService.getAtt$().subscribe(attributeValues => this.processAttribute(attributeValues));
    }

    processAttribute(attributeValues): void {
        const searchObj = {};
        let filters = [];
        this.productFiltered = [];
        this.productAttributeM = attributeValues;
        this.searchCheck = this._dataService.getFilterList();
        searchObj[this.productSelected.level] = this.productSelected.product;
        const searchKey = this.productSelected.level;
        this.productAttribute = _.where(attributeValues, searchObj);
        if (this._dataService.getFilterSelected() == true) {
            filters = this._dataService.getFilterList();
            for (let i = 0; i < filters.length; i++) {
                for (let j = 0; j < filters[i].searchChild.length; j++) {
                    if (isNaN(+filters[i].searchChild[j])) {
                        searchObj[filters[i].searchParent] = filters[i].searchChild[j];
                    } else {
                        searchObj[filters[i].searchParent] = +filters[i].searchChild[j];
                    }
                }
            }

            this.productFiltered.push.apply(this.productFiltered, (_.where(this.productAttribute, searchObj)));
            this.productAttribute = _.unique(this.productFiltered);
        }
        this.countModel = [];
        for (let i = 0; i < this.searchOrder.length; i++) {
            this.checkB = false;
            if (this.searchCheck !== undefined && this.searchCheck.length > 0) {
                this.searchCheck1 = this.searchCheck.find(this.check, this.searchOrder[i]);
                if (this.searchCheck1 !== undefined) {
                    if (this.searchCheck1.searchChild.length > 0) {
                        this.checkB = true;
                    }
                }
            }
            this.countModel[i] = {};
            this.countModel[i].parentName = this.searchOrder[i];
            this.countVal = _.pairs(_.countBy(this.productAttribute, this.searchOrder[i]));
            this.countChildModel = [];
            for (let j = 0; j < this.countVal.length; j++) {
                if (this.countVal[j][0] !== 'null'){
                    this.countChildModel[j] = {};
                    this.countChildModel[j].childName = this.countVal[j][0];
                    this.countChildModel[j].count = this.countVal[j][1];
                    if (this.checkB) {
                        if ((this.searchCheck.find(this.check, this.searchOrder[i]).searchChild.find
                        (this.check1, this.countVal[j][0])) !== undefined) {
                            this.countChildModel[j].check = true;
                        }
                    }
                }

            }
            this.countModel[i].child = _.sortBy(this.countChildModel, 'childName');
        }
        for (let i = 0; i < this.countModel.length; i++) {
            this.countModel[i].parentLabel = _.where(this.labelList, { id: this.countModel[i].parentName })[0].label;
        }
    }
    check(ad) {
        return ad.searchParent == this ? true : false;
    }
    check1(ad2) {
        return ad2 == this ? true : false;
    }
    changeFilter(val) {
        val.length == 0 ? this._dataService.setFilterSelected(false) : this._dataService.setFilterSelected(true);
        this.processAttribute(this.productAttributeM);

    }
    taxonSelected(parent, taxon, checked) {
        this.searchModel = {};
        this.childSearch = [];
        this.searchArray = [];
        if (checked) {
            this.searchArray = this._dataService.getFilterList();
            if (_.where(this._dataService.getFilterList(), { searchParent: parent }).length > 0) {
                this.childSearch = _.where(this._dataService.getFilterList(), { searchParent: parent })[0].searchChild;
                this.searchArray.splice((this.searchArray.map(function (d) { return d['searchParent']; }).indexOf(parent)), 1);
                this.childSearch.push(taxon.childName);
                this.searchModel.searchChild = this.childSearch;
                this.searchArray.push(this.searchModel);
                this.searchModel.searchParent = parent;
                this._dataService.setFilterList(this.searchArray);
            } else {
                this.childSearch.push(taxon.childName);
                this.searchModel.searchChild = this.childSearch;
                this.searchModel.searchParent = parent;
                this.searchArray.push(this.searchModel);
                this._dataService.setFilterList(this.searchArray);
            }
        } else {
            this.childSearch = _.where(this._dataService.getFilterList(), { searchParent: parent })[0].searchChild;
            if (this._dataService.getFilterList().length > 1) {
                this.searchArray = this._dataService.getFilterList();
                this.searchArray.splice((this.searchArray.map(function (d) { return d['searchParent']; }).indexOf(parent)), 1);
                if (this.childSearch.length > 1) {
                    this.childSearch.splice(this.childSearch.indexOf(taxon.childName), 1);
                    this.searchModel.searchChild = this.childSearch;
                    this.searchModel.searchParent = parent;
                    this.searchArray.push(this.searchModel);
                    this._dataService.setFilterList(this.searchArray);
                } else {
                    this._dataService.setFilterList(this.searchArray);
                }

            } else {
                if (this.childSearch.length > 1) {
                    this.childSearch.splice(this.childSearch.indexOf(taxon.childName), 1);
                    this.searchModel.searchChild = this.childSearch;
                    this.searchModel.searchParent = parent;
                    this.searchArray.push(this.searchModel);
                    this._dataService.setFilterList(this.searchArray);
                } else {
                    this._dataService.setFilterList([]);
                }
            }
        }
    }

    ngOnDestroy(): void {
    }
}

完整 side bar.html 代码:

<div class="logo">
    <a href="" class="simple-text">
        <div class="logo-img">
            <img src="" />
        </div>
        Tenets Technologies
    </a>
</div>
<div class="sidebar-wrapper scrollbar-danger" >


    <div class="row">
        <div class="col-lg-6 col-md-6 col-sm-6 refine ">
          <h6>Refine </h6>
        </div>
        <div class="col-lg-5 col-md-5 col-sm-5 clear text-right" (click) = "clearFilters()" *ngIf="filterSelected==true">
          Clear All
          <i class="material-icons">clear</i>
        </div>
      </div>
    <div class="nav-container  ">
        <ul class="nav ">
            <li routerlinkactive="active" *ngFor="let search of countModel" class="">
                <a data-toggle="collapse" href="#{{search.parentName}}" class="">
                    <div class="row">
                        <div class="col-md-2 col-sm-2 col-xs-2 ">
                    <i class="material-icons">apps</i>
                </div>
                <div class="col-xs-8 col-sm-8 col-md-8">
                    <p class="test"> {{search.parentLabel}}</p>
                        </div>
                        <div class="col-md-2 col-sm-2 col-xs-2 ">
                        <b class="caret"></b>
                    </div>

                </div>
                </a>
                <div class="collapse listCont scrollbar-danger" id="{{search.parentName}}">
                    <ul class="nav child">

                        <li routerlinkactive="active" *ngFor="let test of search.child">
                            <div class="checkbox">
                                <label>
                                    <input type="checkbox" name="optionsCheckboxes" [checked]="test.check" (click)="taxonSelected(search.parentName,test, $event.target.checked)">
                                    <span class="checkbox-label">
                                        {{test.childName}} ({{test.count}})
                                    </span>
                                </label>
                            </div>
                        </li>
                    </ul>
                </div>
            </li>
        </ul>
    </div>

</div>

App.component.ts

import { Component, OnInit, ViewChild, AfterViewInit, ChangeDetectorRef } from '@angular/core';
import { Location, LocationStrategy, PathLocationStrategy, PopStateEvent } from '@angular/common';
import 'rxjs/add/operator/filter';
import { NavbarComponent } from './components/navbar/navbar.component';
import { Router, NavigationEnd, NavigationStart } from '@angular/router';
import { Subscription } from 'rxjs/Subscription';
import PerfectScrollbar from 'perfect-scrollbar';
import { DataService } from 'app/core/service/data.service';
import { MatSnackBar } from '@angular/material';
import { AppService } from 'app/core/service/app.service';

declare const $: any;

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.css']
})
export class AppComponent implements OnInit {
    private _router: Subscription;
    private lastPoppedUrl: string;
    private yScrollStack: number[] = [];
    /* headerValue = 0;  */
    toggleSide: number;

    @ViewChild(NavbarComponent) navbar: NavbarComponent;

    constructor(public location: Location, private router: Router, 
        private _dataService: DataService, private cdr: ChangeDetectorRef,private _AppService:AppService
        ) { }

    ngOnInit() {
        $.material.init();
        this._dataService.toggleValue.subscribe(toggle => this.changeToggle(toggle));
        const elemMainPanel = <HTMLElement>document.querySelector('.main-panel');
        const elemSidebar = <HTMLElement>document.querySelector('.sidebar .sidebar-wrapper');
        this._dataService.setDisplay(0);
        /*   this.headerValue = 0;
         this._dataService.headerValue.subscribe(header => this.headerValue = header);  */
        this.location.subscribe((ev: PopStateEvent) => {
            this.lastPoppedUrl = ev.url;
        });
        this.router.events.subscribe((event: any) => {
            this.navbar.sidebarClose();
            if (event instanceof NavigationStart) {
                if (event.url != this.lastPoppedUrl)
                    this.yScrollStack.push(window.scrollY);
            } else if (event instanceof NavigationEnd) {
                if (event.url == this.lastPoppedUrl) {
                    this.lastPoppedUrl = undefined;
                    window.scrollTo(0, this.yScrollStack.pop());
                } else
                    window.scrollTo(0, 0);
            }
        });
        this._router = this.router.events.filter(event => event instanceof NavigationEnd).subscribe((event: NavigationEnd) => {
            elemMainPanel.scrollTop = 0;
            elemSidebar.scrollTop = 0;
        });
        if (window.matchMedia(`(min-width: 960px)`).matches && !this.isMac()) {
            let ps = new PerfectScrollbar(elemMainPanel);
            ps = new PerfectScrollbar(elemSidebar);
        }
    }

    changeToggle(val) {
        this.toggleSide = val;

        this.isDetailsMenu();
    }
    isDetailsMenu() {
        if (this.toggleSide == 1) {
            return false;
        } else {
            return true;
        }
    }
    ngAfterViewInit() {
        this.runOnRouteChange();
    }
    runOnRouteChange(): void {
        if (window.matchMedia(`(min-width: 960px)`).matches && !this.isMac()) {
            const elemMainPanel = <HTMLElement>document.querySelector('.main-panel');
            const ps = new PerfectScrollbar(elemMainPanel);
            ps.update();
        }
    }
    isMac(): boolean {
        let bool = false;
        if (navigator.platform.toUpperCase().indexOf('MAC') >= 0 || navigator.platform.toUpperCase().indexOf('IPAD') >= 0) {
            bool = true;
        }
        return bool;
    }
}

App.component.html

<div class="wrapper">
    <div class="sidebar" >
        <app-sidebar></app-sidebar>
        <div class="sidebar-background" style="background-image: url(assets/img/sidebar-1.jpg)"></div>
    </div>
    <div class="main-panel" [ngClass]="{'dtl1': isDetailsMenu()}">
        <app-navbar class="pos" [ngClass]="{'dtl-nav': isDetailsMenu()}"></app-navbar>
        <app-app-filter ></app-app-filter>
        <app-tree></app-tree>
        <router-outlet></router-outlet>

            <app-footer></app-footer>
      </div>


</div>
2个回答

我也刚遇到这个问题,似乎错误本身就是一个转移注意力的借口,因为这个错误是 angular 中的一个 bug,请查看以下问题以了解更多内容

实际的底层错误很可能是 expressionChangedAfterItHasBeenCheckedError 。在开发模式下,angular 会检查您的绑定两次,第二次检查它们是否具有相同的值。如果它们不具有相同的值,它会假定读取该值会修改这些绑定,因此会抛出 expressionChangedAfterItHasBeenCheckedError - 该异常的其他来源可能是如果您的值非常不稳定并且经常更改,请尝试删除绑定,然后错误应该会消失,然后您确定它是 expressionChangedAfterItHasBeenCheckedError

长话短说,不要关心您发布的异常,尝试解决 expressionChangedAfterItHasBeenCheckedError (谷歌搜索它是如何发生和解决的 - SO 上已经有大量关于此的问题)

peter
2018-01-29

您的方法 isDetailsMenu() 必须包含一个尝试使用其自己的 name 属性的变量。

在执行任何操作之前,您应该始终测试您的变量。

isDetailsMenu() {
  return this.myVar && this.myVar.name === 'dtl';
}
2018-01-29