6

组件模板中有一些自定义元素和属性(在此示例中,它们由第三方非 Angular 代码使用):

<foo></foo>
<div data-bar="{{ bar }}"></div>

它们会导致编译器错误:

Template parse errors:
'foo' is not a known element:
1. If 'foo' is an Angular component, then verify that it is part of this module.
2. If 'foo' is a Web Component then add "CUSTOM_ELEMENTS_SCHEMA" to the '@NgModule.schemas' of this component to suppress this message. ("
    [ERROR ->]<foo></foo>
    <div data-bar="{{ bar }}"></div>
  "): App@1:4
Can't bind to 'bar' since it isn't a known property of 'div'. ("
    <foo></foo>
    <div [ERROR ->]data-bar="{{ bar }}"></div>
  ")
...

如何将foo元素和data-bar属性添加到编译器模式?

NO_ERRORS_SCHEMA不是一个选项,因为不希望将其他未知元素和属性列入白名单。

4

3 回答 3

8

您可以尝试DomElementSchemaRegistry像这样覆盖:

import { DomElementSchemaRegistry, ElementSchemaRegistry } from '@angular/compiler'
import { SchemaMetadata } from '@angular/core';

const MY_DOM_ELEMENT_SCHEMA = [
  'foo'
];

const MY_CUSTOM_PROPERTIES_SCHEMA = {
  'div': {
    'bar': 'string'
  }
};

export class CustomDomElementSchemaRegistry extends DomElementSchemaRegistry {
  constructor() {
    super();
  }

  hasElement(tagName: string, schemaMetas: SchemaMetadata[]): boolean {
    return MY_DOM_ELEMENT_SCHEMA.indexOf(tagName) > -1 || 
         super.hasElement(tagName, schemaMetas);
  }

  hasProperty(tagName: string, propName: string, schemaMetas: SchemaMetadata[]): boolean {
    const elementProperties = MY_CUSTOM_PROPERTIES_SCHEMA[tagName.toLowerCase()];
    return (elementProperties && elementProperties[propName]) || 
        super.hasProperty(tagName, propName, schemaMetas);
  }
}

platformBrowserDynamic().bootstrapModule(AppModule, {
  providers: [{ provide: ElementSchemaRegistry, useClass: CustomDomElementSchemaRegistry }]
});

Plunker 示例

于 2017-03-25T05:30:50.617 回答
1

看起来不错,但请注意您的 Plunker Example,标记中的 div 如下:

data-bar="{{ bar }}"

bar being = "test" 在 dom 上输出,如下所示:

<div></div>

并不是

<div data-bar="test"></div>

如果您想要渲染属性 - 例如聚合物组件,您可以使用以下内容。

[attr.data-bar]="bar" in the markup
于 2017-04-13T15:39:01.527 回答
0

注入 CustomDomElementSchemaRegistry 现在的工作方式有点不同:

即时编译器

自上一个答案以来,注入器发生了变化,因此类型签名变化很小。从版本 5 开始StaticInjector,需要带有 useClass 的 deps 数组(Angular 8.2)

platformBrowserDynamic().bootstrapModule(AppModule, 
 {
   providers: [
    { 
       provide: ElementSchemaRegistry, 
       useClass: CustomDomElementSchemaRegistry, 
       deps: [] 
    }
   ]
 }
);

AOT 编译器

我们不能使用注入器进行提前编译,但是可以DomElementSchemaRegistry在编译器实例化之前对类进行“猴子补丁”,有关详细说明,请在此处查看:

https://medium.com/angular-in-depth/angular-elementschemaregistry-for-dummies-83d54cd31478

于 2020-02-07T12:47:26.467 回答