/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import {Component, destroyPlatform, EventEmitter, Injector, Input, NgModule, Output, Renderer2, ViewEncapsulation} from '@angular/core'; import {TestBed} from '@angular/core/testing'; import {BrowserModule} from '@angular/platform-browser'; import {browserDetection} from '@angular/platform-browser/testing/src/browser_util'; import {expect} from '@angular/platform-browser/testing/src/matchers'; if (browserDetection.supportsShadowDom) { describe('ShadowDOM Support', () => { let testContainer: HTMLDivElement; beforeEach(() => { TestBed.configureTestingModule({imports: [TestModule]}); }); it('should attach and use a shadowRoot when ViewEncapsulation.Native is set', () => { const compEl = TestBed.createComponent(ShadowComponent).nativeElement; expect(compEl.shadowRoot!.textContent).toEqual('Hello World'); }); it('should use the shadow root to encapsulate styles', () => { const compEl = TestBed.createComponent(StyledShadowComponent).nativeElement; // Firefox and Chrome return different computed styles. Chrome supports CSS property // shorthands in the computed style object while Firefox expects explicit CSS properties. // e.g. we can't use the "border" CSS property for this test as "border" is a shorthand // property and therefore would not work within Firefox. expect(window.getComputedStyle(compEl).backgroundColor).toEqual('rgb(0, 0, 0)'); const redDiv = compEl.shadowRoot.querySelector('div.red'); expect(window.getComputedStyle(redDiv).backgroundColor).toEqual('rgb(255, 0, 0)'); }); it('should allow the usage of elements', () => { const el = TestBed.createComponent(ShadowSlotComponent).nativeElement; const projectedContent = document.createTextNode('Hello Slot!'); el.appendChild(projectedContent); const slot = el.shadowRoot!.querySelector('slot'); expect(slot!.assignedNodes().length).toBe(1); expect(slot!.assignedNodes()[0].textContent).toBe('Hello Slot!'); }); it('should allow the usage of named elements', () => { const el = TestBed.createComponent(ShadowSlotsComponent).nativeElement; const headerContent = document.createElement('h1'); headerContent.setAttribute('slot', 'header'); headerContent.textContent = 'Header Text!'; const articleContent = document.createElement('span'); articleContent.setAttribute('slot', 'article'); articleContent.textContent = 'Article Text!'; const articleSubcontent = document.createElement('span'); articleSubcontent.setAttribute('slot', 'article'); articleSubcontent.textContent = 'Article Subtext!'; el.appendChild(headerContent); el.appendChild(articleContent); el.appendChild(articleSubcontent); const headerSlot = el.shadowRoot!.querySelector('slot[name=header]') as HTMLSlotElement; const articleSlot = el.shadowRoot!.querySelector('slot[name=article]') as HTMLSlotElement; expect(headerSlot!.assignedNodes().length).toBe(1); expect(headerSlot!.assignedNodes()[0].textContent).toBe('Header Text!'); expect(headerContent.assignedSlot).toBe(headerSlot); expect(articleSlot!.assignedNodes().length).toBe(2); expect(articleSlot!.assignedNodes()[0].textContent).toBe('Article Text!'); expect(articleSlot!.assignedNodes()[1].textContent).toBe('Article Subtext!'); expect(articleContent.assignedSlot).toBe(articleSlot); expect(articleSubcontent.assignedSlot).toBe(articleSlot); }); }); } @Component( {selector: 'shadow-comp', template: 'Hello World', encapsulation: ViewEncapsulation.ShadowDom}) class ShadowComponent { } @Component({ selector: 'styled-shadow-comp', template: '
', encapsulation: ViewEncapsulation.ShadowDom, styles: [`:host { background: black; } .red { background: red; }`] }) class StyledShadowComponent { } @Component({ selector: 'shadow-slot-comp', template: '', encapsulation: ViewEncapsulation.ShadowDom }) class ShadowSlotComponent { } @Component({ selector: 'shadow-slots-comp', template: '
', encapsulation: ViewEncapsulation.ShadowDom }) class ShadowSlotsComponent { } @NgModule({ imports: [BrowserModule], declarations: [ShadowComponent, ShadowSlotComponent, ShadowSlotsComponent, StyledShadowComponent], entryComponents: [ShadowComponent, ShadowSlotComponent, ShadowSlotsComponent, StyledShadowComponent], }) class TestModule { ngDoBootstrap() {} }