From 5b284f8c9ba65a88fb4d58ac6aa34be429b19e8e Mon Sep 17 00:00:00 2001 From: Max Peintner Date: Fri, 29 Jul 2022 11:14:45 +0200 Subject: [PATCH] fix(console): horizontal toggle for users, projects, improve UI/UX (#4047) * fix(console): horizontal toggle for users, projects * improve input contrast * toggles, profile UI fix * lint * fix safari styles * fix button placement redirects * style lint Co-authored-by: Livio Spring --- .../nav-toggle/nav-toggle.component.html | 6 + .../nav-toggle/nav-toggle.component.scss | 71 ++++++++++++ .../nav-toggle/nav-toggle.component.spec.ts | 23 ++++ .../nav-toggle/nav-toggle.component.ts | 14 +++ .../modules/nav-toggle/nav-toggle.module.ts | 12 ++ .../refresh-table.component.scss | 3 +- .../redirect-uris.component.scss | 2 +- .../pages/projects/projects.component.html | 29 +++-- .../pages/projects/projects.component.scss | 37 +++---- .../app/pages/projects/projects.component.ts | 1 + .../src/app/pages/projects/projects.module.ts | 2 + .../detail-form/detail-form.component.html | 104 ++++++++++-------- .../detail-form/detail-form.component.scss | 63 ++++++----- .../profile-picture.component.html | 26 +++-- .../profile-picture.component.ts | 4 +- .../pages/users/user-list/user-list.module.ts | 2 + .../user-table/user-table.component.html | 18 +-- .../user-table/user-table.component.scss | 36 +++--- console/src/app/services/toast.service.ts | 22 ++-- console/src/component-themes.scss | 2 + console/src/styles.scss | 34 +++++- console/src/styles/input.scss | 6 +- 22 files changed, 358 insertions(+), 159 deletions(-) create mode 100644 console/src/app/modules/nav-toggle/nav-toggle.component.html create mode 100644 console/src/app/modules/nav-toggle/nav-toggle.component.scss create mode 100644 console/src/app/modules/nav-toggle/nav-toggle.component.spec.ts create mode 100644 console/src/app/modules/nav-toggle/nav-toggle.component.ts create mode 100644 console/src/app/modules/nav-toggle/nav-toggle.module.ts diff --git a/console/src/app/modules/nav-toggle/nav-toggle.component.html b/console/src/app/modules/nav-toggle/nav-toggle.component.html new file mode 100644 index 0000000000..35ca967148 --- /dev/null +++ b/console/src/app/modules/nav-toggle/nav-toggle.component.html @@ -0,0 +1,6 @@ + diff --git a/console/src/app/modules/nav-toggle/nav-toggle.component.scss b/console/src/app/modules/nav-toggle/nav-toggle.component.scss new file mode 100644 index 0000000000..0e4a1a7227 --- /dev/null +++ b/console/src/app/modules/nav-toggle/nav-toggle.component.scss @@ -0,0 +1,71 @@ +@use '@angular/material' as mat; + +@mixin nav-toggle-theme($theme) { + $primary: map-get($theme, primary); + $warn: map-get($theme, warn); + $background: map-get($theme, background); + $accent: map-get($theme, accent); + $primary-color: mat.get-color-from-palette($primary, 500); + + $warn-color: mat.get-color-from-palette($warn, 500); + $accent-color: mat.get-color-from-palette($accent, 500); + $foreground: map-get($theme, foreground); + $is-dark-theme: map-get($theme, is-dark); + $back: map-get($background, background); + + .nav-toggle { + display: flex; + align-items: center; + font-size: 14px; + line-height: 14px; + padding: 0.4rem 12px; + color: mat.get-color-from-palette($foreground, text) !important; + transition: all 0.2s ease; + text-decoration: none; + border-radius: 50vw; + border: none; + font-weight: 400; + margin: 0.25rem 2px; + white-space: nowrap; + position: relative; + background: none; + cursor: pointer; + font-family: 'Lato', -apple-system, BlinkMacSystemFont, sans-serif; + + .c_label { + display: flex; + align-items: center; + text-align: center; + + .count { + display: none; + margin-left: 6px; + } + } + + &.external-link { + padding-right: 2rem; + + i { + position: absolute; + right: 8px; + font-size: 1.2rem; + } + } + + &:hover { + background: if($is-dark-theme, #ffffff40, #00000010); + } + + &.active { + background-color: $primary-color; + color: mat.get-color-from-palette($foreground, toolbar-items) !important; + + .c_label { + .count { + display: inline-block; + } + } + } + } +} diff --git a/console/src/app/modules/nav-toggle/nav-toggle.component.spec.ts b/console/src/app/modules/nav-toggle/nav-toggle.component.spec.ts new file mode 100644 index 0000000000..63158cf042 --- /dev/null +++ b/console/src/app/modules/nav-toggle/nav-toggle.component.spec.ts @@ -0,0 +1,23 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { NavToggleComponent } from './nav-toggle.component'; + +describe('NavToggleComponent', () => { + let component: NavToggleComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ NavToggleComponent ] + }) + .compileComponents(); + + fixture = TestBed.createComponent(NavToggleComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/console/src/app/modules/nav-toggle/nav-toggle.component.ts b/console/src/app/modules/nav-toggle/nav-toggle.component.ts new file mode 100644 index 0000000000..97cca6bfab --- /dev/null +++ b/console/src/app/modules/nav-toggle/nav-toggle.component.ts @@ -0,0 +1,14 @@ +import { Component, EventEmitter, Input, Output } from '@angular/core'; + +@Component({ + selector: 'cnsl-nav-toggle', + templateUrl: './nav-toggle.component.html', + styleUrls: ['./nav-toggle.component.scss'], +}) +export class NavToggleComponent { + @Input() public label: string = ''; + @Input() public count: number | null = 0; + @Input() public active: boolean = false; + @Output() public clicked: EventEmitter = new EventEmitter(); + constructor() {} +} diff --git a/console/src/app/modules/nav-toggle/nav-toggle.module.ts b/console/src/app/modules/nav-toggle/nav-toggle.module.ts new file mode 100644 index 0000000000..ca348c0d1f --- /dev/null +++ b/console/src/app/modules/nav-toggle/nav-toggle.module.ts @@ -0,0 +1,12 @@ +import { CommonModule } from '@angular/common'; +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +import { NavToggleComponent } from './nav-toggle.component'; + +@NgModule({ + declarations: [NavToggleComponent], + imports: [CommonModule, RouterModule], + exports: [NavToggleComponent], +}) +export class NavToggleModule {} diff --git a/console/src/app/modules/refresh-table/refresh-table.component.scss b/console/src/app/modules/refresh-table/refresh-table.component.scss index 5f3ab80e47..dbed9fc327 100644 --- a/console/src/app/modules/refresh-table/refresh-table.component.scss +++ b/console/src/app/modules/refresh-table/refresh-table.component.scss @@ -7,9 +7,10 @@ display: flex; align-items: center; padding-bottom: 0.5rem; + box-sizing: border-box; &.border-bottom { - padding-bottom: 0.5rem; + padding-bottom: 1rem; margin-bottom: 1rem; border-bottom: 1px solid map-get($foreground, divider); } diff --git a/console/src/app/pages/projects/apps/redirect-uris/redirect-uris.component.scss b/console/src/app/pages/projects/apps/redirect-uris/redirect-uris.component.scss index 0768e4a319..550950334b 100644 --- a/console/src/app/pages/projects/apps/redirect-uris/redirect-uris.component.scss +++ b/console/src/app/pages/projects/apps/redirect-uris/redirect-uris.component.scss @@ -80,7 +80,7 @@ min-width: 320px; .formfield { - flex: 1; + width: 500px; } button { diff --git a/console/src/app/pages/projects/projects.component.html b/console/src/app/pages/projects/projects.component.html index 0c596c3c5c..9d62108cb3 100644 --- a/console/src/app/pages/projects/projects.component.html +++ b/console/src/app/pages/projects/projects.component.html @@ -9,22 +9,21 @@

{{ 'PROJECT.PAGES.LISTDESCRIPTION' | translate }}

-
- - +
+ +
+ -
+
{{ 'USER.PROFILE.USERNAME' | translate }} - +
- - {{ 'USER.PROFILE.FIRSTNAME' | translate }} - - - - {{ 'USER.PROFILE.LASTNAME' | translate }} - - - - {{ 'USER.PROFILE.NICKNAME' | translate }} - - - - {{ 'USER.PROFILE.DISPLAYNAME' | translate }} - - - - {{ 'USER.PROFILE.GENDER' | translate }} - - - {{ 'GENDERS.'+gender | translate }} - - - - - {{ 'USER.PROFILE.PREFERRED_LANGUAGE' | translate }} - - - {{ 'LANGUAGES.'+language | translate }} - - - +
+ + {{ 'USER.PROFILE.FIRSTNAME' | translate }} + + + + {{ 'USER.PROFILE.LASTNAME' | translate }} + + + + {{ 'USER.PROFILE.NICKNAME' | translate }} + + + + {{ 'USER.PROFILE.DISPLAYNAME' | translate }} + + + + {{ 'USER.PROFILE.GENDER' | translate }} + + + {{ 'GENDERS.' + gender | translate }} + + + + + {{ 'USER.PROFILE.PREFERRED_LANGUAGE' | translate }} + + + {{ 'LANGUAGES.' + language | translate }} + + + +
- +
- \ No newline at end of file + diff --git a/console/src/app/pages/users/user-detail/detail-form/detail-form.component.scss b/console/src/app/pages/users/user-detail/detail-form/detail-form.component.scss index 819ef62d9f..407ec23d8b 100644 --- a/console/src/app/pages/users/user-detail/detail-form/detail-form.component.scss +++ b/console/src/app/pages/users/user-detail/detail-form/detail-form.component.scss @@ -2,31 +2,28 @@ display: flex; flex-direction: row; flex-wrap: wrap; - margin: 0 -.5rem; + margin: 0; + width: 100%; + align-items: center; - &.inner { - margin: 0; - width: 100%; - display: flex; - align-items: center; + .usernamediv { + margin-left: 0.5rem; + margin-bottom: 0.5rem; - .usernamediv { - margin-left: .5rem; - margin-bottom: .5rem; + .formfield { + margin: 0; + flex: 1; + } - .formfield { - margin: 0; - flex: 1; - } - - .edit { - cursor: pointer !important; - } + .edit { + display: block; + margin-top: 0.5rem; + cursor: pointer !important; } } .camera-wrapper { - margin: 0 .5rem; + margin: 0 0.5rem; position: relative; border-radius: 50%; padding: 0; @@ -36,37 +33,49 @@ justify-content: center; background: none; cursor: pointer; - transition: all .3s ease; + transition: all 0.3s ease; + overflow: hidden; .i-wrapper { - border-radius: 50%; - background-color: #00000050; + display: none; + background-color: #00000080; position: absolute; - top: 0; z-index: 1; left: 0; right: 0; bottom: 0; - display: flex; align-items: center; justify-content: center; + transition: all 0.2 ease; i { - font-size: 3rem; + font-size: 1.2rem; + margin: 0.25rem; color: white; } } &:hover { .i-wrapper { - background-color: #00000080; + display: inline; } } } .formfield { flex: 1 1 33%; - margin: 0 .5rem; + margin: 0 0.5rem; + } +} + +.user-grid { + display: grid; + grid-template-columns: 1fr; + gap: 1rem; + margin: 0; + + @media only screen and (min-width: 700px) { + grid-template-columns: 1fr 1fr; } } @@ -75,6 +84,6 @@ justify-content: flex-end; .submit-button { - border-radius: .5rem; + border-radius: 0.5rem; } } diff --git a/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.html b/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.html index 669638c545..7c60010bfa 100644 --- a/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.html +++ b/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.html @@ -1,20 +1,26 @@ -{{'USER.PROFILE.AVATAR.UPLOADTITLE' | translate}} +{{ 'USER.PROFILE.AVATAR.UPLOADTITLE' | translate }}
-

{{'USER.PROFILE.AVATAR.CURRENT' | translate}}

+

{{ 'USER.PROFILE.AVATAR.CURRENT' | translate }}

- - - +
-
\ No newline at end of file +
diff --git a/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.ts b/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.ts index 6f808b20f7..293d42355a 100644 --- a/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.ts +++ b/console/src/app/pages/users/user-detail/detail-form/profile-picture/profile-picture.component.ts @@ -49,7 +49,9 @@ export class ProfilePictureComponent { this.data.profilePic = resp.user?.human?.profile?.avatarUrl ?? ''; }); }) - .catch((error) => this.toast.showError(error)); + .catch((error) => { + this.toast.showError(error.error, false); + }); } public closeDialog(): void { diff --git a/console/src/app/pages/users/user-list/user-list.module.ts b/console/src/app/pages/users/user-list/user-list.module.ts index 969f737b1f..be86f4ddfd 100644 --- a/console/src/app/pages/users/user-list/user-list.module.ts +++ b/console/src/app/pages/users/user-list/user-list.module.ts @@ -18,6 +18,7 @@ import { AvatarModule } from 'src/app/modules/avatar/avatar.module'; import { CardModule } from 'src/app/modules/card/card.module'; import { FilterUserModule } from 'src/app/modules/filter-user/filter-user.module'; import { InputModule } from 'src/app/modules/input/input.module'; +import { NavToggleModule } from 'src/app/modules/nav-toggle/nav-toggle.module'; import { PaginatorModule } from 'src/app/modules/paginator/paginator.module'; import { RefreshTableModule } from 'src/app/modules/refresh-table/refresh-table.module'; import { TableActionsModule } from 'src/app/modules/table-actions/table-actions.module'; @@ -49,6 +50,7 @@ import { UserTableComponent } from './user-table/user-table.component'; TranslateModule, FilterUserModule, RouterModule, + NavToggleModule, RefreshTableModule, TableActionsModule, ActionKeysModule, diff --git a/console/src/app/pages/users/user-list/user-table/user-table.component.html b/console/src/app/pages/users/user-list/user-table/user-table.component.html index bb83827477..a6c263f77b 100644 --- a/console/src/app/pages/users/user-list/user-table/user-table.component.html +++ b/console/src/app/pages/users/user-list/user-table/user-table.component.html @@ -8,13 +8,17 @@ [emitRefreshOnPreviousRoutes]="refreshOnPreviousRoutes" [showBorder]="true" > -
- - +
+ +
diff --git a/console/src/app/pages/users/user-list/user-table/user-table.component.scss b/console/src/app/pages/users/user-list/user-table/user-table.component.scss index 8a9159bf3f..99f2f2a3b1 100644 --- a/console/src/app/pages/users/user-list/user-table/user-table.component.scss +++ b/console/src/app/pages/users/user-list/user-table/user-table.component.scss @@ -4,31 +4,31 @@ $foreground: map-get($theme, foreground); $is-dark-theme: map-get($theme, is-dark); - .user-table-left-actions { + .user-toggle-group { display: flex; - align-items: center; + margin: 0; + height: 100%; - .type-button { - border: none; - background: none; - text-align: left; - padding: 0.75rem 0; - opacity: 0.6; - font-size: 15px; - cursor: pointer; - color: map-get($foreground, text); + .toggle-row { + display: flex; + align-items: center; - &:first-child { - margin-right: 1rem; + i { + margin-right: 0.5rem; } - &:hover { - opacity: 1; + .info-i { + font-size: 1.2rem; + margin-left: 0.5rem; + margin-right: 0; } - &.active { - font-weight: 600; - opacity: 1; + .current-dot { + height: 8px; + width: 8px; + border-radius: 50%; + background-color: rgb(84, 142, 230); + margin-left: 0.5rem; } } } diff --git a/console/src/app/services/toast.service.ts b/console/src/app/services/toast.service.ts index c881dcd92c..b545a050cb 100644 --- a/console/src/app/services/toast.service.ts +++ b/console/src/app/services/toast.service.ts @@ -33,15 +33,19 @@ export class ToastService { } } - public showError(grpcError: any): void { - const { message, code, metadata } = grpcError; - if (code !== 16) { - this.translate - .get('ACTIONS.CLOSE') - .pipe(take(1)) - .subscribe((value) => { - this.showMessage(decodeURI(message), value, false); - }); + public showError(error: any | string, isGrpc: boolean = true): void { + if (isGrpc) { + const { message, code, metadata } = error; + if (code !== 16) { + this.translate + .get('ACTIONS.CLOSE') + .pipe(take(1)) + .subscribe((value) => { + this.showMessage(decodeURI(message), value, false); + }); + } + } else { + this.showMessage(error as string, '', false); } } diff --git a/console/src/component-themes.scss b/console/src/component-themes.scss index 019973e2ac..b377a4b0ba 100644 --- a/console/src/component-themes.scss +++ b/console/src/component-themes.scss @@ -12,6 +12,7 @@ @import 'src/app/modules/app-card/app-card.component'; @import 'src/app/modules/contributors/contributors.component'; @import 'src/app/modules/nav/nav.component'; +@import 'src/app/modules/nav-toggle/nav-toggle.component'; @import './styles/toast.scss'; @import 'src/app/modules/table-actions/table-actions.component'; @import 'src/app/modules/org-context/org-context.component.scss'; @@ -62,6 +63,7 @@ @include main-theme($theme); @include avatar-theme($theme); @include nav-theme($theme); + @include nav-toggle-theme($theme); @include header-theme($theme); @include app-type-radio-theme($theme); @include projects-theme($theme); diff --git a/console/src/styles.scss b/console/src/styles.scss index 110aa3eb20..c25f70439f 100644 --- a/console/src/styles.scss +++ b/console/src/styles.scss @@ -446,9 +446,18 @@ $custom-typography: mat.define-typography-config( } } - .mat-button-toggle-button { + .mat-button-toggle-group-appearance-standard { + border-color: map-get($foreground, divider); + } + + .mat-button-toggle { background-color: mat.get-color-from-palette($background, cards); transition: background-color 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); + transition: border-color 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); + + &.mat-button-toggle-checked { + background-color: #00000010; + } } .main-container, @@ -509,9 +518,18 @@ $custom-typography: mat.define-typography-config( } } - .mat-button-toggle-button { + .mat-button-toggle-group-appearance-standard { + border-color: map-get($foreground, divider); + } + + .mat-button-toggle { background-color: mat.get-color-from-palette($background, cards); transition: background-color 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); + transition: border-color 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); + + &.mat-button-toggle-checked { + background-color: mat.get-color-from-palette($background, background); + } } .main-container, @@ -601,3 +619,15 @@ i { .mat-checkbox-inner-container.mat-checkbox-inner-container-no-side-margin { margin-right: 0.5rem !important; } + +.mat-button-toggle-button { + display: flex; + height: 36px; + line-height: 36px !important; + align-items: center; + font-size: 14px !important; +} + +.mat-button-toggle-label-content { + line-height: 36px; +} diff --git a/console/src/styles/input.scss b/console/src/styles/input.scss index 53779af177..190b8be9a6 100644 --- a/console/src/styles/input.scss +++ b/console/src/styles/input.scss @@ -20,8 +20,8 @@ transform: all 0.2 linear; font-size: 1rem; border: none; - border: 1px solid if($is-dark-theme, #f9f7f725, #1a191938); - background-color: if($is-dark-theme, #00000020, #00000004); + border: 1px solid if($is-dark-theme, #f9f7f775, #1a191954); + background-color: if($is-dark-theme, #00000040, #00000004); border-radius: 4px; height: 40px; padding: 10px; @@ -34,7 +34,7 @@ margin-bottom: 2px; &:hover { - border-color: if($is-dark-theme, #aeafb1, #1a1b1b); + border-color: if($is-dark-theme, #e0e0e0, #1a1b1b); } &:active,