From 151ab9c57dd645a0ddda094c073a1312d648fe9b Mon Sep 17 00:00:00 2001 From: splincode Date: Mon, 5 Feb 2024 15:58:00 +0300 Subject: [PATCH] feat(experimental): add `tuiButtonGroup` --- projects/demo/src/modules/app/app.routes.ts | 9 +++ projects/demo/src/modules/app/pages.ts | 6 ++ .../button-group/button-group.component.ts | 30 ++++++++ .../button-group/button-group.module.ts | 35 +++++++++ .../button-group/button-group.template.html | 59 +++++++++++++++ .../button-group/examples/1/index.html | 44 ++++++++++++ .../button-group/examples/1/index.less | 5 ++ .../button-group/examples/1/index.ts | 12 ++++ .../button-group/examples/2/index.html | 44 ++++++++++++ .../button-group/examples/2/index.less | 5 ++ .../button-group/examples/2/index.ts | 12 ++++ .../button-group/examples/3/index.html | 44 ++++++++++++ .../button-group/examples/3/index.less | 16 +++++ .../button-group/examples/3/index.ts | 12 ++++ .../examples/import/import-module.md | 14 ++++ .../examples/import/insert-template.md | 24 +++++++ projects/demo/used-icons.ts | 1 + .../button-group/button-group.component.ts | 12 ++++ .../button-group/button-group.directive.ts | 15 ++++ .../button-group/button-group.module.ts | 10 +++ .../button-group/button-group.style.less | 72 +++++++++++++++++++ .../directives/button-group/index.ts | 3 + .../directives/button-group/ng-package.json | 5 ++ projects/experimental/directives/index.ts | 1 + 24 files changed, 490 insertions(+) create mode 100644 projects/demo/src/modules/experimental/button-group/button-group.component.ts create mode 100644 projects/demo/src/modules/experimental/button-group/button-group.module.ts create mode 100644 projects/demo/src/modules/experimental/button-group/button-group.template.html create mode 100644 projects/demo/src/modules/experimental/button-group/examples/1/index.html create mode 100644 projects/demo/src/modules/experimental/button-group/examples/1/index.less create mode 100644 projects/demo/src/modules/experimental/button-group/examples/1/index.ts create mode 100644 projects/demo/src/modules/experimental/button-group/examples/2/index.html create mode 100644 projects/demo/src/modules/experimental/button-group/examples/2/index.less create mode 100644 projects/demo/src/modules/experimental/button-group/examples/2/index.ts create mode 100644 projects/demo/src/modules/experimental/button-group/examples/3/index.html create mode 100644 projects/demo/src/modules/experimental/button-group/examples/3/index.less create mode 100644 projects/demo/src/modules/experimental/button-group/examples/3/index.ts create mode 100644 projects/demo/src/modules/experimental/button-group/examples/import/import-module.md create mode 100644 projects/demo/src/modules/experimental/button-group/examples/import/insert-template.md create mode 100644 projects/experimental/directives/button-group/button-group.component.ts create mode 100644 projects/experimental/directives/button-group/button-group.directive.ts create mode 100644 projects/experimental/directives/button-group/button-group.module.ts create mode 100644 projects/experimental/directives/button-group/button-group.style.less create mode 100644 projects/experimental/directives/button-group/index.ts create mode 100644 projects/experimental/directives/button-group/ng-package.json diff --git a/projects/demo/src/modules/app/app.routes.ts b/projects/demo/src/modules/app/app.routes.ts index c9857721a74eb..92f927d4f858e 100644 --- a/projects/demo/src/modules/app/app.routes.ts +++ b/projects/demo/src/modules/app/app.routes.ts @@ -361,6 +361,15 @@ export const ROUTES: Routes = [ title: 'ButtonClose', }, }, + { + path: 'experimental/button-group', + loadChildren: async () => + (await import('../experimental/button-group/button-group.module')) + .ExampleTuiButtonGroupModule, + data: { + title: 'ButtonGroup', + }, + }, { path: 'experimental/cell', loadChildren: async () => diff --git a/projects/demo/src/modules/app/pages.ts b/projects/demo/src/modules/app/pages.ts index 0d6fd8f86cd97..766ec73c11f10 100644 --- a/projects/demo/src/modules/app/pages.ts +++ b/projects/demo/src/modules/app/pages.ts @@ -916,6 +916,12 @@ export const pages: TuiDocPages = [ keywords: 'кнопка, button, close, закрыть', route: '/experimental/button-close', }, + { + section: 'Experimental', + title: 'ButtonGroup', + keywords: 'кнопка, button, group, группа', + route: '/experimental/button-group', + }, ], }, { diff --git a/projects/demo/src/modules/experimental/button-group/button-group.component.ts b/projects/demo/src/modules/experimental/button-group/button-group.component.ts new file mode 100644 index 0000000000000..e58f4cbd151cd --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/button-group.component.ts @@ -0,0 +1,30 @@ +import {Component} from '@angular/core'; +import {changeDetection} from '@demo/emulate/change-detection'; +import {TuiDocExample, TuiRawLoaderContent} from '@taiga-ui/addon-doc'; + +@Component({ + selector: 'example-button-group-card', + templateUrl: './button-group.template.html', + changeDetection, +}) +export class ExampleTuiButtonGroupComponent { + readonly exampleModule: TuiRawLoaderContent = import( + './examples/import/import-module.md?raw' + ); + + readonly exampleHtml: TuiRawLoaderContent = import( + './examples/import/insert-template.md?raw' + ); + + readonly example1: TuiDocExample = { + HTML: import('./examples/1/index.html?raw'), + }; + + readonly example2: TuiDocExample = { + HTML: import('./examples/2/index.html?raw'), + }; + + readonly example3: TuiDocExample = { + HTML: import('./examples/3/index.html?raw'), + }; +} diff --git a/projects/demo/src/modules/experimental/button-group/button-group.module.ts b/projects/demo/src/modules/experimental/button-group/button-group.module.ts new file mode 100644 index 0000000000000..c0b6a690e721d --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/button-group.module.ts @@ -0,0 +1,35 @@ +import {CommonModule} from '@angular/common'; +import {NgModule} from '@angular/core'; +import {tuiGetDocModules} from '@taiga-ui/addon-doc'; +import {TuiNotificationModule} from '@taiga-ui/core'; +import { + TuiBadgedContentModule, + TuiButtonGroupModule, + TuiIconModule, + TuiSurfaceModule, +} from '@taiga-ui/experimental'; + +import {ExampleTuiButtonGroupComponent} from './button-group.component'; +import {TuiButtonGroupExample1} from './examples/1'; +import {TuiButtonGroupExample2} from './examples/2'; +import {TuiButtonGroupExample3} from './examples/3'; + +@NgModule({ + imports: [ + CommonModule, + TuiNotificationModule, + tuiGetDocModules(ExampleTuiButtonGroupComponent), + TuiSurfaceModule, + TuiButtonGroupModule, + TuiBadgedContentModule, + TuiIconModule, + ], + declarations: [ + ExampleTuiButtonGroupComponent, + TuiButtonGroupExample1, + TuiButtonGroupExample2, + TuiButtonGroupExample3, + ], + exports: [ExampleTuiButtonGroupComponent], +}) +export class ExampleTuiButtonGroupModule {} diff --git a/projects/demo/src/modules/experimental/button-group/button-group.template.html b/projects/demo/src/modules/experimental/button-group/button-group.template.html new file mode 100644 index 0000000000000..0e89a2f098026 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/button-group.template.html @@ -0,0 +1,59 @@ + + + + This code is + experimental + and is a subject to change. Expect final solution to be shipped in the next major version + + + + + + + + + + + + + + + + +
    +
  1. +

    Import module:

    + + +
  2. + +
  3. +

    Add to the template:

    + + +
  4. +
+
+
diff --git a/projects/demo/src/modules/experimental/button-group/examples/1/index.html b/projects/demo/src/modules/experimental/button-group/examples/1/index.html new file mode 100644 index 0000000000000..a62f550370fd4 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/1/index.html @@ -0,0 +1,44 @@ +
+ + + + + +
+ +
+ + + +
+ +
+ +
diff --git a/projects/demo/src/modules/experimental/button-group/examples/1/index.less b/projects/demo/src/modules/experimental/button-group/examples/1/index.less new file mode 100644 index 0000000000000..23d25a3bb0a19 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/1/index.less @@ -0,0 +1,5 @@ +:host { + display: flex; + flex-direction: column; + gap: 1rem; +} diff --git a/projects/demo/src/modules/experimental/button-group/examples/1/index.ts b/projects/demo/src/modules/experimental/button-group/examples/1/index.ts new file mode 100644 index 0000000000000..3cefd7843236d --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/1/index.ts @@ -0,0 +1,12 @@ +import {Component} from '@angular/core'; +import {changeDetection} from '@demo/emulate/change-detection'; +import {encapsulation} from '@demo/emulate/encapsulation'; + +@Component({ + selector: 'tui-button-group-example-1', + templateUrl: './index.html', + styleUrls: ['./index.less'], + encapsulation, + changeDetection, +}) +export class TuiButtonGroupExample1 {} diff --git a/projects/demo/src/modules/experimental/button-group/examples/2/index.html b/projects/demo/src/modules/experimental/button-group/examples/2/index.html new file mode 100644 index 0000000000000..1e97a3cb5d3eb --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/2/index.html @@ -0,0 +1,44 @@ +
+ + + + + +
+ +
+ + + +
+ +
+ +
diff --git a/projects/demo/src/modules/experimental/button-group/examples/2/index.less b/projects/demo/src/modules/experimental/button-group/examples/2/index.less new file mode 100644 index 0000000000000..23d25a3bb0a19 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/2/index.less @@ -0,0 +1,5 @@ +:host { + display: flex; + flex-direction: column; + gap: 1rem; +} diff --git a/projects/demo/src/modules/experimental/button-group/examples/2/index.ts b/projects/demo/src/modules/experimental/button-group/examples/2/index.ts new file mode 100644 index 0000000000000..8decc0e110e8f --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/2/index.ts @@ -0,0 +1,12 @@ +import {Component} from '@angular/core'; +import {changeDetection} from '@demo/emulate/change-detection'; +import {encapsulation} from '@demo/emulate/encapsulation'; + +@Component({ + selector: 'tui-button-group-example-2', + templateUrl: './index.html', + styleUrls: ['./index.less'], + encapsulation, + changeDetection, +}) +export class TuiButtonGroupExample2 {} diff --git a/projects/demo/src/modules/experimental/button-group/examples/3/index.html b/projects/demo/src/modules/experimental/button-group/examples/3/index.html new file mode 100644 index 0000000000000..840906b4147d5 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/3/index.html @@ -0,0 +1,44 @@ +
+ + + + + +
+ +
+ + + +
+ +
+ +
diff --git a/projects/demo/src/modules/experimental/button-group/examples/3/index.less b/projects/demo/src/modules/experimental/button-group/examples/3/index.less new file mode 100644 index 0000000000000..27f0692f97e30 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/3/index.less @@ -0,0 +1,16 @@ +@import '@taiga-ui/core/styles/taiga-ui-local'; + +:host { + display: flex; + flex-direction: column; + gap: 1rem; +} + +[tuiSurface][data-surface='custom'], +.shadow { + background: linear-gradient(334.83deg, #7d8ca0 0%, #647382 100%); + + button { + color: #fff; + } +} diff --git a/projects/demo/src/modules/experimental/button-group/examples/3/index.ts b/projects/demo/src/modules/experimental/button-group/examples/3/index.ts new file mode 100644 index 0000000000000..052c8d9a8791e --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/3/index.ts @@ -0,0 +1,12 @@ +import {Component} from '@angular/core'; +import {changeDetection} from '@demo/emulate/change-detection'; +import {encapsulation} from '@demo/emulate/encapsulation'; + +@Component({ + selector: 'tui-button-group-example-3', + templateUrl: './index.html', + styleUrls: ['./index.less'], + encapsulation, + changeDetection, +}) +export class TuiButtonGroupExample3 {} diff --git a/projects/demo/src/modules/experimental/button-group/examples/import/import-module.md b/projects/demo/src/modules/experimental/button-group/examples/import/import-module.md new file mode 100644 index 0000000000000..99e96d5f8c097 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/import/import-module.md @@ -0,0 +1,14 @@ +```ts +import {NgModule} from '@angular/core'; +import {TuiButtonGroupModule} from '@taiga-ui/experimental'; +// ... + +@NgModule({ + imports: [ + // ... + TuiButtonModule, + TuiButtonGroupModule, + ], +}) +export class MyModule {} +``` diff --git a/projects/demo/src/modules/experimental/button-group/examples/import/insert-template.md b/projects/demo/src/modules/experimental/button-group/examples/import/insert-template.md new file mode 100644 index 0000000000000..ae3af4f18f3d9 --- /dev/null +++ b/projects/demo/src/modules/experimental/button-group/examples/import/insert-template.md @@ -0,0 +1,24 @@ +```html +
+ + + +
+``` diff --git a/projects/demo/used-icons.ts b/projects/demo/used-icons.ts index 99021f8b75940..8ccf0a2845c84 100644 --- a/projects/demo/used-icons.ts +++ b/projects/demo/used-icons.ts @@ -101,6 +101,7 @@ export const TUI_USED_ICONS = [ 'tuiIconLock', 'tuiIconUsers', 'tuiIconClock', + 'tuiIconArrowRightCircleLarge', 'tuiIconPlusSquareLarge', 'tuiIconTarget', 'tuiIconGooglePay', diff --git a/projects/experimental/directives/button-group/button-group.component.ts b/projects/experimental/directives/button-group/button-group.component.ts new file mode 100644 index 0000000000000..d1385ce579b11 --- /dev/null +++ b/projects/experimental/directives/button-group/button-group.component.ts @@ -0,0 +1,12 @@ +import {ChangeDetectionStrategy, Component, ViewEncapsulation} from '@angular/core'; + +@Component({ + template: '', + styleUrls: ['./button-group.style.less'], + encapsulation: ViewEncapsulation.None, + changeDetection: ChangeDetectionStrategy.OnPush, + host: { + class: 'tui-button-group-styles', + }, +}) +export class TuiButtonGroupComponent {} diff --git a/projects/experimental/directives/button-group/button-group.directive.ts b/projects/experimental/directives/button-group/button-group.directive.ts new file mode 100644 index 0000000000000..a9df64a25d55e --- /dev/null +++ b/projects/experimental/directives/button-group/button-group.directive.ts @@ -0,0 +1,15 @@ +import {Directive, Inject} from '@angular/core'; +import {TuiDirectiveStylesService} from '@taiga-ui/cdk'; + +import {TuiButtonGroupComponent} from './button-group.component'; + +@Directive({ + selector: '[tuiButtonGroup]', +}) +export class TuiButtonGroupDirective { + constructor( + @Inject(TuiDirectiveStylesService) directiveStyles: TuiDirectiveStylesService, + ) { + directiveStyles.addComponent(TuiButtonGroupComponent); + } +} diff --git a/projects/experimental/directives/button-group/button-group.module.ts b/projects/experimental/directives/button-group/button-group.module.ts new file mode 100644 index 0000000000000..b06322d26f43e --- /dev/null +++ b/projects/experimental/directives/button-group/button-group.module.ts @@ -0,0 +1,10 @@ +import {NgModule} from '@angular/core'; + +import {TuiButtonGroupComponent} from './button-group.component'; +import {TuiButtonGroupDirective} from './button-group.directive'; + +@NgModule({ + declarations: [TuiButtonGroupDirective, TuiButtonGroupComponent], + exports: [TuiButtonGroupDirective], +}) +export class TuiButtonGroupModule {} diff --git a/projects/experimental/directives/button-group/button-group.style.less b/projects/experimental/directives/button-group/button-group.style.less new file mode 100644 index 0000000000000..b9984cbaae43c --- /dev/null +++ b/projects/experimental/directives/button-group/button-group.style.less @@ -0,0 +1,72 @@ +@import '@taiga-ui/core/styles/taiga-ui-local'; + +[tuiButtonGroup] { + display: flex; + border-radius: 1.5rem; + justify-content: center; + overflow: hidden; + + > button { + .button-base(); + + position: relative; + flex-grow: 1; + flex-basis: 0; + border-radius: 0; + flex-direction: column; + padding: 1.125rem 0.25rem 1.125rem 0.25rem; + gap: 0.5rem; + height: auto; + font-size: 0.8125rem; + white-space: pre-line; + min-width: 7.125rem; + font-weight: normal; + max-width: calc(50% - 1.75rem); + overflow: visible; + cursor: pointer; + color: var(--tui-link); + + &:active, + &:active:before, + &:active:after { + background: var(--tui-clear); + } + + &:before, + &:after { + position: absolute; + top: 0; + width: 1.75rem; + height: 100%; + } + + &, + &:before, + &:after { + transition: background var(--tui-duration, 300ms) ease-in-out; + } + + &:first-child::before { + content: ''; + left: -1.75rem; + } + + &:last-child::after { + content: ''; + right: -1.75rem; + } + } + + &:has(button:only-child) { + border-radius: 0.75rem; + } + + > button:only-child { + width: 100%; + flex-direction: row; + gap: 8px; + font-size: 17px; + max-width: 100%; + justify-content: center; + } +} diff --git a/projects/experimental/directives/button-group/index.ts b/projects/experimental/directives/button-group/index.ts new file mode 100644 index 0000000000000..0c4cf1d358274 --- /dev/null +++ b/projects/experimental/directives/button-group/index.ts @@ -0,0 +1,3 @@ +export * from './button-group.component'; +export * from './button-group.directive'; +export * from './button-group.module'; diff --git a/projects/experimental/directives/button-group/ng-package.json b/projects/experimental/directives/button-group/ng-package.json new file mode 100644 index 0000000000000..bebf62dcb5e51 --- /dev/null +++ b/projects/experimental/directives/button-group/ng-package.json @@ -0,0 +1,5 @@ +{ + "lib": { + "entryFile": "index.ts" + } +} diff --git a/projects/experimental/directives/index.ts b/projects/experimental/directives/index.ts index 78a556d9b6c25..7d2856ccfe8b0 100644 --- a/projects/experimental/directives/index.ts +++ b/projects/experimental/directives/index.ts @@ -1,5 +1,6 @@ export * from '@taiga-ui/experimental/directives/appearance'; export * from '@taiga-ui/experimental/directives/button-close'; +export * from '@taiga-ui/experimental/directives/button-group'; export * from '@taiga-ui/experimental/directives/button-vertical'; export * from '@taiga-ui/experimental/directives/card'; export * from '@taiga-ui/experimental/directives/cell';