diff --git a/src-ui/src/app/components/dashboard/dashboard.component.ts b/src-ui/src/app/components/dashboard/dashboard.component.ts
index c7410c3f2..a5f857b99 100644
--- a/src-ui/src/app/components/dashboard/dashboard.component.ts
+++ b/src-ui/src/app/components/dashboard/dashboard.component.ts
@@ -1,7 +1,7 @@
import { Component, OnInit } from '@angular/core';
-import { Title } from '@angular/platform-browser';
-import { SavedViewConfigService } from 'src/app/services/saved-view-config.service';
-import { environment } from 'src/environments/environment';
+import { Meta } from '@angular/platform-browser';
+import { PaperlessSavedView } from 'src/app/data/paperless-saved-view';
+import { SavedViewService } from 'src/app/services/rest/saved-view.service';
@Component({
@@ -12,15 +12,36 @@ import { environment } from 'src/environments/environment';
export class DashboardComponent implements OnInit {
constructor(
- public savedViewConfigService: SavedViewConfigService,
- private titleService: Title) { }
+ private savedViewService: SavedViewService,
+ private meta: Meta
+ ) { }
+ get displayName() {
+ let tagFullName = this.meta.getTag('name=full_name')
+ let tagUsername = this.meta.getTag('name=username')
+ if (tagFullName && tagFullName.content) {
+ return tagFullName.content
+ } else if (tagUsername && tagUsername.content) {
+ return tagUsername.content
+ } else {
+ return null
+ }
+ }
- savedViews = []
+ get subtitle() {
+ if (this.displayName) {
+ return $localize`Hello ${this.displayName}, welcome to Paperless-ng!`
+ } else {
+ return $localize`Welcome to Paperless-ng!`
+ }
+ }
+
+ savedViews: PaperlessSavedView[] = []
ngOnInit(): void {
- this.savedViews = this.savedViewConfigService.getDashboardConfigs()
- this.titleService.setTitle(`Dashboard - ${environment.appTitle}`)
+ this.savedViewService.listAll().subscribe(results => {
+ this.savedViews = results.results.filter(savedView => savedView.show_on_dashboard)
+ })
}
}
diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html
index e63ecc47b..d05d8a667 100644
--- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html
+++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html
@@ -1,21 +1,21 @@
-
+
- Show all
+ Show all
- Created |
- Title |
+ Created |
+ Title |
- {{doc.created | date}} |
- {{doc.title}}
+ | {{doc.created | customDate}} |
+ {{doc.title | documentTitle}}
|
-
\ No newline at end of file
+
diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss
index e69de29bb..4e275dc81 100644
--- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss
+++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss
@@ -0,0 +1,8 @@
+table {
+ overflow-wrap: anywhere;
+ table-layout: fixed;
+}
+
+th:first-child {
+ width: 25%;
+}
diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts
index a55bf57fc..20bc26cdc 100644
--- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts
+++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts
@@ -1,8 +1,10 @@
-import { Component, Input, OnInit } from '@angular/core';
+import { Component, Input, OnDestroy, OnInit } from '@angular/core';
import { Router } from '@angular/router';
+import { Subscription } from 'rxjs';
import { PaperlessDocument } from 'src/app/data/paperless-document';
-import { SavedViewConfig } from 'src/app/data/saved-view-config';
+import { PaperlessSavedView } from 'src/app/data/paperless-saved-view';
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
+import { ConsumerStatusService } from 'src/app/services/consumer-status.service';
import { DocumentService } from 'src/app/services/rest/document.service';
@Component({
@@ -10,31 +12,45 @@ import { DocumentService } from 'src/app/services/rest/document.service';
templateUrl: './saved-view-widget.component.html',
styleUrls: ['./saved-view-widget.component.scss']
})
-export class SavedViewWidgetComponent implements OnInit {
+export class SavedViewWidgetComponent implements OnInit, OnDestroy {
constructor(
private documentService: DocumentService,
private router: Router,
- private list: DocumentListViewService) { }
-
+ private list: DocumentListViewService,
+ private consumerStatusService: ConsumerStatusService) { }
+
@Input()
- savedView: SavedViewConfig
+ savedView: PaperlessSavedView
documents: PaperlessDocument[] = []
+ subscription: Subscription
+
ngOnInit(): void {
- this.documentService.list(1,10,this.savedView.sortField,this.savedView.sortDirection,this.savedView.filterRules).subscribe(result => {
+ this.reload()
+ this.subscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => {
+ this.reload()
+ })
+ }
+
+ ngOnDestroy(): void {
+ this.subscription.unsubscribe()
+ }
+
+ reload() {
+ this.documentService.listFiltered(1,10,this.savedView.sort_field, this.savedView.sort_reverse, this.savedView.filter_rules).subscribe(result => {
this.documents = result.results
})
}
showAll() {
- if (this.savedView.showInSideBar) {
+ if (this.savedView.show_in_sidebar) {
this.router.navigate(['view', this.savedView.id])
} else {
- this.list.load(this.savedView)
+ this.list.loadSavedView(this.savedView, true)
this.router.navigate(["documents"])
- }
+ }
}
}
diff --git a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html
index 50d844b36..a29b50f78 100644
--- a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html
+++ b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html
@@ -1,6 +1,6 @@
-
+
- Documents in inbox: {{statistics.documents_inbox}}
- Total documents: {{statistics.documents_total}}
+ Documents in inbox: {{statistics?.documents_inbox}}
+ Total documents: {{statistics?.documents_total}}
-
\ No newline at end of file
+
diff --git a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts
index 73eee698c..f1488a66f 100644
--- a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts
+++ b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts
@@ -1,6 +1,7 @@
import { HttpClient } from '@angular/common/http';
-import { Component, OnInit } from '@angular/core';
-import { Observable } from 'rxjs';
+import { Component, OnDestroy, OnInit } from '@angular/core';
+import { Observable, Subscription } from 'rxjs';
+import { ConsumerStatusService } from 'src/app/services/consumer-status.service';
import { environment } from 'src/environments/environment';
export interface Statistics {
@@ -14,20 +15,34 @@ export interface Statistics {
templateUrl: './statistics-widget.component.html',
styleUrls: ['./statistics-widget.component.scss']
})
-export class StatisticsWidgetComponent implements OnInit {
+export class StatisticsWidgetComponent implements OnInit, OnDestroy {
- constructor(private http: HttpClient) { }
+ constructor(private http: HttpClient,
+ private consumerStatusService: ConsumerStatusService) { }
statistics: Statistics = {}
- getStatistics(): Observable
{
+ subscription: Subscription
+
+ private getStatistics(): Observable {
return this.http.get(`${environment.apiBaseUrl}statistics/`)
}
-
- ngOnInit(): void {
+
+ reload() {
this.getStatistics().subscribe(statistics => {
this.statistics = statistics
})
}
+ ngOnInit(): void {
+ this.reload()
+ this.subscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => {
+ this.reload()
+ })
+ }
+
+ ngOnDestroy(): void {
+ this.subscription.unsubscribe()
+ }
+
}
diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html
index 013486a47..c4f7fa21c 100644
--- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html
+++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html
@@ -1,18 +1,52 @@
-
-
+
+
-
-
Uploading {{uploadStatus.length}} file(s)
-
0">
-
+
0">{{getStatusSummary()}}
+
+
+
+
+
+ {getStatusHidden().length, plural, =1 {One more document} other {{{getStatusHidden().length}} more documents}}
+ •
+ Show all
+
+
-
\ No newline at end of file
+
+
+
+
+ {{status.filename}}
+ {{status.message}}
+
+
+
+
+
+
diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss
index e69de29bb..c13a7bd47 100644
--- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss
+++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss
@@ -0,0 +1,35 @@
+@import "/src/theme";
+
+form {
+ position: relative;
+}
+
+.alert-heading {
+ font-size: 80%;
+ font-weight: bold;
+}
+
+.alerts-hidden {
+ .btn {
+ line-height: 1;
+ }
+}
+
+.btn-open {
+ line-height: 1;
+
+ svg {
+ margin-top: -1px;
+ }
+}
+
+.progress {
+ position: absolute;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ height: auto;
+ mix-blend-mode: soft-light;
+ pointer-events: none;
+}
diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts
index 2ea4825f1..e74e2bc41 100644
--- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts
+++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts
@@ -1,14 +1,10 @@
import { HttpEventType } from '@angular/common/http';
import { Component, OnInit } from '@angular/core';
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 { Toast, ToastService } from 'src/app/services/toast.service';
-
-interface UploadStatus {
- loaded: number
- total: number
-}
+const MAX_ALERTS = 5
@Component({
selector: 'app-upload-file-widget',
@@ -16,8 +12,89 @@ interface UploadStatus {
styleUrls: ['./upload-file-widget.component.scss']
})
export class UploadFileWidgetComponent implements OnInit {
+ alertsExpanded = false
- constructor(private documentService: DocumentService, private toastService: ToastService) { }
+ constructor(
+ private documentService: DocumentService,
+ private consumerStatusService: ConsumerStatusService
+ ) { }
+
+ getStatus() {
+ return this.consumerStatusService.getConsumerStatus().slice(0, MAX_ALERTS)
+ }
+
+ getStatusSummary() {
+ let strings = []
+ let countUploadingAndProcessing = this.consumerStatusService.getConsumerStatusNotCompleted().length
+ let countFailed = this.getStatusFailed().length
+ let countSuccess = this.getStatusSuccess().length
+ if (countUploadingAndProcessing > 0) {
+ strings.push($localize`Processing: ${countUploadingAndProcessing}`)
+ }
+ if (countFailed > 0) {
+ strings.push($localize`Failed: ${countFailed}`)
+ }
+ if (countSuccess > 0) {
+ strings.push($localize`Added: ${countSuccess}`)
+ }
+ return strings.join($localize`:this string is used to separate processing, failed and added on the file upload widget:, `)
+ }
+
+ getStatusHidden() {
+ if (this.consumerStatusService.getConsumerStatus().length < MAX_ALERTS) return []
+ else return this.consumerStatusService.getConsumerStatus().slice(MAX_ALERTS)
+ }
+
+ getStatusUploading() {
+ return this.consumerStatusService.getConsumerStatus(FileStatusPhase.UPLOADING)
+ }
+
+ getStatusFailed() {
+ return this.consumerStatusService.getConsumerStatus(FileStatusPhase.FAILED)
+ }
+
+ getStatusSuccess() {
+ return this.consumerStatusService.getConsumerStatus(FileStatusPhase.SUCCESS)
+ }
+
+ getStatusCompleted() {
+ return this.consumerStatusService.getConsumerStatusCompleted()
+ }
+ getTotalUploadProgress() {
+ let current = 0
+ let max = 0
+
+ this.getStatusUploading().forEach(status => {
+ current += status.currentPhaseProgress
+ max += status.currentPhaseMaxProgress
+ })
+
+ return current / Math.max(max, 1)
+ }
+
+ isFinished(status: FileStatus) {
+ return status.phase == FileStatusPhase.FAILED || status.phase == FileStatusPhase.SUCCESS
+ }
+
+ getStatusColor(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)
+ }
+
+ dismissCompleted() {
+ this.consumerStatusService.dismissCompleted()
+ }
ngOnInit(): void {
}
@@ -28,54 +105,39 @@ export class UploadFileWidgetComponent implements OnInit {
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[]) {
for (const droppedFile of files) {
if (droppedFile.fileEntry.isFile) {
- let uploadStatusObject: UploadStatus = {loaded: 0, total: 1}
- this.uploadStatus.push(uploadStatusObject)
- this.uploadVisible = true
const fileEntry = droppedFile.fileEntry as FileSystemFileEntry;
fileEntry.file((file: File) => {
let formData = new FormData()
formData.append('document', file, file.name)
+ let status = this.consumerStatusService.newFileUpload(file.name)
+
+ status.message = $localize`Connecting...`
this.documentService.uploadDocument(formData).subscribe(event => {
if (event.type == HttpEventType.UploadProgress) {
- uploadStatusObject.loaded = event.loaded
- uploadStatusObject.total = event.total
+ status.updateProgress(FileStatusPhase.UPLOADING, event.loaded, event.total)
+ status.message = $localize`Uploading...`
} else if (event.type == HttpEventType.Response) {
- this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1)
- this.completedFiles += 1
- this.toastService.showToast(Toast.make("Information", "The document has been uploaded and will be processed by the consumer shortly."))
+ status.taskId = event.body["task_id"]
+ status.message = $localize`Upload complete, waiting...`
}
-
+
}, error => {
- this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1)
- this.completedFiles += 1
switch (error.status) {
case 400: {
- this.toastService.showToast(Toast.makeError(`There was an error while uploading the document: ${error.error.document}`))
+ this.consumerStatusService.fail(status, error.error.document)
break;
}
default: {
- this.toastService.showToast(Toast.makeError("An error has occurred while uploading the document. Sorry!"))
+ this.consumerStatusService.fail(status, $localize`HTTP error: ${error.status} ${error.statusText}`)
break;
}
}
+
})
});
}
diff --git a/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html b/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html
index 0caf55f11..17c7ffe1e 100644
--- a/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html
+++ b/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html
@@ -1,16 +1,16 @@
-
+
- Paperless is running! :)
- You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list.
- After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and have them displayed on the dashboard instead of this message.
- Paperless offers some more features that try to make your life easier, such as:
+ Paperless is running! :)
+ You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list.
+ After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message.
+ Paperless offers some more features that try to make your life easier:
- - Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.
- - You can configure paperless to read your mails and add documents from attached files.
+ - Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.
+ - You can configure paperless to read your mails and add documents from attached files.
- Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.
+ Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.
-
\ No newline at end of file
+
diff --git a/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html b/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html
index 1d7d2d906..1cd440306 100644
--- a/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html
+++ b/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html
@@ -4,9 +4,9 @@
{{title}}
-
+