[Angular 基础] - 视图封装 局部引用 父子组件中内容传递

[Angular 基础] - 视图封装 & 局部引用 & 父子组件中内容传递

之前的笔记:

  • [Angular 基础] - Angular 渲染过程 & 组件的创建

  • [Angular 基础] - 数据绑定(databinding)

  • [Angular 基础] - 指令(directives)

    以上为静态页面,即不涉及到跨组件交流的内容

    以下涉及到组件内的沟通,从这开始数据就“活”了

  • [Angular 基础] - 自定义事件 & 自定义属性

下面的例子依旧会沿用 [Angular 基础] - 自定义事件 & 自定义属性 这里创建的项目

视图封装(view encapsulation)

在 [Angular 基础] - Angular 渲染过程 & 组件的创建 中曾经提到过 CSS 的作用域为当前组件,这是因为 Angular 实现的 view encapsulation。

这个部分可以在 @Component 中修改,如:

@Component({selector: 'my-component',template: `<p>My Component</p>`,encapsulation: ViewEncapsulation.Emulated // default
})

Angular 的 view encapsulation 有 3 个值:Emulated, NoneShadowDom

Emulated

这也是 Angular 默认的实现,在这个实现里,Angular 会为当前组件增添独特的属性,这样当前组件的 CSS 只能绑定于当前的组件上,是一个对 shadow dom 的拟态实现,如下:

在这里插入图片描述

注意这里的 _ngcontent-hash-value,这就是 Angular 随机生成的属性名称,有且只会作用于当前组件上。我这里搜索的是对应的属性名称,可以看到整个 app-server-element 下的 HTML 标签都共享同一个属性名称,无论是 server 还是 blueprint,只要是被 app-server-element 渲染的,都是如此:

在这里插入图片描述

None

None 代表着 Angular 将不会提供任何的 view encapsulation。

如修改一下 app.component.ts,将其 view encapsulation 修改为 None,同时为 p 标签增加颜色:

@Component({selector: 'app-root',templateUrl: './app.component.html',styleUrls: ['./app.component.css'],encapsulation: ViewEncapsulation.None,
})
p {color: blue;
}

其效果如下:

在这里插入图片描述

鉴于该样式写在 app.component.css 中,会被所有的组件访问,因此它会成为所有 p 标签的默认样式,一直到被覆盖为止

ShadowDom

这个使用方法就是仰仗原生浏览器去实现 Shadow DOM,这里将其添加到 app-server-elemen 中:

@Component({selector: 'app-server-element',templateUrl: './server-element.component.html',styleUrl: './server-element.component.css',encapsulation: ViewEncapsulation.ShadowDom,
})

实现效果如下:

在这里插入图片描述

可以看到,不仅 Angular 没有新增对应的属性,并且其他的样式也消失不见了。这是因为对于浏览器来说,CSS 库不会自动被应用的,因此如果要使用 CSS 库的话,要么手动导入,要么重新实现一下,或者使用 JS 动态绑定

前者依旧会引入大量的重复,因此在常见的 2C 项目中是一个比较少见的实现,比较常规的使用是在自己写库的时候会用到

局部引用(local reference)

local reference 在 [Angular 基础] - 指令(directives) 中的 ngIf 中出现过:

<p *ngIf="serverCreated; else noServer">Sever was created, server name is {{ serverName }}
</p>
<ng-template #noServer><p>No server was created!</p>
</ng-template>

其中 #noServer 就是对 ng-template 的 local reference,这里起到的作用就是可以让 Angular 直接在 else 这个条件中获取 <ng-template #noServer> <p>No server was created!</p> </ng-template> 这个元素。

这个做法其实和 React 中的 ref 的作用有些相似,比如说以当前的代码为例,在 cockpit 中获取 server name 和 server content 用的方法是双向绑定,也就是这样的语法 [(ngModel)]="newServerName",但是如果不需要追踪这个值的变化,只需要在点击提交的时候获取元素中的值,则是可以通过 local reference 去实现:

  • V 层修改

    <!-- <input type="text" class="form-control" [(ngModel)]="newServerName" /> -->
    <input type="text" class="form-control" #serverNameInput />
    <!-- 省略若干实现,注意这里传的值 -->
    <button class="btn btn-primary" (click)="onAddServer(serverNameInput)">Add Server
    </button>
    

    其中需要注意的一点就是,local reference 只能在 View 层中传递,它无法 直接 在 VM 层中被访问

  • VM 层修改

    export class CockpitComponent {onAddServer(nameInput: HTMLInputElement) {console.log(nameInput);console.dir(nameInput);}
    }
    

    输出结果为:

    在这里插入图片描述

Local Reference 的主要用法如下:

  • 直接访问 DOM 元素

  • 直接访问子元素

    这点下面会提到怎么实现

  • 搭配 structural directives 进行条件渲染

  • 获取第三方库中的值

父子组件间内容的访问与投射

[Angular 基础] - 自定义事件 & 自定义属性 是父子组件中的属性与事件的交流,这里是内容 (content, DOM) 之间的投射与访问

@ViewChild

虽然说是父组件访问子组件的方法,不过也可以用在同组件的 VM 层和 V 层

获取 Element Ref

具体的方法也是通过绑定 local reference 和 @ViewChild decorator 去实现,代码如下:

  • V 层

    这里的修改和 local reference 中对 V 层的修改类似

    <input type="text" class="form-control" #serverContentInput />
    
  • VM 层

    export class CockpitComponent {@ViewChild('serverContentInput', { static: true })serverContentInput: ElementRef;onAddServer(nameInput: HTMLInputElement) {console.log(this.serverContentInput);}
    }
    

输出结果如下:

在这里插入图片描述

在这里插入图片描述

⚠️:和直接使用 local reference 不同,这里创造的是一个 ElementRef

父组件获取子组件

这里只有 VM 层的修改,在 app.component.ts 中添加一下代码:

export class AppComponent {@ViewChild(CockpitComponent, { static: true })cockpitComponent: CockpitComponent;onServerAdded(serverData: Omit<ServerElement, 'type'>) {console.log(this.cockpitComponent);}
}

输出结果如下:

在这里插入图片描述

⚠️:这个方法只能获取第一个 instance,如果有多个子组件,可以用 @ViewChildren 进行实现,这个用法暂时不会涉及,等到用到时再补充

❗: 一般不推荐用这种方法去访问/获取 V 层的数据

投射内容

这里是在父元素中渲染一个 placeholder,随后等数据接收完毕后,让子元素重写这个 placeholder,以当前项目为例,目前 server-element 渲染的内容为:

<p><strong *ngIf="aliasElement.type === 'server'" style="color: red">{{ aliasElement.content }}</strong><em *ngIf="aliasElement.type === 'blueprint'">{{ aliasElement.content }}</em>
</p>

这样的数据是在 child component 中处理的,不过在有些情况下,对应的数据处理可能需要在父组件完成,而不是通过传递 props 在子组件中进行二次检查——尤其很多时候需要传递 onclick, onsubmit 这种点击事件到子组件中,但是逻辑处理依旧存在于父组件里就会显得比较麻烦——也是可以实现的,如这里将 p 标签的渲染改放到父组件中:

<app-server-element*ngFor="let serverElement of serverElements"[element]="serverElement"
><p><strong *ngIf="serverElement.type === 'server'" style="color: red">{{ serverElement.content }}</strong><em *ngIf="serverElement.type === 'blueprint'">{{ serverElement.content }}</em></p>
</app-server-element>

不过直接这么修改会导致数据丢失:

在这里插入图片描述

这时候,需要在 server-element 中放置一个 ng-content 的指令(directive),这样当前组件就会接受从父组件传来的 内容(content),并且将其投射出来,现在的 server-element 代码如下:

<div class="panel panel-default"><div class="panel-heading">{{ aliasElement.name }}</div><div class="panel-body"><ng-content></ng-content></div>
</div>

这一部分相对于 React 来说确实更加的动态,子组件不需要从父组件接受数据——当然,也可以动态绑定属性和事件进行实现

@ContentChild

使用 @ContentChild 可以让子组件访问 父组件投射到子组件中的 内容(content),也就是上面使用 ng-content 进行投射的渲染内容

具体方法如下:

  • 父组件中声明一个 local reference

    这里的实现在 V 层:

    <p #contentParagraph><strong *ngIf="serverElement.type === 'server'" style="color: red">{{ serverElement.content }}</strong><em *ngIf="serverElement.type === 'blueprint'">{{ serverElement.content }}</em>
    </p>
    
  • 子组件中从 VM 层访问 local reference

    export class ServerElementComponent {@ContentChild('contentParagraph', { static: true }) paragraph: ElementRef;// 这个在 lifecycle 会重新提一下ngAfterContentInit() {console.log('ngAfterContentInit in ServerElementComponent');console.log(this.paragraph, this.paragraph.nativeElement.textContent);}
    }
    

    渲染结果如下:

    在这里插入图片描述

⚠️:这个 directive 其实和 @ViewChild/@ViewChildren 一样,也可以用在父组件中获取子组件的投射,并且在 ngAfterContentInit 确认投射完成后做一些对应操作

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/474154.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue-自定义创建项目(六)

为什么要自定义创建项目&#xff1f; 因为VueCli默认创建的项目不能够满足我们的要求&#xff0c;比如默认的项目中没有帮我们集成路由&#xff0c;vuex&#xff0c;eslink等功能。 默认项目 自定义创建项目 流程&#xff1a; 创建项目命令&#xff1a;vue create custom_dem…

使用Python生成二维码的完整指南

无边落木萧萧下&#xff0c;不如跟着可莉一起游~ 可莉将这篇博客收录在了&#xff1a;《Python》 可莉推荐的优质博主首页&#xff1a;Kevin ’ s blog 本文将介绍如何使用Python中的qrcode库来生成二维码。通过简单的代码示例和详细解释&#xff0c;读者将学习如何在Python中轻…

【技巧】Allegro中大型板布局技巧

Orcade原理图导入到allegro PCB后&#xff0c;画好边框就可以通过quickplace 将元件放到边框的四周了。这个比较easy&#xff0c;但是place的元件比较杂乱无章&#xff0c;后面PCB布局是摆放元件将非常费力。 学习一个布局的方法。 在Orcad原理图中将器件加上Room属性&#x…

Open CASCADE学习|管道建模

​这是用Open CASCADE Technology (OCCT)库来创建一个管道模型的示例。OCCT是一个开源的几何建模库&#xff0c;广泛应用于CAD/CAM/CAE和其他几何建模应用中。 在下面的代码中&#xff0c;首先创建了一些点&#xff0c;并用这些点来构建B样条曲线&#xff0c;进而创建边(Edges…

typeScript 类型推论

什么是类型推论&#xff1f; 类型推论是 TypeScript 中的一个特性&#xff0c;它允许开发人员不必显式地指定变量的类型。相反&#xff0c;开发人员可以根据变量的使用情况让 TypeScript 编译器自动推断出类型。例如&#xff0c;如果开发人员将一个字符串赋值给一个变量&#…

测试物理网络的ping命令

通过发送Internet控制消息协议&#xff08;ICMP&#xff09;并接收其应答&#xff0c;测试验证与另一台TCP/IP计算机的IP级联通性、可达到性和名称解析的疑难问题主要TCP/IP命令。如果不带参数&#xff0c;ping将显示帮助。通过在命令提示符下输入“ping /&#xff1f;”命令&a…

[ 2024春节 Flink打卡 ] -- Paimon

Flink 社区希望能够将 Flink 的 Streaming 实时计算能力和 Lakehouse 新架构优势进一步结合&#xff0c;推出新一代的 Streaming Lakehouse 技术&#xff0c;促进数据在数据湖上真正实时流动起来&#xff0c;并为用户提供实时离线一体化的开发体验。原名 Flink Table Store &am…

WebFlux简单测试

一、介绍 响应式的web框架&#xff0c;不同于传统的WebMVC&#xff0c;WebFlux其内部使用Netty&#xff0c;基于异步和事件驱动&#xff0c;用少量Loop线程处理request、response io操作&#xff0c;而业务中阻塞的操作交由Work线程处理。总的来说客户端感知&#xff08;响应时…

探索【注解】、【反射】、【动态代理】,深入掌握高级 Java 开发技术

文章目录 Java注解1.注解基础2.注解原理 反射1.Class对象的获取1.基础公共类1.1.Object > getClass()1.2.类名.class 的方式1.3.Class.forName() 2.获取类的成员变量3.获取成员方法并调用4.反射优缺点 代理1.结构2.静态代理2.1.案例1-计算前后校验2.1.1.创建接口2.1.2.创建实…

使用Thonny搭建MicroPython 开发环境,简单易用,保姆级教程

MicroPython Thonny开发环境搭建&#xff0c;简单易用&#xff0c;保姆级教程 MicroPython为广大Python编程开发人员打开了通往单片机、物联网开发的大门&#xff0c;但是在开发工具方面却不是很给力&#xff0c;虽然VSCode的插件很多&#xff0c;但是针对MicroPython却不是很…

【leetcode】深搜、暴搜、回溯、剪枝(C++)3

深搜、暴搜、回溯、剪枝&#xff08;C&#xff09;3 一、解数独1、题目描述2、代码3、解析 二、单词搜索1、题目描述2、代码3、解析 三、黄金矿工1、题目描述2、代码3、解析 四、不同路径III1、题目描述2、代码3、解析 一、解数独 1、题目描述 leetcode链接 2、代码 class…

MCU电源控制(PWR)与低功耗

目录 一、STM32 的内核和外设电源系统管理&#xff1a; 二、MCU电源监控&#xff1a; 三、三种低功耗模式&#xff1a; 1、睡眠模式&#xff1a; 2、停止模式&#xff1a; 3、待机模式&#xff1a; 一、STM32 的内核和外设电源系统管理&#xff1a; ① 电池备份区域&#…