mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Merge branch 'dev'
This commit is contained in:
commit
5821033e3d
@ -1,117 +0,0 @@
|
|||||||
# Paperless NGX Development Environment
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
Welcome to the Paperless NGX development environment! This setup uses VSCode DevContainers to provide a consistent and seamless development experience.
|
|
||||||
|
|
||||||
### What are DevContainers?
|
|
||||||
|
|
||||||
DevContainers are a feature in VSCode that allows you to develop within a Docker container. This ensures that your development environment is consistent across different machines and setups. By defining a containerized environment, you can eliminate the "works on my machine" problem.
|
|
||||||
|
|
||||||
### Advantages of DevContainers
|
|
||||||
|
|
||||||
- **Consistency**: Same environment for all developers.
|
|
||||||
- **Isolation**: Separate development environment from your local machine.
|
|
||||||
- **Reproducibility**: Easily recreate the environment on any machine.
|
|
||||||
- **Pre-configured Tools**: Include all necessary tools and dependencies in the container.
|
|
||||||
|
|
||||||
## DevContainer Setup
|
|
||||||
|
|
||||||
The DevContainer configuration provides up all the necessary services for Paperless NGX, including:
|
|
||||||
|
|
||||||
- Redis
|
|
||||||
- Gotenberg
|
|
||||||
- Tika
|
|
||||||
|
|
||||||
Data is stored using Docker volumes to ensure persistence across container restarts.
|
|
||||||
|
|
||||||
## Configuration Files
|
|
||||||
|
|
||||||
The setup includes debugging configurations (`launch.json`) and tasks (`tasks.json`) to help you manage and debug various parts of the project:
|
|
||||||
|
|
||||||
- **Backend Debugging:**
|
|
||||||
- `manage.py runserver`
|
|
||||||
- `manage.py document-consumer`
|
|
||||||
- `celery`
|
|
||||||
- **Maintenance Tasks:**
|
|
||||||
- Create superuser
|
|
||||||
- Run migrations
|
|
||||||
- Recreate virtual environment (`.venv` with pipenv)
|
|
||||||
- Compile frontend assets
|
|
||||||
|
|
||||||
## Getting Started
|
|
||||||
|
|
||||||
### Step 1: Running the DevContainer
|
|
||||||
|
|
||||||
To start the DevContainer:
|
|
||||||
|
|
||||||
1. Open VSCode.
|
|
||||||
2. Open the project folder.
|
|
||||||
3. Open the command palette:
|
|
||||||
- **Windows/Linux**: `Ctrl+Shift+P`
|
|
||||||
- **Mac**: `Cmd+Shift+P`
|
|
||||||
4. Type and select `Dev Containers: Rebuild and Reopen in Container`.
|
|
||||||
|
|
||||||
VSCode will build and start the DevContainer environment.
|
|
||||||
|
|
||||||
### Step 2: Initial Setup
|
|
||||||
|
|
||||||
Once the DevContainer is up and running, perform the following steps:
|
|
||||||
|
|
||||||
1. **Compile Frontend Assets**:
|
|
||||||
|
|
||||||
- Open the command palette:
|
|
||||||
- **Windows/Linux**: `Ctrl+Shift+P`
|
|
||||||
- **Mac**: `Cmd+Shift+P`
|
|
||||||
- Select `Tasks: Run Task`.
|
|
||||||
- Choose `Frontend Compile`.
|
|
||||||
|
|
||||||
2. **Run Database Migrations**:
|
|
||||||
|
|
||||||
- Open the command palette:
|
|
||||||
- **Windows/Linux**: `Ctrl+Shift+P`
|
|
||||||
- **Mac**: `Cmd+Shift+P`
|
|
||||||
- Select `Tasks: Run Task`.
|
|
||||||
- Choose `Migrate Database`.
|
|
||||||
|
|
||||||
3. **Create Superuser**:
|
|
||||||
- Open the command palette:
|
|
||||||
- **Windows/Linux**: `Ctrl+Shift+P`
|
|
||||||
- **Mac**: `Cmd+Shift+P`
|
|
||||||
- Select `Tasks: Run Task`.
|
|
||||||
- Choose `Create Superuser`.
|
|
||||||
|
|
||||||
### Debugging and Running Services
|
|
||||||
|
|
||||||
You can start and debug backend services either as debugging sessions via `launch.json` or as tasks.
|
|
||||||
|
|
||||||
#### Using `launch.json`:
|
|
||||||
|
|
||||||
1. Press `F5` or go to the **Run and Debug** view in VSCode.
|
|
||||||
2. Select the desired configuration:
|
|
||||||
- `Runserver`
|
|
||||||
- `Document Consumer`
|
|
||||||
- `Celery`
|
|
||||||
|
|
||||||
#### Using Tasks:
|
|
||||||
|
|
||||||
1. Open the command palette:
|
|
||||||
- **Windows/Linux**: `Ctrl+Shift+P`
|
|
||||||
- **Mac**: `Cmd+Shift+P`
|
|
||||||
2. Select `Tasks: Run Task`.
|
|
||||||
3. Choose the desired task:
|
|
||||||
- `Runserver`
|
|
||||||
- `Document Consumer`
|
|
||||||
- `Celery`
|
|
||||||
|
|
||||||
### Additional Maintenance Tasks
|
|
||||||
|
|
||||||
Additional tasks are available for common maintenance operations:
|
|
||||||
|
|
||||||
- **Recreate .venv**: For setting up the virtual environment using pipenv.
|
|
||||||
- **Migrate Database**: To apply database migrations.
|
|
||||||
- **Create Superuser**: To create an admin user for the application.
|
|
||||||
|
|
||||||
## Let's Get Started!
|
|
||||||
|
|
||||||
Follow the steps above to get your development environment up and running. Happy coding!
|
|
@ -3,14 +3,26 @@
|
|||||||
"dockerComposeFile": "docker-compose.devcontainer.sqlite-tika.yml",
|
"dockerComposeFile": "docker-compose.devcontainer.sqlite-tika.yml",
|
||||||
"service": "paperless-development",
|
"service": "paperless-development",
|
||||||
"workspaceFolder": "/usr/src/paperless/paperless-ngx",
|
"workspaceFolder": "/usr/src/paperless/paperless-ngx",
|
||||||
"postCreateCommand": "/bin/bash -c pre-commit install && pipenv install --dev",
|
"postCreateCommand": "pipenv install --dev && pipenv run pre-commit install",
|
||||||
"customizations": {
|
"customizations": {
|
||||||
"vscode": {
|
"vscode": {
|
||||||
"extensions": [
|
"extensions": [
|
||||||
"mhutchie.git-graph",
|
"mhutchie.git-graph",
|
||||||
"ms-python.python"
|
"ms-python.python",
|
||||||
]
|
"ms-vscode.js-debug-nightly",
|
||||||
|
"eamodio.gitlens",
|
||||||
|
"yzhang.markdown-all-in-one"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"python.defaultInterpreterPath": "/usr/src/paperless/paperless-ngx/.venv/bin/python",
|
||||||
|
"python.pythonPath": "/usr/src/paperless/paperless-ngx/.venv/bin/python",
|
||||||
|
"python.terminal.activateEnvInCurrentTerminal": true,
|
||||||
|
"editor.formatOnPaste": false,
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.formatOnType": true,
|
||||||
|
"files.trimTrailingWhitespace": true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"remoteUser": "paperless"
|
"remoteUser": "paperless"
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,7 @@ services:
|
|||||||
image: docker.io/library/redis:7
|
image: docker.io/library/redis:7
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
volumes:
|
volumes:
|
||||||
- redisdata:/data
|
- ./redisdata:/data
|
||||||
|
|
||||||
# No ports need to be exposed; the VSCode DevContainer plugin manages them.
|
# No ports need to be exposed; the VSCode DevContainer plugin manages them.
|
||||||
paperless-development:
|
paperless-development:
|
||||||
@ -43,14 +43,16 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- ..:/usr/src/paperless/paperless-ngx:delegated
|
- ..:/usr/src/paperless/paperless-ngx:delegated
|
||||||
- ../.devcontainer/vscode:/usr/src/paperless/paperless-ngx/.vscode:delegated # VSCode config files
|
- ../.devcontainer/vscode:/usr/src/paperless/paperless-ngx/.vscode:delegated # VSCode config files
|
||||||
- pipenv:/usr/src/paperless/paperless-ngx/.venv # Pipenv environment persisted in volume
|
- pipenv:/usr/src/paperless/paperless-ngx/.venv
|
||||||
- /usr/src/paperless/paperless-ngx/src/documents/static/frontend # Static frontend files exist only in container
|
- /usr/src/paperless/paperless-ngx/src/documents/static/frontend # Static frontend files exist only in container
|
||||||
- /usr/src/paperless/paperless-ngx/src/.pytest_cache
|
- /usr/src/paperless/paperless-ngx/src/.pytest_cache
|
||||||
- /usr/src/paperless/paperless-ngx/.ruff_cache
|
- /usr/src/paperless/paperless-ngx/.ruff_cache
|
||||||
- /usr/src/paperless/paperless-ngx/htmlcov
|
- /usr/src/paperless/paperless-ngx/htmlcov
|
||||||
- /usr/src/paperless/paperless-ngx/.coverage
|
- /usr/src/paperless/paperless-ngx/.coverage
|
||||||
- data:/usr/src/paperless/paperless-ngx/data
|
- ./data:/usr/src/paperless/paperless-ngx/data
|
||||||
- media:/usr/src/paperless/paperless-ngx/media
|
- ./media:/usr/src/paperless/paperless-ngx/media
|
||||||
|
- ./consume:/usr/src/paperless/paperless-ngx/consume
|
||||||
|
- ~/.gitconfig:/usr/src/paperless/.gitconfig:ro
|
||||||
environment:
|
environment:
|
||||||
PAPERLESS_REDIS: redis://broker:6379
|
PAPERLESS_REDIS: redis://broker:6379
|
||||||
PAPERLESS_TIKA_ENABLED: 1
|
PAPERLESS_TIKA_ENABLED: 1
|
||||||
@ -78,7 +80,4 @@ services:
|
|||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
data:
|
|
||||||
media:
|
|
||||||
redisdata:
|
|
||||||
pipenv:
|
pipenv:
|
||||||
|
@ -2,42 +2,57 @@
|
|||||||
"version": "0.2.0",
|
"version": "0.2.0",
|
||||||
"configurations": [
|
"configurations": [
|
||||||
{
|
{
|
||||||
"name": "manage.py runserver",
|
"name": "Chrome: Debug Angular Frontend",
|
||||||
|
"description": "Debug the Angular Dev Frontend in Chrome",
|
||||||
|
"type": "chrome",
|
||||||
|
"request": "launch",
|
||||||
|
"url": "http://localhost:4200",
|
||||||
|
"webRoot": "${workspaceFolder}/src-ui",
|
||||||
|
"preLaunchTask": "Start: Frontend Angular"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug: Backend Server (manage.py runserver)",
|
||||||
|
"description": "Debug the Django Backend Server",
|
||||||
"type": "python",
|
"type": "python",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"program": "${workspaceFolder}/src/manage.py",
|
"program": "${workspaceFolder}/src/manage.py",
|
||||||
"console": "integratedTerminal",
|
"args": [
|
||||||
"justMyCode": true,
|
"runserver"
|
||||||
"args": ["runserver"],
|
],
|
||||||
"django": true
|
"django": true,
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "manage.py document_consumer",
|
|
||||||
"type": "python",
|
|
||||||
"request": "launch",
|
|
||||||
"program": "${workspaceFolder}/src/manage.py",
|
|
||||||
"console": "integratedTerminal",
|
|
||||||
"justMyCode": true,
|
|
||||||
"args": ["document_consumer"],
|
|
||||||
"django": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "celery",
|
|
||||||
"type": "python",
|
|
||||||
"cwd": "${workspaceFolder}/src",
|
|
||||||
"request": "launch",
|
|
||||||
"module": "celery",
|
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"env": {
|
"env": {
|
||||||
"PYTHONPATH": "${workspaceFolder}/src"
|
"PYTHONPATH": "${workspaceFolder}/src"
|
||||||
},
|
},
|
||||||
|
"python": "${workspaceFolder}/.venv/bin/python"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug: Consumer Service (manage.py document_consumer)",
|
||||||
|
"description": "Debug the Consumer Service which processes files from a directory",
|
||||||
|
"type": "python",
|
||||||
|
"request": "launch",
|
||||||
|
"program": "${workspaceFolder}/src/manage.py",
|
||||||
"args": [
|
"args": [
|
||||||
"-A",
|
"document_consumer"
|
||||||
"paperless",
|
],
|
||||||
"worker",
|
"django": true,
|
||||||
"-l",
|
"console": "integratedTerminal",
|
||||||
"DEBUG"
|
"env": {
|
||||||
]
|
"PYTHONPATH": "${workspaceFolder}/src"
|
||||||
|
},
|
||||||
|
"python": "${workspaceFolder}/.venv/bin/python"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"compounds": [
|
||||||
|
{
|
||||||
|
"name": "Debug: FullStack",
|
||||||
|
"description": "Debug run the Angular dev frontend, Django backend, and consumer service",
|
||||||
|
"configurations": [
|
||||||
|
"Chrome: Debug Angular Frontend",
|
||||||
|
"Debug: Backend Server (manage.py runserver)",
|
||||||
|
"Debug: Consumer Service (manage.py document_consumer)"
|
||||||
|
],
|
||||||
|
"preLaunchTask": "Start: Celery Worker"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -1,27 +1,84 @@
|
|||||||
{
|
{
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"tasks": [
|
"tasks": [
|
||||||
{
|
{
|
||||||
"label": "manage.py document_consumer",
|
"label": "Start: Celery Worker",
|
||||||
"type": "shell",
|
"description": "Start the Celery Worker which processes background and consume tasks",
|
||||||
"command": "pipenv run python manage.py document_consumer",
|
"type": "shell",
|
||||||
"group": "build",
|
"command": "pipenv run celery --app paperless worker -l DEBUG",
|
||||||
"presentation": {
|
"isBackground": true,
|
||||||
"echo": true,
|
"options": {
|
||||||
"reveal": "always",
|
"cwd": "${workspaceFolder}/src"
|
||||||
"focus": false,
|
},
|
||||||
"panel": "shared",
|
"problemMatcher": [
|
||||||
"showReuseMessage": false,
|
{
|
||||||
"clear": true,
|
"owner": "custom",
|
||||||
"revealProblems": "onProblem"
|
"pattern": [
|
||||||
},
|
{
|
||||||
"options": {
|
"regexp": ".",
|
||||||
"cwd": "${workspaceFolder}/src"
|
"file": 1,
|
||||||
}
|
"location": 2,
|
||||||
|
"message": 3
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "celery.*",
|
||||||
|
"endsPattern": "ready"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "manage.py runserver",
|
"label": "Start: Frontend Angular",
|
||||||
|
"description": "Start the Frontend Angular Dev Server",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "npm start",
|
||||||
|
"isBackground": true,
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/src-ui"
|
||||||
|
},
|
||||||
|
"problemMatcher": [
|
||||||
|
{
|
||||||
|
"owner": "custom",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": ".",
|
||||||
|
"file": 1,
|
||||||
|
"location": 2,
|
||||||
|
"message": 3
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": ".*",
|
||||||
|
"endsPattern": "Compiled successfully"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Start: Consumer Service (manage.py document_consumer)",
|
||||||
|
"description": "Start the Consumer Service which processes files from a directory",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pipenv run python manage.py document_consumer",
|
||||||
|
"group": "build",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/src"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Start: Backend Server (manage.py runserver)",
|
||||||
|
"description": "Start the Backend Server which serves the Django API and the compiled Angular frontend",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "pipenv run python manage.py runserver",
|
"command": "pipenv run python manage.py runserver",
|
||||||
"group": "build",
|
"group": "build",
|
||||||
@ -37,100 +94,130 @@
|
|||||||
"options": {
|
"options": {
|
||||||
"cwd": "${workspaceFolder}/src"
|
"cwd": "${workspaceFolder}/src"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Maintenance: manage.py migrate",
|
||||||
|
"description": "Apply database migrations",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pipenv run python manage.py migrate",
|
||||||
|
"group": "none",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
},
|
},
|
||||||
{
|
"options": {
|
||||||
"label": "Maintenance: manage.py migrate",
|
"cwd": "${workspaceFolder}/src"
|
||||||
"type": "shell",
|
}
|
||||||
"command": "pipenv run python manage.py migrate",
|
|
||||||
"group": "none",
|
|
||||||
"presentation": {
|
|
||||||
"echo": true,
|
|
||||||
"reveal": "always",
|
|
||||||
"focus": true,
|
|
||||||
"panel": "shared",
|
|
||||||
"showReuseMessage": false,
|
|
||||||
"clear": true,
|
|
||||||
"revealProblems": "onProblem"
|
|
||||||
},
|
},
|
||||||
"options": {
|
{
|
||||||
"cwd": "${workspaceFolder}/src"
|
"label": "Maintenance: Build Documentation",
|
||||||
|
"description": "Build the documentation with MkDocs",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pipenv run mkdocs build --config-file mkdocs.yml && pipenv run mkdocs serve",
|
||||||
|
"group": "none",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Maintenance: manage.py createsuperuser",
|
||||||
|
"description": "Create a superuser",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pipenv run python manage.py createsuperuser",
|
||||||
|
"group": "none",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/src"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Maintenance: recreate .venv",
|
||||||
|
"description": "Recreate the python virtual environment and install python dependencies",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "rm -R -v .venv/* || pipenv install --dev",
|
||||||
|
"group": "none",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Maintenance: Install Frontend Dependencies",
|
||||||
|
"description": "Install frontend (npm) dependencies",
|
||||||
|
"type": "npm",
|
||||||
|
"script": "install",
|
||||||
|
"path": "src-ui",
|
||||||
|
"group": "clean",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"detail": "install dependencies from package"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"description": "Clean install frontend dependencies and build the frontend for production",
|
||||||
|
"label": "Maintenance: Compile frontend for production",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "npm ci && ./node_modules/.bin/ng build --configuration production",
|
||||||
|
"group": "none",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true,
|
||||||
|
"revealProblems": "onProblem"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"cwd": "${workspaceFolder}/src-ui"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Project Setup: Run all Init Tasks",
|
||||||
|
"description": "Runs all init tasks to setup the project including migrate the database, create a superuser and compile the frontend for production",
|
||||||
|
"dependsOrder": "sequence",
|
||||||
|
"dependsOn": [
|
||||||
|
"Maintenance: manage.py migrate",
|
||||||
|
"Maintenance: manage.py createsuperuser",
|
||||||
|
"Maintenance: Compile frontend for production"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Project Start: Run all Services",
|
||||||
|
"description": "Runs all services required to start the project including the Celery Worker, the Consumer Service and the Backend Server",
|
||||||
|
"dependsOn": [
|
||||||
|
"Start: Celery Worker",
|
||||||
|
"Start: Consumer Service (manage.py document_consumer)",
|
||||||
|
"Start: Backend Server (manage.py runserver)"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "Maintenance: manage.py createsuperuser",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "pipenv run python manage.py createsuperuser",
|
|
||||||
"group": "none",
|
|
||||||
"presentation": {
|
|
||||||
"echo": true,
|
|
||||||
"reveal": "always",
|
|
||||||
"focus": true,
|
|
||||||
"panel": "shared",
|
|
||||||
"showReuseMessage": false,
|
|
||||||
"clear": true,
|
|
||||||
"revealProblems": "onProblem"
|
|
||||||
},
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/src"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "compile frontend",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "npm ci && ./node_modules/.bin/ng build --configuration production",
|
|
||||||
"group": "none",
|
|
||||||
"presentation": {
|
|
||||||
"echo": true,
|
|
||||||
"reveal": "always",
|
|
||||||
"focus": true,
|
|
||||||
"panel": "shared",
|
|
||||||
"showReuseMessage": false,
|
|
||||||
"clear": true,
|
|
||||||
"revealProblems": "onProblem"
|
|
||||||
},
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/src-ui"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "Maintenance: recreate .venv",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "rm -R -v .venv/* || pipenv install --dev",
|
|
||||||
"group": "none",
|
|
||||||
"presentation": {
|
|
||||||
"echo": true,
|
|
||||||
"reveal": "always",
|
|
||||||
"focus": true,
|
|
||||||
"panel": "shared",
|
|
||||||
"showReuseMessage": false,
|
|
||||||
"clear": true,
|
|
||||||
"revealProblems": "onProblem"
|
|
||||||
},
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "Celery Worker",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "pipenv run celery --app paperless worker -l DEBUG",
|
|
||||||
"group": {
|
|
||||||
"kind": "build",
|
|
||||||
"isDefault": true
|
|
||||||
},
|
|
||||||
"presentation": {
|
|
||||||
"echo": true,
|
|
||||||
"reveal": "always",
|
|
||||||
"focus": true,
|
|
||||||
"panel": "shared",
|
|
||||||
"showReuseMessage": false,
|
|
||||||
"clear": true,
|
|
||||||
"revealProblems": "onProblem"
|
|
||||||
},
|
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder}/src"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
10
.github/workflows/ci.yml
vendored
10
.github/workflows/ci.yml
vendored
@ -16,7 +16,7 @@ on:
|
|||||||
env:
|
env:
|
||||||
# This is the version of pipenv all the steps will use
|
# This is the version of pipenv all the steps will use
|
||||||
# If changing this, change Dockerfile
|
# If changing this, change Dockerfile
|
||||||
DEFAULT_PIP_ENV_VERSION: "2024.0.3"
|
DEFAULT_PIP_ENV_VERSION: "2024.4.0"
|
||||||
# This is the default version of Python to use in most steps which aren't specific
|
# This is the default version of Python to use in most steps which aren't specific
|
||||||
DEFAULT_PYTHON_VERSION: "3.11"
|
DEFAULT_PYTHON_VERSION: "3.11"
|
||||||
|
|
||||||
@ -283,7 +283,7 @@ jobs:
|
|||||||
merge-multiple: true
|
merge-multiple: true
|
||||||
-
|
-
|
||||||
name: Upload frontend coverage to Codecov
|
name: Upload frontend coverage to Codecov
|
||||||
uses: codecov/codecov-action@v4
|
uses: codecov/codecov-action@v5
|
||||||
with:
|
with:
|
||||||
# not required for public repos, but intermittently fails otherwise
|
# not required for public repos, but intermittently fails otherwise
|
||||||
token: ${{ secrets.CODECOV_TOKEN }}
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
@ -299,7 +299,7 @@ jobs:
|
|||||||
path: src/
|
path: src/
|
||||||
-
|
-
|
||||||
name: Upload coverage to Codecov
|
name: Upload coverage to Codecov
|
||||||
uses: codecov/codecov-action@v4
|
uses: codecov/codecov-action@v5
|
||||||
with:
|
with:
|
||||||
# not required for public repos, but intermittently fails otherwise
|
# not required for public repos, but intermittently fails otherwise
|
||||||
token: ${{ secrets.CODECOV_TOKEN }}
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
@ -406,7 +406,7 @@ jobs:
|
|||||||
-
|
-
|
||||||
name: Login to Docker Hub
|
name: Login to Docker Hub
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v3
|
||||||
# Don't attempt to login is not pushing to Docker Hub
|
# Don't attempt to login if not pushing to Docker Hub
|
||||||
if: steps.push-other-places.outputs.enable == 'true'
|
if: steps.push-other-places.outputs.enable == 'true'
|
||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
@ -414,7 +414,7 @@ jobs:
|
|||||||
-
|
-
|
||||||
name: Login to Quay.io
|
name: Login to Quay.io
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v3
|
||||||
# Don't attempt to login is not pushing to Quay.io
|
# Don't attempt to login if not pushing to Quay.io
|
||||||
if: steps.push-other-places.outputs.enable == 'true'
|
if: steps.push-other-places.outputs.enable == 'true'
|
||||||
with:
|
with:
|
||||||
registry: quay.io
|
registry: quay.io
|
||||||
|
6
.gitignore
vendored
6
.gitignore
vendored
@ -100,3 +100,9 @@ scripts/nuke
|
|||||||
|
|
||||||
# celery schedule file
|
# celery schedule file
|
||||||
celerybeat-schedule*
|
celerybeat-schedule*
|
||||||
|
|
||||||
|
# ignore .devcontainer sub folders
|
||||||
|
/.devcontainer/consume/
|
||||||
|
/.devcontainer/data/
|
||||||
|
/.devcontainer/media/
|
||||||
|
/.devcontainer/redisdata/
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
repos:
|
repos:
|
||||||
# General hooks
|
# General hooks
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v4.6.0
|
rev: v5.0.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: check-docstring-first
|
- id: check-docstring-first
|
||||||
- id: check-json
|
- id: check-json
|
||||||
@ -46,9 +46,12 @@ repos:
|
|||||||
- ts
|
- ts
|
||||||
- markdown
|
- markdown
|
||||||
exclude: "(^Pipfile\\.lock$)"
|
exclude: "(^Pipfile\\.lock$)"
|
||||||
|
additional_dependencies:
|
||||||
|
- prettier@3.3.3
|
||||||
|
- 'prettier-plugin-organize-imports@4.1.0'
|
||||||
# Python hooks
|
# Python hooks
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
rev: 'v0.6.8'
|
rev: v0.8.6
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff
|
- id: ruff
|
||||||
- id: ruff-format
|
- id: ruff-format
|
||||||
|
16
.prettierrc
16
.prettierrc
@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
# https://prettier.io/docs/en/options.html#semicolons
|
|
||||||
"semi": false,
|
|
||||||
# https://prettier.io/docs/en/options.html#quotes
|
|
||||||
"singleQuote": true,
|
|
||||||
# https://prettier.io/docs/en/options.html#trailing-commas
|
|
||||||
"trailingComma": "es5",
|
|
||||||
"overrides": [
|
|
||||||
{
|
|
||||||
"files": ["docs/*.md"],
|
|
||||||
"options": {
|
|
||||||
"tabWidth": 4,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
19
.prettierrc.js
Normal file
19
.prettierrc.js
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
const config = {
|
||||||
|
// https://prettier.io/docs/en/options.html#semicolons
|
||||||
|
semi: false,
|
||||||
|
// https://prettier.io/docs/en/options.html#quotes
|
||||||
|
singleQuote: true,
|
||||||
|
// https://prettier.io/docs/en/options.html#trailing-commas
|
||||||
|
trailingComma: 'es5',
|
||||||
|
overrides: [
|
||||||
|
{
|
||||||
|
files: ['docs/*.md'],
|
||||||
|
options: {
|
||||||
|
tabWidth: 4,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
plugins: [require('prettier-plugin-organize-imports')],
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = config
|
44
.ruff.toml
44
.ruff.toml
@ -31,17 +31,55 @@ extend-select = [
|
|||||||
"PLE", # https://docs.astral.sh/ruff/rules/#pylint-pl
|
"PLE", # https://docs.astral.sh/ruff/rules/#pylint-pl
|
||||||
"RUF", # https://docs.astral.sh/ruff/rules/#ruff-specific-rules-ruf
|
"RUF", # https://docs.astral.sh/ruff/rules/#ruff-specific-rules-ruf
|
||||||
"FLY", # https://docs.astral.sh/ruff/rules/#flynt-fly
|
"FLY", # https://docs.astral.sh/ruff/rules/#flynt-fly
|
||||||
|
"PTH", # https://docs.astral.sh/ruff/rules/#flake8-use-pathlib-pth
|
||||||
]
|
]
|
||||||
# TODO PTH https://docs.astral.sh/ruff/rules/#flake8-use-pathlib-pth
|
|
||||||
ignore = ["DJ001", "SIM105", "RUF012"]
|
ignore = ["DJ001", "SIM105", "RUF012"]
|
||||||
|
|
||||||
[lint.per-file-ignores]
|
[lint.per-file-ignores]
|
||||||
".github/scripts/*.py" = ["E501", "INP001", "SIM117"]
|
".github/scripts/*.py" = ["E501", "INP001", "SIM117"]
|
||||||
"docker/wait-for-redis.py" = ["INP001", "T201"]
|
"docker/wait-for-redis.py" = ["INP001", "T201"]
|
||||||
|
"src/documents/consumer.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/file_handling.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/management/commands/document_consumer.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/management/commands/document_exporter.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/migrations/0012_auto_20160305_0040.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/migrations/0014_document_checksum.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/migrations/1003_mime_types.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/migrations/1012_fix_archive_files.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/models.py" = ["SIM115", "PTH"] # TODO PTH Enable & remove
|
||||||
|
"src/documents/parsers.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/signals/handlers.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tasks.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_api_app_config.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_api_bulk_download.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_api_documents.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_classifier.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_consumer.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_file_handling.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_management.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_management_consumer.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_management_exporter.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_management_thumbnails.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_migration_archive_files.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_migration_document_pages_count.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_migration_mime_type.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_sanity_check.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_tasks.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/tests/test_views.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/documents/views.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless/checks.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless/settings.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless/tests/test_checks.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless/urls.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless/views.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless_mail/mail.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless_mail/preprocessor.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless_tesseract/parsers.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless_tesseract/tests/test_parser.py" = ["RUF001", "PTH"] # TODO PTH Enable & remove
|
||||||
|
"src/paperless_tika/tests/test_live_tika.py" = ["PTH"] # TODO Enable & remove
|
||||||
|
"src/paperless_tika/tests/test_tika_parser.py" = ["PTH"] # TODO Enable & remove
|
||||||
"*/tests/*.py" = ["E501", "SIM117"]
|
"*/tests/*.py" = ["E501", "SIM117"]
|
||||||
"*/migrations/*.py" = ["E501", "SIM", "T201"]
|
"*/migrations/*.py" = ["E501", "SIM", "T201"]
|
||||||
"src/paperless_tesseract/tests/test_parser.py" = ["RUF001"]
|
|
||||||
"src/documents/models.py" = ["SIM115"]
|
|
||||||
|
|
||||||
[lint.isort]
|
[lint.isort]
|
||||||
force-single-line = true
|
force-single-line = true
|
||||||
|
10
Dockerfile
10
Dockerfile
@ -39,7 +39,7 @@ COPY Pipfile* ./
|
|||||||
|
|
||||||
RUN set -eux \
|
RUN set -eux \
|
||||||
&& echo "Installing pipenv" \
|
&& echo "Installing pipenv" \
|
||||||
&& python3 -m pip install --no-cache-dir --upgrade pipenv==2024.0.3 \
|
&& python3 -m pip install --no-cache-dir --upgrade pipenv==2024.4.0 \
|
||||||
&& echo "Generating requirement.txt" \
|
&& echo "Generating requirement.txt" \
|
||||||
&& pipenv requirements > requirements.txt
|
&& pipenv requirements > requirements.txt
|
||||||
|
|
||||||
@ -233,11 +233,11 @@ RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \
|
|||||||
&& python3 -m pip install --no-cache-dir --upgrade wheel \
|
&& python3 -m pip install --no-cache-dir --upgrade wheel \
|
||||||
&& echo "Installing Python requirements" \
|
&& echo "Installing Python requirements" \
|
||||||
&& curl --fail --silent --show-error --location \
|
&& curl --fail --silent --show-error --location \
|
||||||
--output psycopg_c-3.2.2-cp312-cp312-linux_x86_64.whl \
|
--output psycopg_c-3.2.3-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 \
|
https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.3/psycopg_c-3.2.3-cp312-cp312-linux_x86_64.whl \
|
||||||
&& curl --fail --silent --show-error --location \
|
&& curl --fail --silent --show-error --location \
|
||||||
--output psycopg_c-3.2.2-cp312-cp312-linux_aarch64.whl \
|
--output psycopg_c-3.2.3-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 \
|
https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.3/psycopg_c-3.2.3-cp312-cp312-linux_aarch64.whl \
|
||||||
&& python3 -m pip install --default-timeout=1000 --find-links . --requirement requirements.txt \
|
&& python3 -m pip install --default-timeout=1000 --find-links . --requirement requirements.txt \
|
||||||
&& echo "Installing NLTK data" \
|
&& echo "Installing NLTK data" \
|
||||||
&& python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \
|
&& python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \
|
||||||
|
14
Pipfile
14
Pipfile
@ -7,8 +7,8 @@ name = "pypi"
|
|||||||
dateparser = "~=1.2"
|
dateparser = "~=1.2"
|
||||||
# WARNING: django does not use semver.
|
# WARNING: django does not use semver.
|
||||||
# Only patch versions are guaranteed to not introduce breaking changes.
|
# Only patch versions are guaranteed to not introduce breaking changes.
|
||||||
django = "~=5.1.1"
|
django = "~=5.1.4"
|
||||||
django-allauth = {extras = ["socialaccount"], version = "*"}
|
django-allauth = {extras = ["mfa", "socialaccount"], version = "*"}
|
||||||
django-auditlog = "*"
|
django-auditlog = "*"
|
||||||
django-celery-results = "*"
|
django-celery-results = "*"
|
||||||
django-compression-middleware = "*"
|
django-compression-middleware = "*"
|
||||||
@ -18,12 +18,12 @@ django-filter = "~=24.3"
|
|||||||
django-guardian = "*"
|
django-guardian = "*"
|
||||||
django-multiselectfield = "*"
|
django-multiselectfield = "*"
|
||||||
django-soft-delete = "*"
|
django-soft-delete = "*"
|
||||||
djangorestframework = "==3.15.2"
|
djangorestframework = "~=3.15.2"
|
||||||
djangorestframework-guardian = "*"
|
djangorestframework-guardian = "*"
|
||||||
drf-writable-nested = "*"
|
drf-writable-nested = "*"
|
||||||
bleach = "*"
|
bleach = "*"
|
||||||
celery = {extras = ["redis"], version = "*"}
|
celery = {extras = ["redis"], version = "*"}
|
||||||
channels = "~=4.1"
|
channels = "~=4.2"
|
||||||
channels-redis = "*"
|
channels-redis = "*"
|
||||||
concurrent-log-handler = "*"
|
concurrent-log-handler = "*"
|
||||||
filelock = "*"
|
filelock = "*"
|
||||||
@ -37,7 +37,7 @@ jinja2 = "~=3.1"
|
|||||||
langdetect = "*"
|
langdetect = "*"
|
||||||
mysqlclient = "*"
|
mysqlclient = "*"
|
||||||
nltk = "*"
|
nltk = "*"
|
||||||
ocrmypdf = "~=16.5"
|
ocrmypdf = "~=16.8"
|
||||||
pathvalidate = "*"
|
pathvalidate = "*"
|
||||||
pdf2image = "*"
|
pdf2image = "*"
|
||||||
psycopg = {version = "*", extras = ["c"]}
|
psycopg = {version = "*", extras = ["c"]}
|
||||||
@ -49,13 +49,13 @@ python-magic = "*"
|
|||||||
pyzbar = "*"
|
pyzbar = "*"
|
||||||
rapidfuzz = "*"
|
rapidfuzz = "*"
|
||||||
redis = {extras = ["hiredis"], version = "*"}
|
redis = {extras = ["hiredis"], version = "*"}
|
||||||
scikit-learn = "~=1.5"
|
scikit-learn = "~=1.6"
|
||||||
setproctitle = "*"
|
setproctitle = "*"
|
||||||
tika-client = "*"
|
tika-client = "*"
|
||||||
tqdm = "*"
|
tqdm = "*"
|
||||||
# See https://github.com/paperless-ngx/paperless-ngx/issues/5494
|
# See https://github.com/paperless-ngx/paperless-ngx/issues/5494
|
||||||
uvicorn = {extras = ["standard"], version = "==0.25.0"}
|
uvicorn = {extras = ["standard"], version = "==0.25.0"}
|
||||||
watchdog = "~=4.0"
|
watchdog = "~=6.0"
|
||||||
whitenoise = "~=6.8"
|
whitenoise = "~=6.8"
|
||||||
whoosh = "~=2.7"
|
whoosh = "~=2.7"
|
||||||
zxing-cpp = {version = "*", platform_machine = "== 'x86_64'"}
|
zxing-cpp = {version = "*", platform_machine = "== 'x86_64'"}
|
||||||
|
3926
Pipfile.lock
generated
3926
Pipfile.lock
generated
File diff suppressed because it is too large
Load Diff
@ -19,6 +19,8 @@
|
|||||||
#
|
#
|
||||||
# - Open portainer Stacks list and click 'Add stack'
|
# - Open portainer Stacks list and click 'Add stack'
|
||||||
# - Paste the contents of this file and assign a name, e.g. 'paperless'
|
# - Paste the contents of this file and assign a name, e.g. 'paperless'
|
||||||
|
# - Upload 'docker-compose.env' by clicking on 'Load variables from .env file'
|
||||||
|
# - Modify the environment variables as needed
|
||||||
# - Click 'Deploy the stack' and wait for it to be deployed
|
# - Click 'Deploy the stack' and wait for it to be deployed
|
||||||
# - Open the list of containers, select paperless_webserver_1
|
# - Open the list of containers, select paperless_webserver_1
|
||||||
# - Click 'Console' and then 'Connect' to open the command line inside the container
|
# - Click 'Console' and then 'Connect' to open the command line inside the container
|
||||||
@ -61,28 +63,8 @@ services:
|
|||||||
environment:
|
environment:
|
||||||
PAPERLESS_REDIS: redis://broker:6379
|
PAPERLESS_REDIS: redis://broker:6379
|
||||||
PAPERLESS_DBHOST: db
|
PAPERLESS_DBHOST: db
|
||||||
# The UID and GID of the user used to run paperless in the container. Set this
|
env_file:
|
||||||
# to your UID and GID on the host so that you have write access to the
|
- stack.env
|
||||||
# consumption directory.
|
|
||||||
USERMAP_UID: 1000
|
|
||||||
USERMAP_GID: 100
|
|
||||||
# Additional languages to install for text recognition, separated by a
|
|
||||||
# whitespace. Note that this is
|
|
||||||
# different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines the
|
|
||||||
# language used for OCR.
|
|
||||||
# The container installs English, German, Italian, Spanish and French by
|
|
||||||
# default.
|
|
||||||
# See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster
|
|
||||||
# for available languages.
|
|
||||||
#PAPERLESS_OCR_LANGUAGES: tur ces
|
|
||||||
# Adjust this key if you plan to make paperless available publicly. It should
|
|
||||||
# be a very long sequence of random characters. You don't need to remember it.
|
|
||||||
#PAPERLESS_SECRET_KEY: change-me
|
|
||||||
# Use this variable to set a timezone for the Paperless Docker containers. If not specified, defaults to UTC.
|
|
||||||
#PAPERLESS_TIME_ZONE: America/Los_Angeles
|
|
||||||
# The default language to use for OCR. Set this to the language most of your
|
|
||||||
# documents are written in.
|
|
||||||
#PAPERLESS_OCR_LANGUAGE: eng
|
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
data:
|
data:
|
||||||
|
@ -15,7 +15,8 @@ for command in decrypt_documents \
|
|||||||
document_sanity_checker \
|
document_sanity_checker \
|
||||||
document_fuzzy_match \
|
document_fuzzy_match \
|
||||||
manage_superuser \
|
manage_superuser \
|
||||||
convert_mariadb_uuid;
|
convert_mariadb_uuid \
|
||||||
|
prune_audit_logs;
|
||||||
do
|
do
|
||||||
echo "installing $command..."
|
echo "installing $command..."
|
||||||
sed "s/management_command/$command/g" management_script.sh > /usr/local/bin/$command
|
sed "s/management_command/$command/g" management_script.sh > /usr/local/bin/$command
|
||||||
|
@ -81,8 +81,8 @@ $ docker compose down
|
|||||||
1. If you pull the image from the docker hub, all you need to do is:
|
1. If you pull the image from the docker hub, all you need to do is:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ docker compose pull
|
docker compose pull
|
||||||
$ docker compose up
|
docker compose up
|
||||||
```
|
```
|
||||||
|
|
||||||
The Docker Compose files refer to the `latest` version, which is
|
The Docker Compose files refer to the `latest` version, which is
|
||||||
@ -91,9 +91,9 @@ $ docker compose down
|
|||||||
1. If you built the image yourself, do the following:
|
1. If you built the image yourself, do the following:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ git pull
|
git pull
|
||||||
$ docker compose build
|
docker compose build
|
||||||
$ docker compose up
|
docker compose up
|
||||||
```
|
```
|
||||||
|
|
||||||
Running `docker compose up` will also apply any new database migrations.
|
Running `docker compose up` will also apply any new database migrations.
|
||||||
@ -155,7 +155,7 @@ following:
|
|||||||
environment before that, if you use one.
|
environment before that, if you use one.
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ pip install -r requirements.txt
|
pip install -r requirements.txt
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! note
|
!!! note
|
||||||
@ -168,8 +168,8 @@ following:
|
|||||||
3. Migrate the database.
|
3. Migrate the database.
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ cd src
|
cd src
|
||||||
$ python3 manage.py migrate # (1)
|
python3 manage.py migrate # (1)
|
||||||
```
|
```
|
||||||
|
|
||||||
1. Including `sudo -Hu <paperless_user>` may be required
|
1. Including `sudo -Hu <paperless_user>` may be required
|
||||||
@ -241,6 +241,7 @@ document_exporter target [-c] [-d] [-f] [-na] [-nt] [-p] [-sm] [-z]
|
|||||||
|
|
||||||
optional arguments:
|
optional arguments:
|
||||||
-c, --compare-checksums
|
-c, --compare-checksums
|
||||||
|
-cj, --compare-json
|
||||||
-d, --delete
|
-d, --delete
|
||||||
-f, --use-filename-format
|
-f, --use-filename-format
|
||||||
-na, --no-archive
|
-na, --no-archive
|
||||||
@ -269,7 +270,8 @@ only export changed and added files. Paperless determines whether a file
|
|||||||
has changed by inspecting the file attributes "date/time modified" and
|
has changed by inspecting the file attributes "date/time modified" and
|
||||||
"size". If that does not work out for you, specify `-c` or
|
"size". If that does not work out for you, specify `-c` or
|
||||||
`--compare-checksums` and paperless will attempt to compare file
|
`--compare-checksums` and paperless will attempt to compare file
|
||||||
checksums instead. This is slower.
|
checksums instead. This is slower. The manifest and metadata json files
|
||||||
|
are always updated, unless `cj` or `--compare-json` is specified.
|
||||||
|
|
||||||
Paperless will not remove any existing files in the export directory. If
|
Paperless will not remove any existing files in the export directory. If
|
||||||
you want paperless to also remove files that do not belong to the
|
you want paperless to also remove files that do not belong to the
|
||||||
@ -622,3 +624,12 @@ document_fuzzy_match [--ratio] [--processes N]
|
|||||||
If providing the `--delete` option, it is highly recommended to have a backup.
|
If providing the `--delete` option, it is highly recommended to have a backup.
|
||||||
While every effort has been taken to ensure proper operation, there is always the
|
While every effort has been taken to ensure proper operation, there is always the
|
||||||
chance of deletion of a file you want to keep.
|
chance of deletion of a file you want to keep.
|
||||||
|
|
||||||
|
### Prune history (audit log) entries {#prune-history}
|
||||||
|
|
||||||
|
If the audit log is enabled Paperless-ngx keeps an audit log of all changes made to documents. Functionality to automatically remove entries for deleted documents was added but
|
||||||
|
entries created prior to this are not removed. This command allows you to prune the audit log of entries that are no longer needed.
|
||||||
|
|
||||||
|
```shell
|
||||||
|
prune_audit_logs
|
||||||
|
```
|
||||||
|
17
docs/api.md
17
docs/api.md
@ -365,6 +365,10 @@ The endpoint supports the following optional form fields:
|
|||||||
- `custom_fields`: An array of custom field ids to assign (with an empty
|
- `custom_fields`: An array of custom field ids to assign (with an empty
|
||||||
value) to the document.
|
value) to the document.
|
||||||
|
|
||||||
|
!!! note
|
||||||
|
|
||||||
|
Sending a `Content-Length` header with correct size is mandatory.
|
||||||
|
|
||||||
The endpoint will immediately return HTTP 200 if the document consumption
|
The endpoint will immediately return HTTP 200 if the document consumption
|
||||||
process was started successfully, with the UUID of the consumption task
|
process was started successfully, with the UUID of the consumption task
|
||||||
as the data. No additional status information about the consumption process
|
as the data. No additional status information about the consumption process
|
||||||
@ -473,6 +477,11 @@ The following methods are supported:
|
|||||||
- Requires `parameters`:
|
- Requires `parameters`:
|
||||||
- `"pages": [..]` The list should be a list of integers e.g. `"[2,3,4]"`
|
- `"pages": [..]` The list should be a list of integers e.g. `"[2,3,4]"`
|
||||||
- The delete_pages operation only accepts a single document.
|
- The delete_pages operation only accepts a single document.
|
||||||
|
- `modify_custom_fields`
|
||||||
|
- Requires `parameters`:
|
||||||
|
- `"add_custom_fields": { CUSTOM_FIELD_ID: VALUE }`: JSON object consisting of custom field id:value pairs to add to the document, can also be a list of custom field IDs
|
||||||
|
to add with empty values.
|
||||||
|
- `"remove_custom_fields": [CUSTOM_FIELD_ID]`: custom field ids to remove from the document.
|
||||||
|
|
||||||
### Objects
|
### Objects
|
||||||
|
|
||||||
@ -556,3 +565,11 @@ Initial API version.
|
|||||||
|
|
||||||
- Consumption templates were refactored to workflows and API endpoints
|
- Consumption templates were refactored to workflows and API endpoints
|
||||||
changed as such.
|
changed as such.
|
||||||
|
|
||||||
|
#### Version 5
|
||||||
|
|
||||||
|
- Added bulk deletion methods for documents and objects.
|
||||||
|
|
||||||
|
#### Version 6
|
||||||
|
|
||||||
|
- Moved acknowledge tasks endpoint to be under `/api/tasks/acknowledge/`.
|
||||||
|
@ -596,7 +596,7 @@ system. See the corresponding
|
|||||||
|
|
||||||
: Disables the regular frontend username / password login, i.e. once you have setup SSO. Note that this setting does not disable the Django admin login nor logging in with local credentials via the API. To prevent access to the Django admin, consider blocking `/admin/` in your [web server or reverse proxy configuration](https://github.com/paperless-ngx/paperless-ngx/wiki/Using-a-Reverse-Proxy-with-Paperless-ngx).
|
: Disables the regular frontend username / password login, i.e. once you have setup SSO. Note that this setting does not disable the Django admin login nor logging in with local credentials via the API. To prevent access to the Django admin, consider blocking `/admin/` in your [web server or reverse proxy configuration](https://github.com/paperless-ngx/paperless-ngx/wiki/Using-a-Reverse-Proxy-with-Paperless-ngx).
|
||||||
|
|
||||||
You can optionally also automatically redirect users to the SSO login with [PAPERLESS_REDIRECT_LOGIN_TO_SSO](#PAPERLESS_REDIRECT_LOGIN_TO_SSO)
|
You can optionally also automatically redirect users to the SSO login with [PAPERLESS_REDIRECT_LOGIN_TO_SSO](#PAPERLESS_REDIRECT_LOGIN_TO_SSO)
|
||||||
|
|
||||||
Defaults to False
|
Defaults to False
|
||||||
|
|
||||||
@ -1523,7 +1523,7 @@ one pod).
|
|||||||
actual user ID on the host system, which you can get by executing
|
actual user ID on the host system, which you can get by executing
|
||||||
|
|
||||||
``` shell-session
|
``` shell-session
|
||||||
$ id -u
|
id -u
|
||||||
```
|
```
|
||||||
|
|
||||||
Paperless will change ownership on its folders to this user, so you
|
Paperless will change ownership on its folders to this user, so you
|
||||||
@ -1538,7 +1538,7 @@ actual user ID on the host system, which you can get by executing
|
|||||||
actual group ID on the host system, which you can get by executing
|
actual group ID on the host system, which you can get by executing
|
||||||
|
|
||||||
``` shell-session
|
``` shell-session
|
||||||
$ id -g
|
id -g
|
||||||
```
|
```
|
||||||
|
|
||||||
Paperless will change ownership on its folders to this group, so you
|
Paperless will change ownership on its folders to this group, so you
|
||||||
|
@ -69,13 +69,13 @@ first-time setup.
|
|||||||
3. Create `consume` and `media` directories:
|
3. Create `consume` and `media` directories:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ mkdir -p consume media
|
mkdir -p consume media
|
||||||
```
|
```
|
||||||
|
|
||||||
4. Install the Python dependencies:
|
4. Install the Python dependencies:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ pipenv install --dev
|
pipenv install --dev
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! note
|
!!! note
|
||||||
@ -85,7 +85,7 @@ first-time setup.
|
|||||||
5. Install pre-commit hooks:
|
5. Install pre-commit hooks:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ pre-commit install
|
pre-commit install
|
||||||
```
|
```
|
||||||
|
|
||||||
6. Apply migrations and create a superuser for your development instance:
|
6. Apply migrations and create a superuser for your development instance:
|
||||||
@ -93,8 +93,8 @@ first-time setup.
|
|||||||
```bash
|
```bash
|
||||||
# src/
|
# src/
|
||||||
|
|
||||||
$ python3 manage.py migrate
|
python3 manage.py migrate
|
||||||
$ python3 manage.py createsuperuser
|
python3 manage.py createsuperuser
|
||||||
```
|
```
|
||||||
|
|
||||||
7. You can now either ...
|
7. You can now either ...
|
||||||
@ -108,7 +108,7 @@ first-time setup.
|
|||||||
- spin up a bare redis container
|
- spin up a bare redis container
|
||||||
|
|
||||||
```
|
```
|
||||||
$ docker run -d -p 6379:6379 --restart unless-stopped redis:latest
|
docker run -d -p 6379:6379 --restart unless-stopped redis:latest
|
||||||
```
|
```
|
||||||
|
|
||||||
8. Continue with either back-end or front-end development – or both :-).
|
8. Continue with either back-end or front-end development – or both :-).
|
||||||
@ -176,7 +176,7 @@ The front end is built using AngularJS. In order to get started, you need Node.j
|
|||||||
1. Install the Angular CLI. You might need sudo privileges to perform this command:
|
1. Install the Angular CLI. You might need sudo privileges to perform this command:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ npm install -g @angular/cli
|
npm install -g @angular/cli
|
||||||
```
|
```
|
||||||
|
|
||||||
2. Make sure that it's on your path.
|
2. Make sure that it's on your path.
|
||||||
@ -184,13 +184,13 @@ The front end is built using AngularJS. In order to get started, you need Node.j
|
|||||||
3. Install all necessary modules:
|
3. Install all necessary modules:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ npm install
|
npm install
|
||||||
```
|
```
|
||||||
|
|
||||||
4. You can launch a development server by running:
|
4. You can launch a development server by running:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ ng serve
|
ng serve
|
||||||
```
|
```
|
||||||
|
|
||||||
This will automatically update whenever you save. However, in-place
|
This will automatically update whenever you save. However, in-place
|
||||||
@ -335,13 +335,13 @@ If you want to build the documentation locally, this is how you do it:
|
|||||||
1. Have an active pipenv shell (`pipenv shell`) and install Python dependencies:
|
1. Have an active pipenv shell (`pipenv shell`) and install Python dependencies:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ pipenv install --dev
|
pipenv install --dev
|
||||||
```
|
```
|
||||||
|
|
||||||
2. Build the documentation
|
2. Build the documentation
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ mkdocs build --config-file mkdocs.yml
|
mkdocs build --config-file mkdocs.yml
|
||||||
```
|
```
|
||||||
|
|
||||||
_alternatively..._
|
_alternatively..._
|
||||||
@ -352,7 +352,7 @@ If you want to build the documentation locally, this is how you do it:
|
|||||||
something.
|
something.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ mkdocs serve
|
mkdocs serve
|
||||||
```
|
```
|
||||||
|
|
||||||
## Building the Docker image
|
## Building the Docker image
|
||||||
@ -450,3 +450,26 @@ def myparser_consumer_declaration(sender, **kwargs):
|
|||||||
mime types have many extensions associated with them and the Python
|
mime types have many extensions associated with them and the Python
|
||||||
methods responsible for guessing the extension do not always return
|
methods responsible for guessing the extension do not always return
|
||||||
the same value.
|
the same value.
|
||||||
|
|
||||||
|
## Using Visual Studio Code devcontainer
|
||||||
|
|
||||||
|
Another easy way to get started with development is to use Visual Studio
|
||||||
|
Code devcontainers. This approach will create a preconfigured development
|
||||||
|
environment with all of the required tools and dependencies.
|
||||||
|
[Learn more about devcontainers](https://code.visualstudio.com/docs/devcontainers/containers).
|
||||||
|
The .devcontainer/vscode/tasks.json and .devcontainer/vscode/launch.json files
|
||||||
|
contain more information about the specific tasks and launch configurations (see the
|
||||||
|
non-standard "description" field).
|
||||||
|
|
||||||
|
To get started:
|
||||||
|
|
||||||
|
1. Clone the repository on your machine and open the Paperless-ngx folder in VS Code.
|
||||||
|
|
||||||
|
2. VS Code will prompt you with "Reopen in container". Do so and wait for the environment to start.
|
||||||
|
|
||||||
|
3. Initialize the project by running the task **Project Setup: Run all Init Tasks**. This
|
||||||
|
will initialize the database tables and create a superuser. Then you can compile the front end
|
||||||
|
for production or run the frontend in debug mode.
|
||||||
|
|
||||||
|
4. The project is ready for debugging, start either run the fullstack debug or individual debug
|
||||||
|
processes. Yo spin up the project without debugging run the task **Project Start: Run all Services**
|
||||||
|
@ -30,7 +30,7 @@ account. The script essentially automatically performs the steps described in [D
|
|||||||
2. Download and run the installation script:
|
2. Download and run the installation script:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ bash -c "$(curl --location --silent --show-error https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/install-paperless-ngx.sh)"
|
bash -c "$(curl --location --silent --show-error https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/install-paperless-ngx.sh)"
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! note
|
!!! note
|
||||||
@ -135,13 +135,13 @@ account. The script essentially automatically performs the steps described in [D
|
|||||||
execute the following command:
|
execute the following command:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ docker compose run --rm webserver createsuperuser
|
docker compose run --rm webserver createsuperuser
|
||||||
```
|
```
|
||||||
|
|
||||||
or using docker exec from within the container:
|
or using docker exec from within the container:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ python3 manage.py createsuperuser
|
python3 manage.py createsuperuser
|
||||||
```
|
```
|
||||||
|
|
||||||
This will guide you through the superuser setup.
|
This will guide you through the superuser setup.
|
||||||
@ -188,7 +188,7 @@ account. The script essentially automatically performs the steps described in [D
|
|||||||
`docker compose pull` to pull the image, run
|
`docker compose pull` to pull the image, run
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ docker compose build
|
docker compose build
|
||||||
```
|
```
|
||||||
|
|
||||||
instead to build the image.
|
instead to build the image.
|
||||||
@ -557,8 +557,8 @@ Migration to paperless-ngx is then performed in a few simple steps:
|
|||||||
1. Stop paperless.
|
1. Stop paperless.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ cd /path/to/current/paperless
|
cd /path/to/current/paperless
|
||||||
$ docker compose down
|
docker compose down
|
||||||
```
|
```
|
||||||
|
|
||||||
2. Do a backup for two purposes: If something goes wrong, you still
|
2. Do a backup for two purposes: If something goes wrong, you still
|
||||||
@ -582,7 +582,7 @@ Migration to paperless-ngx is then performed in a few simple steps:
|
|||||||
names of your volumes with
|
names of your volumes with
|
||||||
|
|
||||||
``` shell-session
|
``` shell-session
|
||||||
$ docker volume ls | grep _data
|
docker volume ls | grep _data
|
||||||
```
|
```
|
||||||
|
|
||||||
and adjust the project name in the `.env` file so that it matches
|
and adjust the project name in the `.env` file so that it matches
|
||||||
@ -603,7 +603,7 @@ Migration to paperless-ngx is then performed in a few simple steps:
|
|||||||
the search index:
|
the search index:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ docker compose run --rm webserver document_index reindex
|
docker compose run --rm webserver document_index reindex
|
||||||
```
|
```
|
||||||
|
|
||||||
This will migrate your database and create the search index. After
|
This will migrate your database and create the search index. After
|
||||||
@ -612,7 +612,7 @@ Migration to paperless-ngx is then performed in a few simple steps:
|
|||||||
8. Start paperless-ngx.
|
8. Start paperless-ngx.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ docker compose up -d
|
docker compose up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
This will run paperless in the background and automatically start it
|
This will run paperless in the background and automatically start it
|
||||||
|
@ -18,7 +18,7 @@ Check for the following issues:
|
|||||||
automatically. Manually invoke the task processor by executing
|
automatically. Manually invoke the task processor by executing
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ celery --app paperless worker
|
celery --app paperless worker
|
||||||
```
|
```
|
||||||
|
|
||||||
- Look at the output of paperless and inspect it for any errors.
|
- Look at the output of paperless and inspect it for any errors.
|
||||||
|
@ -299,6 +299,12 @@ In order to enable the password reset feature you will need to setup an SMTP bac
|
|||||||
[`PAPERLESS_EMAIL_HOST`](configuration.md#PAPERLESS_EMAIL_HOST). If your installation does not have
|
[`PAPERLESS_EMAIL_HOST`](configuration.md#PAPERLESS_EMAIL_HOST). If your installation does not have
|
||||||
[`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) set, the reset link included in emails will use the server host.
|
[`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) set, the reset link included in emails will use the server host.
|
||||||
|
|
||||||
|
### Two-factor authentication
|
||||||
|
|
||||||
|
Users can enable two-factor authentication (2FA) for their accounts from the 'My Profile' dialog. Opening the dropdown reveals a QR code that can be scanned by a 2FA app (e.g. Google Authenticator) to generate a code. The code must then be entered in the dialog to enable 2FA. If the code is accepted and 2FA is enabled, the user will be shown a set of 10 recovery codes that can be used to login in the event that the 2FA device is lost or unavailable. These codes should be stored securely and cannot be retrieved again. Once enabled, users will be required to enter a code from their 2FA app when logging in.
|
||||||
|
|
||||||
|
Should a user lose access to their 2FA device and all recovery codes, a superuser can disable 2FA for the user from the 'Users & Groups' management screen.
|
||||||
|
|
||||||
## Workflows
|
## Workflows
|
||||||
|
|
||||||
!!! note
|
!!! note
|
||||||
@ -316,6 +322,8 @@ fields and permissions, which will be merged.
|
|||||||
|
|
||||||
### Workflow Triggers
|
### Workflow Triggers
|
||||||
|
|
||||||
|
#### Types
|
||||||
|
|
||||||
Currently, there are three events that correspond to workflow trigger 'types':
|
Currently, there are three events that correspond to workflow trigger 'types':
|
||||||
|
|
||||||
1. **Consumption Started**: _before_ a document is consumed, so events can include filters by source (mail, consumption
|
1. **Consumption Started**: _before_ a document is consumed, so events can include filters by source (mail, consumption
|
||||||
@ -325,8 +333,10 @@ Currently, there are three events that correspond to workflow trigger 'types':
|
|||||||
be used for filtering.
|
be used for filtering.
|
||||||
3. **Document Updated**: when a document is updated. Similar to 'added' events, triggers can include filtering by content matching,
|
3. **Document Updated**: when a document is updated. Similar to 'added' events, triggers can include filtering by content matching,
|
||||||
tags, doc type, or correspondent.
|
tags, doc type, or correspondent.
|
||||||
|
4. **Scheduled**: a scheduled trigger that can be used to run workflows at a specific time. The date used can be either the document
|
||||||
|
added, created, updated date or you can specify a (date) custom field. You can also specify a day offset from the date.
|
||||||
|
|
||||||
The following flow diagram illustrates the three trigger types:
|
The following flow diagram illustrates the three document trigger types:
|
||||||
|
|
||||||
```mermaid
|
```mermaid
|
||||||
flowchart TD
|
flowchart TD
|
||||||
@ -372,25 +382,49 @@ Workflows allow you to filter by:
|
|||||||
|
|
||||||
### Workflow Actions
|
### Workflow Actions
|
||||||
|
|
||||||
There are currently two types of workflow actions, "Assignment", which can assign:
|
#### Types
|
||||||
|
|
||||||
- Title, see [title placeholders](usage.md#title-placeholders) below
|
The following workflow action types are available:
|
||||||
|
|
||||||
|
##### Assignment
|
||||||
|
|
||||||
|
"Assignment" actions can assign:
|
||||||
|
|
||||||
|
- Title, see [workflow placeholders](usage.md#workflow-placeholders) below
|
||||||
- Tags, correspondent, document type and storage path
|
- Tags, correspondent, document type and storage path
|
||||||
- Document owner
|
- Document owner
|
||||||
- View and / or edit permissions to users or groups
|
- View and / or edit permissions to users or groups
|
||||||
- Custom fields. Note that no value for the field will be set
|
- Custom fields. Note that no value for the field will be set
|
||||||
|
|
||||||
and "Removal" actions, which can remove either all of or specific sets of the following:
|
##### Removal
|
||||||
|
|
||||||
|
"Removal" actions can remove either all of or specific sets of the following:
|
||||||
|
|
||||||
- Tags, correspondents, document types or storage paths
|
- Tags, correspondents, document types or storage paths
|
||||||
- Document owner
|
- Document owner
|
||||||
- View and / or edit permissions
|
- View and / or edit permissions
|
||||||
- Custom fields
|
- Custom fields
|
||||||
|
|
||||||
#### Title placeholders
|
##### Email
|
||||||
|
|
||||||
Workflow titles can include placeholders but the available options differ depending on the type of
|
"Email" actions can send documents via email. This action requires a mail server to be [configured](configuration.md#email-sending). You can specify:
|
||||||
workflow trigger. This is because at the time of consumption (when the title is to be set), no automatic tags etc. have been
|
|
||||||
|
- The recipient email address(es) separated by commas
|
||||||
|
- The subject and body of the email, which can include placeholders, see [placeholders](usage.md#workflow-placeholders) below
|
||||||
|
- Whether to include the document as an attachment
|
||||||
|
|
||||||
|
##### Webhook
|
||||||
|
|
||||||
|
"Webhook" actions send a POST request to a specified URL. You can specify:
|
||||||
|
|
||||||
|
- The URL to send the request to
|
||||||
|
- The request body as text or as key-value pairs, which can include placeholders, see [placeholders](usage.md#workflow-placeholders) below.
|
||||||
|
- The request headers as key-value pairs
|
||||||
|
|
||||||
|
#### Workflow placeholders
|
||||||
|
|
||||||
|
Some workflow text can include placeholders but the available options differ depending on the type of
|
||||||
|
workflow trigger. This is because at the time of consumption (when the text is to be set), no automatic tags etc. have been
|
||||||
applied. You can use the following placeholders with any trigger type:
|
applied. You can use the following placeholders with any trigger type:
|
||||||
|
|
||||||
- `{correspondent}`: assigned correspondent name
|
- `{correspondent}`: assigned correspondent name
|
||||||
@ -405,6 +439,7 @@ applied. You can use the following placeholders with any trigger type:
|
|||||||
- `{added_day}`: added day
|
- `{added_day}`: added day
|
||||||
- `{added_time}`: added time in HH:MM format
|
- `{added_time}`: added time in HH:MM format
|
||||||
- `{original_filename}`: original file name without extension
|
- `{original_filename}`: original file name without extension
|
||||||
|
- `{filename}`: current file name without extension
|
||||||
|
|
||||||
The following placeholders are only available for "added" or "updated" triggers
|
The following placeholders are only available for "added" or "updated" triggers
|
||||||
|
|
||||||
@ -416,6 +451,7 @@ The following placeholders are only available for "added" or "updated" triggers
|
|||||||
- `{created_month_name_short}`: created month short name
|
- `{created_month_name_short}`: created month short name
|
||||||
- `{created_day}`: created day
|
- `{created_day}`: created day
|
||||||
- `{created_time}`: created time in HH:MM format
|
- `{created_time}`: created time in HH:MM format
|
||||||
|
- `{doc_url}`: URL to the document in the web UI. Requires the `PAPERLESS_URL` setting to be set.
|
||||||
|
|
||||||
### Workflow permissions
|
### Workflow permissions
|
||||||
|
|
||||||
@ -752,8 +788,8 @@ Paperless-ngx consists of the following components:
|
|||||||
with paperless. You may start the webserver directly with
|
with paperless. You may start the webserver directly with
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ cd /path/to/paperless/src/
|
cd /path/to/paperless/src/
|
||||||
$ gunicorn -c ../gunicorn.conf.py paperless.wsgi
|
gunicorn -c ../gunicorn.conf.py paperless.wsgi
|
||||||
```
|
```
|
||||||
|
|
||||||
or by any other means such as Apache `mod_wsgi`.
|
or by any other means such as Apache `mod_wsgi`.
|
||||||
@ -768,8 +804,8 @@ Paperless-ngx consists of the following components:
|
|||||||
Start the consumer with the management command `document_consumer`:
|
Start the consumer with the management command `document_consumer`:
|
||||||
|
|
||||||
```shell-session
|
```shell-session
|
||||||
$ cd /path/to/paperless/src/
|
cd /path/to/paperless/src/
|
||||||
$ python3 manage.py document_consumer
|
python3 manage.py document_consumer
|
||||||
```
|
```
|
||||||
|
|
||||||
- **The task processor:** Paperless relies on [Celery - Distributed
|
- **The task processor:** Paperless relies on [Celery - Distributed
|
||||||
|
@ -330,8 +330,13 @@ SECRET_KEY=$(LC_ALL=C tr -dc 'a-zA-Z0-9!#$%&()*+,-./:;<=>?@[\]^_`{|}~' < /dev/ur
|
|||||||
|
|
||||||
DEFAULT_LANGUAGES=("deu eng fra ita spa")
|
DEFAULT_LANGUAGES=("deu eng fra ita spa")
|
||||||
|
|
||||||
_split_langs="${OCR_LANGUAGE//+/ }"
|
# OCR_LANG requires underscores, replace dashes if the user gave them with underscores
|
||||||
read -r -a OCR_LANGUAGES_ARRAY <<< "${_split_langs}"
|
readonly ocr_langs=${OCR_LANGUAGE//-/_}
|
||||||
|
# OCR_LANGS (the install version) uses dashes, not underscores, so convert underscore to dash and plus to space
|
||||||
|
install_langs=${OCR_LANGUAGE//_/-} # First convert any underscores to dashes
|
||||||
|
install_langs=${install_langs//+/ } # Then convert plus signs to spaces
|
||||||
|
|
||||||
|
read -r -a install_langs_array <<< "${install_langs}"
|
||||||
|
|
||||||
{
|
{
|
||||||
if [[ ! $URL == "" ]] ; then
|
if [[ ! $URL == "" ]] ; then
|
||||||
@ -344,10 +349,10 @@ read -r -a OCR_LANGUAGES_ARRAY <<< "${_split_langs}"
|
|||||||
echo "USERMAP_GID=$USERMAP_GID"
|
echo "USERMAP_GID=$USERMAP_GID"
|
||||||
fi
|
fi
|
||||||
echo "PAPERLESS_TIME_ZONE=$TIME_ZONE"
|
echo "PAPERLESS_TIME_ZONE=$TIME_ZONE"
|
||||||
echo "PAPERLESS_OCR_LANGUAGE=$OCR_LANGUAGE"
|
echo "PAPERLESS_OCR_LANGUAGE=$ocr_langs"
|
||||||
echo "PAPERLESS_SECRET_KEY='$SECRET_KEY'"
|
echo "PAPERLESS_SECRET_KEY='$SECRET_KEY'"
|
||||||
if [[ ! ${DEFAULT_LANGUAGES[*]} =~ ${OCR_LANGUAGES_ARRAY[*]} ]] ; then
|
if [[ ! ${DEFAULT_LANGUAGES[*]} =~ ${install_langs_array[*]} ]] ; then
|
||||||
echo "PAPERLESS_OCR_LANGUAGES=${OCR_LANGUAGES_ARRAY[*]}"
|
echo "PAPERLESS_OCR_LANGUAGES=${install_langs_array[*]}"
|
||||||
fi
|
fi
|
||||||
} > docker-compose.env
|
} > docker-compose.env
|
||||||
|
|
||||||
|
@ -11,8 +11,7 @@
|
|||||||
],
|
],
|
||||||
"parserOptions": {
|
"parserOptions": {
|
||||||
"project": [
|
"project": [
|
||||||
"tsconfig.json",
|
"tsconfig.json"
|
||||||
"e2e/tsconfig.json"
|
|
||||||
],
|
],
|
||||||
"createDefaultProgram": true
|
"createDefaultProgram": true
|
||||||
},
|
},
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { test, expect } from '@playwright/test'
|
import { expect, test } from '@playwright/test'
|
||||||
|
import path from 'node:path'
|
||||||
|
|
||||||
const REQUESTS_HAR = 'e2e/admin/requests/api-settings.har'
|
const REQUESTS_HAR = path.join(__dirname, 'requests/api-settings.har')
|
||||||
|
|
||||||
test('should activate / deactivate save button when settings change', async ({
|
test('should activate / deactivate save button when settings change', async ({
|
||||||
page,
|
page,
|
||||||
@ -33,24 +34,3 @@ test('should apply appearance changes when set', async ({ page }) => {
|
|||||||
await page.getByLabel('Enable dark mode').click()
|
await page.getByLabel('Enable dark mode').click()
|
||||||
await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /dark/)
|
await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /dark/)
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should toggle saved view options when set & saved', async ({ page }) => {
|
|
||||||
await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' })
|
|
||||||
await page.goto('/settings/savedviews')
|
|
||||||
await page.getByLabel('Show on dashboard').first().click()
|
|
||||||
await page.getByLabel('Show in sidebar').first().click()
|
|
||||||
const updatePromise = page.waitForRequest((request) => {
|
|
||||||
if (!request.url().includes('8')) return true // skip other saved views
|
|
||||||
const data = request.postDataJSON()
|
|
||||||
const isValid =
|
|
||||||
data['show_on_dashboard'] === true && data['show_in_sidebar'] === true
|
|
||||||
return (
|
|
||||||
isValid &&
|
|
||||||
request.method() === 'PATCH' &&
|
|
||||||
request.url().includes('/api/saved_views/')
|
|
||||||
)
|
|
||||||
})
|
|
||||||
await page.getByRole('button', { name: 'Save' }).scrollIntoViewIfNeeded()
|
|
||||||
await page.getByRole('button', { name: 'Save' }).click()
|
|
||||||
await updatePromise
|
|
||||||
})
|
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
import { test, expect } from '@playwright/test'
|
import { expect, test } from '@playwright/test'
|
||||||
|
import path from 'node:path'
|
||||||
|
|
||||||
const REQUESTS_HAR1 = 'e2e/dashboard/requests/api-dashboard1.har'
|
const REQUESTS_HAR1 = path.join(__dirname, 'requests/api-dashboard1.har')
|
||||||
const REQUESTS_HAR2 = 'e2e/dashboard/requests/api-dashboard2.har'
|
const REQUESTS_HAR2 = path.join(__dirname, 'requests/api-dashboard2.har')
|
||||||
const REQUESTS_HAR3 = 'e2e/dashboard/requests/api-dashboard3.har'
|
const REQUESTS_HAR3 = path.join(__dirname, 'requests/api-dashboard3.har')
|
||||||
const REQUESTS_HAR4 = 'e2e/dashboard/requests/api-dashboard4.har'
|
const REQUESTS_HAR4 = path.join(__dirname, 'requests/api-dashboard4.har')
|
||||||
|
|
||||||
test('dashboard inbox link', async ({ page }) => {
|
test('dashboard inbox link', async ({ page }) => {
|
||||||
await page.routeFromHAR(REQUESTS_HAR1, { notFound: 'fallback' })
|
await page.routeFromHAR(REQUESTS_HAR1, { notFound: 'fallback' })
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
import { test, expect } from '@playwright/test'
|
import { expect, test } from '@playwright/test'
|
||||||
|
import path from 'node:path'
|
||||||
|
|
||||||
const REQUESTS_HAR = 'e2e/document-detail/requests/api-document-detail.har'
|
const REQUESTS_HAR = path.join(__dirname, 'requests/api-document-detail.har')
|
||||||
const REQUESTS_HAR2 = 'e2e/document-detail/requests/api-document-detail2.har'
|
const REQUESTS_HAR2 = path.join(__dirname, 'requests/api-document-detail2.har')
|
||||||
|
|
||||||
test('should activate / deactivate save button when changes are saved', async ({
|
test('should activate / deactivate save button when changes are saved', async ({
|
||||||
page,
|
page,
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
import { test, expect } from '@playwright/test'
|
import { expect, test } from '@playwright/test'
|
||||||
|
import path from 'node:path'
|
||||||
|
|
||||||
const REQUESTS_HAR1 = 'e2e/document-list/requests/api-document-list1.har'
|
const REQUESTS_HAR1 = path.join(__dirname, 'requests/api-document-list1.har')
|
||||||
const REQUESTS_HAR2 = 'e2e/document-list/requests/api-document-list2.har'
|
const REQUESTS_HAR2 = path.join(__dirname, 'requests/api-document-list2.har')
|
||||||
const REQUESTS_HAR3 = 'e2e/document-list/requests/api-document-list3.har'
|
const REQUESTS_HAR3 = path.join(__dirname, 'requests/api-document-list3.har')
|
||||||
const REQUESTS_HAR4 = 'e2e/document-list/requests/api-document-list4.har'
|
const REQUESTS_HAR4 = path.join(__dirname, 'requests/api-document-list4.har')
|
||||||
const REQUESTS_HAR5 = 'e2e/document-list/requests/api-document-list5.har'
|
const REQUESTS_HAR5 = path.join(__dirname, 'requests/api-document-list5.har')
|
||||||
const REQUESTS_HAR6 = 'e2e/document-list/requests/api-document-list6.har'
|
const REQUESTS_HAR6 = path.join(__dirname, 'requests/api-document-list6.har')
|
||||||
|
|
||||||
test('basic filtering', async ({ page }) => {
|
test('basic filtering', async ({ page }) => {
|
||||||
await page.routeFromHAR(REQUESTS_HAR1, { notFound: 'fallback' })
|
await page.routeFromHAR(REQUESTS_HAR1, { notFound: 'fallback' })
|
||||||
@ -134,11 +135,11 @@ test('sorting', async ({ page }) => {
|
|||||||
test('change views', async ({ page }) => {
|
test('change views', async ({ page }) => {
|
||||||
await page.routeFromHAR(REQUESTS_HAR5, { notFound: 'fallback' })
|
await page.routeFromHAR(REQUESTS_HAR5, { notFound: 'fallback' })
|
||||||
await page.goto('/documents')
|
await page.goto('/documents')
|
||||||
await page.locator('.btn-group label').first().click()
|
await page.locator('.btn-group > label').first().click()
|
||||||
await expect(page.locator('pngx-document-list table')).toBeVisible()
|
await expect(page.locator('pngx-document-list table')).toBeVisible()
|
||||||
await page.locator('.btn-group label').nth(1).click()
|
await page.locator('label:nth-child(4)').first().click()
|
||||||
await expect(page.locator('pngx-document-card-small').first()).toBeAttached()
|
await expect(page.locator('pngx-document-card-small').first()).toBeAttached()
|
||||||
await page.locator('.btn-group label').nth(2).click()
|
await page.locator('label:nth-child(6)').click()
|
||||||
await expect(page.locator('pngx-document-card-large').first()).toBeAttached()
|
await expect(page.locator('pngx-document-card-large').first()).toBeAttached()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { test, expect } from '@playwright/test'
|
import { expect, test } from '@playwright/test'
|
||||||
|
import path from 'node:path'
|
||||||
|
|
||||||
const REQUESTS_HAR = 'e2e/permissions/requests/api-global-permissions.har'
|
const REQUESTS_HAR = path.join(__dirname, 'requests/api-global-permissions.har')
|
||||||
|
|
||||||
test('should not allow user to edit settings', async ({ page }) => {
|
test('should not allow user to edit settings', async ({ page }) => {
|
||||||
await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' })
|
await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' })
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import * as webpack from 'webpack'
|
|
||||||
import {
|
import {
|
||||||
CustomWebpackBrowserSchema,
|
CustomWebpackBrowserSchema,
|
||||||
TargetOptions,
|
TargetOptions,
|
||||||
} from '@angular-builders/custom-webpack'
|
} from '@angular-builders/custom-webpack'
|
||||||
|
import * as webpack from 'webpack'
|
||||||
const { codecovWebpackPlugin } = require('@codecov/webpack-plugin')
|
const { codecovWebpackPlugin } = require('@codecov/webpack-plugin')
|
||||||
|
|
||||||
export default (
|
export default (
|
||||||
|
3158
src-ui/messages.xlf
3158
src-ui/messages.xlf
File diff suppressed because it is too large
Load Diff
6085
src-ui/package-lock.json
generated
6085
src-ui/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -11,46 +11,47 @@
|
|||||||
},
|
},
|
||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular/cdk": "^18.2.11",
|
"@angular/cdk": "^19.0.2",
|
||||||
"@angular/common": "~18.2.10",
|
"@angular/common": "~19.0.3",
|
||||||
"@angular/compiler": "~18.2.10",
|
"@angular/compiler": "~19.0.3",
|
||||||
"@angular/core": "~18.2.10",
|
"@angular/core": "~19.0.3",
|
||||||
"@angular/forms": "~18.2.10",
|
"@angular/forms": "~19.0.3",
|
||||||
"@angular/localize": "~18.2.10",
|
"@angular/localize": "~19.0.3",
|
||||||
"@angular/platform-browser": "~18.2.10",
|
"@angular/platform-browser": "~19.0.3",
|
||||||
"@angular/platform-browser-dynamic": "~18.2.10",
|
"@angular/platform-browser-dynamic": "~19.0.3",
|
||||||
"@angular/router": "~18.2.10",
|
"@angular/router": "~19.0.3",
|
||||||
"@ng-bootstrap/ng-bootstrap": "^17.0.1",
|
"@ng-bootstrap/ng-bootstrap": "^18.0.0",
|
||||||
"@ng-select/ng-select": "^13.9.1",
|
"@ng-select/ng-select": "^14.1.0",
|
||||||
"@ngneat/dirty-check-forms": "^3.0.3",
|
"@ngneat/dirty-check-forms": "^3.0.3",
|
||||||
"@popperjs/core": "^2.11.8",
|
"@popperjs/core": "^2.11.8",
|
||||||
"bootstrap": "^5.3.3",
|
"bootstrap": "^5.3.3",
|
||||||
"file-saver": "^2.0.5",
|
"file-saver": "^2.0.5",
|
||||||
"mime-names": "^1.0.0",
|
"mime-names": "^1.0.0",
|
||||||
"ng2-pdf-viewer": "^10.3.4",
|
"ng2-pdf-viewer": "^10.4.0",
|
||||||
"ngx-bootstrap-icons": "^1.9.3",
|
"ngx-bootstrap-icons": "^1.9.3",
|
||||||
"ngx-color": "^9.0.0",
|
"ngx-color": "^9.0.0",
|
||||||
"ngx-cookie-service": "^18.0.0",
|
"ngx-cookie-service": "^19.0.0",
|
||||||
"ngx-file-drop": "^16.0.0",
|
"ngx-file-drop": "^16.0.0",
|
||||||
"ngx-ui-tour-ng-bootstrap": "^15.0.0",
|
"ngx-ui-tour-ng-bootstrap": "^16.0.0",
|
||||||
"rxjs": "^7.8.1",
|
"rxjs": "^7.8.1",
|
||||||
"tslib": "^2.8.1",
|
"tslib": "^2.8.1",
|
||||||
|
"utif": "^3.1.0",
|
||||||
"uuid": "^11.0.2",
|
"uuid": "^11.0.2",
|
||||||
"zone.js": "^0.14.8"
|
"zone.js": "^0.15.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@angular-builders/custom-webpack": "^18.0.0",
|
"@angular-builders/custom-webpack": "^19.0.0-beta.0",
|
||||||
"@angular-builders/jest": "^18.0.0",
|
"@angular-builders/jest": "^19.0.0-beta.1",
|
||||||
"@angular-devkit/build-angular": "^18.2.2",
|
"@angular-devkit/build-angular": "^19.0.4",
|
||||||
"@angular-devkit/core": "^18.2.11",
|
"@angular-devkit/core": "^19.0.4",
|
||||||
"@angular-devkit/schematics": "^18.2.11",
|
"@angular-devkit/schematics": "^19.0.4",
|
||||||
"@angular-eslint/builder": "18.4.0",
|
"@angular-eslint/builder": "19.0.0",
|
||||||
"@angular-eslint/eslint-plugin": "18.4.0",
|
"@angular-eslint/eslint-plugin": "19.0.0",
|
||||||
"@angular-eslint/eslint-plugin-template": "18.4.0",
|
"@angular-eslint/eslint-plugin-template": "19.0.0",
|
||||||
"@angular-eslint/schematics": "18.4.0",
|
"@angular-eslint/schematics": "19.0.0",
|
||||||
"@angular-eslint/template-parser": "18.4.0",
|
"@angular-eslint/template-parser": "19.0.0",
|
||||||
"@angular/cli": "~18.2.11",
|
"@angular/cli": "~19.0.4",
|
||||||
"@angular/compiler-cli": "~18.2.2",
|
"@angular/compiler-cli": "~19.0.3",
|
||||||
"@codecov/webpack-plugin": "^1.2.1",
|
"@codecov/webpack-plugin": "^1.2.1",
|
||||||
"@playwright/test": "^1.48.2",
|
"@playwright/test": "^1.48.2",
|
||||||
"@types/jest": "^29.5.14",
|
"@types/jest": "^29.5.14",
|
||||||
@ -61,10 +62,12 @@
|
|||||||
"eslint": "^9.14.0",
|
"eslint": "^9.14.0",
|
||||||
"jest": "29.7.0",
|
"jest": "29.7.0",
|
||||||
"jest-environment-jsdom": "^29.7.0",
|
"jest-environment-jsdom": "^29.7.0",
|
||||||
"jest-preset-angular": "^14.2.4",
|
"jest-preset-angular": "^14.4.2",
|
||||||
"jest-websocket-mock": "^2.5.0",
|
"jest-websocket-mock": "^2.5.0",
|
||||||
"patch-package": "^8.0.0",
|
"patch-package": "^8.0.0",
|
||||||
|
"prettier-plugin-organize-imports": "^4.1.0",
|
||||||
"ts-node": "~10.9.1",
|
"ts-node": "~10.9.1",
|
||||||
"typescript": "^5.5.4"
|
"typescript": "^5.5.4"
|
||||||
}
|
},
|
||||||
|
"typings": "./src/typings.d.ts"
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
import { jest } from '@jest/globals'
|
|
||||||
if (process.env.NODE_ENV === 'test') {
|
|
||||||
require('jest-preset-angular/setup-jest')
|
|
||||||
}
|
|
||||||
import '@angular/localize/init'
|
import '@angular/localize/init'
|
||||||
import { TextEncoder, TextDecoder } from 'util'
|
import { jest } from '@jest/globals'
|
||||||
|
import { setupZoneTestEnv } from 'jest-preset-angular/setup-env/zone'
|
||||||
|
import { TextDecoder, TextEncoder } from 'util'
|
||||||
|
if (process.env.NODE_ENV === 'test') {
|
||||||
|
setupZoneTestEnv()
|
||||||
|
}
|
||||||
global.TextEncoder = TextEncoder
|
global.TextEncoder = TextEncoder
|
||||||
global.TextDecoder = TextDecoder
|
global.TextDecoder = TextDecoder
|
||||||
|
|
||||||
|
@ -1,32 +1,33 @@
|
|||||||
import { NgModule } from '@angular/core'
|
import { NgModule } from '@angular/core'
|
||||||
import { Routes, RouterModule } from '@angular/router'
|
import { RouterModule, Routes } from '@angular/router'
|
||||||
|
import { ConfigComponent } from './components/admin/config/config.component'
|
||||||
|
import { LogsComponent } from './components/admin/logs/logs.component'
|
||||||
|
import { SettingsComponent } from './components/admin/settings/settings.component'
|
||||||
|
import { TasksComponent } from './components/admin/tasks/tasks.component'
|
||||||
|
import { TrashComponent } from './components/admin/trash/trash.component'
|
||||||
|
import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component'
|
||||||
import { AppFrameComponent } from './components/app-frame/app-frame.component'
|
import { AppFrameComponent } from './components/app-frame/app-frame.component'
|
||||||
import { DashboardComponent } from './components/dashboard/dashboard.component'
|
import { DashboardComponent } from './components/dashboard/dashboard.component'
|
||||||
|
import { DocumentAsnComponent } from './components/document-asn/document-asn.component'
|
||||||
import { DocumentDetailComponent } from './components/document-detail/document-detail.component'
|
import { DocumentDetailComponent } from './components/document-detail/document-detail.component'
|
||||||
import { DocumentListComponent } from './components/document-list/document-list.component'
|
import { DocumentListComponent } from './components/document-list/document-list.component'
|
||||||
import { CorrespondentListComponent } from './components/manage/correspondent-list/correspondent-list.component'
|
import { CorrespondentListComponent } from './components/manage/correspondent-list/correspondent-list.component'
|
||||||
|
import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component'
|
||||||
import { DocumentTypeListComponent } from './components/manage/document-type-list/document-type-list.component'
|
import { DocumentTypeListComponent } from './components/manage/document-type-list/document-type-list.component'
|
||||||
import { LogsComponent } from './components/admin/logs/logs.component'
|
import { MailComponent } from './components/manage/mail/mail.component'
|
||||||
import { SettingsComponent } from './components/admin/settings/settings.component'
|
import { SavedViewsComponent } from './components/manage/saved-views/saved-views.component'
|
||||||
import { TagListComponent } from './components/manage/tag-list/tag-list.component'
|
|
||||||
import { NotFoundComponent } from './components/not-found/not-found.component'
|
|
||||||
import { DocumentAsnComponent } from './components/document-asn/document-asn.component'
|
|
||||||
import { DirtyFormGuard } from './guards/dirty-form.guard'
|
|
||||||
import { StoragePathListComponent } from './components/manage/storage-path-list/storage-path-list.component'
|
import { StoragePathListComponent } from './components/manage/storage-path-list/storage-path-list.component'
|
||||||
import { TasksComponent } from './components/admin/tasks/tasks.component'
|
import { TagListComponent } from './components/manage/tag-list/tag-list.component'
|
||||||
import { PermissionsGuard } from './guards/permissions.guard'
|
import { WorkflowsComponent } from './components/manage/workflows/workflows.component'
|
||||||
|
import { NotFoundComponent } from './components/not-found/not-found.component'
|
||||||
import { DirtyDocGuard } from './guards/dirty-doc.guard'
|
import { DirtyDocGuard } from './guards/dirty-doc.guard'
|
||||||
|
import { DirtyFormGuard } from './guards/dirty-form.guard'
|
||||||
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
|
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
|
||||||
|
import { PermissionsGuard } from './guards/permissions.guard'
|
||||||
import {
|
import {
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
PermissionType,
|
PermissionType,
|
||||||
} from './services/permissions.service'
|
} from './services/permissions.service'
|
||||||
import { WorkflowsComponent } from './components/manage/workflows/workflows.component'
|
|
||||||
import { MailComponent } from './components/manage/mail/mail.component'
|
|
||||||
import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component'
|
|
||||||
import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component'
|
|
||||||
import { ConfigComponent } from './components/admin/config/config.component'
|
|
||||||
import { TrashComponent } from './components/admin/trash/trash.component'
|
|
||||||
|
|
||||||
export const routes: Routes = [
|
export const routes: Routes = [
|
||||||
{ path: '', redirectTo: 'dashboard', pathMatch: 'full' },
|
{ path: '', redirectTo: 'dashboard', pathMatch: 'full' },
|
||||||
@ -165,6 +166,10 @@ export const routes: Routes = [
|
|||||||
path: 'settings/usersgroups',
|
path: 'settings/usersgroups',
|
||||||
redirectTo: '/usersgroups',
|
redirectTo: '/usersgroups',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'settings/savedviews',
|
||||||
|
redirectTo: '/savedviews',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: 'settings',
|
path: 'settings',
|
||||||
component: SettingsComponent,
|
component: SettingsComponent,
|
||||||
@ -255,6 +260,17 @@ export const routes: Routes = [
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'savedviews',
|
||||||
|
component: SavedViewsComponent,
|
||||||
|
canActivate: [PermissionsGuard],
|
||||||
|
data: {
|
||||||
|
requiredPermission: {
|
||||||
|
action: PermissionAction.View,
|
||||||
|
type: PermissionType.SavedView,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -1,30 +1,31 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
TestBed,
|
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
|
TestBed,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import { Router, RouterModule } from '@angular/router'
|
import { Router, RouterModule } from '@angular/router'
|
||||||
import { TourService, TourNgBootstrapModule } from 'ngx-ui-tour-ng-bootstrap'
|
import { NgbModalModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { NgxFileDropModule } from 'ngx-file-drop'
|
||||||
|
import { TourNgBootstrapModule, TourService } from 'ngx-ui-tour-ng-bootstrap'
|
||||||
import { Subject } from 'rxjs'
|
import { Subject } from 'rxjs'
|
||||||
import { routes } from './app-routing.module'
|
import { routes } from './app-routing.module'
|
||||||
import { AppComponent } from './app.component'
|
import { AppComponent } from './app.component'
|
||||||
import { ToastsComponent } from './components/common/toasts/toasts.component'
|
import { ToastsComponent } from './components/common/toasts/toasts.component'
|
||||||
|
import { FileDropComponent } from './components/file-drop/file-drop.component'
|
||||||
|
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
|
||||||
|
import { PermissionsGuard } from './guards/permissions.guard'
|
||||||
import {
|
import {
|
||||||
ConsumerStatusService,
|
ConsumerStatusService,
|
||||||
FileStatus,
|
FileStatus,
|
||||||
} from './services/consumer-status.service'
|
} from './services/consumer-status.service'
|
||||||
import { PermissionsService } from './services/permissions.service'
|
|
||||||
import { ToastService, Toast } from './services/toast.service'
|
|
||||||
import { SettingsService } from './services/settings.service'
|
|
||||||
import { FileDropComponent } from './components/file-drop/file-drop.component'
|
|
||||||
import { NgxFileDropModule } from 'ngx-file-drop'
|
|
||||||
import { NgbModalModule } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { HotKeyService } from './services/hot-key.service'
|
import { HotKeyService } from './services/hot-key.service'
|
||||||
import { PermissionsGuard } from './guards/permissions.guard'
|
import { PermissionsService } from './services/permissions.service'
|
||||||
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
|
import { SettingsService } from './services/settings.service'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { Toast, ToastService } from './services/toast.service'
|
||||||
|
|
||||||
describe('AppComponent', () => {
|
describe('AppComponent', () => {
|
||||||
let component: AppComponent
|
let component: AppComponent
|
||||||
@ -39,12 +40,15 @@ describe('AppComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [AppComponent, ToastsComponent, FileDropComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
TourNgBootstrapModule,
|
TourNgBootstrapModule,
|
||||||
RouterModule.forRoot(routes),
|
RouterModule.forRoot(routes),
|
||||||
NgxFileDropModule,
|
NgxFileDropModule,
|
||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
|
AppComponent,
|
||||||
|
ToastsComponent,
|
||||||
|
FileDropComponent,
|
||||||
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
PermissionsGuard,
|
PermissionsGuard,
|
||||||
|
@ -1,23 +1,31 @@
|
|||||||
import { SettingsService } from './services/settings.service'
|
|
||||||
import { SETTINGS_KEYS } from './data/ui-settings'
|
|
||||||
import { Component, OnDestroy, OnInit, Renderer2 } from '@angular/core'
|
import { Component, OnDestroy, OnInit, Renderer2 } from '@angular/core'
|
||||||
import { Router } from '@angular/router'
|
import { Router, RouterOutlet } from '@angular/router'
|
||||||
import { Subscription, first } from 'rxjs'
|
import { TourNgBootstrapModule, TourService } from 'ngx-ui-tour-ng-bootstrap'
|
||||||
|
import { first, Subscription } from 'rxjs'
|
||||||
|
import { ToastsComponent } from './components/common/toasts/toasts.component'
|
||||||
|
import { FileDropComponent } from './components/file-drop/file-drop.component'
|
||||||
|
import { SETTINGS_KEYS } from './data/ui-settings'
|
||||||
import { ConsumerStatusService } from './services/consumer-status.service'
|
import { ConsumerStatusService } from './services/consumer-status.service'
|
||||||
import { ToastService } from './services/toast.service'
|
import { HotKeyService } from './services/hot-key.service'
|
||||||
import { TasksService } from './services/tasks.service'
|
|
||||||
import { TourService } from 'ngx-ui-tour-ng-bootstrap'
|
|
||||||
import {
|
import {
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
PermissionsService,
|
PermissionsService,
|
||||||
PermissionType,
|
PermissionType,
|
||||||
} from './services/permissions.service'
|
} from './services/permissions.service'
|
||||||
import { HotKeyService } from './services/hot-key.service'
|
import { SettingsService } from './services/settings.service'
|
||||||
|
import { TasksService } from './services/tasks.service'
|
||||||
|
import { ToastService } from './services/toast.service'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-root',
|
selector: 'pngx-root',
|
||||||
templateUrl: './app.component.html',
|
templateUrl: './app.component.html',
|
||||||
styleUrls: ['./app.component.scss'],
|
styleUrls: ['./app.component.scss'],
|
||||||
|
imports: [
|
||||||
|
FileDropComponent,
|
||||||
|
ToastsComponent,
|
||||||
|
TourNgBootstrapModule,
|
||||||
|
RouterOutlet,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class AppComponent implements OnInit, OnDestroy {
|
export class AppComponent implements OnInit, OnDestroy {
|
||||||
newDocumentSubscription: Subscription
|
newDocumentSubscription: Subscription
|
||||||
@ -165,7 +173,7 @@ export class AppComponent implements OnInit, OnDestroy {
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
anchorId: 'tour.dashboard',
|
anchorId: 'tour.dashboard',
|
||||||
content: $localize`The dashboard can be used to show saved views, such as an 'Inbox'. Those settings are found under Settings > Saved Views once you have created some.`,
|
content: $localize`The dashboard can be used to show saved views, such as an 'Inbox'. Views are found under Manage > Saved Views once you have created some.`,
|
||||||
route: '/dashboard',
|
route: '/dashboard',
|
||||||
delayAfterNavigation: 500,
|
delayAfterNavigation: 500,
|
||||||
isOptional: false,
|
isOptional: false,
|
||||||
@ -227,7 +235,7 @@ export class AppComponent implements OnInit, OnDestroy {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
anchorId: 'tour.settings',
|
anchorId: 'tour.settings',
|
||||||
content: $localize`Check out the settings for various tweaks to the web app and toggle settings for saved views.`,
|
content: $localize`Check out the settings for various tweaks to the web app.`,
|
||||||
route: '/settings',
|
route: '/settings',
|
||||||
backdropConfig: {
|
backdropConfig: {
|
||||||
offset: 0,
|
offset: 0,
|
||||||
|
@ -1,571 +0,0 @@
|
|||||||
import { BrowserModule } from '@angular/platform-browser'
|
|
||||||
import { APP_INITIALIZER, NgModule } from '@angular/core'
|
|
||||||
import { AppRoutingModule } from './app-routing.module'
|
|
||||||
import { AppComponent } from './app.component'
|
|
||||||
import {
|
|
||||||
NgbDateAdapter,
|
|
||||||
NgbDateParserFormatter,
|
|
||||||
NgbModule,
|
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import {
|
|
||||||
HTTP_INTERCEPTORS,
|
|
||||||
provideHttpClient,
|
|
||||||
withInterceptorsFromDi,
|
|
||||||
} from '@angular/common/http'
|
|
||||||
import { DocumentListComponent } from './components/document-list/document-list.component'
|
|
||||||
import { DocumentDetailComponent } from './components/document-detail/document-detail.component'
|
|
||||||
import { DashboardComponent } from './components/dashboard/dashboard.component'
|
|
||||||
import { TagListComponent } from './components/manage/tag-list/tag-list.component'
|
|
||||||
import { DocumentTypeListComponent } from './components/manage/document-type-list/document-type-list.component'
|
|
||||||
import { CorrespondentListComponent } from './components/manage/correspondent-list/correspondent-list.component'
|
|
||||||
import { LogsComponent } from './components/admin/logs/logs.component'
|
|
||||||
import { SettingsComponent } from './components/admin/settings/settings.component'
|
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
|
||||||
import { DatePipe, registerLocaleData } from '@angular/common'
|
|
||||||
import { NotFoundComponent } from './components/not-found/not-found.component'
|
|
||||||
import { ConfirmDialogComponent } from './components/common/confirm-dialog/confirm-dialog.component'
|
|
||||||
import { CorrespondentEditDialogComponent } from './components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component'
|
|
||||||
import { TagEditDialogComponent } from './components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component'
|
|
||||||
import { DocumentTypeEditDialogComponent } from './components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component'
|
|
||||||
import { TagComponent } from './components/common/tag/tag.component'
|
|
||||||
import { ClearableBadgeComponent } from './components/common/clearable-badge/clearable-badge.component'
|
|
||||||
import { PageHeaderComponent } from './components/common/page-header/page-header.component'
|
|
||||||
import { AppFrameComponent } from './components/app-frame/app-frame.component'
|
|
||||||
import { ToastsComponent } from './components/common/toasts/toasts.component'
|
|
||||||
import { FilterEditorComponent } from './components/document-list/filter-editor/filter-editor.component'
|
|
||||||
import { FilterableDropdownComponent } from './components/common/filterable-dropdown/filterable-dropdown.component'
|
|
||||||
import { ToggleableDropdownButtonComponent } from './components/common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'
|
|
||||||
import { DatesDropdownComponent } from './components/common/dates-dropdown/dates-dropdown.component'
|
|
||||||
import { DocumentCardLargeComponent } from './components/document-list/document-card-large/document-card-large.component'
|
|
||||||
import { DocumentCardSmallComponent } from './components/document-list/document-card-small/document-card-small.component'
|
|
||||||
import { BulkEditorComponent } from './components/document-list/bulk-editor/bulk-editor.component'
|
|
||||||
import { NgxFileDropModule } from 'ngx-file-drop'
|
|
||||||
import { TextComponent } from './components/common/input/text/text.component'
|
|
||||||
import { 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'
|
|
||||||
import { PasswordComponent } from './components/common/input/password/password.component'
|
|
||||||
import { SaveViewConfigDialogComponent } from './components/document-list/save-view-config-dialog/save-view-config-dialog.component'
|
|
||||||
import { TagsComponent } from './components/common/input/tags/tags.component'
|
|
||||||
import { IfPermissionsDirective } from './directives/if-permissions.directive'
|
|
||||||
import { SortableDirective } from './directives/sortable.directive'
|
|
||||||
import { CookieService } from 'ngx-cookie-service'
|
|
||||||
import { CsrfInterceptor } from './interceptors/csrf.interceptor'
|
|
||||||
import { SavedViewWidgetComponent } from './components/dashboard/widgets/saved-view-widget/saved-view-widget.component'
|
|
||||||
import { StatisticsWidgetComponent } from './components/dashboard/widgets/statistics-widget/statistics-widget.component'
|
|
||||||
import { UploadFileWidgetComponent } from './components/dashboard/widgets/upload-file-widget/upload-file-widget.component'
|
|
||||||
import { WidgetFrameComponent } from './components/dashboard/widgets/widget-frame/widget-frame.component'
|
|
||||||
import { WelcomeWidgetComponent } from './components/dashboard/widgets/welcome-widget/welcome-widget.component'
|
|
||||||
import { YesNoPipe } from './pipes/yes-no.pipe'
|
|
||||||
import { FileSizePipe } from './pipes/file-size.pipe'
|
|
||||||
import { FilterPipe } from './pipes/filter.pipe'
|
|
||||||
import { DocumentTitlePipe } from './pipes/document-title.pipe'
|
|
||||||
import { MetadataCollapseComponent } from './components/document-detail/metadata-collapse/metadata-collapse.component'
|
|
||||||
import { SelectDialogComponent } from './components/common/select-dialog/select-dialog.component'
|
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
|
||||||
import { NumberComponent } from './components/common/input/number/number.component'
|
|
||||||
import { SafeUrlPipe } from './pipes/safeurl.pipe'
|
|
||||||
import { SafeHtmlPipe } from './pipes/safehtml.pipe'
|
|
||||||
import { CustomDatePipe } from './pipes/custom-date.pipe'
|
|
||||||
import { DateComponent } from './components/common/input/date/date.component'
|
|
||||||
import { ISODateAdapter } from './utils/ngb-iso-date-adapter'
|
|
||||||
import { LocalizedDateParserFormatter } from './utils/ngb-date-parser-formatter'
|
|
||||||
import { ApiVersionInterceptor } from './interceptors/api-version.interceptor'
|
|
||||||
import { ColorSliderModule } from 'ngx-color/slider'
|
|
||||||
import { ColorComponent } from './components/common/input/color/color.component'
|
|
||||||
import { DocumentAsnComponent } from './components/document-asn/document-asn.component'
|
|
||||||
import { DocumentNotesComponent } from './components/document-notes/document-notes.component'
|
|
||||||
import { PermissionsGuard } from './guards/permissions.guard'
|
|
||||||
import { DirtyDocGuard } from './guards/dirty-doc.guard'
|
|
||||||
import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard'
|
|
||||||
import { StoragePathListComponent } from './components/manage/storage-path-list/storage-path-list.component'
|
|
||||||
import { StoragePathEditDialogComponent } from './components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component'
|
|
||||||
import { SettingsService } from './services/settings.service'
|
|
||||||
import { TasksComponent } from './components/admin/tasks/tasks.component'
|
|
||||||
import { TourNgBootstrapModule } from 'ngx-ui-tour-ng-bootstrap'
|
|
||||||
import { UserEditDialogComponent } from './components/common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
|
||||||
import { GroupEditDialogComponent } from './components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
|
||||||
import { PermissionsSelectComponent } from './components/common/permissions-select/permissions-select.component'
|
|
||||||
import { MailAccountEditDialogComponent } from './components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component'
|
|
||||||
import { MailRuleEditDialogComponent } from './components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component'
|
|
||||||
import { PermissionsUserComponent } from './components/common/input/permissions/permissions-user/permissions-user.component'
|
|
||||||
import { PermissionsGroupComponent } from './components/common/input/permissions/permissions-group/permissions-group.component'
|
|
||||||
import { IfOwnerDirective } from './directives/if-owner.directive'
|
|
||||||
import { IfObjectPermissionsDirective } from './directives/if-object-permissions.directive'
|
|
||||||
import { PermissionsDialogComponent } from './components/common/permissions-dialog/permissions-dialog.component'
|
|
||||||
import { PermissionsFormComponent } from './components/common/input/permissions/permissions-form/permissions-form.component'
|
|
||||||
import { PermissionsFilterDropdownComponent } from './components/common/permissions-filter-dropdown/permissions-filter-dropdown.component'
|
|
||||||
import { UsernamePipe } from './pipes/username.pipe'
|
|
||||||
import { LogoComponent } from './components/common/logo/logo.component'
|
|
||||||
import { IsNumberPipe } from './pipes/is-number.pipe'
|
|
||||||
import { ShareLinksDropdownComponent } from './components/common/share-links-dropdown/share-links-dropdown.component'
|
|
||||||
import { WorkflowsComponent } from './components/manage/workflows/workflows.component'
|
|
||||||
import { WorkflowEditDialogComponent } from './components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
|
|
||||||
import { MailComponent } from './components/manage/mail/mail.component'
|
|
||||||
import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component'
|
|
||||||
import { DragDropModule } from '@angular/cdk/drag-drop'
|
|
||||||
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'
|
|
||||||
import { PreviewPopupComponent } from './components/common/preview-popup/preview-popup.component'
|
|
||||||
import { SwitchComponent } from './components/common/input/switch/switch.component'
|
|
||||||
import { ConfigComponent } from './components/admin/config/config.component'
|
|
||||||
import { FileComponent } from './components/common/input/file/file.component'
|
|
||||||
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
|
||||||
import { ConfirmButtonComponent } from './components/common/confirm-button/confirm-button.component'
|
|
||||||
import { MonetaryComponent } from './components/common/input/monetary/monetary.component'
|
|
||||||
import { SystemStatusDialogComponent } from './components/common/system-status-dialog/system-status-dialog.component'
|
|
||||||
import { RotateConfirmDialogComponent } from './components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component'
|
|
||||||
import { MergeConfirmDialogComponent } from './components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component'
|
|
||||||
import { SplitConfirmDialogComponent } from './components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component'
|
|
||||||
import { DocumentHistoryComponent } from './components/document-history/document-history.component'
|
|
||||||
import { DragDropSelectComponent } from './components/common/input/drag-drop-select/drag-drop-select.component'
|
|
||||||
import { CustomFieldDisplayComponent } from './components/common/custom-field-display/custom-field-display.component'
|
|
||||||
import { GlobalSearchComponent } from './components/app-frame/global-search/global-search.component'
|
|
||||||
import { HotkeyDialogComponent } from './components/common/hotkey-dialog/hotkey-dialog.component'
|
|
||||||
import { DeletePagesConfirmDialogComponent } from './components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component'
|
|
||||||
import { TrashComponent } from './components/admin/trash/trash.component'
|
|
||||||
import {
|
|
||||||
airplane,
|
|
||||||
archive,
|
|
||||||
arrowClockwise,
|
|
||||||
arrowCounterclockwise,
|
|
||||||
arrowDown,
|
|
||||||
arrowLeft,
|
|
||||||
arrowRepeat,
|
|
||||||
arrowRight,
|
|
||||||
arrowRightShort,
|
|
||||||
arrowUpRight,
|
|
||||||
asterisk,
|
|
||||||
braces,
|
|
||||||
bodyText,
|
|
||||||
boxArrowInRight,
|
|
||||||
boxArrowUp,
|
|
||||||
boxArrowUpRight,
|
|
||||||
boxes,
|
|
||||||
calendar,
|
|
||||||
calendarEvent,
|
|
||||||
calendarEventFill,
|
|
||||||
cardChecklist,
|
|
||||||
cardHeading,
|
|
||||||
caretDown,
|
|
||||||
caretUp,
|
|
||||||
chatLeftText,
|
|
||||||
check,
|
|
||||||
check2All,
|
|
||||||
checkAll,
|
|
||||||
checkCircleFill,
|
|
||||||
checkLg,
|
|
||||||
chevronDoubleLeft,
|
|
||||||
chevronDoubleRight,
|
|
||||||
clipboard,
|
|
||||||
clipboardCheck,
|
|
||||||
clipboardCheckFill,
|
|
||||||
clipboardFill,
|
|
||||||
dash,
|
|
||||||
dashCircle,
|
|
||||||
diagram3,
|
|
||||||
dice5,
|
|
||||||
doorOpen,
|
|
||||||
download,
|
|
||||||
envelope,
|
|
||||||
envelopeAt,
|
|
||||||
envelopeAtFill,
|
|
||||||
exclamationCircleFill,
|
|
||||||
exclamationTriangle,
|
|
||||||
exclamationTriangleFill,
|
|
||||||
eye,
|
|
||||||
fileEarmark,
|
|
||||||
fileEarmarkCheck,
|
|
||||||
fileEarmarkFill,
|
|
||||||
fileEarmarkLock,
|
|
||||||
fileEarmarkMinus,
|
|
||||||
files,
|
|
||||||
fileText,
|
|
||||||
filter,
|
|
||||||
folder,
|
|
||||||
folderFill,
|
|
||||||
funnel,
|
|
||||||
gear,
|
|
||||||
google,
|
|
||||||
grid,
|
|
||||||
gripVertical,
|
|
||||||
hash,
|
|
||||||
hddStack,
|
|
||||||
house,
|
|
||||||
infoCircle,
|
|
||||||
journals,
|
|
||||||
link,
|
|
||||||
listTask,
|
|
||||||
listUl,
|
|
||||||
microsoft,
|
|
||||||
nodePlus,
|
|
||||||
pencil,
|
|
||||||
people,
|
|
||||||
peopleFill,
|
|
||||||
person,
|
|
||||||
personCircle,
|
|
||||||
personFill,
|
|
||||||
personFillLock,
|
|
||||||
personLock,
|
|
||||||
personSquare,
|
|
||||||
plus,
|
|
||||||
plusCircle,
|
|
||||||
questionCircle,
|
|
||||||
scissors,
|
|
||||||
search,
|
|
||||||
slashCircle,
|
|
||||||
sliders2Vertical,
|
|
||||||
sortAlphaDown,
|
|
||||||
sortAlphaUpAlt,
|
|
||||||
tagFill,
|
|
||||||
tag,
|
|
||||||
tags,
|
|
||||||
textIndentLeft,
|
|
||||||
textLeft,
|
|
||||||
threeDots,
|
|
||||||
threeDotsVertical,
|
|
||||||
trash,
|
|
||||||
uiRadios,
|
|
||||||
upcScan,
|
|
||||||
x,
|
|
||||||
xCircle,
|
|
||||||
xLg,
|
|
||||||
} from 'ngx-bootstrap-icons'
|
|
||||||
|
|
||||||
const icons = {
|
|
||||||
airplane,
|
|
||||||
archive,
|
|
||||||
arrowClockwise,
|
|
||||||
arrowCounterclockwise,
|
|
||||||
arrowDown,
|
|
||||||
arrowLeft,
|
|
||||||
arrowRepeat,
|
|
||||||
arrowRight,
|
|
||||||
arrowRightShort,
|
|
||||||
arrowUpRight,
|
|
||||||
asterisk,
|
|
||||||
braces,
|
|
||||||
bodyText,
|
|
||||||
boxArrowInRight,
|
|
||||||
boxArrowUp,
|
|
||||||
boxArrowUpRight,
|
|
||||||
boxes,
|
|
||||||
calendar,
|
|
||||||
calendarEvent,
|
|
||||||
calendarEventFill,
|
|
||||||
cardChecklist,
|
|
||||||
cardHeading,
|
|
||||||
caretDown,
|
|
||||||
caretUp,
|
|
||||||
chatLeftText,
|
|
||||||
check,
|
|
||||||
check2All,
|
|
||||||
checkAll,
|
|
||||||
checkCircleFill,
|
|
||||||
checkLg,
|
|
||||||
chevronDoubleLeft,
|
|
||||||
chevronDoubleRight,
|
|
||||||
clipboard,
|
|
||||||
clipboardCheck,
|
|
||||||
clipboardCheckFill,
|
|
||||||
clipboardFill,
|
|
||||||
dash,
|
|
||||||
dashCircle,
|
|
||||||
diagram3,
|
|
||||||
dice5,
|
|
||||||
doorOpen,
|
|
||||||
download,
|
|
||||||
envelope,
|
|
||||||
envelopeAt,
|
|
||||||
envelopeAtFill,
|
|
||||||
exclamationCircleFill,
|
|
||||||
exclamationTriangle,
|
|
||||||
exclamationTriangleFill,
|
|
||||||
eye,
|
|
||||||
fileEarmark,
|
|
||||||
fileEarmarkCheck,
|
|
||||||
fileEarmarkFill,
|
|
||||||
fileEarmarkLock,
|
|
||||||
fileEarmarkMinus,
|
|
||||||
files,
|
|
||||||
fileText,
|
|
||||||
filter,
|
|
||||||
folder,
|
|
||||||
folderFill,
|
|
||||||
funnel,
|
|
||||||
gear,
|
|
||||||
google,
|
|
||||||
grid,
|
|
||||||
gripVertical,
|
|
||||||
hash,
|
|
||||||
hddStack,
|
|
||||||
house,
|
|
||||||
infoCircle,
|
|
||||||
journals,
|
|
||||||
link,
|
|
||||||
listTask,
|
|
||||||
listUl,
|
|
||||||
microsoft,
|
|
||||||
nodePlus,
|
|
||||||
pencil,
|
|
||||||
people,
|
|
||||||
peopleFill,
|
|
||||||
person,
|
|
||||||
personCircle,
|
|
||||||
personFill,
|
|
||||||
personFillLock,
|
|
||||||
personLock,
|
|
||||||
personSquare,
|
|
||||||
plus,
|
|
||||||
plusCircle,
|
|
||||||
questionCircle,
|
|
||||||
scissors,
|
|
||||||
search,
|
|
||||||
slashCircle,
|
|
||||||
sliders2Vertical,
|
|
||||||
sortAlphaDown,
|
|
||||||
sortAlphaUpAlt,
|
|
||||||
tagFill,
|
|
||||||
tag,
|
|
||||||
tags,
|
|
||||||
textIndentLeft,
|
|
||||||
textLeft,
|
|
||||||
threeDots,
|
|
||||||
threeDotsVertical,
|
|
||||||
trash,
|
|
||||||
uiRadios,
|
|
||||||
upcScan,
|
|
||||||
x,
|
|
||||||
xCircle,
|
|
||||||
xLg,
|
|
||||||
}
|
|
||||||
|
|
||||||
import localeAf from '@angular/common/locales/af'
|
|
||||||
import localeAr from '@angular/common/locales/ar'
|
|
||||||
import localeBe from '@angular/common/locales/be'
|
|
||||||
import localeBg from '@angular/common/locales/bg'
|
|
||||||
import localeCa from '@angular/common/locales/ca'
|
|
||||||
import localeCs from '@angular/common/locales/cs'
|
|
||||||
import localeDa from '@angular/common/locales/da'
|
|
||||||
import localeDe from '@angular/common/locales/de'
|
|
||||||
import localeEl from '@angular/common/locales/el'
|
|
||||||
import localeEnGb from '@angular/common/locales/en-GB'
|
|
||||||
import localeEs from '@angular/common/locales/es'
|
|
||||||
import localeFi from '@angular/common/locales/fi'
|
|
||||||
import localeFr from '@angular/common/locales/fr'
|
|
||||||
import localeHu from '@angular/common/locales/hu'
|
|
||||||
import localeIt from '@angular/common/locales/it'
|
|
||||||
import localeJa from '@angular/common/locales/ja'
|
|
||||||
import localeKo from '@angular/common/locales/ko'
|
|
||||||
import localeLb from '@angular/common/locales/lb'
|
|
||||||
import localeNl from '@angular/common/locales/nl'
|
|
||||||
import localeNo from '@angular/common/locales/no'
|
|
||||||
import localePl from '@angular/common/locales/pl'
|
|
||||||
import localePt from '@angular/common/locales/pt'
|
|
||||||
import localeRo from '@angular/common/locales/ro'
|
|
||||||
import localeRu from '@angular/common/locales/ru'
|
|
||||||
import localeSk from '@angular/common/locales/sk'
|
|
||||||
import localeSl from '@angular/common/locales/sl'
|
|
||||||
import localeSr from '@angular/common/locales/sr'
|
|
||||||
import localeSv from '@angular/common/locales/sv'
|
|
||||||
import localeTr from '@angular/common/locales/tr'
|
|
||||||
import localeUk from '@angular/common/locales/uk'
|
|
||||||
import localeZh from '@angular/common/locales/zh'
|
|
||||||
|
|
||||||
registerLocaleData(localeAf)
|
|
||||||
registerLocaleData(localeAr)
|
|
||||||
registerLocaleData(localeBe)
|
|
||||||
registerLocaleData(localeBg)
|
|
||||||
registerLocaleData(localeCa)
|
|
||||||
registerLocaleData(localeCs)
|
|
||||||
registerLocaleData(localeDa)
|
|
||||||
registerLocaleData(localeDe)
|
|
||||||
registerLocaleData(localeEl)
|
|
||||||
registerLocaleData(localeEnGb)
|
|
||||||
registerLocaleData(localeEs)
|
|
||||||
registerLocaleData(localeFi)
|
|
||||||
registerLocaleData(localeFr)
|
|
||||||
registerLocaleData(localeHu)
|
|
||||||
registerLocaleData(localeIt)
|
|
||||||
registerLocaleData(localeJa)
|
|
||||||
registerLocaleData(localeKo)
|
|
||||||
registerLocaleData(localeLb)
|
|
||||||
registerLocaleData(localeNl)
|
|
||||||
registerLocaleData(localeNo)
|
|
||||||
registerLocaleData(localePl)
|
|
||||||
registerLocaleData(localePt, 'pt-BR')
|
|
||||||
registerLocaleData(localePt, 'pt-PT')
|
|
||||||
registerLocaleData(localeRo)
|
|
||||||
registerLocaleData(localeRu)
|
|
||||||
registerLocaleData(localeSk)
|
|
||||||
registerLocaleData(localeSl)
|
|
||||||
registerLocaleData(localeSr)
|
|
||||||
registerLocaleData(localeSv)
|
|
||||||
registerLocaleData(localeTr)
|
|
||||||
registerLocaleData(localeUk)
|
|
||||||
registerLocaleData(localeZh)
|
|
||||||
|
|
||||||
function initializeApp(settings: SettingsService) {
|
|
||||||
return () => {
|
|
||||||
return settings.initializeSettings()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@NgModule({
|
|
||||||
declarations: [
|
|
||||||
AppComponent,
|
|
||||||
DocumentListComponent,
|
|
||||||
DocumentDetailComponent,
|
|
||||||
DashboardComponent,
|
|
||||||
TagListComponent,
|
|
||||||
DocumentTypeListComponent,
|
|
||||||
CorrespondentListComponent,
|
|
||||||
StoragePathListComponent,
|
|
||||||
LogsComponent,
|
|
||||||
SettingsComponent,
|
|
||||||
NotFoundComponent,
|
|
||||||
CorrespondentEditDialogComponent,
|
|
||||||
ConfirmDialogComponent,
|
|
||||||
TagEditDialogComponent,
|
|
||||||
DocumentTypeEditDialogComponent,
|
|
||||||
StoragePathEditDialogComponent,
|
|
||||||
TagComponent,
|
|
||||||
ClearableBadgeComponent,
|
|
||||||
PageHeaderComponent,
|
|
||||||
AppFrameComponent,
|
|
||||||
ToastsComponent,
|
|
||||||
FilterEditorComponent,
|
|
||||||
FilterableDropdownComponent,
|
|
||||||
ToggleableDropdownButtonComponent,
|
|
||||||
DatesDropdownComponent,
|
|
||||||
DocumentCardLargeComponent,
|
|
||||||
DocumentCardSmallComponent,
|
|
||||||
BulkEditorComponent,
|
|
||||||
TextComponent,
|
|
||||||
TextAreaComponent,
|
|
||||||
SelectComponent,
|
|
||||||
CheckComponent,
|
|
||||||
UrlComponent,
|
|
||||||
PasswordComponent,
|
|
||||||
SaveViewConfigDialogComponent,
|
|
||||||
TagsComponent,
|
|
||||||
IfPermissionsDirective,
|
|
||||||
SortableDirective,
|
|
||||||
SavedViewWidgetComponent,
|
|
||||||
StatisticsWidgetComponent,
|
|
||||||
UploadFileWidgetComponent,
|
|
||||||
WidgetFrameComponent,
|
|
||||||
WelcomeWidgetComponent,
|
|
||||||
YesNoPipe,
|
|
||||||
FileSizePipe,
|
|
||||||
FilterPipe,
|
|
||||||
DocumentTitlePipe,
|
|
||||||
MetadataCollapseComponent,
|
|
||||||
SelectDialogComponent,
|
|
||||||
NumberComponent,
|
|
||||||
SafeUrlPipe,
|
|
||||||
SafeHtmlPipe,
|
|
||||||
CustomDatePipe,
|
|
||||||
DateComponent,
|
|
||||||
ColorComponent,
|
|
||||||
DocumentAsnComponent,
|
|
||||||
DocumentNotesComponent,
|
|
||||||
TasksComponent,
|
|
||||||
UserEditDialogComponent,
|
|
||||||
GroupEditDialogComponent,
|
|
||||||
PermissionsSelectComponent,
|
|
||||||
MailAccountEditDialogComponent,
|
|
||||||
MailRuleEditDialogComponent,
|
|
||||||
PermissionsUserComponent,
|
|
||||||
PermissionsGroupComponent,
|
|
||||||
IfOwnerDirective,
|
|
||||||
IfObjectPermissionsDirective,
|
|
||||||
PermissionsDialogComponent,
|
|
||||||
PermissionsFormComponent,
|
|
||||||
PermissionsFilterDropdownComponent,
|
|
||||||
UsernamePipe,
|
|
||||||
LogoComponent,
|
|
||||||
IsNumberPipe,
|
|
||||||
ShareLinksDropdownComponent,
|
|
||||||
WorkflowsComponent,
|
|
||||||
WorkflowEditDialogComponent,
|
|
||||||
MailComponent,
|
|
||||||
UsersAndGroupsComponent,
|
|
||||||
FileDropComponent,
|
|
||||||
CustomFieldsComponent,
|
|
||||||
CustomFieldEditDialogComponent,
|
|
||||||
CustomFieldsDropdownComponent,
|
|
||||||
CustomFieldsQueryDropdownComponent,
|
|
||||||
ProfileEditDialogComponent,
|
|
||||||
DocumentLinkComponent,
|
|
||||||
PreviewPopupComponent,
|
|
||||||
SwitchComponent,
|
|
||||||
ConfigComponent,
|
|
||||||
FileComponent,
|
|
||||||
ConfirmButtonComponent,
|
|
||||||
MonetaryComponent,
|
|
||||||
SystemStatusDialogComponent,
|
|
||||||
RotateConfirmDialogComponent,
|
|
||||||
MergeConfirmDialogComponent,
|
|
||||||
SplitConfirmDialogComponent,
|
|
||||||
DocumentHistoryComponent,
|
|
||||||
DragDropSelectComponent,
|
|
||||||
CustomFieldDisplayComponent,
|
|
||||||
GlobalSearchComponent,
|
|
||||||
HotkeyDialogComponent,
|
|
||||||
DeletePagesConfirmDialogComponent,
|
|
||||||
TrashComponent,
|
|
||||||
],
|
|
||||||
bootstrap: [AppComponent],
|
|
||||||
imports: [
|
|
||||||
BrowserModule,
|
|
||||||
AppRoutingModule,
|
|
||||||
NgbModule,
|
|
||||||
FormsModule,
|
|
||||||
ReactiveFormsModule,
|
|
||||||
PdfViewerModule,
|
|
||||||
NgxFileDropModule,
|
|
||||||
NgSelectModule,
|
|
||||||
ColorSliderModule,
|
|
||||||
TourNgBootstrapModule,
|
|
||||||
DragDropModule,
|
|
||||||
NgxBootstrapIconsModule.pick(icons),
|
|
||||||
],
|
|
||||||
providers: [
|
|
||||||
{
|
|
||||||
provide: APP_INITIALIZER,
|
|
||||||
useFactory: initializeApp,
|
|
||||||
deps: [SettingsService],
|
|
||||||
multi: true,
|
|
||||||
},
|
|
||||||
DatePipe,
|
|
||||||
CookieService,
|
|
||||||
{
|
|
||||||
provide: HTTP_INTERCEPTORS,
|
|
||||||
useClass: CsrfInterceptor,
|
|
||||||
multi: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
provide: HTTP_INTERCEPTORS,
|
|
||||||
useClass: ApiVersionInterceptor,
|
|
||||||
multi: true,
|
|
||||||
},
|
|
||||||
FilterPipe,
|
|
||||||
DocumentTitlePipe,
|
|
||||||
{ provide: NgbDateAdapter, useClass: ISODateAdapter },
|
|
||||||
{ provide: NgbDateParserFormatter, useClass: LocalizedDateParserFormatter },
|
|
||||||
PermissionsGuard,
|
|
||||||
DirtyDocGuard,
|
|
||||||
DirtySavedViewGuard,
|
|
||||||
UsernamePipe,
|
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
|
||||||
],
|
|
||||||
})
|
|
||||||
export class AppModule {}
|
|
@ -1,24 +1,24 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
|
||||||
import { ConfigComponent } from './config.component'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { ConfigService } from 'src/app/services/config.service'
|
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { of, throwError } from 'rxjs'
|
|
||||||
import { OutputTypeConfig } from 'src/app/data/paperless-config'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { BrowserModule } from '@angular/platform-browser'
|
import { BrowserModule } from '@angular/platform-browser'
|
||||||
import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
import { TextComponent } from '../../common/input/text/text.component'
|
|
||||||
import { NumberComponent } from '../../common/input/number/number.component'
|
|
||||||
import { SwitchComponent } from '../../common/input/switch/switch.component'
|
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
|
||||||
import { SelectComponent } from '../../common/input/select/select.component'
|
|
||||||
import { FileComponent } from '../../common/input/file/file.component'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { of, throwError } from 'rxjs'
|
||||||
|
import { OutputTypeConfig } from 'src/app/data/paperless-config'
|
||||||
|
import { ConfigService } from 'src/app/services/config.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { FileComponent } from '../../common/input/file/file.component'
|
||||||
|
import { NumberComponent } from '../../common/input/number/number.component'
|
||||||
|
import { SelectComponent } from '../../common/input/select/select.component'
|
||||||
|
import { SwitchComponent } from '../../common/input/switch/switch.component'
|
||||||
|
import { TextComponent } from '../../common/input/text/text.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { ConfigComponent } from './config.component'
|
||||||
|
|
||||||
describe('ConfigComponent', () => {
|
describe('ConfigComponent', () => {
|
||||||
let component: ConfigComponent
|
let component: ConfigComponent
|
||||||
@ -29,15 +29,6 @@ describe('ConfigComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
ConfigComponent,
|
|
||||||
TextComponent,
|
|
||||||
SelectComponent,
|
|
||||||
NumberComponent,
|
|
||||||
SwitchComponent,
|
|
||||||
FileComponent,
|
|
||||||
PageHeaderComponent,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
NgbModule,
|
NgbModule,
|
||||||
@ -45,6 +36,13 @@ describe('ConfigComponent', () => {
|
|||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
ConfigComponent,
|
||||||
|
TextComponent,
|
||||||
|
SelectComponent,
|
||||||
|
NumberComponent,
|
||||||
|
SwitchComponent,
|
||||||
|
FileComponent,
|
||||||
|
PageHeaderComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
@ -1,33 +1,60 @@
|
|||||||
|
import { AsyncPipe } from '@angular/common'
|
||||||
import { Component, OnDestroy, OnInit } from '@angular/core'
|
import { Component, OnDestroy, OnInit } from '@angular/core'
|
||||||
import { AbstractControl, FormControl, FormGroup } from '@angular/forms'
|
import {
|
||||||
|
AbstractControl,
|
||||||
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
} from '@angular/forms'
|
||||||
|
import { NgbNavModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { DirtyComponent, dirtyCheck } from '@ngneat/dirty-check-forms'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import {
|
import {
|
||||||
BehaviorSubject,
|
BehaviorSubject,
|
||||||
Observable,
|
Observable,
|
||||||
Subject,
|
|
||||||
Subscription,
|
Subscription,
|
||||||
first,
|
first,
|
||||||
takeUntil,
|
takeUntil,
|
||||||
} from 'rxjs'
|
} from 'rxjs'
|
||||||
import {
|
import {
|
||||||
PaperlessConfigOptions,
|
|
||||||
ConfigCategory,
|
ConfigCategory,
|
||||||
ConfigOption,
|
ConfigOption,
|
||||||
ConfigOptionType,
|
ConfigOptionType,
|
||||||
PaperlessConfig,
|
PaperlessConfig,
|
||||||
|
PaperlessConfigOptions,
|
||||||
} from 'src/app/data/paperless-config'
|
} from 'src/app/data/paperless-config'
|
||||||
import { ConfigService } from 'src/app/services/config.service'
|
import { ConfigService } from 'src/app/services/config.service'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
|
||||||
import { DirtyComponent, dirtyCheck } from '@ngneat/dirty-check-forms'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { FileComponent } from '../../common/input/file/file.component'
|
||||||
|
import { NumberComponent } from '../../common/input/number/number.component'
|
||||||
|
import { SelectComponent } from '../../common/input/select/select.component'
|
||||||
|
import { SwitchComponent } from '../../common/input/switch/switch.component'
|
||||||
|
import { TextComponent } from '../../common/input/text/text.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-config',
|
selector: 'pngx-config',
|
||||||
templateUrl: './config.component.html',
|
templateUrl: './config.component.html',
|
||||||
styleUrl: './config.component.scss',
|
styleUrl: './config.component.scss',
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
SelectComponent,
|
||||||
|
SwitchComponent,
|
||||||
|
TextComponent,
|
||||||
|
NumberComponent,
|
||||||
|
FileComponent,
|
||||||
|
AsyncPipe,
|
||||||
|
NgbNavModule,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class ConfigComponent
|
export class ConfigComponent
|
||||||
extends ComponentWithPermissions
|
extends LoadingComponentWithPermissions
|
||||||
implements OnInit, OnDestroy, DirtyComponent
|
implements OnInit, OnDestroy, DirtyComponent
|
||||||
{
|
{
|
||||||
public readonly ConfigOptionType = ConfigOptionType
|
public readonly ConfigOptionType = ConfigOptionType
|
||||||
@ -45,15 +72,11 @@ export class ConfigComponent
|
|||||||
return PaperlessConfigOptions.filter((o) => o.category === category)
|
return PaperlessConfigOptions.filter((o) => o.category === category)
|
||||||
}
|
}
|
||||||
|
|
||||||
public loading: boolean = false
|
|
||||||
|
|
||||||
initialConfig: PaperlessConfig
|
initialConfig: PaperlessConfig
|
||||||
store: BehaviorSubject<any>
|
store: BehaviorSubject<any>
|
||||||
storeSub: Subscription
|
storeSub: Subscription
|
||||||
isDirty$: Observable<boolean>
|
isDirty$: Observable<boolean>
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private configService: ConfigService,
|
private configService: ConfigService,
|
||||||
private toastService: ToastService,
|
private toastService: ToastService,
|
||||||
@ -67,7 +90,6 @@ export class ConfigComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.loading = true
|
|
||||||
this.configService
|
this.configService
|
||||||
.getConfig()
|
.getConfig()
|
||||||
.pipe(takeUntil(this.unsubscribeNotifier))
|
.pipe(takeUntil(this.unsubscribeNotifier))
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
info="Review the log files for the application and for email checking."
|
info="Review the log files for the application and for email checking."
|
||||||
i18n-info>
|
i18n-info>
|
||||||
<div class="form-check form-switch">
|
<div class="form-check form-switch">
|
||||||
<input class="form-check-input" type="checkbox" role="switch" id="autoRefreshSwitch" (click)="toggleAutoRefresh()" [attr.checked]="autoRefreshInterval">
|
<input class="form-check-input" type="checkbox" role="switch" [(ngModel)]="autoRefreshEnabled">
|
||||||
<label class="form-check-label" for="autoRefreshSwitch" i18n>Auto refresh</label>
|
<label class="form-check-label" for="autoRefreshSwitch" i18n>Auto refresh</label>
|
||||||
</div>
|
</div>
|
||||||
</pngx-page-header>
|
</pngx-page-header>
|
||||||
@ -17,7 +17,7 @@
|
|||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
}
|
}
|
||||||
@if (isLoading || !logFiles.length) {
|
@if (loading || !logFiles.length) {
|
||||||
<div class="ps-2 d-flex align-items-center">
|
<div class="ps-2 d-flex align-items-center">
|
||||||
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
||||||
@if (!logFiles.length) {
|
@if (!logFiles.length) {
|
||||||
@ -30,15 +30,13 @@
|
|||||||
<div [ngbNavOutlet]="nav" class="mt-2"></div>
|
<div [ngbNavOutlet]="nav" class="mt-2"></div>
|
||||||
|
|
||||||
<div class="bg-dark p-3 text-light font-monospace log-container" #logContainer>
|
<div class="bg-dark p-3 text-light font-monospace log-container" #logContainer>
|
||||||
@if (isLoading && logFiles.length) {
|
@if (loading && logFiles.length) {
|
||||||
<div>
|
<div>
|
||||||
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
||||||
<ng-container i18n>Loading...</ng-container>
|
<ng-container i18n>Loading...</ng-container>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
@for (log of logs; track log) {
|
@for (log of logs; track $index) {
|
||||||
<p
|
<p class="m-0 p-0 log-entry-{{getLogLevel(log)}}">{{log}}</p>
|
||||||
class="m-0 p-0 log-entry-{{getLogLevel(log)}}"
|
|
||||||
>{{log}}</p>
|
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,18 +1,13 @@
|
|||||||
import {
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
ComponentFixture,
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
TestBed,
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
fakeAsync,
|
import { BrowserModule, By } from '@angular/platform-browser'
|
||||||
tick,
|
import { NgbModule, NgbNavLink } from '@ng-bootstrap/ng-bootstrap'
|
||||||
} from '@angular/core/testing'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { of, throwError } from 'rxjs'
|
||||||
import { LogService } from 'src/app/services/rest/log.service'
|
import { LogService } from 'src/app/services/rest/log.service'
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
import { LogsComponent } from './logs.component'
|
import { LogsComponent } from './logs.component'
|
||||||
import { of, throwError } from 'rxjs'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { NgbModule, NgbNavLink } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { BrowserModule, By } from '@angular/platform-browser'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const paperless_logs = [
|
const paperless_logs = [
|
||||||
'[2023-05-29 03:05:01,224] [DEBUG] [paperless.tasks] Training data unchanged.',
|
'[2023-05-29 03:05:01,224] [DEBUG] [paperless.tasks] Training data unchanged.',
|
||||||
@ -37,11 +32,12 @@ describe('LogsComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [LogsComponent, PageHeaderComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
NgbModule,
|
NgbModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
LogsComponent,
|
||||||
|
PageHeaderComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
@ -90,8 +86,7 @@ describe('LogsComponent', () => {
|
|||||||
jest.advanceTimersByTime(6000)
|
jest.advanceTimersByTime(6000)
|
||||||
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
||||||
|
|
||||||
component.toggleAutoRefresh()
|
component.autoRefreshEnabled = false
|
||||||
expect(component.autoRefreshInterval).toBeNull()
|
|
||||||
jest.advanceTimersByTime(6000)
|
jest.advanceTimersByTime(6000)
|
||||||
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
||||||
})
|
})
|
||||||
|
@ -1,24 +1,39 @@
|
|||||||
import {
|
import {
|
||||||
|
ChangeDetectorRef,
|
||||||
Component,
|
Component,
|
||||||
ElementRef,
|
ElementRef,
|
||||||
|
OnDestroy,
|
||||||
OnInit,
|
OnInit,
|
||||||
ViewChild,
|
ViewChild,
|
||||||
OnDestroy,
|
|
||||||
ChangeDetectorRef,
|
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { Subject, takeUntil } from 'rxjs'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { NgbNavModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { filter, takeUntil, timer } from 'rxjs'
|
||||||
import { LogService } from 'src/app/services/rest/log.service'
|
import { LogService } from 'src/app/services/rest/log.service'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-logs',
|
selector: 'pngx-logs',
|
||||||
templateUrl: './logs.component.html',
|
templateUrl: './logs.component.html',
|
||||||
styleUrls: ['./logs.component.scss'],
|
styleUrls: ['./logs.component.scss'],
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
NgbNavModule,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class LogsComponent implements OnInit, OnDestroy {
|
export class LogsComponent
|
||||||
|
extends LoadingComponentWithPermissions
|
||||||
|
implements OnInit, OnDestroy
|
||||||
|
{
|
||||||
constructor(
|
constructor(
|
||||||
private logService: LogService,
|
private logService: LogService,
|
||||||
private changedetectorRef: ChangeDetectorRef
|
private changedetectorRef: ChangeDetectorRef
|
||||||
) {}
|
) {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
public logs: string[] = []
|
public logs: string[] = []
|
||||||
|
|
||||||
@ -26,50 +41,50 @@ export class LogsComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
public activeLog: string
|
public activeLog: string
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
public autoRefreshEnabled: boolean = true
|
||||||
|
|
||||||
public isLoading: boolean = false
|
|
||||||
|
|
||||||
public autoRefreshInterval: any
|
|
||||||
|
|
||||||
@ViewChild('logContainer') logContainer: ElementRef
|
@ViewChild('logContainer') logContainer: ElementRef
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.isLoading = true
|
|
||||||
this.logService
|
this.logService
|
||||||
.list()
|
.list()
|
||||||
.pipe(takeUntil(this.unsubscribeNotifier))
|
.pipe(takeUntil(this.unsubscribeNotifier))
|
||||||
.subscribe((result) => {
|
.subscribe((result) => {
|
||||||
this.logFiles = result
|
this.logFiles = result
|
||||||
this.isLoading = false
|
this.loading = false
|
||||||
if (this.logFiles.length > 0) {
|
if (this.logFiles.length > 0) {
|
||||||
this.activeLog = this.logFiles[0]
|
this.activeLog = this.logFiles[0]
|
||||||
this.reloadLogs()
|
this.reloadLogs()
|
||||||
}
|
}
|
||||||
this.toggleAutoRefresh()
|
timer(5000, 5000)
|
||||||
|
.pipe(
|
||||||
|
filter(() => this.autoRefreshEnabled),
|
||||||
|
takeUntil(this.unsubscribeNotifier)
|
||||||
|
)
|
||||||
|
.subscribe(() => {
|
||||||
|
this.reloadLogs()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy(): void {
|
ngOnDestroy(): void {
|
||||||
this.unsubscribeNotifier.next(true)
|
super.ngOnDestroy()
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
clearInterval(this.autoRefreshInterval)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
reloadLogs() {
|
reloadLogs() {
|
||||||
this.isLoading = true
|
this.loading = true
|
||||||
this.logService
|
this.logService
|
||||||
.get(this.activeLog)
|
.get(this.activeLog)
|
||||||
.pipe(takeUntil(this.unsubscribeNotifier))
|
.pipe(takeUntil(this.unsubscribeNotifier))
|
||||||
.subscribe({
|
.subscribe({
|
||||||
next: (result) => {
|
next: (result) => {
|
||||||
this.logs = result
|
this.logs = result
|
||||||
this.isLoading = false
|
this.loading = false
|
||||||
this.scrollToBottom()
|
this.scrollToBottom()
|
||||||
},
|
},
|
||||||
error: () => {
|
error: () => {
|
||||||
this.logs = []
|
this.logs = []
|
||||||
this.isLoading = false
|
this.loading = false
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -96,15 +111,4 @@ export class LogsComponent implements OnInit, OnDestroy {
|
|||||||
behavior: 'auto',
|
behavior: 'auto',
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
toggleAutoRefresh(): void {
|
|
||||||
if (this.autoRefreshInterval) {
|
|
||||||
clearInterval(this.autoRefreshInterval)
|
|
||||||
this.autoRefreshInterval = null
|
|
||||||
} else {
|
|
||||||
this.autoRefreshInterval = setInterval(() => {
|
|
||||||
this.reloadLogs()
|
|
||||||
}, 5000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<pngx-page-header
|
<pngx-page-header
|
||||||
title="Settings"
|
title="Settings"
|
||||||
i18n-title
|
i18n-title
|
||||||
info="Options to customize appearance, notifications, saved views and more. Settings apply to the <strong>current user only</strong>."
|
info="Options to customize appearance, notifications and more. Settings apply to the <strong>current user only</strong>."
|
||||||
i18n-info
|
i18n-info
|
||||||
>
|
>
|
||||||
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
||||||
@ -39,193 +39,204 @@
|
|||||||
<li [ngbNavItem]="SettingsNavIDs.General">
|
<li [ngbNavItem]="SettingsNavIDs.General">
|
||||||
<a ngbNavLink i18n>General</a>
|
<a ngbNavLink i18n>General</a>
|
||||||
<ng-template ngbNavContent>
|
<ng-template ngbNavContent>
|
||||||
|
<div class="row">
|
||||||
<h4 i18n>Appearance</h4>
|
<div class="col-xl-6 pe-xl-5">
|
||||||
|
<h4 i18n>Appearance</h4>
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
<span i18n>Display language</span>
|
<span i18n>Display language</span>
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<select class="form-select" formControlName="displayLanguage">
|
|
||||||
@for (lang of displayLanguageOptions; track lang) {
|
|
||||||
<option [ngValue]="lang.code">{{lang.name}}@if (lang.code && currentLocale !== 'en-US') {
|
|
||||||
<span> - {{lang.englishName}}</span>
|
|
||||||
}</option>
|
|
||||||
}
|
|
||||||
</select>
|
|
||||||
|
|
||||||
@if (displayLanguageIsDirty) {
|
|
||||||
<small class="form-text text-primary" i18n>You need to reload the page after applying a new language.</small>
|
|
||||||
}
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Date display</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<select class="form-select" formControlName="dateLocale">
|
|
||||||
@for (lang of dateLocaleOptions; track lang) {
|
|
||||||
<option [ngValue]="lang.code">{{lang.name}}@if (lang.code) {
|
|
||||||
<span> - {{today | customDate:'shortDate':null:lang.code}}</span>
|
|
||||||
}</option>
|
|
||||||
}
|
|
||||||
</select>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Date format</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<div class="form-check">
|
|
||||||
<input type="radio" id="dateFormatShort" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="shortDate">
|
|
||||||
<label class="form-check-label" for="dateFormatShort" i18n>Short: {{today | customDate:'shortDate':null:computedDateLocale}}</label>
|
|
||||||
</div>
|
|
||||||
<div class="form-check">
|
|
||||||
<input type="radio" id="dateFormatMedium" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="mediumDate">
|
|
||||||
<label class="form-check-label" for="dateFormatMedium" i18n>Medium: {{today | customDate:'mediumDate':null:computedDateLocale}}</label>
|
|
||||||
</div>
|
|
||||||
<div class="form-check">
|
|
||||||
<input type="radio" id="dateFormatLong" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="longDate">
|
|
||||||
<label class="form-check-label" for="dateFormatLong" i18n>Long: {{today | customDate:'longDate':null:computedDateLocale}}</label>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Items per page</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<select class="form-select" formControlName="documentListItemPerPage">
|
|
||||||
<option [ngValue]="10">10</option>
|
|
||||||
<option [ngValue]="25">25</option>
|
|
||||||
<option [ngValue]="50">50</option>
|
|
||||||
<option [ngValue]="100">100</option>
|
|
||||||
</select>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Document editor</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<pngx-input-check i18n-title title="Use PDF viewer provided by the browser" i18n-hint hint="This is usually faster for displaying large PDF documents, but it might not work on some browsers." formControlName="useNativePdfViewer"></pngx-input-check>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Sidebar</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
|
|
||||||
<pngx-input-check i18n-title title="Use 'slim' sidebar (icons only)" formControlName="slimSidebarEnabled"></pngx-input-check>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Dark mode</span>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
<pngx-input-check i18n-title title="Use system settings" formControlName="darkModeUseSystem"></pngx-input-check>
|
|
||||||
<pngx-input-check [hidden]="settingsForm.value.darkModeUseSystem" i18n-title title="Enable dark mode" formControlName="darkModeEnabled"></pngx-input-check>
|
|
||||||
<pngx-input-check i18n-title title="Invert thumbnails in dark mode" formControlName="darkModeInvertThumbs"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="col-md-3 col-form-label pt-0">
|
|
||||||
<span i18n>Theme Color</span>
|
|
||||||
</div>
|
|
||||||
<div class="col col-md-3">
|
|
||||||
<pngx-input-color i18n-title formControlName="themeColor" [error]="error?.color"></pngx-input-color>
|
|
||||||
</div>
|
|
||||||
<div class="col-2">
|
|
||||||
<button class="btn btn-link btn-sm pt-2 ps-0" [disabled]="!this.settingsForm.get('themeColor').value" (click)="clearThemeColor()">
|
|
||||||
<i-bs width="1em" height="1em" name="x"></i-bs><ng-container i18n>Reset</ng-container>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 class="mt-4" id="update-checking" i18n>Update checking</h4>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<p i18n>
|
|
||||||
Update checking works by pinging the public <a href="https://api.github.com/repos/paperless-ngx/paperless-ngx/releases/latest" target="_blank" rel="noopener noreferrer">GitHub API</a> for the latest release to determine whether a new version is available.<br/>
|
|
||||||
Actual updating of the app must still be performed manually.
|
|
||||||
</p>
|
|
||||||
<p i18n>
|
|
||||||
<em>No tracking data is collected by the app in any way.</em>
|
|
||||||
</p>
|
|
||||||
<pngx-input-check i18n-title title="Enable update checking" formControlName="updateCheckingEnabled"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 class="mt-4" i18n>Document editing</h4>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<pngx-input-check i18n-title title="Automatically remove inbox tag(s) on save" formControlName="documentEditingRemoveInboxTags"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 class="mt-4" i18n>Bulk editing</h4>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<pngx-input-check i18n-title title="Show confirmation dialogs" formControlName="bulkEditConfirmationDialogs"></pngx-input-check>
|
|
||||||
<pngx-input-check i18n-title title="Apply on close" formControlName="bulkEditApplyOnClose"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 class="mt-4" i18n>Global search</h4>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<pngx-input-check i18n-title title="Do not include advanced search results" formControlName="searchDbOnly"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-md-2 col-form-label pt-0">
|
|
||||||
<span i18n>Full search links to</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="col">
|
<div class="col">
|
||||||
<select class="form-select" formControlName="searchLink">
|
|
||||||
<option [ngValue]="GlobalSearchType.TITLE_CONTENT" i18n>Title and content search</option>
|
<select class="form-select" formControlName="displayLanguage">
|
||||||
<option [ngValue]="GlobalSearchType.ADVANCED" i18n>Advanced search</option>
|
@for (lang of displayLanguageOptions; track lang) {
|
||||||
|
<option [ngValue]="lang.code">{{lang.name}}@if (lang.code && currentLocale !== 'en-US') {
|
||||||
|
<span> - {{lang.englishName}}</span>
|
||||||
|
}</option>
|
||||||
|
}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
|
@if (displayLanguageIsDirty) {
|
||||||
|
<small class="form-text text-primary" i18n>You need to reload the page after applying a new language.</small>
|
||||||
|
}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Date display</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<select class="form-select" formControlName="dateLocale">
|
||||||
|
@for (lang of dateLocaleOptions; track lang) {
|
||||||
|
<option [ngValue]="lang.code">{{lang.name}}@if (lang.code) {
|
||||||
|
<span> - {{today | customDate:'shortDate':null:lang.code}}</span>
|
||||||
|
}</option>
|
||||||
|
}
|
||||||
|
</select>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Date format</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<div class="form-check">
|
||||||
|
<input type="radio" id="dateFormatShort" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="shortDate">
|
||||||
|
<label class="form-check-label" for="dateFormatShort" i18n>Short: {{today | customDate:'shortDate':null:computedDateLocale}}</label>
|
||||||
|
</div>
|
||||||
|
<div class="form-check">
|
||||||
|
<input type="radio" id="dateFormatMedium" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="mediumDate">
|
||||||
|
<label class="form-check-label" for="dateFormatMedium" i18n>Medium: {{today | customDate:'mediumDate':null:computedDateLocale}}</label>
|
||||||
|
</div>
|
||||||
|
<div class="form-check">
|
||||||
|
<input type="radio" id="dateFormatLong" name="dateFormat" class="form-check-input" formControlName="dateFormat" value="longDate">
|
||||||
|
<label class="form-check-label" for="dateFormatLong" i18n>Long: {{today | customDate:'longDate':null:computedDateLocale}}</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Items per page</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<select class="form-select" formControlName="documentListItemPerPage">
|
||||||
|
<option [ngValue]="10">10</option>
|
||||||
|
<option [ngValue]="25">25</option>
|
||||||
|
<option [ngValue]="50">50</option>
|
||||||
|
<option [ngValue]="100">100</option>
|
||||||
|
</select>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Sidebar</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<pngx-input-check i18n-title title="Use 'slim' sidebar (icons only)" formControlName="slimSidebarEnabled"></pngx-input-check>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Dark mode</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Use system settings" formControlName="darkModeUseSystem"></pngx-input-check>
|
||||||
|
<pngx-input-check [hidden]="settingsForm.value.darkModeUseSystem" i18n-title title="Enable dark mode" formControlName="darkModeEnabled"></pngx-input-check>
|
||||||
|
<pngx-input-check i18n-title title="Invert thumbnails in dark mode" formControlName="darkModeInvertThumbs"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Theme Color</span>
|
||||||
|
</div>
|
||||||
|
<div class="col col-md-4">
|
||||||
|
<pngx-input-color i18n-title formControlName="themeColor" [error]="error?.color"></pngx-input-color>
|
||||||
|
</div>
|
||||||
|
<div class="col-2">
|
||||||
|
<button class="btn btn-link btn-sm pt-2 ps-0" [disabled]="!this.settingsForm.get('themeColor').value" (click)="clearThemeColor()">
|
||||||
|
<i-bs width="1em" height="1em" name="x"></i-bs><ng-container i18n>Reset</ng-container>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h4 class="mt-4" i18n>Document editing</h4>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Use PDF viewer provided by the browser" i18n-hint hint="This is usually faster for displaying large PDF documents, but it might not work on some browsers." formControlName="useNativePdfViewer"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Automatically remove inbox tag(s) on save" formControlName="documentEditingRemoveInboxTags"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Show document thumbnail during loading" formControlName="documentEditingOverlayThumbnail"></pngx-input-check>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
<div class="col-xl-6 ps-xl-5">
|
||||||
|
<h4 class="mt-4 mt-md-0" id="update-checking" i18n>Update checking</h4>
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col d-flex flex-row align-items-start">
|
||||||
|
<pngx-input-check i18n-title title="Enable update checking" formControlName="updateCheckingEnabled"></pngx-input-check>
|
||||||
|
<button class="btn btn-sm btn-link text-muted me-auto p-0 ms-2" title="What's this?" i18n-title type="button" triggers="mouseenter:mouseleave" [ngbPopover]="updatesPopover" [autoClose]="true">
|
||||||
|
<i-bs name="question-circle"></i-bs>
|
||||||
|
</button>
|
||||||
|
<ng-template #updatesPopover>
|
||||||
|
<p i18n>
|
||||||
|
Update checking works by pinging the public GitHub API for the latest release to determine whether a new version is available. Actual updating of the app must still be performed manually.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<em i18n>No tracking data is collected by the app in any way.</em>
|
||||||
|
</p>
|
||||||
|
</ng-template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<h4 class="mt-4" i18n>Notes</h4>
|
<h4 class="mt-4" i18n>Bulk editing</h4>
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Show confirmation dialogs" formControlName="bulkEditConfirmationDialogs"></pngx-input-check>
|
||||||
|
<pngx-input-check i18n-title title="Apply on close" formControlName="bulkEditApplyOnClose"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row mb-3">
|
<h4 class="mt-4" i18n>Global search</h4>
|
||||||
<div class="offset-md-3 col">
|
<div class="row mb-3">
|
||||||
<pngx-input-check i18n-title title="Enable notes" formControlName="notesEnabled"></pngx-input-check>
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Do not include advanced search results" formControlName="searchDbOnly"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-md-3 col-form-label pt-0">
|
||||||
|
<span i18n>Full search links to</span>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
<select class="form-select" formControlName="searchLink">
|
||||||
|
<option [ngValue]="GlobalSearchType.TITLE_CONTENT" i18n>Title and content search</option>
|
||||||
|
<option [ngValue]="GlobalSearchType.ADVANCED" i18n>Advanced search</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h4 class="mt-4" i18n>Saved Views</h4>
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Show warning when closing saved views with unsaved changes" formControlName="savedViewsWarnOnUnsavedChange"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h4 class="mt-4" i18n>Notes</h4>
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-check i18n-title title="Enable notes" formControlName="notesEnabled"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -239,7 +250,7 @@
|
|||||||
<h4 i18n>Default Permissions</h4>
|
<h4 i18n>Default Permissions</h4>
|
||||||
|
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<div class="offset-md-3 col">
|
<div class="col">
|
||||||
<p i18n>
|
<p i18n>
|
||||||
Settings apply to this user account for objects (Tags, Mail Rules, etc.) created via the web UI
|
Settings apply to this user account for objects (Tags, Mail Rules, etc.) created via the web UI
|
||||||
</p>
|
</p>
|
||||||
@ -321,7 +332,7 @@
|
|||||||
<h4 i18n>Document processing</h4>
|
<h4 i18n>Document processing</h4>
|
||||||
|
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<div class="offset-md-3 col">
|
<div class="col">
|
||||||
<pngx-input-check i18n-title title="Show notifications when new documents are detected" formControlName="notificationsConsumerNewDocument"></pngx-input-check>
|
<pngx-input-check i18n-title title="Show notifications when new documents are detected" formControlName="notificationsConsumerNewDocument"></pngx-input-check>
|
||||||
<pngx-input-check i18n-title title="Show notifications when document processing completes successfully" formControlName="notificationsConsumerSuccess"></pngx-input-check>
|
<pngx-input-check i18n-title title="Show notifications when document processing completes successfully" formControlName="notificationsConsumerSuccess"></pngx-input-check>
|
||||||
<pngx-input-check i18n-title title="Show notifications when document processing fails" formControlName="notificationsConsumerFailed"></pngx-input-check>
|
<pngx-input-check i18n-title title="Show notifications when document processing fails" formControlName="notificationsConsumerFailed"></pngx-input-check>
|
||||||
@ -331,87 +342,6 @@
|
|||||||
|
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
<li [ngbNavItem]="SettingsNavIDs.SavedViews" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.SavedView }">
|
|
||||||
<a ngbNavLink i18n>Saved views</a>
|
|
||||||
<ng-template ngbNavContent>
|
|
||||||
|
|
||||||
<h4 i18n>Settings</h4>
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<pngx-input-check i18n-title title="Show warning when closing saved views with unsaved changes" formControlName="savedViewsWarnOnUnsavedChange"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 i18n>Views</h4>
|
|
||||||
<ul class="list-group" formGroupName="savedViews">
|
|
||||||
|
|
||||||
@for (view of savedViews; track view) {
|
|
||||||
<li class="list-group-item py-3">
|
|
||||||
<div [formGroupName]="view.id">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col">
|
|
||||||
<pngx-input-text title="Name" formControlName="name"></pngx-input-text>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
<div class="form-check form-switch mt-3">
|
|
||||||
<input type="checkbox" class="form-check-input" id="show_on_dashboard_{{view.id}}" formControlName="show_on_dashboard">
|
|
||||||
<label class="form-check-label" for="show_on_dashboard_{{view.id}}" i18n>Show on dashboard</label>
|
|
||||||
</div>
|
|
||||||
<div class="form-check form-switch">
|
|
||||||
<input type="checkbox" class="form-check-input" id="show_in_sidebar_{{view.id}}" formControlName="show_in_sidebar">
|
|
||||||
<label class="form-check-label" for="show_in_sidebar_{{view.id}}" i18n>Show in sidebar</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="col-auto">
|
|
||||||
<label class="form-label" for="name_{{view.id}}" i18n>Actions</label>
|
|
||||||
<pngx-confirm-button
|
|
||||||
label="Delete"
|
|
||||||
i18n-label
|
|
||||||
(confirm)="deleteSavedView(view)"
|
|
||||||
*pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.SavedView }"
|
|
||||||
buttonClasses="btn-sm btn-outline-danger form-control"
|
|
||||||
iconName="trash">
|
|
||||||
</pngx-confirm-button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="row">
|
|
||||||
<div class="col">
|
|
||||||
<pngx-input-number i18n-title title="Documents page size" [showAdd]="false" formControlName="page_size"></pngx-input-number>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
<label class="form-label" for="display_mode_{{view.id}}" i18n>Display as</label>
|
|
||||||
<select class="form-select" formControlName="display_mode">
|
|
||||||
<option [ngValue]="DisplayMode.TABLE" i18n>Table</option>
|
|
||||||
<option [ngValue]="DisplayMode.SMALL_CARDS" i18n>Small Cards</option>
|
|
||||||
<option [ngValue]="DisplayMode.LARGE_CARDS" i18n>Large Cards</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
@if (displayFields) {
|
|
||||||
<pngx-input-drag-drop-select i18n-title title="Show" i18n-emptyText emptyText="Default" [items]="displayFields" formControlName="display_fields"></pngx-input-drag-drop-select>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
@if (savedViews && savedViews.length === 0) {
|
|
||||||
<li class="list-group-item">
|
|
||||||
<div i18n>No saved views defined.</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
@if (!savedViews) {
|
|
||||||
<li class="list-group-item">
|
|
||||||
<div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div>
|
|
||||||
<div class="visually-hidden" i18n>Loading...</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
</ng-template>
|
|
||||||
</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div>
|
<div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div>
|
||||||
|
@ -1,35 +1,45 @@
|
|||||||
import { ViewportScroller, DatePipe } from '@angular/common'
|
import { DragDropModule } from '@angular/cdk/drag-drop'
|
||||||
|
import { DatePipe, ViewportScroller } from '@angular/common'
|
||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { By } from '@angular/platform-browser'
|
import { By } from '@angular/platform-browser'
|
||||||
import { Router, ActivatedRoute, convertToParamMap } from '@angular/router'
|
import { ActivatedRoute, Router, convertToParamMap } from '@angular/router'
|
||||||
import { RouterTestingModule } from '@angular/router/testing'
|
import { RouterTestingModule } from '@angular/router/testing'
|
||||||
import {
|
import {
|
||||||
NgbModule,
|
|
||||||
NgbAlertModule,
|
NgbAlertModule,
|
||||||
NgbNavLink,
|
|
||||||
NgbModal,
|
NgbModal,
|
||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
|
NgbModule,
|
||||||
|
NgbNavLink,
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { of, throwError } from 'rxjs'
|
import { of, throwError } from 'rxjs'
|
||||||
import { routes } from 'src/app/app-routing.module'
|
import { routes } from 'src/app/app-routing.module'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
import {
|
||||||
|
InstallType,
|
||||||
|
SystemStatus,
|
||||||
|
SystemStatusItemStatus,
|
||||||
|
} from 'src/app/data/system-status'
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
|
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { GroupService } from 'src/app/services/rest/group.service'
|
import { GroupService } from 'src/app/services/rest/group.service'
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ToastService, Toast } from 'src/app/services/toast.service'
|
import { SystemStatusService } from 'src/app/services/system-status.service'
|
||||||
|
import { Toast, ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { ConfirmButtonComponent } from '../../common/confirm-button/confirm-button.component'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { CheckComponent } from '../../common/input/check/check.component'
|
import { CheckComponent } from '../../common/input/check/check.component'
|
||||||
import { ColorComponent } from '../../common/input/color/color.component'
|
import { ColorComponent } from '../../common/input/color/color.component'
|
||||||
|
import { DragDropSelectComponent } from '../../common/input/drag-drop-select/drag-drop-select.component'
|
||||||
import { NumberComponent } from '../../common/input/number/number.component'
|
import { NumberComponent } from '../../common/input/number/number.component'
|
||||||
import { PermissionsGroupComponent } from '../../common/input/permissions/permissions-group/permissions-group.component'
|
import { PermissionsGroupComponent } from '../../common/input/permissions/permissions-group/permissions-group.component'
|
||||||
import { PermissionsUserComponent } from '../../common/input/permissions/permissions-user/permissions-user.component'
|
import { PermissionsUserComponent } from '../../common/input/permissions/permissions-user/permissions-user.component'
|
||||||
@ -37,25 +47,9 @@ import { SelectComponent } from '../../common/input/select/select.component'
|
|||||||
import { TagsComponent } from '../../common/input/tags/tags.component'
|
import { TagsComponent } from '../../common/input/tags/tags.component'
|
||||||
import { TextComponent } from '../../common/input/text/text.component'
|
import { TextComponent } from '../../common/input/text/text.component'
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
import { SettingsComponent } from './settings.component'
|
|
||||||
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { ConfirmButtonComponent } from '../../common/confirm-button/confirm-button.component'
|
|
||||||
import { SystemStatusDialogComponent } from '../../common/system-status-dialog/system-status-dialog.component'
|
import { SystemStatusDialogComponent } from '../../common/system-status-dialog/system-status-dialog.component'
|
||||||
import { SystemStatusService } from 'src/app/services/system-status.service'
|
import { SettingsComponent } from './settings.component'
|
||||||
import {
|
|
||||||
SystemStatus,
|
|
||||||
InstallType,
|
|
||||||
SystemStatusItemStatus,
|
|
||||||
} from 'src/app/data/system-status'
|
|
||||||
import { DragDropSelectComponent } from '../../common/input/drag-drop-select/drag-drop-select.component'
|
|
||||||
import { DragDropModule } from '@angular/cdk/drag-drop'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const savedViews = [
|
|
||||||
{ id: 1, name: 'view1', show_in_sidebar: true, show_on_dashboard: true },
|
|
||||||
{ id: 2, name: 'view2', show_in_sidebar: false, show_on_dashboard: false },
|
|
||||||
]
|
|
||||||
const users = [
|
const users = [
|
||||||
{ id: 1, username: 'user1', is_superuser: false },
|
{ id: 1, username: 'user1', is_superuser: false },
|
||||||
{ id: 2, username: 'user2', is_superuser: false },
|
{ id: 2, username: 'user2', is_superuser: false },
|
||||||
@ -70,7 +64,6 @@ describe('SettingsComponent', () => {
|
|||||||
let fixture: ComponentFixture<SettingsComponent>
|
let fixture: ComponentFixture<SettingsComponent>
|
||||||
let router: Router
|
let router: Router
|
||||||
let settingsService: SettingsService
|
let settingsService: SettingsService
|
||||||
let savedViewService: SavedViewService
|
|
||||||
let activatedRoute: ActivatedRoute
|
let activatedRoute: ActivatedRoute
|
||||||
let viewportScroller: ViewportScroller
|
let viewportScroller: ViewportScroller
|
||||||
let toastService: ToastService
|
let toastService: ToastService
|
||||||
@ -82,7 +75,16 @@ describe('SettingsComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
imports: [
|
||||||
|
NgbModule,
|
||||||
|
RouterTestingModule.withRoutes(routes),
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgbAlertModule,
|
||||||
|
NgSelectModule,
|
||||||
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
NgbModalModule,
|
||||||
|
DragDropModule,
|
||||||
SettingsComponent,
|
SettingsComponent,
|
||||||
PageHeaderComponent,
|
PageHeaderComponent,
|
||||||
IfPermissionsDirective,
|
IfPermissionsDirective,
|
||||||
@ -101,17 +103,6 @@ describe('SettingsComponent', () => {
|
|||||||
ConfirmButtonComponent,
|
ConfirmButtonComponent,
|
||||||
DragDropSelectComponent,
|
DragDropSelectComponent,
|
||||||
],
|
],
|
||||||
imports: [
|
|
||||||
NgbModule,
|
|
||||||
RouterTestingModule.withRoutes(routes),
|
|
||||||
FormsModule,
|
|
||||||
ReactiveFormsModule,
|
|
||||||
NgbAlertModule,
|
|
||||||
NgSelectModule,
|
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
|
||||||
NgbModalModule,
|
|
||||||
DragDropModule,
|
|
||||||
],
|
|
||||||
providers: [
|
providers: [
|
||||||
CustomDatePipe,
|
CustomDatePipe,
|
||||||
DatePipe,
|
DatePipe,
|
||||||
@ -139,7 +130,6 @@ describe('SettingsComponent', () => {
|
|||||||
.spyOn(permissionsService, 'currentUserOwnsObject')
|
.spyOn(permissionsService, 'currentUserOwnsObject')
|
||||||
.mockReturnValue(true)
|
.mockReturnValue(true)
|
||||||
groupService = TestBed.inject(GroupService)
|
groupService = TestBed.inject(GroupService)
|
||||||
savedViewService = TestBed.inject(SavedViewService)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
function completeSetup(excludeService = null) {
|
function completeSetup(excludeService = null) {
|
||||||
@ -161,15 +151,6 @@ describe('SettingsComponent', () => {
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
if (excludeService !== savedViewService) {
|
|
||||||
jest.spyOn(savedViewService, 'listAll').mockReturnValue(
|
|
||||||
of({
|
|
||||||
all: savedViews.map((v) => v.id),
|
|
||||||
count: savedViews.length,
|
|
||||||
results: (savedViews as SavedView[]).concat([]),
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fixture = TestBed.createComponent(SettingsComponent)
|
fixture = TestBed.createComponent(SettingsComponent)
|
||||||
component = fixture.componentInstance
|
component = fixture.componentInstance
|
||||||
@ -184,8 +165,6 @@ describe('SettingsComponent', () => {
|
|||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'permissions'])
|
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'permissions'])
|
||||||
tabButtons[2].nativeElement.dispatchEvent(new MouseEvent('click'))
|
tabButtons[2].nativeElement.dispatchEvent(new MouseEvent('click'))
|
||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'notifications'])
|
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'notifications'])
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'savedviews'])
|
|
||||||
|
|
||||||
const initSpy = jest.spyOn(component, 'initialize')
|
const initSpy = jest.spyOn(component, 'initialize')
|
||||||
component.isDirty = true // mock dirty
|
component.isDirty = true // mock dirty
|
||||||
@ -213,90 +192,8 @@ describe('SettingsComponent', () => {
|
|||||||
expect(scrollSpy).toHaveBeenCalledWith('#notifications')
|
expect(scrollSpy).toHaveBeenCalledWith('#notifications')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should enable organizing of sidebar saved views even on direct navigation', () => {
|
|
||||||
completeSetup()
|
|
||||||
jest
|
|
||||||
.spyOn(activatedRoute, 'paramMap', 'get')
|
|
||||||
.mockReturnValue(of(convertToParamMap({ section: 'savedviews' })))
|
|
||||||
activatedRoute.snapshot.fragment = '#savedviews'
|
|
||||||
component.ngOnInit()
|
|
||||||
expect(component.activeNavID).toEqual(4) // Saved Views
|
|
||||||
component.ngAfterViewInit()
|
|
||||||
expect(settingsService.organizingSidebarSavedViews).toBeTruthy()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should support save saved views, show error', () => {
|
|
||||||
completeSetup()
|
|
||||||
|
|
||||||
const tabButtons = fixture.debugElement.queryAll(By.directive(NgbNavLink))
|
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
|
||||||
const toastSpy = jest.spyOn(toastService, 'show')
|
|
||||||
const savedViewPatchSpy = jest.spyOn(savedViewService, 'patchMany')
|
|
||||||
|
|
||||||
const toggle = fixture.debugElement.query(
|
|
||||||
By.css('.form-check.form-switch input')
|
|
||||||
)
|
|
||||||
toggle.nativeElement.checked = true
|
|
||||||
toggle.nativeElement.dispatchEvent(new Event('change'))
|
|
||||||
|
|
||||||
// saved views error first
|
|
||||||
savedViewPatchSpy.mockReturnValueOnce(
|
|
||||||
throwError(() => new Error('unable to save saved views'))
|
|
||||||
)
|
|
||||||
component.saveSettings()
|
|
||||||
expect(toastErrorSpy).toHaveBeenCalled()
|
|
||||||
expect(savedViewPatchSpy).toHaveBeenCalled()
|
|
||||||
toastSpy.mockClear()
|
|
||||||
toastErrorSpy.mockClear()
|
|
||||||
savedViewPatchSpy.mockClear()
|
|
||||||
|
|
||||||
// succeed saved views
|
|
||||||
savedViewPatchSpy.mockReturnValueOnce(of(savedViews as SavedView[]))
|
|
||||||
component.saveSettings()
|
|
||||||
expect(toastErrorSpy).not.toHaveBeenCalled()
|
|
||||||
expect(savedViewPatchSpy).toHaveBeenCalled()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should update only patch saved views that have changed', () => {
|
|
||||||
completeSetup()
|
|
||||||
|
|
||||||
const tabButtons = fixture.debugElement.queryAll(By.directive(NgbNavLink))
|
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
const patchSpy = jest.spyOn(savedViewService, 'patchMany')
|
|
||||||
component.saveSettings()
|
|
||||||
expect(patchSpy).not.toHaveBeenCalled()
|
|
||||||
|
|
||||||
const view = savedViews[0]
|
|
||||||
const toggle = fixture.debugElement.query(
|
|
||||||
By.css('.form-check.form-switch input')
|
|
||||||
)
|
|
||||||
toggle.nativeElement.checked = true
|
|
||||||
toggle.nativeElement.dispatchEvent(new Event('change'))
|
|
||||||
// register change
|
|
||||||
component.savedViewGroup.get(view.id.toString()).value[
|
|
||||||
'show_on_dashboard'
|
|
||||||
] = !view.show_on_dashboard
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
component.saveSettings()
|
|
||||||
expect(patchSpy).toHaveBeenCalledWith([
|
|
||||||
{
|
|
||||||
id: view.id,
|
|
||||||
name: view.name,
|
|
||||||
show_in_sidebar: view.show_in_sidebar,
|
|
||||||
show_on_dashboard: !view.show_on_dashboard,
|
|
||||||
},
|
|
||||||
])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should support save local settings updating appearance settings and calling API, show error', () => {
|
it('should support save local settings updating appearance settings and calling API, show error', () => {
|
||||||
completeSetup()
|
completeSetup()
|
||||||
jest.spyOn(savedViewService, 'patchMany').mockReturnValue(of([]))
|
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
||||||
const toastSpy = jest.spyOn(toastService, 'show')
|
const toastSpy = jest.spyOn(toastService, 'show')
|
||||||
const storeSpy = jest.spyOn(settingsService, 'storeSettings')
|
const storeSpy = jest.spyOn(settingsService, 'storeSettings')
|
||||||
@ -315,7 +212,7 @@ describe('SettingsComponent', () => {
|
|||||||
expect(toastErrorSpy).toHaveBeenCalled()
|
expect(toastErrorSpy).toHaveBeenCalled()
|
||||||
expect(storeSpy).toHaveBeenCalled()
|
expect(storeSpy).toHaveBeenCalled()
|
||||||
expect(appearanceSettingsSpy).not.toHaveBeenCalled()
|
expect(appearanceSettingsSpy).not.toHaveBeenCalled()
|
||||||
expect(setSpy).toHaveBeenCalledTimes(27)
|
expect(setSpy).toHaveBeenCalledTimes(28)
|
||||||
|
|
||||||
// succeed
|
// succeed
|
||||||
storeSpy.mockReturnValueOnce(of(true))
|
storeSpy.mockReturnValueOnce(of(true))
|
||||||
@ -326,7 +223,6 @@ describe('SettingsComponent', () => {
|
|||||||
|
|
||||||
it('should offer reload if settings changes require', () => {
|
it('should offer reload if settings changes require', () => {
|
||||||
completeSetup()
|
completeSetup()
|
||||||
jest.spyOn(savedViewService, 'patchMany').mockReturnValue(of([]))
|
|
||||||
let toast: Toast
|
let toast: Toast
|
||||||
toastService.getToasts().subscribe((t) => (toast = t[0]))
|
toastService.getToasts().subscribe((t) => (toast = t[0]))
|
||||||
component.initialize(true) // reset
|
component.initialize(true) // reset
|
||||||
@ -361,18 +257,6 @@ describe('SettingsComponent', () => {
|
|||||||
component.clearThemeColor()
|
component.clearThemeColor()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should support delete saved view', () => {
|
|
||||||
completeSetup()
|
|
||||||
const toastSpy = jest.spyOn(toastService, 'showInfo')
|
|
||||||
const deleteSpy = jest.spyOn(savedViewService, 'delete')
|
|
||||||
deleteSpy.mockReturnValue(of(true))
|
|
||||||
component.deleteSavedView(savedViews[0] as SavedView)
|
|
||||||
expect(deleteSpy).toHaveBeenCalled()
|
|
||||||
expect(toastSpy).toHaveBeenCalledWith(
|
|
||||||
`Saved view "${savedViews[0].name}" deleted.`
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should show errors on load if load users failure', () => {
|
it('should show errors on load if load users failure', () => {
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
||||||
jest
|
jest
|
||||||
|
@ -1,56 +1,69 @@
|
|||||||
import { ViewportScroller } from '@angular/common'
|
import { AsyncPipe, ViewportScroller } from '@angular/common'
|
||||||
import {
|
import {
|
||||||
Component,
|
|
||||||
OnInit,
|
|
||||||
AfterViewInit,
|
AfterViewInit,
|
||||||
OnDestroy,
|
Component,
|
||||||
Inject,
|
Inject,
|
||||||
LOCALE_ID,
|
LOCALE_ID,
|
||||||
|
OnDestroy,
|
||||||
|
OnInit,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { FormGroup, FormControl } from '@angular/forms'
|
import {
|
||||||
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
} from '@angular/forms'
|
||||||
import { ActivatedRoute, Router } from '@angular/router'
|
import { ActivatedRoute, Router } from '@angular/router'
|
||||||
import {
|
import {
|
||||||
NgbModal,
|
NgbModal,
|
||||||
NgbModalRef,
|
NgbModalRef,
|
||||||
NgbNavChangeEvent,
|
NgbNavChangeEvent,
|
||||||
|
NgbNavModule,
|
||||||
|
NgbPopoverModule,
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { DirtyComponent, dirtyCheck } from '@ngneat/dirty-check-forms'
|
import { DirtyComponent, dirtyCheck } from '@ngneat/dirty-check-forms'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import { TourService } from 'ngx-ui-tour-ng-bootstrap'
|
import { TourService } from 'ngx-ui-tour-ng-bootstrap'
|
||||||
import {
|
import {
|
||||||
BehaviorSubject,
|
BehaviorSubject,
|
||||||
Subscription,
|
|
||||||
Observable,
|
Observable,
|
||||||
Subject,
|
Subject,
|
||||||
|
Subscription,
|
||||||
first,
|
first,
|
||||||
takeUntil,
|
takeUntil,
|
||||||
tap,
|
tap,
|
||||||
} from 'rxjs'
|
} from 'rxjs'
|
||||||
import { Group } from 'src/app/data/group'
|
import { Group } from 'src/app/data/group'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
import {
|
||||||
|
SystemStatus,
|
||||||
|
SystemStatusItemStatus,
|
||||||
|
} from 'src/app/data/system-status'
|
||||||
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
import { User } from 'src/app/data/user'
|
import { User } from 'src/app/data/user'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
||||||
import {
|
import {
|
||||||
PermissionsService,
|
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
PermissionType,
|
PermissionType,
|
||||||
|
PermissionsService,
|
||||||
} from 'src/app/services/permissions.service'
|
} from 'src/app/services/permissions.service'
|
||||||
import { GroupService } from 'src/app/services/rest/group.service'
|
import { GroupService } from 'src/app/services/rest/group.service'
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import {
|
import {
|
||||||
SettingsService,
|
|
||||||
LanguageOption,
|
LanguageOption,
|
||||||
|
SettingsService,
|
||||||
} from 'src/app/services/settings.service'
|
} from 'src/app/services/settings.service'
|
||||||
import { ToastService, Toast } from 'src/app/services/toast.service'
|
|
||||||
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
|
||||||
import { SystemStatusDialogComponent } from '../../common/system-status-dialog/system-status-dialog.component'
|
|
||||||
import { SystemStatusService } from 'src/app/services/system-status.service'
|
import { SystemStatusService } from 'src/app/services/system-status.service'
|
||||||
import {
|
import { Toast, ToastService } from 'src/app/services/toast.service'
|
||||||
SystemStatusItemStatus,
|
import { CheckComponent } from '../../common/input/check/check.component'
|
||||||
SystemStatus,
|
import { ColorComponent } from '../../common/input/color/color.component'
|
||||||
} from 'src/app/data/system-status'
|
import { PermissionsGroupComponent } from '../../common/input/permissions/permissions-group/permissions-group.component'
|
||||||
import { DisplayMode } from 'src/app/data/document'
|
import { PermissionsUserComponent } from '../../common/input/permissions/permissions-user/permissions-user.component'
|
||||||
|
import { SelectComponent } from '../../common/input/select/select.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { SystemStatusDialogComponent } from '../../common/system-status-dialog/system-status-dialog.component'
|
||||||
|
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
||||||
|
|
||||||
enum SettingsNavIDs {
|
enum SettingsNavIDs {
|
||||||
General = 1,
|
General = 1,
|
||||||
@ -69,15 +82,28 @@ const systemDateFormat = {
|
|||||||
selector: 'pngx-settings',
|
selector: 'pngx-settings',
|
||||||
templateUrl: './settings.component.html',
|
templateUrl: './settings.component.html',
|
||||||
styleUrls: ['./settings.component.scss'],
|
styleUrls: ['./settings.component.scss'],
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
CheckComponent,
|
||||||
|
ColorComponent,
|
||||||
|
SelectComponent,
|
||||||
|
PermissionsGroupComponent,
|
||||||
|
PermissionsUserComponent,
|
||||||
|
CustomDatePipe,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
AsyncPipe,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgbNavModule,
|
||||||
|
NgbPopoverModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class SettingsComponent
|
export class SettingsComponent
|
||||||
extends ComponentWithPermissions
|
extends ComponentWithPermissions
|
||||||
implements OnInit, AfterViewInit, OnDestroy, DirtyComponent
|
implements OnInit, AfterViewInit, OnDestroy, DirtyComponent
|
||||||
{
|
{
|
||||||
activeNavID: number
|
activeNavID: number
|
||||||
DisplayMode = DisplayMode
|
|
||||||
|
|
||||||
savedViewGroup = new FormGroup({})
|
|
||||||
|
|
||||||
settingsForm = new FormGroup({
|
settingsForm = new FormGroup({
|
||||||
bulkEditConfirmationDialogs: new FormControl(null),
|
bulkEditConfirmationDialogs: new FormControl(null),
|
||||||
@ -88,7 +114,6 @@ export class SettingsComponent
|
|||||||
darkModeEnabled: new FormControl(null),
|
darkModeEnabled: new FormControl(null),
|
||||||
darkModeInvertThumbs: new FormControl(null),
|
darkModeInvertThumbs: new FormControl(null),
|
||||||
themeColor: new FormControl(null),
|
themeColor: new FormControl(null),
|
||||||
useNativePdfViewer: new FormControl(null),
|
|
||||||
displayLanguage: new FormControl(null),
|
displayLanguage: new FormControl(null),
|
||||||
dateLocale: new FormControl(null),
|
dateLocale: new FormControl(null),
|
||||||
dateFormat: new FormControl(null),
|
dateFormat: new FormControl(null),
|
||||||
@ -99,7 +124,9 @@ export class SettingsComponent
|
|||||||
defaultPermsViewGroups: new FormControl(null),
|
defaultPermsViewGroups: new FormControl(null),
|
||||||
defaultPermsEditUsers: new FormControl(null),
|
defaultPermsEditUsers: new FormControl(null),
|
||||||
defaultPermsEditGroups: new FormControl(null),
|
defaultPermsEditGroups: new FormControl(null),
|
||||||
|
useNativePdfViewer: new FormControl(null),
|
||||||
documentEditingRemoveInboxTags: new FormControl(null),
|
documentEditingRemoveInboxTags: new FormControl(null),
|
||||||
|
documentEditingOverlayThumbnail: new FormControl(null),
|
||||||
searchDbOnly: new FormControl(null),
|
searchDbOnly: new FormControl(null),
|
||||||
searchLink: new FormControl(null),
|
searchLink: new FormControl(null),
|
||||||
|
|
||||||
@ -109,14 +136,9 @@ export class SettingsComponent
|
|||||||
notificationsConsumerSuppressOnDashboard: new FormControl(null),
|
notificationsConsumerSuppressOnDashboard: new FormControl(null),
|
||||||
|
|
||||||
savedViewsWarnOnUnsavedChange: new FormControl(null),
|
savedViewsWarnOnUnsavedChange: new FormControl(null),
|
||||||
savedViews: this.savedViewGroup,
|
|
||||||
})
|
})
|
||||||
|
|
||||||
savedViews: SavedView[]
|
|
||||||
SettingsNavIDs = SettingsNavIDs
|
SettingsNavIDs = SettingsNavIDs
|
||||||
get displayFields() {
|
|
||||||
return this.settings.allDisplayFields
|
|
||||||
}
|
|
||||||
|
|
||||||
store: BehaviorSubject<any>
|
store: BehaviorSubject<any>
|
||||||
storeSub: Subscription
|
storeSub: Subscription
|
||||||
@ -151,7 +173,6 @@ export class SettingsComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
public savedViewService: SavedViewService,
|
|
||||||
private documentListViewService: DocumentListViewService,
|
private documentListViewService: DocumentListViewService,
|
||||||
private toastService: ToastService,
|
private toastService: ToastService,
|
||||||
private settings: SettingsService,
|
private settings: SettingsService,
|
||||||
@ -213,18 +234,6 @@ export class SettingsComponent
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
|
||||||
this.permissionsService.currentUserCan(
|
|
||||||
PermissionAction.View,
|
|
||||||
PermissionType.SavedView
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
this.savedViewService.listAll().subscribe((r) => {
|
|
||||||
this.savedViews = r.results
|
|
||||||
this.initialize(false)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
this.activatedRoute.paramMap.subscribe((paramMap) => {
|
this.activatedRoute.paramMap.subscribe((paramMap) => {
|
||||||
const section = paramMap.get('section')
|
const section = paramMap.get('section')
|
||||||
if (section) {
|
if (section) {
|
||||||
@ -234,9 +243,6 @@ export class SettingsComponent
|
|||||||
if (navIDKey) {
|
if (navIDKey) {
|
||||||
this.activeNavID = SettingsNavIDs[navIDKey]
|
this.activeNavID = SettingsNavIDs[navIDKey]
|
||||||
}
|
}
|
||||||
if (this.activeNavID === SettingsNavIDs.SavedViews) {
|
|
||||||
this.settings.organizingSidebarSavedViews = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -308,9 +314,11 @@ export class SettingsComponent
|
|||||||
documentEditingRemoveInboxTags: this.settings.get(
|
documentEditingRemoveInboxTags: this.settings.get(
|
||||||
SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS
|
SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS
|
||||||
),
|
),
|
||||||
|
documentEditingOverlayThumbnail: this.settings.get(
|
||||||
|
SETTINGS_KEYS.DOCUMENT_EDITING_OVERLAY_THUMBNAIL
|
||||||
|
),
|
||||||
searchDbOnly: this.settings.get(SETTINGS_KEYS.SEARCH_DB_ONLY),
|
searchDbOnly: this.settings.get(SETTINGS_KEYS.SEARCH_DB_ONLY),
|
||||||
searchLink: this.settings.get(SETTINGS_KEYS.SEARCH_FULL_TYPE),
|
searchLink: this.settings.get(SETTINGS_KEYS.SEARCH_FULL_TYPE),
|
||||||
savedViews: {},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -323,15 +331,11 @@ export class SettingsComponent
|
|||||||
this.router
|
this.router
|
||||||
.navigate(['settings', foundNavIDkey.toLowerCase()])
|
.navigate(['settings', foundNavIDkey.toLowerCase()])
|
||||||
.then((navigated) => {
|
.then((navigated) => {
|
||||||
this.settings.organizingSidebarSavedViews = false
|
|
||||||
if (!navigated && this.isDirty) {
|
if (!navigated && this.isDirty) {
|
||||||
this.activeNavID = navChangeEvent.activeId
|
this.activeNavID = navChangeEvent.activeId
|
||||||
} else if (navigated && this.isDirty) {
|
} else if (navigated && this.isDirty) {
|
||||||
this.initialize()
|
this.initialize()
|
||||||
}
|
}
|
||||||
if (this.activeNavID === SettingsNavIDs.SavedViews) {
|
|
||||||
this.settings.organizingSidebarSavedViews = true
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -342,34 +346,6 @@ export class SettingsComponent
|
|||||||
|
|
||||||
let storeData = this.getCurrentSettings()
|
let storeData = this.getCurrentSettings()
|
||||||
|
|
||||||
if (this.savedViews) {
|
|
||||||
this.emptyGroup(this.savedViewGroup)
|
|
||||||
|
|
||||||
for (let view of this.savedViews) {
|
|
||||||
storeData.savedViews[view.id.toString()] = {
|
|
||||||
id: view.id,
|
|
||||||
name: view.name,
|
|
||||||
show_on_dashboard: view.show_on_dashboard,
|
|
||||||
show_in_sidebar: view.show_in_sidebar,
|
|
||||||
page_size: view.page_size,
|
|
||||||
display_mode: view.display_mode,
|
|
||||||
display_fields: view.display_fields,
|
|
||||||
}
|
|
||||||
this.savedViewGroup.addControl(
|
|
||||||
view.id.toString(),
|
|
||||||
new FormGroup({
|
|
||||||
id: new FormControl(null),
|
|
||||||
name: new FormControl(null),
|
|
||||||
show_on_dashboard: new FormControl(null),
|
|
||||||
show_in_sidebar: new FormControl(null),
|
|
||||||
page_size: new FormControl(null),
|
|
||||||
display_mode: new FormControl(null),
|
|
||||||
display_fields: new FormControl([]),
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.store = new BehaviorSubject(storeData)
|
this.store = new BehaviorSubject(storeData)
|
||||||
|
|
||||||
this.storeSub = this.store.asObservable().subscribe((state) => {
|
this.storeSub = this.store.asObservable().subscribe((state) => {
|
||||||
@ -409,32 +385,12 @@ export class SettingsComponent
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private emptyGroup(group: FormGroup) {
|
|
||||||
Object.keys(group.controls).forEach((key) => group.removeControl(key))
|
|
||||||
}
|
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
if (this.isDirty) this.settings.updateAppearanceSettings() // in case user changed appearance but didn't save
|
if (this.isDirty) this.settings.updateAppearanceSettings() // in case user changed appearance but didn't save
|
||||||
this.storeSub && this.storeSub.unsubscribe()
|
this.storeSub && this.storeSub.unsubscribe()
|
||||||
this.settings.organizingSidebarSavedViews = false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteSavedView(savedView: SavedView) {
|
public saveSettings() {
|
||||||
this.savedViewService.delete(savedView).subscribe(() => {
|
|
||||||
this.savedViewGroup.removeControl(savedView.id.toString())
|
|
||||||
this.savedViews.splice(this.savedViews.indexOf(savedView), 1)
|
|
||||||
this.toastService.showInfo(
|
|
||||||
$localize`Saved view "${savedView.name}" deleted.`
|
|
||||||
)
|
|
||||||
this.savedViewService.clearCache()
|
|
||||||
this.savedViewService.listAll().subscribe((r) => {
|
|
||||||
this.savedViews = r.results
|
|
||||||
this.initialize(true)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
private saveLocalSettings() {
|
|
||||||
this.savePending = true
|
this.savePending = true
|
||||||
const reloadRequired =
|
const reloadRequired =
|
||||||
this.settingsForm.value.displayLanguage !=
|
this.settingsForm.value.displayLanguage !=
|
||||||
@ -539,6 +495,10 @@ export class SettingsComponent
|
|||||||
SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS,
|
SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS,
|
||||||
this.settingsForm.value.documentEditingRemoveInboxTags
|
this.settingsForm.value.documentEditingRemoveInboxTags
|
||||||
)
|
)
|
||||||
|
this.settings.set(
|
||||||
|
SETTINGS_KEYS.DOCUMENT_EDITING_OVERLAY_THUMBNAIL,
|
||||||
|
this.settingsForm.value.documentEditingOverlayThumbnail
|
||||||
|
)
|
||||||
this.settings.set(
|
this.settings.set(
|
||||||
SETTINGS_KEYS.SEARCH_DB_ONLY,
|
SETTINGS_KEYS.SEARCH_DB_ONLY,
|
||||||
this.settingsForm.value.searchDbOnly
|
this.settingsForm.value.searchDbOnly
|
||||||
@ -592,31 +552,6 @@ export class SettingsComponent
|
|||||||
return new Date()
|
return new Date()
|
||||||
}
|
}
|
||||||
|
|
||||||
saveSettings() {
|
|
||||||
// only patch views that have actually changed
|
|
||||||
const changed: SavedView[] = []
|
|
||||||
Object.values(this.savedViewGroup.controls)
|
|
||||||
.filter((g: FormGroup) => !g.pristine)
|
|
||||||
.forEach((group: FormGroup) => {
|
|
||||||
changed.push(group.value)
|
|
||||||
})
|
|
||||||
if (changed.length > 0) {
|
|
||||||
this.savedViewService.patchMany(changed).subscribe({
|
|
||||||
next: () => {
|
|
||||||
this.saveLocalSettings()
|
|
||||||
},
|
|
||||||
error: (error) => {
|
|
||||||
this.toastService.showError(
|
|
||||||
$localize`Error while storing settings on server.`,
|
|
||||||
error
|
|
||||||
)
|
|
||||||
},
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
this.saveLocalSettings()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
reset() {
|
reset() {
|
||||||
this.settingsForm.patchValue(this.store.getValue())
|
this.settingsForm.patchValue(this.store.getValue())
|
||||||
}
|
}
|
||||||
|
@ -4,15 +4,40 @@
|
|||||||
info="File Tasks shows you documents that have been consumed, are waiting to be, or may have failed during the process."
|
info="File Tasks shows you documents that have been consumed, are waiting to be, or may have failed during the process."
|
||||||
i18n-info
|
i18n-info
|
||||||
>
|
>
|
||||||
<div class="btn-toolbar col col-md-auto align-items-center">
|
<div class="btn-toolbar col col-md-auto align-items-center gap-2">
|
||||||
<button class="btn btn-sm btn-outline-secondary me-2" (click)="clearSelection()" [hidden]="selectedTasks.size === 0">
|
<button class="btn btn-sm btn-outline-secondary me-2" (click)="clearSelection()" [hidden]="selectedTasks.size === 0">
|
||||||
<i-bs name="x"></i-bs> <ng-container i18n>Clear selection</ng-container>
|
<i-bs name="x"></i-bs> <ng-container i18n>Clear selection</ng-container>
|
||||||
</button>
|
</button>
|
||||||
<button class="btn btn-sm btn-outline-primary me-4" (click)="dismissTasks()" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.PaperlessTask }" [disabled]="tasksService.total === 0">
|
<button class="btn btn-sm btn-outline-primary me-2" (click)="dismissTasks()" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.PaperlessTask }" [disabled]="tasksService.total === 0">
|
||||||
<i-bs name="check2-all"></i-bs> {{dismissButtonText}}
|
<i-bs name="check2-all"></i-bs> {{dismissButtonText}}
|
||||||
</button>
|
</button>
|
||||||
<div class="form-check form-switch mb-0">
|
<div class="form-inline d-flex align-items-center">
|
||||||
<input class="form-check-input" type="checkbox" role="switch" id="autoRefreshSwitch" (click)="toggleAutoRefresh()" [attr.checked]="autoRefreshInterval">
|
<div class="input-group input-group-sm flex-fill w-auto flex-nowrap">
|
||||||
|
<span class="input-group-text text-muted" i18n>Filter by</span>
|
||||||
|
@if (filterTargets.length > 1) {
|
||||||
|
<div ngbDropdown>
|
||||||
|
<button class="btn btn-sm btn-outline-primary" ngbDropdownToggle>{{filterTargetName}}</button>
|
||||||
|
<div class="dropdown-menu shadow" ngbDropdownMenu>
|
||||||
|
@for (t of filterTargets; track t.id) {
|
||||||
|
<button ngbDropdownItem [class.active]="filterTargetID === t.id" (click)="filterTargetID = t.id">{{t.name}}</button>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
} @else {
|
||||||
|
<span class="input-group-text">{{filterTargetName}}</span>
|
||||||
|
}
|
||||||
|
@if (filterText?.length) {
|
||||||
|
<button class="btn btn-link btn-sm px-2 position-absolute top-0 end-0 z-10" (click)="resetFilter()">
|
||||||
|
<i-bs width="1em" height="1em" name="x"></i-bs>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
<input #filterInput class="form-control form-control-sm" type="text"
|
||||||
|
(keyup)="filterInputKeyup($event)"
|
||||||
|
[(ngModel)]="filterText">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-check form-switch mb-0 ms-2">
|
||||||
|
<input class="form-check-input" type="checkbox" role="switch" [(ngModel)]="autoRefreshEnabled">
|
||||||
<label class="form-check-label" for="autoRefreshSwitch" i18n>Auto refresh</label>
|
<label class="form-check-label" for="autoRefreshSwitch" i18n>Auto refresh</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -118,13 +143,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
|
||||||
<ul ngbNav #nav="ngbNav" [(activeId)]="activeTab" class="nav-tabs" (hidden)="duringTabChange($event)">
|
<ul ngbNav #nav="ngbNav" [(activeId)]="activeTab" class="nav-tabs" (hidden)="duringTabChange()" (navChange)="beforeTabChange()">
|
||||||
<li ngbNavItem="failed">
|
<li ngbNavItem="failed">
|
||||||
<a ngbNavLink i18n>Failed@if (tasksService.failedFileTasks.length > 0) {
|
<a ngbNavLink i18n>Failed@if (tasksService.failedFileTasks.length > 0) {
|
||||||
<span class="badge bg-danger ms-2">{{tasksService.failedFileTasks.length}}</span>
|
<span class="badge bg-danger ms-2">{{tasksService.failedFileTasks.length}}</span>
|
||||||
}</a>
|
}</a>
|
||||||
<ng-template ngbNavContent>
|
<ng-template ngbNavContent>
|
||||||
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:tasksService.failedFileTasks}"></ng-container>
|
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:currentTasks}"></ng-container>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
<li ngbNavItem="completed">
|
<li ngbNavItem="completed">
|
||||||
@ -132,7 +157,7 @@
|
|||||||
<span class="badge bg-secondary ms-2">{{tasksService.completedFileTasks.length}}</span>
|
<span class="badge bg-secondary ms-2">{{tasksService.completedFileTasks.length}}</span>
|
||||||
}</a>
|
}</a>
|
||||||
<ng-template ngbNavContent>
|
<ng-template ngbNavContent>
|
||||||
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:tasksService.completedFileTasks}"></ng-container>
|
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:currentTasks}"></ng-container>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
<li ngbNavItem="started">
|
<li ngbNavItem="started">
|
||||||
@ -140,7 +165,7 @@
|
|||||||
<span class="badge bg-secondary ms-2">{{tasksService.startedFileTasks.length}}</span>
|
<span class="badge bg-secondary ms-2">{{tasksService.startedFileTasks.length}}</span>
|
||||||
}</a>
|
}</a>
|
||||||
<ng-template ngbNavContent>
|
<ng-template ngbNavContent>
|
||||||
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:tasksService.startedFileTasks}"></ng-container>
|
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:currentTasks}"></ng-container>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
<li ngbNavItem="queued">
|
<li ngbNavItem="queued">
|
||||||
@ -148,7 +173,7 @@
|
|||||||
<span class="badge bg-secondary ms-2">{{tasksService.queuedFileTasks.length}}</span>
|
<span class="badge bg-secondary ms-2">{{tasksService.queuedFileTasks.length}}</span>
|
||||||
}</a>
|
}</a>
|
||||||
<ng-template ngbNavContent>
|
<ng-template ngbNavContent>
|
||||||
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:tasksService.queuedFileTasks}"></ng-container>
|
<ng-container [ngTemplateOutlet]="tasksTemplate" [ngTemplateOutletContext]="{tasks:currentTasks}"></ng-container>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
@ -26,3 +26,14 @@ pre {
|
|||||||
max-width: 150px;
|
max-width: 150px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.input-group .dropdown .btn {
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
border-bottom-right-radius: 0;
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
border-bottom-left-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.z-10 {
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
@ -1,36 +1,36 @@
|
|||||||
import { DatePipe } from '@angular/common'
|
import { DatePipe } from '@angular/common'
|
||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import {
|
import {
|
||||||
HttpTestingController,
|
HttpTestingController,
|
||||||
provideHttpClientTesting,
|
provideHttpClientTesting,
|
||||||
} from '@angular/common/http/testing'
|
} from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
import { FormsModule } from '@angular/forms'
|
||||||
import { By } from '@angular/platform-browser'
|
import { By } from '@angular/platform-browser'
|
||||||
import { Router } from '@angular/router'
|
import { Router } from '@angular/router'
|
||||||
import { RouterTestingModule } from '@angular/router/testing'
|
import { RouterTestingModule } from '@angular/router/testing'
|
||||||
import {
|
import {
|
||||||
NgbModal,
|
NgbModal,
|
||||||
|
NgbModalRef,
|
||||||
NgbModule,
|
NgbModule,
|
||||||
NgbNavItem,
|
NgbNavItem,
|
||||||
NgbModalRef,
|
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import { routes } from 'src/app/app-routing.module'
|
import { routes } from 'src/app/app-routing.module'
|
||||||
import {
|
import {
|
||||||
PaperlessTask,
|
PaperlessTask,
|
||||||
PaperlessTaskType,
|
|
||||||
PaperlessTaskStatus,
|
PaperlessTaskStatus,
|
||||||
|
PaperlessTaskType,
|
||||||
} from 'src/app/data/paperless-task'
|
} from 'src/app/data/paperless-task'
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { TasksService } from 'src/app/services/tasks.service'
|
import { TasksService } from 'src/app/services/tasks.service'
|
||||||
import { environment } from 'src/environments/environment'
|
import { environment } from 'src/environments/environment'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
import { TasksComponent } from './tasks.component'
|
import { TasksComponent, TaskTab } from './tasks.component'
|
||||||
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { FormsModule } from '@angular/forms'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const tasks: PaperlessTask[] = [
|
const tasks: PaperlessTask[] = [
|
||||||
{
|
{
|
||||||
@ -119,18 +119,16 @@ describe('TasksComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
TasksComponent,
|
|
||||||
PageHeaderComponent,
|
|
||||||
IfPermissionsDirective,
|
|
||||||
CustomDatePipe,
|
|
||||||
ConfirmDialogComponent,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgbModule,
|
NgbModule,
|
||||||
RouterTestingModule.withRoutes(routes),
|
RouterTestingModule.withRoutes(routes),
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
FormsModule,
|
FormsModule,
|
||||||
|
TasksComponent,
|
||||||
|
PageHeaderComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
CustomDatePipe,
|
||||||
|
ConfirmDialogComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
{
|
{
|
||||||
@ -167,7 +165,7 @@ describe('TasksComponent', () => {
|
|||||||
let currentTasksLength = tasks.filter(
|
let currentTasksLength = tasks.filter(
|
||||||
(t) => t.status === PaperlessTaskStatus.Failed
|
(t) => t.status === PaperlessTaskStatus.Failed
|
||||||
).length
|
).length
|
||||||
component.activeTab = 'failed'
|
component.activeTab = TaskTab.Failed
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
expect(tabButtons[0].nativeElement.textContent).toEqual(
|
expect(tabButtons[0].nativeElement.textContent).toEqual(
|
||||||
`Failed${currentTasksLength}`
|
`Failed${currentTasksLength}`
|
||||||
@ -179,7 +177,7 @@ describe('TasksComponent', () => {
|
|||||||
currentTasksLength = tasks.filter(
|
currentTasksLength = tasks.filter(
|
||||||
(t) => t.status === PaperlessTaskStatus.Complete
|
(t) => t.status === PaperlessTaskStatus.Complete
|
||||||
).length
|
).length
|
||||||
component.activeTab = 'completed'
|
component.activeTab = TaskTab.Completed
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
expect(tabButtons[1].nativeElement.textContent).toEqual(
|
expect(tabButtons[1].nativeElement.textContent).toEqual(
|
||||||
`Complete${currentTasksLength}`
|
`Complete${currentTasksLength}`
|
||||||
@ -188,7 +186,7 @@ describe('TasksComponent', () => {
|
|||||||
currentTasksLength = tasks.filter(
|
currentTasksLength = tasks.filter(
|
||||||
(t) => t.status === PaperlessTaskStatus.Started
|
(t) => t.status === PaperlessTaskStatus.Started
|
||||||
).length
|
).length
|
||||||
component.activeTab = 'started'
|
component.activeTab = TaskTab.Started
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
expect(tabButtons[2].nativeElement.textContent).toEqual(
|
expect(tabButtons[2].nativeElement.textContent).toEqual(
|
||||||
`Started${currentTasksLength}`
|
`Started${currentTasksLength}`
|
||||||
@ -197,7 +195,7 @@ describe('TasksComponent', () => {
|
|||||||
currentTasksLength = tasks.filter(
|
currentTasksLength = tasks.filter(
|
||||||
(t) => t.status === PaperlessTaskStatus.Pending
|
(t) => t.status === PaperlessTaskStatus.Pending
|
||||||
).length
|
).length
|
||||||
component.activeTab = 'queued'
|
component.activeTab = TaskTab.Queued
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
expect(tabButtons[3].nativeElement.textContent).toEqual(
|
expect(tabButtons[3].nativeElement.textContent).toEqual(
|
||||||
`Queued${currentTasksLength}`
|
`Queued${currentTasksLength}`
|
||||||
@ -206,7 +204,7 @@ describe('TasksComponent', () => {
|
|||||||
|
|
||||||
it('should to go page 1 between tab switch', () => {
|
it('should to go page 1 between tab switch', () => {
|
||||||
component.page = 10
|
component.page = 10
|
||||||
component.duringTabChange(2)
|
component.duringTabChange()
|
||||||
expect(component.page).toEqual(1)
|
expect(component.page).toEqual(1)
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -283,10 +281,63 @@ describe('TasksComponent', () => {
|
|||||||
expect(reloadSpy).toHaveBeenCalledTimes(1)
|
expect(reloadSpy).toHaveBeenCalledTimes(1)
|
||||||
jest.advanceTimersByTime(5000)
|
jest.advanceTimersByTime(5000)
|
||||||
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
||||||
|
component.autoRefreshEnabled = false
|
||||||
component.toggleAutoRefresh()
|
|
||||||
expect(component.autoRefreshInterval).toBeNull()
|
|
||||||
jest.advanceTimersByTime(6000)
|
jest.advanceTimersByTime(6000)
|
||||||
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
expect(reloadSpy).toHaveBeenCalledTimes(2)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should filter tasks by file name', () => {
|
||||||
|
const input = fixture.debugElement.query(
|
||||||
|
By.css('pngx-page-header input[type=text]')
|
||||||
|
)
|
||||||
|
input.nativeElement.value = '191092'
|
||||||
|
input.nativeElement.dispatchEvent(new Event('input'))
|
||||||
|
jest.advanceTimersByTime(150) // debounce time
|
||||||
|
fixture.detectChanges()
|
||||||
|
expect(component.filterText).toEqual('191092')
|
||||||
|
expect(
|
||||||
|
fixture.debugElement.queryAll(By.css('table tbody tr')).length
|
||||||
|
).toEqual(2) // 1 task x 2 lines
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should filter tasks by result', () => {
|
||||||
|
component.activeTab = TaskTab.Failed
|
||||||
|
fixture.detectChanges()
|
||||||
|
component.filterTargetID = 1
|
||||||
|
const input = fixture.debugElement.query(
|
||||||
|
By.css('pngx-page-header input[type=text]')
|
||||||
|
)
|
||||||
|
input.nativeElement.value = 'duplicate'
|
||||||
|
input.nativeElement.dispatchEvent(new Event('input'))
|
||||||
|
jest.advanceTimersByTime(150) // debounce time
|
||||||
|
fixture.detectChanges()
|
||||||
|
expect(component.filterText).toEqual('duplicate')
|
||||||
|
expect(
|
||||||
|
fixture.debugElement.queryAll(By.css('table tbody tr')).length
|
||||||
|
).toEqual(4) // 2 tasks x 2 lines
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should support keyboard events for filtering', () => {
|
||||||
|
const input = fixture.debugElement.query(
|
||||||
|
By.css('pngx-page-header input[type=text]')
|
||||||
|
)
|
||||||
|
input.nativeElement.value = '191092'
|
||||||
|
input.nativeElement.dispatchEvent(
|
||||||
|
new KeyboardEvent('keyup', { key: 'Enter' })
|
||||||
|
)
|
||||||
|
expect(component.filterText).toEqual('191092') // no debounce needed
|
||||||
|
input.nativeElement.dispatchEvent(
|
||||||
|
new KeyboardEvent('keyup', { key: 'Escape' })
|
||||||
|
)
|
||||||
|
expect(component.filterText).toEqual('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should reset filter and target on tab switch', () => {
|
||||||
|
component.filterText = '191092'
|
||||||
|
component.filterTargetID = 1
|
||||||
|
component.activeTab = TaskTab.Completed
|
||||||
|
component.beforeTabChange()
|
||||||
|
expect(component.filterText).toEqual('')
|
||||||
|
expect(component.filterTargetID).toEqual(0)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
@ -1,22 +1,75 @@
|
|||||||
import { Component, OnInit, OnDestroy } from '@angular/core'
|
import { NgTemplateOutlet, SlicePipe } from '@angular/common'
|
||||||
|
import { Component, OnDestroy, OnInit } from '@angular/core'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { Router } from '@angular/router'
|
import { Router } from '@angular/router'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import {
|
||||||
import { first } from 'rxjs'
|
NgbCollapseModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbModal,
|
||||||
|
NgbNavModule,
|
||||||
|
NgbPaginationModule,
|
||||||
|
NgbPopoverModule,
|
||||||
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import {
|
||||||
|
debounceTime,
|
||||||
|
distinctUntilChanged,
|
||||||
|
filter,
|
||||||
|
first,
|
||||||
|
Subject,
|
||||||
|
takeUntil,
|
||||||
|
timer,
|
||||||
|
} from 'rxjs'
|
||||||
import { PaperlessTask } from 'src/app/data/paperless-task'
|
import { PaperlessTask } from 'src/app/data/paperless-task'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { TasksService } from 'src/app/services/tasks.service'
|
import { TasksService } from 'src/app/services/tasks.service'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
|
export enum TaskTab {
|
||||||
|
Queued = 'queued',
|
||||||
|
Started = 'started',
|
||||||
|
Completed = 'completed',
|
||||||
|
Failed = 'failed',
|
||||||
|
}
|
||||||
|
|
||||||
|
enum TaskFilterTargetID {
|
||||||
|
Name,
|
||||||
|
Result,
|
||||||
|
}
|
||||||
|
|
||||||
|
const FILTER_TARGETS = [
|
||||||
|
{ id: TaskFilterTargetID.Name, name: $localize`Name` },
|
||||||
|
{ id: TaskFilterTargetID.Result, name: $localize`Result` },
|
||||||
|
]
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-tasks',
|
selector: 'pngx-tasks',
|
||||||
templateUrl: './tasks.component.html',
|
templateUrl: './tasks.component.html',
|
||||||
styleUrls: ['./tasks.component.scss'],
|
styleUrls: ['./tasks.component.scss'],
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
CustomDatePipe,
|
||||||
|
SlicePipe,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgTemplateOutlet,
|
||||||
|
NgbCollapseModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbNavModule,
|
||||||
|
NgbPaginationModule,
|
||||||
|
NgbPopoverModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class TasksComponent
|
export class TasksComponent
|
||||||
extends ComponentWithPermissions
|
extends LoadingComponentWithPermissions
|
||||||
implements OnInit, OnDestroy
|
implements OnInit, OnDestroy
|
||||||
{
|
{
|
||||||
public activeTab: string
|
public activeTab: TaskTab
|
||||||
public selectedTasks: Set<number> = new Set()
|
public selectedTasks: Set<number> = new Set()
|
||||||
public togggleAll: boolean = false
|
public togggleAll: boolean = false
|
||||||
public expandedTask: number
|
public expandedTask: number
|
||||||
@ -24,7 +77,27 @@ export class TasksComponent
|
|||||||
public pageSize: number = 25
|
public pageSize: number = 25
|
||||||
public page: number = 1
|
public page: number = 1
|
||||||
|
|
||||||
public autoRefreshInterval: any
|
public autoRefreshEnabled: boolean = true
|
||||||
|
|
||||||
|
private _filterText: string = ''
|
||||||
|
get filterText() {
|
||||||
|
return this._filterText
|
||||||
|
}
|
||||||
|
set filterText(value: string) {
|
||||||
|
this.filterDebounce.next(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
public filterTargetID: TaskFilterTargetID = TaskFilterTargetID.Name
|
||||||
|
public get filterTargetName(): string {
|
||||||
|
return this.filterTargets.find((t) => t.id == this.filterTargetID).name
|
||||||
|
}
|
||||||
|
private filterDebounce: Subject<string> = new Subject<string>()
|
||||||
|
|
||||||
|
public get filterTargets(): Array<{ id: number; name: string }> {
|
||||||
|
return [TaskTab.Failed, TaskTab.Completed].includes(this.activeTab)
|
||||||
|
? FILTER_TARGETS
|
||||||
|
: FILTER_TARGETS.slice(0, 1)
|
||||||
|
}
|
||||||
|
|
||||||
get dismissButtonText(): string {
|
get dismissButtonText(): string {
|
||||||
return this.selectedTasks.size > 0
|
return this.selectedTasks.size > 0
|
||||||
@ -42,12 +115,28 @@ export class TasksComponent
|
|||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.tasksService.reload()
|
this.tasksService.reload()
|
||||||
this.toggleAutoRefresh()
|
timer(5000, 5000)
|
||||||
|
.pipe(
|
||||||
|
filter(() => this.autoRefreshEnabled),
|
||||||
|
takeUntil(this.unsubscribeNotifier)
|
||||||
|
)
|
||||||
|
.subscribe(() => {
|
||||||
|
this.tasksService.reload()
|
||||||
|
})
|
||||||
|
|
||||||
|
this.filterDebounce
|
||||||
|
.pipe(
|
||||||
|
takeUntil(this.unsubscribeNotifier),
|
||||||
|
debounceTime(100),
|
||||||
|
distinctUntilChanged(),
|
||||||
|
filter((query) => !query.length || query.length > 2)
|
||||||
|
)
|
||||||
|
.subscribe((query) => (this._filterText = query))
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
|
super.ngOnDestroy()
|
||||||
this.tasksService.cancelPending()
|
this.tasksService.cancelPending()
|
||||||
clearInterval(this.autoRefreshInterval)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dismissTask(task: PaperlessTask) {
|
dismissTask(task: PaperlessTask) {
|
||||||
@ -96,19 +185,30 @@ export class TasksComponent
|
|||||||
get currentTasks(): PaperlessTask[] {
|
get currentTasks(): PaperlessTask[] {
|
||||||
let tasks: PaperlessTask[] = []
|
let tasks: PaperlessTask[] = []
|
||||||
switch (this.activeTab) {
|
switch (this.activeTab) {
|
||||||
case 'queued':
|
case TaskTab.Queued:
|
||||||
tasks = this.tasksService.queuedFileTasks
|
tasks = this.tasksService.queuedFileTasks
|
||||||
break
|
break
|
||||||
case 'started':
|
case TaskTab.Started:
|
||||||
tasks = this.tasksService.startedFileTasks
|
tasks = this.tasksService.startedFileTasks
|
||||||
break
|
break
|
||||||
case 'completed':
|
case TaskTab.Completed:
|
||||||
tasks = this.tasksService.completedFileTasks
|
tasks = this.tasksService.completedFileTasks
|
||||||
break
|
break
|
||||||
case 'failed':
|
case TaskTab.Failed:
|
||||||
tasks = this.tasksService.failedFileTasks
|
tasks = this.tasksService.failedFileTasks
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
if (this._filterText.length) {
|
||||||
|
tasks = tasks.filter((t) => {
|
||||||
|
if (this.filterTargetID == TaskFilterTargetID.Name) {
|
||||||
|
return t.task_file_name
|
||||||
|
.toLowerCase()
|
||||||
|
.includes(this._filterText.toLowerCase())
|
||||||
|
} else if (this.filterTargetID == TaskFilterTargetID.Result) {
|
||||||
|
return t.result.toLowerCase().includes(this._filterText.toLowerCase())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
return tasks
|
return tasks
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -125,31 +225,37 @@ export class TasksComponent
|
|||||||
this.selectedTasks.clear()
|
this.selectedTasks.clear()
|
||||||
}
|
}
|
||||||
|
|
||||||
duringTabChange(navID: number) {
|
duringTabChange() {
|
||||||
this.page = 1
|
this.page = 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
beforeTabChange() {
|
||||||
|
this.resetFilter()
|
||||||
|
this.filterTargetID = TaskFilterTargetID.Name
|
||||||
|
}
|
||||||
|
|
||||||
get activeTabLocalized(): string {
|
get activeTabLocalized(): string {
|
||||||
switch (this.activeTab) {
|
switch (this.activeTab) {
|
||||||
case 'queued':
|
case TaskTab.Queued:
|
||||||
return $localize`queued`
|
return $localize`queued`
|
||||||
case 'started':
|
case TaskTab.Started:
|
||||||
return $localize`started`
|
return $localize`started`
|
||||||
case 'completed':
|
case TaskTab.Completed:
|
||||||
return $localize`completed`
|
return $localize`completed`
|
||||||
case 'failed':
|
case TaskTab.Failed:
|
||||||
return $localize`failed`
|
return $localize`failed`
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
toggleAutoRefresh(): void {
|
public resetFilter() {
|
||||||
if (this.autoRefreshInterval) {
|
this._filterText = ''
|
||||||
clearInterval(this.autoRefreshInterval)
|
}
|
||||||
this.autoRefreshInterval = null
|
|
||||||
} else {
|
filterInputKeyup(event: KeyboardEvent) {
|
||||||
this.autoRefreshInterval = setInterval(() => {
|
if (event.key == 'Enter') {
|
||||||
this.tasksService.reload()
|
this._filterText = (event.target as HTMLInputElement).value
|
||||||
}, 5000)
|
} else if (event.key === 'Escape') {
|
||||||
|
this.resetFilter()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,7 +38,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
@if (isLoading) {
|
@if (loading) {
|
||||||
<tr>
|
<tr>
|
||||||
<td colspan="5">
|
<td colspan="5">
|
||||||
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
<div class="spinner-border spinner-border-sm me-2" role="status"></div>
|
||||||
@ -47,15 +47,20 @@
|
|||||||
</tr>
|
</tr>
|
||||||
}
|
}
|
||||||
@for (document of documentsInTrash; track document.id) {
|
@for (document of documentsInTrash; track document.id) {
|
||||||
<tr (click)="toggleSelected(document); $event.stopPropagation();">
|
<tr (click)="toggleSelected(document); $event.stopPropagation();" (mouseleave)="popupPreview.close()" class="data-row fade" [class.show]="show">
|
||||||
<td>
|
<td>
|
||||||
<div class="form-check m-0 ms-2 me-n2">
|
<div class="form-check m-0 ms-2 me-n2">
|
||||||
<input type="checkbox" class="form-check-input" id="{{document.id}}" [checked]="selectedDocuments.has(document.id)" (click)="toggleSelected(document); $event.stopPropagation();">
|
<input type="checkbox" class="form-check-input" id="{{document.id}}" [checked]="selectedDocuments.has(document.id)" (click)="toggleSelected(document); $event.stopPropagation();">
|
||||||
<label class="form-check-label" for="{{document.id}}"></label>
|
<label class="form-check-label" for="{{document.id}}"></label>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td scope="row">{{ document.title }}</td>
|
<td scope="row">
|
||||||
<td scope="row" i18n>{{ getDaysRemaining(document) }} days</td>
|
{{ document.title }}
|
||||||
|
<pngx-preview-popup [document]="document" linkClasses="btn btn-sm btn-link" #popupPreview>
|
||||||
|
<i-bs name="eye"></i-bs>
|
||||||
|
</pngx-preview-popup>
|
||||||
|
</td>
|
||||||
|
<td scope="row" class="d-none d-sm-table-cell" i18n>{{ getDaysRemaining(document) }} days</td>
|
||||||
<td scope="row">
|
<td scope="row">
|
||||||
<div class="btn-group d-block d-sm-none">
|
<div class="btn-group d-block d-sm-none">
|
||||||
<div ngbDropdown container="body" class="d-inline-block">
|
<div ngbDropdown container="body" class="d-inline-block">
|
||||||
@ -83,7 +88,7 @@
|
|||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@if (!isLoading) {
|
@if (!loading) {
|
||||||
<div class="d-flex mb-2">
|
<div class="d-flex mb-2">
|
||||||
<div>
|
<div>
|
||||||
<ng-container i18n>{totalDocuments, plural, =1 {One document in trash} other {{{totalDocuments || 0}} total documents in trash}}</ng-container>
|
<ng-container i18n>{totalDocuments, plural, =1 {One document in trash} other {{{totalDocuments || 0}} total documents in trash}}</ng-container>
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
// hide caret on mobile dropdown
|
||||||
|
.d-block.d-sm-none .dropdown-toggle::after {
|
||||||
|
display: none;
|
||||||
|
}
|
@ -1,21 +1,22 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
|
||||||
import { TrashComponent } from './trash.component'
|
|
||||||
import { HttpClientTestingModule } from '@angular/common/http/testing'
|
import { HttpClientTestingModule } from '@angular/common/http/testing'
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { By } from '@angular/platform-browser'
|
||||||
|
import { Router } from '@angular/router'
|
||||||
import {
|
import {
|
||||||
NgbModal,
|
NgbModal,
|
||||||
NgbPaginationModule,
|
NgbPaginationModule,
|
||||||
NgbPopoverModule,
|
NgbPopoverModule,
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
|
||||||
import { TrashService } from 'src/app/services/trash.service'
|
|
||||||
import { of, throwError } from 'rxjs'
|
import { of, throwError } from 'rxjs'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
|
||||||
import { By } from '@angular/platform-browser'
|
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { TrashService } from 'src/app/services/trash.service'
|
||||||
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { TrashComponent } from './trash.component'
|
||||||
|
|
||||||
const documentsInTrash = [
|
const documentsInTrash = [
|
||||||
{
|
{
|
||||||
@ -38,15 +39,10 @@ describe('TrashComponent', () => {
|
|||||||
let trashService: TrashService
|
let trashService: TrashService
|
||||||
let modalService: NgbModal
|
let modalService: NgbModal
|
||||||
let toastService: ToastService
|
let toastService: ToastService
|
||||||
|
let router: Router
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
TrashComponent,
|
|
||||||
PageHeaderComponent,
|
|
||||||
ConfirmDialogComponent,
|
|
||||||
SafeHtmlPipe,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
HttpClientTestingModule,
|
HttpClientTestingModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
@ -54,6 +50,10 @@ describe('TrashComponent', () => {
|
|||||||
NgbPopoverModule,
|
NgbPopoverModule,
|
||||||
NgbPaginationModule,
|
NgbPaginationModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
TrashComponent,
|
||||||
|
PageHeaderComponent,
|
||||||
|
ConfirmDialogComponent,
|
||||||
|
SafeHtmlPipe,
|
||||||
],
|
],
|
||||||
}).compileComponents()
|
}).compileComponents()
|
||||||
|
|
||||||
@ -61,11 +61,13 @@ describe('TrashComponent', () => {
|
|||||||
trashService = TestBed.inject(TrashService)
|
trashService = TestBed.inject(TrashService)
|
||||||
modalService = TestBed.inject(NgbModal)
|
modalService = TestBed.inject(NgbModal)
|
||||||
toastService = TestBed.inject(ToastService)
|
toastService = TestBed.inject(ToastService)
|
||||||
|
router = TestBed.inject(Router)
|
||||||
component = fixture.componentInstance
|
component = fixture.componentInstance
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should call correct service method on reload', () => {
|
it('should call correct service method on reload', () => {
|
||||||
|
jest.useFakeTimers()
|
||||||
const trashSpy = jest.spyOn(trashService, 'getTrash')
|
const trashSpy = jest.spyOn(trashService, 'getTrash')
|
||||||
trashSpy.mockReturnValue(
|
trashSpy.mockReturnValue(
|
||||||
of({
|
of({
|
||||||
@ -75,6 +77,7 @@ describe('TrashComponent', () => {
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
component.reload()
|
component.reload()
|
||||||
|
jest.advanceTimersByTime(100)
|
||||||
expect(trashSpy).toHaveBeenCalled()
|
expect(trashSpy).toHaveBeenCalled()
|
||||||
expect(component.documentsInTrash).toEqual(documentsInTrash)
|
expect(component.documentsInTrash).toEqual(documentsInTrash)
|
||||||
})
|
})
|
||||||
@ -161,6 +164,22 @@ describe('TrashComponent', () => {
|
|||||||
expect(restoreSpy).toHaveBeenCalledWith([1, 2])
|
expect(restoreSpy).toHaveBeenCalledWith([1, 2])
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should offer link to restored document', () => {
|
||||||
|
let toasts
|
||||||
|
const navigateSpy = jest.spyOn(router, 'navigate')
|
||||||
|
toastService.getToasts().subscribe((allToasts) => {
|
||||||
|
toasts = [...allToasts]
|
||||||
|
})
|
||||||
|
jest.spyOn(trashService, 'restoreDocuments').mockReturnValue(of('OK'))
|
||||||
|
component.restore(documentsInTrash[0])
|
||||||
|
expect(toasts.length).toEqual(1)
|
||||||
|
toasts[0].action()
|
||||||
|
expect(navigateSpy).toHaveBeenCalledWith([
|
||||||
|
'documents',
|
||||||
|
documentsInTrash[0].id,
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
it('should support toggle all items in view', () => {
|
it('should support toggle all items in view', () => {
|
||||||
component.documentsInTrash = documentsInTrash
|
component.documentsInTrash = documentsInTrash
|
||||||
expect(component.selectedDocuments.size).toEqual(0)
|
expect(component.selectedDocuments.size).toEqual(0)
|
||||||
|
@ -1,49 +1,74 @@
|
|||||||
import { Component, OnDestroy } from '@angular/core'
|
import { Component, OnDestroy } from '@angular/core'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { Router } from '@angular/router'
|
||||||
|
import {
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbModal,
|
||||||
|
NgbPaginationModule,
|
||||||
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { delay, takeUntil, tap } from 'rxjs'
|
||||||
import { Document } from 'src/app/data/document'
|
import { Document } from 'src/app/data/document'
|
||||||
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
import { TrashService } from 'src/app/services/trash.service'
|
import { TrashService } from 'src/app/services/trash.service'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { Subject, takeUntil } from 'rxjs'
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { PreviewPopupComponent } from '../../common/preview-popup/preview-popup.component'
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-trash',
|
selector: 'pngx-trash',
|
||||||
templateUrl: './trash.component.html',
|
templateUrl: './trash.component.html',
|
||||||
styleUrl: './trash.component.scss',
|
styleUrl: './trash.component.scss',
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
PreviewPopupComponent,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbPaginationModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class TrashComponent implements OnDestroy {
|
export class TrashComponent
|
||||||
|
extends LoadingComponentWithPermissions
|
||||||
|
implements OnDestroy
|
||||||
|
{
|
||||||
public documentsInTrash: Document[] = []
|
public documentsInTrash: Document[] = []
|
||||||
public selectedDocuments: Set<number> = new Set()
|
public selectedDocuments: Set<number> = new Set()
|
||||||
public allToggled: boolean = false
|
public allToggled: boolean = false
|
||||||
public page: number = 1
|
public page: number = 1
|
||||||
public totalDocuments: number
|
public totalDocuments: number
|
||||||
public isLoading: boolean = false
|
|
||||||
unsubscribeNotifier: Subject<void> = new Subject()
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private trashService: TrashService,
|
private trashService: TrashService,
|
||||||
private toastService: ToastService,
|
private toastService: ToastService,
|
||||||
private modalService: NgbModal,
|
private modalService: NgbModal,
|
||||||
private settingsService: SettingsService
|
private settingsService: SettingsService,
|
||||||
|
private router: Router
|
||||||
) {
|
) {
|
||||||
|
super()
|
||||||
this.reload()
|
this.reload()
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
|
||||||
this.unsubscribeNotifier.next()
|
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
}
|
|
||||||
|
|
||||||
reload() {
|
reload() {
|
||||||
this.isLoading = true
|
this.loading = true
|
||||||
this.trashService.getTrash(this.page).subscribe((r) => {
|
this.trashService
|
||||||
this.documentsInTrash = r.results
|
.getTrash(this.page)
|
||||||
this.totalDocuments = r.count
|
.pipe(
|
||||||
this.isLoading = false
|
tap((r) => {
|
||||||
this.selectedDocuments.clear()
|
this.documentsInTrash = r.results
|
||||||
})
|
this.totalDocuments = r.count
|
||||||
|
this.selectedDocuments.clear()
|
||||||
|
this.loading = false
|
||||||
|
}),
|
||||||
|
delay(100)
|
||||||
|
)
|
||||||
|
.subscribe(() => {
|
||||||
|
this.show = true
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
delete(document: Document) {
|
delete(document: Document) {
|
||||||
@ -110,7 +135,14 @@ export class TrashComponent implements OnDestroy {
|
|||||||
restore(document: Document) {
|
restore(document: Document) {
|
||||||
this.trashService.restoreDocuments([document.id]).subscribe({
|
this.trashService.restoreDocuments([document.id]).subscribe({
|
||||||
next: () => {
|
next: () => {
|
||||||
this.toastService.showInfo($localize`Document restored`)
|
this.toastService.show({
|
||||||
|
content: $localize`Document restored`,
|
||||||
|
delay: 5000,
|
||||||
|
actionName: $localize`Open document`,
|
||||||
|
action: () => {
|
||||||
|
this.router.navigate(['documents', document.id])
|
||||||
|
},
|
||||||
|
})
|
||||||
this.reload()
|
this.reload()
|
||||||
},
|
},
|
||||||
error: (err) => {
|
error: (err) => {
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { DatePipe } from '@angular/common'
|
import { DatePipe } from '@angular/common'
|
||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
@ -6,22 +7,13 @@ import {
|
|||||||
fakeAsync,
|
fakeAsync,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { RouterTestingModule } from '@angular/router/testing'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import {
|
import { of, throwError } from 'rxjs'
|
||||||
NgbModule,
|
import { Group } from 'src/app/data/group'
|
||||||
NgbAlertModule,
|
import { User } from 'src/app/data/user'
|
||||||
NgbModal,
|
|
||||||
NgbModalRef,
|
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
|
||||||
import { throwError, of } from 'rxjs'
|
|
||||||
import { routes } from 'src/app/app-routing.module'
|
|
||||||
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
|
||||||
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { GroupService } from 'src/app/services/rest/group.service'
|
import { GroupService } from 'src/app/services/rest/group.service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
@ -30,21 +22,7 @@ import { ToastService } from 'src/app/services/toast.service'
|
|||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
||||||
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
||||||
import { CheckComponent } from '../../common/input/check/check.component'
|
|
||||||
import { NumberComponent } from '../../common/input/number/number.component'
|
|
||||||
import { PasswordComponent } from '../../common/input/password/password.component'
|
|
||||||
import { PermissionsGroupComponent } from '../../common/input/permissions/permissions-group/permissions-group.component'
|
|
||||||
import { PermissionsUserComponent } from '../../common/input/permissions/permissions-user/permissions-user.component'
|
|
||||||
import { SelectComponent } from '../../common/input/select/select.component'
|
|
||||||
import { TagsComponent } from '../../common/input/tags/tags.component'
|
|
||||||
import { TextComponent } from '../../common/input/text/text.component'
|
|
||||||
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
|
||||||
import { SettingsComponent } from '../settings/settings.component'
|
|
||||||
import { UsersAndGroupsComponent } from './users-groups.component'
|
import { UsersAndGroupsComponent } from './users-groups.component'
|
||||||
import { User } from 'src/app/data/user'
|
|
||||||
import { Group } from 'src/app/data/group'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const users = [
|
const users = [
|
||||||
{ id: 1, username: 'user1', is_superuser: false },
|
{ id: 1, username: 'user1', is_superuser: false },
|
||||||
@ -67,33 +45,7 @@ describe('UsersAndGroupsComponent', () => {
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
imports: [NgxBootstrapIconsModule.pick(allIcons)],
|
||||||
UsersAndGroupsComponent,
|
|
||||||
SettingsComponent,
|
|
||||||
PageHeaderComponent,
|
|
||||||
IfPermissionsDirective,
|
|
||||||
CustomDatePipe,
|
|
||||||
ConfirmDialogComponent,
|
|
||||||
CheckComponent,
|
|
||||||
SafeHtmlPipe,
|
|
||||||
SelectComponent,
|
|
||||||
TextComponent,
|
|
||||||
PasswordComponent,
|
|
||||||
NumberComponent,
|
|
||||||
TagsComponent,
|
|
||||||
PermissionsUserComponent,
|
|
||||||
PermissionsGroupComponent,
|
|
||||||
IfOwnerDirective,
|
|
||||||
],
|
|
||||||
imports: [
|
|
||||||
NgbModule,
|
|
||||||
RouterTestingModule.withRoutes(routes),
|
|
||||||
FormsModule,
|
|
||||||
ReactiveFormsModule,
|
|
||||||
NgbAlertModule,
|
|
||||||
NgSelectModule,
|
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
|
||||||
],
|
|
||||||
providers: [
|
providers: [
|
||||||
CustomDatePipe,
|
CustomDatePipe,
|
||||||
DatePipe,
|
DatePipe,
|
||||||
|
@ -1,23 +1,31 @@
|
|||||||
import { Component, OnDestroy, OnInit } from '@angular/core'
|
import { Component, OnDestroy, OnInit } from '@angular/core'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import { Subject, first, takeUntil } from 'rxjs'
|
import { Subject, first, takeUntil } from 'rxjs'
|
||||||
import { Group } from 'src/app/data/group'
|
import { Group } from 'src/app/data/group'
|
||||||
import { User } from 'src/app/data/user'
|
import { User } from 'src/app/data/user'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { GroupService } from 'src/app/services/rest/group.service'
|
import { GroupService } from 'src/app/services/rest/group.service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'
|
||||||
import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component'
|
import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component'
|
||||||
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
||||||
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-users-groups',
|
selector: 'pngx-users-groups',
|
||||||
templateUrl: './users-groups.component.html',
|
templateUrl: './users-groups.component.html',
|
||||||
styleUrls: ['./users-groups.component.scss'],
|
styleUrls: ['./users-groups.component.scss'],
|
||||||
|
imports: [
|
||||||
|
PageHeaderComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class UsersAndGroupsComponent
|
export class UsersAndGroupsComponent
|
||||||
extends ComponentWithPermissions
|
extends ComponentWithPermissions
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
<nav class="navbar navbar-dark sticky-top bg-primary flex-md-nowrap p-0 shadow-sm">
|
<nav class="navbar navbar-dark fixed-top bg-primary flex-md-nowrap p-0 shadow-sm">
|
||||||
<button class="navbar-toggler d-md-none collapsed border-0" type="button" data-toggle="collapse"
|
<button class="navbar-toggler d-md-none collapsed border-0" type="button" data-toggle="collapse"
|
||||||
data-target="#sidebarMenu" aria-controls="sidebarMenu" aria-expanded="false" aria-label="Toggle navigation"
|
data-target="#sidebarMenu" aria-controls="sidebarMenu" aria-expanded="false" aria-label="Toggle navigation"
|
||||||
(click)="isMenuCollapsed = !isMenuCollapsed">
|
(click)="isMenuCollapsed = !isMenuCollapsed">
|
||||||
@ -199,6 +199,13 @@
|
|||||||
<i-bs class="me-1" name="ui-radios"></i-bs><span> <ng-container i18n>Custom Fields</ng-container></span>
|
<i-bs class="me-1" name="ui-radios"></i-bs><span> <ng-container i18n>Custom Fields</ng-container></span>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li class="nav-item app-link" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.SavedView }">
|
||||||
|
<a class="nav-link" routerLink="savedviews" routerLinkActive="active" (click)="closeMenu()"
|
||||||
|
ngbPopover="Saved Views" i18n-ngbPopover [disablePopover]="!slimSidebarEnabled" placement="end"
|
||||||
|
container="body" triggers="mouseenter:mouseleave" popoverClass="popover-slim">
|
||||||
|
<i-bs class="me-1" name="window-stack"></i-bs><span> <ng-container i18n>Saved Views</ng-container></span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
<li class="nav-item app-link"
|
<li class="nav-item app-link"
|
||||||
*pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Workflow }"
|
*pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Workflow }"
|
||||||
tourAnchor="tour.workflows">
|
tourAnchor="tour.workflows">
|
||||||
@ -327,7 +334,7 @@
|
|||||||
</a>
|
</a>
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
<a class="small text-decoration-none" routerLink="/settings" fragment="update-checking"
|
<a *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.UISettings }" class="small text-decoration-none" routerLink="/settings" fragment="update-checking"
|
||||||
[ngbPopover]="updateCheckingNotEnabledPopContent" popoverClass="shadow" triggers="mouseenter"
|
[ngbPopover]="updateCheckingNotEnabledPopContent" popoverClass="shadow" triggers="mouseenter"
|
||||||
container="body">
|
container="body">
|
||||||
<i-bs width="1.2em" height="1.2em" name="info-circle"></i-bs>
|
<i-bs width="1.2em" height="1.2em" name="info-circle"></i-bs>
|
||||||
|
@ -1,6 +1,3 @@
|
|||||||
@import "node_modules/bootstrap/scss/functions";
|
|
||||||
@import "node_modules/bootstrap/scss/variables";
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Sidebar
|
* Sidebar
|
||||||
*/
|
*/
|
||||||
@ -15,6 +12,7 @@
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
--pngx-sidebar-width: 100%;
|
--pngx-sidebar-width: 100%;
|
||||||
max-width: var(--pngx-sidebar-width);
|
max-width: var(--pngx-sidebar-width);
|
||||||
|
transition: all .2s ease;
|
||||||
|
|
||||||
.sidebar-heading .spinner-border {
|
.sidebar-heading .spinner-border {
|
||||||
width: 0.8em;
|
width: 0.8em;
|
||||||
@ -37,8 +35,6 @@
|
|||||||
@media (min-width: 2400px) {
|
@media (min-width: 2400px) {
|
||||||
--pngx-sidebar-width: 8.33333333%;
|
--pngx-sidebar-width: 8.33333333%;
|
||||||
}
|
}
|
||||||
|
|
||||||
transition: all .2s ease;
|
|
||||||
}
|
}
|
||||||
@media (max-width: 767.98px) {
|
@media (max-width: 767.98px) {
|
||||||
.sidebar {
|
.sidebar {
|
||||||
@ -48,6 +44,13 @@
|
|||||||
|
|
||||||
main {
|
main {
|
||||||
transition: all .2s ease;
|
transition: all .2s ease;
|
||||||
|
padding-top: 110px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
main {
|
||||||
|
padding-top: 56px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-slim-toggler {
|
.sidebar-slim-toggler {
|
||||||
|
@ -1,43 +1,43 @@
|
|||||||
|
import { CdkDragDrop, DragDropModule } from '@angular/cdk/drag-drop'
|
||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import {
|
import {
|
||||||
HttpTestingController,
|
HttpTestingController,
|
||||||
provideHttpClientTesting,
|
provideHttpClientTesting,
|
||||||
} from '@angular/common/http/testing'
|
} from '@angular/common/http/testing'
|
||||||
import { AppFrameComponent } from './app-frame.component'
|
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
TestBed,
|
TestBed,
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import { NgbModal, NgbModalModule, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { BrowserModule } from '@angular/platform-browser'
|
|
||||||
import { RouterTestingModule } from '@angular/router/testing'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
|
||||||
import { RemoteVersionService } from 'src/app/services/rest/remote-version.service'
|
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { BrowserModule } from '@angular/platform-browser'
|
||||||
|
import { ActivatedRoute, Router } from '@angular/router'
|
||||||
|
import { RouterTestingModule } from '@angular/router/testing'
|
||||||
|
import { NgbModal, NgbModalModule, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { of, throwError } from 'rxjs'
|
import { of, throwError } from 'rxjs'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
import { routes } from 'src/app/app-routing.module'
|
||||||
|
import { SavedView } from 'src/app/data/saved-view'
|
||||||
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
import {
|
import {
|
||||||
DjangoMessageLevel,
|
DjangoMessageLevel,
|
||||||
DjangoMessagesService,
|
DjangoMessagesService,
|
||||||
} from 'src/app/services/django-messages.service'
|
} from 'src/app/services/django-messages.service'
|
||||||
import { environment } from 'src/environments/environment'
|
|
||||||
import { OpenDocumentsService } from 'src/app/services/open-documents.service'
|
import { OpenDocumentsService } from 'src/app/services/open-documents.service'
|
||||||
import { ActivatedRoute, Router } from '@angular/router'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { DocumentDetailComponent } from '../document-detail/document-detail.component'
|
import { RemoteVersionService } from 'src/app/services/rest/remote-version.service'
|
||||||
|
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
||||||
import { SearchService } from 'src/app/services/rest/search.service'
|
import { SearchService } from 'src/app/services/rest/search.service'
|
||||||
import { routes } from 'src/app/app-routing.module'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
import { CdkDragDrop, DragDropModule } from '@angular/cdk/drag-drop'
|
import { environment } from 'src/environments/environment'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
|
||||||
import { ProfileEditDialogComponent } from '../common/profile-edit-dialog/profile-edit-dialog.component'
|
import { ProfileEditDialogComponent } from '../common/profile-edit-dialog/profile-edit-dialog.component'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { DocumentDetailComponent } from '../document-detail/document-detail.component'
|
||||||
|
import { AppFrameComponent } from './app-frame.component'
|
||||||
import { GlobalSearchComponent } from './global-search/global-search.component'
|
import { GlobalSearchComponent } from './global-search/global-search.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const saved_views = [
|
const saved_views = [
|
||||||
{
|
{
|
||||||
@ -95,11 +95,6 @@ describe('AppFrameComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
AppFrameComponent,
|
|
||||||
IfPermissionsDirective,
|
|
||||||
GlobalSearchComponent,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
RouterTestingModule.withRoutes(routes),
|
RouterTestingModule.withRoutes(routes),
|
||||||
@ -109,6 +104,9 @@ describe('AppFrameComponent', () => {
|
|||||||
DragDropModule,
|
DragDropModule,
|
||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
AppFrameComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
GlobalSearchComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
SettingsService,
|
SettingsService,
|
||||||
@ -343,6 +341,7 @@ describe('AppFrameComponent', () => {
|
|||||||
component.editProfile()
|
component.editProfile()
|
||||||
expect(modalSpy).toHaveBeenCalledWith(ProfileEditDialogComponent, {
|
expect(modalSpy).toHaveBeenCalledWith(ProfileEditDialogComponent, {
|
||||||
backdrop: 'static',
|
backdrop: 'static',
|
||||||
|
size: 'xl',
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -1,45 +1,72 @@
|
|||||||
|
import {
|
||||||
|
CdkDragDrop,
|
||||||
|
CdkDragEnd,
|
||||||
|
CdkDragStart,
|
||||||
|
DragDropModule,
|
||||||
|
moveItemInArray,
|
||||||
|
} from '@angular/cdk/drag-drop'
|
||||||
|
import { NgClass } from '@angular/common'
|
||||||
import { Component, HostListener, OnInit } from '@angular/core'
|
import { Component, HostListener, OnInit } from '@angular/core'
|
||||||
import { ActivatedRoute, Router } from '@angular/router'
|
import { ActivatedRoute, Router, RouterModule } from '@angular/router'
|
||||||
|
import {
|
||||||
|
NgbCollapseModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbModal,
|
||||||
|
NgbNavModule,
|
||||||
|
NgbPopoverModule,
|
||||||
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { TourNgBootstrapModule } from 'ngx-ui-tour-ng-bootstrap'
|
||||||
import { Observable } from 'rxjs'
|
import { Observable } from 'rxjs'
|
||||||
import { first } from 'rxjs/operators'
|
import { first } from 'rxjs/operators'
|
||||||
import { Document } from 'src/app/data/document'
|
import { Document } from 'src/app/data/document'
|
||||||
import { OpenDocumentsService } from 'src/app/services/open-documents.service'
|
import { SavedView } from 'src/app/data/saved-view'
|
||||||
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { ComponentCanDeactivate } from 'src/app/guards/dirty-doc.guard'
|
||||||
|
import { DocumentTitlePipe } from 'src/app/pipes/document-title.pipe'
|
||||||
import {
|
import {
|
||||||
DjangoMessageLevel,
|
DjangoMessageLevel,
|
||||||
DjangoMessagesService,
|
DjangoMessagesService,
|
||||||
} from 'src/app/services/django-messages.service'
|
} from 'src/app/services/django-messages.service'
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
import { OpenDocumentsService } from 'src/app/services/open-documents.service'
|
||||||
import { environment } from 'src/environments/environment'
|
|
||||||
import { DocumentDetailComponent } from '../document-detail/document-detail.component'
|
|
||||||
import {
|
|
||||||
RemoteVersionService,
|
|
||||||
AppRemoteVersion,
|
|
||||||
} from 'src/app/services/rest/remote-version.service'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
import { TasksService } from 'src/app/services/tasks.service'
|
|
||||||
import { ComponentCanDeactivate } from 'src/app/guards/dirty-doc.guard'
|
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { ComponentWithPermissions } from '../with-permissions/with-permissions.component'
|
|
||||||
import {
|
import {
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
PermissionsService,
|
PermissionsService,
|
||||||
PermissionType,
|
PermissionType,
|
||||||
} from 'src/app/services/permissions.service'
|
} from 'src/app/services/permissions.service'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
|
||||||
import {
|
import {
|
||||||
CdkDragStart,
|
AppRemoteVersion,
|
||||||
CdkDragEnd,
|
RemoteVersionService,
|
||||||
CdkDragDrop,
|
} from 'src/app/services/rest/remote-version.service'
|
||||||
moveItemInArray,
|
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
||||||
} from '@angular/cdk/drag-drop'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { TasksService } from 'src/app/services/tasks.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { environment } from 'src/environments/environment'
|
||||||
import { ProfileEditDialogComponent } from '../common/profile-edit-dialog/profile-edit-dialog.component'
|
import { ProfileEditDialogComponent } from '../common/profile-edit-dialog/profile-edit-dialog.component'
|
||||||
|
import { DocumentDetailComponent } from '../document-detail/document-detail.component'
|
||||||
|
import { ComponentWithPermissions } from '../with-permissions/with-permissions.component'
|
||||||
|
import { GlobalSearchComponent } from './global-search/global-search.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-app-frame',
|
selector: 'pngx-app-frame',
|
||||||
templateUrl: './app-frame.component.html',
|
templateUrl: './app-frame.component.html',
|
||||||
styleUrls: ['./app-frame.component.scss'],
|
styleUrls: ['./app-frame.component.scss'],
|
||||||
|
imports: [
|
||||||
|
GlobalSearchComponent,
|
||||||
|
DocumentTitlePipe,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
RouterModule,
|
||||||
|
NgClass,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbPopoverModule,
|
||||||
|
NgbCollapseModule,
|
||||||
|
NgbNavModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
DragDropModule,
|
||||||
|
TourNgBootstrapModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class AppFrameComponent
|
export class AppFrameComponent
|
||||||
extends ComponentWithPermissions
|
extends ComponentWithPermissions
|
||||||
@ -81,7 +108,14 @@ export class AppFrameComponent
|
|||||||
if (this.settingsService.get(SETTINGS_KEYS.UPDATE_CHECKING_ENABLED)) {
|
if (this.settingsService.get(SETTINGS_KEYS.UPDATE_CHECKING_ENABLED)) {
|
||||||
this.checkForUpdates()
|
this.checkForUpdates()
|
||||||
}
|
}
|
||||||
this.tasksService.reload()
|
if (
|
||||||
|
this.permissionsService.currentUserCan(
|
||||||
|
PermissionAction.View,
|
||||||
|
PermissionType.PaperlessTask
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
this.tasksService.reload()
|
||||||
|
}
|
||||||
|
|
||||||
this.djangoMessagesService.get().forEach((message) => {
|
this.djangoMessagesService.get().forEach((message) => {
|
||||||
switch (message.level) {
|
switch (message.level) {
|
||||||
@ -136,6 +170,7 @@ export class AppFrameComponent
|
|||||||
editProfile() {
|
editProfile() {
|
||||||
this.modalService.open(ProfileEditDialogComponent, {
|
this.modalService.open(ProfileEditDialogComponent, {
|
||||||
backdrop: 'static',
|
backdrop: 'static',
|
||||||
|
size: 'xl',
|
||||||
})
|
})
|
||||||
this.closeMenu()
|
this.closeMenu()
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
[disabled]="disablePrimaryButton(type, item)"
|
[disabled]="disablePrimaryButton(type, item)"
|
||||||
(mouseenter)="onButtonHover($event)">
|
(mouseenter)="onButtonHover($event)">
|
||||||
@if (type === DataType.Document) {
|
@if (type === DataType.Document) {
|
||||||
<i-bs width="1em" height="1em" name="box-arrow-in-right"></i-bs>
|
<i-bs width="1em" height="1em" name="file-earmark-richtext"></i-bs>
|
||||||
<span> <ng-container i18n>Open</ng-container></span>
|
<span> <ng-container i18n>Open</ng-container></span>
|
||||||
} @else if (type === DataType.SavedView) {
|
} @else if (type === DataType.SavedView) {
|
||||||
<i-bs width="1em" height="1em" name="eye"></i-bs>
|
<i-bs width="1em" height="1em" name="eye"></i-bs>
|
||||||
@ -72,7 +72,7 @@
|
|||||||
<i-bs width="1em" height="1em" name="download"></i-bs>
|
<i-bs width="1em" height="1em" name="download"></i-bs>
|
||||||
<span> <ng-container i18n>Download</ng-container></span>
|
<span> <ng-container i18n>Download</ng-container></span>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs width="1em" height="1em" name="box-arrow-in-right"></i-bs>
|
<i-bs width="1em" height="1em" name="file-earmark-richtext"></i-bs>
|
||||||
<span> <ng-container i18n>Open</ng-container></span>
|
<span> <ng-container i18n>Open</ng-container></span>
|
||||||
}
|
}
|
||||||
</button>
|
</button>
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ElementRef } from '@angular/core'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
TestBed,
|
TestBed,
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import { GlobalSearchComponent } from './global-search.component'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { of } from 'rxjs'
|
|
||||||
import { SearchService } from 'src/app/services/rest/search.service'
|
|
||||||
import { Router } from '@angular/router'
|
import { Router } from '@angular/router'
|
||||||
import {
|
import {
|
||||||
NgbDropdownModule,
|
NgbDropdownModule,
|
||||||
@ -14,11 +15,9 @@ import {
|
|||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
NgbModalRef,
|
NgbModalRef,
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
import { of } from 'rxjs'
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
import { DataType } from 'src/app/data/datatype'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
|
||||||
import {
|
import {
|
||||||
FILTER_FULLTEXT_QUERY,
|
FILTER_FULLTEXT_QUERY,
|
||||||
FILTER_HAS_CORRESPONDENT_ANY,
|
FILTER_HAS_CORRESPONDENT_ANY,
|
||||||
@ -27,20 +26,21 @@ import {
|
|||||||
FILTER_HAS_TAGS_ALL,
|
FILTER_HAS_TAGS_ALL,
|
||||||
FILTER_TITLE_CONTENT,
|
FILTER_TITLE_CONTENT,
|
||||||
} from 'src/app/data/filter-rule-type'
|
} from 'src/app/data/filter-rule-type'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
|
||||||
import { MailRuleEditDialogComponent } from '../../common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component'
|
|
||||||
import { MailAccountEditDialogComponent } from '../../common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component'
|
|
||||||
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
|
||||||
import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
|
||||||
import { WorkflowEditDialogComponent } from '../../common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
|
|
||||||
import { ElementRef } from '@angular/core'
|
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { DataType } from 'src/app/data/datatype'
|
|
||||||
import { queryParamsFromFilterRules } from 'src/app/utils/query-params'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
||||||
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { SearchService } from 'src/app/services/rest/search.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { queryParamsFromFilterRules } from 'src/app/utils/query-params'
|
||||||
|
import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component'
|
||||||
|
import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
||||||
|
import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component'
|
||||||
|
import { MailAccountEditDialogComponent } from '../../common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component'
|
||||||
|
import { MailRuleEditDialogComponent } from '../../common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component'
|
||||||
|
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
||||||
|
import { WorkflowEditDialogComponent } from '../../common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
|
||||||
|
import { GlobalSearchComponent } from './global-search.component'
|
||||||
|
|
||||||
const searchResults = {
|
const searchResults = {
|
||||||
total: 11,
|
total: 11,
|
||||||
@ -138,13 +138,13 @@ describe('GlobalSearchComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [GlobalSearchComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
NgbDropdownModule,
|
NgbDropdownModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
GlobalSearchComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
@ -1,14 +1,23 @@
|
|||||||
|
import { NgTemplateOutlet } from '@angular/common'
|
||||||
import {
|
import {
|
||||||
Component,
|
Component,
|
||||||
ViewChild,
|
|
||||||
ElementRef,
|
ElementRef,
|
||||||
ViewChildren,
|
|
||||||
QueryList,
|
|
||||||
OnInit,
|
OnInit,
|
||||||
|
QueryList,
|
||||||
|
ViewChild,
|
||||||
|
ViewChildren,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { Router } from '@angular/router'
|
import { Router } from '@angular/router'
|
||||||
import { NgbDropdown, NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'
|
import {
|
||||||
import { Subject, debounceTime, distinctUntilChanged, filter, map } from 'rxjs'
|
NgbDropdown,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgbModal,
|
||||||
|
NgbModalRef,
|
||||||
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { Subject, debounceTime, distinctUntilChanged, filter } from 'rxjs'
|
||||||
|
import { DataType } from 'src/app/data/datatype'
|
||||||
import {
|
import {
|
||||||
FILTER_FULLTEXT_QUERY,
|
FILTER_FULLTEXT_QUERY,
|
||||||
FILTER_HAS_CORRESPONDENT_ANY,
|
FILTER_HAS_CORRESPONDENT_ANY,
|
||||||
@ -17,19 +26,23 @@ import {
|
|||||||
FILTER_HAS_TAGS_ALL,
|
FILTER_HAS_TAGS_ALL,
|
||||||
FILTER_TITLE_CONTENT,
|
FILTER_TITLE_CONTENT,
|
||||||
} from 'src/app/data/filter-rule-type'
|
} from 'src/app/data/filter-rule-type'
|
||||||
import { DataType } from 'src/app/data/datatype'
|
|
||||||
import { ObjectWithId } from 'src/app/data/object-with-id'
|
import { ObjectWithId } from 'src/app/data/object-with-id'
|
||||||
|
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
||||||
|
import { HotKeyService } from 'src/app/services/hot-key.service'
|
||||||
import {
|
import {
|
||||||
PermissionsService,
|
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
|
PermissionsService,
|
||||||
} from 'src/app/services/permissions.service'
|
} from 'src/app/services/permissions.service'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
import {
|
import {
|
||||||
GlobalSearchResult,
|
GlobalSearchResult,
|
||||||
SearchService,
|
SearchService,
|
||||||
} from 'src/app/services/rest/search.service'
|
} from 'src/app/services/rest/search.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { paramsFromViewState } from 'src/app/utils/query-params'
|
||||||
import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component'
|
import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component'
|
||||||
import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
import { CustomFieldEditDialogComponent } from '../../common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
||||||
import { DocumentTypeEditDialogComponent } from '../../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component'
|
import { DocumentTypeEditDialogComponent } from '../../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component'
|
||||||
@ -41,15 +54,19 @@ import { StoragePathEditDialogComponent } from '../../common/edit-dialog/storage
|
|||||||
import { TagEditDialogComponent } from '../../common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component'
|
import { TagEditDialogComponent } from '../../common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component'
|
||||||
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component'
|
||||||
import { WorkflowEditDialogComponent } from '../../common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
|
import { WorkflowEditDialogComponent } from '../../common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
|
||||||
import { HotKeyService } from 'src/app/services/hot-key.service'
|
|
||||||
import { paramsFromViewState } from 'src/app/utils/query-params'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
|
||||||
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-global-search',
|
selector: 'pngx-global-search',
|
||||||
templateUrl: './global-search.component.html',
|
templateUrl: './global-search.component.html',
|
||||||
styleUrl: './global-search.component.scss',
|
styleUrl: './global-search.component.scss',
|
||||||
|
imports: [
|
||||||
|
CustomDatePipe,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgTemplateOutlet,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class GlobalSearchComponent implements OnInit {
|
export class GlobalSearchComponent implements OnInit {
|
||||||
public DataType = DataType
|
public DataType = DataType
|
||||||
@ -89,7 +106,6 @@ export class GlobalSearchComponent implements OnInit {
|
|||||||
this.queryDebounce
|
this.queryDebounce
|
||||||
.pipe(
|
.pipe(
|
||||||
debounceTime(400),
|
debounceTime(400),
|
||||||
map((query) => query?.trim()),
|
|
||||||
filter((query) => !query?.length || query?.length > 2),
|
filter((query) => !query?.length || query?.length > 2),
|
||||||
distinctUntilChanged()
|
distinctUntilChanged()
|
||||||
)
|
)
|
||||||
@ -109,7 +125,7 @@ export class GlobalSearchComponent implements OnInit {
|
|||||||
|
|
||||||
private search(query: string) {
|
private search(query: string) {
|
||||||
this.loading = true
|
this.loading = true
|
||||||
this.searchService.globalSearch(query).subscribe((results) => {
|
this.searchService.globalSearch(query.trim()).subscribe((results) => {
|
||||||
this.searchResults = results
|
this.searchResults = results
|
||||||
this.loading = false
|
this.loading = false
|
||||||
this.resultsDropdown.open()
|
this.resultsDropdown.open()
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { ClearableBadgeComponent } from './clearable-badge.component'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { ClearableBadgeComponent } from './clearable-badge.component'
|
||||||
|
|
||||||
describe('ClearableBadgeComponent', () => {
|
describe('ClearableBadgeComponent', () => {
|
||||||
let component: ClearableBadgeComponent
|
let component: ClearableBadgeComponent
|
||||||
@ -8,8 +8,10 @@ describe('ClearableBadgeComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [ClearableBadgeComponent],
|
imports: [
|
||||||
imports: [NgxBootstrapIconsModule.pick(allIcons)],
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
ClearableBadgeComponent,
|
||||||
|
],
|
||||||
}).compileComponents()
|
}).compileComponents()
|
||||||
|
|
||||||
fixture = TestBed.createComponent(ClearableBadgeComponent)
|
fixture = TestBed.createComponent(ClearableBadgeComponent)
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
import { Component, Input, Output, EventEmitter } from '@angular/core'
|
import { Component, EventEmitter, Input, Output } from '@angular/core'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-clearable-badge',
|
selector: 'pngx-clearable-badge',
|
||||||
templateUrl: './clearable-badge.component.html',
|
templateUrl: './clearable-badge.component.html',
|
||||||
styleUrls: ['./clearable-badge.component.scss'],
|
styleUrls: ['./clearable-badge.component.scss'],
|
||||||
|
imports: [NgxBootstrapIconsModule],
|
||||||
})
|
})
|
||||||
export class ClearableBadgeComponent {
|
export class ClearableBadgeComponent {
|
||||||
constructor() {}
|
constructor() {}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
|
||||||
import { ConfirmButtonComponent } from './confirm-button.component'
|
|
||||||
import { NgbPopoverModule } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbPopoverModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { ConfirmButtonComponent } from './confirm-button.component'
|
||||||
|
|
||||||
describe('ConfirmButtonComponent', () => {
|
describe('ConfirmButtonComponent', () => {
|
||||||
let component: ConfirmButtonComponent
|
let component: ConfirmButtonComponent
|
||||||
@ -10,8 +10,11 @@ describe('ConfirmButtonComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [ConfirmButtonComponent],
|
imports: [
|
||||||
imports: [NgbPopoverModule, NgxBootstrapIconsModule.pick(allIcons)],
|
NgbPopoverModule,
|
||||||
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
ConfirmButtonComponent,
|
||||||
|
],
|
||||||
}).compileComponents()
|
}).compileComponents()
|
||||||
|
|
||||||
fixture = TestBed.createComponent(ConfirmButtonComponent)
|
fixture = TestBed.createComponent(ConfirmButtonComponent)
|
||||||
|
@ -5,12 +5,14 @@ import {
|
|||||||
Output,
|
Output,
|
||||||
ViewChild,
|
ViewChild,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { NgbPopover } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbPopover, NgbPopoverModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-confirm-button',
|
selector: 'pngx-confirm-button',
|
||||||
templateUrl: './confirm-button.component.html',
|
templateUrl: './confirm-button.component.html',
|
||||||
styleUrl: './confirm-button.component.scss',
|
styleUrl: './confirm-button.component.scss',
|
||||||
|
imports: [NgbPopoverModule, NgxBootstrapIconsModule],
|
||||||
})
|
})
|
||||||
export class ConfirmButtonComponent {
|
export class ConfirmButtonComponent {
|
||||||
@Input()
|
@Input()
|
||||||
|
@ -1,14 +1,8 @@
|
|||||||
import {
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
ComponentFixture,
|
|
||||||
TestBed,
|
|
||||||
discardPeriodicTasks,
|
|
||||||
fakeAsync,
|
|
||||||
tick,
|
|
||||||
} from '@angular/core/testing'
|
|
||||||
import { ConfirmDialogComponent } from './confirm-dialog.component'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
|
||||||
import { Subject } from 'rxjs'
|
import { Subject } from 'rxjs'
|
||||||
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
|
import { ConfirmDialogComponent } from './confirm-dialog.component'
|
||||||
|
|
||||||
describe('ConfirmDialogComponent', () => {
|
describe('ConfirmDialogComponent', () => {
|
||||||
let component: ConfirmDialogComponent
|
let component: ConfirmDialogComponent
|
||||||
@ -17,9 +11,8 @@ describe('ConfirmDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [ConfirmDialogComponent, SafeHtmlPipe],
|
|
||||||
providers: [NgbActiveModal, SafeHtmlPipe],
|
providers: [NgbActiveModal, SafeHtmlPipe],
|
||||||
imports: [],
|
imports: [ConfirmDialogComponent, SafeHtmlPipe],
|
||||||
}).compileComponents()
|
}).compileComponents()
|
||||||
|
|
||||||
modal = TestBed.inject(NgbActiveModal)
|
modal = TestBed.inject(NgbActiveModal)
|
||||||
|
@ -1,14 +1,20 @@
|
|||||||
|
import { DecimalPipe } from '@angular/common'
|
||||||
import { Component, EventEmitter, Input, Output } from '@angular/core'
|
import { Component, EventEmitter, Input, Output } from '@angular/core'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { interval, Subject, take } from 'rxjs'
|
import { Subject } from 'rxjs'
|
||||||
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-confirm-dialog',
|
selector: 'pngx-confirm-dialog',
|
||||||
templateUrl: './confirm-dialog.component.html',
|
templateUrl: './confirm-dialog.component.html',
|
||||||
styleUrls: ['./confirm-dialog.component.scss'],
|
styleUrls: ['./confirm-dialog.component.scss'],
|
||||||
|
imports: [DecimalPipe, SafeHtmlPipe],
|
||||||
})
|
})
|
||||||
export class ConfirmDialogComponent {
|
export class ConfirmDialogComponent extends LoadingComponentWithPermissions {
|
||||||
constructor(public activeModal: NgbActiveModal) {}
|
constructor(public activeModal: NgbActiveModal) {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
@Output()
|
@Output()
|
||||||
public confirmClicked = new EventEmitter()
|
public confirmClicked = new EventEmitter()
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
.pdf-viewer-container {
|
.pdf-viewer-container {
|
||||||
background-color: gray;
|
background-color: gray;
|
||||||
height: 350px;
|
height: 550px;
|
||||||
|
|
||||||
pdf-viewer {
|
pdf-viewer {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { DeletePagesConfirmDialogComponent } from './delete-pages-confirm-dialog.component'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { DeletePagesConfirmDialogComponent } from './delete-pages-confirm-dialog.component'
|
||||||
import { PdfViewerComponent } from 'ng2-pdf-viewer'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
describe('DeletePagesConfirmDialogComponent', () => {
|
describe('DeletePagesConfirmDialogComponent', () => {
|
||||||
let component: DeletePagesConfirmDialogComponent
|
let component: DeletePagesConfirmDialogComponent
|
||||||
@ -14,11 +13,12 @@ describe('DeletePagesConfirmDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [DeletePagesConfirmDialogComponent, PdfViewerComponent],
|
declarations: [],
|
||||||
imports: [
|
imports: [
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
|
DeletePagesConfirmDialogComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
|
@ -1,13 +1,20 @@
|
|||||||
import { Component, TemplateRef, ViewChild } from '@angular/core'
|
import { Component, TemplateRef, ViewChild } from '@angular/core'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import {
|
||||||
|
PDFDocumentProxy,
|
||||||
|
PdfViewerComponent,
|
||||||
|
PdfViewerModule,
|
||||||
|
} from 'ng2-pdf-viewer'
|
||||||
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
||||||
import { PDFDocumentProxy, PdfViewerComponent } from 'ng2-pdf-viewer'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-delete-pages-confirm-dialog',
|
selector: 'pngx-delete-pages-confirm-dialog',
|
||||||
templateUrl: './delete-pages-confirm-dialog.component.html',
|
templateUrl: './delete-pages-confirm-dialog.component.html',
|
||||||
styleUrl: './delete-pages-confirm-dialog.component.scss',
|
styleUrl: './delete-pages-confirm-dialog.component.scss',
|
||||||
|
imports: [PdfViewerModule, FormsModule, ReactiveFormsModule, SafeHtmlPipe],
|
||||||
})
|
})
|
||||||
export class DeletePagesConfirmDialogComponent extends ConfirmDialogComponent {
|
export class DeletePagesConfirmDialogComponent extends ConfirmDialogComponent {
|
||||||
public documentID: number
|
public documentID: number
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { MergeConfirmDialogComponent } from './merge-confirm-dialog.component'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { of } from 'rxjs'
|
import { of } from 'rxjs'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { MergeConfirmDialogComponent } from './merge-confirm-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
describe('MergeConfirmDialogComponent', () => {
|
describe('MergeConfirmDialogComponent', () => {
|
||||||
let component: MergeConfirmDialogComponent
|
let component: MergeConfirmDialogComponent
|
||||||
@ -15,11 +15,11 @@ describe('MergeConfirmDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [MergeConfirmDialogComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
|
MergeConfirmDialogComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
|
@ -1,16 +1,19 @@
|
|||||||
import { Component, OnInit } from '@angular/core'
|
|
||||||
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
|
||||||
import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop'
|
import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop'
|
||||||
import { Subject, takeUntil } from 'rxjs'
|
import { Component, OnInit } from '@angular/core'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { takeUntil } from 'rxjs'
|
||||||
import { Document } from 'src/app/data/document'
|
import { Document } from 'src/app/data/document'
|
||||||
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-merge-confirm-dialog',
|
selector: 'pngx-merge-confirm-dialog',
|
||||||
templateUrl: './merge-confirm-dialog.component.html',
|
templateUrl: './merge-confirm-dialog.component.html',
|
||||||
styleUrl: './merge-confirm-dialog.component.scss',
|
styleUrl: './merge-confirm-dialog.component.scss',
|
||||||
|
imports: [FormsModule, ReactiveFormsModule, NgxBootstrapIconsModule],
|
||||||
})
|
})
|
||||||
export class MergeConfirmDialogComponent
|
export class MergeConfirmDialogComponent
|
||||||
extends ConfirmDialogComponent
|
extends ConfirmDialogComponent
|
||||||
@ -25,8 +28,6 @@ export class MergeConfirmDialogComponent
|
|||||||
|
|
||||||
public metadataDocumentID: number = -1
|
public metadataDocumentID: number = -1
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
activeModal: NgbActiveModal,
|
activeModal: NgbActiveModal,
|
||||||
private documentService: DocumentService,
|
private documentService: DocumentService,
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
|
||||||
import { RotateConfirmDialogComponent } from './rotate-confirm-dialog.component'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
|
import { RotateConfirmDialogComponent } from './rotate-confirm-dialog.component'
|
||||||
|
|
||||||
describe('RotateConfirmDialogComponent', () => {
|
describe('RotateConfirmDialogComponent', () => {
|
||||||
let component: RotateConfirmDialogComponent
|
let component: RotateConfirmDialogComponent
|
||||||
@ -12,8 +12,11 @@ describe('RotateConfirmDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [RotateConfirmDialogComponent, SafeHtmlPipe],
|
imports: [
|
||||||
imports: [NgxBootstrapIconsModule.pick(allIcons)],
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
RotateConfirmDialogComponent,
|
||||||
|
SafeHtmlPipe,
|
||||||
|
],
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
SafeHtmlPipe,
|
SafeHtmlPipe,
|
||||||
|
@ -1,12 +1,16 @@
|
|||||||
|
import { NgStyle } from '@angular/common'
|
||||||
import { Component } from '@angular/core'
|
import { Component } from '@angular/core'
|
||||||
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-rotate-confirm-dialog',
|
selector: 'pngx-rotate-confirm-dialog',
|
||||||
templateUrl: './rotate-confirm-dialog.component.html',
|
templateUrl: './rotate-confirm-dialog.component.html',
|
||||||
styleUrl: './rotate-confirm-dialog.component.scss',
|
styleUrl: './rotate-confirm-dialog.component.scss',
|
||||||
|
imports: [NgStyle, NgxBootstrapIconsModule, SafeHtmlPipe],
|
||||||
})
|
})
|
||||||
export class RotateConfirmDialogComponent extends ConfirmDialogComponent {
|
export class RotateConfirmDialogComponent extends ConfirmDialogComponent {
|
||||||
public documentID: number
|
public documentID: number
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<p>{{message}}</p>
|
<p>{{message}}</p>
|
||||||
<div class="row mb-2">
|
<div class="row mb-2">
|
||||||
<div class="col-8">
|
<div class="col-7">
|
||||||
<div class="input-group input-group-sm">
|
<div class="input-group input-group-sm">
|
||||||
<div class="input-group-text" i18n>Page</div>
|
<div class="input-group-text" i18n>Page</div>
|
||||||
<input class="form-control" type="number" min="1" [(ngModel)]="page" />
|
<input class="form-control" type="number" min="1" [(ngModel)]="page" />
|
||||||
@ -21,7 +21,7 @@
|
|||||||
</pdf-viewer>
|
</pdf-viewer>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4">
|
<div class="col-5">
|
||||||
<div class="d-grid">
|
<div class="d-grid">
|
||||||
<button class="btn btn-sm btn-primary" (click)="addSplit()" [disabled]="!canSplit">
|
<button class="btn btn-sm btn-primary" (click)="addSplit()" [disabled]="!canSplit">
|
||||||
<i-bs name="plus-circle"></i-bs>
|
<i-bs name="plus-circle"></i-bs>
|
||||||
@ -44,12 +44,12 @@
|
|||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-check form-switch mt-4">
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="form-check form-switch me-auto">
|
||||||
<input class="form-check-input" type="checkbox" role="switch" id="deleteOriginalSwitch" [(ngModel)]="deleteOriginal" [disabled]="!userOwnsDocument">
|
<input class="form-check-input" type="checkbox" role="switch" id="deleteOriginalSwitch" [(ngModel)]="deleteOriginal" [disabled]="!userOwnsDocument">
|
||||||
<label class="form-check-label" for="deleteOriginalSwitch" i18n>Delete original document after successful split</label>
|
<label class="form-check-label" for="deleteOriginalSwitch" i18n>Delete original document after successful split</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
<div class="modal-footer">
|
|
||||||
<button type="button" class="btn" [class]="cancelBtnClass" (click)="cancel()" [disabled]="!buttonsEnabled">
|
<button type="button" class="btn" [class]="cancelBtnClass" (click)="cancel()" [disabled]="!buttonsEnabled">
|
||||||
<span class="d-inline-block" style="padding-bottom: 1px;">{{cancelBtnCaption}}</span>
|
<span class="d-inline-block" style="padding-bottom: 1px;">{{cancelBtnCaption}}</span>
|
||||||
</button>
|
</button>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
.pdf-viewer-container {
|
.pdf-viewer-container {
|
||||||
background-color: gray;
|
background-color: gray;
|
||||||
height: 350px;
|
height: 500px;
|
||||||
|
|
||||||
pdf-viewer {
|
pdf-viewer {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
|
||||||
import { SplitConfirmDialogComponent } from './split-confirm-dialog.component'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { ReactiveFormsModule, FormsModule } from '@angular/forms'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
|
||||||
import { PdfViewerModule } from 'ng2-pdf-viewer'
|
|
||||||
import { of } from 'rxjs'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { PdfViewerModule } from 'ng2-pdf-viewer'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { of } from 'rxjs'
|
||||||
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { SplitConfirmDialogComponent } from './split-confirm-dialog.component'
|
||||||
|
|
||||||
describe('SplitConfirmDialogComponent', () => {
|
describe('SplitConfirmDialogComponent', () => {
|
||||||
let component: SplitConfirmDialogComponent
|
let component: SplitConfirmDialogComponent
|
||||||
@ -17,12 +17,12 @@ describe('SplitConfirmDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [SplitConfirmDialogComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
PdfViewerModule,
|
PdfViewerModule,
|
||||||
|
SplitConfirmDialogComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
|
@ -1,15 +1,23 @@
|
|||||||
import { Component, OnInit } from '@angular/core'
|
import { Component, OnInit } from '@angular/core'
|
||||||
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { Document } from 'src/app/data/document'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { PDFDocumentProxy, PdfViewerModule } from 'ng2-pdf-viewer'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { Document } from 'src/app/data/document'
|
||||||
import { PermissionsService } from 'src/app/services/permissions.service'
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
import { PDFDocumentProxy } from 'ng2-pdf-viewer'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { ConfirmDialogComponent } from '../confirm-dialog.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-split-confirm-dialog',
|
selector: 'pngx-split-confirm-dialog',
|
||||||
templateUrl: './split-confirm-dialog.component.html',
|
templateUrl: './split-confirm-dialog.component.html',
|
||||||
styleUrl: './split-confirm-dialog.component.scss',
|
styleUrl: './split-confirm-dialog.component.scss',
|
||||||
|
imports: [
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
PdfViewerModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class SplitConfirmDialogComponent
|
export class SplitConfirmDialogComponent
|
||||||
extends ConfirmDialogComponent
|
extends ConfirmDialogComponent
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { of } from 'rxjs'
|
import { of } from 'rxjs'
|
||||||
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
|
import { DisplayField, Document } from 'src/app/data/document'
|
||||||
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
import { CustomFieldDisplayComponent } from './custom-field-display.component'
|
import { CustomFieldDisplayComponent } from './custom-field-display.component'
|
||||||
import { DisplayField, Document } from 'src/app/data/document'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
const customFields: CustomField[] = [
|
const customFields: CustomField[] = [
|
||||||
{ id: 1, name: 'Field 1', data_type: CustomFieldDataType.String },
|
{ id: 1, name: 'Field 1', data_type: CustomFieldDataType.String },
|
||||||
@ -17,7 +17,11 @@ const customFields: CustomField[] = [
|
|||||||
name: 'Field 4',
|
name: 'Field 4',
|
||||||
data_type: CustomFieldDataType.Select,
|
data_type: CustomFieldDataType.Select,
|
||||||
extra_data: {
|
extra_data: {
|
||||||
select_options: ['Option 1', 'Option 2', 'Option 3'],
|
select_options: [
|
||||||
|
{ label: 'Option 1', id: 'abc-123' },
|
||||||
|
{ label: 'Option 2', id: 'def-456' },
|
||||||
|
{ label: 'Option 3', id: 'ghi-789' },
|
||||||
|
],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -45,8 +49,7 @@ describe('CustomFieldDisplayComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [CustomFieldDisplayComponent],
|
imports: [CustomFieldDisplayComponent],
|
||||||
imports: [],
|
|
||||||
providers: [
|
providers: [
|
||||||
DocumentService,
|
DocumentService,
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
@ -131,6 +134,8 @@ describe('CustomFieldDisplayComponent', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
it('should show select value', () => {
|
it('should show select value', () => {
|
||||||
expect(component.getSelectValue(customFields[3], 2)).toEqual('Option 3')
|
expect(component.getSelectValue(customFields[3], 'ghi-789')).toEqual(
|
||||||
|
'Option 3'
|
||||||
|
)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -1,25 +1,24 @@
|
|||||||
import { getLocaleCurrencyCode } from '@angular/common'
|
import { CurrencyPipe, getLocaleCurrencyCode } from '@angular/common'
|
||||||
import {
|
import { Component, Inject, Input, LOCALE_ID, OnInit } from '@angular/core'
|
||||||
Component,
|
import { takeUntil } from 'rxjs'
|
||||||
Inject,
|
|
||||||
Input,
|
|
||||||
LOCALE_ID,
|
|
||||||
OnDestroy,
|
|
||||||
OnInit,
|
|
||||||
} from '@angular/core'
|
|
||||||
import { Subject, takeUntil } from 'rxjs'
|
|
||||||
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
import { DisplayField, Document } from 'src/app/data/document'
|
import { DisplayField, Document } from 'src/app/data/document'
|
||||||
import { Results } from 'src/app/data/results'
|
import { Results } from 'src/app/data/results'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
import { DocumentService } from 'src/app/services/rest/document.service'
|
import { DocumentService } from 'src/app/services/rest/document.service'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-custom-field-display',
|
selector: 'pngx-custom-field-display',
|
||||||
templateUrl: './custom-field-display.component.html',
|
templateUrl: './custom-field-display.component.html',
|
||||||
styleUrl: './custom-field-display.component.scss',
|
styleUrl: './custom-field-display.component.scss',
|
||||||
|
imports: [CustomDatePipe, CurrencyPipe],
|
||||||
})
|
})
|
||||||
export class CustomFieldDisplayComponent implements OnInit, OnDestroy {
|
export class CustomFieldDisplayComponent
|
||||||
|
extends LoadingComponentWithPermissions
|
||||||
|
implements OnInit
|
||||||
|
{
|
||||||
CustomFieldDataType = CustomFieldDataType
|
CustomFieldDataType = CustomFieldDataType
|
||||||
|
|
||||||
private _document: Document
|
private _document: Document
|
||||||
@ -61,7 +60,6 @@ export class CustomFieldDisplayComponent implements OnInit, OnDestroy {
|
|||||||
|
|
||||||
private docLinkDocuments: Document[] = []
|
private docLinkDocuments: Document[] = []
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
private defaultCurrencyCode: any
|
private defaultCurrencyCode: any
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@ -69,6 +67,7 @@ export class CustomFieldDisplayComponent implements OnInit, OnDestroy {
|
|||||||
private documentService: DocumentService,
|
private documentService: DocumentService,
|
||||||
@Inject(LOCALE_ID) currentLocale: string
|
@Inject(LOCALE_ID) currentLocale: string
|
||||||
) {
|
) {
|
||||||
|
super()
|
||||||
this.defaultCurrencyCode = getLocaleCurrencyCode(currentLocale)
|
this.defaultCurrencyCode = getLocaleCurrencyCode(currentLocale)
|
||||||
this.customFieldService.listAll().subscribe((r) => {
|
this.customFieldService.listAll().subscribe((r) => {
|
||||||
this.customFields = r.results
|
this.customFields = r.results
|
||||||
@ -117,12 +116,7 @@ export class CustomFieldDisplayComponent implements OnInit, OnDestroy {
|
|||||||
return this.docLinkDocuments?.find((d) => d.id === docId)?.title
|
return this.docLinkDocuments?.find((d) => d.id === docId)?.title
|
||||||
}
|
}
|
||||||
|
|
||||||
public getSelectValue(field: CustomField, index: number): string {
|
public getSelectValue(field: CustomField, id: string): string {
|
||||||
return field.extra_data.select_options[index]
|
return field.extra_data.select_options?.find((o) => o.id === id)?.label
|
||||||
}
|
|
||||||
|
|
||||||
ngOnDestroy(): void {
|
|
||||||
this.unsubscribeNotifier.next(true)
|
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,28 +1,29 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
TestBed,
|
TestBed,
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import { CustomFieldsDropdownComponent } from './custom-fields-dropdown.component'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
|
||||||
import { of } from 'rxjs'
|
|
||||||
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
|
||||||
import { SelectComponent } from '../input/select/select.component'
|
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import { By } from '@angular/platform-browser'
|
||||||
import {
|
import {
|
||||||
NgbDropdownModule,
|
NgbDropdownModule,
|
||||||
NgbModal,
|
NgbModal,
|
||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
NgbModalRef,
|
NgbModalRef,
|
||||||
} from '@ng-bootstrap/ng-bootstrap'
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { CustomFieldEditDialogComponent } from '../edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
import { By } from '@angular/platform-browser'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { of } from 'rxjs'
|
||||||
|
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { CustomFieldEditDialogComponent } from '../edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
||||||
|
import { SelectComponent } from '../input/select/select.component'
|
||||||
|
import { CustomFieldsDropdownComponent } from './custom-fields-dropdown.component'
|
||||||
|
|
||||||
const fields: CustomField[] = [
|
const fields: CustomField[] = [
|
||||||
{
|
{
|
||||||
@ -43,10 +44,10 @@ describe('CustomFieldsDropdownComponent', () => {
|
|||||||
let customFieldService: CustomFieldsService
|
let customFieldService: CustomFieldsService
|
||||||
let toastService: ToastService
|
let toastService: ToastService
|
||||||
let modalService: NgbModal
|
let modalService: NgbModal
|
||||||
|
let settingsService: SettingsService
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [CustomFieldsDropdownComponent, SelectComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgSelectModule,
|
NgSelectModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
@ -54,6 +55,8 @@ describe('CustomFieldsDropdownComponent', () => {
|
|||||||
NgbModalModule,
|
NgbModalModule,
|
||||||
NgbDropdownModule,
|
NgbDropdownModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
CustomFieldsDropdownComponent,
|
||||||
|
SelectComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
@ -70,6 +73,8 @@ describe('CustomFieldsDropdownComponent', () => {
|
|||||||
results: fields.concat([]),
|
results: fields.concat([]),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
settingsService = TestBed.inject(SettingsService)
|
||||||
|
settingsService.currentUser = { id: 1, username: 'test' }
|
||||||
fixture = TestBed.createComponent(CustomFieldsDropdownComponent)
|
fixture = TestBed.createComponent(CustomFieldsDropdownComponent)
|
||||||
component = fixture.componentInstance
|
component = fixture.componentInstance
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
|
@ -3,31 +3,39 @@ import {
|
|||||||
ElementRef,
|
ElementRef,
|
||||||
EventEmitter,
|
EventEmitter,
|
||||||
Input,
|
Input,
|
||||||
OnDestroy,
|
|
||||||
Output,
|
Output,
|
||||||
QueryList,
|
QueryList,
|
||||||
ViewChild,
|
ViewChild,
|
||||||
ViewChildren,
|
ViewChildren,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { Subject, first, takeUntil } from 'rxjs'
|
import { NgbDropdownModule, NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { first, takeUntil } from 'rxjs'
|
||||||
import { CustomField, DATA_TYPE_LABELS } from 'src/app/data/custom-field'
|
import { CustomField, DATA_TYPE_LABELS } from 'src/app/data/custom-field'
|
||||||
import { CustomFieldInstance } from 'src/app/data/custom-field-instance'
|
import { CustomFieldInstance } from 'src/app/data/custom-field-instance'
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
|
||||||
import { ToastService } from 'src/app/services/toast.service'
|
|
||||||
import { CustomFieldEditDialogComponent } from '../edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
|
||||||
import {
|
import {
|
||||||
PermissionAction,
|
PermissionAction,
|
||||||
PermissionType,
|
PermissionType,
|
||||||
PermissionsService,
|
PermissionsService,
|
||||||
} from 'src/app/services/permissions.service'
|
} from 'src/app/services/permissions.service'
|
||||||
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
import { CustomFieldEditDialogComponent } from '../edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-custom-fields-dropdown',
|
selector: 'pngx-custom-fields-dropdown',
|
||||||
templateUrl: './custom-fields-dropdown.component.html',
|
templateUrl: './custom-fields-dropdown.component.html',
|
||||||
styleUrls: ['./custom-fields-dropdown.component.scss'],
|
styleUrls: ['./custom-fields-dropdown.component.scss'],
|
||||||
|
imports: [
|
||||||
|
NgbDropdownModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class CustomFieldsDropdownComponent implements OnDestroy {
|
export class CustomFieldsDropdownComponent extends LoadingComponentWithPermissions {
|
||||||
@Input()
|
@Input()
|
||||||
documentId: number
|
documentId: number
|
||||||
|
|
||||||
@ -60,8 +68,6 @@ export class CustomFieldsDropdownComponent implements OnDestroy {
|
|||||||
|
|
||||||
public filterText: string
|
public filterText: string
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
|
|
||||||
get canCreateFields(): boolean {
|
get canCreateFields(): boolean {
|
||||||
return this.permissionsService.currentUserCan(
|
return this.permissionsService.currentUserCan(
|
||||||
PermissionAction.Add,
|
PermissionAction.Add,
|
||||||
@ -75,14 +81,10 @@ export class CustomFieldsDropdownComponent implements OnDestroy {
|
|||||||
private toastService: ToastService,
|
private toastService: ToastService,
|
||||||
private permissionsService: PermissionsService
|
private permissionsService: PermissionsService
|
||||||
) {
|
) {
|
||||||
|
super()
|
||||||
this.getFields()
|
this.getFields()
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy(): void {
|
|
||||||
this.unsubscribeNotifier.next(this)
|
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
}
|
|
||||||
|
|
||||||
private getFields() {
|
private getFields() {
|
||||||
this.customFieldsService
|
this.customFieldsService
|
||||||
.listAll()
|
.listAll()
|
||||||
|
@ -44,6 +44,8 @@
|
|||||||
<ng-select #fieldSelects
|
<ng-select #fieldSelects
|
||||||
class="paperless-input-select rounded-end"
|
class="paperless-input-select rounded-end"
|
||||||
[items]="getSelectOptionsForField(atom.field)"
|
[items]="getSelectOptionsForField(atom.field)"
|
||||||
|
bindLabel="label"
|
||||||
|
bindValue="id"
|
||||||
[(ngModel)]="atom.value"
|
[(ngModel)]="atom.value"
|
||||||
[disabled]="disabled"
|
[disabled]="disabled"
|
||||||
(mousedown)="$event.stopImmediatePropagation()"
|
(mousedown)="$event.stopImmediatePropagation()"
|
||||||
@ -65,7 +67,9 @@
|
|||||||
(mousedown)="$event.stopImmediatePropagation()"
|
(mousedown)="$event.stopImmediatePropagation()"
|
||||||
></ng-select>
|
></ng-select>
|
||||||
<select class="w-25 form-select" [(ngModel)]="atom.operator" [disabled]="disabled">
|
<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>
|
@for (operator of getOperatorsForField(atom.field); track operator.label) {
|
||||||
|
<option [ngValue]="operator.value">{{operator.label}}</option>
|
||||||
|
}
|
||||||
</select>
|
</select>
|
||||||
@switch (atom.operator) {
|
@switch (atom.operator) {
|
||||||
@case (CustomFieldQueryOperator.Exists) {
|
@case (CustomFieldQueryOperator.Exists) {
|
||||||
@ -99,6 +103,8 @@
|
|||||||
<ng-select
|
<ng-select
|
||||||
class="paperless-input-select rounded-end"
|
class="paperless-input-select rounded-end"
|
||||||
[items]="getSelectOptionsForField(atom.field)"
|
[items]="getSelectOptionsForField(atom.field)"
|
||||||
|
bindLabel="label"
|
||||||
|
bindValue="id"
|
||||||
[(ngModel)]="atom.value"
|
[(ngModel)]="atom.value"
|
||||||
[disabled]="disabled"
|
[disabled]="disabled"
|
||||||
[multiple]="true"
|
[multiple]="true"
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
TestBed,
|
TestBed,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import {
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
CustomFieldQueriesModel,
|
import { NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
CustomFieldsQueryDropdownComponent,
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
} from './custom-fields-query-dropdown.component'
|
import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
|
||||||
import { of } from 'rxjs'
|
import { of } from 'rxjs'
|
||||||
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
import {
|
import {
|
||||||
@ -16,17 +17,16 @@ import {
|
|||||||
CustomFieldQueryLogicalOperator,
|
CustomFieldQueryLogicalOperator,
|
||||||
CustomFieldQueryOperatorGroups,
|
CustomFieldQueryOperatorGroups,
|
||||||
} from 'src/app/data/custom-field-query'
|
} from 'src/app/data/custom-field-query'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
import { NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
import {
|
import {
|
||||||
CustomFieldQueryExpression,
|
|
||||||
CustomFieldQueryAtom,
|
CustomFieldQueryAtom,
|
||||||
CustomFieldQueryElement,
|
CustomFieldQueryElement,
|
||||||
|
CustomFieldQueryExpression,
|
||||||
} from 'src/app/utils/custom-field-query-element'
|
} from 'src/app/utils/custom-field-query-element'
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import {
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
CustomFieldQueriesModel,
|
||||||
|
CustomFieldsQueryDropdownComponent,
|
||||||
|
} from './custom-fields-query-dropdown.component'
|
||||||
|
|
||||||
const customFields = [
|
const customFields = [
|
||||||
{
|
{
|
||||||
@ -39,7 +39,12 @@ const customFields = [
|
|||||||
id: 2,
|
id: 2,
|
||||||
name: 'Test Select Field',
|
name: 'Test Select Field',
|
||||||
data_type: CustomFieldDataType.Select,
|
data_type: CustomFieldDataType.Select,
|
||||||
extra_data: { select_options: ['Option 1', 'Option 2'] },
|
extra_data: {
|
||||||
|
select_options: [
|
||||||
|
{ label: 'Option 1', id: 'abc-123' },
|
||||||
|
{ label: 'Option 2', id: 'def-456' },
|
||||||
|
],
|
||||||
|
},
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -50,13 +55,13 @@ describe('CustomFieldsQueryDropdownComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await TestBed.configureTestingModule({
|
await TestBed.configureTestingModule({
|
||||||
declarations: [CustomFieldsQueryDropdownComponent],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgbDropdownModule,
|
NgbDropdownModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
NgSelectModule,
|
NgSelectModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
|
CustomFieldsQueryDropdownComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
@ -128,11 +133,19 @@ describe('CustomFieldsQueryDropdownComponent', () => {
|
|||||||
id: 1,
|
id: 1,
|
||||||
name: 'Test Field',
|
name: 'Test Field',
|
||||||
data_type: CustomFieldDataType.Select,
|
data_type: CustomFieldDataType.Select,
|
||||||
extra_data: { select_options: ['Option 1', 'Option 2'] },
|
extra_data: {
|
||||||
|
select_options: [
|
||||||
|
{ label: 'Option 1', id: 'abc-123' },
|
||||||
|
{ label: 'Option 2', id: 'def-456' },
|
||||||
|
],
|
||||||
|
},
|
||||||
}
|
}
|
||||||
component.customFields = [field]
|
component.customFields = [field]
|
||||||
const options = component.getSelectOptionsForField(1)
|
const options = component.getSelectOptionsForField(1)
|
||||||
expect(options).toEqual(['Option 1', 'Option 2'])
|
expect(options).toEqual([
|
||||||
|
{ label: 'Option 1', id: 'abc-123' },
|
||||||
|
{ label: 'Option 2', id: 'def-456' },
|
||||||
|
])
|
||||||
|
|
||||||
// Fallback to empty array if field is not found
|
// Fallback to empty array if field is not found
|
||||||
const options2 = component.getSelectOptionsForField(2)
|
const options2 = component.getSelectOptionsForField(2)
|
||||||
|
@ -1,34 +1,38 @@
|
|||||||
|
import { NgTemplateOutlet } from '@angular/common'
|
||||||
import {
|
import {
|
||||||
Component,
|
Component,
|
||||||
EventEmitter,
|
EventEmitter,
|
||||||
Input,
|
Input,
|
||||||
OnDestroy,
|
|
||||||
Output,
|
Output,
|
||||||
QueryList,
|
QueryList,
|
||||||
ViewChild,
|
ViewChild,
|
||||||
ViewChildren,
|
ViewChildren,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { NgbDropdown } from '@ng-bootstrap/ng-bootstrap'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { NgSelectComponent } from '@ng-select/ng-select'
|
import { NgbDropdown, NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { Subject, first, takeUntil } from 'rxjs'
|
import { NgSelectComponent, NgSelectModule } from '@ng-select/ng-select'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { first, Subject, takeUntil } from 'rxjs'
|
||||||
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
import {
|
import {
|
||||||
|
CUSTOM_FIELD_QUERY_MAX_ATOMS,
|
||||||
|
CUSTOM_FIELD_QUERY_MAX_DEPTH,
|
||||||
|
CUSTOM_FIELD_QUERY_OPERATOR_GROUPS_BY_TYPE,
|
||||||
|
CUSTOM_FIELD_QUERY_OPERATOR_LABELS,
|
||||||
|
CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP,
|
||||||
CustomFieldQueryElementType,
|
CustomFieldQueryElementType,
|
||||||
CustomFieldQueryOperator,
|
CustomFieldQueryOperator,
|
||||||
CUSTOM_FIELD_QUERY_OPERATOR_GROUPS_BY_TYPE,
|
|
||||||
CUSTOM_FIELD_QUERY_OPERATORS_BY_GROUP,
|
|
||||||
CustomFieldQueryOperatorGroups,
|
CustomFieldQueryOperatorGroups,
|
||||||
CUSTOM_FIELD_QUERY_OPERATOR_LABELS,
|
|
||||||
CUSTOM_FIELD_QUERY_MAX_DEPTH,
|
|
||||||
CUSTOM_FIELD_QUERY_MAX_ATOMS,
|
|
||||||
} from 'src/app/data/custom-field-query'
|
} from 'src/app/data/custom-field-query'
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
import {
|
import {
|
||||||
|
CustomFieldQueryAtom,
|
||||||
CustomFieldQueryElement,
|
CustomFieldQueryElement,
|
||||||
CustomFieldQueryExpression,
|
CustomFieldQueryExpression,
|
||||||
CustomFieldQueryAtom,
|
|
||||||
} from 'src/app/utils/custom-field-query-element'
|
} from 'src/app/utils/custom-field-query-element'
|
||||||
import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options'
|
import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options'
|
||||||
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
import { ClearableBadgeComponent } from '../clearable-badge/clearable-badge.component'
|
||||||
|
|
||||||
export class CustomFieldQueriesModel {
|
export class CustomFieldQueriesModel {
|
||||||
public queries: CustomFieldQueryElement[] = []
|
public queries: CustomFieldQueryElement[] = []
|
||||||
@ -156,8 +160,17 @@ export class CustomFieldQueriesModel {
|
|||||||
selector: 'pngx-custom-fields-query-dropdown',
|
selector: 'pngx-custom-fields-query-dropdown',
|
||||||
templateUrl: './custom-fields-query-dropdown.component.html',
|
templateUrl: './custom-fields-query-dropdown.component.html',
|
||||||
styleUrls: ['./custom-fields-query-dropdown.component.scss'],
|
styleUrls: ['./custom-fields-query-dropdown.component.scss'],
|
||||||
|
imports: [
|
||||||
|
ClearableBadgeComponent,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgTemplateOutlet,
|
||||||
|
NgSelectModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class CustomFieldsQueryDropdownComponent implements OnDestroy {
|
export class CustomFieldsQueryDropdownComponent extends LoadingComponentWithPermissions {
|
||||||
public CustomFieldQueryComponentType = CustomFieldQueryElementType
|
public CustomFieldQueryComponentType = CustomFieldQueryElementType
|
||||||
public CustomFieldQueryOperator = CustomFieldQueryOperator
|
public CustomFieldQueryOperator = CustomFieldQueryOperator
|
||||||
public CustomFieldDataType = CustomFieldDataType
|
public CustomFieldDataType = CustomFieldDataType
|
||||||
@ -223,19 +236,13 @@ export class CustomFieldsQueryDropdownComponent implements OnDestroy {
|
|||||||
|
|
||||||
customFields: CustomField[] = []
|
customFields: CustomField[] = []
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
|
|
||||||
constructor(protected customFieldsService: CustomFieldsService) {
|
constructor(protected customFieldsService: CustomFieldsService) {
|
||||||
|
super()
|
||||||
this.selectionModel = new CustomFieldQueriesModel()
|
this.selectionModel = new CustomFieldQueriesModel()
|
||||||
this.getFields()
|
this.getFields()
|
||||||
this.reset()
|
this.reset()
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy(): void {
|
|
||||||
this.unsubscribeNotifier.next(this)
|
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
}
|
|
||||||
|
|
||||||
public onOpenChange(open: boolean) {
|
public onOpenChange(open: boolean) {
|
||||||
if (open) {
|
if (open) {
|
||||||
if (this.selectionModel.queries.length === 0) {
|
if (this.selectionModel.queries.length === 0) {
|
||||||
@ -311,7 +318,9 @@ export class CustomFieldsQueryDropdownComponent implements OnDestroy {
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
getSelectOptionsForField(fieldID: number): string[] {
|
getSelectOptionsForField(
|
||||||
|
fieldID: number
|
||||||
|
): Array<{ label: string; id: string }> {
|
||||||
const field = this.customFields.find((field) => field.id === fieldID)
|
const field = this.customFields.find((field) => field.id === fieldID)
|
||||||
if (field) {
|
if (field) {
|
||||||
return field.extra_data['select_options']
|
return field.extra_data['select_options']
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
|
import { DatePipe } from '@angular/common'
|
||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import {
|
import {
|
||||||
ComponentFixture,
|
ComponentFixture,
|
||||||
TestBed,
|
TestBed,
|
||||||
fakeAsync,
|
fakeAsync,
|
||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
let fixture: ComponentFixture<DatesDropdownComponent>
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import {
|
|
||||||
DatesDropdownComponent,
|
|
||||||
DateSelection,
|
|
||||||
RelativeDate,
|
|
||||||
} from './dates-dropdown.component'
|
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
|
||||||
import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ClearableBadgeComponent } from '../clearable-badge/clearable-badge.component'
|
import { ClearableBadgeComponent } from '../clearable-badge/clearable-badge.component'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import {
|
||||||
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
DateSelection,
|
||||||
import { DatePipe } from '@angular/common'
|
DatesDropdownComponent,
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
RelativeDate,
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
} from './dates-dropdown.component'
|
||||||
|
let fixture: ComponentFixture<DatesDropdownComponent>
|
||||||
|
|
||||||
describe('DatesDropdownComponent', () => {
|
describe('DatesDropdownComponent', () => {
|
||||||
let component: DatesDropdownComponent
|
let component: DatesDropdownComponent
|
||||||
@ -27,16 +27,14 @@ describe('DatesDropdownComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
DatesDropdownComponent,
|
|
||||||
ClearableBadgeComponent,
|
|
||||||
CustomDatePipe,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
NgbModule,
|
NgbModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
DatesDropdownComponent,
|
||||||
|
ClearableBadgeComponent,
|
||||||
|
CustomDatePipe,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
SettingsService,
|
SettingsService,
|
||||||
|
@ -1,17 +1,26 @@
|
|||||||
|
import { NgClass } from '@angular/common'
|
||||||
import {
|
import {
|
||||||
Component,
|
Component,
|
||||||
EventEmitter,
|
EventEmitter,
|
||||||
Input,
|
Input,
|
||||||
Output,
|
|
||||||
OnInit,
|
|
||||||
OnDestroy,
|
OnDestroy,
|
||||||
|
OnInit,
|
||||||
|
Output,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { NgbDateAdapter } from '@ng-bootstrap/ng-bootstrap'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
|
import {
|
||||||
|
NgbDateAdapter,
|
||||||
|
NgbDatepickerModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
} from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
import { Subject, Subscription } from 'rxjs'
|
import { Subject, Subscription } from 'rxjs'
|
||||||
import { debounceTime } from 'rxjs/operators'
|
import { debounceTime } from 'rxjs/operators'
|
||||||
|
import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { ISODateAdapter } from 'src/app/utils/ngb-iso-date-adapter'
|
import { ISODateAdapter } from 'src/app/utils/ngb-iso-date-adapter'
|
||||||
import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options'
|
import { popperOptionsReenablePreventOverflow } from 'src/app/utils/popper-options'
|
||||||
|
import { ClearableBadgeComponent } from '../clearable-badge/clearable-badge.component'
|
||||||
|
|
||||||
export interface DateSelection {
|
export interface DateSelection {
|
||||||
createdBefore?: string
|
createdBefore?: string
|
||||||
@ -34,6 +43,16 @@ export enum RelativeDate {
|
|||||||
templateUrl: './dates-dropdown.component.html',
|
templateUrl: './dates-dropdown.component.html',
|
||||||
styleUrls: ['./dates-dropdown.component.scss'],
|
styleUrls: ['./dates-dropdown.component.scss'],
|
||||||
providers: [{ provide: NgbDateAdapter, useClass: ISODateAdapter }],
|
providers: [{ provide: NgbDateAdapter, useClass: ISODateAdapter }],
|
||||||
|
imports: [
|
||||||
|
ClearableBadgeComponent,
|
||||||
|
CustomDatePipe,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
NgbDatepickerModule,
|
||||||
|
NgbDropdownModule,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgClass,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class DatesDropdownComponent implements OnInit, OnDestroy {
|
export class DatesDropdownComponent implements OnInit, OnDestroy {
|
||||||
public popperOptions = popperOptionsReenablePreventOverflow
|
public popperOptions = popperOptionsReenablePreventOverflow
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
@ -11,7 +12,6 @@ import { SelectComponent } from '../../input/select/select.component'
|
|||||||
import { TextComponent } from '../../input/text/text.component'
|
import { TextComponent } from '../../input/text/text.component'
|
||||||
import { EditDialogMode } from '../edit-dialog.component'
|
import { EditDialogMode } from '../edit-dialog.component'
|
||||||
import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog.component'
|
import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
describe('CorrespondentEditDialogComponent', () => {
|
describe('CorrespondentEditDialogComponent', () => {
|
||||||
let component: CorrespondentEditDialogComponent
|
let component: CorrespondentEditDialogComponent
|
||||||
@ -20,7 +20,11 @@ describe('CorrespondentEditDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
imports: [
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgSelectModule,
|
||||||
|
NgbModule,
|
||||||
CorrespondentEditDialogComponent,
|
CorrespondentEditDialogComponent,
|
||||||
IfPermissionsDirective,
|
IfPermissionsDirective,
|
||||||
IfOwnerDirective,
|
IfOwnerDirective,
|
||||||
@ -28,7 +32,6 @@ describe('CorrespondentEditDialogComponent', () => {
|
|||||||
TextComponent,
|
TextComponent,
|
||||||
PermissionsFormComponent,
|
PermissionsFormComponent,
|
||||||
],
|
],
|
||||||
imports: [FormsModule, ReactiveFormsModule, NgSelectModule, NgbModule],
|
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
@ -1,17 +1,34 @@
|
|||||||
import { Component } from '@angular/core'
|
import { Component } from '@angular/core'
|
||||||
import { FormControl, FormGroup } from '@angular/forms'
|
import {
|
||||||
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
} from '@angular/forms'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'
|
import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'
|
||||||
import { DEFAULT_MATCHING_ALGORITHM } from 'src/app/data/matching-model'
|
|
||||||
import { Correspondent } from 'src/app/data/correspondent'
|
import { Correspondent } from 'src/app/data/correspondent'
|
||||||
|
import { DEFAULT_MATCHING_ALGORITHM } from 'src/app/data/matching-model'
|
||||||
|
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
||||||
import { CorrespondentService } from 'src/app/services/rest/correspondent.service'
|
import { CorrespondentService } from 'src/app/services/rest/correspondent.service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
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'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-correspondent-edit-dialog',
|
selector: 'pngx-correspondent-edit-dialog',
|
||||||
templateUrl: './correspondent-edit-dialog.component.html',
|
templateUrl: './correspondent-edit-dialog.component.html',
|
||||||
styleUrls: ['./correspondent-edit-dialog.component.scss'],
|
styleUrls: ['./correspondent-edit-dialog.component.scss'],
|
||||||
|
imports: [
|
||||||
|
SelectComponent,
|
||||||
|
PermissionsFormComponent,
|
||||||
|
TextComponent,
|
||||||
|
IfOwnerDirective,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class CorrespondentEditDialogComponent extends EditDialogComponent<Correspondent> {
|
export class CorrespondentEditDialogComponent extends EditDialogComponent<Correspondent> {
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -21,12 +21,16 @@
|
|||||||
</button>
|
</button>
|
||||||
<div formArrayName="select_options">
|
<div formArrayName="select_options">
|
||||||
@for (option of objectForm.controls.extra_data.controls.select_options.controls; track option; let i = $index) {
|
@for (option of objectForm.controls.extra_data.controls.select_options.controls; track option; let i = $index) {
|
||||||
<div class="input-group input-group-sm my-2">
|
<div class="input-group input-group-sm my-2" [formGroup]="objectForm.controls.extra_data.controls.select_options.controls[i]">
|
||||||
<input #selectOption type="text" class="form-control" [formControl]="option" autocomplete="off">
|
<input #selectOption type="text" class="form-control" formControlName="label" autocomplete="off">
|
||||||
|
<input type="hidden" formControlName="id">
|
||||||
<button type="button" class="btn btn-outline-danger" (click)="removeSelectOption(i)" i18n>Delete</button>
|
<button type="button" class="btn btn-outline-danger" (click)="removeSelectOption(i)" i18n>Delete</button>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
@if (object?.id) {
|
||||||
|
<small class="d-block mt-2" i18n>Warning: existing instances of this field will retain their current value index (e.g. option #1, #2, #3) after editing the options here</small>
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@case (CustomFieldDataType.Monetary) {
|
@case (CustomFieldDataType.Monetary) {
|
||||||
<div class="my-3">
|
<div class="my-3">
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
|
||||||
import { CustomFieldEditDialogComponent } from './custom-field-edit-dialog.component'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ElementRef, QueryList } from '@angular/core'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { CustomFieldDataType } from 'src/app/data/custom-field'
|
||||||
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe'
|
||||||
@ -12,10 +15,7 @@ import { SettingsService } from 'src/app/services/settings.service'
|
|||||||
import { SelectComponent } from '../../input/select/select.component'
|
import { SelectComponent } from '../../input/select/select.component'
|
||||||
import { TextComponent } from '../../input/text/text.component'
|
import { TextComponent } from '../../input/text/text.component'
|
||||||
import { EditDialogMode } from '../edit-dialog.component'
|
import { EditDialogMode } from '../edit-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { CustomFieldEditDialogComponent } from './custom-field-edit-dialog.component'
|
||||||
import { CustomFieldDataType } from 'src/app/data/custom-field'
|
|
||||||
import { ElementRef, QueryList } from '@angular/core'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
|
|
||||||
describe('CustomFieldEditDialogComponent', () => {
|
describe('CustomFieldEditDialogComponent', () => {
|
||||||
let component: CustomFieldEditDialogComponent
|
let component: CustomFieldEditDialogComponent
|
||||||
@ -24,20 +24,18 @@ describe('CustomFieldEditDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
|
||||||
CustomFieldEditDialogComponent,
|
|
||||||
IfPermissionsDirective,
|
|
||||||
IfOwnerDirective,
|
|
||||||
SelectComponent,
|
|
||||||
TextComponent,
|
|
||||||
SafeHtmlPipe,
|
|
||||||
],
|
|
||||||
imports: [
|
imports: [
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
NgSelectModule,
|
NgSelectModule,
|
||||||
NgbModule,
|
NgbModule,
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
CustomFieldEditDialogComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
IfOwnerDirective,
|
||||||
|
SelectComponent,
|
||||||
|
TextComponent,
|
||||||
|
SafeHtmlPipe,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
@ -80,7 +78,11 @@ describe('CustomFieldEditDialogComponent', () => {
|
|||||||
name: 'Field 1',
|
name: 'Field 1',
|
||||||
data_type: CustomFieldDataType.Select,
|
data_type: CustomFieldDataType.Select,
|
||||||
extra_data: {
|
extra_data: {
|
||||||
select_options: ['Option 1', 'Option 2', 'Option 3'],
|
select_options: [
|
||||||
|
{ label: 'Option 1', id: '123-xyz' },
|
||||||
|
{ label: 'Option 2', id: '456-abc' },
|
||||||
|
{ label: 'Option 3', id: '789-123' },
|
||||||
|
],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
@ -94,6 +96,10 @@ describe('CustomFieldEditDialogComponent', () => {
|
|||||||
component.dialogMode = EditDialogMode.CREATE
|
component.dialogMode = EditDialogMode.CREATE
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
component.ngOnInit()
|
component.ngOnInit()
|
||||||
|
expect(
|
||||||
|
component.objectForm.get('extra_data').get('select_options').value.length
|
||||||
|
).toBe(0)
|
||||||
|
component.addSelectOption()
|
||||||
expect(
|
expect(
|
||||||
component.objectForm.get('extra_data').get('select_options').value.length
|
component.objectForm.get('extra_data').get('select_options').value.length
|
||||||
).toBe(1)
|
).toBe(1)
|
||||||
@ -101,14 +107,10 @@ describe('CustomFieldEditDialogComponent', () => {
|
|||||||
expect(
|
expect(
|
||||||
component.objectForm.get('extra_data').get('select_options').value.length
|
component.objectForm.get('extra_data').get('select_options').value.length
|
||||||
).toBe(2)
|
).toBe(2)
|
||||||
component.addSelectOption()
|
|
||||||
expect(
|
|
||||||
component.objectForm.get('extra_data').get('select_options').value.length
|
|
||||||
).toBe(3)
|
|
||||||
component.removeSelectOption(0)
|
component.removeSelectOption(0)
|
||||||
expect(
|
expect(
|
||||||
component.objectForm.get('extra_data').get('select_options').value.length
|
component.objectForm.get('extra_data').get('select_options').value.length
|
||||||
).toBe(2)
|
).toBe(1)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should focus on last select option input', () => {
|
it('should focus on last select option input', () => {
|
||||||
|
@ -2,40 +2,53 @@ import {
|
|||||||
AfterViewInit,
|
AfterViewInit,
|
||||||
Component,
|
Component,
|
||||||
ElementRef,
|
ElementRef,
|
||||||
OnDestroy,
|
|
||||||
OnInit,
|
OnInit,
|
||||||
QueryList,
|
QueryList,
|
||||||
ViewChildren,
|
ViewChildren,
|
||||||
} from '@angular/core'
|
} from '@angular/core'
|
||||||
import { FormGroup, FormControl, FormArray } from '@angular/forms'
|
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
||||||
import {
|
import {
|
||||||
DATA_TYPE_LABELS,
|
FormArray,
|
||||||
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
} from '@angular/forms'
|
||||||
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
||||||
|
import { takeUntil } from 'rxjs'
|
||||||
|
import {
|
||||||
CustomField,
|
CustomField,
|
||||||
CustomFieldDataType,
|
CustomFieldDataType,
|
||||||
|
DATA_TYPE_LABELS,
|
||||||
} from 'src/app/data/custom-field'
|
} from 'src/app/data/custom-field'
|
||||||
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { SelectComponent } from '../../input/select/select.component'
|
||||||
|
import { TextComponent } from '../../input/text/text.component'
|
||||||
import { EditDialogComponent, EditDialogMode } from '../edit-dialog.component'
|
import { EditDialogComponent, EditDialogMode } from '../edit-dialog.component'
|
||||||
import { Subject, takeUntil } from 'rxjs'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-custom-field-edit-dialog',
|
selector: 'pngx-custom-field-edit-dialog',
|
||||||
templateUrl: './custom-field-edit-dialog.component.html',
|
templateUrl: './custom-field-edit-dialog.component.html',
|
||||||
styleUrls: ['./custom-field-edit-dialog.component.scss'],
|
styleUrls: ['./custom-field-edit-dialog.component.scss'],
|
||||||
|
imports: [
|
||||||
|
SelectComponent,
|
||||||
|
TextComponent,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgxBootstrapIconsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class CustomFieldEditDialogComponent
|
export class CustomFieldEditDialogComponent
|
||||||
extends EditDialogComponent<CustomField>
|
extends EditDialogComponent<CustomField>
|
||||||
implements OnInit, AfterViewInit, OnDestroy
|
implements OnInit, AfterViewInit
|
||||||
{
|
{
|
||||||
CustomFieldDataType = CustomFieldDataType
|
CustomFieldDataType = CustomFieldDataType
|
||||||
|
|
||||||
@ViewChildren('selectOption')
|
@ViewChildren('selectOption')
|
||||||
private selectOptionInputs: QueryList<ElementRef>
|
private selectOptionInputs: QueryList<ElementRef>
|
||||||
|
|
||||||
private unsubscribeNotifier: Subject<any> = new Subject()
|
|
||||||
|
|
||||||
private get selectOptions(): FormArray {
|
private get selectOptions(): FormArray {
|
||||||
return (this.objectForm.controls.extra_data as FormGroup).controls
|
return (this.objectForm.controls.extra_data as FormGroup).controls
|
||||||
.select_options as FormArray
|
.select_options as FormArray
|
||||||
@ -57,9 +70,16 @@ export class CustomFieldEditDialogComponent
|
|||||||
}
|
}
|
||||||
if (this.object?.data_type === CustomFieldDataType.Select) {
|
if (this.object?.data_type === CustomFieldDataType.Select) {
|
||||||
this.selectOptions.clear()
|
this.selectOptions.clear()
|
||||||
this.object.extra_data.select_options.forEach((option) =>
|
this.object.extra_data.select_options
|
||||||
this.selectOptions.push(new FormControl(option))
|
.filter((option) => option)
|
||||||
)
|
.forEach((option) =>
|
||||||
|
this.selectOptions.push(
|
||||||
|
new FormGroup({
|
||||||
|
label: new FormControl(option.label),
|
||||||
|
id: new FormControl(option.id),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,11 +91,6 @@ export class CustomFieldEditDialogComponent
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy(): void {
|
|
||||||
this.unsubscribeNotifier.next(true)
|
|
||||||
this.unsubscribeNotifier.complete()
|
|
||||||
}
|
|
||||||
|
|
||||||
getCreateTitle() {
|
getCreateTitle() {
|
||||||
return $localize`Create new custom field`
|
return $localize`Create new custom field`
|
||||||
}
|
}
|
||||||
@ -89,7 +104,7 @@ export class CustomFieldEditDialogComponent
|
|||||||
name: new FormControl(null),
|
name: new FormControl(null),
|
||||||
data_type: new FormControl(null),
|
data_type: new FormControl(null),
|
||||||
extra_data: new FormGroup({
|
extra_data: new FormGroup({
|
||||||
select_options: new FormArray([new FormControl(null)]),
|
select_options: new FormArray([]),
|
||||||
default_currency: new FormControl(null),
|
default_currency: new FormControl(null),
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
@ -104,7 +119,9 @@ export class CustomFieldEditDialogComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
public addSelectOption() {
|
public addSelectOption() {
|
||||||
this.selectOptions.push(new FormControl(''))
|
this.selectOptions.push(
|
||||||
|
new FormGroup({ label: new FormControl(null), id: new FormControl(null) })
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
public removeSelectOption(index: number) {
|
public removeSelectOption(index: number) {
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
@ -11,7 +12,6 @@ import { SelectComponent } from '../../input/select/select.component'
|
|||||||
import { TextComponent } from '../../input/text/text.component'
|
import { TextComponent } from '../../input/text/text.component'
|
||||||
import { EditDialogMode } from '../edit-dialog.component'
|
import { EditDialogMode } from '../edit-dialog.component'
|
||||||
import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog.component'
|
import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
describe('DocumentTypeEditDialogComponent', () => {
|
describe('DocumentTypeEditDialogComponent', () => {
|
||||||
let component: DocumentTypeEditDialogComponent
|
let component: DocumentTypeEditDialogComponent
|
||||||
@ -20,7 +20,11 @@ describe('DocumentTypeEditDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
imports: [
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgSelectModule,
|
||||||
|
NgbModule,
|
||||||
DocumentTypeEditDialogComponent,
|
DocumentTypeEditDialogComponent,
|
||||||
IfPermissionsDirective,
|
IfPermissionsDirective,
|
||||||
IfOwnerDirective,
|
IfOwnerDirective,
|
||||||
@ -28,7 +32,6 @@ describe('DocumentTypeEditDialogComponent', () => {
|
|||||||
TextComponent,
|
TextComponent,
|
||||||
PermissionsFormComponent,
|
PermissionsFormComponent,
|
||||||
],
|
],
|
||||||
imports: [FormsModule, ReactiveFormsModule, NgSelectModule, NgbModule],
|
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
@ -1,17 +1,34 @@
|
|||||||
import { Component } from '@angular/core'
|
import { Component } from '@angular/core'
|
||||||
import { FormControl, FormGroup } from '@angular/forms'
|
import {
|
||||||
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
} from '@angular/forms'
|
||||||
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'
|
import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'
|
||||||
import { DEFAULT_MATCHING_ALGORITHM } from 'src/app/data/matching-model'
|
|
||||||
import { DocumentType } from 'src/app/data/document-type'
|
import { DocumentType } from 'src/app/data/document-type'
|
||||||
|
import { DEFAULT_MATCHING_ALGORITHM } from 'src/app/data/matching-model'
|
||||||
|
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
||||||
import { DocumentTypeService } from 'src/app/services/rest/document-type.service'
|
import { DocumentTypeService } from 'src/app/services/rest/document-type.service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
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'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'pngx-document-type-edit-dialog',
|
selector: 'pngx-document-type-edit-dialog',
|
||||||
templateUrl: './document-type-edit-dialog.component.html',
|
templateUrl: './document-type-edit-dialog.component.html',
|
||||||
styleUrls: ['./document-type-edit-dialog.component.scss'],
|
styleUrls: ['./document-type-edit-dialog.component.scss'],
|
||||||
|
imports: [
|
||||||
|
SelectComponent,
|
||||||
|
PermissionsFormComponent,
|
||||||
|
TextComponent,
|
||||||
|
IfOwnerDirective,
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
],
|
||||||
})
|
})
|
||||||
export class DocumentTypeEditDialogComponent extends EditDialogComponent<DocumentType> {
|
export class DocumentTypeEditDialogComponent extends EditDialogComponent<DocumentType> {
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import {
|
import {
|
||||||
HttpTestingController,
|
HttpTestingController,
|
||||||
provideHttpClientTesting,
|
provideHttpClientTesting,
|
||||||
@ -10,8 +11,8 @@ import {
|
|||||||
tick,
|
tick,
|
||||||
} from '@angular/core/testing'
|
} from '@angular/core/testing'
|
||||||
import {
|
import {
|
||||||
FormGroup,
|
|
||||||
FormControl,
|
FormControl,
|
||||||
|
FormGroup,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ReactiveFormsModule,
|
ReactiveFormsModule,
|
||||||
} from '@angular/forms'
|
} from '@angular/forms'
|
||||||
@ -19,9 +20,9 @@ import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|||||||
import { of } from 'rxjs'
|
import { of } from 'rxjs'
|
||||||
import {
|
import {
|
||||||
DEFAULT_MATCHING_ALGORITHM,
|
DEFAULT_MATCHING_ALGORITHM,
|
||||||
|
MATCH_ALL,
|
||||||
MATCH_AUTO,
|
MATCH_AUTO,
|
||||||
MATCH_NONE,
|
MATCH_NONE,
|
||||||
MATCH_ALL,
|
|
||||||
} from 'src/app/data/matching-model'
|
} from 'src/app/data/matching-model'
|
||||||
import { Tag } from 'src/app/data/tag'
|
import { Tag } from 'src/app/data/tag'
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
@ -30,7 +31,6 @@ import { UserService } from 'src/app/services/rest/user.service'
|
|||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { environment } from 'src/environments/environment'
|
import { environment } from 'src/environments/environment'
|
||||||
import { EditDialogComponent, EditDialogMode } from './edit-dialog.component'
|
import { EditDialogComponent, EditDialogMode } from './edit-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: `
|
template: `
|
||||||
@ -38,6 +38,7 @@ import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|||||||
<h4 class="modal-title" id="modal-basic-title">{{ getTitle() }}</h4>
|
<h4 class="modal-title" id="modal-basic-title">{{ getTitle() }}</h4>
|
||||||
</div>
|
</div>
|
||||||
`,
|
`,
|
||||||
|
imports: [FormsModule, ReactiveFormsModule],
|
||||||
})
|
})
|
||||||
class TestComponent extends EditDialogComponent<Tag> {
|
class TestComponent extends EditDialogComponent<Tag> {
|
||||||
constructor(
|
constructor(
|
||||||
@ -96,8 +97,7 @@ describe('EditDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [TestComponent],
|
imports: [FormsModule, ReactiveFormsModule, TestComponent],
|
||||||
imports: [FormsModule, ReactiveFormsModule],
|
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
{
|
{
|
||||||
|
@ -9,12 +9,13 @@ import {
|
|||||||
} from 'src/app/data/matching-model'
|
} from 'src/app/data/matching-model'
|
||||||
import { ObjectWithId } from 'src/app/data/object-with-id'
|
import { ObjectWithId } from 'src/app/data/object-with-id'
|
||||||
import { ObjectWithPermissions } from 'src/app/data/object-with-permissions'
|
import { ObjectWithPermissions } from 'src/app/data/object-with-permissions'
|
||||||
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
import { User } from 'src/app/data/user'
|
import { User } from 'src/app/data/user'
|
||||||
import { AbstractPaperlessService } from 'src/app/services/rest/abstract-paperless-service'
|
import { AbstractPaperlessService } from 'src/app/services/rest/abstract-paperless-service'
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import { PermissionsFormObject } from '../input/permissions/permissions-form/permissions-form.component'
|
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { LoadingComponentWithPermissions } from '../../loading-component/loading.component'
|
||||||
|
import { PermissionsFormObject } from '../input/permissions/permissions-form/permissions-form.component'
|
||||||
|
|
||||||
export enum EditDialogMode {
|
export enum EditDialogMode {
|
||||||
CREATE = 0,
|
CREATE = 0,
|
||||||
@ -23,15 +24,19 @@ export enum EditDialogMode {
|
|||||||
|
|
||||||
@Directive()
|
@Directive()
|
||||||
export abstract class EditDialogComponent<
|
export abstract class EditDialogComponent<
|
||||||
T extends ObjectWithPermissions | ObjectWithId,
|
T extends ObjectWithPermissions | ObjectWithId,
|
||||||
> implements OnInit
|
>
|
||||||
|
extends LoadingComponentWithPermissions
|
||||||
|
implements OnInit
|
||||||
{
|
{
|
||||||
constructor(
|
constructor(
|
||||||
protected service: AbstractPaperlessService<T>,
|
protected service: AbstractPaperlessService<T>,
|
||||||
private activeModal: NgbActiveModal,
|
private activeModal: NgbActiveModal,
|
||||||
private userService: UserService,
|
private userService: UserService,
|
||||||
private settingsService: SettingsService
|
protected settingsService: SettingsService
|
||||||
) {}
|
) {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
users: User[]
|
users: User[]
|
||||||
|
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms'
|
||||||
import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbActiveModal, NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
import { IfOwnerDirective } from 'src/app/directives/if-owner.directive'
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
import { SettingsService } from 'src/app/services/settings.service'
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
@ -12,8 +14,6 @@ import { TextComponent } from '../../input/text/text.component'
|
|||||||
import { PermissionsSelectComponent } from '../../permissions-select/permissions-select.component'
|
import { PermissionsSelectComponent } from '../../permissions-select/permissions-select.component'
|
||||||
import { EditDialogMode } from '../edit-dialog.component'
|
import { EditDialogMode } from '../edit-dialog.component'
|
||||||
import { GroupEditDialogComponent } from './group-edit-dialog.component'
|
import { GroupEditDialogComponent } from './group-edit-dialog.component'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
|
||||||
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
|
||||||
|
|
||||||
describe('GroupEditDialogComponent', () => {
|
describe('GroupEditDialogComponent', () => {
|
||||||
let component: GroupEditDialogComponent
|
let component: GroupEditDialogComponent
|
||||||
@ -22,7 +22,12 @@ describe('GroupEditDialogComponent', () => {
|
|||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [
|
imports: [
|
||||||
|
FormsModule,
|
||||||
|
ReactiveFormsModule,
|
||||||
|
NgSelectModule,
|
||||||
|
NgbModule,
|
||||||
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
GroupEditDialogComponent,
|
GroupEditDialogComponent,
|
||||||
IfPermissionsDirective,
|
IfPermissionsDirective,
|
||||||
IfOwnerDirective,
|
IfOwnerDirective,
|
||||||
@ -31,13 +36,6 @@ describe('GroupEditDialogComponent', () => {
|
|||||||
PermissionsFormComponent,
|
PermissionsFormComponent,
|
||||||
PermissionsSelectComponent,
|
PermissionsSelectComponent,
|
||||||
],
|
],
|
||||||
imports: [
|
|
||||||
FormsModule,
|
|
||||||
ReactiveFormsModule,
|
|
||||||
NgSelectModule,
|
|
||||||
NgbModule,
|
|
||||||
NgxBootstrapIconsModule.pick(allIcons),
|
|
||||||
],
|
|
||||||
providers: [
|
providers: [
|
||||||
NgbActiveModal,
|
NgbActiveModal,
|
||||||
provideHttpClient(withInterceptorsFromDi()),
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
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