Skip to content

Commit be8aacd

Browse files
committed
refactor(theme.directive): signal inputs, host bindings, cleanup, tests
1 parent 824495f commit be8aacd

File tree

2 files changed

+45
-15
lines changed

2 files changed

+45
-15
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,29 @@
1-
import { ElementRef, Renderer2 } from '@angular/core';
2-
import { TestBed } from '@angular/core/testing';
1+
import { Component, DebugElement, ElementRef, Renderer2 } from '@angular/core';
2+
import { ComponentFixture, TestBed } from '@angular/core/testing';
33
import { ThemeDirective } from './theme.directive';
4+
import { By } from '@angular/platform-browser';
5+
6+
@Component({
7+
imports: [ThemeDirective],
8+
template: '<div cTheme [colorScheme]="theme"></div>'
9+
})
10+
export class TestComponent {
11+
theme!: 'dark' | 'light' | undefined;
12+
}
413

514
class MockElementRef extends ElementRef {}
615

716
describe('ThemeDirective', () => {
17+
let fixture: ComponentFixture<TestComponent>;
18+
let debugElement: DebugElement;
819

920
beforeEach(() => {
1021
TestBed.configureTestingModule({
22+
imports: [TestComponent],
1123
providers: [{ provide: ElementRef, useClass: MockElementRef }, Renderer2]
1224
});
25+
fixture = TestBed.createComponent(TestComponent);
26+
debugElement = fixture.debugElement.query(By.css('div'));
1327
});
1428

1529
it('should create an instance', () => {
@@ -18,4 +32,18 @@ describe('ThemeDirective', () => {
1832
expect(directive).toBeTruthy();
1933
});
2034
});
35+
36+
it('should set data-coreui-theme attribute', () => {
37+
fixture.detectChanges();
38+
expect(debugElement.nativeElement.getAttribute('data-coreui-theme')).toBeNull();
39+
fixture.componentInstance.theme = 'dark';
40+
fixture.detectChanges();
41+
expect(debugElement.nativeElement.getAttribute('data-coreui-theme')).toBe('dark');
42+
fixture.componentInstance.theme = 'light';
43+
fixture.detectChanges();
44+
expect(debugElement.nativeElement.getAttribute('data-coreui-theme')).toBe('light');
45+
fixture.componentInstance.theme = undefined;
46+
fixture.detectChanges();
47+
expect(debugElement.nativeElement.getAttribute('data-coreui-theme')).toBeNull();
48+
});
2149
});

projects/coreui-angular/src/lib/shared/theme.directive.ts

+15-13
Original file line numberDiff line numberDiff line change
@@ -1,28 +1,30 @@
1-
import { booleanAttribute, Directive, ElementRef, inject, Input, Renderer2 } from '@angular/core';
1+
import { booleanAttribute, Directive, effect, ElementRef, inject, input, Renderer2 } from '@angular/core';
22

33
@Directive({
4-
selector: '[cTheme]'
4+
selector: '[cTheme]',
5+
exportAs: 'cTheme'
56
})
67
export class ThemeDirective {
78
readonly #hostElement = inject(ElementRef);
89
readonly #renderer = inject(Renderer2);
910

1011
/**
1112
* Add dark theme attribute.
12-
* @type 'dark' | 'light' | undefined
13+
* @return 'dark' | 'light' | undefined
1314
*/
14-
@Input() set colorScheme(scheme: 'dark' | 'light' | undefined) {
15-
!!scheme ? this.setTheme(scheme) : this.unsetTheme();
16-
}
15+
readonly colorScheme = input<'dark' | 'light'>();
1716

18-
/**
19-
* Add dark theme attribute.
20-
* @type boolean
21-
*/
22-
@Input({ transform: booleanAttribute })
23-
set dark(darkTheme: boolean) {
17+
readonly #colorSchemeChange = effect(() => {
18+
const colorScheme = this.colorScheme();
19+
colorScheme ? this.setTheme(colorScheme) : this.unsetTheme();
20+
});
21+
22+
readonly dark = input(false, { transform: booleanAttribute });
23+
24+
readonly #darkChange = effect(() => {
25+
const darkTheme = this.dark();
2426
darkTheme ? this.setTheme('dark') : this.unsetTheme();
25-
}
27+
});
2628

2729
setTheme(theme?: string): void {
2830
if (theme) {

0 commit comments

Comments
 (0)