mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-25 15:52:35 -04:00 
			
		
		
		
	rework of the front end components
This commit is contained in:
		
							parent
							
								
									339e96b63c
								
							
						
					
					
						commit
						ede3bd1391
					
				| @ -32,7 +32,7 @@ export class AppComponent implements OnInit, OnDestroy { | |||||||
| 
 | 
 | ||||||
|     this.successSubscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => { |     this.successSubscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => { | ||||||
|       this.toastService.show({title: "Document added", delay: 10000, content: `Document ${status.filename} was added to paperless.`, actionName: "Open document", action: () => { |       this.toastService.show({title: "Document added", delay: 10000, content: `Document ${status.filename} was added to paperless.`, actionName: "Open document", action: () => { | ||||||
|         this.router.navigate(['documents', status.document_id]) |         this.router.navigate(['documents', status.documentId]) | ||||||
|       }}) |       }}) | ||||||
|     }) |     }) | ||||||
| 
 | 
 | ||||||
| @ -42,6 +42,4 @@ export class AppComponent implements OnInit, OnDestroy { | |||||||
| 
 | 
 | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -9,10 +9,16 @@ | |||||||
| 
 | 
 | ||||||
|       </ngx-file-drop> |       </ngx-file-drop> | ||||||
|     </form> |     </form> | ||||||
|     <div *ngIf="uploadVisible" class="mt-3"> |     <div *ngFor="let status of getStatus()"> | ||||||
|       <p i18n>{uploadStatus.length, plural, =1 {Uploading file...} =other {Uploading {{uploadStatus.length}} files...}}</p> |       <p>{{status.filename}}: {{status.message}}</p> | ||||||
|       <ngb-progressbar [value]="loadedSum" [max]="totalSum" [striped]="true" [animated]="uploadStatus.length > 0"> |       <ngb-progressbar [value]="status.getProgress()" [max]="1" [striped]="true" [animated]="!isFinished(status)" [type]="getType(status)"> | ||||||
|       </ngb-progressbar> |       </ngb-progressbar> | ||||||
|  | 
 | ||||||
|  |       <div *ngIf="isFinished(status)" class="mb-2"> | ||||||
|  |         <button *ngIf="status.documentId" class="btn btn-sm btn-outline-primary mr-2" routerLink="/documents/{{status.documentId}}" (click)="dismiss(status)">Open document</button> | ||||||
|  |         <button class="btn btn-sm btn-outline-secondary" (click)="dismiss(status)">Dismiss</button> | ||||||
|  |       </div> | ||||||
|  |      | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </app-widget-frame> | </app-widget-frame> | ||||||
| @ -1,15 +1,10 @@ | |||||||
| import { HttpEventType } from '@angular/common/http'; | import { HttpEventType } from '@angular/common/http'; | ||||||
| import { Component, OnInit } from '@angular/core'; | import { Component, OnInit } from '@angular/core'; | ||||||
| import { FileSystemFileEntry, NgxFileDropEntry } from 'ngx-file-drop'; | import { FileSystemFileEntry, NgxFileDropEntry } from 'ngx-file-drop'; | ||||||
|  | import { ConsumerStatusService, FileStatus, FileStatusPhase } from 'src/app/services/consumer-status.service'; | ||||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | import { DocumentService } from 'src/app/services/rest/document.service'; | ||||||
| import { ToastService } from 'src/app/services/toast.service'; |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| interface UploadStatus { |  | ||||||
|   loaded: number |  | ||||||
|   total: number |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| @Component({ | @Component({ | ||||||
|   selector: 'app-upload-file-widget', |   selector: 'app-upload-file-widget', | ||||||
|   templateUrl: './upload-file-widget.component.html', |   templateUrl: './upload-file-widget.component.html', | ||||||
| @ -17,7 +12,34 @@ interface UploadStatus { | |||||||
| }) | }) | ||||||
| export class UploadFileWidgetComponent implements OnInit { | export class UploadFileWidgetComponent implements OnInit { | ||||||
| 
 | 
 | ||||||
|   constructor(private documentService: DocumentService, private toastService: ToastService) { } |   constructor( | ||||||
|  |     private documentService: DocumentService, | ||||||
|  |     private consumerStatusService: ConsumerStatusService | ||||||
|  |   ) { } | ||||||
|  | 
 | ||||||
|  |   getStatus() { | ||||||
|  |     return this.consumerStatusService.consumerStatus | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   isFinished(status: FileStatus) { | ||||||
|  |     return status.phase == FileStatusPhase.FAILED || status.phase == FileStatusPhase.SUCCESS | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   getType(status: FileStatus) { | ||||||
|  |     switch (status.phase) { | ||||||
|  |       case FileStatusPhase.PROCESSING: | ||||||
|  |       case FileStatusPhase.UPLOADING: | ||||||
|  |           return "primary" | ||||||
|  |       case FileStatusPhase.FAILED: | ||||||
|  |         return "danger" | ||||||
|  |       case FileStatusPhase.SUCCESS: | ||||||
|  |         return "success" | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   dismiss(status: FileStatus) { | ||||||
|  |     this.consumerStatusService.dismiss(status) | ||||||
|  |   } | ||||||
|    |    | ||||||
|   ngOnInit(): void { |   ngOnInit(): void { | ||||||
|   } |   } | ||||||
| @ -28,54 +50,37 @@ export class UploadFileWidgetComponent implements OnInit { | |||||||
|   public fileLeave(event){ |   public fileLeave(event){ | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   uploadStatus: UploadStatus[] = [] |  | ||||||
|   completedFiles = 0 |  | ||||||
| 
 |  | ||||||
|   uploadVisible = false |  | ||||||
| 
 |  | ||||||
|   get loadedSum() { |  | ||||||
|     return this.uploadStatus.map(s => s.loaded).reduce((a,b) => a+b, this.completedFiles > 0 ? 1 : 0) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get totalSum() { |  | ||||||
|     return this.uploadStatus.map(s => s.total).reduce((a,b) => a+b, 1) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   public dropped(files: NgxFileDropEntry[]) { |   public dropped(files: NgxFileDropEntry[]) { | ||||||
|     for (const droppedFile of files) { |     for (const droppedFile of files) { | ||||||
|       if (droppedFile.fileEntry.isFile) { |       if (droppedFile.fileEntry.isFile) { | ||||||
|       let uploadStatusObject: UploadStatus = {loaded: 0, total: 1} |  | ||||||
|       this.uploadStatus.push(uploadStatusObject) |  | ||||||
|       this.uploadVisible = true |  | ||||||
| 
 | 
 | ||||||
|       const fileEntry = droppedFile.fileEntry as FileSystemFileEntry; |       const fileEntry = droppedFile.fileEntry as FileSystemFileEntry; | ||||||
|         fileEntry.file((file: File) => { |         fileEntry.file((file: File) => { | ||||||
|           let formData = new FormData() |           let formData = new FormData() | ||||||
|           formData.append('document', file, file.name) |           formData.append('document', file, file.name) | ||||||
|  |           let status = this.consumerStatusService.newFileUpload() | ||||||
|  |           status.filename = file.name | ||||||
| 
 | 
 | ||||||
|           this.documentService.uploadDocument(formData).subscribe(event => { |           this.documentService.uploadDocument(formData).subscribe(event => { | ||||||
|             if (event.type == HttpEventType.UploadProgress) { |             if (event.type == HttpEventType.UploadProgress) { | ||||||
|               uploadStatusObject.loaded = event.loaded |               status.updateProgress(FileStatusPhase.UPLOADING, event.loaded, event.total) | ||||||
|               uploadStatusObject.total = event.total |  | ||||||
|             } else if (event.type == HttpEventType.Response) { |             } else if (event.type == HttpEventType.Response) { | ||||||
|               this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) |               status.taskId = event.body["task_id"] | ||||||
|               this.completedFiles += 1 |  | ||||||
|               this.toastService.showInfo($localize`The document has been uploaded and will be processed by the consumer shortly.`) |  | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|           }, error => { |           }, error => { | ||||||
|             this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) |             status.updateProgress(FileStatusPhase.FAILED) | ||||||
|             this.completedFiles += 1 |  | ||||||
|             switch (error.status) { |             switch (error.status) { | ||||||
|               case 400: { |               case 400: { | ||||||
|                 this.toastService.showInfo($localize`There was an error while uploading the document: ${error.error.document}`) |                 status.message = error.error.document | ||||||
|                 break; |                 break; | ||||||
|               } |               } | ||||||
|               default: { |               default: { | ||||||
|                 this.toastService.showInfo($localize`An error has occurred while uploading the document. Sorry!`) |                 status.message = $localize`An error has occurred while uploading the document. Sorry!` | ||||||
|                 break; |                 break; | ||||||
|               } |               } | ||||||
|             } |             } | ||||||
|  | 
 | ||||||
|           }) |           }) | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								src-ui/src/app/data/websocket-consumer-status-message.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								src-ui/src/app/data/websocket-consumer-status-message.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | export interface WebsocketConsumerStatusMessage { | ||||||
|  | 
 | ||||||
|  |   filename?: string | ||||||
|  |   task_id?: string | ||||||
|  |   current_progress?: number | ||||||
|  |   max_progress?: number | ||||||
|  |   status?: string | ||||||
|  |   message?: string | ||||||
|  |   document_id: number | ||||||
|  | 
 | ||||||
|  | } | ||||||
| @ -1,13 +1,57 @@ | |||||||
| import { Injectable } from '@angular/core'; | import { Injectable } from '@angular/core'; | ||||||
| import { Subject } from 'rxjs'; | import { Subject } from 'rxjs'; | ||||||
|  | import { WebsocketConsumerStatusMessage } from '../data/websocket-consumer-status-message'; | ||||||
|  | 
 | ||||||
|  | export enum FileStatusPhase { | ||||||
|  |   STARTED = 0, | ||||||
|  |   UPLOADING = 1, | ||||||
|  |   PROCESSING = 2, | ||||||
|  |   SUCCESS = 3, | ||||||
|  |   FAILED = 4 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export class FileStatus { | ||||||
|  | 
 | ||||||
|  |   filename: string | ||||||
|  | 
 | ||||||
|  |   taskId: string | ||||||
|  | 
 | ||||||
|  |   phase: FileStatusPhase = FileStatusPhase.STARTED   | ||||||
|  | 
 | ||||||
|  |   currentPhaseProgress: number | ||||||
|  | 
 | ||||||
|  |   currentPhaseMaxProgress: number | ||||||
|  | 
 | ||||||
|  |   message: string | ||||||
|  | 
 | ||||||
|  |   documentId: number | ||||||
|  | 
 | ||||||
|  |   getProgress(): number { | ||||||
|  |     switch (this.phase) { | ||||||
|  |       case FileStatusPhase.STARTED: | ||||||
|  |         return 0.0 | ||||||
|  |       case FileStatusPhase.UPLOADING: | ||||||
|  |         return this.currentPhaseProgress / this.currentPhaseMaxProgress * 0.2 | ||||||
|  |       case FileStatusPhase.PROCESSING: | ||||||
|  |         return this.currentPhaseProgress / this.currentPhaseMaxProgress * 0.8 + 0.2 | ||||||
|  |       case FileStatusPhase.SUCCESS: | ||||||
|  |       case FileStatusPhase.FAILED: | ||||||
|  |         return 1.0 | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   updateProgress(status: FileStatusPhase, currentProgress?: number, maxProgress?: number) { | ||||||
|  |     if (status >= this.phase) { | ||||||
|  |       this.phase = status | ||||||
|  |       if (currentProgress) { | ||||||
|  |         this.currentPhaseProgress = currentProgress | ||||||
|  |       } | ||||||
|  |       if (maxProgress) { | ||||||
|  |         this.currentPhaseMaxProgress = maxProgress | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
| export interface FileStatus { |  | ||||||
|   filename?: string |  | ||||||
|   current_progress?: number |  | ||||||
|   max_progress?: number |  | ||||||
|   status?: string |  | ||||||
|   message?: string |  | ||||||
|   document_id?: number |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @Injectable({ | @Injectable({ | ||||||
| @ -23,24 +67,41 @@ export class ConsumerStatusService { | |||||||
|   private documentConsumptionFinishedSubject = new Subject<FileStatus>() |   private documentConsumptionFinishedSubject = new Subject<FileStatus>() | ||||||
|   private documentConsumptionFailedSubject = new Subject<FileStatus>() |   private documentConsumptionFailedSubject = new Subject<FileStatus>() | ||||||
| 
 | 
 | ||||||
|  |   private get(taskId: string, filename?: string) { | ||||||
|  |     let status = this.consumerStatus.find(e => e.taskId == taskId) || this.consumerStatus.find(e => e.filename == filename) | ||||||
|  |     if (!status) { | ||||||
|  |       status = new FileStatus() | ||||||
|  |       this.consumerStatus.push(status) | ||||||
|  |     } | ||||||
|  |     status.taskId = taskId | ||||||
|  |     status.filename = filename | ||||||
|  |     return status | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   newFileUpload(): FileStatus { | ||||||
|  |     let status = new FileStatus() | ||||||
|  |     this.consumerStatus.push(status) | ||||||
|  |     return status | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   connect() { |   connect() { | ||||||
|     this.disconnect() |     this.disconnect() | ||||||
|     this.statusWebSocked = new WebSocket("ws://localhost:8000/ws/status/"); |     this.statusWebSocked = new WebSocket("ws://localhost:8000/ws/status/"); | ||||||
|     this.statusWebSocked.onmessage = (ev) => { |     this.statusWebSocked.onmessage = (ev) => { | ||||||
|       let statusUpdate: FileStatus = JSON.parse(ev['data']) |       let statusMessage: WebsocketConsumerStatusMessage = JSON.parse(ev['data']) | ||||||
| 
 | 
 | ||||||
|       let index = this.consumerStatus.findIndex(fs => fs.filename == statusUpdate.filename) |       let status = this.get(statusMessage.task_id, statusMessage.filename) | ||||||
|       if (index > -1) { |       status.updateProgress(FileStatusPhase.PROCESSING, statusMessage.current_progress, statusMessage.max_progress) | ||||||
|         this.consumerStatus[index] = statusUpdate |       status.message = statusMessage.message | ||||||
|       } else { |       status.documentId = statusMessage.document_id | ||||||
|         this.consumerStatus.push(statusUpdate) |  | ||||||
|       } |  | ||||||
| 
 | 
 | ||||||
|       if (statusUpdate.status == "SUCCESS") { |       if (statusMessage.status == "SUCCESS") { | ||||||
|         this.documentConsumptionFinishedSubject.next(statusUpdate) |         status.phase = FileStatusPhase.SUCCESS | ||||||
|  |         this.documentConsumptionFinishedSubject.next(status) | ||||||
|       } |       } | ||||||
|       if (statusUpdate.status == "FAILED") { |       if (statusMessage.status == "FAILED") { | ||||||
|         this.documentConsumptionFailedSubject.next(statusUpdate) |         status.phase = FileStatusPhase.FAILED | ||||||
|  |         this.documentConsumptionFailedSubject.next(status) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user