add help mode

This commit is contained in:
Dan Percic 2021-08-22 19:04:29 +03:00
parent eef1e59fdd
commit a935fb413b
17 changed files with 353 additions and 4 deletions

View File

@ -44,6 +44,7 @@
"@angular-eslint/no-output-rename": "error",
"@angular-eslint/prefer-output-readonly": "error",
"@typescript-eslint/unbound-method": "error",
"@typescript-eslint/no-floating-promises": "off",
"@typescript-eslint/lines-between-class-members": "off",
"@typescript-eslint/naming-convention": [
"error",

View File

@ -0,0 +1,40 @@
@import 'apps/red-ui/src/assets/styles/variables';
.mat-dialog-container {
color: $accent;
padding: 0 !important;
border-radius: 8px !important;
}
.dialog {
position: relative;
min-height: 80px;
.dialog-close {
position: absolute;
top: 16px;
right: 16px;
}
.dialog-header {
padding: 32px 60px 0 32px;
}
.dialog-content {
padding: 24px 32px 40px;
}
.dialog-actions {
height: 81px;
box-sizing: border-box;
border-top: 1px solid $separator;
padding: 0 32px;
align-items: center;
display: flex;
> * {
margin-right: 16px;
}
}
}

View File

@ -27,3 +27,15 @@
opacity: 1;
}
}
.heading {
font-size: 16px;
line-height: 20px;
font-weight: 600;
}
.heading-l {
font-size: 20px;
font-weight: 600;
line-height: 24px;
}

View File

@ -3,3 +3,4 @@
@import 'texts';
@import 'tables';
@import 'layout';
@import 'dialogs';

View File

@ -1,10 +1,12 @@
export * from './lib/common-ui.module';
export * from './lib/buttons/icon-button/icon-button.type';
export * from './lib/buttons/icon-button/icon-button.component';
export * from './lib/utils/injection-tokens';
export * from './lib/utils/functions';
export * from './lib/utils/operators';
export * from './lib/utils/auto-unsubscribe.directive';
export * from './lib/utils/pipes/humanize.pipe';
export * from './lib/utils/types/events.type';
export * from './lib/utils/types/utility-types';
export * from './lib/utils/types/tooltip-positions.type';
export * from './lib/utils/decorators/bind.decorator';
@ -36,3 +38,7 @@ export * from './lib/misc/status-bar/status-bar-config.model';
export * from './lib/inputs/round-checkbox/round-checkbox.component';
export * from './lib/inputs/editable-input/editable-input.component';
export * from './lib/inputs/input-with-action/input-with-action.component';
export * from './lib/help-mode/help-mode.service';
export * from './lib/help-mode/help-mode.directive';
export * from './lib/help-mode/help-mode/help-mode.component';
export * from './lib/help-mode/help-mode-dialog/help-mode-dialog.component';

View File

@ -8,6 +8,7 @@ import { TranslateModule } from '@ngx-translate/core';
import { FormsModule } from '@angular/forms';
import { MatMenuModule } from '@angular/material/menu';
import { MatCheckboxModule } from '@angular/material/checkbox';
import { MatDialogModule } from '@angular/material/dialog';
import { IconButtonComponent } from './buttons/icon-button/icon-button.component';
import { ChevronButtonComponent } from './buttons/chevron-button/chevron-button.component';
import { CircleButtonComponent } from './buttons/circle-button/circle-button.component';
@ -22,12 +23,15 @@ import { StatusBarComponent } from './misc/status-bar/status-bar.component';
import { EditableInputComponent } from './inputs/editable-input/editable-input.component';
import { PopupFilterComponent } from './filtering/popup-filter/popup-filter.component';
import { InputWithActionComponent } from './inputs/input-with-action/input-with-action.component';
import { HelpModeDirective } from './help-mode/help-mode.directive';
import { HelpModeComponent } from './help-mode/help-mode/help-mode.component';
import { HelpModeDialogComponent } from './help-mode/help-mode-dialog/help-mode-dialog.component';
const buttons = [IconButtonComponent, ChevronButtonComponent, CircleButtonComponent];
const inputs = [RoundCheckboxComponent, EditableInputComponent, InputWithActionComponent];
const matModules = [MatIconModule, MatButtonModule, MatTooltipModule, MatMenuModule, MatCheckboxModule];
const matModules = [MatIconModule, MatButtonModule, MatTooltipModule, MatMenuModule, MatCheckboxModule, MatDialogModule];
const modules = [...matModules, FormsModule, TranslateModule];
@ -38,10 +42,12 @@ const components = [
QuickFiltersComponent,
PopupFilterComponent,
TableHeaderComponent,
StatusBarComponent
StatusBarComponent,
HelpModeComponent,
HelpModeDialogComponent
];
const utils = [SortByPipe, HumanizePipe, SyncWidthDirective];
const utils = [SortByPipe, HumanizePipe, SyncWidthDirective, HelpModeDirective];
@NgModule({
declarations: [...components, ...utils],

View File

@ -0,0 +1,8 @@
<section class="dialog">
<div class="content">
<p class="heading-l pre" [innerHTML]="'help-mode.welcome-to-help-mode' | translate"></p>
<img src="assets/illustrations/illustration.gif" alt="" width="335" />
<p class="pre" [innerHTML]="'help-mode.clicking-anywhere-on' | translate"></p>
</div>
<iqser-circle-button class="dialog-close" icon="iqser:close" mat-dialog-close></iqser-circle-button>
</section>

View File

@ -0,0 +1,16 @@
section {
background: #ecedf0;
display: flex;
justify-content: center;
}
.content {
width: 440px;
display: flex;
flex-direction: column;
align-items: center;
text-align: center;
padding-top: 20px;
padding-bottom: 30px;
line-height: 18px;
}

View File

@ -0,0 +1,7 @@
import { Component } from '@angular/core';
@Component({
templateUrl: './help-mode-dialog.component.html',
styleUrls: ['./help-mode-dialog.component.scss']
})
export class HelpModeDialogComponent {}

View File

@ -0,0 +1,34 @@
import { Directive, ElementRef, HostListener, Input, OnInit, Renderer2 } from '@angular/core';
import { HelpModeService } from './help-mode.service';
@Directive({
selector: '[iqserHelpMode]',
exportAs: 'iqserHelpMode'
})
export class HelpModeDirective implements OnInit {
@Input('iqserHelpMode') elementName!: string;
constructor(
private readonly _elementRef: ElementRef,
private readonly _renderer: Renderer2,
private readonly _helpModeService: HelpModeService
) {}
ngOnInit(): void {
this._createHelperElement();
}
private _createHelperElement() {
const element = this._elementRef.nativeElement as HTMLElement;
const helperElement = this._renderer.createElement('div') as HTMLElement;
this._renderer.addClass(helperElement, 'help-mode-on-mouse-over');
this._renderer.addClass(helperElement, `help-mode-on-mouse-over-${this.elementName}`);
this._helpModeService.addElement(this.elementName, element, helperElement);
}
@HostListener('click') onClick(): void {
this._helpModeService.openDocsFor(this.elementName);
}
}

View File

@ -0,0 +1,91 @@
import { Inject, Injectable, Renderer2, RendererFactory2 } from '@angular/core';
import { MatDialog, MatDialogRef } from '@angular/material/dialog';
import { TranslateService } from '@ngx-translate/core';
import { HelpModeDialogComponent } from './help-mode-dialog/help-mode-dialog.component';
import { HELP_DOCS } from '../utils/injection-tokens';
interface Helper {
readonly element: HTMLElement;
readonly helperElement: HTMLElement;
}
@Injectable({
providedIn: 'root'
})
export class HelpModeService {
isHelpModeActive = false;
helpModeDialogIsOpened = false;
private readonly _elements: Record<string, Helper> = {};
private readonly _renderer: Renderer2;
constructor(
@Inject(HELP_DOCS) private readonly _docs: Record<string, Record<string, string>>,
private readonly _dialog: MatDialog,
private readonly _rendererFactory: RendererFactory2,
private readonly _translateService: TranslateService
) {
this._renderer = this._rendererFactory.createRenderer(null, null);
}
openHelpModeDialog(): MatDialogRef<HelpModeDialogComponent> {
this.helpModeDialogIsOpened = true;
const ref = this._dialog.open(HelpModeDialogComponent, {
width: '600px'
});
ref.afterClosed()
.toPromise()
.then(() => {
this.helpModeDialogIsOpened = false;
});
return ref;
}
openDocsFor(elementName: string): void {
if (this.isHelpModeActive) {
window.open(this._docs[elementName][this._translateService.currentLang]);
}
}
activateHelpMode(): void {
this.isHelpModeActive = true;
this.openHelpModeDialog();
this._enableHelperElements();
}
deactivateHelpMode(): void {
this.isHelpModeActive = false;
this._disableHelperElements();
}
highlightHelperElements(): void {
if (!this.isHelpModeActive) return;
Object.values(this._elements).forEach(({ helperElement }) => {
this._renderer.addClass(helperElement, 'highlight');
setTimeout(() => {
this._renderer.removeClass(helperElement, 'highlight');
}, 500);
});
}
addElement(elementName: string, element: HTMLElement, helperElement: HTMLElement): void {
this._elements[elementName] = { element, helperElement };
}
private _enableHelperElements() {
Object.values(this._elements).forEach(({ element, helperElement }) => {
this._renderer.setStyle(element, 'position', 'relative');
this._renderer.appendChild(element, helperElement);
});
}
private _disableHelperElements() {
Object.values(this._elements).forEach(({ element, helperElement }) => {
this._renderer.removeStyle(element, 'position');
this._renderer.removeChild(element, helperElement);
});
}
}

View File

@ -0,0 +1,20 @@
<div class="help-button" *ngIf="!helpModeService.isHelpModeActive" (click)="helpModeService.activateHelpMode()">
<mat-icon svgIcon="red:help-outline"></mat-icon>
<div class="text">{{ 'help-mode.button-text' | translate }}</div>
</div>
<div class="help-mode-border" *ngIf="helpModeService.isHelpModeActive">
<div class="bottom">
<p class="heading">{{ 'help-mode.text' | translate }}</p>
<a class="instructions" *ngIf="!helpModeService.helpModeDialogIsOpened" (click)="helpModeService.openHelpModeDialog()">
{{ 'help-mode.instructions' | translate }}
</a>
<div class="close">
(esc)
<iqser-circle-button
class="dialog-close"
icon="iqser:close"
(click)="helpModeService.deactivateHelpMode()"
></iqser-circle-button>
</div>
</div>
</div>

View File

@ -0,0 +1,72 @@
@import '../../../../../../apps/red-ui/src/assets/styles/variables';
.help-button {
width: 44px;
height: 40px;
position: absolute;
bottom: 20px;
right: 0;
z-index: 1;
background: $green-2;
border-top-left-radius: 8px;
border-bottom-left-radius: 8px;
box-shadow: -1px 1px 5px 0 rgba(40, 50, 65, 0.25);
display: flex;
align-items: center;
justify-content: center;
transition: all 0.25s;
}
.help-button:hover {
cursor: pointer;
width: fit-content;
padding-left: 10px;
padding-right: 10px;
.text {
display: block;
}
mat-icon {
padding-right: 8px;
}
}
.text {
display: none;
}
.help-mode-border {
box-sizing: border-box;
height: 100%;
width: 100%;
border-left: 8px solid $green-2;
border-right: 8px solid $green-2;
border-top: 8px solid $green-2;
border-bottom: 60px solid $green-2;
z-index: 10;
position: absolute;
display: flex;
justify-content: center;
.bottom {
position: fixed;
height: 60px;
width: 95%;
bottom: 0;
display: flex;
justify-content: space-between;
align-items: center;
pointer-events: visiblePainted;
a {
color: black;
text-decoration: underline;
}
.close {
display: flex;
align-items: center;
}
}
}

View File

@ -0,0 +1,29 @@
import { Component, HostListener } from '@angular/core';
import { HelpModeService } from '../help-mode.service';
import { IqserEventTarget } from '../../utils/types/events.type';
@Component({
selector: 'iqser-help-mode',
templateUrl: './help-mode.component.html',
styleUrls: ['./help-mode.component.scss']
})
export class HelpModeComponent {
constructor(readonly helpModeService: HelpModeService) {}
@HostListener('document:keydown.escape') onEscKeydownHandler(): void {
if (!this.helpModeService.helpModeDialogIsOpened) {
this.helpModeService.deactivateHelpMode();
}
}
@HostListener('document:keydown.h', ['$event']) onHKeydownHandler(event: KeyboardEvent): void {
const node = (event.target as IqserEventTarget).localName;
if (!this.helpModeService.isHelpModeActive && node !== 'input' && node !== 'textarea') {
this.helpModeService.activateHelpMode();
}
}
@HostListener('click') onClick(): void {
this.helpModeService.highlightHelperElements();
}
}

View File

@ -17,7 +17,7 @@ export class InputWithActionComponent {
@Output() readonly valueChange = new EventEmitter<string>();
get hasContent(): boolean {
return !!this.value.length;
return !!this.value?.length;
}
get computedWidth(): string {

View File

@ -0,0 +1,3 @@
import { InjectionToken } from '@angular/core';
export const HELP_DOCS = new InjectionToken<Record<string, Record<string, string>>>('Links to user manual or help docs');

View File

@ -0,0 +1,3 @@
export interface IqserEventTarget extends EventTarget {
localName: string;
}