diff --git a/README.md b/README.md
index 6af1ff092..136070d07 100644
--- a/README.md
+++ b/README.md
@@ -24,7 +24,7 @@ If you would like to propose a challenge, this project is open source, so feel f
## Challenges
-Check [all 36 challenges](https://angular-challenges.vercel.app/)
+Check [all 37 challenges](https://angular-challenges.vercel.app/)
## Contributors ✨
diff --git a/apps/performance/ngfor-biglist/.eslintrc.json b/apps/performance/ngfor-biglist/.eslintrc.json
new file mode 100644
index 000000000..bf8df1428
--- /dev/null
+++ b/apps/performance/ngfor-biglist/.eslintrc.json
@@ -0,0 +1,36 @@
+{
+ "extends": ["../../../.eslintrc.json"],
+ "ignorePatterns": ["!**/*"],
+ "overrides": [
+ {
+ "files": ["*.ts"],
+ "rules": {
+ "@angular-eslint/directive-selector": [
+ "error",
+ {
+ "type": "attribute",
+ "prefix": "app",
+ "style": "camelCase"
+ }
+ ],
+ "@angular-eslint/component-selector": [
+ "error",
+ {
+ "type": "element",
+ "prefix": "app",
+ "style": "kebab-case"
+ }
+ ]
+ },
+ "extends": [
+ "plugin:@nx/angular",
+ "plugin:@angular-eslint/template/process-inline-templates"
+ ]
+ },
+ {
+ "files": ["*.html"],
+ "extends": ["plugin:@nx/angular-template"],
+ "rules": {}
+ }
+ ]
+}
diff --git a/apps/performance/ngfor-biglist/README.md b/apps/performance/ngfor-biglist/README.md
new file mode 100644
index 000000000..e23cb226d
--- /dev/null
+++ b/apps/performance/ngfor-biglist/README.md
@@ -0,0 +1,13 @@
+# NgFor optimize big list
+
+> Author: Thomas Laforge
+
+### Run Application
+
+```bash
+npx nx serve performance-ngfor-biglist
+```
+
+### Documentation and Instruction
+
+Challenge documentation is [here](https://angular-challenges.vercel.app/challenges/angular-performance/37-ngfor-biglist/).
diff --git a/apps/performance/ngfor-biglist/project.json b/apps/performance/ngfor-biglist/project.json
new file mode 100644
index 000000000..e567b9b53
--- /dev/null
+++ b/apps/performance/ngfor-biglist/project.json
@@ -0,0 +1,81 @@
+{
+ "name": "performance-ngfor-biglist",
+ "$schema": "../../../node_modules/nx/schemas/project-schema.json",
+ "projectType": "application",
+ "prefix": "app",
+ "sourceRoot": "apps/performance/ngfor-biglist/src",
+ "tags": [],
+ "targets": {
+ "build": {
+ "executor": "@angular-devkit/build-angular:browser",
+ "outputs": ["{options.outputPath}"],
+ "options": {
+ "outputPath": "dist/apps/performance/ngfor-biglist",
+ "index": "apps/performance/ngfor-biglist/src/index.html",
+ "main": "apps/performance/ngfor-biglist/src/main.ts",
+ "polyfills": ["zone.js"],
+ "tsConfig": "apps/performance/ngfor-biglist/tsconfig.app.json",
+ "assets": [
+ "apps/performance/ngfor-biglist/src/favicon.ico",
+ "apps/performance/ngfor-biglist/src/assets"
+ ],
+ "styles": ["apps/performance/ngfor-biglist/src/styles.scss"],
+ "scripts": []
+ },
+ "configurations": {
+ "production": {
+ "budgets": [
+ {
+ "type": "initial",
+ "maximumWarning": "500kb",
+ "maximumError": "1mb"
+ },
+ {
+ "type": "anyComponentStyle",
+ "maximumWarning": "2kb",
+ "maximumError": "4kb"
+ }
+ ],
+ "outputHashing": "all"
+ },
+ "development": {
+ "buildOptimizer": false,
+ "optimization": false,
+ "vendorChunk": true,
+ "extractLicenses": false,
+ "sourceMap": true,
+ "namedChunks": true
+ }
+ },
+ "defaultConfiguration": "production"
+ },
+ "serve": {
+ "executor": "@angular-devkit/build-angular:dev-server",
+ "configurations": {
+ "production": {
+ "browserTarget": "performance-ngfor-biglist:build:production"
+ },
+ "development": {
+ "browserTarget": "performance-ngfor-biglist:build:development"
+ }
+ },
+ "defaultConfiguration": "development"
+ },
+ "extract-i18n": {
+ "executor": "@angular-devkit/build-angular:extract-i18n",
+ "options": {
+ "browserTarget": "performance-ngfor-biglist:build"
+ }
+ },
+ "lint": {
+ "executor": "@nx/linter:eslint",
+ "outputs": ["{options.outputFile}"],
+ "options": {
+ "lintFilePatterns": [
+ "apps/performance/ngfor-biglist/**/*.ts",
+ "apps/performance/ngfor-biglist/**/*.html"
+ ]
+ }
+ }
+ }
+}
diff --git a/apps/performance/ngfor-biglist/src/app/app.component.ts b/apps/performance/ngfor-biglist/src/app/app.component.ts
new file mode 100644
index 000000000..1baa61dab
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/app.component.ts
@@ -0,0 +1,42 @@
+import { NgIf } from '@angular/common';
+import { Component, signal } from '@angular/core';
+import { FormsModule } from '@angular/forms';
+import { MatFormFieldModule } from '@angular/material/form-field';
+import { MatInputModule } from '@angular/material/input';
+import { generateList } from './generateList';
+import { PersonService } from './list.service';
+import { PersonListComponent } from './person-list.component';
+
+@Component({
+ standalone: true,
+ imports: [
+ NgIf,
+ PersonListComponent,
+ FormsModule,
+ MatFormFieldModule,
+ MatInputModule,
+ ],
+ providers: [PersonService],
+ selector: 'app-root',
+ template: `
+
+ Load List
+
+
+
+ `,
+ host: {
+ class: 'flex items-center flex-col gap-5',
+ },
+})
+export class AppComponent {
+ readonly persons = signal(generateList());
+ readonly loadList = signal(false);
+
+ label = '';
+}
diff --git a/apps/performance/ngfor-biglist/src/app/app.config.ts b/apps/performance/ngfor-biglist/src/app/app.config.ts
new file mode 100644
index 000000000..59198e627
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/app.config.ts
@@ -0,0 +1,6 @@
+import { ApplicationConfig } from '@angular/core';
+import { provideAnimations } from '@angular/platform-browser/animations';
+
+export const appConfig: ApplicationConfig = {
+ providers: [provideAnimations()],
+};
diff --git a/apps/performance/ngfor-biglist/src/app/generateList.ts b/apps/performance/ngfor-biglist/src/app/generateList.ts
new file mode 100644
index 000000000..96bfb3722
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/generateList.ts
@@ -0,0 +1,15 @@
+import { randEmail, randFirstName } from '@ngneat/falso';
+import { Person } from './person.model';
+
+export function generateList() {
+ const arr: Person[] = [];
+
+ for (let i = 0; i < 100000; i++) {
+ arr.push({
+ email: randEmail(),
+ name: randFirstName(),
+ });
+ }
+
+ return arr;
+}
diff --git a/apps/performance/ngfor-biglist/src/app/list.service.ts b/apps/performance/ngfor-biglist/src/app/list.service.ts
new file mode 100644
index 000000000..e56bddef8
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/list.service.ts
@@ -0,0 +1,44 @@
+import { Injectable, inject, signal } from '@angular/core';
+import { randEmail, randFirstName } from '@ngneat/falso';
+import { generateList } from './generateList';
+import { Person } from './person.model';
+
+@Injectable()
+export class PersonService {
+ private readonly fakeBackend = inject(FakeBackendService);
+ readonly persons = signal([]);
+
+ loadPersons() {
+ this.persons.set(generateList());
+ }
+
+ deletePerson(email: string) {
+ this.persons.set(
+ this.fakeBackend
+ .returnNewList(this.persons())
+ .filter((p) => p.email !== email)
+ );
+ }
+
+ updatePerson(email: string) {
+ this.persons.set(
+ this.fakeBackend
+ .returnNewList(this.persons())
+ .map((p) => (p.email === email ? { email, name: randFirstName() } : p))
+ );
+ }
+
+ addPerson(name: string) {
+ this.persons.set([
+ { email: randEmail(), name },
+ ...this.fakeBackend.returnNewList(this.persons()),
+ ]);
+ }
+}
+
+@Injectable({ providedIn: 'root' })
+export class FakeBackendService {
+ returnNewList = (input: Person[]): Person[] => [
+ ...input.map((i) => ({ ...i })),
+ ];
+}
diff --git a/apps/performance/ngfor-biglist/src/app/person-list.component.ts b/apps/performance/ngfor-biglist/src/app/person-list.component.ts
new file mode 100644
index 000000000..e9bbb1785
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/person-list.component.ts
@@ -0,0 +1,30 @@
+import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
+
+import { NgForTrackByModule } from '@angular-challenges/shared/directives';
+import { CommonModule } from '@angular/common';
+import { Person } from './person.model';
+
+@Component({
+ selector: 'app-person-list',
+ standalone: true,
+ imports: [CommonModule, NgForTrackByModule],
+ template: `
+
+ `,
+ host: {
+ class: 'w-full flex flex-col',
+ },
+ changeDetection: ChangeDetectionStrategy.OnPush,
+})
+export class PersonListComponent {
+ @Input() persons: Person[] = [];
+}
diff --git a/apps/performance/ngfor-biglist/src/app/person.model.ts b/apps/performance/ngfor-biglist/src/app/person.model.ts
new file mode 100644
index 000000000..0ccffbbf9
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/app/person.model.ts
@@ -0,0 +1,4 @@
+export interface Person {
+ email: string;
+ name: string;
+}
diff --git a/apps/performance/ngfor-biglist/src/assets/.gitkeep b/apps/performance/ngfor-biglist/src/assets/.gitkeep
new file mode 100644
index 000000000..e69de29bb
diff --git a/apps/performance/ngfor-biglist/src/favicon.ico b/apps/performance/ngfor-biglist/src/favicon.ico
new file mode 100644
index 000000000..317ebcb23
Binary files /dev/null and b/apps/performance/ngfor-biglist/src/favicon.ico differ
diff --git a/apps/performance/ngfor-biglist/src/index.html b/apps/performance/ngfor-biglist/src/index.html
new file mode 100644
index 000000000..f23a85cca
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/index.html
@@ -0,0 +1,13 @@
+
+
+
+
+ performance-ngfor-biglist
+
+
+
+
+
+
+
+
diff --git a/apps/performance/ngfor-biglist/src/main.ts b/apps/performance/ngfor-biglist/src/main.ts
new file mode 100644
index 000000000..514c89a08
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/main.ts
@@ -0,0 +1,7 @@
+import { bootstrapApplication } from '@angular/platform-browser';
+import { appConfig } from './app/app.config';
+import { AppComponent } from './app/app.component';
+
+bootstrapApplication(AppComponent, appConfig).catch((err) =>
+ console.error(err)
+);
diff --git a/apps/performance/ngfor-biglist/src/styles.scss b/apps/performance/ngfor-biglist/src/styles.scss
new file mode 100644
index 000000000..77e408aa8
--- /dev/null
+++ b/apps/performance/ngfor-biglist/src/styles.scss
@@ -0,0 +1,5 @@
+@tailwind base;
+@tailwind components;
+@tailwind utilities;
+
+/* You can add global styles to this file, and also import other style files */
diff --git a/apps/performance/ngfor-biglist/tailwind.config.js b/apps/performance/ngfor-biglist/tailwind.config.js
new file mode 100644
index 000000000..38183db2c
--- /dev/null
+++ b/apps/performance/ngfor-biglist/tailwind.config.js
@@ -0,0 +1,14 @@
+const { createGlobPatternsForDependencies } = require('@nx/angular/tailwind');
+const { join } = require('path');
+
+/** @type {import('tailwindcss').Config} */
+module.exports = {
+ content: [
+ join(__dirname, 'src/**/!(*.stories|*.spec).{ts,html}'),
+ ...createGlobPatternsForDependencies(__dirname),
+ ],
+ theme: {
+ extend: {},
+ },
+ plugins: [],
+};
diff --git a/apps/performance/ngfor-biglist/tsconfig.app.json b/apps/performance/ngfor-biglist/tsconfig.app.json
new file mode 100644
index 000000000..58220429a
--- /dev/null
+++ b/apps/performance/ngfor-biglist/tsconfig.app.json
@@ -0,0 +1,10 @@
+{
+ "extends": "./tsconfig.json",
+ "compilerOptions": {
+ "outDir": "../../../dist/out-tsc",
+ "types": []
+ },
+ "files": ["src/main.ts"],
+ "include": ["src/**/*.d.ts"],
+ "exclude": ["jest.config.ts", "src/**/*.test.ts", "src/**/*.spec.ts"]
+}
diff --git a/apps/performance/ngfor-biglist/tsconfig.editor.json b/apps/performance/ngfor-biglist/tsconfig.editor.json
new file mode 100644
index 000000000..4ee639340
--- /dev/null
+++ b/apps/performance/ngfor-biglist/tsconfig.editor.json
@@ -0,0 +1,7 @@
+{
+ "extends": "./tsconfig.json",
+ "include": ["src/**/*.ts"],
+ "compilerOptions": {
+ "types": []
+ }
+}
diff --git a/apps/performance/ngfor-biglist/tsconfig.json b/apps/performance/ngfor-biglist/tsconfig.json
new file mode 100644
index 000000000..51c7908c5
--- /dev/null
+++ b/apps/performance/ngfor-biglist/tsconfig.json
@@ -0,0 +1,29 @@
+{
+ "compilerOptions": {
+ "target": "es2022",
+ "useDefineForClassFields": false,
+ "forceConsistentCasingInFileNames": true,
+ "strict": true,
+ "noImplicitOverride": true,
+ "noPropertyAccessFromIndexSignature": true,
+ "noImplicitReturns": true,
+ "noFallthroughCasesInSwitch": true
+ },
+ "files": [],
+ "include": [],
+ "references": [
+ {
+ "path": "./tsconfig.app.json"
+ },
+ {
+ "path": "./tsconfig.editor.json"
+ }
+ ],
+ "extends": "../../../tsconfig.base.json",
+ "angularCompilerOptions": {
+ "enableI18nLegacyMessageIdFormat": false,
+ "strictInjectionParameters": true,
+ "strictInputAccessModifiers": true,
+ "strictTemplates": true
+ }
+}
diff --git a/challenge-number.json b/challenge-number.json
index a6d987c39..14d6f15fe 100644
--- a/challenge-number.json
+++ b/challenge-number.json
@@ -1,6 +1,6 @@
{
- "total": 36,
+ "total": 37,
"🟢": 13,
- "🟠": 116,
+ "🟠": 117,
"🔴": 207
}
diff --git a/docs/src/content/docs/challenges/angular-performance/35-memoize.md b/docs/src/content/docs/challenges/angular-performance/35-memoize.md
index 54fa6aacb..0c774bbc1 100644
--- a/docs/src/content/docs/challenges/angular-performance/35-memoize.md
+++ b/docs/src/content/docs/challenges/angular-performance/35-memoize.md
@@ -1,6 +1,6 @@
---
title: 🟢 Memoization
-description: Challenge 35 is about learning
+description: Challenge 35 is about learning how pure pipe works
sidebar:
order: 8
---
diff --git a/docs/src/content/docs/challenges/angular-performance/36-ngfor-optimize.md b/docs/src/content/docs/challenges/angular-performance/36-ngfor-optimize.md
index 5fef237ff..8a043f10d 100644
--- a/docs/src/content/docs/challenges/angular-performance/36-ngfor-optimize.md
+++ b/docs/src/content/docs/challenges/angular-performance/36-ngfor-optimize.md
@@ -1,9 +1,8 @@
---
title: 🟢 NgFor Optimization
-description: Challenge 36 is about ...
+description: Challenge 36 is about learning how trackby works
sidebar:
order: 13
- badge: New
---
Challenge #36
diff --git a/docs/src/content/docs/challenges/angular-performance/37-ngfor-biglist.md b/docs/src/content/docs/challenges/angular-performance/37-ngfor-biglist.md
new file mode 100644
index 000000000..c3691b75e
--- /dev/null
+++ b/docs/src/content/docs/challenges/angular-performance/37-ngfor-biglist.md
@@ -0,0 +1,55 @@
+---
+title: 🟠 NgFor Optimization Big List
+description: Challenge 37 is about learning how virtualization optimize big list rendering
+sidebar:
+ order: 117
+ badge: New
+---
+
+Challenge #37
+
+## Information
+
+In this application, we can render a list of 100,000 individuals by clicking on the **loadList** button. If you open the Chrome developer panel by pressing **F12**, go to the Source tab, and expand the element to see the list, you will notice that all 100,000 elements are rendered in the DOM, even though we can only see about 20 elements in the viewport. This process takes a lot of time, which is why the application is very slow at displaying the list.
+
+We can use the Angular DevTool to profile our application and understand what is happening inside our application. I will show you how to do it inside the following video.
+
+:::note
+If you don't know how to use it, read [the performance introduction page](/challenges/angular-performance/) first and come back after.
+:::
+
+## Statement
+
+The goal of this challenge is to implement a better alternative to display big list of items.
+
+## Hints:
+
+
+ Hint 1
+
+If you're unsure where to begin, I recommend reading the [Angular CDK virtualization documentation](https://material.angular.io/cdk/scrolling/overview)
+
+
+
+---
+
+:::note
+Start the project by running: `npx nx serve performance-ngfor-biglist`.
+:::
+
+:::tip[Reminder]
+Your PR title must start with Answer:37 .
+:::
+
+
diff --git a/docs/src/content/docs/index.mdx b/docs/src/content/docs/index.mdx
index 70e1f8627..082f1f097 100644
--- a/docs/src/content/docs/index.mdx
+++ b/docs/src/content/docs/index.mdx
@@ -23,8 +23,8 @@ hero:
import { Card, CardGrid } from '@astrojs/starlight/components';
-
- This repository gathers 36 Challenges related to Angular , Nx , RxJS , Ngrx and Typescript .
+
+ This repository gathers 37 Challenges related to Angular , Nx , RxJS , Ngrx and Typescript .
These challenges resolve around real-life issues or specific features to elevate your skills.