投影元素直接隔离_Angular ngcontent 内容投影

前言

内容投影和ng-content是可以让我们最大程度构建可重用组件的Angular功能之一。我们来构造一个小组件,一个Font Awesomne输入框。我们设计这个组件的目标是为了构造一个带有图标的文本框。

最终的样子如图所示:

57526ec29bc0039164e8db6a8f5cbe03.png

ng-content

不使用ng-content的话会遇到什么问题?先来尝试下不用内容投影的话,我们的组件会遇到什么问题。

首先看模板:

class="fa" [ngClass]="classes">
    #input
    (focus)="inputFocus = true"
    (blur)="inputFocus = false"
    (keyup)="value.emit(input.value)"
/>
用classes对象来控制展示的图标,然后用inputFocus获得焦点进入input,通过组件的HostBinding来给组件应用外边框。
import {Component, EventEmitter, HostBinding, Input, OnInit, Output} from '@angular/core';

({
    selector: 'app-fa-input',
    templateUrl: './fa-input.component.html',
    styleUrls: ['./fa-input.component.scss']
})
export class FaInputComponent implements OnInit {
    () icon: string;
    () value = new EventEmitter<string>();
    inputFocus: boolean = false;
    get classes() {
        const cssClasses = {
            fa: true,
        }
        cssClasses['fa' + this.icon] = true;
        return cssClasses;
    }
    ('class.focus')
    get focus() {
        console.log(this.inputFocus);
        return this.inputFocus;
    }
}

// css部分:
:host{
    border: 1px solid grey;
}
input{
    border: none;
    outline: none;
}
:host(.focus) {
    border: 1px solid blue;
}
看样式文件可以知道,组件内部的input元素被移除了自带的样式。但我们给宿主元素加上了边框,让组件看起来像原生的html input元素。当input获取到焦点的时候,通过将.focus类添加到宿主元素来模拟输入框获得焦点。然后看看如何使用这个组件:
<div>
    <h1>FA Inputh1>
    <i class="fa fa-heart">i>
    <app-fa-input icon="envelope" (value)="onNewValue($event)">app-fa-input>
div>
使用的时候,我们只需要向组件传递一个图标的名称和接收input输入值的函数即可。 让我们回顾下我们是如何设计这个组件的:
  • 作为组件公共api的一部分,我们有一个图标的属性,该属性定义了需要显示的图标。

  • 组件有个名为value的自定义输出事件,该事件在input元素输入值发生变化时发出新的值。

  • 为了实现焦点功能,我们在组件内部的input元素上绑定了blur和focus事件,通过@HostBinding在宿主元素上增加或删除focuscss 类。

  • 这个组件可以满足我们的需求。但是假设我们的需求发生了变更,我们马上会陷入到新的麻烦中。

问题1:如何支持所有的input属性?
我们的组件目前只是预定义了blur和focus属性,那我们需要增加其他属性,比如type,autocomplete、placeholder等,咋办?那我们只能被迫去修改组件,使其可以支持这样调用:
input icon="envelope" type="text" placeholder="email" autocomplete="off" (value)="onNewValue($event)">input>
在组件类中需要接收这些属性:
export class FaInputComponent implements OnInit {
    // ...
    () icon: string;
    () placeholder: string;
    () type: string;
    () autocomplete: string;
    () value = new EventEmitter<string>();
    // ...
}
在模板中应用:
class="fa" [ngClass]="classes">
    #input
    [placeholder]="placeholder"
    [type]="type"
    [autocomplete]="autocomplete"
    (focus)="inputFocus = true"
    (blur)="inputFocus = false"
    (keyup)="value.emit(input.value)"
/>
总而言之,我们要将需要处理的属性,从消费处一直传递到组件内部,然后在组件内部从组件类到组件的模板。虽然是很麻烦,但这样是可行的。 但是,还有其他更棘手的问题。
问题2:如何和Angular Form 集成?
我们的组件是个带图标的输入框,那么它的作用不仅仅是展示,它的重点功能是表单的一个输入元素,那么我们很可能需要和Angular Form集成,那么我们咋办?还是如上面一样,我们需要将表单的所有属性,比如formControlName全部转发到组件内部。
问题3:检测普通浏览器事件
我们想在组件上检测到标准浏览器的dom事件怎么办?比如keydown事件?也还是和上面一样,我们需要通过组件内部检测然后在消费的地方去提供处理方法。 也是 可行的,但是好像我们的这个设计变得很不好,这样慢慢的会很臃肿。 这种设计不是个很好的解决方法。
问题4:自定义属性
在构建表单时,第三方系统可能希望填写某些自定义的html数据属性,比如类似于:data-之类的属性用于其他作用。 这会变得非常难办,因为我们无法预知这些属性的名字。 那么,到目前为止,我们这种设计的关键问题是什么? 关键问题是,我们将input元素隐藏到了组件模板中。 在需要调用这个组件的地方和组件内部形成了一个屏障。 我们可以用内容投影来重构组件,以解决上面的问题。
使用ng-content内容投影来重构组件
让我们重新设计组件Api,与其将输入元素隐藏在组件内部,不如将其提供为组件本身的内容元素(content element)。那么我们在调用的地方应该是这样的:
input icon="envelope">
      <input type="text" placeholder="email"/>input>
需要注意的是,我们这里的input元素不是存在于组件内部,而是作为组件的html标签的一部分“内容”。 实际上,这种api在html标准元素中非常常见,比如选择框:
<select>
   <option value=1>oneoption>
   <option value=2>twooption>
select>
Angular Core 确实允许我们做同样的事情。 我们可以使用@ContentChild和@ContentChildren装饰器来查询组件HTML内容的所标记的内容。 并将其在内部模板用作配置API。如果有必要,我们还可以将区域中的内容直接用作组件的内容。 我们需要改造fa-input组件:
<mat-icon>{{icon}}mat-icon>
<ng-content>ng-content>
为省事我这里使用了Angular Material的图标。
import {Component, Input, OnInit} from '@angular/core';
({
    selector: 'app-fa-input',
    templateUrl: './fa-input.component.html',
    styleUrls: ['./fa-input.component.scss']
})
export class FaInputComponent {
    () icon: string;
    constructor() {
    }
}
然后在其他组件中使用这个组件:
input icon="mail_outline">
    <input type="text" name="email"/>input>
页面需要的元素都是OK的,我们这里的input也作为投影的内容显示在了组件的内部。 但是好像css没有应用上啊,那投影的元素的样式如何处理?给投影的元素应用css样式;目前的样式是定义在组件的样式fa-input.component.scss之中:
input{
    border: none;
    outline: none;
}
为啥不起作用?因为这些样式位于链接到组件的样式文件内,所以它们会被赋予一个运行时的属性,这个属性是该组件模板中所有html元素独有的属性。目前元素没应用上,我们可以给mat-icon写个样式来观察下:
.mat-icon{
    color: red;
}

// 然后查看运行后的页面,我们查看控制板板里面的css有:
.mat-icon[_ngcontent-hhd-c122] {
    color: red;
}

对应的html有:

<app-fa-input _ngcontent-hhd-c144 icon="mail_outline" _nghost-hhd-c122 ng-reflect-icon="mail_outline">
    <mat-icon _ngcontent-hhd-c122 role="img" class="mat-icon">mail_outlinemat-icon>
    <input _ngcontent-hhd-c144 type="text" name="email">
app-fa-input>
我们可以看到,组件内部的元素是拥有一个特定的属性_ngcontent-hhd-c122,组件内部链接的样式也是有一个属性_ngcontent-hhd-c122,这可以让组件内部的样式不去干涉外部的元素。这是非常有用的。而input元素是外部投影进来的,所以它的属性是_ngcontent-hhd-144,组件内部的样式是应用不上去的,这就是为啥我们样式不起作用的原因。我们需要加上::ng-deep来使样式穿透。
::ng-deep input{
    border: none;
    outline: none;
}
这样看起来是好的,但是有个隐患,我们在外层使用组件的地方加上一个
input:

input icon="mail_outline">
    <input type="text" name="email"/>input>
<input type="text" name="email"/>
好嘛,两个input都被应用上了样式。甚至于我们去别的组件,不是父子组件,只是在这个页面组件树的其他组件中加上input,发现都应用上了,看来这个样式使用::ng-deep之后就变成了全局的css了。这样会造成一些不可控的问题。如何解决呢?我们只需要在样式前面加上:host来限定下即可:
:host ::ng-deep input{
    border: none;
    outline: none;
}
这样,发现只在投影到组件内部的元素才会应用这个样式。所以,我们的需求是样式既要应用在当前组件,也需要应用到投影进来的元素。我们使用:host ::ng-deep就可以完美解决。 再在控制台下查 看下样式:
[_nghost-unf-c122] input {
    border: none;
    outline: none;
}
正如我们所看到的,这个样式的作用域依旧是当前的组件内容,但是他们也会穿透到投影到当前组件的元素。 如何与投影内容交互?前面我们尝试了将组件内的样式应用到投影的元素中,现在我们尝试下和投影的内容进行交互。 我们无法在ng-content标签上创建交互,也没法在其上绑定事件监听。 相对的,与投影内容做交互做好的方法是以单独的指令去操作。 这里为了示例,我就不创建新的指令了,而是使用Angular Material的matInput指令。首先在将matInput挂到input元素上:
input icon="mail_outline">
    <input matInput type="text" name="email"/>input>
然后在指令中通过@ContentChild修饰符获取到投影进来的input元素:
export class FaInputComponent implements OnInit {
    @Input() icon: string;
    @ContentChild(MatInput)
    input:MatInput;
}
然后通过这个指令去模拟input获取到焦点的过程:
@HostBinding('class.focus')
get focus() {
    console.log('input', this.input.focused);
    return this.input ? this.input.focused : false;
}

// 相对应的css样式:
.fa-input{
    padding: 3px 8px;
    display: flex;
    justify-content: flex-start;
    align-items: center;
}
:host ::ng-deep input{
    border: none;
    outline: none;
}
:host(.focus){
    border: 1px solid blue;
}
最后的效果:

f6e5fa14dfdac40300fabf46a129f944.png

多插槽(Multi-Slot)内容投影

到目前为止,我们基本是一个ng-content将内容投影进来,但是假如我们想投影一部分或者几个部分呢?前面是在fa-input组件内部定义了icon,然后将input从外部投影到了组件内部。那么我希望可以将两个都投影进来。fa-input组件只是提供一个空壳子。可以通过ng-content的select属性来获取到组件tag标记中的内容进行部分投影。 我们可以修改fa-input组件的模板内容:
<div class="fa-input">
    图标:
    <ng-content select="mat-icon">ng-content>
    输入框:
    <ng-content select="input">ng-content>
    
    <ng-content>ng-content>
div>
然后在使用的地方:
<app-fa-input icon="mail_outline">
    <mat-icon>mail_outlinemat-icon>
    <input matInput autocomplete="off" type="text" name="email"/>
app-fa-input>
最后可以看到,我们包括在组件tag中的内容会被分配到我们希望他们出现的地方。 e4f89ca7ccb3acf9240e63e64697e36c.png来解读下上面的“插槽”。上面两个ng-content的select属性查找组件tag标记中的内容的特定元素,匹配后就投影进来,不带select的ng-content会将没有匹配的内容投影到组件中去。我们也可以查找具有特定类的元素,以结合多个选择器。 例如,根据类选择某个input:
<div class="fa-input">
    图标:
    <ng-content select="mat-icon">ng-content>
    输入框:
    <ng-content select="input.text">ng-content>
    <ng-content>ng-content>
div>
对应的使用的地方:
<app-fa-input>
    <mat-icon>mail_outlinemat-icon>
    <input matInput class="text" autocomplete="off" type="text" name="email"/>
    <input autocomplete="off" type="file" name="email"/>
    <p>其他的一些投影p>
app-fa-input>
可以看到投影到了具体的input。

最后

当然,这篇文章是我根据angular blog angular-ng-content这篇文章的翻译和解读,觉得啰嗦或者说不清楚的话,可以去看原文。

参考链接

https://blog.angular-university.io/angular-ng-content

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值