From 27ea95aef2dd2ff027c14f7f6882f03cd10f0840 Mon Sep 17 00:00:00 2001 From: clouless Date: Sun, 22 Oct 2017 19:47:08 +0200 Subject: [PATCH] tooltip theme --- package.json | 3 +- src/components/children/tooltip.component.ts | 63 ++++++++++++++++---- src/components/tooltip.directive.ts | 6 ++ src/components/tooltip.module.ts | 13 +++- src/components/tooltip.theme.ts | 56 +++++++++++++++++ src/demo/demo.component.html | 8 +-- src/demo/demo.component.ts | 4 +- src/demo/demo.module.ts | 15 +++-- 8 files changed, 141 insertions(+), 27 deletions(-) create mode 100644 src/components/tooltip.theme.ts diff --git a/package.json b/package.json index 4a12a1d..afefca5 100644 --- a/package.json +++ b/package.json @@ -18,7 +18,8 @@ "test": "echo \"ok\"" }, "dependencies": { - "@cloukit/dropout": "1.1.0" + "@cloukit/dropout": ">=1.1.0", + "@cloukit/theme": ">=1.6.1" }, "devDependencies": { "@cloukit/library-build-chain": "1.7.1" diff --git a/src/components/children/tooltip.component.ts b/src/components/children/tooltip.component.ts index c532002..fd4a162 100644 --- a/src/components/children/tooltip.component.ts +++ b/src/components/children/tooltip.component.ts @@ -5,13 +5,19 @@ */ import { Component, Input, ViewChild, TemplateRef, OnInit } from '@angular/core'; import { DropoutPlacement } from '@cloukit/dropout'; +import { + CloukitComponentTheme, CloukitStatefulAndModifierAwareElementThemeStyleDefinition, + CloukitThemeService, +} from '@cloukit/theme'; @Component({ selector: 'cloukit-tooltip', template: ` -
- {{tooltipText}} +
+
+ {{tooltipText}} +
`, @@ -25,23 +31,58 @@ export class CloukitTooltipComponent implements OnInit { public tooltipTemplate: TemplateRef; @Input('cloukitTooltipPlacement') - cloukitDropoutPlacement: DropoutPlacement; + public cloukitDropoutPlacement: DropoutPlacement; - public style = { - backgroundColor: '#333', - padding: '4px', - color: '#fff', - transform: '', + @Input() + public wrapperMargin: string; + + @Input() + public theme: string; + + private themeSelected: CloukitComponentTheme; + private state = { + uiModifier: 'base', + uiState: 'init', + tooltipTransform: '', }; + constructor(private themeService: CloukitThemeService) { + this.themeSelected = this.themeService.getComponentTheme('tooltip'); + } + + public getStyle(element: string): CloukitStatefulAndModifierAwareElementThemeStyleDefinition { + if (this.themeSelected !== undefined && this.themeSelected !== null) { + const style = this.themeSelected.getStyle(element, this.state.uiState, this.state.uiModifier); + if (element === 'tooltip') { + style.style[ 'transform' ] = this.state.tooltipTransform; + } + if (element === 'wrapper') { + style.style[this.wrapperMargin] = this.state.uiState === 'ready' ? '5px' : '0px'; + } + return this.themeService.prefixStyle(style); + } + return { style: {}, icon: {} } as CloukitStatefulAndModifierAwareElementThemeStyleDefinition; + } + ngOnInit() { - console.log('changes'); + const self = this; if (this.cloukitDropoutPlacement === DropoutPlacement.DOWN_CENTER || this.cloukitDropoutPlacement === DropoutPlacement.UP_CENTER ) { - this.style.transform = 'translate(-50%, 0)'; + this.state.tooltipTransform = 'translate(-50%, 0)'; } if (this.cloukitDropoutPlacement === DropoutPlacement.LEFT_CENTER || this.cloukitDropoutPlacement === DropoutPlacement.RIGHT_CENTER ) { - this.style.transform = 'translate(0, -50%)'; + this.state.tooltipTransform = 'translate(0, -50%)'; + } + if (this.theme !== undefined && this.theme !== null) { + this.themeSelected = this.themeService.getComponentTheme(this.theme); + if (this.themeSelected === null) { + console.log(`WARN: requested theme ${this.theme} does not exist. Falling back to default theme for tooltip.`); + this.themeSelected = this.themeService.getComponentTheme('tooltip'); + } } + // Transition to ready state once component is created + setTimeout(() => { + self.state.uiState = 'ready'; + }, 10) } } diff --git a/src/components/tooltip.directive.ts b/src/components/tooltip.directive.ts index 5905466..adbaaf0 100644 --- a/src/components/tooltip.directive.ts +++ b/src/components/tooltip.directive.ts @@ -32,13 +32,18 @@ export class CloukitTooltipDirective { // PLACEMENT // let placement: DropoutPlacement; + let wrapperMargin: string; if (this.cloukitDropoutPlacement === undefined || this.cloukitDropoutPlacement === 'bottom') { placement = DropoutPlacement.DOWN_CENTER; + wrapperMargin = 'marginTop'; } else if (this.cloukitDropoutPlacement === 'top') { + wrapperMargin = 'marginBottom'; placement = DropoutPlacement.UP_CENTER; } else if (this.cloukitDropoutPlacement === 'left') { + wrapperMargin = 'marginRight'; placement = DropoutPlacement.LEFT_CENTER; } else if (this.cloukitDropoutPlacement === 'right') { + wrapperMargin = 'marginLeft'; placement = DropoutPlacement.RIGHT_CENTER; } // @@ -48,6 +53,7 @@ export class CloukitTooltipDirective { const tooltipRef = this.viewContainerRef.createComponent(componentFactory); tooltipRef.instance.tooltipText = this.cloukitDropout; tooltipRef.instance.cloukitDropoutPlacement = placement; + tooltipRef.instance.wrapperMargin = wrapperMargin; // // REQUEST // diff --git a/src/components/tooltip.module.ts b/src/components/tooltip.module.ts index bd86647..35a4ba3 100644 --- a/src/components/tooltip.module.ts +++ b/src/components/tooltip.module.ts @@ -5,14 +5,21 @@ */ import { NgModule } from '@angular/core'; import { CommonModule } from '@angular/common'; -import { CloukitDropoutModule } from '@cloukit/dropout'; +import { CloukitThemeService } from '@cloukit/theme'; import { CloukitTooltipDirective } from './tooltip.directive'; import { CloukitTooltipComponent } from './children/tooltip.component'; +import { CloukitTooltipComponentThemeDefault } from './tooltip.theme'; @NgModule({ - imports: [ CommonModule, CloukitDropoutModule ], + imports: [ CommonModule ], exports: [ CloukitTooltipDirective ], declarations: [ CloukitTooltipDirective, CloukitTooltipComponent ], entryComponents: [ CloukitTooltipComponent ], }) -export class CloukitTooltipModule {} +export class CloukitTooltipModule { + constructor(private themeService: CloukitThemeService) { + if (this.themeService.getComponentTheme('tooltip') === null) { + this.themeService.registerComponentTheme('tooltip', new CloukitTooltipComponentThemeDefault()); + } + } +} diff --git a/src/components/tooltip.theme.ts b/src/components/tooltip.theme.ts new file mode 100644 index 0000000..44dd557 --- /dev/null +++ b/src/components/tooltip.theme.ts @@ -0,0 +1,56 @@ +/*! + * @license MIT + * Copyright (c) 2017 Bernhard Grünewaldt - codeclou.io + * https://github.com/cloukit/legal + */ +import { + CloukitComponentTheme, + CloukitStatefulAndModifierAwareElementThemeStyleDefinition, +} from '@cloukit/theme'; + +/** + * The default theme + */ +export class CloukitTooltipComponentThemeDefault extends CloukitComponentTheme { + + constructor() { + super(); + // + // NOTE: A margin-* of 5px will be added for wrapper.ready.base on component creation. + // this cannot be overridden by the theme! Since the placement influences top/bottom/left/right + // + this.createStyle('wrapper', 'init', 'base', { + style: { + transition: ` + margin-left 200ms linear, + margin-right 200ms linear, + margin-bottom 200ms linear, + margin-top 200ms linear, + opacity 200ms ease-in-out`, + opacity: 0, + } + } as CloukitStatefulAndModifierAwareElementThemeStyleDefinition); + + this.createStyle('wrapper', 'ready', 'base', + this.merge(this.getStyle('wrapper', 'init', 'base'), { + style: { + opacity: 1, + } + } as CloukitStatefulAndModifierAwareElementThemeStyleDefinition)); + + this.createStyle('tooltip', 'init', 'base', { + style: { + backgroundColor: '#333', + padding: '4px', + color: '#fff', + } + } as CloukitStatefulAndModifierAwareElementThemeStyleDefinition); + + this.createStyle('tooltip', 'ready', 'base', + this.merge(this.getStyle('tooltip', 'init', 'base'), { + style: { + } + } as CloukitStatefulAndModifierAwareElementThemeStyleDefinition)); + } + +} diff --git a/src/demo/demo.component.html b/src/demo/demo.component.html index 7f148a6..a50b517 100644 --- a/src/demo/demo.component.html +++ b/src/demo/demo.component.html @@ -4,28 +4,28 @@ tooltip to the left: Bratwurst

tooltip to the right: Bratwurst

tooltip to the top: Bratwurst

tooltip to the bottom: Bratwurst
diff --git a/src/demo/demo.component.ts b/src/demo/demo.component.ts index 4a49b0d..6cc39df 100644 --- a/src/demo/demo.component.ts +++ b/src/demo/demo.component.ts @@ -8,6 +8,4 @@ import { Component } from '@angular/core'; '.label { display:inline-block; width:250px; }', ], }) -export class DemoComponent { - -} +export class DemoComponent {} diff --git a/src/demo/demo.module.ts b/src/demo/demo.module.ts index 5404188..4e80b95 100644 --- a/src/demo/demo.module.ts +++ b/src/demo/demo.module.ts @@ -1,15 +1,20 @@ import { NgModule } from '@angular/core'; import { CommonModule } from '@angular/common'; -import { DemoComponent } from './demo.component'; -import { CloukitTooltipModule } from '../index'; import { CloukitDropoutModule } from '@cloukit/dropout'; +import { CloukitThemeModule } from '@cloukit/theme'; +import { CloukitTooltipModule } from '../index'; +import { DemoComponent } from './demo.component'; @NgModule({ declarations: [ DemoComponent ], exports: [ DemoComponent ], - imports: [ CommonModule, CloukitTooltipModule, CloukitDropoutModule ], + imports: [ + CommonModule, + CloukitThemeModule, + CloukitTooltipModule, + CloukitDropoutModule, + ], providers: [ ], bootstrap: [ ] }) -export class DemoModule { -} +export class DemoModule {}