mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-25 07:49:06 -04:00 
			
		
		
		
	Merge pull request #7902 from paperless-ngx/beta
[Beta] Paperless-ngx v2.13.0 Beta Release
This commit is contained in:
		
						commit
						2814cd110d
					
				
							
								
								
									
										14
									
								
								.codecov.yml
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								.codecov.yml
									
									
									
									
									
								
							| @ -14,8 +14,9 @@ flag_management: | ||||
| # codecov will only comment if coverage changes | ||||
| comment: | ||||
|   require_changes: true | ||||
|   # https://docs.codecov.com/docs/javascript-bundle-analysis | ||||
|   require_bundle_changes: true | ||||
|   bundle_change_threshold: "1Kb" | ||||
|   bundle_change_threshold: "50Kb" | ||||
| coverage: | ||||
|   status: | ||||
|     project: | ||||
| @ -24,7 +25,12 @@ coverage: | ||||
|         threshold: 1% | ||||
|     patch: | ||||
|       default: | ||||
|         # For the changed lines only, target 75% covered, but | ||||
|         # allow as low as 50% | ||||
|         target: 75% | ||||
|         # For the changed lines only, target 100% covered, but | ||||
|         # allow as low as 75% | ||||
|         target: 100% | ||||
|         threshold: 25% | ||||
| # https://docs.codecov.com/docs/javascript-bundle-analysis | ||||
| bundle_analysis: | ||||
|   # Fail if the bundle size increases by more than 1MB | ||||
|   warning_threshold: "1MB" | ||||
|   status: true | ||||
|  | ||||
							
								
								
									
										8
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @ -16,9 +16,9 @@ on: | ||||
| env: | ||||
|   # This is the version of pipenv all the steps will use | ||||
|   # If changing this, change Dockerfile | ||||
|   DEFAULT_PIP_ENV_VERSION: "2024.0.1" | ||||
|   DEFAULT_PIP_ENV_VERSION: "2024.0.3" | ||||
|   # This is the default version of Python to use in most steps which aren't specific | ||||
|   DEFAULT_PYTHON_VERSION: "3.10" | ||||
|   DEFAULT_PYTHON_VERSION: "3.11" | ||||
| 
 | ||||
| jobs: | ||||
|   pre-commit: | ||||
| @ -100,7 +100,7 @@ jobs: | ||||
|       - pre-commit | ||||
|     strategy: | ||||
|       matrix: | ||||
|         python-version: ['3.9', '3.10', '3.11'] | ||||
|         python-version: ['3.10', '3.11', '3.12'] | ||||
|       fail-fast: false | ||||
|     steps: | ||||
|       - | ||||
| @ -486,7 +486,7 @@ jobs: | ||||
|         name: Patch whitenoise | ||||
|         run: | | ||||
|           curl --fail --silent --show-error --location --output 484.patch https://github.com/evansd/whitenoise/pull/484.patch | ||||
|           patch -d $(pipenv --venv)/lib/python3.10/site-packages --verbose -p2 < 484.patch | ||||
|           patch -d $(pipenv --venv)/lib/python3.11/site-packages --verbose -p2 < 484.patch | ||||
|           rm 484.patch | ||||
|       - | ||||
|         name: Install system dependencies | ||||
|  | ||||
							
								
								
									
										1
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							| @ -15,6 +15,7 @@ on: | ||||
| jobs: | ||||
|   synchronize-with-crowdin: | ||||
|     name: Crowdin Sync | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
| 
 | ||||
|     steps: | ||||
|  | ||||
							
								
								
									
										21
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							| @ -16,6 +16,7 @@ concurrency: | ||||
| jobs: | ||||
|   stale: | ||||
|     name: 'Stale' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v9 | ||||
| @ -31,6 +32,7 @@ jobs: | ||||
|             for your contributions. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|   lock-threads: | ||||
|     name: 'Lock Old Threads' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@v5 | ||||
| @ -56,6 +58,7 @@ jobs: | ||||
|             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|   close-answered-discussions: | ||||
|     name: 'Close Answered Discussions' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
| @ -112,6 +115,7 @@ jobs: | ||||
|             } | ||||
|   close-outdated-discussions: | ||||
|     name: 'Close Outdated Discussions' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
| @ -203,6 +207,7 @@ jobs: | ||||
|             } | ||||
|   close-unsupported-feature-requests: | ||||
|     name: 'Close Unsupported Feature Requests' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
| @ -212,15 +217,20 @@ jobs: | ||||
|               return new Promise(resolve => setTimeout(resolve, ms)); | ||||
|             } | ||||
| 
 | ||||
|             const CUTOFF_MAX_COUNT = 80; | ||||
|             const CUTOFF_1_DAYS = 180; | ||||
|             const CUTOFF_1_COUNT = 5; | ||||
|             const CUTOFF_2_DAYS = 365; | ||||
|             const CUTOFF_2_COUNT = 10; | ||||
|             const CUTOFF_2_COUNT = 20; | ||||
|             const CUTOFF_3_DAYS = 730; | ||||
|             const CUTOFF_3_COUNT = 40; | ||||
| 
 | ||||
|             const cutoff1Date = new Date(); | ||||
|             cutoff1Date.setDate(cutoff1Date.getDate() - CUTOFF_1_DAYS); | ||||
|             const cutoff2Date = new Date(); | ||||
|             cutoff2Date.setDate(cutoff2Date.getDate() - CUTOFF_2_DAYS); | ||||
|             const cutoff3Date = new Date(); | ||||
|             cutoff3Date.setDate(cutoff3Date.getDate() - CUTOFF_3_DAYS); | ||||
| 
 | ||||
|             const query = `query( | ||||
|                 $owner:String!, | ||||
| @ -250,9 +260,12 @@ jobs: | ||||
|             const result = await github.graphql(query, variables); | ||||
| 
 | ||||
|             for (const discussion of result.repository.discussions.nodes) { | ||||
|               const discussionDate = new Date(discussion.updatedAt); | ||||
|               if ((discussionDate < cutoff1Date && discussion.upvoteCount < CUTOFF_1_COUNT) || | ||||
|                   (discussionDate < cutoff2Date && discussion.upvoteCount < CUTOFF_2_COUNT)) { | ||||
|               const discussionUpdatedDate = new Date(discussion.updatedAt); | ||||
|               const discussionCreatedDate = new Date(discussion.createdAt); | ||||
|               if ((discussionUpdatedDate < cutoff1Date && discussion.upvoteCount < CUTOFF_MAX_COUNT) || | ||||
|                   (discussionCreatedDate < cutoff1Date && discussion.upvoteCount < CUTOFF_1_COUNT) || | ||||
|                   (discussionCreatedDate < cutoff2Date && discussion.upvoteCount < CUTOFF_2_COUNT) || | ||||
|                   (discussionCreatedDate < cutoff3Date && discussion.upvoteCount < CUTOFF_3_COUNT)) { | ||||
|                 console.log(`Closing discussion #${discussion.number} (${discussion.id}), last updated at ${discussion.updatedAt} with votes ${discussion.upvoteCount}`); | ||||
|                 const addCommentMutation = `mutation($discussion:ID!, $body:String!) { | ||||
|                   addDiscussionComment(input:{discussionId:$discussion, body:$body}) { | ||||
|  | ||||
| @ -48,7 +48,7 @@ repos: | ||||
|         exclude: "(^Pipfile\\.lock$)" | ||||
|   # Python hooks | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: 'v0.6.4' | ||||
|     rev: 'v0.6.8' | ||||
|     hooks: | ||||
|       - id: ruff | ||||
|       - id: ruff-format | ||||
| @ -62,6 +62,8 @@ repos: | ||||
|     rev: v6.2.1 | ||||
|     hooks: | ||||
|       - id: beautysh | ||||
|         additional_dependencies: | ||||
|           - setuptools | ||||
|         args: | ||||
|           - "--tab" | ||||
|   - repo: https://github.com/shellcheck-py/shellcheck-py | ||||
|  | ||||
| @ -1 +1 @@ | ||||
| 3.9.19 | ||||
| 3.10.15 | ||||
|  | ||||
| @ -2,7 +2,7 @@ fix = true | ||||
| line-length = 88 | ||||
| respect-gitignore = true | ||||
| src = ["src"] | ||||
| target-version = "py39" | ||||
| target-version = "py310" | ||||
| output-format = "grouped" | ||||
| show-fixes = true | ||||
| 
 | ||||
|  | ||||
| @ -11,7 +11,7 @@ If you want to implement something big: | ||||
| 
 | ||||
| ## Python | ||||
| 
 | ||||
| Paperless supports python 3.9 - 3.11 at this time. We format Python code with [ruff](https://docs.astral.sh/ruff/formatter/). | ||||
| Paperless supports python 3.10 - 3.12 at this time. We format Python code with [ruff](https://docs.astral.sh/ruff/formatter/). | ||||
| 
 | ||||
| ## Branches | ||||
| 
 | ||||
| @ -147,7 +147,7 @@ community members. That said, in an effort to keep the repository organized and | ||||
| - Issues, pull requests and discussions that are closed will be locked after 30 days of inactivity. | ||||
| - Discussions with a marked answer will be automatically closed. | ||||
| - Discussions in the 'General' or 'Support' categories will be closed after 180 days of inactivity. | ||||
| - Feature requests that do not meet the following thresholds will be closed: 5 "up-votes" after 180 days of inactivity or 10 "up-votes" after 365 days. | ||||
| - Feature requests that do not meet the following thresholds will be closed: 180 days of inactivity, < 5 "up-votes" after 180 days, < 20 "up-votes" after 1 year or < 80 "up-votes" at 2 years. | ||||
| 
 | ||||
| In all cases, threads can be re-opened by project maintainers and, of course, users can always create a new discussion for related concerns. | ||||
| Finally, remember that all information remains searchable and 'closed' feature requests can still serve as inspiration for new features. | ||||
|  | ||||
							
								
								
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -18,7 +18,7 @@ ARG PNGX_TAG_VERSION= | ||||
| # Add the tag to the environment file if its a tagged dev build | ||||
| RUN set -eux && \ | ||||
| case "${PNGX_TAG_VERSION}" in \ | ||||
|   dev|fix*|feature*) \ | ||||
|   dev|beta|fix*|feature*) \ | ||||
|     sed -i -E "s/version: '([0-9\.]+)'/version: '\1 #${PNGX_TAG_VERSION}'/g" /src/src-ui/src/environments/environment.prod.ts \ | ||||
|     ;; \ | ||||
| esac | ||||
| @ -31,7 +31,7 @@ RUN set -eux \ | ||||
| # Comments: | ||||
| #  - pipenv dependencies are not left in the final image | ||||
| #  - pipenv can't touch the final image somehow | ||||
| FROM --platform=$BUILDPLATFORM docker.io/python:3.11-alpine AS pipenv-base | ||||
| FROM --platform=$BUILDPLATFORM docker.io/python:3.12-alpine AS pipenv-base | ||||
| 
 | ||||
| WORKDIR /usr/src/pipenv | ||||
| 
 | ||||
| @ -39,7 +39,7 @@ COPY Pipfile* ./ | ||||
| 
 | ||||
| RUN set -eux \ | ||||
|   && echo "Installing pipenv" \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade pipenv==2024.0.1 \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade pipenv==2024.0.3 \ | ||||
|   && echo "Generating requirement.txt" \ | ||||
|     && pipenv requirements > requirements.txt | ||||
| 
 | ||||
| @ -47,7 +47,7 @@ RUN set -eux \ | ||||
| # Purpose: The final image | ||||
| # Comments: | ||||
| #  - Don't leave anything extra in here | ||||
| FROM docker.io/python:3.11-slim-bookworm AS main-app | ||||
| FROM docker.io/python:3.12-slim-bookworm AS main-app | ||||
| 
 | ||||
| LABEL org.opencontainers.image.authors="paperless-ngx team <hello@paperless-ngx.com>" | ||||
| LABEL org.opencontainers.image.documentation="https://docs.paperless-ngx.com/" | ||||
| @ -233,15 +233,15 @@ RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade wheel \ | ||||
|   && echo "Installing Python requirements" \ | ||||
|     && curl --fail --silent --show-error --location \ | ||||
|     --output psycopg_c-3.2.1-cp311-cp311-linux_x86_64.whl \ | ||||
|     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.1/psycopg_c-3.2.1-cp311-cp311-linux_x86_64.whl \ | ||||
|     --output psycopg_c-3.2.2-cp312-cp312-linux_x86_64.whl \ | ||||
|     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.2/psycopg_c-3.2.2-cp312-cp312-linux_x86_64.whl \ | ||||
|     && curl --fail --silent --show-error --location \ | ||||
|     --output psycopg_c-3.2.1-cp311-cp311-linux_aarch64.whl  \ | ||||
|     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.1/psycopg_c-3.2.1-cp311-cp311-linux_aarch64.whl \ | ||||
|     --output psycopg_c-3.2.2-cp312-cp312-linux_aarch64.whl  \ | ||||
|     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.2/psycopg_c-3.2.2-cp312-cp312-linux_aarch64.whl \ | ||||
|     && python3 -m pip install --default-timeout=1000 --find-links . --requirement requirements.txt \ | ||||
|   && echo "Patching whitenoise for compression speedup" \ | ||||
|     && curl --fail --silent --show-error --location --output 484.patch https://github.com/evansd/whitenoise/pull/484.patch \ | ||||
|     && patch -d /usr/local/lib/python3.11/site-packages --verbose -p2 < 484.patch \ | ||||
|     && patch -d /usr/local/lib/python3.12/site-packages --verbose -p2 < 484.patch \ | ||||
|     && rm 484.patch \ | ||||
|   && echo "Installing NLTK data" \ | ||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \ | ||||
|  | ||||
							
								
								
									
										6
									
								
								Pipfile
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								Pipfile
									
									
									
									
									
								
							| @ -7,7 +7,7 @@ name = "pypi" | ||||
| dateparser = "~=1.2" | ||||
| # WARNING: django does not use semver. | ||||
| #          Only patch versions are guaranteed to not introduce breaking changes. | ||||
| django = "~=4.2.15" | ||||
| django = "~=5.1.1" | ||||
| django-allauth = {extras = ["socialaccount"], version = "*"} | ||||
| django-auditlog = "*" | ||||
| django-celery-results = "*" | ||||
| @ -30,12 +30,14 @@ filelock = "*" | ||||
| flower = "*" | ||||
| gotenberg-client = "*" | ||||
| gunicorn = "*" | ||||
| httpx-oauth = "*" | ||||
| imap-tools = "*" | ||||
| inotifyrecursive = "~=0.3" | ||||
| jinja2 = "~=3.1" | ||||
| langdetect = "*" | ||||
| mysqlclient = "*" | ||||
| nltk = "*" | ||||
| ocrmypdf = "~=15.4" | ||||
| ocrmypdf = "~=16.5" | ||||
| pathvalidate = "*" | ||||
| pdf2image = "*" | ||||
| psycopg = {version = "*", extras = ["c"]} | ||||
|  | ||||
							
								
								
									
										1866
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1866
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -122,27 +122,38 @@ install_languages() { | ||||
| 	if [ ${#langs[@]} -eq 0 ]; then | ||||
| 		return | ||||
| 	fi | ||||
| 	apt-get update | ||||
| 
 | ||||
| 	# Build list of packages to install | ||||
| 	to_install=() | ||||
| 	for lang in "${langs[@]}"; do | ||||
| 		pkg="tesseract-ocr-$lang" | ||||
| 
 | ||||
| 		if dpkg --status "$pkg" &>/dev/null; then | ||||
| 			echo "Package $pkg already installed!" | ||||
| 			continue | ||||
| 		fi | ||||
| 
 | ||||
| 		if ! apt-cache show "$pkg" &>/dev/null; then | ||||
| 			echo "Package $pkg not found! :(" | ||||
| 			continue | ||||
| 		fi | ||||
| 
 | ||||
| 		echo "Installing package $pkg..." | ||||
| 		if ! apt-get --assume-yes install "$pkg" &>/dev/null; then | ||||
| 			echo "Could not install $pkg" | ||||
| 			exit 1 | ||||
| 		else | ||||
| 			to_install+=("$pkg") | ||||
| 		fi | ||||
| 	done | ||||
| 
 | ||||
| 	# Use apt only when we install packages | ||||
| 	if [ ${#to_install[@]} -gt 0 ]; then | ||||
| 		apt-get update | ||||
| 
 | ||||
| 		for pkg in "${to_install[@]}"; do | ||||
| 
 | ||||
| 			if ! apt-cache show "$pkg" &>/dev/null; then | ||||
| 				echo "Skipped $pkg: Package not found! :(" | ||||
| 				continue | ||||
| 			fi | ||||
| 
 | ||||
| 			echo "Installing package $pkg..." | ||||
| 			if ! apt-get --assume-yes install "$pkg" &>/dev/null; then | ||||
| 				echo "Could not install $pkg" | ||||
| 				exit 1 | ||||
| 			fi | ||||
| 		done | ||||
| 	fi | ||||
| } | ||||
| 
 | ||||
| echo "Paperless-ngx docker container starting..." | ||||
|  | ||||
| @ -265,7 +265,7 @@ This variable allows you to configure the filename (folders are allowed) | ||||
| using placeholders. For example, configuring this to | ||||
| 
 | ||||
| ```bash | ||||
| PAPERLESS_FILENAME_FORMAT={created_year}/{correspondent}/{title} | ||||
| PAPERLESS_FILENAME_FORMAT={{ created_year }}/{{ correspondent }}/{{ title }} | ||||
| ``` | ||||
| 
 | ||||
| will create a directory structure as follows: | ||||
| @ -298,39 +298,39 @@ will create a directory structure as follows: | ||||
|     when changing `PAPERLESS_FILENAME_FORMAT` you will need to manually run the | ||||
|     [`document renamer`](administration.md#renamer) to move any existing documents. | ||||
| 
 | ||||
| #### Placeholders | ||||
| ### Placeholders {#filename-format-variables} | ||||
| 
 | ||||
| Paperless provides the following placeholders within filenames: | ||||
| Paperless provides the following variables for use within filenames: | ||||
| 
 | ||||
| - `{asn}`: The archive serial number of the document, or "none". | ||||
| - `{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 | ||||
| - `{{ asn }}`: The archive serial number of the document, or "none". | ||||
| - `{{ 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 (ISO format) the document was created. | ||||
| - `{created_year}`: Year created only, formatted as the year with | ||||
| - `{{ title }}`: The title of the document. | ||||
| - `{{ created }}`: The full date (ISO format) the document was created. | ||||
| - `{{ created_year }}`: Year created only, formatted as the year with | ||||
|   century. | ||||
| - `{created_year_short}`: Year created only, formatted as the year | ||||
| - `{{ created_year_short }}`: Year created only, formatted as the year | ||||
|   without century, zero padded. | ||||
| - `{created_month}`: Month created only (number 01-12). | ||||
| - `{created_month_name}`: Month created name, as per locale | ||||
| - `{created_month_name_short}`: Month created abbreviated name, as per | ||||
| - `{{ created_month }}`: Month created only (number 01-12). | ||||
| - `{{ created_month_name }}`: Month created name, as per locale | ||||
| - `{{ created_month_name_short }}`: Month created abbreviated name, as per | ||||
|   locale | ||||
| - `{created_day}`: Day created only (number 01-31). | ||||
| - `{added}`: The full date (ISO format) the document was added to | ||||
| - `{{ created_day }}`: Day created only (number 01-31). | ||||
| - `{{ added }}`: The full date (ISO format) the document was added to | ||||
|   paperless. | ||||
| - `{added_year}`: Year added only. | ||||
| - `{added_year_short}`: Year added only, formatted as the year without | ||||
| - `{{ added_year }}`: Year added only. | ||||
| - `{{ added_year_short }}`: Year added only, formatted as the year without | ||||
|   century, zero padded. | ||||
| - `{added_month}`: Month added only (number 01-12). | ||||
| - `{added_month_name}`: Month added name, as per locale | ||||
| - `{added_month_name_short}`: Month added abbreviated name, as per | ||||
| - `{{ added_month }}`: Month added only (number 01-12). | ||||
| - `{{ added_month_name }}`: Month added name, as per locale | ||||
| - `{{ added_month_name_short }}`: Month added abbreviated name, as per | ||||
|   locale | ||||
| - `{added_day}`: Day added only (number 01-31). | ||||
| - `{owner_username}`: Username of document owner, if any, or "none" | ||||
| - `{original_name}`: Document original filename, minus the extension, if any, or "none" | ||||
| - `{doc_pk}`: The paperless identifier (primary key) for the document. | ||||
| - `{{ added_day }}`: Day added only (number 01-31). | ||||
| - `{{ owner_username }}`: Username of document owner, if any, or "none" | ||||
| - `{{ original_name }}`: Document original filename, minus the extension, if any, or "none" | ||||
| - `{{ doc_pk }}`: The paperless identifier (primary key) for the document. | ||||
| 
 | ||||
| !!! warning | ||||
| 
 | ||||
| @ -338,6 +338,11 @@ Paperless provides the following placeholders within filenames: | ||||
|     you may run into the limits of your operating system's maximum path lengths. | ||||
|     In that case, files will retain the previous path instead and the issue logged. | ||||
| 
 | ||||
| !!! tip | ||||
| 
 | ||||
|     These variables are all simple strings, but the format can be a full template. | ||||
|     See [Filename Templates](#filename-templates) for even more advanced formatting. | ||||
| 
 | ||||
| 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 | ||||
| @ -363,7 +368,7 @@ paperless will fall back to using the default naming scheme instead. | ||||
|     However, keep in mind that inside docker, if files get stored outside of | ||||
|     the predefined volumes, they will be lost after a restart. | ||||
| 
 | ||||
| ##### Empty placeholders | ||||
| #### Empty placeholders | ||||
| 
 | ||||
| You can affect how empty placeholders are treated by changing the | ||||
| [`PAPERLESS_FILENAME_FORMAT_REMOVE_NONE`](configuration.md#PAPERLESS_FILENAME_FORMAT_REMOVE_NONE) setting. | ||||
| @ -390,8 +395,8 @@ For example, you could define the following two storage paths: | ||||
|     the correspondence. | ||||
| 
 | ||||
| ``` | ||||
| By Year = {created_year}/{correspondent}/{title} | ||||
| Insurances = Insurances/{correspondent}/{created_year}-{created_month}-{created_day} {title} | ||||
| By Year = {{ created_year }}/{{ correspondent }}/{{ title }} | ||||
| Insurances = Insurances/{{ correspondent }}/{{ created_year }}-{{ created_month }}-{{ created_day }} {{ title }} | ||||
| ``` | ||||
| 
 | ||||
| If you then map these storage paths to the documents, you might get the | ||||
| @ -418,6 +423,97 @@ Insurances/                             # Insurances | ||||
|     Defining a storage path is optional. If no storage path is defined for a | ||||
|     document, the global [`PAPERLESS_FILENAME_FORMAT`](configuration.md#PAPERLESS_FILENAME_FORMAT) is applied. | ||||
| 
 | ||||
| ### Filename Templates {#filename-templates} | ||||
| 
 | ||||
| The filename formatting uses [Jinja templates](https://jinja.palletsprojects.com/en/3.1.x/templates/) to build the filename. | ||||
| This allows for complex logic to be included in the format, including [logical structures](https://jinja.palletsprojects.com/en/3.1.x/templates/#list-of-control-structures) | ||||
| and [filters](https://jinja.palletsprojects.com/en/3.1.x/templates/#id11) to manipulate the [variables](#filename-format-variables) | ||||
| provided. The template is provided as a string, potentially multiline, and rendered into a single line. | ||||
| 
 | ||||
| In addition, the entire Document instance is available to be utilized in a more advanced way, as well as some variables which only make sense to be accessed | ||||
| with more complex logic. | ||||
| 
 | ||||
| #### Additional Variables | ||||
| 
 | ||||
| - `{{ tag_name_list }}`: A list of tag names applied to the document, ordered by the tag name. Note this is a list, not a single string | ||||
| - `{{ custom_fields }}`: A mapping of custom field names to their type and value. A user can access the mapping by field name or check if a field is applied by checking its existence in the variable. | ||||
| 
 | ||||
| !!! tip | ||||
| 
 | ||||
|     To access a custom field which has a space in the name, use the `get_cf_value` filter.  See the examples below. | ||||
|     This helps get fields by name and handle a default value if the named field is not attached to a Document. | ||||
| 
 | ||||
| #### Examples | ||||
| 
 | ||||
| This example will construct a path based on the archive serial number range: | ||||
| 
 | ||||
| ```jinja | ||||
| somepath/ | ||||
| {% if document.archive_serial_number >= 0 and document.archive_serial_number <= 200 %} | ||||
|   asn-000-200/{{title}} | ||||
| {% elif document.archive_serial_number >= 201 and document.archive_serial_number <= 400 %} | ||||
|   asn-201-400 | ||||
|   {% if document.archive_serial_number >= 201 and document.archive_serial_number < 300 %} | ||||
|     /asn-2xx | ||||
|   {% elif document.archive_serial_number >= 300 and document.archive_serial_number < 400 %} | ||||
|     /asn-3xx | ||||
|   {% endif %} | ||||
| {% endif %} | ||||
| /{{ title }} | ||||
| ``` | ||||
| 
 | ||||
| For a document with an ASN of 205, it would result in `somepath/asn-201-400/asn-2xx/Title.pdf`, but | ||||
| a document with an ASN of 355 would be placed in `somepath/asn-201-400/asn-3xx/Title.pdf`. | ||||
| 
 | ||||
| ```jinja | ||||
| {% if document.mime_type == "application/pdf" %} | ||||
|   pdfs | ||||
| {% elif document.mime_type == "image/png" %} | ||||
|   pngs | ||||
| {% else %} | ||||
|   others | ||||
| {% endif %} | ||||
| /{{ title }} | ||||
| ``` | ||||
| 
 | ||||
| For a PDF document, it would result in `pdfs/Title.pdf`, but for a PNG document, the path would be `pngs/Title.pdf`. | ||||
| 
 | ||||
| To use custom fields: | ||||
| 
 | ||||
| ```jinja | ||||
| {% if "Invoice" in custom_fields %} | ||||
|   invoices/{{ custom_fields.Invoice.value }} | ||||
| {% else %} | ||||
|   not-invoices/{{ title }} | ||||
| {% endif %} | ||||
| ``` | ||||
| 
 | ||||
| If the document has a custom field named "Invoice" with a value of 123, it would be filed into the `invoices/123.pdf`, but a document without the custom field | ||||
| would be filed to `not-invoices/Title.pdf` | ||||
| 
 | ||||
| If the custom field is named "Invoice Number", you would access the value of it via the `get_cf_value` filter due to quirks of the Django Template Language: | ||||
| 
 | ||||
| ```jinja | ||||
| "invoices/{{ custom_fields|get_cf_value('Invoice Number') }}" | ||||
| ``` | ||||
| 
 | ||||
| You can also use a custom `datetime` filter to format dates: | ||||
| 
 | ||||
| ```jinja | ||||
| invoices/ | ||||
| {{ custom_fields|get_cf_value("Date Field","2024-01-01")|datetime('%Y') }}/ | ||||
| {{ custom_fields|get_cf_value("Date Field","2024-01-01")|datetime('%m') }}/ | ||||
| {{ custom_fields|get_cf_value("Date Field","2024-01-01")|datetime('%d') }}/ | ||||
| Invoice_{{ custom_fields|get_cf_value("Select Field") }}_{{ custom_fields|get_cf_value("Date Field","2024-01-01")|replace("-", "") }}.pdf | ||||
| ``` | ||||
| 
 | ||||
| This will create a path like `invoices/2022/01/01/Invoice_OptionTwo_20220101.pdf` if the custom field "Date Field" is set to January 1, 2022 and "Select Field" is set to `OptionTwo`. | ||||
| 
 | ||||
| ## Automatic recovery of invalid PDFs {#pdf-recovery} | ||||
| 
 | ||||
| Paperless will attempt to "clean" certain invalid PDFs with `qpdf` before processing if, for example, the mime_type | ||||
| detection is incorrect. This can happen if the PDF is not properly formatted or contains errors. | ||||
| 
 | ||||
| ## Celery Monitoring {#celery-monitoring} | ||||
| 
 | ||||
| The monitoring tool | ||||
|  | ||||
							
								
								
									
										52
									
								
								docs/api.md
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								docs/api.md
									
									
									
									
									
								
							| @ -54,6 +54,7 @@ fields: | ||||
| - `archived_file_name`: Verbose filename of the archived document. | ||||
|   Read-only. Null if no archived document is available. | ||||
| - `notes`: Array of notes associated with the document. | ||||
| - `page_count`: Number of pages. | ||||
| - `set_permissions`: Allows setting document permissions. Optional, | ||||
|   write-only. See [below](#permissions). | ||||
| - `custom_fields`: Array of custom fields & values, specified as | ||||
| @ -235,12 +236,6 @@ results: | ||||
| Pagination works exactly the same as it does for normal requests on this | ||||
| endpoint. | ||||
| 
 | ||||
| Certain limitations apply to full text queries: | ||||
| 
 | ||||
| - Results are always sorted by search score. The results matching the | ||||
|   query best will show up first. | ||||
| - Only a small subset of filtering parameters are supported. | ||||
| 
 | ||||
| Furthermore, each returned document has an additional `__search_hit__` | ||||
| attribute with various information about the search results: | ||||
| 
 | ||||
| @ -280,6 +275,51 @@ attribute with various information about the search results: | ||||
| - `rank` is the index of the search results. The first result will | ||||
|   have rank 0. | ||||
| 
 | ||||
| ### Filtering by custom fields | ||||
| 
 | ||||
| You can filter documents by their custom field values by specifying the | ||||
| `custom_field_query` query parameter. Here are some recipes for common | ||||
| use cases: | ||||
| 
 | ||||
| 1. Documents with a custom field "due" (date) between Aug 1, 2024 and | ||||
|    Sept 1, 2024 (inclusive): | ||||
| 
 | ||||
|    `?custom_field_query=["due", "range", ["2024-08-01", "2024-09-01"]]` | ||||
| 
 | ||||
| 2. Documents with a custom field "customer" (text) that equals "bob" | ||||
|    (case sensitive): | ||||
| 
 | ||||
|    `?custom_field_query=["customer", "exact", "bob"]` | ||||
| 
 | ||||
| 3. Documents with a custom field "answered" (boolean) set to `true`: | ||||
| 
 | ||||
|    `?custom_field_query=["answered", "exact", true]` | ||||
| 
 | ||||
| 4. Documents with a custom field "favorite animal" (select) set to either | ||||
|    "cat" or "dog": | ||||
| 
 | ||||
|    `?custom_field_query=["favorite animal", "in", ["cat", "dog"]]` | ||||
| 
 | ||||
| 5. Documents with a custom field "address" (text) that is empty: | ||||
| 
 | ||||
|    `?custom_field_query=["OR", ["address", "isnull", true], ["address", "exact", ""]]` | ||||
| 
 | ||||
| 6. Documents that don't have a field called "foo": | ||||
| 
 | ||||
|    `?custom_field_query=["foo", "exists", false]` | ||||
| 
 | ||||
| 7. Documents that have document links "references" to both document 3 and 7: | ||||
| 
 | ||||
|    `?custom_field_query=["references", "contains", [3, 7]]` | ||||
| 
 | ||||
| All field types support basic operations including `exact`, `in`, `isnull`, | ||||
| and `exists`. String, URL, and monetary fields support case-insensitive | ||||
| substring matching operations including `icontains`, `istartswith`, and | ||||
| `iendswith`. Integer, float, and date fields support arithmetic comparisons | ||||
| including `gt` (>), `gte` (>=), `lt` (<), `lte` (<=), and `range`. | ||||
| Lastly, document link fields support a `contains` operator that behaves | ||||
| like a "is superset of" check. | ||||
| 
 | ||||
| ### `/api/search/autocomplete/` | ||||
| 
 | ||||
| Get auto completions for a partial search term. | ||||
|  | ||||
| @ -608,9 +608,18 @@ You can optionally also automatically redirect users to the SSO login with [PAPE | ||||
| 
 | ||||
| #### [`PAPERLESS_ACCOUNT_SESSION_REMEMBER=<bool>`](#PAPERLESS_ACCOUNT_SESSION_REMEMBER) {#PAPERLESS_ACCOUNT_SESSION_REMEMBER} | ||||
| 
 | ||||
| : Only applies to regular (non-SSO) accounts. See the corresponding | ||||
| : If false, sessions will expire at browser close, if true will use `PAPERLESS_SESSION_COOKIE_AGE` for expiration. See the corresponding | ||||
| [django-allauth documentation](https://docs.allauth.org/en/latest/account/configuration.html) | ||||
| 
 | ||||
|     Defaults to True | ||||
| 
 | ||||
| #### [`PAPERLESS_SESSION_COOKIE_AGE=<int>`](#PAPERLESS_SESSION_COOKIE_AGE) {#PAPERLESS_SESSION_COOKIE_AGE} | ||||
| 
 | ||||
| : Login session cookie expiration. Applies if `PAPERLESS_ACCOUNT_SESSION_REMEMBER` is enabled. See the corresponding | ||||
| [django documentation](https://docs.djangoproject.com/en/5.1/ref/settings/#std-setting-SESSION_COOKIE_AGE) | ||||
| 
 | ||||
|     Defaults to 1209600 (2 weeks) | ||||
| 
 | ||||
| ## OCR settings {#ocr} | ||||
| 
 | ||||
| Paperless uses [OCRmyPDF](https://ocrmypdf.readthedocs.io/en/latest/) | ||||
| @ -1155,12 +1164,6 @@ within your documents. | ||||
| 
 | ||||
|     Defaults to false. | ||||
| 
 | ||||
| #### [`PAPERLESS_EMAIL_GNUPG_HOME=<str>`](#PAPERLESS_EMAIL_GNUPG_HOME) {#PAPERLESS_EMAIL_GNUPG_HOME} | ||||
| 
 | ||||
| : Optional, sets the `GNUPG_HOME` path to use with GPG decryptor for encrypted emails. See [GPG Decryptor](advanced_usage.md#gpg-decryptor) for more information. If not set, defaults to the default `GNUPG_HOME` path. | ||||
| 
 | ||||
|     Defaults to <not set>. | ||||
| 
 | ||||
| ### Polling {#polling} | ||||
| 
 | ||||
| #### [`PAPERLESS_CONSUMER_POLLING=<num>`](#PAPERLESS_CONSUMER_POLLING) {#PAPERLESS_CONSUMER_POLLING} | ||||
| @ -1204,6 +1207,48 @@ consumers working on the same file. Configure this to prevent that. | ||||
| 
 | ||||
|     Defaults to 0.5 seconds. | ||||
| 
 | ||||
| ## Incoming Mail {#incoming_mail} | ||||
| 
 | ||||
| ### Email OAuth {#email_oauth} | ||||
| 
 | ||||
| #### [`PAPERLESS_OAUTH_CALLBACK_BASE_URL=<str>`](#PAPERLESS_OAUTH_CALLBACK_BASE_URL) {#PAPERLESS_OAUTH_CALLBACK_BASE_URL} | ||||
| 
 | ||||
| : The base URL for the OAuth callback. This is used to construct the full URL for the OAuth callback. This should be the URL that the Paperless instance is accessible at. If not set, defaults to the `PAPERLESS_URL` setting. At least one of these settings must be set to enable OAuth Email setup. | ||||
| 
 | ||||
|     Defaults to none (thus will use [PAPERLESS_URL](#PAPERLESS_URL)). | ||||
| 
 | ||||
| #### [`PAPERLESS_GMAIL_OAUTH_CLIENT_ID=<str>`](#PAPERLESS_GMAIL_OAUTH_CLIENT_ID) {#PAPERLESS_GMAIL_OAUTH_CLIENT_ID} | ||||
| 
 | ||||
| : The OAuth client ID for Gmail. This is required for Gmail OAuth Email setup. See [OAuth Email Setup](usage.md#oauth-email-setup) for more information. | ||||
| 
 | ||||
|     Defaults to none. | ||||
| 
 | ||||
| #### [`PAPERLESS_GMAIL_OAUTH_CLIENT_SECRET=<str>`](#PAPERLESS_GMAIL_OAUTH_CLIENT_SECRET) {#PAPERLESS_GMAIL_OAUTH_CLIENT_SECRET} | ||||
| 
 | ||||
| : The OAuth client secret for Gmail. This is required for Gmail OAuth Email setup. See [OAuth Email Setup](usage.md#oauth-email-setup) for more information. | ||||
| 
 | ||||
|     Defaults to none. | ||||
| 
 | ||||
| #### [`PAPERLESS_OUTLOOK_OAUTH_CLIENT_ID=<str>`](#PAPERLESS_OUTLOOK_OAUTH_CLIENT_ID) {#PAPERLESS_OUTLOOK_OAUTH_CLIENT_ID} | ||||
| 
 | ||||
| : The OAuth client ID for Outlook. This is required for Outlook OAuth Email setup. See [OAuth Email Setup](usage.md#oauth-email-setup) for more information. | ||||
| 
 | ||||
|     Defaults to none. | ||||
| 
 | ||||
| #### [`PAPERLESS_OUTLOOK_OAUTH_CLIENT_SECRET=<str>`](#PAPERLESS_OUTLOOK_OAUTH_CLIENT_SECRET) {#PAPERLESS_OUTLOOK_OAUTH_CLIENT_SECRET} | ||||
| 
 | ||||
| : The OAuth client secret for Outlook. This is required for Outlook OAuth Email setup. See [OAuth Email Setup](usage.md#oauth-email-setup) for more information. | ||||
| 
 | ||||
|     Defaults to none. | ||||
| 
 | ||||
| ### Encrypted Emails {#encrypted_emails} | ||||
| 
 | ||||
| #### [`PAPERLESS_EMAIL_GNUPG_HOME=<str>`](#PAPERLESS_EMAIL_GNUPG_HOME) {#PAPERLESS_EMAIL_GNUPG_HOME} | ||||
| 
 | ||||
| : Optional, sets the `GNUPG_HOME` path to use with GPG decryptor for encrypted emails. See [GPG Decryptor](advanced_usage.md#gpg-decryptor) for more information. If not set, defaults to the default `GNUPG_HOME` path. | ||||
| 
 | ||||
|     Defaults to <not set>. | ||||
| 
 | ||||
| ## Barcodes {#barcodes} | ||||
| 
 | ||||
| #### [`PAPERLESS_CONSUMER_ENABLE_BARCODES=<bool>`](#PAPERLESS_CONSUMER_ENABLE_BARCODES) {#PAPERLESS_CONSUMER_ENABLE_BARCODES} | ||||
| @ -1242,6 +1287,12 @@ change this. | ||||
| 
 | ||||
|     Defaults to "PATCHT" | ||||
| 
 | ||||
| #### [`PAPERLESS_CONSUMER_BARCODE_RETAIN_SPLIT_PAGES=<bool>`](#PAPERLESS_CONSUMER_BARCODE_RETAIN_SPLIT_PAGES) {#PAPERLESS_CONSUMER_BARCODE_RETAIN_SPLIT_PAGES} | ||||
| 
 | ||||
| : If set to true, all pages that are split by a barcode (such as PATCHT) will be kept. | ||||
| 
 | ||||
|     Defaults to false. | ||||
| 
 | ||||
| #### [`PAPERLESS_CONSUMER_ENABLE_ASN_BARCODE=<bool>`](#PAPERLESS_CONSUMER_ENABLE_ASN_BARCODE) {#PAPERLESS_CONSUMER_ENABLE_ASN_BARCODE} | ||||
| 
 | ||||
| : Enables the detection of barcodes in the scanned document and | ||||
|  | ||||
| @ -360,10 +360,10 @@ If you want to build the documentation locally, this is how you do it: | ||||
| The docker image is primarily built by the GitHub actions workflow, but | ||||
| it can be faster when developing to build and tag an image locally. | ||||
| 
 | ||||
| Building the image works as with any image: | ||||
| Make sure you have the `docker-buildx` package installed. Building the image works as with any image: | ||||
| 
 | ||||
| ``` | ||||
| docker build --file Dockerfile --tag paperless:local --progress simple . | ||||
| docker build --file Dockerfile --tag paperless:local . | ||||
| ``` | ||||
| 
 | ||||
| ## Extending Paperless-ngx | ||||
|  | ||||
| @ -132,3 +132,11 @@ Multiple options for ASGI servers exist: | ||||
| - `daphne` as a standalone server, which is the reference | ||||
|   implementation for ASGI. | ||||
| - `uvicorn` as a standalone server | ||||
| 
 | ||||
| ## _What about the Redis licensing change and using one of the open source forks_? | ||||
| 
 | ||||
| Currently (October 2024), forks of Redis such as Valkey or Redirect are not officially supported by our upstream | ||||
| libraries, so using one of these to replace Redis is not officially supported. | ||||
| 
 | ||||
| However, they do claim to be compatible with the Redis protocol and will likely work, but we will | ||||
| not be updating from using Redis as the broker officially just yet. | ||||
|  | ||||
| @ -250,7 +250,7 @@ a minimal installation of Debian/Buster, which is the current stable | ||||
| release at the time of writing. Windows is not and will never be | ||||
| supported. | ||||
| 
 | ||||
| Paperless requires Python 3. At this time, 3.9 - 3.11 are tested versions. | ||||
| Paperless requires Python 3. At this time, 3.10 - 3.12 are tested versions. | ||||
| Newer versions may work, but some dependencies may not fully support newer versions. | ||||
| Support for older Python versions may be dropped as they reach end of life or as newer versions | ||||
| are released, dependency support is confirmed, etc. | ||||
|  | ||||
| @ -112,7 +112,7 @@ process. | ||||
| Please see [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Related-Projects) for a user-maintained list of related projects and | ||||
| software (e.g. for mobile devices) that is compatible with Paperless-ngx. | ||||
| 
 | ||||
| ### IMAP (Email) {#usage-email} | ||||
| ### Email {#usage-email} | ||||
| 
 | ||||
| You can tell paperless-ngx to consume documents from your email | ||||
| accounts. This is a very flexible and powerful feature, if you regularly | ||||
| @ -200,6 +200,14 @@ different means. These are as follows: | ||||
| Paperless is set up to check your mails every 10 minutes. This can be | ||||
| configured via [`PAPERLESS_EMAIL_TASK_CRON`](configuration.md#PAPERLESS_EMAIL_TASK_CRON) | ||||
| 
 | ||||
| #### OAuth Email Setup | ||||
| 
 | ||||
| Paperless-ngx supports OAuth2 authentication for Gmail and Outlook email accounts. To set up an email account with OAuth2, you will need to create a 'developer' app with the respective provider and obtain the client ID and client secret and set the appropriate [configuration variables](configuration.md#email_oauth). You will also need to set either [`PAPERLESS_OAUTH_CALLBACK_BASE_URL`](configuration.md#PAPERLESS_OAUTH_CALLBACK_BASE_URL) or [`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) to the correct value for the OAuth2 flow to work correctly. | ||||
| 
 | ||||
| Specific instructions for setting up the required 'developer' app with Google or Microsoft are beyond the scope of this documentation, but you can find user-maintained instructions in [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Email-OAuth-App-Setup) or by searching the web. | ||||
| 
 | ||||
| Once setup, navigating to the email settings page in Paperless-ngx will allow you to add an email account for Gmail or Outlook using OAuth2. After authenticating, you will be presented with the newly-created account where you will need to enter and save your email address. After this, the account will work as any other email account in Paperless-ngx and refreshing tokens will be handled automatically. | ||||
| 
 | ||||
| ### REST API | ||||
| 
 | ||||
| You can also submit a document using the REST API, see [POSTing documents](api.md#file-uploads) | ||||
|  | ||||
							
								
								
									
										1208
									
								
								src-ui/messages.xlf
									
									
									
									
									
								
							
							
						
						
									
										1208
									
								
								src-ui/messages.xlf
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										2044
									
								
								src-ui/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2044
									
								
								src-ui/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -11,23 +11,23 @@ | ||||
|   }, | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "@angular/cdk": "^18.2.2", | ||||
|     "@angular/common": "~18.2.2", | ||||
|     "@angular/compiler": "~18.2.2", | ||||
|     "@angular/core": "~18.2.2", | ||||
|     "@angular/forms": "~18.2.2", | ||||
|     "@angular/localize": "~18.2.2", | ||||
|     "@angular/platform-browser": "~18.2.2", | ||||
|     "@angular/platform-browser-dynamic": "~18.2.2", | ||||
|     "@angular/router": "~18.2.2", | ||||
|     "@angular/cdk": "^18.2.6", | ||||
|     "@angular/common": "~18.2.6", | ||||
|     "@angular/compiler": "~18.2.6", | ||||
|     "@angular/core": "~18.2.6", | ||||
|     "@angular/forms": "~18.2.6", | ||||
|     "@angular/localize": "~18.2.6", | ||||
|     "@angular/platform-browser": "~18.2.6", | ||||
|     "@angular/platform-browser-dynamic": "~18.2.6", | ||||
|     "@angular/router": "~18.2.6", | ||||
|     "@ng-bootstrap/ng-bootstrap": "^17.0.1", | ||||
|     "@ng-select/ng-select": "^13.7.0", | ||||
|     "@ng-select/ng-select": "^13.9.0", | ||||
|     "@ngneat/dirty-check-forms": "^3.0.3", | ||||
|     "@popperjs/core": "^2.11.8", | ||||
|     "bootstrap": "^5.3.3", | ||||
|     "file-saver": "^2.0.5", | ||||
|     "mime-names": "^1.0.0", | ||||
|     "ng2-pdf-viewer": "^10.3.0", | ||||
|     "ng2-pdf-viewer": "^10.3.1", | ||||
|     "ngx-bootstrap-icons": "^1.9.3", | ||||
|     "ngx-color": "^9.0.0", | ||||
|     "ngx-cookie-service": "^18.0.0", | ||||
| @ -42,26 +42,26 @@ | ||||
|     "@angular-builders/custom-webpack": "^18.0.0", | ||||
|     "@angular-builders/jest": "^18.0.0", | ||||
|     "@angular-devkit/build-angular": "^18.2.2", | ||||
|     "@angular-devkit/core": "^18.2.2", | ||||
|     "@angular-devkit/schematics": "^18.2.2", | ||||
|     "@angular-eslint/builder": "18.3.0", | ||||
|     "@angular-eslint/eslint-plugin": "18.3.0", | ||||
|     "@angular-eslint/eslint-plugin-template": "18.3.0", | ||||
|     "@angular-eslint/schematics": "18.3.0", | ||||
|     "@angular-eslint/template-parser": "18.3.0", | ||||
|     "@angular/cli": "~18.2.2", | ||||
|     "@angular-devkit/core": "^18.2.6", | ||||
|     "@angular-devkit/schematics": "^18.2.6", | ||||
|     "@angular-eslint/builder": "18.3.1", | ||||
|     "@angular-eslint/eslint-plugin": "18.3.1", | ||||
|     "@angular-eslint/eslint-plugin-template": "18.3.1", | ||||
|     "@angular-eslint/schematics": "18.3.1", | ||||
|     "@angular-eslint/template-parser": "18.3.1", | ||||
|     "@angular/cli": "~18.2.6", | ||||
|     "@angular/compiler-cli": "~18.2.2", | ||||
|     "@codecov/webpack-plugin": "^1.0.1", | ||||
|     "@playwright/test": "^1.46.1", | ||||
|     "@types/jest": "^29.5.12", | ||||
|     "@types/node": "^22.0.2", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.3.0", | ||||
|     "@typescript-eslint/parser": "^8.3.0", | ||||
|     "@codecov/webpack-plugin": "^1.2.0", | ||||
|     "@playwright/test": "^1.47.2", | ||||
|     "@types/jest": "^29.5.13", | ||||
|     "@types/node": "^22.7.4", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.8.0", | ||||
|     "@typescript-eslint/parser": "^8.8.0", | ||||
|     "@typescript-eslint/utils": "^8.0.0", | ||||
|     "eslint": "^9.9.1", | ||||
|     "eslint": "^9.11.1", | ||||
|     "jest": "29.7.0", | ||||
|     "jest-environment-jsdom": "^29.7.0", | ||||
|     "jest-preset-angular": "^14.2.2", | ||||
|     "jest-preset-angular": "^14.2.4", | ||||
|     "jest-websocket-mock": "^2.5.0", | ||||
|     "patch-package": "^8.0.0", | ||||
|     "ts-node": "~10.9.1", | ||||
|  | ||||
| @ -41,6 +41,7 @@ import { DocumentCardSmallComponent } from './components/document-list/document- | ||||
| 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 { TextAreaComponent } from './components/common/input/textarea/textarea.component' | ||||
| import { SelectComponent } from './components/common/input/select/select.component' | ||||
| import { CheckComponent } from './components/common/input/check/check.component' | ||||
| import { UrlComponent } from './components/common/input/url/url.component' | ||||
| @ -108,6 +109,7 @@ import { FileDropComponent } from './components/file-drop/file-drop.component' | ||||
| import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component' | ||||
| import { CustomFieldEditDialogComponent } from './components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component' | ||||
| import { CustomFieldsDropdownComponent } from './components/common/custom-fields-dropdown/custom-fields-dropdown.component' | ||||
| import { CustomFieldsQueryDropdownComponent } from './components/common/custom-fields-query-dropdown/custom-fields-query-dropdown.component' | ||||
| import { ProfileEditDialogComponent } from './components/common/profile-edit-dialog/profile-edit-dialog.component' | ||||
| import { PdfViewerModule } from 'ng2-pdf-viewer' | ||||
| import { DocumentLinkComponent } from './components/common/input/document-link/document-link.component' | ||||
| @ -141,7 +143,9 @@ import { | ||||
|   arrowRightShort, | ||||
|   arrowUpRight, | ||||
|   asterisk, | ||||
|   braces, | ||||
|   bodyText, | ||||
|   boxArrowInRight, | ||||
|   boxArrowUp, | ||||
|   boxArrowUpRight, | ||||
|   boxes, | ||||
| @ -172,6 +176,7 @@ import { | ||||
|   download, | ||||
|   envelope, | ||||
|   envelopeAt, | ||||
|   envelopeAtFill, | ||||
|   exclamationCircleFill, | ||||
|   exclamationTriangle, | ||||
|   exclamationTriangleFill, | ||||
| @ -188,6 +193,7 @@ import { | ||||
|   folderFill, | ||||
|   funnel, | ||||
|   gear, | ||||
|   google, | ||||
|   grid, | ||||
|   gripVertical, | ||||
|   hash, | ||||
| @ -198,6 +204,8 @@ import { | ||||
|   link, | ||||
|   listTask, | ||||
|   listUl, | ||||
|   microsoft, | ||||
|   nodePlus, | ||||
|   pencil, | ||||
|   people, | ||||
|   peopleFill, | ||||
| @ -227,6 +235,7 @@ import { | ||||
|   uiRadios, | ||||
|   upcScan, | ||||
|   x, | ||||
|   xCircle, | ||||
|   xLg, | ||||
| } from 'ngx-bootstrap-icons' | ||||
| 
 | ||||
| @ -242,7 +251,9 @@ const icons = { | ||||
|   arrowRightShort, | ||||
|   arrowUpRight, | ||||
|   asterisk, | ||||
|   braces, | ||||
|   bodyText, | ||||
|   boxArrowInRight, | ||||
|   boxArrowUp, | ||||
|   boxArrowUpRight, | ||||
|   boxes, | ||||
| @ -273,6 +284,7 @@ const icons = { | ||||
|   download, | ||||
|   envelope, | ||||
|   envelopeAt, | ||||
|   envelopeAtFill, | ||||
|   exclamationCircleFill, | ||||
|   exclamationTriangle, | ||||
|   exclamationTriangleFill, | ||||
| @ -289,6 +301,7 @@ const icons = { | ||||
|   folderFill, | ||||
|   funnel, | ||||
|   gear, | ||||
|   google, | ||||
|   grid, | ||||
|   gripVertical, | ||||
|   hash, | ||||
| @ -299,6 +312,8 @@ const icons = { | ||||
|   link, | ||||
|   listTask, | ||||
|   listUl, | ||||
|   microsoft, | ||||
|   nodePlus, | ||||
|   pencil, | ||||
|   people, | ||||
|   peopleFill, | ||||
| @ -328,6 +343,7 @@ const icons = { | ||||
|   uiRadios, | ||||
|   upcScan, | ||||
|   x, | ||||
|   xCircle, | ||||
|   xLg, | ||||
| } | ||||
| 
 | ||||
| @ -433,6 +449,7 @@ function initializeApp(settings: SettingsService) { | ||||
|     DocumentCardSmallComponent, | ||||
|     BulkEditorComponent, | ||||
|     TextComponent, | ||||
|     TextAreaComponent, | ||||
|     SelectComponent, | ||||
|     CheckComponent, | ||||
|     UrlComponent, | ||||
| @ -485,6 +502,7 @@ function initializeApp(settings: SettingsService) { | ||||
|     CustomFieldsComponent, | ||||
|     CustomFieldEditDialogComponent, | ||||
|     CustomFieldsDropdownComponent, | ||||
|     CustomFieldsQueryDropdownComponent, | ||||
|     ProfileEditDialogComponent, | ||||
|     DocumentLinkComponent, | ||||
|     PreviewPopupComponent, | ||||
|  | ||||
| @ -43,7 +43,7 @@ | ||||
|           <div class="dropdown-divider"></div> | ||||
|         </div> | ||||
|         <button ngbDropdownItem class="nav-link" (click)="editProfile()"> | ||||
|           <i-bs class="me-2" name="person"></i-bs> <ng-container i18n>My Profile</ng-container> | ||||
|           <i-bs class="me-2" name="person"></i-bs><ng-container i18n>My Profile</ng-container> | ||||
|         </button> | ||||
|         <a ngbDropdownItem class="nav-link" routerLink="settings" (click)="closeMenu()" | ||||
|           *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.UISettings }"> | ||||
|  | ||||
| @ -12,6 +12,9 @@ | ||||
|   z-index: 995; /* Behind the navbar */ | ||||
|   padding: 50px 0 0; /* Height of navbar */ | ||||
|   box-shadow: inset -1px 0 0 rgba(0, 0, 0, .1); | ||||
|   overflow-y: auto; | ||||
|   --pngx-sidebar-width: 100%; | ||||
|   max-width: var(--pngx-sidebar-width); | ||||
| 
 | ||||
|   .sidebar-heading .spinner-border { | ||||
|     width: 0.8em; | ||||
| @ -24,15 +27,15 @@ | ||||
| 
 | ||||
|   // These come from the col-* classes for non-slim sidebar, needed for animation | ||||
|   @media (min-width: 768px) { | ||||
|     max-width: 25%; | ||||
|     --pngx-sidebar-width: 25%; | ||||
|   } | ||||
| 
 | ||||
|   @media (min-width: 992px) { | ||||
|     max-width: 16.66666667%; | ||||
|     --pngx-sidebar-width: 16.66666667%; | ||||
|   } | ||||
| 
 | ||||
|   @media (min-width: 2400px) { | ||||
|     max-width: 8.33333333%; | ||||
|     --pngx-sidebar-width: 8.33333333%; | ||||
|   } | ||||
| 
 | ||||
|   transition: all .2s ease; | ||||
| @ -109,12 +112,17 @@ main { | ||||
| 
 | ||||
|   .sidebar-slim-toggler { | ||||
|     display: block; | ||||
|     position: absolute; | ||||
|     right: -12px; | ||||
|     position: fixed; | ||||
|     left: calc(var(--pngx-sidebar-width) - 12px); | ||||
|     top: 60px; | ||||
|     z-index: 996; | ||||
|     --bs-btn-padding-x: 0.35rem; | ||||
|     --bs-btn-padding-y: 0.125rem; | ||||
|     transition: all .2s ease; | ||||
|   } | ||||
| 
 | ||||
|   .sidebar.slim .sidebar-slim-toggler { | ||||
|     --pngx-sidebar-width: 50px !important; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -49,14 +49,14 @@ | ||||
|                 [disabled]="disablePrimaryButton(type, item)" | ||||
|                 (mouseenter)="onButtonHover($event)"> | ||||
|                     @if (type === DataType.Document) { | ||||
|                         <i-bs width="1em" height="1em" name="pencil"></i-bs> | ||||
|                         <i-bs width="1em" height="1em" name="box-arrow-in-right"></i-bs> | ||||
|                         <span> <ng-container i18n>Open</ng-container></span> | ||||
|                     } @else if (type === DataType.SavedView) { | ||||
|                         <i-bs width="1em" height="1em" name="eye"></i-bs> | ||||
|                         <span> <ng-container i18n>Open</ng-container></span> | ||||
|                     } @else if (type === DataType.Workflow || type === DataType.CustomField || type === DataType.Group || type === DataType.User || type === DataType.MailAccount || type === DataType.MailRule) { | ||||
|                         <i-bs width="1em" height="1em" name="pencil"></i-bs> | ||||
|                         <span> <ng-container i18n>Edit</ng-container></span> | ||||
|                         <span> <ng-container i18n>Open</ng-container></span> | ||||
|                     } @else { | ||||
|                         <i-bs width="1em" height="1em" name="filter"></i-bs> | ||||
|                         <span> <ng-container i18n>Filter documents</ng-container></span> | ||||
| @ -72,8 +72,8 @@ | ||||
|                             <i-bs width="1em" height="1em" name="download"></i-bs> | ||||
|                             <span> <ng-container i18n>Download</ng-container></span> | ||||
|                         } @else { | ||||
|                             <i-bs width="1em" height="1em" name="pencil"></i-bs> | ||||
|                             <span> <ng-container i18n>Edit</ng-container></span> | ||||
|                             <i-bs width="1em" height="1em" name="box-arrow-in-right"></i-bs> | ||||
|                             <span> <ng-container i18n>Open</ng-container></span> | ||||
|                         } | ||||
|                     </button> | ||||
|                 } | ||||
|  | ||||
| @ -65,10 +65,6 @@ form { | ||||
|   --pngx-focus-alpha: 0; | ||||
| } | ||||
| 
 | ||||
| .cursor-pointer { | ||||
|   cursor: pointer; | ||||
| } | ||||
| 
 | ||||
| .mh-75 { | ||||
|   max-height: 75vh; | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,163 @@ | ||||
| <div class="btn-group w-100" role="group" ngbDropdown #dropdown="ngbDropdown" (openChange)="onOpenChange($event)" [popperOptions]="popperOptions"> | ||||
|   <button class="btn btn-sm btn-outline-primary" id="dropdown_toggle" ngbDropdownToggle [disabled]="disabled"> | ||||
|     <i-bs name="{{icon}}"></i-bs> | ||||
|     <div class="d-none d-sm-inline"> {{title}}</div> | ||||
|     @if (isActive) { | ||||
|       <pngx-clearable-badge [selected]="isActive" (cleared)="reset()"></pngx-clearable-badge> | ||||
|     } | ||||
|   </button> | ||||
|   <div class="px-3 shadow" ngbDropdownMenu attr.aria-labelledby="dropdown_{{name}}"> | ||||
|     <div class="list-group list-group-flush"> | ||||
|       @for (element of selectionModel.queries; track element.id; let i = $index) { | ||||
|         <div class="list-group-item px-0 d-flex flex-nowrap"> | ||||
|           @switch (element.type) { | ||||
|             @case (CustomFieldQueryComponentType.Atom) { | ||||
|               <ng-container *ngTemplateOutlet="queryAtom; context: { atom: element }"></ng-container> | ||||
|             } | ||||
|             @case (CustomFieldQueryComponentType.Expression) { | ||||
|               <ng-container *ngTemplateOutlet="queryExpression; context: { expression: element }"></ng-container> | ||||
|             } | ||||
|           } | ||||
|         </div> | ||||
|       } | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| 
 | ||||
| <ng-template #comparisonValueTemplate let-atom="atom"> | ||||
|   @if (getCustomFieldByID(atom.field)?.data_type === CustomFieldDataType.Date) { | ||||
|     <input class="form-control" placeholder="yyyy-mm-dd" | ||||
|       [(ngModel)]="atom.value" | ||||
|       ngbDatepicker | ||||
|       #d="ngbDatepicker" /> | ||||
|     <button class="btn btn-sm btn-outline-secondary rounded-end" (click)="d.toggle()" type="button"> | ||||
|       <i-bs name="calendar-event"></i-bs> | ||||
|     </button> | ||||
|   } @else if (getCustomFieldByID(atom.field)?.data_type === CustomFieldDataType.Float || getCustomFieldByID(atom.field)?.data_type === CustomFieldDataType.Integer) { | ||||
|     <input class="w-25 form-control rounded-end" type="number" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|   } @else if (getCustomFieldByID(atom.field)?.data_type === CustomFieldDataType.Boolean) { | ||||
|     <select class="w-25 form-select rounded-end" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|       <option value="true" i18n>True</option> | ||||
|       <option value="false" i18n>False</option> | ||||
|     </select> | ||||
|   } @else if (getCustomFieldByID(atom.field)?.data_type === CustomFieldDataType.Select) { | ||||
|     <ng-select #fieldSelects | ||||
|       class="paperless-input-select rounded-end" | ||||
|       [items]="getSelectOptionsForField(atom.field)" | ||||
|       [(ngModel)]="atom.value" | ||||
|       [disabled]="disabled" | ||||
|       (mousedown)="$event.stopImmediatePropagation()" | ||||
|     ></ng-select> | ||||
|   } @else { | ||||
|     <input class="w-25 form-control rounded-end" type="text" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|   } | ||||
| </ng-template> | ||||
| 
 | ||||
| <ng-template #queryAtom let-atom="atom"> | ||||
|   <div class="input-group input-group-sm"> | ||||
|     <ng-select | ||||
|       class="paperless-input-select" | ||||
|       [items]="customFields" | ||||
|       [(ngModel)]="atom.field" | ||||
|       [disabled]="disabled" | ||||
|       bindLabel="name" | ||||
|       bindValue="id" | ||||
|       (mousedown)="$event.stopImmediatePropagation()" | ||||
|     ></ng-select> | ||||
|     <select class="w-25 form-select" [(ngModel)]="atom.operator" [disabled]="disabled"> | ||||
|       <option *ngFor="let operator of getOperatorsForField(atom.field)" [ngValue]="operator.value">{{operator.label}}</option> | ||||
|     </select> | ||||
|     @switch (atom.operator) { | ||||
|       @case (CustomFieldQueryOperator.Exists) { | ||||
|         <select class="w-25 form-select rounded-end" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|           <option value="true" i18n>True</option> | ||||
|           <option value="false" i18n>False</option> | ||||
|         </select> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.IsNull) { | ||||
|         <select class="w-25 form-select rounded-end" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|           <option value="true" i18n>True</option> | ||||
|           <option value="false" i18n>False</option> | ||||
|         </select> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.GreaterThanOrEqual) { | ||||
|         <ng-container *ngTemplateOutlet="comparisonValueTemplate; context: { atom: atom }"></ng-container> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.LessThanOrEqual) { | ||||
|         <ng-container *ngTemplateOutlet="comparisonValueTemplate; context: { atom: atom }"></ng-container> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.GreaterThan) { | ||||
|         <ng-container *ngTemplateOutlet="comparisonValueTemplate; context: { atom: atom }"></ng-container> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.LessThan) { | ||||
|         <ng-container *ngTemplateOutlet="comparisonValueTemplate; context: { atom: atom }"></ng-container> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.Contains) { | ||||
|         <pngx-input-document-link [(ngModel)]="atom.value" class="w-25 form-select doc-link-select p-0" placeholder="Search docs..." i18n-placeholder [minimal]="true"></pngx-input-document-link> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.In) { | ||||
|         <ng-select | ||||
|           class="paperless-input-select rounded-end" | ||||
|           [items]="getSelectOptionsForField(atom.field)" | ||||
|           [(ngModel)]="atom.value" | ||||
|           [disabled]="disabled" | ||||
|           [multiple]="true" | ||||
|           (mousedown)="$event.stopImmediatePropagation()" | ||||
|         ></ng-select> | ||||
|       } | ||||
|       @case (CustomFieldQueryOperator.Exact) { | ||||
|         <ng-container *ngTemplateOutlet="comparisonValueTemplate; context: { atom: atom }"></ng-container> | ||||
|       } | ||||
|       @default { | ||||
|         <input class="w-25 form-control rounded-end" type="text" [(ngModel)]="atom.value" [disabled]="disabled"> | ||||
|       } | ||||
|     } | ||||
|     <button class="btn btn-link btn-sm text-danger pe-0" type="button" (click)="removeElement(atom)" [disabled]="disabled"> | ||||
|       <i-bs name="x-circle"></i-bs> | ||||
|     </button> | ||||
|   </div> | ||||
| </ng-template> | ||||
| 
 | ||||
| <ng-template #queryExpression let-expression="expression"> | ||||
|   <div class="d-flex w-100"> | ||||
|     <div class="d-flex flex-grow-1 flex-column"> | ||||
|       <div class="btn-group btn-group-xs" role="group"> | ||||
|         <input [(ngModel)]="expression.operator" type="radio" class="btn-check" id="logicalOperatorOr_{{expression.id}}" name="logicalOperatorOr_{{expression.id}}" value="OR" [disabled]="expression.depth > 0 && expression.value.length < 2"> | ||||
|         <label class="btn btn-outline-primary" for="logicalOperatorOr_{{expression.id}}" i18n>Any</label> | ||||
|         <input [(ngModel)]="expression.operator" type="radio" class="btn-check" id="logicalOperatorAnd_{{expression.id}}" name="logicalOperatorAnd_{{expression.id}}" value="AND" [disabled]="expression.depth > 0 && expression.value.length < 2"> | ||||
|         <label class="btn btn-outline-primary" for="logicalOperatorAnd_{{expression.id}}" i18n>All</label> | ||||
|         @if (expression.negatable)  { | ||||
|           <input [(ngModel)]="expression.operator" type="radio" class="btn-check" id="logicalOperatorNot_{{expression.id}}" name="logicalOperatorNot_{{expression.id}}" value="NOT"> | ||||
|           <label class="btn btn-outline-secondary" for="logicalOperatorNot_{{expression.id}}" i18n>Not</label> | ||||
|         } | ||||
|       </div> | ||||
|       <div class="list-group list-group-flush mb-n2"> | ||||
|         @for (element of expression.value; track element.id; let i = $index) { | ||||
|           <div class="list-group-item px-0 d-flex flex-nowrap"> | ||||
|             @switch (element.type) { | ||||
|               @case (CustomFieldQueryComponentType.Atom) { | ||||
|                 <ng-container *ngTemplateOutlet="queryAtom; context: { atom: element }"></ng-container> | ||||
|               } | ||||
|               @case (CustomFieldQueryComponentType.Expression) { | ||||
|                 <ng-container *ngTemplateOutlet="queryExpression; context: { expression: element }"></ng-container> | ||||
|               } | ||||
|             } | ||||
|           </div> | ||||
|         } | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="btn-group-vertical ms-2 ps-2 border-start" role="group" aria-label="Vertical button group"> | ||||
|       <button type="button" class="btn btn-sm btn-outline-secondary text-primary" title="Add query" i18n-title (click)="addAtom(expression)" [disabled]="disabled || expression.value.length === CUSTOM_FIELD_QUERY_MAX_ATOMS"> | ||||
|         <i-bs name="node-plus"></i-bs> | ||||
|       </button> | ||||
|       <button type="button" class="btn btn-sm btn-outline-secondary text-primary" title="Add expression" i18n-title (click)="addExpression(expression)" [disabled]="disabled || expression.depth === CUSTOM_FIELD_QUERY_MAX_DEPTH"> | ||||
|         <i-bs name="braces"></i-bs> | ||||
|       </button> | ||||
|       @if (expression.depth > 0) { | ||||
|         <button type="button" class="btn btn-sm btn-outline-secondary text-danger" (click)="removeElement(expression)" [disabled]="disabled"> | ||||
|           <i-bs name="x-circle"></i-bs> | ||||
|         </button> | ||||
|       } | ||||
|     </div> | ||||
|   </div> | ||||
| </ng-template> | ||||
| @ -0,0 +1,43 @@ | ||||
| .dropdown-menu { | ||||
|   width: 370px; | ||||
|   @media(min-width: 768px) { | ||||
|     width: 600px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| ::ng-deep .ng-select-container { | ||||
|   border-top-right-radius: 0 !important; | ||||
|   border-bottom-right-radius: 0 !important; | ||||
|   height: 100% !important; | ||||
| } | ||||
| 
 | ||||
| ::ng-deep .rounded-end .ng-select-container { | ||||
|   border-top-right-radius: var(--bs-border-radius) !important; | ||||
|   border-bottom-right-radius: var(--bs-border-radius) !important; | ||||
|   border-top-left-radius: 0 !important; | ||||
|   border-bottom-left-radius: 0 !important; | ||||
| } | ||||
| 
 | ||||
| ::ng-deep .ng-select { | ||||
|   max-width: 100px; | ||||
|   min-width: 35%; | ||||
|   font-size: 14px; | ||||
| } | ||||
| 
 | ||||
| ::ng-deep .doc-link-select { | ||||
|   padding-top: 0 !important; | ||||
|   border-top-right-radius: var(--bs-border-radius) !important; | ||||
|   border-bottom-right-radius: var(--bs-border-radius) !important; | ||||
|   background-image: none !important; | ||||
| 
 | ||||
|   .ng-select-container, | ||||
|   .ng-select.ng-select-opened > .ng-select-container { | ||||
|     border: none !important; | ||||
|     min-height: 34px !important; | ||||
|     background: none !important; | ||||
|   } | ||||
|   .ng-select { | ||||
|     max-width: 200px; | ||||
|     min-width: 140px; | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,342 @@ | ||||
| import { | ||||
|   ComponentFixture, | ||||
|   fakeAsync, | ||||
|   TestBed, | ||||
|   tick, | ||||
| } from '@angular/core/testing' | ||||
| import { | ||||
|   CustomFieldQueriesModel, | ||||
|   CustomFieldsQueryDropdownComponent, | ||||
| } from './custom-fields-query-dropdown.component' | ||||
| import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' | ||||
| import { of } from 'rxjs' | ||||
| import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field' | ||||
| import { | ||||
|   CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP, | ||||
|   CustomFieldQueryLogicalOperator, | ||||
|   CustomFieldQueryOperatorGroups, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| import { provideHttpClientTesting } from '@angular/common/http/testing' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { | ||||
|   CustomFieldQueryExpression, | ||||
|   CustomFieldQueryAtom, | ||||
|   CustomFieldQueryElement, | ||||
| } from 'src/app/utils/custom-field-query-element' | ||||
| import { NgSelectModule } from '@ng-select/ng-select' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| 
 | ||||
| const customFields = [ | ||||
|   { | ||||
|     id: 1, | ||||
|     name: 'Test Field', | ||||
|     data_type: CustomFieldDataType.String, | ||||
|     extra_data: {}, | ||||
|   }, | ||||
|   { | ||||
|     id: 2, | ||||
|     name: 'Test Select Field', | ||||
|     data_type: CustomFieldDataType.Select, | ||||
|     extra_data: { select_options: ['Option 1', 'Option 2'] }, | ||||
|   }, | ||||
| ] | ||||
| 
 | ||||
| describe('CustomFieldsQueryDropdownComponent', () => { | ||||
|   let component: CustomFieldsQueryDropdownComponent | ||||
|   let fixture: ComponentFixture<CustomFieldsQueryDropdownComponent> | ||||
|   let customFieldsService: CustomFieldsService | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [CustomFieldsQueryDropdownComponent], | ||||
|       imports: [ | ||||
|         NgbDropdownModule, | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|         NgSelectModule, | ||||
|         FormsModule, | ||||
|         ReactiveFormsModule, | ||||
|       ], | ||||
|       providers: [ | ||||
|         provideHttpClient(withInterceptorsFromDi()), | ||||
|         provideHttpClientTesting(), | ||||
|       ], | ||||
|     }).compileComponents() | ||||
| 
 | ||||
|     customFieldsService = TestBed.inject(CustomFieldsService) | ||||
|     jest.spyOn(customFieldsService, 'listAll').mockReturnValue( | ||||
|       of({ | ||||
|         count: customFields.length, | ||||
|         all: customFields.map((f) => f.id), | ||||
|         results: customFields, | ||||
|       }) | ||||
|     ) | ||||
|     fixture = TestBed.createComponent(CustomFieldsQueryDropdownComponent) | ||||
|     component = fixture.componentInstance | ||||
|     component.icon = 'ui-radios' | ||||
|     fixture.detectChanges() | ||||
|   }) | ||||
| 
 | ||||
|   it('should initialize custom fields on creation', () => { | ||||
|     expect(component.customFields).toEqual(customFields) | ||||
|   }) | ||||
| 
 | ||||
|   it('should add an expression when opened if queries are empty', () => { | ||||
|     component.selectionModel.clear() | ||||
|     component.onOpenChange(true) | ||||
|     expect(component.selectionModel.queries.length).toBe(1) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support reset the selection model', () => { | ||||
|     component.selectionModel.addExpression() | ||||
|     component.reset() | ||||
|     expect(component.selectionModel.isEmpty()).toBeTruthy() | ||||
|   }) | ||||
| 
 | ||||
|   it('should get operators for a field', () => { | ||||
|     const field: CustomField = { | ||||
|       id: 1, | ||||
|       name: 'Test Field', | ||||
|       data_type: CustomFieldDataType.String, | ||||
|       extra_data: {}, | ||||
|     } | ||||
|     component.customFields = [field] | ||||
|     const operators = component.getOperatorsForField(1) | ||||
|     expect(operators.length).toEqual( | ||||
|       [ | ||||
|         ...CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP[ | ||||
|           CustomFieldQueryOperatorGroups.Basic | ||||
|         ], | ||||
|         ...CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP[ | ||||
|           CustomFieldQueryOperatorGroups.String | ||||
|         ], | ||||
|       ].length | ||||
|     ) | ||||
| 
 | ||||
|     // Fallback to basic operators if field is not found
 | ||||
|     const operators2 = component.getOperatorsForField(2) | ||||
|     expect(operators2.length).toEqual( | ||||
|       CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP[ | ||||
|         CustomFieldQueryOperatorGroups.Basic | ||||
|       ].length | ||||
|     ) | ||||
|   }) | ||||
| 
 | ||||
|   it('should get select options for a field', () => { | ||||
|     const field: CustomField = { | ||||
|       id: 1, | ||||
|       name: 'Test Field', | ||||
|       data_type: CustomFieldDataType.Select, | ||||
|       extra_data: { select_options: ['Option 1', 'Option 2'] }, | ||||
|     } | ||||
|     component.customFields = [field] | ||||
|     const options = component.getSelectOptionsForField(1) | ||||
|     expect(options).toEqual(['Option 1', 'Option 2']) | ||||
| 
 | ||||
|     // Fallback to empty array if field is not found
 | ||||
|     const options2 = component.getSelectOptionsForField(2) | ||||
|     expect(options2).toEqual([]) | ||||
|   }) | ||||
| 
 | ||||
|   it('should remove an element from the selection model', () => { | ||||
|     const expression = new CustomFieldQueryExpression() | ||||
|     const atom = new CustomFieldQueryAtom() | ||||
|     ;(expression.value as CustomFieldQueryElement[]).push(atom) | ||||
|     component.selectionModel.addExpression(expression) | ||||
|     component.removeElement(atom) | ||||
|     expect(component.selectionModel.isEmpty()).toBeTruthy() | ||||
|     const expression2 = new CustomFieldQueryExpression([ | ||||
|       CustomFieldQueryLogicalOperator.And, | ||||
|       [ | ||||
|         [1, 'icontains', 'test'], | ||||
|         [2, 'icontains', 'test'], | ||||
|       ], | ||||
|     ]) | ||||
|     component.selectionModel.addExpression(expression2) | ||||
|     component.removeElement(expression2) | ||||
|     expect(component.selectionModel.isEmpty()).toBeTruthy() | ||||
|   }) | ||||
| 
 | ||||
|   it('should emit selectionModelChange when model changes', () => { | ||||
|     const nextSpy = jest.spyOn(component.selectionModelChange, 'next') | ||||
|     const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|     component.selectionModel.addAtom(atom) | ||||
|     atom.changed.next(atom) | ||||
|     expect(nextSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should complete selection model subscription when new selection model is set', () => { | ||||
|     const completeSpy = jest.spyOn(component.selectionModel.changed, 'complete') | ||||
|     const selectionModel = new CustomFieldQueriesModel() | ||||
|     component.selectionModel = selectionModel | ||||
|     expect(completeSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support adding an atom', () => { | ||||
|     const expression = new CustomFieldQueryExpression() | ||||
|     component.addAtom(expression) | ||||
|     expect(expression.value.length).toBe(1) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support adding an expression', () => { | ||||
|     const expression = new CustomFieldQueryExpression() | ||||
|     component.addExpression(expression) | ||||
|     expect(expression.value.length).toBe(1) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support getting a custom field by ID', () => { | ||||
|     expect(component.getCustomFieldByID(1)).toEqual(customFields[0]) | ||||
|   }) | ||||
| 
 | ||||
|   it('should sanitize name from title', () => { | ||||
|     component.title = 'Test Title' | ||||
|     expect(component.name).toBe('test_title') | ||||
|   }) | ||||
| 
 | ||||
|   it('should add a default atom on open and focus the select field', fakeAsync(() => { | ||||
|     expect(component.selectionModel.queries.length).toBe(0) | ||||
|     component.onOpenChange(true) | ||||
|     fixture.detectChanges() | ||||
|     tick() | ||||
|     expect(component.selectionModel.queries.length).toBe(1) | ||||
|     expect(window.document.activeElement.tagName).toBe('INPUT') | ||||
|   })) | ||||
| 
 | ||||
|   describe('CustomFieldQueriesModel', () => { | ||||
|     let model: CustomFieldQueriesModel | ||||
| 
 | ||||
|     beforeEach(() => { | ||||
|       model = new CustomFieldQueriesModel() | ||||
|     }) | ||||
| 
 | ||||
|     it('should initialize with empty queries', () => { | ||||
|       expect(model.queries).toEqual([]) | ||||
|     }) | ||||
| 
 | ||||
|     it('should clear queries and fire event', () => { | ||||
|       const nextSpy = jest.spyOn(model.changed, 'next') | ||||
|       model.addExpression() | ||||
|       model.clear() | ||||
|       expect(model.queries).toEqual([]) | ||||
|       expect(nextSpy).toHaveBeenCalledWith(model) | ||||
|     }) | ||||
| 
 | ||||
|     it('should clear queries without firing event', () => { | ||||
|       const nextSpy = jest.spyOn(model.changed, 'next') | ||||
|       model.addExpression() | ||||
|       model.clear(false) | ||||
|       expect(model.queries).toEqual([]) | ||||
|       expect(nextSpy).not.toHaveBeenCalled() | ||||
|     }) | ||||
| 
 | ||||
|     it('should validate an empty model as invalid', () => { | ||||
|       expect(model.isValid()).toBeFalsy() | ||||
|     }) | ||||
| 
 | ||||
|     it('should validate a model with valid expression as valid', () => { | ||||
|       const expression = new CustomFieldQueryExpression() | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       const atom2 = new CustomFieldQueryAtom([2, 'icontains', 'test']) | ||||
|       const expression2 = new CustomFieldQueryExpression() | ||||
|       expression2.addAtom(atom) | ||||
|       expression2.addAtom(atom2) | ||||
|       expression.addExpression(expression2) | ||||
|       model.addExpression(expression) | ||||
|       expect(model.isValid()).toBeTruthy() | ||||
|     }) | ||||
| 
 | ||||
|     it('should validate a model with invalid expression as invalid', () => { | ||||
|       const expression = new CustomFieldQueryExpression() | ||||
|       model.addExpression(expression) | ||||
|       expect(model.isValid()).toBeFalsy() | ||||
|     }) | ||||
| 
 | ||||
|     it('should validate an atom with in or contains operator', () => { | ||||
|       const atom = new CustomFieldQueryAtom([1, 'in', '[1,2,3]']) | ||||
|       expect(model['validateAtom'].apply(null, [atom])).toBeTruthy() | ||||
|       atom.operator = 'contains' | ||||
|       atom.value = [1, 2, 3] | ||||
|       expect(model['validateAtom'].apply(null, [atom])).toBeTruthy() | ||||
|       atom.value = null | ||||
|       expect(model['validateAtom'].apply(null, [atom])).toBeFalsy() | ||||
|     }) | ||||
| 
 | ||||
|     it('should check if model is empty', () => { | ||||
|       expect(model.isEmpty()).toBeTruthy() | ||||
|       model.addExpression() | ||||
|       expect(model.isEmpty()).toBeTruthy() | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       model.addAtom(atom) | ||||
|       expect(model.isEmpty()).toBeFalsy() | ||||
|     }) | ||||
| 
 | ||||
|     it('should add an atom to the model', () => { | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       model.addAtom(atom) | ||||
|       expect(model.queries.length).toBe(1) | ||||
|       expect( | ||||
|         (model.queries[0] as CustomFieldQueryExpression).value.length | ||||
|       ).toBe(1) | ||||
|     }) | ||||
| 
 | ||||
|     it('should add an expression to the model, propagate changes', () => { | ||||
|       const expression = new CustomFieldQueryExpression() | ||||
|       model.addExpression(expression) | ||||
|       expect(model.queries.length).toBe(1) | ||||
|       const expression2 = new CustomFieldQueryExpression([ | ||||
|         CustomFieldQueryLogicalOperator.And, | ||||
|         [ | ||||
|           [1, 'icontains', 'test'], | ||||
|           [2, 'icontains', 'test'], | ||||
|         ], | ||||
|       ]) | ||||
|       model.addExpression(expression2) | ||||
|       const nextSpy = jest.spyOn(model.changed, 'next') | ||||
|       expression2.changed.next(expression2) | ||||
|       expect(nextSpy).toHaveBeenCalled() | ||||
|     }) | ||||
| 
 | ||||
|     it('should remove an element from the model', () => { | ||||
|       const expression = new CustomFieldQueryExpression([ | ||||
|         CustomFieldQueryLogicalOperator.And, | ||||
|         [ | ||||
|           [1, 'icontains', 'test'], | ||||
|           [2, 'icontains', 'test'], | ||||
|         ], | ||||
|       ]) | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       const expression2 = new CustomFieldQueryExpression([ | ||||
|         CustomFieldQueryLogicalOperator.And, | ||||
|         [ | ||||
|           [3, 'icontains', 'test'], | ||||
|           [4, 'icontains', 'test'], | ||||
|         ], | ||||
|       ]) | ||||
|       expression.addAtom(atom) | ||||
|       expression2.addExpression(expression) | ||||
|       model.addExpression(expression2) | ||||
|       model.removeElement(atom) | ||||
|       expect(model.queries.length).toBe(1) | ||||
|       model.removeElement(expression2) | ||||
|     }) | ||||
| 
 | ||||
|     it('should fire changed event when an atom changes', () => { | ||||
|       const nextSpy = jest.spyOn(model.changed, 'next') | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       model.addAtom(atom) | ||||
|       atom.changed.next(atom) | ||||
|       expect(nextSpy).toHaveBeenCalledWith(model) | ||||
|     }) | ||||
| 
 | ||||
|     it('should complete changed subject when element is removed', () => { | ||||
|       const expression = new CustomFieldQueryExpression() | ||||
|       const atom = new CustomFieldQueryAtom([1, 'icontains', 'test']) | ||||
|       ;(expression.value as CustomFieldQueryElement[]).push(atom) | ||||
|       model.addExpression(expression) | ||||
|       const completeSpy = jest.spyOn(atom.changed, 'complete') | ||||
|       model.removeElement(atom) | ||||
|       expect(completeSpy).toHaveBeenCalled() | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
| @ -0,0 +1,316 @@ | ||||
| import { | ||||
|   Component, | ||||
|   EventEmitter, | ||||
|   Input, | ||||
|   OnDestroy, | ||||
|   Output, | ||||
|   QueryList, | ||||
|   ViewChild, | ||||
|   ViewChildren, | ||||
| } from '@angular/core' | ||||
| import { NgbDropdown } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgSelectComponent } from '@ng-select/ng-select' | ||||
| import { Subject, first, takeUntil } from 'rxjs' | ||||
| import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field' | ||||
| import { | ||||
|   CustomFieldQueryElementType, | ||||
|   CustomFieldQueryOperator, | ||||
|   CUSTOM_FIELD_QUERY_OPERATOR_GROUPS_BY_TYPE, | ||||
|   CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP, | ||||
|   CustomFieldQueryOperatorGroups, | ||||
|   CUSTOM_FIELD_QUERY_OPERATOR_LABELS, | ||||
|   CUSTOM_FIELD_QUERY_MAX_DEPTH, | ||||
|   CUSTOM_FIELD_QUERY_MAX_ATOMS, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' | ||||
| import { | ||||
|   CustomFieldQueryElement, | ||||
|   CustomFieldQueryExpression, | ||||
|   CustomFieldQueryAtom, | ||||
| } from 'src/app/utils/custom-field-query-element' | ||||
| import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options' | ||||
| 
 | ||||
| export class CustomFieldQueriesModel { | ||||
|   public queries: CustomFieldQueryElement[] = [] | ||||
| 
 | ||||
|   public readonly changed = new Subject<CustomFieldQueriesModel>() | ||||
| 
 | ||||
|   public clear(fireEvent = true) { | ||||
|     this.queries = [] | ||||
|     if (fireEvent) { | ||||
|       this.changed.next(this) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public isValid(): boolean { | ||||
|     return ( | ||||
|       this.queries.length > 0 && | ||||
|       this.validateExpression(this.queries[0] as CustomFieldQueryExpression) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   public isEmpty(): boolean { | ||||
|     return ( | ||||
|       this.queries.length === 0 || | ||||
|       (this.queries.length === 1 && this.queries[0].value.length === 0) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   private validateAtom(atom: CustomFieldQueryAtom) { | ||||
|     let valid = !!(atom.field && atom.operator && atom.value !== null) | ||||
|     if ( | ||||
|       [ | ||||
|         CustomFieldQueryOperator.In.valueOf(), | ||||
|         CustomFieldQueryOperator.Contains.valueOf(), | ||||
|       ].includes(atom.operator) && | ||||
|       atom.value | ||||
|     ) { | ||||
|       valid = valid && atom.value.length > 0 | ||||
|     } | ||||
|     return valid | ||||
|   } | ||||
| 
 | ||||
|   private validateExpression(expression: CustomFieldQueryExpression) { | ||||
|     return ( | ||||
|       expression.operator && | ||||
|       expression.value.length > 0 && | ||||
|       (expression.value as CustomFieldQueryElement[]).every((e) => | ||||
|         e.type === CustomFieldQueryElementType.Atom | ||||
|           ? this.validateAtom(e as CustomFieldQueryAtom) | ||||
|           : this.validateExpression(e as CustomFieldQueryExpression) | ||||
|       ) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   public addAtom(atom: CustomFieldQueryAtom) { | ||||
|     if (this.queries.length === 0) { | ||||
|       this.addExpression() | ||||
|     } | ||||
|     ;(this.queries[0].value as CustomFieldQueryElement[]).push(atom) | ||||
|     atom.changed.subscribe(() => { | ||||
|       if (atom.field && atom.operator && atom.value) { | ||||
|         this.changed.next(this) | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   public addExpression( | ||||
|     expression: CustomFieldQueryExpression = new CustomFieldQueryExpression() | ||||
|   ) { | ||||
|     if (this.queries.length > 0) { | ||||
|       ;( | ||||
|         (this.queries[0] as CustomFieldQueryExpression) | ||||
|           .value as CustomFieldQueryElement[] | ||||
|       ).push(expression) | ||||
|     } else { | ||||
|       this.queries.push(expression) | ||||
|     } | ||||
|     expression.changed.subscribe(() => { | ||||
|       this.changed.next(this) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   private findElement( | ||||
|     queryElement: CustomFieldQueryElement, | ||||
|     elements: any[] | ||||
|   ): CustomFieldQueryElement { | ||||
|     for (let i = 0; i < elements.length; i++) { | ||||
|       if (elements[i] === queryElement) { | ||||
|         return elements.splice(i, 1)[0] | ||||
|       } else if (elements[i].type === CustomFieldQueryElementType.Expression) { | ||||
|         return this.findElement( | ||||
|           queryElement, | ||||
|           elements[i].value as CustomFieldQueryElement[] | ||||
|         ) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public removeElement(queryElement: CustomFieldQueryElement) { | ||||
|     let foundComponent | ||||
|     for (let i = 0; i < this.queries.length; i++) { | ||||
|       let query = this.queries[i] | ||||
|       if (query === queryElement) { | ||||
|         foundComponent = this.queries.splice(i, 1)[0] | ||||
|         break | ||||
|       } else if (query.type === CustomFieldQueryElementType.Expression) { | ||||
|         foundComponent = this.findElement(queryElement, query.value as any[]) | ||||
|       } | ||||
|     } | ||||
|     if (foundComponent) { | ||||
|       foundComponent.changed.complete() | ||||
|       if (this.isEmpty()) { | ||||
|         this.clear() | ||||
|       } | ||||
|       this.changed.next(this) | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'pngx-custom-fields-query-dropdown', | ||||
|   templateUrl: './custom-fields-query-dropdown.component.html', | ||||
|   styleUrls: ['./custom-fields-query-dropdown.component.scss'], | ||||
| }) | ||||
| export class CustomFieldsQueryDropdownComponent implements OnDestroy { | ||||
|   public CustomFieldQueryComponentType = CustomFieldQueryElementType | ||||
|   public CustomFieldQueryOperator = CustomFieldQueryOperator | ||||
|   public CustomFieldDataType = CustomFieldDataType | ||||
|   public CUSTOM_FIELD_QUERY_MAX_DEPTH = CUSTOM_FIELD_QUERY_MAX_DEPTH | ||||
|   public CUSTOM_FIELD_QUERY_MAX_ATOMS = CUSTOM_FIELD_QUERY_MAX_ATOMS | ||||
|   public popperOptions = popperOptionsReenablePreventOverflow | ||||
| 
 | ||||
|   @Input() | ||||
|   title: string | ||||
| 
 | ||||
|   @Input() | ||||
|   filterPlaceholder: string = '' | ||||
| 
 | ||||
|   @Input() | ||||
|   icon: string | ||||
| 
 | ||||
|   @Input() | ||||
|   allowSelectNone: boolean = false | ||||
| 
 | ||||
|   @Input() | ||||
|   editing = false | ||||
| 
 | ||||
|   @Input() | ||||
|   applyOnClose = false | ||||
| 
 | ||||
|   get name(): string { | ||||
|     return this.title ? this.title.replace(/\s/g, '_').toLowerCase() : null | ||||
|   } | ||||
| 
 | ||||
|   @Input() | ||||
|   disabled: boolean = false | ||||
| 
 | ||||
|   @ViewChild('dropdown') dropdown: NgbDropdown | ||||
| 
 | ||||
|   @ViewChildren(NgSelectComponent) fieldSelects!: QueryList<NgSelectComponent> | ||||
| 
 | ||||
|   private _selectionModel: CustomFieldQueriesModel | ||||
| 
 | ||||
|   @Input() | ||||
|   set selectionModel(model: CustomFieldQueriesModel) { | ||||
|     if (this._selectionModel) { | ||||
|       this._selectionModel.changed.complete() | ||||
|     } | ||||
|     model.changed.subscribe(() => { | ||||
|       this.onModelChange() | ||||
|     }) | ||||
|     this._selectionModel = model | ||||
|   } | ||||
| 
 | ||||
|   get selectionModel(): CustomFieldQueriesModel { | ||||
|     return this._selectionModel | ||||
|   } | ||||
| 
 | ||||
|   private onModelChange() { | ||||
|     if (this.selectionModel.isEmpty() || this.selectionModel.isValid()) { | ||||
|       this.selectionModelChange.next(this.selectionModel) | ||||
|       this.selectionModel.isEmpty() && this.dropdown?.close() | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @Output() | ||||
|   selectionModelChange = new EventEmitter<CustomFieldQueriesModel>() | ||||
| 
 | ||||
|   customFields: CustomField[] = [] | ||||
| 
 | ||||
|   private unsubscribeNotifier: Subject<any> = new Subject() | ||||
| 
 | ||||
|   constructor(protected customFieldsService: CustomFieldsService) { | ||||
|     this.selectionModel = new CustomFieldQueriesModel() | ||||
|     this.getFields() | ||||
|     this.reset() | ||||
|   } | ||||
| 
 | ||||
|   ngOnDestroy(): void { | ||||
|     this.unsubscribeNotifier.next(this) | ||||
|     this.unsubscribeNotifier.complete() | ||||
|   } | ||||
| 
 | ||||
|   public onOpenChange(open: boolean) { | ||||
|     if (open) { | ||||
|       if (this.selectionModel.queries.length === 0) { | ||||
|         this.selectionModel.addAtom( | ||||
|           new CustomFieldQueryAtom([ | ||||
|             null, | ||||
|             CustomFieldQueryOperator.Exists, | ||||
|             'true', | ||||
|           ]) | ||||
|         ) | ||||
|       } | ||||
|       if ( | ||||
|         this.selectionModel.queries.length === 1 && | ||||
|         ( | ||||
|           (this.selectionModel.queries[0] as CustomFieldQueryExpression) | ||||
|             ?.value[0] as CustomFieldQueryAtom | ||||
|         )?.field === null | ||||
|       ) { | ||||
|         setTimeout(() => { | ||||
|           this.fieldSelects.first?.focus() | ||||
|         }, 0) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public get isActive(): boolean { | ||||
|     return this.selectionModel.isValid() | ||||
|   } | ||||
| 
 | ||||
|   private getFields() { | ||||
|     this.customFieldsService | ||||
|       .listAll() | ||||
|       .pipe(first(), takeUntil(this.unsubscribeNotifier)) | ||||
|       .subscribe((result) => { | ||||
|         this.customFields = result.results | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|   public getCustomFieldByID(id: number): CustomField { | ||||
|     return this.customFields.find((field) => field.id === id) | ||||
|   } | ||||
| 
 | ||||
|   public addAtom(expression: CustomFieldQueryExpression) { | ||||
|     expression.addAtom() | ||||
|   } | ||||
| 
 | ||||
|   public addExpression(expression: CustomFieldQueryExpression) { | ||||
|     expression.addExpression() | ||||
|   } | ||||
| 
 | ||||
|   public removeElement(element: CustomFieldQueryElement) { | ||||
|     this.selectionModel.removeElement(element) | ||||
|   } | ||||
| 
 | ||||
|   public reset() { | ||||
|     this.selectionModel.clear(false) | ||||
|     this.selectionModel.changed.next(this.selectionModel) | ||||
|   } | ||||
| 
 | ||||
|   getOperatorsForField( | ||||
|     fieldID: number | ||||
|   ): Array<{ value: string; label: string }> { | ||||
|     const field = this.customFields.find((field) => field.id === fieldID) | ||||
|     const groups: CustomFieldQueryOperatorGroups[] = field | ||||
|       ? CUSTOM_FIELD_QUERY_OPERATOR_GROUPS_BY_TYPE[field.data_type] | ||||
|       : [CustomFieldQueryOperatorGroups.Basic] | ||||
|     const operators = groups.flatMap( | ||||
|       (group) => CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP[group] | ||||
|     ) | ||||
|     return operators.map((operator) => ({ | ||||
|       value: operator, | ||||
|       label: CUSTOM_FIELD_QUERY_OPERATOR_LABELS[operator], | ||||
|     })) | ||||
|   } | ||||
| 
 | ||||
|   getSelectOptionsForField(fieldID: number): string[] { | ||||
|     const field = this.customFields.find((field) => field.id === fieldID) | ||||
|     if (field) { | ||||
|       return field.extra_data['select_options'] | ||||
|     } | ||||
|     return [] | ||||
|   } | ||||
| } | ||||
| @ -1,4 +1,4 @@ | ||||
| <div class="btn-group w-100" ngbDropdown role="group"> | ||||
| <div class="btn-group w-100" ngbDropdown role="group" [popperOptions]="popperOptions"> | ||||
|   <button class="btn btn-sm" id="dropdown{{title}}" ngbDropdownToggle [ngClass]="createdDateBefore || createdDateAfter ? 'btn-primary' : 'btn-outline-primary'" [disabled]="disabled"> | ||||
|     <i-bs width="1em" height="1em" name="calendar-event-fill"></i-bs> | ||||
|     <div class="d-none d-sm-inline"> {{title}}</div> | ||||
| @ -17,7 +17,7 @@ | ||||
|                 } | ||||
|               </div> | ||||
|               <div class="d-flex justify-content-between w-100 align-items-center ps-2"> | ||||
|                 <div class="pe-2 pe-lg-4"> | ||||
|                 <div class="pe-4"> | ||||
|                   {{rd.name}} | ||||
|                 </div> | ||||
|                 <div class="text-muted small pe-2"> | ||||
| @ -28,20 +28,19 @@ | ||||
|               </div> | ||||
|             </button> | ||||
|           } | ||||
|           <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
|           <div class="list-group-item d-flex p-2" role="menuitem"> | ||||
| 
 | ||||
|             <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|               <div i18n>After</div> | ||||
|             <div class="selected-icon"> | ||||
|               @if (createdDateAfter) { | ||||
|                 <a class="btn btn-link p-0 m-0" (click)="clearCreatedAfter()"> | ||||
|                   <i-bs width="1em" height="1em" name="x"></i-bs> | ||||
|                   <small i18n>Clear</small> | ||||
|                 <a class="text-light focus-variants" href="javascript:void(0)" (click)="clearCreatedAfter()"> | ||||
|                   <i-bs width="1em" height="1em" name="check" class="variant-unfocused"></i-bs> | ||||
|                   <i-bs width="1em" height="1em" name="x" class="variant-focused text-primary"></i-bs> | ||||
|                 </a> | ||||
|               } | ||||
|             </div> | ||||
| 
 | ||||
|             <div class="input-group input-group-sm"> | ||||
|               <input class="form-control" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|             <div class="input-group input-group-sm small ps-1 pe-2"> | ||||
|               <span class="input-group-text w-25 small text-muted" i18n>After</span> | ||||
|               <input class="form-control small" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|                 maxlength="10" [(ngModel)]="createdDateAfter" ngbDatepicker #createdDateAfterPicker="ngbDatepicker"> | ||||
|               <button class="btn btn-outline-secondary" (click)="createdDateAfterPicker.toggle()" type="button"> | ||||
|                 <i-bs width="1em" height="1em" name="calendar"></i-bs> | ||||
| @ -49,20 +48,19 @@ | ||||
|             </div> | ||||
| 
 | ||||
|           </div> | ||||
|           <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
|           <div class="list-group-item d-flex p-2" role="menuitem"> | ||||
| 
 | ||||
|             <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|               <div i18n>Before</div> | ||||
|             <div class="selected-icon"> | ||||
|               @if (createdDateBefore) { | ||||
|                 <a class="btn btn-link p-0 m-0" (click)="clearCreatedBefore()"> | ||||
|                   <i-bs width="1em" height="1em" name="x"></i-bs> | ||||
|                   <small i18n>Clear</small> | ||||
|                 <a class="text-light focus-variants" href="javascript:void(0)" (click)="clearCreatedBefore()"> | ||||
|                   <i-bs width="1em" height="1em" name="check" class="variant-unfocused"></i-bs> | ||||
|                   <i-bs width="1em" height="1em" name="x" class="variant-focused text-primary"></i-bs> | ||||
|                 </a> | ||||
|               } | ||||
|             </div> | ||||
| 
 | ||||
|             <div class="input-group input-group-sm"> | ||||
|               <input class="form-control" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|             <div class="input-group input-group-sm small ps-1 pe-2"> | ||||
|               <span class="input-group-text w-25 small text-muted" i18n>Before</span> | ||||
|               <input class="form-control small" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|                 maxlength="10" [(ngModel)]="createdDateBefore" ngbDatepicker #createdDateBeforePicker="ngbDatepicker"> | ||||
|               <button class="btn btn-outline-secondary" (click)="createdDateBeforePicker.toggle()" type="button"> | ||||
|                 <i-bs width="1em" height="1em" name="calendar"></i-bs> | ||||
| @ -83,7 +81,7 @@ | ||||
|                 } | ||||
|               </div> | ||||
|               <div class="d-flex justify-content-between w-100 align-items-center ps-2"> | ||||
|                 <div class="pe-2 pe-lg-4"> | ||||
|                 <div class="pe-4"> | ||||
|                   {{rd.name}} | ||||
|                 </div> | ||||
|                 <div class="text-muted small pe-2"> | ||||
| @ -94,20 +92,19 @@ | ||||
|               </div> | ||||
|             </button> | ||||
|           } | ||||
|           <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
|           <div class="list-group-item d-flex p-2" role="menuitem"> | ||||
| 
 | ||||
|             <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|               <div i18n>After</div> | ||||
|             <div class="selected-icon"> | ||||
|               @if (addedDateAfter) { | ||||
|                 <a class="btn btn-link p-0 m-0" (click)="clearAddedAfter()"> | ||||
|                   <i-bs width="1em" height="1em" name="x"></i-bs> | ||||
|                   <small i18n>Clear</small> | ||||
|                 <a class="text-light focus-variants" href="javascript:void(0)" (click)="clearAddedAfter()"> | ||||
|                   <i-bs width="1em" height="1em" name="check" class="variant-unfocused"></i-bs> | ||||
|                   <i-bs width="1em" height="1em" name="x" class="variant-focused text-primary"></i-bs> | ||||
|                 </a> | ||||
|               } | ||||
|             </div> | ||||
| 
 | ||||
|             <div class="input-group input-group-sm"> | ||||
|               <input class="form-control" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|             <div class="input-group input-group-sm small ps-1 pe-2"> | ||||
|               <span class="input-group-text w-25 small text-muted" i18n>After</span> | ||||
|               <input class="form-control small" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|                 maxlength="10" [(ngModel)]="addedDateAfter" ngbDatepicker #addedDateAfterPicker="ngbDatepicker"> | ||||
|               <button class="btn btn-outline-secondary" (click)="addedDateAfterPicker.toggle()" type="button"> | ||||
|                 <i-bs width="1em" height="1em" name="calendar"></i-bs> | ||||
| @ -115,20 +112,19 @@ | ||||
|             </div> | ||||
| 
 | ||||
|           </div> | ||||
|           <div class="list-group-item d-flex flex-column align-items-start" role="menuitem"> | ||||
|           <div class="list-group-item d-flex p-2" role="menuitem"> | ||||
| 
 | ||||
|             <div class="mb-2 d-flex flex-row w-100 justify-content-between small"> | ||||
|               <div i18n>Before</div> | ||||
|             <div class="selected-icon"> | ||||
|               @if (addedDateBefore) { | ||||
|                 <a class="btn btn-link p-0 m-0" (click)="clearAddedBefore()"> | ||||
|                   <i-bs width="1em" height="1em" name="x"></i-bs> | ||||
|                   <small i18n>Clear</small> | ||||
|                 <a class="text-light focus-variants" href="javascript:void(0)" (click)="clearAddedBefore()"> | ||||
|                   <i-bs width="1em" height="1em" name="check" class="variant-unfocused"></i-bs> | ||||
|                   <i-bs width="1em" height="1em" name="x" class="variant-focused text-primary"></i-bs> | ||||
|                 </a> | ||||
|               } | ||||
|             </div> | ||||
| 
 | ||||
|             <div class="input-group input-group-sm"> | ||||
|               <input class="form-control" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|             <div class="input-group input-group-sm small ps-1 pe-2"> | ||||
|               <span class="input-group-text w-25 small text-muted" i18n>Before</span> | ||||
|               <input class="form-control small" [placeholder]="datePlaceHolder" (dateSelect)="onChangeDebounce()" (change)="onChangeDebounce()" (keypress)="onKeyPress($event)" | ||||
|                 maxlength="10" [(ngModel)]="addedDateBefore" ngbDatepicker #addedDateBeforePicker="ngbDatepicker"> | ||||
|               <button class="btn btn-outline-secondary" (click)="addedDateBeforePicker.toggle()" type="button"> | ||||
|                 <i-bs width="1em" height="1em" name="calendar"></i-bs> | ||||
|  | ||||
| @ -5,6 +5,12 @@ | ||||
|     --bs-dropdown-min-width: 40rem; | ||||
|   } | ||||
| 
 | ||||
|   @media screen and (max-width: 767px) { | ||||
|     .border-end { | ||||
|       border: none !important; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   .btn-link { | ||||
|     line-height: 1; | ||||
|   } | ||||
| @ -14,3 +20,24 @@ | ||||
|   min-width: 1em; | ||||
|   min-height: 1em; | ||||
| } | ||||
| 
 | ||||
| .input-group-sm { | ||||
|   .form-control { | ||||
|     font-size: 0.875rem; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .focus-variants { | ||||
|   .variant-focused { | ||||
|     display: none; | ||||
|   } | ||||
| 
 | ||||
|   &:hover, &:focus { | ||||
|     .variant-unfocused { | ||||
|       display: none; | ||||
|     } | ||||
|     .variant-focused { | ||||
|       display: block; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -11,6 +11,7 @@ import { Subject, Subscription } from 'rxjs' | ||||
| import { debounceTime } from 'rxjs/operators' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { ISODateAdapter } from 'src/app/utils/ngb-iso-date-adapter' | ||||
| import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options' | ||||
| 
 | ||||
| export interface DateSelection { | ||||
|   createdBefore?: string | ||||
| @ -35,6 +36,8 @@ export enum RelativeDate { | ||||
|   providers: [{ provide: NgbDateAdapter, useClass: ISODateAdapter }], | ||||
| }) | ||||
| export class DatesDropdownComponent implements OnInit, OnDestroy { | ||||
|   public popperOptions = popperOptionsReenablePreventOverflow | ||||
| 
 | ||||
|   constructor(settings: SettingsService) { | ||||
|     this.datePlaceHolder = settings.getLocalizedDateInputFormat() | ||||
|   } | ||||
|  | ||||
| @ -67,7 +67,7 @@ export class CustomFieldEditDialogComponent | ||||
|     this.selectOptionInputs.changes | ||||
|       .pipe(takeUntil(this.unsubscribeNotifier)) | ||||
|       .subscribe(() => { | ||||
|         this.selectOptionInputs.last.nativeElement.focus() | ||||
|         this.selectOptionInputs.last?.nativeElement.focus() | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -11,7 +11,7 @@ import { | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgSelectModule } from '@ng-select/ng-select' | ||||
| import { IMAPSecurity } from 'src/app/data/mail-account' | ||||
| import { IMAPSecurity, MailAccountType } from 'src/app/data/mail-account' | ||||
| import { IfOwnerDirective } from 'src/app/directives/if-owner.directive' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| @ -82,6 +82,7 @@ describe('MailAccountEditDialogComponent', () => { | ||||
|       imap_port: 443, | ||||
|       imap_security: IMAPSecurity.SSL, | ||||
|       is_token: false, | ||||
|       account_type: MailAccountType.IMAP, | ||||
|     } | ||||
| 
 | ||||
|     // success
 | ||||
|  | ||||
| @ -12,12 +12,15 @@ | ||||
|       <div class="col-md-4"> | ||||
|         <pngx-input-text [horizontal]="true" i18n-title title="Name" formControlName="name" [error]="error?.name" autocomplete="off"></pngx-input-text> | ||||
|       </div> | ||||
|       <div class="col-md-4"> | ||||
|         <pngx-input-number [horizontal]="true" i18n-title title="Rule order" formControlName="order" [showAdd]="false" [error]="error?.order"></pngx-input-number> | ||||
|       </div> | ||||
|       <div class="col-md-4"> | ||||
|       <div class="col-md-3"> | ||||
|         <pngx-input-select [horizontal]="true" i18n-title title="Account" [items]="accounts" formControlName="account"></pngx-input-select> | ||||
|       </div> | ||||
|       <div class="col-md-3"> | ||||
|         <pngx-input-number [horizontal]="true" i18n-title title="Order" formControlName="order" [showAdd]="false" [error]="error?.order"></pngx-input-number> | ||||
|       </div> | ||||
|       <div class="col-md-2 pt-2"> | ||||
|         <pngx-input-switch [horizontal]="true" i18n-title title="Enabled" formControlName="enabled"></pngx-input-switch> | ||||
|       </div> | ||||
|     </div> | ||||
|     <hr class="mt-0"/> | ||||
|     <div class="row"> | ||||
|  | ||||
| @ -24,6 +24,7 @@ import { TextComponent } from '../../input/text/text.component' | ||||
| import { EditDialogMode } from '../edit-dialog.component' | ||||
| import { MailRuleEditDialogComponent } from './mail-rule-edit-dialog.component' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { SwitchComponent } from '../../input/switch/switch.component' | ||||
| 
 | ||||
| describe('MailRuleEditDialogComponent', () => { | ||||
|   let component: MailRuleEditDialogComponent | ||||
| @ -43,6 +44,7 @@ describe('MailRuleEditDialogComponent', () => { | ||||
|         TagsComponent, | ||||
|         SafeHtmlPipe, | ||||
|         CheckComponent, | ||||
|         SwitchComponent, | ||||
|       ], | ||||
|       imports: [FormsModule, ReactiveFormsModule, NgSelectModule, NgbModule], | ||||
|       providers: [ | ||||
|  | ||||
| @ -153,6 +153,7 @@ export class MailRuleEditDialogComponent extends EditDialogComponent<MailRule> { | ||||
|     return new FormGroup({ | ||||
|       name: new FormControl(null), | ||||
|       account: new FormControl(null), | ||||
|       enabled: new FormControl(true), | ||||
|       folder: new FormControl('INBOX'), | ||||
|       filter_from: new FormControl(null), | ||||
|       filter_to: new FormControl(null), | ||||
|  | ||||
| @ -10,7 +10,57 @@ | ||||
|   <div class="modal-body"> | ||||
| 
 | ||||
|     <pngx-input-text i18n-title title="Name" formControlName="name" [error]="error?.name" autocomplete="off"></pngx-input-text> | ||||
|     <pngx-input-text i18n-title title="Path" formControlName="path" [error]="error?.path" [hint]="pathHint"></pngx-input-text> | ||||
|     <pngx-input-textarea i18n-title title="Path" formControlName="path" [error]="error?.path" hint="See <a target='_blank' href='https://docs.paperless-ngx.com/advanced_usage/#file-name-handling'>the documentation</a>." i18n-hint [monospace]="true"></pngx-input-textarea> | ||||
| 
 | ||||
|     <div ngbAccordion> | ||||
|       <div ngbAccordionItem> | ||||
|         <h2 ngbAccordionHeader> | ||||
|           <button ngbAccordionButton i18n>Preview</button> | ||||
|         </h2> | ||||
|         <div ngbAccordionCollapse> | ||||
|           <div ngbAccordionBody> | ||||
|             <ng-template> | ||||
|               <div class="card mb-2"> | ||||
|                 <div class="card-body p-2"> | ||||
|                   @if (testLoading) { | ||||
|                     <ng-container [ngTemplateOutlet]="loadingTemplate"></ng-container> | ||||
|                   } @else if (testResult) { | ||||
|                     <code>{{testResult}}</code> | ||||
|                   } @else if (testFailed) { | ||||
|                     <div class="text-danger" i18n>Path test failed</div> | ||||
|                   } @else { | ||||
|                     <div class="text-muted small" i18n>No document selected</div> | ||||
|                   } | ||||
|                 </div> | ||||
|               </div> | ||||
|               <ng-select name="testDocument" | ||||
|                 [items]="foundDocuments$ | async" | ||||
|                 placeholder="Search for a document" i18n-placeholder | ||||
|                 notFoundText="No documents found" i18n-notFoundText | ||||
|                 bindValue="id" | ||||
|                 bindLabel="title" | ||||
|                 [compareWith]="compareDocuments" | ||||
|                 [trackByFn]="trackByFn" | ||||
|                 [minTermLength]="2" | ||||
|                 [loading]="loading" | ||||
|                 [typeahead]="documentsInput$" | ||||
|                 (change)="testPath($event)"> | ||||
|                 <ng-template #loadingTemplate ng-loadingspinner-tmp> | ||||
|                   <div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div> | ||||
|                   <div class="visually-hidden" i18n>Loading...</div> | ||||
|                 </ng-template> | ||||
|                 <ng-template ng-option-tmp let-document="item" let-index="index" let-search="searchTerm"> | ||||
|                   <div>{{document.title}} <small class="text-muted">({{document.created | customDate:'shortDate'}})</small></div> | ||||
|                 </ng-template> | ||||
|               </ng-select> | ||||
|             </ng-template> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
|     <hr/> | ||||
| 
 | ||||
|     <pngx-input-select i18n-title title="Matching algorithm" [items]="getMatchingAlgorithms()" formControlName="matching_algorithm"></pngx-input-select> | ||||
|     @if (patternRequired) { | ||||
|       <pngx-input-text i18n-title title="Matching pattern" formControlName="match" [error]="error?.match"></pngx-input-text> | ||||
|  | ||||
| @ -0,0 +1,4 @@ | ||||
| .accordion { | ||||
|     --bs-accordion-btn-padding-x: 0.75rem; | ||||
|     --bs-accordion-btn-padding-y: 0.375rem; | ||||
|   } | ||||
| @ -1,7 +1,11 @@ | ||||
| import { provideHttpClientTesting } from '@angular/common/http/testing' | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { | ||||
|   NgbAccordionButton, | ||||
|   NgbActiveModal, | ||||
|   NgbModule, | ||||
| } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgSelectModule } from '@ng-select/ng-select' | ||||
| import { IfOwnerDirective } from 'src/app/directives/if-owner.directive' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| @ -10,13 +14,20 @@ import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { PermissionsFormComponent } from '../../input/permissions/permissions-form/permissions-form.component' | ||||
| import { SelectComponent } from '../../input/select/select.component' | ||||
| import { TextComponent } from '../../input/text/text.component' | ||||
| import { TextAreaComponent } from '../../input/textarea/textarea.component' | ||||
| import { EditDialogMode } from '../edit-dialog.component' | ||||
| import { StoragePathEditDialogComponent } from './storage-path-edit-dialog.component' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { StoragePathService } from 'src/app/services/rest/storage-path.service' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { of, throwError } from 'rxjs' | ||||
| import { FILTER_TITLE } from 'src/app/data/filter-rule-type' | ||||
| import { By } from '@angular/platform-browser' | ||||
| 
 | ||||
| describe('StoragePathEditDialogComponent', () => { | ||||
|   let component: StoragePathEditDialogComponent | ||||
|   let settingsService: SettingsService | ||||
|   let documentService: DocumentService | ||||
|   let fixture: ComponentFixture<StoragePathEditDialogComponent> | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
| @ -27,6 +38,7 @@ describe('StoragePathEditDialogComponent', () => { | ||||
|         IfOwnerDirective, | ||||
|         SelectComponent, | ||||
|         TextComponent, | ||||
|         TextAreaComponent, | ||||
|         PermissionsFormComponent, | ||||
|         SafeHtmlPipe, | ||||
|       ], | ||||
| @ -38,6 +50,7 @@ describe('StoragePathEditDialogComponent', () => { | ||||
|       ], | ||||
|     }).compileComponents() | ||||
| 
 | ||||
|     documentService = TestBed.inject(DocumentService) | ||||
|     fixture = TestBed.createComponent(StoragePathEditDialogComponent) | ||||
|     settingsService = TestBed.inject(SettingsService) | ||||
|     settingsService.currentUser = { id: 99, username: 'user99' } | ||||
| @ -57,4 +70,87 @@ describe('StoragePathEditDialogComponent', () => { | ||||
|     fixture.detectChanges() | ||||
|     expect(editTitleSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support test path', () => { | ||||
|     const testSpy = jest.spyOn( | ||||
|       component['service'] as StoragePathService, | ||||
|       'testPath' | ||||
|     ) | ||||
|     testSpy.mockReturnValueOnce(of('test/abc123')) | ||||
|     component.objectForm.patchValue({ path: 'test/{{title}}' }) | ||||
|     fixture.detectChanges() | ||||
|     component.testPath({ id: 1 }) | ||||
|     expect(testSpy).toHaveBeenCalledWith('test/{{title}}', 1) | ||||
|     expect(component.testResult).toBe('test/abc123') | ||||
|     expect(component.testFailed).toBeFalsy() | ||||
| 
 | ||||
|     // test failed
 | ||||
|     testSpy.mockReturnValueOnce(of('')) | ||||
|     component.testPath({ id: 1 }) | ||||
|     expect(component.testResult).toBeNull() | ||||
|     expect(component.testFailed).toBeTruthy() | ||||
| 
 | ||||
|     component.testPath(null) | ||||
|     expect(component.testResult).toBeNull() | ||||
|   }) | ||||
| 
 | ||||
|   it('should compare two documents by id', () => { | ||||
|     const doc1 = { id: 1 } | ||||
|     const doc2 = { id: 2 } | ||||
|     expect(component.compareDocuments(doc1, doc1)).toBeTruthy() | ||||
|     expect(component.compareDocuments(doc1, doc2)).toBeFalsy() | ||||
|   }) | ||||
| 
 | ||||
|   it('should use id as trackBy', () => { | ||||
|     expect(component.trackByFn({ id: 1 })).toBe(1) | ||||
|   }) | ||||
| 
 | ||||
|   it('should search on select text input', () => { | ||||
|     fixture.debugElement | ||||
|       .query(By.directive(NgbAccordionButton)) | ||||
|       .triggerEventHandler('click', null) | ||||
|     fixture.detectChanges() | ||||
|     const documents = [ | ||||
|       { id: 1, title: 'foo' }, | ||||
|       { id: 2, title: 'bar' }, | ||||
|     ] | ||||
|     const listSpy = jest.spyOn(documentService, 'listFiltered') | ||||
|     listSpy.mockReturnValueOnce( | ||||
|       of({ | ||||
|         count: 1, | ||||
|         results: documents[0], | ||||
|         all: [1], | ||||
|       } as any) | ||||
|     ) | ||||
|     component.documentsInput$.next('bar') | ||||
|     expect(listSpy).toHaveBeenCalledWith( | ||||
|       1, | ||||
|       null, | ||||
|       'created', | ||||
|       true, | ||||
|       [{ rule_type: FILTER_TITLE, value: 'bar' }], | ||||
|       { truncate_content: true } | ||||
|     ) | ||||
|     listSpy.mockReturnValueOnce( | ||||
|       of({ | ||||
|         count: 2, | ||||
|         results: [...documents], | ||||
|         all: [1, 2], | ||||
|       } as any) | ||||
|     ) | ||||
|     component.documentsInput$.next('ba') | ||||
|     listSpy.mockReturnValueOnce(throwError(() => new Error())) | ||||
|     component.documentsInput$.next('foo') | ||||
|   }) | ||||
| 
 | ||||
|   it('should run path test on path change', () => { | ||||
|     const testSpy = jest.spyOn(component, 'testPath') | ||||
|     component['testDocument'] = { id: 1 } as any | ||||
|     component.objectForm.patchValue( | ||||
|       { path: 'test/{{title}}' }, | ||||
|       { emitEvent: true } | ||||
|     ) | ||||
|     fixture.detectChanges() | ||||
|     expect(testSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,9 +1,25 @@ | ||||
| import { Component } from '@angular/core' | ||||
| import { Component, OnDestroy } from '@angular/core' | ||||
| import { FormControl, FormGroup } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { | ||||
|   Subject, | ||||
|   Observable, | ||||
|   concat, | ||||
|   of, | ||||
|   distinctUntilChanged, | ||||
|   takeUntil, | ||||
|   tap, | ||||
|   switchMap, | ||||
|   map, | ||||
|   catchError, | ||||
|   filter, | ||||
| } from 'rxjs' | ||||
| import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component' | ||||
| import { Document } from 'src/app/data/document' | ||||
| import { FILTER_TITLE } from 'src/app/data/filter-rule-type' | ||||
| import { DEFAULT_MATCHING_ALGORITHM } from 'src/app/data/matching-model' | ||||
| import { StoragePath } from 'src/app/data/storage-path' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { StoragePathService } from 'src/app/services/rest/storage-path.service' | ||||
| import { UserService } from 'src/app/services/rest/user.service' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| @ -13,24 +29,34 @@ import { SettingsService } from 'src/app/services/settings.service' | ||||
|   templateUrl: './storage-path-edit-dialog.component.html', | ||||
|   styleUrls: ['./storage-path-edit-dialog.component.scss'], | ||||
| }) | ||||
| export class StoragePathEditDialogComponent extends EditDialogComponent<StoragePath> { | ||||
| export class StoragePathEditDialogComponent | ||||
|   extends EditDialogComponent<StoragePath> | ||||
|   implements OnDestroy | ||||
| { | ||||
|   public documentsInput$ = new Subject<string>() | ||||
|   public foundDocuments$: Observable<Document[]> | ||||
|   private testDocument: Document | ||||
|   public testResult: string | ||||
|   public testFailed: boolean = false | ||||
|   public loading = false | ||||
|   public testLoading = false | ||||
| 
 | ||||
|   private unsubscribeNotifier: Subject<any> = new Subject() | ||||
| 
 | ||||
|   constructor( | ||||
|     service: StoragePathService, | ||||
|     activeModal: NgbActiveModal, | ||||
|     userService: UserService, | ||||
|     settingsService: SettingsService | ||||
|     settingsService: SettingsService, | ||||
|     private documentsService: DocumentService | ||||
|   ) { | ||||
|     super(service, activeModal, userService, settingsService) | ||||
|     this.initPathObservables() | ||||
|   } | ||||
| 
 | ||||
|   get pathHint() { | ||||
|     return ( | ||||
|       $localize`e.g.` + | ||||
|       ' <code>{created_year}-{title}</code> ' + | ||||
|       $localize`or use slashes to add directories e.g.` + | ||||
|       ' <code>{created_year}/{correspondent}/{title}</code>. ' + | ||||
|       $localize`See <a target="_blank" href="https://docs.paperless-ngx.com/advanced_usage/#file-name-handling">documentation</a> for full list.` | ||||
|     ) | ||||
|   ngOnDestroy(): void { | ||||
|     this.unsubscribeNotifier.next(this) | ||||
|     this.unsubscribeNotifier.complete() | ||||
|   } | ||||
| 
 | ||||
|   getCreateTitle() { | ||||
| @ -51,4 +77,70 @@ export class StoragePathEditDialogComponent extends EditDialogComponent<StorageP | ||||
|       permissions_form: new FormControl(null), | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   public testPath(document: Document) { | ||||
|     if (!document) { | ||||
|       this.testResult = null | ||||
|       return | ||||
|     } | ||||
|     this.testDocument = document | ||||
|     this.testLoading = true | ||||
|     ;(this.service as StoragePathService) | ||||
|       .testPath(this.objectForm.get('path').value, document.id) | ||||
|       .subscribe((result) => { | ||||
|         if (result?.length) { | ||||
|           this.testResult = result | ||||
|           this.testFailed = false | ||||
|         } else { | ||||
|           this.testResult = null | ||||
|           this.testFailed = true | ||||
|         } | ||||
|         this.testLoading = false | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|   compareDocuments(document: Document, selectedDocument: Document) { | ||||
|     return document.id === selectedDocument.id | ||||
|   } | ||||
| 
 | ||||
|   private initPathObservables() { | ||||
|     this.objectForm | ||||
|       .get('path') | ||||
|       .valueChanges.pipe( | ||||
|         takeUntil(this.unsubscribeNotifier), | ||||
|         filter((path) => path && !!this.testDocument) | ||||
|       ) | ||||
|       .subscribe(() => { | ||||
|         this.testPath(this.testDocument) | ||||
|       }) | ||||
| 
 | ||||
|     this.foundDocuments$ = concat( | ||||
|       of([]), // default items
 | ||||
|       this.documentsInput$.pipe( | ||||
|         distinctUntilChanged(), | ||||
|         takeUntil(this.unsubscribeNotifier), | ||||
|         tap(() => (this.loading = true)), | ||||
|         switchMap((title) => | ||||
|           this.documentsService | ||||
|             .listFiltered( | ||||
|               1, | ||||
|               null, | ||||
|               'created', | ||||
|               true, | ||||
|               [{ rule_type: FILTER_TITLE, value: title }], | ||||
|               { truncate_content: true } | ||||
|             ) | ||||
|             .pipe( | ||||
|               map((result) => result.results), | ||||
|               catchError(() => of([])), // empty on error
 | ||||
|               tap(() => (this.loading = false)) | ||||
|             ) | ||||
|         ) | ||||
|       ) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   trackByFn(item: Document) { | ||||
|     return item.id | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -3,3 +3,7 @@ | ||||
|         color: rgba(var(--bs-danger-rgb), var(--bs-text-opacity)) !important; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| .accordion-button { | ||||
|     font-size: 1rem; | ||||
| } | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| <div class="btn-group w-100" ngbDropdown role="group" (openChange)="dropdownOpenChange($event)" #dropdown="ngbDropdown" (keydown)="listKeyDown($event)"> | ||||
| <div class="btn-group w-100" ngbDropdown role="group" (openChange)="dropdownOpenChange($event)" #dropdown="ngbDropdown" (keydown)="listKeyDown($event)" [popperOptions]="popperOptions"> | ||||
|   <button class="btn btn-sm" id="dropdown_{{name}}" ngbDropdownToggle [ngClass]="!editing && selectionModel.selectionSize() > 0 ? 'btn-primary' : 'btn-outline-primary'" [disabled]="disabled"> | ||||
|     <i-bs name="{{icon}}"></i-bs> | ||||
|     <div class="d-none d-sm-inline"> {{title}}</div> | ||||
|  | ||||
| @ -539,15 +539,10 @@ describe('FilterableDropdownComponent & FilterableDropdownSelectionModel', () => | ||||
|     fixture.nativeElement | ||||
|       .querySelector('button') | ||||
|       .dispatchEvent(new MouseEvent('click')) // open
 | ||||
|     fixture.detectChanges() | ||||
|     tick(100) | ||||
|     component.filterText = 'FooBar' | ||||
|     fixture.detectChanges() | ||||
|     component.listFilterTextInput.nativeElement.dispatchEvent( | ||||
|       new KeyboardEvent('keyup', { key: 'Enter' }) | ||||
|     ) | ||||
|     component.listFilterEnter() | ||||
|     expect(component.selectionModel.getSelectedItems()).toEqual([]) | ||||
|     tick(300) | ||||
|     expect(createSpy).toHaveBeenCalled() | ||||
|   })) | ||||
| 
 | ||||
|  | ||||
| @ -16,6 +16,7 @@ import { Subject, filter, take, takeUntil } from 'rxjs' | ||||
| import { SelectionDataItem } from 'src/app/services/rest/document.service' | ||||
| import { ObjectWithPermissions } from 'src/app/data/object-with-permissions' | ||||
| import { HotKeyService } from 'src/app/services/hot-key.service' | ||||
| import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options' | ||||
| 
 | ||||
| export interface ChangedItems { | ||||
|   itemsToAdd: MatchingModel[] | ||||
| @ -330,6 +331,8 @@ export class FilterableDropdownComponent implements OnDestroy, OnInit { | ||||
|   @ViewChild('dropdown') dropdown: NgbDropdown | ||||
|   @ViewChild('buttonItems') buttonItems: ElementRef | ||||
| 
 | ||||
|   public popperOptions = popperOptionsReenablePreventOverflow | ||||
| 
 | ||||
|   filterText: string | ||||
| 
 | ||||
|   @Input() | ||||
| @ -483,7 +486,7 @@ export class FilterableDropdownComponent implements OnDestroy, OnInit { | ||||
|   dropdownOpenChange(open: boolean): void { | ||||
|     if (open) { | ||||
|       setTimeout(() => { | ||||
|         this.listFilterTextInput.nativeElement.focus() | ||||
|         this.listFilterTextInput?.nativeElement.focus() | ||||
|       }, 0) | ||||
|       if (this.editing) { | ||||
|         this.selectionModel.reset() | ||||
| @ -492,7 +495,7 @@ export class FilterableDropdownComponent implements OnDestroy, OnInit { | ||||
|       this.opened.next(this) | ||||
|     } else { | ||||
|       if (this.creating) { | ||||
|         this.dropdown.open() | ||||
|         this.dropdown?.open() | ||||
|         this.creating = false | ||||
|       } else { | ||||
|         this.filterText = '' | ||||
|  | ||||
| @ -1,50 +1,57 @@ | ||||
| <div class="mb-3 paperless-input-select" [class.disabled]="disabled"> | ||||
|   <div class="row"> | ||||
|     <div class="d-flex align-items-center position-relative hidden-button-container" [class.col-md-3]="horizontal"> | ||||
|       @if (title) { | ||||
|         <label class="form-label" [class.mb-md-0]="horizontal" [for]="inputId">{{title}}</label> | ||||
|       } | ||||
|       @if (removable) { | ||||
|         <button type="button" class="btn btn-sm btn-danger position-absolute left-0" (click)="removed.emit(this)"> | ||||
|           <i-bs name="x"></i-bs> <ng-container i18n>Remove</ng-container> | ||||
|         </button> | ||||
|       } | ||||
|     </div> | ||||
|     <div [class.col-md-9]="horizontal"> | ||||
|       <div> | ||||
|         <ng-select name="inputId" [(ngModel)]="selectedDocuments" | ||||
|           [disabled]="disabled" | ||||
|           [items]="foundDocuments$ | async" | ||||
|           placeholder="Search for documents" | ||||
|           [notFoundText]="notFoundText" | ||||
|           [multiple]="true" | ||||
|           bindValue="id" | ||||
|           [compareWith]="compareDocuments" | ||||
|           [trackByFn]="trackByFn" | ||||
|           [minTermLength]="2" | ||||
|           [loading]="loading" | ||||
|           [typeahead]="documentsInput$" | ||||
|           (change)="onChange(selectedDocuments)"> | ||||
|           <ng-template ng-label-tmp let-document="item"> | ||||
|             <div class="d-flex align-items-center"> | ||||
|               <button class="btn p-0 lh-1" (click)="unselect(document)" title="Remove link" i18n-title><i-bs name="x"></i-bs></button> | ||||
|               <a routerLink="/documents/{{document.id}}" class="badge bg-light text-primary" (mousedown)="$event.stopImmediatePropagation();" title="Open link" i18n-title> | ||||
|                 <i-bs width="0.9em" height="0.9em" name="file-text"></i-bs> <span>{{document.title}}</span> | ||||
|               </a> | ||||
|             </div> | ||||
|           </ng-template> | ||||
|           <ng-template ng-loadingspinner-tmp> | ||||
|             <div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div> | ||||
|             <div class="visually-hidden" i18n>Loading...</div> | ||||
|           </ng-template> | ||||
|           <ng-template ng-option-tmp let-document="item" let-index="index" let-search="searchTerm"> | ||||
|             <div>{{document.title}} <small class="text-muted">({{document.created | customDate:'shortDate'}})</small></div> | ||||
|           </ng-template> | ||||
|         </ng-select> | ||||
| @if (minimal) { | ||||
|   <ng-container *ngTemplateOutlet="select"></ng-container> | ||||
| } @else { | ||||
|   <div class="mb-3 paperless-input-select" [class.disabled]="disabled"> | ||||
|     <div class="row"> | ||||
|       <div class="d-flex align-items-center position-relative hidden-button-container" [class.col-md-3]="horizontal"> | ||||
|         @if (title) { | ||||
|           <label class="form-label" [class.mb-md-0]="horizontal" [for]="inputId">{{title}}</label> | ||||
|         } | ||||
|         @if (removable) { | ||||
|           <button type="button" class="btn btn-sm btn-danger position-absolute left-0" (click)="removed.emit(this)"> | ||||
|             <i-bs name="x"></i-bs> <ng-container i18n>Remove</ng-container> | ||||
|           </button> | ||||
|         } | ||||
|       </div> | ||||
|       <div [class.col-md-9]="horizontal"> | ||||
|         <ng-container *ngTemplateOutlet="select"></ng-container> | ||||
|         @if (hint) { | ||||
|           <small class="form-text text-muted">{{hint}}</small> | ||||
|         } | ||||
|       </div> | ||||
|       @if (hint) { | ||||
|         <small class="form-text text-muted">{{hint}}</small> | ||||
|       } | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| } | ||||
| 
 | ||||
| <ng-template #select> | ||||
|   <ng-select name="inputId" [(ngModel)]="selectedDocuments" | ||||
|     [disabled]="disabled" | ||||
|     [items]="foundDocuments$ | async" | ||||
|     [placeholder]="placeholder" | ||||
|     [notFoundText]="notFoundText" | ||||
|     [multiple]="true" | ||||
|     bindValue="id" | ||||
|     [compareWith]="compareDocuments" | ||||
|     [trackByFn]="trackByFn" | ||||
|     [minTermLength]="2" | ||||
|     [loading]="loading" | ||||
|     [typeahead]="documentsInput$" | ||||
|     (mousedown)="$event.stopImmediatePropagation()" | ||||
|     (change)="onChange(selectedDocuments)"> | ||||
|     <ng-template ng-label-tmp let-document="item"> | ||||
|       <div class="d-flex align-items-center"> | ||||
|         <button class="btn p-0 lh-1" *ngIf="!disabled" (click)="unselect(document)" title="Remove link" i18n-title><i-bs name="x"></i-bs></button> | ||||
|         <a routerLink="/documents/{{document.id}}" class="badge bg-light text-primary" (mousedown)="$event.stopImmediatePropagation();" title="Open link" i18n-title> | ||||
|           <i-bs width="0.9em" height="0.9em" name="file-text"></i-bs> <span>{{document.title}}</span> | ||||
|         </a> | ||||
|       </div> | ||||
|     </ng-template> | ||||
|     <ng-template ng-loadingspinner-tmp> | ||||
|       <div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div> | ||||
|       <div class="visually-hidden" i18n>Loading...</div> | ||||
|     </ng-template> | ||||
|     <ng-template ng-option-tmp let-document="item" let-index="index" let-search="searchTerm"> | ||||
|       <div>{{document.title}} <small class="text-muted">({{document.created | customDate:'shortDate'}})</small></div> | ||||
|     </ng-template> | ||||
|   </ng-select> | ||||
| </ng-template> | ||||
|  | ||||
| @ -3,7 +3,19 @@ | ||||
| 
 | ||||
|     .ng-value { | ||||
|         background-color: transparent !important; | ||||
|         border-color: transparent; | ||||
|         border-color: transparent !important; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| .paperless-input-select.disabled { | ||||
|     --bs-btn-disabled-border-color: transparent; | ||||
|     ::ng-deep ng-select { | ||||
|         .ng-select-container { | ||||
|             div, .ng-arrow-wrapper, input { | ||||
|                 cursor: not-allowed; | ||||
|             } | ||||
|             background-color: var(--pngx-bg-alt) !important; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -46,6 +46,12 @@ export class DocumentLinkComponent | ||||
|   @Input() | ||||
|   parentDocumentID: number | ||||
| 
 | ||||
|   @Input() | ||||
|   minimal: boolean = false | ||||
| 
 | ||||
|   @Input() | ||||
|   placeholder: string = $localize`Search for documents` | ||||
| 
 | ||||
|   constructor(private documentsService: DocumentService) { | ||||
|     super() | ||||
|   } | ||||
|  | ||||
| @ -0,0 +1,33 @@ | ||||
| <div class="mb-3" [class.pb-3]="error"> | ||||
|   <div class="row"> | ||||
|     <div class="d-flex align-items-center position-relative hidden-button-container" [class.col-md-3]="horizontal"> | ||||
|       @if (title) { | ||||
|         <label class="form-label" [class.mb-md-0]="horizontal" [for]="inputId">{{title}}</label> | ||||
|       } | ||||
|       @if (removable) { | ||||
|         <button type="button" class="btn btn-sm btn-danger position-absolute left-0" (click)="removed.emit(this)"> | ||||
|           <i-bs name="x"></i-bs> <ng-container i18n>Remove</ng-container> | ||||
|           </button> | ||||
|         } | ||||
|       </div> | ||||
|       <div class="position-relative" [class.col-md-9]="horizontal"> | ||||
|         <textarea #inputField | ||||
|           [id]="inputId" | ||||
|           class="form-control" | ||||
|           [class.is-invalid]="error" | ||||
|           [class.font-monospace]="monospace" | ||||
|           [(ngModel)]="value" | ||||
|           (change)="onChange(value)" | ||||
|           [disabled]="disabled" | ||||
|           [placeholder]="placeholder" | ||||
|           rows="4"> | ||||
|         </textarea> | ||||
|         @if (hint) { | ||||
|           <small class="form-text text-muted" [innerHTML]="hint | safeHtml"></small> | ||||
|         } | ||||
|         <div class="invalid-feedback position-absolute top-100"> | ||||
|           {{error}} | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| @ -0,0 +1,31 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing' | ||||
| import { | ||||
|   FormsModule, | ||||
|   ReactiveFormsModule, | ||||
|   NG_VALUE_ACCESSOR, | ||||
| } from '@angular/forms' | ||||
| import { TextAreaComponent } from './textarea.component' | ||||
| 
 | ||||
| describe('TextComponent', () => { | ||||
|   let component: TextAreaComponent | ||||
|   let fixture: ComponentFixture<TextAreaComponent> | ||||
|   let input: HTMLTextAreaElement | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     TestBed.configureTestingModule({ | ||||
|       declarations: [TextAreaComponent], | ||||
|       providers: [], | ||||
|       imports: [FormsModule, ReactiveFormsModule], | ||||
|     }).compileComponents() | ||||
| 
 | ||||
|     fixture = TestBed.createComponent(TextAreaComponent) | ||||
|     fixture.debugElement.injector.get(NG_VALUE_ACCESSOR) | ||||
|     component = fixture.componentInstance | ||||
|     fixture.detectChanges() | ||||
|     input = component.inputField.nativeElement | ||||
|   }) | ||||
| 
 | ||||
|   it('should support use of input field', () => { | ||||
|     expect(component.value).toBeUndefined() | ||||
|   }) | ||||
| }) | ||||
| @ -0,0 +1,27 @@ | ||||
| import { Component, Input, forwardRef } from '@angular/core' | ||||
| import { NG_VALUE_ACCESSOR } from '@angular/forms' | ||||
| import { AbstractInputComponent } from '../abstract-input' | ||||
| 
 | ||||
| @Component({ | ||||
|   providers: [ | ||||
|     { | ||||
|       provide: NG_VALUE_ACCESSOR, | ||||
|       useExisting: forwardRef(() => TextAreaComponent), | ||||
|       multi: true, | ||||
|     }, | ||||
|   ], | ||||
|   selector: 'pngx-input-textarea', | ||||
|   templateUrl: './textarea.component.html', | ||||
|   styleUrls: ['./textarea.component.scss'], | ||||
| }) | ||||
| export class TextAreaComponent extends AbstractInputComponent<string> { | ||||
|   @Input() | ||||
|   placeholder: string = '' | ||||
| 
 | ||||
|   @Input() | ||||
|   monospace: boolean = false | ||||
| 
 | ||||
|   constructor() { | ||||
|     super() | ||||
|   } | ||||
| } | ||||
| @ -7,3 +7,32 @@ | ||||
| ::ng-deep .popover.popover-preview { | ||||
|     max-width: 32rem; | ||||
| } | ||||
| 
 | ||||
| // https://github.com/paperless-ngx/paperless-ngx/issues/7920 | ||||
| // TODO: remove me | ||||
| @mixin ff_txt { | ||||
|   .preview-popup-container { | ||||
|     width: 30rem !important; | ||||
|     height: 22rem !important; | ||||
|     background-color: #e7e7e7; | ||||
|   } | ||||
| 
 | ||||
|   object { | ||||
|     mix-blend-mode: difference; | ||||
|     &.p-2 { | ||||
|       padding: 0 !important; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @-moz-document url-prefix() { | ||||
|   html[data-bs-theme='dark'] { | ||||
|     @include ff_txt; | ||||
|   } | ||||
|   html[data-bs-theme='auto'] { | ||||
|     @media screen and (prefers-color-scheme: dark) { | ||||
|       @include ff_txt; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -65,6 +65,7 @@ const savedView: SavedView = { | ||||
|     DisplayField.CORRESPONDENT, | ||||
|     DisplayField.DOCUMENT_TYPE, | ||||
|     DisplayField.STORAGE_PATH, | ||||
|     DisplayField.PAGE_COUNT, | ||||
|     `${DisplayField.CUSTOM_FIELD}11` as any, | ||||
|     `${DisplayField.CUSTOM_FIELD}15` as any, | ||||
|   ], | ||||
| @ -344,6 +345,7 @@ describe('SavedViewWidgetComponent', () => { | ||||
|     expect(component.getColumnTitle(DisplayField.STORAGE_PATH)).toEqual( | ||||
|       'Storage path' | ||||
|     ) | ||||
|     expect(component.getColumnTitle(DisplayField.PAGE_COUNT)).toEqual('Pages') | ||||
|   }) | ||||
| 
 | ||||
|   it('should get correct column title for custom field', () => { | ||||
|  | ||||
| @ -344,8 +344,8 @@ | ||||
|       @if (!hasNext()) { | ||||
|         <button type="button" class="order-2 btn btn-sm btn-outline-primary" (click)="save(true)" i18n [disabled]="!userCanEdit || networkActive || (isDirty$ | async) !== true">Save & close</button> | ||||
|       } | ||||
|       <button type="button" class="order-0 btn btn-sm btn-outline-secondary" (click)="discard()" i18n [disabled]="!userCanEdit || networkActive || (isDirty$ | async) !== true">Discard</button> | ||||
|     </ng-container> | ||||
|     <button type="button" class="order-0 btn btn-sm btn-outline-secondary" (click)="discard()" i18n [disabled]="!userCanEdit || networkActive || (isDirty$ | async) !== true">Discard</button> | ||||
|   </div> | ||||
| </ng-template> | ||||
| 
 | ||||
| @ -379,7 +379,7 @@ | ||||
|         } | ||||
|       } | ||||
|       @case (ContentRenderType.Text) { | ||||
|         <div class="preview-sticky bg-light p-3 overflow-auto" width="100%">{{previewText}}</div> | ||||
|         <div class="preview-sticky bg-light p-3 overflow-auto whitespace-preserve" width="100%">{{previewText}}</div> | ||||
|       } | ||||
|       @case (ContentRenderType.Image) { | ||||
|         <div class="preview-sticky"> | ||||
|  | ||||
| @ -62,3 +62,7 @@ textarea.rtl { | ||||
|   height: 100%; | ||||
|   object-fit: contain; | ||||
| } | ||||
| 
 | ||||
| .whitespace-preserve { | ||||
|   white-space: preserve; | ||||
| } | ||||
|  | ||||
| @ -85,6 +85,7 @@ import { PdfViewerModule } from 'ng2-pdf-viewer' | ||||
| import { DataType } from 'src/app/data/datatype' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { TagService } from 'src/app/services/rest/tag.service' | ||||
| import { TextAreaComponent } from '../common/input/textarea/textarea.component' | ||||
| 
 | ||||
| const doc: Document = { | ||||
|   id: 3, | ||||
| @ -183,6 +184,7 @@ describe('DocumentDetailComponent', () => { | ||||
|         SplitConfirmDialogComponent, | ||||
|         RotateConfirmDialogComponent, | ||||
|         DeletePagesConfirmDialogComponent, | ||||
|         TextAreaComponent, | ||||
|       ], | ||||
|       imports: [ | ||||
|         RouterModule.forRoot(routes), | ||||
|  | ||||
| @ -327,13 +327,8 @@ export class DocumentDetailComponent | ||||
|         switchMap((paramMap) => { | ||||
|           const documentId = +paramMap.get('id') | ||||
|           this.docChangeNotifier.next(documentId) | ||||
|           return this.documentsService.get(documentId) | ||||
|         }) | ||||
|       ) | ||||
|       .pipe( | ||||
|         switchMap((doc) => { | ||||
|           this.documentId = doc.id | ||||
|           this.previewUrl = this.documentsService.getPreviewUrl(this.documentId) | ||||
|           // Dont wait to get the preview
 | ||||
|           this.previewUrl = this.documentsService.getPreviewUrl(documentId) | ||||
|           this.http.get(this.previewUrl, { responseType: 'text' }).subscribe({ | ||||
|             next: (res) => { | ||||
|               this.previewText = res.toString() | ||||
| @ -344,6 +339,12 @@ export class DocumentDetailComponent | ||||
|               }` | ||||
|             }, | ||||
|           }) | ||||
|           return this.documentsService.get(documentId) | ||||
|         }) | ||||
|       ) | ||||
|       .pipe( | ||||
|         switchMap((doc) => { | ||||
|           this.documentId = doc.id | ||||
|           this.downloadUrl = this.documentsService.getDownloadUrl( | ||||
|             this.documentId | ||||
|           ) | ||||
| @ -1019,10 +1020,14 @@ export class DocumentDetailComponent | ||||
|     } | ||||
|     return ( | ||||
|       !this.document || | ||||
|       this.permissionsService.currentUserHasObjectPermissions( | ||||
|       (this.permissionsService.currentUserCan( | ||||
|         PermissionAction.Change, | ||||
|         doc | ||||
|       ) | ||||
|         PermissionType.Document | ||||
|       ) && | ||||
|         this.permissionsService.currentUserHasObjectPermissions( | ||||
|           PermissionAction.Change, | ||||
|           doc | ||||
|         )) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -21,7 +21,7 @@ | ||||
|               } @else { | ||||
|                 {{(document.correspondent$ | async)?.name}} | ||||
|               } | ||||
|               : | ||||
|               @if (displayFields.includes(DisplayField.TITLE)) {:} | ||||
|             } | ||||
|             @if (displayFields.includes(DisplayField.TITLE)) { | ||||
|               {{document.title | documentTitle}} | ||||
| @ -54,7 +54,7 @@ | ||||
|               <i-bs name="diagram-3"></i-bs> <span class="d-none d-md-inline" i18n>More like this</span> | ||||
|             </a> | ||||
|             <a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Document }"> | ||||
|               <i-bs name="pencil"></i-bs> <span class="d-none d-md-inline" i18n>Edit</span> | ||||
|               <i-bs name="box-arrow-in-right"></i-bs> <span class="d-none d-md-inline" i18n>Open</span> | ||||
|             </a> | ||||
|             <a class="btn btn-sm btn-outline-secondary" target="_blank" [href]="previewUrl" | ||||
|               [ngbPopover]="previewContent" [popoverTitle]="document.title | documentTitle" | ||||
| @ -111,6 +111,12 @@ | ||||
|                   </div> | ||||
|                 } | ||||
|               } | ||||
|               @if (displayFields.includes(DisplayField.PAGE_COUNT) && document.page_count) { | ||||
|                 <div class="list-group-item bg-light text-dark p-1 border-0 d-flex align-items-center"> | ||||
|                   <i-bs width=".9em" height=".9em" class="me-2 text-muted" name="files"></i-bs> | ||||
|                   <small i18n>{document.page_count, plural, =1 {1 page} other {{{document.page_count}} pages}}</small> | ||||
|                 </div> | ||||
|               } | ||||
|               @if (displayFields.includes(DisplayField.OWNER) && document.owner && document.owner !== settingsService.currentUser.id) { | ||||
|                 <div class="list-group-item bg-light text-dark p-1 border-0 d-flex align-items-center"> | ||||
|                   <i-bs width=".9em" height=".9em" class="me-2 text-muted" name="person-fill-lock"></i-bs><small>{{document.owner | username}}</small> | ||||
|  | ||||
| @ -31,6 +31,7 @@ const doc = { | ||||
|   correspondent: 8, | ||||
|   document_type: 10, | ||||
|   storage_path: null, | ||||
|   page_count: 8, | ||||
|   notes: [ | ||||
|     { | ||||
|       id: 11, | ||||
| @ -80,6 +81,7 @@ describe('DocumentCardLargeComponent', () => { | ||||
|   it('should display a document', () => { | ||||
|     expect(fixture.nativeElement.textContent).toContain('Document 10') | ||||
|     expect(fixture.nativeElement.textContent).toContain('Cupcake ipsum') | ||||
|     expect(fixture.nativeElement.textContent).toContain('8 pages') | ||||
|   }) | ||||
| 
 | ||||
|   it('should show preview on mouseover after delay to preload content', fakeAsync(() => { | ||||
|  | ||||
| @ -35,7 +35,8 @@ | ||||
|     <div class="card-body bg-light p-2"> | ||||
|       <p class="card-text"> | ||||
|         @if (displayFields.includes(DisplayField.CORRESPONDENT) && document.correspondent) { | ||||
|           <a title="Toggle correspondent filter" i18n-title (click)="clickCorrespondent.emit(document.correspondent);$event.stopPropagation()" class="fw-bold btn-link">{{(document.correspondent$ | async)?.name ?? privateName}}</a>: | ||||
|           <a title="Toggle correspondent filter" i18n-title (click)="clickCorrespondent.emit(document.correspondent);$event.stopPropagation()" class="fw-bold btn-link">{{(document.correspondent$ | async)?.name ?? privateName}}</a> | ||||
|           @if (displayFields.includes(DisplayField.TITLE)) {:} | ||||
|         } | ||||
|         @if (displayFields.includes(DisplayField.TITLE)) { | ||||
|           {{document.title | documentTitle}} | ||||
| @ -88,6 +89,14 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|         } | ||||
|         @if (displayFields.includes(DisplayField.PAGE_COUNT) && document.page_count) { | ||||
|           <div class="list-group-item bg-transparent p-0 border-0 d-flex flex-wrap-reverse justify-content-between"> | ||||
|             <div class="ps-0 p-1" placement="top"> | ||||
|               <i-bs width="1em" height="1em" class="me-2 text-muted" name="files"></i-bs> | ||||
|               <small i18n>{document.page_count, plural, =1 {1 page} other {{{document.page_count}} pages}}</small> | ||||
|             </div> | ||||
|           </div> | ||||
|         } | ||||
|         @if (displayFields.includes(DisplayField.ASN) && document.archive_serial_number | isNumber) { | ||||
|           <div class="ps-0 p-1"> | ||||
|             <i-bs width="1em" height="1em" class="me-2 text-muted" name="upc-scan"></i-bs> | ||||
| @ -117,8 +126,8 @@ | ||||
|       </div> | ||||
|       <div class="d-flex justify-content-between align-items-center"> | ||||
|         <div class="btn-group w-100"> | ||||
|           <a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary" title="Edit" i18n-title *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Document }" i18n-title> | ||||
|             <i-bs name="pencil"></i-bs> | ||||
|           <a routerLink="/documents/{{document.id}}" class="btn btn-sm btn-outline-secondary" title="Open" i18n-title *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Document }" i18n-title> | ||||
|             <i-bs name="box-arrow-in-right"></i-bs> | ||||
|           </a> | ||||
|           <a [href]="previewUrl" target="_blank" class="btn btn-sm btn-outline-secondary" | ||||
|             [ngbPopover]="previewContent" [popoverTitle]="document.title | documentTitle" | ||||
|  | ||||
| @ -34,6 +34,7 @@ const doc = { | ||||
|   correspondent: 8, | ||||
|   document_type: 10, | ||||
|   storage_path: null, | ||||
|   page_count: 12, | ||||
|   notes: [ | ||||
|     { | ||||
|       id: 11, | ||||
| @ -91,6 +92,10 @@ describe('DocumentCardSmallComponent', () => { | ||||
|     fixture.detectChanges() | ||||
|   }) | ||||
| 
 | ||||
|   it('should display page count', () => { | ||||
|     expect(fixture.nativeElement.textContent).toContain('12 pages') | ||||
|   }) | ||||
| 
 | ||||
|   it('should display a document, limit tags to 5', () => { | ||||
|     expect(fixture.nativeElement.textContent).toContain('Document 10') | ||||
|     expect( | ||||
|  | ||||
| @ -140,7 +140,7 @@ | ||||
|   } @else { | ||||
|     @if (list.displayMode === DisplayMode.LARGE_CARDS) { | ||||
|       <div> | ||||
|         @for (d of list.documents; track trackByDocumentId($index, d)) { | ||||
|         @for (d of list.documents; track d.id) { | ||||
|           <pngx-document-card-large | ||||
|             [selected]="list.isSelected(d)" | ||||
|             (toggleSelected)="toggleSelected(d, $event)" | ||||
| @ -160,105 +160,116 @@ | ||||
|       <div class="table-responsive"> | ||||
|         <table class="table table-sm align-middle border shadow-sm"> | ||||
|           <thead> | ||||
|             <th></th> | ||||
|             @if (activeDisplayFields.includes(DisplayField.ASN)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="archive_serial_number" | ||||
|                 title="Sort by ASN" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>ASN</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.CORRESPONDENT) && permissionService.currentUserCan(PermissionAction.View, PermissionType.Correspondent)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="correspondent__name" | ||||
|                 title="Sort by correspondent" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Correspondent</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.TITLE)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="title" | ||||
|                 title="Sort by title" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 style="min-width: 150px;" | ||||
|                 i18n>Title</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.TAGS) && !activeDisplayFields.includes(DisplayField.TITLE)) { | ||||
|               <th i18n>Tags</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.OWNER) && permissionService.currentUserCan(PermissionAction.View, PermissionType.User)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="owner" | ||||
|                 title="Sort by owner" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Owner</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.NOTES) && notesEnabled) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="num_notes" | ||||
|                 title="Sort by notes" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Notes</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.DOCUMENT_TYPE) && permissionService.currentUserCan(PermissionAction.View, PermissionType.DocumentType)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="document_type__name" | ||||
|                 title="Sort by document type" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Document type</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.STORAGE_PATH) && permissionService.currentUserCan(PermissionAction.View, PermissionType.StoragePath)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="storage_path__name" | ||||
|                 title="Sort by storage path" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Storage path</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.CREATED)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="created" | ||||
|                 title="Sort by created date" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Created</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.ADDED)) { | ||||
|               <th class="cursor-pointer" | ||||
|                 pngxSortable="added" | ||||
|                 title="Sort by added date" i18n-title | ||||
|                 [currentSortField]="list.sortField" | ||||
|                 [currentSortReverse]="list.sortReverse" | ||||
|                 (sort)="onSort($event)" | ||||
|                 i18n>Added</th> | ||||
|             } | ||||
|             @if (activeDisplayFields.includes(DisplayField.SHARED)) { | ||||
|               <th i18n> | ||||
|                 Shared | ||||
|               </th> | ||||
|             } | ||||
|             @for (field of activeDisplayCustomFields; track field) { | ||||
|               <th> | ||||
|                 {{getDisplayCustomFieldTitle(field)}} | ||||
|               </th> | ||||
|             } | ||||
|             <tr> | ||||
|               <th></th> | ||||
|               @if (activeDisplayFields.includes(DisplayField.ASN)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="archive_serial_number" | ||||
|                   title="Sort by ASN" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>ASN</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.CORRESPONDENT) && permissionService.currentUserCan(PermissionAction.View, PermissionType.Correspondent)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="correspondent__name" | ||||
|                   title="Sort by correspondent" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Correspondent</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.TITLE)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="title" | ||||
|                   title="Sort by title" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   style="min-width: 150px;" | ||||
|                   i18n>Title</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.TAGS) && !activeDisplayFields.includes(DisplayField.TITLE)) { | ||||
|                 <th i18n>Tags</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.OWNER) && permissionService.currentUserCan(PermissionAction.View, PermissionType.User)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="owner" | ||||
|                   title="Sort by owner" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Owner</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.NOTES) && notesEnabled) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="num_notes" | ||||
|                   title="Sort by notes" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Notes</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.DOCUMENT_TYPE) && permissionService.currentUserCan(PermissionAction.View, PermissionType.DocumentType)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="document_type__name" | ||||
|                   title="Sort by document type" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Document type</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.STORAGE_PATH) && permissionService.currentUserCan(PermissionAction.View, PermissionType.StoragePath)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="storage_path__name" | ||||
|                   title="Sort by storage path" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Storage path</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.CREATED)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="created" | ||||
|                   title="Sort by created date" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Created</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.ADDED)) { | ||||
|                 <th class="cursor-pointer" | ||||
|                   pngxSortable="added" | ||||
|                   title="Sort by added date" i18n-title | ||||
|                   [currentSortField]="list.sortField" | ||||
|                   [currentSortReverse]="list.sortReverse" | ||||
|                   (sort)="onSort($event)" | ||||
|                   i18n>Added</th> | ||||
|               } | ||||
|               @if (activeDisplayFields.includes(DisplayField.PAGE_COUNT)) { | ||||
|                   <th class="cursor-pointer" | ||||
|                     pngxSortable="page_count" | ||||
|                     title="Sort by number of pages" i18n-title | ||||
|                     [currentSortField]="list.sortField" | ||||
|                     [currentSortReverse]="list.sortReverse" | ||||
|                     (sort)="onSort($event)" | ||||
|                     i18n>Pages</th> | ||||
|                 } | ||||
|               @if (activeDisplayFields.includes(DisplayField.SHARED)) { | ||||
|                 <th i18n> | ||||
|                   Shared | ||||
|                 </th> | ||||
|               } | ||||
|               @for (field of activeDisplayCustomFields; track field) { | ||||
|                 <th> | ||||
|                   {{getDisplayCustomFieldTitle(field)}} | ||||
|                 </th> | ||||
|               } | ||||
|             </tr> | ||||
|           </thead> | ||||
|           <tbody> | ||||
|             @for (d of list.documents; track trackByDocumentId($index, d)) { | ||||
|             @for (d of list.documents; track d.id) { | ||||
|               <tr (click)="toggleSelected(d, $event); $event.stopPropagation();" (dblclick)="openDocumentDetail(d)" [ngClass]="list.isSelected(d) ? 'table-row-selected' : ''"> | ||||
|                 <td> | ||||
|                   <div class="form-check"> | ||||
| @ -330,6 +341,11 @@ | ||||
|                     {{d.added | customDate}} | ||||
|                   </td> | ||||
|                 } | ||||
|                 @if (activeDisplayFields.includes(DisplayField.PAGE_COUNT)) { | ||||
|                     <td> | ||||
|                         {{ d.page_count }} | ||||
|                     </td> | ||||
|                   } | ||||
|                 @if (activeDisplayFields.includes(DisplayField.SHARED)) { | ||||
|                   <td> | ||||
|                     @if (d.is_shared_by_requester) { <ng-container i18n>Yes</ng-container> } @else { <ng-container i18n>No</ng-container> } | ||||
| @ -348,7 +364,7 @@ | ||||
|     } | ||||
|     @if (list.displayMode === DisplayMode.SMALL_CARDS) { | ||||
|       <div class="row row-cols-paperless-cards"> | ||||
|         @for (d of list.documents; track trackByDocumentId($index, d)) { | ||||
|         @for (d of list.documents; track d.id) { | ||||
|           <pngx-document-card-small class="p-0" | ||||
|             [selected]="list.isSelected(d)" | ||||
|             (toggleSelected)="toggleSelected(d, $event)" | ||||
|  | ||||
| @ -6,10 +6,6 @@ tr { | ||||
|   user-select: none; | ||||
| } | ||||
| 
 | ||||
| .cursor-pointer { | ||||
|   cursor: pointer; | ||||
| } | ||||
| 
 | ||||
| .table-row-selected { | ||||
|   background-color: var(--pngx-primary-faded); | ||||
| } | ||||
|  | ||||
| @ -302,7 +302,7 @@ describe('DocumentListComponent', () => { | ||||
|     displayModeButtons[0].triggerEventHandler('change') | ||||
|     fixture.detectChanges() | ||||
|     expect(component.list.displayMode).toEqual('table') | ||||
|     expect(fixture.debugElement.queryAll(By.css('tr'))).toHaveLength(3) | ||||
|     expect(fixture.debugElement.queryAll(By.css('tr'))).toHaveLength(4) | ||||
| 
 | ||||
|     displayModeButtons[1].nativeElement.checked = true | ||||
|     displayModeButtons[1].triggerEventHandler('change') | ||||
| @ -602,7 +602,7 @@ describe('DocumentListComponent', () => { | ||||
| 
 | ||||
|     expect( | ||||
|       fixture.debugElement.queryAll(By.directive(SortableDirective)) | ||||
|     ).toHaveLength(9) | ||||
|     ).toHaveLength(10) | ||||
| 
 | ||||
|     expect(component.notesEnabled).toBeTruthy() | ||||
|     settingsService.set(SETTINGS_KEYS.NOTES_ENABLED, false) | ||||
| @ -610,14 +610,14 @@ describe('DocumentListComponent', () => { | ||||
|     expect(component.notesEnabled).toBeFalsy() | ||||
|     expect( | ||||
|       fixture.debugElement.queryAll(By.directive(SortableDirective)) | ||||
|     ).toHaveLength(8) | ||||
|     ).toHaveLength(9) | ||||
| 
 | ||||
|     // insufficient perms
 | ||||
|     jest.spyOn(permissionService, 'currentUserCan').mockReturnValue(false) | ||||
|     fixture.detectChanges() | ||||
|     expect( | ||||
|       fixture.debugElement.queryAll(By.directive(SortableDirective)) | ||||
|     ).toHaveLength(4) | ||||
|     ).toHaveLength(5) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support toggle on document objects', () => { | ||||
|  | ||||
| @ -15,7 +15,12 @@ import { | ||||
|   isFullTextFilterRule, | ||||
| } from 'src/app/utils/filter-rules' | ||||
| import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type' | ||||
| import { DisplayField, DisplayMode, Document } from 'src/app/data/document' | ||||
| import { | ||||
|   DEFAULT_DISPLAY_FIELDS, | ||||
|   DisplayField, | ||||
|   DisplayMode, | ||||
|   Document, | ||||
| } from 'src/app/data/document' | ||||
| import { SavedView } from 'src/app/data/saved-view' | ||||
| import { SETTINGS_KEYS } from 'src/app/data/ui-settings' | ||||
| import { | ||||
| @ -108,6 +113,11 @@ export class DocumentListComponent | ||||
|         (this.unmodifiedSavedView.display_fields && | ||||
|           this.unmodifiedSavedView.display_fields.join(',') !== | ||||
|             this.activeDisplayFields.join(',')) || | ||||
|         (!this.unmodifiedSavedView.display_fields && | ||||
|           this.activeDisplayFields.join(',') !== | ||||
|             DEFAULT_DISPLAY_FIELDS.filter((f) => f.id !== DisplayField.ADDED) | ||||
|               .map((f) => f.id) | ||||
|               .join(',')) || | ||||
|         filterRulesDiffer( | ||||
|           this.unmodifiedSavedView.filter_rules, | ||||
|           this.list.filterRules | ||||
| @ -383,10 +393,6 @@ export class DocumentListComponent | ||||
|     ]) | ||||
|   } | ||||
| 
 | ||||
|   trackByDocumentId(index, item: Document) { | ||||
|     return item.id | ||||
|   } | ||||
| 
 | ||||
|   get notesEnabled(): boolean { | ||||
|     return this.settingsService.get(SETTINGS_KEYS.NOTES_ENABLED) | ||||
|   } | ||||
|  | ||||
| @ -86,15 +86,10 @@ | ||||
|         } | ||||
| 
 | ||||
|         @if (permissionsService.currentUserCan(PermissionAction.View, PermissionType.CustomField) && customFields.length > 0) { | ||||
|           <pngx-filterable-dropdown class="flex-fill" title="Custom fields" icon="ui-radios" i18n-title | ||||
|           filterPlaceholder="Filter custom fields" i18n-filterPlaceholder | ||||
|           [items]="customFields" | ||||
|           [manyToOne]="true" | ||||
|           [(selectionModel)]="customFieldSelectionModel" | ||||
|           <pngx-custom-fields-query-dropdown class="flex-fill" title="Custom fields" icon="ui-radios" i18n-title | ||||
|           [(selectionModel)]="customFieldQueriesModel" | ||||
|           (selectionModelChange)="updateRules()" | ||||
|           (opened)="onCustomFieldsDropdownOpen()" | ||||
|           [documentCounts]="customFieldDocumentCounts" | ||||
|           [allowSelectNone]="true"></pngx-filterable-dropdown> | ||||
|           ></pngx-custom-fields-query-dropdown> | ||||
|         } | ||||
|         <pngx-dates-dropdown | ||||
|           title="Dates" i18n-title | ||||
|  | ||||
| @ -17,7 +17,7 @@ import { | ||||
|   NgbDropdownItem, | ||||
|   NgbTypeaheadModule, | ||||
| } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgSelectComponent } from '@ng-select/ng-select' | ||||
| import { NgSelectComponent, NgSelectModule } from '@ng-select/ng-select' | ||||
| import { of, throwError } from 'rxjs' | ||||
| import { | ||||
|   FILTER_TITLE, | ||||
| @ -55,6 +55,7 @@ import { | ||||
|   FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|   FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|   FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|   FILTER_CUSTOM_FIELDS_QUERY, | ||||
| } from 'src/app/data/filter-rule-type' | ||||
| import { Correspondent } from 'src/app/data/correspondent' | ||||
| import { DocumentType } from 'src/app/data/document-type' | ||||
| @ -95,6 +96,15 @@ import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service | ||||
| import { RouterModule } from '@angular/router' | ||||
| import { SearchService } from 'src/app/services/rest/search.service' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { CustomFieldsQueryDropdownComponent } from '../../common/custom-fields-query-dropdown/custom-fields-query-dropdown.component' | ||||
| import { | ||||
|   CustomFieldQueryLogicalOperator, | ||||
|   CustomFieldQueryOperator, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| import { | ||||
|   CustomFieldQueryAtom, | ||||
|   CustomFieldQueryExpression, | ||||
| } from 'src/app/utils/custom-field-query-element' | ||||
| 
 | ||||
| const tags: Tag[] = [ | ||||
|   { | ||||
| @ -181,6 +191,7 @@ describe('FilterEditorComponent', () => { | ||||
|         ToggleableDropdownButtonComponent, | ||||
|         DatesDropdownComponent, | ||||
|         CustomDatePipe, | ||||
|         CustomFieldsQueryDropdownComponent, | ||||
|       ], | ||||
|       imports: [ | ||||
|         RouterModule, | ||||
| @ -190,6 +201,7 @@ describe('FilterEditorComponent', () => { | ||||
|         NgbDatepickerModule, | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|         NgbTypeaheadModule, | ||||
|         NgSelectModule, | ||||
|       ], | ||||
|       providers: [ | ||||
|         FilterPipe, | ||||
| @ -838,108 +850,79 @@ describe('FilterEditorComponent', () => { | ||||
|     ] | ||||
|   })) | ||||
| 
 | ||||
|   it('should ingest filter rules for has all custom fields', fakeAsync(() => { | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toHaveLength( | ||||
|       0 | ||||
|     ) | ||||
|   it('should ingest filter rules for custom fields all', fakeAsync(() => { | ||||
|     expect(component.customFieldQueriesModel.isEmpty()).toBeTruthy() | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: '42', | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: '43', | ||||
|         value: '42,43', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldSelectionModel.logicalOperator).toEqual( | ||||
|       LogicalOperator.And | ||||
|     expect(component.customFieldQueriesModel.queries[0].operator).toEqual( | ||||
|       CustomFieldQueryLogicalOperator.And | ||||
|     ) | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toEqual( | ||||
|       custom_fields | ||||
|     ) | ||||
|     // coverage
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: null, | ||||
|       }, | ||||
|     ] | ||||
|     component.toggleTag(2) // coverage
 | ||||
|     expect(component.customFieldQueriesModel.queries[0].value.length).toEqual(2) | ||||
|     expect( | ||||
|       ( | ||||
|         component.customFieldQueriesModel.queries[0] | ||||
|           .value[0] as CustomFieldQueryAtom | ||||
|       ).serialize() | ||||
|     ).toEqual(['42', CustomFieldQueryOperator.Exists, 'true']) | ||||
|   })) | ||||
| 
 | ||||
|   it('should ingest filter rules for has any custom fields', fakeAsync(() => { | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toHaveLength( | ||||
|       0 | ||||
|     ) | ||||
|     expect(component.customFieldQueriesModel.isEmpty()).toBeTruthy() | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|         value: '42', | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|         value: '43', | ||||
|         value: '42,43', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldSelectionModel.logicalOperator).toEqual( | ||||
|       LogicalOperator.Or | ||||
|     expect(component.customFieldQueriesModel.queries[0].operator).toEqual( | ||||
|       CustomFieldQueryLogicalOperator.Or | ||||
|     ) | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toEqual( | ||||
|       custom_fields | ||||
|     ) | ||||
|     // coverage
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|         value: null, | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldQueriesModel.queries[0].value.length).toEqual(2) | ||||
|     expect( | ||||
|       ( | ||||
|         component.customFieldQueriesModel.queries[0] | ||||
|           .value[0] as CustomFieldQueryAtom | ||||
|       ).serialize() | ||||
|     ).toEqual(['42', CustomFieldQueryOperator.Exists, 'true']) | ||||
|   })) | ||||
| 
 | ||||
|   it('should ingest filter rules for has any custom field', fakeAsync(() => { | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toHaveLength( | ||||
|       0 | ||||
|     ) | ||||
|   it('should ingest filter rules for custom field queries', fakeAsync(() => { | ||||
|     expect(component.customFieldQueriesModel.isEmpty()).toBeTruthy() | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|         value: '1', | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: '["AND", [[42, "exists", "true"],[43, "exists", "true"]]]', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldSelectionModel.getSelectedItems()).toHaveLength( | ||||
|       1 | ||||
|     expect(component.customFieldQueriesModel.queries[0].operator).toEqual( | ||||
|       CustomFieldQueryLogicalOperator.And | ||||
|     ) | ||||
|     expect(component.customFieldSelectionModel.get(null)).toBeTruthy() | ||||
|   })) | ||||
|     expect(component.customFieldQueriesModel.queries[0].value.length).toEqual(2) | ||||
|     expect( | ||||
|       ( | ||||
|         component.customFieldQueriesModel.queries[0] | ||||
|           .value[0] as CustomFieldQueryAtom | ||||
|       ).serialize() | ||||
|     ).toEqual([42, CustomFieldQueryOperator.Exists, 'true']) | ||||
| 
 | ||||
|   it('should ingest filter rules for exclude tag(s)', fakeAsync(() => { | ||||
|     expect(component.customFieldSelectionModel.getExcludedItems()).toHaveLength( | ||||
|       0 | ||||
|     ) | ||||
|     // atom
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|         value: '42', | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|         value: '43', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldSelectionModel.logicalOperator).toEqual( | ||||
|       LogicalOperator.And | ||||
|     ) | ||||
|     expect(component.customFieldSelectionModel.getExcludedItems()).toEqual( | ||||
|       custom_fields | ||||
|     ) | ||||
|     // coverage
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|         value: null, | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: '[42, "exists", "true"]', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.customFieldQueriesModel.queries[0].value.length).toEqual(1) | ||||
|     expect( | ||||
|       ( | ||||
|         component.customFieldQueriesModel.queries[0] | ||||
|           .value[0] as CustomFieldQueryAtom | ||||
|       ).serialize() | ||||
|     ).toEqual([42, CustomFieldQueryOperator.Exists, 'true']) | ||||
|   })) | ||||
| 
 | ||||
|   it('should ingest filter rules for owner', fakeAsync(() => { | ||||
| @ -1453,71 +1436,34 @@ describe('FilterEditorComponent', () => { | ||||
|     ]) | ||||
|   })) | ||||
| 
 | ||||
|   it('should convert user input to correct filter rules on custom field select not assigned', fakeAsync(() => { | ||||
|     const customFieldsFilterableDropdown = fixture.debugElement.queryAll( | ||||
|       By.directive(FilterableDropdownComponent) | ||||
|     )[4] | ||||
|     customFieldsFilterableDropdown.triggerEventHandler('opened') | ||||
|     const customFieldButton = customFieldsFilterableDropdown.queryAll( | ||||
|       By.directive(ToggleableDropdownButtonComponent) | ||||
|     )[0] | ||||
|     customFieldButton.triggerEventHandler('toggle') | ||||
|     fixture.detectChanges() | ||||
|     expect(component.filterRules).toEqual([ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|         value: 'false', | ||||
|       }, | ||||
|     ]) | ||||
|   })) | ||||
| 
 | ||||
|   it('should convert user input to correct filter rules on custom field selections', fakeAsync(() => { | ||||
|     const customFieldsFilterableDropdown = fixture.debugElement.queryAll( | ||||
|       By.directive(FilterableDropdownComponent) | ||||
|     )[4] // CF dropdown
 | ||||
|     customFieldsFilterableDropdown.triggerEventHandler('opened') | ||||
|     const customFieldButtons = customFieldsFilterableDropdown.queryAll( | ||||
|       By.directive(ToggleableDropdownButtonComponent) | ||||
|     const customFieldsQueryDropdown = fixture.debugElement.queryAll( | ||||
|       By.directive(CustomFieldsQueryDropdownComponent) | ||||
|     )[0] | ||||
|     const customFieldToggleButton = customFieldsQueryDropdown.query( | ||||
|       By.css('button') | ||||
|     ) | ||||
|     customFieldButtons[1].triggerEventHandler('toggle') | ||||
|     customFieldButtons[2].triggerEventHandler('toggle') | ||||
|     customFieldToggleButton.triggerEventHandler('click') | ||||
|     tick() | ||||
|     fixture.detectChanges() | ||||
|     const expression = component.customFieldQueriesModel | ||||
|       .queries[0] as CustomFieldQueryExpression | ||||
|     const atom = expression.value[0] as CustomFieldQueryAtom | ||||
|     atom.field = custom_fields[0].id | ||||
|     const fieldSelect: NgSelectComponent = customFieldsQueryDropdown.queryAll( | ||||
|       By.directive(NgSelectComponent) | ||||
|     )[0].componentInstance | ||||
|     fieldSelect.open() | ||||
|     const options = customFieldsQueryDropdown.queryAll(By.css('.ng-option')) | ||||
|     options[0].nativeElement.click() | ||||
|     expect(component.customFieldQueriesModel.queries[0].value.length).toEqual(1) | ||||
|     expect(component.filterRules).toEqual([ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: custom_fields[0].id.toString(), | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: custom_fields[1].id.toString(), | ||||
|       }, | ||||
|     ]) | ||||
|     const toggleOperatorButtons = customFieldsFilterableDropdown.queryAll( | ||||
|       By.css('input[type=radio]') | ||||
|     ) | ||||
|     toggleOperatorButtons[1].nativeElement.checked = true | ||||
|     toggleOperatorButtons[1].triggerEventHandler('change') | ||||
|     fixture.detectChanges() | ||||
|     expect(component.filterRules).toEqual([ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|         value: custom_fields[0].id.toString(), | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|         value: custom_fields[1].id.toString(), | ||||
|       }, | ||||
|     ]) | ||||
|     customFieldButtons[2].triggerEventHandler('exclude') | ||||
|     fixture.detectChanges() | ||||
|     expect(component.filterRules).toEqual([ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: custom_fields[0].id.toString(), | ||||
|       }, | ||||
|       { | ||||
|         rule_type: FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|         value: custom_fields[1].id.toString(), | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: JSON.stringify([ | ||||
|           CustomFieldQueryLogicalOperator.Or, | ||||
|           [[custom_fields[0].id, 'exists', 'true']], | ||||
|         ]), | ||||
|       }, | ||||
|     ]) | ||||
|   })) | ||||
| @ -1930,21 +1876,11 @@ describe('FilterEditorComponent', () => { | ||||
| 
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|         value: '42', | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: '["AND",[["42","exists","true"],["43","exists","true"]]]', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.generateFilterName()).toEqual( | ||||
|       `Custom fields: ${custom_fields[0].name}` | ||||
|     ) | ||||
| 
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|         rule_type: FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|         value: 'false', | ||||
|       }, | ||||
|     ] | ||||
|     expect(component.generateFilterName()).toEqual('Without any custom field') | ||||
|     expect(component.generateFilterName()).toEqual(`Custom fields query`) | ||||
| 
 | ||||
|     component.filterRules = [ | ||||
|       { | ||||
|  | ||||
| @ -12,7 +12,7 @@ import { | ||||
| import { Tag } from 'src/app/data/tag' | ||||
| import { Correspondent } from 'src/app/data/correspondent' | ||||
| import { DocumentType } from 'src/app/data/document-type' | ||||
| import { Observable, Subject, Subscription, from } from 'rxjs' | ||||
| import { Observable, Subject, from } from 'rxjs' | ||||
| import { | ||||
|   catchError, | ||||
|   debounceTime, | ||||
| @ -62,7 +62,7 @@ import { | ||||
|   FILTER_HAS_CUSTOM_FIELDS_ANY, | ||||
|   FILTER_HAS_CUSTOM_FIELDS_ALL, | ||||
|   FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|   FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|   FILTER_CUSTOM_FIELDS_QUERY, | ||||
| } from 'src/app/data/filter-rule-type' | ||||
| import { | ||||
|   FilterableDropdownSelectionModel, | ||||
| @ -92,6 +92,15 @@ import { ComponentWithPermissions } from '../../with-permissions/with-permission | ||||
| import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' | ||||
| import { CustomField } from 'src/app/data/custom-field' | ||||
| import { SearchService } from 'src/app/services/rest/search.service' | ||||
| import { | ||||
|   CustomFieldQueryLogicalOperator, | ||||
|   CustomFieldQueryOperator, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| import { CustomFieldQueriesModel } from '../../common/custom-fields-query-dropdown/custom-fields-query-dropdown.component' | ||||
| import { | ||||
|   CustomFieldQueryExpression, | ||||
|   CustomFieldQueryAtom, | ||||
| } from 'src/app/utils/custom-field-query-element' | ||||
| 
 | ||||
| const TEXT_FILTER_TARGET_TITLE = 'title' | ||||
| const TEXT_FILTER_TARGET_TITLE_CONTENT = 'title-content' | ||||
| @ -225,15 +234,8 @@ export class FilterEditorComponent | ||||
|             return $localize`Without any tag` | ||||
|           } | ||||
| 
 | ||||
|         case FILTER_HAS_CUSTOM_FIELDS_ALL: | ||||
|           return $localize`Custom fields: ${ | ||||
|             this.customFields.find((f) => f.id == +rule.value)?.name | ||||
|           }` | ||||
| 
 | ||||
|         case FILTER_HAS_ANY_CUSTOM_FIELDS: | ||||
|           if (rule.value == 'false') { | ||||
|             return $localize`Without any custom field` | ||||
|           } | ||||
|         case FILTER_CUSTOM_FIELDS_QUERY: | ||||
|           return $localize`Custom fields query` | ||||
| 
 | ||||
|         case FILTER_TITLE: | ||||
|           return $localize`Title: ${rule.value}` | ||||
| @ -321,7 +323,7 @@ export class FilterEditorComponent | ||||
|   correspondentSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   documentTypeSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   storagePathSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   customFieldSelectionModel = new FilterableDropdownSelectionModel() | ||||
|   customFieldQueriesModel = new CustomFieldQueriesModel() | ||||
| 
 | ||||
|   dateCreatedBefore: string | ||||
|   dateCreatedAfter: string | ||||
| @ -356,7 +358,7 @@ export class FilterEditorComponent | ||||
|     this.storagePathSelectionModel.clear(false) | ||||
|     this.tagSelectionModel.clear(false) | ||||
|     this.correspondentSelectionModel.clear(false) | ||||
|     this.customFieldSelectionModel.clear(false) | ||||
|     this.customFieldQueriesModel.clear(false) | ||||
|     this._textFilter = null | ||||
|     this._moreLikeId = null | ||||
|     this.dateAddedBefore = null | ||||
| @ -523,34 +525,45 @@ export class FilterEditorComponent | ||||
|             false | ||||
|           ) | ||||
|           break | ||||
|         case FILTER_CUSTOM_FIELDS_QUERY: | ||||
|           try { | ||||
|             const query = JSON.parse(rule.value) | ||||
|             if (Array.isArray(query)) { | ||||
|               if (query.length === 2) { | ||||
|                 // expression
 | ||||
|                 this.customFieldQueriesModel.addExpression( | ||||
|                   new CustomFieldQueryExpression(query as any) | ||||
|                 ) | ||||
|               } else if (query.length === 3) { | ||||
|                 // atom
 | ||||
|                 this.customFieldQueriesModel.addAtom( | ||||
|                   new CustomFieldQueryAtom(query as any) | ||||
|                 ) | ||||
|               } | ||||
|             } | ||||
|           } catch (e) { | ||||
|             // error handled by list view service
 | ||||
|           } | ||||
|           break | ||||
|         // Legacy custom field filters
 | ||||
|         case FILTER_HAS_CUSTOM_FIELDS_ALL: | ||||
|           this.customFieldSelectionModel.logicalOperator = LogicalOperator.And | ||||
|           this.customFieldSelectionModel.set( | ||||
|             rule.value ? +rule.value : null, | ||||
|             ToggleableItemState.Selected, | ||||
|             false | ||||
|           this.customFieldQueriesModel.addExpression( | ||||
|             new CustomFieldQueryExpression([ | ||||
|               CustomFieldQueryLogicalOperator.And, | ||||
|               rule.value | ||||
|                 .split(',') | ||||
|                 .map((id) => [id, CustomFieldQueryOperator.Exists, 'true']), | ||||
|             ]) | ||||
|           ) | ||||
|           break | ||||
|         case FILTER_HAS_CUSTOM_FIELDS_ANY: | ||||
|           this.customFieldSelectionModel.logicalOperator = LogicalOperator.Or | ||||
|           this.customFieldSelectionModel.set( | ||||
|             rule.value ? +rule.value : null, | ||||
|             ToggleableItemState.Selected, | ||||
|             false | ||||
|           ) | ||||
|           break | ||||
|         case FILTER_HAS_ANY_CUSTOM_FIELDS: | ||||
|           this.customFieldSelectionModel.set( | ||||
|             null, | ||||
|             ToggleableItemState.Selected, | ||||
|             false | ||||
|           ) | ||||
|           break | ||||
|         case FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS: | ||||
|           this.customFieldSelectionModel.set( | ||||
|             rule.value ? +rule.value : null, | ||||
|             ToggleableItemState.Excluded, | ||||
|             false | ||||
|           this.customFieldQueriesModel.addExpression( | ||||
|             new CustomFieldQueryExpression([ | ||||
|               CustomFieldQueryLogicalOperator.Or, | ||||
|               rule.value | ||||
|                 .split(',') | ||||
|                 .map((id) => [id, CustomFieldQueryOperator.Exists, 'true']), | ||||
|             ]) | ||||
|           ) | ||||
|           break | ||||
|         case FILTER_ASN_ISNULL: | ||||
| @ -768,34 +781,14 @@ export class FilterEditorComponent | ||||
|           }) | ||||
|         }) | ||||
|     } | ||||
|     if (this.customFieldSelectionModel.isNoneSelected()) { | ||||
|     let queries = this.customFieldQueriesModel.queries.map((query) => | ||||
|       query.serialize() | ||||
|     ) | ||||
|     if (queries.length > 0) { | ||||
|       filterRules.push({ | ||||
|         rule_type: FILTER_HAS_ANY_CUSTOM_FIELDS, | ||||
|         value: 'false', | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: JSON.stringify(queries[0]), | ||||
|       }) | ||||
|     } else { | ||||
|       const customFieldFilterType = | ||||
|         this.customFieldSelectionModel.logicalOperator == LogicalOperator.And | ||||
|           ? FILTER_HAS_CUSTOM_FIELDS_ALL | ||||
|           : FILTER_HAS_CUSTOM_FIELDS_ANY | ||||
|       this.customFieldSelectionModel | ||||
|         .getSelectedItems() | ||||
|         .filter((field) => field.id) | ||||
|         .forEach((field) => { | ||||
|           filterRules.push({ | ||||
|             rule_type: customFieldFilterType, | ||||
|             value: field.id?.toString(), | ||||
|           }) | ||||
|         }) | ||||
|       this.customFieldSelectionModel | ||||
|         .getExcludedItems() | ||||
|         .filter((field) => field.id) | ||||
|         .forEach((field) => { | ||||
|           filterRules.push({ | ||||
|             rule_type: FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS, | ||||
|             value: field.id?.toString(), | ||||
|           }) | ||||
|         }) | ||||
|     } | ||||
|     if (this.dateCreatedBefore) { | ||||
|       filterRules.push({ | ||||
| @ -1079,10 +1072,6 @@ export class FilterEditorComponent | ||||
|     this.storagePathSelectionModel.apply() | ||||
|   } | ||||
| 
 | ||||
|   onCustomFieldsDropdownOpen() { | ||||
|     this.customFieldSelectionModel.apply() | ||||
|   } | ||||
| 
 | ||||
|   updateTextFilter(text, updateRules = true) { | ||||
|     this._textFilter = text | ||||
|     if (updateRules) { | ||||
|  | ||||
| @ -26,7 +26,21 @@ | ||||
|         <div class="col d-flex align-items-center"><button class="btn btn-link p-0 text-start" type="button" (click)="editField(field)" [disabled]="!permissionsService.currentUserCan(PermissionAction.Change, PermissionType.CustomField)">{{field.name}}</button></div> | ||||
|         <div class="col d-flex align-items-center">{{getDataType(field)}}</div> | ||||
|         <div class="col"> | ||||
|           <div class="btn-group"> | ||||
|           <div class="btn-group d-block d-sm-none"> | ||||
|             <div ngbDropdown container="body" class="d-inline-block"> | ||||
|               <button type="button" class="btn btn-link" id="actionsMenuMobile" (click)="$event.stopPropagation()" ngbDropdownToggle> | ||||
|                 <i-bs name="three-dots-vertical"></i-bs> | ||||
|               </button> | ||||
|               <div ngbDropdownMenu aria-labelledby="actionsMenuMobile"> | ||||
|                 <button (click)="editField(field)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.CustomField }" ngbDropdownItem i18n>Edit</button> | ||||
|                 <button class="text-danger" (click)="deleteField(field)" *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.CustomField }" ngbDropdownItem i18n>Delete</button> | ||||
|                 @if (field.document_count > 0) { | ||||
|                   <button (click)="filterDocuments(field)" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Document }" ngbDropdownItem i18n>Filter Documents ({{ field.document_count }})</button> | ||||
|                 } | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="btn-group d-none d-sm-inline-block"> | ||||
|             <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.CustomField }" class="btn btn-sm btn-outline-secondary" type="button" (click)="editField(field)"> | ||||
|               <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|             </button> | ||||
| @ -34,6 +48,13 @@ | ||||
|               <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|             </button> | ||||
|           </div> | ||||
|           @if (field.document_count > 0) { | ||||
|             <div class="btn-group d-none d-sm-inline-block ms-2"> | ||||
|               <button class="btn btn-sm btn-outline-secondary" type="button" (click)="filterDocuments(field)"> | ||||
|                 <i-bs width="1em" height="1em" name="filter"></i-bs> <ng-container i18n>Documents</ng-container><span class="badge bg-light text-secondary ms-2">{{ field.document_count }}</span> | ||||
|               </button> | ||||
|             </div> | ||||
|           } | ||||
|         </div> | ||||
|       </div> | ||||
|     </li> | ||||
|  | ||||
| @ -0,0 +1,4 @@ | ||||
| // hide caret on mobile dropdown | ||||
| .d-block.d-sm-none .dropdown-toggle::after { | ||||
|     display: none; | ||||
| } | ||||
| @ -22,6 +22,12 @@ import { PageHeaderComponent } from '../../common/page-header/page-header.compon | ||||
| import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { DocumentListViewService } from 'src/app/services/document-list-view.service' | ||||
| import { FILTER_CUSTOM_FIELDS_QUERY } from 'src/app/data/filter-rule-type' | ||||
| import { | ||||
|   CustomFieldQueryLogicalOperator, | ||||
|   CustomFieldQueryOperator, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| 
 | ||||
| const fields: CustomField[] = [ | ||||
|   { | ||||
| @ -42,6 +48,7 @@ describe('CustomFieldsComponent', () => { | ||||
|   let customFieldsService: CustomFieldsService | ||||
|   let modalService: NgbModal | ||||
|   let toastService: ToastService | ||||
|   let listViewService: DocumentListViewService | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @ -83,6 +90,7 @@ describe('CustomFieldsComponent', () => { | ||||
|     ) | ||||
|     modalService = TestBed.inject(NgbModal) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     listViewService = TestBed.inject(DocumentListViewService) | ||||
| 
 | ||||
|     fixture = TestBed.createComponent(CustomFieldsComponent) | ||||
|     component = fixture.componentInstance | ||||
| @ -145,7 +153,7 @@ describe('CustomFieldsComponent', () => { | ||||
|     const deleteSpy = jest.spyOn(customFieldsService, 'delete') | ||||
|     const reloadSpy = jest.spyOn(component, 'reload') | ||||
| 
 | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[4] | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[5] | ||||
|     deleteButton.triggerEventHandler('click') | ||||
| 
 | ||||
|     expect(modal).not.toBeUndefined() | ||||
| @ -162,4 +170,18 @@ describe('CustomFieldsComponent', () => { | ||||
|     editDialog.confirmClicked.emit() | ||||
|     expect(reloadSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support filter documents', () => { | ||||
|     const filterSpy = jest.spyOn(listViewService, 'quickFilter') | ||||
|     component.filterDocuments(fields[0]) | ||||
|     expect(filterSpy).toHaveBeenCalledWith([ | ||||
|       { | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: JSON.stringify([ | ||||
|           CustomFieldQueryLogicalOperator.Or, | ||||
|           [[fields[0].id, CustomFieldQueryOperator.Exists, true]], | ||||
|         ]), | ||||
|       }, | ||||
|     ]) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -9,6 +9,13 @@ import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dial | ||||
| import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component' | ||||
| import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component' | ||||
| import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' | ||||
| import { DocumentListViewService } from 'src/app/services/document-list-view.service' | ||||
| import { FILTER_CUSTOM_FIELDS_QUERY } from 'src/app/data/filter-rule-type' | ||||
| import { | ||||
|   CustomFieldQueryLogicalOperator, | ||||
|   CustomFieldQueryOperator, | ||||
| } from 'src/app/data/custom-field-query' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'pngx-custom-fields', | ||||
| @ -26,7 +33,9 @@ export class CustomFieldsComponent | ||||
|     private customFieldsService: CustomFieldsService, | ||||
|     public permissionsService: PermissionsService, | ||||
|     private modalService: NgbModal, | ||||
|     private toastService: ToastService | ||||
|     private toastService: ToastService, | ||||
|     private documentListViewService: DocumentListViewService, | ||||
|     private settingsService: SettingsService | ||||
|   ) { | ||||
|     super() | ||||
|   } | ||||
| @ -55,6 +64,7 @@ export class CustomFieldsComponent | ||||
|       .subscribe((newField) => { | ||||
|         this.toastService.showInfo($localize`Saved field "${newField.name}".`) | ||||
|         this.customFieldsService.clearCache() | ||||
|         this.settingsService.initializeDisplayFields() | ||||
|         this.reload() | ||||
|       }) | ||||
|     modal.componentInstance.failed | ||||
| @ -80,6 +90,7 @@ export class CustomFieldsComponent | ||||
|           modal.close() | ||||
|           this.toastService.showInfo($localize`Deleted field`) | ||||
|           this.customFieldsService.clearCache() | ||||
|           this.settingsService.initializeDisplayFields() | ||||
|           this.reload() | ||||
|         }, | ||||
|         error: (e) => { | ||||
| @ -92,4 +103,16 @@ export class CustomFieldsComponent | ||||
|   getDataType(field: CustomField): string { | ||||
|     return DATA_TYPE_LABELS.find((l) => l.id === field.data_type).name | ||||
|   } | ||||
| 
 | ||||
|   filterDocuments(field: CustomField) { | ||||
|     this.documentListViewService.quickFilter([ | ||||
|       { | ||||
|         rule_type: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|         value: JSON.stringify([ | ||||
|           CustomFieldQueryLogicalOperator.Or, | ||||
|           [[field.id, CustomFieldQueryOperator.Exists, true]], | ||||
|         ]), | ||||
|       }, | ||||
|     ]) | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -13,13 +13,23 @@ | ||||
|     <button type="button" class="btn btn-sm btn-outline-primary ms-4" (click)="editMailAccount()" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.MailAccount }"> | ||||
|       <i-bs name="plus-circle"></i-bs> <ng-container i18n>Add Account</ng-container> | ||||
|     </button> | ||||
|     @if (gmailOAuthUrl) { | ||||
|       <a class="btn btn-sm btn-outline-secondary ms-2" [href]="gmailOAuthUrl" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.MailAccount }"> | ||||
|         <i-bs name="google"></i-bs> <ng-container i18n>Connect Gmail Account</ng-container> | ||||
|       </a> | ||||
|     } | ||||
|     @if (outlookOAuthUrl) { | ||||
|       <a class="btn btn-sm btn-outline-secondary ms-2" [href]="outlookOAuthUrl" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.MailAccount }"> | ||||
|         <i-bs name="microsoft"></i-bs> <ng-container i18n>Connect Outlook Account</ng-container> | ||||
|       </a> | ||||
|     } | ||||
|   </h4> | ||||
|   <ul class="list-group"> | ||||
|     <li class="list-group-item"> | ||||
|       <div class="row"> | ||||
|         <div class="col" i18n>Name</div> | ||||
|         <div class="col" i18n>Server</div> | ||||
|         <div class="col" i18n>Username</div> | ||||
|         <div class="col d-none d-sm-block" i18n>Username</div> | ||||
|         <div class="col" i18n>Actions</div> | ||||
|       </div> | ||||
|     </li> | ||||
| @ -27,11 +37,31 @@ | ||||
|     @for (account of mailAccounts; track account) { | ||||
|       <li class="list-group-item"> | ||||
|         <div class="row"> | ||||
|           <div class="col d-flex align-items-center"><button class="btn btn-link p-0 text-start" type="button" (click)="editMailAccount(account)" [disabled]="!permissionsService.currentUserCan(PermissionAction.Change, PermissionType.MailAccount)">{{account.name}}</button></div> | ||||
|           <div class="col d-flex align-items-center"> | ||||
|             <button class="btn btn-link p-0 text-start" type="button" (click)="editMailAccount(account)" [disabled]="!permissionsService.currentUserCan(PermissionAction.Change, PermissionType.MailAccount)"> | ||||
|               {{account.name}}@switch (account.account_type) { | ||||
|                 @case (MailAccountType.IMAP) {<i-bs name="envelope-at-fill" class="ms-2"></i-bs>} | ||||
|                 @case (MailAccountType.Gmail_OAuth) {<i-bs name="google" class="ms-2"></i-bs>} | ||||
|                 @case (MailAccountType.Outlook_OAuth) {<i-bs name="microsoft" class="ms-2"></i-bs>} | ||||
|               } | ||||
|             </button> | ||||
|           </div> | ||||
|           <div class="col d-flex align-items-center">{{account.imap_server}}</div> | ||||
|           <div class="col d-flex align-items-center">{{account.username}}</div> | ||||
|           <div class="col d-flex align-items-center d-none d-sm-block">{{account.username}}</div> | ||||
|           <div class="col"> | ||||
|             <div class="btn-group"> | ||||
|             <div class="btn-group d-block d-sm-none"> | ||||
|               <div ngbDropdown container="body" class="d-inline-block"> | ||||
|                 <button type="button" class="btn btn-link" id="actionsMenuMobile" (click)="$event.stopPropagation()" ngbDropdownToggle> | ||||
|                   <i-bs name="three-dots-vertical"></i-bs> | ||||
|                 </button> | ||||
|                 <div ngbDropdownMenu aria-labelledby="actionsMenuMobile"> | ||||
|                   <button (click)="editMailAccount(account)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailAccount }" ngbDropdownItem i18n>Edit</button> | ||||
|                   <button (click)="editPermissions(account)" *pngxIfOwner="account" ngbDropdownItem i18n>Permissions</button> | ||||
|                   <button (click)="deleteMailAccount(account)" *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.MailAccount }" ngbDropdownItem i18n>Delete</button> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="btn-group d-none d-sm-block"> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailAccount }" [disabled]="!userCanEdit(account)" class="btn btn-sm btn-outline-secondary" type="button" (click)="editMailAccount(account)"> | ||||
|                 <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|               </button> | ||||
| @ -64,8 +94,9 @@ | ||||
|     <li class="list-group-item"> | ||||
|       <div class="row"> | ||||
|         <div class="col" i18n>Name</div> | ||||
|         <div class="col" i18n>Sort Order</div> | ||||
|         <div class="col d-none d-sm-block" i18n>Sort Order</div> | ||||
|         <div class="col" i18n>Account</div> | ||||
|         <div class="col d-none d-sm-block" i18n>Status</div> | ||||
|         <div class="col" i18n>Actions</div> | ||||
|       </div> | ||||
|     </li> | ||||
| @ -74,19 +105,47 @@ | ||||
|       <li class="list-group-item"> | ||||
|         <div class="row"> | ||||
|           <div class="col d-flex align-items-center"><button class="btn btn-link p-0 text-start" type="button" (click)="editMailRule(rule)" [disabled]="!permissionsService.currentUserCan(PermissionAction.Change, PermissionType.MailRule)">{{rule.name}}</button></div> | ||||
|           <div class="col d-flex align-items-center">{{rule.order}}</div> | ||||
|           <div class="col d-flex align-items-center d-none d-sm-flex">{{rule.order}}</div> | ||||
|           <div class="col d-flex align-items-center">{{(mailAccountService.getCached(rule.account) | async)?.name}}</div> | ||||
|           <div class="col d-flex align-items-center d-none d-sm-flex"> | ||||
|             <div class="form-check form-switch mb-0"> | ||||
|               <input #inputField type="checkbox" class="form-check-input cursor-pointer" [id]="rule.id+'_enable'" [(ngModel)]="rule.enabled" (change)="onMailRuleEnableToggled(rule)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailRule }"> | ||||
|               <label class="form-check-label cursor-pointer" [for]="rule.id+'_enable'"> | ||||
|                 <code> @if(rule.enabled) { <ng-container i18n>Enabled</ng-container> } @else { <span i18n class="text-muted">Disabled</span> }</code> | ||||
|               </label> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="col"> | ||||
|             <div class="btn-group"> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailRule }" [disabled]="!userCanEdit(rule)" class="btn btn-sm btn-outline-secondary" type="button" (click)="editMailRule(rule)"> | ||||
|                 <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|               </button> | ||||
|               <button *pngxIfOwner="rule" class="btn btn-sm btn-outline-secondary" type="button" (click)="editPermissions(rule)"> | ||||
|                 <i-bs width="1em" height="1em" name="person-lock"></i-bs> <ng-container i18n>Permissions</ng-container> | ||||
|               </button> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.MailRule }" [disabled]="!userIsOwner(rule)" class="btn btn-sm btn-outline-danger" type="button" (click)="deleteMailRule(rule)"> | ||||
|                 <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|               </button> | ||||
|             <div class="btn-group d-block d-sm-none"> | ||||
|               <div ngbDropdown container="body" class="d-inline-block"> | ||||
|                 <button type="button" class="btn btn-link" id="actionsMenuMobile" (click)="$event.stopPropagation()" ngbDropdownToggle> | ||||
|                   <i-bs name="three-dots-vertical"></i-bs> | ||||
|                 </button> | ||||
|                 <div ngbDropdownMenu aria-labelledby="actionsMenuMobile"> | ||||
|                   <button (click)="editMailRule(rule)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailRule }" ngbDropdownItem i18n>Edit</button> | ||||
|                   <button (click)="editPermissions(rule)" *pngxIfOwner="rule" ngbDropdownItem i18n>Permissions</button> | ||||
|                   <button (click)="deleteMailRule(rule)" *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.MailRule }" ngbDropdownItem i18n>Delete</button> | ||||
|                   <button (click)="copyMailRule(rule)" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.MailRule }" ngbDropdownItem i18n>Copy</button> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="btn-toolbar d-none d-sm-flex gap-2" role="toolbar"> | ||||
|               <div class="btn-group"> | ||||
|                 <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.MailRule }" [disabled]="!userCanEdit(rule)" class="btn btn-sm btn-outline-secondary" type="button" (click)="editMailRule(rule)"> | ||||
|                   <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|                 </button> | ||||
|                 <button *pngxIfOwner="rule" class="btn btn-sm btn-outline-secondary" type="button" (click)="editPermissions(rule)"> | ||||
|                   <i-bs width="1em" height="1em" name="person-lock"></i-bs> <ng-container i18n>Permissions</ng-container> | ||||
|                 </button> | ||||
|                 <button *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.MailRule }" [disabled]="!userIsOwner(rule)" class="btn btn-sm btn-outline-danger" type="button" (click)="deleteMailRule(rule)"> | ||||
|                   <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|                 </button> | ||||
|               </div> | ||||
|               <div class="btn-group"> | ||||
|                 <button *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.MailRule }" class="btn btn-sm btn-outline-secondary" type="button" (click)="copyMailRule(rule)"> | ||||
|                   <i-bs width="1em" height="1em" name="files"></i-bs> <ng-container i18n>Copy</ng-container> | ||||
|                 </button> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
| @ -0,0 +1,4 @@ | ||||
| // hide caret on mobile dropdown | ||||
| .d-block.d-sm-none .dropdown-toggle::after { | ||||
|   display: none; | ||||
| } | ||||
| @ -13,7 +13,7 @@ import { | ||||
| import { NgSelectModule } from '@ng-select/ng-select' | ||||
| import { of, throwError } from 'rxjs' | ||||
| import { routes } from 'src/app/app-routing.module' | ||||
| import { MailAccount } from 'src/app/data/mail-account' | ||||
| import { MailAccount, MailAccountType } from 'src/app/data/mail-account' | ||||
| import { MailRule } from 'src/app/data/mail-rule' | ||||
| import { IfOwnerDirective } from 'src/app/directives/if-owner.directive' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| @ -43,14 +43,18 @@ import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { SwitchComponent } from '../../common/input/switch/switch.component' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { By } from '@angular/platform-browser' | ||||
| import { ActivatedRoute, convertToParamMap } from '@angular/router' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| 
 | ||||
| const mailAccounts = [ | ||||
|   { id: 1, name: 'account1' }, | ||||
|   { id: 2, name: 'account2' }, | ||||
|   { id: 1, name: 'account1', account_type: MailAccountType.IMAP }, | ||||
|   { id: 2, name: 'account2', account_type: MailAccountType.IMAP }, | ||||
|   { id: 3, name: 'account3', accout_type: MailAccountType.Gmail_OAuth }, | ||||
| ] | ||||
| const mailRules = [ | ||||
|   { id: 1, name: 'rule1', owner: 1, account: 1 }, | ||||
|   { id: 2, name: 'rule2', owner: 2, account: 2 }, | ||||
|   { id: 1, name: 'rule1', owner: 1, account: 1, enabled: true }, | ||||
|   { id: 2, name: 'rule2', owner: 2, account: 2, enabled: true }, | ||||
| ] | ||||
| 
 | ||||
| describe('MailComponent', () => { | ||||
| @ -61,6 +65,8 @@ describe('MailComponent', () => { | ||||
|   let modalService: NgbModal | ||||
|   let toastService: ToastService | ||||
|   let permissionsService: PermissionsService | ||||
|   let activatedRoute: ActivatedRoute | ||||
|   let settingsService: SettingsService | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @ -109,6 +115,9 @@ describe('MailComponent', () => { | ||||
|     modalService = TestBed.inject(NgbModal) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     permissionsService = TestBed.inject(PermissionsService) | ||||
|     activatedRoute = TestBed.inject(ActivatedRoute) | ||||
|     settingsService = TestBed.inject(SettingsService) | ||||
|     settingsService.currentUser = { id: 1 } | ||||
|     jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) | ||||
|     jest | ||||
|       .spyOn(permissionsService, 'currentUserHasObjectPermissions') | ||||
| @ -226,6 +235,17 @@ describe('MailComponent', () => { | ||||
|     component.editMailRule() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support copy mail rule', () => { | ||||
|     completeSetup() | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((refs) => (modal = refs[0])) | ||||
|     component.copyMailRule(mailRules[0] as MailRule) | ||||
|     const editDialog = modal.componentInstance as MailRuleEditDialogComponent | ||||
|     expect(editDialog.object.id).toBeNull() | ||||
|     expect(editDialog.object.name).toEqual(`${mailRules[0].name} (copy)`) | ||||
|     expect(editDialog.dialogMode).toEqual(EditDialogMode.CREATE) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support delete mail rule, show error if needed', () => { | ||||
|     completeSetup() | ||||
|     let modal: NgbModalRef | ||||
| @ -310,4 +330,62 @@ describe('MailComponent', () => { | ||||
|     dialog.confirmClicked.emit({ permissions: perms, merge: true }) | ||||
|     expect(accountPatchSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should update mail rule when enable is toggled', () => { | ||||
|     completeSetup() | ||||
|     const patchSpy = jest.spyOn(mailRuleService, 'patch') | ||||
|     const toggleInput = fixture.debugElement.query( | ||||
|       By.css('input[type="checkbox"]') | ||||
|     ) | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     // fail first
 | ||||
|     patchSpy.mockReturnValueOnce( | ||||
|       throwError(() => new Error('Error getting config')) | ||||
|     ) | ||||
|     toggleInput.nativeElement.click() | ||||
|     expect(patchSpy).toHaveBeenCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     // succeed second
 | ||||
|     patchSpy.mockReturnValueOnce(of(mailRules[0] as MailRule)) | ||||
|     toggleInput.nativeElement.click() | ||||
|     patchSpy.mockReturnValueOnce( | ||||
|       of({ ...mailRules[0], enabled: false } as MailRule) | ||||
|     ) | ||||
|     toggleInput.nativeElement.click() | ||||
|     expect(patchSpy).toHaveBeenCalled() | ||||
|     expect(toastInfoSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should show success message when oauth account is connected', () => { | ||||
|     const queryParams = { oauth_success: '1' } | ||||
|     jest | ||||
|       .spyOn(activatedRoute, 'queryParamMap', 'get') | ||||
|       .mockReturnValue(of(convertToParamMap(queryParams))) | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     completeSetup() | ||||
|     expect(toastInfoSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should show error message when oauth account connect fails', () => { | ||||
|     const queryParams = { oauth_success: '0' } | ||||
|     jest | ||||
|       .spyOn(activatedRoute, 'queryParamMap', 'get') | ||||
|       .mockReturnValue(of(convertToParamMap(queryParams))) | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     completeSetup() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should open account edit dialog if oauth account is connected', () => { | ||||
|     const queryParams = { oauth_success: '1', oauth_account: '3' } | ||||
|     jest | ||||
|       .spyOn(activatedRoute, 'queryParamMap', 'get') | ||||
|       .mockReturnValue(of(convertToParamMap(queryParams))) | ||||
|     completeSetup() | ||||
|     component.oAuthAccountId = 3 | ||||
|     const editSpy = jest.spyOn(component, 'editMailAccount') | ||||
|     component.ngOnInit() | ||||
|     expect(editSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -2,7 +2,7 @@ import { Component, OnInit, OnDestroy } from '@angular/core' | ||||
| import { NgbModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { Subject, first, takeUntil } from 'rxjs' | ||||
| import { ObjectWithPermissions } from 'src/app/data/object-with-permissions' | ||||
| import { MailAccount } from 'src/app/data/mail-account' | ||||
| import { MailAccount, MailAccountType } from 'src/app/data/mail-account' | ||||
| import { MailRule } from 'src/app/data/mail-rule' | ||||
| import { | ||||
|   PermissionsService, | ||||
| @ -18,6 +18,9 @@ import { MailAccountEditDialogComponent } from '../../common/edit-dialog/mail-ac | ||||
| import { MailRuleEditDialogComponent } from '../../common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component' | ||||
| import { PermissionsDialogComponent } from '../../common/permissions-dialog/permissions-dialog.component' | ||||
| import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { SETTINGS_KEYS } from 'src/app/data/ui-settings' | ||||
| import { ActivatedRoute } from '@angular/router' | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'pngx-mail', | ||||
| @ -28,17 +31,30 @@ export class MailComponent | ||||
|   extends ComponentWithPermissions | ||||
|   implements OnInit, OnDestroy | ||||
| { | ||||
|   public MailAccountType = MailAccountType | ||||
| 
 | ||||
|   mailAccounts: MailAccount[] = [] | ||||
|   mailRules: MailRule[] = [] | ||||
| 
 | ||||
|   unsubscribeNotifier: Subject<any> = new Subject() | ||||
|   oAuthAccountId: number | ||||
| 
 | ||||
|   public get gmailOAuthUrl(): string { | ||||
|     return this.settingsService.get(SETTINGS_KEYS.GMAIL_OAUTH_URL) | ||||
|   } | ||||
| 
 | ||||
|   public get outlookOAuthUrl(): string { | ||||
|     return this.settingsService.get(SETTINGS_KEYS.OUTLOOK_OAUTH_URL) | ||||
|   } | ||||
| 
 | ||||
|   constructor( | ||||
|     public mailAccountService: MailAccountService, | ||||
|     public mailRuleService: MailRuleService, | ||||
|     private toastService: ToastService, | ||||
|     private modalService: NgbModal, | ||||
|     public permissionsService: PermissionsService | ||||
|     public permissionsService: PermissionsService, | ||||
|     private settingsService: SettingsService, | ||||
|     private route: ActivatedRoute | ||||
|   ) { | ||||
|     super() | ||||
|   } | ||||
| @ -50,6 +66,13 @@ export class MailComponent | ||||
|       .subscribe({ | ||||
|         next: (r) => { | ||||
|           this.mailAccounts = r.results | ||||
|           if (this.oAuthAccountId) { | ||||
|             this.editMailAccount( | ||||
|               this.mailAccounts.find( | ||||
|                 (account) => account.id === this.oAuthAccountId | ||||
|               ) | ||||
|             ) | ||||
|           } | ||||
|         }, | ||||
|         error: (e) => { | ||||
|           this.toastService.showError( | ||||
| @ -70,6 +93,27 @@ export class MailComponent | ||||
|           this.toastService.showError($localize`Error retrieving mail rules`, e) | ||||
|         }, | ||||
|       }) | ||||
| 
 | ||||
|     this.route.queryParamMap.subscribe((params) => { | ||||
|       if (params.get('oauth_success')) { | ||||
|         const success = params.get('oauth_success') === '1' | ||||
|         if (success) { | ||||
|           this.toastService.showInfo($localize`OAuth2 authentication success`) | ||||
|           this.oAuthAccountId = parseInt(params.get('account_id')) | ||||
|           if (this.mailAccounts.length > 0) { | ||||
|             this.editMailAccount( | ||||
|               this.mailAccounts.find( | ||||
|                 (account) => account.id === this.oAuthAccountId | ||||
|               ) | ||||
|             ) | ||||
|           } | ||||
|         } else { | ||||
|           this.toastService.showError( | ||||
|             $localize`OAuth2 authentication failed, see logs for details` | ||||
|           ) | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   ngOnDestroy() { | ||||
| @ -137,14 +181,13 @@ export class MailComponent | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   editMailRule(rule: MailRule = null) { | ||||
|   editMailRule(rule: MailRule = null, forceCreate = false) { | ||||
|     const modal = this.modalService.open(MailRuleEditDialogComponent, { | ||||
|       backdrop: 'static', | ||||
|       size: 'xl', | ||||
|     }) | ||||
|     modal.componentInstance.dialogMode = rule | ||||
|       ? EditDialogMode.EDIT | ||||
|       : EditDialogMode.CREATE | ||||
|     modal.componentInstance.dialogMode = | ||||
|       rule && !forceCreate ? EditDialogMode.EDIT : EditDialogMode.CREATE | ||||
|     modal.componentInstance.object = rule | ||||
|     modal.componentInstance.succeeded | ||||
|       .pipe(takeUntil(this.unsubscribeNotifier)) | ||||
| @ -164,6 +207,28 @@ export class MailComponent | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|   copyMailRule(rule: MailRule) { | ||||
|     const clone = { ...rule } | ||||
|     clone.id = null | ||||
|     clone.name = `${rule.name} (copy)` | ||||
|     this.editMailRule(clone, true) | ||||
|   } | ||||
| 
 | ||||
|   onMailRuleEnableToggled(rule: MailRule) { | ||||
|     this.mailRuleService.patch(rule).subscribe({ | ||||
|       next: () => { | ||||
|         this.toastService.showInfo( | ||||
|           rule.enabled | ||||
|             ? $localize`Rule "${rule.name}" enabled.` | ||||
|             : $localize`Rule "${rule.name}" disabled.` | ||||
|         ) | ||||
|       }, | ||||
|       error: (e) => { | ||||
|         this.toastService.showError($localize`Error toggling rule.`, e) | ||||
|       }, | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   deleteMailRule(rule: MailRule) { | ||||
|     const modal = this.modalService.open(ConfirmDialogComponent, { | ||||
|       backdrop: 'static', | ||||
|  | ||||
| @ -38,7 +38,7 @@ | ||||
|         <th scope="col" class="fw-normal d-none d-sm-table-cell" pngxSortable="matching_algorithm" [currentSortField]="sortField" [currentSortReverse]="sortReverse" (sort)="onSort($event)" i18n>Matching</th> | ||||
|         <th scope="col" class="fw-normal" pngxSortable="document_count" [currentSortField]="sortField" [currentSortReverse]="sortReverse" (sort)="onSort($event)" i18n>Document count</th> | ||||
|         @for (column of extraColumns; track column) { | ||||
|           <th scope="col" class="fw-normal" pngxSortable="{{column.key}}" [currentSortField]="sortField" [currentSortReverse]="sortReverse" (sort)="onSort($event)">{{column.name}}</th> | ||||
|           <th scope="col" class="fw-normal" [ngClass]="{ 'd-none d-sm-table-cell' : column.hideOnMobile }" pngxSortable="{{column.key}}" [currentSortField]="sortField" [currentSortReverse]="sortReverse" (sort)="onSort($event)">{{column.name}}</th> | ||||
|         } | ||||
|         <th scope="col" class="fw-normal" i18n>Actions</th> | ||||
|       </tr> | ||||
| @ -64,7 +64,7 @@ | ||||
|           <td scope="row" class="d-none d-sm-table-cell">{{ getMatching(object) }}</td> | ||||
|           <td scope="row">{{ object.document_count }}</td> | ||||
|           @for (column of extraColumns; track column) { | ||||
|             <td scope="row"> | ||||
|             <td scope="row" [ngClass]="{ 'd-none d-sm-table-cell' : column.hideOnMobile }"> | ||||
|               @if (column.rendersHtml) { | ||||
|                 <div [innerHtml]="column.valueFn.call(null, object) | safeHtml"></div> | ||||
|               } @else { | ||||
| @ -79,16 +79,15 @@ | ||||
|                   <i-bs name="three-dots-vertical"></i-bs> | ||||
|                 </button> | ||||
|                 <div ngbDropdownMenu aria-labelledby="actionsMenuMobile"> | ||||
|                   <button (click)="filterDocuments(object)" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Document }" ngbDropdownItem i18n>Filter Documents</button> | ||||
|                   <button (click)="openEditDialog(object)" *pngxIfPermissions="{ action: PermissionAction.Change, type: permissionType }" ngbDropdownItem i18n>Edit</button> | ||||
|                   <button class="text-danger" (click)="openDeleteDialog(object)" *pngxIfPermissions="{ action: PermissionAction.Delete, type: permissionType }" ngbDropdownItem i18n>Delete</button> | ||||
|                   @if (object.document_count > 0) { | ||||
|                     <button (click)="filterDocuments(object)" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Document }" ngbDropdownItem i18n>Filter Documents ({{ object.document_count }})</button> | ||||
|                   } | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="btn-group d-none d-sm-block"> | ||||
|               <button class="btn btn-sm btn-outline-secondary" (click)="filterDocuments(object); $event.stopPropagation();" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Document }"> | ||||
|                 <i-bs width="1em" height="1em" name="filter"></i-bs> <ng-container i18n>Documents</ng-container> | ||||
|               </button> | ||||
|             <div class="btn-group d-none d-sm-inline-block"> | ||||
|               <button class="btn btn-sm btn-outline-secondary" (click)="openEditDialog(object); $event.stopPropagation();" *pngxIfPermissions="{ action: PermissionAction.Change, type: permissionType }" [disabled]="!userCanEdit(object)"> | ||||
|                 <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|               </button> | ||||
| @ -96,6 +95,13 @@ | ||||
|                 <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|               </button> | ||||
|             </div> | ||||
|             @if (object.document_count > 0) { | ||||
|               <div class="btn-group d-none d-sm-inline-block ms-2"> | ||||
|                 <button class="btn btn-sm btn-outline-secondary" (click)="filterDocuments(object); $event.stopPropagation();" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Document }"> | ||||
|                   <i-bs width="1em" height="1em" name="filter"></i-bs> <ng-container i18n>Documents</ng-container><span class="badge bg-light text-secondary ms-2">{{ object.document_count }}</span> | ||||
|                 </button> | ||||
|               </div> | ||||
|             } | ||||
|           </td> | ||||
|         </tr> | ||||
|       } | ||||
|  | ||||
| @ -49,16 +49,19 @@ const tags: Tag[] = [ | ||||
|     name: 'Tag1 Foo', | ||||
|     matching_algorithm: MATCH_LITERAL, | ||||
|     match: 'foo', | ||||
|     document_count: 35, | ||||
|   }, | ||||
|   { | ||||
|     id: 2, | ||||
|     name: 'Tag2', | ||||
|     matching_algorithm: MATCH_NONE, | ||||
|     document_count: 0, | ||||
|   }, | ||||
|   { | ||||
|     id: 3, | ||||
|     name: 'Tag3', | ||||
|     matching_algorithm: MATCH_AUTO, | ||||
|     document_count: 5, | ||||
|   }, | ||||
| ] | ||||
| 
 | ||||
| @ -180,7 +183,7 @@ describe('ManagementListComponent', () => { | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     const reloadSpy = jest.spyOn(component, 'reloadData') | ||||
| 
 | ||||
|     const editButton = fixture.debugElement.queryAll(By.css('button'))[7] | ||||
|     const editButton = fixture.debugElement.queryAll(By.css('button'))[6] | ||||
|     editButton.triggerEventHandler('click') | ||||
| 
 | ||||
|     expect(modal).not.toBeUndefined() | ||||
| @ -205,7 +208,7 @@ describe('ManagementListComponent', () => { | ||||
|     const deleteSpy = jest.spyOn(tagService, 'delete') | ||||
|     const reloadSpy = jest.spyOn(component, 'reloadData') | ||||
| 
 | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[8] | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[7] | ||||
|     deleteButton.triggerEventHandler('click') | ||||
| 
 | ||||
|     expect(modal).not.toBeUndefined() | ||||
| @ -225,7 +228,7 @@ describe('ManagementListComponent', () => { | ||||
| 
 | ||||
|   it('should support quick filter for objects', () => { | ||||
|     const qfSpy = jest.spyOn(documentListViewService, 'quickFilter') | ||||
|     const filterButton = fixture.debugElement.queryAll(By.css('button'))[6] | ||||
|     const filterButton = fixture.debugElement.queryAll(By.css('button'))[8] | ||||
|     filterButton.triggerEventHandler('click') | ||||
|     expect(qfSpy).toHaveBeenCalledWith([ | ||||
|       { rule_type: FILTER_HAS_TAGS_ALL, value: tags[0].id.toString() }, | ||||
|  | ||||
| @ -44,6 +44,8 @@ export interface ManagementListColumn { | ||||
|   valueFn: any | ||||
| 
 | ||||
|   rendersHtml?: boolean | ||||
| 
 | ||||
|   hideOnMobile?: boolean | ||||
| } | ||||
| 
 | ||||
| @Directive() | ||||
|  | ||||
| @ -11,6 +11,8 @@ import { PageHeaderComponent } from '../../common/page-header/page-header.compon | ||||
| import { StoragePathListComponent } from './storage-path-list.component' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| import { StoragePath } from 'src/app/data/storage-path' | ||||
| 
 | ||||
| describe('StoragePathListComponent', () => { | ||||
|   let component: StoragePathListComponent | ||||
| @ -24,6 +26,7 @@ describe('StoragePathListComponent', () => { | ||||
|         SortableDirective, | ||||
|         PageHeaderComponent, | ||||
|         IfPermissionsDirective, | ||||
|         SafeHtmlPipe, | ||||
|       ], | ||||
|       imports: [ | ||||
|         NgbPaginationModule, | ||||
| @ -71,4 +74,15 @@ describe('StoragePathListComponent', () => { | ||||
|       'Do you really want to delete the storage path "StoragePath1"?' | ||||
|     ) | ||||
|   }) | ||||
| 
 | ||||
|   it('should truncate path if necessary', () => { | ||||
|     const path: StoragePath = { | ||||
|       id: 1, | ||||
|       name: 'StoragePath1', | ||||
|       path: 'a'.repeat(100), | ||||
|     } | ||||
|     expect(component.extraColumns[0].valueFn(path)).toEqual( | ||||
|       `<code>${'a'.repeat(49)}...</code>` | ||||
|     ) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -40,8 +40,10 @@ export class StoragePathListComponent extends ManagementListComponent<StoragePat | ||||
|         { | ||||
|           key: 'path', | ||||
|           name: $localize`Path`, | ||||
|           rendersHtml: true, | ||||
|           hideOnMobile: true, | ||||
|           valueFn: (c: StoragePath) => { | ||||
|             return c.path | ||||
|             return `<code>${c.path?.slice(0, 49)}${c.path?.length > 50 ? '...' : ''}</code>` | ||||
|           }, | ||||
|         }, | ||||
|       ] | ||||
|  | ||||
| @ -15,9 +15,9 @@ | ||||
|   <li class="list-group-item"> | ||||
|     <div class="row"> | ||||
|       <div class="col" i18n>Name</div> | ||||
|       <div class="col" i18n>Sort order</div> | ||||
|       <div class="col d-none d-sm-flex" i18n>Sort order</div> | ||||
|       <div class="col" i18n>Status</div> | ||||
|       <div class="col" i18n>Triggers</div> | ||||
|       <div class="col d-none d-sm-flex" i18n>Triggers</div> | ||||
|       <div class="col" i18n>Actions</div> | ||||
|     </div> | ||||
|   </li> | ||||
| @ -26,17 +26,44 @@ | ||||
|     <li class="list-group-item"> | ||||
|       <div class="row"> | ||||
|         <div class="col d-flex align-items-center"><button class="btn btn-link p-0 text-start" type="button" (click)="editWorkflow(workflow)" [disabled]="!permissionsService.currentUserCan(PermissionAction.Change, PermissionType.Workflow)">{{workflow.name}}</button></div> | ||||
|         <div class="col d-flex align-items-center"><code>{{workflow.order}}</code></div> | ||||
|         <div class="col d-flex align-items-center"><code> @if(workflow.enabled) { <ng-container i18n>Enabled</ng-container> } @else { <span i18n class="text-muted">Disabled</span> }</code></div> | ||||
|         <div class="col d-flex align-items-center">{{getTypesList(workflow)}}</div> | ||||
|         <div class="col d-flex align-items-center d-none d-sm-flex"><code>{{workflow.order}}</code></div> | ||||
|         <div class="col d-flex align-items-center"> | ||||
|           <div class="form-check form-switch mb-0"> | ||||
|             <input #inputField type="checkbox" class="form-check-input cursor-pointer" [id]="workflow.id+'_enable'" [(ngModel)]="workflow.enabled" (change)="onWorkflowEnableToggled(workflow)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Workflow }"> | ||||
|             <label class="form-check-label cursor-pointer" [for]="workflow.id+'_enable'"> | ||||
|               <code> @if(workflow.enabled) { <ng-container i18n>Enabled</ng-container> } @else { <span i18n class="text-muted">Disabled</span> }</code> | ||||
|             </label> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="col d-flex align-items-center d-none d-sm-flex">{{getTypesList(workflow)}}</div> | ||||
|         <div class="col"> | ||||
|           <div class="btn-group"> | ||||
|             <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Workflow }" class="btn btn-sm btn-outline-secondary" type="button" (click)="editWorkflow(workflow)"> | ||||
|               <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|             </button> | ||||
|             <button *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.Workflow }" class="btn btn-sm btn-outline-danger" type="button" (click)="deleteWorkflow(workflow)"> | ||||
|               <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|             </button> | ||||
| 
 | ||||
|           <div class="btn-group d-block d-sm-none"> | ||||
|             <div ngbDropdown container="body" class="d-inline-block"> | ||||
|               <button type="button" class="btn btn-link" id="actionsMenuMobile" (click)="$event.stopPropagation()" ngbDropdownToggle> | ||||
|                 <i-bs name="three-dots-vertical"></i-bs> | ||||
|               </button> | ||||
|               <div ngbDropdownMenu aria-labelledby="actionsMenuMobile"> | ||||
|                 <button (click)="editWorkflow(workflow)" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Workflow }" ngbDropdownItem i18n>Edit</button> | ||||
|                 <button (click)="deleteWorkflow(workflow)" *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.Workflow }" ngbDropdownItem i18n>Delete</button> | ||||
|                 <button (click)="copyWorkflow(workflow)" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.Workflow }" ngbDropdownItem i18n>Copy</button> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="btn-toolbar d-none d-sm-flex gap-2" role="toolbar"> | ||||
|             <div class="btn-group"> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.Workflow }" class="btn btn-sm btn-outline-secondary" type="button" (click)="editWorkflow(workflow)"> | ||||
|                 <i-bs width="1em" height="1em" name="pencil"></i-bs> <ng-container i18n>Edit</ng-container> | ||||
|               </button> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.Workflow }" class="btn btn-sm btn-outline-danger" type="button" (click)="deleteWorkflow(workflow)"> | ||||
|                 <i-bs width="1em" height="1em" name="trash"></i-bs> <ng-container i18n>Delete</ng-container> | ||||
|               </button> | ||||
|             </div> | ||||
|             <div class="btn-group"> | ||||
|               <button *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.Workflow }" class="btn btn-sm btn-outline-secondary" type="button" (click)="copyWorkflow(workflow)"> | ||||
|                 <i-bs width="1em" height="1em" name="files"></i-bs> <ng-container i18n>Copy</ng-container> | ||||
|               </button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|  | ||||
| @ -0,0 +1,4 @@ | ||||
| // hide caret on mobile dropdown | ||||
| .d-block.d-sm-none .dropdown-toggle::after { | ||||
|   display: none; | ||||
| } | ||||
| @ -26,6 +26,7 @@ import { | ||||
| import { WorkflowActionType } from 'src/app/data/workflow-action' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component' | ||||
| 
 | ||||
| const workflows: Workflow[] = [ | ||||
|   { | ||||
| @ -173,6 +174,19 @@ describe('WorkflowsComponent', () => { | ||||
|     expect(reloadSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support copy', () => { | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
| 
 | ||||
|     const copyButton = fixture.debugElement.queryAll(By.css('button'))[6] | ||||
|     copyButton.triggerEventHandler('click') | ||||
| 
 | ||||
|     expect(modal).not.toBeUndefined() | ||||
|     const editDialog = modal.componentInstance as WorkflowEditDialogComponent | ||||
|     expect(editDialog.object.name).toEqual(workflows[0].name + ' (copy)') | ||||
|     expect(editDialog.dialogMode).toEqual(EditDialogMode.CREATE) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support delete, show notification on error / success', () => { | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
| @ -180,7 +194,7 @@ describe('WorkflowsComponent', () => { | ||||
|     const deleteSpy = jest.spyOn(workflowService, 'delete') | ||||
|     const reloadSpy = jest.spyOn(component, 'reload') | ||||
| 
 | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[4] | ||||
|     const deleteButton = fixture.debugElement.queryAll(By.css('button'))[5] | ||||
|     deleteButton.triggerEventHandler('click') | ||||
| 
 | ||||
|     expect(modal).not.toBeUndefined() | ||||
| @ -197,4 +211,27 @@ describe('WorkflowsComponent', () => { | ||||
|     editDialog.confirmClicked.emit() | ||||
|     expect(reloadSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| 
 | ||||
|   it('should update workflow when enable is toggled', () => { | ||||
|     const patchSpy = jest.spyOn(workflowService, 'patch') | ||||
|     const toggleInput = fixture.debugElement.query( | ||||
|       By.css('input[type="checkbox"]') | ||||
|     ) | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     // fail first
 | ||||
|     patchSpy.mockReturnValueOnce( | ||||
|       throwError(() => new Error('Error getting config')) | ||||
|     ) | ||||
|     toggleInput.nativeElement.click() | ||||
|     expect(patchSpy).toHaveBeenCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     // succeed second
 | ||||
|     patchSpy.mockReturnValueOnce(of(workflows[0])) | ||||
|     toggleInput.nativeElement.click() | ||||
|     patchSpy.mockReturnValueOnce(of({ ...workflows[0], enabled: false })) | ||||
|     toggleInput.nativeElement.click() | ||||
|     expect(patchSpy).toHaveBeenCalled() | ||||
|     expect(toastInfoSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -57,14 +57,13 @@ export class WorkflowsComponent | ||||
|       .join(', ') | ||||
|   } | ||||
| 
 | ||||
|   editWorkflow(workflow: Workflow) { | ||||
|   editWorkflow(workflow: Workflow, forceCreate: boolean = false) { | ||||
|     const modal = this.modalService.open(WorkflowEditDialogComponent, { | ||||
|       backdrop: 'static', | ||||
|       size: 'xl', | ||||
|     }) | ||||
|     modal.componentInstance.dialogMode = workflow | ||||
|       ? EditDialogMode.EDIT | ||||
|       : EditDialogMode.CREATE | ||||
|     modal.componentInstance.dialogMode = | ||||
|       workflow && !forceCreate ? EditDialogMode.EDIT : EditDialogMode.CREATE | ||||
|     if (workflow) { | ||||
|       // quick "deep" clone so original doesn't get modified
 | ||||
|       const clone = Object.assign({}, workflow) | ||||
| @ -88,6 +87,25 @@ export class WorkflowsComponent | ||||
|       }) | ||||
|   } | ||||
| 
 | ||||
|   copyWorkflow(workflow: Workflow) { | ||||
|     const clone = Object.assign({}, workflow) | ||||
|     clone.id = null | ||||
|     clone.name = `${workflow.name} (copy)` | ||||
|     clone.actions = [ | ||||
|       ...workflow.actions.map((a) => { | ||||
|         a.id = null | ||||
|         return a | ||||
|       }), | ||||
|     ] | ||||
|     clone.triggers = [ | ||||
|       ...workflow.triggers.map((t) => { | ||||
|         t.id = null | ||||
|         return t | ||||
|       }), | ||||
|     ] | ||||
|     this.editWorkflow(clone, true) | ||||
|   } | ||||
| 
 | ||||
|   deleteWorkflow(workflow: Workflow) { | ||||
|     const modal = this.modalService.open(ConfirmDialogComponent, { | ||||
|       backdrop: 'static', | ||||
| @ -112,4 +130,21 @@ export class WorkflowsComponent | ||||
|       }) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   onWorkflowEnableToggled(workflow: Workflow) { | ||||
|     this.workflowService.patch(workflow).subscribe({ | ||||
|       next: () => { | ||||
|         this.toastService.showInfo( | ||||
|           workflow.enabled | ||||
|             ? $localize`Enabled workflow` | ||||
|             : $localize`Disabled workflow` | ||||
|         ) | ||||
|         this.workflowService.clearCache() | ||||
|         this.reload() | ||||
|       }, | ||||
|       error: (e) => { | ||||
|         this.toastService.showError($localize`Error toggling workflow.`, e) | ||||
|       }, | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|  | ||||
							
								
								
									
										127
									
								
								src-ui/src/app/data/custom-field-query.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								src-ui/src/app/data/custom-field-query.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,127 @@ | ||||
| import { CustomFieldDataType } from './custom-field' | ||||
| 
 | ||||
| export enum CustomFieldQueryLogicalOperator { | ||||
|   And = 'AND', | ||||
|   Or = 'OR', | ||||
|   Not = 'NOT', | ||||
| } | ||||
| 
 | ||||
| export enum CustomFieldQueryOperator { | ||||
|   Exact = 'exact', | ||||
|   In = 'in', | ||||
|   IsNull = 'isnull', | ||||
|   Exists = 'exists', | ||||
|   Contains = 'contains', | ||||
|   IContains = 'icontains', | ||||
|   GreaterThan = 'gt', | ||||
|   GreaterThanOrEqual = 'gte', | ||||
|   LessThan = 'lt', | ||||
|   LessThanOrEqual = 'lte', | ||||
|   Range = 'range', | ||||
| } | ||||
| 
 | ||||
| export const CUSTOM_FIELD_QUERY_OPERATOR_LABELS = { | ||||
|   [CustomFieldQueryOperator.Exact]: $localize`Equal to`, | ||||
|   [CustomFieldQueryOperator.In]: $localize`In`, | ||||
|   [CustomFieldQueryOperator.IsNull]: $localize`Is null`, | ||||
|   [CustomFieldQueryOperator.Exists]: $localize`Exists`, | ||||
|   [CustomFieldQueryOperator.Contains]: $localize`Contains`, | ||||
|   [CustomFieldQueryOperator.IContains]: $localize`Contains (case-insensitive)`, | ||||
|   [CustomFieldQueryOperator.GreaterThan]: $localize`Greater than`, | ||||
|   [CustomFieldQueryOperator.GreaterThanOrEqual]: $localize`Greater than or equal to`, | ||||
|   [CustomFieldQueryOperator.LessThan]: $localize`Less than`, | ||||
|   [CustomFieldQueryOperator.LessThanOrEqual]: $localize`Less than or equal to`, | ||||
|   [CustomFieldQueryOperator.Range]: $localize`Range`, | ||||
| } | ||||
| 
 | ||||
| export enum CustomFieldQueryOperatorGroups { | ||||
|   Basic = 'basic', | ||||
|   String = 'string', | ||||
|   Arithmetic = 'arithmetic', | ||||
|   Containment = 'containment', | ||||
|   Subset = 'subset', | ||||
|   Date = 'date', | ||||
| } | ||||
| 
 | ||||
| // Modified from filters.py > SUPPORTED_EXPR_OPERATORS
 | ||||
| export const CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP = { | ||||
|   [CustomFieldQueryOperatorGroups.Basic]: [ | ||||
|     CustomFieldQueryOperator.Exists, | ||||
|     CustomFieldQueryOperator.IsNull, | ||||
|     CustomFieldQueryOperator.Exact, | ||||
|   ], | ||||
|   [CustomFieldQueryOperatorGroups.String]: [CustomFieldQueryOperator.IContains], | ||||
|   [CustomFieldQueryOperatorGroups.Arithmetic]: [ | ||||
|     CustomFieldQueryOperator.GreaterThan, | ||||
|     CustomFieldQueryOperator.GreaterThanOrEqual, | ||||
|     CustomFieldQueryOperator.LessThan, | ||||
|     CustomFieldQueryOperator.LessThanOrEqual, | ||||
|   ], | ||||
|   [CustomFieldQueryOperatorGroups.Containment]: [ | ||||
|     CustomFieldQueryOperator.Contains, | ||||
|   ], | ||||
|   [CustomFieldQueryOperatorGroups.Subset]: [CustomFieldQueryOperator.In], | ||||
|   [CustomFieldQueryOperatorGroups.Date]: [ | ||||
|     CustomFieldQueryOperator.GreaterThanOrEqual, | ||||
|     CustomFieldQueryOperator.LessThanOrEqual, | ||||
|   ], | ||||
| } | ||||
| 
 | ||||
| // filters.py > SUPPORTED_EXPR_CATEGORIES
 | ||||
| export const CUSTOM_FIELD_QUERY_OPERATOR_GROUPS_BY_TYPE = { | ||||
|   [CustomFieldDataType.String]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.String, | ||||
|   ], | ||||
|   [CustomFieldDataType.Url]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.String, | ||||
|   ], | ||||
|   [CustomFieldDataType.Date]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.Date, | ||||
|   ], | ||||
|   [CustomFieldDataType.Boolean]: [CustomFieldQueryOperatorGroups.Basic], | ||||
|   [CustomFieldDataType.Integer]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.Arithmetic, | ||||
|   ], | ||||
|   [CustomFieldDataType.Float]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.Arithmetic, | ||||
|   ], | ||||
|   [CustomFieldDataType.Monetary]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.String, | ||||
|     CustomFieldQueryOperatorGroups.Arithmetic, | ||||
|   ], | ||||
|   [CustomFieldDataType.DocumentLink]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.Containment, | ||||
|   ], | ||||
|   [CustomFieldDataType.Select]: [ | ||||
|     CustomFieldQueryOperatorGroups.Basic, | ||||
|     CustomFieldQueryOperatorGroups.Subset, | ||||
|   ], | ||||
| } | ||||
| 
 | ||||
| export const CUSTOM_FIELD_QUERY_VALUE_TYPES_BY_OPERATOR = { | ||||
|   [CustomFieldQueryOperator.Exact]: 'string|boolean', | ||||
|   [CustomFieldQueryOperator.IsNull]: 'boolean', | ||||
|   [CustomFieldQueryOperator.Exists]: 'boolean', | ||||
|   [CustomFieldQueryOperator.IContains]: 'string', | ||||
|   [CustomFieldQueryOperator.GreaterThanOrEqual]: 'string|number', | ||||
|   [CustomFieldQueryOperator.LessThanOrEqual]: 'string|number', | ||||
|   [CustomFieldQueryOperator.GreaterThan]: 'number', | ||||
|   [CustomFieldQueryOperator.LessThan]: 'number', | ||||
|   [CustomFieldQueryOperator.Contains]: 'array', | ||||
|   [CustomFieldQueryOperator.In]: 'array', | ||||
| } | ||||
| 
 | ||||
| export const CUSTOM_FIELD_QUERY_MAX_DEPTH = 4 | ||||
| export const CUSTOM_FIELD_QUERY_MAX_ATOMS = 5 | ||||
| 
 | ||||
| export enum CustomFieldQueryElementType { | ||||
|   Atom = 'Atom', | ||||
|   Expression = 'Expression', | ||||
| } | ||||
| @ -59,4 +59,5 @@ export interface CustomField extends ObjectWithId { | ||||
|     select_options?: string[] | ||||
|     default_currency?: string | ||||
|   } | ||||
|   document_count?: number | ||||
| } | ||||
|  | ||||
| @ -26,6 +26,7 @@ export enum DisplayField { | ||||
|   OWNER = 'owner', | ||||
|   SHARED = 'shared', | ||||
|   ASN = 'asn', | ||||
|   PAGE_COUNT = 'pagecount', | ||||
| } | ||||
| 
 | ||||
| export const DEFAULT_DISPLAY_FIELDS = [ | ||||
| @ -73,6 +74,10 @@ export const DEFAULT_DISPLAY_FIELDS = [ | ||||
|     id: DisplayField.ASN, | ||||
|     name: $localize`ASN`, | ||||
|   }, | ||||
|   { | ||||
|     id: DisplayField.PAGE_COUNT, | ||||
|     name: $localize`Pages`, | ||||
|   }, | ||||
| ] | ||||
| 
 | ||||
| export const DEFAULT_DASHBOARD_VIEW_PAGE_SIZE = 10 | ||||
| @ -94,6 +99,7 @@ export const DOCUMENT_SORT_FIELDS = [ | ||||
|   { field: 'modified', name: $localize`Modified` }, | ||||
|   { field: 'num_notes', name: $localize`Notes` }, | ||||
|   { field: 'owner', name: $localize`Owner` }, | ||||
|   { field: 'page_count', name: $localize`Pages` }, | ||||
| ] | ||||
| 
 | ||||
| export const DOCUMENT_SORT_FIELDS_FULLTEXT = [ | ||||
| @ -164,4 +170,6 @@ export interface Document extends ObjectWithPermissions { | ||||
| 
 | ||||
|   // write-only field
 | ||||
|   remove_inbox_tags?: boolean | ||||
| 
 | ||||
|   page_count?: number | ||||
| } | ||||
|  | ||||
| @ -55,6 +55,8 @@ export const FILTER_HAS_CUSTOM_FIELDS_ANY = 39 | ||||
| export const FILTER_DOES_NOT_HAVE_CUSTOM_FIELDS = 40 | ||||
| export const FILTER_HAS_ANY_CUSTOM_FIELDS = 41 | ||||
| 
 | ||||
| export const FILTER_CUSTOM_FIELDS_QUERY = 42 | ||||
| 
 | ||||
| export const FILTER_RULE_TYPES: FilterRuleType[] = [ | ||||
|   { | ||||
|     id: FILTER_TITLE, | ||||
| @ -317,6 +319,12 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ | ||||
|     multi: false, | ||||
|     default: true, | ||||
|   }, | ||||
|   { | ||||
|     id: FILTER_CUSTOM_FIELDS_QUERY, | ||||
|     filtervar: 'custom_field_query', | ||||
|     datatype: 'string', | ||||
|     multi: false, | ||||
|   }, | ||||
| ] | ||||
| 
 | ||||
| export interface FilterRuleType { | ||||
|  | ||||
| @ -6,6 +6,12 @@ export enum IMAPSecurity { | ||||
|   STARTTLS = 3, | ||||
| } | ||||
| 
 | ||||
| export enum MailAccountType { | ||||
|   IMAP = 1, | ||||
|   Gmail_OAuth = 2, | ||||
|   Outlook_OAuth = 3, | ||||
| } | ||||
| 
 | ||||
| export interface MailAccount extends ObjectWithPermissions { | ||||
|   name: string | ||||
| 
 | ||||
| @ -22,4 +28,8 @@ export interface MailAccount extends ObjectWithPermissions { | ||||
|   character_set?: string | ||||
| 
 | ||||
|   is_token: boolean | ||||
| 
 | ||||
|   account_type: MailAccountType | ||||
| 
 | ||||
|   expiration?: string // Date
 | ||||
| } | ||||
|  | ||||
| @ -39,6 +39,8 @@ export interface MailRule extends ObjectWithPermissions { | ||||
| 
 | ||||
|   order: number | ||||
| 
 | ||||
|   enabled: boolean | ||||
| 
 | ||||
|   folder: string | ||||
| 
 | ||||
|   filter_from: string | ||||
|  | ||||
| @ -64,6 +64,8 @@ export const SETTINGS_KEYS = { | ||||
|   SEARCH_DB_ONLY: 'general-settings:search:db-only', | ||||
|   SEARCH_FULL_TYPE: 'general-settings:search:more-link', | ||||
|   EMPTY_TRASH_DELAY: 'trash_delay', | ||||
|   GMAIL_OAUTH_URL: 'gmail_oauth_url', | ||||
|   OUTLOOK_OAUTH_URL: 'outlook_oauth_url', | ||||
| } | ||||
| 
 | ||||
| export const SETTINGS: UiSetting[] = [ | ||||
| @ -242,4 +244,14 @@ export const SETTINGS: UiSetting[] = [ | ||||
|     type: 'number', | ||||
|     default: 30, | ||||
|   }, | ||||
|   { | ||||
|     key: SETTINGS_KEYS.GMAIL_OAUTH_URL, | ||||
|     type: 'string', | ||||
|     default: null, | ||||
|   }, | ||||
|   { | ||||
|     key: SETTINGS_KEYS.OUTLOOK_OAUTH_URL, | ||||
|     type: 'string', | ||||
|     default: null, | ||||
|   }, | ||||
| ] | ||||
|  | ||||
| @ -4,7 +4,7 @@ import { TestBed } from '@angular/core/testing' | ||||
| import { environment } from 'src/environments/environment' | ||||
| import { commonAbstractPaperlessServiceTests } from './abstract-paperless-service.spec' | ||||
| import { MailAccountService } from './mail-account.service' | ||||
| import { IMAPSecurity } from 'src/app/data/mail-account' | ||||
| import { IMAPSecurity, MailAccountType } from 'src/app/data/mail-account' | ||||
| 
 | ||||
| let httpTestingController: HttpTestingController | ||||
| let service: MailAccountService | ||||
| @ -20,6 +20,7 @@ const mail_accounts = [ | ||||
|     username: 'user', | ||||
|     password: 'pass', | ||||
|     is_token: false, | ||||
|     account_type: MailAccountType.IMAP, | ||||
|   }, | ||||
|   { | ||||
|     name: 'Mail Account 2', | ||||
| @ -30,6 +31,7 @@ const mail_accounts = [ | ||||
|     username: 'user', | ||||
|     password: 'pass', | ||||
|     is_token: false, | ||||
|     account_type: MailAccountType.IMAP, | ||||
|   }, | ||||
|   { | ||||
|     name: 'Mail Account 3', | ||||
| @ -40,6 +42,7 @@ const mail_accounts = [ | ||||
|     username: 'user', | ||||
|     password: 'pass', | ||||
|     is_token: false, | ||||
|     account_type: MailAccountType.IMAP, | ||||
|   }, | ||||
| ] | ||||
| 
 | ||||
| @ -55,20 +58,6 @@ describe(`Additional service tests for MailAccountService`, () => { | ||||
|     expect(req.request.method).toEqual('POST') | ||||
|   }) | ||||
| 
 | ||||
|   it('should support patchMany', () => { | ||||
|     subscription = service.patchMany(mail_accounts).subscribe() | ||||
|     mail_accounts.forEach((mail_account) => { | ||||
|       const req = httpTestingController.expectOne( | ||||
|         `${environment.apiBaseUrl}${endpoint}/${mail_account.id}/` | ||||
|       ) | ||||
|       expect(req.request.method).toEqual('PATCH') | ||||
|       req.flush(mail_account) | ||||
|     }) | ||||
|     httpTestingController.expectOne( | ||||
|       `${environment.apiBaseUrl}${endpoint}/?page=1&page_size=100000` | ||||
|     ) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support reload', () => { | ||||
|     service['reload']() | ||||
|     const req = httpTestingController.expectOne( | ||||
|  | ||||
| @ -1,6 +1,5 @@ | ||||
| import { HttpClient } from '@angular/common/http' | ||||
| import { Injectable } from '@angular/core' | ||||
| import { combineLatest, Observable } from 'rxjs' | ||||
| import { tap } from 'rxjs/operators' | ||||
| import { MailAccount } from 'src/app/data/mail-account' | ||||
| import { AbstractPaperlessService } from './abstract-paperless-service' | ||||
| @ -34,15 +33,11 @@ export class MailAccountService extends AbstractPaperlessService<MailAccount> { | ||||
|   } | ||||
| 
 | ||||
|   update(o: MailAccount) { | ||||
|     // Remove expiration from the object before updating
 | ||||
|     delete o.expiration | ||||
|     return super.update(o).pipe(tap(() => this.reload())) | ||||
|   } | ||||
| 
 | ||||
|   patchMany(objects: MailAccount[]): Observable<MailAccount[]> { | ||||
|     return combineLatest(objects.map((o) => super.patch(o))).pipe( | ||||
|       tap(() => this.reload()) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   delete(o: MailAccount) { | ||||
|     return super.delete(o).pipe(tap(() => this.reload())) | ||||
|   } | ||||
|  | ||||
| @ -18,6 +18,7 @@ const mail_rules = [ | ||||
|     id: 1, | ||||
|     account: 1, | ||||
|     order: 1, | ||||
|     enabled: true, | ||||
|     folder: 'INBOX', | ||||
|     filter_from: null, | ||||
|     filter_to: null, | ||||
| @ -36,6 +37,7 @@ const mail_rules = [ | ||||
|     id: 2, | ||||
|     account: 1, | ||||
|     order: 1, | ||||
|     enabled: true, | ||||
|     folder: 'INBOX', | ||||
|     filter_from: null, | ||||
|     filter_to: null, | ||||
| @ -54,6 +56,7 @@ const mail_rules = [ | ||||
|     id: 3, | ||||
|     account: 1, | ||||
|     order: 1, | ||||
|     enabled: true, | ||||
|     folder: 'INBOX', | ||||
|     filter_from: null, | ||||
|     filter_to: null, | ||||
| @ -73,21 +76,6 @@ const mail_rules = [ | ||||
| commonAbstractPaperlessServiceTests(endpoint, MailRuleService) | ||||
| 
 | ||||
| describe(`Additional service tests for MailRuleService`, () => { | ||||
|   it('should support patchMany', () => { | ||||
|     subscription = service.patchMany(mail_rules).subscribe() | ||||
|     mail_rules.forEach((mail_rule) => { | ||||
|       const req = httpTestingController.expectOne( | ||||
|         `${environment.apiBaseUrl}${endpoint}/${mail_rule.id}/` | ||||
|       ) | ||||
|       expect(req.request.method).toEqual('PATCH') | ||||
|       req.flush(mail_rule) | ||||
|     }) | ||||
|     const reloadReq = httpTestingController.expectOne( | ||||
|       `${environment.apiBaseUrl}${endpoint}/?page=1&page_size=100000` | ||||
|     ) | ||||
|     reloadReq.flush({ results: mail_rules }) | ||||
|   }) | ||||
| 
 | ||||
|   it('should support reload', () => { | ||||
|     service['reload']() | ||||
|     const req = httpTestingController.expectOne( | ||||
|  | ||||
| @ -37,12 +37,6 @@ export class MailRuleService extends AbstractPaperlessService<MailRule> { | ||||
|     return super.update(o).pipe(tap(() => this.reload())) | ||||
|   } | ||||
| 
 | ||||
|   patchMany(objects: MailRule[]): Observable<MailRule[]> { | ||||
|     return combineLatest(objects.map((o) => super.patch(o))).pipe( | ||||
|       tap(() => this.reload()) | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   delete(o: MailRule) { | ||||
|     return super.delete(o).pipe(tap(() => this.reload())) | ||||
|   } | ||||
|  | ||||
| @ -1,7 +1,35 @@ | ||||
| import { StoragePathService } from './storage-path.service' | ||||
| import { commonAbstractNameFilterPaperlessServiceTests } from './abstract-name-filter-service.spec' | ||||
| import { Subscription } from 'rxjs' | ||||
| import { HttpTestingController } from '@angular/common/http/testing' | ||||
| import { TestBed } from '@angular/core/testing' | ||||
| import { environment } from 'src/environments/environment' | ||||
| 
 | ||||
| let httpTestingController: HttpTestingController | ||||
| let service: StoragePathService | ||||
| let subscription: Subscription | ||||
| const endpoint = 'storage_paths' | ||||
| 
 | ||||
| commonAbstractNameFilterPaperlessServiceTests( | ||||
|   'storage_paths', | ||||
|   StoragePathService | ||||
| ) | ||||
| 
 | ||||
| describe(`Additional service tests for StoragePathservice`, () => { | ||||
|   beforeEach(() => { | ||||
|     httpTestingController = TestBed.inject(HttpTestingController) | ||||
|     service = TestBed.inject(StoragePathService) | ||||
|   }) | ||||
| 
 | ||||
|   afterEach(() => { | ||||
|     subscription?.unsubscribe() | ||||
|     httpTestingController.verify() | ||||
|   }) | ||||
| 
 | ||||
|   it('should support testing path', () => { | ||||
|     subscription = service.testPath('path', 11).subscribe() | ||||
|     httpTestingController | ||||
|       .expectOne(`${environment.apiBaseUrl}${endpoint}/test/`) | ||||
|       .flush('ok') | ||||
|   }) | ||||
| }) | ||||
|  | ||||
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