diff --git a/.editorconfig b/.editorconfig
index 8bed5ad4df..45c01ff049 100644
--- a/.editorconfig
+++ b/.editorconfig
@@ -19,7 +19,7 @@ tab_width = 4
[*.{yml, yaml}]
indent_size = 2
-[*.{js,jsx,ts,tsx}]
+[*.{js,jsx,ts,tsx,mjs,mjsx,tsx,mtsx}]
indent_size = 4
[*.html]
diff --git a/e2e/zoneless-env/__tests__/zoneless-env.spec.ts b/e2e/zoneless-env/__tests__/zoneless-env.spec.ts
new file mode 100644
index 0000000000..ea305d8c1b
--- /dev/null
+++ b/e2e/zoneless-env/__tests__/zoneless-env.spec.ts
@@ -0,0 +1,18 @@
+import { TestBed } from '@angular/core/testing';
+
+import { FooComponent } from '../foo.component';
+
+describe('FooComponent', () => {
+ it('should trigger change detection without fixture.detectChanges', () => {
+ TestBed.configureTestingModule({
+ imports: [FooComponent],
+ });
+ const fixture = TestBed.createComponent(FooComponent);
+
+ expect(fixture.componentInstance.value1()).toBe('val1');
+
+ fixture.componentRef.setInput('value1', 'hello');
+
+ expect(fixture.componentInstance.value1()).toBe('hello');
+ });
+});
diff --git a/e2e/zoneless-env/foo.component.html b/e2e/zoneless-env/foo.component.html
new file mode 100644
index 0000000000..cdd69d0efa
--- /dev/null
+++ b/e2e/zoneless-env/foo.component.html
@@ -0,0 +1,10 @@
+
+
Line 1
+
+
+ {{ value1() }}
+
+
+ {{ value2() }}
+
+
diff --git a/e2e/zoneless-env/foo.component.scss b/e2e/zoneless-env/foo.component.scss
new file mode 100644
index 0000000000..c2393c27c0
--- /dev/null
+++ b/e2e/zoneless-env/foo.component.scss
@@ -0,0 +1,3 @@
+p {
+ font-size: 1.6rem;
+}
diff --git a/e2e/zoneless-env/foo.component.ts b/e2e/zoneless-env/foo.component.ts
new file mode 100644
index 0000000000..62407aed74
--- /dev/null
+++ b/e2e/zoneless-env/foo.component.ts
@@ -0,0 +1,25 @@
+import { NgIf } from '@angular/common';
+import { Component, input } from '@angular/core';
+
+@Component({
+ selector: 'foo',
+ standalone: true,
+ templateUrl: './foo.component.html',
+ styleUrls: ['./foo.component.scss'],
+ // we have to setup styles this way, since simple styles/styleUrs properties will be removed (jest does not unit test styles)
+ styles: [
+ `
+ p {
+ color: red;
+ }
+ `,
+ ],
+ imports: [NgIf],
+})
+export class FooComponent {
+ readonly value1 = input('val1');
+ readonly value2 = input('val2');
+
+ protected readonly condition1 = true;
+ protected readonly condition2 = false;
+}
diff --git a/e2e/zoneless-env/jest-cjs.config.ts b/e2e/zoneless-env/jest-cjs.config.ts
new file mode 100644
index 0000000000..13b7970b6e
--- /dev/null
+++ b/e2e/zoneless-env/jest-cjs.config.ts
@@ -0,0 +1,24 @@
+import type { JestConfigWithTsJest } from 'ts-jest';
+
+const config: JestConfigWithTsJest = {
+ displayName: 'e2e-zoneless-env',
+ testEnvironment: 'jsdom',
+ snapshotSerializers: [
+ '/../../build/serializers/html-comment',
+ '/../../build/serializers/ng-snapshot',
+ '/../../build/serializers/no-ng-attributes',
+ ],
+ setupFilesAfterEnv: ['./setup-zoneless-env.ts'],
+ transform: {
+ '^.+\\.(ts|js|mjs|html)$': [
+ '/../../build/index.js',
+ {
+ tsconfig: '/tsconfig-cjs.spec.json',
+ stringifyContentPathRegex: '\\.(html|svg)$',
+ },
+ ],
+ },
+ transformIgnorePatterns: ['node_modules/(?!.*\\.mjs$)'],
+};
+
+export default config;
diff --git a/e2e/zoneless-env/jest-esm.config.ts b/e2e/zoneless-env/jest-esm.config.ts
new file mode 100644
index 0000000000..c521da99eb
--- /dev/null
+++ b/e2e/zoneless-env/jest-esm.config.ts
@@ -0,0 +1,28 @@
+import type { JestConfigWithTsJest } from 'ts-jest';
+
+const config: JestConfigWithTsJest = {
+ displayName: 'e2e-zoneless-env',
+ testEnvironment: 'jsdom',
+ snapshotSerializers: [
+ '/../../build/serializers/html-comment',
+ '/../../build/serializers/ng-snapshot',
+ '/../../build/serializers/no-ng-attributes',
+ ],
+ setupFilesAfterEnv: ['./setup-zoneless-env.mts'],
+ moduleNameMapper: {
+ rxjs: '/../../node_modules/rxjs/dist/bundles/rxjs.umd.js',
+ },
+ extensionsToTreatAsEsm: ['.ts', '.mts'],
+ transform: {
+ '^.+\\.(ts|mts|js|mjs|html)$': [
+ '/../../build/index.js',
+ {
+ useESM: true,
+ tsconfig: '/tsconfig-esm.spec.json',
+ stringifyContentPathRegex: '\\.(html|svg)$',
+ },
+ ],
+ },
+};
+
+export default config;
diff --git a/e2e/zoneless-env/jest-transpile-cjs.config.ts b/e2e/zoneless-env/jest-transpile-cjs.config.ts
new file mode 100644
index 0000000000..46d6084eee
--- /dev/null
+++ b/e2e/zoneless-env/jest-transpile-cjs.config.ts
@@ -0,0 +1,25 @@
+import type { JestConfigWithTsJest } from 'ts-jest';
+
+const config: JestConfigWithTsJest = {
+ displayName: 'e2e-snapshot-serializers',
+ testEnvironment: 'jsdom',
+ snapshotSerializers: [
+ '/../../build/serializers/html-comment',
+ '/../../build/serializers/ng-snapshot',
+ '/../../build/serializers/no-ng-attributes',
+ ],
+ setupFilesAfterEnv: ['./setup-zoneless-env.ts'],
+ transform: {
+ '^.+\\.(ts|js|mjs|html)$': [
+ '/../../build/index.js',
+ {
+ tsconfig: '/tsconfig-cjs.spec.json',
+ stringifyContentPathRegex: '\\.(html|svg)$',
+ isolatedModules: true,
+ },
+ ],
+ },
+ transformIgnorePatterns: ['node_modules/(?!.*\\.mjs$)'],
+};
+
+export default config;
diff --git a/e2e/zoneless-env/jest-transpile-esm.config.ts b/e2e/zoneless-env/jest-transpile-esm.config.ts
new file mode 100644
index 0000000000..1e57840617
--- /dev/null
+++ b/e2e/zoneless-env/jest-transpile-esm.config.ts
@@ -0,0 +1,29 @@
+import type { JestConfigWithTsJest } from 'ts-jest';
+
+const config: JestConfigWithTsJest = {
+ displayName: 'e2e-snapshot-serializers',
+ testEnvironment: 'jsdom',
+ snapshotSerializers: [
+ '/../../build/serializers/html-comment',
+ '/../../build/serializers/ng-snapshot',
+ '/../../build/serializers/no-ng-attributes',
+ ],
+ setupFilesAfterEnv: ['./setup-zoneless-env.mts'],
+ moduleNameMapper: {
+ rxjs: '/../../node_modules/rxjs/dist/bundles/rxjs.umd.js',
+ },
+ extensionsToTreatAsEsm: ['.ts', '.mts'],
+ transform: {
+ '^.+\\.(ts|mts|js|mjs|html)$': [
+ '/../../build/index.js',
+ {
+ useESM: true,
+ tsconfig: '/tsconfig-esm.spec.json',
+ stringifyContentPathRegex: '\\.(html|svg)$',
+ isolatedModules: true,
+ },
+ ],
+ },
+};
+
+export default config;
diff --git a/e2e/zoneless-env/setup-zoneless-env.mts b/e2e/zoneless-env/setup-zoneless-env.mts
new file mode 100644
index 0000000000..6375b3715a
--- /dev/null
+++ b/e2e/zoneless-env/setup-zoneless-env.mts
@@ -0,0 +1,3 @@
+import { setupZonelessTestEnv } from '../../setup-env/zoneless/index.mjs';
+
+setupZonelessTestEnv();
diff --git a/e2e/zoneless-env/setup-zoneless-env.ts b/e2e/zoneless-env/setup-zoneless-env.ts
new file mode 100644
index 0000000000..b16ec598be
--- /dev/null
+++ b/e2e/zoneless-env/setup-zoneless-env.ts
@@ -0,0 +1,3 @@
+import { setupZonelessTestEnv } from '../../setup-env/zoneless';
+
+setupZonelessTestEnv();
diff --git a/e2e/zoneless-env/tsconfig-cjs.spec.json b/e2e/zoneless-env/tsconfig-cjs.spec.json
new file mode 100644
index 0000000000..f65fc7b4f1
--- /dev/null
+++ b/e2e/zoneless-env/tsconfig-cjs.spec.json
@@ -0,0 +1,3 @@
+{
+ "extends": "../../tsconfig-base.spec.json"
+}
diff --git a/e2e/zoneless-env/tsconfig-esm.spec.json b/e2e/zoneless-env/tsconfig-esm.spec.json
new file mode 100644
index 0000000000..2007b41c6e
--- /dev/null
+++ b/e2e/zoneless-env/tsconfig-esm.spec.json
@@ -0,0 +1,7 @@
+{
+ "extends": "../../tsconfig-base.spec.json",
+ "compilerOptions": {
+ "module": "ES2022",
+ "esModuleInterop": true
+ }
+}
diff --git a/setup-env/utils.mjs b/setup-env/utils.js
similarity index 71%
rename from setup-env/utils.mjs
rename to setup-env/utils.js
index eceb631a4e..41b592793a 100644
--- a/setup-env/utils.mjs
+++ b/setup-env/utils.js
@@ -1,13 +1,13 @@
-import { TextDecoder, TextEncoder } from 'util';
+const { TextDecoder, TextEncoder } = require('util');
-export const polyfillEncoder = () => {
+const polyfillEncoder = () => {
if (typeof globalThis.TextEncoder === 'undefined') {
globalThis.TextEncoder = TextEncoder;
globalThis.TextDecoder = TextDecoder;
}
};
-export const resolveTestEnvOptions = (options) => {
+const resolveTestEnvOptions = (options) => {
const globalTestEnvOptions = globalThis.ngJest?.testEnvironmentOptions;
if (globalTestEnvOptions) {
console.warn(
@@ -17,3 +17,8 @@ export const resolveTestEnvOptions = (options) => {
return globalTestEnvOptions ?? options;
};
+
+module.exports = {
+ polyfillEncoder,
+ resolveTestEnvOptions,
+};
diff --git a/setup-env/zone/index.js b/setup-env/zone/index.js
index b2b2d2a17a..31301cbf85 100644
--- a/setup-env/zone/index.js
+++ b/setup-env/zone/index.js
@@ -1,38 +1,20 @@
require('zone.js');
require('zone.js/testing');
-const { TextEncoder, TextDecoder } = require('util');
-
const { getTestBed } = require('@angular/core/testing');
const {
BrowserDynamicTestingModule,
platformBrowserDynamicTesting,
} = require('@angular/platform-browser-dynamic/testing');
-const polyfillEncoder = () => {
- if (typeof globalThis.TextEncoder === 'undefined') {
- globalThis.TextEncoder = TextEncoder;
- globalThis.TextDecoder = TextDecoder;
- }
-};
-
-const resolveTestEnvOptions = (options) => {
- const globalTestEnvOptions = globalThis.ngJest?.testEnvironmentOptions;
- if (globalTestEnvOptions) {
- console.warn(
- 'Setting testEnvironmentOptions via globalThis.ngJest is deprecated. Please provide testEnvironmentOptions via function argument',
- );
- }
-
- return globalTestEnvOptions ?? options;
-};
+const { polyfillEncoder, resolveTestEnvOptions } = require('../utils');
const setupZoneTestEnv = (options) => {
polyfillEncoder();
const testEnvironmentOptions = resolveTestEnvOptions(options);
getTestBed().initTestEnvironment(
- BrowserDynamicTestingModule,
+ [BrowserDynamicTestingModule],
platformBrowserDynamicTesting(),
testEnvironmentOptions,
);
diff --git a/setup-env/zone/index.mjs b/setup-env/zone/index.mjs
index fe0d2a993c..68fe0afbb0 100644
--- a/setup-env/zone/index.mjs
+++ b/setup-env/zone/index.mjs
@@ -4,15 +4,19 @@ import 'zone.js/testing';
import { getTestBed } from '@angular/core/testing';
import { BrowserDynamicTestingModule, platformBrowserDynamicTesting } from '@angular/platform-browser-dynamic/testing';
-import { polyfillEncoder, resolveTestEnvOptions } from '../utils.mjs';
+import { polyfillEncoder, resolveTestEnvOptions } from '../utils';
-export const setupZoneTestEnv = (options) => {
+const setupZoneTestEnv = (options) => {
polyfillEncoder();
const testEnvironmentOptions = resolveTestEnvOptions(options);
getTestBed().initTestEnvironment(
- BrowserDynamicTestingModule,
+ [BrowserDynamicTestingModule],
platformBrowserDynamicTesting(),
testEnvironmentOptions,
);
};
+
+export {
+ setupZoneTestEnv,
+}
diff --git a/setup-env/zoneless/index.js b/setup-env/zoneless/index.js
index b725a157ad..6ef884f540 100644
--- a/setup-env/zoneless/index.js
+++ b/setup-env/zoneless/index.js
@@ -1,8 +1,48 @@
-// eslint-disable-next-line @typescript-eslint/no-unused-vars
-const setupZonelessTestEnv = (_options) => {
+const { provideExperimentalZonelessChangeDetection, NgModule, ErrorHandler } = require('@angular/core');
+const { getTestBed } = require('@angular/core/testing');
+const {
+ BrowserDynamicTestingModule,
+ platformBrowserDynamicTesting,
+} = require('@angular/platform-browser-dynamic/testing');
+
+const { polyfillEncoder, resolveTestEnvOptions } = require('../utils');
+
+const provideZonelessConfig = () => {
+ class TestModule {}
+ NgModule({
+ providers: [
+ provideExperimentalZonelessChangeDetection(),
+ {
+ provide: ErrorHandler,
+ useValue: {
+ handleError: (e) => {
+ throw e;
+ },
+ },
+ },
+ ],
+ })(TestModule);
+
+ return TestModule;
+};
+
+const setupZonelessTestEnv = (options) => {
+ polyfillEncoder();
+ if (typeof provideExperimentalZonelessChangeDetection !== 'undefined') {
+ const testEnvironmentOptions = resolveTestEnvOptions(options);
+
+ getTestBed().initTestEnvironment(
+ [BrowserDynamicTestingModule, provideZonelessConfig()],
+ platformBrowserDynamicTesting(),
+ testEnvironmentOptions,
+ );
+
+ return;
+ }
+
throw Error(
- 'Zoneless testing environment only works when running Jest in ESM mode with Jest 29. ' +
- 'Jest 30+ will support to work with CommonJS mode.',
+ 'Cannot find provideExperimentalZonelessChangeDetection() to setup zoneless testing environment. ' +
+ 'Please use setupZoneTestEnv() from jest-preset-angular/setup-env/setup-zone-env.mjs instead.',
);
};
diff --git a/setup-env/zoneless/index.mjs b/setup-env/zoneless/index.mjs
index b63cd62e6e..ce66a4cc90 100644
--- a/setup-env/zoneless/index.mjs
+++ b/setup-env/zoneless/index.mjs
@@ -2,29 +2,34 @@ import { ErrorHandler, NgModule, provideExperimentalZonelessChangeDetection } fr
import { getTestBed } from '@angular/core/testing';
import { BrowserDynamicTestingModule, platformBrowserDynamicTesting } from '@angular/platform-browser-dynamic/testing';
-import { polyfillEncoder, resolveTestEnvOptions } from '../utils.mjs';
+import { polyfillEncoder, resolveTestEnvOptions } from '../utils';
-export const setupZonelessTestEnv = (options) => {
- if (typeof provideExperimentalZonelessChangeDetection !== 'undefined') {
- polyfillEncoder();
- const testEnvironmentOptions = resolveTestEnvOptions(options);
- @NgModule({
- providers: [
- provideExperimentalZonelessChangeDetection(),
- {
- provide: ErrorHandler,
- useValue: {
- handleError: (e) => {
- throw e;
- },
+const provideZonelessConfig = () => {
+ class TestModule {}
+ NgModule({
+ providers: [
+ provideExperimentalZonelessChangeDetection(),
+ {
+ provide: ErrorHandler,
+ useValue: {
+ handleError: (e) => {
+ throw e;
},
},
- ],
- })
- export class TestModule {}
+ },
+ ],
+ })(TestModule);
+
+ return TestModule;
+};
+
+const setupZonelessTestEnv = (options) => {
+ polyfillEncoder();
+ if (typeof provideExperimentalZonelessChangeDetection !== 'undefined') {
+ const testEnvironmentOptions = resolveTestEnvOptions(options);
getTestBed().initTestEnvironment(
- [BrowserDynamicTestingModule, TestModule],
+ [BrowserDynamicTestingModule, provideZonelessConfig()],
platformBrowserDynamicTesting(),
testEnvironmentOptions,
);
@@ -34,6 +39,10 @@ export const setupZonelessTestEnv = (options) => {
throw Error(
'Cannot find provideExperimentalZonelessChangeDetection() to setup zoneless testing environment. ' +
- 'Please use setupZoneTestEnv() from jest-preset-angular/setup-env/setup-zone-env.mjs instead.',
+ 'Please use setupZoneTestEnv() from jest-preset-angular/setup-env/setup-zone-env.mjs instead.',
);
};
+
+export {
+ setupZonelessTestEnv
+}
diff --git a/setup-jest.js b/setup-jest.js
index 4528ff46e6..4ac9724c84 100644
--- a/setup-jest.js
+++ b/setup-jest.js
@@ -25,4 +25,8 @@ if (typeof globalThis.TextEncoder === 'undefined') {
const testEnvironmentOptions = globalThis.ngJest?.testEnvironmentOptions ?? Object.create(null);
-getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting(), testEnvironmentOptions);
+getTestBed().initTestEnvironment(
+ [BrowserDynamicTestingModule],
+ platformBrowserDynamicTesting(),
+ testEnvironmentOptions,
+);
diff --git a/setup-jest.mjs b/setup-jest.mjs
index 19752ef617..e4bd512a42 100644
--- a/setup-jest.mjs
+++ b/setup-jest.mjs
@@ -22,4 +22,4 @@ if (typeof globalThis.TextEncoder === 'undefined') {
const testEnvironmentOptions = globalThis.ngJest?.testEnvironmentOptions ?? Object.create(null);
-getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting(), testEnvironmentOptions);
+getTestBed().initTestEnvironment([BrowserDynamicTestingModule], platformBrowserDynamicTesting(), testEnvironmentOptions);
diff --git a/src/config/setup-jest.spec.ts b/src/config/setup-jest.spec.ts
index 1de0e5bd21..634f6fbbec 100644
--- a/src/config/setup-jest.spec.ts
+++ b/src/config/setup-jest.spec.ts
@@ -28,6 +28,7 @@ jest.mock('@angular/core/testing', () => {
class BrowserDynamicTestingModuleStub {}
class PlatformRefStub {}
+class ErrorHandlerStub {}
const mockPlatformBrowserDynamicTesting = jest.fn(() => new PlatformRefStub());
jest.mock('@angular/platform-browser-dynamic/testing', () => {
return {
@@ -35,14 +36,31 @@ jest.mock('@angular/platform-browser-dynamic/testing', () => {
platformBrowserDynamicTesting: mockPlatformBrowserDynamicTesting,
};
});
+const mockProvideExperimentalZonelessChangeDetection = jest.fn();
+jest.mock('@angular/core', () => {
+ return {
+ provideExperimentalZonelessChangeDetection: mockProvideExperimentalZonelessChangeDetection,
+ ErrorHandler: ErrorHandlerStub,
+ NgModule: () => {
+ return jest.fn();
+ },
+ };
+});
describe('setup-jest', () => {
const assertOnInitTestEnv = (): void => {
expect(mockGetTestBed).toHaveBeenCalled();
- expect(mockInitTestEnvironment).toHaveBeenCalled();
- expect(mockInitTestEnvironment.mock.calls[0][0]).toBeInstanceOf(BrowserDynamicTestingModuleStub);
+ expect(mockInitTestEnvironment.mock.calls[0][0][0]).toBeInstanceOf(BrowserDynamicTestingModuleStub);
expect(mockPlatformBrowserDynamicTesting).toHaveBeenCalled();
expect(mockPlatformBrowserDynamicTesting.mock.results[0].value).toBeInstanceOf(PlatformRefStub);
+ expect(mockInitTestEnvironment.mock.calls[0][2]).toEqual({
+ teardown: {
+ destroyAfterEach: false,
+ rethrowErrors: true,
+ },
+ errorOnUnknownElements: true,
+ errorOnUnknownProperties: true,
+ });
};
beforeEach(() => {
@@ -53,7 +71,7 @@ describe('setup-jest', () => {
});
describe('for CJS setup-jest, test environment initialization', () => {
- test('should call getTestBed() and initTestEnvironment() with the testEnvironmentOptions passed to ngJest', async () => {
+ it('should setup test environment with setup-jest file', async () => {
globalThis.ngJest = {
testEnvironmentOptions: {
teardown: {
@@ -65,28 +83,15 @@ describe('setup-jest', () => {
},
};
- await import('../../setup-jest');
+ await import('../../setup-jest.js');
+ expect(globalThis.TextEncoder).toBeDefined();
expect(mockZoneJs).toHaveBeenCalled();
expect(mockZoneJsTesting).toHaveBeenCalled();
assertOnInitTestEnv();
- expect(mockInitTestEnvironment.mock.calls[0][2]).toEqual({
- teardown: {
- destroyAfterEach: false,
- rethrowErrors: true,
- },
- errorOnUnknownElements: true,
- errorOnUnknownProperties: true,
- });
});
- test('should always have TextEncoder in globalThis', async () => {
- await import('../../setup-jest');
-
- expect(globalThis.TextEncoder).toBeDefined();
- });
-
- it('should call getTestBed() and initTestEnvironment() with the testEnvironmentOptions passed as argument with setupZoneTestEnv()', async () => {
+ it('should setup test environment with setupZoneTestEnv()', async () => {
const { setupZoneTestEnv } = await import('../../setup-env/zone/index.js');
setupZoneTestEnv({
@@ -98,10 +103,16 @@ describe('setup-jest', () => {
errorOnUnknownProperties: true,
});
+ expect(globalThis.TextEncoder).toBeDefined();
expect(mockZoneJs).toHaveBeenCalled();
expect(mockZoneJsTesting).toHaveBeenCalled();
assertOnInitTestEnv();
- expect(mockInitTestEnvironment.mock.calls[0][2]).toEqual({
+ });
+
+ it('should setup test environment with setupZonelessTestEnv()', async () => {
+ const { setupZonelessTestEnv } = await import('../../setup-env/zoneless/index.js');
+
+ setupZonelessTestEnv({
teardown: {
destroyAfterEach: false,
rethrowErrors: true,
@@ -109,19 +120,17 @@ describe('setup-jest', () => {
errorOnUnknownElements: true,
errorOnUnknownProperties: true,
});
- });
-
- it('should always have TextEncoder in globalThis with setupZoneTestEnv()', async () => {
- const { setupZoneTestEnv } = await import('../../setup-env/zone/index.js');
-
- setupZoneTestEnv();
expect(globalThis.TextEncoder).toBeDefined();
+ expect(mockZoneJs).not.toHaveBeenCalled();
+ expect(mockZoneJsTesting).not.toHaveBeenCalled();
+ assertOnInitTestEnv();
+ expect(mockProvideExperimentalZonelessChangeDetection).toHaveBeenCalled();
});
});
describe('for ESM setup-jest, test environment initialization', () => {
- test('should call getTestBed() and initTestEnvironment() with the testEnvironmentOptions passed to ngJest', async () => {
+ it('should setup test environment with setup-jest file', async () => {
globalThis.ngJest = {
testEnvironmentOptions: {
teardown: {
@@ -135,26 +144,13 @@ describe('setup-jest', () => {
await import('../../setup-jest.mjs');
+ expect(globalThis.TextEncoder).toBeDefined();
expect(mockZoneJs).toHaveBeenCalled();
expect(mockZoneJsTesting).toHaveBeenCalled();
assertOnInitTestEnv();
- expect(mockInitTestEnvironment.mock.calls[0][2]).toEqual({
- teardown: {
- destroyAfterEach: false,
- rethrowErrors: true,
- },
- errorOnUnknownElements: true,
- errorOnUnknownProperties: true,
- });
});
- test('should always have TextEncoder in globalThis', async () => {
- await import('../../setup-jest.mjs');
-
- expect(globalThis.TextEncoder).toBeDefined();
- });
-
- it('should call getTestBed() and initTestEnvironment() with the testEnvironmentOptions passed as argument with setupZoneTestEnv()', async () => {
+ it('should setup test environment with setupZoneTestEnv()', async () => {
const { setupZoneTestEnv } = await import('../../setup-env/zone/index.mjs');
setupZoneTestEnv({
@@ -166,10 +162,16 @@ describe('setup-jest', () => {
errorOnUnknownProperties: true,
});
+ expect(globalThis.TextEncoder).toBeDefined();
expect(mockZoneJs).toHaveBeenCalled();
expect(mockZoneJsTesting).toHaveBeenCalled();
assertOnInitTestEnv();
- expect(mockInitTestEnvironment.mock.calls[0][2]).toEqual({
+ });
+
+ it('should setup test environment with setupZonelessTestEnv()', async () => {
+ const { setupZonelessTestEnv } = await import('../../setup-env/zoneless/index.mjs');
+
+ setupZonelessTestEnv({
teardown: {
destroyAfterEach: false,
rethrowErrors: true,
@@ -177,14 +179,12 @@ describe('setup-jest', () => {
errorOnUnknownElements: true,
errorOnUnknownProperties: true,
});
- });
-
- it('should always have TextEncoder in globalThis with setupZoneTestEnv()', async () => {
- const { setupZoneTestEnv } = await import('../../setup-env/zone/index.mjs');
-
- setupZoneTestEnv();
expect(globalThis.TextEncoder).toBeDefined();
+ expect(mockZoneJs).not.toHaveBeenCalled();
+ expect(mockZoneJsTesting).not.toHaveBeenCalled();
+ assertOnInitTestEnv();
+ expect(mockProvideExperimentalZonelessChangeDetection).toHaveBeenCalled();
});
});
});
diff --git a/website/docs/getting-started/test-environment.md b/website/docs/getting-started/test-environment.md
index f0207352d0..6341cd35e9 100644
--- a/website/docs/getting-started/test-environment.md
+++ b/website/docs/getting-started/test-environment.md
@@ -82,12 +82,6 @@ export default jestConfig;
Configures a test environment that **DOESN'T** use `zone.js`, as described in [Angular experimental zoneless guide](https://angular.dev/guide/experimental/zoneless).
It is designed for projects that have disabled `zone.js`, which can lead to improved performance and simplified testing.
-:::important
-
-This function is only supported in Jest `ESM` mode in [Jest 29](https://github.com/jestjs/jest/issues/10962). Jest 30+ will support to use for `CommonJS` mode.
-
-:::
-
You can customize the environment by providing options as function arguments.
#### Parameters
@@ -98,6 +92,15 @@ You can customize the environment by providing options as function arguments.
- Create a Jest setup file:
+```ts tab={"label": "TypeScript CJS"}
+// setup-jest.ts
+import { setupZonelessTestEnv } from 'jest-preset-angular/setup-env/zoneless';
+
+setupZonelessTestEnv({
+ //...options
+});
+```
+
```ts tab={"label": "TypeScript ESM"}
// setup-jest.ts
import { setupZonelessTestEnv } from 'jest-preset-angular/setup-env/zoneless/index.mjs';
@@ -109,6 +112,18 @@ setupZonelessTestEnv({
- Update your Jest configuration:
+```ts tab={"label": "TypeScript CJS"}
+// jest.config.mts
+import type { Config } from 'jest';
+
+const jestConfig: Config = {
+ preset: 'jest-preset-angular',
+ setupFilesAfterEnv: ['/setup-jest.ts'],
+};
+
+export default jestConfig;
+```
+
```ts tab={"label": "TypeScript ESM"}
// jest.config.mts
import type { Config } from 'jest';
diff --git a/website/versioned_docs/version-14.3/getting-started/test-environment.md b/website/versioned_docs/version-14.3/getting-started/test-environment.md
index f0207352d0..6341cd35e9 100644
--- a/website/versioned_docs/version-14.3/getting-started/test-environment.md
+++ b/website/versioned_docs/version-14.3/getting-started/test-environment.md
@@ -82,12 +82,6 @@ export default jestConfig;
Configures a test environment that **DOESN'T** use `zone.js`, as described in [Angular experimental zoneless guide](https://angular.dev/guide/experimental/zoneless).
It is designed for projects that have disabled `zone.js`, which can lead to improved performance and simplified testing.
-:::important
-
-This function is only supported in Jest `ESM` mode in [Jest 29](https://github.com/jestjs/jest/issues/10962). Jest 30+ will support to use for `CommonJS` mode.
-
-:::
-
You can customize the environment by providing options as function arguments.
#### Parameters
@@ -98,6 +92,15 @@ You can customize the environment by providing options as function arguments.
- Create a Jest setup file:
+```ts tab={"label": "TypeScript CJS"}
+// setup-jest.ts
+import { setupZonelessTestEnv } from 'jest-preset-angular/setup-env/zoneless';
+
+setupZonelessTestEnv({
+ //...options
+});
+```
+
```ts tab={"label": "TypeScript ESM"}
// setup-jest.ts
import { setupZonelessTestEnv } from 'jest-preset-angular/setup-env/zoneless/index.mjs';
@@ -109,6 +112,18 @@ setupZonelessTestEnv({
- Update your Jest configuration:
+```ts tab={"label": "TypeScript CJS"}
+// jest.config.mts
+import type { Config } from 'jest';
+
+const jestConfig: Config = {
+ preset: 'jest-preset-angular',
+ setupFilesAfterEnv: ['/setup-jest.ts'],
+};
+
+export default jestConfig;
+```
+
```ts tab={"label": "TypeScript ESM"}
// jest.config.mts
import type { Config } from 'jest';