Merge branch 'dev' of github.com:sartography/cr-connect-bpmn into dev
This commit is contained in:
commit
2fa2921fa9
27
angular.json
27
angular.json
|
@ -35,7 +35,14 @@
|
||||||
},
|
},
|
||||||
"configurations": {
|
"configurations": {
|
||||||
"production": {
|
"production": {
|
||||||
"optimization": true,
|
"optimization": {
|
||||||
|
"scripts": true,
|
||||||
|
"styles": {
|
||||||
|
"minify": true,
|
||||||
|
"inlineCritical": false
|
||||||
|
},
|
||||||
|
"fonts": true
|
||||||
|
},
|
||||||
"outputHashing": "all",
|
"outputHashing": "all",
|
||||||
"sourceMap": false,
|
"sourceMap": false,
|
||||||
"extractCss": true,
|
"extractCss": true,
|
||||||
|
@ -53,7 +60,14 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"staging": {
|
"staging": {
|
||||||
"optimization": true,
|
"optimization": {
|
||||||
|
"scripts": true,
|
||||||
|
"styles": {
|
||||||
|
"minify": true,
|
||||||
|
"inlineCritical": false
|
||||||
|
},
|
||||||
|
"fonts": true
|
||||||
|
},
|
||||||
"outputHashing": "all",
|
"outputHashing": "all",
|
||||||
"sourceMap": false,
|
"sourceMap": false,
|
||||||
"extractCss": true,
|
"extractCss": true,
|
||||||
|
@ -71,7 +85,14 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"test": {
|
"test": {
|
||||||
"optimization": true,
|
"optimization": {
|
||||||
|
"scripts": true,
|
||||||
|
"styles": {
|
||||||
|
"minify": true,
|
||||||
|
"inlineCritical": false
|
||||||
|
},
|
||||||
|
"fonts": true
|
||||||
|
},
|
||||||
"outputHashing": "all",
|
"outputHashing": "all",
|
||||||
"sourceMap": false,
|
"sourceMap": false,
|
||||||
"extractCss": true,
|
"extractCss": true,
|
||||||
|
|
|
@ -41,7 +41,7 @@
|
||||||
"ngx-markdown": "^12.0.1",
|
"ngx-markdown": "^12.0.1",
|
||||||
"protractor": "~7.0.0",
|
"protractor": "~7.0.0",
|
||||||
"rxjs": "^6.5.5",
|
"rxjs": "^6.5.5",
|
||||||
"sartography-workflow-lib": "^0.0.537",
|
"sartography-workflow-lib": "^0.0.538",
|
||||||
"tslib": "^2.0.0",
|
"tslib": "^2.0.0",
|
||||||
"uuid": "^7.0.2",
|
"uuid": "^7.0.2",
|
||||||
"zone.js": "~0.11.4"
|
"zone.js": "~0.11.4"
|
||||||
|
@ -18748,9 +18748,9 @@
|
||||||
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
|
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
|
||||||
},
|
},
|
||||||
"node_modules/sartography-workflow-lib": {
|
"node_modules/sartography-workflow-lib": {
|
||||||
"version": "0.0.537",
|
"version": "0.0.538",
|
||||||
"resolved": "https://registry.npmjs.org/sartography-workflow-lib/-/sartography-workflow-lib-0.0.537.tgz",
|
"resolved": "https://registry.npmjs.org/sartography-workflow-lib/-/sartography-workflow-lib-0.0.538.tgz",
|
||||||
"integrity": "sha512-eVCgXpOyyS+rKx+KFM0xG+CHWxqGlrl3vDGJSG9WaGKgUI/gvSyPkRiSpXZURbdZgYz2Jz+djKKH+NxZ3c8K1A==",
|
"integrity": "sha512-KE6a/ZN9DVlwLXF0A5iq/JYYVIB6T+DaqHQNzGpAAnSSO5vRhQxNeiTqeL30aZhg8js2OzvVivLOAU4i4vKqoQ==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"tslib": "^2.2.0"
|
"tslib": "^2.2.0"
|
||||||
}
|
}
|
||||||
|
@ -36093,9 +36093,9 @@
|
||||||
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
|
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
|
||||||
},
|
},
|
||||||
"sartography-workflow-lib": {
|
"sartography-workflow-lib": {
|
||||||
"version": "0.0.537",
|
"version": "0.0.538",
|
||||||
"resolved": "https://registry.npmjs.org/sartography-workflow-lib/-/sartography-workflow-lib-0.0.537.tgz",
|
"resolved": "https://registry.npmjs.org/sartography-workflow-lib/-/sartography-workflow-lib-0.0.538.tgz",
|
||||||
"integrity": "sha512-eVCgXpOyyS+rKx+KFM0xG+CHWxqGlrl3vDGJSG9WaGKgUI/gvSyPkRiSpXZURbdZgYz2Jz+djKKH+NxZ3c8K1A==",
|
"integrity": "sha512-KE6a/ZN9DVlwLXF0A5iq/JYYVIB6T+DaqHQNzGpAAnSSO5vRhQxNeiTqeL30aZhg8js2OzvVivLOAU4i4vKqoQ==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"tslib": "^2.2.0"
|
"tslib": "^2.2.0"
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,8 +53,9 @@ import {MatSidenavModule} from '@angular/material/sidenav';
|
||||||
import { ConfirmDialogComponent } from './_dialogs/confirm-dialog/confirm-dialog.component';
|
import { ConfirmDialogComponent } from './_dialogs/confirm-dialog/confirm-dialog.component';
|
||||||
import {MatExpansionModule} from '@angular/material/expansion';
|
import {MatExpansionModule} from '@angular/material/expansion';
|
||||||
import { SettingsComponent } from './settings/settings.component';
|
import { SettingsComponent } from './settings/settings.component';
|
||||||
import {MatSelect, MatSelectModule} from '@angular/material/select';
|
import { MatSelectModule} from '@angular/material/select';
|
||||||
import {LibraryListComponent} from './library-list/library-list.component';
|
import {LibraryListComponent} from './library-list/library-list.component';
|
||||||
|
import {MatCheckboxModule} from '@angular/material/checkbox';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ThisEnvironment implements AppEnvironment {
|
export class ThisEnvironment implements AppEnvironment {
|
||||||
|
@ -107,36 +108,37 @@ export function getBaseHref(platformLocation: PlatformLocation): string {
|
||||||
ConfirmDialogComponent,
|
ConfirmDialogComponent,
|
||||||
SettingsComponent,
|
SettingsComponent,
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
BrowserAnimationsModule,
|
BrowserAnimationsModule,
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
FlexLayoutModule,
|
FlexLayoutModule,
|
||||||
FormlyModule,
|
FormlyModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
HttpClientModule,
|
HttpClientModule,
|
||||||
MatBottomSheetModule,
|
MatBottomSheetModule,
|
||||||
MatButtonModule,
|
MatButtonModule,
|
||||||
MatCardModule,
|
MatCardModule,
|
||||||
MatDialogModule,
|
MatDialogModule,
|
||||||
MatDividerModule,
|
MatDividerModule,
|
||||||
MatIconModule,
|
MatIconModule,
|
||||||
MatInputModule,
|
MatInputModule,
|
||||||
MatListModule,
|
MatListModule,
|
||||||
MatMenuModule,
|
MatMenuModule,
|
||||||
MatSnackBarModule,
|
MatSnackBarModule,
|
||||||
MatTabsModule,
|
MatTabsModule,
|
||||||
MatToolbarModule,
|
MatToolbarModule,
|
||||||
MatTooltipModule,
|
MatTooltipModule,
|
||||||
MatSelectModule,
|
MatSelectModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
SartographyFormsModule,
|
SartographyFormsModule,
|
||||||
SartographyPipesModule,
|
SartographyPipesModule,
|
||||||
SartographyWorkflowLibModule,
|
SartographyWorkflowLibModule,
|
||||||
AppRoutingModule,
|
AppRoutingModule,
|
||||||
MatSidenavModule,
|
MatSidenavModule,
|
||||||
MatExpansionModule,
|
MatExpansionModule,
|
||||||
// <-- This line MUST be last (https://angular.io/guide/router#module-import-order-matters)
|
MatCheckboxModule,
|
||||||
],
|
// <-- This line MUST be last (https://angular.io/guide/router#module-import-order-matters)
|
||||||
|
],
|
||||||
bootstrap: [AppComponent],
|
bootstrap: [AppComponent],
|
||||||
entryComponents: [
|
entryComponents: [
|
||||||
DeleteFileDialogComponent,
|
DeleteFileDialogComponent,
|
||||||
|
|
|
@ -1,46 +1,10 @@
|
||||||
<div class="file-list">
|
<div class="file-list">
|
||||||
<mat-list>
|
<mat-list>
|
||||||
<mat-list-item
|
<mat-list-item
|
||||||
*ngFor="let fm of fileMetas"
|
*ngFor="let fm of workflowLibraries"
|
||||||
[attr.data-file-meta-id]="fm.id"
|
|
||||||
[attr.data-workflow-spec-id]="workflowSpec.id"
|
|
||||||
>
|
>
|
||||||
<mat-icon (click)="editFile(fm)" mat-list-icon>{{fm.type | getIconCode}}</mat-icon>
|
<mat-checkbox (click)="updateItem(fm,isChecked(fm))" [checked]="isChecked(fm)">{{fm.display_name}}</mat-checkbox>
|
||||||
<ng-container *ngIf="fm.type === fileType.BPMN">
|
|
||||||
<button (click)="makePrimary(fm)" *ngIf="!fm.primary" mat-flat-button class="make-primary">
|
|
||||||
<mat-icon>radio_button_unchecked</mat-icon>
|
|
||||||
Make primary process
|
|
||||||
</button>
|
|
||||||
<button *ngIf="fm.primary" mat-flat-button class="make-primary">
|
|
||||||
<mat-icon>radio_button_checked</mat-icon>
|
|
||||||
Primary process
|
|
||||||
</button>
|
|
||||||
</ng-container>
|
|
||||||
<h4 (click)="editFile(fm)" mat-line>{{fm.name}}</h4>
|
|
||||||
<p (click)="editFile(fm)" mat-line> Updated: {{fm.last_modified | date:'medium'}}</p>
|
|
||||||
<button (click)="downloadFile(fm)" class="mat-elevation-z0" color="primary" mat-icon-button>
|
|
||||||
<mat-icon>save_alt</mat-icon>
|
|
||||||
</button>
|
|
||||||
<button (click)="editFile(fm)" class="mat-elevation-z0" color="primary" mat-icon-button>
|
|
||||||
<mat-icon>edit</mat-icon>
|
|
||||||
</button>
|
|
||||||
<button (click)="confirmDelete(fm)" class="mat-elevation-z0" color="warn" mat-icon-button>
|
|
||||||
<mat-icon>delete</mat-icon>
|
|
||||||
</button>
|
|
||||||
</mat-list-item>
|
</mat-list-item>
|
||||||
</mat-list>
|
</mat-list>
|
||||||
</div>
|
</div>
|
||||||
<div class="file-list-actions">
|
|
||||||
<button [queryParams]="{action: 'newFile'}" [routerLink]="['/modeler/' + workflowSpec.id]" mat-button>
|
|
||||||
<mat-icon>note_add</mat-icon>
|
|
||||||
Add new BPMN or DMN file
|
|
||||||
</button>
|
|
||||||
<button [queryParams]="{action: 'openFile'}" [routerLink]="['/modeler/' + workflowSpec.id]" mat-button>
|
|
||||||
<mat-icon>cloud_upload</mat-icon>
|
|
||||||
Upload BPMN or DMN file
|
|
||||||
</button>
|
|
||||||
<button (click)="editFile()" mat-button>
|
|
||||||
<mat-icon>cloud_upload</mat-icon>
|
|
||||||
Upload Template file
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
|
@ -1,241 +1,55 @@
|
||||||
import {APP_BASE_HREF} from '@angular/common';
|
import {APP_BASE_HREF} from '@angular/common';
|
||||||
import {HttpHeaders} from '@angular/common/http';
|
|
||||||
import {HttpClientTestingModule, HttpTestingController} from '@angular/common/http/testing';
|
import {HttpClientTestingModule, HttpTestingController} from '@angular/common/http/testing';
|
||||||
import {async, ComponentFixture, TestBed} from '@angular/core/testing';
|
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
import {MAT_DIALOG_DATA, MatDialogModule, MatDialogRef} from '@angular/material/dialog';
|
|
||||||
import {MatIconModule} from '@angular/material/icon';
|
import {MatIconModule} from '@angular/material/icon';
|
||||||
import {MatListModule} from '@angular/material/list';
|
import {MatMenuModule} from '@angular/material/menu';
|
||||||
import {MatSnackBarModule} from '@angular/material/snack-bar';
|
import {ApiService, MockEnvironment, mockWorkflowSpec0, mockWorkflowSpec1, WorkflowSpec} from 'sartography-workflow-lib';
|
||||||
import {BrowserDynamicTestingModule} from '@angular/platform-browser-dynamic/testing';
|
import {LibraryListComponent} from './library-list.component';
|
||||||
import {BrowserAnimationsModule} from '@angular/platform-browser/animations';
|
|
||||||
import {RouterTestingModule} from '@angular/router/testing';
|
|
||||||
import createClone from 'rfdc';
|
|
||||||
import {of} from 'rxjs';
|
|
||||||
import {
|
|
||||||
ApiService,
|
|
||||||
FileMeta,
|
|
||||||
FileType,
|
|
||||||
MockEnvironment, mockFile0,
|
|
||||||
mockFileMeta0,
|
|
||||||
mockFileMetas,
|
|
||||||
mockWorkflowSpec0
|
|
||||||
} from 'sartography-workflow-lib';
|
|
||||||
import {DeleteFileDialogComponent} from '../_dialogs/delete-file-dialog/delete-file-dialog.component';
|
|
||||||
import {DeleteFileDialogData} from '../_interfaces/dialog-data';
|
|
||||||
import {GetIconCodePipe} from '../_pipes/get-icon-code.pipe';
|
|
||||||
import {FileListComponent} from '../file-list/file-list.component';
|
|
||||||
|
|
||||||
|
|
||||||
describe('FileListComponent', () => {
|
|
||||||
|
describe('LibraryListComponent', () => {
|
||||||
|
let component: LibraryListComponent;
|
||||||
|
let fixture: ComponentFixture<LibraryListComponent>;
|
||||||
let httpMock: HttpTestingController;
|
let httpMock: HttpTestingController;
|
||||||
let component: FileListComponent;
|
let libraries: WorkflowSpec[];
|
||||||
let fixture: ComponentFixture<FileListComponent>;
|
|
||||||
const timeString = '2020-01-23T12:34:12.345Z';
|
|
||||||
const timeCode = new Date(timeString).getTime();
|
|
||||||
|
|
||||||
|
|
||||||
beforeEach(async(() => {
|
beforeEach(async(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [
|
|
||||||
BrowserAnimationsModule,
|
|
||||||
HttpClientTestingModule,
|
|
||||||
MatDialogModule,
|
|
||||||
MatIconModule,
|
|
||||||
MatListModule,
|
|
||||||
MatSnackBarModule,
|
|
||||||
RouterTestingModule,
|
|
||||||
],
|
|
||||||
declarations: [
|
declarations: [
|
||||||
DeleteFileDialogComponent,
|
LibraryListComponent
|
||||||
FileListComponent,
|
],
|
||||||
GetIconCodePipe,
|
imports: [
|
||||||
|
HttpClientTestingModule,
|
||||||
|
MatIconModule,
|
||||||
|
MatMenuModule,
|
||||||
|
// RouterTestingModule,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
ApiService,
|
ApiService,
|
||||||
{provide: 'APP_ENVIRONMENT', useClass: MockEnvironment},
|
{provide: 'APP_ENVIRONMENT', useClass: MockEnvironment},
|
||||||
{provide: APP_BASE_HREF, useValue: ''},
|
{provide: APP_BASE_HREF, useValue: ''},
|
||||||
{
|
],
|
||||||
provide: MatDialogRef,
|
|
||||||
useValue: {
|
|
||||||
close: (dialogResult: any) => {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{provide: MAT_DIALOG_DATA, useValue: []},
|
|
||||||
]
|
|
||||||
}).overrideModule(BrowserDynamicTestingModule, {
|
|
||||||
set: {
|
|
||||||
entryComponents: [
|
|
||||||
DeleteFileDialogComponent,
|
|
||||||
]
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.compileComponents();
|
.compileComponents();
|
||||||
}));
|
}));
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
fixture = TestBed.createComponent(FileListComponent);
|
localStorage.setItem('token', 'some_token');
|
||||||
component = fixture.componentInstance;
|
|
||||||
httpMock = TestBed.inject(HttpTestingController);
|
httpMock = TestBed.inject(HttpTestingController);
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
fixture = TestBed.createComponent(LibraryListComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
libraries = [mockWorkflowSpec0, mockWorkflowSpec1];
|
||||||
|
libraries[0].library = true;
|
||||||
|
libraries[1].library = true;
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
const uReq = httpMock.expectOne('apiRoot/workflow-specification?libraries=true');
|
||||||
|
expect(uReq.request.method).toEqual('GET');
|
||||||
const fmsReq = httpMock.expectOne(`apiRoot/file?workflow_spec_id=${mockWorkflowSpec0.id}`);
|
uReq.flush(libraries);
|
||||||
expect(fmsReq.request.method).toEqual('GET');
|
expect(component.workflowLibraries).toEqual(libraries);
|
||||||
fmsReq.flush(mockFileMetas);
|
|
||||||
expect(component.fileMetas.length).toBeGreaterThan(0);
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => {
|
|
||||||
httpMock.verify();
|
|
||||||
fixture.destroy();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create', () => {
|
it('should create', () => {
|
||||||
expect(component).toBeTruthy();
|
expect(component).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should sort files by name', () => {
|
|
||||||
let prevFileMeta;
|
|
||||||
|
|
||||||
for (const thisFileMeta of component.fileMetas) {
|
|
||||||
if (!prevFileMeta) {
|
|
||||||
prevFileMeta = thisFileMeta;
|
|
||||||
} else {
|
|
||||||
expect(thisFileMeta.name).toBeGreaterThan(prevFileMeta.name);
|
|
||||||
prevFileMeta = thisFileMeta;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should show a confirmation dialog before deleting a file', () => {
|
|
||||||
const mockConfirmDeleteData: DeleteFileDialogData = {
|
|
||||||
confirm: false,
|
|
||||||
fileMeta: mockFileMeta0,
|
|
||||||
};
|
|
||||||
|
|
||||||
const _deleteFileSpy = spyOn((component as any), '_deleteFile').and.stub();
|
|
||||||
const openDialogSpy = spyOn(component.dialog, 'open')
|
|
||||||
.and.returnValue({afterClosed: () => of(mockConfirmDeleteData)} as any);
|
|
||||||
|
|
||||||
component.confirmDelete(mockFileMeta0);
|
|
||||||
expect(openDialogSpy).toHaveBeenCalled();
|
|
||||||
expect(_deleteFileSpy).not.toHaveBeenCalled();
|
|
||||||
|
|
||||||
mockConfirmDeleteData.confirm = true;
|
|
||||||
component.confirmDelete(mockFileMeta0);
|
|
||||||
expect(openDialogSpy).toHaveBeenCalled();
|
|
||||||
expect(_deleteFileSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should delete a file', () => {
|
|
||||||
const loadFileMetasSpy = spyOn((component as any), '_loadFileMetas').and.stub();
|
|
||||||
(component as any)._deleteFile(mockFileMeta0);
|
|
||||||
const fmsReq = httpMock.expectOne(`apiRoot/file/${mockFileMeta0.id}`);
|
|
||||||
expect(fmsReq.request.method).toEqual('DELETE');
|
|
||||||
fmsReq.flush(null);
|
|
||||||
|
|
||||||
expect(loadFileMetasSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should navigate to modeler to edit a BPMN or DMN file', () => {
|
|
||||||
const routerNavigateSpy = spyOn((component as any).router, 'navigate');
|
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
|
||||||
component.editFile(mockFileMeta0);
|
|
||||||
expect(routerNavigateSpy).toHaveBeenCalledWith([`/modeler/${mockWorkflowSpec0.id}/${mockFileMeta0.id}`]);
|
|
||||||
|
|
||||||
routerNavigateSpy.calls.reset();
|
|
||||||
const mockDmnMeta = createClone()(mockFileMeta0);
|
|
||||||
mockDmnMeta.type = FileType.DMN;
|
|
||||||
component.editFile(mockDmnMeta);
|
|
||||||
expect(routerNavigateSpy).toHaveBeenCalledWith([`/modeler/${mockWorkflowSpec0.id}/${mockDmnMeta.id}`]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should open file metadata dialog for non-BPMN files', () => {
|
|
||||||
const routerNavigateSpy = spyOn((component as any).router, 'navigate');
|
|
||||||
const editFileMetaSpy = spyOn(component, 'editFileMeta');
|
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
|
||||||
const mockDocMeta = createClone()(mockFileMeta0);
|
|
||||||
mockDocMeta.type = FileType.DOCX;
|
|
||||||
component.editFile(mockDocMeta);
|
|
||||||
expect(routerNavigateSpy).not.toHaveBeenCalled();
|
|
||||||
expect(editFileMetaSpy).toHaveBeenCalledWith(mockDocMeta);
|
|
||||||
|
|
||||||
routerNavigateSpy.calls.reset();
|
|
||||||
editFileMetaSpy.calls.reset();
|
|
||||||
component.editFile(null);
|
|
||||||
expect(routerNavigateSpy).not.toHaveBeenCalled();
|
|
||||||
expect(editFileMetaSpy).toHaveBeenCalledWith(null);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should open file metadata dialog', () => {
|
|
||||||
const _openFileDialogSpy = spyOn((component as any), '_openFileDialog').and.stub();
|
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
|
||||||
const mockDocMeta: FileMeta = createClone()(mockFileMeta0);
|
|
||||||
mockDocMeta.type = FileType.DOCX;
|
|
||||||
component.editFileMeta(mockDocMeta);
|
|
||||||
|
|
||||||
const expectedFile = new File([], mockDocMeta.name, {
|
|
||||||
type: mockDocMeta.content_type,
|
|
||||||
lastModified: timeCode
|
|
||||||
});
|
|
||||||
const fReq = httpMock.expectOne(`apiRoot/file/${mockDocMeta.id}/data`);
|
|
||||||
|
|
||||||
const mockHeaders = new HttpHeaders()
|
|
||||||
.append('last-modified', expectedFile.lastModified.toString())
|
|
||||||
.append('content-type', mockDocMeta.content_type);
|
|
||||||
expect(fReq.request.method).toEqual('GET');
|
|
||||||
fReq.flush(new ArrayBuffer(8), {headers: mockHeaders});
|
|
||||||
expect(fReq.request.method).toEqual('GET');
|
|
||||||
expect(_openFileDialogSpy).toHaveBeenCalledWith(mockDocMeta, expectedFile);
|
|
||||||
|
|
||||||
_openFileDialogSpy.calls.reset();
|
|
||||||
|
|
||||||
component.editFileMeta(null);
|
|
||||||
expect(_openFileDialogSpy).toHaveBeenCalledWith();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should upload new file from file dialog', () => {
|
|
||||||
const openDialogSpy = spyOn(component.dialog, 'open')
|
|
||||||
.and.returnValue({afterClosed: () => of({file: mockFile0})} as any);
|
|
||||||
const _loadFileMetasSpy = spyOn((component as any), '_loadFileMetas').and.stub();
|
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
|
||||||
|
|
||||||
(component as any)._openFileDialog();
|
|
||||||
const addReq = httpMock.expectOne(`apiRoot/file?workflow_spec_id=${mockWorkflowSpec0.id}`);
|
|
||||||
expect(addReq.request.method).toEqual('POST');
|
|
||||||
addReq.flush(mockFileMeta0);
|
|
||||||
|
|
||||||
expect(openDialogSpy).toHaveBeenCalled();
|
|
||||||
expect(_loadFileMetasSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should update existing file from file dialog', () => {
|
|
||||||
const openDialogSpy = spyOn(component.dialog, 'open')
|
|
||||||
.and.returnValue({afterClosed: () => of({fileMetaId: mockFileMeta0.id, file: mockFile0})} as any);
|
|
||||||
const _loadFileMetasSpy = spyOn((component as any), '_loadFileMetas').and.stub();
|
|
||||||
component.workflowSpec = mockWorkflowSpec0;
|
|
||||||
|
|
||||||
(component as any)._openFileDialog(mockFileMeta0, mockFile0);
|
|
||||||
const updateReq = httpMock.expectOne(`apiRoot/file/${mockFileMeta0.id}/data`);
|
|
||||||
expect(updateReq.request.method).toEqual('PUT');
|
|
||||||
updateReq.flush(mockFileMeta0);
|
|
||||||
|
|
||||||
expect(openDialogSpy).toHaveBeenCalled();
|
|
||||||
expect(_loadFileMetasSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should flag a file as primary', () => {
|
|
||||||
const updateFileMetaSpy = spyOn((component as any).api, 'updateFileMeta').and.returnValue(of(mockFileMeta0));
|
|
||||||
const _loadFileMetasSpy = spyOn((component as any), '_loadFileMetas').and.stub();
|
|
||||||
expect(component.fileMetas.length).toEqual(mockFileMetas.length);
|
|
||||||
component.makePrimary(mockFileMeta0);
|
|
||||||
|
|
||||||
expect(updateFileMetaSpy).toHaveBeenCalledTimes(mockFileMetas.length);
|
|
||||||
expect(component.fileMetas.length).toEqual(mockFileMetas.length);
|
|
||||||
expect(component.fileMetas.reduce((sum, fm) => fm.primary ? sum + 1 : sum, 0)).toEqual(1);
|
|
||||||
expect(_loadFileMetasSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,20 +1,8 @@
|
||||||
import {Component, Input, OnChanges, OnInit} from '@angular/core';
|
import {Component, Input, OnChanges, OnInit} from '@angular/core';
|
||||||
import {MatDialog} from '@angular/material/dialog';
|
|
||||||
import {MatSnackBar} from '@angular/material/snack-bar';
|
|
||||||
import {ActivatedRoute, Router} from '@angular/router';
|
|
||||||
import {
|
import {
|
||||||
ApiService,
|
ApiService,
|
||||||
FileMeta,
|
|
||||||
FileParams,
|
|
||||||
FileType,
|
|
||||||
getFileType,
|
|
||||||
isNumberDefined, newFileFromResponse,
|
|
||||||
WorkflowSpec
|
WorkflowSpec
|
||||||
} from 'sartography-workflow-lib';
|
} from 'sartography-workflow-lib';
|
||||||
import {DeleteFileDialogComponent} from '../_dialogs/delete-file-dialog/delete-file-dialog.component';
|
|
||||||
import {OpenFileDialogComponent} from '../_dialogs/open-file-dialog/open-file-dialog.component';
|
|
||||||
import {DeleteFileDialogData, OpenFileDialogData} from '../_interfaces/dialog-data';
|
|
||||||
import * as fileSaver from 'file-saver';
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-library-list',
|
selector: 'app-library-list',
|
||||||
|
@ -22,136 +10,47 @@ import * as fileSaver from 'file-saver';
|
||||||
styleUrls: ['./library-list.component.scss']
|
styleUrls: ['./library-list.component.scss']
|
||||||
})
|
})
|
||||||
export class LibraryListComponent implements OnInit, OnChanges {
|
export class LibraryListComponent implements OnInit, OnChanges {
|
||||||
@Input() workflowSpec: WorkflowSpec;
|
@Input() workflowSpecId: string;
|
||||||
fileMetas: FileMeta[];
|
workflowLibraries: WorkflowSpec[];
|
||||||
fileType = FileType;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private api: ApiService,
|
private api: ApiService,
|
||||||
public dialog: MatDialog,
|
|
||||||
private route: ActivatedRoute,
|
|
||||||
private router: Router,
|
|
||||||
private snackBar: MatSnackBar,
|
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this._loadFileMetas();
|
this._loadWorkflowLibraries();
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnChanges() {
|
ngOnChanges() {
|
||||||
this._loadFileMetas();
|
this._loadWorkflowLibraries();
|
||||||
}
|
}
|
||||||
|
|
||||||
editFile(fileMeta?: FileMeta) {
|
|
||||||
if (fileMeta && ((fileMeta.type === FileType.BPMN) || (fileMeta.type === FileType.DMN))) {
|
isChecked(libraryspec): boolean {
|
||||||
this.router.navigate([`/modeler/${this.workflowSpec.id}/${fileMeta.id}`]);
|
let checked = false;
|
||||||
} else {
|
for (const item of libraryspec.parents) {
|
||||||
// Show edit file meta dialog
|
checked = checked || (item.id === this.workflowSpecId);
|
||||||
this.editFileMeta(fileMeta);
|
|
||||||
}
|
}
|
||||||
|
return checked;
|
||||||
}
|
}
|
||||||
|
|
||||||
editFileMeta(fm: FileMeta) {
|
updateItem(library: WorkflowSpec , checked: boolean) {
|
||||||
if (fm && isNumberDefined(fm.id)) {
|
if (checked) {
|
||||||
this.api.getFileData(fm.id).subscribe(response => {
|
this.api.deleteWorkflowLibrary(this.workflowSpecId, library.id).subscribe(() => {
|
||||||
const file = newFileFromResponse(fm, response);
|
this._loadWorkflowLibraries();
|
||||||
this._openFileDialog(fm, file);
|
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
this._openFileDialog();
|
this.api.addWorkflowLibrary(this.workflowSpecId, library.id).subscribe(() => {
|
||||||
}
|
this._loadWorkflowLibraries();
|
||||||
}
|
|
||||||
|
|
||||||
confirmDelete(fm: FileMeta) {
|
|
||||||
const dialogRef = this.dialog.open(DeleteFileDialogComponent, {
|
|
||||||
data: {
|
|
||||||
confirm: false,
|
|
||||||
fileMeta: fm,
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
dialogRef.afterClosed().subscribe((data: DeleteFileDialogData) => {
|
|
||||||
if (data && data.confirm && data.fileMeta) {
|
|
||||||
this._deleteFile(data.fileMeta);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
makePrimary(fmPrimary: FileMeta) {
|
|
||||||
if (fmPrimary.type === FileType.BPMN) {
|
|
||||||
let numUpdated = 0;
|
|
||||||
// Fixme: This buisness rule does not belong here.
|
|
||||||
this.fileMetas.forEach(fm => {
|
|
||||||
fm.primary = (fmPrimary.id === fm.id);
|
|
||||||
this.api.updateFileMeta(fm).subscribe(() => {
|
|
||||||
numUpdated++;
|
|
||||||
|
|
||||||
// Reload all fileMetas when all have been updated.
|
|
||||||
if (numUpdated === this.fileMetas.length) {
|
|
||||||
this._loadFileMetas();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _openFileDialog(fm?: FileMeta, file?: File) {
|
private _loadWorkflowLibraries() {
|
||||||
const dialogData: OpenFileDialogData = {
|
|
||||||
fileMetaId: fm ? fm.id : undefined,
|
|
||||||
file: file,
|
|
||||||
mode: 'local',
|
|
||||||
fileTypes: [FileType.DOC, FileType.DOCX, FileType.XLSX, FileType.XLS],
|
|
||||||
};
|
|
||||||
const dialogRef = this.dialog.open(OpenFileDialogComponent, {data: dialogData});
|
|
||||||
|
|
||||||
dialogRef.afterClosed().subscribe((data: OpenFileDialogData) => {
|
this.api.getWorkflowSpecificationLibraries().subscribe(wfs => {
|
||||||
if (data && data.file) {
|
this.workflowLibraries = wfs;
|
||||||
const newFileMeta: FileMeta = {
|
|
||||||
id: data.fileMetaId,
|
|
||||||
content_type: data.file.type,
|
|
||||||
name: data.file.name,
|
|
||||||
type: getFileType(data.file),
|
|
||||||
workflow_spec_id: this.workflowSpec.id,
|
|
||||||
};
|
|
||||||
|
|
||||||
if (isNumberDefined(data.fileMetaId)) {
|
|
||||||
// Update existing file
|
|
||||||
this.api.updateFileData(newFileMeta, data.file).subscribe(() => {
|
|
||||||
this._loadFileMetas();
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
// Add new file
|
|
||||||
const fileParams: FileParams = {
|
|
||||||
workflow_spec_id: this.workflowSpec.id,
|
|
||||||
};
|
|
||||||
|
|
||||||
this.api.addFile(fileParams, newFileMeta, data.file).subscribe(dbFm => {
|
|
||||||
this._loadFileMetas();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private _deleteFile(fileMeta: FileMeta) {
|
|
||||||
this.api.deleteFileMeta(fileMeta.id).subscribe(() => {
|
|
||||||
this._loadFileMetas();
|
|
||||||
this.snackBar.open(`Deleted file ${fileMeta.name}.`, 'Ok', {duration: 3000});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private _loadFileMetas() {
|
|
||||||
this.api.getFileMetas({workflow_spec_id: this.workflowSpec.id}).subscribe(fms => {
|
|
||||||
this.fileMetas = fms.sort((a, b) => (a.name > b.name) ? 1 : -1);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
downloadFile(fm: FileMeta) {
|
|
||||||
this.api.getFileData(fm.id).subscribe(response => {
|
|
||||||
const blob = new Blob([response.body], {type: fm.content_type});
|
|
||||||
fileSaver.saveAs(blob, fm.name);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@
|
||||||
</mat-card-title>
|
</mat-card-title>
|
||||||
</mat-card-header>
|
</mat-card-header>
|
||||||
<mat-card-content>
|
<mat-card-content>
|
||||||
<dl gdAreas="field value" gdColumns="8ch auto">
|
<dl gdAreas="field value" gdColumns="10ch auto">
|
||||||
<dt>ID</dt><dd>{{workflowSpec.id}}</dd>
|
<dt>ID</dt><dd>{{workflowSpec.id}}</dd>
|
||||||
<dt>Name</dt><dd>{{workflowSpec.name}}</dd>
|
<dt>Name</dt><dd>{{workflowSpec.name}}</dd>
|
||||||
<dt>Description</dt><dd>{{workflowSpec.description}}</dd>
|
<dt>Description</dt><dd>{{workflowSpec.description}}</dd>
|
||||||
|
@ -23,9 +23,12 @@
|
||||||
<ng-template #thenBlock><dd>True</dd></ng-template>
|
<ng-template #thenBlock><dd>True</dd></ng-template>
|
||||||
<ng-template #elseBlock><dd>False</dd></ng-template>
|
<ng-template #elseBlock><dd>False</dd></ng-template>
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
<h4>Workflow Spec Files</h4>
|
<h4>Workflow Spec Files</h4>
|
||||||
<app-file-list [workflowSpec]="workflowSpec"></app-file-list>
|
<app-file-list [workflowSpec]="workflowSpec"></app-file-list>
|
||||||
|
<div *ngIf="!workflowSpec.library">
|
||||||
|
<h4>Included Libraries</h4>
|
||||||
|
<app-library-list [workflowSpecId]="workflowSpec.id"></app-library-list>
|
||||||
|
</div>
|
||||||
</mat-card-content>
|
</mat-card-content>
|
||||||
<mat-card-actions>
|
<mat-card-actions>
|
||||||
</mat-card-actions>
|
</mat-card-actions>
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
<div class="container mat-typography" fxLayout="column" fxLayoutGap="10px">
|
<div class="workflow-specs" fxLayout="column" fxLayoutGap="10px">
|
||||||
<h1>Workflow Specifications</h1>
|
<h1>Workflow Specifications</h1>
|
||||||
<div fxLayout="row" fxLayoutGap="10px">
|
<div class="buttons" fxLayout="row" fxLayoutGap="10px">
|
||||||
<button id="add_spec" mat-flat-button color="primary" (click)="editWorkflowSpec()">
|
<button id="add_spec" mat-flat-button color="primary" (click)="editWorkflowSpec()" fxLayoutAlign="stretch">
|
||||||
<mat-icon>library_add</mat-icon>
|
<mat-icon>library_add</mat-icon>
|
||||||
Add new workflow specification
|
Add new workflow specification
|
||||||
</button>
|
</button>
|
||||||
<button id="add_category" mat-flat-button color="accent" (click)="editWorkflowSpecCategory()">
|
<button id="add_category" mat-flat-button color="accent" (click)="editWorkflowSpecCategory()" fxLayoutAlign="stretch">
|
||||||
<mat-icon>post_add</mat-icon>
|
<mat-icon>post_add</mat-icon>
|
||||||
Add category
|
Add category
|
||||||
</button>
|
</button>
|
||||||
|
@ -20,7 +20,7 @@
|
||||||
<h4>Master Specification</h4>
|
<h4>Master Specification</h4>
|
||||||
<mat-list>
|
<mat-list>
|
||||||
<mat-list-item class="workflow-spec" fxLayout="row">
|
<mat-list-item class="workflow-spec" fxLayout="row">
|
||||||
<a class="spec_menu_item" (click)="selectSpec(masterStatusSpec)">{{masterStatusSpec.display_name}}</a>
|
<span class="spec_menu_item" (click)="selectSpec(masterStatusSpec)">{{masterStatusSpec.display_name}}</span>
|
||||||
</mat-list-item>
|
</mat-list-item>
|
||||||
</mat-list>
|
</mat-list>
|
||||||
</div>
|
</div>
|
||||||
|
@ -31,7 +31,7 @@
|
||||||
<h4>Libraries</h4>
|
<h4>Libraries</h4>
|
||||||
<mat-list>
|
<mat-list>
|
||||||
<mat-list-item *ngFor="let wfs of workflowLibraries" class="workflow-spec" fxLayout="row" fxLayoutGap="10px" fxLayoutAlign="start center">
|
<mat-list-item *ngFor="let wfs of workflowLibraries" class="workflow-spec" fxLayout="row" fxLayoutGap="10px" fxLayoutAlign="start center">
|
||||||
<a [ngClass]="{'spec_menu_item':true, 'spec-selected': selectedSpec && wfs.id === selectedSpec.id}" (click)="selectSpec(wfs)">{{wfs.display_name}}</a>
|
<span [ngClass]="{'spec_menu_item':true, 'spec-selected': selectedSpec && wfs.id === selectedSpec.id}" (click)="selectSpec(wfs)">{{wfs.display_name}}</span>
|
||||||
<span class="spec-actions" fxLayout="row" fxLayoutGap="10px">
|
<span class="spec-actions" fxLayout="row" fxLayoutGap="10px">
|
||||||
<button
|
<button
|
||||||
mat-icon-button
|
mat-icon-button
|
||||||
|
@ -85,7 +85,7 @@
|
||||||
</mat-expansion-panel-header>
|
</mat-expansion-panel-header>
|
||||||
<mat-list>
|
<mat-list>
|
||||||
<mat-list-item *ngFor="let wfs of cat.workflow_specs" class="workflow-spec" fxLayout="row" fxLayoutGap="10px" fxLayoutAlign="start center">
|
<mat-list-item *ngFor="let wfs of cat.workflow_specs" class="workflow-spec" fxLayout="row" fxLayoutGap="10px" fxLayoutAlign="start center">
|
||||||
<a [ngClass]="{'spec_menu_item':true, 'spec-selected': selectedSpec && wfs.id === selectedSpec.id}" (click)="selectSpec(wfs)">{{wfs.display_name}}</a>
|
<span [ngClass]="{'spec_menu_item':true, 'spec-selected': selectedSpec && wfs.id === selectedSpec.id}" (click)="selectSpec(wfs)">{{wfs.display_name}}</span>
|
||||||
<span class="spec-actions" fxLayout="row" fxLayoutGap="10px" *ngIf="cat && cat.id !== null">
|
<span class="spec-actions" fxLayout="row" fxLayoutGap="10px" *ngIf="cat && cat.id !== null">
|
||||||
<button
|
<button
|
||||||
*ngIf="cat && cat.workflow_specs.length > 0 && wfs.display_order !== 0"
|
*ngIf="cat && cat.workflow_specs.length > 0 && wfs.display_order !== 0"
|
||||||
|
|
|
@ -4,10 +4,14 @@
|
||||||
margin: 0;
|
margin: 0;
|
||||||
max-height: 40px;
|
max-height: 40px;
|
||||||
}
|
}
|
||||||
.container {
|
.workflow-specs {
|
||||||
padding: 3rem;
|
padding: 3rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.buttons {
|
||||||
|
height: 60px;
|
||||||
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
min-height: 600px;
|
min-height: 600px;
|
||||||
}
|
}
|
||||||
|
@ -42,13 +46,14 @@
|
||||||
.workflow-spec {
|
.workflow-spec {
|
||||||
border-left: 4px solid $brand-gray-light;
|
border-left: 4px solid $brand-gray-light;
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
border-left: 4px solid $brand-primary;
|
border-left: 4px solid $brand-primary;
|
||||||
background-color: $brand-primary-tint-4;
|
background-color: $brand-primary-tint-4;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
cursor: pointer;
|
|
||||||
.spec_menu_item {
|
.spec_menu_item {
|
||||||
flex-flow: row;
|
flex-flow: row;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
|
Loading…
Reference in New Issue