惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

博客园 - Franky
N
Netflix TechBlog - Medium
Google Online Security Blog
Google Online Security Blog
月光博客
月光博客
量子位
酷 壳 – CoolShell
酷 壳 – CoolShell
V
V2EX
腾讯CDC
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
博客园 - 聂微东
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
M
MIT News - Artificial intelligence
Vercel News
Vercel News
The GitHub Blog
The GitHub Blog
Hugging Face - Blog
Hugging Face - Blog
博客园 - 【当耐特】
Apple Machine Learning Research
Apple Machine Learning Research
aimingoo的专栏
aimingoo的专栏
博客园 - 三生石上(FineUI控件)
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
MongoDB | Blog
MongoDB | Blog
H
Help Net Security
The Cloudflare Blog
Blog — PlanetScale
Blog — PlanetScale
F
Full Disclosure
G
Google Developers Blog
罗磊的独立博客
Jina AI
Jina AI
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
Y
Y Combinator Blog
H
Hackread – Cybersecurity News, Data Breaches, AI and More
J
Java Code Geeks
A
About on SuperTechFans
IT之家
IT之家
大猫的无限游戏
大猫的无限游戏
S
SegmentFault 最新的问题
有赞技术团队
有赞技术团队
GbyAI
GbyAI
雷峰网
雷峰网
T
The Blog of Author Tim Ferriss
The Register - Security
The Register - Security
U
Unit 42
D
Docker
Martin Fowler
Martin Fowler
L
LINUX DO - 热门话题
NISL@THU
NISL@THU
阮一峰的网络日志
阮一峰的网络日志
C
Cybersecurity and Infrastructure Security Agency CISA
博客园_首页
Google DeepMind News
Google DeepMind News

博客园 - dudu

.NET CQRS 的实现中引入 ReadOnlyRepository 初试 .NET CQRS 开源库 LiteBus 什么是 Agentic ? 初试 Microsoft Agent Framework 初识 Microsoft Agent Framework:一句话介绍 ASP.NET Core 中读取 UserAgent 的正确姿势 记录一下对 ASP.NET Core Middleware 进行单元测试的代码 C# 实现通用的 IdEqualityComparer 量子网络操作系统 QNodeOS 资料收集 Kubernetes 集群上部署 Open WebUI 在 Kubernetes 集群的 GPU 节点上部署 Ollama 尝试在 Kubernetes 集群上用阿里云 GPU 实例部署 Ollama + DeekSeek-R1 阿里云 GPU 实例云服务器本地部署 DeepSeek R1 尝试使用阿里云计算巢部署 DeepSeek-R1 Angular 中依赖注入问题造成 Observable 订阅不更新 园子博客后台 Angular 升级:手工迁移至 Standalone Component Angular 中使用 ChildContent 记录 园子博客后台升级至 angular 19 后 eslint 9 迁移记录 学习大模型(LLM)的英文好文收集
用 Angular Signal Inputs 完成一个组件的重构
dudu · 2025-05-18 · via 博客园 - dudu

昨天,园子的博客后台终于完成了前端框架升级,从 angular 15 升级到了 angular 19。Angular 从版本 17 开始了文艺复兴(renaissance)之路, 变化很大,这下可以在开发中使用 angular 的新特性了。

这篇博文记录一下昨天晚上用 angular 17 引入的新特性 signal inputs 完成了一个组件的重构。

重构前的组件模版文件

<div class="cnb-panel-header"
     [class.cnb-panel-header-clickable]="headerClickable"
     (click)="headerClickable ? toggleCollapsed() : undefined">    
    <span>{{name}}</span>
    <div class="cnb-panel-header-extra" *ngIf="headerExtra">
        <ng-container *ngTemplateOutlet="headerExtra; context: headerExtraContext ?? {}"></ng-container>
    </div>
</div>
<div class="panel-content"
     [ngClass]="collapsed ? '' : panelContentClass"
     [@openClosePanel]="collapsed ? 'close' : 'open'">
    <ng-content></ng-content>
</div>

重构前的组件类文件

export class CollapsePanelComponent implements OnInit {  
    @Input() name = '';
    @Input() initCollapsed = false;
    @Input() toggleIcon?: TemplateRef<any>;
    @Input() headerClickable = false;
    @Output() collapsedChange = new EventEmitter<boolean>(false);
    @Input() headerExtra: TemplateRef<any> | null = null;
    @Input() headerExtraContext?: ({ $implicit?: any } & Record<string, any>) | null;
    @Input() panelContentClass: string[] = [];
    collapsed = false;

    constructor() { }
    
    toggleCollapsed(collapsed?: boolean) {
        const cache = this.collapsed;
        this.collapsed = collapsed !== undefined ? collapsed : !this.collapsed;
        if (cache !== this.collapsed) this.collapsedChange.next(this.collapsed);
    }

    ngOnInit() {
        this.collapsed = this.initCollapsed || false;
    }
}

重构后的组件模版文件,只是把 collapsed 改成了 collapsed()

重构后的组件类文件

export class CollapsePanelComponent {  
    @Input() name = '';
    @Input() toggleIcon?: TemplateRef<any>;
    @Input() headerClickable = false;
    @Output() collapsedChange = new EventEmitter<boolean>(false);
    @Input() headerExtra: TemplateRef<any> | null = null;
    @Input() headerExtraContext?: ({ $implicit?: any } & Record<string, any>) | null;
    @Input() panelContentClass: string[] = [];
    collapsed = model(false);

    constructor() {
        effect(() => {
            this.collapsedChange.next(this.collapsed());
        });
    }

    toggleCollapsed() {
        this.collapsed.update(x => !x);
    }
}

这里使用 mode 没有使用 input, 是因为组件内部需要更新 collapsed 的值,如果没有这个需要,都由父组件传值,那就用 input

重构中也用到了 effect,只需要在 toggleCollapsed 方法中更新 collapsed 的值,将其他逻辑移到了 effect 的回调函数中处理。

参考: