mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Merge branch 'dev' into feature-bulk-edit
This commit is contained in:
commit
75b22e8684
25
README.md
25
README.md
@ -7,7 +7,7 @@
|
|||||||
|
|
||||||
[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](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-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, have a look at the changelog in the documentation.
|
||||||
|
|
||||||
This project is still in development and some things may not work as expected.
|
This project is still in development and some things may not work as expected.
|
||||||
|
|
||||||
@ -15,11 +15,13 @@ This project is still in development and some things may not work as expected.
|
|||||||
|
|
||||||
Paperless does not control your scanner, it only helps you deal with what your scanner produces.
|
Paperless does not control your scanner, it only helps you deal with what your scanner produces.
|
||||||
|
|
||||||
1. Buy a document scanner that can write to a place on your network. If you need some inspiration, have a look at the [scanner recommendations](https://paperless-ng.readthedocs.io/en/latest/scanners.html) page.
|
1. Buy a document scanner that can write to a place on your network. If you need some inspiration, have a look at the [scanner recommendations](https://paperless-ng.readthedocs.io/en/latest/scanners.html) page. Set it up to "scan to FTP" or something similar. It should be able to push scanned images to a server without you having to do anything. Of course if your scanner doesn't know how to automatically upload the file somewhere, you can always do that manually. Paperless doesn't care how the documents get into its local consumption directory.
|
||||||
2. Set it up to "scan to FTP" or something similar. It should be able to push scanned images to a server without you having to do anything. Of course if your scanner doesn't know how to automatically upload the file somewhere, you can always do that manually. Paperless doesn't care how the documents get into its local consumption directory.
|
|
||||||
3. Have the target server run the Paperless consumption script to OCR the file and index it into a local database.
|
- Alternatively, you can use any of the mobile scanning apps out there. We have an app that allows you to share documents with paperless, if you're on Android. See the section on affiliated projects.
|
||||||
4. Use the web frontend to sift through the database and find what you want.
|
|
||||||
5. Download the PDF you need/want via the web interface and do whatever you like with it. You can even print it and send it as if it's the original. In most cases, no one will care or notice.
|
2. Wait for paperless to process your files. OCR is expensive, and depending on the power of your machine, this might take a bit of time.
|
||||||
|
3. Use the web frontend to sift through the database and find what you want.
|
||||||
|
4. Download the PDF you need/want via the web interface and do whatever you like with it. You can even print it and send it as if it's the original. In most cases, no one will care or notice.
|
||||||
|
|
||||||
Here's what you get:
|
Here's what you get:
|
||||||
|
|
||||||
@ -39,7 +41,6 @@ Here's what you get:
|
|||||||
* When adding documents from mails, paperless can move these mails to a new folder, mark them as read, flag them or delete them.
|
* When adding documents from mails, paperless can move these mails to a new folder, mark them as read, flag them or delete them.
|
||||||
* Machine learning powered document matching.
|
* Machine learning powered document matching.
|
||||||
* Paperless learns from your documents and will be able to automatically assign tags, correspondents and types to documents once you've stored a few documents in paperless.
|
* Paperless learns from your documents and will be able to automatically assign tags, correspondents and types to documents once you've stored a few documents in paperless.
|
||||||
* We have a mobile app that offers a 'Share with paperless' option over at https://github.com/qcasey/paperless_share. You can use that in combination with any of the mobile scanning apps out there. It's still a little rough around the edges, but it works!
|
|
||||||
* A task processor that processes documents in parallel and also tells you when something goes wrong. On modern multi core systems, consumption is blazing fast.
|
* A task processor that processes documents in parallel and also tells you when something goes wrong. On modern multi core systems, consumption is blazing fast.
|
||||||
* Code cleanup in many, MANY areas. Some of the code from OG paperless was just overly complicated.
|
* Code cleanup in many, MANY areas. Some of the code from OG paperless was just overly complicated.
|
||||||
* More tests, more stability.
|
* More tests, more stability.
|
||||||
@ -78,7 +79,7 @@ The recommended way to deploy paperless is docker-compose. Don't clone the repos
|
|||||||
|
|
||||||
Read the [documentation](https://paperless-ng.readthedocs.io/en/latest/setup.html#installation) on how to get started.
|
Read the [documentation](https://paperless-ng.readthedocs.io/en/latest/setup.html#installation) on how to get started.
|
||||||
|
|
||||||
Alternatively, you can install the dependencies and setup apache and a database server yourself. The documenation has information about the individual components of paperless that you need to take care of.
|
Alternatively, you can install the dependencies and setup apache and a database server yourself. The documenation has a step by step guide on how to do it.
|
||||||
|
|
||||||
# Migrating to paperless-ng
|
# Migrating to paperless-ng
|
||||||
|
|
||||||
@ -102,13 +103,15 @@ If you want to implement something big: Please start a discussion about that in
|
|||||||
|
|
||||||
Paperless has been around a while now, and people are starting to build stuff on top of it. If you're one of those people, we can add your project to this list:
|
Paperless has been around a while now, and people are starting to build stuff on top of it. If you're one of those people, we can add your project to this list:
|
||||||
|
|
||||||
* [Paperless App](https://github.com/bauerj/paperless_app): An Android/iOS app for Paperless. We're working on making this compatible.
|
* [Paperless App](https://github.com/bauerj/paperless_app): An Android/iOS app for Paperless. Updated to work with paperless-ng.
|
||||||
|
* [Paperless Share](https://github.com/qcasey/paperless_share). Share any files from your Android application with paperless. Very simple, but works with all of the mobile scanning apps out there that allow you to share scanned documents.
|
||||||
|
|
||||||
|
These projects also exist, but their status and compatibility with paperless-ng is unknown.
|
||||||
|
|
||||||
* [Paperless Desktop](https://github.com/thomasbrueggemann/paperless-desktop): A desktop UI for your Paperless installation. Runs on Mac, Linux, and Windows.
|
* [Paperless Desktop](https://github.com/thomasbrueggemann/paperless-desktop): A desktop UI for your Paperless installation. Runs on Mac, Linux, and Windows.
|
||||||
* [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.
|
||||||
|
|
||||||
Compatibility with Paperless-ng is unknown.
|
|
||||||
|
|
||||||
# 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.
|
||||||
|
@ -15,7 +15,7 @@ services:
|
|||||||
POSTGRES_PASSWORD: paperless
|
POSTGRES_PASSWORD: paperless
|
||||||
|
|
||||||
webserver:
|
webserver:
|
||||||
image: jonaswinkler/paperless-ng:0.9.6
|
image: jonaswinkler/paperless-ng:0.9.8
|
||||||
restart: always
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
- db
|
- db
|
||||||
|
@ -5,7 +5,7 @@ services:
|
|||||||
restart: always
|
restart: always
|
||||||
|
|
||||||
webserver:
|
webserver:
|
||||||
image: jonaswinkler/paperless-ng:0.9.6
|
image: jonaswinkler/paperless-ng:0.9.8
|
||||||
restart: always
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
- broker
|
- broker
|
||||||
|
@ -263,10 +263,10 @@ using the identifier which it has assigned to each document. You will end up get
|
|||||||
files like ``0000123.pdf`` in your media directory. This isn't necessarily a bad
|
files like ``0000123.pdf`` in your media directory. This isn't necessarily a bad
|
||||||
thing, because you normally don't have to access these files manually. However, if
|
thing, because you normally don't have to access these files manually. However, if
|
||||||
you wish to name your files differently, you can do that by adjusting the
|
you wish to name your files differently, you can do that by adjusting the
|
||||||
``PAPERLESS_FILENAME_FORMAT`` settings variable.
|
``PAPERLESS_FILENAME_FORMAT`` configuration option.
|
||||||
|
|
||||||
This variable allows you to configure the filename (folders are allowed!) using
|
This variable allows you to configure the filename (folders are allowed) using
|
||||||
placeholders. For example, setting
|
placeholders. For example, configuring this to
|
||||||
|
|
||||||
.. code:: bash
|
.. code:: bash
|
||||||
|
|
||||||
@ -277,17 +277,16 @@ will create a directory structure as follows:
|
|||||||
.. code::
|
.. code::
|
||||||
|
|
||||||
2019/
|
2019/
|
||||||
my_bank/
|
My bank/
|
||||||
statement-january-0000001.pdf
|
Statement January.pdf
|
||||||
statement-february-0000002.pdf
|
Statement February.pdf
|
||||||
2020/
|
2020/
|
||||||
my_bank/
|
My bank/
|
||||||
statement-january-0000003.pdf
|
Statement January.pdf
|
||||||
shoe_store/
|
Letter.pdf
|
||||||
my_new_shoes-0000004.pdf
|
Letter_01.pdf
|
||||||
|
Shoe store/
|
||||||
Paperless appends the unique identifier of each document to the filename. This
|
My new shoes.pdf
|
||||||
avoids filename clashes.
|
|
||||||
|
|
||||||
.. danger::
|
.. danger::
|
||||||
|
|
||||||
@ -299,6 +298,7 @@ Paperless provides the following placeholders withing filenames:
|
|||||||
|
|
||||||
* ``{correspondent}``: The name of the correspondent, or "none".
|
* ``{correspondent}``: The name of the correspondent, or "none".
|
||||||
* ``{document_type}``: The name of the document type, or "none".
|
* ``{document_type}``: The name of the document type, or "none".
|
||||||
|
* ``{tag_list}``: A comma separated list of all tags assigned to the document.
|
||||||
* ``{title}``: The title of the document.
|
* ``{title}``: The title of the document.
|
||||||
* ``{created}``: The full date and time the document was created.
|
* ``{created}``: The full date and time the document was created.
|
||||||
* ``{created_year}``: Year created only.
|
* ``{created_year}``: Year created only.
|
||||||
@ -309,8 +309,14 @@ Paperless provides the following placeholders withing filenames:
|
|||||||
* ``{added_month}``: Month added only (number 1-12).
|
* ``{added_month}``: Month added only (number 1-12).
|
||||||
* ``{added_day}``: Day added only (number 1-31).
|
* ``{added_day}``: Day added only (number 1-31).
|
||||||
|
|
||||||
Paperless will convert all values for the placeholders into values which are safe
|
|
||||||
for use in filenames.
|
Paperless will try to conserve the information from your database as much as possible.
|
||||||
|
However, some characters that you can use in document titles and correspondent names (such
|
||||||
|
as ``: \ /`` and a couple more) are not allowed in filenames and will be replaced with dashes.
|
||||||
|
|
||||||
|
If paperless detects that two documents share the same filename, paperless will automatically
|
||||||
|
append ``_01``, ``_02``, etc to the filename. This happens if all the placeholders in a filename
|
||||||
|
evaluate to the same value.
|
||||||
|
|
||||||
.. hint::
|
.. hint::
|
||||||
|
|
||||||
|
@ -5,6 +5,58 @@
|
|||||||
Changelog
|
Changelog
|
||||||
*********
|
*********
|
||||||
|
|
||||||
|
|
||||||
|
paperless-ng 0.9.8
|
||||||
|
##################
|
||||||
|
|
||||||
|
This release addresses two severe issues with the previous release.
|
||||||
|
|
||||||
|
* The delete buttons for document types, correspondents and tags were not working.
|
||||||
|
* The document section in the admin was causing internal server errors (500).
|
||||||
|
|
||||||
|
|
||||||
|
paperless-ng 0.9.7
|
||||||
|
##################
|
||||||
|
|
||||||
|
|
||||||
|
* Front end
|
||||||
|
|
||||||
|
* Thanks to the hard work of `Michael Shamoon`_, paperless now comes with a much more streamlined UI for
|
||||||
|
filtering documents.
|
||||||
|
|
||||||
|
* `Michael Shamoon`_ replaced the document preview with another component. This should fix compatibility with Safari browsers.
|
||||||
|
|
||||||
|
* Added buttons to the management pages to quickly show all documents with one specific tag, correspondent, or title.
|
||||||
|
|
||||||
|
* Paperless now stores your saved views on the server and associates them with your user account.
|
||||||
|
This means that you can access your views on multiple devices and have separate views for different users.
|
||||||
|
You will have to recreate your views.
|
||||||
|
|
||||||
|
* The GitHub and documentation links now open in new tabs/windows. Thanks to `rYR79435`_.
|
||||||
|
|
||||||
|
* Paperless now generates default saved view names when saving views with certain filter rules.
|
||||||
|
|
||||||
|
* Added a small version indicator to the front end.
|
||||||
|
|
||||||
|
* Other additions and changes
|
||||||
|
|
||||||
|
* The new filename format field ``{tag_list}`` inserts a list of tags into the filename, separated by comma.
|
||||||
|
* The ``document_retagger`` no longer removes inbox tags or tags without matching rules.
|
||||||
|
* The new configuration option ``PAPERLESS_COOKIE_PREFIX`` allows you to run multiple instances of paperless on different ports.
|
||||||
|
This option enables you to be logged in into multiple instances by specifying different cookie names for each instance.
|
||||||
|
|
||||||
|
* Fixes
|
||||||
|
|
||||||
|
* Sometimes paperless would assign dates in the future to newly consumed documents.
|
||||||
|
* The filename format fields ``{created_month}`` and ``{created_day}`` now use a leading zero for single digit values.
|
||||||
|
* The filename format field ``{tags}`` can no longer be used without arguments.
|
||||||
|
* Paperless was not able to consume many images (especially images from mobile scanners) due to missing DPI information.
|
||||||
|
Paperless now assumes A4 paper size for PDF generation if no DPI information is present.
|
||||||
|
* Documents with empty titles could not be opened from the table view due to the link being empty.
|
||||||
|
* Fixed an issue with filenames containing special characters such as ``:`` not being accepted for upload.
|
||||||
|
* Fixed issues with thumbnail generation for plain text files.
|
||||||
|
|
||||||
|
|
||||||
paperless-ng 0.9.6
|
paperless-ng 0.9.6
|
||||||
##################
|
##################
|
||||||
|
|
||||||
@ -841,6 +893,8 @@ bulk of the work on this big change.
|
|||||||
|
|
||||||
* Initial release
|
* Initial release
|
||||||
|
|
||||||
|
.. _rYR79435: https://github.com/rYR79435
|
||||||
|
.. _Michael Shamoon: https://github.com/shamoon
|
||||||
.. _jayme-github: http://github.com/jayme-github
|
.. _jayme-github: http://github.com/jayme-github
|
||||||
.. _Brian Conn: https://github.com/TheConnMan
|
.. _Brian Conn: https://github.com/TheConnMan
|
||||||
.. _Christopher Luu: https://github.com/nuudles
|
.. _Christopher Luu: https://github.com/nuudles
|
||||||
|
@ -57,9 +57,6 @@ Adding documents to paperless
|
|||||||
#############################
|
#############################
|
||||||
|
|
||||||
Once you've got Paperless setup, you need to start feeding documents into it.
|
Once you've got Paperless setup, you need to start feeding documents into it.
|
||||||
Currently, there are four options: the consumption directory, the dashboard, IMAP (email), and
|
|
||||||
HTTP POST.
|
|
||||||
|
|
||||||
When adding documents to paperless, it will perform the following operations on
|
When adding documents to paperless, it will perform the following operations on
|
||||||
your documents:
|
your documents:
|
||||||
|
|
||||||
@ -112,6 +109,17 @@ Dashboard upload
|
|||||||
The dashboard has a file drop field to upload documents to paperless. Simply drag a file
|
The dashboard has a file drop field to upload documents to paperless. Simply drag a file
|
||||||
onto this field or select a file with the file dialog. Multiple files are supported.
|
onto this field or select a file with the file dialog. Multiple files are supported.
|
||||||
|
|
||||||
|
|
||||||
|
Mobile upload
|
||||||
|
=============
|
||||||
|
|
||||||
|
The mobile app over at `<https://github.com/qcasey/paperless_share>`_ allows Android users
|
||||||
|
to share any documents with paperless. This can be combined with any of the mobile
|
||||||
|
scanning apps out there, such as Office Lens.
|
||||||
|
|
||||||
|
Furthermore, there is the `Paperless App <https://github.com/bauerj/paperless_app>`_ as well,
|
||||||
|
which no only has document upload, but also document editing and browsing.
|
||||||
|
|
||||||
.. _usage-email:
|
.. _usage-email:
|
||||||
|
|
||||||
IMAP (Email)
|
IMAP (Email)
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
# adjust src/paperless/version.py
|
# adjust src/paperless/version.py
|
||||||
# changelog in the documentation
|
# changelog in the documentation
|
||||||
# adjust versions in docker/hub/*
|
# adjust versions in docker/hub/*
|
||||||
|
# adjust version in src-ui/src/environments/prod
|
||||||
# If docker-compose was modified: all compose files are the same.
|
# If docker-compose was modified: all compose files are the same.
|
||||||
|
|
||||||
# Steps:
|
# Steps:
|
||||||
|
@ -121,7 +121,8 @@ import { SelectDialogComponent } from './components/common/select-dialog/select-
|
|||||||
useClass: CsrfInterceptor,
|
useClass: CsrfInterceptor,
|
||||||
multi: true
|
multi: true
|
||||||
},
|
},
|
||||||
FilterPipe
|
FilterPipe,
|
||||||
|
DocumentTitlePipe
|
||||||
],
|
],
|
||||||
bootstrap: [AppComponent]
|
bootstrap: [AppComponent]
|
||||||
})
|
})
|
||||||
|
@ -17,6 +17,11 @@
|
|||||||
<div class="container-fluid">
|
<div class="container-fluid">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<nav id="sidebarMenu" class="col-md-3 col-lg-2 d-md-block bg-light sidebar collapse" [ngbCollapse]="isMenuCollapsed">
|
<nav id="sidebarMenu" class="col-md-3 col-lg-2 d-md-block bg-light sidebar collapse" [ngbCollapse]="isMenuCollapsed">
|
||||||
|
|
||||||
|
<div style="position: absolute; bottom: 0; left: 0;" class="text-muted p-1">
|
||||||
|
{{versionString}}
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="sidebar-sticky pt-3">
|
<div class="sidebar-sticky pt-3">
|
||||||
<ul class="nav flex-column">
|
<ul class="nav flex-column">
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
@ -60,7 +65,7 @@
|
|||||||
<svg class="sidebaricon" fill="currentColor">
|
<svg class="sidebaricon" fill="currentColor">
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#file-text"/>
|
<use xlink:href="assets/bootstrap-icons.svg#file-text"/>
|
||||||
</svg>
|
</svg>
|
||||||
{{d.title}}
|
{{d.title | documentTitle}}
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
<li class="nav-item w-100" *ngIf="openDocuments.length > 1">
|
<li class="nav-item w-100" *ngIf="openDocuments.length > 1">
|
||||||
|
@ -7,6 +7,7 @@ import { PaperlessDocument } from 'src/app/data/paperless-document';
|
|||||||
import { OpenDocumentsService } from 'src/app/services/open-documents.service';
|
import { OpenDocumentsService } from 'src/app/services/open-documents.service';
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service';
|
import { SavedViewService } from 'src/app/services/rest/saved-view.service';
|
||||||
import { SearchService } from 'src/app/services/rest/search.service';
|
import { SearchService } from 'src/app/services/rest/search.service';
|
||||||
|
import { environment } from 'src/environments/environment';
|
||||||
import { DocumentDetailComponent } from '../document-detail/document-detail.component';
|
import { DocumentDetailComponent } from '../document-detail/document-detail.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -25,6 +26,8 @@ export class AppFrameComponent implements OnInit, OnDestroy {
|
|||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
versionString = `${environment.appTitle} ${environment.version}`
|
||||||
|
|
||||||
isMenuCollapsed: boolean = true
|
isMenuCollapsed: boolean = true
|
||||||
|
|
||||||
closeMenu() {
|
closeMenu() {
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
<tr *ngFor="let doc of documents" routerLink="/documents/{{doc.id}}">
|
<tr *ngFor="let doc of documents" routerLink="/documents/{{doc.id}}">
|
||||||
<td>{{doc.created | date}}</td>
|
<td>{{doc.created | date}}</td>
|
||||||
<td>{{doc.title}}<app-tag [tag]="t" *ngFor="let t of doc.tags$ | async" class="ml-1"></app-tag>
|
<td>{{doc.title | documentTitle}}<app-tag [tag]="t" *ngFor="let t of doc.tags$ | async" class="ml-1"></app-tag>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
@ -110,8 +110,8 @@
|
|||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
<app-metadata-collapse title="Original document metadata" [metadata]="metadata.original_metadata" *ngIf="metadata?.original_metadata.length > 0"></app-metadata-collapse>
|
<app-metadata-collapse title="Original document metadata" [metadata]="metadata.original_metadata" *ngIf="metadata?.original_metadata?.length > 0"></app-metadata-collapse>
|
||||||
<app-metadata-collapse title="Archived document metadata" [metadata]="metadata.archive_metadata" *ngIf="metadata?.archive_metadata.length > 0"></app-metadata-collapse>
|
<app-metadata-collapse title="Archived document metadata" [metadata]="metadata.archive_metadata" *ngIf="metadata?.archive_metadata?.length > 0"></app-metadata-collapse>
|
||||||
|
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
|
@ -6,6 +6,7 @@ import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent';
|
|||||||
import { PaperlessDocument } from 'src/app/data/paperless-document';
|
import { PaperlessDocument } from 'src/app/data/paperless-document';
|
||||||
import { PaperlessDocumentMetadata } from 'src/app/data/paperless-document-metadata';
|
import { PaperlessDocumentMetadata } from 'src/app/data/paperless-document-metadata';
|
||||||
import { PaperlessDocumentType } from 'src/app/data/paperless-document-type';
|
import { PaperlessDocumentType } from 'src/app/data/paperless-document-type';
|
||||||
|
import { DocumentTitlePipe } from 'src/app/pipes/document-title.pipe';
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { OpenDocumentsService } from 'src/app/services/open-documents.service';
|
import { OpenDocumentsService } from 'src/app/services/open-documents.service';
|
||||||
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
||||||
@ -54,7 +55,8 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
private router: Router,
|
private router: Router,
|
||||||
private modalService: NgbModal,
|
private modalService: NgbModal,
|
||||||
private openDocumentService: OpenDocumentsService,
|
private openDocumentService: OpenDocumentsService,
|
||||||
private documentListViewService: DocumentListViewService) { }
|
private documentListViewService: DocumentListViewService,
|
||||||
|
private documentTitlePipe: DocumentTitlePipe) { }
|
||||||
|
|
||||||
getContentType() {
|
getContentType() {
|
||||||
return this.metadata?.has_archive_version ? 'application/pdf' : this.metadata?.original_mime_type
|
return this.metadata?.has_archive_version ? 'application/pdf' : this.metadata?.original_mime_type
|
||||||
@ -90,7 +92,7 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
this.documentsService.getMetadata(doc.id).subscribe(result => {
|
this.documentsService.getMetadata(doc.id).subscribe(result => {
|
||||||
this.metadata = result
|
this.metadata = result
|
||||||
})
|
})
|
||||||
this.title = doc.title
|
this.title = this.documentTitlePipe.transform(doc.title)
|
||||||
this.documentForm.patchValue(doc)
|
this.documentForm.patchValue(doc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
.result-content {
|
.result-content {
|
||||||
color: darkgray;
|
color: darkgray;
|
||||||
|
overflow-wrap: anywhere;
|
||||||
}
|
}
|
||||||
|
|
||||||
.doc-img {
|
.doc-img {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
<div class="col p-2 h-100 document-card" style="width: 16rem;">
|
<div class="col p-2 h-100 document-card">
|
||||||
<div class="card h-100 shadow-sm" [class.card-selected]="selected">
|
<div class="card h-100 shadow-sm" [class.card-selected]="selected">
|
||||||
<div class="border-bottom" [class.doc-img-background-selected]="selected">
|
<div class="border-bottom" [class.doc-img-background-selected]="selected">
|
||||||
<img class="card-img doc-img" [src]="getThumbUrl()" (click)="selected = !selected">
|
<img class="card-img doc-img" [src]="getThumbUrl()" (click)="selected = !selected">
|
||||||
@ -10,7 +10,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
<div style="top: 0; right: 0; font-size: large" class="text-right position-absolute mr-1">
|
<div style="top: 0; right: 0; font-size: large" class="text-right position-absolute mr-1">
|
||||||
<div *ngFor="let t of getTagsLimited$() | async">
|
<div *ngFor="let t of getTagsLimited$() | async">
|
||||||
<app-tag [tag]="t" (click)="clickTag.emit(t.id)" [clickable]="true" linkTitle="Filter by tag"></app-tag>
|
<app-tag [tag]="t" (click)="clickTag.emit(t.id)" [clickable]="true" linkTitle="Filter by tag"></app-tag>
|
||||||
@ -31,7 +30,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="card-footer">
|
<div class="card-footer">
|
||||||
|
|
||||||
<div class="d-flex justify-content-between align-items-center ml-n2">
|
<div class="d-flex justify-content-between align-items-center mx-n2">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary" title="Edit">
|
<a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary" title="Edit">
|
||||||
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-pencil" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-pencil" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
@ -51,7 +50,7 @@
|
|||||||
</svg>
|
</svg>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<small class="text-muted">{{document.created | date}}</small>
|
<small class="text-muted pl-1">{{document.created | date}}</small>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -86,7 +86,7 @@
|
|||||||
|
|
||||||
</app-page-header>
|
</app-page-header>
|
||||||
|
|
||||||
<div class="w-100 mb-4">
|
<div class="w-100 mb-2 mb-sm-4">
|
||||||
<app-filter-editor [(filterRules)]="list.filterRules" #filterEditor></app-filter-editor>
|
<app-filter-editor [(filterRules)]="list.filterRules" #filterEditor></app-filter-editor>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -129,7 +129,7 @@
|
|||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<a routerLink="/documents/{{d.id}}" title="Edit document" style="overflow-wrap: anywhere;">{{d.title | documentTitle}}</a>
|
<a routerLink="/documents/{{d.id}}" title="Edit document" style="overflow-wrap: anywhere;">{{d.title | documentTitle}}</a>
|
||||||
<app-tag [tag]="t" *ngFor="let t of d.tags$ | async" class="ml-1" clickable="true" linkTitle="Filter by tag" (click)="clickTag(t)"></app-tag>
|
<app-tag [tag]="t" *ngFor="let t of d.tags$ | async" class="ml-1" clickable="true" linkTitle="Filter by tag" (click)="clickTag(t.id)"></app-tag>
|
||||||
</td>
|
</td>
|
||||||
<td class="d-none d-xl-table-cell">
|
<td class="d-none d-xl-table-cell">
|
||||||
<ng-container *ngIf="d.document_type">
|
<ng-container *ngIf="d.document_type">
|
||||||
@ -147,6 +147,6 @@
|
|||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
|
||||||
<div class=" m-n2 row" *ngIf="displayMode == 'smallCards'">
|
<div class="m-n2 row row-cols-paperless-cards" *ngIf="displayMode == 'smallCards'">
|
||||||
<app-document-card-small [selected]="list.isSelected(d)" (selectedChange)="list.setSelected(d, $event)" [document]="d" *ngFor="let d of list.documents" (clickTag)="clickTag($event)" (clickCorrespondent)="clickCorrespondent($event)"></app-document-card-small>
|
<app-document-card-small [selected]="list.isSelected(d)" (selectedChange)="list.setSelected(d, $event)" [document]="d" *ngFor="let d of list.documents" (clickTag)="clickTag($event)" (clickCorrespondent)="clickCorrespondent($event)"></app-document-card-small>
|
||||||
</div>
|
</div>
|
||||||
|
@ -2,4 +2,26 @@
|
|||||||
|
|
||||||
.table-row-selected {
|
.table-row-selected {
|
||||||
background-color: $primaryFaded;
|
background-color: $primaryFaded;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$paperless-card-breakpoints: (
|
||||||
|
0: 2, // xs
|
||||||
|
768px: 3, //md
|
||||||
|
992px: 4, //lg
|
||||||
|
1200px: 5, //xl
|
||||||
|
1400px: 6, // xxl
|
||||||
|
1600px: 7,
|
||||||
|
1800px: 8,
|
||||||
|
2000px: 9
|
||||||
|
);
|
||||||
|
|
||||||
|
.row-cols-paperless-cards {
|
||||||
|
@each $width, $n_cols in $paperless-card-breakpoints {
|
||||||
|
@media(min-width: $width) {
|
||||||
|
> * {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
width: 100% / $n_cols;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -3,6 +3,7 @@ import { ActivatedRoute, Router } from '@angular/router';
|
|||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
import { Observable } from 'rxjs';
|
import { Observable } from 'rxjs';
|
||||||
import { map } from 'rxjs/operators';
|
import { map } from 'rxjs/operators';
|
||||||
|
import { FILTER_CORRESPONDENT } from 'src/app/data/filter-rule-type';
|
||||||
import { PaperlessSavedView } from 'src/app/data/paperless-saved-view';
|
import { PaperlessSavedView } from 'src/app/data/paperless-saved-view';
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
||||||
@ -94,6 +95,7 @@ export class DocumentListComponent implements OnInit {
|
|||||||
|
|
||||||
saveViewConfigAs() {
|
saveViewConfigAs() {
|
||||||
let modal = this.modalService.open(SaveViewConfigDialogComponent, {backdrop: 'static'})
|
let modal = this.modalService.open(SaveViewConfigDialogComponent, {backdrop: 'static'})
|
||||||
|
modal.componentInstance.defaultName = this.filterEditor.generateFilterName()
|
||||||
modal.componentInstance.saveClicked.subscribe(formValue => {
|
modal.componentInstance.saveClicked.subscribe(formValue => {
|
||||||
let savedView = {
|
let savedView = {
|
||||||
name: formValue.name,
|
name: formValue.name,
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Component, EventEmitter, OnInit, Output } from '@angular/core';
|
import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';
|
||||||
import { FormControl, FormGroup } from '@angular/forms';
|
import { FormControl, FormGroup } from '@angular/forms';
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
|
||||||
@ -14,6 +14,19 @@ export class SaveViewConfigDialogComponent implements OnInit {
|
|||||||
@Output()
|
@Output()
|
||||||
public saveClicked = new EventEmitter()
|
public saveClicked = new EventEmitter()
|
||||||
|
|
||||||
|
_defaultName = ""
|
||||||
|
|
||||||
|
get defaultName() {
|
||||||
|
return this._defaultName
|
||||||
|
}
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
set defaultName(value: string) {
|
||||||
|
this._defaultName = value
|
||||||
|
this.saveViewConfigForm.patchValue({name: value})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
saveViewConfigForm = new FormGroup({
|
saveViewConfigForm = new FormGroup({
|
||||||
name: new FormControl(''),
|
name: new FormControl(''),
|
||||||
showInSideBar: new FormControl(false),
|
showInSideBar: new FormControl(false),
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import { Component, EventEmitter, Input, Output, ElementRef, ViewChild, SimpleChange } from '@angular/core';
|
import { Component, EventEmitter, Input, Output, ElementRef, ViewChild, SimpleChange } from '@angular/core';
|
||||||
import { NgbDate, NgbDateStruct, NgbDatepicker } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbDate, NgbDateStruct, NgbDatepicker } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
|
||||||
|
|
||||||
export interface DateSelection {
|
export interface DateSelection {
|
||||||
before?: NgbDateStruct
|
before?: NgbDateStruct
|
||||||
after?: NgbDateStruct
|
after?: NgbDateStruct
|
||||||
@ -72,7 +71,6 @@ export class FilterDropdownDateComponent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
setDateQuickFilter(range: any) {
|
setDateQuickFilter(range: any) {
|
||||||
this._dateAfter = this._dateBefore = undefined
|
|
||||||
let date = new Date()
|
let date = new Date()
|
||||||
let newDate: NgbDateStruct = { year: date.getFullYear(), month: date.getMonth() + 1, day: date.getDate() }
|
let newDate: NgbDateStruct = { year: date.getFullYear(), month: date.getMonth() + 1, day: date.getDate() }
|
||||||
switch (typeof range) {
|
switch (typeof range) {
|
||||||
@ -92,18 +90,23 @@ export class FilterDropdownDateComponent {
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
this._dateAfter = newDate
|
this._dateAfter = newDate
|
||||||
|
this._dateBefore = null
|
||||||
this.datesSet.emit({after: newDate, before: null})
|
this.datesSet.emit({after: newDate, before: null})
|
||||||
}
|
}
|
||||||
|
|
||||||
onBeforeSelected(date: NgbDateStruct) {
|
onBeforeSelected(date: NgbDateStruct) {
|
||||||
|
this._dateBefore = date
|
||||||
this.datesSet.emit({after: this._dateAfter, before: date})
|
this.datesSet.emit({after: this._dateAfter, before: date})
|
||||||
}
|
}
|
||||||
|
|
||||||
onAfterSelected(date: NgbDateStruct) {
|
onAfterSelected(date: NgbDateStruct) {
|
||||||
|
this._dateAfter = date
|
||||||
this.datesSet.emit({after: date, before: this._dateBefore})
|
this.datesSet.emit({after: date, before: this._dateBefore})
|
||||||
}
|
}
|
||||||
|
|
||||||
clear() {
|
clear() {
|
||||||
|
this._dateBefore = null
|
||||||
|
this._dateAfter = null
|
||||||
this.datesSet.emit({after: null, before: null})
|
this.datesSet.emit({after: null, before: null})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,16 @@
|
|||||||
<div class="btn-group" ngbDropdown role="group" (openChange)="dropdownOpenChange($event)" #filterDropdown="ngbDropdown">
|
<div class="btn-group" ngbDropdown role="group" (openChange)="dropdownOpenChange($event)" #filterDropdown="ngbDropdown">
|
||||||
<button class="btn btn-sm" id="dropdown{{title}}" ngbDropdownToggle [ngClass]="itemsSelected?.length > 0 ? 'btn-primary' : 'btn-outline-primary'">
|
<button class="btn btn-sm" id="dropdown{{title}}" ngbDropdownToggle [ngClass]="itemsSelected?.length > 0 ? 'btn-primary' : 'btn-outline-primary'">
|
||||||
{{title}}
|
<div class="d-none d-md-inline">{{title}}</div>
|
||||||
|
<div class="d-inline-block d-md-none">
|
||||||
|
<svg class="toolbaricon" fill="currentColor">
|
||||||
|
<use attr.xlink:href="assets/bootstrap-icons.svg#{{icon}}" />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<ng-container *ngIf="itemsSelected?.length > 0">
|
||||||
|
<div class="badge bg-secondary text-light rounded-pill badge-corner">
|
||||||
|
{{itemsSelected?.length}}
|
||||||
|
</div>
|
||||||
|
</ng-container>
|
||||||
</button>
|
</button>
|
||||||
<div class="dropdown-menu py-0 shadow" ngbDropdownMenu attr.aria-labelledby="dropdown{{title}}">
|
<div class="dropdown-menu py-0 shadow" ngbDropdownMenu attr.aria-labelledby="dropdown{{title}}">
|
||||||
<div class="list-group list-group-flush">
|
<div class="list-group list-group-flush">
|
||||||
|
@ -1,3 +1,9 @@
|
|||||||
|
.badge-corner {
|
||||||
|
position: absolute;
|
||||||
|
top: -8px;
|
||||||
|
right: -8px;
|
||||||
|
}
|
||||||
|
|
||||||
.dropdown-menu {
|
.dropdown-menu {
|
||||||
min-width: 250px;
|
min-width: 250px;
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ export class FilterDropdownComponent {
|
|||||||
title: string
|
title: string
|
||||||
|
|
||||||
@Input()
|
@Input()
|
||||||
display: string
|
icon: string
|
||||||
|
|
||||||
@Output()
|
@Output()
|
||||||
toggle = new EventEmitter()
|
toggle = new EventEmitter()
|
||||||
|
@ -1,22 +1,27 @@
|
|||||||
<div class="form-row form-group mb-0">
|
<div class="row">
|
||||||
<div class="col-auto">
|
<div class="col mb-2 mb-xl-0">
|
||||||
<div class="text-muted mt-1">Filter by:</div>
|
<div class="form-inline d-flex">
|
||||||
|
<label class="text-muted mr-2">Filter by:</label>
|
||||||
|
<input class="form-control form-control-sm flex-grow-1" type="text" [(ngModel)]="titleFilter" placeholder="Title">
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="col">
|
<div class="w-100 d-xl-none"></div>
|
||||||
<input class="form-control form-control-sm" type="text" [(ngModel)]="titleFilter" placeholder="Title">
|
<div class="col col-xl-auto mb-2 mb-xl-0">
|
||||||
</div>
|
<div class="d-flex">
|
||||||
|
<app-filter-dropdown class="mr-2 mr-md-3" [items]="tags" [itemsSelected]="selectedTags" title="Tags" icon="tag-fill" (toggle)="toggleTag($event.id)"></app-filter-dropdown>
|
||||||
<app-filter-dropdown class="col-auto" [items]="tags" [itemsSelected]="selectedTags" title="Tags" (toggle)="toggleTag($event.id)"></app-filter-dropdown>
|
<app-filter-dropdown class="mr-2 mr-md-3" [items]="correspondents" [itemsSelected]="selectedCorrespondents" title="Correspondents" icon="person-fill" (toggle)="toggleCorrespondent($event.id)"></app-filter-dropdown>
|
||||||
<app-filter-dropdown class="col-auto" [items]="correspondents" [itemsSelected]="selectedCorrespondents" title="Correspondents" (toggle)="toggleCorrespondent($event.id)"></app-filter-dropdown>
|
<app-filter-dropdown class="mr-2 mr-md-3" [items]="documentTypes" [itemsSelected]="selectedDocumentTypes" title="Document types" icon="file-earmark-fill" (toggle)="toggleDocumentType($event.id)"></app-filter-dropdown>
|
||||||
<app-filter-dropdown class="col-auto" [items]="documentTypes" [itemsSelected]="selectedDocumentTypes" title="Document types" (toggle)="toggleDocumentType($event.id)"></app-filter-dropdown>
|
<app-filter-dropdown-date class="mr-2 mr-md-3" [dateBefore]="dateCreatedBefore" [dateAfter]="dateCreatedAfter" title="Created" (datesSet)="onDatesCreatedSet($event)"></app-filter-dropdown-date>
|
||||||
|
<app-filter-dropdown-date [dateBefore]="dateAddedBefore" [dateAfter]="dateAddedAfter" title="Added" (datesSet)="onDatesAddedSet($event)"></app-filter-dropdown-date>
|
||||||
<app-filter-dropdown-date class="col-auto" [dateBefore]="dateCreatedBefore" [dateAfter]="dateCreatedAfter" title="Created" (datesSet)="onDatesCreatedSet($event)"></app-filter-dropdown-date>
|
</div>
|
||||||
<app-filter-dropdown-date class="col-auto" [dateBefore]="dateAddedBefore" [dateAfter]="dateAddedAfter" title="Added" (datesSet)="onDatesAddedSet($event)"></app-filter-dropdown-date>
|
</div>
|
||||||
|
<div class="w-100 d-xl-none"></div>
|
||||||
<button class="btn btn-link btn-sm" [disabled]="!hasFilters()" (click)="clearSelected()">
|
<div class="col col-xl-auto mb-2 mb-xl-0">
|
||||||
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-x" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
<button class="btn btn-link btn-sm px-0 mx-0 ml-xl-n4" [disabled]="!hasFilters()" (click)="clearSelected()">
|
||||||
<path fill-rule="evenodd" d="M4.646 4.646a.5.5 0 0 1 .708 0L8 7.293l2.646-2.647a.5.5 0 0 1 .708.708L8.707 8l2.647 2.646a.5.5 0 0 1-.708.708L8 8.707l-2.646 2.647a.5.5 0 0 1-.708-.708L7.293 8 4.646 5.354a.5.5 0 0 1 0-.708z"/>
|
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-x" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
</svg>
|
<path fill-rule="evenodd" d="M4.646 4.646a.5.5 0 0 1 .708 0L8 7.293l2.646-2.647a.5.5 0 0 1 .708.708L8.707 8l2.647 2.646a.5.5 0 0 1-.708.708L8 8.707l-2.646 2.647a.5.5 0 0 1-.708-.708L7.293 8 4.646 5.354a.5.5 0 0 1 0-.708z"/>
|
||||||
Clear all filters
|
</svg>
|
||||||
</button>
|
Clear all filters
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -19,6 +19,26 @@ import { DateSelection } from './filter-dropdown-date/filter-dropdown-date.compo
|
|||||||
})
|
})
|
||||||
export class FilterEditorComponent implements OnInit, OnDestroy {
|
export class FilterEditorComponent implements OnInit, OnDestroy {
|
||||||
|
|
||||||
|
generateFilterName() {
|
||||||
|
if (this.filterRules.length == 1) {
|
||||||
|
let rule = this.filterRules[0]
|
||||||
|
switch(this.filterRules[0].rule_type) {
|
||||||
|
|
||||||
|
case FILTER_CORRESPONDENT:
|
||||||
|
return `Correspondent: ${this.correspondents.find(c => c.id == +rule.value)?.name}`
|
||||||
|
|
||||||
|
case FILTER_DOCUMENT_TYPE:
|
||||||
|
return `Type: ${this.documentTypes.find(dt => dt.id == +rule.value)?.name}`
|
||||||
|
|
||||||
|
case FILTER_HAS_TAG:
|
||||||
|
return `Tag: ${this.tags.find(t => t.id == +rule.value)?.name}`
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private documentTypeService: DocumentTypeService,
|
private documentTypeService: DocumentTypeService,
|
||||||
private tagService: TagService,
|
private tagService: TagService,
|
||||||
@ -200,24 +220,21 @@ export class FilterEditorComponent implements OnInit, OnDestroy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
setDateFilter(date: NgbDateStruct, dateRuleTypeID: number) {
|
setDateFilter(date: NgbDateStruct, dateRuleTypeID: number) {
|
||||||
let filterRules = this.filterRules
|
let existingRule = this.filterRules.find(rule => rule.rule_type == dateRuleTypeID)
|
||||||
let existingRule = filterRules.find(rule => rule.rule_type == dateRuleTypeID)
|
|
||||||
let newValue = this.dateParser.format(date)
|
let newValue = this.dateParser.format(date)
|
||||||
|
|
||||||
if (existingRule) {
|
if (existingRule) {
|
||||||
existingRule.value = newValue
|
existingRule.value = newValue
|
||||||
} else {
|
} else {
|
||||||
filterRules.push({rule_type: dateRuleTypeID, value: newValue})
|
this.filterRules.push({rule_type: dateRuleTypeID, value: newValue})
|
||||||
}
|
}
|
||||||
|
|
||||||
this.filterRules = filterRules
|
|
||||||
}
|
}
|
||||||
|
|
||||||
clearDateFilter(dateRuleTypeID: number) {
|
clearDateFilter(dateRuleTypeID: number) {
|
||||||
let filterRules = this.filterRules
|
let ruleIndex = this.filterRules.findIndex(rule => rule.rule_type == dateRuleTypeID)
|
||||||
let existingRule = filterRules.find(rule => rule.rule_type == dateRuleTypeID)
|
if (ruleIndex != -1) {
|
||||||
filterRules.splice(filterRules.indexOf(existingRule), 1)
|
this.filterRules.splice(ruleIndex, 1)
|
||||||
this.filterRules = filterRules
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -26,9 +26,26 @@
|
|||||||
<td scope="row">{{ correspondent.last_correspondence | date }}</td>
|
<td scope="row">{{ correspondent.last_correspondence | date }}</td>
|
||||||
<td scope="row">
|
<td scope="row">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(correspondent)">Edit</button>
|
<button class="btn btn-sm btn-outline-secondary" (click)="filterDocuments(correspondent)">
|
||||||
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(correspondent)">Delete</button>
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-funnel" viewBox="0 0 16 16">
|
||||||
</div>
|
<path fill-rule="evenodd" d="M1.5 1.5A.5.5 0 0 1 2 1h12a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-.128.334L10 8.692V13.5a.5.5 0 0 1-.342.474l-3 1A.5.5 0 0 1 6 14.5V8.692L1.628 3.834A.5.5 0 0 1 1.5 3.5v-2zm1 .5v1.308l4.372 4.858A.5.5 0 0 1 7 8.5v5.306l2-.666V8.5a.5.5 0 0 1 .128-.334L13.5 3.308V2h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Documents
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(correspondent)">
|
||||||
|
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-pencil" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168l10-10zM11.207 2.5L13.5 4.793 14.793 3.5 12.5 1.207 11.207 2.5zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293l6.5-6.5zm-9.761 5.175l-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325z"/>
|
||||||
|
</svg>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(correspondent)">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-trash" viewBox="0 0 16 16">
|
||||||
|
<path d="M5.5 5.5A.5.5 0 0 1 6 6v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm2.5 0a.5.5 0 0 1 .5.5v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm3 .5a.5.5 0 0 0-1 0v6a.5.5 0 0 0 1 0V6z"/>
|
||||||
|
<path fill-rule="evenodd" d="M14.5 3a1 1 0 0 1-1 1H13v9a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V4h-.5a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1H6a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1h3.5a1 1 0 0 1 1 1v1zM4.118 4L4 4.059V13a1 1 0 0 0 1 1h6a1 1 0 0 0 1-1V4.059L11.882 4H4.118zM2.5 3V2h11v1h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Delete
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
import { Component, OnInit } from '@angular/core';
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
import { Router } from '@angular/router';
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
import { FILTER_CORRESPONDENT } from 'src/app/data/filter-rule-type';
|
||||||
import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent';
|
import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent';
|
||||||
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
import { CorrespondentService } from 'src/app/services/rest/correspondent.service';
|
||||||
import { GenericListComponent } from '../generic-list/generic-list.component';
|
import { GenericListComponent } from '../generic-list/generic-list.component';
|
||||||
import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog/correspondent-edit-dialog.component';
|
import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog/correspondent-edit-dialog.component';
|
||||||
@ -12,7 +15,10 @@ import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog/co
|
|||||||
})
|
})
|
||||||
export class CorrespondentListComponent extends GenericListComponent<PaperlessCorrespondent> {
|
export class CorrespondentListComponent extends GenericListComponent<PaperlessCorrespondent> {
|
||||||
|
|
||||||
constructor(correspondentsService: CorrespondentService, modalService: NgbModal,) {
|
constructor(correspondentsService: CorrespondentService, modalService: NgbModal,
|
||||||
|
private router: Router,
|
||||||
|
private list: DocumentListViewService
|
||||||
|
) {
|
||||||
super(correspondentsService,modalService,CorrespondentEditDialogComponent)
|
super(correspondentsService,modalService,CorrespondentEditDialogComponent)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -20,4 +26,10 @@ export class CorrespondentListComponent extends GenericListComponent<PaperlessCo
|
|||||||
return `correspondent '${object.name}'`
|
return `correspondent '${object.name}'`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
filterDocuments(object: PaperlessCorrespondent) {
|
||||||
|
this.list.documentListView.filter_rules = [
|
||||||
|
{rule_type: FILTER_CORRESPONDENT, value: object.id.toString()}
|
||||||
|
]
|
||||||
|
this.router.navigate(["documents"])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -25,8 +25,25 @@
|
|||||||
<td scope="row">{{ document_type.document_count }}</td>
|
<td scope="row">{{ document_type.document_count }}</td>
|
||||||
<td scope="row">
|
<td scope="row">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(document_type)">Edit</button>
|
<button class="btn btn-sm btn-outline-secondary" (click)="filterDocuments(document_type)">
|
||||||
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(document_type)">Delete</button>
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-funnel" viewBox="0 0 16 16">
|
||||||
|
<path fill-rule="evenodd" d="M1.5 1.5A.5.5 0 0 1 2 1h12a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-.128.334L10 8.692V13.5a.5.5 0 0 1-.342.474l-3 1A.5.5 0 0 1 6 14.5V8.692L1.628 3.834A.5.5 0 0 1 1.5 3.5v-2zm1 .5v1.308l4.372 4.858A.5.5 0 0 1 7 8.5v5.306l2-.666V8.5a.5.5 0 0 1 .128-.334L13.5 3.308V2h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Documents
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(document_type)">
|
||||||
|
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-pencil" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168l10-10zM11.207 2.5L13.5 4.793 14.793 3.5 12.5 1.207 11.207 2.5zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293l6.5-6.5zm-9.761 5.175l-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325z"/>
|
||||||
|
</svg>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(document_type)">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-trash" viewBox="0 0 16 16">
|
||||||
|
<path d="M5.5 5.5A.5.5 0 0 1 6 6v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm2.5 0a.5.5 0 0 1 .5.5v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm3 .5a.5.5 0 0 0-1 0v6a.5.5 0 0 0 1 0V6z"/>
|
||||||
|
<path fill-rule="evenodd" d="M14.5 3a1 1 0 0 1-1 1H13v9a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V4h-.5a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1H6a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1h3.5a1 1 0 0 1 1 1v1zM4.118 4L4 4.059V13a1 1 0 0 0 1 1h6a1 1 0 0 0 1-1V4.059L11.882 4H4.118zM2.5 3V2h11v1h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Delete
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
|
import { Router } from '@angular/router';
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
import { FILTER_DOCUMENT_TYPE } from 'src/app/data/filter-rule-type';
|
||||||
import { PaperlessDocumentType } from 'src/app/data/paperless-document-type';
|
import { PaperlessDocumentType } from 'src/app/data/paperless-document-type';
|
||||||
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { DocumentTypeService } from 'src/app/services/rest/document-type.service';
|
import { DocumentTypeService } from 'src/app/services/rest/document-type.service';
|
||||||
import { GenericListComponent } from '../generic-list/generic-list.component';
|
import { GenericListComponent } from '../generic-list/generic-list.component';
|
||||||
import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog/document-type-edit-dialog.component';
|
import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog/document-type-edit-dialog.component';
|
||||||
@ -12,7 +15,10 @@ import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog/doc
|
|||||||
})
|
})
|
||||||
export class DocumentTypeListComponent extends GenericListComponent<PaperlessDocumentType> {
|
export class DocumentTypeListComponent extends GenericListComponent<PaperlessDocumentType> {
|
||||||
|
|
||||||
constructor(service: DocumentTypeService, modalService: NgbModal) {
|
constructor(service: DocumentTypeService, modalService: NgbModal,
|
||||||
|
private router: Router,
|
||||||
|
private list: DocumentListViewService
|
||||||
|
) {
|
||||||
super(service, modalService, DocumentTypeEditDialogComponent)
|
super(service, modalService, DocumentTypeEditDialogComponent)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -20,4 +26,10 @@ export class DocumentTypeListComponent extends GenericListComponent<PaperlessDoc
|
|||||||
return `document type '${object.name}'`
|
return `document type '${object.name}'`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
filterDocuments(object: PaperlessDocumentType) {
|
||||||
|
this.list.documentListView.filter_rules = [
|
||||||
|
{rule_type: FILTER_DOCUMENT_TYPE, value: object.id.toString()}
|
||||||
|
]
|
||||||
|
this.router.navigate(["documents"])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -95,7 +95,7 @@ export abstract class GenericListComponent<T extends ObjectWithId> implements On
|
|||||||
activeModal.componentInstance.message = "Associated documents will not be deleted."
|
activeModal.componentInstance.message = "Associated documents will not be deleted."
|
||||||
activeModal.componentInstance.btnClass = "btn-danger"
|
activeModal.componentInstance.btnClass = "btn-danger"
|
||||||
activeModal.componentInstance.btnCaption = "Delete"
|
activeModal.componentInstance.btnCaption = "Delete"
|
||||||
activeModal.componentInstance.confirmPressed.subscribe(() => {
|
activeModal.componentInstance.confirmClicked.subscribe(() => {
|
||||||
this.service.delete(object).subscribe(_ => {
|
this.service.delete(object).subscribe(_ => {
|
||||||
activeModal.close()
|
activeModal.close()
|
||||||
this.reloadData()
|
this.reloadData()
|
||||||
|
@ -50,16 +50,26 @@ export class SettingsComponent implements OnInit {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private saveLocalSettings() {
|
||||||
|
localStorage.setItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE, this.settingsForm.value.documentListItemPerPage)
|
||||||
|
this.documentListViewService.updatePageSize()
|
||||||
|
this.toastService.showToast(Toast.make("Information", "Settings saved successfully."))
|
||||||
|
}
|
||||||
|
|
||||||
saveSettings() {
|
saveSettings() {
|
||||||
let x = []
|
let x = []
|
||||||
for (let id in this.savedViewGroup.value) {
|
for (let id in this.savedViewGroup.value) {
|
||||||
x.push(this.savedViewGroup.value[id])
|
x.push(this.savedViewGroup.value[id])
|
||||||
}
|
}
|
||||||
this.savedViewService.patchMany(x).subscribe(s => {
|
if (x.length > 0) {
|
||||||
this.toastService.showToast(Toast.make("Information", "Settings saved successfully."))
|
this.savedViewService.patchMany(x).subscribe(s => {
|
||||||
localStorage.setItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE, this.settingsForm.value.documentListItemPerPage)
|
this.saveLocalSettings()
|
||||||
this.documentListViewService.updatePageSize()
|
}, error => {
|
||||||
})
|
this.toastService.showToast(Toast.makeError(`Error while storing settings on server: ${JSON.stringify(error.error)}`))
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
this.saveLocalSettings()
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
aria-label="Default pagination"></ngb-pagination>
|
aria-label="Default pagination"></ngb-pagination>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<table class="table table-striped border shadow">
|
<table class="table table-striped border shadow-sm">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="col" sortable="name" (sort)="onSort($event)">Name</th>
|
<th scope="col" sortable="name" (sort)="onSort($event)">Name</th>
|
||||||
@ -28,8 +28,25 @@
|
|||||||
<td scope="row">{{ tag.document_count }}</td>
|
<td scope="row">{{ tag.document_count }}</td>
|
||||||
<td scope="row">
|
<td scope="row">
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(tag)">Edit</button>
|
<button class="btn btn-sm btn-outline-secondary" (click)="filterDocuments(tag)">
|
||||||
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(tag)">Delete</button>
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-funnel" viewBox="0 0 16 16">
|
||||||
|
<path fill-rule="evenodd" d="M1.5 1.5A.5.5 0 0 1 2 1h12a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-.128.334L10 8.692V13.5a.5.5 0 0 1-.342.474l-3 1A.5.5 0 0 1 6 14.5V8.692L1.628 3.834A.5.5 0 0 1 1.5 3.5v-2zm1 .5v1.308l4.372 4.858A.5.5 0 0 1 7 8.5v5.306l2-.666V8.5a.5.5 0 0 1 .128-.334L13.5 3.308V2h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Documents
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(tag)">
|
||||||
|
<svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-pencil" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168l10-10zM11.207 2.5L13.5 4.793 14.793 3.5 12.5 1.207 11.207 2.5zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293l6.5-6.5zm-9.761 5.175l-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325z"/>
|
||||||
|
</svg>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-sm btn-outline-danger" (click)="openDeleteDialog(tag)">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-trash" viewBox="0 0 16 16">
|
||||||
|
<path d="M5.5 5.5A.5.5 0 0 1 6 6v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm2.5 0a.5.5 0 0 1 .5.5v6a.5.5 0 0 1-1 0V6a.5.5 0 0 1 .5-.5zm3 .5a.5.5 0 0 0-1 0v6a.5.5 0 0 0 1 0V6z"/>
|
||||||
|
<path fill-rule="evenodd" d="M14.5 3a1 1 0 0 1-1 1H13v9a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V4h-.5a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1H6a1 1 0 0 1 1-1h2a1 1 0 0 1 1 1h3.5a1 1 0 0 1 1 1v1zM4.118 4L4 4.059V13a1 1 0 0 0 1 1h6a1 1 0 0 0 1-1V4.059L11.882 4H4.118zM2.5 3V2h11v1h-11z"/>
|
||||||
|
</svg>
|
||||||
|
Delete
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
|
import { Router } from '@angular/router';
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
import { FILTER_HAS_TAG } from 'src/app/data/filter-rule-type';
|
||||||
import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag';
|
import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag';
|
||||||
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service';
|
||||||
import { TagService } from 'src/app/services/rest/tag.service';
|
import { TagService } from 'src/app/services/rest/tag.service';
|
||||||
import { GenericListComponent } from '../generic-list/generic-list.component';
|
import { GenericListComponent } from '../generic-list/generic-list.component';
|
||||||
import { TagEditDialogComponent } from './tag-edit-dialog/tag-edit-dialog.component';
|
import { TagEditDialogComponent } from './tag-edit-dialog/tag-edit-dialog.component';
|
||||||
@ -12,7 +15,10 @@ import { TagEditDialogComponent } from './tag-edit-dialog/tag-edit-dialog.compon
|
|||||||
})
|
})
|
||||||
export class TagListComponent extends GenericListComponent<PaperlessTag> {
|
export class TagListComponent extends GenericListComponent<PaperlessTag> {
|
||||||
|
|
||||||
constructor(tagService: TagService, modalService: NgbModal) {
|
constructor(tagService: TagService, modalService: NgbModal,
|
||||||
|
private router: Router,
|
||||||
|
private list: DocumentListViewService
|
||||||
|
) {
|
||||||
super(tagService, modalService, TagEditDialogComponent)
|
super(tagService, modalService, TagEditDialogComponent)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -23,4 +29,11 @@ export class TagListComponent extends GenericListComponent<PaperlessTag> {
|
|||||||
getObjectName(object: PaperlessTag) {
|
getObjectName(object: PaperlessTag) {
|
||||||
return `tag '${object.name}'`
|
return `tag '${object.name}'`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
filterDocuments(object: PaperlessTag) {
|
||||||
|
this.list.documentListView.filter_rules = [
|
||||||
|
{rule_type: FILTER_HAS_TAG, value: object.id.toString()}
|
||||||
|
]
|
||||||
|
this.router.navigate(["documents"])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,16 +7,21 @@ import {
|
|||||||
} from '@angular/common/http';
|
} from '@angular/common/http';
|
||||||
import { Observable } from 'rxjs';
|
import { Observable } from 'rxjs';
|
||||||
import { CookieService } from 'ngx-cookie-service';
|
import { CookieService } from 'ngx-cookie-service';
|
||||||
|
import { Meta } from '@angular/platform-browser';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class CsrfInterceptor implements HttpInterceptor {
|
export class CsrfInterceptor implements HttpInterceptor {
|
||||||
|
|
||||||
constructor(private cookieService: CookieService) {
|
constructor(private cookieService: CookieService, private meta: Meta) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
intercept(request: HttpRequest<unknown>, next: HttpHandler): Observable<HttpEvent<unknown>> {
|
intercept(request: HttpRequest<unknown>, next: HttpHandler): Observable<HttpEvent<unknown>> {
|
||||||
let csrfToken = this.cookieService.get('csrftoken')
|
let prefix = ""
|
||||||
|
if (this.meta.getTag('name=cookie_prefix')) {
|
||||||
|
prefix = this.meta.getTag('name=cookie_prefix').content
|
||||||
|
}
|
||||||
|
let csrfToken = this.cookieService.get(`${prefix?prefix:''}csrftoken`)
|
||||||
if (csrfToken) {
|
if (csrfToken) {
|
||||||
request = request.clone({
|
request = request.clone({
|
||||||
setHeaders: {
|
setHeaders: {
|
||||||
|
@ -5,7 +5,7 @@ import { Pipe, PipeTransform } from '@angular/core';
|
|||||||
})
|
})
|
||||||
export class DocumentTitlePipe implements PipeTransform {
|
export class DocumentTitlePipe implements PipeTransform {
|
||||||
|
|
||||||
transform(value: string): unknown {
|
transform(value: string): string {
|
||||||
if (value) {
|
if (value) {
|
||||||
return value
|
return value
|
||||||
} else {
|
} else {
|
||||||
|
@ -116,14 +116,14 @@ export class DocumentListViewService {
|
|||||||
set filterRules(filterRules: FilterRule[]) {
|
set filterRules(filterRules: FilterRule[]) {
|
||||||
//we're going to clone the filterRules object, since we don't
|
//we're going to clone the filterRules object, since we don't
|
||||||
//want changes in the filter editor to propagate into here right away.
|
//want changes in the filter editor to propagate into here right away.
|
||||||
this.view.filter_rules = cloneFilterRules(filterRules)
|
this.view.filter_rules = filterRules
|
||||||
this.reload()
|
this.reload()
|
||||||
this.reduceSelectionToFilter()
|
this.reduceSelectionToFilter()
|
||||||
this.saveDocumentListView()
|
this.saveDocumentListView()
|
||||||
}
|
}
|
||||||
|
|
||||||
get filterRules(): FilterRule[] {
|
get filterRules(): FilterRule[] {
|
||||||
return cloneFilterRules(this.view.filter_rules)
|
return this.view.filter_rules
|
||||||
}
|
}
|
||||||
|
|
||||||
set sortField(field: string) {
|
set sortField(field: string) {
|
||||||
@ -245,7 +245,7 @@ export class DocumentListViewService {
|
|||||||
this.documentListView = null
|
this.documentListView = null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!this.documentListView) {
|
if (!this.documentListView || !this.documentListView.filter_rules || !this.documentListView.sort_reverse || !this.documentListView.sort_field) {
|
||||||
this.documentListView = {
|
this.documentListView = {
|
||||||
filter_rules: [],
|
filter_rules: [],
|
||||||
sort_reverse: true,
|
sort_reverse: true,
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 8.1 KiB After Width: | Height: | Size: 8.1 KiB |
@ -1,5 +1,6 @@
|
|||||||
export const environment = {
|
export const environment = {
|
||||||
production: true,
|
production: true,
|
||||||
apiBaseUrl: "/api/",
|
apiBaseUrl: "/api/",
|
||||||
appTitle: "Paperless-ng"
|
appTitle: "Paperless-ng",
|
||||||
|
version: "0.9.8"
|
||||||
};
|
};
|
||||||
|
@ -5,7 +5,8 @@
|
|||||||
export const environment = {
|
export const environment = {
|
||||||
production: false,
|
production: false,
|
||||||
apiBaseUrl: "http://localhost:8000/api/",
|
apiBaseUrl: "http://localhost:8000/api/",
|
||||||
appTitle: "DEVELOPMENT P-NG"
|
appTitle: "Paperless-ng",
|
||||||
|
version: "DEVELOPMENT"
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -69,7 +69,7 @@ class DocumentAdmin(admin.ModelAdmin):
|
|||||||
|
|
||||||
filter_horizontal = ("tags",)
|
filter_horizontal = ("tags",)
|
||||||
|
|
||||||
ordering = ["-created", "correspondent"]
|
ordering = ["-created"]
|
||||||
|
|
||||||
date_hierarchy = "created"
|
date_hierarchy = "created"
|
||||||
|
|
||||||
|
@ -2,6 +2,7 @@ import textwrap
|
|||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.checks import Error, register
|
from django.core.checks import Error, register
|
||||||
|
from django.core.exceptions import FieldError
|
||||||
from django.db.utils import OperationalError, ProgrammingError
|
from django.db.utils import OperationalError, ProgrammingError
|
||||||
|
|
||||||
from documents.signals import document_consumer_declaration
|
from documents.signals import document_consumer_declaration
|
||||||
@ -16,7 +17,7 @@ def changed_password_check(app_configs, **kwargs):
|
|||||||
try:
|
try:
|
||||||
encrypted_doc = Document.objects.filter(
|
encrypted_doc = Document.objects.filter(
|
||||||
storage_type=Document.STORAGE_TYPE_GPG).first()
|
storage_type=Document.STORAGE_TYPE_GPG).first()
|
||||||
except (OperationalError, ProgrammingError):
|
except (OperationalError, ProgrammingError, FieldError):
|
||||||
return [] # No documents table yet
|
return [] # No documents table yet
|
||||||
|
|
||||||
if encrypted_doc:
|
if encrypted_doc:
|
||||||
|
@ -99,6 +99,11 @@ def generate_filename(doc, counter=0):
|
|||||||
tags = defaultdictNoStr(lambda: slugify(None),
|
tags = defaultdictNoStr(lambda: slugify(None),
|
||||||
many_to_dictionary(doc.tags))
|
many_to_dictionary(doc.tags))
|
||||||
|
|
||||||
|
tag_list = pathvalidate.sanitize_filename(
|
||||||
|
",".join([tag.name for tag in doc.tags.all()]),
|
||||||
|
replacement_text="-"
|
||||||
|
)
|
||||||
|
|
||||||
if doc.correspondent:
|
if doc.correspondent:
|
||||||
correspondent = pathvalidate.sanitize_filename(
|
correspondent = pathvalidate.sanitize_filename(
|
||||||
doc.correspondent.name, replacement_text="-"
|
doc.correspondent.name, replacement_text="-"
|
||||||
@ -127,7 +132,7 @@ def generate_filename(doc, counter=0):
|
|||||||
added_month=f"{doc.added.month:02}" if doc.added else "none",
|
added_month=f"{doc.added.month:02}" if doc.added else "none",
|
||||||
added_day=f"{doc.added.day:02}" if doc.added else "none",
|
added_day=f"{doc.added.day:02}" if doc.added else "none",
|
||||||
tags=tags,
|
tags=tags,
|
||||||
tag_list=",".join([tag.name for tag in doc.tags.all()])
|
tag_list=tag_list
|
||||||
).strip()
|
).strip()
|
||||||
|
|
||||||
path = path.strip(os.sep)
|
path = path.strip(os.sep)
|
||||||
|
@ -2,7 +2,6 @@ import os
|
|||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.management.base import BaseCommand, CommandError
|
from django.core.management.base import BaseCommand, CommandError
|
||||||
from termcolor import colored as coloured
|
|
||||||
|
|
||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
from paperless.db import GnuPG
|
from paperless.db import GnuPG
|
||||||
@ -26,16 +25,14 @@ class Command(BaseCommand):
|
|||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
|
|
||||||
try:
|
try:
|
||||||
print(coloured(
|
print(
|
||||||
"\n\nWARNING: This script is going to work directly on your "
|
"\n\nWARNING: This script is going to work directly on your "
|
||||||
"document originals, so\nWARNING: you probably shouldn't run "
|
"document originals, so\nWARNING: you probably shouldn't run "
|
||||||
"this unless you've got a recent backup\nWARNING: handy. It "
|
"this unless you've got a recent backup\nWARNING: handy. It "
|
||||||
"*should* work without a hitch, but be safe and backup your\n"
|
"*should* work without a hitch, but be safe and backup your\n"
|
||||||
"WARNING: stuff first.\n\nHit Ctrl+C to exit now, or Enter to "
|
"WARNING: stuff first.\n\nHit Ctrl+C to exit now, or Enter to "
|
||||||
"continue.\n\n",
|
"continue.\n\n"
|
||||||
"yellow",
|
)
|
||||||
attrs=("bold",)
|
|
||||||
))
|
|
||||||
__ = input()
|
__ = input()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
return
|
return
|
||||||
@ -57,8 +54,8 @@ class Command(BaseCommand):
|
|||||||
|
|
||||||
for document in encrypted_files:
|
for document in encrypted_files:
|
||||||
|
|
||||||
print(coloured("Decrypting {}".format(
|
print("Decrypting {}".format(
|
||||||
document).encode('utf-8'), "green"))
|
document).encode('utf-8'))
|
||||||
|
|
||||||
old_paths = [document.source_path, document.thumbnail_path]
|
old_paths = [document.source_path, document.thumbnail_path]
|
||||||
|
|
||||||
|
@ -6,13 +6,17 @@ import magic
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.db import migrations, models
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
from paperless.db import GnuPG
|
||||||
|
|
||||||
|
STORAGE_TYPE_UNENCRYPTED = "unencrypted"
|
||||||
|
STORAGE_TYPE_GPG = "gpg"
|
||||||
|
|
||||||
def source_path(self):
|
def source_path(self):
|
||||||
if self.filename:
|
if self.filename:
|
||||||
fname = str(self.filename)
|
fname = str(self.filename)
|
||||||
else:
|
else:
|
||||||
fname = "{:07}.{}".format(self.pk, self.file_type)
|
fname = "{:07}.{}".format(self.pk, self.file_type)
|
||||||
if self.storage_type == self.STORAGE_TYPE_GPG:
|
if self.storage_type == STORAGE_TYPE_GPG:
|
||||||
fname += ".gpg"
|
fname += ".gpg"
|
||||||
|
|
||||||
return os.path.join(
|
return os.path.join(
|
||||||
@ -26,9 +30,18 @@ def add_mime_types(apps, schema_editor):
|
|||||||
documents = Document.objects.all()
|
documents = Document.objects.all()
|
||||||
|
|
||||||
for d in documents:
|
for d in documents:
|
||||||
d.mime_type = magic.from_file(source_path(d), mime=True)
|
f = open(source_path(d), "rb")
|
||||||
|
if d.storage_type == STORAGE_TYPE_GPG:
|
||||||
|
|
||||||
|
data = GnuPG.decrypted(f)
|
||||||
|
else:
|
||||||
|
data = f.read(1024)
|
||||||
|
|
||||||
|
d.mime_type = magic.from_buffer(data, mime=True)
|
||||||
d.save()
|
d.save()
|
||||||
|
|
||||||
|
f.close()
|
||||||
|
|
||||||
|
|
||||||
def add_file_extensions(apps, schema_editor):
|
def add_file_extensions(apps, schema_editor):
|
||||||
Document = apps.get_model("documents", "Document")
|
Document = apps.get_model("documents", "Document")
|
||||||
|
34
src/documents/migrations/1008_auto_20201216_1736.py
Normal file
34
src/documents/migrations/1008_auto_20201216_1736.py
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
# Generated by Django 3.1.4 on 2020-12-16 17:36
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
import django.db.models.functions.text
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('documents', '1007_savedview_savedviewfilterrule'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='correspondent',
|
||||||
|
options={'ordering': (django.db.models.functions.text.Lower('name'),)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='document',
|
||||||
|
options={'ordering': ('-created',)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='documenttype',
|
||||||
|
options={'ordering': (django.db.models.functions.text.Lower('name'),)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='savedview',
|
||||||
|
options={'ordering': (django.db.models.functions.text.Lower('name'),)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='tag',
|
||||||
|
options={'ordering': (django.db.models.functions.text.Lower('name'),)},
|
||||||
|
),
|
||||||
|
]
|
29
src/documents/migrations/1009_auto_20201216_2005.py
Normal file
29
src/documents/migrations/1009_auto_20201216_2005.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Generated by Django 3.1.4 on 2020-12-16 20:05
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('documents', '1008_auto_20201216_1736'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='correspondent',
|
||||||
|
options={'ordering': ('name',)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='documenttype',
|
||||||
|
options={'ordering': ('name',)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='savedview',
|
||||||
|
options={'ordering': ('name',)},
|
||||||
|
),
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='tag',
|
||||||
|
options={'ordering': ('name',)},
|
||||||
|
),
|
||||||
|
]
|
@ -12,7 +12,6 @@ from django.conf import settings
|
|||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.text import slugify
|
|
||||||
|
|
||||||
from documents.file_handling import archive_name_from_filename
|
from documents.file_handling import archive_name_from_filename
|
||||||
from documents.parsers import get_default_file_extension
|
from documents.parsers import get_default_file_extension
|
||||||
@ -205,7 +204,7 @@ class Document(models.Model):
|
|||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ("correspondent", "title")
|
ordering = ("-created",)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
created = datetime.date.isoformat(self.created)
|
created = datetime.date.isoformat(self.created)
|
||||||
@ -221,7 +220,7 @@ class Document(models.Model):
|
|||||||
else:
|
else:
|
||||||
fname = "{:07}{}".format(self.pk, self.file_type)
|
fname = "{:07}{}".format(self.pk, self.file_type)
|
||||||
if self.storage_type == self.STORAGE_TYPE_GPG:
|
if self.storage_type == self.STORAGE_TYPE_GPG:
|
||||||
fname += ".gpg"
|
fname += ".gpg" # pragma: no cover
|
||||||
|
|
||||||
return os.path.join(
|
return os.path.join(
|
||||||
settings.ORIGINALS_DIR,
|
settings.ORIGINALS_DIR,
|
||||||
@ -308,6 +307,10 @@ class Log(models.Model):
|
|||||||
|
|
||||||
class SavedView(models.Model):
|
class SavedView(models.Model):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
|
||||||
|
ordering = ("name",)
|
||||||
|
|
||||||
user = models.ForeignKey(User, on_delete=models.CASCADE)
|
user = models.ForeignKey(User, on_delete=models.CASCADE)
|
||||||
name = models.CharField(max_length=128)
|
name = models.CharField(max_length=128)
|
||||||
|
|
||||||
@ -340,7 +343,11 @@ class SavedViewFilterRule(models.Model):
|
|||||||
(17, "Does not have tag"),
|
(17, "Does not have tag"),
|
||||||
]
|
]
|
||||||
|
|
||||||
saved_view = models.ForeignKey(SavedView, on_delete=models.CASCADE, related_name="filter_rules")
|
saved_view = models.ForeignKey(
|
||||||
|
SavedView,
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="filter_rules"
|
||||||
|
)
|
||||||
|
|
||||||
rule_type = models.PositiveIntegerField(choices=RULE_TYPES)
|
rule_type = models.PositiveIntegerField(choices=RULE_TYPES)
|
||||||
|
|
||||||
|
@ -163,8 +163,6 @@ def parse_date(filename, text):
|
|||||||
|
|
||||||
date = None
|
date = None
|
||||||
|
|
||||||
next_year = timezone.now().year + 5 # Arbitrary 5 year future limit
|
|
||||||
|
|
||||||
# if filename date parsing is enabled, search there first:
|
# if filename date parsing is enabled, search there first:
|
||||||
if settings.FILENAME_DATE_ORDER:
|
if settings.FILENAME_DATE_ORDER:
|
||||||
for m in re.finditer(DATE_REGEX, filename):
|
for m in re.finditer(DATE_REGEX, filename):
|
||||||
@ -176,7 +174,7 @@ def parse_date(filename, text):
|
|||||||
# Skip all matches that do not parse to a proper date
|
# Skip all matches that do not parse to a proper date
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if date is not None and next_year > date.year > 1900:
|
if date and date.year > 1900 and date <= timezone.now():
|
||||||
return date
|
return date
|
||||||
|
|
||||||
# Iterate through all regex matches in text and try to parse the date
|
# Iterate through all regex matches in text and try to parse the date
|
||||||
@ -189,7 +187,7 @@ def parse_date(filename, text):
|
|||||||
# Skip all matches that do not parse to a proper date
|
# Skip all matches that do not parse to a proper date
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if date is not None and next_year > date.year > 1900:
|
if date and date.year > 1900 and date <= timezone.now():
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
date = None
|
date = None
|
||||||
|
@ -187,17 +187,19 @@ class SavedViewSerializer(serializers.ModelSerializer):
|
|||||||
else:
|
else:
|
||||||
rules_data = None
|
rules_data = None
|
||||||
super(SavedViewSerializer, self).update(instance, validated_data)
|
super(SavedViewSerializer, self).update(instance, validated_data)
|
||||||
if rules_data:
|
if rules_data is not None:
|
||||||
SavedViewFilterRule.objects.filter(saved_view=instance).delete()
|
SavedViewFilterRule.objects.filter(saved_view=instance).delete()
|
||||||
for rule_data in rules_data:
|
for rule_data in rules_data:
|
||||||
SavedViewFilterRule.objects.create(saved_view=instance, **rule_data)
|
SavedViewFilterRule.objects.create(
|
||||||
|
saved_view=instance, **rule_data)
|
||||||
return instance
|
return instance
|
||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
rules_data = validated_data.pop('filter_rules')
|
rules_data = validated_data.pop('filter_rules')
|
||||||
saved_view = SavedView.objects.create(**validated_data)
|
saved_view = SavedView.objects.create(**validated_data)
|
||||||
for rule_data in rules_data:
|
for rule_data in rules_data:
|
||||||
SavedViewFilterRule.objects.create(saved_view=saved_view, **rule_data)
|
SavedViewFilterRule.objects.create(
|
||||||
|
saved_view=saved_view, **rule_data)
|
||||||
return saved_view
|
return saved_view
|
||||||
|
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
<title>PaperlessUi</title>
|
<title>PaperlessUi</title>
|
||||||
<base href="/">
|
<base href="/">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<meta name="cookie_prefix" content="{{cookie_prefix}}">
|
||||||
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
||||||
<link rel="stylesheet" href="{% static 'frontend/styles.css' %}"></head>
|
<link rel="stylesheet" href="{% static 'frontend/styles.css' %}"></head>
|
||||||
<body>
|
<body>
|
||||||
|
@ -5,13 +5,11 @@ import tempfile
|
|||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.test import client
|
|
||||||
from pathvalidate import ValidationError
|
|
||||||
from rest_framework.test import APITestCase
|
from rest_framework.test import APITestCase
|
||||||
from whoosh.writing import AsyncWriter
|
from whoosh.writing import AsyncWriter
|
||||||
|
|
||||||
from documents import index, bulk_edit
|
from documents import index, bulk_edit
|
||||||
from documents.models import Document, Correspondent, DocumentType, Tag
|
from documents.models import Document, Correspondent, DocumentType, Tag, SavedView
|
||||||
from documents.tests.utils import DirectoriesMixin
|
from documents.tests.utils import DirectoriesMixin
|
||||||
|
|
||||||
|
|
||||||
@ -20,8 +18,8 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(TestDocumentApi, self).setUp()
|
super(TestDocumentApi, self).setUp()
|
||||||
|
|
||||||
user = User.objects.create_superuser(username="temp_admin")
|
self.user = User.objects.create_superuser(username="temp_admin")
|
||||||
self.client.force_login(user=user)
|
self.client.force_login(user=self.user)
|
||||||
|
|
||||||
def testDocuments(self):
|
def testDocuments(self):
|
||||||
|
|
||||||
@ -172,15 +170,13 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
results = response.data['results']
|
results = response.data['results']
|
||||||
self.assertEqual(len(results), 2)
|
self.assertEqual(len(results), 2)
|
||||||
self.assertEqual(results[0]['id'], doc2.id)
|
self.assertCountEqual([results[0]['id'], results[1]['id']], [doc2.id, doc3.id])
|
||||||
self.assertEqual(results[1]['id'], doc3.id)
|
|
||||||
|
|
||||||
response = self.client.get("/api/documents/?tags__id__in={},{}".format(tag_inbox.id, tag_3.id))
|
response = self.client.get("/api/documents/?tags__id__in={},{}".format(tag_inbox.id, tag_3.id))
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
results = response.data['results']
|
results = response.data['results']
|
||||||
self.assertEqual(len(results), 2)
|
self.assertEqual(len(results), 2)
|
||||||
self.assertEqual(results[0]['id'], doc1.id)
|
self.assertCountEqual([results[0]['id'], results[1]['id']], [doc1.id, doc3.id])
|
||||||
self.assertEqual(results[1]['id'], doc3.id)
|
|
||||||
|
|
||||||
response = self.client.get("/api/documents/?tags__id__all={},{}".format(tag_2.id, tag_3.id))
|
response = self.client.get("/api/documents/?tags__id__all={},{}".format(tag_2.id, tag_3.id))
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
@ -202,8 +198,7 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
results = response.data['results']
|
results = response.data['results']
|
||||||
self.assertEqual(len(results), 2)
|
self.assertEqual(len(results), 2)
|
||||||
self.assertEqual(results[0]['id'], doc1.id)
|
self.assertCountEqual([results[0]['id'], results[1]['id']], [doc1.id, doc2.id])
|
||||||
self.assertEqual(results[1]['id'], doc2.id)
|
|
||||||
|
|
||||||
response = self.client.get("/api/documents/?tags__id__none={},{}".format(tag_3.id, tag_2.id))
|
response = self.client.get("/api/documents/?tags__id__none={},{}".format(tag_3.id, tag_2.id))
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
@ -518,6 +513,90 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
self.assertGreater(len(meta['original_metadata']), 0)
|
self.assertGreater(len(meta['original_metadata']), 0)
|
||||||
self.assertIsNone(meta['archive_metadata'])
|
self.assertIsNone(meta['archive_metadata'])
|
||||||
|
|
||||||
|
def test_saved_views(self):
|
||||||
|
u1 = User.objects.create_user("user1")
|
||||||
|
u2 = User.objects.create_user("user2")
|
||||||
|
|
||||||
|
v1 = SavedView.objects.create(user=u1, name="test1", sort_field="", show_on_dashboard=False, show_in_sidebar=False)
|
||||||
|
v2 = SavedView.objects.create(user=u2, name="test2", sort_field="", show_on_dashboard=False, show_in_sidebar=False)
|
||||||
|
v3 = SavedView.objects.create(user=u2, name="test3", sort_field="", show_on_dashboard=False, show_in_sidebar=False)
|
||||||
|
|
||||||
|
response = self.client.get("/api/saved_views/")
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertEqual(response.data['count'], 0)
|
||||||
|
|
||||||
|
self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 404)
|
||||||
|
|
||||||
|
self.client.force_login(user=u1)
|
||||||
|
|
||||||
|
response = self.client.get("/api/saved_views/")
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertEqual(response.data['count'], 1)
|
||||||
|
|
||||||
|
self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 200)
|
||||||
|
|
||||||
|
self.client.force_login(user=u2)
|
||||||
|
|
||||||
|
response = self.client.get("/api/saved_views/")
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertEqual(response.data['count'], 2)
|
||||||
|
|
||||||
|
self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 404)
|
||||||
|
|
||||||
|
def test_create_update_patch(self):
|
||||||
|
|
||||||
|
u1 = User.objects.create_user("user1")
|
||||||
|
|
||||||
|
view = {
|
||||||
|
"name": "test",
|
||||||
|
"show_on_dashboard": True,
|
||||||
|
"show_in_sidebar": True,
|
||||||
|
"sort_field": "created2",
|
||||||
|
"filter_rules": [
|
||||||
|
{
|
||||||
|
"rule_type": 4,
|
||||||
|
"value": "test"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
response = self.client.post("/api/saved_views/", view, format='json')
|
||||||
|
self.assertEqual(response.status_code, 201)
|
||||||
|
|
||||||
|
v1 = SavedView.objects.get(name="test")
|
||||||
|
self.assertEqual(v1.sort_field, "created2")
|
||||||
|
self.assertEqual(v1.filter_rules.count(), 1)
|
||||||
|
self.assertEqual(v1.user, self.user)
|
||||||
|
|
||||||
|
response = self.client.patch(f"/api/saved_views/{v1.id}/", {
|
||||||
|
"show_in_sidebar": False
|
||||||
|
}, format='json')
|
||||||
|
|
||||||
|
v1 = SavedView.objects.get(id=v1.id)
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertFalse(v1.show_in_sidebar)
|
||||||
|
self.assertEqual(v1.filter_rules.count(), 1)
|
||||||
|
|
||||||
|
view['filter_rules'] = [{
|
||||||
|
"rule_type": 12,
|
||||||
|
"value": "secret"
|
||||||
|
}]
|
||||||
|
|
||||||
|
response = self.client.put(f"/api/saved_views/{v1.id}/", view, format='json')
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
|
||||||
|
v1 = SavedView.objects.get(id=v1.id)
|
||||||
|
self.assertEqual(v1.filter_rules.count(), 1)
|
||||||
|
self.assertEqual(v1.filter_rules.first().value, "secret")
|
||||||
|
|
||||||
|
view['filter_rules'] = []
|
||||||
|
|
||||||
|
response = self.client.put(f"/api/saved_views/{v1.id}/", view, format='json')
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
|
||||||
|
v1 = SavedView.objects.get(id=v1.id)
|
||||||
|
self.assertEqual(v1.filter_rules.count(), 0)
|
||||||
|
|
||||||
|
|
||||||
class TestBulkEdit(DirectoriesMixin, APITestCase):
|
class TestBulkEdit(DirectoriesMixin, APITestCase):
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@ from unittest import mock
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.db import DatabaseError
|
from django.db import DatabaseError
|
||||||
from django.test import TestCase, override_settings
|
from django.test import TestCase, override_settings
|
||||||
|
from django.utils import timezone
|
||||||
|
|
||||||
from .utils import DirectoriesMixin
|
from .utils import DirectoriesMixin
|
||||||
from ..file_handling import generate_filename, create_source_path_directory, delete_empty_directories, \
|
from ..file_handling import generate_filename, create_source_path_directory, delete_empty_directories, \
|
||||||
@ -298,23 +299,23 @@ class TestFileHandling(DirectoriesMixin, TestCase):
|
|||||||
|
|
||||||
@override_settings(PAPERLESS_FILENAME_FORMAT="{created_year}-{created_month}-{created_day}")
|
@override_settings(PAPERLESS_FILENAME_FORMAT="{created_year}-{created_month}-{created_day}")
|
||||||
def test_created_year_month_day(self):
|
def test_created_year_month_day(self):
|
||||||
d1 = datetime.datetime(2020, 3, 6, 1, 1, 1)
|
d1 = timezone.make_aware(datetime.datetime(2020, 3, 6, 1, 1, 1))
|
||||||
doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", created=d1)
|
doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", created=d1)
|
||||||
|
|
||||||
self.assertEqual(generate_filename(doc1), "2020-03-06.pdf")
|
self.assertEqual(generate_filename(doc1), "2020-03-06.pdf")
|
||||||
|
|
||||||
doc1.created = datetime.datetime(2020, 11, 16, 1, 1, 1)
|
doc1.created = timezone.make_aware(datetime.datetime(2020, 11, 16, 1, 1, 1))
|
||||||
|
|
||||||
self.assertEqual(generate_filename(doc1), "2020-11-16.pdf")
|
self.assertEqual(generate_filename(doc1), "2020-11-16.pdf")
|
||||||
|
|
||||||
@override_settings(PAPERLESS_FILENAME_FORMAT="{added_year}-{added_month}-{added_day}")
|
@override_settings(PAPERLESS_FILENAME_FORMAT="{added_year}-{added_month}-{added_day}")
|
||||||
def test_added_year_month_day(self):
|
def test_added_year_month_day(self):
|
||||||
d1 = datetime.datetime(232, 1, 9, 1, 1, 1)
|
d1 = timezone.make_aware(datetime.datetime(232, 1, 9, 1, 1, 1))
|
||||||
doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", added=d1)
|
doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", added=d1)
|
||||||
|
|
||||||
self.assertEqual(generate_filename(doc1), "232-01-09.pdf")
|
self.assertEqual(generate_filename(doc1), "232-01-09.pdf")
|
||||||
|
|
||||||
doc1.added = datetime.datetime(2020, 11, 16, 1, 1, 1)
|
doc1.added = timezone.make_aware(datetime.datetime(2020, 11, 16, 1, 1, 1))
|
||||||
|
|
||||||
self.assertEqual(generate_filename(doc1), "2020-11-16.pdf")
|
self.assertEqual(generate_filename(doc1), "2020-11-16.pdf")
|
||||||
|
|
||||||
@ -599,7 +600,7 @@ class TestFilenameGeneration(TestCase):
|
|||||||
PAPERLESS_FILENAME_FORMAT="{created}"
|
PAPERLESS_FILENAME_FORMAT="{created}"
|
||||||
)
|
)
|
||||||
def test_date(self):
|
def test_date(self):
|
||||||
doc = Document.objects.create(title="does not matter", created=datetime.datetime(2020,5,21, 7,36,51, 153), mime_type="application/pdf", pk=2, checksum="2")
|
doc = Document.objects.create(title="does not matter", created=timezone.make_aware(datetime.datetime(2020,5,21, 7,36,51, 153)), mime_type="application/pdf", pk=2, checksum="2")
|
||||||
self.assertEqual(generate_filename(doc), "2020-05-21.pdf")
|
self.assertEqual(generate_filename(doc), "2020-05-21.pdf")
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from documents import index
|
||||||
from documents.index import JsonFormatter
|
from documents.index import JsonFormatter
|
||||||
|
from documents.models import Document
|
||||||
|
from documents.tests.utils import DirectoriesMixin
|
||||||
|
|
||||||
|
|
||||||
class JsonFormatterTest(TestCase):
|
class JsonFormatterTest(TestCase):
|
||||||
@ -12,3 +15,21 @@ class JsonFormatterTest(TestCase):
|
|||||||
self.assertListEqual(self.formatter.format([]), [])
|
self.assertListEqual(self.formatter.format([]), [])
|
||||||
|
|
||||||
|
|
||||||
|
class TestAutoComplete(DirectoriesMixin, TestCase):
|
||||||
|
|
||||||
|
def test_auto_complete(self):
|
||||||
|
|
||||||
|
doc1 = Document.objects.create(title="doc1", checksum="A", content="test test2 test3")
|
||||||
|
doc2 = Document.objects.create(title="doc2", checksum="B", content="test test2")
|
||||||
|
doc3 = Document.objects.create(title="doc3", checksum="C", content="test2")
|
||||||
|
|
||||||
|
index.add_or_update_document(doc1)
|
||||||
|
index.add_or_update_document(doc2)
|
||||||
|
index.add_or_update_document(doc3)
|
||||||
|
|
||||||
|
ix = index.open_index()
|
||||||
|
|
||||||
|
self.assertListEqual(index.autocomplete(ix, "tes"), [b"test3", b"test", b"test2"])
|
||||||
|
self.assertListEqual(index.autocomplete(ix, "tes", limit=3), [b"test3", b"test", b"test2"])
|
||||||
|
self.assertListEqual(index.autocomplete(ix, "tes", limit=1), [b"test3"])
|
||||||
|
self.assertListEqual(index.autocomplete(ix, "tes", limit=0), [])
|
||||||
|
@ -2,6 +2,8 @@ import os
|
|||||||
import shutil
|
import shutil
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
import filelock
|
||||||
|
from django.conf import settings
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
|
|
||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
@ -13,9 +15,11 @@ class TestSanityCheck(DirectoriesMixin, TestCase):
|
|||||||
|
|
||||||
def make_test_data(self):
|
def make_test_data(self):
|
||||||
|
|
||||||
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000001.pdf"), os.path.join(self.dirs.originals_dir, "0000001.pdf"))
|
with filelock.FileLock(settings.MEDIA_LOCK):
|
||||||
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "archive", "0000001.pdf"), os.path.join(self.dirs.archive_dir, "0000001.pdf"))
|
# just make sure that the lockfile is present.
|
||||||
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png"), os.path.join(self.dirs.thumbnail_dir, "0000001.png"))
|
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000001.pdf"), os.path.join(self.dirs.originals_dir, "0000001.pdf"))
|
||||||
|
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "archive", "0000001.pdf"), os.path.join(self.dirs.archive_dir, "0000001.pdf"))
|
||||||
|
shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png"), os.path.join(self.dirs.thumbnail_dir, "0000001.png"))
|
||||||
|
|
||||||
return Document.objects.create(title="test", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", content="test", pk=1, filename="0000001.pdf", mime_type="application/pdf")
|
return Document.objects.create(title="test", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", content="test", pk=1, filename="0000001.pdf", mime_type="application/pdf")
|
||||||
|
|
||||||
|
@ -34,7 +34,8 @@ def setup_directories():
|
|||||||
ARCHIVE_DIR=dirs.archive_dir,
|
ARCHIVE_DIR=dirs.archive_dir,
|
||||||
CONSUMPTION_DIR=dirs.consumption_dir,
|
CONSUMPTION_DIR=dirs.consumption_dir,
|
||||||
INDEX_DIR=dirs.index_dir,
|
INDEX_DIR=dirs.index_dir,
|
||||||
MODEL_FILE=os.path.join(dirs.data_dir, "classification_model.pickle")
|
MODEL_FILE=os.path.join(dirs.data_dir, "classification_model.pickle"),
|
||||||
|
MEDIA_LOCK=os.path.join(dirs.media_dir, "media.lock")
|
||||||
|
|
||||||
)
|
)
|
||||||
dirs.settings_override.enable()
|
dirs.settings_override.enable()
|
||||||
|
@ -55,6 +55,11 @@ from .serialisers import (
|
|||||||
class IndexView(TemplateView):
|
class IndexView(TemplateView):
|
||||||
template_name = "index.html"
|
template_name = "index.html"
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
context = super().get_context_data(**kwargs)
|
||||||
|
context['cookie_prefix'] = settings.COOKIE_PREFIX
|
||||||
|
return context
|
||||||
|
|
||||||
|
|
||||||
class CorrespondentViewSet(ModelViewSet):
|
class CorrespondentViewSet(ModelViewSet):
|
||||||
model = Correspondent
|
model = Correspondent
|
||||||
@ -185,7 +190,12 @@ class DocumentViewSet(RetrieveModelMixin,
|
|||||||
parser_class = get_parser_class_for_mime_type(mime_type)
|
parser_class = get_parser_class_for_mime_type(mime_type)
|
||||||
if parser_class:
|
if parser_class:
|
||||||
parser = parser_class(logging_group=None)
|
parser = parser_class(logging_group=None)
|
||||||
return parser.extract_metadata(file, mime_type)
|
|
||||||
|
try:
|
||||||
|
return parser.extract_metadata(file, mime_type)
|
||||||
|
except Exception as e:
|
||||||
|
# TODO: cover GPG errors, remove later.
|
||||||
|
return []
|
||||||
else:
|
else:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
@ -231,7 +241,12 @@ class DocumentViewSet(RetrieveModelMixin,
|
|||||||
@cache_control(public=False, max_age=315360000)
|
@cache_control(public=False, max_age=315360000)
|
||||||
def thumb(self, request, pk=None):
|
def thumb(self, request, pk=None):
|
||||||
try:
|
try:
|
||||||
return HttpResponse(Document.objects.get(id=pk).thumbnail_file,
|
doc = Document.objects.get(id=pk)
|
||||||
|
if doc.storage_type == Document.STORAGE_TYPE_GPG:
|
||||||
|
handle = GnuPG.decrypted(doc.thumbnail_file)
|
||||||
|
else:
|
||||||
|
handle = doc.thumbnail_file
|
||||||
|
return HttpResponse(handle,
|
||||||
content_type='image/png')
|
content_type='image/png')
|
||||||
except (FileNotFoundError, Document.DoesNotExist):
|
except (FileNotFoundError, Document.DoesNotExist):
|
||||||
raise Http404()
|
raise Http404()
|
||||||
|
@ -1 +1 @@
|
|||||||
__version__ = (0, 9, 6)
|
__version__ = (0, 9, 8)
|
||||||
|
@ -26,7 +26,7 @@ class BaseMailAction:
|
|||||||
return {}
|
return {}
|
||||||
|
|
||||||
def post_consume(self, M, message_uids, parameter):
|
def post_consume(self, M, message_uids, parameter):
|
||||||
pass
|
pass # pragma: nocover
|
||||||
|
|
||||||
|
|
||||||
class DeleteMailAction(BaseMailAction):
|
class DeleteMailAction(BaseMailAction):
|
||||||
@ -69,7 +69,7 @@ def get_rule_action(rule):
|
|||||||
elif rule.action == MailRule.ACTION_MARK_READ:
|
elif rule.action == MailRule.ACTION_MARK_READ:
|
||||||
return MarkReadMailAction()
|
return MarkReadMailAction()
|
||||||
else:
|
else:
|
||||||
raise ValueError("Unknown action.")
|
raise NotImplementedError("Unknown action.") # pragma: nocover
|
||||||
|
|
||||||
|
|
||||||
def make_criterias(rule):
|
def make_criterias(rule):
|
||||||
@ -95,7 +95,7 @@ def get_mailbox(server, port, security):
|
|||||||
elif security == MailAccount.IMAP_SECURITY_SSL:
|
elif security == MailAccount.IMAP_SECURITY_SSL:
|
||||||
mailbox = MailBox(server, port)
|
mailbox = MailBox(server, port)
|
||||||
else:
|
else:
|
||||||
raise ValueError("Unknown IMAP security")
|
raise NotImplementedError("Unknown IMAP security") # pragma: nocover
|
||||||
return mailbox
|
return mailbox
|
||||||
|
|
||||||
|
|
||||||
@ -119,7 +119,7 @@ class MailAccountHandler(LoggingMixin):
|
|||||||
return os.path.splitext(os.path.basename(att.filename))[0]
|
return os.path.splitext(os.path.basename(att.filename))[0]
|
||||||
|
|
||||||
else:
|
else:
|
||||||
raise ValueError("Unknown title selector.")
|
raise NotImplementedError("Unknown title selector.") # pragma: nocover # NOQA: E501
|
||||||
|
|
||||||
def get_correspondent(self, message, rule):
|
def get_correspondent(self, message, rule):
|
||||||
c_from = rule.assign_correspondent_from
|
c_from = rule.assign_correspondent_from
|
||||||
@ -141,7 +141,7 @@ class MailAccountHandler(LoggingMixin):
|
|||||||
return rule.assign_correspondent
|
return rule.assign_correspondent
|
||||||
|
|
||||||
else:
|
else:
|
||||||
raise ValueError("Unknwown correspondent selector")
|
raise NotImplementedError("Unknwown correspondent selector") # pragma: nocover # NOQA: E501
|
||||||
|
|
||||||
def handle_mail_account(self, account):
|
def handle_mail_account(self, account):
|
||||||
|
|
||||||
|
@ -399,7 +399,7 @@ class TestMail(TestCase):
|
|||||||
|
|
||||||
c = Correspondent.objects.get(name="amazon@amazon.de")
|
c = Correspondent.objects.get(name="amazon@amazon.de")
|
||||||
# should work
|
# should work
|
||||||
self.assertEquals(kwargs['override_correspondent_id'], c.id)
|
self.assertEqual(kwargs['override_correspondent_id'], c.id)
|
||||||
|
|
||||||
self.async_task.reset_mock()
|
self.async_task.reset_mock()
|
||||||
self.reset_bogus_mailbox()
|
self.reset_bogus_mailbox()
|
||||||
@ -411,7 +411,7 @@ class TestMail(TestCase):
|
|||||||
|
|
||||||
args, kwargs = self.async_task.call_args
|
args, kwargs = self.async_task.call_args
|
||||||
self.async_task.assert_called_once()
|
self.async_task.assert_called_once()
|
||||||
self.assertEquals(kwargs['override_correspondent_id'], None)
|
self.assertEqual(kwargs['override_correspondent_id'], None)
|
||||||
|
|
||||||
|
|
||||||
def test_filters(self):
|
def test_filters(self):
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import os
|
import os
|
||||||
import subprocess
|
import subprocess
|
||||||
|
|
||||||
|
from PIL import ImageDraw, ImageFont, Image
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
|
||||||
from documents.parsers import DocumentParser, ParseError
|
from documents.parsers import DocumentParser, ParseError
|
||||||
@ -12,63 +13,22 @@ class TextDocumentParser(DocumentParser):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def get_thumbnail(self, document_path, mime_type):
|
def get_thumbnail(self, document_path, mime_type):
|
||||||
"""
|
|
||||||
The thumbnail of a text file is just a 500px wide image of the text
|
|
||||||
rendered onto a letter-sized page.
|
|
||||||
"""
|
|
||||||
# The below is heavily cribbed from https://askubuntu.com/a/590951
|
|
||||||
|
|
||||||
bg_color = "white" # bg color
|
|
||||||
text_color = "black" # text color
|
|
||||||
psize = [500, 647] # icon size
|
|
||||||
n_lines = 50 # number of lines to show
|
|
||||||
out_path = os.path.join(self.tempdir, "convert.png")
|
|
||||||
|
|
||||||
temp_bg = os.path.join(self.tempdir, "bg.png")
|
|
||||||
temp_txlayer = os.path.join(self.tempdir, "tx.png")
|
|
||||||
picsize = "x".join([str(n) for n in psize])
|
|
||||||
txsize = "x".join([str(n - 8) for n in psize])
|
|
||||||
|
|
||||||
def create_bg():
|
|
||||||
work_size = ",".join([str(n - 1) for n in psize])
|
|
||||||
r = str(round(psize[0] / 10))
|
|
||||||
rounded = ",".join([r, r])
|
|
||||||
run_command(
|
|
||||||
settings.CONVERT_BINARY,
|
|
||||||
"-size ", picsize,
|
|
||||||
' xc:none -draw ',
|
|
||||||
'"fill ', bg_color, ' roundrectangle 0,0,', work_size, ",", rounded, '" ', # NOQA: E501
|
|
||||||
temp_bg
|
|
||||||
)
|
|
||||||
|
|
||||||
def read_text():
|
def read_text():
|
||||||
with open(document_path, 'r') as src:
|
with open(document_path, 'r') as src:
|
||||||
lines = [line.strip() for line in src.readlines()]
|
lines = [line.strip() for line in src.readlines()]
|
||||||
text = "\n".join([line for line in lines[:n_lines]])
|
text = "\n".join(lines[:50])
|
||||||
return text.replace('"', "'")
|
return text
|
||||||
|
|
||||||
def create_txlayer():
|
img = Image.new("RGB", (500, 700), color="white")
|
||||||
run_command(
|
draw = ImageDraw.Draw(img)
|
||||||
settings.CONVERT_BINARY,
|
font = ImageFont.truetype(
|
||||||
"-background none",
|
"/usr/share/fonts/liberation/LiberationSerif-Regular.ttf", 20,
|
||||||
"-fill",
|
layout_engine=ImageFont.LAYOUT_BASIC)
|
||||||
text_color,
|
draw.text((5, 5), read_text(), font=font, fill="black")
|
||||||
"-pointsize", "12",
|
|
||||||
"-border 4 -bordercolor none",
|
|
||||||
"-size ", txsize,
|
|
||||||
' caption:"', read_text(), '" ',
|
|
||||||
temp_txlayer
|
|
||||||
)
|
|
||||||
|
|
||||||
create_txlayer()
|
out_path = os.path.join(self.tempdir, "thumb.png")
|
||||||
create_bg()
|
img.save(out_path)
|
||||||
run_command(
|
|
||||||
settings.CONVERT_BINARY,
|
|
||||||
temp_bg,
|
|
||||||
temp_txlayer,
|
|
||||||
"-background None -layers merge ",
|
|
||||||
out_path
|
|
||||||
)
|
|
||||||
|
|
||||||
return out_path
|
return out_path
|
||||||
|
|
||||||
|
1
src/paperless_text/tests/samples/test.txt
Normal file
1
src/paperless_text/tests/samples/test.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
This is a test file.
|
26
src/paperless_text/tests/test_parser.py
Normal file
26
src/paperless_text/tests/test_parser.py
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import os
|
||||||
|
|
||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from documents.tests.utils import DirectoriesMixin
|
||||||
|
from paperless_text.parsers import TextDocumentParser
|
||||||
|
|
||||||
|
|
||||||
|
class TestTextParser(DirectoriesMixin, TestCase):
|
||||||
|
|
||||||
|
def test_thumbnail(self):
|
||||||
|
|
||||||
|
parser = TextDocumentParser(None)
|
||||||
|
|
||||||
|
# just make sure that it does not crash
|
||||||
|
f = parser.get_thumbnail(os.path.join(os.path.dirname(__file__), "samples", "test.txt"), "text/plain")
|
||||||
|
self.assertTrue(os.path.isfile(f))
|
||||||
|
|
||||||
|
def test_parse(self):
|
||||||
|
|
||||||
|
parser = TextDocumentParser(None)
|
||||||
|
|
||||||
|
parser.parse(os.path.join(os.path.dirname(__file__), "samples", "test.txt"), "text/plain")
|
||||||
|
|
||||||
|
self.assertEqual(parser.get_text(), "This is a test file.\n")
|
||||||
|
self.assertIsNone(parser.get_archive_path())
|
Loading…
x
Reference in New Issue
Block a user