mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-25 15:52:35 -04:00 
			
		
		
		
	Merge branch 'dev' into feature-websockets-status
This commit is contained in:
		
						commit
						9f9581e1f8
					
				
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -85,3 +85,4 @@ scripts/nuke | ||||
| 
 | ||||
| # this is where the compiled frontend is moved to. | ||||
| /src/documents/static/frontend/ | ||||
| /docs/.vscode/settings.json | ||||
|  | ||||
| @ -24,3 +24,7 @@ feature-X branches is for experimental stuff that will eventually be merged into | ||||
| I'm trying to get most of paperless tested, so please do the same for your code! I know its a hassle, but it makes sure that your code works now and will allow us to detect regressions easily. | ||||
| 
 | ||||
| To test your code, execute `pytest` in the src/ directory. Executing that in the project root is no good. This also generates a html coverage report, which you can use to see if you missed anything important during testing. | ||||
| 
 | ||||
| ## More info: | ||||
| 
 | ||||
| ... is available in the documentation. https://paperless-ng.readthedocs.io/en/latest/extending.html | ||||
|  | ||||
							
								
								
									
										3
									
								
								Pipfile
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								Pipfile
									
									
									
									
									
								
							| @ -19,6 +19,7 @@ django-extensions = "*" | ||||
| django-filter = "~=2.4.0" | ||||
| django-q = "~=1.3.4" | ||||
| djangorestframework = "~=3.12.2" | ||||
| filelock = "*" | ||||
| fuzzywuzzy = "*" | ||||
| gunicorn = "*" | ||||
| imap-tools = "*" | ||||
| @ -26,6 +27,7 @@ langdetect = "*" | ||||
| pdftotext = "*" | ||||
| pathvalidate = "*" | ||||
| pillow = "*" | ||||
| pikepdf = "*" | ||||
| python-gnupg = "*" | ||||
| python-dotenv = "*" | ||||
| python-dateutil = "*" | ||||
| @ -40,6 +42,7 @@ whoosh="~=2.7.4" | ||||
| inotifyrecursive = "~=0.3.4" | ||||
| ocrmypdf = "*" | ||||
| tqdm = "*" | ||||
| tika = "*" | ||||
| channels = "~=3.0" | ||||
| channels-redis = "*" | ||||
| daphne = "~=3.0" | ||||
|  | ||||
							
								
								
									
										886
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										886
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										60
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										60
									
								
								README.md
									
									
									
									
									
								
							| @ -1,13 +1,14 @@ | ||||
| [](https://travis-ci.org/jonaswinkler/paperless-ng) | ||||
| [](https://travis-ci.com/jonaswinkler/paperless-ng) | ||||
| [](https://paperless-ng.readthedocs.io/en/latest/?badge=latest) | ||||
| [](https://gitter.im/paperless-ng/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) | ||||
| [](https://hub.docker.com/r/jonaswinkler/paperless-ng) | ||||
| [](https://coveralls.io/github/jonaswinkler/paperless-ng?branch=master) | ||||
| 
 | ||||
| # Paperless-ng | ||||
| 
 | ||||
| [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 contributors 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. | ||||
| 
 | ||||
| @ -15,11 +16,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. | ||||
| 
 | ||||
| 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. | ||||
| 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. | ||||
| 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. | ||||
| 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. | ||||
| 
 | ||||
| 	- 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. | ||||
| 
 | ||||
| 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: | ||||
| 
 | ||||
| @ -28,19 +31,22 @@ Here's what you get: | ||||
| # Features | ||||
| 
 | ||||
| * Performs OCR on your documents, adds selectable text to image only documents and adds tags, correspondents and document types to your documents. | ||||
| * Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and can be configured freely. | ||||
| * Single page application front end. Should be pretty snappy. Will be mobile friendly in the future. | ||||
| 	* Includes a dashboard that shows basic statistics and has document upload. | ||||
| 	* Filtering by tags, correspondents, types, and more. | ||||
| 	* Customizable views can be saved and displayed on the dashboard. | ||||
| 	* Full text search with auto completion, scored results and query highlighting allows you to quickly find what you need. | ||||
| * Full text search helps you find what you need. | ||||
| 	* Auto completion suggests relevant words from your documents. | ||||
| 	* Results are sorted by relevance to your search query. | ||||
| 	* Highlighting shows you which parts of the document matched the query. | ||||
| 	* Searching for similar documents ("More like this") | ||||
| * Email processing: Paperless adds documents from your email accounts. | ||||
| 	* Configure multiple accounts and filters for each account. | ||||
| 	* 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. | ||||
| 	* 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. | ||||
| * 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. | ||||
| * More tests, more stability. | ||||
| 
 | ||||
| If you want to see some screenshots of paperless-ng in action, [some are available in the documentation](https://paperless-ng.readthedocs.io/en/latest/screenshots.html). | ||||
| 
 | ||||
| @ -49,35 +55,44 @@ For a complete list of changes from paperless, check out the [changelog](https:/ | ||||
| # Roadmap for 1.0 | ||||
| 
 | ||||
| - Make the front end nice (except mobile). | ||||
| - Test coverage at 90%. | ||||
| - Store archived documents with an embedded OCR text layer, while keeping originals available. Making good progress in the `feature-ocrmypdf` branch. | ||||
| - Fix whatever bugs I and you find. | ||||
| 
 | ||||
| ## Roadmap for versions beyond 1.0 | ||||
| 
 | ||||
| These are things that I want to add to paperless eventually. They are sorted by priority. | ||||
| 
 | ||||
| - **Bulk editing**. Add/remove metadata from multiple documents at once. | ||||
| - **More search.** The search backend is incredibly versatile and customizable. Searching is the most important feature of this project and thus, I want to implement things like: | ||||
|   - Group and limit search results by correspondent, show “more from this” links in the results. | ||||
|   - Ability to search for “Similar documents” in the search results | ||||
| - **Nested tags**. Organize tags in a hierarchical structure. This will combine the benefits of folders and tags in one coherent system. | ||||
| - **Localization.** I won't translate paperless into any other languages except English and German, however, I'll add the necessary means so that anyone can translate paperless into their favorite language. | ||||
| - **An interactive consumer** that shows its progress for documents it processes on the web page. | ||||
| 	- With live updates ans websockets. This already works on a dev branch, but requires a lot of new dependencies, which I'm not particular happy about. | ||||
| 	- With live updates and websockets. This already works on a dev branch, but requires a lot of new dependencies, which I'm not particularly happy about. | ||||
| 	- Notifications when a document was added with buttons to open the new document right away. | ||||
| - **Arbitrary tag colors**. Allow the selection of any color with a color picker. | ||||
| - **More file types**. Possibly allow more file types to be processed by paperless, such as office .odt, .doc and .docx documents. | ||||
| 
 | ||||
| Apart from that, paperless is pretty much feature complete. | ||||
| 
 | ||||
| ## On the chopping block. | ||||
| 
 | ||||
| - **GnuPG encrypion.** [Here's a note about encryption in paperless](https://paperless-ng.readthedocs.io/en/latest/administration.html#managing-encryption). The gist of it is that I don't see which attacks this implementation protects against. It gives a false sense of security to users who don't care about how it works. | ||||
| 
 | ||||
| ## Wont-do list. | ||||
| 
 | ||||
| These features will probably never make it into paperless, since paperless is meant to be an easy to use set-and-forget solution. | ||||
| 
 | ||||
| - **Document versions.** I might consider adding the ability to update a document with a newer version, but that's about it. The kind of documents that get added to paperless usually don't change at all. | ||||
| - **Workflows.** I don't see a use case for these, yet. | ||||
| - **Folders.** Tags are superior in just about every way. | ||||
| - **Apps / extension support.** Again, paperless is meant to be simple. | ||||
| 
 | ||||
| # Getting started | ||||
| 
 | ||||
| The recommended way to deploy paperless is docker-compose. Don't clone the repository, grab the latest release to get started instead. The dockerfiles archive contains just the docker files which will pull the image from docker hub. The source archive contains everything you need to build the docker image yourself (i.e. if you want to run on Raspberry Pi). | ||||
| 
 | ||||
| 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 | ||||
| 
 | ||||
| @ -101,12 +116,13 @@ 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 App](https://github.com/bauerj/paperless_app): An Android/iOS app for Paperless. We're working on making this compatible. | ||||
| * [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. | ||||
| * [paperless-cli](https://github.com/stgarf/paperless-cli): A golang command line binary to interact with a Paperless instance. | ||||
| * [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. | ||||
| 
 | ||||
| Compatibility with Paperless-ng is unknown. | ||||
| 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-cli](https://github.com/stgarf/paperless-cli): A golang command line binary to interact with a Paperless instance. | ||||
| 
 | ||||
| # Important Note | ||||
| 
 | ||||
|  | ||||
| @ -25,6 +25,11 @@ wait_for_postgres() { | ||||
| 	host="${PAPERLESS_DBHOST}" | ||||
| 	port="${PAPERLESS_DBPORT}" | ||||
| 
 | ||||
| 	if [[ -z $port ]] ; | ||||
| 	then | ||||
| 		port="5432" | ||||
| 	fi | ||||
| 
 | ||||
| 	while !</dev/tcp/$host/$port ; | ||||
| 	do | ||||
| 
 | ||||
| @ -114,13 +119,13 @@ install_languages() { | ||||
|     done | ||||
| } | ||||
| 
 | ||||
| initialize | ||||
| 
 | ||||
| # Install additional languages if specified | ||||
| if [[ ! -z "$PAPERLESS_OCR_LANGUAGES"  ]]; then | ||||
| 		install_languages "$PAPERLESS_OCR_LANGUAGES" | ||||
| fi | ||||
| 
 | ||||
| initialize | ||||
| 
 | ||||
| if [[ "$1" != "/"* ]]; then | ||||
| 	exec sudo -HEu paperless python3 manage.py "$@" | ||||
| else | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| bind = '127.0.0.1:8000' | ||||
| bind = '0.0.0.0:8000' | ||||
| backlog = 2048 | ||||
| workers = 3 | ||||
| worker_class = 'sync' | ||||
|  | ||||
| @ -15,7 +15,7 @@ services: | ||||
|       POSTGRES_PASSWORD: paperless | ||||
| 
 | ||||
|   webserver: | ||||
|     image: jonaswinkler/paperless-ng:0.9.5 | ||||
|     image: jonaswinkler/paperless-ng:0.9.11 | ||||
|     restart: always | ||||
|     depends_on: | ||||
|       - db | ||||
|  | ||||
| @ -5,7 +5,7 @@ services: | ||||
|     restart: always | ||||
| 
 | ||||
|   webserver: | ||||
|     image: jonaswinkler/paperless-ng:0.9.5 | ||||
|     image: jonaswinkler/paperless-ng:0.9.11 | ||||
|     restart: always | ||||
|     depends_on: | ||||
|       - broker | ||||
|  | ||||
							
								
								
									
										43
									
								
								docker/hub/docker-compose.tika.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								docker/hub/docker-compose.tika.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | ||||
| version: "3.4" | ||||
| services: | ||||
|   broker: | ||||
|     image: redis:6.0 | ||||
|     restart: always | ||||
| 
 | ||||
|   webserver: | ||||
|     image: jonaswinkler/paperless-ng:0.9.9 | ||||
|     restart: always | ||||
|     depends_on: | ||||
|       - broker | ||||
|     ports: | ||||
|       - 8000:8000 | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-f", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
|       - ./export:/usr/src/paperless/export | ||||
|       - ./consume:/usr/src/paperless/consume | ||||
|     env_file: docker-compose.env | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
| 
 | ||||
|   gotenberg: | ||||
|     image: thecodingmachine/gotenberg | ||||
|     restart: unless-stopped | ||||
|     environment: | ||||
|       DISABLE_GOOGLE_CHROME: 1 | ||||
| 
 | ||||
|   tika: | ||||
|     image: apache/tika | ||||
|     restart: unless-stopped | ||||
| 
 | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
| @ -9,6 +9,8 @@ RUN apt-get update \ | ||||
|   && apt-get -y --no-install-recommends install \ | ||||
| 		build-essential \ | ||||
| 		curl \ | ||||
| 		file \ | ||||
| 		fonts-liberation \ | ||||
| 		ghostscript \ | ||||
| 		gnupg \ | ||||
| 		icc-profiles-free \ | ||||
| @ -18,7 +20,9 @@ RUN apt-get update \ | ||||
| 		libmagic-dev \ | ||||
| 		libpoppler-cpp-dev \ | ||||
| 		libpq-dev \ | ||||
| 		libqpdf-dev \ | ||||
| 		libxml2 \ | ||||
| 		libxslt1-dev \ | ||||
| 		optipng \ | ||||
| 		pngquant \ | ||||
| 		qpdf \ | ||||
| @ -34,7 +38,7 @@ RUN apt-get update \ | ||||
| 		zlib1g \ | ||||
| 	&& pip3 install --upgrade supervisor setuptools \ | ||||
| 	&& pip install --no-cache-dir -r requirements.txt \ | ||||
| 	&& apt-get -y purge build-essential \ | ||||
| 	&& apt-get -y purge build-essential libqpdf-dev \ | ||||
| 	&& apt-get -y autoremove --purge \ | ||||
| 	&& rm -rf /var/lib/apt/lists/* \ | ||||
| 	&& mkdir /var/log/supervisord /var/run/supervisord | ||||
| @ -59,8 +63,11 @@ WORKDIR /usr/src/paperless/src/ | ||||
| 
 | ||||
| RUN sudo -HEu paperless python3 manage.py collectstatic --clear --no-input | ||||
| 
 | ||||
| RUN sudo -HEu paperless python3 manage.py compilemessages | ||||
| 
 | ||||
| VOLUME ["/usr/src/paperless/data", "/usr/src/paperless/media", "/usr/src/paperless/consume", "/usr/src/paperless/export"] | ||||
| ENTRYPOINT ["/sbin/docker-entrypoint.sh"] | ||||
| EXPOSE 8000 | ||||
| CMD ["/usr/local/bin/supervisord", "-c", "/etc/supervisord.conf"] | ||||
| 
 | ||||
| LABEL maintainer="Jonas Winkler <dev@jpwinkler.de>" | ||||
|  | ||||
							
								
								
									
										43
									
								
								docker/local/docker-compose.tika.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								docker/local/docker-compose.tika.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | ||||
| version: "3.4" | ||||
| services: | ||||
|   broker: | ||||
|     image: redis:6.0 | ||||
|     restart: always | ||||
| 
 | ||||
|   webserver: | ||||
|     build: . | ||||
|     restart: always | ||||
|     depends_on: | ||||
|       - broker | ||||
|     ports: | ||||
|       - 8000:8000 | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-f", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
|       - ./export:/usr/src/paperless/export | ||||
|       - ./consume:/usr/src/paperless/consume | ||||
|     env_file: docker-compose.env | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
| 
 | ||||
|   gotenberg: | ||||
|     image: thecodingmachine/gotenberg | ||||
|     restart: unless-stopped | ||||
|     environment: | ||||
|       DISABLE_GOOGLE_CHROME: 1 | ||||
| 
 | ||||
|   tika: | ||||
|     image: apache/tika | ||||
|     restart: unless-stopped | ||||
| 
 | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
| @ -119,8 +119,11 @@ Updating paperless without docker | ||||
| 
 | ||||
| After grabbing the new release and unpacking the contents, do the following: | ||||
| 
 | ||||
| 1.  Update python requirements. Paperless uses | ||||
|     `Pipenv`_ for managing dependencies: | ||||
| 1.  Update dependencies. New paperless version may require additional | ||||
|     dependencies. The dependencies required are listed in the section about  | ||||
|     :ref:`bare metal installations <setup-bare_metal>`. | ||||
| 
 | ||||
| 2.  Update python requirements. If you use Pipenv, this is done with the following steps. | ||||
| 
 | ||||
|     .. code:: shell-session | ||||
| 
 | ||||
| @ -132,20 +135,26 @@ After grabbing the new release and unpacking the contents, do the following: | ||||
|     This creates a new virtual environment (or uses your existing environment) | ||||
|     and installs all dependencies into it. | ||||
| 
 | ||||
| 2.  Collect static files. | ||||
| 3.  Collect static files. | ||||
| 
 | ||||
|     .. code:: shell-session | ||||
| 
 | ||||
|         $ cd src | ||||
|         $ pipenv run python3 manage.py collectstatic --clear | ||||
|      | ||||
| 3.  Migrate the database. | ||||
| 4.  Migrate the database. | ||||
| 
 | ||||
|     .. code:: shell-session | ||||
| 
 | ||||
|         $ cd src | ||||
|         $ pipenv run python3 manage.py migrate | ||||
|      | ||||
| 5.  Update translation files. | ||||
| 
 | ||||
|     .. code:: shell-session | ||||
| 
 | ||||
|         $ cd src | ||||
|         $ pipenv run python3 manage.py compilemessages | ||||
|          | ||||
| Management utilities | ||||
| #################### | ||||
| @ -153,14 +162,14 @@ Management utilities | ||||
| Paperless comes with some management commands that perform various maintenance | ||||
| tasks on your paperless instance. You can invoke these commands either by | ||||
| 
 | ||||
| .. code:: bash | ||||
| .. code:: shell-session | ||||
| 
 | ||||
|     $ cd /path/to/paperless | ||||
|     $ docker-compose run --rm webserver <command> <arguments> | ||||
| 
 | ||||
| or | ||||
| 
 | ||||
| .. code:: bash | ||||
| .. code:: shell-session | ||||
| 
 | ||||
|     $ cd /path/to/paperless/src | ||||
|     $ pipenv run python manage.py <command> <arguments> | ||||
| @ -366,7 +375,7 @@ is specified, the archiver will only process that document. | ||||
| .. note:: | ||||
| 
 | ||||
|     Some documents will cause errors and cannot be converted into PDF/A documents, | ||||
|     such as encrypted PDF documents. The archiver will skip over these Documents | ||||
|     such as encrypted PDF documents. The archiver will skip over these documents | ||||
|     each time it sees them. | ||||
| 
 | ||||
| .. _utilities-encyption: | ||||
|  | ||||
| @ -5,85 +5,6 @@ Advanced topics | ||||
| Paperless offers a couple features that automate certain tasks and make your life | ||||
| easier. | ||||
| 
 | ||||
| Guesswork | ||||
| ######### | ||||
| 
 | ||||
| 
 | ||||
| Any document you put into the consumption directory will be consumed, but if | ||||
| you name the file right, it'll automatically set some values in the database | ||||
| for you.  This is is the logic the consumer follows: | ||||
| 
 | ||||
| 1. Try to find the correspondent, title, and tags in the file name following | ||||
|    the pattern: ``Date - Correspondent - Title - tag,tag,tag.pdf``.  Note that | ||||
|    the format of the date is **rigidly defined** as ``YYYYMMDDHHMMSSZ`` or | ||||
|    ``YYYYMMDDZ``.  The ``Z`` refers "Zulu time" AKA "UTC". | ||||
|    The tags are optional, so the format ``Date - Correspondent - Title.pdf`` | ||||
|    works as well. | ||||
| 2. If that doesn't work, we skip the date and try this pattern: | ||||
|    ``Correspondent - Title - tag,tag,tag.pdf``. | ||||
| 3. If that doesn't work, we try to find the correspondent and title in the file | ||||
|    name following the pattern: ``Correspondent - Title.pdf``. | ||||
| 4. If that doesn't work, just assume that the name of the file is the title. | ||||
| 
 | ||||
| So given the above, the following examples would work as you'd expect: | ||||
| 
 | ||||
| * ``20150314000700Z - Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` | ||||
| * ``20150314Z - Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` | ||||
| * ``Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` | ||||
| * ``Another Company - Letter of Reference.jpg`` | ||||
| * ``Dad's Recipe for Pancakes.png`` | ||||
| 
 | ||||
| These however wouldn't work: | ||||
| 
 | ||||
| * ``2015-03-14 00:07:00 UTC - Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` | ||||
| * ``2015-03-14 - Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` | ||||
| * ``Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` | ||||
| * ``Another Company- Letter of Reference.jpg`` | ||||
| 
 | ||||
| Do I have to be so strict about naming? | ||||
| ======================================= | ||||
| 
 | ||||
| Rather than using the strict document naming rules, one can also set the option | ||||
| ``PAPERLESS_FILENAME_DATE_ORDER`` in ``paperless.conf`` to any date order | ||||
| that is accepted by dateparser_. Doing so will cause ``paperless`` to default | ||||
| to any date format that is found in the title, instead of a date pulled from | ||||
| the document's text, without requiring the strict formatting of the document | ||||
| filename as described above. | ||||
| 
 | ||||
| .. _dateparser: https://github.com/scrapinghub/dateparser/blob/v0.7.0/docs/usage.rst#settings | ||||
| 
 | ||||
| .. _advanced-transforming_filenames: | ||||
| 
 | ||||
| Transforming filenames for parsing | ||||
| ================================== | ||||
| 
 | ||||
| Some devices can't produce filenames that can be parsed by the default | ||||
| parser. By configuring the option ``PAPERLESS_FILENAME_PARSE_TRANSFORMS`` in | ||||
| ``paperless.conf`` one can add transformations that are applied to the filename | ||||
| before it's parsed. | ||||
| 
 | ||||
| The option contains a list of dictionaries of regular expressions (key: | ||||
| ``pattern``) and replacements (key: ``repl``) in JSON format, which are | ||||
| applied in order by passing them to ``re.subn``. Transformation stops | ||||
| after the first match, so at most one transformation is applied. The general | ||||
| syntax is | ||||
| 
 | ||||
| .. code:: python | ||||
| 
 | ||||
|    [{"pattern":"pattern1", "repl":"repl1"}, {"pattern":"pattern2", "repl":"repl2"}, ..., {"pattern":"patternN", "repl":"replN"}] | ||||
| 
 | ||||
| The example below is for a Brother ADS-2400N, a scanner that allows | ||||
| different names to different hardware buttons (useful for handling | ||||
| multiple entities in one instance), but insists on adding ``_<count>`` | ||||
| to the filename. | ||||
| 
 | ||||
| .. code:: python | ||||
| 
 | ||||
|    # Brother profile configuration, support "Name_Date_Count" (the default | ||||
|    # setting) and "Name_Count" (use "Name" as tag and "Count" as title). | ||||
|    PAPERLESS_FILENAME_PARSE_TRANSFORMS=[{"pattern":"^([a-z]+)_(\\d{8})_(\\d{6})_([0-9]+)\\.", "repl":"\\2\\3Z - \\4 - \\1."}, {"pattern":"^([a-z]+)_([0-9]+)\\.", "repl":" - \\2 - \\1."}] | ||||
| 
 | ||||
| 
 | ||||
| .. _advanced-matching: | ||||
| 
 | ||||
| Matching tags, correspondents and document types | ||||
| @ -263,10 +184,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 | ||||
| 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 | ||||
| ``PAPERLESS_FILENAME_FORMAT`` settings variable. | ||||
| ``PAPERLESS_FILENAME_FORMAT`` configuration option. | ||||
| 
 | ||||
| This variable allows you to configure the filename (folders are allowed!) using | ||||
| placeholders. For example, setting | ||||
| This variable allows you to configure the filename (folders are allowed) using | ||||
| placeholders. For example, configuring this to | ||||
| 
 | ||||
| .. code:: bash | ||||
| 
 | ||||
| @ -277,17 +198,16 @@ will create a directory structure as follows: | ||||
| .. code:: | ||||
| 
 | ||||
|     2019/ | ||||
|       my_bank/ | ||||
|         statement-january-0000001.pdf | ||||
|         statement-february-0000002.pdf | ||||
|       My bank/ | ||||
|         Statement January.pdf | ||||
|         Statement February.pdf | ||||
|     2020/ | ||||
|       my_bank/ | ||||
|         statement-january-0000003.pdf | ||||
|       shoe_store/ | ||||
|         my_new_shoes-0000004.pdf | ||||
| 
 | ||||
| Paperless appends the unique identifier of each document to the filename. This | ||||
| avoids filename clashes. | ||||
|       My bank/ | ||||
|         Statement January.pdf | ||||
|         Letter.pdf | ||||
|         Letter_01.pdf | ||||
|       Shoe store/ | ||||
|         My new shoes.pdf | ||||
| 
 | ||||
| .. danger:: | ||||
| 
 | ||||
| @ -298,6 +218,8 @@ avoids filename clashes. | ||||
| Paperless provides the following placeholders withing filenames: | ||||
| 
 | ||||
| * ``{correspondent}``: The name of the correspondent, 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. | ||||
| * ``{created}``: The full date and time the document was created. | ||||
| * ``{created_year}``: Year created only. | ||||
| @ -307,10 +229,15 @@ Paperless provides the following placeholders withing filenames: | ||||
| * ``{added_year}``: Year added only. | ||||
| * ``{added_month}``: Month added only (number 1-12). | ||||
| * ``{added_day}``: Day added only (number 1-31). | ||||
| * ``{tags}``: I don't know how this works. Look at the source. | ||||
| 
 | ||||
| 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:: | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										111
									
								
								docs/api.rst
									
									
									
									
									
								
							
							
						
						
									
										111
									
								
								docs/api.rst
									
									
									
									
									
								
							| @ -13,23 +13,55 @@ available filters and ordering fields. | ||||
| 
 | ||||
| The API provides 5 main endpoints: | ||||
| 
 | ||||
| *   ``/api/documents/``: Full CRUD support, except POSTing new documents. See below. | ||||
| *   ``/api/correspondents/``: Full CRUD support. | ||||
| *   ``/api/document_types/``: Full CRUD support. | ||||
| *   ``/api/documents/``: Full CRUD support, except POSTing new documents. See below. | ||||
| *   ``/api/logs/``: Read-Only. | ||||
| *   ``/api/tags/``: Full CRUD support. | ||||
| 
 | ||||
| All of these endpoints except for the logging endpoint  | ||||
| All of these endpoints except for the logging endpoint | ||||
| allow you to fetch, edit and delete individual objects | ||||
| by appending their primary key to the path, for example ``/api/documents/454/``. | ||||
| 
 | ||||
| The objects served by the document endpoint contain the following fields: | ||||
| 
 | ||||
| *   ``id``: ID of the document. Read-only. | ||||
| *   ``title``: Title of the document. | ||||
| *   ``content``: Plain text content of the document. | ||||
| *   ``tags``: List of IDs of tags assigned to this document, or empty list. | ||||
| *   ``document_type``: Document type of this document, or null. | ||||
| *   ``correspondent``:  Correspondent of this document or null. | ||||
| *   ``created``: The date at which this document was created. | ||||
| *   ``modified``: The date at which this document was last edited in paperless. Read-only. | ||||
| *   ``added``: The date at which this document was added to paperless. Read-only. | ||||
| *   ``archive_serial_number``: The identifier of this document in a physical document archive. | ||||
| *   ``original_file_name``: Verbose filename of the original document. Read-only. | ||||
| *   ``archived_file_name``: Verbose filename of the archived document. Read-only. Null if no archived document is available. | ||||
| 
 | ||||
| 
 | ||||
| Downloading documents | ||||
| ##################### | ||||
| 
 | ||||
| In addition to that, the document endpoint offers these additional actions on | ||||
| individual documents: | ||||
| 
 | ||||
| *   ``/api/documents/<pk>/download/``: Download the original document. | ||||
| *   ``/api/documents/<pk>/thumb/``: Download the PNG thumbnail of a document. | ||||
| *   ``/api/documents/<pk>/preview/``: Display the original document inline, | ||||
| *   ``/api/documents/<pk>/download/``: Download the document. | ||||
| *   ``/api/documents/<pk>/preview/``: Display the document inline, | ||||
|     without downloading it. | ||||
| *   ``/api/documents/<pk>/thumb/``: Download the PNG thumbnail of a document. | ||||
| 
 | ||||
| Paperless generates archived PDF/A documents from consumed files and stores both | ||||
| the original files as well as the archived files. By default, the endpoints | ||||
| for previews and downloads serve the archived file, if it is available. | ||||
| Otherwise, the original file is served. | ||||
| Some document cannot be archived. | ||||
| 
 | ||||
| The endpoints correctly serve the response header fields ``Content-Disposition`` | ||||
| and ``Content-Type`` to indicate the filename for download and the type of content of | ||||
| the document. | ||||
| 
 | ||||
| In order to download or preview the original document when an archied document is available, | ||||
| supply the query parameter ``original=true``. | ||||
| 
 | ||||
| .. hint:: | ||||
| 
 | ||||
| @ -38,13 +70,43 @@ individual documents: | ||||
|     are in place. However, if you use these old URLs to access documents, you | ||||
|     should update your app or script to use the new URLs. | ||||
| 
 | ||||
| .. note:: | ||||
| 
 | ||||
|     The document endpoint provides tags, document types and correspondents as | ||||
|     ids in their corresponding fields. These are writeable. Paperless also | ||||
|     offers read-only objects for assigned tags, types and correspondents, | ||||
|     however, these might be removed in the future. As for now, the front end | ||||
|     requires them. | ||||
| Getting document metadata | ||||
| ######################### | ||||
| 
 | ||||
| The api also has an endpoint to retrieve read-only metadata about specific documents. this | ||||
| information is not served along with the document objects, since it requires reading | ||||
| files and would therefore slow down document lists considerably. | ||||
| 
 | ||||
| Access the metadata of a document with an ID ``id`` at ``/api/documents/<id>/metadata/``. | ||||
| 
 | ||||
| The endpoint reports the following data: | ||||
| 
 | ||||
| *   ``original_checksum``: MD5 checksum of the original document. | ||||
| *   ``original_size``: Size of the original document, in bytes. | ||||
| *   ``original_mime_type``: Mime type of the original document. | ||||
| *   ``media_filename``: Current filename of the document, under which it is stored inside the media directory. | ||||
| *   ``has_archive_version``: True, if this document is archived, false otherwise. | ||||
| *   ``original_metadata``: A list of metadata associated with the original document. See below. | ||||
| *   ``archive_checksum``: MD5 checksum of the archived document, or null. | ||||
| *   ``archive_size``: Size of the archived document in bytes, or null. | ||||
| *   ``archive_metadata``: Metadata associated with the archived document, or null. See below. | ||||
| 
 | ||||
| File metadata is reported as a list of objects in the following form: | ||||
| 
 | ||||
| .. code:: json | ||||
| 
 | ||||
|     [ | ||||
|         { | ||||
|             "namespace": "http://ns.adobe.com/pdf/1.3/", | ||||
|             "prefix": "pdf", | ||||
|             "key": "Producer", | ||||
|             "value": "SparklePDF, Fancy edition" | ||||
|         }, | ||||
|     ] | ||||
| 
 | ||||
| ``namespace`` and ``prefix`` can be null. The actual metadata reported depends on the file type and the metadata | ||||
| available in that specific document. Paperless only reports PDF metadata at this point. | ||||
| 
 | ||||
| Authorization | ||||
| ############# | ||||
| @ -54,11 +116,11 @@ The REST api provides three different forms of authentication. | ||||
| 1.  Basic authentication | ||||
| 
 | ||||
|     Authorize by providing a HTTP header in the form | ||||
|      | ||||
| 
 | ||||
|     .. code:: | ||||
| 
 | ||||
|         Authorization: Basic <credentials> | ||||
|      | ||||
| 
 | ||||
|     where ``credentials`` is a base64-encoded string of ``<username>:<password>`` | ||||
| 
 | ||||
| 2.  Session authentication | ||||
| @ -79,7 +141,7 @@ The REST api provides three different forms of authentication. | ||||
|     .. code:: | ||||
| 
 | ||||
|         Authorization: Token <token> | ||||
|      | ||||
| 
 | ||||
|     Tokens can be managed and revoked in the paperless admin. | ||||
| 
 | ||||
| Searching for documents | ||||
| @ -111,7 +173,7 @@ Result list object returned by the endpoint: | ||||
|         "page_count": 1, | ||||
|         "corrected_query": "", | ||||
|         "results": [ | ||||
|              | ||||
| 
 | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| @ -131,12 +193,12 @@ Result object: | ||||
|     { | ||||
|         "id": 1, | ||||
|         "highlights": [ | ||||
|              | ||||
| 
 | ||||
|         ], | ||||
|         "score": 6.34234, | ||||
|         "rank": 23, | ||||
|         "document": { | ||||
|              | ||||
| 
 | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| @ -159,21 +221,16 @@ Each fragment contains a list of strings, and some of them are marked as a highl | ||||
| 
 | ||||
|     [ | ||||
|         [ | ||||
|             {"text": "This is a sample text with a "}, | ||||
|             {"text": "highlighted", "term": 0}, | ||||
|             {"text": " word."} | ||||
|             {"text": "This is a sample text with a ", "highlight": false}, | ||||
|             {"text": "highlighted", "highlight": true}, | ||||
|             {"text": " word.", "highlight": false} | ||||
|         ], | ||||
|         [ | ||||
|             {"text": "Another", "term": 1}, | ||||
|             {"text": " fragment with a highlight."} | ||||
|             {"text": "Another", "highlight": true}, | ||||
|             {"text": " fragment with a highlight.", "highlight": false} | ||||
|         ] | ||||
|     ] | ||||
|      | ||||
| 
 | ||||
| 
 | ||||
| When ``term`` is present within a string, the word within ``text`` should be highlighted. | ||||
| The term index groups multiple matches together and words with the same index | ||||
| should get identical highlighting. | ||||
| A client may use this example to produce the following output: | ||||
| 
 | ||||
| ... This is a sample text with a **highlighted** word. ... **Another** fragment with a highlight. ... | ||||
|  | ||||
| @ -5,6 +5,206 @@ | ||||
| Changelog | ||||
| ********* | ||||
| 
 | ||||
| paperless-ng 0.9.12 | ||||
| ################### | ||||
| 
 | ||||
| * Paperless localization | ||||
| 
 | ||||
|   * Thanks to the combined efforts of many users, Paperless is now available in English, Dutch, French and German. | ||||
| 
 | ||||
| * Thanks to `Jo Vandeginste`_, Paperless has optional support for Office documents such as .docx, .doc, .odt and more. | ||||
| 
 | ||||
|   * See the :ref:`configuration<configuration-tika>` on how to enable this feature. This feature requires two additional services | ||||
|     (one for parsing Office documents and metadata extraction and another for converting Office documents to PDF), and is therefore | ||||
|     not enabled on default installations. | ||||
| 
 | ||||
| * Dark mode | ||||
| 
 | ||||
|   * Thanks to `Michael Shamoon`_, paperless now has a dark mode. Configuration is available in the settings. | ||||
| 
 | ||||
| * Other changes and additions | ||||
| 
 | ||||
|   * The PDF viewer now uses a local copy of some dependencies instead of fetching them from the internet. Thanks to `slorenz`_. | ||||
|   * Revamped search bar styling thanks to `Michael Shamoon`_. | ||||
|   * Sorting in the document list by clicking on table headers. | ||||
|   * A button was added to the document detail page that assigns a new ASN to a document. | ||||
|   * Form field validation: When providing invalid input in a form (such as a duplicate ASN or no name), paperless now has visual | ||||
|     indicators and clearer error messages about what's wrong. | ||||
|   * Paperless disables buttons with network actions (such as save and delete) when a network action is active. This indicates that | ||||
|     something is happening and prevents double clicking. | ||||
| 
 | ||||
| * Fixes | ||||
| 
 | ||||
|   * Paperless was unable to save views when "Not assigned" was chosen in one of the filter dropdowns. | ||||
|   * Clearer error messages when pre and post consumption scripts do not exist. | ||||
|   * The post consumption script is executed later in the consumption process. Before the change, an ID was passed to the script referring to | ||||
|     a document that did not yet exist in the database. | ||||
| 
 | ||||
| paperless-ng 0.9.11 | ||||
| ################### | ||||
| 
 | ||||
| * Fixed an issue with the docker image not starting at all due to a configuration change of the web server. | ||||
| 
 | ||||
| 
 | ||||
| paperless-ng 0.9.10 | ||||
| ################### | ||||
| 
 | ||||
| * Bulk editing | ||||
| 
 | ||||
|   * Thanks to `Michael Shamoon`_, we've got a new interface for the bulk editor. | ||||
|   * There are some configuration options in the settings to alter the behavior. | ||||
| 
 | ||||
| * Other changes and additions | ||||
|    | ||||
|   * Thanks to `zjean`_, paperless now publishes a webmanifest, which is useful for adding the application to home screens on mobile devices. | ||||
|   * The Paperless-ng logo now navigates to the dashboard. | ||||
|   * Filter for documents that don't have any correspondents, types or tags assigned. | ||||
|   * Tags, types and correspondents are now sorted case insensitive. | ||||
|   * Lots of preparation work for localization support. | ||||
| 
 | ||||
| * Fixes | ||||
| 
 | ||||
|   * Added missing dependencies for Raspberry Pi builds. | ||||
|   * Fixed an issue with plain text file consumption: Thumbnail generation failed due to missing fonts. | ||||
|   * An issue with the search index reporting missing documents after bulk deletes was fixed. | ||||
|   * Issue with the tag selector not clearing input correctly. | ||||
|   * The consumer used to stop working when encountering an incomplete classifier model file. | ||||
| 
 | ||||
| .. note:: | ||||
| 
 | ||||
|   The bulk delete operations did not update the search index. Therefore, documents that you deleted remained in the index and | ||||
|   caused the search to return messages about missing documents when searching. Further bulk operations will properly update | ||||
|   the index. | ||||
|    | ||||
|   However, this change is not retroactive: If you used the delete method of the bulk editor, you need to reindex your search index | ||||
|   by :ref:`running the management command document_index with the argument reindex <administration-index>`. | ||||
| 
 | ||||
| paperless-ng 0.9.9 | ||||
| ################## | ||||
| 
 | ||||
| Christmas release! | ||||
| 
 | ||||
| * Bulk editing | ||||
| 
 | ||||
|   * Paperless now supports bulk editing. | ||||
|   * The following operations are available: Add and remove correspondents, tags, document types from selected documents, as well as mass-deleting documents. | ||||
|   * We've got a more fancy UI in the works that makes these features more accessible, but that's not quite ready yet. | ||||
| 
 | ||||
| * Searching | ||||
| 
 | ||||
|   * Paperless now supports searching for similar documents ("More like this") both from the document detail page as well as from individual search results. | ||||
|   * A search score indicates how well a document matches the search query, or how similar a document is to a given reference document. | ||||
| 
 | ||||
| * Other additions and changes | ||||
| 
 | ||||
|   * Clarification in the UI that the fields "Match" and "Is insensitive" are not relevant for the Auto matching algorithm. | ||||
|   * New select interface for tags, types and correspondents allows filtering. This also improves tag selection. Thanks again to `Michael Shamoon`_! | ||||
|   * Page navigation controls for the document viewer, thanks to `Michael Shamoon`_. | ||||
|   * Layout changes to the small cards document list. | ||||
|   * The dashboard now displays the username (or full name if specified in the admin) on the dashboard. | ||||
| 
 | ||||
| * Fixes | ||||
| 
 | ||||
|   * An error that caused the document importer to crash was fixed. | ||||
|   * An issue with changes not being possible when ``PAPERLESS_COOKIE_PREFIX`` is used was fixed. | ||||
|   * The date selection filters now allow manual entry of dates. | ||||
| 
 | ||||
| * Feature Removal | ||||
| 
 | ||||
|   * Most of the guesswork features have been removed. Paperless no longer tries to extract correspondents and tags from file names. | ||||
| 
 | ||||
| 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 | ||||
| ################## | ||||
| 
 | ||||
| This release focusses primarily on many small issues with the UI. | ||||
| 
 | ||||
| * Front end | ||||
| 
 | ||||
|   * Paperless now has proper window titles. | ||||
|   * Fixed an issue with the small cards when more than 7 tags were used. | ||||
|   * Navigation of the "Show all" links adjusted. They navigate to the saved view now, if available in the sidebar. | ||||
|   * Some indication on the document lists that a filter is active was added. | ||||
|   * There's a new filter to filter for documents that do *not* have a certain tag. | ||||
|   * The file upload box now shows upload progress. | ||||
|   * The document edit page was reorganized. | ||||
|   * The document edit page shows various information about a document. | ||||
|   * An issue with the height of the preview was fixed. | ||||
|   * Table issues with too long document titles fixed. | ||||
| 
 | ||||
| * API | ||||
| 
 | ||||
|   * The API now serves file names with documents. | ||||
|   * The API now serves various metadata about documents. | ||||
|   * API documentation updated. | ||||
| 
 | ||||
| * Other | ||||
| 
 | ||||
|   * Fixed an issue with the docker image when a non-standard PostgreSQL port was used. | ||||
|   * The docker image was trying check for installed languages before actually installing them. | ||||
|   * ``FILENAME_FORMAT`` placeholder for document types. | ||||
|   * The filename formatter is now less restrictive with file names and tries to | ||||
|     conserve the original correspondents, types and titles as much as possible. | ||||
|   * The filename formatter does not include the document ID in filenames anymore. It will | ||||
|     rather append ``_01``, ``_02``, etc when it detects duplicate filenames. | ||||
| 
 | ||||
| .. note:: | ||||
| 
 | ||||
|   The changes to the filename format will apply to newly added documents and changed documents. | ||||
|   If you want all files to reflect these changes, execute the ``document_renamer`` management | ||||
|   command. | ||||
| 
 | ||||
| 
 | ||||
| paperless-ng 0.9.5 | ||||
| ################## | ||||
| 
 | ||||
| @ -800,6 +1000,11 @@ bulk of the work on this big change. | ||||
| 
 | ||||
| * Initial release | ||||
| 
 | ||||
| .. _slorenz: https://github.com/sisao | ||||
| .. _Jo Vandeginste: https://github.com/jovandeginste | ||||
| .. _zjean: https://github.com/zjean | ||||
| .. _rYR79435: https://github.com/rYR79435 | ||||
| .. _Michael Shamoon: https://github.com/shamoon | ||||
| .. _jayme-github: http://github.com/jayme-github | ||||
| .. _Brian Conn: https://github.com/TheConnMan | ||||
| .. _Christopher Luu: https://github.com/nuudles | ||||
|  | ||||
| @ -152,6 +152,16 @@ PAPERLESS_AUTO_LOGIN_USERNAME=<username> | ||||
| 
 | ||||
|     Defaults to none, which disables this feature. | ||||
| 
 | ||||
| 
 | ||||
| PAPERLESS_COOKIE_PREFIX=<str> | ||||
|     Specify a prefix that is added to the cookies used by paperless to identify | ||||
|     the currently logged in user. This is useful for when you're running two | ||||
|     instances of paperless on the same host. | ||||
| 
 | ||||
|     After changing this, you will have to login again. | ||||
| 
 | ||||
|     Defaults to ``""``, which does not alter the cookie names. | ||||
| 
 | ||||
| .. _configuration-ocr: | ||||
| 
 | ||||
| OCR settings | ||||
| @ -267,6 +277,35 @@ PAPERLESS_OCR_USER_ARG=<json> | ||||
| 
 | ||||
|         {"deskew": true, "optimize": 3, "unpaper_args": "--pre-rotate 90"}     | ||||
|      | ||||
| .. _configuration-tika: | ||||
| 
 | ||||
| Tika settings | ||||
| ############# | ||||
| 
 | ||||
| Paperless can make use of `Tika <https://tika.apache.org/>`_ and  | ||||
| `Gotenberg <https://thecodingmachine.github.io/gotenberg/>`_ for parsing and | ||||
| converting "Office" documents (such as ".doc", ".xlsx" and ".odt"). If you | ||||
| wish to use this, you must provide a Tika server and a Gotenberg server, | ||||
| configure their endpoints, and enable the feature. | ||||
| 
 | ||||
| If you run paperless on docker, you can add those services to the docker-compose | ||||
| file (see the examples provided). | ||||
| 
 | ||||
| PAPERLESS_TIKA_ENABLED=<bool> | ||||
|     Enable (or disable) the Tika parser. | ||||
| 
 | ||||
|     Defaults to false. | ||||
| 
 | ||||
| PAPERLESS_TIKA_ENDPOINT=<url> | ||||
|     Set the endpoint URL were Paperless can reach your Tika server. | ||||
| 
 | ||||
|     Defaults to "http://localhost:9998". | ||||
| 
 | ||||
| PAPERLESS_TIKA_GOTENBERG_ENDPOINT=<url> | ||||
|     Set the endpoint URL were Paperless can reach your Gotenberg server. | ||||
| 
 | ||||
|     Defaults to "http://localhost:3000". | ||||
| 
 | ||||
|      | ||||
| Software tweaks | ||||
| ############### | ||||
| @ -309,11 +348,14 @@ PAPERLESS_TIME_ZONE=<timezone> | ||||
|     Defaults to UTC. | ||||
| 
 | ||||
| 
 | ||||
| .. _configuration-polling: | ||||
| 
 | ||||
| PAPERLESS_CONSUMER_POLLING=<num> | ||||
|     If paperless won't find documents added to your consume folder, it might | ||||
|     not be able to automatically detect filesystem changes. In that case, | ||||
|     specify a polling interval in seconds here, which will then cause paperless | ||||
|     to periodically check your consumption directory for changes. | ||||
|     to periodically check your consumption directory for changes. This will also | ||||
|     disable listening for file system changes with ``inotify``. | ||||
| 
 | ||||
|     Defaults to 0, which disables polling and uses filesystem notifications. | ||||
| 
 | ||||
| @ -390,11 +432,15 @@ PAPERLESS_FILENAME_DATE_ORDER=<format> | ||||
| 
 | ||||
|     Defaults to none, which disables this feature. | ||||
| 
 | ||||
| PAPERLESS_FILENAME_PARSE_TRANSFORMS | ||||
|     Transforms filenames before they are processed by paperless. See | ||||
|     :ref:`advanced-transforming_filenames` for details. | ||||
| PAPERLESS_THUMBNAIL_FONT_NAME=<filename> | ||||
|     Paperless creates thumbnails for plain text files by rendering the content | ||||
|     of the file on an image and uses a predefined font for that. This | ||||
|     font can be changed here. | ||||
| 
 | ||||
|     Note that this won't have any effect on already generated thumbnails. | ||||
| 
 | ||||
|     Defaults to ``/usr/share/fonts/liberation/LiberationSerif-Regular.ttf``. | ||||
| 
 | ||||
|     Defaults to none, which disables this feature. | ||||
| 
 | ||||
| Binaries | ||||
| ######## | ||||
|  | ||||
| @ -118,114 +118,80 @@ This will test and assemble everything and also build and tag a docker image. | ||||
| Extending Paperless | ||||
| =================== | ||||
| 
 | ||||
| .. warning:: | ||||
| Paperless does not have any fancy plugin systems and will probably never have. However, | ||||
| some parts of the application have been designed to allow easy integration of additional | ||||
| features without any modification to the base code. | ||||
| 
 | ||||
|     This section is not updated to paperless-ng yet. | ||||
| Making custom parsers | ||||
| --------------------- | ||||
| 
 | ||||
| For the most part, Paperless is monolithic, so extending it is often best | ||||
| managed by way of modifying the code directly and issuing a pull request on | ||||
| `GitHub`_.  However, over time the project has been evolving to be a little | ||||
| more "pluggable" so that users can write their own stuff that talks to it. | ||||
| Paperless uses parsers to add documents to paperless. A parser is responsible for: | ||||
| 
 | ||||
| .. _GitHub: https://github.com/the-paperless-project/paperless | ||||
| *   Retrieve the content from the original | ||||
| *   Create a thumbnail | ||||
| *   Optional: Retrieve a created date from the original | ||||
| *   Optional: Create an archived document from the original | ||||
| 
 | ||||
| Custom parsers can be added to paperless to support more file types. In order to do that, | ||||
| you need to write the parser itself and announce its existence to paperless. | ||||
| 
 | ||||
| .. _extending-parsers: | ||||
| 
 | ||||
| Parsers | ||||
| ------- | ||||
| 
 | ||||
| You can leverage Paperless' consumption model to have it consume files *other* | ||||
| than ones handled by default like ``.pdf``, ``.jpg``, and ``.tiff``.  To do so, | ||||
| you simply follow Django's convention of creating a new app, with a few key | ||||
| requirements. | ||||
| 
 | ||||
| 
 | ||||
| .. _extending-parsers-parserspy: | ||||
| 
 | ||||
| parsers.py | ||||
| .......... | ||||
| 
 | ||||
| In this file, you create a class that extends | ||||
| ``documents.parsers.DocumentParser`` and go about implementing the three | ||||
| required methods: | ||||
| 
 | ||||
| * ``get_thumbnail()``: Returns the path to a file we can use as a thumbnail for | ||||
|   this document. | ||||
| * ``get_text()``: Returns the text from the document and only the text. | ||||
| * ``get_date()``: If possible, this returns the date of the document, otherwise | ||||
|   it should return ``None``. | ||||
| 
 | ||||
| 
 | ||||
| .. _extending-parsers-signalspy: | ||||
| 
 | ||||
| signals.py | ||||
| .......... | ||||
| 
 | ||||
| At consumption time, Paperless emits a ``document_consumer_declaration`` | ||||
| signal which your module has to react to in order to let the consumer know | ||||
| whether or not it's capable of handling a particular file.  Think of it like | ||||
| this: | ||||
| 
 | ||||
| 1. Consumer finds a file in the consumption directory. | ||||
| 2. It asks all the available parsers: *"Hey, can you handle this file?"* | ||||
| 3. Each parser responds with either ``None`` meaning they can't handle the | ||||
|    file, or a dictionary in the following format: | ||||
| The parser itself must extend ``documents.parsers.DocumentParser`` and must implement the | ||||
| methods ``parse`` and ``get_thumbnail``. You can provide your own implementation to | ||||
| ``get_date`` if you don't want to rely on paperless' default date guessing mechanisms. | ||||
| 
 | ||||
| .. code:: python | ||||
| 
 | ||||
|     { | ||||
|         "parser": <the class name>, | ||||
|         "weight": <an integer> | ||||
|     } | ||||
|     class MyCustomParser(DocumentParser): | ||||
| 
 | ||||
| The consumer compares the ``weight`` values from all respondents and uses the | ||||
| class with the highest value to consume the document.  The default parser, | ||||
| ``RasterisedDocumentParser`` has a weight of ``0``. | ||||
|         def parse(self, document_path, mime_type): | ||||
|             # This method does not return anything. Rather, you should assign | ||||
|             # whatever you got from the document to the following fields: | ||||
| 
 | ||||
|             # The content of the document. | ||||
|             self.text = "content" | ||||
|              | ||||
|             # Optional: path to a PDF document that you created from the original. | ||||
|             self.archive_path = os.path.join(self.tempdir, "archived.pdf") | ||||
| 
 | ||||
| .. _extending-parsers-appspy: | ||||
|             # Optional: "created" date of the document. | ||||
|             self.date = get_created_from_metadata(document_path) | ||||
| 
 | ||||
| apps.py | ||||
| ....... | ||||
|         def get_thumbnail(self, document_path, mime_type): | ||||
|             # This should return the path to a thumbnail you created for this | ||||
|             # document. | ||||
|             return os.path.join(self.tempdir, "thumb.png") | ||||
| 
 | ||||
| This is a standard Django file, but you'll need to add some code to it to | ||||
| connect your parser to the ``document_consumer_declaration`` signal. | ||||
| If you encounter any issues during parsing, raise a ``documents.parsers.ParseError``. | ||||
| 
 | ||||
| The ``self.tempdir`` directory is a temporary directory that is guaranteed to be empty | ||||
| and removed after consumption finished. You can use that directory to store any | ||||
| intermediate files and also use it to store the thumbnail / archived document. | ||||
| 
 | ||||
| .. _extending-parsers-finally: | ||||
| 
 | ||||
| Finally | ||||
| ....... | ||||
| 
 | ||||
| The last step is to update ``settings.py`` to include your new module. | ||||
| Eventually, this will be dynamic, but at the moment, you have to edit the | ||||
| ``INSTALLED_APPS`` section manually.  Simply add the path to your AppConfig to | ||||
| the list like this: | ||||
| After that, you need to announce your parser to paperless. You need to connect a | ||||
| handler to the ``document_consumer_declaration`` signal. Have a look in the file | ||||
| ``src/paperless_tesseract/apps.py`` on how that's done. The handler is a method | ||||
| that returns information about your parser: | ||||
| 
 | ||||
| .. code:: python | ||||
| 
 | ||||
|     INSTALLED_APPS = [ | ||||
|         ... | ||||
|         "my_module.apps.MyModuleConfig", | ||||
|         ... | ||||
|     ] | ||||
|     def myparser_consumer_declaration(sender, **kwargs): | ||||
|         return { | ||||
|             "parser": MyCustomParser, | ||||
|             "weight": 0, | ||||
|             "mime_types": { | ||||
|                 "application/pdf": ".pdf", | ||||
|                 "image/jpeg": ".jpg", | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
| Order doesn't matter, but generally it's a good idea to place your module lower | ||||
| in the list so that you don't end up accidentally overriding project defaults | ||||
| somewhere. | ||||
| *   ``parser`` is a reference to a class that extends ``DocumentParser``. | ||||
| 
 | ||||
| *   ``weight`` is used whenever two or more parsers are able to parse a file: The parser with | ||||
|     the higher weight wins. This can be used to override the parsers provided by | ||||
|     paperless. | ||||
| 
 | ||||
| .. _extending-parsers-example: | ||||
| 
 | ||||
| An Example | ||||
| .......... | ||||
| 
 | ||||
| The core Paperless functionality is based on this design, so if you want to see | ||||
| what a parser module should look like, have a look at `parsers.py`_, | ||||
| `signals.py`_, and `apps.py`_ in the `paperless_tesseract`_ module. | ||||
| 
 | ||||
| .. _parsers.py: https://github.com/the-paperless-project/paperless/blob/master/src/paperless_tesseract/parsers.py | ||||
| .. _signals.py: https://github.com/the-paperless-project/paperless/blob/master/src/paperless_tesseract/signals.py | ||||
| .. _apps.py: https://github.com/the-paperless-project/paperless/blob/master/src/paperless_tesseract/apps.py | ||||
| .. _paperless_tesseract: https://github.com/the-paperless-project/paperless/blob/master/src/paperless_tesseract/ | ||||
| *   ``mime_types`` is a dictionary. The keys are the mime types your parser supports and the value | ||||
|     is the default file extension that paperless should use when storing files and serving them for | ||||
|     download. We could guess that from the file extensions, but some mime types have many extensions | ||||
|     associated with them and the python methods responsible for guessing the extension do not always | ||||
|     return the same value. | ||||
|  | ||||
| @ -73,11 +73,17 @@ in your browser and paperless has to do much less work to serve the data. | ||||
| 
 | ||||
| **Q:** *How do I install paperless-ng on Raspberry Pi?* | ||||
| 
 | ||||
| **A:** There is not docker image for ARM available. If you know how to build | ||||
| **A:** There is no docker image for ARM available. If you know how to build | ||||
| that automatically, I'm all ears. For now, you have to grab the latest release | ||||
| archive from the project page and build the image yourself. The release comes | ||||
| with the front end already compiled, so you don't have to do this on the Pi. | ||||
| 
 | ||||
| **Q:** *How do I run this on unRaid?* | ||||
| 
 | ||||
| **A:** Head over to `<https://github.com/selfhosters/unRAID-CA-templates>`_, | ||||
| `Uli Fahrer <https://github.com/Tooa>`_ created a container template for that. | ||||
| I don't exactly know how to use that though, since I don't use unRaid. | ||||
| 
 | ||||
| **Q:** *How do I run this on my toaster?* | ||||
| 
 | ||||
| **A:** I honestly don't know! As for all other devices that might be able | ||||
|  | ||||
| @ -25,6 +25,8 @@ that works right for you based on recommendations from other Paperless users. | ||||
| +---------+----------------+-----+-----+-----+----------------+ | ||||
| | Fujitsu | `ix500`_       | yes |     | yes | `eonist`_      | | ||||
| +---------+----------------+-----+-----+-----+----------------+ | ||||
| | Epson   | `WF-7710DWF`_  | yes |     | yes | `Skylinar`_    | | ||||
| +---------+----------------+-----+-----+-----+----------------+ | ||||
| | Fujitsu | `S1300i`_      | yes |     | yes | `jonaswinkler`_| | ||||
| +---------+----------------+-----+-----+-----+----------------+ | ||||
| 
 | ||||
| @ -32,7 +34,8 @@ that works right for you based on recommendations from other Paperless users. | ||||
| .. _MFC-J6930DW: https://www.brother.ca/en/p/MFCJ6930DW | ||||
| .. _MFC-J5910DW: https://www.brother.co.uk/printers/inkjet-printers/mfcj5910dw | ||||
| .. _MFC-9142CDN: https://www.brother.co.uk/printers/laser-printers/mfc9140cdn | ||||
| .. _ix500: https://www.fujitsu.com/global/products/computing/peripheral/scanners/scansnap/ix500/ | ||||
| .. _ix500: http://www.fujitsu.com/us/products/computing/peripheral/scanners/scansnap/ix500/ | ||||
| .. _WF-7710DWF: https://www.epson.de/en/products/printers/inkjet-printers/for-home/workforce-wf-7710dwf | ||||
| .. _S1300i: https://www.fujitsu.com/global/products/computing/peripheral/scanners/soho/s1300i/ | ||||
| 
 | ||||
| .. _danielquinn: https://github.com/danielquinn | ||||
| @ -40,4 +43,5 @@ that works right for you based on recommendations from other Paperless users. | ||||
| .. _bmsleight: https://github.com/bmsleight | ||||
| .. _eonist: https://github.com/eonist | ||||
| .. _REOLDEV: https://github.com/REOLDEV | ||||
| .. _Skylinar: https://github.com/Skylinar | ||||
| .. _jonaswinkler: https://github.com/jonaswinkler | ||||
|  | ||||
| @ -120,6 +120,8 @@ The `bare metal route`_ is more complicated to setup but makes it easier | ||||
| should you want to contribute some code back. You need to configure and | ||||
| run the above mentioned components yourself. | ||||
| 
 | ||||
| .. _setup-docker_route: | ||||
| 
 | ||||
| Docker Route | ||||
| ============ | ||||
| 
 | ||||
| @ -177,6 +179,14 @@ Docker Route | ||||
| 
 | ||||
|         You can use any settings from the file ``paperless.conf`` in this file. | ||||
|         Have a look at :ref:`configuration` to see whats available. | ||||
|      | ||||
|     .. caution:: | ||||
| 
 | ||||
|         Certain file systems such as NFS network shares don't support file system | ||||
|         notifications with ``inotify``. When storing the consumption directory | ||||
|         on such a file system, paperless will be unable to pick up new files | ||||
|         with the default configuration. You will need to use ``PAPERLESS_CONSUMER_POLLING``, | ||||
|         which will disable inotify. See :ref:`here <configuration-polling>`. | ||||
| 
 | ||||
| 4.  Run ``docker-compose up -d``. This will create and start the necessary | ||||
|     containers. This will also build the image of paperless if you grabbed the | ||||
| @ -219,8 +229,9 @@ writing. Windows is not and will never be supported. | ||||
|     *   ``python3-pip``, optionally ``pipenv`` for package installation | ||||
|     *   ``python3-dev`` | ||||
| 
 | ||||
|     *   ``fonts-liberation`` for generating thumbnails for plain text files | ||||
|     *   ``imagemagick`` >= 6 for PDF conversion | ||||
|     *   ``optipng`` for optimising thumbnails | ||||
|     *   ``optipng`` for optimizing thumbnails | ||||
|     *   ``gnupg`` for handling encrypted documents | ||||
|     *   ``libpoppler-cpp-dev`` for PDF to text conversion | ||||
|     *   ``libmagic-dev`` for mime type detection | ||||
| @ -240,8 +251,7 @@ writing. Windows is not and will never be supported. | ||||
|     *   ``tesseract-ocr`` language packs (``tesseract-ocr-eng``, ``tesseract-ocr-deu``, etc) | ||||
| 
 | ||||
|     You will also need ``build-essential``, ``python3-setuptools`` and ``python3-wheel`` | ||||
|     for installing some of the python dependencies. You can remove that | ||||
|     again after installation. | ||||
|     for installing some of the python dependencies. | ||||
| 
 | ||||
| 2.  Install ``redis`` >= 5.0 and configure it to start automatically. | ||||
| 
 | ||||
| @ -290,6 +300,9 @@ writing. Windows is not and will never be supported. | ||||
|          | ||||
|         # This creates the database schema. | ||||
|         python3 manage.py migrate | ||||
|          | ||||
|         # This creates the translation files for paperless. | ||||
|         python3 manage.py compilemessages | ||||
| 
 | ||||
|         # This creates your first paperless user | ||||
|         python3 manage.py createsuperuser | ||||
| @ -460,6 +473,15 @@ management commands as below. | ||||
|     load data from an old database schema in SQLite into a newer database | ||||
|     schema in PostgreSQL, you will run into trouble. | ||||
| 
 | ||||
| .. warning:: | ||||
| 
 | ||||
|     On some database fields, PostgreSQL enforces predefined limits on maximum | ||||
|     length, whereas SQLite does not. The fields in question are the title of documents | ||||
|     (128 characters), names of document types, tags and correspondents (128 characters), | ||||
|     and filenames (1024 characters). If you have data in these fields that surpasses these | ||||
|     limits, migration to PostgreSQL is not possible and will fail with an error. | ||||
| 
 | ||||
| 
 | ||||
| 1.  Stop paperless, if it is running. | ||||
| 2.  Tell paperless to use PostgreSQL: | ||||
| 
 | ||||
|  | ||||
| @ -34,12 +34,15 @@ directory at startup, but won't find any other files added later, check out | ||||
| the configuration file and enable filesystem polling with the setting | ||||
| ``PAPERLESS_CONSUMER_POLLING``. | ||||
| 
 | ||||
| This will disable listening to filesystem changes with inotify and paperless will | ||||
| manually check the consumption directory for changes instead. | ||||
| 
 | ||||
| Operation not permitted | ||||
| ####################### | ||||
| 
 | ||||
| You might see errors such as: | ||||
| 
 | ||||
| .. code:: | ||||
| .. code:: shell-session | ||||
| 
 | ||||
|     chown: changing ownership of '../export': Operation not permitted | ||||
| 
 | ||||
| @ -49,3 +52,29 @@ to these folders. This happens when pointing these directories to NFS shares, | ||||
| for example. | ||||
| 
 | ||||
| Ensure that `chown` is possible on these directories. | ||||
| 
 | ||||
| Classifier error: No training data available | ||||
| ############################################ | ||||
| 
 | ||||
| This indicates that the Auto matching algorithm found no documents to learn from. | ||||
| This may have two reasons: | ||||
| 
 | ||||
| *   You don't use the Auto matching algorithm: The error can be safely ignored in this case. | ||||
| *   You are using the Auto matching algorithm: The classifier explicitly excludes documents | ||||
|     with Inbox tags. Verify that there are documents in your archive without inbox tags. | ||||
|     The algorithm will only learn from documents not in your inbox. | ||||
| 
 | ||||
| Permission denied errors in the consumption directory | ||||
| ##################################################### | ||||
| 
 | ||||
| You might encounter errors such as: | ||||
| 
 | ||||
| .. code:: shell-session | ||||
| 
 | ||||
|     The following error occured while consuming document.pdf: [Errno 13] Permission denied: '/usr/src/paperless/src/../consume/document.pdf' | ||||
| 
 | ||||
| This happens when paperless does not have permission to delete files inside the consumption directory. | ||||
| Ensure that ``USERMAP_UID`` and ``USERMAP_GID`` are set to the user id and group id you use on the host operating system, if these are | ||||
| different from ``1000``. See :ref:`setup-docker_route`. | ||||
| 
 | ||||
| Also ensure that you are able to read and write to the consumption directory on the host. | ||||
|  | ||||
| @ -57,9 +57,6 @@ Adding documents to paperless | ||||
| ############################# | ||||
| 
 | ||||
| Once you've got Paperless setup, you need to start feeding documents into it. | ||||
| Currently, there are three options: the consumption directory, IMAP (email), and | ||||
| HTTP POST. | ||||
| 
 | ||||
| When adding documents to paperless, it will perform the following operations on | ||||
| your documents: | ||||
| 
 | ||||
| @ -82,8 +79,7 @@ your documents: | ||||
|     No matter which options you choose, Paperless will always store the original | ||||
|     document that it found in the consumption directory or in the mail and | ||||
|     will never overwrite that document. Archived versions are stored alongside the | ||||
|     digital versions. | ||||
| 
 | ||||
|     original versions. | ||||
| 
 | ||||
| 
 | ||||
| The consumption directory | ||||
| @ -107,6 +103,23 @@ files from the scanner.  Typically, you're looking at an FTP server like | ||||
| 
 | ||||
| .. TODO: hyperref to configuration of the location of this magic folder. | ||||
| 
 | ||||
| Dashboard upload | ||||
| ================ | ||||
| 
 | ||||
| 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. | ||||
| 
 | ||||
| 
 | ||||
| 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: | ||||
| 
 | ||||
| IMAP (Email) | ||||
| @ -183,6 +196,63 @@ You can also submit a document using the REST API, see :ref:`api-file_uploads` f | ||||
| 
 | ||||
| .. _basic-searching: | ||||
| 
 | ||||
| 
 | ||||
| Best practices | ||||
| ############## | ||||
| 
 | ||||
| Paperless offers a couple tools that help you organize your document collection. However, | ||||
| it is up to you to use them in a way that helps you organize documents and find specific | ||||
| documents when you need them. This section offers a couple ideas for managing your collection. | ||||
| 
 | ||||
| Document types allow you to classify documents according to what they are. You can define | ||||
| types such as "Receipt", "Invoice", or "Contract". If you used to collect all your receipts | ||||
| in a single binder, you can recreate that system in paperless by defining a document type, | ||||
| assigning documents to that type and then filtering by that type to only see all receipts. | ||||
| 
 | ||||
| Not all documents need document types. Sometimes its hard to determine what the type of a | ||||
| document is or it is hard to justify creating a document type that you only need once or twice. | ||||
| This is okay. As long as the types you define help you organize your collection in the way | ||||
| you want, paperless is doing its job. | ||||
| 
 | ||||
| Tags can be used in many different ways. Think of tags are more versatile folders or binders. | ||||
| If you have a binder for documents related to university / your car or health care, you can | ||||
| create these binders in paperless by creating tags and assigning them to relevant documents. | ||||
| Just as with documents, you can filter the document list by tags and only see documents of | ||||
| a certain topic. | ||||
| 
 | ||||
| With physical documents, you'll often need to decide which folder the document belongs to. | ||||
| The advantage of tags over folders and binders is that a single document can have multiple | ||||
| tags. A physical document cannot magically appear in two different folders, but with tags, | ||||
| this is entirely possible. | ||||
| 
 | ||||
| .. hint:: | ||||
| 
 | ||||
|   This can be used in many different ways. One example: Imagine you're working on a particular | ||||
|   task, such as signing up for university. Usually you'll need to collect a bunch of different | ||||
|   documents that are already sorted into various folders. With the tag system of paperless, | ||||
|   you can create a new group of documents that are relevant to this task without destroying | ||||
|   the already existing organization. When you're done with the task, you could delete the | ||||
|   tag again, which would be equal to sorting documents back into the folder they belong into. | ||||
|   Or keep the tag, up to you. | ||||
| 
 | ||||
| All of the logic above applies to correspondents as well. Attach them to documents if you | ||||
| feel that they help you organize your collection. | ||||
| 
 | ||||
| When you've started organizing your documents, create a couple saved views for document collections | ||||
| you regularly access. This is equal to having labeled physical binders on your desk, except | ||||
| that these saved views are dynamic and simply update themselves as you add documents to the system. | ||||
| 
 | ||||
| Here are a couple examples of tags and types that you could use in your collection. | ||||
| 
 | ||||
| * An ``inbox`` tag for newly added documents that you haven't manually edited yet. | ||||
| * A tag ``car`` for everything car related (repairs, registration, insurance, etc) | ||||
| * A tag ``todo`` for documents that you still need to do something with, such as reply, or | ||||
|   perform some task online. | ||||
| * A tag ``bank account x`` for all bank statement related to that account. | ||||
| * A tag ``mail`` for anything that you added to paperless via its mail processing capabilities. | ||||
| * A tag ``missing_metadata`` when you still need to add some metadata to a document, but can't | ||||
|   or don't want to do this right now. | ||||
| 
 | ||||
| Searching | ||||
| ######### | ||||
| 
 | ||||
|  | ||||
| @ -30,6 +30,7 @@ | ||||
| #PAPERLESS_FORCE_SCRIPT_NAME= | ||||
| #PAPERLESS_STATIC_URL=/static/ | ||||
| #PAPERLESS_AUTO_LOGIN_USERNAME= | ||||
| #PAPERLESS_COOKIE_PREFIX= | ||||
| 
 | ||||
| # OCR settings | ||||
| 
 | ||||
| @ -38,7 +39,7 @@ | ||||
| #PAPERLESS_OCR_OUTPUT_TYPE=pdfa | ||||
| #PAPERLESS_OCR_PAGES=1 | ||||
| #PAPERLESS_OCR_IMAGE_DPI=300 | ||||
| #PAPERLESS_OCR_USER_ARG={} | ||||
| #PAPERLESS_OCR_USER_ARGS={} | ||||
| #PAPERLESS_CONVERT_MEMORY_LIMIT=0 | ||||
| #PAPERLESS_CONVERT_TMPDIR=/var/tmp/paperless | ||||
| 
 | ||||
| @ -53,6 +54,13 @@ | ||||
| #PAPERLESS_POST_CONSUME_SCRIPT=/path/to/an/arbitrary/script.sh | ||||
| #PAPERLESS_FILENAME_DATE_ORDER=YMD | ||||
| #PAPERLESS_FILENAME_PARSE_TRANSFORMS=[] | ||||
| #PAPERLESS_THUMBNAIL_FONT_NAME= | ||||
| 
 | ||||
| # Tika settings | ||||
| 
 | ||||
| #PAPERLESS_TIKA_ENABLED=false | ||||
| #PAPERLESS_TIKA_ENDPOINT=http://localhost:9998 | ||||
| #PAPERLESS_TIKA_GOTENBERG_ENDPOINT=http://localhost:3000 | ||||
| 
 | ||||
| # Binaries | ||||
| 
 | ||||
|  | ||||
| @ -5,6 +5,7 @@ | ||||
| # adjust src/paperless/version.py | ||||
| # changelog in the documentation | ||||
| # adjust versions in docker/hub/* | ||||
| # adjust version in src-ui/src/environments/prod | ||||
| # If docker-compose was modified: all compose files are the same. | ||||
| 
 | ||||
| # Steps: | ||||
|  | ||||
| @ -1,2 +1,4 @@ | ||||
| docker run -p 5432:5432 -v paperless_pgdata:/var/lib/postgresql/data -d postgres:13 | ||||
| docker run -d -p 6379:6379 redis:latest | ||||
| docker run -p 3000:3000 -d thecodingmachine/gotenberg | ||||
| docker run -p 9998:9998 -d apache/tika | ||||
|  | ||||
| @ -13,6 +13,14 @@ | ||||
| 			"root": "", | ||||
| 			"sourceRoot": "src", | ||||
| 			"prefix": "app", | ||||
| 			"i18n": { | ||||
| 				"sourceLocale": "en-US", | ||||
| 				"locales": { | ||||
| 					"de": "src/locale/messages.de.xlf", | ||||
| 					"nl-NL": "src/locale/messages.nl_NL.xlf", | ||||
| 					"fr": "src/locale/messages.fr.xlf" | ||||
| 				} | ||||
| 			}, | ||||
| 			"architect": { | ||||
| 				"build": { | ||||
| 					"builder": "@angular-devkit/build-angular:browser", | ||||
| @ -23,15 +31,24 @@ | ||||
| 						"main": "src/main.ts", | ||||
| 						"polyfills": "src/polyfills.ts", | ||||
| 						"tsConfig": "tsconfig.app.json", | ||||
| 						"localize": true, | ||||
| 						"aot": true, | ||||
| 						"assets": [ | ||||
| 							"src/favicon.ico", | ||||
| 							"src/assets" | ||||
| 							"src/assets", | ||||
| 							"src/manifest.webmanifest", { | ||||
| 								"glob": "pdf.worker.min.js", | ||||
| 								"input": "node_modules/pdfjs-dist/build/", | ||||
| 								"output": "/assets/js/" | ||||
| 							} | ||||
| 						], | ||||
| 						"styles": [ | ||||
| 							"src/styles.scss" | ||||
| 						], | ||||
| 						"scripts": [] | ||||
| 						"scripts": [], | ||||
| 						"allowedCommonJsDependencies": [ | ||||
| 							"ng2-pdf-viewer" | ||||
| 						] | ||||
| 					}, | ||||
| 					"configurations": { | ||||
| 						"production": { | ||||
| @ -61,13 +78,16 @@ | ||||
| 									"maximumError": "10kb" | ||||
| 								} | ||||
| 							] | ||||
| 						}, | ||||
| 						"en-US": { | ||||
| 							"localize": ["en-US"] | ||||
| 						} | ||||
| 					} | ||||
| 				}, | ||||
| 				"serve": { | ||||
| 					"builder": "@angular-devkit/build-angular:dev-server", | ||||
| 					"options": { | ||||
| 						"browserTarget": "paperless-ui:build" | ||||
| 						"browserTarget": "paperless-ui:build:en-US" | ||||
| 					}, | ||||
| 					"configurations": { | ||||
| 						"production": { | ||||
| @ -90,7 +110,8 @@ | ||||
| 						"karmaConfig": "karma.conf.js", | ||||
| 						"assets": [ | ||||
| 							"src/favicon.ico", | ||||
| 							"src/assets" | ||||
| 							"src/assets", | ||||
| 							"src/manifest.webmanifest" | ||||
| 						], | ||||
| 						"styles": [ | ||||
| 							"src/styles.scss" | ||||
| @ -127,4 +148,4 @@ | ||||
| 		} | ||||
| 	}, | ||||
| 	"defaultProject": "paperless-ui" | ||||
| } | ||||
| } | ||||
|  | ||||
							
								
								
									
										1671
									
								
								src-ui/messages.xlf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1671
									
								
								src-ui/messages.xlf
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										71
									
								
								src-ui/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										71
									
								
								src-ui/package-lock.json
									
									
									
										generated
									
									
									
								
							| @ -2056,6 +2056,14 @@ | ||||
|         "tslib": "^2.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "@ng-select/ng-select": { | ||||
|       "version": "5.0.9", | ||||
|       "resolved": "https://registry.npmjs.org/@ng-select/ng-select/-/ng-select-5.0.9.tgz", | ||||
|       "integrity": "sha512-YZeSAiS8/Nx/eHZJPmOOYL8YmcvSq+dr1P8WIrsKmRA7mueorBpPc5xlUj+nLQbpLtsiQvdWDQspf/ykOvD/lA==", | ||||
|       "requires": { | ||||
|         "tslib": "^2.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "@ngtools/webpack": { | ||||
|       "version": "10.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-10.2.0.tgz", | ||||
| @ -2215,6 +2223,11 @@ | ||||
|       "integrity": "sha512-UV1/ZJMC+HcP902wWdpC43cAcGu0IQk/I5bXjP2aSuCjsk3cE74mDvFrLKga7oDC170ugOAYBwfT4DSQW3akDA==", | ||||
|       "dev": true | ||||
|     }, | ||||
|     "@types/pdfjs-dist": { | ||||
|       "version": "2.1.7", | ||||
|       "resolved": "https://registry.npmjs.org/@types/pdfjs-dist/-/pdfjs-dist-2.1.7.tgz", | ||||
|       "integrity": "sha512-nQIwcPUhkAIyn7x9NS0lR/qxYfd5unRtfGkMjvpgF4Sh28IXftRymaNmFKTTdejDNY25NDGSIyjwj/BRwAPexg==" | ||||
|     }, | ||||
|     "@types/q": { | ||||
|       "version": "1.5.4", | ||||
|       "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.4.tgz", | ||||
| @ -3023,6 +3036,16 @@ | ||||
|       "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", | ||||
|       "dev": true | ||||
|     }, | ||||
|     "bindings": { | ||||
|       "version": "1.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", | ||||
|       "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", | ||||
|       "dev": true, | ||||
|       "optional": true, | ||||
|       "requires": { | ||||
|         "file-uri-to-path": "1.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "blob": { | ||||
|       "version": "0.0.5", | ||||
|       "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.5.tgz", | ||||
| @ -5508,6 +5531,13 @@ | ||||
|         "schema-utils": "^2.6.5" | ||||
|       } | ||||
|     }, | ||||
|     "file-uri-to-path": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", | ||||
|       "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", | ||||
|       "dev": true, | ||||
|       "optional": true | ||||
|     }, | ||||
|     "fill-range": { | ||||
|       "version": "7.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", | ||||
| @ -8208,6 +8238,13 @@ | ||||
|       "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", | ||||
|       "dev": true | ||||
|     }, | ||||
|     "nan": { | ||||
|       "version": "2.14.2", | ||||
|       "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz", | ||||
|       "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==", | ||||
|       "dev": true, | ||||
|       "optional": true | ||||
|     }, | ||||
|     "nanomatch": { | ||||
|       "version": "1.2.13", | ||||
|       "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", | ||||
| @ -8260,6 +8297,23 @@ | ||||
|         "moment": "2.18.1" | ||||
|       } | ||||
|     }, | ||||
|     "ng2-pdf-viewer": { | ||||
|       "version": "6.3.2", | ||||
|       "resolved": "https://registry.npmjs.org/ng2-pdf-viewer/-/ng2-pdf-viewer-6.3.2.tgz", | ||||
|       "integrity": "sha512-H2tBhDd+Lq6CUzK2g54HsCcZDR2wTn1sDjYqKY3yF0Ydasl2R5ppCKynZBU/zge4EKvmHglJI120FbQMpJKDYQ==", | ||||
|       "requires": { | ||||
|         "@types/pdfjs-dist": "^2.1.4", | ||||
|         "pdfjs-dist": "^2.4.456", | ||||
|         "tslib": "^1.10.0" | ||||
|       }, | ||||
|       "dependencies": { | ||||
|         "tslib": { | ||||
|           "version": "1.14.1", | ||||
|           "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", | ||||
|           "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "ngx-cookie-service": { | ||||
|       "version": "10.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/ngx-cookie-service/-/ngx-cookie-service-10.1.1.tgz", | ||||
| @ -9270,6 +9324,11 @@ | ||||
|         "sha.js": "^2.4.8" | ||||
|       } | ||||
|     }, | ||||
|     "pdfjs-dist": { | ||||
|       "version": "2.5.207", | ||||
|       "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-2.5.207.tgz", | ||||
|       "integrity": "sha512-xGDUhnCYPfHy+unMXCLCJtlpZaaZ17Ew3WIL0tnSgKFUZXHAPD49GO9xScyszSsQMoutNDgRb+rfBXIaX/lJbw==" | ||||
|     }, | ||||
|     "performance-now": { | ||||
|       "version": "2.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", | ||||
| @ -13228,7 +13287,11 @@ | ||||
|           "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", | ||||
|           "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", | ||||
|           "dev": true, | ||||
|           "optional": true | ||||
|           "optional": true, | ||||
|           "requires": { | ||||
|             "bindings": "^1.5.0", | ||||
|             "nan": "^2.12.1" | ||||
|           } | ||||
|         }, | ||||
|         "glob-parent": { | ||||
|           "version": "3.1.0", | ||||
| @ -13832,7 +13895,11 @@ | ||||
|           "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", | ||||
|           "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", | ||||
|           "dev": true, | ||||
|           "optional": true | ||||
|           "optional": true, | ||||
|           "requires": { | ||||
|             "bindings": "^1.5.0", | ||||
|             "nan": "^2.12.1" | ||||
|           } | ||||
|         }, | ||||
|         "glob-parent": { | ||||
|           "version": "3.1.0", | ||||
|  | ||||
| @ -21,8 +21,10 @@ | ||||
|     "@angular/platform-browser-dynamic": "~10.1.5", | ||||
|     "@angular/router": "~10.1.5", | ||||
|     "@ng-bootstrap/ng-bootstrap": "^8.0.0", | ||||
|     "@ng-select/ng-select": "^5.0.9", | ||||
|     "bootstrap": "^4.5.0", | ||||
|     "ng-bootstrap": "^1.6.3", | ||||
|     "ng2-pdf-viewer": "^6.3.2", | ||||
|     "ngx-cookie-service": "^10.1.1", | ||||
|     "ngx-file-drop": "^10.0.0", | ||||
|     "ngx-infinite-scroll": "^9.1.0", | ||||
|  | ||||
| @ -1,3 +1,4 @@ | ||||
| import { SettingsService } from './services/settings.service'; | ||||
| import { Component, OnDestroy, OnInit } from '@angular/core'; | ||||
| import { Router } from '@angular/router'; | ||||
| import { Subscription } from 'rxjs'; | ||||
| @ -13,8 +14,11 @@ export class AppComponent implements OnInit, OnDestroy { | ||||
| 
 | ||||
|   successSubscription: Subscription; | ||||
|   failedSubscription: Subscription; | ||||
|    | ||||
|   constructor ( private consumerStatusService: ConsumerStatusService, private toastService: ToastService, private router: Router ) { | ||||
| 
 | ||||
|   constructor (private settings: SettingsService, private consumerStatusService: ConsumerStatusService, private toastService: ToastService, private router: Router) { | ||||
|     let anyWindow = (window as any) | ||||
|     anyWindow.pdfWorkerSrc = '/assets/js/pdf.worker.min.js'; | ||||
|     this.settings.updateDarkModeSettings() | ||||
|   } | ||||
| 
 | ||||
|   ngOnDestroy(): void { | ||||
| @ -38,6 +42,6 @@ export class AppComponent implements OnInit, OnDestroy { | ||||
| 
 | ||||
|   } | ||||
| 
 | ||||
|    | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -14,10 +14,9 @@ import { LogsComponent } from './components/manage/logs/logs.component'; | ||||
| import { SettingsComponent } from './components/manage/settings/settings.component'; | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms'; | ||||
| import { DatePipe } from '@angular/common'; | ||||
| import { SafePipe } from './pipes/safe.pipe'; | ||||
| import { NotFoundComponent } from './components/not-found/not-found.component'; | ||||
| import { CorrespondentListComponent } from './components/manage/correspondent-list/correspondent-list.component'; | ||||
| import { DeleteDialogComponent } from './components/common/delete-dialog/delete-dialog.component'; | ||||
| import { ConfirmDialogComponent } from './components/common/confirm-dialog/confirm-dialog.component'; | ||||
| import { CorrespondentEditDialogComponent } from './components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component'; | ||||
| import { TagEditDialogComponent } from './components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component'; | ||||
| import { DocumentTypeEditDialogComponent } from './components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component'; | ||||
| @ -27,9 +26,13 @@ import { ResultHighlightComponent } from './components/search/result-highlight/r | ||||
| import { PageHeaderComponent } from './components/common/page-header/page-header.component'; | ||||
| import { AppFrameComponent } from './components/app-frame/app-frame.component'; | ||||
| import { ToastsComponent } from './components/common/toasts/toasts.component'; | ||||
| import { FilterEditorComponent } from './components/filter-editor/filter-editor.component'; | ||||
| import { FilterEditorComponent } from './components/document-list/filter-editor/filter-editor.component'; | ||||
| import { FilterableDropdownComponent } from './components/common/filterable-dropdown/filterable-dropdown.component'; | ||||
| import { ToggleableDropdownButtonComponent } from './components/common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; | ||||
| import { DateDropdownComponent } from './components/common/date-dropdown/date-dropdown.component'; | ||||
| import { DocumentCardLargeComponent } from './components/document-list/document-card-large/document-card-large.component'; | ||||
| import { DocumentCardSmallComponent } from './components/document-list/document-card-small/document-card-small.component'; | ||||
| import { BulkEditorComponent } from './components/document-list/bulk-editor/bulk-editor.component'; | ||||
| import { NgxFileDropModule } from 'ngx-file-drop'; | ||||
| import { TextComponent } from './components/common/input/text/text.component'; | ||||
| import { SelectComponent } from './components/common/input/select/select.component'; | ||||
| @ -45,7 +48,16 @@ import { SavedViewWidgetComponent } from './components/dashboard/widgets/saved-v | ||||
| import { StatisticsWidgetComponent } from './components/dashboard/widgets/statistics-widget/statistics-widget.component'; | ||||
| import { UploadFileWidgetComponent } from './components/dashboard/widgets/upload-file-widget/upload-file-widget.component'; | ||||
| import { WidgetFrameComponent } from './components/dashboard/widgets/widget-frame/widget-frame.component'; | ||||
| import { PdfViewerModule } from 'ng2-pdf-viewer'; | ||||
| import { WelcomeWidgetComponent } from './components/dashboard/widgets/welcome-widget/welcome-widget.component'; | ||||
| import { YesNoPipe } from './pipes/yes-no.pipe'; | ||||
| import { FileSizePipe } from './pipes/file-size.pipe'; | ||||
| import { FilterPipe } from './pipes/filter.pipe'; | ||||
| import { DocumentTitlePipe } from './pipes/document-title.pipe'; | ||||
| import { MetadataCollapseComponent } from './components/document-detail/metadata-collapse/metadata-collapse.component'; | ||||
| import { SelectDialogComponent } from './components/common/select-dialog/select-dialog.component'; | ||||
| import { NgSelectModule } from '@ng-select/ng-select'; | ||||
| import { NumberComponent } from './components/common/input/number/number.component'; | ||||
| import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/consumer-status-widget/consumer-status-widget.component'; | ||||
| 
 | ||||
| @NgModule({ | ||||
| @ -59,10 +71,9 @@ import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/co | ||||
|     DocumentTypeListComponent, | ||||
|     LogsComponent, | ||||
|     SettingsComponent, | ||||
|     SafePipe, | ||||
|     NotFoundComponent, | ||||
|     CorrespondentEditDialogComponent, | ||||
|     DeleteDialogComponent, | ||||
|     ConfirmDialogComponent, | ||||
|     TagEditDialogComponent, | ||||
|     DocumentTypeEditDialogComponent, | ||||
|     TagComponent, | ||||
| @ -72,8 +83,12 @@ import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/co | ||||
|     AppFrameComponent, | ||||
|     ToastsComponent, | ||||
|     FilterEditorComponent, | ||||
|     FilterableDropdownComponent, | ||||
|     ToggleableDropdownButtonComponent, | ||||
|     DateDropdownComponent, | ||||
|     DocumentCardLargeComponent, | ||||
|     DocumentCardSmallComponent, | ||||
|     BulkEditorComponent, | ||||
|     TextComponent, | ||||
|     SelectComponent, | ||||
|     CheckComponent, | ||||
| @ -86,6 +101,13 @@ import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/co | ||||
|     UploadFileWidgetComponent, | ||||
|     WidgetFrameComponent, | ||||
|     WelcomeWidgetComponent, | ||||
|     YesNoPipe, | ||||
|     FileSizePipe, | ||||
|     FilterPipe, | ||||
|     DocumentTitlePipe, | ||||
|     MetadataCollapseComponent, | ||||
|     SelectDialogComponent, | ||||
|     NumberComponent, | ||||
|     ConsumerStatusWidgetComponent | ||||
|   ], | ||||
|   imports: [ | ||||
| @ -96,7 +118,9 @@ import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/co | ||||
|     FormsModule, | ||||
|     ReactiveFormsModule, | ||||
|     NgxFileDropModule, | ||||
|     InfiniteScrollModule | ||||
|     InfiniteScrollModule, | ||||
|     PdfViewerModule, | ||||
|     NgSelectModule | ||||
|   ], | ||||
|   providers: [ | ||||
|     DatePipe, | ||||
| @ -104,7 +128,9 @@ import { ConsumerStatusWidgetComponent } from './components/dashboard/widgets/co | ||||
|       provide: HTTP_INTERCEPTORS, | ||||
|       useClass: CsrfInterceptor, | ||||
|       multi: true | ||||
|     } | ||||
|     }, | ||||
|     FilterPipe, | ||||
|     DocumentTitlePipe | ||||
|   ], | ||||
|   bootstrap: [AppComponent] | ||||
| }) | ||||
|  | ||||
| @ -1,158 +1,170 @@ | ||||
| <nav class="navbar navbar-dark sticky-top bg-primary flex-md-nowrap p-0 shadow"> | ||||
|   <span class="navbar-brand col-md-3 col-lg-2 mr-0 px-3" href="#"> | ||||
|     <img src="assets/logo-dark-notext.svg" height="18px" class="mr-2"> | ||||
|     Paperless-ng | ||||
|   </span> | ||||
|   <button class="navbar-toggler position-absolute d-md-none collapsed" type="button" data-toggle="collapse" | ||||
|   <button class="navbar-toggler d-md-none collapsed border-0" type="button" data-toggle="collapse" | ||||
|     data-target="#sidebarMenu" aria-controls="sidebarMenu" aria-expanded="false" aria-label="Toggle navigation" | ||||
|     (click)="isMenuCollapsed = !isMenuCollapsed"> | ||||
|     <span class="navbar-toggler-icon"></span> | ||||
|   </button> | ||||
|   <form (ngSubmit)="search()" class="w-100 m-1"> | ||||
|     <input class="form-control form-control-dark" type="text" placeholder="Search" aria-label="Search" | ||||
|       [formControl]="searchField" [ngbTypeahead]="searchAutoComplete" (selectItem)="itemSelected($event)"> | ||||
|   </form> | ||||
|   <a class="navbar-brand col-auto col-md-3 col-lg-2 mr-0 px-3 py-3 order-sm-0" routerLink="/dashboard"> | ||||
|     <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 198.43 238.91" width="1rem" class="mr-2" fill="currentColor"> | ||||
|       <path d="M194.7,0C164.22,70.94,17.64,79.74,64.55,194.06c.58,1.47-10.85,17-18.47,29.9-1.76-6.45-3.81-13.48-3.52-14.07,38.11-45.14-27.26-70.65-30.78-107.58C-4.64,131.62-10.5,182.92,39,212.53c.3,0,2.64,11.14,3.81,16.71a58.55,58.55,0,0,0-2.93,6.45c-1.17,2.93,7.62,2.64,7.62,3.22.88-.29,21.7-36.93,22.28-37.23C187.67,174.72,208.48,68.6,194.7,0ZM134.61,74.75C79.5,124,70.12,160.64,71.88,178.53,53.41,134.85,107.64,86.77,134.61,74.75ZM28.2,145.11c10.55,9.67,28.14,39.28,13.19,56.57C44.91,193.77,46.08,175.89,28.2,145.11Z" transform="translate(0 0)"/> | ||||
|     </svg> | ||||
|     <ng-container i18n="app title">Paperless-ng</ng-container> | ||||
|   </a> | ||||
|   <div class="search-form-container flex-grow-1 py-2 pb-3 pb-sm-2 px-3 pl-md-4 mr-sm-auto order-3 order-sm-1"> | ||||
|     <form (ngSubmit)="search()" class="form-inline flex-grow-1"> | ||||
|       <input class="form-control form-control-sm" type="text" placeholder="Search documents" aria-label="Search" | ||||
|         [formControl]="searchField" [ngbTypeahead]="searchAutoComplete" (selectItem)="itemSelected($event)" i18n-placeholder> | ||||
|       <svg width="1em" height="1em"> | ||||
|         <use xlink:href="assets/bootstrap-icons.svg#search"/> | ||||
|       </svg> | ||||
|     </form> | ||||
|   </div> | ||||
|   <ul ngbNav class="order-sm-3"> | ||||
|     <li ngbDropdown class="nav-item dropdown"> | ||||
|       <button class="btn text-light" id="userDropdown" ngbDropdownToggle> | ||||
|         <span *ngIf="displayName" class="navbar-text small mr-2 text-light d-none d-sm-inline"> | ||||
|           {{displayName}} | ||||
|         </span> | ||||
|         <svg width="1.3em" height="1.3em"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#person-circle"/> | ||||
|         </svg> | ||||
|       </button> | ||||
|       <div ngbDropdownMenu class="dropdown-menu-right shadow mr-2" aria-labelledby="userDropdown"> | ||||
|         <div *ngIf="displayName" class="d-sm-none"> | ||||
|           <p class="small mb-0 px-3" i18n>Logged in as {{displayName}}</p> | ||||
|           <div class="dropdown-divider"></div> | ||||
|         </div> | ||||
|         <a ngbDropdownItem class="nav-link" routerLink="settings" (click)="closeMenu()"> | ||||
|           <svg class="sidebaricon mr-2" fill="currentColor"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#gear"/> | ||||
|           </svg><ng-container i18n>Settings</ng-container> | ||||
|         </a> | ||||
|         <a ngbDropdownItem class="nav-link" href="accounts/logout/"> | ||||
|           <svg class="sidebaricon mr-2" fill="currentColor"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#door-open"/> | ||||
|           </svg><ng-container i18n>Logout</ng-container> | ||||
|         </a> | ||||
|       </div> | ||||
|     </li> | ||||
|   </ul> | ||||
| </nav> | ||||
| 
 | ||||
| <div class="container-fluid"> | ||||
|   <div class="row"> | ||||
|     <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"> | ||||
|         <ul class="nav flex-column"> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="dashboard" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#house"/> | ||||
|               </svg> | ||||
|               Dashboard | ||||
|               </svg> <ng-container i18n>Dashboard</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="documents" routerLinkActive="active" [routerLinkActiveOptions]="{exact: true}" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#files"/> | ||||
|               </svg> | ||||
|               Documents | ||||
|               </svg> <ng-container i18n>Documents</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|         </ul> | ||||
| 
 | ||||
|         <h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted" *ngIf='viewConfigService.getSideBarConfigs().length > 0'> | ||||
|           <span>Saved views</span> | ||||
|         <h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted" *ngIf='savedViewService.sidebarViews.length > 0'> | ||||
|           <ng-container i18n>Saved views</ng-container> | ||||
|         </h6> | ||||
|         <ul class="nav flex-column mb-2"> | ||||
|           <li class="nav-item w-100" *ngFor='let config of viewConfigService.getSideBarConfigs()'> | ||||
|             <a class="nav-link text-truncate" routerLink="view/{{config.id}}" routerLinkActive="active" (click)="closeMenu()"> | ||||
|           <li class="nav-item w-100" *ngFor="let view of savedViewService.sidebarViews"> | ||||
|             <a class="nav-link text-truncate" routerLink="view/{{view.id}}" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#funnel"/> | ||||
|               </svg> | ||||
|               {{config.title}} | ||||
|               </svg> {{view.name}} | ||||
|             </a> | ||||
|           </li> | ||||
|         </ul> | ||||
| 
 | ||||
|         <h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted" *ngIf='openDocuments.length > 0'> | ||||
|           <span>Open documents</span> | ||||
|           <ng-container i18n>Open documents</ng-container> | ||||
|         </h6> | ||||
|         <ul class="nav flex-column mb-2"> | ||||
|           <li class="nav-item w-100" *ngFor='let d of openDocuments'> | ||||
|             <a class="nav-link text-truncate" routerLink="documents/{{d.id}}" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#file-text"/> | ||||
|               </svg> | ||||
|               {{d.title}} | ||||
|               </svg> {{d.title | documentTitle}} | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item w-100" *ngIf="openDocuments.length > 1"> | ||||
|             <a class="nav-link text-truncate" [routerLink]="" (click)="closeAll()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#x"/> | ||||
|               </svg> | ||||
|               Close all | ||||
|               </svg> <ng-container i18n>Close all</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|         </ul> | ||||
| 
 | ||||
|         <h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
|           <span>Manage</span> | ||||
|           <ng-container i18n>Manage</ng-container> | ||||
|         </h6> | ||||
|         <ul class="nav flex-column mb-2"> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="correspondents" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#person"/> | ||||
|               </svg> | ||||
|               Correspondents | ||||
|               </svg> <ng-container i18n>Correspondents</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="tags" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#tags"/> | ||||
|               </svg> | ||||
|               Tags | ||||
|               </svg> <ng-container i18n>Tags</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="documenttypes" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#hash"/> | ||||
|               </svg> | ||||
|               Document types | ||||
|               </svg> <ng-container i18n>Document types</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="logs" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#text-left"/> | ||||
|               </svg> | ||||
|               Logs | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" routerLink="settings" routerLinkActive="active" (click)="closeMenu()"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#gear"/> | ||||
|               </svg> | ||||
|               Settings | ||||
|               </svg> <ng-container i18n>Logs</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" href="admin/"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#toggles"/> | ||||
|               </svg> | ||||
|               Admin | ||||
|               </svg> <ng-container i18n>Admin</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|         </ul> | ||||
| 
 | ||||
|         <h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
|           <span>Misc</span> | ||||
|           <ng-container i18n>Misc</ng-container> | ||||
|         </h6> | ||||
|         <ul class="nav flex-column mb-2"> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" href="https://paperless-ng.readthedocs.io/en/latest/"> | ||||
|             <a class="nav-link" target="_blank" rel="noopener noreferrer" href="https://paperless-ng.readthedocs.io/en/latest/"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#question-circle"/> | ||||
|               </svg> | ||||
|               Documentation | ||||
|               </svg> <ng-container i18n>Documentation</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" href="https://github.com/jonaswinkler/paperless-ng"> | ||||
|             <a class="nav-link" target="_blank" rel="noopener noreferrer" href="https://github.com/jonaswinkler/paperless-ng"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#link"/> | ||||
|               </svg> | ||||
|               GitHub | ||||
|             </a> | ||||
|           </li> | ||||
|           <li class="nav-item"> | ||||
|             <a class="nav-link" href="accounts/logout/"> | ||||
|               <svg class="sidebaricon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#door-open"/> | ||||
|               </svg> | ||||
|               Logout | ||||
|               </svg> <ng-container i18n>GitHub</ng-container> | ||||
|             </a> | ||||
|           </li> | ||||
|         </ul> | ||||
|  | ||||
| @ -1,36 +1,30 @@ | ||||
| 
 | ||||
| @import "/src/theme"; | ||||
| 
 | ||||
|   /* | ||||
| /* | ||||
|  * Sidebar | ||||
|  */ | ||||
| 
 | ||||
|  .sidebar { | ||||
| .sidebar { | ||||
|   position: fixed; | ||||
|   top: 0; | ||||
|   bottom: 0; | ||||
|   left: 0; | ||||
|   z-index: 100; /* Behind the navbar */ | ||||
|   padding: 48px 0 0; /* Height of navbar */ | ||||
|   padding: 50px 0 0; /* Height of navbar */ | ||||
|   box-shadow: inset -1px 0 0 rgba(0, 0, 0, .1); | ||||
| } | ||||
| 
 | ||||
| @media (max-width: 767.98px) { | ||||
|   .sidebar { | ||||
|     top: 3rem; | ||||
|     top: 3.5rem; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .sidebar-sticky { | ||||
|   position: relative; | ||||
|   top: 0; | ||||
|   /* height: calc(100vh - 48px); */ | ||||
|   height: 100%; | ||||
|   padding-top: .5rem; | ||||
|   padding-top: 0.5rem; | ||||
|   overflow-x: hidden; | ||||
|   overflow-y: auto; /* Scrollable contents if viewport is shorter than content. */ | ||||
| } | ||||
| 
 | ||||
| @supports ((position: -webkit-sticky) or (position: sticky)) { | ||||
|   .sidebar-sticky { | ||||
|     position: -webkit-sticky; | ||||
| @ -53,36 +47,85 @@ | ||||
|   font-weight: bold; | ||||
| } | ||||
| 
 | ||||
| .sidebar .nav-link:hover .sidebaricon, | ||||
| .sidebar .nav-link.active .sidebaricon { | ||||
| .sidebar .nav-link.active .sidebaricon, | ||||
| .sidebar .nav-link:hover .sidebaricon { | ||||
|   color: inherit; | ||||
| } | ||||
| 
 | ||||
| .sidebar-heading { | ||||
|   font-size: .75rem; | ||||
|   font-size: 0.75rem; | ||||
|   text-transform: uppercase; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /* | ||||
|  * Navbar | ||||
|  */ | ||||
| 
 | ||||
|  .navbar-brand { | ||||
|   padding-top: .75rem; | ||||
|   padding-bottom: .75rem; | ||||
| .navbar-brand { | ||||
|   padding-top: 0.75rem; | ||||
|   padding-bottom: 0.75rem; | ||||
|   font-size: 1rem; | ||||
|   background-color: rgba(0, 0, 0, .25); | ||||
|   box-shadow: inset -1px 0 0 rgba(0, 0, 0, .25); | ||||
| } | ||||
| 
 | ||||
| .navbar .navbar-toggler { | ||||
|   top: .25rem; | ||||
|   right: 1rem; | ||||
| .dropdown.show .dropdown-toggle, | ||||
| .dropdown-toggle:hover { | ||||
|   opacity: 0.7; | ||||
| } | ||||
| 
 | ||||
| .navbar .form-control { | ||||
|   padding: .75rem 1rem; | ||||
|   border-width: 0; | ||||
|   border-radius: 0; | ||||
| .dropdown-toggle::after { | ||||
|   margin-left: 0.4em; | ||||
|   vertical-align: 0.155em; | ||||
| } | ||||
| 
 | ||||
| .navbar .dropdown-menu { | ||||
|   font-size: 0.875rem; // body size | ||||
| 
 | ||||
|   a svg { | ||||
|     opacity: 0.6; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .navbar .search-form-container { | ||||
|   max-width: 550px; | ||||
| 
 | ||||
|   form { | ||||
|     position: relative; | ||||
|   } | ||||
| 
 | ||||
|   svg { | ||||
|     position: absolute; | ||||
|     left: 0.6rem; | ||||
|     color: rgba(255, 255, 255, 0.6); | ||||
|   } | ||||
| 
 | ||||
|   &:focus-within { | ||||
|     svg { | ||||
|       display: none; | ||||
|     } | ||||
| 
 | ||||
|     .form-control::placeholder { | ||||
|       color: rgba(255, 255, 255, 0); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   .form-control { | ||||
|     color: rgba(255, 255, 255, 0.3); | ||||
|     background-color: rgba(0, 0, 0, 0.15); | ||||
|     padding-left: 1.8rem; | ||||
|     border-color: rgba(255, 255, 255, 0.2); | ||||
|     transition: flex 0.3s ease; | ||||
|     max-width: 600px; | ||||
|     min-width: 300px; // 1/2 max | ||||
| 
 | ||||
|     &::placeholder { | ||||
|       color: rgba(255, 255, 255, 0.4); | ||||
|     } | ||||
| 
 | ||||
|     &:focus { | ||||
|       background-color: #fff; | ||||
|       color: #212529; | ||||
|       flex-grow: 1; | ||||
|       padding-left: 0.5rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -5,10 +5,12 @@ import { from, Observable, Subscription } from 'rxjs'; | ||||
| import { debounceTime, distinctUntilChanged, map, switchMap } from 'rxjs/operators'; | ||||
| import { PaperlessDocument } from 'src/app/data/paperless-document'; | ||||
| import { OpenDocumentsService } from 'src/app/services/open-documents.service'; | ||||
| import { SavedViewService } from 'src/app/services/rest/saved-view.service'; | ||||
| import { SearchService } from 'src/app/services/rest/search.service'; | ||||
| import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; | ||||
| import { environment } from 'src/environments/environment'; | ||||
| import { DocumentDetailComponent } from '../document-detail/document-detail.component'; | ||||
|    | ||||
| import { Meta } from '@angular/platform-browser'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-app-frame', | ||||
|   templateUrl: './app-frame.component.html', | ||||
| @ -21,10 +23,14 @@ export class AppFrameComponent implements OnInit, OnDestroy { | ||||
|     private activatedRoute: ActivatedRoute, | ||||
|     private openDocumentsService: OpenDocumentsService, | ||||
|     private searchService: SearchService, | ||||
|     public viewConfigService: SavedViewConfigService | ||||
|     public savedViewService: SavedViewService, | ||||
|     private meta: Meta | ||||
|     ) { | ||||
|        | ||||
|   } | ||||
| 
 | ||||
|   versionString = `${environment.appTitle} ${environment.version}` | ||||
| 
 | ||||
|   isMenuCollapsed: boolean = true | ||||
| 
 | ||||
|   closeMenu() { | ||||
| @ -52,7 +58,7 @@ export class AppFrameComponent implements OnInit, OnDestroy { | ||||
|         term.length < 2 ? from([[]]) : this.searchService.autocomplete(term) | ||||
|       ) | ||||
|     ) | ||||
|    | ||||
| 
 | ||||
|   itemSelected(event) { | ||||
|     event.preventDefault() | ||||
|     let currentSearch: string = this.searchField.value | ||||
| @ -95,4 +101,17 @@ export class AppFrameComponent implements OnInit, OnDestroy { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   get displayName() { | ||||
|     // TODO: taken from dashboard component, is this the best way to pass around username?
 | ||||
|     let tagFullName = this.meta.getTag('name=full_name') | ||||
|     let tagUsername = this.meta.getTag('name=username') | ||||
|     if (tagFullName && tagFullName.content) { | ||||
|       return tagFullName.content | ||||
|     } else if (tagUsername && tagUsername.content) { | ||||
|       return tagUsername.content | ||||
|     } else { | ||||
|       return null | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,17 @@ | ||||
|     <div class="modal-header"> | ||||
|       <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|       <button type="button" class="close" aria-label="Close" (click)="cancelClicked()"> | ||||
|         <span aria-hidden="true">×</span> | ||||
|       </button> | ||||
|     </div> | ||||
|     <div class="modal-body"> | ||||
|       <p *ngIf="messageBold"><b>{{messageBold}}</b></p> | ||||
|       <p *ngIf="message">{{message}}</p> | ||||
|     </div> | ||||
|     <div class="modal-footer"> | ||||
|       <button type="button" class="btn btn-outline-dark" (click)="cancelClicked()" [disabled]="!buttonsEnabled">Cancel</button> | ||||
|       <button type="button" class="btn" [class]="btnClass" (click)="confirmClicked.emit()" [disabled]="!confirmButtonEnabled || !buttonsEnabled"> | ||||
|         {{btnCaption}} | ||||
|         <span *ngIf="!confirmButtonEnabled"> ({{seconds}})</span> | ||||
|       </button> | ||||
|     </div> | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { ConfirmDialogComponent } from './confirm-dialog.component'; | ||||
| 
 | ||||
| describe('ConfirmDialogComponent', () => { | ||||
|   let component: ConfirmDialogComponent; | ||||
|   let fixture: ComponentFixture<ConfirmDialogComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ ConfirmDialogComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(ConfirmDialogComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,55 @@ | ||||
| import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-confirm-dialog', | ||||
|   templateUrl: './confirm-dialog.component.html', | ||||
|   styleUrls: ['./confirm-dialog.component.scss'] | ||||
| }) | ||||
| export class ConfirmDialogComponent implements OnInit { | ||||
| 
 | ||||
|   constructor(public activeModal: NgbActiveModal) { } | ||||
| 
 | ||||
|   @Output() | ||||
|   public confirmClicked = new EventEmitter() | ||||
| 
 | ||||
|   @Input() | ||||
|   title = $localize`Confirmation` | ||||
| 
 | ||||
|   @Input() | ||||
|   messageBold | ||||
| 
 | ||||
|   @Input() | ||||
|   message | ||||
| 
 | ||||
|   @Input() | ||||
|   btnClass = "btn-primary" | ||||
| 
 | ||||
|   @Input() | ||||
|   btnCaption = $localize`Confirm` | ||||
| 
 | ||||
|   @Input() | ||||
|   buttonsEnabled = true | ||||
|    | ||||
|   confirmButtonEnabled = true | ||||
|   seconds = 0 | ||||
| 
 | ||||
|   delayConfirm(seconds: number) { | ||||
|     this.confirmButtonEnabled = false | ||||
|     this.seconds = seconds | ||||
|     setTimeout(() => { | ||||
|       if (this.seconds <= 1) { | ||||
|         this.confirmButtonEnabled = true | ||||
|       } else { | ||||
|         this.delayConfirm(seconds - 1) | ||||
|       } | ||||
|     }, 1000) | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
|   cancelClicked() { | ||||
|     this.activeModal.close() | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,44 @@ | ||||
|   <div class="btn-group" ngbDropdown role="group"> | ||||
|   <button class="btn btn-sm" id="dropdown{{title}}" ngbDropdownToggle [ngClass]="dateBefore || dateAfter ? 'btn-primary' : 'btn-outline-primary'"> | ||||
|     {{title}} | ||||
|   </button> | ||||
|   <div class="dropdown-menu date-dropdown shadow pt-0" ngbDropdownMenu attr.aria-labelledby="dropdown{{title}}"> | ||||
|     <div class="list-group list-group-flush"> | ||||
|         <button *ngFor="let qf of quickFilters" class="list-group-item small list-goup list-group-item-action d-flex p-2 pl-3" role="menuitem" (click)="setDateQuickFilter(qf.id)"> | ||||
|           {{qf.name}} | ||||
|         </button> | ||||
|         <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
| 
 | ||||
|           <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|             <div i18n>After</div> | ||||
|             <a *ngIf="dateAfter" class="btn btn-link p-0 m-0" (click)="clearAfter()"> | ||||
|               <svg width="0.8em" height="0.8em" viewBox="0 0 16 16" class="bi bi-x" fill="currentColor" xmlns="http://www.w3.org/2000/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" /> | ||||
|               </svg> | ||||
|               <small i18n>Clear</small> | ||||
|             </a> | ||||
|           </div> | ||||
| 
 | ||||
|           <div class="input-group input-group-sm"> | ||||
|             <input type="date" class="form-control" id="date_after" [(ngModel)]="dateAfter" (change)="onChangeDebounce()"> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
| 
 | ||||
|           <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|             <div i18n>Before</div> | ||||
|             <a *ngIf="dateBefore" class="btn btn-link p-0 m-0" (click)="clearBefore()"> | ||||
|               <svg width="0.8em" height="0.8em" viewBox="0 0 16 16" class="bi bi-x" fill="currentColor" xmlns="http://www.w3.org/2000/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" /> | ||||
|               </svg> | ||||
|               <small i18n>Clear</small> | ||||
|             </a> | ||||
|           </div> | ||||
| 
 | ||||
|           <div class="input-group input-group-sm"> | ||||
|             <input type="date" class="form-control" id="date_before" [(ngModel)]="dateBefore" (change)="onChangeDebounce()"> | ||||
|           </div> | ||||
|         </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| @ -0,0 +1,7 @@ | ||||
| .date-dropdown { | ||||
|   min-width: 250px; | ||||
| 
 | ||||
|   .btn-link { | ||||
|     line-height: 1; | ||||
|   } | ||||
| } | ||||
| @ -1,20 +1,20 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { DeleteDialogComponent } from './delete-dialog.component'; | ||||
| import { DateDropdownComponent } from './date-dropdown.component'; | ||||
| 
 | ||||
| describe('DeleteDialogComponent', () => { | ||||
|   let component: DeleteDialogComponent; | ||||
|   let fixture: ComponentFixture<DeleteDialogComponent>; | ||||
| describe('DateDropdownComponent', () => { | ||||
|   let component: DateDropdownComponent; | ||||
|   let fixture: ComponentFixture<DateDropdownComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ DeleteDialogComponent ] | ||||
|       declarations: [ DateDropdownComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(DeleteDialogComponent); | ||||
|     fixture = TestBed.createComponent(DateDropdownComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| @ -0,0 +1,111 @@ | ||||
| import { formatDate } from '@angular/common'; | ||||
| import { Component, EventEmitter, Input, Output, OnInit, OnDestroy } from '@angular/core'; | ||||
| import { Subject, Subscription } from 'rxjs'; | ||||
| import { debounceTime, distinctUntilChanged } from 'rxjs/operators'; | ||||
| 
 | ||||
| export interface DateSelection { | ||||
|   before?: string | ||||
|   after?: string | ||||
| } | ||||
| 
 | ||||
| const LAST_7_DAYS = 0 | ||||
| const LAST_MONTH = 1 | ||||
| const LAST_3_MONTHS = 2 | ||||
| const LAST_YEAR = 3 | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-date-dropdown', | ||||
|   templateUrl: './date-dropdown.component.html', | ||||
|   styleUrls: ['./date-dropdown.component.scss'] | ||||
| }) | ||||
| export class DateDropdownComponent implements OnInit, OnDestroy { | ||||
| 
 | ||||
|   quickFilters = [ | ||||
|     {id: LAST_7_DAYS, name: $localize`Last 7 days`}, | ||||
|     {id: LAST_MONTH, name: $localize`Last month`}, | ||||
|     {id: LAST_3_MONTHS, name: $localize`Last 3 months`}, | ||||
|     {id: LAST_YEAR, name: $localize`Last year`} | ||||
|   ] | ||||
| 
 | ||||
|   @Input() | ||||
|   dateBefore: string | ||||
| 
 | ||||
|   @Output() | ||||
|   dateBeforeChange = new EventEmitter<string>() | ||||
| 
 | ||||
|   @Input() | ||||
|   dateAfter: string | ||||
| 
 | ||||
|   @Output() | ||||
|   dateAfterChange = new EventEmitter<string>() | ||||
| 
 | ||||
|   @Input() | ||||
|   title: string | ||||
| 
 | ||||
|   @Output() | ||||
|   datesSet = new EventEmitter<DateSelection>() | ||||
| 
 | ||||
|   private datesSetDebounce$ = new Subject() | ||||
| 
 | ||||
|   private sub: Subscription | ||||
| 
 | ||||
|   ngOnInit() { | ||||
|     this.sub = this.datesSetDebounce$.pipe( | ||||
|       debounceTime(400) | ||||
|     ).subscribe(() => { | ||||
|       this.onChange() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   ngOnDestroy() { | ||||
|     if (this.sub) { | ||||
|       this.sub.unsubscribe() | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   setDateQuickFilter(qf: number) { | ||||
|     this.dateBefore = null | ||||
|     let date = new Date() | ||||
|     switch (qf) { | ||||
|       case LAST_7_DAYS: | ||||
|         date.setDate(date.getDate() - 7) | ||||
|         break; | ||||
| 
 | ||||
|       case LAST_MONTH: | ||||
|         date.setMonth(date.getMonth() - 1) | ||||
|         break; | ||||
| 
 | ||||
|       case LAST_3_MONTHS: | ||||
|         date.setMonth(date.getMonth() - 3) | ||||
|         break | ||||
| 
 | ||||
|       case LAST_YEAR: | ||||
|         date.setFullYear(date.getFullYear() - 1) | ||||
|         break | ||||
| 
 | ||||
|       } | ||||
|     this.dateAfter = formatDate(date, 'yyyy-MM-dd', "en-us", "UTC") | ||||
|     this.onChange() | ||||
|   } | ||||
| 
 | ||||
|   onChange() { | ||||
|     this.dateAfterChange.emit(this.dateAfter) | ||||
|     this.dateBeforeChange.emit(this.dateBefore) | ||||
|     this.datesSet.emit({after: this.dateAfter, before: this.dateBefore}) | ||||
|   } | ||||
| 
 | ||||
|   onChangeDebounce() { | ||||
|     this.datesSetDebounce$.next({after: this.dateAfter, before: this.dateBefore}) | ||||
|   } | ||||
| 
 | ||||
|   clearBefore() { | ||||
|     this.dateBefore = null | ||||
|     this.onChange() | ||||
|   } | ||||
| 
 | ||||
|   clearAfter() { | ||||
|     this.dateAfter = null | ||||
|     this.onChange() | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| @ -1,14 +0,0 @@ | ||||
|     <div class="modal-header"> | ||||
|       <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|       <button type="button" class="close" aria-label="Close" (click)="cancelClicked()"> | ||||
|         <span aria-hidden="true">×</span> | ||||
|       </button> | ||||
|     </div> | ||||
|     <div class="modal-body"> | ||||
|       <p><b>{{message}}</b></p> | ||||
|       <p *ngIf="message2">{{message2}}</p> | ||||
|     </div> | ||||
|     <div class="modal-footer"> | ||||
|       <button type="button" class="btn btn-outline-dark" (click)="cancelClicked()">Cancel</button> | ||||
|       <button type="button" class="btn btn-danger" (click)="deleteClicked.emit()">Delete</button> | ||||
|     </div> | ||||
| @ -1,31 +0,0 @@ | ||||
| import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-delete-dialog', | ||||
|   templateUrl: './delete-dialog.component.html', | ||||
|   styleUrls: ['./delete-dialog.component.scss'] | ||||
| }) | ||||
| export class DeleteDialogComponent implements OnInit { | ||||
| 
 | ||||
|   constructor(public activeModal: NgbActiveModal) { } | ||||
| 
 | ||||
|   @Output() | ||||
|   public deleteClicked = new EventEmitter() | ||||
| 
 | ||||
|   @Input() | ||||
|   title = "Delete confirmation" | ||||
| 
 | ||||
|   @Input() | ||||
|   message = "Do you really want to delete this?" | ||||
| 
 | ||||
|   @Input() | ||||
|   message2 | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
|   cancelClicked() { | ||||
|     this.activeModal.close() | ||||
|   } | ||||
| } | ||||
| @ -2,7 +2,8 @@ import { Directive, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { FormGroup } from '@angular/forms'; | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| import { Observable } from 'rxjs'; | ||||
| import { MATCHING_ALGORITHMS } from 'src/app/data/matching-model'; | ||||
| import { map } from 'rxjs/operators'; | ||||
| import { MATCHING_ALGORITHMS, MATCH_AUTO } from 'src/app/data/matching-model'; | ||||
| import { ObjectWithId } from 'src/app/data/object-with-id'; | ||||
| import { AbstractPaperlessService } from 'src/app/services/rest/abstract-paperless-service'; | ||||
| import { ToastService } from 'src/app/services/toast.service'; | ||||
| @ -13,8 +14,7 @@ export abstract class EditDialogComponent<T extends ObjectWithId> implements OnI | ||||
|   constructor( | ||||
|     private service: AbstractPaperlessService<T>, | ||||
|     private activeModal: NgbActiveModal, | ||||
|     private toastService: ToastService, | ||||
|     private entityName: string) { } | ||||
|     private toastService: ToastService) { } | ||||
| 
 | ||||
|   @Input() | ||||
|   dialogMode: string = 'create' | ||||
| @ -25,6 +25,10 @@ export abstract class EditDialogComponent<T extends ObjectWithId> implements OnI | ||||
|   @Output() | ||||
|   success = new EventEmitter() | ||||
| 
 | ||||
|   networkActive = false | ||||
| 
 | ||||
|   error = null | ||||
| 
 | ||||
|   abstract getForm(): FormGroup | ||||
| 
 | ||||
|   objectForm: FormGroup = this.getForm() | ||||
| @ -35,12 +39,24 @@ export abstract class EditDialogComponent<T extends ObjectWithId> implements OnI | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   getCreateTitle() { | ||||
|     return $localize`Create new item` | ||||
|   } | ||||
| 
 | ||||
|   getEditTitle() { | ||||
|     return $localize`Edit item` | ||||
|   } | ||||
| 
 | ||||
|   getSaveErrorMessage(error: string) { | ||||
|     return $localize`Could not save element: ${error}` | ||||
|   } | ||||
| 
 | ||||
|   getTitle() { | ||||
|     switch (this.dialogMode) { | ||||
|       case 'create': | ||||
|         return "Create new " + this.entityName | ||||
|         return this.getCreateTitle() | ||||
|       case 'edit': | ||||
|         return "Edit " + this.entityName | ||||
|         return this.getEditTitle() | ||||
|       default: | ||||
|         break; | ||||
|     } | ||||
| @ -50,6 +66,10 @@ export abstract class EditDialogComponent<T extends ObjectWithId> implements OnI | ||||
|     return MATCHING_ALGORITHMS | ||||
|   } | ||||
| 
 | ||||
|   get patternRequired(): boolean { | ||||
|     return this.objectForm?.value.matching_algorithm !== MATCH_AUTO | ||||
|   } | ||||
| 
 | ||||
|   save() { | ||||
|     var newObject = Object.assign(Object.assign({}, this.object), this.objectForm.value) | ||||
|     var serverResponse: Observable<T> | ||||
| @ -62,11 +82,14 @@ export abstract class EditDialogComponent<T extends ObjectWithId> implements OnI | ||||
|       default: | ||||
|         break; | ||||
|     } | ||||
|     this.networkActive = true | ||||
|     serverResponse.subscribe(result => { | ||||
|       this.activeModal.close() | ||||
|       this.success.emit(result) | ||||
|       this.networkActive = false | ||||
|     }, error => { | ||||
|       this.toastService.showError(`Could not save ${this.entityName}: ${error.error.name}`) | ||||
|       this.error = error.error | ||||
|       this.networkActive = false | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -0,0 +1,35 @@ | ||||
| <div class="btn-group" ngbDropdown role="group" (openChange)="dropdownOpenChange($event)" #dropdown="ngbDropdown"> | ||||
|   <button class="btn btn-sm" id="dropdown{{title}}" ngbDropdownToggle [ngClass]="!editing && selectionModel.selectionSize() > 0 ? 'btn-primary' : 'btn-outline-primary'"> | ||||
|     <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="!editing && selectionModel.selectionSize() > 0"> | ||||
|       <div class="badge bg-secondary text-light rounded-pill badge-corner"> | ||||
|         {{selectionModel.selectionSize()}} | ||||
|       </div> | ||||
|     </ng-container> | ||||
|   </button> | ||||
|   <div class="dropdown-menu py-0 shadow" ngbDropdownMenu attr.aria-labelledby="dropdown{{title}}"> | ||||
|     <div class="list-group list-group-flush"> | ||||
|       <div class="list-group-item"> | ||||
|         <div class="input-group input-group-sm"> | ||||
|           <input class="form-control" type="text" [(ngModel)]="filterText" [placeholder]="filterPlaceholder" (keyup.enter)="listFilterEnter()" #listFilterTextInput> | ||||
|         </div> | ||||
|       </div> | ||||
|       <div *ngIf="selectionModel.items" class="items"> | ||||
|         <ng-container *ngFor="let item of (editing ? selectionModel.itemsSorted : selectionModel.items) | filter: filterText"> | ||||
|           <app-toggleable-dropdown-button *ngIf="allowSelectNone || item.id" [item]="item" [state]="selectionModel.get(item.id)" (toggle)="selectionModel.toggle(item.id)"></app-toggleable-dropdown-button> | ||||
|         </ng-container> | ||||
|       </div> | ||||
|       <button *ngIf="editing" class="list-group-item list-group-item-action bg-light" (click)="applyClicked()" [disabled]="!selectionModel.isDirty()"> | ||||
|         <small class="ml-1" [ngClass]="{'font-weight-bold': selectionModel.isDirty()}" i18n>Apply</small> | ||||
|         <svg width="1.5em" height="1em" viewBox="0 0 16 16" fill="currentColor"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#arrow-right" /> | ||||
|         </svg> | ||||
|       </button> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| @ -0,0 +1,14 @@ | ||||
| .badge-corner { | ||||
|   position: absolute; | ||||
|   top: -8px; | ||||
|   right: -8px; | ||||
| } | ||||
| 
 | ||||
| .dropdown-menu { | ||||
|   min-width: 250px; | ||||
| 
 | ||||
|   .items { | ||||
|     max-height: 400px; | ||||
|     overflow-y: scroll; | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { FilterableDropodownComponent } from './filterable-dropdown.component'; | ||||
| 
 | ||||
| describe('FilterableDropodownComponent', () => { | ||||
|   let component: FilterableDropodownComponent; | ||||
|   let fixture: ComponentFixture<FilterableDropodownComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ FilterableDropodownComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(FilterableDropodownComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,267 @@ | ||||
| import { Component, EventEmitter, Input, Output, ElementRef, ViewChild } from '@angular/core'; | ||||
| import { FilterPipe } from  'src/app/pipes/filter.pipe'; | ||||
| import { NgbDropdown } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { ToggleableItemState } from './toggleable-dropdown-button/toggleable-dropdown-button.component'; | ||||
| import { MatchingModel } from 'src/app/data/matching-model'; | ||||
| import { Subject } from 'rxjs'; | ||||
| 
 | ||||
| export interface ChangedItems { | ||||
|   itemsToAdd: MatchingModel[], | ||||
|   itemsToRemove: MatchingModel[] | ||||
| } | ||||
| 
 | ||||
| export class FilterableDropdownSelectionModel { | ||||
| 
 | ||||
|   changed = new Subject<FilterableDropdownSelectionModel>() | ||||
| 
 | ||||
|   multiple = false | ||||
| 
 | ||||
|   items: MatchingModel[] = [] | ||||
| 
 | ||||
|   get itemsSorted(): MatchingModel[] { | ||||
|     return this.items.sort((a,b) => { | ||||
|       if (this.getNonTemporary(a.id) == ToggleableItemState.NotSelected && this.getNonTemporary(b.id) != ToggleableItemState.NotSelected) { | ||||
|         return 1 | ||||
|       } else if (this.getNonTemporary(a.id) != ToggleableItemState.NotSelected && this.getNonTemporary(b.id) == ToggleableItemState.NotSelected) { | ||||
|         return -1 | ||||
|       } else { | ||||
|         return a.name.localeCompare(b.name) | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   private selectionStates = new Map<number, ToggleableItemState>() | ||||
| 
 | ||||
|   private temporarySelectionStates = new Map<number, ToggleableItemState>() | ||||
| 
 | ||||
|   getSelectedItems() { | ||||
|     return this.items.filter(i => this.temporarySelectionStates.get(i.id) == ToggleableItemState.Selected) | ||||
|   } | ||||
| 
 | ||||
|   set(id: number, state: ToggleableItemState, fireEvent = true) { | ||||
|     if (state == ToggleableItemState.NotSelected) { | ||||
|       this.temporarySelectionStates.delete(id) | ||||
|     } else { | ||||
|       this.temporarySelectionStates.set(id, state) | ||||
|     } | ||||
|     if (fireEvent) { | ||||
|       this.changed.next(this) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   toggle(id: number, fireEvent = true) { | ||||
|     let state = this.temporarySelectionStates.get(id) | ||||
|     if (state == null || state != ToggleableItemState.Selected) { | ||||
|       this.temporarySelectionStates.set(id, ToggleableItemState.Selected) | ||||
|     } else if (state == ToggleableItemState.Selected) { | ||||
|       this.temporarySelectionStates.delete(id) | ||||
|     } | ||||
| 
 | ||||
|     if (!this.multiple) { | ||||
|       for (let key of this.temporarySelectionStates.keys()) { | ||||
|         if (key != id) { | ||||
|           this.temporarySelectionStates.delete(key) | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (!id) { | ||||
|       for (let key of this.temporarySelectionStates.keys()) { | ||||
|         if (key) { | ||||
|           this.temporarySelectionStates.delete(key) | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       this.temporarySelectionStates.delete(null) | ||||
|     } | ||||
| 
 | ||||
|     if (fireEvent) { | ||||
|       this.changed.next(this) | ||||
|     } | ||||
|      | ||||
|   } | ||||
| 
 | ||||
|   private getNonTemporary(id: number) { | ||||
|     return this.selectionStates.get(id) || ToggleableItemState.NotSelected | ||||
|   } | ||||
| 
 | ||||
|   get(id: number) { | ||||
|     return this.temporarySelectionStates.get(id) || ToggleableItemState.NotSelected | ||||
|   } | ||||
| 
 | ||||
|   selectionSize() { | ||||
|     return this.getSelectedItems().length | ||||
|   } | ||||
| 
 | ||||
|   clear(fireEvent = true) { | ||||
|     this.temporarySelectionStates.clear() | ||||
|     if (fireEvent) { | ||||
|       this.changed.next(this) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   isDirty() { | ||||
|     if (!Array.from(this.temporarySelectionStates.keys()).every(id => this.temporarySelectionStates.get(id) == this.selectionStates.get(id))) { | ||||
|       return true | ||||
|     } else if (!Array.from(this.selectionStates.keys()).every(id => this.selectionStates.get(id) == this.temporarySelectionStates.get(id))) { | ||||
|       return true | ||||
|     } else { | ||||
|       return false | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   isNoneSelected() { | ||||
|     return this.selectionSize() == 1 && this.get(null) == ToggleableItemState.Selected | ||||
|   } | ||||
| 
 | ||||
|   init(map) { | ||||
|     this.temporarySelectionStates = map | ||||
|     this.apply() | ||||
|   } | ||||
| 
 | ||||
|   apply() { | ||||
|     this.selectionStates.clear() | ||||
|     this.temporarySelectionStates.forEach((value, key) => { | ||||
|       this.selectionStates.set(key, value) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   reset() { | ||||
|     this.temporarySelectionStates.clear() | ||||
|     this.selectionStates.forEach((value, key) => { | ||||
|       this.temporarySelectionStates.set(key, value) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   diff(): ChangedItems { | ||||
|     return { | ||||
|       itemsToAdd: this.items.filter(item => this.temporarySelectionStates.get(item.id) == ToggleableItemState.Selected && this.selectionStates.get(item.id) != ToggleableItemState.Selected), | ||||
|       itemsToRemove: this.items.filter(item => !this.temporarySelectionStates.has(item.id) && this.selectionStates.has(item.id)), | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-filterable-dropdown', | ||||
|   templateUrl: './filterable-dropdown.component.html', | ||||
|   styleUrls: ['./filterable-dropdown.component.scss'] | ||||
| }) | ||||
| export class FilterableDropdownComponent { | ||||
| 
 | ||||
|   @ViewChild('listFilterTextInput') listFilterTextInput: ElementRef | ||||
|   @ViewChild('dropdown') dropdown: NgbDropdown | ||||
| 
 | ||||
|   filterText: string | ||||
| 
 | ||||
|   @Input() | ||||
|   set items(items: MatchingModel[]) { | ||||
|     if (items) { | ||||
|       this._selectionModel.items = Array.from(items) | ||||
|       this._selectionModel.items.unshift({ | ||||
|         name: $localize`:Filter drop down element to filter for documents with no correspondent/type/tag assigned:Not assigned`, | ||||
|         id: null | ||||
|       }) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   get items(): MatchingModel[] { | ||||
|     return this._selectionModel.items | ||||
|   } | ||||
| 
 | ||||
|   _selectionModel = new FilterableDropdownSelectionModel() | ||||
| 
 | ||||
|   @Input() | ||||
|   set selectionModel(model: FilterableDropdownSelectionModel) { | ||||
|     if (this.selectionModel) { | ||||
|       this.selectionModel.changed.complete() | ||||
|       model.items = this.selectionModel.items | ||||
|       model.multiple = this.selectionModel.multiple | ||||
|     } | ||||
|     model.changed.subscribe(updatedModel => { | ||||
|       this.selectionModelChange.next(updatedModel) | ||||
|     }) | ||||
|     this._selectionModel = model | ||||
|   } | ||||
| 
 | ||||
|   get selectionModel(): FilterableDropdownSelectionModel { | ||||
|     return this._selectionModel | ||||
|   } | ||||
| 
 | ||||
|   @Output() | ||||
|   selectionModelChange = new EventEmitter<FilterableDropdownSelectionModel>() | ||||
| 
 | ||||
|   @Input() | ||||
|   set multiple(value: boolean) { | ||||
|     this.selectionModel.multiple = value | ||||
|   } | ||||
| 
 | ||||
|   get multiple() { | ||||
|     return this.selectionModel.multiple | ||||
|   } | ||||
| 
 | ||||
|   @Input() | ||||
|   title: string | ||||
| 
 | ||||
|   @Input() | ||||
|   filterPlaceholder: string = "" | ||||
| 
 | ||||
|   @Input() | ||||
|   icon: string | ||||
| 
 | ||||
|   @Input() | ||||
|   allowSelectNone: boolean = false | ||||
| 
 | ||||
|   @Input() | ||||
|   editing = false | ||||
| 
 | ||||
|   @Input() | ||||
|   applyOnClose = false | ||||
| 
 | ||||
|   @Output() | ||||
|   apply = new EventEmitter<ChangedItems>() | ||||
| 
 | ||||
|   @Output() | ||||
|   open = new EventEmitter() | ||||
| 
 | ||||
|   constructor(private filterPipe: FilterPipe) { | ||||
|     this.selectionModel = new FilterableDropdownSelectionModel() | ||||
|   } | ||||
| 
 | ||||
|   applyClicked() { | ||||
|     if (this.selectionModel.isDirty()) { | ||||
|       this.dropdown.close() | ||||
|       if (!this.applyOnClose) { | ||||
|         this.apply.emit(this.selectionModel.diff()) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   dropdownOpenChange(open: boolean): void { | ||||
|     if (open) { | ||||
|       setTimeout(() => { | ||||
|         this.listFilterTextInput.nativeElement.focus(); | ||||
|       }, 0) | ||||
|       if (this.editing) { | ||||
|         this.selectionModel.reset() | ||||
|       } | ||||
|       this.open.next() | ||||
|     } else { | ||||
|       this.filterText = '' | ||||
|       if (this.applyOnClose && this.selectionModel.isDirty()) { | ||||
|         this.apply.emit(this.selectionModel.diff()) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   listFilterEnter(): void { | ||||
|     let filtered = this.filterPipe.transform(this.items, this.filterText) | ||||
|     if (filtered.length == 1) { | ||||
|       this.selectionModel.toggle(filtered[0].id) | ||||
|       if (this.editing) { | ||||
|         this.applyClicked() | ||||
|       } else { | ||||
|         this.dropdown.close() | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,20 @@ | ||||
| <button class="list-group-item list-group-item-action d-flex align-items-center p-2 border-top-0 border-left-0 border-right-0 border-bottom" role="menuitem" (click)="toggleItem()"> | ||||
|   <div class="selected-icon mr-1"> | ||||
|     <ng-container *ngIf="isChecked()"> | ||||
|       <svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" fill="currentColor" class="bi bi-check" viewBox="0 0 16 16"> | ||||
|         <path d="M10.97 4.97a.75.75 0 0 1 1.07 1.05l-3.99 4.99a.75.75 0 0 1-1.08.02L4.324 8.384a.75.75 0 1 1 1.06-1.06l2.094 2.093 3.473-4.425a.267.267 0 0 1 .02-.022z"/> | ||||
|       </svg> | ||||
|     </ng-container> | ||||
|     <ng-container *ngIf="isPartiallyChecked()"> | ||||
|       <svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" fill="currentColor" class="bi bi-dash" viewBox="0 0 16 16"> | ||||
|         <path d="M4 8a.5.5 0 0 1 .5-.5h7a.5.5 0 0 1 0 1h-7A.5.5 0 0 1 4 8z"/> | ||||
|       </svg> | ||||
|     </ng-container> | ||||
|      | ||||
|   </div> | ||||
|   <div class="mr-1"> | ||||
|     <app-tag *ngIf="isTag; else displayName" [tag]="item" [clickable]="true" linkTitle="Filter by tag"></app-tag> | ||||
|     <ng-template #displayName><small>{{item.name}}</small></ng-template> | ||||
|   </div> | ||||
|   <div class="badge badge-light rounded-pill ml-auto mr-1">{{item.document_count}}</div> | ||||
| </button> | ||||
| @ -0,0 +1,4 @@ | ||||
| .selected-icon { | ||||
|   min-width: 1em; | ||||
|   min-height: 1em; | ||||
| } | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { ToggleableDropdownButtonComponent } from './toggleable-dropdown-button.component'; | ||||
| 
 | ||||
| describe('ToggleableDropdownButtonComponent', () => { | ||||
|   let component: ToggleableDropdownButtonComponent; | ||||
|   let fixture: ComponentFixture<ToggleableDropdownButtonComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ ToggleableDropdownButtonComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(ToggleableDropdownButtonComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,51 @@ | ||||
| import { Component, EventEmitter, Input, Output, OnInit } from '@angular/core'; | ||||
| import { MatchingModel } from 'src/app/data/matching-model'; | ||||
| 
 | ||||
| export interface ToggleableItem { | ||||
|   item: MatchingModel, | ||||
|   state: ToggleableItemState, | ||||
|   count: number | ||||
| } | ||||
| 
 | ||||
| export enum ToggleableItemState { | ||||
|   NotSelected = 0, | ||||
|   Selected = 1, | ||||
|   PartiallySelected = 2 | ||||
| } | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-toggleable-dropdown-button', | ||||
|   templateUrl: './toggleable-dropdown-button.component.html', | ||||
|   styleUrls: ['./toggleable-dropdown-button.component.scss'] | ||||
| }) | ||||
| export class ToggleableDropdownButtonComponent { | ||||
| 
 | ||||
|   @Input() | ||||
|   item: MatchingModel | ||||
| 
 | ||||
|   @Input() | ||||
|   state: ToggleableItemState | ||||
| 
 | ||||
|   @Input() | ||||
|   count: number | ||||
| 
 | ||||
|   @Output() | ||||
|   toggle = new EventEmitter() | ||||
| 
 | ||||
|   get isTag(): boolean { | ||||
|     return 'is_inbox_tag' in this.item | ||||
|   } | ||||
| 
 | ||||
|   toggleItem(): void { | ||||
|     this.toggle.emit() | ||||
|   } | ||||
| 
 | ||||
|   isChecked() { | ||||
|     return this.state == ToggleableItemState.Selected | ||||
|   } | ||||
| 
 | ||||
|   isPartiallyChecked() { | ||||
|     return this.state == ToggleableItemState.PartiallySelected | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| @ -1,5 +1,5 @@ | ||||
| import { Component, Directive, forwardRef, Input, OnInit } from '@angular/core'; | ||||
| import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { Directive, Input, OnInit } from '@angular/core'; | ||||
| import { ControlValueAccessor } from '@angular/forms'; | ||||
| import { v4 as uuidv4 } from 'uuid'; | ||||
| 
 | ||||
| @Directive() | ||||
| @ -30,6 +30,9 @@ export class AbstractInputComponent<T> implements OnInit, ControlValueAccessor { | ||||
|   @Input() | ||||
|   disabled = false; | ||||
| 
 | ||||
|   @Input() | ||||
|   error: string | ||||
| 
 | ||||
|   value: T | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|  | ||||
| @ -3,11 +3,10 @@ | ||||
|       <label for="created_date">{{titleDate}}</label> | ||||
|       <input type="date" class="form-control" id="created_date" [(ngModel)]="dateValue" (change)="dateOrTimeChanged()"> | ||||
|   </div> | ||||
|   <div class="form-group col"> | ||||
|   <div class="form-group col" *ngIf="titleTime"> | ||||
|       <label for="created_time">{{titleTime}}</label> | ||||
|       <input type="time" class="form-control" id="created_time" [(ngModel)]="timeValue" (change)="dateOrTimeChanged()"> | ||||
|   </div> | ||||
| 
 | ||||
| </div> | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -1,7 +1,6 @@ | ||||
| import { formatDate } from '@angular/common'; | ||||
| import { Component, forwardRef, Input, OnInit } from '@angular/core'; | ||||
| import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { AbstractInputComponent } from '../abstract-input'; | ||||
| 
 | ||||
| @Component({ | ||||
|   providers: [{ | ||||
| @ -40,7 +39,7 @@ export class DateTimeComponent implements OnInit,ControlValueAccessor  { | ||||
|   titleDate: string = "Date" | ||||
| 
 | ||||
|   @Input() | ||||
|   titleTime: string = "Time" | ||||
|   titleTime: string | ||||
| 
 | ||||
|   @Input() | ||||
|   disabled: boolean = false | ||||
|  | ||||
| @ -0,0 +1,14 @@ | ||||
| <div class="form-group"> | ||||
|   <label [for]="inputId">{{title}}</label> | ||||
|   <div class="input-group" [class.is-invalid]="error"> | ||||
|     <input type="number" class="form-control" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)" [class.is-invalid]="error"> | ||||
|     <div class="input-group-append"> | ||||
|       <button class="btn btn-outline-secondary" type="button" id="button-addon1" (click)="nextAsn()" [disabled]="value">+1</button> | ||||
|     </div> | ||||
|   </div> | ||||
|   <div class="invalid-feedback"> | ||||
|     {{error}} | ||||
|   </div> | ||||
|   <small *ngIf="hint" class="form-text text-muted">{{hint}}</small> | ||||
| 
 | ||||
| </div> | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { NumberComponent } from './number.component'; | ||||
| 
 | ||||
| describe('NumberComponent', () => { | ||||
|   let component: NumberComponent; | ||||
|   let fixture: ComponentFixture<NumberComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ NumberComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(NumberComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,39 @@ | ||||
| import { Component, forwardRef } from '@angular/core'; | ||||
| import { NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { FILTER_ASN_ISNULL } from 'src/app/data/filter-rule-type'; | ||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | ||||
| import { AbstractInputComponent } from '../abstract-input'; | ||||
| 
 | ||||
| @Component({ | ||||
|   providers: [{ | ||||
|     provide: NG_VALUE_ACCESSOR, | ||||
|     useExisting: forwardRef(() => NumberComponent), | ||||
|     multi: true | ||||
|   }], | ||||
|   selector: 'app-input-number', | ||||
|   templateUrl: './number.component.html', | ||||
|   styleUrls: ['./number.component.scss'] | ||||
| }) | ||||
| export class NumberComponent extends AbstractInputComponent<number> { | ||||
| 
 | ||||
|   constructor(private documentService: DocumentService) { | ||||
|     super() | ||||
|   } | ||||
| 
 | ||||
|   nextAsn() { | ||||
|     if (this.value) { | ||||
|       return | ||||
|     } | ||||
|     this.documentService.listFiltered(1, 1, "archive_serial_number", true, [{rule_type: FILTER_ASN_ISNULL, value: "false"}]).subscribe( | ||||
|       results => { | ||||
|         if (results.count > 0) { | ||||
|           this.value = results.results[0].archive_serial_number + 1 | ||||
|         } else { | ||||
|           this.value + 1 | ||||
|         } | ||||
|         this.onChange(this.value) | ||||
|       } | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| @ -1,11 +1,16 @@ | ||||
| <div class="form-group"> | ||||
| <div class="form-group paperless-input-select"> | ||||
|   <label [for]="inputId">{{title}}</label> | ||||
|   <div [class.input-group]="showPlusButton()"> | ||||
|     <select class="form-control" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)" (blur)="onTouched()" | ||||
|       [disabled]="disabled" [style.color]="textColor" [style.background]="backgroundColor"> | ||||
|       <option *ngIf="allowNull" [ngValue]="null" class="form-control">---</option> | ||||
|       <option *ngFor="let i of items" [ngValue]="i.id" class="form-control">{{i.name}}</option> | ||||
|     </select> | ||||
|     <ng-select name="inputId" [(ngModel)]="value" | ||||
|       [disabled]="disabled" | ||||
|       [style.color]="textColor" | ||||
|       [style.background]="backgroundColor" | ||||
|       [clearable]="allowNull" | ||||
|       (change)="onChange(value)" | ||||
|       (blur)="onTouched()"> | ||||
|       <ng-option *ngFor="let i of items" [value]="i.id">{{i.name}}</ng-option> | ||||
|     </ng-select> | ||||
| 
 | ||||
|     <div *ngIf="showPlusButton()" class="input-group-append"> | ||||
|       <button class="btn btn-outline-secondary" type="button" (click)="createNew.emit()"> | ||||
|         <svg class="buttonicon" fill="currentColor"> | ||||
| @ -15,4 +20,4 @@ | ||||
|     </div> | ||||
|   </div> | ||||
|   <small *ngIf="hint" class="form-text text-muted">{{hint}}</small> | ||||
| </div> | ||||
| </div> | ||||
|  | ||||
| @ -0,0 +1 @@ | ||||
| // styles for ng-select child are in styles.scss | ||||
| @ -1,4 +1,4 @@ | ||||
| import { Component, EventEmitter, forwardRef, Input, OnInit, Output } from '@angular/core'; | ||||
| import { Component, EventEmitter, forwardRef, Input, Output } from '@angular/core'; | ||||
| import { NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { AbstractInputComponent } from '../abstract-input'; | ||||
| 
 | ||||
|  | ||||
| @ -1,30 +1,43 @@ | ||||
| <div class="form-group"> | ||||
|   <label for="exampleFormControlTextarea1">Tags</label> | ||||
| <div class="form-group paperless-input-select paperless-input-tags"> | ||||
|   <label for="tags">Tags</label> | ||||
| 
 | ||||
|   <div class="input-group"> | ||||
|     <div class="form-control tags-form-control" id="tags"> | ||||
|       <app-tag class="mr-2" *ngFor="let id of displayValue" [tag]="getTag(id)" (click)="removeTag(id)"></app-tag> | ||||
|     </div> | ||||
|   <div class="input-group flex-nowrap"> | ||||
|     <ng-select name="tags" [items]="tags" bindLabel="name" bindValue="id" [(ngModel)]="displayValue" | ||||
|       [multiple]="true" | ||||
|       [closeOnSelect]="false" | ||||
|       [clearSearchOnAdd]="true" | ||||
|       [disabled]="disabled" | ||||
|       [hideSelected]="true" | ||||
|       (change)="ngSelectChange()"> | ||||
| 
 | ||||
|     <div class="input-group-append" ngbDropdown placement="top-right"> | ||||
|       <button class="btn btn-outline-secondary" type="button" ngbDropdownToggle></button> | ||||
|       <div ngbDropdownMenu class="scrollable-menu"> | ||||
|         <button type="button" *ngFor="let tag of tags" ngbDropdownItem (click)="addTag(tag.id)"> | ||||
|           <app-tag [tag]="tag"></app-tag> | ||||
|         </button> | ||||
|       </div> | ||||
|     </div> | ||||
|       <ng-template ng-label-tmp let-item="item"> | ||||
|         <span class="tag-wrap tag-wrap-delete" (click)="removeTag(item.id)"> | ||||
|           <svg width="1.2em" height="1em" viewBox="0 0 16 16" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#x"/> | ||||
|           </svg> | ||||
|           <app-tag style="background-color: none;" [tag]="getTag(item.id)"></app-tag> | ||||
|         </span> | ||||
|       </ng-template> | ||||
|       <ng-template ng-option-tmp let-item="item" let-index="index" let-search="searchTerm"> | ||||
|         <div class="tag-wrap"> | ||||
|           <div class="selected-icon d-inline-block mr-1"> | ||||
|             <svg *ngIf="displayValue.includes(item.id)" width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|               <use xlink:href="assets/bootstrap-icons.svg#check"/> | ||||
|             </svg> | ||||
|           </div> | ||||
|           <app-tag class="mr-2" [tag]="getTag(item.id)"></app-tag> | ||||
|         </div> | ||||
|       </ng-template> | ||||
|     </ng-select> | ||||
| 
 | ||||
|     <div class="input-group-append"> | ||||
| 
 | ||||
|       <button class="btn btn-outline-secondary" type="button" (click)="createTag()"> | ||||
|         <svg class="buttonicon" fill="currentColor"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#plus" /> | ||||
|         </svg> | ||||
|       </button> | ||||
|     </div> | ||||
| 
 | ||||
|   </div> | ||||
|   <small class="form-text text-muted" *ngIf="hint">{{hint}}</small> | ||||
| 
 | ||||
| </div> | ||||
| </div> | ||||
|  | ||||
| @ -1,10 +1,12 @@ | ||||
| .tags-form-control { | ||||
|   height: auto; | ||||
| .selected-icon { | ||||
|   min-width: 1em; | ||||
|   min-height: 1em; | ||||
| } | ||||
| 
 | ||||
| .tag-wrap { | ||||
|   font-size: 1rem; | ||||
| } | ||||
| 
 | ||||
| .scrollable-menu { | ||||
|   height: auto; | ||||
|   max-height: 300px; | ||||
|   overflow-x: hidden; | ||||
| } | ||||
| .tag-wrap-delete { | ||||
|   cursor: pointer; | ||||
| } | ||||
|  | ||||
| @ -1,8 +1,6 @@ | ||||
| import { ThrowStmt } from '@angular/compiler'; | ||||
| import { Component, forwardRef, Input, OnInit } from '@angular/core'; | ||||
| import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| import { Observable } from 'rxjs'; | ||||
| import { TagEditDialogComponent } from 'src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component'; | ||||
| import { PaperlessTag } from 'src/app/data/paperless-tag'; | ||||
| import { TagService } from 'src/app/services/rest/tag.service'; | ||||
| @ -23,7 +21,7 @@ export class TagsComponent implements OnInit, ControlValueAccessor { | ||||
| 
 | ||||
| 
 | ||||
|   onChange = (newValue: number[]) => {}; | ||||
|    | ||||
| 
 | ||||
|   onTouched = () => {}; | ||||
| 
 | ||||
|   writeValue(newValue: number[]): void { | ||||
| @ -68,29 +66,28 @@ export class TagsComponent implements OnInit, ControlValueAccessor { | ||||
|   removeTag(id) { | ||||
|     let index = this.displayValue.indexOf(id) | ||||
|     if (index > -1) { | ||||
|       this.displayValue.splice(index, 1) | ||||
|       let oldValue = this.displayValue | ||||
|       oldValue.splice(index, 1) | ||||
|       this.displayValue = [...oldValue] | ||||
|       this.onChange(this.displayValue) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   addTag(id) { | ||||
|     let index = this.displayValue.indexOf(id) | ||||
|     if (index == -1) { | ||||
|       this.displayValue.push(id) | ||||
|       this.onChange(this.displayValue) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|   createTag() { | ||||
|     var modal = this.modalService.open(TagEditDialogComponent, {backdrop: 'static'}) | ||||
|     modal.componentInstance.dialogMode = 'create' | ||||
|     modal.componentInstance.success.subscribe(newTag => { | ||||
|       this.tagService.listAll().subscribe(tags => { | ||||
|         this.tags = tags.results | ||||
|         this.addTag(newTag.id) | ||||
|         this.displayValue = [...this.displayValue, newTag.id] | ||||
|         this.onChange(this.displayValue) | ||||
|       }) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   ngSelectChange() { | ||||
|     this.value = this.displayValue | ||||
|     this.onChange(this.displayValue) | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -1,5 +1,8 @@ | ||||
| <div class="form-group"> | ||||
|   <label [for]="inputId">{{title}}</label> | ||||
|   <input type="text" class="form-control" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)"> | ||||
|   <input type="text" class="form-control" [class.is-invalid]="error" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)"> | ||||
|   <small *ngIf="hint" class="form-text text-muted">{{hint}}</small> | ||||
|   <div class="invalid-feedback"> | ||||
|     {{error}} | ||||
|   </div> | ||||
| </div> | ||||
| @ -1,6 +1,5 @@ | ||||
| import { Component, forwardRef, Input, OnInit } from '@angular/core'; | ||||
| import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { v4 as uuidv4 } from 'uuid'; | ||||
| import { Component, forwardRef } from '@angular/core'; | ||||
| import { NG_VALUE_ACCESSOR } from '@angular/forms'; | ||||
| import { AbstractInputComponent } from '../abstract-input'; | ||||
| 
 | ||||
| @Component({ | ||||
|  | ||||
| @ -1,21 +1,29 @@ | ||||
| import { Component, Input, OnInit } from '@angular/core'; | ||||
| import { Component, Input } from '@angular/core'; | ||||
| import { Title } from '@angular/platform-browser'; | ||||
| import { environment } from 'src/environments/environment'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-page-header', | ||||
|   templateUrl: './page-header.component.html', | ||||
|   styleUrls: ['./page-header.component.scss'] | ||||
| }) | ||||
| export class PageHeaderComponent implements OnInit { | ||||
| export class PageHeaderComponent { | ||||
| 
 | ||||
|   constructor() { } | ||||
|   constructor(private titleService: Title) { } | ||||
| 
 | ||||
|   _title = "" | ||||
| 
 | ||||
|   @Input() | ||||
|   title: string = "" | ||||
|   set title(title: string) { | ||||
|     this._title = title | ||||
|     this.titleService.setTitle(`${this.title} - ${environment.appTitle}`) | ||||
|   } | ||||
| 
 | ||||
|   get title() { | ||||
|     return this._title | ||||
|   } | ||||
| 
 | ||||
|   @Input() | ||||
|   subTitle: string = "" | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,15 @@ | ||||
| <div class="modal-header"> | ||||
|   <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|   <button type="button" class="close" aria-label="Close" (click)="cancelClicked()"> | ||||
|     <span aria-hidden="true">×</span> | ||||
|   </button> | ||||
| </div> | ||||
| <div class="modal-body"> | ||||
| 
 | ||||
|   <app-input-select [items]="objects" [title]="message" [(ngModel)]="selected"></app-input-select> | ||||
| 
 | ||||
| </div> | ||||
| <div class="modal-footer"> | ||||
|   <button type="button" class="btn btn-outline-dark" (click)="cancelClicked()" i18n>Cancel</button> | ||||
|   <button type="button" class="btn btn-primary" (click)="selectClicked.emit(selected)" i18n>Select</button> | ||||
| </div> | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { SelectDialogComponent } from './select-dialog.component'; | ||||
| 
 | ||||
| describe('SelectDialogComponent', () => { | ||||
|   let component: SelectDialogComponent; | ||||
|   let fixture: ComponentFixture<SelectDialogComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ SelectDialogComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(SelectDialogComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,34 @@ | ||||
| import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| import { ObjectWithId } from 'src/app/data/object-with-id'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-select-dialog', | ||||
|   templateUrl: './select-dialog.component.html', | ||||
|   styleUrls: ['./select-dialog.component.scss'] | ||||
| }) | ||||
| 
 | ||||
| export class SelectDialogComponent implements OnInit { | ||||
|   constructor(public activeModal: NgbActiveModal) { } | ||||
| 
 | ||||
|   @Output() | ||||
|   public selectClicked = new EventEmitter() | ||||
| 
 | ||||
|   @Input() | ||||
|   title = $localize`Select` | ||||
| 
 | ||||
|   @Input() | ||||
|   message = $localize`Please select an object` | ||||
| 
 | ||||
|   @Input() | ||||
|   objects: ObjectWithId[] = [] | ||||
| 
 | ||||
|   selected: number | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
|   cancelClicked() { | ||||
|     this.activeModal.close() | ||||
|   } | ||||
| } | ||||
| @ -1,4 +1,4 @@ | ||||
| import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { Component, Input, OnInit } from '@angular/core'; | ||||
| import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag'; | ||||
| 
 | ||||
| @Component({ | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -1,5 +1,7 @@ | ||||
| import { Component, OnInit } from '@angular/core'; | ||||
| import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; | ||||
| import { Meta } from '@angular/platform-browser'; | ||||
| import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; | ||||
| import { SavedViewService } from 'src/app/services/rest/saved-view.service'; | ||||
| 
 | ||||
| 
 | ||||
| @Component({ | ||||
| @ -10,13 +12,36 @@ import { SavedViewConfigService } from 'src/app/services/saved-view-config.servi | ||||
| export class DashboardComponent implements OnInit { | ||||
| 
 | ||||
|   constructor( | ||||
|     public savedViewConfigService: SavedViewConfigService) { } | ||||
|     private savedViewService: SavedViewService, | ||||
|     private meta: Meta | ||||
|   ) { } | ||||
| 
 | ||||
|   get displayName() { | ||||
|     let tagFullName = this.meta.getTag('name=full_name') | ||||
|     let tagUsername = this.meta.getTag('name=username') | ||||
|     if (tagFullName && tagFullName.content) { | ||||
|       return tagFullName.content | ||||
|     } else if (tagUsername && tagUsername.content) { | ||||
|       return tagUsername.content | ||||
|     } else { | ||||
|       return null | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   savedViews = [] | ||||
|   get subtitle() { | ||||
|     if (this.displayName) { | ||||
|       return $localize`Hello ${this.displayName}, welcome to Paperless-ng!` | ||||
|     } else { | ||||
|       return $localize`Welcome to Paperless-ng!` | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   savedViews: PaperlessSavedView[] = [] | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     this.savedViews = this.savedViewConfigService.getDashboardConfigs() | ||||
|     this.savedViewService.listAll().subscribe(results => { | ||||
|       this.savedViews = results.results.filter(savedView => savedView.show_on_dashboard) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -1,19 +1,19 @@ | ||||
| <app-widget-frame [title]="savedView.title"> | ||||
| <app-widget-frame [title]="savedView.name"> | ||||
| 
 | ||||
|   <a header-buttons [routerLink]="" (click)="showAll()">Show all</a> | ||||
|   <a header-buttons [routerLink]="" (click)="showAll()" i18n>Show all</a> | ||||
| 
 | ||||
| 
 | ||||
|   <table content class="table table-sm table-hover table-borderless"> | ||||
|     <thead> | ||||
|       <tr> | ||||
|         <th>Created</th> | ||||
|         <th scope="col">Title</th> | ||||
|         <th i18n>Created</th> | ||||
|         <th scope="col" i18n>Title</th> | ||||
|       </tr> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|       <tr *ngFor="let doc of documents" routerLink="/documents/{{doc.id}}"> | ||||
|         <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> | ||||
|     </tbody> | ||||
|   </table> | ||||
|  | ||||
| @ -0,0 +1,7 @@ | ||||
| table { | ||||
|   overflow-wrap: anywhere; | ||||
| } | ||||
| 
 | ||||
| th:first-child { | ||||
|   min-width: 5rem; | ||||
| } | ||||
| @ -2,7 +2,7 @@ import { Component, Input, OnDestroy, OnInit } from '@angular/core'; | ||||
| import { Router } from '@angular/router'; | ||||
| import { Subscription } from 'rxjs'; | ||||
| import { PaperlessDocument } from 'src/app/data/paperless-document'; | ||||
| import { SavedViewConfig } from 'src/app/data/saved-view-config'; | ||||
| import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; | ||||
| import { DocumentListViewService } from 'src/app/services/document-list-view.service'; | ||||
| import { ConsumerStatusService } from 'src/app/services/consumer-status.service'; | ||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | ||||
| @ -21,7 +21,7 @@ export class SavedViewWidgetComponent implements OnInit { | ||||
|     private consumerStatusService: ConsumerStatusService) { } | ||||
| 
 | ||||
|   @Input() | ||||
|   savedView: SavedViewConfig | ||||
|   savedView: PaperlessSavedView | ||||
| 
 | ||||
|   documents: PaperlessDocument[] = [] | ||||
| 
 | ||||
| @ -39,14 +39,18 @@ export class SavedViewWidgetComponent implements OnInit { | ||||
|   } | ||||
| 
 | ||||
|   reload() { | ||||
|     this.documentService.list(1,10,this.savedView.sortField,this.savedView.sortDirection,this.savedView.filterRules).subscribe(result => { | ||||
|     this.documentService.listFiltered(1,10,this.savedView.sort_field, this.savedView.sort_reverse, this.savedView.filter_rules).subscribe(result => { | ||||
|       this.documents = result.results | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   showAll() { | ||||
|     this.list.load(this.savedView) | ||||
|     this.router.navigate(["documents"]) | ||||
|     if (this.savedView.show_in_sidebar) { | ||||
|       this.router.navigate(['view', this.savedView.id]) | ||||
|     } else { | ||||
|       this.list.load(this.savedView) | ||||
|       this.router.navigate(["documents"]) | ||||
|       } | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,6 @@ | ||||
| <app-widget-frame title="Statistics"> | ||||
| <app-widget-frame title="Statistics" i18n-title> | ||||
|   <ng-container content> | ||||
|     <p class="card-text">Documents in inbox: {{statistics.documents_inbox}}</p> | ||||
|     <p class="card-text">Total documents: {{statistics.documents_total}}</p> | ||||
|     <p class="card-text" i18n>Documents in inbox: {{statistics.documents_inbox}}</p> | ||||
|     <p class="card-text" i18n>Total documents: {{statistics.documents_total}}</p> | ||||
|   </ng-container> | ||||
| </app-widget-frame> | ||||
|  | ||||
| @ -1,15 +1,18 @@ | ||||
| <app-widget-frame title="Upload new documents"> | ||||
| <app-widget-frame title="Upload new documents" i18n-title> | ||||
| 
 | ||||
|   <form content> | ||||
|     <ngx-file-drop  | ||||
|       dropZoneLabel="Drop documents here or" (onFileDrop)="dropped($event)" | ||||
|       (onFileOver)="fileOver($event)" (onFileLeave)="fileLeave($event)" | ||||
|       dropZoneClassName="bg-light card" | ||||
|       multiple="true" | ||||
|       contentClassName="justify-content-center d-flex align-items-center p-5" | ||||
|       [showBrowseBtn]=true | ||||
|       browseBtnClassName="btn btn-sm btn-outline-primary ml-2"> | ||||
|   <div content> | ||||
|     <form> | ||||
|       <ngx-file-drop dropZoneLabel="Drop documents here or" browseBtnLabel="Browse files" (onFileDrop)="dropped($event)" | ||||
|         (onFileOver)="fileOver($event)" (onFileLeave)="fileLeave($event)" dropZoneClassName="bg-light card" | ||||
|         multiple="true" contentClassName="justify-content-center d-flex align-items-center p-5" [showBrowseBtn]=true | ||||
|         browseBtnClassName="btn btn-sm btn-outline-primary ml-2" i18n-dropZoneLabel i18n-browseBtnLabel> | ||||
| 
 | ||||
|     </ngx-file-drop> | ||||
|   </form> | ||||
|       </ngx-file-drop> | ||||
|     </form> | ||||
|     <div *ngIf="uploadVisible" class="mt-3"> | ||||
|       <p i18n>{uploadStatus.length, plural, =1 {Uploading file...} =other {Uploading {{uploadStatus.length}} files...}}</p> | ||||
|       <ngb-progressbar [value]="loadedSum" [max]="totalSum" [striped]="true" [animated]="uploadStatus.length > 0"> | ||||
|       </ngb-progressbar> | ||||
|     </div> | ||||
|   </div> | ||||
| </app-widget-frame> | ||||
| @ -1,7 +1,14 @@ | ||||
| import { HttpEventType } from '@angular/common/http'; | ||||
| import { Component, OnInit } from '@angular/core'; | ||||
| import { FileSystemFileEntry, NgxFileDropEntry } from 'ngx-file-drop'; | ||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | ||||
| import { Toast, ToastService } from 'src/app/services/toast.service'; | ||||
| import { ToastService } from 'src/app/services/toast.service'; | ||||
| 
 | ||||
| 
 | ||||
| interface UploadStatus { | ||||
|   loaded: number | ||||
|   total: number | ||||
| } | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-upload-file-widget', | ||||
| @ -21,23 +28,51 @@ export class UploadFileWidgetComponent implements OnInit { | ||||
|   public fileLeave(event){ | ||||
|   } | ||||
| 
 | ||||
|   uploadStatus: UploadStatus[] = [] | ||||
|   completedFiles = 0 | ||||
| 
 | ||||
|   uploadVisible = false | ||||
| 
 | ||||
|   get loadedSum() { | ||||
|     return this.uploadStatus.map(s => s.loaded).reduce((a,b) => a+b, this.completedFiles > 0 ? 1 : 0) | ||||
|   } | ||||
| 
 | ||||
|   get totalSum() { | ||||
|     return this.uploadStatus.map(s => s.total).reduce((a,b) => a+b, 1) | ||||
|   } | ||||
| 
 | ||||
|   public dropped(files: NgxFileDropEntry[]) { | ||||
|     for (const droppedFile of files) { | ||||
|       if (droppedFile.fileEntry.isFile) { | ||||
|         const fileEntry = droppedFile.fileEntry as FileSystemFileEntry; | ||||
|       let uploadStatusObject: UploadStatus = {loaded: 0, total: 1} | ||||
|       this.uploadStatus.push(uploadStatusObject) | ||||
|       this.uploadVisible = true | ||||
| 
 | ||||
|       const fileEntry = droppedFile.fileEntry as FileSystemFileEntry; | ||||
|         fileEntry.file((file: File) => { | ||||
|           const formData = new FormData() | ||||
|           let formData = new FormData() | ||||
|           formData.append('document', file, file.name) | ||||
|           this.documentService.uploadDocument(formData).subscribe(result => { | ||||
|             this.toastService.showInfo("The document has been uploaded and will be processed by the consumer shortly.") | ||||
| 
 | ||||
|           this.documentService.uploadDocument(formData).subscribe(event => { | ||||
|             if (event.type == HttpEventType.UploadProgress) { | ||||
|               uploadStatusObject.loaded = event.loaded | ||||
|               uploadStatusObject.total = event.total | ||||
|             } else if (event.type == HttpEventType.Response) { | ||||
|               this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) | ||||
|               this.completedFiles += 1 | ||||
|               this.toastService.showInfo($localize`The document has been uploaded and will be processed by the consumer shortly.`) | ||||
|             } | ||||
| 
 | ||||
|           }, error => { | ||||
|             this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) | ||||
|             this.completedFiles += 1 | ||||
|             switch (error.status) { | ||||
|               case 400: { | ||||
|                 this.toastService.showError(`There was an error while uploading the document: ${error.error.document}`) | ||||
|                 this.toastService.showInfo($localize`There was an error while uploading the document: ${error.error.document}`) | ||||
|                 break; | ||||
|               } | ||||
|               default: { | ||||
|                 this.toastService.showError("An error has occured while uploading the document. Sorry!") | ||||
|                 this.toastService.showInfo($localize`An error has occurred while uploading the document. Sorry!`) | ||||
|                 break; | ||||
|               } | ||||
|             } | ||||
|  | ||||
| @ -1,16 +1,16 @@ | ||||
| <app-widget-frame title="First steps"> | ||||
| <app-widget-frame title="First steps" i18n-title> | ||||
| 
 | ||||
|   <ng-container content> | ||||
|     <img src="assets/save-filter.png" class="float-right"> | ||||
|     <p>Paperless is running! :)</p> | ||||
|     <p>You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. | ||||
|       After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and have them displayed on the dashboard instead of this message.</p> | ||||
|     <p>Paperless offers some more features that try to make your life easier, such as:</p> | ||||
|     <p i18n>Paperless is running! :)</p> | ||||
|     <p i18n>You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. | ||||
|       After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message.</p> | ||||
|     <p i18n>Paperless offers some more features that try to make your life easier:</p> | ||||
|     <ul> | ||||
|       <li>Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.</li> | ||||
|       <li>You can configure paperless to read your mails and add documents from attached files.</li> | ||||
|       <li i18n>Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.</li> | ||||
|       <li i18n>You can configure paperless to read your mails and add documents from attached files.</li> | ||||
|     </ul> | ||||
|     <p>Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.</p> | ||||
|     <p i18n>Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.</p> | ||||
|   </ng-container> | ||||
| 
 | ||||
| </app-widget-frame> | ||||
| @ -1,4 +1,4 @@ | ||||
| <div class="card mb-3 shadow"> | ||||
| <div class="card mb-3 shadow-sm"> | ||||
|   <div class="card-header"> | ||||
|     <div class="d-flex justify-content-between align-items-center"> | ||||
|       <h5 class="card-title mb-0">{{title}}</h5> | ||||
|  | ||||
| @ -1,9 +1,18 @@ | ||||
| <app-page-header [(title)]="title"> | ||||
|     <div class="input-group input-group-sm mr-5" *ngIf="getContentType() == 'application/pdf'"> | ||||
|       <div class="input-group-prepend"> | ||||
|         <div class="input-group-text" i18n>Page</div> | ||||
|       </div> | ||||
|       <input class="form-control flex-grow-0 w-auto" type="number" min="1" [max]="previewNumPages" [(ngModel)]="previewCurrentPage" /> | ||||
|       <div class="input-group-append"> | ||||
|         <div class="input-group-text" i18n>of {{previewNumPages}}</div> | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
|     <button type="button" class="btn btn-sm btn-outline-danger mr-2" (click)="delete()"> | ||||
|         <svg class="buttonicon" fill="currentColor"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#trash" /> | ||||
|         </svg> | ||||
|         <span class="d-none d-lg-inline"> Delete</span> | ||||
|         </svg> <span class="d-none d-lg-inline" i18n>Delete</span> | ||||
|     </button> | ||||
| 
 | ||||
|     <div class="btn-group mr-2"> | ||||
| @ -11,65 +20,122 @@ | ||||
|         <a [href]="downloadUrl" class="btn btn-sm btn-outline-primary"> | ||||
|             <svg class="buttonicon" fill="currentColor"> | ||||
|                 <use xlink:href="assets/bootstrap-icons.svg#download" /> | ||||
|             </svg> | ||||
|             <span class="d-none d-lg-inline"> Download</span> | ||||
|             </svg> <span class="d-none d-lg-inline" i18n>Download</span> | ||||
|         </a> | ||||
|      | ||||
|         <div class="btn-group" ngbDropdown role="group" *ngIf="metadata?.paperless__has_archive_version"> | ||||
|           <button class="btn btn-sm btn-outline-primary dropdown-toggle-split" ngbDropdownToggle></button> | ||||
|           <div class="dropdown-menu" ngbDropdownMenu> | ||||
|             <a ngbDropdownItem [href]="downloadOriginalUrl">Download original</a> | ||||
|           </div> | ||||
|         </div> | ||||
|      | ||||
|       </div> | ||||
| 
 | ||||
|         <div class="btn-group" ngbDropdown role="group" *ngIf="metadata?.has_archive_version"> | ||||
|             <button class="btn btn-sm btn-outline-primary dropdown-toggle-split" ngbDropdownToggle></button> | ||||
|             <div class="dropdown-menu shadow" ngbDropdownMenu> | ||||
|                 <a ngbDropdownItem [href]="downloadOriginalUrl" i18n>Download original</a> | ||||
|             </div> | ||||
|         </div> | ||||
| 
 | ||||
|     </div> | ||||
| 
 | ||||
|     <button type="button" class="btn btn-sm btn-outline-primary mr-2" (click)="moreLike()"> | ||||
|         <svg class="buttonicon" fill="currentColor"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#three-dots" /> | ||||
|         </svg> <span class="d-none d-lg-inline" i18n>More like this</span> | ||||
|     </button> | ||||
| 
 | ||||
|     <button type="button" class="btn btn-sm btn-outline-primary" (click)="close()"> | ||||
|         <svg class="buttonicon" fill="currentColor"> | ||||
|             <use xlink:href="assets/bootstrap-icons.svg#x" /> | ||||
|         </svg> | ||||
|         <span class="d-none d-lg-inline"> Close</span> | ||||
|         </svg> <span class="d-none d-lg-inline" i18n>Close</span> | ||||
|     </button> | ||||
| </app-page-header> | ||||
| 
 | ||||
| 
 | ||||
| <div class="row"> | ||||
|     <div class="col-xl"> | ||||
|     <div class="col mb-4"> | ||||
| 
 | ||||
|         <form [formGroup]='documentForm' (ngSubmit)="save()"> | ||||
| 
 | ||||
|             <app-input-text title="Title" formControlName="title"></app-input-text> | ||||
|             <ul ngbNav #nav="ngbNav" class="nav-tabs"> | ||||
|                 <li [ngbNavItem]="1"> | ||||
|                     <a ngbNavLink i18n>Details</a> | ||||
|                     <ng-template ngbNavContent> | ||||
| 
 | ||||
|             <div class="form-group"> | ||||
|                 <label for="archive_serial_number">Archive Serial Number</label> | ||||
|                 <input type="number" class="form-control" id="archive_serial_number" | ||||
|                     formControlName='archive_serial_number'> | ||||
|             </div> | ||||
|                         <app-input-text i18n-title title="Title" formControlName="title" [error]="error?.title"></app-input-text> | ||||
|                         <app-input-number i18n-title title="Archive serial number" [error]="error?.archive_serial_number" formControlName='archive_serial_number'></app-input-number> | ||||
|                         <app-input-date-time i18n-titleDate titleDate="Date created" formControlName="created"></app-input-date-time> | ||||
|                         <app-input-select [items]="correspondents" i18n-title title="Correspondent" formControlName="correspondent" [allowNull]="true" | ||||
|                             (createNew)="createCorrespondent()"></app-input-select> | ||||
|                         <app-input-select [items]="documentTypes" i18n-title title="Document type" formControlName="document_type" [allowNull]="true" | ||||
|                             (createNew)="createDocumentType()"></app-input-select> | ||||
|                         <app-input-tags formControlName="tags" i18n-title title="Tags"></app-input-tags> | ||||
| 
 | ||||
|             <app-input-date-time title="Date created" titleTime="Time created" formControlName="created"></app-input-date-time> | ||||
|                     </ng-template> | ||||
|                 </li> | ||||
| 
 | ||||
|             <div class="form-group"> | ||||
|                 <label for="content">Content</label> | ||||
|                 <textarea class="form-control" id="content" rows="5" formControlName='content'></textarea> | ||||
|             </div> | ||||
|                 <li [ngbNavItem]="2"> | ||||
|                     <a ngbNavLink i18n>Content</a> | ||||
|                     <ng-template ngbNavContent> | ||||
|                         <div class="form-group"> | ||||
|                             <textarea class="form-control" id="content" rows="20" formControlName='content'></textarea> | ||||
|                         </div> | ||||
|                     </ng-template> | ||||
|                 </li> | ||||
| 
 | ||||
|             <app-input-select [items]="correspondents" title="Correspondent" formControlName="correspondent" allowNull="true" (createNew)="createCorrespondent()"></app-input-select> | ||||
|                 <li [ngbNavItem]="3"> | ||||
|                     <a ngbNavLink i18n>Metadata</a> | ||||
|                     <ng-template ngbNavContent> | ||||
| 
 | ||||
|             <app-input-select [items]="documentTypes" title="Document type" formControlName="document_type" allowNull="true" (createNew)="createDocumentType()"></app-input-select> | ||||
|                         <table class="table table-borderless"> | ||||
|                             <tbody> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Date modified</td> | ||||
|                                     <td>{{document.modified | date:'medium'}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Date added</td> | ||||
|                                     <td>{{document.added | date:'medium'}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Media filename</td> | ||||
|                                     <td>{{metadata?.media_filename}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Original MD5 checksum</td> | ||||
|                                     <td>{{metadata?.original_checksum}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Original file size</td> | ||||
|                                     <td>{{metadata?.original_size | fileSize}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr> | ||||
|                                     <td i18n>Original mime type</td> | ||||
|                                     <td>{{metadata?.original_mime_type}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr *ngIf="metadata?.has_archive_version"> | ||||
|                                     <td i18n>Archive MD5 checksum</td> | ||||
|                                     <td>{{metadata?.archive_checksum}}</td> | ||||
|                                 </tr> | ||||
|                                 <tr *ngIf="metadata?.has_archive_version"> | ||||
|                                     <td i18n>Archive file size</td> | ||||
|                                     <td>{{metadata?.archive_size | fileSize}}</td> | ||||
|                                 </tr> | ||||
|                             </tbody> | ||||
|                         </table> | ||||
| 
 | ||||
|             <app-input-tags formControlName="tags" title="Tags"></app-input-tags> | ||||
|                         <app-metadata-collapse i18n-title title="Original document metadata" [metadata]="metadata.original_metadata" *ngIf="metadata?.original_metadata?.length > 0"></app-metadata-collapse> | ||||
|                         <app-metadata-collapse i18n-title title="Archived document metadata" [metadata]="metadata.archive_metadata" *ngIf="metadata?.archive_metadata?.length > 0"></app-metadata-collapse> | ||||
| 
 | ||||
|             <button type="button" class="btn btn-outline-secondary" (click)="discard()">Discard</button>  | ||||
|             <button type="button" class="btn btn-outline-primary" (click)="saveEditNext()" *ngIf="hasNext()">Save & edit next</button>  | ||||
|             <button type="submit" class="btn btn-primary">Save</button>  | ||||
|                     </ng-template> | ||||
|                 </li> | ||||
|             </ul> | ||||
| 
 | ||||
|             <div [ngbNavOutlet]="nav" class="mt-2"></div> | ||||
| 
 | ||||
|             <button type="button" class="btn btn-outline-secondary" (click)="discard()" i18n [disabled]="networkActive">Discard</button>  | ||||
|             <button type="button" class="btn btn-outline-primary" (click)="saveEditNext()" *ngIf="hasNext()" i18n [disabled]="networkActive">Save & next</button>  | ||||
|             <button type="submit" class="btn btn-primary" i18n [disabled]="networkActive">Save</button>  | ||||
|         </form> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="col-xl"> | ||||
|         <object [data]="previewUrl | safe" type="application/pdf" width="100%" height="100%"> | ||||
|             <p>Your browser does not support PDFs. | ||||
|                 <a href="previewUrl">Download the PDF</a>.</p> | ||||
|         </object> | ||||
| 
 | ||||
|     <div class="col-md-6 col-xl-8 mb-3"> | ||||
|       <div class="pdf-viewer-container" *ngIf="getContentType() == 'application/pdf'"> | ||||
|         <pdf-viewer [src]="previewUrl" [original-size]="false" [show-borders]="true" [show-all]="true" [(page)]="previewCurrentPage" [render-text-mode]="2" (after-load-complete)="pdfPreviewLoaded($event)"></pdf-viewer> | ||||
|       </div> | ||||
|     </div> | ||||
| </div> | ||||
|  | ||||
| @ -0,0 +1,6 @@ | ||||
| .pdf-viewer-container { | ||||
|   height: calc(100vh - 160px); | ||||
|   top: 70px; | ||||
|   position: sticky; | ||||
|   background-color: gray; | ||||
| } | ||||
| @ -6,14 +6,17 @@ import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; | ||||
| import { PaperlessDocument } from 'src/app/data/paperless-document'; | ||||
| import { PaperlessDocumentMetadata } from 'src/app/data/paperless-document-metadata'; | ||||
| 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 { OpenDocumentsService } from 'src/app/services/open-documents.service'; | ||||
| import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; | ||||
| import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; | ||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | ||||
| import { DeleteDialogComponent } from '../common/delete-dialog/delete-dialog.component'; | ||||
| import { ConfirmDialogComponent } from '../common/confirm-dialog/confirm-dialog.component'; | ||||
| import { CorrespondentEditDialogComponent } from '../manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component'; | ||||
| import { DocumentTypeEditDialogComponent } from '../manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component'; | ||||
| import { PDFDocumentProxy } from 'ng2-pdf-viewer'; | ||||
| import { ToastService } from 'src/app/services/toast.service'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-document-detail', | ||||
| @ -22,6 +25,13 @@ import { DocumentTypeEditDialogComponent } from '../manage/document-type-list/do | ||||
| }) | ||||
| export class DocumentDetailComponent implements OnInit { | ||||
| 
 | ||||
|   expandOriginalMetadata = false | ||||
|   expandArchivedMetadata = false | ||||
| 
 | ||||
|   error: any | ||||
| 
 | ||||
|   networkActive = false | ||||
| 
 | ||||
|   documentId: number | ||||
|   document: PaperlessDocument | ||||
|   metadata: PaperlessDocumentMetadata | ||||
| @ -43,15 +53,24 @@ export class DocumentDetailComponent implements OnInit { | ||||
|     tags: new FormControl([]) | ||||
|   }) | ||||
| 
 | ||||
|   previewCurrentPage: number = 1 | ||||
|   previewNumPages: number = 1 | ||||
| 
 | ||||
|   constructor( | ||||
|     private documentsService: DocumentService,  | ||||
|     private documentsService: DocumentService, | ||||
|     private route: ActivatedRoute, | ||||
|     private correspondentService: CorrespondentService, | ||||
|     private documentTypeService: DocumentTypeService, | ||||
|     private router: Router, | ||||
|     private modalService: NgbModal, | ||||
|     private openDocumentService: OpenDocumentsService, | ||||
|     private documentListViewService: DocumentListViewService) { } | ||||
|     private documentListViewService: DocumentListViewService, | ||||
|     private documentTitlePipe: DocumentTitlePipe, | ||||
|     private toastService: ToastService) { } | ||||
| 
 | ||||
|   getContentType() { | ||||
|     return this.metadata?.has_archive_version ? 'application/pdf' : this.metadata?.original_mime_type | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     this.documentForm.valueChanges.subscribe(wow => { | ||||
| @ -83,7 +102,7 @@ export class DocumentDetailComponent implements OnInit { | ||||
|     this.documentsService.getMetadata(doc.id).subscribe(result => { | ||||
|       this.metadata = result | ||||
|     }) | ||||
|     this.title = doc.title | ||||
|     this.title = this.documentTitlePipe.transform(doc.title) | ||||
|     this.documentForm.patchValue(doc) | ||||
|   } | ||||
| 
 | ||||
| @ -117,20 +136,34 @@ export class DocumentDetailComponent implements OnInit { | ||||
|     }, error => {this.router.navigate(['404'])}) | ||||
|   } | ||||
| 
 | ||||
|   save() {     | ||||
|   save() { | ||||
|     this.networkActive = true | ||||
|     this.documentsService.update(this.document).subscribe(result => { | ||||
|       this.close() | ||||
|       this.networkActive = false | ||||
|       this.error = null | ||||
|     }, error => { | ||||
|       this.networkActive = false | ||||
|       this.error = error.error | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   saveEditNext() { | ||||
|     this.networkActive = true | ||||
|     this.documentsService.update(this.document).subscribe(result => { | ||||
|       this.error = null | ||||
|       this.documentListViewService.getNext(this.document.id).subscribe(nextDocId => { | ||||
|         this.networkActive = false | ||||
|         if (nextDocId) { | ||||
|           this.openDocumentService.closeDocument(this.document) | ||||
|           this.router.navigate(['documents', nextDocId]) | ||||
|         } | ||||
|       }, error => { | ||||
|         this.networkActive = false | ||||
|       }) | ||||
|     }, error => { | ||||
|       this.networkActive = false | ||||
|       this.error = error.error | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
| @ -144,19 +177,35 @@ export class DocumentDetailComponent implements OnInit { | ||||
|   } | ||||
| 
 | ||||
|   delete() { | ||||
|     let modal = this.modalService.open(DeleteDialogComponent, {backdrop: 'static'}) | ||||
|     modal.componentInstance.message = `Do you really want to delete document '${this.document.title}'?` | ||||
|     modal.componentInstance.message2 = `The files for this document will be deleted permanently. This operation cannot be undone.` | ||||
|     modal.componentInstance.deleteClicked.subscribe(() => { | ||||
|     let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) | ||||
|     modal.componentInstance.title = $localize`Confirm delete` | ||||
|     modal.componentInstance.messageBold = $localize`Do you really want to delete document "${this.document.title}"?` | ||||
|     modal.componentInstance.message = $localize`The files for this document will be deleted permanently. This operation cannot be undone.` | ||||
|     modal.componentInstance.btnClass = "btn-danger" | ||||
|     modal.componentInstance.btnCaption = $localize`Delete document` | ||||
|     modal.componentInstance.confirmClicked.subscribe(() => { | ||||
|       modal.componentInstance.buttonsEnabled = false | ||||
|       this.documentsService.delete(this.document).subscribe(() => { | ||||
|         modal.close()   | ||||
|         modal.close() | ||||
|         this.close() | ||||
|       }, error => { | ||||
|         this.toastService.showError($localize`Error deleting document: ${JSON.stringify(error)}`) | ||||
|         modal.componentInstance.buttonsEnabled = true | ||||
|       }) | ||||
|     }) | ||||
| 
 | ||||
|   } | ||||
| 
 | ||||
|   moreLike() { | ||||
|     this.router.navigate(["search"], {queryParams: {more_like:this.document.id}}) | ||||
|   } | ||||
| 
 | ||||
|   hasNext() { | ||||
|     return this.documentListViewService.hasNext(this.documentId) | ||||
|   } | ||||
| 
 | ||||
|   pdfPreviewLoaded(pdf: PDFDocumentProxy) { | ||||
|     this.previewNumPages = pdf.numPages | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,23 @@ | ||||
| <h6> | ||||
|   <button type="button" class="btn btn-outline-secondary btn-sm mr-2" | ||||
|       (click)="expand = !expand"> | ||||
|       <svg class="buttonicon" fill="currentColor" *ngIf="!expand"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#caret-down" /> | ||||
|       </svg> | ||||
|       <svg class="buttonicon" fill="currentColor" *ngIf="expand"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#caret-up" /> | ||||
|       </svg> | ||||
|   </button> | ||||
|   {{title}} | ||||
| </h6> | ||||
| 
 | ||||
| <div #collapse="ngbCollapse" [(ngbCollapse)]="!expand"> | ||||
|   <table class="table table-borderless"> | ||||
|       <tbody> | ||||
|           <tr *ngFor="let m of metadata"> | ||||
|               <td>{{m.prefix}}:{{m.key}}</td> | ||||
|               <td class="metadata-column">{{m.value}}</td> | ||||
|           </tr> | ||||
|       </tbody> | ||||
|   </table> | ||||
| </div> | ||||
| @ -0,0 +1,3 @@ | ||||
| .metadata-column { | ||||
|   overflow-wrap: anywhere; | ||||
| } | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { MetadataCollapseComponent } from './metadata-collapse.component'; | ||||
| 
 | ||||
| describe('MetadataCollapseComponent', () => { | ||||
|   let component: MetadataCollapseComponent; | ||||
|   let fixture: ComponentFixture<MetadataCollapseComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ MetadataCollapseComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(MetadataCollapseComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,23 @@ | ||||
| import { Component, Input, OnInit } from '@angular/core'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-metadata-collapse', | ||||
|   templateUrl: './metadata-collapse.component.html', | ||||
|   styleUrls: ['./metadata-collapse.component.scss'] | ||||
| }) | ||||
| export class MetadataCollapseComponent implements OnInit { | ||||
| 
 | ||||
|   constructor() { } | ||||
| 
 | ||||
|   expand = false | ||||
| 
 | ||||
|   @Input() | ||||
|   metadata | ||||
| 
 | ||||
|   @Input() | ||||
|   title = $localize`Metadata` | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| @ -0,0 +1,67 @@ | ||||
| <div class="row"> | ||||
|   <div class="col-auto mb-2 mb-xl-0" role="group" aria-label="Select"> | ||||
|     <button class="btn btn-sm btn-outline-danger" (click)="list.selectNone()"> | ||||
|       <svg width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor"> | ||||
|         <use xlink:href="assets/bootstrap-icons.svg#slash-circle" /> | ||||
|       </svg> <ng-container i18n>Cancel</ng-container> | ||||
|     </button> | ||||
|   </div> | ||||
|   <div class="w-100 d-xl-none"></div> | ||||
|   <div class="col-auto mb-2 mb-xl-0" role="group" aria-label="Select"> | ||||
|     <label class="mr-2 mb-0" i18n>Select:</label> | ||||
|     <div class="btn-group"> | ||||
|       <button class="btn btn-sm btn-outline-primary" (click)="list.selectPage()"> | ||||
|         <svg width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#file-earmark-check" /> | ||||
|         </svg> <ng-container i18n>Page</ng-container> | ||||
|       </button> | ||||
|       <button class="btn btn-sm btn-outline-primary" (click)="list.selectAll()"> | ||||
|         <svg width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor"> | ||||
|           <use xlink:href="assets/bootstrap-icons.svg#check-all" /> | ||||
|         </svg> <ng-container i18n>All</ng-container> | ||||
|       </button> | ||||
|     </div> | ||||
|   </div> | ||||
|   <div class="w-100 d-xl-none"></div> | ||||
|   <div class="col-auto mb-2 mb-xl-0"> | ||||
|     <div class="d-flex"> | ||||
|       <label class="ml-auto mt-1 mb-0 mr-2" i18n>Edit:</label> | ||||
|       <app-filterable-dropdown class="mr-2 mr-md-3" title="Tags" icon="tag-fill" i18n-title | ||||
|         filterPlaceholder="Filter tags" i18n-filterPlaceholder | ||||
|         [items]="tags" | ||||
|         [editing]="true" | ||||
|         [multiple]="true" | ||||
|         [applyOnClose]="applyOnClose" | ||||
|         (open)="openTagsDropdown()" | ||||
|         [(selectionModel)]="tagSelectionModel" | ||||
|         (apply)="setTags($event)"> | ||||
|       </app-filterable-dropdown> | ||||
|       <app-filterable-dropdown class="mr-2 mr-md-3" title="Correspondent" icon="person-fill" i18n-title | ||||
|         filterPlaceholder="Filter correspondents" i18n-filterPlaceholder | ||||
|         [items]="correspondents" | ||||
|         [editing]="true" | ||||
|         [applyOnClose]="applyOnClose" | ||||
|         (open)="openCorrespondentDropdown()" | ||||
|         [(selectionModel)]="correspondentSelectionModel" | ||||
|         (apply)="setCorrespondents($event)"> | ||||
|       </app-filterable-dropdown> | ||||
|       <app-filterable-dropdown class="mr-2 mr-md-3" title="Document type" icon="file-earmark-fill" i18n-title | ||||
|         filterPlaceholder="Filter document types" i18n-filterPlaceholder | ||||
|         [items]="documentTypes" | ||||
|         [editing]="true" | ||||
|         [applyOnClose]="applyOnClose" | ||||
|         (open)="openDocumentTypeDropdown()" | ||||
|         [(selectionModel)]="documentTypeSelectionModel" | ||||
|         (apply)="setDocumentTypes($event)"> | ||||
|       </app-filterable-dropdown> | ||||
|     </div> | ||||
|   </div> | ||||
|   <div class="w-100 d-xl-none"></div> | ||||
|   <div class="col mb-2 mb-xl-0 d-flex"> | ||||
|     <button type="button" class="btn btn-sm btn-outline-danger ml-0 ml-lg-auto" (click)="applyDelete()"> | ||||
|       <svg width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor"> | ||||
|         <use xlink:href="assets/bootstrap-icons.svg#trash" /> | ||||
|       </svg> <ng-container i18n>Delete</ng-container> | ||||
|     </button> | ||||
|   </div> | ||||
| </div> | ||||
| @ -0,0 +1,25 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing'; | ||||
| 
 | ||||
| import { BulkEditorComponent } from './bulk-editor.component'; | ||||
| 
 | ||||
| describe('BulkEditorComponent', () => { | ||||
|   let component: BulkEditorComponent; | ||||
|   let fixture: ComponentFixture<BulkEditorComponent>; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [ BulkEditorComponent ] | ||||
|     }) | ||||
|     .compileComponents(); | ||||
|   }); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     fixture = TestBed.createComponent(BulkEditorComponent); | ||||
|     component = fixture.componentInstance; | ||||
|     fixture.detectChanges(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should create', () => { | ||||
|     expect(component).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @ -0,0 +1,210 @@ | ||||
| import { Component } from '@angular/core'; | ||||
| import { PaperlessTag } from 'src/app/data/paperless-tag'; | ||||
| import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; | ||||
| import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; | ||||
| import { TagService } from 'src/app/services/rest/tag.service'; | ||||
| import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; | ||||
| import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; | ||||
| import { DocumentListViewService } from 'src/app/services/document-list-view.service'; | ||||
| import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; | ||||
| import { DocumentService, SelectionDataItem } from 'src/app/services/rest/document.service'; | ||||
| import { OpenDocumentsService } from 'src/app/services/open-documents.service'; | ||||
| import { ConfirmDialogComponent } from 'src/app/components/common/confirm-dialog/confirm-dialog.component'; | ||||
| import { ChangedItems, FilterableDropdownSelectionModel } from '../../common/filterable-dropdown/filterable-dropdown.component'; | ||||
| import { ToggleableItemState } from '../../common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; | ||||
| import { MatchingModel } from 'src/app/data/matching-model'; | ||||
| import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; | ||||
| import { ToastService } from 'src/app/services/toast.service'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-bulk-editor', | ||||
|   templateUrl: './bulk-editor.component.html', | ||||
|   styleUrls: ['./bulk-editor.component.scss'] | ||||
| }) | ||||
| export class BulkEditorComponent { | ||||
| 
 | ||||
|   tags: PaperlessTag[] | ||||
|   correspondents: PaperlessCorrespondent[] | ||||
|   documentTypes: PaperlessDocumentType[] | ||||
| 
 | ||||
|   tagSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   correspondentSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   documentTypeSelectionModel = new FilterableDropdownSelectionModel() | ||||
| 
 | ||||
|   constructor( | ||||
|     private documentTypeService: DocumentTypeService, | ||||
|     private tagService: TagService, | ||||
|     private correspondentService: CorrespondentService, | ||||
|     public list: DocumentListViewService, | ||||
|     private documentService: DocumentService, | ||||
|     private modalService: NgbModal, | ||||
|     private openDocumentService: OpenDocumentsService, | ||||
|     private settings: SettingsService, | ||||
|     private toastService: ToastService | ||||
|   ) { } | ||||
| 
 | ||||
|   applyOnClose: boolean = this.settings.get(SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE) | ||||
|   showConfirmationDialogs: boolean = this.settings.get(SETTINGS_KEYS.BULK_EDIT_CONFIRMATION_DIALOGS) | ||||
| 
 | ||||
|   ngOnInit() { | ||||
|     this.tagService.listAll().subscribe(result => this.tags = result.results) | ||||
|     this.correspondentService.listAll().subscribe(result => this.correspondents = result.results) | ||||
|     this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results) | ||||
|   } | ||||
| 
 | ||||
|   private executeBulkOperation(modal, method: string, args) { | ||||
|     if (modal) { | ||||
|       modal.componentInstance.buttonsEnabled = false | ||||
|     } | ||||
|     this.documentService.bulkEdit(Array.from(this.list.selected), method, args).subscribe( | ||||
|       response => { | ||||
|         this.list.reload() | ||||
|         this.list.reduceSelectionToFilter() | ||||
|         this.list.selected.forEach(id => { | ||||
|           this.openDocumentService.refreshDocument(id) | ||||
|         }) | ||||
|         if (modal) { | ||||
|           modal.close() | ||||
|         } | ||||
|       }, error => { | ||||
|         if (modal) { | ||||
|           modal.componentInstance.buttonsEnabled = true | ||||
|         } | ||||
|         this.toastService.showError($localize`Error executing bulk operation: ${JSON.stringify(error.error)}`) | ||||
|       } | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   private applySelectionData(items: SelectionDataItem[], selectionModel: FilterableDropdownSelectionModel) { | ||||
|     let selectionData = new Map<number, ToggleableItemState>() | ||||
|     items.forEach(i => { | ||||
|       if (i.document_count == this.list.selected.size) { | ||||
|         selectionData.set(i.id, ToggleableItemState.Selected) | ||||
|       } else if (i.document_count > 0) { | ||||
|         selectionData.set(i.id, ToggleableItemState.PartiallySelected) | ||||
|       } | ||||
|     }) | ||||
|     selectionModel.init(selectionData) | ||||
|   } | ||||
| 
 | ||||
|   openTagsDropdown() { | ||||
|     this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { | ||||
|       this.applySelectionData(s.selected_tags, this.tagSelectionModel) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   openDocumentTypeDropdown() { | ||||
|     this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { | ||||
|       this.applySelectionData(s.selected_document_types, this.documentTypeSelectionModel) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   openCorrespondentDropdown() { | ||||
|     this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { | ||||
|       this.applySelectionData(s.selected_correspondents, this.correspondentSelectionModel) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   private _localizeList(items: MatchingModel[]) { | ||||
|     if (items.length == 0) { | ||||
|       return "" | ||||
|     } else if (items.length == 1) { | ||||
|       return items[0].name | ||||
|     } else if (items.length == 2) { | ||||
|       return $localize`:This is for messages like 'modify "tag1" and "tag2"':"${items[0].name}" and "${items[1].name}"` | ||||
|     } else { | ||||
|       let list = items.slice(0, items.length - 1).map(i => $localize`"${i.name}"`).join($localize`:this is used to separate enumerations and should probably be a comma and a whitespace in most languages:, `) | ||||
|       return $localize`:this is for messages like 'modify "tag1", "tag2" and "tag3"':${list} and "${items[items.length - 1].name}"` | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   setTags(changedTags: ChangedItems) { | ||||
|     if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length == 0) return | ||||
| 
 | ||||
|     if (this.showConfirmationDialogs) { | ||||
|       let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) | ||||
|       modal.componentInstance.title = $localize`Confirm tags assignment` | ||||
|       if (changedTags.itemsToAdd.length == 1 && changedTags.itemsToRemove.length == 0) { | ||||
|         let tag = changedTags.itemsToAdd[0] | ||||
|         modal.componentInstance.message = $localize`This operation will add the tag "${tag.name}" to ${this.list.selected.size} selected document(s).` | ||||
|       } else if (changedTags.itemsToAdd.length > 1 && changedTags.itemsToRemove.length == 0) { | ||||
|         modal.componentInstance.message = $localize`This operation will add the tags ${this._localizeList(changedTags.itemsToAdd)} to ${this.list.selected.size} selected document(s).` | ||||
|       } else if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length == 1) { | ||||
|         let tag = changedTags.itemsToRemove[0] | ||||
|         modal.componentInstance.message = $localize`This operation will remove the tag "${tag.name}" from ${this.list.selected.size} selected document(s).` | ||||
|       } else if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length > 1) { | ||||
|         modal.componentInstance.message = $localize`This operation will remove the tags ${this._localizeList(changedTags.itemsToRemove)} from ${this.list.selected.size} selected document(s).` | ||||
|       } else { | ||||
|         modal.componentInstance.message = $localize`This operation will add the tags ${this._localizeList(changedTags.itemsToAdd)} and remove the tags ${this._localizeList(changedTags.itemsToRemove)} on ${this.list.selected.size} selected document(s).` | ||||
|       } | ||||
|        | ||||
|       modal.componentInstance.btnClass = "btn-warning" | ||||
|       modal.componentInstance.btnCaption = $localize`Confirm` | ||||
|       modal.componentInstance.confirmClicked.subscribe(() => { | ||||
|         this.executeBulkOperation(modal, 'modify_tags', {"add_tags": changedTags.itemsToAdd.map(t => t.id), "remove_tags": changedTags.itemsToRemove.map(t => t.id)}) | ||||
|       }) | ||||
|     } else { | ||||
|       this.executeBulkOperation(null, 'modify_tags', {"add_tags": changedTags.itemsToAdd.map(t => t.id), "remove_tags": changedTags.itemsToRemove.map(t => t.id)}) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   setCorrespondents(changedCorrespondents: ChangedItems) { | ||||
|     if (changedCorrespondents.itemsToAdd.length == 0 && changedCorrespondents.itemsToRemove.length == 0) return | ||||
| 
 | ||||
|     let correspondent = changedCorrespondents.itemsToAdd.length > 0 ? changedCorrespondents.itemsToAdd[0] : null | ||||
| 
 | ||||
|     if (this.showConfirmationDialogs) { | ||||
|       let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) | ||||
|       modal.componentInstance.title = $localize`Confirm correspondent assignment` | ||||
|       if (correspondent) { | ||||
|         modal.componentInstance.message = $localize`This operation will assign the correspondent "${correspondent.name}" to ${this.list.selected.size} selected document(s).` | ||||
|       } else { | ||||
|         modal.componentInstance.message = $localize`This operation will remove the correspondent from ${this.list.selected.size} selected document(s).` | ||||
|       } | ||||
|       modal.componentInstance.btnClass = "btn-warning" | ||||
|       modal.componentInstance.btnCaption = $localize`Confirm` | ||||
|       modal.componentInstance.confirmClicked.subscribe(() => { | ||||
|         this.executeBulkOperation(modal, 'set_correspondent', {"correspondent": correspondent ? correspondent.id : null}) | ||||
|       }) | ||||
|     } else { | ||||
|       this.executeBulkOperation(null, 'set_correspondent', {"correspondent": correspondent ? correspondent.id : null}) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   setDocumentTypes(changedDocumentTypes: ChangedItems) { | ||||
|     if (changedDocumentTypes.itemsToAdd.length == 0 && changedDocumentTypes.itemsToRemove.length == 0) return | ||||
| 
 | ||||
|     let documentType = changedDocumentTypes.itemsToAdd.length > 0 ? changedDocumentTypes.itemsToAdd[0] : null | ||||
| 
 | ||||
|     if (this.showConfirmationDialogs) { | ||||
|       let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) | ||||
|       modal.componentInstance.title = $localize`Confirm document type assignment` | ||||
|       if (documentType) { | ||||
|         modal.componentInstance.message = $localize`This operation will assign the document type "${documentType.name}" to ${this.list.selected.size} selected document(s).` | ||||
|       } else { | ||||
|         modal.componentInstance.message = $localize`This operation will remove the document type from ${this.list.selected.size} selected document(s).` | ||||
|       } | ||||
|       modal.componentInstance.btnClass = "btn-warning" | ||||
|       modal.componentInstance.btnCaption = $localize`Confirm` | ||||
|       modal.componentInstance.confirmClicked.subscribe(() => { | ||||
|         this.executeBulkOperation(modal, 'set_document_type', {"document_type": documentType ? documentType.id : null}) | ||||
|       }) | ||||
|     } else { | ||||
|       this.executeBulkOperation(null, 'set_document_type', {"document_type": documentType ? documentType.id : null}) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   applyDelete() { | ||||
|     let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) | ||||
|     modal.componentInstance.delayConfirm(5) | ||||
|     modal.componentInstance.title = $localize`Delete confirm` | ||||
|     modal.componentInstance.messageBold = $localize`This operation will permanently delete ${this.list.selected.size} selected document(s).` | ||||
|     modal.componentInstance.message = $localize`This operation cannot be undone.` | ||||
|     modal.componentInstance.btnClass = "btn-danger" | ||||
|     modal.componentInstance.btnCaption = $localize`Delete document(s)` | ||||
|     modal.componentInstance.confirmClicked.subscribe(() => { | ||||
|       modal.componentInstance.buttonsEnabled = false | ||||
|       this.executeBulkOperation(modal, "delete", {}) | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| @ -1,19 +1,27 @@ | ||||
| <div class="card mb-3 bg-light shadow-sm"> | ||||
| <div class="card mb-3 shadow-sm" [class.card-selected]="selected" [class.document-card]="selectable"> | ||||
|   <div class="row no-gutters"> | ||||
|     <div class="col-md-2 d-none d-lg-block"> | ||||
|       <img [src]="getThumbUrl()" class="card-img doc-img border-right"> | ||||
|     <div class="col-md-2 d-none d-lg-block doc-img-background rounded-left" [class.doc-img-background-selected]="selected"> | ||||
|       <img [src]="getThumbUrl()" class="card-img doc-img border-right rounded-left" (click)="setSelected(selectable ? !selected : false)"> | ||||
| 
 | ||||
|       <div style="top: 0; left: 0" class="position-absolute border-right border-bottom bg-light p-1" [class.document-card-check]="!selected"> | ||||
|         <div class="custom-control custom-checkbox"> | ||||
|           <input type="checkbox" class="custom-control-input" id="smallCardCheck{{document.id}}" [checked]="selected" (change)="setSelected($event.target.checked)"> | ||||
|           <label class="custom-control-label" for="smallCardCheck{{document.id}}"></label> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|     </div> | ||||
|     <div class="col"> | ||||
|       <div class="card-body"> | ||||
|       <div class="card-body bg-light"> | ||||
| 
 | ||||
|         <div class="d-flex justify-content-between align-items-center"> | ||||
|           <h5 class="card-title">     | ||||
|           <h5 class="card-title"> | ||||
|             <ng-container *ngIf="document.correspondent"> | ||||
|               <a *ngIf="clickCorrespondent.observers.length ; else nolink" [routerLink]="" title="Filter by correspondent" (click)="clickCorrespondent.emit(document.correspondent)" class="font-weight-bold">{{(document.correspondent$ | async)?.name}}</a> | ||||
|               <a *ngIf="clickCorrespondent.observers.length ; else nolink" [routerLink]="" title="Filter by correspondent" i18n-title (click)="clickCorrespondent.emit(document.correspondent)" class="font-weight-bold">{{(document.correspondent$ | async)?.name}}</a> | ||||
|               <ng-template #nolink>{{(document.correspondent$ | async)?.name}}</ng-template>: | ||||
|             </ng-container> | ||||
|             {{document.title}} | ||||
|             <app-tag [tag]="t" linkTitle="Filter by tag" *ngFor="let t of document.tags$ | async" class="ml-1" (click)="clickTag.emit(t.id)" [clickable]="clickTag.observers.length"></app-tag> | ||||
|             {{document.title | documentTitle}} | ||||
|             <app-tag [tag]="t" linkTitle="Filter by tag" i18n-linkTitle *ngFor="let t of document.tags$ | async" class="ml-1" (click)="clickTag.emit(t.id)" [clickable]="clickTag.observers.length"></app-tag> | ||||
|           </h5> | ||||
|           <h5 class="card-title" *ngIf="document.archive_serial_number">#{{document.archive_serial_number}}</h5> | ||||
|         </div> | ||||
| @ -23,33 +31,41 @@ | ||||
|         </p> | ||||
| 
 | ||||
| 
 | ||||
|         <div class="d-flex justify-content-between align-items-center"> | ||||
|         <div class="d-flex align-items-center"> | ||||
|           <div class="btn-group"> | ||||
|             <a routerLink="/search" [queryParams]="{'more_like': document.id}" class="btn btn-sm btn-outline-secondary" *ngIf="moreLikeThis"> | ||||
|               <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-three-dots" viewBox="0 0 16 16"> | ||||
|                 <path fill-rule="evenodd" d="M3 9.5a1.5 1.5 0 1 1 0-3 1.5 1.5 0 0 1 0 3zm5 0a1.5 1.5 0 1 1 0-3 1.5 1.5 0 0 1 0 3zm5 0a1.5 1.5 0 1 1 0-3 1.5 1.5 0 0 1 0 3z"/> | ||||
|               </svg> <ng-container i18n>More like this</ng-container> | ||||
|             </a> | ||||
|             <a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary"> | ||||
|               <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 | ||||
|               </svg> <ng-container i18n>Edit</ng-container> | ||||
|             </a> | ||||
|             <a type="button" class="btn btn-sm btn-outline-secondary" [href]="getPreviewUrl()"> | ||||
|               <svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-search" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|                 <path fill-rule="evenodd" d="M10.442 10.442a1 1 0 0 1 1.415 0l3.85 3.85a1 1 0 0 1-1.414 1.415l-3.85-3.85a1 1 0 0 1 0-1.415z"/> | ||||
|                 <path fill-rule="evenodd" d="M6.5 12a5.5 5.5 0 1 0 0-11 5.5 5.5 0 0 0 0 11zM13 6.5a6.5 6.5 0 1 1-13 0 6.5 6.5 0 0 1 13 0z"/> | ||||
|               </svg> | ||||
|               View | ||||
|               </svg> <ng-container i18n>View</ng-container> | ||||
|             </a> | ||||
|             <a type="button" class="btn btn-sm btn-outline-secondary" [href]="getDownloadUrl()"> | ||||
|               <svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-download" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|                 <path fill-rule="evenodd" d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/> | ||||
|                 <path fill-rule="evenodd" d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/> | ||||
|               </svg> | ||||
|               Download | ||||
|               </svg> <ng-container i18n>Download</ng-container> | ||||
|             </a> | ||||
| 
 | ||||
|           </div> | ||||
|           <small class="text-muted">Created: {{document.created | date}}</small> | ||||
| 
 | ||||
|           <small *ngIf="searchScore" class="text-muted ml-auto" i18n>Score:</small> | ||||
| 
 | ||||
|           <ngb-progressbar *ngIf="searchScore" [type]="searchScoreClass" [value]="searchScore" class="search-score-bar mx-2" [max]="1"></ngb-progressbar> | ||||
| 
 | ||||
|           <small class="text-muted" [class.ml-auto]="!searchScore" i18n>Created: {{document.created | date}}</small> | ||||
|         </div> | ||||
| 
 | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| </div> | ||||
|  | ||||
| @ -1,5 +1,7 @@ | ||||
| @import "/src/theme"; | ||||
| 
 | ||||
| .result-content { | ||||
|   color: darkgray; | ||||
|   overflow-wrap: anywhere; | ||||
| } | ||||
| 
 | ||||
| .doc-img { | ||||
| @ -7,5 +9,27 @@ | ||||
|   object-position: top; | ||||
|   height: 100%; | ||||
|   position: absolute; | ||||
|   mix-blend-mode: multiply; | ||||
| } | ||||
| 
 | ||||
| } | ||||
| .search-score-bar { | ||||
|   width: 100px; | ||||
|   height: 5px; | ||||
|   margin-top: 2px; | ||||
| } | ||||
| 
 | ||||
| .document-card-check { | ||||
|   display: none | ||||
| } | ||||
| 
 | ||||
| .document-card:hover .document-card-check { | ||||
|   display: block; | ||||
| } | ||||
| 
 | ||||
| .card-selected { | ||||
|   border-color: $primary; | ||||
| } | ||||
| 
 | ||||
| .doc-img-background-selected { | ||||
|   background-color: $primaryFaded; | ||||
| } | ||||
|  | ||||
| @ -1,7 +1,6 @@ | ||||
| import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; | ||||
| import { DomSanitizer } from '@angular/platform-browser'; | ||||
| import { PaperlessDocument } from 'src/app/data/paperless-document'; | ||||
| import { PaperlessTag } from 'src/app/data/paperless-tag'; | ||||
| import { DocumentService } from 'src/app/services/rest/document.service'; | ||||
| 
 | ||||
| @Component({ | ||||
| @ -13,6 +12,24 @@ export class DocumentCardLargeComponent implements OnInit { | ||||
| 
 | ||||
|   constructor(private documentService: DocumentService, private sanitizer: DomSanitizer) { } | ||||
| 
 | ||||
|   @Input() | ||||
|   selected = false | ||||
| 
 | ||||
|   setSelected(value: boolean) { | ||||
|     this.selected = value | ||||
|     this.selectedChange.emit(value) | ||||
|   } | ||||
| 
 | ||||
|   @Output() | ||||
|   selectedChange = new EventEmitter<boolean>() | ||||
| 
 | ||||
|   get selectable() { | ||||
|     return this.selectedChange.observers.length > 0 | ||||
|   } | ||||
| 
 | ||||
|   @Input() | ||||
|   moreLikeThis: boolean = false | ||||
| 
 | ||||
|   @Input() | ||||
|   document: PaperlessDocument | ||||
| 
 | ||||
| @ -25,6 +42,19 @@ export class DocumentCardLargeComponent implements OnInit { | ||||
|   @Output() | ||||
|   clickCorrespondent = new EventEmitter<number>() | ||||
| 
 | ||||
|   @Input() | ||||
|   searchScore: number | ||||
| 
 | ||||
|   get searchScoreClass() { | ||||
|     if (this.searchScore > 0.7) { | ||||
|       return "success" | ||||
|     } else if (this.searchScore > 0.3) { | ||||
|       return "warning" | ||||
|     } else { | ||||
|       return "danger" | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -1,44 +1,58 @@ | ||||
| <div class="col p-2 h-100" style="width: 16rem;"> | ||||
|   <div class="card h-100 shadow-sm"> | ||||
|     <div class=" border-bottom doc-img pr-1" [ngStyle]="{'background-image': 'url(' + getThumbUrl() + ')'}"> | ||||
|       <div class="row" *ngFor="let t of document.tags$ | async"> | ||||
|         <app-tag style="font-size: large;" [tag]="t" class="col text-right" (click)="clickTag.emit(t.id)" [clickable]="true" linkTitle="Filter by tag"></app-tag> | ||||
| <div class="col p-2 h-100"> | ||||
|   <div class="card h-100 shadow-sm document-card" [class.card-selected]="selected"> | ||||
|     <div class="border-bottom doc-img-container" [class.doc-img-background-selected]="selected"> | ||||
|       <img class="card-img doc-img rounded-top" [src]="getThumbUrl()" (click)="setSelected(!selected)"> | ||||
| 
 | ||||
|       <div class="border-right border-bottom bg-light p-1 rounded document-card-check"> | ||||
|         <div class="custom-control custom-checkbox"> | ||||
|           <input type="checkbox" class="custom-control-input" id="smallCardCheck{{document.id}}" [checked]="selected" (change)="setSelected($event.target.checked)"> | ||||
|           <label class="custom-control-label" for="smallCardCheck{{document.id}}"></label> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div style="top: 0; right: 0; font-size: large" class="text-right position-absolute mr-1"> | ||||
|         <div *ngFor="let t of getTagsLimited$() | async"> | ||||
|           <app-tag [tag]="t" (click)="clickTag.emit(t.id)" [clickable]="true" linkTitle="Filter by tag" i18n-linkTitle></app-tag> | ||||
|         </div> | ||||
|         <div *ngIf="moreTags"> | ||||
|           <span class="badge badge-secondary">+ {{moreTags}}</span> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|      | ||||
| 
 | ||||
|     <div class="card-body p-2"> | ||||
|       <p class="card-text"> | ||||
|         <ng-container *ngIf="document.correspondent"> | ||||
|           <a [routerLink]="" title="Filter by correspondent" (click)="clickCorrespondent.emit(document.correspondent)" class="font-weight-bold">{{(document.correspondent$ | async)?.name}}</a>: | ||||
|           <a [routerLink]="" title="Filter by correspondent" i18n-title (click)="clickCorrespondent.emit(document.correspondent)" class="font-weight-bold">{{(document.correspondent$ | async)?.name}}</a>: | ||||
|         </ng-container> | ||||
|         {{document.title}} | ||||
|         {{document.title | documentTitle}} | ||||
|       </p> | ||||
|     </div> | ||||
|     <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"> | ||||
|           <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" i18n-title> | ||||
|             <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> | ||||
|           </a> | ||||
|           <a [href]="getPreviewUrl()" class="btn btn-sm btn-outline-secondary" title="View in browser"> | ||||
|           <a [href]="getPreviewUrl()" class="btn btn-sm btn-outline-secondary" title="View in browser" i18n-title> | ||||
|             <svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-search" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|               <path fill-rule="evenodd" d="M10.442 10.442a1 1 0 0 1 1.415 0l3.85 3.85a1 1 0 0 1-1.414 1.415l-3.85-3.85a1 1 0 0 1 0-1.415z"/> | ||||
|               <path fill-rule="evenodd" d="M6.5 12a5.5 5.5 0 1 0 0-11 5.5 5.5 0 0 0 0 11zM13 6.5a6.5 6.5 0 1 1-13 0 6.5 6.5 0 0 1 13 0z"/> | ||||
|             </svg> | ||||
|           </a> | ||||
|           <a [href]="getDownloadUrl()" class="btn btn-sm btn-outline-secondary" title="Download"> | ||||
|           <a [href]="getDownloadUrl()" class="btn btn-sm btn-outline-secondary" title="Download" i18n-title> | ||||
|             <svg width="1em" height="1em" viewBox="0 0 16 16" class="bi bi-download" fill="currentColor" xmlns="http://www.w3.org/2000/svg"> | ||||
|               <path fill-rule="evenodd" d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/> | ||||
|               <path fill-rule="evenodd" d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/> | ||||
|             </svg> | ||||
|           </a> | ||||
|         </div> | ||||
|         <small class="text-muted">{{document.created | date}}</small> | ||||
|         <small class="text-muted pl-1">{{document.created | date}}</small> | ||||
|       </div> | ||||
|        | ||||
| 
 | ||||
|     </div> | ||||
|   </div>   | ||||
| </div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user