From 1519aee0d9699c01bbd925e4ac879b2acc210e44 Mon Sep 17 00:00:00 2001 From: Olivia Guyot Date: Wed, 20 Sep 2023 11:42:58 +0200 Subject: [PATCH] feat(ui): improvements to dropdown * show selected item Also modified button: * removing minus margin and add hidden border to buttons --- .../chart-view.component.stories.ts | 13 +++---- .../lib/chart-view/chart-view.component.ts | 11 +++--- libs/ui/inputs/src/index.ts | 2 ++ .../inputs/src/lib/button/button.component.ts | 10 +++--- .../dropdown-multiselect.component.stories.ts | 3 +- .../dropdown-selector.component.html | 36 +++++++++---------- .../dropdown-selector.component.stories.ts | 26 ++++++++------ .../dropdown-selector.component.ts | 27 ++++++-------- .../dropdown-selector.model.ts | 5 +++ 9 files changed, 68 insertions(+), 65 deletions(-) create mode 100644 libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.model.ts diff --git a/libs/feature/dataviz/src/lib/chart-view/chart-view.component.stories.ts b/libs/feature/dataviz/src/lib/chart-view/chart-view.component.stories.ts index caa0507f30..cbe29d25b4 100644 --- a/libs/feature/dataviz/src/lib/chart-view/chart-view.component.stories.ts +++ b/libs/feature/dataviz/src/lib/chart-view/chart-view.component.stories.ts @@ -13,22 +13,23 @@ import { ChartViewComponent } from './chart-view.component' import { ChartComponent, UiDatavizModule } from '@geonetwork-ui/ui/dataviz' import { LoadingMaskComponent } from '@geonetwork-ui/ui/widgets' import { importProvidersFrom } from '@angular/core' -import { DropdownSelectorComponent } from '@geonetwork-ui/ui/inputs' +import { + DropdownSelectorComponent, + UiInputsModule, +} from '@geonetwork-ui/ui/inputs' import { MatProgressSpinner } from '@angular/material/progress-spinner' +import { OverlayModule } from '@angular/cdk/overlay' export default { title: 'Smart/Dataviz/ChartView', component: ChartViewComponent, decorators: [ moduleMetadata({ - declarations: [ - DropdownSelectorComponent, - LoadingMaskComponent, - MatProgressSpinner, - ], imports: [ ChartComponent, + OverlayModule, TranslateModule.forRoot(TRANSLATE_DEFAULT_CONFIG), + UiInputsModule, ], }), applicationConfig({ diff --git a/libs/feature/dataviz/src/lib/chart-view/chart-view.component.ts b/libs/feature/dataviz/src/lib/chart-view/chart-view.component.ts index c35fe4995f..2deda3803b 100644 --- a/libs/feature/dataviz/src/lib/chart-view/chart-view.component.ts +++ b/libs/feature/dataviz/src/lib/chart-view/chart-view.component.ts @@ -12,7 +12,7 @@ import { FieldAggregation, getJsonDataItemsProxy, } from '@geonetwork-ui/data-fetcher' -import { DDChoices } from '@geonetwork-ui/ui/inputs' +import { DropdownChoice } from '@geonetwork-ui/ui/inputs' import { BehaviorSubject, combineLatest, EMPTY, Observable } from 'rxjs' import { catchError, @@ -24,10 +24,7 @@ import { tap, } from 'rxjs/operators' import { DataService } from '../service/data.service' -import { - AggregationTypes, - InputChartType, -} from '@geonetwork-ui/common/domain/dataviz-configuration.model' +import { InputChartType } from '@geonetwork-ui/common/domain/dataviz-configuration.model' import { DatasetDistribution } from '@geonetwork-ui/common/domain/record' import { TranslateService } from '@ngx-translate/core' @@ -93,7 +90,7 @@ export class ChartViewComponent { error = null errorInfo = null - typeChoices: DDChoices = [ + typeChoices: DropdownChoice[] = [ { label: 'chart.type.bar', value: 'bar' }, { label: 'chart.type.barHorizontal', value: 'bar-horizontal' }, { label: 'chart.type.line', value: 'line' }, @@ -111,7 +108,7 @@ export class ChartViewComponent { { label: 'chart.aggregation.min', value: 'min' }, { label: 'chart.aggregation.average', value: 'average' }, { label: 'chart.aggregation.count', value: 'count' }, - ] as DDChoices + ] as DropdownChoice[] } dataset$: Observable = this.currentLink$.pipe( diff --git a/libs/ui/inputs/src/index.ts b/libs/ui/inputs/src/index.ts index cd80f3653e..3328a2da52 100644 --- a/libs/ui/inputs/src/index.ts +++ b/libs/ui/inputs/src/index.ts @@ -1,4 +1,6 @@ export * from './lib/dropdown-selector/dropdown-selector.component' +export * from './lib/dropdown-selector/dropdown-selector.model' +export * from './lib/dropdown-multiselect/dropdown-multiselect.component' export * from './lib/dropdown-multiselect/dropdown-multiselect.model' export * from './lib/text-input/text-input.component' export * from './lib/chips-input/chips-input.component' diff --git a/libs/ui/inputs/src/lib/button/button.component.ts b/libs/ui/inputs/src/lib/button/button.component.ts index 46764cb0d4..83770fcd80 100644 --- a/libs/ui/inputs/src/lib/button/button.component.ts +++ b/libs/ui/inputs/src/lib/button/button.component.ts @@ -55,15 +55,15 @@ export class ButtonComponent { get borderColor() { switch (this.type) { case 'default': - return 'focus:ring-4 focus:ring-gray-200' + return 'border border-gray-700 focus:ring-4 focus:ring-gray-200' case 'secondary': - return 'focus:ring-4 focus:ring-secondary-lightest' + return 'border border-secondary focus:ring-4 focus:ring-secondary-lightest' case 'primary': - return 'focus:ring-4 focus:ring-primary-lightest' + return 'border border-primary focus:ring-4 focus:ring-primary-lightest' case 'outline': - return 'border border-gray-300 -m-[1px] hover:border-primary-lighter focus:border-primary-lighter focus:ring-4 focus:ring-primary-lightest active:border-primary-darker' + return 'border border-gray-300 hover:border-primary-lighter focus:border-primary-lighter focus:ring-4 focus:ring-primary-lightest active:border-primary-darker' case 'light': - return 'focus:ring-4 focus:ring-gray-300' + return 'border border-white focus:ring-4 focus:ring-gray-300' } } diff --git a/libs/ui/inputs/src/lib/dropdown-multiselect/dropdown-multiselect.component.stories.ts b/libs/ui/inputs/src/lib/dropdown-multiselect/dropdown-multiselect.component.stories.ts index ff51631608..8064b038e6 100644 --- a/libs/ui/inputs/src/lib/dropdown-multiselect/dropdown-multiselect.component.stories.ts +++ b/libs/ui/inputs/src/lib/dropdown-multiselect/dropdown-multiselect.component.stories.ts @@ -9,13 +9,14 @@ import { OverlayModule } from '@angular/cdk/overlay' import { MatCheckboxModule } from '@angular/material/checkbox' import { TranslateModule } from '@ngx-translate/core' import { MatIcon } from '@angular/material/icon' +import { ButtonComponent } from '../button/button.component' export default { title: 'Inputs/DropdownMultiselectComponent', component: DropdownMultiselectComponent, decorators: [ moduleMetadata({ - declarations: [MatIcon], + declarations: [MatIcon, ButtonComponent], imports: [OverlayModule, MatCheckboxModule, TranslateModule.forRoot()], }), componentWrapperDecorator( diff --git a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.html b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.html index fd6374aaf7..5841b35d5a 100644 --- a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.html +++ b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.html @@ -11,8 +11,7 @@ -
-
- {{ getChoiceLabel() | translate }} -
+
+ {{ getChoiceLabel() | translate }}
- + expand_less expand_more @@ -46,7 +43,7 @@
- +
diff --git a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.stories.ts b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.stories.ts index 4175ff67b2..5f4f3bbcea 100644 --- a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.stories.ts +++ b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.stories.ts @@ -1,5 +1,6 @@ import { applicationConfig, + componentWrapperDecorator, Meta, moduleMetadata, StoryObj, @@ -11,24 +12,27 @@ import { TRANSLATE_DEFAULT_CONFIG, UtilI18nModule, } from '@geonetwork-ui/util/i18n' -import { MatCheckboxModule } from '@angular/material/checkbox' +import { MatIcon } from '@angular/material/icon' +import { ButtonComponent } from '../button/button.component' +import { importProvidersFrom } from '@angular/core' export default { title: 'Inputs/DropdownSelectorComponent', component: DropdownSelectorComponent, decorators: [ moduleMetadata({ - declarations: [], - imports: [ - UtilI18nModule, - OverlayModule, - MatCheckboxModule, - TranslateModule.forRoot(TRANSLATE_DEFAULT_CONFIG), - ], + declarations: [MatIcon, ButtonComponent], + imports: [UtilI18nModule, OverlayModule, TranslateModule], }), applicationConfig({ - providers: [], + providers: [ + importProvidersFrom(TranslateModule.forRoot(TRANSLATE_DEFAULT_CONFIG)), + ], }), + componentWrapperDecorator( + (story) => + `
${story}
` + ), ], } as Meta @@ -42,11 +46,11 @@ export const Primary: StoryObj = { value: 'choice1', }, { - label: 'My Choice 2', + label: 'My Choice 2, second choice', value: 'choice2', }, { - label: 'My Choice 3', + label: 'My Choice 3, very very very very very very long text', value: 'choice3', }, ], diff --git a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.ts b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.ts index 3a9555532f..28364f421a 100644 --- a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.ts +++ b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.component.ts @@ -15,13 +15,8 @@ import { ViewChild, ViewChildren, } from '@angular/core' -import { Choice } from '../dropdown-multiselect/dropdown-multiselect.model' -import { take } from 'rxjs/operators' - -export type DDChoices = Array<{ - label: string - value: string -}> +import { firstValueFrom } from 'rxjs' +import { DropdownChoice } from './dropdown-selector.model' const DEFAULT_ROW_NUMBERS = 6 @@ -35,12 +30,12 @@ export class DropdownSelectorComponent implements OnInit { @Input() title: string @Input() showTitle = true @Input() ariaName: string - @Input() choices: DDChoices - @Input() selected: any + @Input() choices: Array + @Input() selected: DropdownChoice['value'] @Input() maxRows: number @Input() extraBtnClass = '' @Input() minWidth = '' - @Output() selectValue = new EventEmitter() + @Output() selectValue = new EventEmitter() @ViewChild('overlayOrigin') overlayOrigin: CdkOverlayOrigin @ViewChild(CdkConnectedOverlay) overlay: CdkConnectedOverlay overlayOpen = false @@ -65,7 +60,7 @@ export class DropdownSelectorComponent implements OnInit { @ViewChildren('choiceInputs', { read: ElementRef }) choiceInputs: QueryList - get selectedChoice(): Choice { + get selectedChoice(): DropdownChoice { return ( this.choices.find((choice) => choice.value === this.selected) ?? this.choices[0] @@ -87,11 +82,11 @@ export class DropdownSelectorComponent implements OnInit { } } - isSelected(choice) { + isSelected(choice: DropdownChoice) { return choice === this.selectedChoice } - onSelectValue(choice: Choice): void { + onSelectValue(choice: DropdownChoice) { this.closeOverlay() this.selected = choice.value this.selectValue.emit(this.selected) @@ -106,8 +101,8 @@ export class DropdownSelectorComponent implements OnInit { : 'none' this.overlayOpen = true return Promise.all([ - this.overlay.attach.pipe(take(1)).toPromise(), - this.choiceInputs.changes.pipe(take(1)).toPromise(), + firstValueFrom(this.overlay.attach), + firstValueFrom(this.choiceInputs.changes), ]) } @@ -177,7 +172,7 @@ export class DropdownSelectorComponent implements OnInit { ) } - selectIfEnter(event: KeyboardEvent, choice: Choice) { + selectIfEnter(event: KeyboardEvent, choice: DropdownChoice) { if (event.code === 'Enter') { event.preventDefault() this.onSelectValue(choice) diff --git a/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.model.ts b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.model.ts new file mode 100644 index 0000000000..1eb2cb10b8 --- /dev/null +++ b/libs/ui/inputs/src/lib/dropdown-selector/dropdown-selector.model.ts @@ -0,0 +1,5 @@ +// FIXME: this should support more than string values, and match the multiselect choice model +export interface DropdownChoice { + value: unknown + label: string +}