diff --git a/package-lock.json b/package-lock.json index 9aec185031..3ffd3e27e0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5,6 +5,7 @@ "requires": true, "packages": { "": { + "name": "dsp-app", "version": "5.3.0", "dependencies": { "@angular/animations": "^11.2.9", @@ -4916,6 +4917,7 @@ "version": "9.0.0", "dev": true, "license": "MIT", + "peer": true, "peerDependencies": { "tslib": "^1.10.0" } @@ -4924,6 +4926,7 @@ "version": "9.0.0", "dev": true, "license": "MIT", + "peer": true, "peerDependencies": { "rxjs": "^6.5.3", "tslib": "^1.10.0", @@ -12311,6 +12314,7 @@ "node_modules/pdfjs-dist": { "version": "2.7.570", "license": "Apache-2.0", + "peer": true, "peerDependencies": { "worker-loader": "^3.0.7" } @@ -21888,8 +21892,6 @@ "version": "6.0.2", "dev": true, "requires": { - "@angular/compiler": "9.0.0", - "@angular/core": "9.0.0", "app-root-path": "^3.0.0", "aria-query": "^3.0.0", "axobject-query": "2.0.2", @@ -21907,11 +21909,13 @@ "@angular/compiler": { "version": "9.0.0", "dev": true, + "peer": true, "requires": {} }, "@angular/core": { "version": "9.0.0", "dev": true, + "peer": true, "requires": {} }, "source-map": { @@ -26180,7 +26184,6 @@ "ng2-pdf-viewer": { "version": "7.0.1", "requires": { - "pdfjs-dist": "~2.7.570", "tslib": "^2.0.0" } }, @@ -26985,6 +26988,7 @@ }, "pdfjs-dist": { "version": "2.7.570", + "peer": true, "requires": {} }, "performance-now": { diff --git a/src/app/app.module.ts b/src/app/app.module.ts index 6969afeda0..ef3c6ba794 100644 --- a/src/app/app.module.ts +++ b/src/app/app.module.ts @@ -98,6 +98,22 @@ import { ResultsComponent } from './workspace/results/results.component'; import { AudioComponent } from './workspace/resource/representation/audio/audio.component'; import { IntermediateComponent } from './workspace/intermediate/intermediate.component'; import { ResourceLinkFormComponent } from './workspace/resource/resource-link-form/resource-link-form.component'; +import { ConfirmationDialogComponent } from './main/action/confirmation-dialog/confirmation-dialog.component'; +import { ConfirmationMessageComponent } from './main/action/confirmation-dialog/confirmation-message/confirmation-message.component'; +import { LoginFormComponent } from './main/action/login-form/login-form.component'; +import { MessageComponent } from './main/action/message/message.component'; +import { ProgressIndicatorComponent } from './main/action/progress-indicator/progress-indicator.component'; +import { StringLiteralInputComponent } from './main/action/string-literal-input/string-literal-input.component'; +import { SortButtonComponent } from './main/action/sort-button/sort-button.component'; +import { SelectedResourcesComponent } from './main/action/selected-resources/selected-resources.component'; +import { AdminImageDirective } from './main/directive/admin-image/admin-image.directive'; +import { ExistingNameDirective } from './main/directive/existing-name/existing-name.directive'; +import { GndDirective } from './main/directive/gnd/gnd.directive'; +import { FormattedBooleanPipe } from './main/pipes/formatting/formatted-boolean.pipe'; +import { KnoraDatePipe } from './main/pipes/formatting/knoradate.pipe'; +import { LinkifyPipe } from './main/pipes/string-transformation/linkify.pipe'; +import { StringifyStringLiteralPipe } from './main/pipes/string-transformation/stringify-string-literal.pipe'; +import { TruncatePipe } from './main/pipes/string-transformation/truncate.pipe'; // translate: AoT requires an exported function for factories export function httpLoaderFactory(httpClient: HttpClient) { @@ -182,6 +198,22 @@ export function httpLoaderFactory(httpClient: HttpClient) { AudioComponent, IntermediateComponent, ResourceLinkFormComponent, + ConfirmationDialogComponent, + ConfirmationMessageComponent, + LoginFormComponent, + MessageComponent, + ProgressIndicatorComponent, + StringLiteralInputComponent, + SortButtonComponent, + SelectedResourcesComponent, + AdminImageDirective, + ExistingNameDirective, + GndDirective, + FormattedBooleanPipe, + KnoraDatePipe, + LinkifyPipe, + StringifyStringLiteralPipe, + TruncatePipe, ], imports: [ AppRoutingModule, diff --git a/src/app/main/action/confirmation-dialog/confirmation-dialog.component.html b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.html new file mode 100644 index 0000000000..cd02c1ec84 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.html @@ -0,0 +1,11 @@ +
+ +

Are you sure you want to delete this value from {{data.value.propertyLabel}}?

+ +
+ + + + +
+ diff --git a/src/app/main/action/confirmation-dialog/confirmation-dialog.component.scss b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.scss new file mode 100644 index 0000000000..0a5a960ec2 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.scss @@ -0,0 +1 @@ +@import "../../../../assets/style/viewer"; diff --git a/src/app/main/action/confirmation-dialog/confirmation-dialog.component.spec.ts b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.spec.ts new file mode 100644 index 0000000000..dac46a0528 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.spec.ts @@ -0,0 +1,168 @@ +import { OverlayContainer } from '@angular/cdk/overlay'; +import { HarnessLoader } from '@angular/cdk/testing'; +import { TestbedHarnessEnvironment } from '@angular/cdk/testing/testbed'; +import { Component, Input, OnInit } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { MatButtonHarness } from '@angular/material/button/testing'; +import { MAT_DIALOG_DATA, MatDialog, MatDialogModule, MatDialogRef } from '@angular/material/dialog'; +import { MatDialogHarness } from '@angular/material/dialog/testing'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; +import { MockResource, ReadIntValue, ReadValue } from '@dasch-swiss/dsp-js'; +import { ConfirmationDialogComponent, ConfirmationDialogValueDeletionPayload } from './confirmation-dialog.component'; + +/** + * test host component to simulate parent component with a confirmation dialog. + */ +@Component({ + template: ` +

{{confirmationDialogResponse}}

` +}) +class ConfirmationDialogTestHostComponent implements OnInit { + + confirmationDialogResponse: string; + + testValue: ReadIntValue; + + constructor(private dialog: MatDialog) { + } + + ngOnInit() { + MockResource.getTestThing().subscribe(res => { + this.testValue = res.getValuesAs('http://0.0.0.0:3333/ontology/0001/anything/v2#hasInteger', ReadIntValue)[0]; + }); + } + + openDialog() { + + this.dialog.open(ConfirmationDialogComponent, { + data: { + value: this.testValue, + buttonTextOk: 'OK', + buttonTextCancel: 'Cancel' + } + }).afterClosed().subscribe((payload: ConfirmationDialogValueDeletionPayload) => { + if (payload.confirmed) { + this.confirmationDialogResponse = 'Action was confirmed!'; + } else { + this.confirmationDialogResponse = 'Action was cancelled'; + } + }); + } +} + +@Component({ selector: 'app-confirmation-message', template: '' }) +class MockConfirmationMessageComponent { + @Input() value: ReadValue; + + constructor() { } +} + +describe('ConfirmationDialogComponent', () => { + let testHostComponent: ConfirmationDialogTestHostComponent; + let testHostFixture: ComponentFixture; + let rootLoader: HarnessLoader; + let overlayContainer: OverlayContainer; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + declarations: [ + ConfirmationDialogComponent, + ConfirmationDialogTestHostComponent, + MockConfirmationMessageComponent + ], + imports: [ + MatDialogModule, + BrowserAnimationsModule + ], + providers: [ + { + provide: MAT_DIALOG_DATA, + useValue: {} + }, + { + provide: MatDialogRef, + useValue: {} + } + ] + }) + .compileComponents(); + + overlayContainer = TestBed.inject(OverlayContainer); + })); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(ConfirmationDialogTestHostComponent); + testHostComponent = testHostFixture.componentInstance; + rootLoader = TestbedHarnessEnvironment.documentRootLoader(testHostFixture); + testHostFixture.detectChanges(); + expect(testHostComponent).toBeTruthy(); + }); + + afterEach(async () => { + const dialogs = await rootLoader.getAllHarnesses(MatDialogHarness); + await Promise.all(dialogs.map(async d => await d.close())); + + // angular won't call this for us so we need to do it ourselves to avoid leaks. + overlayContainer.ngOnDestroy(); + }); + + it('should display a confirmation dialog', async () => { + + testHostComponent.openDialog(); + + testHostFixture.detectChanges(); + + await testHostFixture.whenStable(); + + const dialogDiv = document.querySelector('mat-dialog-container'); + expect(dialogDiv).toBeTruthy(); + + const dspConfirmMsg = document.querySelector('app-confirmation-message'); + expect(dspConfirmMsg).toBeTruthy(); + + const dialogTitle = dialogDiv.querySelector('.title'); + expect(dialogTitle.innerHTML.trim()).toEqual('Are you sure you want to delete this value from Integer?'); + + }); + + it('should return a confirmation message when the OK button is clicked', async () => { + + testHostComponent.openDialog(); + + let dialogHarnesses = await rootLoader.getAllHarnesses(MatDialogHarness); + + expect(dialogHarnesses.length).toEqual(1); + + const okButton = await rootLoader.getHarness(MatButtonHarness.with({ selector: '.ok' })); + + await okButton.click(); + + dialogHarnesses = await rootLoader.getAllHarnesses(MatDialogHarness); + + expect(dialogHarnesses.length).toEqual(0); + + expect(testHostComponent.confirmationDialogResponse).toEqual('Action was confirmed!'); + + }); + + it('should return a cancelled message when the cancel button is clicked', async () => { + + testHostComponent.openDialog(); + + let dialogHarnesses = await rootLoader.getAllHarnesses(MatDialogHarness); + + expect(dialogHarnesses.length).toEqual(1); + + const cancelButton = await rootLoader.getHarness(MatButtonHarness.with({ selector: '.cancel' })); + + await cancelButton.click(); + + dialogHarnesses = await rootLoader.getAllHarnesses(MatDialogHarness); + + expect(dialogHarnesses.length).toEqual(0); + + expect(testHostComponent.confirmationDialogResponse).toEqual('Action was cancelled'); + + + }); +}); diff --git a/src/app/main/action/confirmation-dialog/confirmation-dialog.component.ts b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.ts new file mode 100644 index 0000000000..79ac60c1b2 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-dialog.component.ts @@ -0,0 +1,39 @@ +import { Component, Inject, ViewChild } from '@angular/core'; +import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog'; +import { ReadValue } from '@dasch-swiss/dsp-js'; +import { ConfirmationMessageComponent } from './confirmation-message/confirmation-message.component'; + +export class ConfirmationDialogData { + value: ReadValue; + buttonTextOk: string; + buttonTextCancel: string; +} + +export class ConfirmationDialogValueDeletionPayload { + confirmed: boolean; + deletionComment?: string; +} + +@Component({ + selector: 'app-confirmation-dialog', + templateUrl: './confirmation-dialog.component.html', + styleUrls: ['./confirmation-dialog.component.scss'] +}) +export class ConfirmationDialogComponent { + @ViewChild('confirmMessage') confirmationMessageComponent: ConfirmationMessageComponent; + + // type assertion doesn't seem to be enforced + // https://stackoverflow.com/a/57787554 + constructor( + @Inject(MAT_DIALOG_DATA) public data: ConfirmationDialogData, + private _dialogRef: MatDialogRef + ) { } + + onConfirmClick(): void { + const payload = new ConfirmationDialogValueDeletionPayload(); + payload.confirmed = true; + payload.deletionComment = this.confirmationMessageComponent.comment ? this.confirmationMessageComponent.comment : undefined; + this._dialogRef.close(payload); + } + +} diff --git a/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.html b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.html new file mode 100644 index 0000000000..49fdbbf3ef --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.html @@ -0,0 +1,13 @@ +
+

Confirming this action will delete the following value from {{value.propertyLabel}}:

+

Value: {{value.strval}}

+

Value Comment: {{value.valueHasComment ? value.valueHasComment : 'No comment'}}

+

Value Creation Date: {{value.valueCreationDate}}

+ +
diff --git a/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.scss b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.scss new file mode 100644 index 0000000000..44169e9db7 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.scss @@ -0,0 +1 @@ +@import "../../../../../assets/style/viewer"; diff --git a/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.spec.ts b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.spec.ts new file mode 100644 index 0000000000..1f2d313ec5 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.spec.ts @@ -0,0 +1,88 @@ +import { Component, OnInit } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { By } from '@angular/platform-browser'; +import { MockResource, ReadIntValue } from '@dasch-swiss/dsp-js'; +import { ConfirmationMessageComponent } from './confirmation-message.component'; + +/** + * test host component to simulate parent component with a confirmation dialog. + */ +@Component({ + template: ` + ` +}) +class ConfirmationMessageTestHostComponent implements OnInit { + + testValue: ReadIntValue; + + constructor() { + } + + ngOnInit() { + MockResource.getTestThing().subscribe(res => { + this.testValue = res.getValuesAs('http://0.0.0.0:3333/ontology/0001/anything/v2#hasInteger', ReadIntValue)[0]; + }); + } +} + +describe('ConfirmationMessageComponent', () => { + let testHostComponent: ConfirmationMessageTestHostComponent; + let testHostFixture: ComponentFixture; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + declarations: [ + ConfirmationMessageTestHostComponent, + ConfirmationMessageComponent + ] + }) + .compileComponents(); + })); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(ConfirmationMessageTestHostComponent); + testHostComponent = testHostFixture.componentInstance; + testHostFixture.detectChanges(); + }); + + it('should create', () => { + expect(testHostComponent).toBeTruthy(); + }); + + it('should bind the values correctly', () => { + testHostComponent.testValue.valueHasComment = 'My comment'; + testHostFixture.detectChanges(); + + const hostCompDe = testHostFixture.debugElement; + const valueComponentDe = hostCompDe.query(By.directive(ConfirmationMessageComponent)); + + expect(valueComponentDe).toBeTruthy(); + + const label = valueComponentDe.query(By.css('.val-label')).nativeElement; + expect(label.innerText).toEqual('Confirming this action will delete the following value from Integer:'); + + const value = valueComponentDe.query(By.css('.val-value')).nativeElement; + expect(value.innerText).toEqual('Value: 1'); + + const comment = valueComponentDe.query(By.css('.val-comment')).nativeElement; + expect(comment.innerText).toEqual('Value Comment: My comment'); + + const creationDate = valueComponentDe.query(By.css('.val-creation-date')).nativeElement; + expect(creationDate.innerText).toEqual('Value Creation Date: 2018-05-28T15:52:03.897Z'); + + }); + + it('should default to "no comment" if the value does not contain a comment', () => { + testHostComponent.testValue.valueHasComment = null; + testHostFixture.detectChanges(); + + const hostCompDe = testHostFixture.debugElement; + const valueComponentDe = hostCompDe.query(By.directive(ConfirmationMessageComponent)); + + expect(valueComponentDe).toBeTruthy(); + + const comment = valueComponentDe.query(By.css('.val-comment')).nativeElement; + expect(comment.innerText).toEqual('Value Comment: No comment'); + + }); +}); diff --git a/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.ts b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.ts new file mode 100644 index 0000000000..87b7a62339 --- /dev/null +++ b/src/app/main/action/confirmation-dialog/confirmation-message/confirmation-message.component.ts @@ -0,0 +1,20 @@ +import { Component, Input } from '@angular/core'; +import { ReadValue } from '@dasch-swiss/dsp-js'; + +@Component({ + selector: 'app-confirmation-message', + templateUrl: './confirmation-message.component.html', + styleUrls: ['./confirmation-message.component.scss'] +}) +export class ConfirmationMessageComponent { + + @Input() value: ReadValue; + comment?: string; + + constructor() { } + + onKey(event: KeyboardEvent) { + this.comment = (event.target as HTMLInputElement).value; + } + +} diff --git a/src/app/main/action/login-form/login-form.component.html b/src/app/main/action/login-form/login-form.component.html new file mode 100644 index 0000000000..347f1b3bdd --- /dev/null +++ b/src/app/main/action/login-form/login-form.component.html @@ -0,0 +1,67 @@ + + + +
+

A user is already logged in:

+

Username: {{session.user.name}}

+
+

Please log out if it's not you.

+ +
diff --git a/src/app/main/action/login-form/login-form.component.scss b/src/app/main/action/login-form/login-form.component.scss new file mode 100644 index 0000000000..7378eb40f1 --- /dev/null +++ b/src/app/main/action/login-form/login-form.component.scss @@ -0,0 +1,38 @@ +$warn: #ef5350; + +$width: 280px; + +.full-width { + width: 100%; +} + +.login-form-title { + text-align: center; +} + +.login-error { + color: $warn; + text-align: center; +} + +.login-progress { + display: inline-block; + margin-right: 6px; +} + +.login-container, +.logout-container { + margin-left: auto; + margin-right: auto; + position: relative; + width: $width; +} + +.submit-button, +.logout-button { + height: 48px; +} + +.center { + text-align: center; +} diff --git a/src/app/main/action/login-form/login-form.component.spec.ts b/src/app/main/action/login-form/login-form.component.spec.ts new file mode 100644 index 0000000000..b32d8c7b6e --- /dev/null +++ b/src/app/main/action/login-form/login-form.component.spec.ts @@ -0,0 +1,225 @@ +import { Component, OnInit, ViewChild } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { FormBuilder, ReactiveFormsModule } from '@angular/forms'; +import { MatInputModule } from '@angular/material/input'; +import { MatSnackBarModule } from '@angular/material/snack-bar'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; +import { + ApiResponseData, + AuthenticationEndpointV2, + LoginResponse, + LogoutResponse, + MockUsers, + UsersEndpointAdmin +} from '@dasch-swiss/dsp-js'; +import { of } from 'rxjs'; +import { AjaxResponse } from 'rxjs/ajax'; +import { DspApiConnectionToken } from '../../declarations/dsp-api-tokens'; +import { Session, SessionService } from '../../services/session.service'; +import { LoginFormComponent } from './login-form.component'; + +/** + * test host component to simulate login-form component. + */ +@Component({ + template: ` + ` +}) +class TestHostComponent implements OnInit { + + @ViewChild('loginForm') loginFormComponent: LoginFormComponent; + + loggedIn: boolean; + + ngOnInit() { } + + // if response is true, the login was successful + // assign loggedIn to the response + onLogin(response: boolean) { + this.loggedIn = response; + } + + // if response is true, the logout was successful + // assign loggedIn to the opposite of the response + onLogout(response: boolean) { + this.loggedIn = !response; + } + +} + +describe('LoginFormComponent', () => { + let testHostComponent: TestHostComponent; + let testHostFixture: ComponentFixture; + + let sessionService: SessionService; + + beforeEach(waitForAsync(() => { + const dspConnSpy = { + admin: { + usersEndpoint: jasmine.createSpyObj('usersEndpoint', ['getUser']) + }, + v2: { + auth: jasmine.createSpyObj('auth', ['login', 'logout']), + jsonWebToken: '' + }, + }; + + TestBed.configureTestingModule({ + declarations: [ + LoginFormComponent, + TestHostComponent + ], + providers: [ + { + provide: DspApiConnectionToken, + useValue: dspConnSpy + }, + FormBuilder, + SessionService + ], + imports: [ + ReactiveFormsModule, + MatInputModule, + MatSnackBarModule, + BrowserAnimationsModule + ] + }) + .compileComponents(); + + sessionService = TestBed.inject(SessionService); + })); + + // mock localStorage + beforeEach(() => { + let store = {}; + + spyOn(localStorage, 'getItem').and.callFake( + (key: string): string => store[key] || null + ); + spyOn(localStorage, 'removeItem').and.callFake( + (key: string): void => { + delete store[key]; + } + ); + spyOn(localStorage, 'setItem').and.callFake( + (key: string, value: string): void => { + store[key] = value; + } + ); + spyOn(localStorage, 'clear').and.callFake(() => { + store = {}; + }); + }); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(TestHostComponent); + testHostComponent = testHostFixture.componentInstance; + testHostFixture.detectChanges(); + + expect(testHostComponent).toBeTruthy(); + }); + + it('should create an instance', () => { + expect(testHostComponent.loginFormComponent).toBeTruthy(); + }); + + describe('Login', () => { + + beforeEach(() => { + const dspConnSpy = TestBed.inject(DspApiConnectionToken); + + (dspConnSpy.v2.auth as jasmine.SpyObj).login.and.callFake( + () => { + const response: LoginResponse = new LoginResponse(); + + response.token = 'myToken'; + + return of(ApiResponseData.fromAjaxResponse({ response } as AjaxResponse)); + } + ); + + (dspConnSpy.admin.usersEndpoint as jasmine.SpyObj).getUser.and.callFake( + () => { + const loggedInUser = MockUsers.mockUser(); + return of(loggedInUser); + } + ); + }); + + it('should log the user in if the credentials are correct', () => { + + const dspConnSpy = TestBed.inject(DspApiConnectionToken); + + testHostComponent.loginFormComponent.form.get('username').setValue('root'); + + testHostComponent.loginFormComponent.form.get('password').setValue('test'); + + testHostFixture.detectChanges(); + + testHostComponent.loginFormComponent.login(); + + expect(dspConnSpy.v2.auth.login).toHaveBeenCalledTimes(1); + + expect(dspConnSpy.v2.auth.login).toHaveBeenCalledWith('username', 'root', 'test'); + + expect(dspConnSpy.admin.usersEndpoint.getUser).toHaveBeenCalledTimes(1); + + expect(dspConnSpy.admin.usersEndpoint.getUser).toHaveBeenCalledWith('username', 'root'); + + const session = JSON.parse(localStorage.getItem('session')); + + expect(session.user.name).toEqual('root'); + + expect(session.user.jwt).toEqual('myToken'); + + expect(testHostComponent.loggedIn).toBeTruthy(); + + }); + }); + + describe('Logout', () => { + beforeEach(() => { + const dspConnSpy = TestBed.inject(DspApiConnectionToken); + + (dspConnSpy.v2.auth as jasmine.SpyObj).logout.and.callFake( + () => { + const response: LogoutResponse = new LogoutResponse(); + + response.status = 0; + + return of(ApiResponseData.fromAjaxResponse({ response } as AjaxResponse)); + } + ); + }); + + it('should log the user out', () => { + const dspConnSpy = TestBed.inject(DspApiConnectionToken); + + // mock session directly instead of calling the login method + const session: Session = { + id: 12345, + user: { + name: 'username', + jwt: 'myToken', + lang: 'en', + sysAdmin: false, + projectAdmin: [] + } + }; + + // store session in localStorage + localStorage.setItem('session', JSON.stringify(session)); + + expect(JSON.parse(localStorage.getItem('session')).user.name).toEqual('username'); + + testHostComponent.loginFormComponent.logout(); + + expect(dspConnSpy.v2.auth.logout).toHaveBeenCalledTimes(1); + + expect(JSON.parse(localStorage.getItem('session'))).toBeNull(); + + expect(testHostComponent.loggedIn).toBeFalsy(); + + }); + }); +}); diff --git a/src/app/main/action/login-form/login-form.component.ts b/src/app/main/action/login-form/login-form.component.ts new file mode 100644 index 0000000000..0976402b26 --- /dev/null +++ b/src/app/main/action/login-form/login-form.component.ts @@ -0,0 +1,201 @@ +import { Component, EventEmitter, Inject, Input, OnInit, Output } from '@angular/core'; +import { FormBuilder, FormGroup, Validators } from '@angular/forms'; +import { ApiResponseData, ApiResponseError, KnoraApiConnection, LoginResponse, LogoutResponse } from '@dasch-swiss/dsp-js'; +import { DspApiConnectionToken } from '../../declarations/dsp-api-tokens'; +import { NotificationService } from '../../services/notification.service'; +import { Session, SessionService } from '../../services/session.service'; + +@Component({ + selector: 'app-login-form', + templateUrl: './login-form.component.html', + styleUrls: ['./login-form.component.scss'] +}) +export class LoginFormComponent implements OnInit { + + /** + * navigate to the defined url (or path) after successful login + * + * @param navigate + */ + @Input() navigate?: string; + + /** + * set your theme color here, + * it will be used in the progress-indicator and the buttons + * + * @param color + */ + @Input() color?: string; + + /** + * set whether or not you want icons to display in the input fields + * + * @param icons + */ + @Input() icons?: boolean; + + /** + * emits true when the login process was successful and false in case of error + * + * @param loginSuccess + * + */ + @Output() loginSuccess: EventEmitter = new EventEmitter(); + + /** + * emits true when the logout process was successful and false in case of error + * + * @param logoutSuccess + * + */ + @Output() logoutSuccess: EventEmitter = new EventEmitter(); + + // is there already a valid session? + session: Session; + + // form + form: FormGroup; + + // show progress indicator + loading = false; + + isError: boolean; + + // specific error messages + loginFailed = false; + loginErrorServer = false; + + // labels for the login form + // todo: should be handled by translation service (i18n) + formLabel = { + title: 'Login here', + name: 'Username', + pw: 'Password', + submit: 'Login', + retry: 'Retry', + logout: 'LOGOUT', + remember: 'Remember me', + forgotPassword: 'Forgot password?', + error: { + failed: 'Password or username is wrong', + server: 'An error has occurred when connecting to the server. Try again later or contact the DaSCH support.' + } + }; + + // error definitions for the following form fields + formErrors = { + username: '', + password: '' + }; + + // error messages for the form fields defined in formErrors + validationMessages = { + username: { + required: 'user name is required.' + }, + password: { + required: 'password is required' + } + }; + + + constructor( + @Inject(DspApiConnectionToken) private _dspApiConnection: KnoraApiConnection, + private _notification: NotificationService, + private _sessionService: SessionService, + private _fb: FormBuilder + ) { } + + ngOnInit() { + // if session is valid (a user is logged-in) show a message, otherwise build the form + this._sessionService.isSessionValid().subscribe( + result => { + // returns a result if session is still valid + if (result) { + this.session = JSON.parse(localStorage.getItem('session')); + } else { + // session is invalid, build the login form + this.buildLoginForm(); + } + } + ); + } + + buildLoginForm(): void { + this.form = this._fb.group({ + username: ['', Validators.required], + password: ['', Validators.required] + }); + } + + /** + * @ignore + * + * Login and set session + */ + login() { + + this.loading = true; + this.isError = false; + + // grab values from form + const identifier = this.form.get('username').value; + const password = this.form.get('password').value; + + const identifierType: 'iri' | 'email' | 'username' = (identifier.indexOf('@') > -1 ? 'email' : 'username'); + + this._dspApiConnection.v2.auth.login(identifierType, identifier, password).subscribe( + (response: ApiResponseData) => { + this._sessionService.setSession(response.body.token, identifier, identifierType).subscribe( + () => { + this.session = this._sessionService.getSession(); + this.loginSuccess.emit(true); + this.loading = false; + } + ); + }, + (error: ApiResponseError) => { + // error handling + this.loginFailed = (error.status === 401 || error.status === 404); + this.loginErrorServer = (error.status === 0 || error.status >= 500 && error.status < 600); + + if (this.loginErrorServer) { + this._notification.openSnackBar(error); + } + + this.loginSuccess.emit(false); + this.isError = true; + + this.loading = false; + } + ); + } + + /** + * @ignore + * + * Logout and destroy session + * + */ + logout() { + this.loading = true; + + this._dspApiConnection.v2.auth.logout().subscribe( + (response: ApiResponseData) => { + this.logoutSuccess.emit(true); + this._sessionService.destroySession(); + this.loading = false; + this.buildLoginForm(); + this.session = undefined; + this.form.get('password').setValue(''); + }, + (error: ApiResponseError) => { + this._notification.openSnackBar(error); + this.logoutSuccess.emit(false); + this.loading = false; + } + ); + + } + +} diff --git a/src/app/main/action/message/message.component.html b/src/app/main/action/message/message.component.html new file mode 100644 index 0000000000..7f9a41cd87 --- /dev/null +++ b/src/app/main/action/message/message.component.html @@ -0,0 +1,61 @@ + + + + {{message?.type | uppercase }} {{message?.status}} | {{message?.statusMsg}} + + + + + + + + + → {{message?.route}} + + + + + + +

{{apiError.error.toString()}}

+
+
+ + +

{{links.title}}

+ + {{item.icon}} +

{{item.label}}

+
+
+ +
+ + + +

{{message.footnote}}

+ + +

+ Please come back in a few minutes and try to reload the page. +

+ + + + mail_outline Contact the support team + +
+ +
+ + + +
+ + {{message?.statusText}} + + +
+ +
diff --git a/src/app/main/action/message/message.component.scss b/src/app/main/action/message/message.component.scss new file mode 100644 index 0000000000..f45b4cd4c8 --- /dev/null +++ b/src/app/main/action/message/message.component.scss @@ -0,0 +1,110 @@ +@import '../../../../assets/style/responsive'; + +$accent: rgba(255, 196, 0, 1); +$primary: rgba(0, 105, 92, 1); +$warn: rgb(244, 67, 54); + +.app-panel { + display: flex; + box-sizing: border-box; + flex-direction: row; + white-space: nowrap; +} + +.fill-remaining-space { + flex-basis: auto; + flex-grow: 1; + flex-shrink: 1; +} + +.app-error { + background-color: rgba($warn, .5); +} + +.app-warning { + background-color: rgba($accent, .5); +} + +.app-note, +.app-hint { + background-color: $primary; + color: #fff; +} + +.app-error, +.app-warning, +.app-note, +.app-hint { + margin: 12px auto; + max-width: 640px; + + .message-subtitle { + padding-bottom: 12px; + + .left { + float: left; + left: 16px; + position: absolute; + text-align: left; + } + + .right { + float: right; + right: 16px; + position: absolute; + text-align: right; + + } + } + + .message-title { + padding-top: 12px; + + } + + .message-content { + margin-bottom: 48px; + margin-top: 48px; + + .link { + cursor: pointer; + } + } + + .message-footnote { + padding: 24px; + + .bolder { + font-weight: bolder; + } + + .action { + margin: 48px auto 0 auto; + display: table; + } + } +} + +.app-short-message { + + .app-short-message-text { + font-weight: bolder; + text-align: center; + } + +} + + +// mobile device: phone +@media (max-width: map-get($grid-breakpoints, phone)) { + + .app-panel { + white-space: normal; + } + + .app-short-message { + .app-short-message-text { + text-align: left; + } + } +} diff --git a/src/app/main/action/message/message.component.spec.ts b/src/app/main/action/message/message.component.spec.ts new file mode 100644 index 0000000000..699956e37e --- /dev/null +++ b/src/app/main/action/message/message.component.spec.ts @@ -0,0 +1,218 @@ +import { Component, OnInit, ViewChild } from '@angular/core'; +import { waitForAsync, ComponentFixture, fakeAsync, TestBed } from '@angular/core/testing'; +import { MatCardModule } from '@angular/material/card'; +import { MatIconModule } from '@angular/material/icon'; +import { MatListModule } from '@angular/material/list'; +import { By } from '@angular/platform-browser'; +import { RouterTestingModule } from '@angular/router/testing'; +import { ApiResponseError } from '@dasch-swiss/dsp-js'; +import { StatusMsg } from 'src/assets/http/statusMsg'; +import { AppMessageData, MessageComponent } from './message.component'; + +/** + * test host component to simulate parent component. + */ +@Component({ + template: ` + ` +}) +class ShortMessageTestHostComponent implements OnInit { + + @ViewChild('message', { static: false }) messageComponent: MessageComponent; + + shortMessage: AppMessageData = { + status: 200, + statusMsg: 'Success', + statusText: 'You just updated the user profile.', + type: 'Note', + footnote: 'Close it' + }; + + size = 'short'; + + constructor() { + } + + ngOnInit() { } +} + +/** + * test host component to simulate parent component. + */ +@Component({ + template: '' +}) +class LongMessageTestHostComponent implements OnInit { + + @ViewChild('message', { static: false }) messageComponent: MessageComponent; + + errorMessage: ApiResponseError = { + status: 403, + url: 'http://0.0.0.0:3333/admin/projects/shortcode/001/members', + method: 'Http failure response for http://0.0.0.0:3333/admin/projects/shortcode/001/members: 400 Bad Request', + error: 'error message' + }; + + constructor() { + } + + ngOnInit() { } +} + +/** + * test host component to simulate parent component. + */ +@Component({ + template: '' +}) +class ShortMessageWithDurationTestHostComponent implements OnInit { + + @ViewChild('message', { static: false }) messageComponent: MessageComponent; + + shortMessage: AppMessageData = { + status: 200, + statusMsg: 'Success', + statusText: 'You just updated the user profile.', + type: 'Note', + footnote: 'Close it' + }; + + size = 'short'; + + constructor() { + } + + ngOnInit() { } +} + +describe('MessageComponent', () => { + let shortMsgTestHostComponent: ShortMessageTestHostComponent; + let shortMsgTestHostFixture: ComponentFixture; + + let longMsgTestHostComponent: LongMessageTestHostComponent; + let longMsgTestHostFixture: ComponentFixture; + + let shortMsgDurationTestHostComponent: ShortMessageWithDurationTestHostComponent; + let shortMsgDurationTestHostFixture: ComponentFixture; + + let status: StatusMsg; + let apiResonseError: ApiResponseError; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + imports: [ + MatCardModule, + MatIconModule, + MatListModule, + RouterTestingModule + ], + providers: [ + StatusMsg, + ApiResponseError + ], + declarations: [ + MessageComponent, + ShortMessageTestHostComponent, + LongMessageTestHostComponent, + ShortMessageWithDurationTestHostComponent + ] + }).compileComponents(); + + status = TestBed.inject(StatusMsg); + apiResonseError = TestBed.inject(ApiResponseError); + + })); + + describe('display a short message', () => { + beforeEach(() => { + shortMsgTestHostFixture = TestBed.createComponent(ShortMessageTestHostComponent); + shortMsgTestHostComponent = shortMsgTestHostFixture.componentInstance; + shortMsgTestHostFixture.detectChanges(); + }); + + it('should create', () => { + expect(shortMsgTestHostComponent.messageComponent).toBeTruthy(); + }); + + it('should display a short message', () => { + expect(shortMsgTestHostComponent.messageComponent).toBeTruthy(); + expect(shortMsgTestHostComponent.messageComponent.message.status).toEqual(200); + expect(shortMsgTestHostComponent.messageComponent.message.statusMsg).toEqual('Success'); + + const hostCompDe = shortMsgTestHostFixture.debugElement; + + const messageEl = hostCompDe.query(By.directive(MessageComponent)); + + const spanShortMessageElement = messageEl.query(By.css('.app-short-message-text')); + + expect(spanShortMessageElement.nativeElement.innerText).toEqual('You just updated the user profile.'); + + }); + + }); + + describe('display a long message', () => { + beforeEach(() => { + longMsgTestHostFixture = TestBed.createComponent(LongMessageTestHostComponent); + longMsgTestHostComponent = longMsgTestHostFixture.componentInstance; + longMsgTestHostFixture.detectChanges(); + }); + + it('should create', () => { + expect(longMsgTestHostComponent.messageComponent).toBeTruthy(); + }); + + it('should display a long message', () => { + expect(longMsgTestHostComponent.messageComponent).toBeTruthy(); + + expect(longMsgTestHostComponent.messageComponent.message.status).toEqual(403); + expect(longMsgTestHostComponent.messageComponent.message.statusMsg).toEqual('Forbidden'); + expect(longMsgTestHostComponent.messageComponent.message.statusText).toEqual( + 'The request was a legal request, but the server is refusing to respond to it'); + + const hostCompDe = longMsgTestHostFixture.debugElement; + + const messageEl = hostCompDe.query(By.directive(MessageComponent)); + + const messageSubtitleElement = messageEl.query(By.css('.message-subtitle .left')); + + expect(messageSubtitleElement.nativeElement.innerText).toEqual('ERROR 403 | Forbidden'); + + const messageTitleElement = messageEl.query(By.css('.message-title')); + + expect(messageTitleElement.nativeElement.innerText).toEqual( + 'The request was a legal request, but the server is refusing to respond to it'); + + }); + }); + + describe('display a short message with a duration of 2 seconds', () => { + beforeEach(() => { + shortMsgDurationTestHostFixture = TestBed.createComponent(ShortMessageWithDurationTestHostComponent); + shortMsgDurationTestHostComponent = shortMsgDurationTestHostFixture.componentInstance; + shortMsgDurationTestHostFixture.detectChanges(); + }); + + it('should create', () => { + expect(shortMsgDurationTestHostComponent.messageComponent).toBeTruthy(); + }); + + it('should display a short message', fakeAsync(() => { + expect(shortMsgDurationTestHostComponent.messageComponent).toBeTruthy(); + expect(shortMsgDurationTestHostComponent.messageComponent.message.status).toEqual(200); + expect(shortMsgDurationTestHostComponent.messageComponent.message.statusMsg).toEqual('Success'); + + const hostCompDe = shortMsgDurationTestHostFixture.debugElement; + + const messageEl = hostCompDe.query(By.directive(MessageComponent)); + + const spanShortMessageElement = messageEl.query(By.css('.app-short-message-text')); + + expect(spanShortMessageElement.nativeElement.innerText).toEqual('You just updated the user profile.'); + + shortMsgDurationTestHostFixture.whenStable().then(() => { + expect(shortMsgDurationTestHostComponent.messageComponent.disable).toBeTruthy(); + }); + })); + }); +}); diff --git a/src/app/main/action/message/message.component.ts b/src/app/main/action/message/message.component.ts new file mode 100644 index 0000000000..90083f5c5c --- /dev/null +++ b/src/app/main/action/message/message.component.ts @@ -0,0 +1,233 @@ +import { Location } from '@angular/common'; +import { Component, Input, OnInit } from '@angular/core'; +import { ActivatedRoute, Router } from '@angular/router'; +import { ApiResponseError } from '@dasch-swiss/dsp-js'; +import { StatusMsg } from 'src/assets/http/statusMsg'; + +/** + * @ignore + * Data type for messages + */ +export class AppMessageData { + status: number; + statusMsg?: string; + statusText?: string; + type?: string; + route?: string; + footnote?: string; + errorInfo?: string; + url?: string; +} + +/** + * @deprecated Will be replaced by notification service with material snackbar + */ +@Component({ + selector: 'app-message', + templateUrl: './message.component.html', + styleUrls: ['./message.component.scss'] +}) +export class MessageComponent implements OnInit { + + /** + * message type: AppMessageData + * + * @param message This type needs at least a status number (0-599). + * In this case, or if type is ApiResponseError, it takes the default status messages + * from the list of HTTP status codes (https://en.wikipedia.org/wiki/List_of_HTTP_status_codes) + */ + @Input() message: AppMessageData = new AppMessageData(); + + /** + * message type: ApiResponseError + * @param apiError + */ + @Input() apiError?: ApiResponseError; + + /** + * size of the message: long, medium or short? + * @param size Default size is 'long' + */ + @Input() size: 'short' | 'medium' | 'long' = 'long'; + + /** + * @deprecated + * @param short Show short message only + * A small message box to notify the user an event has occured. + */ + @Input() short = (this.size === 'short'); + + /** + * @deprecated + * @param medium Show medium message + * A message box without footnote or links. + */ + @Input() medium = (this.size === 'medium'); + + /** + * @param duration How long should the message be displayed + */ + @Input() duration?: number; + + statusMsg: any; + + isLoading = true; + + showLinks = false; + + // disable message + disable = false; + + /** + * @ignore + * default link list, which will be used in message content to give a user some possibilities + * what he can do in the case of an error + * + */ + links: any = { + title: 'You have the following possibilities now', + list: [ + { + label: 'go to the start page', + route: '/', + icon: 'keyboard_arrow_right' + }, + { + label: 'try to login', + route: '/login', + icon: 'keyboard_arrow_right' + }, + { + label: 'go back', + route: '<--', + icon: 'keyboard_arrow_left' + } + ] + }; + + constructor( + private _router: Router, + private _location: Location, + private _activatedRoute: ActivatedRoute, + private _status: StatusMsg + ) { } + + ngOnInit() { + // temporary solution as long we have to support the deprecated inputs "short" and "medium" + if (this.short || this.medium) { + this.size = (this.short ? 'short' : 'medium'); + } + + + if (this.apiError) { + this.message.status = this.apiError.status; + } + + this.statusMsg = this._status.default; + + if (!this.message) { + this._activatedRoute.data.subscribe((data: any) => { + this.message.status = data.status; + }); + } + + this.message = this.setMessage(this.message); + this.isLoading = false; + if (this.duration) { + setTimeout(() => this.disable = true, this.duration); + } + } + + setMessage(msg: AppMessageData) { + const tmpMsg: AppMessageData = {} as AppMessageData; + + const s: number = msg.status === 0 ? 503 : msg.status; + + tmpMsg.status = s; + tmpMsg.route = msg.route; + tmpMsg.statusMsg = msg.statusMsg; + tmpMsg.statusText = msg.statusText; + tmpMsg.route = msg.route; + tmpMsg.footnote = msg.footnote; + + switch (true) { + case s > 0 && s < 300: + // the message is a note + tmpMsg.type = 'note'; + tmpMsg.statusMsg = + msg.statusMsg !== undefined + ? msg.statusMsg + : this.statusMsg[s].message; + tmpMsg.statusText = + msg.statusText !== undefined + ? msg.statusText + : this.statusMsg[s].description; + // console.log('the message is a note'); + break; + case s >= 300 && s < 400: + // the message is a warning + tmpMsg.type = 'warning'; + tmpMsg.statusMsg = + msg.statusMsg !== undefined + ? msg.statusMsg + : this.statusMsg[s].message; + tmpMsg.statusText = + msg.statusText !== undefined + ? msg.statusText + : this.statusMsg[s].description; + // console.log('the message is a warning'); + + break; + case s >= 400 && s < 500: + // the message is a client side (app) error + // console.error('the message is a client side (app) error', s); + tmpMsg.type = 'error'; + tmpMsg.statusMsg = + msg.statusMsg !== undefined + ? msg.statusMsg + : this.statusMsg[s].message; + tmpMsg.statusText = + msg.statusText !== undefined + ? msg.statusText + : this.statusMsg[s].description; + this.showLinks = (this.size === 'long'); + break; + case s >= 500 && s < 600: + // the message is a server side (api) error + // console.error('the message is a server side (api) error'); + tmpMsg.type = 'error'; + tmpMsg.statusMsg = + msg.statusMsg !== undefined + ? msg.statusMsg + : this.statusMsg[s].message; + tmpMsg.statusText = + msg.statusText !== undefined + ? msg.statusText + : this.statusMsg[s].description; + this.showLinks = false; + break; + default: + // no default configuration? + break; + } + + return tmpMsg; + } + + goToLocation(route: string) { + if (route === '<--') { + this._location.back(); + } else { + this._router.navigate([route]); + } + } + + closeMessage() { + this.disable = !this.disable; + } + + reload() { + window.location.reload(); + } + +} diff --git a/src/app/main/action/progress-indicator/progress-indicator.component.html b/src/app/main/action/progress-indicator/progress-indicator.component.html new file mode 100644 index 0000000000..847cc13cc1 --- /dev/null +++ b/src/app/main/action/progress-indicator/progress-indicator.component.html @@ -0,0 +1,33 @@ + +
+ +
+
+
+ +
+ + keyboard_arrow_right + + done + + not_interested +
+ +
+ + + +
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/app/main/action/progress-indicator/progress-indicator.component.scss b/src/app/main/action/progress-indicator/progress-indicator.component.scss new file mode 100644 index 0000000000..52168ece0b --- /dev/null +++ b/src/app/main/action/progress-indicator/progress-indicator.component.scss @@ -0,0 +1,179 @@ +// -------------------------------------------------------------- +// configuration TODO: we should implement this from the app! +// -------------------------------------------------------------- + +// color definitions +$bright: rgba(249, 249, 249, 1); +$dark: rgba(41, 41, 41, 1); + +$mute: rgba(128, 128, 128, 0.8); + +$accent: rgba(255, 196, 0, 1); +$primary: rgba(0, 105, 92, 1); +$warn: rgb(244, 67, 54); + +// -------------------------------------------------------------- + +// +// two lines with three bouncing squares each +// + +.app-progress-indicator.default { + height: 56px; + margin-left: auto; + margin-right: auto; + padding: 24px 36px; + top: 60px; + width: 96px; + + &.page-center { + left: 50%; + position: absolute; + top: 39%; + transform: translate(-50%, -50%); + } + + p, + h1, + h2, + h3 { + color: #555555; + text-align: center; + } + + .line { + margin: 0 auto; + text-align: center; + width: 70px; + + > div { + animation: bounce-keyframes 1.4s infinite ease-in-out both; + background-color: $primary; + border-radius: 6px; + display: inline-block; + height: 18px; + width: 18px; + } + + .bounce1 { + animation-delay: -.32s; + } + + .bounce2 { + animation-delay: -.16s; + } + } + + @keyframes bounce-keyframes { + + 0%, 80%, 100% { + transform: scale(0); + } + + 40% { + transform: scale(1); + } + } +} + +// +// submit data has a different progress indicator +// +.app-progress-indicator.submit { + height: 32px; + width: 32px; + + .on-submit { + animation: spinner-keyframes .7s linear infinite; + height: 32px; + width: 32px; + + .spinner { + border: 2px solid $primary; + border-bottom-color: transparent; + border-radius: 50%; + border-right-color: transparent; + height: 28px; + width: 28px; + } + } + + .before-submit { + color: $mute; + } + + .after-submit { + color: $primary; + } + + .submit-error { + color: $warn; + } + + @keyframes spinner-keyframes { + 0% { transform: rotate(0deg); transform: rotate(0deg); } + 100% { transform: rotate(360deg); transform: rotate(360deg); } + } + +} + +// +// loader element +// +.loading-progress-indicator { + text-align: center; + width: 100%; + + .text { + color: $primary; + font-size: 12pt; + } + + .dot { + animation: dot-keyframes 1.4s infinite ease-in-out; + background-color: $primary; + border-radius: 2px; + display: inline-block; + height: 6px; + margin: 3px 6px 2px; + width: 6px; + + &:nth-child(2) { + animation-delay: .16s; + } + + &:nth-child(3) { + animation-delay: .32s; + } + + &:nth-child(4) { + animation-delay: .48s; + } + + &:nth-child(5) { + animation-delay: .64s; + } + + &:nth-child(6) { + animation-delay: .8s; + } + } +} + +@keyframes dot-keyframes { + 0% { + opacity: .4; + transform: scale(1, 1); + } + + 50% { + opacity: 1; + transform: scale(1.2, 1.2); + } + + 100% { + opacity: .4; + transform: scale(1, 1); + } +} + diff --git a/src/app/main/action/progress-indicator/progress-indicator.component.spec.ts b/src/app/main/action/progress-indicator/progress-indicator.component.spec.ts new file mode 100644 index 0000000000..ecf2c1343d --- /dev/null +++ b/src/app/main/action/progress-indicator/progress-indicator.component.spec.ts @@ -0,0 +1,165 @@ +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; + +import { Component, OnInit, ViewChild } from '@angular/core'; +import { MatIconModule } from '@angular/material/icon'; +import { By } from '@angular/platform-browser'; +import { ProgressIndicatorComponent } from './progress-indicator.component'; + +/** + * test host component to simulate parent component with a progress bar. + */ +@Component({ + template: ` + ` +}) +class TestHostComponent implements OnInit { + + @ViewChild('progressIndicator', { static: false }) progressIndicatorComponent: ProgressIndicatorComponent; + + status = 0; + color = 'red'; + + constructor() { + } + + ngOnInit() { } +} + +describe('ProgressIndicatorComponent', () => { + let testHostComponent: TestHostComponent; + let testHostFixture: ComponentFixture; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + imports: [ + MatIconModule + ], + declarations: [ + ProgressIndicatorComponent, + TestHostComponent + ] + }) + .compileComponents(); + })); + + + beforeEach(() => { + testHostFixture = TestBed.createComponent(TestHostComponent); + testHostComponent = testHostFixture.componentInstance; + testHostFixture.detectChanges(); + + expect(testHostComponent).toBeTruthy(); + }); + + it('should create', () => { + expect(testHostComponent.progressIndicatorComponent).toBeTruthy(); + }); + + it('should display a red spinner as progress indicator', () => { + expect(testHostComponent.progressIndicatorComponent).toBeTruthy(); + expect(testHostComponent.progressIndicatorComponent.color).toEqual('red'); + expect(testHostComponent.progressIndicatorComponent.status).toEqual(0); + + const hostCompDe = testHostFixture.debugElement; + + const progressEl = hostCompDe.query(By.directive(ProgressIndicatorComponent)); + + const divProgressElement = progressEl.query(By.css('.app-progress-indicator')); + + const submitEl = divProgressElement.query(By.css('div')); + + const spinnerEl = submitEl.query(By.css('div')); + + expect(spinnerEl.styles.borderTopColor).toEqual('red'); + expect(spinnerEl.styles.borderLeftColor).toEqual('red'); + + }); + + it('should change the color of the progress indicator from red to blue', () => { + expect(testHostComponent.progressIndicatorComponent).toBeTruthy(); + expect(testHostComponent.progressIndicatorComponent.color).toEqual('red'); + expect(testHostComponent.progressIndicatorComponent.status).toEqual(0); + + const hostCompDe = testHostFixture.debugElement; + + const progressEl = hostCompDe.query(By.directive(ProgressIndicatorComponent)); + + const divProgressElement = progressEl.query(By.css('.app-progress-indicator')); + + const submitEl = divProgressElement.query(By.css('div')); + + const spinnerEl = submitEl.query(By.css('div')); + + expect(spinnerEl.styles.borderTopColor).toEqual('red'); + expect(spinnerEl.styles.borderLeftColor).toEqual('red'); + + // change the color of the spinner + testHostComponent.progressIndicatorComponent.color = 'blue'; + + testHostFixture.detectChanges(); + + // expect the spinner to be blue + expect(spinnerEl.styles.borderTopColor).toEqual('blue'); + expect(spinnerEl.styles.borderLeftColor).toEqual('blue'); + }); + + it('should update the progress indicator according to the status value', () => { + expect(testHostComponent.progressIndicatorComponent).toBeTruthy(); + expect(testHostComponent.progressIndicatorComponent.color).toEqual('red'); + expect(testHostComponent.progressIndicatorComponent.status).toEqual(0); + + const hostCompDe = testHostFixture.debugElement; + + const progressEl = hostCompDe.query(By.directive(ProgressIndicatorComponent)); + + const divProgressElement = progressEl.query(By.css('.app-progress-indicator')); + + const submitEl = divProgressElement.query(By.css('div')); + + const spinnerEl = submitEl.query(By.css('div')); + + expect(spinnerEl.attributes.class).toEqual('spinner'); + + // update status value to 1 + testHostComponent.progressIndicatorComponent.status = 1; + + testHostFixture.detectChanges(); + + const divEl = divProgressElement.query(By.css('div')); + + const matIconEl = divProgressElement.query(By.css('mat-icon')); + + // new status: done + expect(matIconEl.attributes.class).toEqual('mat-icon notranslate after-submit material-icons mat-icon-no-color'); + expect(matIconEl.nativeElement.innerText).toEqual('done'); + expect(matIconEl.styles.color).toEqual('red'); + }); + + it('should display the default progress indicator when the status is undefined', () => { + expect(testHostComponent.progressIndicatorComponent).toBeTruthy(); + expect(testHostComponent.progressIndicatorComponent.color).toEqual('red'); + expect(testHostComponent.progressIndicatorComponent.status).toEqual(0); + + // change the status to undefined + testHostComponent.progressIndicatorComponent.status = undefined; + + expect(testHostComponent.progressIndicatorComponent.status).toEqual(undefined); + + testHostFixture.detectChanges(); + + const hostCompDe = testHostFixture.debugElement; + + const progressEl = hostCompDe.query(By.directive(ProgressIndicatorComponent)); + + const divProgressElement = progressEl.query(By.css('.app-progress-indicator')); + + const lineEl = divProgressElement.query(By.css('.line')); + + const bounceEl = lineEl.query(By.css('div')); + + // expect the default progress indicator + expect(bounceEl.styles.backgroundColor).toEqual('red'); + expect(bounceEl.attributes.class).toEqual('bounce1'); + + }); +}); diff --git a/src/app/main/action/progress-indicator/progress-indicator.component.ts b/src/app/main/action/progress-indicator/progress-indicator.component.ts new file mode 100644 index 0000000000..0023a3a363 --- /dev/null +++ b/src/app/main/action/progress-indicator/progress-indicator.component.ts @@ -0,0 +1,42 @@ +import { Component, Input, OnInit } from '@angular/core'; + +@Component({ + selector: 'app-progress-indicator', + templateUrl: './progress-indicator.component.html', + styleUrls: ['./progress-indicator.component.scss'] +}) +export class ProgressIndicatorComponent implements OnInit { + + /** + * @param status number relating to status + * + * [status] is a number and can be used when submitting form data: + * + * - not ready: -1 + * - loading: 0 + * - done: 1 + * + * - error: 400 + */ + @Input() status?: number; + + /** + * @param color Hex value or predefined color from scss + * + * Parameter to customize the appearance of the loader. + * Hexadecimal color value e.g. #00ff00 or similar color values 'red', 'green' etc. + * + * TODO: Default color should come from app settings + */ + @Input() color = '#5849a7'; + + /** + * @ignore + */ + constructor() { + } + + ngOnInit() { + } + +} diff --git a/src/app/main/action/selected-resources/selected-resources.component.html b/src/app/main/action/selected-resources/selected-resources.component.html new file mode 100644 index 0000000000..f681553a1c --- /dev/null +++ b/src/app/main/action/selected-resources/selected-resources.component.html @@ -0,0 +1,20 @@ +
+ +

Number of resources selected: {{ resCount }}

+ +

Ids of the Selected resources are:

+ + + description +
{{ id }}
+
+
+ +
+ + + + + +
+
diff --git a/src/app/main/action/selected-resources/selected-resources.component.scss b/src/app/main/action/selected-resources/selected-resources.component.scss new file mode 100644 index 0000000000..2e8c1a55d7 --- /dev/null +++ b/src/app/main/action/selected-resources/selected-resources.component.scss @@ -0,0 +1,37 @@ + +.selected-resources { + margin-top: 10px; + + .mat-list-item { + height: auto; + + .mat-icon { + font-size: 20px; + color: #673ab7; + } + + .mat-line { + white-space: normal !important; + } + + .mat-list-item { + margin-left: 0px; + } + } +} + +.res-ids-container { + margin: 8px; +} + +.res-value { + color: #673ab7; +} + +.action-buttons { +margin-top: 20px; + +.res-action { + margin: 4px; +} +} diff --git a/src/app/main/action/selected-resources/selected-resources.component.spec.ts b/src/app/main/action/selected-resources/selected-resources.component.spec.ts new file mode 100644 index 0000000000..836a9e8d69 --- /dev/null +++ b/src/app/main/action/selected-resources/selected-resources.component.spec.ts @@ -0,0 +1,62 @@ +import { HarnessLoader } from '@angular/cdk/testing'; +import { TestbedHarnessEnvironment } from '@angular/cdk/testing/testbed'; +import { Component, ViewChild } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { MatIconModule } from '@angular/material/icon'; +import { MatButtonModule } from '@angular/material/button'; +import { MatListModule } from '@angular/material/list'; + +import { SelectedResourcesComponent } from './selected-resources.component'; + +/** + * test host component to simulate parent component. + */ +@Component({ + selector: 'app-selected-resources-host-component', + template: ` + + ` +}) +class TestHostSelectedResourcesComponent { + + @ViewChild('selectedResources') selectedResources: SelectedResourcesComponent; + + selectedResCount = 2; + selectedRes = ['http://rdfh.ch/0001/H6gBWUuJSuuO-CilHV8kQw', 'http://rdfh.ch/0010/H6gBWUuJSuuO-CilddsgfdQw']; + + getActionType(action: string) { + console.log(action); + } +} + +describe('SelectedResourcesComponent', () => { + let testHostComponent: TestHostSelectedResourcesComponent; + let testHostFixture: ComponentFixture; + let loader: HarnessLoader; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + declarations: [ + SelectedResourcesComponent, + TestHostSelectedResourcesComponent + ], + imports: [ + MatListModule, + MatButtonModule, + MatIconModule + ] + }) + .compileComponents(); + })); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(TestHostSelectedResourcesComponent); + testHostComponent = testHostFixture.componentInstance; + loader = TestbedHarnessEnvironment.loader(testHostFixture); + testHostFixture.detectChanges(); + }); + + it('should create', () => { + expect(testHostComponent.selectedResources).toBeTruthy(); + }); +}); diff --git a/src/app/main/action/selected-resources/selected-resources.component.ts b/src/app/main/action/selected-resources/selected-resources.component.ts new file mode 100644 index 0000000000..3b724582d1 --- /dev/null +++ b/src/app/main/action/selected-resources/selected-resources.component.ts @@ -0,0 +1,36 @@ +import { Component, EventEmitter, Input, Output } from '@angular/core'; + +@Component({ + selector: 'app-selected-resources', + templateUrl: './selected-resources.component.html', + styleUrls: ['./selected-resources.component.scss'] +}) +export class SelectedResourcesComponent { + + // total number of resources selected + @Input() resCount: number; + + // list of selected resources ids + @Input() resIds: string[]; + + // return selected actions and other info if any + @Output() actionType: EventEmitter = new EventEmitter(); + + // actions which can be applied on selected resources + resourceAction: 'compare' | 'edit' | 'delete' | 'starred' | 'cancel'; + + constructor() { } + + // return compare action + compareResources() { + this.resourceAction = 'compare'; + this.actionType.emit(this.resourceAction); + } + + // cancel action + cancelSelection() { + this.resourceAction = 'cancel'; + this.actionType.emit(this.resourceAction); + } + +} diff --git a/src/app/main/action/sort-button/sort-button.component.html b/src/app/main/action/sort-button/sort-button.component.html new file mode 100644 index 0000000000..621e5e323f --- /dev/null +++ b/src/app/main/action/sort-button/sort-button.component.html @@ -0,0 +1,13 @@ + + + + + + diff --git a/src/app/main/action/sort-button/sort-button.component.scss b/src/app/main/action/sort-button/sort-button.component.scss new file mode 100644 index 0000000000..d7b9d11b0b --- /dev/null +++ b/src/app/main/action/sort-button/sort-button.component.scss @@ -0,0 +1,12 @@ +.active { + background: rgba(128,128,128,.39); + font-weight: bolder; +} + +.right.sort { + float: right; + + .mat-icon { + transform: scale(-1, 1); + } +} diff --git a/src/app/main/action/sort-button/sort-button.component.spec.ts b/src/app/main/action/sort-button/sort-button.component.spec.ts new file mode 100644 index 0000000000..a782af9aea --- /dev/null +++ b/src/app/main/action/sort-button/sort-button.component.spec.ts @@ -0,0 +1,168 @@ +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; + +import { MatIconModule } from '@angular/material/icon'; +import { MatMenuModule } from '@angular/material/menu'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; +import { SortButtonComponent } from './sort-button.component'; + +import { Component, DebugElement, OnInit, ViewChild, } from '@angular/core'; +import { By } from '@angular/platform-browser'; +import { SortingService } from '../../services/sorting.service'; + +/** + * test host component to simulate parent component with a progress bar. + */ +@Component({ + template: ` + + +
    +
  • + {{item.firstname}} + {{item.lastname}} + by + {{item.creator}} +
  • +
+ ` +}) +class TestHostComponent implements OnInit { + + @ViewChild('sortButton', { static: false }) sortButtonComponent: SortButtonComponent; + + sortingService: SortingService = new SortingService(); + + sortProps: any = [{ + key: 'firstname', + label: 'First name' + }, + { + key: 'lastname', + label: 'Last name' + }, + { + key: 'creator', + label: 'Creator' + } + ]; + position = 'left'; + + list = [{ + firstname: 'a', + lastname: 'z', + creator: 'André Franquin' + + }, + { + firstname: 'b', + lastname: 'y', + creator: 'Walt Disney' + + }, + { + firstname: 'c', + lastname: 'x', + creator: 'William Shakespeare' + + }, + { + firstname: 'd', + lastname: 'w', + creator: 'Charles M. Schulz' + + } + ]; + + constructor() { + } + + ngOnInit() { } + + sortList(key) { + this.list = this.sortingService.keySortByAlphabetical(this.list, key); + } +} + +describe('SortButtonComponent', () => { + let testHostComponent: TestHostComponent; + let testHostFixture: ComponentFixture; + const listData = [ + { firstname: 'a', lastname: 'z', creator: 'André Franquin' }, + { firstname: 'b', lastname: 'y', creator: 'Walt Disney' }, + { firstname: 'c', lastname: 'x', creator: 'William Shakespeare' }, + { firstname: 'd', lastname: 'w', creator: 'Charles M. Schulz' }, + ]; + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + imports: [ + MatIconModule, + MatMenuModule, + BrowserAnimationsModule + ], + declarations: [ + SortButtonComponent, + TestHostComponent, + ] + }) + .compileComponents(); + })); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(TestHostComponent); + testHostComponent = testHostFixture.componentInstance; + testHostFixture.detectChanges(); + + expect(testHostComponent).toBeTruthy(); + }); + + it('should create an instance', () => { + expect(testHostComponent.sortButtonComponent).toBeTruthy(); + }); + + it('should sort the list by lastname', () => { + expect(testHostComponent.sortButtonComponent).toBeTruthy(); + expect(testHostComponent.list).toEqual(listData); + + const hostCompDe = testHostFixture.debugElement; + + const spanEl: DebugElement = hostCompDe.query(By.css('span')); + + // expect the button position to be 'right' + // expect(spanEl.properties).toEqual({ 'className': 'left' }); + + const sortSelectionBtnEl: DebugElement = spanEl.query(By.css('button')); + + const matIconEl: DebugElement = sortSelectionBtnEl.query(By.css('mat-icon')); + + // expect that the button label is 'sort' + expect(matIconEl.nativeElement.innerText).toEqual('sort'); + + // click on the sort button to trigger the sort selection menu + sortSelectionBtnEl.triggerEventHandler('click', null); + + const matMenuEl = spanEl.query(By.css('mat-menu')); + + const sortSelectionEl = matMenuEl.references.sortSelection; + + // expect that items's names of the sort list are 'Firstname', 'Last name' and 'Creator' + expect(sortSelectionEl.items._results[0]._elementRef.nativeElement.innerText).toEqual('First name'); + expect(sortSelectionEl.items._results[1]._elementRef.nativeElement.innerText).toEqual('Last name'); + expect(sortSelectionEl.items._results[2]._elementRef.nativeElement.innerText).toEqual('Creator'); + + // sort by 'lastname' through a click event + const item2 = sortSelectionEl.items._results[1]._elementRef.nativeElement; + item2.click(); + testHostFixture.detectChanges(); + + + const listEl: DebugElement = hostCompDe.query(By.css('.list')); + const children = listEl.nativeNode.children; + expect(children[0].innerText).toEqual('d w by Charles M. Schulz'); + expect(children[1].innerText).toEqual('c x by William Shakespeare'); + expect(children[2].innerText).toEqual('b y by Walt Disney'); + expect(children[3].innerText).toEqual('a z by André Franquin'); + }); +}); + + diff --git a/src/app/main/action/sort-button/sort-button.component.ts b/src/app/main/action/sort-button/sort-button.component.ts new file mode 100644 index 0000000000..7dc60a0758 --- /dev/null +++ b/src/app/main/action/sort-button/sort-button.component.ts @@ -0,0 +1,80 @@ +import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; + +export interface SortProp { + key: string; + label: string; +} + +/** + * a component with a list of properties to sort a list by one of them. + * It can be used together with the DspSortBy pipe. + */ +@Component({ + selector: 'app-sort-button', + templateUrl: './sort-button.component.html', + styleUrls: ['./sort-button.component.scss'] +}) +export class SortButtonComponent implements OnInit { + + /** + * @param sortProps[] + * An array of SortProp objects for the selection menu: + * SortProp: { key: string, label: string } + */ + @Input() sortProps: SortProp[]; + + /** + * @param position string + * Optional position of the sort menu: right or left + * e.g. [position='left'] + */ + @Input() position: 'left' | 'right' = 'left'; + + /** + * @param icon + * Default icon is "sort" from material design. + * But you can replace it with another one + * e.g. sort_by_alpha + */ + @Input() icon = 'sort'; + + + /** + * @param activeKey + * Optional parameter: selected sort property key + * By default it takes the first key from sortProps + */ + @Input() activeKey?: string; + + /** + * @emits {string} sortKeyChange + * + * EventEmitter when a user selected a sort property; + * This is the selected key + */ + @Output() sortKeyChange: EventEmitter = new EventEmitter(); + + menuXPos: 'before' | 'after' = 'after'; + + constructor() { + } + + ngOnInit() { + if (this.position === 'right') { + this.menuXPos = 'before'; + } + + this.sortBy(this.activeKey); + } + + /** + * @ignore + * + * @param key a string to sort by + */ + sortBy(key: string) { + this.activeKey = (key ? key : this.sortProps[0].key); + this.sortKeyChange.emit(this.activeKey); + } + +} diff --git a/src/app/main/action/string-literal-input/string-literal-input.component.html b/src/app/main/action/string-literal-input/string-literal-input.component.html new file mode 100644 index 0000000000..3f9ee7e7e7 --- /dev/null +++ b/src/app/main/action/string-literal-input/string-literal-input.component.html @@ -0,0 +1,79 @@ +
+ + + + + + + + + + + + + + + + + + +
+ + + + {{lang}} + + + + + + +
+ +
diff --git a/src/app/main/action/string-literal-input/string-literal-input.component.scss b/src/app/main/action/string-literal-input/string-literal-input.component.scss new file mode 100644 index 0000000000..6176307811 --- /dev/null +++ b/src/app/main/action/string-literal-input/string-literal-input.component.scss @@ -0,0 +1,51 @@ +.mat-form-field { + width: 100% !important; +} + +.existing-value { + font-weight: bold; +} + +// string literal textarea: overwrite material styling !important +.string-literal { + &.short-text { + .mat-button.select-lang { + min-width: 36px !important; + padding: 0 8px !important; + border-radius: 4px 0 0 0 !important; + } + } + + &.long-text { + display: flex; + + // select language + .mat-button-toggle { + height: 36px; + width: 56px; + + .mat-button-toggle-button { + width: 50px !important; + } + } + + .mat-button-toggle-vertical { + height: calc(4 * 36px + 2px); + border-radius: 4px 0 0 4px !important; + border-right: 0px; + overflow-y: auto; + + &::-webkit-scrollbar { + width: 0px; + } + } + + .string-literal-textarea { + display: flex !important; + + textarea { + min-height: calc(4 * 36px) !important; + } + } + } +} diff --git a/src/app/main/action/string-literal-input/string-literal-input.component.spec.ts b/src/app/main/action/string-literal-input/string-literal-input.component.spec.ts new file mode 100644 index 0000000000..66ef44854c --- /dev/null +++ b/src/app/main/action/string-literal-input/string-literal-input.component.spec.ts @@ -0,0 +1,346 @@ +import { HarnessLoader } from '@angular/cdk/testing'; +import { TestbedHarnessEnvironment } from '@angular/cdk/testing/testbed'; +import { Component, DebugElement, OnInit, ViewChild } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { ReactiveFormsModule } from '@angular/forms'; +import { MatButtonToggleModule } from '@angular/material/button-toggle'; +import { MatButtonHarness } from '@angular/material/button/testing'; +import { MatFormFieldModule } from '@angular/material/form-field'; +import { MatIconModule } from '@angular/material/icon'; +import { MatInputModule } from '@angular/material/input'; +import { MatInputHarness } from '@angular/material/input/testing'; +import { MatMenuModule } from '@angular/material/menu'; +import { By } from '@angular/platform-browser'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; +import { StringLiteral } from '@dasch-swiss/dsp-js'; +import { DspApiConnectionToken } from '../../declarations/dsp-api-tokens'; +import { SessionService } from '../../services/session.service'; +import { StringLiteralInputComponent } from './string-literal-input.component'; + +/** + * test host component to simulate parent component. + */ +@Component({ + template: ` + + ` +}) +class TestHostStringLiteralInputComponent implements OnInit { + + @ViewChild('stringLiteralInputVal') stringLiteralInputComponent: StringLiteralInputComponent; + + labels: StringLiteral[]; + + language: string; + + isTextarea: boolean; + + ngOnInit() { + + this.labels = [ + { + value: 'Welt', + language: 'de' + }, + { + value: 'World', + language: 'en' + }, + { + value: 'Monde', + language: 'fr' + }, + { + value: 'Mondo', + language: 'it' + }, + ]; + + this.language = 'en'; + } +} + +describe('StringLiteralInputComponent', () => { + let testHostComponent: TestHostStringLiteralInputComponent; + let testHostFixture: ComponentFixture; + let loader: HarnessLoader; + let sessionService: SessionService; + + let sliComponentDe: DebugElement; + let sliMenuDebugElement: DebugElement; + let sliMenuNativeElement; + let langButton; + + beforeEach(waitForAsync(() => { + + // empty spy object to use in the providers for the SessionService injection + const dspConnSpy = { }; + + TestBed.configureTestingModule({ + declarations: [ + StringLiteralInputComponent, + TestHostStringLiteralInputComponent + ], + imports: [ + MatMenuModule, + MatInputModule, + MatIconModule, + MatButtonToggleModule, + MatFormFieldModule, + BrowserAnimationsModule, + ReactiveFormsModule + ], + providers: [ + { + provide: DspApiConnectionToken, + useValue: dspConnSpy + }, + SessionService + ] + }) + .compileComponents(); + + sessionService = TestBed.inject(SessionService); + })); + + beforeEach(() => { + testHostFixture = TestBed.createComponent(TestHostStringLiteralInputComponent); + testHostComponent = testHostFixture.componentInstance; + loader = TestbedHarnessEnvironment.loader(testHostFixture); + testHostFixture.detectChanges(); + expect(testHostComponent).toBeTruthy(); + expect(testHostComponent.stringLiteralInputComponent).toBeTruthy(); + + const hostCompDe = testHostFixture.debugElement; + sliComponentDe = hostCompDe.query(By.directive(StringLiteralInputComponent)); + + expect(sliComponentDe).toBeTruthy(); + }); + + it('should load values and assign them to the correct language', async () => { + + const inputElement = await loader.getHarness(MatInputHarness.with({ selector: '.inputValue' })); + + expect(await inputElement.getValue()).toEqual('World'); + + const langSelectButtonElement = await loader.getHarness(MatButtonHarness.with({ selector: '.select-lang' })); + + expect(langSelectButtonElement).toBeTruthy(); + + // open language select button + await langSelectButtonElement.click(); + + // get reference to the mat-menu + sliMenuDebugElement = sliComponentDe.query(By.css('.lang-menu')); + + // get reference to mat-menu native element in order to be able to access the buttons + sliMenuNativeElement = sliMenuDebugElement.nativeElement; + + // select 'de' button + langButton = sliMenuNativeElement.children[0].children[0]; + + // simulate a user click on the button element to switch input value to the german value + langButton.click(); + + // expect the value of the german input to equal 'Welt' + expect(await inputElement.getValue()).toEqual('Welt'); + + // select 'fr' button + langButton = sliMenuNativeElement.children[0].children[1]; + + // switch to french + langButton.click(); + + // expect the value of the french input to equal 'Monde' + expect(await inputElement.getValue()).toEqual('Monde'); + + // select 'it' button + langButton = sliMenuNativeElement.children[0].children[2]; + + // switch to italian + langButton.click(); + + // expect the value of the italian input to equal 'Mondo' + expect(await inputElement.getValue()).toEqual('Mondo'); + }); + + it('should change a value and assign it to the correct language', async () => { + + const inputElement = await loader.getHarness(MatInputHarness.with({ selector: '.inputValue' })); + + const langSelectButtonElement = await loader.getHarness(MatButtonHarness.with({ selector: '.select-lang' })); + + expect(langSelectButtonElement).toBeTruthy(); + + // open language select button + await langSelectButtonElement.click(); + + // get reference to the mat-menu + sliMenuDebugElement = sliComponentDe.query(By.css('.lang-menu')); + + // get reference to mat-menu native element in order to be able to access the buttons + sliMenuNativeElement = sliMenuDebugElement.nativeElement; + + // select 'de' button + langButton = sliMenuNativeElement.children[0].children[0]; + + // simulate a user click on the button element to switch input value to the german value + langButton.click(); + + // expect the value of the german input to equal 'Welt' + expect(await inputElement.getValue()).toEqual('Welt'); + + // set new value for the german text + await inputElement.setValue('neue Welt'); + + // select 'fr' button + langButton = sliMenuNativeElement.children[0].children[1]; + + // switch to french + langButton.click(); + + // expect the value of the french input to equal 'Monde' + expect(await inputElement.getValue()).toEqual('Monde'); + + // select 'de' button + langButton = sliMenuNativeElement.children[0].children[0]; + + // switch back to german + langButton.click(); + + // expect the value to equal the new value given earlier + expect(await inputElement.getValue()).toEqual('neue Welt'); + + }); + + it('should switch input to a textarea and assign the values to the correct language', async () => { + testHostComponent.isTextarea = true; + + testHostFixture.detectChanges(); + + const inputElement = await loader.getHarness(MatInputHarness.with({ selector: '.textAreaValue' })); + + // get reference to the mat-menu + sliMenuDebugElement = sliComponentDe.query(By.css('.string-literal-select-lang')); + + // get reference to mat-menu native element in order to be able to access the buttons + sliMenuNativeElement = sliMenuDebugElement.nativeElement; + + // select 'de' button + langButton = sliMenuNativeElement.children[0]; + + // simulate a user click on the button element to switch input value to the german value + langButton.click(); + + // expect the value of the german input to equal 'Welt' + expect(await inputElement.getValue()).toEqual('Welt'); + + // select 'fr' button + langButton = sliMenuNativeElement.children[1]; + + // switch to french + langButton.click(); + + // expect the value of the french input to equal 'Monde' + expect(await inputElement.getValue()).toEqual('Monde'); + + // select 'it' button + langButton = sliMenuNativeElement.children[2]; + + // switch to italian + langButton.click(); + + // expect the value of the italian input to equal 'Mondo' + expect(await inputElement.getValue()).toEqual('Mondo'); + + // select 'en' button + langButton = sliMenuNativeElement.children[3]; + + // switch to english + langButton.click(); + + // expect the value of the english input to equal 'World' + expect(await inputElement.getValue()).toEqual('World'); + + }); + + it('should store a new value inside a textarea in the correct language', async () => { + testHostComponent.isTextarea = true; + + testHostFixture.detectChanges(); + + const inputElement = await loader.getHarness(MatInputHarness.with({ selector: '.textAreaValue' })); + + // get reference to the mat-menu + sliMenuDebugElement = sliComponentDe.query(By.css('.string-literal-select-lang')); + + // get reference to mat-menu native element in order to be able to access the buttons + sliMenuNativeElement = sliMenuDebugElement.nativeElement; + + // select 'en' button + langButton = sliMenuNativeElement.children[3]; + + // switch to english + langButton.click(); + + // expect the value of the english input to equal 'World' + expect(await inputElement.getValue()).toEqual('World'); + + // set new value for the german text + await inputElement.setValue('Brave New World'); + + // select 'de' button + langButton = sliMenuNativeElement.children[0]; + + // simulate a user click on the button element to switch input value to the german value + langButton.click(); + + // expect the value of the german input to equal 'Welt' + expect(await inputElement.getValue()).toEqual('Welt'); + + // select 'en' button again + langButton = sliMenuNativeElement.children[3]; + + // switch back to english + langButton.click(); + + // expect the value of the english input to equal the new value 'Brave New World' + expect(await inputElement.getValue()).toEqual('Brave New World'); + + }); + + it('should update values and assign them to the correct language when langauges object is changed', async () => { + + const inputElement = await loader.getHarness(MatInputHarness.with({ selector: '.inputValue' })); + + expect(await inputElement.getValue()).toEqual('World'); + + testHostComponent.labels = [ + { + value: 'Welt', + language: 'de' + }, + { + value: 'Brave New World', + language: 'en' + }, + { + value: 'Monde', + language: 'fr' + }, + { + value: 'Mondo', + language: 'it' + }, + ]; + + testHostFixture.detectChanges(); + + expect(await inputElement.getValue()).toEqual('Brave New World'); + }); +}); diff --git a/src/app/main/action/string-literal-input/string-literal-input.component.ts b/src/app/main/action/string-literal-input/string-literal-input.component.ts new file mode 100644 index 0000000000..a67b8d65f7 --- /dev/null +++ b/src/app/main/action/string-literal-input/string-literal-input.component.ts @@ -0,0 +1,296 @@ +import { Component, ElementRef, EventEmitter, Input, OnChanges, OnInit, Output, ViewChild } from '@angular/core'; +import { FormBuilder, FormControl, FormGroup } from '@angular/forms'; +import { MatMenuTrigger } from '@angular/material/menu'; +import { StringLiteral } from '@dasch-swiss/dsp-js'; +import { SessionService } from '../../services/session.service'; + +@Component({ + selector: 'app-string-literal-input', + templateUrl: './string-literal-input.component.html', + styleUrls: ['./string-literal-input.component.scss'] +}) +export class StringLiteralInputComponent implements OnInit, OnChanges { + + /** + * optional placeholder for the input field e.g. Label + * + * @param {string} [placeholder='Label'] + */ + @Input() placeholder = 'Label'; + + /** + * optional predefined (selected) language: en, de, it, fr, etc. + * + * @param {string} language + */ + @Input() language: string; + + /** + * optional form field input type: textarea? set to true for textarea + * otherwise it's a simple (short) input field + * + * @param {boolean} [textarea=false] + */ + @Input() textarea: boolean; + + /** + * optional form field value of type StringLiteral[] + * + * @param {StringLiteral[]} value + */ + @Input() value: StringLiteral[] = []; + + /** + * optional disable the input field in case of no right to edit the field/value + * + * @param {boolean}: [disabled=false] + */ + @Input() disabled: boolean; + + /** + * the readonly attribute specifies whether the control may be modified by the user. + * + * @param {boolean}: [readonly=false] + */ + @Input() readonly: boolean; + + /** + * returns (output) an array of StringLiteral on any change on the input field. + * + * @emits {StringLiteral[]} dataChanged + */ + @Output() dataChanged: EventEmitter = new EventEmitter(); + + /** + * returns (output) true when the field was touched. This can be used to validate data, e.g. in case a value is required + * + * @emits {boolean} touched + */ + @Output() touched: EventEmitter = new EventEmitter(); + + /** + * returns true when a user press ENTER. This can be used to submit data in the parent component. + * + * * @emits {boolean} enter + */ + @Output() enter: EventEmitter = new EventEmitter(); + + @ViewChild('textInput', { static: false }) textInput: ElementRef; + + @ViewChild('btnToSelectLanguage', { static: false }) btnToSelectLanguage: MatMenuTrigger; + + form: FormGroup; + languages: string[] = ['de', 'fr', 'it', 'en']; + + constructor(private _fb: FormBuilder, + private _sessionService: SessionService) { + + // set selected language, if it's not defined yet + if (!this.language) { + if (this._sessionService.getSession() !== null) { + // get language from the logged-in user profile data + this.language = this._sessionService.getSession().user.lang; + } else { + // get default language from browser + this.language = navigator.language.substr(0, 2); + } + } + + // does the defined language exists in our supported languages list? + if (this.languages.indexOf(this.language) === -1) { + // if not, select the first language from our list of supported languages + this.language = this.languages[0]; + } + + } + + ngOnInit() { + + // reset stringLiterals if they have empty values + this.resetValues(); + + // build the form + this.form = this._fb.group({ + text: new FormControl( + { + value: '', + disabled: this.disabled // https://stackoverflow.com/a/47521965 + } + ) + }); + // update values on form change + this.form.valueChanges.subscribe(data => this.onValueChanged()); + + // get value from stringLiterals + const val = this.getValueFromStringLiteral(this.language); + this.updateFormField(val); + } + + ngOnChanges() { + // get value from stringLiterals + const val = this.getValueFromStringLiteral(this.language); + this.updateFormField(val); + } + + /** + * @ignore + * + * emit data to parent on any change on the input field + */ + onValueChanged() { + if (!this.form) { + return; + } + + const form = this.form; + const control = form.get('text'); + this.touched.emit(control && control.dirty); + + this.updateStringLiterals(this.language, this.form.controls.text.value); + + this.dataChanged.emit(this.value); + + } + + toggleAll() { + // tODO: open/show all languages with their values + } + + /** + * @ignore + * + * Set the language after selecting; + * This updates the array of StringLiterals: adds item with the selected language if it doesn't exist + */ + setLanguage(lang: string) { + + if (this.language !== lang) { + // clean stringLIteral value for previous language, if text field is empty + this.updateStringLiterals(this.language, this.form.controls.text.value); + + this.language = lang; + // update form field value / reset in case of no value + const val = this.getValueFromStringLiteral(lang); + this.updateFormField(val); + } + } + + /** + * @ignore + * + * Switch focus to input field after selecting a language + */ + switchFocus() { + // close the menu + if (!this.textarea && this.btnToSelectLanguage && this.btnToSelectLanguage.menuOpen) { + this.btnToSelectLanguage.closeMenu(); + } + + if (!this.disabled) { + this.form.controls.text.enable(); + this.textInput.nativeElement.focus(); + } + } + + /** + * @ignore + * + * Switch focus to input field after closing the menu by clicking anywhere outside of it + */ + menuClosed() { + if (!this.disabled) { + this.form.controls.text.enable(); + this.textInput.nativeElement.focus(); + } + } + + /** + * @ignore + * + * Set the value in the input field + */ + updateFormField(value: string) { + if (!value) { + value = ''; + } + if (!this.form) { + return; + } + this.form.controls.text.setValue(value); + } + + /** + * @ignore + * + * Update the array of StringLiterals depending on value / empty value add or remove item from array. + */ + updateStringLiterals(lang: string, value?: string) { + const index = this.value.findIndex(i => i.language === lang); + + if (index > -1 && this.value[index].value.length > 0) { + // value is not empty and exists in list of stringLiterals + this.value[index].value = value; + } + + if ((!value || value.length === 0) && index > -1) { + // value is empty: delete stringLiteral item for this language + this.value.splice(index, 1); + } + + if (index < 0 && value) { + // value doesn't exist in stringLiterals: add one + const newValue: StringLiteral = { + value: value, + language: lang + }; + this.value.push(newValue); + } + + } + + /** + * @ignore + * + * In case of strange array of StringLiterals, this method will reset to a API-conform array. This means an array without empty values. + */ + resetValues() { + const length: number = this.value.length; + + if (length > 0) { + let index = length - 1; + while (index >= 0) { + // remove items with empty value + if (!this.value[index].value.length) { + this.value.splice(index, 1); + } + index--; + } + + // does an item for selected lanuage exists + if (this.value.findIndex(i => i.language === this.language) === -1) { + this.language = this.value[0].language; + } + + } else { + this.value = []; + } + } + + /** + * @ignore + * + * Get the value from array of StringLiterals for the selected language + */ + getValueFromStringLiteral(lang: string): string { + // get index for this language + const index = this.value.findIndex(i => i.language === lang); + + if (this.value[index] && this.value[index].value.length > 0) { + return this.value[index].value; + } else { + return undefined; + } + + } + +} diff --git a/src/app/main/directive/admin-image/admin-image.config.ts b/src/app/main/directive/admin-image/admin-image.config.ts new file mode 100644 index 0000000000..d0bb349f0e --- /dev/null +++ b/src/app/main/directive/admin-image/admin-image.config.ts @@ -0,0 +1,21 @@ +/* eslint-disable max-len */ +/** + * @ignore + */ +export class AdminImageConfig { + + /** + * default-project-logo + */ + public static defaultProject = ''; + + /** + * default-user-avatar icon + */ + public static defaultUser = ''; + + /** + * default "not found" image in case of 404 error + */ + public static defaultNotFound = ''; +} diff --git a/src/app/main/directive/admin-image/admin-image.directive.spec.ts b/src/app/main/directive/admin-image/admin-image.directive.spec.ts new file mode 100644 index 0000000000..9cbcf8c5a5 --- /dev/null +++ b/src/app/main/directive/admin-image/admin-image.directive.spec.ts @@ -0,0 +1,71 @@ +/* eslint-disable max-len */ +import { AdminImageDirective } from './admin-image.directive'; +import { Component, DebugElement } from '@angular/core'; +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { By } from '@angular/platform-browser'; + +@Component({ + template: ` + ` +}) +class TestAdminImageComponent { + + img = 'http://dasch.swiss/content/images/2017/11/DaSCH_Logo_RGB.png'; + type = 'project'; + + constructor() { } +} + +describe('Directive: AdminImageDirective', () => { + let component: TestAdminImageComponent; + let fixture: ComponentFixture; + let imageEl: DebugElement; + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [ + AdminImageDirective, + TestAdminImageComponent + ] + }); + + fixture = TestBed.createComponent(TestAdminImageComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + + imageEl = fixture.debugElement.query(By.css('img')); + + }); + + it('should create an instance', () => { + expect(component).toBeTruthy(); + }); + + it('should display the project logo of the DaSCH', () => { + expect(imageEl.nativeElement.src).toBe('http://dasch.swiss/content/images/2017/11/DaSCH_Logo_RGB.png'); + }); + + it('should display the default project logo if the image is null or undefined', () => { + component.img = null; + fixture.detectChanges(); + + expect(imageEl.nativeElement.src).toBe(''); + }); + + it('should detect the change of type to "user" and display the user logo', () => { + component.img = 'salsah@milchkannen.ch'; + component.type = 'user'; + fixture.detectChanges(); + + expect(imageEl.nativeElement.src).toBe('http://www.gravatar.com/avatar/dd74bbb106986f9ef074743e3c7fc555?d=mp&s=256'); + }); + + it('should display the default user logo if the image is null or undefined', () => { + component.img = undefined; + component.type = 'user'; + fixture.detectChanges(); + + expect(imageEl.nativeElement.src).toBe(''); + }); + +}); diff --git a/src/app/main/directive/admin-image/admin-image.directive.ts b/src/app/main/directive/admin-image/admin-image.directive.ts new file mode 100644 index 0000000000..41cbeb7179 --- /dev/null +++ b/src/app/main/directive/admin-image/admin-image.directive.ts @@ -0,0 +1,87 @@ +import { Directive, OnChanges, Input, Renderer2, ElementRef } from '@angular/core'; +import { AdminImageConfig } from './admin-image.config'; +import { Md5 } from 'ts-md5'; + +@Directive({ + selector: '[appAdminImage]' +}) +export class AdminImageDirective implements OnChanges { + + /** + * @param {string} image + * + * source of the image; + * - in the case of user (gr)avatar it's the e-mail address, + * - in the case of project logo it's the image url + */ + @Input() image: string; + + /** + * @param {string} type + * + * type of image; you can use it with + * - project + * - user + */ + @Input() type: string; + + + /** + * @ignore + */ + source: string; + + + /** + * @ignore + */ + onError: string = AdminImageConfig.defaultNotFound; + + + /** + * @ignore + */ + constructor(private _renderer: Renderer2, + private _ele: ElementRef + ) { } + + /** + * @ignore + */ + ngOnChanges() { + + this.source = this.image; + + switch (this.type) { + + case 'user': + this.onError = AdminImageConfig.defaultUser; + + if (this.image === null || this.image === undefined) { + this.source = AdminImageConfig.defaultUser; + } else { + this.source = location.protocol + '//www.gravatar.com/avatar/' + Md5.hashStr(this.image) + '?d=mp&s=256'; + } + + break; + + case 'project': + this.onError = AdminImageConfig.defaultProject; + + if (this.image === null || this.image === undefined) { + + this.source = AdminImageConfig.defaultProject; + } + + break; + + default: + this.source = this.image; + } + + this._renderer.setAttribute(this._ele.nativeElement, 'src', this.source); + this._renderer.setAttribute(this._ele.nativeElement, 'onError', 'this.src=\'' + this.onError + '\''); + + } + +} diff --git a/src/app/main/directive/existing-name/existing-name.directive.spec.ts b/src/app/main/directive/existing-name/existing-name.directive.spec.ts new file mode 100644 index 0000000000..5dc660caf5 --- /dev/null +++ b/src/app/main/directive/existing-name/existing-name.directive.spec.ts @@ -0,0 +1,216 @@ + +import { Component, OnInit } from '@angular/core'; +import { FormGroup, FormBuilder, FormControl, Validators, ReactiveFormsModule, FormsModule } from '@angular/forms'; +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { MatFormFieldModule } from '@angular/material/form-field'; +import { MatInputModule } from '@angular/material/input'; +import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; +import { ExistingNameDirective, existingNamesValidator } from './existing-name.directive'; + +@Component({ + template: ` +
+
+ + + + {{formErrors.name}} + + + + +
+
+ +
    +
  • {{n}}
  • +
+ ` +}) +class TestHostComponent implements OnInit { + + dataMock: string[] = [ + 'Ben', 'Tobias', 'André', 'Flavie', 'Ivan', 'Lucas', 'Mike' + ]; + + existingNames: [RegExp] = [ + new RegExp('user') + ]; + + form: FormGroup; + + formErrors = { + name: '' + }; + + validationMessages = { + name: { + required: 'A name is required', + existingName: 'This name exists already.' + } + }; + + constructor(private _formBuilder: FormBuilder) { } + + ngOnInit() { + // create a list of existing names + let i = 1; + for (const user of this.dataMock) { + this.existingNames[i] = new RegExp('(?:^|W)' + user.toLowerCase() + '(?:$|W)'); + + i++; + } + + // build form + this.form = this._formBuilder.group({ + name: new FormControl({ + value: '', disabled: false + }, [ + Validators.required, + existingNamesValidator(this.existingNames) + ]) + }); + + // detect changes in the form + this.form.valueChanges.subscribe( + data => this.onValueChanged(data) + ); + + this.onValueChanged(); + } + + onValueChanged(data?: any) { + + if (!this.form) { + return; + } + + // check if the form is valid + Object.keys(this.formErrors).map(field => { + this.formErrors[field] = ''; + const control = this.form.get(field); + if (control && control.dirty && !control.valid) { + const messages = this.validationMessages[field]; + Object.keys(control.errors).map(key => { + this.formErrors[field] += messages[key] + ' '; + }); + } + }); + } +} + +describe('ExistingNameDirective', () => { + + let component: TestHostComponent; + let fixture: ComponentFixture; + const existingNamesList: string[] = [ + 'Ben', 'Tobias', 'André', 'Flavie', 'Ivan', 'Lucas', 'Mike' + ]; + const existingNames: [RegExp] = [ + new RegExp('user') + ]; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [ + ReactiveFormsModule, FormsModule, + MatFormFieldModule, MatInputModule, BrowserAnimationsModule + ], + declarations: [ + ExistingNameDirective, + TestHostComponent + ] + }); + + fixture = TestBed.createComponent(TestHostComponent); + component = fixture.componentInstance; + component.ngOnInit(); + }); + + it('should create an instance', () => { + expect(component).toBeTruthy(); + }); + + it('form invalid when empty', () => { + expect(component.form.valid).toBeFalsy(); + }); + + it('name input field validity', () => { + const name = component.form.controls.name; + expect(name.valid).toBeFalsy(); + }); + + it('should recognize the new name "Benjamin" and validate the form', () => { + expect(component.dataMock).toEqual(existingNamesList); + expect(component.form.valid).toBeFalsy(); + + let i = 1; + for (const user of existingNamesList) { + existingNames[i] = new RegExp('(?:^|W)' + user.toLowerCase() + '(?:$|W)'); + + i++; + } + + expect(component.existingNames).toEqual(existingNames); + + fixture.detectChanges(); + + const name = component.form.controls.name; + + let errors = {}; + errors = name.errors || {}; + expect(name.valid).toBeFalsy(); + + // name field is required + expect(errors['required']).toBeTruthy(); + expect(existingNamesValidator(existingNames)); + + // set a new name + name.setValue('Benjamin'); + fixture.detectChanges(); + + errors = name.errors || {}; + + expect(component.form.valid).toBeTruthy(); + expect(errors['required']).toBeFalsy(); + expect(existingNamesValidator(existingNames)).toBeTruthy(); + expect(component.form.controls.name.errors).toEqual(null); + }); + + it('should recognize the existing name "Ben" and invalid the form', () => { + fixture.detectChanges(); + expect(component.dataMock).toEqual(existingNamesList); + expect(component.form.valid).toBeFalsy(); + + let i = 1; + for (const user of existingNamesList) { + existingNames[i] = new RegExp('(?:^|W)' + user.toLowerCase() + '(?:$|W)'); + + i++; + } + + expect(component.existingNames).toEqual(existingNames); + + let errors = {}; + const name = component.form.controls.name; + expect(name.valid).toBeFalsy(); + + // name field is required + errors = name.errors || {}; + expect(errors['required']).toBeTruthy(); + expect(existingNamesValidator(existingNames)); + + // set an existing name + name.setValue('Ben'); + fixture.detectChanges(); + + errors = name.errors || {}; + expect(component.form.valid).toBeFalsy(); + expect(errors['required']).toBeFalsy(); + expect(existingNamesValidator(existingNames)).toBeTruthy(); + expect(component.form.controls.name.errors.existingName.name).toEqual('ben'); + }); + +}); diff --git a/src/app/main/directive/existing-name/existing-name.directive.ts b/src/app/main/directive/existing-name/existing-name.directive.ts new file mode 100644 index 0000000000..20eef3cc4b --- /dev/null +++ b/src/app/main/directive/existing-name/existing-name.directive.ts @@ -0,0 +1,112 @@ +/* eslint-disable @typescript-eslint/no-use-before-define */ +import { Directive, OnChanges, Input, SimpleChanges } from '@angular/core'; +import { Validators, AbstractControl, ValidatorFn } from '@angular/forms'; + +@Directive({ + selector: '[appExistingNames]' +}) +export class ExistingNameDirective implements Validators, OnChanges { + + /** + * @ignore + */ + @Input() existingName: string; + + /** + * @ignore + */ + private _valFn = Validators.nullValidator; + + /** + * @ignore + * @param changes + */ + ngOnChanges(changes: SimpleChanges): void { + const change = changes['existingName']; + if (change) { + const val: string | RegExp = change.currentValue; + const re = val instanceof RegExp ? val : new RegExp(val); + this._valFn = existingNameValidator(re); + } else { + this._valFn = Validators.nullValidator; + } + } + + /** + * @ignore + * @param control + */ + validate(control: AbstractControl): { [key: string]: any } { + return this._valFn(control); + } +} + +/** + * validation of existing name value. String method (only one value); + * Use it in a "formbuilder" group as a validator property + * + * @param {RegExp} valRegexp Only one regular expression value + * @returns ValidatorFn + */ +export function existingNameValidator(valRegexp: RegExp): ValidatorFn { + return (control: AbstractControl): { [key: string]: any } => { + let name; + + if (control.value) { + name = control.value.toLowerCase(); + } + + const no = valRegexp.test(name); + return no ? { 'existingName': { name } } : null; + }; +} + +/** + * validation of existing name values. Array method (list of values) + * Use it in a "formbuilder" group as a validator property + * + * @param {RegExp} valArrayRegexp List of regular expression values + * @returns ValidatorFn + */ +export function existingNamesValidator(valArrayRegexp: [RegExp]): ValidatorFn { + + return (control: AbstractControl): { [key: string]: any } => { + + let name; + + if (control.value) { + name = control.value.toLowerCase(); + } + + let no; + for (const existing of valArrayRegexp) { + no = existing.test(name); + if (no) { + // console.log(no); + return no ? { 'existingName': { name } } : null; + } + } + return no ? { 'existingName': { name } } : null; + }; +} + +/** + * @ignore + * + * @param {RegExp} pattern + * @param {string} regType + * @returns ValidatorFn + */ +export function notAllowed(pattern: RegExp, regType: string): ValidatorFn { + return (control: AbstractControl): { [key: string]: any } => { + let name; + + if (control.value) { + name = control.value.toLowerCase(); + } + + const no = pattern.test(name); + return no ? { regType: { name } } : null; + }; +} + diff --git a/src/app/main/directive/gnd/gnd.directive.spec.ts b/src/app/main/directive/gnd/gnd.directive.spec.ts new file mode 100644 index 0000000000..f04e2e849e --- /dev/null +++ b/src/app/main/directive/gnd/gnd.directive.spec.ts @@ -0,0 +1,145 @@ +import { Component, DebugElement, ElementRef, OnInit } from '@angular/core'; +import { waitForAsync, ComponentFixture, TestBed } from '@angular/core/testing'; +import { By } from '@angular/platform-browser'; +import { GndDirective } from './gnd.directive'; + +/** + * test component for a GND/IAF identifier. + */ +@Component({ + template: '' +}) +class TestGnd1Component { } + +/** + * test component for a VIAF identifier. + */ +@Component({ + template: '' +}) +class TestGnd2Component { } + +/** + * test component for normal text. + */ +@Component({ + template: '' +}) +class TestGnd3Component { } + +/** + * test component for long normal text. + */ +@Component({ + template: `` +}) +class TestGnd4Component { } + +/** + * test component with an updated text. + */ +@Component({ + template: '' +}) +class TestGnd5Component implements OnInit { + + gndValue; + + ngOnInit() { + this.gndValue = 'initial text'; + } +} + +class MockElementRef implements ElementRef { + nativeElement = {}; +} + + +describe('GndDirective', () => { + + beforeEach(waitForAsync(() => { + TestBed.configureTestingModule({ + declarations: [TestGnd1Component, TestGnd2Component, TestGnd3Component, TestGnd4Component, TestGnd5Component, GndDirective], + providers: [ + { provide: ElementRef, useClass: MockElementRef } + ] + }).compileComponents(); + + })); + + it('should create an instance', () => { + + const directive = new GndDirective(new MockElementRef()); + expect(directive).toBeTruthy(); + + }); + + it('should transform a GND/IAF identifier to a hyperlink pointing to the resolver', () => { + + const fixture: ComponentFixture = TestBed.createComponent(TestGnd1Component); + fixture.detectChanges(); + + const spanEle: DebugElement = fixture.debugElement.query(By.css('span')); + + expect(spanEle.nativeElement.innerHTML).toEqual('(DE-588)118696149'); + + }); + + it('should transform a VIAF identifier to a hyperlink pointing to the resolver', () => { + + const fixture: ComponentFixture = TestBed.createComponent(TestGnd2Component); + fixture.detectChanges(); + + const spanEle: DebugElement = fixture.debugElement.query(By.css('span')); + + expect(spanEle.nativeElement.innerHTML).toEqual('(VIAF)22936072'); + + }); + + it('should not transform normal text', () => { + + const fixture: ComponentFixture = TestBed.createComponent(TestGnd3Component); + fixture.detectChanges(); + + const spanEle: DebugElement = fixture.debugElement.query(By.css('span')); + + expect(spanEle.nativeElement.innerHTML).toEqual('normal text'); + + }); + + it('should not transform long normal text', () => { + + const fixture: ComponentFixture = TestBed.createComponent(TestGnd4Component); + fixture.detectChanges(); + + const spanEle: DebugElement = fixture.debugElement.query(By.css('span')); + + expect(spanEle.nativeElement.innerHTML).toEqual('normal text that is quite long and will not even be looked at because it cannot possibly be a GND/IAF or VIAF identifier'); + + }); + + it('should be equal to the updated text', () => { + + const fixture: ComponentFixture = TestBed.createComponent(TestGnd5Component); + fixture.detectChanges(); + + const spanEle: DebugElement = fixture.debugElement.query(By.css('span')); + + expect(spanEle.nativeElement.innerHTML).toEqual('initial text'); + + + fixture.componentInstance.gndValue = 'updated text'; + fixture.detectChanges(); + + expect(spanEle.nativeElement.innerHTML).toEqual('updated text'); + + }); + + +}); + + + + + diff --git a/src/app/main/directive/gnd/gnd.directive.ts b/src/app/main/directive/gnd/gnd.directive.ts new file mode 100644 index 0000000000..7f4194f941 --- /dev/null +++ b/src/app/main/directive/gnd/gnd.directive.ts @@ -0,0 +1,57 @@ +import { Directive, ElementRef, Input, OnChanges } from '@angular/core'; + +export class GNDConstants { + public static GND_PREFIX = '(DE-588)'; + public static GND_RESOLVER = 'http://d-nb.info/gnd/'; + + public static VIAF_PREFIX = '(VIAF)'; + public static VIAF_RESOLVER = 'https://viaf.org/viaf/'; +} + +/** + * this directive renders a GND/IAF or a VIAF identifier as a link to the respective resolver. + */ +@Directive({ + selector: '[appGnd]' +}) +export class GndDirective implements OnChanges { + + @Input() + set gnd(value: string) { + this._gnd = value; + } + + get gnd() { + return this._gnd; + } + + + // the GND identifier to be rendered + private _gnd: string; + + constructor(private _ele: ElementRef) { + + } + + ngOnChanges() { + if (this._gnd.length < 30) { + + if (this._gnd.indexOf(GNDConstants.GND_PREFIX) === 0) { + // gnd/iaf identifier + this._ele.nativeElement.innerHTML = `${this._gnd}`; + } else if (this._gnd.indexOf(GNDConstants.VIAF_PREFIX) === 0) { + // viaf identifier + this._ele.nativeElement.innerHTML = `${this._gnd}`; + } else { + // no identifier, leave unchanged + this._ele.nativeElement.innerHTML = this._gnd; + } + + } else { + // no identifier, leave unchanged + this._ele.nativeElement.innerHTML = this._gnd; + } + + } + +} diff --git a/src/app/main/pipes/formatting/formatted-boolean.pipe.spec.ts b/src/app/main/pipes/formatting/formatted-boolean.pipe.spec.ts new file mode 100644 index 0000000000..862c0bb6e7 --- /dev/null +++ b/src/app/main/pipes/formatting/formatted-boolean.pipe.spec.ts @@ -0,0 +1,45 @@ +import { FormattedBooleanPipe } from './formatted-boolean.pipe'; + +describe('FormattedBooleanPipe', () => { + let pipe: FormattedBooleanPipe; + + beforeEach(() => { + pipe = new FormattedBooleanPipe(); + }); + + it('create an instance', () => { + expect(pipe).toBeTruthy(); + }); + + it('should return "true" (no format specified)', () => { + const myBoolean = true; + + const convertedBoolean = pipe.transform(myBoolean); + + expect(convertedBoolean).toEqual('true'); + }); + + it('should return "false" (format specified)', () => { + const myBoolean = false; + + const convertedBoolean = pipe.transform(myBoolean, 'true-false'); + + expect(convertedBoolean).toEqual('false'); + }); + + it('should return "yes"', () => { + const myBoolean = true; + + const convertedBoolean = pipe.transform(myBoolean, 'yes-no'); + + expect(convertedBoolean).toEqual('yes'); + }); + + it('should return "off"', () => { + const myBoolean = false; + + const convertedBoolean = pipe.transform(myBoolean, 'on-off'); + + expect(convertedBoolean).toEqual('off'); + }); +}); diff --git a/src/app/main/pipes/formatting/formatted-boolean.pipe.ts b/src/app/main/pipes/formatting/formatted-boolean.pipe.ts new file mode 100644 index 0000000000..0c48c9b1e5 --- /dev/null +++ b/src/app/main/pipes/formatting/formatted-boolean.pipe.ts @@ -0,0 +1,32 @@ +import { Pipe, PipeTransform } from '@angular/core'; + +interface DisplayTypes { + value: string; + viewValue: string; +} + +@Pipe({ + name: 'formattedBoolean' +}) +export class FormattedBooleanPipe implements PipeTransform { + + displayTypes: DisplayTypes[] = [ + { value: 'true-false', viewValue: 'True/False' }, + { value: 'yes-no', viewValue: 'Yes/No' }, + { value: 'on-off', viewValue: 'On/Off' } + ]; + + transform(value: boolean, format?: string): string { + switch (format) { + case 'true-false': + return value ? 'true' : 'false'; + case 'yes-no': + return value ? 'yes' : 'no'; + case 'on-off': + return value ? 'on' : 'off'; + default: + return value ? 'true' : 'false'; + } + } + +} diff --git a/src/app/main/pipes/formatting/knoradate.pipe.spec.ts b/src/app/main/pipes/formatting/knoradate.pipe.spec.ts new file mode 100644 index 0000000000..4fafa0eb7e --- /dev/null +++ b/src/app/main/pipes/formatting/knoradate.pipe.spec.ts @@ -0,0 +1,98 @@ +import { KnoraDate } from '@dasch-swiss/dsp-js'; +import { KnoraDatePipe } from './knoradate.pipe'; + +describe('KnoradatePipe', () => { + let pipe: KnoraDatePipe; + + beforeEach(() => { + pipe = new KnoraDatePipe(); + expect(pipe).toBeTruthy(); + }); + + it('should return a date string', () => { + const date = new KnoraDate('GREGORIAN', 'AD', 1993, 10, 10); + + const convertedDate = pipe.transform(date); + + expect(convertedDate).toEqual('10.10.1993'); + }); + + it('should return the correct format for a date with day precision depending on the format provided', () => { + const date = new KnoraDate('GREGORIAN', 'AD', 1776, 7, 4); + + let convertedDate = pipe.transform(date, 'dd.MM.YYYY'); + + expect(convertedDate).toEqual('04.07.1776'); + + convertedDate = pipe.transform(date, 'dd-MM-YYYY'); + + expect(convertedDate).toEqual('04-07-1776'); + + convertedDate = pipe.transform(date, 'MM/dd/YYYY'); + + expect(convertedDate).toEqual('07/04/1776'); + + // should default to dd.MM.YYYY in the event of an invalid format + convertedDate = pipe.transform(date, 'invalid format'); + + expect(convertedDate).toEqual('04.07.1776'); + }); + + it ('should return a string with the desired display options', () => { + const date = new KnoraDate('GREGORIAN', 'AD', 1776, 7, 4); + + let dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'era'); + + expect(dateWithDisplayOptions).toEqual('04.07.1776 AD'); + + dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'calendar'); + + expect(dateWithDisplayOptions).toEqual('04.07.1776 GREGORIAN'); + + dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'all'); + + expect(dateWithDisplayOptions).toEqual('04.07.1776 AD GREGORIAN'); + }); + + it ('should return a string with the desired display options for a date without era', () => { + const date = new KnoraDate('ISLAMIC', 'noEra', 1441, 7, 4); + + let dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'era'); + + expect(dateWithDisplayOptions).toEqual('04.07.1441'); + + dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'calendar'); + + expect(dateWithDisplayOptions).toEqual('04.07.1441 ISLAMIC'); + + dateWithDisplayOptions = pipe.transform(date, 'dd.MM.YYYY', 'all'); + + expect(dateWithDisplayOptions).toEqual('04.07.1441 ISLAMIC'); + }); + + it ('should return a string with only the month and the year', () => { + const date = new KnoraDate('GREGORIAN', 'AD', 1776, 7); + + const convertedDate = pipe.transform(date, 'dd.MM.YYYY'); + + expect(convertedDate).toEqual('07.1776'); + }); + + it ('should return a string with only the year', () => { + const date = new KnoraDate('GREGORIAN', 'AD', 1776); + + const convertedDate = pipe.transform(date, 'dd.MM.YYYY'); + + expect(convertedDate).toEqual('1776'); + }); + + it('should return a number of two digits', () => { + let num = pipe.leftPadding(7); + + expect(num).toEqual('07'); + + num = pipe.leftPadding(12); + + expect(num).toEqual('12'); + }); +}); diff --git a/src/app/main/pipes/formatting/knoradate.pipe.ts b/src/app/main/pipes/formatting/knoradate.pipe.ts new file mode 100644 index 0000000000..ac751d81f8 --- /dev/null +++ b/src/app/main/pipes/formatting/knoradate.pipe.ts @@ -0,0 +1,82 @@ +import { Pipe, PipeTransform } from '@angular/core'; +import { KnoraDate } from '@dasch-swiss/dsp-js'; + +@Pipe({ + name: 'knoraDate' +}) +export class KnoraDatePipe implements PipeTransform { + + transform(date: KnoraDate, format?: string, displayOptions?: 'era' | 'calendar' | 'all'): string { + if (!(date instanceof KnoraDate)) { + console.error('Non-KnoraDate provided. Expected a valid KnoraDate'); + return ''; + } + + const formattedString = this.getFormattedString(date, format); + + if (displayOptions) { + return this.addDisplayOptions(date, formattedString, displayOptions); + } else { + return formattedString; + } + } + + // ensures that day and month are always two digits + leftPadding(value: number): string { + if (value !== undefined) { + return ('0' + value).slice(-2); + } else { + return null; + } + } + + // add the era, calendar, or both to the result returned by the pipe + addDisplayOptions(date: KnoraDate, value: string, options: string): string { + switch (options) { + case 'era': + return value + (date.era !== 'noEra' ? ' ' + date.era : ''); + case 'calendar': + return value + ' ' + date.calendar; + case 'all': + return value + (date.era !== 'noEra' ? ' ' + date.era : '') + ' ' + date.calendar; + } + } + + getFormattedString(date: KnoraDate, format: string): string { + switch (format) { + case 'dd.MM.YYYY': + if (date.precision === 2) { + return `${this.leftPadding(date.day)}.${this.leftPadding(date.month)}.${date.year}`; + } else if (date.precision === 1) { + return `${this.leftPadding(date.month)}.${date.year}`; + } else { + return `${date.year}`; + } + case 'dd-MM-YYYY': + if (date.precision === 2) { + return `${this.leftPadding(date.day)}-${this.leftPadding(date.month)}-${date.year}`; + } else if (date.precision === 1) { + return `${this.leftPadding(date.month)}-${date.year}`; + } else { + return `${date.year}`; + } + case 'MM/dd/YYYY': + if (date.precision === 2) { + return `${this.leftPadding(date.month)}/${this.leftPadding(date.day)}/${date.year}`; + } else if (date.precision === 1) { + return `${this.leftPadding(date.month)}/${date.year}`; + } else { + return `${date.year}`; + } + default: + if (date.precision === 2) { + return `${this.leftPadding(date.day)}.${this.leftPadding(date.month)}.${date.year}`; + } else if (date.precision === 1) { + return `${this.leftPadding(date.month)}.${date.year}`; + } else { + return `${date.year}`; + } + } + } + +} diff --git a/src/app/main/pipes/string-transformation/linkify.pipe.spec.ts b/src/app/main/pipes/string-transformation/linkify.pipe.spec.ts new file mode 100644 index 0000000000..dca2367e44 --- /dev/null +++ b/src/app/main/pipes/string-transformation/linkify.pipe.spec.ts @@ -0,0 +1,39 @@ +/* eslint-disable max-len */ +import { LinkifyPipe } from './linkify.pipe'; + +describe('LinkifyPipe', () => { + + let pipe: LinkifyPipe; + + beforeEach(() => { + pipe = new LinkifyPipe(); + }); + + it('create an instance', () => { + expect(pipe).toBeTruthy(); + }); + + it('should recognize the url without protocol but with hashtag at the end', () => { + const text = 'You can visit the app on app.dasch.swiss/#'; + const linkifiedSnippet = pipe.transform(text); + expect(linkifiedSnippet).toEqual('You can visit the app on app.dasch.swiss/#'); + }); + + it('should recognize the url with protocol followed by full stop', () => { + const text = 'You can visit the app on https://app.dasch.swiss.'; + const linkifiedSnippet = pipe.transform(text); + expect(linkifiedSnippet).toEqual('You can visit the app on https://app.dasch.swiss.'); + }); + + it('should recognize both urls in the example text', () => { + const text = 'You can visit the app on https://app.dasch.swiss and the documentation on docs.dasch.swiss.'; + const linkifiedSnippet = pipe.transform(text); + expect(linkifiedSnippet).toEqual('You can visit the app on https://app.dasch.swiss and the documentation on docs.dasch.swiss.'); + }); + + it('should keep the spaces after a full stop or after a comma', () => { + const text = 'This is just a title. And it could have an URL, but it doesn\'t have one. '; + const linkifiedSnippet = pipe.transform(text); + expect(linkifiedSnippet).toEqual('This is just a title. And it could have an URL, but it doesn\'t have one.'); + }); +}); diff --git a/src/app/main/pipes/string-transformation/linkify.pipe.ts b/src/app/main/pipes/string-transformation/linkify.pipe.ts new file mode 100644 index 0000000000..b27412e783 --- /dev/null +++ b/src/app/main/pipes/string-transformation/linkify.pipe.ts @@ -0,0 +1,57 @@ +import { Pipe, PipeTransform } from '@angular/core'; + +/** + * this pipe analyses a string and converts any url into a href tag + * + */ +@Pipe({ + name: 'appLinkify' +}) +export class LinkifyPipe implements PipeTransform { + + transform(value: string): string { + let stylizedText = ''; + if (value && value.length > 0) { + for (let str of value.split(' ')) { + // if string/url ends with a full stop '.' or colon ':' or comma ',' or semicolon ';' the pipe will not recognize the url + const lastChar = str.substring(str.length - 1); + const endsWithFullStop = (lastChar === '.' || lastChar === ':' || lastChar === ',' || lastChar === ';'); + let end = ' '; + if (endsWithFullStop) { + str = str.slice(0, -1); + end = lastChar + ' '; + } + if (this._recognizeUrl(str)) { + const url = this._setProtocol(str); + stylizedText += `${str}${end}`; + } else { + stylizedText += str + end; + } + } + return stylizedText.trim(); + } else { + return value; + } + } + + private _recognizeUrl(str: string): boolean { + const pattern = new RegExp( + '^(https?:\\/\\/)?' + // protocol + '((([a-z\\d]([a-z\\d-]*[a-z\\d])*)\\.)+[a-z]{2,}|' + // domain name + '((\\d{1,3}\\.){3}\\d{1,3}))' + // oR ip (v4) address + '(\\:\\d+)?(\\/[-a-z\\d%_.~+]*)*' + // port and path + '(\\?[;&a-z\\d%_.~+=-]*)?' + // query string + '(\\#[-a-z\\d_]*)?$', + 'i' + ); // fragment locator + return pattern.test(str); + } + + private _setProtocol(url: string): string { + if (!/^(?:f|ht)tps?\:\/\//.test(url)) { + url = 'http://' + url; + } + return url; + } + +} diff --git a/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.spec.ts b/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.spec.ts new file mode 100644 index 0000000000..514f191a3c --- /dev/null +++ b/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.spec.ts @@ -0,0 +1,131 @@ +import { waitForAsync, TestBed } from '@angular/core/testing'; +import { StringLiteral } from '@dasch-swiss/dsp-js'; +import { DspApiConnectionToken } from '../../declarations/dsp-api-tokens'; +import { Session, SessionService } from '../../services/session.service'; +import { StringifyStringLiteralPipe } from './stringify-string-literal.pipe'; + +describe('StringifyStringLiteralPipe', () => { + let pipe: StringifyStringLiteralPipe; + let labels: StringLiteral[]; + let service: SessionService; + + beforeEach(waitForAsync(() => { + + // empty spy object to use in the providers for the SessionService injection + const dspConnSpy = { }; + + TestBed.configureTestingModule({ + providers: [ + { + provide: DspApiConnectionToken, + useValue: dspConnSpy + }, + SessionService + ] + }); + + service = TestBed.inject(SessionService); + pipe = new StringifyStringLiteralPipe(service); + })); + + beforeEach(() => { + + labels = [ + { + value: 'Welt', + language: 'de' + }, + { + value: 'World', + language: 'en' + }, + { + value: 'Monde', + language: 'fr' + }, + { + value: 'Mondo', + language: 'it' + }, + ]; + }); + + // mock localStorage + beforeEach(() => { + let store = {}; + + spyOn(localStorage, 'getItem').and.callFake( + (key: string): string => store[key] || null + ); + spyOn(localStorage, 'removeItem').and.callFake( + (key: string): void => { + delete store[key]; + } + ); + spyOn(localStorage, 'setItem').and.callFake( + (key: string, value: string): void => { + store[key] = value; + } + ); + spyOn(localStorage, 'clear').and.callFake(() => { + store = {}; + }); + }); + + it('create an instance', () => { + expect(pipe).toBeTruthy(); + }); + + it('should return a string in English', () => { + const session: Session = { + id: 12345, + user: { + name: 'username', + jwt: 'myToken', + lang: 'en', + sysAdmin: false, + projectAdmin: [] + } + }; + + // store session in localStorage + localStorage.setItem('session', JSON.stringify(session)); + + // since no argument is provided, the pipe should use the language stored in the session + const myString = pipe.transform(labels); + expect(myString).toEqual('World'); + + // remove session + localStorage.removeItem('session'); + }); + + it('should return a string in German', () => { + const session: Session = { + id: 12345, + user: { + name: 'username', + jwt: 'myToken', + lang: 'de', + sysAdmin: false, + projectAdmin: [] + } + }; + + // store session in localStorage + localStorage.setItem('session', JSON.stringify(session)); + + // since no argument is provided, the pipe should use the language stored in the session + const myString = pipe.transform(labels); + expect(myString).toEqual('Welt'); + + // remove session + localStorage.removeItem('session'); + }); + + it('should return a string with all languages of which the StringLiteral array contains', () => { + + // since no argument is provided, the pipe should use the language stored in the session + const myString = pipe.transform(labels, 'all'); + expect(myString).toEqual('Welt (de) / World (en) / Monde (fr) / Mondo (it)'); + }); +}); diff --git a/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.ts b/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.ts new file mode 100644 index 0000000000..e983a938af --- /dev/null +++ b/src/app/main/pipes/string-transformation/stringify-string-literal.pipe.ts @@ -0,0 +1,64 @@ +import { Pipe, PipeTransform } from '@angular/core'; +import { StringLiteral } from '@dasch-swiss/dsp-js'; +import { SessionService } from '../../services/session.service'; + +/** + * this pipe stringifies an array of StringLiterals. + * With the parameter 'all', the pipe concats all values and appends the corresponding language in brackets. + * + * Otherwise the pipe displays the value corresponding to the default language which + * comes from the user profile if a user is logged in or from the browser if the user is not logged in. + * + * With the predefined language, the pipe checks if a value exists in the array, otherwise it shows the first value. + */ +@Pipe({ + name: 'appStringifyStringLiteral' +}) +export class StringifyStringLiteralPipe implements PipeTransform { + + constructor(private _sessionService: SessionService) { } + + transform(value: StringLiteral[], args?: string): string { + let stringified = ''; + + let language: string; + + if (!value || !value.length) { + return; + } + + if (args === 'all') { + // show all values + let i = 0; + for (const sl of value) { + const delimiter = (i > 0 ? ' / ' : ''); + stringified += delimiter + sl.value + ' (' + sl.language + ')'; + + i++; + } + return stringified; + } else { + // show only one value, depending on default language + // the language is defined in user profile if a user is logged-in + // otherwise it takes the language from browser + if (this._sessionService.getSession() !== null) { + // get language from the logged-in user profile data + language = this._sessionService.getSession().user.lang; + } else { + // get default language from browser + language = navigator.language.substr(0, 2); + } + // does the defined language exists and does it have a value? + const index = value.findIndex(i => i.language === language); + + if (value[index] && value[index].value.length > 0) { + return value[index].value; + } else { + return value[0].value; + } + + } + + } + +} diff --git a/src/app/main/pipes/string-transformation/truncate.pipe.spec.ts b/src/app/main/pipes/string-transformation/truncate.pipe.spec.ts new file mode 100644 index 0000000000..9c39bed62a --- /dev/null +++ b/src/app/main/pipes/string-transformation/truncate.pipe.spec.ts @@ -0,0 +1,62 @@ +import { TruncatePipe } from './truncate.pipe'; +import { Component } from '@angular/core'; +import { TestBed } from '@angular/core/testing'; + +describe('TruncatePipe', () => { + let pipe: TruncatePipe; + let snippet: string; + + beforeEach(() => { + pipe = new TruncatePipe(); + snippet = 'The quick brown fox jumps over the lazy dog.'; + }); + + it('create an instance', () => { + expect(pipe).toBeTruthy(); + }); + + it('should truncate after 15 characters', () => { + const truncatedSnippet = pipe.transform(snippet, 15); + expect(truncatedSnippet).toEqual('The quick brown...'); + }); + + it('should truncate after 19 characters and add an exclamation mark at the end', () => { + const truncatedSnippet = pipe.transform(snippet, 19, '!'); + expect(truncatedSnippet).toEqual('The quick brown fox!'); + }); + + it('should truncate after 20 characters by default', () => { + const truncatedSnippet = pipe.transform(snippet); + expect(truncatedSnippet).toEqual('The quick brown fox ...'); + }); + + it('should support the limit argument in the template', () => { + @Component({ + template: '{{ text | appTruncate:4 }}', + }) + class AppComponent { + text = 'This is my string'; + } + + TestBed.configureTestingModule({ declarations: [AppComponent, TruncatePipe] }); + const fixture = TestBed.createComponent(AppComponent); + fixture.detectChanges(); + + expect(fixture.nativeElement.textContent).toEqual('This...'); + }); + + it('should support the trail argument in the template', () => { + @Component({ + template: '{{ text | appTruncate:7:\'!!\' }}', + }) + class AppComponent { + text = 'This is my string'; + } + + TestBed.configureTestingModule({ declarations: [AppComponent, TruncatePipe] }); + const fixture = TestBed.createComponent(AppComponent); + fixture.detectChanges(); + + expect(fixture.nativeElement.textContent).toEqual('This is!!'); + }); +}); diff --git a/src/app/main/pipes/string-transformation/truncate.pipe.ts b/src/app/main/pipes/string-transformation/truncate.pipe.ts new file mode 100644 index 0000000000..4118914ab6 --- /dev/null +++ b/src/app/main/pipes/string-transformation/truncate.pipe.ts @@ -0,0 +1,45 @@ +import { Pipe, PipeTransform } from '@angular/core'; + +/** + * this pipe can be used to shorten long text by a defined length. + * + * In markup: + * + * {{ str | appTruncate:24 }} + * + * or + * + * {{ str | appTruncate:24:'...' }} + * + * The first optional parameter defines the length where to truncate the string. + * The second optional parameter defines the characters to append to the shortened string. Default is '...'. + * + * The advantage of this pipe over the default Angular slice pipe is the simplicity of adding + * additional characters at the end of the shortened string. + * The same construct with Angular slice pipe looks as follow: `{{ (str.length>24)? (str | slice:0:24)+'...':(str) }}`. + * + */ +@Pipe({ + name: 'appTruncate' +}) +export class TruncatePipe implements PipeTransform { + + defaultLimit = 20; + defaultTrail = '...'; + + transform(value: string, limit?: number, trail?: string): string { + + if (typeof value !== 'string' || value.length === 0) { + return ''; + } + + // if a custom limit was provided, use that. Otherwise, use the default limit of 20 characters + const limitSetting: number = limit !== undefined ? limit : this.defaultLimit; + + // if a custom trail was provided, use that. Otherwise, use the default trail of '...' + const trailSetting: string = trail !== undefined ? trail : this.defaultTrail; + + return value.length > limitSetting ? value.substring(0, limitSetting) + trailSetting : value; + } + +} diff --git a/src/app/main/services/notification.service.spec.ts b/src/app/main/services/notification.service.spec.ts new file mode 100644 index 0000000000..ea2b7b5699 --- /dev/null +++ b/src/app/main/services/notification.service.spec.ts @@ -0,0 +1,29 @@ +import { TestBed } from '@angular/core/testing'; +import { NotificationService } from './notification.service'; + +describe('NotificationService', () => { + let service: NotificationService; + const mockNotificationService = jasmine.createSpyObj('NotificationService', ['openSnackBar']); + + beforeEach(() => { + TestBed.configureTestingModule({ + providers: [ + { provide: NotificationService, useValue: mockNotificationService } + ] + }); + service = TestBed.inject(NotificationService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); + + describe('openSnackBar', () => { + it('should open the snack bar', () => { + const arg = 'test'; + mockNotificationService.openSnackBar.and.callThrough(); + service.openSnackBar(arg); + expect(service.openSnackBar).toHaveBeenCalled(); + }); + }); +}); diff --git a/src/app/main/services/notification.service.ts b/src/app/main/services/notification.service.ts new file mode 100644 index 0000000000..c361b653dc --- /dev/null +++ b/src/app/main/services/notification.service.ts @@ -0,0 +1,41 @@ +import { Injectable } from '@angular/core'; +import { MatSnackBar } from '@angular/material/snack-bar'; +import { ApiResponseError } from '@dasch-swiss/dsp-js'; +import { StatusMsg } from 'src/assets/http/statusMsg'; + +@Injectable({ + providedIn: 'root' +}) +export class NotificationService { + + constructor( + private _snackBar: MatSnackBar, + private _statusMsg: StatusMsg + ) { } + + // todo: maybe we can add more parameters like: + // action: string = 'x', duration: number = 4200 + // and / or type: 'note' | 'warning' | 'error' | 'success'; which can be used for the panelClass + openSnackBar(notification: string | ApiResponseError): void { + const duration = 5000; + let message: string; + let panelClass: string; + + if (notification instanceof ApiResponseError) { + const status = (notification.status === 0 ? 503 : notification.status); + const defaultStatusMsg = this._statusMsg.default; + message = `${defaultStatusMsg[status].message} (${status}): ${defaultStatusMsg[status].description}`; + panelClass = 'error'; + } else { + message = notification; + panelClass = 'success'; + } + + this._snackBar.open(message, 'x', { + duration, + horizontalPosition: 'center', + verticalPosition: 'top', + panelClass + }); + } +} diff --git a/src/app/main/services/sorting.service.spec.ts b/src/app/main/services/sorting.service.spec.ts new file mode 100644 index 0000000000..6b3c6be3f9 --- /dev/null +++ b/src/app/main/services/sorting.service.spec.ts @@ -0,0 +1,165 @@ +import { TestBed } from '@angular/core/testing'; + +import { SortingService } from './sorting.service'; +import { ReadProject } from '@dasch-swiss/dsp-js'; + +describe('SortingService', () => { + let service: SortingService; + + beforeEach(() => { + TestBed.configureTestingModule({ + providers: [ + SortingService + ] + }); + + service = TestBed.inject(SortingService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); + + describe('reverseArray', () => { + + let data: string[]; + + beforeEach(() => { + data = ['Bernouilli', 'Euler', 'Goldbach', 'Hermann']; + }); + + it('should reverse an array', () => { + expect(service.reverseArray(data)).toEqual(['Hermann', 'Goldbach', 'Euler', 'Bernouilli']); + }); + }); + + describe('sortByAlphabetical', () => { + + let data: + { + firstname: string; + lastname: string; + creator: string; + }[]; + + beforeEach(() => { + data = [ + { + firstname: 'Gaston', + lastname: 'Lagaffe', + creator: 'André Franquin' + }, + { + firstname: 'Mickey', + lastname: 'Mouse', + creator: 'Walt Disney' + }, + { + firstname: 'Gyro', + lastname: 'Gearloose', + creator: 'Carl Barks' + }, + { + firstname: 'Charlie', + lastname: 'Brown', + creator: 'Charles M. Schulz' + } + ]; + }); + + it('should return an array sorted by creator', () => { + const sorted = service.keySortByAlphabetical(data, 'creator'); + expect(sorted).toEqual( + [ + { firstname: 'Gaston', lastname: 'Lagaffe', creator: 'André Franquin' }, + { firstname: 'Gyro', lastname: 'Gearloose', creator: 'Carl Barks' }, + { firstname: 'Charlie', lastname: 'Brown', creator: 'Charles M. Schulz' }, + { firstname: 'Mickey', lastname: 'Mouse', creator: 'Walt Disney' } + ]); + }); + + it('should return an array sorted by firstname', () => { + const sorted = service.keySortByAlphabetical(data, 'firstname'); + expect(sorted).toEqual( + [ + { firstname: 'Charlie', lastname: 'Brown', creator: 'Charles M. Schulz' }, + { firstname: 'Gaston', lastname: 'Lagaffe', creator: 'André Franquin' }, + { firstname: 'Gyro', lastname: 'Gearloose', creator: 'Carl Barks' }, + { firstname: 'Mickey', lastname: 'Mouse', creator: 'Walt Disney' } + ]); + }); + + it('should return an array sorted by lastname', () => { + const sorted = service.keySortByAlphabetical(data, 'lastname'); + expect(sorted).toEqual( + [ + { firstname: 'Charlie', lastname: 'Brown', creator: 'Charles M. Schulz' }, + { firstname: 'Gyro', lastname: 'Gearloose', creator: 'Carl Barks' }, + { firstname: 'Gaston', lastname: 'Lagaffe', creator: 'André Franquin' }, + { firstname: 'Mickey', lastname: 'Mouse', creator: 'Walt Disney' } + ]); + }); + + it('should return an array sorted by lastname reversed', () => { + const sorted = service.keySortByAlphabetical(data, 'lastname', true); + expect(sorted).toEqual( + [ + { firstname: 'Mickey', lastname: 'Mouse', creator: 'Walt Disney' }, + { firstname: 'Gaston', lastname: 'Lagaffe', creator: 'André Franquin' }, + { firstname: 'Gyro', lastname: 'Gearloose', creator: 'Carl Barks' }, + { firstname: 'Charlie', lastname: 'Brown', creator: 'Charles M. Schulz' } + ]); + }); + + }); + + describe('Sort an array of ReadProject', () => { + let project1: ReadProject; + let project2: ReadProject; + let project3: ReadProject; + let project4: ReadProject; + let projects: ReadProject[]; + + beforeEach(() => { + project1 = new ReadProject(); + project1.id = '1'; + project1.longname = 'a'; + + project2 = new ReadProject(); + project2.id = '2'; + project2.longname = 'b'; + + project3 = new ReadProject(); + project3.id = '3'; + project3.longname = 'c'; + + project4 = new ReadProject(); + project4.id = '23'; + project4.longname = 'z'; + + projects = [project4, project2, project3, project1]; + }); + + it('should sort an array of ReadProject by "longname"', () => { + + const sorted = service.keySortByAlphabetical(projects, 'longname'); + + expect(sorted).toEqual( + [project1, project2, project3, project4] + ); + + }); + + it('should sort an array of ReadProject by "longname" reversed', () => { + + const sorted = service.keySortByAlphabetical(projects, 'longname', true); + + expect(sorted).toEqual( + [project4, project3, project2, project1] + ); + + }); + + }); + +}); diff --git a/src/app/main/services/sorting.service.ts b/src/app/main/services/sorting.service.ts new file mode 100644 index 0000000000..a3b1cd8c09 --- /dev/null +++ b/src/app/main/services/sorting.service.ts @@ -0,0 +1,37 @@ +import { Injectable } from '@angular/core'; + +@Injectable({ + providedIn: 'root' +}) +export class SortingService { + + constructor() { } + + /** + * reverses an array + */ + reverseArray(value: Array): Array { + return value.slice().reverse(); + } + + /** + * compares value by value and sorts in alphabetical order using the provided key + * optionally, you can have the array returned to you in reversed order by setting the reversed parameter to 'true' + */ + keySortByAlphabetical(value: Array, sortKey: keyof T, reversed = false): Array { + const sortedArray = value.slice(); + sortedArray.sort((a: T, b: T) => { + if (String(a[sortKey]).toLowerCase() < String(b[sortKey]).toLowerCase()) { + return -1; + } else if (String(a[sortKey]).toLowerCase() > String(b[sortKey]).toLowerCase()) { + return 1; + } else { + return 0; + } + }); + if (reversed) { + sortedArray.reverse(); + } + return sortedArray; + } +} diff --git a/src/assets/http/statusMsg.ts b/src/assets/http/statusMsg.ts new file mode 100644 index 0000000000..a13f13eaac --- /dev/null +++ b/src/assets/http/statusMsg.ts @@ -0,0 +1,263 @@ +import { Injectable } from '@angular/core'; + +@Injectable({ + providedIn: 'root' +}) +export class StatusMsg { + + default: any = { + 100: { + message: 'Continue', + description: 'The server has received the request headers, and the client should proceed to send the request body' + }, + 101: { + message: 'Switching Protocols', + description: 'The requester has asked the server to switch protocols' + }, + 103: { + message: 'Checkpoint', + description: 'Used in the resumable requests proposal to resume aborted PUT or POST requests' + }, + 200: { + message: 'OK', + description: 'The request is OK (this is the standard response for successful HTTP requests)' + }, + 201: { + message: 'Created', + description: 'The request has been fulfilled, and a new resource is created' + }, + 202: { + message: 'Accepted', + description: 'The request has been accepted for processing, but the processing has not been completed' + }, + 203: { + message: 'Non-Authoritative Information', + description: 'The request has been successfully processed, but is returning information that may be from another source' + }, + 204: { + message: 'No Content', + description: 'The request has been successfully processed, but is not returning any content' + }, + 205: { + message: 'Reset Content', + description: 'The request has been successfully processed, but is not returning any content, and requires that the requester reset the document view' + }, + 206: { + message: 'Partial Content', + description: 'The server is delivering only part of the resource due to a range header sent by the client' + }, + 300: { + message: 'Multiple Choices', + description: 'A link list. The user can select a link and go to that location. Maximum five addresses' + }, + 301: { + message: 'Moved Permanently', + description: 'The requested page has moved to a new URL' + }, + 302: { + message: 'Found', + description: 'The requested page has moved temporarily to a new URL' + }, + 303: { + message: 'See Other', + description: 'The requested page can be found under a different URL' + }, + 304: { + message: 'Not Modified', + description: 'Indicates the requested page has not been modified since last requested' + }, + 306: { + message: 'Switch Proxy', + description: '-- No longer used --' + }, + 307: { + message: 'Temporary Redirect', + description: 'The requested page has moved temporarily to a new URL' + }, + 308: { + message: 'Resume Incomplete', + description: 'Used in the resumable requests proposal to resume aborted PUT or POST requests' + }, + 400: { + message: 'Bad Request', + description: 'The request cannot be fulfilled due to bad syntax' + }, + 401: { + message: 'Unauthorized', + description: 'The request was a legal request, but the server is refusing to respond to it. For use when authentication is possible but has failed or not yet been provided' + }, + 402: { + message: 'Payment Required', + description: '-- Reserved for future use --' + }, + 403: { + message: 'Forbidden', + description: 'The request was a legal request, but the server is refusing to respond to it' + }, + 404: { + message: 'Not Found', + description: 'The requested page could not be found but may be available again in the future' + }, + 405: { + message: 'Method Not Allowed', + description: 'A request was made of a page using a request method not supported by that page' + }, + 406: { + message: 'Not Acceptable', + description: 'The server can only generate a response that is not accepted by the client' + }, + 407: { + message: 'Proxy Authentication Required', + description: 'The client must first authenticate itself with the proxy' + }, + 408: { + message: 'Request Timeout', + description: 'The server timed out waiting for the request' + }, + 409: { + message: 'Conflict', + description: 'The request could not be completed because of a conflict in the request' + }, + 410: { + message: 'Gone', + description: 'The requested page is no longer available' + }, + 411: { + message: 'Length Required', + description: 'The "Content-Length" is not defined. The server will not accept the request without it' + }, + 412: { + message: 'Precondition Failed', + description: 'The precondition given in the request evaluated to false by the server' + }, + 413: { + message: 'Request Entity Too Large', + description: 'The server will not accept the request, because the request entity is too large' + }, + 414: { + message: 'Request-URI Too Long', + description: 'The server will not accept the request, because the URL is too long. Occurs when you convert a POST request to a GET request with a long query information' + }, + 415: { + message: 'Unsupported Media Type', + description: 'The server will not accept the request, because the media type is not supported' + }, + 416: { + message: 'Requested Range Not Satisfiable', + description: 'The client has asked for a portion of the file, but the server cannot supply that portion' + }, + 417: { + message: 'Expectation Failed', + description: 'The server cannot meet the requirements of the Expect request-header field' + }, + 418: { + message: 'I\'m a teapot', + description: 'Any attempt to brew coffee with a teapot should result in the error code "418 I\'m a teapot". The resulting entity body MAY be short and stout' + }, + 421: { + message: 'Misdirected Request', + description: 'The request was directed at a server that is not able to produce a response (for example because a connection reuse)' + }, + 422: { + message: 'Unprocessable Entity', + description: 'The request was well-formed but was unable to be followed due to semantic errors' + }, + 423: { + message: 'Locked', + description: 'The resource that is being accessed is locked' + }, + 424: { + message: 'Failed Dependency', + description: 'The request failed due to failure of a previous request (e.g., a PROPPATCH)' + }, + 426: { + message: 'Upgrade Required', + description: 'The client should switch to a different protocol such as TLS/1.0, given in the Upgrade header field' + }, + 428: { + message: 'Precondition Required', + description: 'The origin server requires the request to be conditional' + }, + 429: { + message: 'Too Many Requests', + description: 'The user has sent too many requests in a given amount of time. Intended for use with rate limiting schemes' + }, + 431: { + message: 'Request Header Fields Too Large', + description: 'The server is unwilling to process the request because either an individual header field, or all the header fields collectively, are too large' + }, + 451: { + message: 'Unavailable For Legal Reasons', + description: 'A server operator has received a legal demand to deny access to a resource or to a set of resources that includes the requested resource' + }, + 500: { + message: 'Internal Server Error', + description: 'An error has occured in a server side script, a no more specific message is suitable' + }, + 501: { + message: 'Not Implemented', + description: 'The server either does not recognize the request method, or it lacks the ability to fulfill the request' + }, + 502: { + message: 'Bad Gateway', + description: 'The server was acting as a gateway or proxy and received an invalid response from the upstream server' + }, + 503: { + message: 'Service Unavailable', + description: 'The server is currently unavailable (overloaded or down)' + }, + 504: { + message: 'Gateway Timeout', + description: 'The server was acting as a gateway or proxy and did not receive a timely response from the upstream server' + }, + 505: { + message: 'HTTP Version Not Supported', + description: 'The server does not support the HTTP protocol version used in the request' + }, + 511: { + message: 'Network Authentication Required', + description: 'The client needs to authenticate to gain network access' + }, + info: { + lastUpdate: { + date: '20160411', + log: '4xx Codes updated via Wikipedia' + }, + references: { + W3Schools: 'http://www.w3schools.com/tags/ref_httpmessages.asp', + Wikipedia: 'https://en.wikipedia.org/wiki/List_of_HTTP_status_codes' + }, + codeLookup: { + info: 'Use the \'code\' variable in the url to lookup and individual code', + demoURL: '/statusMsg/?code=500' + }, + htmlDisplay: { + info: 'Add the \'html\' variable to your code lookup to get the error displayed in a nice html template', + demoURL: '/statusMsg/?code=404&html' + }, + invalidCode: 'If an invalid code is given, the site will just show the json list of all codes', + credits: 'This site was crafted by Unreal Designs and is powered by UDCDN' + }, + '1xx': { + message: 'Information', + description: '' + }, + '2xx': { + message: 'Successful', + description: '' + }, + '3xx': { + message: 'Redirection', + description: '' + }, + '4xx': { + message: 'Client Error', + description: '' + }, + '5xx': { + message: 'Server Error', + description: '' + } + }; + +} diff --git a/src/assets/style/_viewer.scss b/src/assets/style/_viewer.scss new file mode 100644 index 0000000000..487cdee173 --- /dev/null +++ b/src/assets/style/_viewer.scss @@ -0,0 +1,213 @@ +@import "./config"; +@import "./mixins"; +@import "./responsive"; + +// css for parent value components + +.read-mode-view { + font: 400 15px/24px sans-serif; + .rm-value, .rm-comment { + display: block; + margin: 0px; + } +} + +.child-value-component, +.value-component-comment { + display: block; +} + +.parent-component-wrapper .value-component-comment { + margin-top: -2em; +} + +.parent-value-component .child-value-component .mat-form-field-wrapper .mat-form-field-infix { + border-top: 0em solid transparent; +} + + +// css for child components of a parent value component + +.child-input-component { + display: inline-block; + vertical-align: bottom; + width: 49%; + &:nth-child(2) { + padding-left: 2%; + } +} + +.child-input-component.full-width { + width: 100%; +} + +// custom error message + +.custom-error-message { + font-size: 12px; + margin-top: -1em; + padding-bottom: 1em; +} + +// viewer operations + +.grid-container { + display: grid; + grid-template-columns: 70% 30%; + grid-template-rows: auto auto; +} + +.value-component, +.crud-buttons { + vertical-align: top; +} + +.value-component { + grid-row-start: 1; + grid-row-end: auto; + grid-column-start: 1; + grid-column-end: 2; + position: relative; + + .action-bubble { + position: absolute; + right: 0; + top: -20px; + border: 1px solid #e4e4e4; + border-radius: 10px; + padding: 0 1px; + background-color: #e4e4e4; + z-index: 2; + box-shadow: #949494 1px 4px 5px 0px; + + .button-container { + + button.mat-button-disabled { + + .mat-icon { + color: #aaaaaa; + } + + } + + button { + cursor: pointer; + border: none; + padding: 2px; + outline: none; + background-color: transparent; + color: #000000; + margin: 0 2px; + border-radius: 10px; + transition: background-color ease-out 0.5s; + min-width: inherit; + line-height: normal; + + .material-icons { + font-size: 18px; + } + + .mat-icon { + width: 18px; + height: 18px; + vertical-align: middle; + } + } + + button.info { + cursor: default; + } + + button:hover { + background-color: #c7c7c7; + } + } + } +} + +.value-component.highlighted { + background-color: #f5f5f5; +} + +.crud-buttons { + grid-row-start: 1; + grid-row-end: auto; + grid-column-start: 2; + grid-column-end: 3; +} + +.mat-error { + grid-row-start: 1; + grid-row-end: auto; + grid-column-start: 3; + grid-column-end: end; +} + +button.save, +button.cancel { + cursor: pointer; + border: none; + padding: 0em; + margin-left: 1em; + outline: none; + background-color: transparent; + color: #000000; +} + +button.save:disabled { + color: #adadad; + cursor: default; +} + +button.save .material-icons, +button.cancel .material-icons { + font-size: 18px; +} + +button.save .mat-icon, +button.cancel .mat-icon { + font-size: 18px; +} + +.deletion-dialog .title { + font-weight: bold; + font-size: 18px; + text-align: center; +} + +.deletion-dialog .action-buttons { + float: right; + + .cancel { + padding: 0 16px; + } +} + +.deletion-dialog-message { + font-size: 16px; + background-color: #ededf5; + border: 1px solid #d8d8df; + border-radius: 5px; + padding: 20px 10px 20px 10px; + text-align: center; +} + +.deletion-dialog-message .deletion-comment { + min-width: 80%; +} + + +.ck-content code { + font-family: monospace !important; +} + +.ck-content pre { + background-color: inherit !important; +} + +code, pre { + font-family: monospace !important; + background-color: hsla(0,0%,78%,.3); + padding: .15em; + border-radius: 2px; +}