mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Merge branch 'dev'
This commit is contained in:
commit
3876f1e9ec
13
CONTRIBUTING.md
Normal file
13
CONTRIBUTING.md
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Contributing
|
||||||
|
|
||||||
|
If you feel that somethings is not working, please submit an issue. You can also ask questions on the issue tracker by tagging your question with the question tag.
|
||||||
|
|
||||||
|
Pull requests are welcome, however, I will be a little bit more strict about what goes into the code and what does not. If you want to make a big change, please ask me about it first.
|
||||||
|
|
||||||
|
* When making additions to the project, consider if the majority of users will benefit from your change. If not, you're probably better of forking the project.
|
||||||
|
* Also consider if your change will get in the way of other users. A good change is a change that enhances the experience of some users who want that change and does not affect users who do not care about the change.
|
||||||
|
|
||||||
|
However:
|
||||||
|
|
||||||
|
* Bug fixes and are always welcome. Docker makes things easier, however, I alone cannot ensure that this runs on all platforms.
|
||||||
|
* Improvements to the styling of the front-end are always welcome. I'm no expert in things UX, and simply copied one of the Bootstrap examples. I think it turned out rather good, but I just can't seem to get some things working properly.
|
@ -1,3 +1,5 @@
|
|||||||
|
LABEL maintainer="Jonas Winkler <dev@jpwinkler.de>"
|
||||||
|
|
||||||
###############################################################################
|
###############################################################################
|
||||||
### Front end ###
|
### Front end ###
|
||||||
###############################################################################
|
###############################################################################
|
||||||
|
21
README.md
21
README.md
@ -1,8 +1,8 @@
|
|||||||
[ en | [de](README-de.md) | [el](README-el.md) ]
|
[Paperless](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and others that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents.
|
||||||
|
|
||||||

|
Paperless-ng is a fork of the original project, adding a new interface and many other changes under the hood. For a detailed list of changes, see below.
|
||||||
|
|
||||||
[Paperless](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and others that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents. This project extends on the project and modernizes many things.
|
This project is still in development and some things may not work as expected.
|
||||||
|
|
||||||
This project is still under development. There also is no automatic way yet to migrate your current paperless setup to this version. I'm working on that.
|
This project is still under development. There also is no automatic way yet to migrate your current paperless setup to this version. I'm working on that.
|
||||||
|
|
||||||
@ -20,7 +20,9 @@ Here's what you get:
|
|||||||
|
|
||||||

|

|
||||||
|
|
||||||
# What is different in this version of Paperless?
|
# Why Paperless-ng?
|
||||||
|
|
||||||
|
I wanted to make big changes to the project that will impact the way it is used by its users greatly. Among the users who currently use paperless in production there are probably many that don't want these changes right away. I also wanted to have more control over what goes into the code and what does not. Therefore, paperless-ng was created. NG stands for both Angular (the framework used for the Frontend) and next-gen. Publishing this project under a different name also avoids confusion between paperless and paperless-ng.
|
||||||
|
|
||||||
This is a list of changes that have been made to the original project.
|
This is a list of changes that have been made to the original project.
|
||||||
|
|
||||||
@ -86,9 +88,13 @@ Please be aware that this uses a postgres database instead of sqlite. If you wan
|
|||||||
|
|
||||||
Alternatively, you can install the dependencies and setup apache and a database server yourself. Details for that will be available in the documentation.
|
Alternatively, you can install the dependencies and setup apache and a database server yourself. Details for that will be available in the documentation.
|
||||||
|
|
||||||
|
# Migrating to paperless-ng
|
||||||
|
|
||||||
|
Don't do it yet. The migrations are in place, but I have not verified yet that they work.
|
||||||
|
|
||||||
# Documentation
|
# Documentation
|
||||||
|
|
||||||
The documentation for Paperless is available on [ReadTheDocs](https://paperless.readthedocs.io/). Updated documentation for this project is not yet available.
|
The documentation for Paperless is available on [ReadTheDocs](https://paperless-ng.readthedocs.io/). Updated documentation for this project is not yet available.
|
||||||
|
|
||||||
# Affiliated Projects
|
# Affiliated Projects
|
||||||
|
|
||||||
@ -99,11 +105,6 @@ Paperless has been around a while now, and people are starting to build stuff on
|
|||||||
* [ansible-role-paperless](https://github.com/ovv/ansible-role-paperless): An easy way to get Paperless running via Ansible.
|
* [ansible-role-paperless](https://github.com/ovv/ansible-role-paperless): An easy way to get Paperless running via Ansible.
|
||||||
* [paperless-cli](https://github.com/stgarf/paperless-cli): A golang command line binary to interact with a Paperless instance.
|
* [paperless-cli](https://github.com/stgarf/paperless-cli): A golang command line binary to interact with a Paperless instance.
|
||||||
|
|
||||||
# Similar Projects
|
|
||||||
|
|
||||||
There's another project out there called [Mayan EDMS](https://www.mayan-edms.com/) that has a surprising amount of technical overlap with Paperless. Also based on Django and using a consumer model with Tesseract and Unpaper, Mayan EDMS is *much* more featureful and comes with a slick UI as well, but still in Python 2. It may be that Paperless consumes fewer resources, but to be honest, this is just a guess as I haven't tested this myself. One thing's for certain though, *Paperless* is a **way** better name.
|
|
||||||
|
|
||||||
|
|
||||||
# Important Note
|
# Important Note
|
||||||
|
|
||||||
Document scanners are typically used to scan sensitive documents. Things like your social insurance number, tax records, invoices, etc. Everything is stored in the clear without encryption by default (it needs to be searchable, so if someone has ideas on how to do that on encrypted data, I'm all ears). This means that Paperless should never be run on an untrusted host. Instead, I recommend that if you do want to use it, run it locally on a server in your own home.
|
Document scanners are typically used to scan sensitive documents. Things like your social insurance number, tax records, invoices, etc. Everything is stored in the clear without encryption by default (it needs to be searchable, so if someone has ideas on how to do that on encrypted data, I'm all ears). This means that Paperless should never be run on an untrusted host. Instead, I recommend that if you do want to use it, run it locally on a server in your own home.
|
||||||
|
@ -38,6 +38,8 @@ import { SelectComponent } from './components/common/input/select/select.compone
|
|||||||
import { CheckComponent } from './components/common/input/check/check.component';
|
import { CheckComponent } from './components/common/input/check/check.component';
|
||||||
import { SaveViewConfigDialogComponent } from './components/document-list/save-view-config-dialog/save-view-config-dialog.component';
|
import { SaveViewConfigDialogComponent } from './components/document-list/save-view-config-dialog/save-view-config-dialog.component';
|
||||||
import { InfiniteScrollModule } from 'ngx-infinite-scroll';
|
import { InfiniteScrollModule } from 'ngx-infinite-scroll';
|
||||||
|
import { DateTimeComponent } from './components/common/input/date-time/date-time.component';
|
||||||
|
import { TagsComponent } from './components/common/input/tags/tags.component';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
@ -69,7 +71,9 @@ import { InfiniteScrollModule } from 'ngx-infinite-scroll';
|
|||||||
TextComponent,
|
TextComponent,
|
||||||
SelectComponent,
|
SelectComponent,
|
||||||
CheckComponent,
|
CheckComponent,
|
||||||
SaveViewConfigDialogComponent
|
SaveViewConfigDialogComponent,
|
||||||
|
DateTimeComponent,
|
||||||
|
TagsComponent
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
|
@ -69,7 +69,7 @@ export class AppFrameComponent implements OnInit, OnDestroy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.openDocumentsSubscription = this.openDocumentsService.getOpenDocuments().subscribe(docs => this.openDocuments = docs)
|
this.openDocuments = this.openDocumentsService.getOpenDocuments()
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
|
@ -0,0 +1,14 @@
|
|||||||
|
<div class="form-row">
|
||||||
|
<div class="form-group col">
|
||||||
|
<label for="created_date">{{titleDate}}</label>
|
||||||
|
<input type="date" class="form-control" id="created_date" [(ngModel)]="dateValue" (change)="dateOrTimeChanged()">
|
||||||
|
</div>
|
||||||
|
<div class="form-group col">
|
||||||
|
<label for="created_time">{{titleTime}}</label>
|
||||||
|
<input type="time" class="form-control" id="created_time" [(ngModel)]="timeValue" (change)="dateOrTimeChanged()">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<!-- <small *ngIf="hint" class="form-text text-muted">{{hint}}</small> -->
|
@ -0,0 +1,25 @@
|
|||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
|
||||||
|
import { DateTimeComponent } from './date-time.component';
|
||||||
|
|
||||||
|
describe('DateTimeComponent', () => {
|
||||||
|
let component: DateTimeComponent;
|
||||||
|
let fixture: ComponentFixture<DateTimeComponent>;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
await TestBed.configureTestingModule({
|
||||||
|
declarations: [ DateTimeComponent ]
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(DateTimeComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create', () => {
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,62 @@
|
|||||||
|
import { formatDate } from '@angular/common';
|
||||||
|
import { Component, forwardRef, Input, OnInit } from '@angular/core';
|
||||||
|
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
|
||||||
|
import { AbstractInputComponent } from '../abstract-input';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
providers: [{
|
||||||
|
provide: NG_VALUE_ACCESSOR,
|
||||||
|
useExisting: forwardRef(() => DateTimeComponent),
|
||||||
|
multi: true
|
||||||
|
}],
|
||||||
|
selector: 'app-input-date-time',
|
||||||
|
templateUrl: './date-time.component.html',
|
||||||
|
styleUrls: ['./date-time.component.css']
|
||||||
|
})
|
||||||
|
export class DateTimeComponent implements OnInit,ControlValueAccessor {
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
}
|
||||||
|
|
||||||
|
onChange = (newValue: any) => {};
|
||||||
|
|
||||||
|
onTouched = () => {};
|
||||||
|
|
||||||
|
writeValue(newValue: any): void {
|
||||||
|
this.dateValue = formatDate(newValue, 'yyyy-MM-dd', "en-US")
|
||||||
|
this.timeValue = formatDate(newValue, 'HH:mm:ss', 'en-US')
|
||||||
|
}
|
||||||
|
registerOnChange(fn: any): void {
|
||||||
|
this.onChange = fn;
|
||||||
|
}
|
||||||
|
registerOnTouched(fn: any): void {
|
||||||
|
this.onTouched = fn;
|
||||||
|
}
|
||||||
|
setDisabledState?(isDisabled: boolean): void {
|
||||||
|
this.disabled = isDisabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
titleDate: string = "Date"
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
titleTime: string = "Time"
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
disabled: boolean = false
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
hint: string
|
||||||
|
|
||||||
|
timeValue
|
||||||
|
|
||||||
|
dateValue
|
||||||
|
|
||||||
|
ngOnInit(): void {
|
||||||
|
}
|
||||||
|
|
||||||
|
dateOrTimeChanged() {
|
||||||
|
this.onChange(formatDate(this.dateValue + "T" + this.timeValue,"yyyy-MM-ddTHH:mm:ssZZZZZ", "en-us", "UTC"))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,10 @@
|
|||||||
|
.tags-form-control {
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.scrollable-menu {
|
||||||
|
height: auto;
|
||||||
|
max-height: 300px;
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
@ -0,0 +1,30 @@
|
|||||||
|
<div class="form-group">
|
||||||
|
<label for="exampleFormControlTextarea1">Tags</label>
|
||||||
|
|
||||||
|
<div class="input-group">
|
||||||
|
<div class="form-control tags-form-control" id="tags">
|
||||||
|
<app-tag class="mr-2" *ngFor="let id of displayValue" [tag]="getTag(id)" (click)="removeTag(id)"></app-tag>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="input-group-append" ngbDropdown placement="top-right">
|
||||||
|
<button class="btn btn-outline-secondary" type="button" ngbDropdownToggle></button>
|
||||||
|
<div ngbDropdownMenu class="scrollable-menu">
|
||||||
|
<button type="button" *ngFor="let tag of tags" ngbDropdownItem (click)="addTag(tag.id)">
|
||||||
|
<app-tag [tag]="tag"></app-tag>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="input-group-append">
|
||||||
|
|
||||||
|
<button class="btn btn-outline-secondary" type="button" (click)="createTag()">
|
||||||
|
<svg class="buttonicon" fill="currentColor">
|
||||||
|
<use xlink:href="assets/bootstrap-icons.svg#plus" />
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<small class="form-text text-muted" *ngIf="hint">{{hint}}</small>
|
||||||
|
|
||||||
|
</div>
|
@ -0,0 +1,25 @@
|
|||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
|
||||||
|
import { TagsComponent } from './tags.component';
|
||||||
|
|
||||||
|
describe('TagsComponent', () => {
|
||||||
|
let component: TagsComponent;
|
||||||
|
let fixture: ComponentFixture<TagsComponent>;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
await TestBed.configureTestingModule({
|
||||||
|
declarations: [ TagsComponent ]
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(TagsComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create', () => {
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,96 @@
|
|||||||
|
import { ThrowStmt } from '@angular/compiler';
|
||||||
|
import { Component, forwardRef, Input, OnInit } from '@angular/core';
|
||||||
|
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
|
||||||
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
import { Observable } from 'rxjs';
|
||||||
|
import { TagEditDialogComponent } from 'src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component';
|
||||||
|
import { PaperlessTag } from 'src/app/data/paperless-tag';
|
||||||
|
import { TagService } from 'src/app/services/rest/tag.service';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
providers: [{
|
||||||
|
provide: NG_VALUE_ACCESSOR,
|
||||||
|
useExisting: forwardRef(() => TagsComponent),
|
||||||
|
multi: true
|
||||||
|
}],
|
||||||
|
selector: 'app-input-tags',
|
||||||
|
templateUrl: './tags.component.html',
|
||||||
|
styleUrls: ['./tags.component.css']
|
||||||
|
})
|
||||||
|
export class TagsComponent implements OnInit, ControlValueAccessor {
|
||||||
|
|
||||||
|
constructor(private tagService: TagService, private modalService: NgbModal) { }
|
||||||
|
|
||||||
|
|
||||||
|
onChange = (newValue: number[]) => {};
|
||||||
|
|
||||||
|
onTouched = () => {};
|
||||||
|
|
||||||
|
writeValue(newValue: number[]): void {
|
||||||
|
this.value = newValue
|
||||||
|
if (this.tags) {
|
||||||
|
this.displayValue = newValue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
registerOnChange(fn: any): void {
|
||||||
|
this.onChange = fn;
|
||||||
|
}
|
||||||
|
registerOnTouched(fn: any): void {
|
||||||
|
this.onTouched = fn;
|
||||||
|
}
|
||||||
|
setDisabledState?(isDisabled: boolean): void {
|
||||||
|
this.disabled = isDisabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnInit(): void {
|
||||||
|
this.tagService.listAll().subscribe(result => {
|
||||||
|
this.tags = result.results
|
||||||
|
this.displayValue = this.value
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
disabled = false
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
hint
|
||||||
|
|
||||||
|
value: number[]
|
||||||
|
|
||||||
|
displayValue: number[] = []
|
||||||
|
|
||||||
|
tags: PaperlessTag[]
|
||||||
|
|
||||||
|
getTag(id) {
|
||||||
|
return this.tags.find(tag => tag.id == id)
|
||||||
|
}
|
||||||
|
|
||||||
|
removeTag(id) {
|
||||||
|
let index = this.displayValue.indexOf(id)
|
||||||
|
if (index > -1) {
|
||||||
|
this.displayValue.splice(index, 1)
|
||||||
|
this.onChange(this.displayValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addTag(id) {
|
||||||
|
let index = this.displayValue.indexOf(id)
|
||||||
|
if (index == -1) {
|
||||||
|
this.displayValue.push(id)
|
||||||
|
this.onChange(this.displayValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
createTag() {
|
||||||
|
var modal = this.modalService.open(TagEditDialogComponent, {backdrop: 'static'})
|
||||||
|
modal.componentInstance.dialogMode = 'create'
|
||||||
|
modal.componentInstance.success.subscribe(newTag => {
|
||||||
|
this.tagService.list().subscribe(tags => {
|
||||||
|
this.tags = tags.results
|
||||||
|
this.addTag(newTag.id)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -3,19 +3,19 @@
|
|||||||
<svg class="buttonicon" fill="currentColor">
|
<svg class="buttonicon" fill="currentColor">
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#trash" />
|
<use xlink:href="assets/bootstrap-icons.svg#trash" />
|
||||||
</svg>
|
</svg>
|
||||||
Delete
|
<span class="d-none d-lg-inline">Delete</span>
|
||||||
</button>
|
</button>
|
||||||
<a [href]="downloadUrl" class="btn btn-sm btn-outline-secondary mr-2">
|
<a [href]="downloadUrl" class="btn btn-sm btn-outline-secondary mr-2">
|
||||||
<svg class="buttonicon" fill="currentColor">
|
<svg class="buttonicon" fill="currentColor">
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#download" />
|
<use xlink:href="assets/bootstrap-icons.svg#download" />
|
||||||
</svg>
|
</svg>
|
||||||
Download
|
<span class="d-none d-lg-inline">Download</span>
|
||||||
</a>
|
</a>
|
||||||
<button type="button" class="btn btn-sm btn-outline-secondary" (click)="close()">
|
<button type="button" class="btn btn-sm btn-outline-secondary" (click)="close()">
|
||||||
<svg class="buttonicon" fill="currentColor">
|
<svg class="buttonicon" fill="currentColor">
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#x" />
|
<use xlink:href="assets/bootstrap-icons.svg#x" />
|
||||||
</svg>
|
</svg>
|
||||||
Close
|
<span class="d-none d-lg-inline">Close</span>
|
||||||
</button>
|
</button>
|
||||||
</app-page-header>
|
</app-page-header>
|
||||||
|
|
||||||
@ -32,16 +32,7 @@
|
|||||||
formControlName='archive_serial_number'>
|
formControlName='archive_serial_number'>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-row">
|
<app-input-date-time title="Date created" titleTime="Time created" formControlName="created"></app-input-date-time>
|
||||||
<div class="form-group col">
|
|
||||||
<label for="created_date">Date created</label>
|
|
||||||
<input type="date" class="form-control" id="created_date" formControlName='created_date'>
|
|
||||||
</div>
|
|
||||||
<div class="form-group col">
|
|
||||||
<label for="created_time">Time created</label>
|
|
||||||
<input type="time" class="form-control" id="created_time" formControlName='created_time'>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="content">Content</label>
|
<label for="content">Content</label>
|
||||||
@ -52,26 +43,9 @@
|
|||||||
|
|
||||||
<app-input-select [items]="documentTypes" title="Document type" formControlName="document_type_id" allowNull="true" (createNew)="createDocumentType()"></app-input-select>
|
<app-input-select [items]="documentTypes" title="Document type" formControlName="document_type_id" allowNull="true" (createNew)="createDocumentType()"></app-input-select>
|
||||||
|
|
||||||
<div class="form-group">
|
<app-input-tags formControlName="tags_id" title="Tags"></app-input-tags>
|
||||||
<label for="exampleFormControlTextarea1">Tags</label>
|
|
||||||
|
|
||||||
<div class="input-group">
|
<button type="button" class="btn btn-outline-secondary" (click)="discard()">Discard</button>
|
||||||
<select multiple class="form-control" id="tags" formControlName="tags_id">
|
|
||||||
<option *ngFor="let t of tags" [ngValue]="t.id">{{t.name}}</option>
|
|
||||||
</select>
|
|
||||||
|
|
||||||
<div class="input-group-append">
|
|
||||||
<button class="btn btn-outline-secondary" type="button" (click)="createTag()">
|
|
||||||
<svg class="buttonicon" fill="currentColor">
|
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#plus" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
<small class="form-text text-muted">Hold CTRL to (de)select multiple tags.</small>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
<button type="button" class="btn btn-outline-primary" (click)="saveEditNext()" *ngIf="hasNext()">Save & edit next</button>
|
<button type="button" class="btn btn-outline-primary" (click)="saveEditNext()" *ngIf="hasNext()">Save & edit next</button>
|
||||||
<button type="submit" class="btn btn-primary">Save</button>
|
<button type="submit" class="btn btn-primary">Save</button>
|
||||||
</form>
|
</form>
|
||||||
|
@ -33,13 +33,11 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
|
|
||||||
correspondents: PaperlessCorrespondent[]
|
correspondents: PaperlessCorrespondent[]
|
||||||
documentTypes: PaperlessDocumentType[]
|
documentTypes: PaperlessDocumentType[]
|
||||||
tags: PaperlessTag[]
|
|
||||||
|
|
||||||
documentForm: FormGroup = new FormGroup({
|
documentForm: FormGroup = new FormGroup({
|
||||||
title: new FormControl(''),
|
title: new FormControl(''),
|
||||||
content: new FormControl(''),
|
content: new FormControl(''),
|
||||||
created_date: new FormControl(),
|
created: new FormControl(),
|
||||||
created_time: new FormControl(),
|
|
||||||
correspondent_id: new FormControl(),
|
correspondent_id: new FormControl(),
|
||||||
document_type_id: new FormControl(),
|
document_type_id: new FormControl(),
|
||||||
archive_serial_number: new FormControl(),
|
archive_serial_number: new FormControl(),
|
||||||
@ -51,7 +49,6 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
private route: ActivatedRoute,
|
private route: ActivatedRoute,
|
||||||
private correspondentService: CorrespondentService,
|
private correspondentService: CorrespondentService,
|
||||||
private documentTypeService: DocumentTypeService,
|
private documentTypeService: DocumentTypeService,
|
||||||
private tagService: TagService,
|
|
||||||
private datePipe: DatePipe,
|
private datePipe: DatePipe,
|
||||||
private router: Router,
|
private router: Router,
|
||||||
private modalService: NgbModal,
|
private modalService: NgbModal,
|
||||||
@ -59,35 +56,33 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
private documentListViewService: DocumentListViewService) { }
|
private documentListViewService: DocumentListViewService) { }
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.correspondentService.list().subscribe(result => this.correspondents = result.results)
|
this.documentForm.valueChanges.subscribe(wow => {
|
||||||
this.documentTypeService.list().subscribe(result => this.documentTypes = result.results)
|
Object.assign(this.document, this.documentForm.value)
|
||||||
this.tagService.list().subscribe(result => this.tags = result.results)
|
})
|
||||||
|
|
||||||
|
this.correspondentService.listAll().subscribe(result => this.correspondents = result.results)
|
||||||
|
this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results)
|
||||||
|
|
||||||
this.route.paramMap.subscribe(paramMap => {
|
this.route.paramMap.subscribe(paramMap => {
|
||||||
this.documentId = +paramMap.get('id')
|
this.documentId = +paramMap.get('id')
|
||||||
this.previewUrl = this.documentsService.getPreviewUrl(this.documentId)
|
this.previewUrl = this.documentsService.getPreviewUrl(this.documentId)
|
||||||
this.downloadUrl = this.documentsService.getDownloadUrl(this.documentId)
|
this.downloadUrl = this.documentsService.getDownloadUrl(this.documentId)
|
||||||
this.documentsService.get(this.documentId).subscribe(doc => {
|
if (this.openDocumentService.getOpenDocument(this.documentId)) {
|
||||||
this.openDocumentService.openDocument(doc)
|
this.updateComponent(this.openDocumentService.getOpenDocument(this.documentId))
|
||||||
this.document = doc
|
} else {
|
||||||
this.title = doc.title
|
this.documentsService.get(this.documentId).subscribe(doc => {
|
||||||
this.documentForm.patchValue(doc)
|
this.openDocumentService.openDocument(doc)
|
||||||
this.documentForm.get('created_date').patchValue(this.datePipe.transform(doc.created, 'yyyy-MM-dd'))
|
this.updateComponent(doc)
|
||||||
this.documentForm.get('created_time').patchValue(this.datePipe.transform(doc.created, 'HH:mm:ss'))
|
}, error => {this.router.navigate(['404'])})
|
||||||
}, error => {this.router.navigate(['404'])})
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
createTag() {
|
updateComponent(doc: PaperlessDocument) {
|
||||||
var modal = this.modalService.open(TagEditDialogComponent, {backdrop: 'static'})
|
this.document = doc
|
||||||
modal.componentInstance.dialogMode = 'create'
|
this.title = doc.title
|
||||||
modal.componentInstance.success.subscribe(newTag => {
|
this.documentForm.patchValue(doc)
|
||||||
this.tagService.list().subscribe(tags => {
|
|
||||||
this.tags = tags.results
|
|
||||||
this.documentForm.get('tags_id').setValue(this.documentForm.get('tags_id').value.concat([newTag.id]))
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
createDocumentType() {
|
createDocumentType() {
|
||||||
@ -112,50 +107,22 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
getTag(id: number): PaperlessTag {
|
discard() {
|
||||||
return this.tags.find(tag => tag.id == id)
|
this.documentsService.get(this.documentId).subscribe(doc => {
|
||||||
}
|
Object.assign(this.document, doc)
|
||||||
|
this.title = doc.title
|
||||||
getColour(id: number) {
|
this.documentForm.patchValue(doc)
|
||||||
return TAG_COLOURS.find(c => c.id == this.getTag(id).colour)
|
}, error => {this.router.navigate(['404'])})
|
||||||
}
|
|
||||||
|
|
||||||
addTag(id: number) {
|
|
||||||
if (this.documentForm.value.tags.indexOf(id) == -1) {
|
|
||||||
this.documentForm.value.tags.push(id)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
removeTag(id: number) {
|
|
||||||
let index = this.documentForm.value.tags.indexOf(id)
|
|
||||||
if (index > -1) {
|
|
||||||
this.documentForm.value.tags.splice(index, 1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
getDateCreated() {
|
|
||||||
let newDate = this.documentForm.value.created_date
|
|
||||||
let newTime = this.documentForm.value.created_time
|
|
||||||
return formatDate(newDate + "T" + newTime,"yyyy-MM-ddTHH:mm:ssZZZZZ", "en-us", "UTC")
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
save() {
|
save() {
|
||||||
let newDocument = Object.assign(Object.assign({}, this.document), this.documentForm.value)
|
this.documentsService.update(this.document).subscribe(result => {
|
||||||
|
|
||||||
newDocument.created = this.getDateCreated()
|
|
||||||
|
|
||||||
this.documentsService.update(newDocument).subscribe(result => {
|
|
||||||
this.close()
|
this.close()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
saveEditNext() {
|
saveEditNext() {
|
||||||
let newDocument = Object.assign(Object.assign({}, this.document), this.documentForm.value)
|
this.documentsService.update(this.document).subscribe(result => {
|
||||||
|
|
||||||
newDocument.created = this.getDateCreated()
|
|
||||||
|
|
||||||
this.documentsService.update(newDocument).subscribe(result => {
|
|
||||||
this.documentListViewService.getNext(this.document.id).subscribe(nextDocId => {
|
this.documentListViewService.getNext(this.document.id).subscribe(nextDocId => {
|
||||||
if (nextDocId) {
|
if (nextDocId) {
|
||||||
this.openDocumentService.closeDocument(this.document)
|
this.openDocumentService.closeDocument(this.document)
|
||||||
|
@ -73,7 +73,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<ngb-pagination [pageSize]="25" [collectionSize]="docs.collectionSize" [(page)]="docs.currentPage" [maxSize]="5"
|
<ngb-pagination [pageSize]="docs.currentPageSize" [collectionSize]="docs.collectionSize" [(page)]="docs.currentPage" [maxSize]="5"
|
||||||
[rotate]="true" [boundaryLinks]="true" (pageChange)="reload()" aria-label="Default pagination"></ngb-pagination>
|
[rotate]="true" [boundaryLinks]="true" (pageChange)="reload()" aria-label="Default pagination"></ngb-pagination>
|
||||||
|
|
||||||
<div *ngIf="displayMode == 'largeCards'">
|
<div *ngIf="displayMode == 'largeCards'">
|
||||||
|
@ -1,8 +1,5 @@
|
|||||||
<div *ngFor="let rule of filterRules" class="form-row form-group">
|
<div *ngFor="let rule of filterRules" class="form-row form-group">
|
||||||
<div class="col-md-3 col-form-label">
|
<div class="col-md-3 col-form-label">
|
||||||
<!-- <select class="form-control form-control-sm" [(ngModel)]="rule.type" (change)="rule.value = null">
|
|
||||||
<option *ngFor="let ruleType of getRuleTypes()" [ngValue]="ruleType">{{ruleType.name}}</option>
|
|
||||||
</select> -->
|
|
||||||
<span>{{rule.type.name}}</span>
|
<span>{{rule.type.name}}</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="col">
|
<div class="col">
|
||||||
|
@ -52,9 +52,9 @@ export class FilterEditorComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.correspondentService.list().subscribe(result => {this.correspondents = result.results})
|
this.correspondentService.listAll().subscribe(result => {this.correspondents = result.results})
|
||||||
this.tagService.list().subscribe(result => this.tags = result.results)
|
this.tagService.listAll().subscribe(result => this.tags = result.results)
|
||||||
this.documentTypeService.list().subscribe(result => this.documentTypes = result.results)
|
this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results)
|
||||||
}
|
}
|
||||||
|
|
||||||
getRuleTypes() {
|
getRuleTypes() {
|
||||||
|
@ -3,37 +3,61 @@
|
|||||||
</app-page-header>
|
</app-page-header>
|
||||||
|
|
||||||
<!-- <p>items per page, documents per view type</p> -->
|
<!-- <p>items per page, documents per view type</p> -->
|
||||||
<ul ngbNav #nav="ngbNav" class="nav-tabs">
|
<form [formGroup]="settingsForm" (ngSubmit)="saveSettings()">
|
||||||
<li [ngbNavItem]="1">
|
|
||||||
<a ngbNavLink>Document List Settings</a>
|
|
||||||
<ng-template ngbNavContent>
|
|
||||||
</ng-template>
|
|
||||||
</li>
|
|
||||||
<li [ngbNavItem]="2">
|
|
||||||
<a ngbNavLink>Saved views</a>
|
|
||||||
<ng-template ngbNavContent>
|
|
||||||
|
|
||||||
<table class="table table-striped">
|
<ul ngbNav #nav="ngbNav" class="nav-tabs">
|
||||||
<thead>
|
<li [ngbNavItem]="1">
|
||||||
<tr>
|
<a ngbNavLink>General settings</a>
|
||||||
<th scope="col">Title</th>
|
<ng-template ngbNavContent>
|
||||||
<th scope="col">Show in dashboard</th>
|
|
||||||
<th scope="col">Show in sidebar</th>
|
|
||||||
<th scope="col">Actions</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr *ngFor="let config of savedViewConfigService.getConfigs()">
|
|
||||||
<td>{{ config.title }}</td>
|
|
||||||
<td>{{ config.showInDashboard }}</td>
|
|
||||||
<td>{{ config.showInSideBar }}</td>
|
|
||||||
<td><button type="button" class="btn btn-sm btn-outline-danger" (click)="deleteViewConfig(config)">Delete</button></td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
|
|
||||||
</ng-template>
|
<h4>Document list</h4>
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<div [ngbNavOutlet]="nav" class="mt-2"></div>
|
<div class="form-row form-group">
|
||||||
|
<div class="col-md-3 col-form-label">
|
||||||
|
<span>Items per page</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<select class="form-control" formControlName="documentListItemPerPage">
|
||||||
|
<option [ngValue]="10">10</option>
|
||||||
|
<option [ngValue]="25">25</option>
|
||||||
|
<option [ngValue]="50">50</option>
|
||||||
|
<option [ngValue]="100">100</option>
|
||||||
|
</select>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</ng-template>
|
||||||
|
</li>
|
||||||
|
<li [ngbNavItem]="2">
|
||||||
|
<a ngbNavLink>Saved views</a>
|
||||||
|
<ng-template ngbNavContent>
|
||||||
|
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th scope="col">Title</th>
|
||||||
|
<th scope="col">Show in dashboard</th>
|
||||||
|
<th scope="col">Show in sidebar</th>
|
||||||
|
<th scope="col">Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr *ngFor="let config of savedViewConfigService.getConfigs()">
|
||||||
|
<td>{{ config.title }}</td>
|
||||||
|
<td>{{ config.showInDashboard }}</td>
|
||||||
|
<td>{{ config.showInSideBar }}</td>
|
||||||
|
<td><button type="button" class="btn btn-sm btn-outline-danger" (click)="deleteViewConfig(config)">Delete</button></td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</ng-template>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div [ngbNavOutlet]="nav" class="mt-2"></div>
|
||||||
|
|
||||||
|
<button type="submit" class="btn btn-primary">Save</button>
|
||||||
|
</form>
|
@ -1,5 +1,8 @@
|
|||||||
import { Component, OnInit } from '@angular/core';
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
import { FormControl, FormGroup } from '@angular/forms';
|
||||||
import { SavedViewConfig } from 'src/app/data/saved-view-config';
|
import { SavedViewConfig } from 'src/app/data/saved-view-config';
|
||||||
|
import { GENERAL_SETTINGS } from 'src/app/data/storage-keys';
|
||||||
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { SavedViewConfigService } from 'src/app/services/saved-view-config.service';
|
import { SavedViewConfigService } from 'src/app/services/saved-view-config.service';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -9,11 +12,14 @@ import { SavedViewConfigService } from 'src/app/services/saved-view-config.servi
|
|||||||
})
|
})
|
||||||
export class SettingsComponent implements OnInit {
|
export class SettingsComponent implements OnInit {
|
||||||
|
|
||||||
constructor(
|
settingsForm = new FormGroup({
|
||||||
private savedViewConfigService: SavedViewConfigService
|
'documentListItemPerPage': new FormControl(+localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT)
|
||||||
) { }
|
})
|
||||||
|
|
||||||
active
|
constructor(
|
||||||
|
private savedViewConfigService: SavedViewConfigService,
|
||||||
|
private documentListViewService: DocumentListViewService
|
||||||
|
) { }
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
}
|
}
|
||||||
@ -22,4 +28,8 @@ export class SettingsComponent implements OnInit {
|
|||||||
this.savedViewConfigService.deleteConfig(config)
|
this.savedViewConfigService.deleteConfig(config)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
saveSettings() {
|
||||||
|
localStorage.setItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE, this.settingsForm.value.documentListItemPerPage)
|
||||||
|
this.documentListViewService.updatePageSize()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
8
src-ui/src/app/data/storage-keys.ts
Normal file
8
src-ui/src/app/data/storage-keys.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
export const OPEN_DOCUMENT_SERVICE = {
|
||||||
|
DOCUMENTS: 'open-documents-service:openDocuments'
|
||||||
|
}
|
||||||
|
|
||||||
|
export const GENERAL_SETTINGS = {
|
||||||
|
DOCUMENT_LIST_SIZE: 'general-settings:documentListSize',
|
||||||
|
DOCUMENT_LIST_SIZE_DEFAULT: 50
|
||||||
|
}
|
@ -3,6 +3,7 @@ import { Observable } from 'rxjs';
|
|||||||
import { cloneFilterRules, FilterRule } from '../data/filter-rule';
|
import { cloneFilterRules, FilterRule } from '../data/filter-rule';
|
||||||
import { PaperlessDocument } from '../data/paperless-document';
|
import { PaperlessDocument } from '../data/paperless-document';
|
||||||
import { SavedViewConfig } from '../data/saved-view-config';
|
import { SavedViewConfig } from '../data/saved-view-config';
|
||||||
|
import { GENERAL_SETTINGS } from '../data/storage-keys';
|
||||||
import { DocumentService, SORT_DIRECTION_DESCENDING } from './rest/document.service';
|
import { DocumentService, SORT_DIRECTION_DESCENDING } from './rest/document.service';
|
||||||
|
|
||||||
|
|
||||||
@ -15,6 +16,7 @@ export class DocumentListViewService {
|
|||||||
|
|
||||||
documents: PaperlessDocument[] = []
|
documents: PaperlessDocument[] = []
|
||||||
currentPage = 1
|
currentPage = 1
|
||||||
|
currentPageSize: number = +localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT
|
||||||
collectionSize: number
|
collectionSize: number
|
||||||
|
|
||||||
currentFilterRules: FilterRule[] = []
|
currentFilterRules: FilterRule[] = []
|
||||||
@ -39,7 +41,7 @@ export class DocumentListViewService {
|
|||||||
|
|
||||||
this.documentService.list(
|
this.documentService.list(
|
||||||
this.currentPage,
|
this.currentPage,
|
||||||
null,
|
this.currentPageSize,
|
||||||
sortField,
|
sortField,
|
||||||
sortDirection,
|
sortDirection,
|
||||||
filterRules).subscribe(
|
filterRules).subscribe(
|
||||||
@ -64,7 +66,7 @@ export class DocumentListViewService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
getLastPage(): number {
|
getLastPage(): number {
|
||||||
return Math.ceil(this.collectionSize / 25)
|
return Math.ceil(this.collectionSize / this.currentPageSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
hasNext(doc: number) {
|
hasNext(doc: number) {
|
||||||
@ -98,5 +100,13 @@ export class DocumentListViewService {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
updatePageSize() {
|
||||||
|
let newPageSize = +localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT
|
||||||
|
if (newPageSize != this.currentPageSize) {
|
||||||
|
this.currentPageSize = newPageSize
|
||||||
|
//this.reload()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
constructor(private documentService: DocumentService) { }
|
constructor(private documentService: DocumentService) { }
|
||||||
}
|
}
|
||||||
|
@ -1,26 +1,38 @@
|
|||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
import { Observable, Subject } from 'rxjs';
|
import { Observable, Subject } from 'rxjs';
|
||||||
import { PaperlessDocument } from '../data/paperless-document';
|
import { PaperlessDocument } from '../data/paperless-document';
|
||||||
|
import { OPEN_DOCUMENT_SERVICE } from '../data/storage-keys';
|
||||||
|
|
||||||
@Injectable({
|
@Injectable({
|
||||||
providedIn: 'root'
|
providedIn: 'root'
|
||||||
})
|
})
|
||||||
export class OpenDocumentsService {
|
export class OpenDocumentsService {
|
||||||
|
|
||||||
constructor() { }
|
constructor() {
|
||||||
|
if (sessionStorage.getItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS)) {
|
||||||
|
try {
|
||||||
|
this.openDocuments = JSON.parse(sessionStorage.getItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS))
|
||||||
|
} catch (e) {
|
||||||
|
sessionStorage.removeItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS)
|
||||||
|
this.openDocuments = []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private openDocuments: PaperlessDocument[] = []
|
private openDocuments: PaperlessDocument[] = []
|
||||||
|
|
||||||
private openDocumentsSubject: Subject<PaperlessDocument[]> = new Subject()
|
getOpenDocuments(): PaperlessDocument[] {
|
||||||
|
return this.openDocuments
|
||||||
|
}
|
||||||
|
|
||||||
getOpenDocuments(): Observable<PaperlessDocument[]> {
|
getOpenDocument(id: number): PaperlessDocument {
|
||||||
return this.openDocumentsSubject
|
return this.openDocuments.find(d => d.id == id)
|
||||||
}
|
}
|
||||||
|
|
||||||
openDocument(doc: PaperlessDocument) {
|
openDocument(doc: PaperlessDocument) {
|
||||||
if (this.openDocuments.find(d => d.id == doc.id) == null) {
|
if (this.openDocuments.find(d => d.id == doc.id) == null) {
|
||||||
this.openDocuments.push(doc)
|
this.openDocuments.push(doc)
|
||||||
this.openDocumentsSubject.next(this.openDocuments)
|
this.save()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -28,8 +40,12 @@ export class OpenDocumentsService {
|
|||||||
let index = this.openDocuments.findIndex(d => d.id == doc.id)
|
let index = this.openDocuments.findIndex(d => d.id == doc.id)
|
||||||
if (index > -1) {
|
if (index > -1) {
|
||||||
this.openDocuments.splice(index, 1)
|
this.openDocuments.splice(index, 1)
|
||||||
this.openDocumentsSubject.next(this.openDocuments)
|
this.save()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
save() {
|
||||||
|
sessionStorage.setItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS, JSON.stringify(this.openDocuments))
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -40,6 +40,10 @@ export abstract class AbstractPaperlessService<T extends ObjectWithId> {
|
|||||||
return this.http.get<Results<T>>(this.getResourceUrl(), {params: httpParams})
|
return this.http.get<Results<T>>(this.getResourceUrl(), {params: httpParams})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
listAll(ordering?: string, extraParams?): Observable<Results<T>> {
|
||||||
|
return this.list(1, 100000, ordering, extraParams)
|
||||||
|
}
|
||||||
|
|
||||||
get(id: number): Observable<T> {
|
get(id: number): Observable<T> {
|
||||||
return this.http.get<T>(this.getResourceUrl(id))
|
return this.http.get<T>(this.getResourceUrl(id))
|
||||||
}
|
}
|
||||||
|
@ -52,35 +52,35 @@ class IndexView(TemplateView):
|
|||||||
|
|
||||||
class CorrespondentViewSet(ModelViewSet):
|
class CorrespondentViewSet(ModelViewSet):
|
||||||
model = Correspondent
|
model = Correspondent
|
||||||
queryset = Correspondent.objects.annotate(document_count=Count('documents'), last_correspondence=Max('documents__created'))
|
queryset = Correspondent.objects.annotate(document_count=Count('documents'), last_correspondence=Max('documents__created')).order_by('name')
|
||||||
serializer_class = CorrespondentSerializer
|
serializer_class = CorrespondentSerializer
|
||||||
pagination_class = StandardPagination
|
pagination_class = StandardPagination
|
||||||
permission_classes = (IsAuthenticated,)
|
permission_classes = (IsAuthenticated,)
|
||||||
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
||||||
filter_class = CorrespondentFilterSet
|
filter_class = CorrespondentFilterSet
|
||||||
ordering_fields = ("name", "document_count", "last_correspondence")
|
ordering_fields = ("name", "matching_algorithm", "match", "document_count", "last_correspondence")
|
||||||
|
|
||||||
|
|
||||||
class TagViewSet(ModelViewSet):
|
class TagViewSet(ModelViewSet):
|
||||||
model = Tag
|
model = Tag
|
||||||
queryset = Tag.objects.annotate(document_count=Count('documents'))
|
queryset = Tag.objects.annotate(document_count=Count('documents')).order_by('name')
|
||||||
serializer_class = TagSerializer
|
serializer_class = TagSerializer
|
||||||
pagination_class = StandardPagination
|
pagination_class = StandardPagination
|
||||||
permission_classes = (IsAuthenticated,)
|
permission_classes = (IsAuthenticated,)
|
||||||
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
||||||
filter_class = TagFilterSet
|
filter_class = TagFilterSet
|
||||||
ordering_fields = ("name", "document_count")
|
ordering_fields = ("name", "matching_algorithm", "match", "document_count")
|
||||||
|
|
||||||
|
|
||||||
class DocumentTypeViewSet(ModelViewSet):
|
class DocumentTypeViewSet(ModelViewSet):
|
||||||
model = DocumentType
|
model = DocumentType
|
||||||
queryset = DocumentType.objects.annotate(document_count=Count('documents'))
|
queryset = DocumentType.objects.annotate(document_count=Count('documents')).order_by('name')
|
||||||
serializer_class = DocumentTypeSerializer
|
serializer_class = DocumentTypeSerializer
|
||||||
pagination_class = StandardPagination
|
pagination_class = StandardPagination
|
||||||
permission_classes = (IsAuthenticated,)
|
permission_classes = (IsAuthenticated,)
|
||||||
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
filter_backends = (DjangoFilterBackend, OrderingFilter)
|
||||||
filter_class = DocumentTypeFilterSet
|
filter_class = DocumentTypeFilterSet
|
||||||
ordering_fields = ("name", "document_count")
|
ordering_fields = ("name", "matching_algorithm", "match", "document_count")
|
||||||
|
|
||||||
|
|
||||||
class DocumentViewSet(RetrieveModelMixin,
|
class DocumentViewSet(RetrieveModelMixin,
|
||||||
@ -186,12 +186,10 @@ class SearchView(APIView):
|
|||||||
page = 1
|
page = 1
|
||||||
|
|
||||||
with self.ix.searcher() as searcher:
|
with self.ix.searcher() as searcher:
|
||||||
query_parser = QueryParser("content", self.ix.schema,
|
query_parser = QueryParser("content", self.ix.schema).parse(query)
|
||||||
termclass=terms.FuzzyTerm).parse(query)
|
|
||||||
result_page = searcher.search_page(query_parser, page)
|
result_page = searcher.search_page(query_parser, page)
|
||||||
result_page.results.fragmenter = highlight.ContextFragmenter(
|
result_page.results.fragmenter = highlight.ContextFragmenter(
|
||||||
surround=50)
|
surround=50)
|
||||||
result_page.results.fragmenter = highlight.PinpointFragmenter()
|
|
||||||
result_page.results.formatter = index.JsonFormatter()
|
result_page.results.formatter = index.JsonFormatter()
|
||||||
|
|
||||||
return Response(
|
return Response(
|
||||||
|
@ -125,7 +125,12 @@ TEMPLATES = [
|
|||||||
# NEVER RUN WITH DEBUG IN PRODUCTION.
|
# NEVER RUN WITH DEBUG IN PRODUCTION.
|
||||||
DEBUG = __get_boolean("PAPERLESS_DEBUG", "NO")
|
DEBUG = __get_boolean("PAPERLESS_DEBUG", "NO")
|
||||||
|
|
||||||
X_FRAME_OPTIONS = 'SAMEORIGIN'
|
if DEBUG:
|
||||||
|
X_FRAME_OPTIONS = ''
|
||||||
|
# this should really be 'allow-from uri' but its not supported in any mayor
|
||||||
|
# browser.
|
||||||
|
else:
|
||||||
|
X_FRAME_OPTIONS = 'SAMEORIGIN'
|
||||||
|
|
||||||
# We allow CORS from localhost:8080
|
# We allow CORS from localhost:8080
|
||||||
CORS_ORIGIN_WHITELIST = tuple(os.getenv("PAPERLESS_CORS_ALLOWED_HOSTS", "http://localhost:8080,https://localhost:8080").split(","))
|
CORS_ORIGIN_WHITELIST = tuple(os.getenv("PAPERLESS_CORS_ALLOWED_HOSTS", "http://localhost:8080,https://localhost:8080").split(","))
|
||||||
|
Loading…
x
Reference in New Issue
Block a user