diff --git a/docs/api.md b/docs/api.md index 6a275be61..83193f025 100644 --- a/docs/api.md +++ b/docs/api.md @@ -11,7 +11,7 @@ The API provides the following main endpoints: - `/api/correspondents/`: Full CRUD support. - `/api/custom_fields/`: Full CRUD support. - `/api/documents/`: Full CRUD support, except POSTing new documents. - See below. + See [below](#posting-documents-file-uploads). - `/api/document_types/`: Full CRUD support. - `/api/groups/`: Full CRUD support. - `/api/logs/`: Read-Only. @@ -24,6 +24,7 @@ The API provides the following main endpoints: - `/api/tasks/`: Read-only. - `/api/users/`: Full CRUD support. - `/api/workflows/`: Full CRUD support. +- `/api/search/` GET, see [below](#global-search). All of these endpoints except for the logging endpoint allow you to fetch (and edit and delete where appropriate) individual objects by @@ -188,6 +189,38 @@ The REST api provides four different forms of authentication. [configuration](configuration.md#PAPERLESS_ENABLE_HTTP_REMOTE_USER_API)), you can authenticate against the API using Remote User auth. +## Global search + +A global search endpoint is available at `/api/search/` and requires a search term +of > 2 characters e.g. `?query=foo`. This endpoint returns a maximum of 3 results +across nearly all objects, e.g. documents, tags, saved views, mail rules, etc. +Results are only included if the requesting user has the appropriate permissions. + +Results are returned in the following format: + +```json +{ + total: number + documents: [] + saved_views: [] + correspondents: [] + document_types: [] + storage_paths: [] + tags: [] + users: [] + groups: [] + mail_accounts: [] + mail_rules: [] + custom_fields: [] + workflows: [] +} +``` + +Global search first searches objects by name (or title for documents) matching the query. +If the optional `db_only` parameter is set, only document titles will be searched. Otherwise, +if the amount of documents returned by a simple title string search is < 3, results from the +search index will also be included. + ## Searching for documents Full text searching is available on the `/api/documents/` endpoint. Two diff --git a/docs/usage.md b/docs/usage.md index c9003d35d..f11d50a03 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -550,6 +550,16 @@ collection. ## Searching {#basic-usage_searching} +### Global search + +The top search bar in the web UI performs a "global" search of the various +objects Paperless-ngx uses, including documents, tags, workflows, etc. Only +objects for which the user has appropriate permissions are returned. For +documents, if there are < 3 results, "advanced" search results (which use +the document index) will also be included. This can be disabled under settings. + +### Document searches + Paperless offers an extensive searching mechanism that is designed to allow you to quickly find a document you're looking for (for example, that thing that just broke and you bought a couple months ago, that @@ -605,6 +615,12 @@ language](https://whoosh.readthedocs.io/en/latest/querylang.html). For details on what date parsing utilities are available, see [Date parsing](https://whoosh.readthedocs.io/en/latest/dates.html#parsing-date-queries). +## Keyboard shortcuts / hotkeys + +A list of available hotkeys can be shown on any page using Shift + +?. The help dialog shows only the keys that are currently available +based on which area of Paperless-ngx you are using. + ## The recommended workflow {#usage-recommended-workflow} Once you have familiarized yourself with paperless and are ready to use diff --git a/src-ui/e2e/document-list/document-list.spec.ts b/src-ui/e2e/document-list/document-list.spec.ts index da2454e7f..5dea9985e 100644 --- a/src-ui/e2e/document-list/document-list.spec.ts +++ b/src-ui/e2e/document-list/document-list.spec.ts @@ -45,8 +45,8 @@ test('basic filtering', async ({ page }) => { test('text filtering', async ({ page }) => { await page.routeFromHAR(REQUESTS_HAR2, { notFound: 'fallback' }) await page.goto('/documents') - await page.getByRole('textbox').click() - await page.getByRole('textbox').fill('test') + await page.getByRole('main').getByRole('combobox').click() + await page.getByRole('main').getByRole('combobox').fill('test') await expect(page.locator('pngx-document-list')).toHaveText(/32 documents/) await expect(page).toHaveURL(/title_content=test/) await page.getByRole('button', { name: 'Title & content' }).click() @@ -59,12 +59,12 @@ test('text filtering', async ({ page }) => { await expect(page.locator('pngx-document-list')).toHaveText(/26 documents/) await page.getByRole('button', { name: 'Advanced search' }).click() await page.getByRole('button', { name: 'ASN' }).click() - await page.getByRole('textbox').fill('1123') + await page.getByRole('main').getByRole('combobox').nth(1).fill('1123') await expect(page).toHaveURL(/archive_serial_number=1123/) await expect(page.locator('pngx-document-list')).toHaveText(/one document/i) await page.locator('select').selectOption('greater') - await page.getByRole('textbox').click() - await page.getByRole('textbox').fill('1123') + await page.getByRole('main').getByRole('combobox').nth(1).click() + await page.getByRole('main').getByRole('combobox').nth(1).fill('1123') await expect(page).toHaveURL(/archive_serial_number__gt=1123/) await expect(page.locator('pngx-document-list')).toHaveText(/5 documents/) await page.locator('select').selectOption('less') diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index bdc803132..81087ab3e 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -240,18 +240,18 @@ Document was added to Paperless-ngx. src/app/app.component.ts - 81 + 83 src/app/app.component.ts - 90 + 92 Open document src/app/app.component.ts - 83 + 85 src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html @@ -262,28 +262,109 @@ Could not add : src/app/app.component.ts - 105 + 107 Document is being processed by Paperless-ngx. src/app/app.component.ts - 120 + 122 + + + + Dashboard + + src/app/app.component.ts + 129 + + + src/app/components/app-frame/app-frame.component.html + 81 + + + src/app/components/app-frame/app-frame.component.html + 83 + + + src/app/components/dashboard/dashboard.component.html + 1 + + + + Documents + + src/app/app.component.ts + 140 + + + src/app/components/app-frame/app-frame.component.html + 88 + + + src/app/components/app-frame/app-frame.component.html + 90 + + + src/app/components/document-list/document-list.component.ts + 128 + + + src/app/components/manage/management-list/management-list.component.html + 90 + + + src/app/components/manage/management-list/management-list.component.html + 90 + + + src/app/components/manage/management-list/management-list.component.html + 90 + + + src/app/components/manage/management-list/management-list.component.html + 90 + + + + Settings + + src/app/app.component.ts + 152 + + + src/app/components/admin/settings/settings.component.html + 2 + + + src/app/components/admin/settings/settings.component.html + 323 + + + src/app/components/app-frame/app-frame.component.html + 50 + + + src/app/components/app-frame/app-frame.component.html + 228 + + + src/app/components/app-frame/app-frame.component.html + 230 Prev src/app/app.component.ts - 126 + 158 Next src/app/app.component.ts - 127 + 159 src/app/components/document-detail/document-detail.component.html @@ -294,56 +375,56 @@ End src/app/app.component.ts - 128 + 160 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. src/app/app.component.ts - 134 + 166 Drag-and-drop documents here to start uploading or place them in the consume folder. You can also drag-and-drop documents anywhere on all other pages of the web app. Once you do, Paperless-ngx will start training its machine learning algorithms. src/app/app.component.ts - 141 + 173 The documents list shows all of your documents and allows for filtering as well as bulk-editing. There are three different view styles: list, small cards and large cards. A list of documents currently opened for editing is shown in the sidebar. src/app/app.component.ts - 146 + 178 The filtering tools allow you to quickly find documents using various searches, dates, tags, etc. src/app/app.component.ts - 153 + 185 Any combination of filters can be saved as a 'view' which can then be displayed on the dashboard and / or sidebar. src/app/app.component.ts - 159 + 191 Tags, correspondents, document types and storage paths can all be managed using these pages. They can also be created from the document edit view. src/app/app.component.ts - 164 + 196 Manage e-mail accounts and rules for automatically importing documents. src/app/app.component.ts - 172 + 204 src/app/components/manage/mail/mail.component.html @@ -354,14 +435,14 @@ Workflows give you more control over the document pipeline. src/app/app.component.ts - 180 + 212 File Tasks shows you documents that have been consumed, are waiting to be, or may have failed during the process. src/app/app.component.ts - 188 + 220 src/app/components/admin/tasks/tasks.component.html @@ -372,28 +453,28 @@ Check out the settings for various tweaks to the web app and toggle settings for saved views. src/app/app.component.ts - 196 + 228 Thank you! 🙏 src/app/app.component.ts - 204 + 236 There are <em>tons</em> more features and info we didn't cover here, but this should get you started. Check out the documentation or visit the project on GitHub to learn more or to report issues. src/app/app.component.ts - 206 + 238 Lastly, on behalf of every contributor to this community-supported project, thank you for using Paperless-ngx! src/app/app.component.ts - 208 + 240 @@ -466,7 +547,7 @@ src/app/components/admin/settings/settings.component.html - 395 + 403 src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html @@ -571,11 +652,11 @@ src/app/components/app-frame/app-frame.component.html - 272 + 263 src/app/components/app-frame/app-frame.component.html - 275 + 266 @@ -608,7 +689,7 @@ src/app/components/admin/settings/settings.component.html - 383 + 391 src/app/components/admin/tasks/tasks.component.html @@ -671,29 +752,6 @@ 51 - - Settings - - src/app/components/admin/settings/settings.component.html - 2 - - - src/app/components/admin/settings/settings.component.html - 315 - - - src/app/components/app-frame/app-frame.component.html - 59 - - - src/app/components/app-frame/app-frame.component.html - 237 - - - src/app/components/app-frame/app-frame.component.html - 239 - - Options to customize appearance, notifications, saved views and more. Settings apply to the <strong>current user only</strong>. @@ -943,11 +1001,29 @@ 196 + + Global search + + src/app/components/admin/settings/settings.component.html + 200 + + + src/app/components/app-frame/global-search/global-search.component.ts + 92 + + + + Search database only (do not include advanced search results) + + src/app/components/admin/settings/settings.component.html + 204 + + Notes src/app/components/admin/settings/settings.component.html - 200 + 208 src/app/components/document-list/document-list.component.html @@ -966,14 +1042,14 @@ Enable notes src/app/components/admin/settings/settings.component.html - 204 + 212 Permissions src/app/components/admin/settings/settings.component.html - 212 + 220 src/app/components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component.html @@ -993,11 +1069,11 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 96 + 100 src/app/components/document-list/filter-editor/filter-editor.component.html - 96 + 110 src/app/components/manage/mail/mail.component.html @@ -1028,28 +1104,28 @@ Default Permissions src/app/components/admin/settings/settings.component.html - 215 + 223 Settings apply to this user account for objects (Tags, Mail Rules, etc.) created via the web UI src/app/components/admin/settings/settings.component.html - 219,221 + 227,229 Default Owner src/app/components/admin/settings/settings.component.html - 226 + 234 Objects without an owner can be viewed and edited by all users src/app/components/admin/settings/settings.component.html - 230 + 238 src/app/components/common/input/permissions/permissions-form/permissions-form.component.html @@ -1060,18 +1136,18 @@ Default View Permissions src/app/components/admin/settings/settings.component.html - 235 + 243 Users: src/app/components/admin/settings/settings.component.html - 240 + 248 src/app/components/admin/settings/settings.component.html - 267 + 275 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html @@ -1102,11 +1178,11 @@ Groups: src/app/components/admin/settings/settings.component.html - 250 + 258 src/app/components/admin/settings/settings.component.html - 277 + 285 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html @@ -1137,14 +1213,14 @@ Default Edit Permissions src/app/components/admin/settings/settings.component.html - 262 + 270 Edit permissions also grant viewing permissions src/app/components/admin/settings/settings.component.html - 286 + 294 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html @@ -1163,74 +1239,74 @@ Notifications src/app/components/admin/settings/settings.component.html - 294 + 302 Document processing src/app/components/admin/settings/settings.component.html - 297 + 305 Show notifications when new documents are detected src/app/components/admin/settings/settings.component.html - 301 + 309 Show notifications when document processing completes successfully src/app/components/admin/settings/settings.component.html - 302 + 310 Show notifications when document processing fails src/app/components/admin/settings/settings.component.html - 303 + 311 Suppress notifications on dashboard src/app/components/admin/settings/settings.component.html - 304 + 312 This will suppress all messages about document processing status on the dashboard. src/app/components/admin/settings/settings.component.html - 304 + 312 Saved views src/app/components/admin/settings/settings.component.html - 312 + 320 src/app/components/app-frame/app-frame.component.html - 107 + 98 Show warning when closing saved views with unsaved changes src/app/components/admin/settings/settings.component.html - 318 + 326 Views src/app/components/admin/settings/settings.component.html - 322 + 330 src/app/components/document-list/document-list.component.html @@ -1241,7 +1317,7 @@ Show on dashboard src/app/components/admin/settings/settings.component.html - 335 + 343 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -1252,7 +1328,7 @@ Show in sidebar src/app/components/admin/settings/settings.component.html - 339 + 347 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -1263,7 +1339,7 @@ Actions src/app/components/admin/settings/settings.component.html - 343 + 351 src/app/components/admin/tasks/tasks.component.html @@ -1287,7 +1363,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 102 + 106 src/app/components/manage/custom-fields/custom-fields.component.html @@ -1326,7 +1402,7 @@ Delete src/app/components/admin/settings/settings.component.html - 345 + 353 src/app/components/admin/users-groups/users-groups.component.html @@ -1362,7 +1438,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 156 + 160 src/app/components/manage/custom-fields/custom-fields.component.html @@ -1437,42 +1513,42 @@ Documents page size src/app/components/admin/settings/settings.component.html - 356 + 364 Display as src/app/components/admin/settings/settings.component.html - 359 + 367 Table src/app/components/admin/settings/settings.component.html - 361 + 369 Small Cards src/app/components/admin/settings/settings.component.html - 362 + 370 Large Cards src/app/components/admin/settings/settings.component.html - 363 + 371 Show src/app/components/admin/settings/settings.component.html - 367 + 375 src/app/components/document-list/document-list.component.html @@ -1483,7 +1559,7 @@ Default src/app/components/admin/settings/settings.component.html - 367 + 375 src/app/components/document-detail/document-detail.component.html @@ -1494,14 +1570,14 @@ No saved views defined. src/app/components/admin/settings/settings.component.html - 376 + 384 Cancel src/app/components/admin/settings/settings.component.html - 396 + 404 src/app/components/common/confirm-dialog/confirm-dialog.component.ts @@ -1586,7 +1662,7 @@ Error retrieving users src/app/components/admin/settings/settings.component.ts - 188 + 189 src/app/components/admin/users-groups/users-groups.component.ts @@ -1597,7 +1673,7 @@ Error retrieving groups src/app/components/admin/settings/settings.component.ts - 207 + 208 src/app/components/admin/users-groups/users-groups.component.ts @@ -1608,46 +1684,46 @@ Saved view "" deleted. src/app/components/admin/settings/settings.component.ts - 421 + 423 Settings were saved successfully. src/app/components/admin/settings/settings.component.ts - 547 + 553 Settings were saved successfully. Reload is required to apply some changes. src/app/components/admin/settings/settings.component.ts - 551 + 557 Reload now src/app/components/admin/settings/settings.component.ts - 552 + 558 An error occurred while saving settings. src/app/components/admin/settings/settings.component.ts - 562 + 568 src/app/components/app-frame/app-frame.component.ts - 140 + 126 Error while storing settings on server. src/app/components/admin/settings/settings.component.ts - 596 + 602 @@ -1658,11 +1734,11 @@ src/app/components/app-frame/app-frame.component.html - 260 + 251 src/app/components/app-frame/app-frame.component.html - 262 + 253 @@ -1959,11 +2035,11 @@ src/app/components/app-frame/app-frame.component.html - 251 + 242 src/app/components/app-frame/app-frame.component.html - 253 + 244 @@ -2031,6 +2107,14 @@ src/app/components/admin/users-groups/users-groups.component.html 73 + + src/app/components/app-frame/global-search/global-search.component.html + 50 + + + src/app/components/app-frame/global-search/global-search.component.html + 66 + src/app/components/common/input/permissions/permissions-form/permissions-form.component.html 53 @@ -2161,15 +2245,15 @@ src/app/components/document-detail/document-detail.component.ts - 773 + 809 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 711 + 714 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 750 + 753 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2204,27 +2288,27 @@ src/app/components/document-detail/document-detail.component.ts - 775 + 811 src/app/components/document-detail/document-detail.component.ts - 1068 + 1104 src/app/components/document-detail/document-detail.component.ts - 1106 + 1142 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 752 + 755 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 785 + 788 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 804 + 807 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2310,129 +2394,76 @@ 20 - - Search documents - - src/app/components/app-frame/app-frame.component.html - 31 - - Logged in as src/app/components/app-frame/app-frame.component.html - 51 + 42 My Profile src/app/components/app-frame/app-frame.component.html - 55 + 46 Logout src/app/components/app-frame/app-frame.component.html - 62 + 53 Documentation src/app/components/app-frame/app-frame.component.html - 67 + 58 src/app/components/app-frame/app-frame.component.html - 281 + 272 src/app/components/app-frame/app-frame.component.html - 284 - - - - Dashboard - - src/app/components/app-frame/app-frame.component.html - 90 - - - src/app/components/app-frame/app-frame.component.html - 92 - - - src/app/components/dashboard/dashboard.component.html - 1 - - - - Documents - - src/app/components/app-frame/app-frame.component.html - 97 - - - src/app/components/app-frame/app-frame.component.html - 99 - - - src/app/components/document-list/document-list.component.ts - 126 - - - src/app/components/manage/management-list/management-list.component.html - 90 - - - src/app/components/manage/management-list/management-list.component.html - 90 - - - src/app/components/manage/management-list/management-list.component.html - 90 - - - src/app/components/manage/management-list/management-list.component.html - 90 + 275 Open documents src/app/components/app-frame/app-frame.component.html - 137 + 128 Close all src/app/components/app-frame/app-frame.component.html - 157 + 148 src/app/components/app-frame/app-frame.component.html - 159 + 150 Manage src/app/components/app-frame/app-frame.component.html - 168 + 159 Correspondents src/app/components/app-frame/app-frame.component.html - 174 + 165 src/app/components/app-frame/app-frame.component.html - 176 + 167 src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html @@ -2443,11 +2474,11 @@ Tags src/app/components/app-frame/app-frame.component.html - 181 + 172 src/app/components/app-frame/app-frame.component.html - 184 + 175 src/app/components/common/input/tags/tags.component.ts @@ -2467,7 +2498,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 33 + 39 src/app/data/document.ts @@ -2478,11 +2509,11 @@ Document Types src/app/components/app-frame/app-frame.component.html - 190 + 181 src/app/components/app-frame/app-frame.component.html - 192 + 183 src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html @@ -2493,11 +2524,11 @@ Storage Paths src/app/components/app-frame/app-frame.component.html - 197 + 188 src/app/components/app-frame/app-frame.component.html - 199 + 190 src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html @@ -2508,11 +2539,11 @@ Custom Fields src/app/components/app-frame/app-frame.component.html - 204 + 195 src/app/components/app-frame/app-frame.component.html - 206 + 197 src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html @@ -2527,11 +2558,11 @@ Workflows src/app/components/app-frame/app-frame.component.html - 213 + 204 src/app/components/app-frame/app-frame.component.html - 215 + 206 src/app/components/manage/workflows/workflows.component.html @@ -2542,92 +2573,268 @@ Mail src/app/components/app-frame/app-frame.component.html - 220 + 211 src/app/components/app-frame/app-frame.component.html - 223 + 214 Administration src/app/components/app-frame/app-frame.component.html - 231 + 222 Configuration src/app/components/app-frame/app-frame.component.html - 244 + 235 src/app/components/app-frame/app-frame.component.html - 246 + 237 GitHub src/app/components/app-frame/app-frame.component.html - 291 + 282 is available. src/app/components/app-frame/app-frame.component.html - 300,301 + 291,292 Click to view. src/app/components/app-frame/app-frame.component.html - 301 + 292 Paperless-ngx can automatically check for updates src/app/components/app-frame/app-frame.component.html - 305 + 296 How does this work? src/app/components/app-frame/app-frame.component.html - 312,314 + 303,305 Update available src/app/components/app-frame/app-frame.component.html - 325 + 316 Sidebar views updated src/app/components/app-frame/app-frame.component.ts - 282 + 209 Error updating sidebar views src/app/components/app-frame/app-frame.component.ts - 285 + 212 An error occurred while saving update checking settings. src/app/components/app-frame/app-frame.component.ts - 306 + 233 + + + + Search + + src/app/components/app-frame/global-search/global-search.component.html + 8 + + + + Advanced search + + src/app/components/app-frame/global-search/global-search.component.html + 19 + + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 143 + + + + Open + + src/app/components/app-frame/global-search/global-search.component.html + 44 + + + src/app/components/app-frame/global-search/global-search.component.html + 47 + + + + Filter documents + + src/app/components/app-frame/global-search/global-search.component.html + 53 + + + + Download + + src/app/components/app-frame/global-search/global-search.component.html + 63 + + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 79 + + + src/app/components/document-detail/document-detail.component.html + 29 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 132 + + + src/app/components/document-list/document-card-large/document-card-large.component.html + 68 + + + src/app/components/document-list/document-card-small/document-card-small.component.html + 131 + + + + No results + + src/app/components/app-frame/global-search/global-search.component.html + 76 + + + + Documents + + src/app/components/app-frame/global-search/global-search.component.html + 79 + + + + Saved Views + + src/app/components/app-frame/global-search/global-search.component.html + 85 + + + + Tags + + src/app/components/app-frame/global-search/global-search.component.html + 92 + + + + Correspondents + + src/app/components/app-frame/global-search/global-search.component.html + 99 + + + + Document types + + src/app/components/app-frame/global-search/global-search.component.html + 106 + + + + Storage paths + + src/app/components/app-frame/global-search/global-search.component.html + 113 + + + + Users + + src/app/components/app-frame/global-search/global-search.component.html + 120 + + + + Groups + + src/app/components/app-frame/global-search/global-search.component.html + 127 + + + + Custom fields + + src/app/components/app-frame/global-search/global-search.component.html + 134 + + + + Mail accounts + + src/app/components/app-frame/global-search/global-search.component.html + 141 + + + + Mail rules + + src/app/components/app-frame/global-search/global-search.component.html + 148 + + + + Workflows + + src/app/components/app-frame/global-search/global-search.component.html + 155 + + + + Successfully updated object. + + src/app/components/app-frame/global-search/global-search.component.ts + 168 + + + src/app/components/app-frame/global-search/global-search.component.ts + 206 + + + + Error occurred saving object. + + src/app/components/app-frame/global-search/global-search.component.ts + 171 + + + src/app/components/app-frame/global-search/global-search.component.ts + 209 @@ -2679,23 +2886,23 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 398 + 401 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 438 + 441 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 476 + 479 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 514 + 517 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 576 + 579 @@ -4126,10 +4333,24 @@ Not assigned src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts - 337 + 340 Filter drop down element to filter for documents with no correspondent/type/tag assigned + + Open filter + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 452 + + + + Keyboard shortcuts + + src/app/components/common/hotkey-dialog/hotkey-dialog.component.ts + 20 + + Remove @@ -4922,29 +5143,6 @@ 71 - - Download - - src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html - 79 - - - src/app/components/document-detail/document-detail.component.html - 29 - - - src/app/components/document-list/bulk-editor/bulk-editor.component.html - 128 - - - src/app/components/document-list/document-card-large/document-card-large.component.html - 68 - - - src/app/components/document-list/document-card-small/document-card-small.component.html - 131 - - No documents @@ -5067,7 +5265,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 343 + 346 this string is used to separate processing, failed and added on the file upload widget @@ -5142,7 +5340,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 106 + 110 @@ -5171,7 +5369,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 109 + 113 @@ -5182,7 +5380,7 @@ src/app/components/document-detail/document-detail.component.ts - 1124 + 1160 src/app/guards/dirty-saved-view.guard.ts @@ -5215,7 +5413,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.ts - 121 + 131 src/app/data/document.ts @@ -5248,7 +5446,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 36 + 37 src/app/components/document-list/document-list.component.html @@ -5256,7 +5454,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 44 + 52 src/app/data/document.ts @@ -5275,7 +5473,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 50 + 52 src/app/components/document-list/document-list.component.html @@ -5283,7 +5481,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 54 + 64 src/app/data/document.ts @@ -5302,7 +5500,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 64 + 67 src/app/components/document-list/document-list.component.html @@ -5310,7 +5508,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 64 + 76 src/app/data/document.ts @@ -5458,78 +5656,110 @@ An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 328,330 + 330,332 Document changes detected src/app/components/document-detail/document-detail.component.ts - 351 + 353 The version of this document in your browser session appears older than the existing version. src/app/components/document-detail/document-detail.component.ts - 352 + 354 Saving the document here may overwrite other changes that were made. To restore the existing version, discard your changes or close the document. src/app/components/document-detail/document-detail.component.ts - 353 + 355 Ok src/app/components/document-detail/document-detail.component.ts - 355 + 357 + + + + Next document + + src/app/components/document-detail/document-detail.component.ts + 464 + + + + Previous document + + src/app/components/document-detail/document-detail.component.ts + 474 + + + + Close document + + src/app/components/document-detail/document-detail.component.ts + 482 + + + src/app/services/open-documents.service.ts + 116 + + + + Save document + + src/app/components/document-detail/document-detail.component.ts + 489 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 495 + 531 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 520 + 556 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 642 + 678 src/app/components/document-detail/document-detail.component.ts - 656 + 692 Error saving document src/app/components/document-detail/document-detail.component.ts - 660 + 696 src/app/components/document-detail/document-detail.component.ts - 701 + 737 Confirm delete src/app/components/document-detail/document-detail.component.ts - 728 + 764 src/app/components/manage/management-list/management-list.component.ts @@ -5544,138 +5774,138 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 729 + 765 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 730 + 766 Delete document src/app/components/document-detail/document-detail.component.ts - 732 + 768 Error deleting document src/app/components/document-detail/document-detail.component.ts - 751 + 787 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 771 + 807 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 748 + 751 This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 772 + 808 Redo OCR operation will begin in the background. Close and re-open or reload this document after the operation has completed to see new content. src/app/components/document-detail/document-detail.component.ts - 783 + 819 Error executing operation src/app/components/document-detail/document-detail.component.ts - 794 + 830 Page Fit src/app/components/document-detail/document-detail.component.ts - 863 + 899 Split confirm src/app/components/document-detail/document-detail.component.ts - 1066 + 1102 This operation will split the selected document(s) into new documents. src/app/components/document-detail/document-detail.component.ts - 1067 + 1103 Split operation will begin in the background. src/app/components/document-detail/document-detail.component.ts - 1082 + 1118 Error executing split operation src/app/components/document-detail/document-detail.component.ts - 1091 + 1127 Rotate confirm src/app/components/document-detail/document-detail.component.ts - 1103 + 1139 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 781 + 784 This operation will permanently rotate the original version of the current document. src/app/components/document-detail/document-detail.component.ts - 1104 + 1140 This will alter the original copy. src/app/components/document-detail/document-detail.component.ts - 1105 + 1141 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 783 + 786 Rotation will begin in the background. Close and re-open the document after the operation has completed to see the changes. src/app/components/document-detail/document-detail.component.ts - 1121 + 1157 Error executing rotate operation src/app/components/document-detail/document-detail.component.ts - 1133 + 1169 @@ -5707,126 +5937,126 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 34 + 40 Filter correspondents src/app/components/document-list/bulk-editor/bulk-editor.component.html - 37 + 38 src/app/components/document-list/filter-editor/filter-editor.component.html - 45 + 53 Filter document types src/app/components/document-list/bulk-editor/bulk-editor.component.html - 51 + 53 src/app/components/document-list/filter-editor/filter-editor.component.html - 55 + 65 Filter storage paths src/app/components/document-list/bulk-editor/bulk-editor.component.html - 65 + 68 src/app/components/document-list/filter-editor/filter-editor.component.html - 65 + 77 Custom fields src/app/components/document-list/bulk-editor/bulk-editor.component.html - 78 + 82 src/app/components/document-list/filter-editor/filter-editor.component.html - 75 + 89 src/app/components/document-list/filter-editor/filter-editor.component.ts - 129 + 139 Filter custom fields src/app/components/document-list/bulk-editor/bulk-editor.component.html - 79 + 83 src/app/components/document-list/filter-editor/filter-editor.component.html - 76 + 90 Merge src/app/components/document-list/bulk-editor/bulk-editor.component.html - 112 + 116 Include: src/app/components/document-list/bulk-editor/bulk-editor.component.html - 134 + 138 Archived files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 138 + 142 Original files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 142 + 146 Use formatted filename src/app/components/document-list/bulk-editor/bulk-editor.component.html - 147 + 151 Error executing bulk operation src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 247 + 250 "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 335 + 338 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 341 + 344 "" and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 337 + 340 This is for messages like 'modify "tag1" and "tag2"' @@ -5834,7 +6064,7 @@ and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 345,347 + 348,350 this is for messages like 'modify "tag1", "tag2" and "tag3"' @@ -5842,14 +6072,14 @@ Confirm tags assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 362 + 365 This operation will add the tag "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 368 + 371 @@ -5858,14 +6088,14 @@ )"/> to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 373,375 + 376,378 This operation will remove the tag "" from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 381 + 384 @@ -5874,7 +6104,7 @@ )"/> from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 386,388 + 389,391 @@ -5885,84 +6115,84 @@ )"/> on selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 390,394 + 393,397 Confirm correspondent assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 431 + 434 This operation will assign the correspondent "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 433 + 436 This operation will remove the correspondent from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 435 + 438 Confirm document type assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 469 + 472 This operation will assign the document type "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 471 + 474 This operation will remove the document type from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 473 + 476 Confirm storage path assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 507 + 510 This operation will assign the storage path "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 509 + 512 This operation will remove the storage path from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 511 + 514 Confirm custom field assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 540 + 543 This operation will assign the custom field "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 546 + 549 @@ -5971,14 +6201,14 @@ )"/> to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 551,553 + 554,556 This operation will remove the custom field "" from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 559 + 562 @@ -5987,7 +6217,7 @@ )"/> from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 564,566 + 567,569 @@ -5998,63 +6228,63 @@ )"/> on selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 568,572 + 571,575 Delete confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 709 + 712 This operation will permanently delete selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 710 + 713 Delete document(s) src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 713 + 716 This operation will permanently redo OCR for selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 749 + 752 This operation will permanently rotate the original version of document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 782 + 785 Merge confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 802 + 805 This operation will merge selected documents into a new document. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 803 + 806 Merged document will be queued for consumption. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 816 + 819 @@ -6227,6 +6457,10 @@ src/app/components/document-list/document-list.component.html 10 + + src/app/components/document-list/document-list.component.ts + 243 + Select all @@ -6234,6 +6468,10 @@ src/app/components/document-list/document-list.component.html 11 + + src/app/components/document-list/document-list.component.ts + 236 + Sort @@ -6285,7 +6523,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 102 + 116 @@ -6310,7 +6548,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.ts - 126 + 136 src/app/data/document.ts @@ -6428,81 +6666,88 @@ 8 + + Reset filters / selection + + src/app/components/document-list/document-list.component.ts + 224 + + + + Open first [selected] document + + src/app/components/document-list/document-list.component.ts + 252 + + View "" saved successfully. src/app/components/document-list/document-list.component.ts - 242 + 288 View "" created successfully. src/app/components/document-list/document-list.component.ts - 285 + 331 Dates src/app/components/document-list/filter-editor/filter-editor.component.html - 86 + 100 Title & content src/app/components/document-list/filter-editor/filter-editor.component.ts - 124 - - - - Advanced search - - src/app/components/document-list/filter-editor/filter-editor.component.ts - 133 + 134 More like src/app/components/document-list/filter-editor/filter-editor.component.ts - 139 + 149 equals src/app/components/document-list/filter-editor/filter-editor.component.ts - 145 + 155 is empty src/app/components/document-list/filter-editor/filter-editor.component.ts - 149 + 159 is not empty src/app/components/document-list/filter-editor/filter-editor.component.ts - 153 + 163 greater than src/app/components/document-list/filter-editor/filter-editor.component.ts - 157 + 167 less than src/app/components/document-list/filter-editor/filter-editor.component.ts - 161 + 171 @@ -6511,14 +6756,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 181,183 + 191,193 Without correspondent src/app/components/document-list/filter-editor/filter-editor.component.ts - 185 + 195 @@ -6527,14 +6772,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 191,193 + 201,203 Without document type src/app/components/document-list/filter-editor/filter-editor.component.ts - 195 + 205 @@ -6543,14 +6788,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 201,203 + 211,213 Without storage path src/app/components/document-list/filter-editor/filter-editor.component.ts - 205 + 215 @@ -6558,14 +6803,14 @@ ?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 209,210 + 219,220 Without any tag src/app/components/document-list/filter-editor/filter-editor.component.ts - 214 + 224 @@ -6574,49 +6819,49 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 218,220 + 228,230 Without any custom field src/app/components/document-list/filter-editor/filter-editor.component.ts - 224 + 234 Title: src/app/components/document-list/filter-editor/filter-editor.component.ts - 228 + 238 ASN: src/app/components/document-list/filter-editor/filter-editor.component.ts - 231 + 241 Owner: src/app/components/document-list/filter-editor/filter-editor.component.ts - 234 + 244 Owner not in: src/app/components/document-list/filter-editor/filter-editor.component.ts - 237 + 247 Without an owner src/app/components/document-list/filter-editor/filter-editor.component.ts - 240 + 250 @@ -7583,11 +7828,11 @@ src/app/services/open-documents.service.ts - 104 + 108 src/app/services/open-documents.service.ts - 131 + 135 @@ -7598,7 +7843,7 @@ src/app/services/open-documents.service.ts - 132 + 136 @@ -7847,35 +8092,28 @@ You have unsaved changes to the document src/app/services/open-documents.service.ts - 106 + 110 Are you sure you want to close this document? src/app/services/open-documents.service.ts - 110 - - - - Close document - - src/app/services/open-documents.service.ts - 112 + 114 Are you sure you want to close all documents? src/app/services/open-documents.service.ts - 133 + 137 Close documents src/app/services/open-documents.service.ts - 135 + 139 diff --git a/src-ui/setup-jest.ts b/src-ui/setup-jest.ts index 8e754589b..3486d17fc 100644 --- a/src-ui/setup-jest.ts +++ b/src-ui/setup-jest.ts @@ -85,6 +85,7 @@ const mock = () => { } } +Object.defineProperty(window, 'open', { value: jest.fn() }) Object.defineProperty(window, 'localStorage', { value: mock() }) Object.defineProperty(window, 'sessionStorage', { value: mock() }) Object.defineProperty(window, 'getComputedStyle', { diff --git a/src-ui/src/app/app.component.spec.ts b/src-ui/src/app/app.component.spec.ts index 80fbdfa5f..e5fac4cc5 100644 --- a/src-ui/src/app/app.component.spec.ts +++ b/src-ui/src/app/app.component.spec.ts @@ -5,8 +5,7 @@ import { fakeAsync, tick, } from '@angular/core/testing' -import { Router } from '@angular/router' -import { RouterTestingModule } from '@angular/router/testing' +import { Router, RouterModule } from '@angular/router' import { TourService, TourNgBootstrapModule } from 'ngx-ui-tour-ng-bootstrap' import { Subject } from 'rxjs' import { routes } from './app-routing.module' @@ -21,6 +20,10 @@ 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 { PermissionsGuard } from './guards/permissions.guard' +import { DirtySavedViewGuard } from './guards/dirty-saved-view.guard' describe('AppComponent', () => { let component: AppComponent @@ -31,16 +34,18 @@ describe('AppComponent', () => { let toastService: ToastService let router: Router let settingsService: SettingsService + let hotKeyService: HotKeyService beforeEach(async () => { TestBed.configureTestingModule({ declarations: [AppComponent, ToastsComponent, FileDropComponent], - providers: [], + providers: [PermissionsGuard, DirtySavedViewGuard], imports: [ HttpClientTestingModule, TourNgBootstrapModule, - RouterTestingModule.withRoutes(routes), + RouterModule.forRoot(routes), NgxFileDropModule, + NgbModalModule, ], }).compileComponents() @@ -50,6 +55,7 @@ describe('AppComponent', () => { settingsService = TestBed.inject(SettingsService) toastService = TestBed.inject(ToastService) router = TestBed.inject(Router) + hotKeyService = TestBed.inject(HotKeyService) fixture = TestBed.createComponent(AppComponent) component = fixture.componentInstance }) @@ -139,4 +145,20 @@ describe('AppComponent', () => { fileStatusSubject.next(new FileStatus()) expect(toastSpy).toHaveBeenCalled() }) + + it('should support hotkeys', () => { + const addShortcutSpy = jest.spyOn(hotKeyService, 'addShortcut') + const routerSpy = jest.spyOn(router, 'navigate') + // prevent actual navigation + routerSpy.mockReturnValue(new Promise(() => {})) + jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) + component.ngOnInit() + expect(addShortcutSpy).toHaveBeenCalled() + document.dispatchEvent(new KeyboardEvent('keydown', { key: 'h' })) + expect(routerSpy).toHaveBeenCalledWith(['/dashboard']) + document.dispatchEvent(new KeyboardEvent('keydown', { key: 'd' })) + expect(routerSpy).toHaveBeenCalledWith(['/documents']) + document.dispatchEvent(new KeyboardEvent('keydown', { key: 's' })) + expect(routerSpy).toHaveBeenCalledWith(['/settings']) + }) }) diff --git a/src-ui/src/app/app.component.ts b/src-ui/src/app/app.component.ts index e93fde30c..7e8abdf34 100644 --- a/src-ui/src/app/app.component.ts +++ b/src-ui/src/app/app.component.ts @@ -12,6 +12,7 @@ import { PermissionsService, PermissionType, } from './services/permissions.service' +import { HotKeyService } from './services/hot-key.service' @Component({ selector: 'pngx-root', @@ -31,7 +32,8 @@ export class AppComponent implements OnInit, OnDestroy { private tasksService: TasksService, public tourService: TourService, private renderer: Renderer2, - private permissionsService: PermissionsService + private permissionsService: PermissionsService, + private hotKeyService: HotKeyService ) { this.settings.updateAppearanceSettings() } @@ -123,6 +125,36 @@ export class AppComponent implements OnInit, OnDestroy { } }) + this.hotKeyService + .addShortcut({ keys: 'h', description: $localize`Dashboard` }) + .subscribe(() => { + this.router.navigate(['/dashboard']) + }) + if ( + this.permissionsService.currentUserCan( + PermissionAction.View, + PermissionType.Document + ) + ) { + this.hotKeyService + .addShortcut({ keys: 'd', description: $localize`Documents` }) + .subscribe(() => { + this.router.navigate(['/documents']) + }) + } + if ( + this.permissionsService.currentUserCan( + PermissionAction.Change, + PermissionType.UISettings + ) + ) { + this.hotKeyService + .addShortcut({ keys: 's', description: $localize`Settings` }) + .subscribe(() => { + this.router.navigate(['/settings']) + }) + } + const prevBtnTitle = $localize`Prev` const nextBtnTitle = $localize`Next` const endBtnTitle = $localize`End` diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 416cfd129..24d63ed11 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -122,6 +122,8 @@ import { SplitConfirmDialogComponent } from './components/common/confirm-dialog/ 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 { airplane, archive, @@ -163,6 +165,7 @@ import { doorOpen, download, envelope, + envelopeAt, exclamationCircleFill, exclamationTriangle, exclamationTriangleFill, @@ -196,6 +199,7 @@ import { personFill, personFillLock, personLock, + personSquare, plus, plusCircle, questionCircle, @@ -206,6 +210,7 @@ import { sortAlphaDown, sortAlphaUpAlt, tagFill, + tag, tags, textIndentLeft, textLeft, @@ -259,6 +264,7 @@ const icons = { doorOpen, download, envelope, + envelopeAt, exclamationCircleFill, exclamationTriangle, exclamationTriangleFill, @@ -292,6 +298,7 @@ const icons = { personFill, personFillLock, personLock, + personSquare, plus, plusCircle, questionCircle, @@ -302,6 +309,7 @@ const icons = { sortAlphaDown, sortAlphaUpAlt, tagFill, + tag, tags, textIndentLeft, textLeft, @@ -482,6 +490,8 @@ function initializeApp(settings: SettingsService) { DocumentHistoryComponent, DragDropSelectComponent, CustomFieldDisplayComponent, + GlobalSearchComponent, + HotkeyDialogComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/admin/settings/settings.component.html b/src-ui/src/app/components/admin/settings/settings.component.html index b5c6ca6b4..87d7ba68a 100644 --- a/src-ui/src/app/components/admin/settings/settings.component.html +++ b/src-ui/src/app/components/admin/settings/settings.component.html @@ -197,6 +197,14 @@ +

Global search

+ +
+
+ +
+
+

Notes

diff --git a/src-ui/src/app/components/admin/settings/settings.component.spec.ts b/src-ui/src/app/components/admin/settings/settings.component.spec.ts index 7b23edc21..71778d394 100644 --- a/src-ui/src/app/components/admin/settings/settings.component.spec.ts +++ b/src-ui/src/app/components/admin/settings/settings.component.spec.ts @@ -309,7 +309,7 @@ describe('SettingsComponent', () => { expect(toastErrorSpy).toHaveBeenCalled() expect(storeSpy).toHaveBeenCalled() expect(appearanceSettingsSpy).not.toHaveBeenCalled() - expect(setSpy).toHaveBeenCalledTimes(25) + expect(setSpy).toHaveBeenCalledTimes(26) // succeed storeSpy.mockReturnValueOnce(of(true)) diff --git a/src-ui/src/app/components/admin/settings/settings.component.ts b/src-ui/src/app/components/admin/settings/settings.component.ts index 7df90e3de..036f27f48 100644 --- a/src-ui/src/app/components/admin/settings/settings.component.ts +++ b/src-ui/src/app/components/admin/settings/settings.component.ts @@ -100,6 +100,7 @@ export class SettingsComponent defaultPermsEditUsers: new FormControl(null), defaultPermsEditGroups: new FormControl(null), documentEditingRemoveInboxTags: new FormControl(null), + searchDbOnly: new FormControl(null), notificationsConsumerNewDocument: new FormControl(null), notificationsConsumerSuccess: new FormControl(null), @@ -304,6 +305,7 @@ export class SettingsComponent documentEditingRemoveInboxTags: this.settings.get( SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS ), + searchDbOnly: this.settings.get(SETTINGS_KEYS.SEARCH_DB_ONLY), savedViews: {}, } } @@ -533,6 +535,10 @@ export class SettingsComponent SETTINGS_KEYS.DOCUMENT_EDITING_REMOVE_INBOX_TAGS, this.settingsForm.value.documentEditingRemoveInboxTags ) + this.settings.set( + SETTINGS_KEYS.SEARCH_DB_ONLY, + this.settingsForm.value.searchDbOnly + ) this.settings.setLanguage(this.settingsForm.value.displayLanguage) this.settings .storeSettings() diff --git a/src-ui/src/app/components/app-frame/app-frame.component.html b/src-ui/src/app/components/app-frame/app-frame.component.html index 1e4080c48..ab5759ec0 100644 --- a/src-ui/src/app/components/app-frame/app-frame.component.html +++ b/src-ui/src/app/components/app-frame/app-frame.component.html @@ -24,19 +24,10 @@ }
-
-
- - - @if (!searchFieldEmpty) { - - } -
+
+
+ +
@@ -38,7 +44,9 @@ (selectionModelChange)="updateRules()" (opened)="onTagsDropdownOpen()" [documentCounts]="tagDocumentCounts" - [allowSelectNone]="true"> + [allowSelectNone]="true" + [disabled]="disabled" + shortcutKey="t"> } @if (permissionsService.currentUserCan(PermissionAction.View, PermissionType.Correspondent)) { + [allowSelectNone]="true" + [disabled]="disabled" + shortcutKey="y"> } @if (permissionsService.currentUserCan(PermissionAction.View, PermissionType.DocumentType)) { + [allowSelectNone]="true" + [disabled]="disabled" + shortcutKey="u"> } @if (permissionsService.currentUserCan(PermissionAction.View, PermissionType.StoragePath) && storagePaths.length > 0) { + [allowSelectNone]="true" + [disabled]="disabled" + shortcutKey="i"> } @if (permissionsService.currentUserCan(PermissionAction.View, PermissionType.CustomField) && customFields.length > 0) { diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts index f52907bf2..0fcbbc299 100644 --- a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts @@ -11,14 +11,14 @@ import { } from '@angular/core/testing' import { FormsModule, ReactiveFormsModule } from '@angular/forms' import { By } from '@angular/platform-browser' -import { RouterTestingModule } from '@angular/router/testing' import { NgbDropdownModule, NgbDatepickerModule, NgbDropdownItem, + NgbTypeaheadModule, } from '@ng-bootstrap/ng-bootstrap' import { NgSelectComponent } from '@ng-select/ng-select' -import { of } from 'rxjs' +import { of, throwError } from 'rxjs' import { FILTER_TITLE, FILTER_TITLE_CONTENT, @@ -92,6 +92,8 @@ import { import { environment } from 'src/environments/environment' import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field' import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' +import { RouterModule } from '@angular/router' +import { SearchService } from 'src/app/services/rest/search.service' const tags: Tag[] = [ { @@ -164,6 +166,7 @@ describe('FilterEditorComponent', () => { let settingsService: SettingsService let permissionsService: PermissionsService let httpTestingController: HttpTestingController + let searchService: SearchService beforeEach(fakeAsync(() => { TestBed.configureTestingModule({ @@ -222,12 +225,13 @@ describe('FilterEditorComponent', () => { ], imports: [ HttpClientTestingModule, - RouterTestingModule, + RouterModule, NgbDropdownModule, FormsModule, ReactiveFormsModule, NgbDatepickerModule, NgxBootstrapIconsModule.pick(allIcons), + NgbTypeaheadModule, ], }).compileComponents() @@ -235,6 +239,7 @@ describe('FilterEditorComponent', () => { settingsService = TestBed.inject(SettingsService) settingsService.currentUser = users[0] permissionsService = TestBed.inject(PermissionsService) + searchService = TestBed.inject(SearchService) jest .spyOn(permissionsService, 'currentUserCan') .mockImplementation((action, type) => { @@ -2034,6 +2039,11 @@ describe('FilterEditorComponent', () => { new KeyboardEvent('keyup', { key: 'Escape' }) ) expect(component.textFilter).toEqual('') + const blurSpy = jest.spyOn(component.textFilterInput.nativeElement, 'blur') + component.textFilterInput.nativeElement.dispatchEvent( + new KeyboardEvent('keyup', { key: 'Escape' }) + ) + expect(blurSpy).toHaveBeenCalled() }) it('should adjust text filter targets if more like search', () => { @@ -2044,4 +2054,40 @@ describe('FilterEditorComponent', () => { name: $localize`More like`, }) }) + + it('should call autocomplete endpoint on input', fakeAsync(() => { + component.textFilterTarget = 'fulltext-query' // TEXT_FILTER_TARGET_FULLTEXT_QUERY + const autocompleteSpy = jest.spyOn(searchService, 'autocomplete') + component.searchAutoComplete(of('hello')).subscribe() + tick(250) + expect(autocompleteSpy).toHaveBeenCalled() + + component.searchAutoComplete(of('hello world 1')).subscribe() + tick(250) + expect(autocompleteSpy).toHaveBeenCalled() + })) + + it('should handle autocomplete backend failure gracefully', fakeAsync(() => { + component.textFilterTarget = 'fulltext-query' // TEXT_FILTER_TARGET_FULLTEXT_QUERY + const serviceAutocompleteSpy = jest.spyOn(searchService, 'autocomplete') + serviceAutocompleteSpy.mockReturnValue( + throwError(() => new Error('autcomplete failed')) + ) + // serviceAutocompleteSpy.mockReturnValue(of([' world'])) + let result + component.searchAutoComplete(of('hello')).subscribe((res) => { + result = res + }) + tick(250) + expect(serviceAutocompleteSpy).toHaveBeenCalled() + expect(result).toEqual([]) + })) + + it('should support choosing a autocomplete item', () => { + expect(component.textFilter).toBeNull() + component.itemSelected({ item: 'hello', preventDefault: () => true }) + expect(component.textFilter).toEqual('hello ') + component.itemSelected({ item: 'world', preventDefault: () => true }) + expect(component.textFilter).toEqual('hello world ') + }) }) diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts index b59ae53f1..994de01f0 100644 --- a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts @@ -7,12 +7,21 @@ import { OnDestroy, ViewChild, ElementRef, + AfterViewInit, } from '@angular/core' import { Tag } from 'src/app/data/tag' import { Correspondent } from 'src/app/data/correspondent' import { DocumentType } from 'src/app/data/document-type' -import { Subject, Subscription } from 'rxjs' -import { debounceTime, distinctUntilChanged, filter } from 'rxjs/operators' +import { Observable, Subject, Subscription, from } from 'rxjs' +import { + catchError, + debounceTime, + distinctUntilChanged, + filter, + map, + switchMap, + takeUntil, +} from 'rxjs/operators' import { DocumentTypeService } from 'src/app/services/rest/document-type.service' import { TagService } from 'src/app/services/rest/tag.service' import { CorrespondentService } from 'src/app/services/rest/correspondent.service' @@ -82,6 +91,7 @@ import { import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' import { CustomField } from 'src/app/data/custom-field' +import { SearchService } from 'src/app/services/rest/search.service' const TEXT_FILTER_TARGET_TITLE = 'title' const TEXT_FILTER_TARGET_TITLE_CONTENT = 'title-content' @@ -169,7 +179,7 @@ const DEFAULT_TEXT_FILTER_MODIFIER_OPTIONS = [ }) export class FilterEditorComponent extends ComponentWithPermissions - implements OnInit, OnDestroy + implements OnInit, OnDestroy, AfterViewInit { generateFilterName() { if (this.filterRules.length == 1) { @@ -251,7 +261,8 @@ export class FilterEditorComponent private documentService: DocumentService, private storagePathService: StoragePathService, public permissionsService: PermissionsService, - private customFieldService: CustomFieldsService + private customFieldService: CustomFieldsService, + private searchService: SearchService ) { super() } @@ -275,6 +286,8 @@ export class FilterEditorComponent _moreLikeId: number _moreLikeDoc: Document + unsubscribeNotifier: Subject = new Subject() + get textFilterTargets() { if (this.textFilterTarget == TEXT_FILTER_TARGET_FULLTEXT_MORELIKE) { return DEFAULT_TEXT_FILTER_TARGET_OPTIONS.concat([ @@ -944,7 +957,9 @@ export class FilterEditorComponent } textFilterDebounce: Subject - subscription: Subscription + + @Input() + public disabled: boolean = false ngOnInit() { if ( @@ -1000,19 +1015,29 @@ export class FilterEditorComponent this.textFilterDebounce = new Subject() - this.subscription = this.textFilterDebounce + this.textFilterDebounce .pipe( + takeUntil(this.unsubscribeNotifier), debounceTime(400), distinctUntilChanged(), filter((query) => !query.length || query.length > 2) ) - .subscribe((text) => this.updateTextFilter(text)) + .subscribe((text) => + this.updateTextFilter( + text, + this.textFilterTarget !== TEXT_FILTER_TARGET_FULLTEXT_QUERY + ) + ) if (this._textFilter) this.documentService.searchQuery = this._textFilter } + ngAfterViewInit() { + this.textFilterInput.nativeElement.focus() + } + ngOnDestroy() { - this.textFilterDebounce.complete() + this.unsubscribeNotifier.next(true) } resetSelected() { @@ -1057,10 +1082,12 @@ export class FilterEditorComponent this.customFieldSelectionModel.apply() } - updateTextFilter(text) { + updateTextFilter(text, updateRules = true) { this._textFilter = text - this.documentService.searchQuery = text - this.updateRules() + if (updateRules) { + this.documentService.searchQuery = text + this.updateRules() + } } textFilterKeyup(event: KeyboardEvent) { @@ -1071,8 +1098,12 @@ export class FilterEditorComponent if (filterString.length) { this.updateTextFilter(filterString) } - } else if (event.key == 'Escape') { - this.resetTextField() + } else if (event.key === 'Escape') { + if (this._textFilter?.length) { + this.resetTextField() + } else { + this.textFilterInput.nativeElement.blur() + } } } @@ -1105,4 +1136,40 @@ export class FilterEditorComponent this.updateRules() } } + + searchAutoComplete = (text$: Observable) => + text$.pipe( + debounceTime(200), + distinctUntilChanged(), + filter(() => this.textFilterTarget === TEXT_FILTER_TARGET_FULLTEXT_QUERY), + map((term) => { + if (term.lastIndexOf(' ') != -1) { + return term.substring(term.lastIndexOf(' ') + 1) + } else { + return term + } + }), + switchMap((term) => + term.length < 2 + ? from([[]]) + : this.searchService.autocomplete(term).pipe( + catchError(() => { + return from([[]]) + }) + ) + ) + ) + + itemSelected(event) { + event.preventDefault() + let currentSearch: string = this._textFilter ?? '' + let lastSpaceIndex = currentSearch.lastIndexOf(' ') + if (lastSpaceIndex != -1) { + currentSearch = currentSearch.substring(0, lastSpaceIndex + 1) + currentSearch += event.item + ' ' + } else { + currentSearch = event.item + ' ' + } + this.updateTextFilter(currentSearch) + } } diff --git a/src-ui/src/app/data/datatype.ts b/src-ui/src/app/data/datatype.ts new file mode 100644 index 000000000..288186c52 --- /dev/null +++ b/src-ui/src/app/data/datatype.ts @@ -0,0 +1,14 @@ +export enum DataType { + Document = 'document', + SavedView = 'saved_view', + Correspondent = 'correspondent', + DocumentType = 'document_type', + StoragePath = 'storage_path', + Tag = 'tag', + User = 'user', + Group = 'group', + MailAccount = 'mail_account', + MailRule = 'mail_rule', + CustomField = 'custom_field', + Workflow = 'workflow', +} diff --git a/src-ui/src/app/data/filter-rule-type.ts b/src-ui/src/app/data/filter-rule-type.ts index cd4700096..9a87a421c 100644 --- a/src-ui/src/app/data/filter-rule-type.ts +++ b/src-ui/src/app/data/filter-rule-type.ts @@ -1,3 +1,5 @@ +import { DataType } from './datatype' + // These correspond to src/documents/models.py and changes here require a DB migration (and vice versa) export const FILTER_TITLE = 0 export const FILTER_CONTENT = 1 @@ -78,57 +80,57 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ id: FILTER_CORRESPONDENT, filtervar: 'correspondent__id', isnull_filtervar: 'correspondent__isnull', - datatype: 'correspondent', + datatype: DataType.Correspondent, multi: false, }, { id: FILTER_HAS_CORRESPONDENT_ANY, filtervar: 'correspondent__id__in', - datatype: 'correspondent', + datatype: DataType.Correspondent, multi: true, }, { id: FILTER_DOES_NOT_HAVE_CORRESPONDENT, filtervar: 'correspondent__id__none', - datatype: 'correspondent', + datatype: DataType.Correspondent, multi: true, }, { id: FILTER_STORAGE_PATH, filtervar: 'storage_path__id', isnull_filtervar: 'storage_path__isnull', - datatype: 'storage_path', + datatype: DataType.StoragePath, multi: false, }, { id: FILTER_HAS_STORAGE_PATH_ANY, filtervar: 'storage_path__id__in', - datatype: 'storage_path', + datatype: DataType.StoragePath, multi: true, }, { id: FILTER_DOES_NOT_HAVE_STORAGE_PATH, filtervar: 'storage_path__id__none', - datatype: 'storage_path', + datatype: DataType.StoragePath, multi: true, }, { id: FILTER_DOCUMENT_TYPE, filtervar: 'document_type__id', isnull_filtervar: 'document_type__isnull', - datatype: 'document_type', + datatype: DataType.DocumentType, multi: false, }, { id: FILTER_HAS_DOCUMENT_TYPE_ANY, filtervar: 'document_type__id__in', - datatype: 'document_type', + datatype: DataType.DocumentType, multi: true, }, { id: FILTER_DOES_NOT_HAVE_DOCUMENT_TYPE, filtervar: 'document_type__id__none', - datatype: 'document_type', + datatype: DataType.DocumentType, multi: true, }, { @@ -141,19 +143,19 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ { id: FILTER_HAS_TAGS_ALL, filtervar: 'tags__id__all', - datatype: 'tag', + datatype: DataType.Tag, multi: true, }, { id: FILTER_HAS_TAGS_ANY, filtervar: 'tags__id__in', - datatype: 'tag', + datatype: DataType.Tag, multi: true, }, { id: FILTER_DOES_NOT_HAVE_TAG, filtervar: 'tags__id__none', - datatype: 'tag', + datatype: DataType.Tag, multi: true, }, { diff --git a/src-ui/src/app/data/ui-settings.ts b/src-ui/src/app/data/ui-settings.ts index 41f9ba361..6f8f246ff 100644 --- a/src-ui/src/app/data/ui-settings.ts +++ b/src-ui/src/app/data/ui-settings.ts @@ -56,6 +56,7 @@ export const SETTINGS_KEYS = { DEFAULT_PERMS_EDIT_GROUPS: 'general-settings:permissions:default-edit-groups', DOCUMENT_EDITING_REMOVE_INBOX_TAGS: 'general-settings:document-editing:remove-inbox-tags', + SEARCH_DB_ONLY: 'general-settings:search:db-only', } export const SETTINGS: UiSetting[] = [ @@ -219,4 +220,9 @@ export const SETTINGS: UiSetting[] = [ type: 'boolean', default: false, }, + { + key: SETTINGS_KEYS.SEARCH_DB_ONLY, + type: 'boolean', + default: false, + }, ] diff --git a/src-ui/src/app/services/hot-key.service.spec.ts b/src-ui/src/app/services/hot-key.service.spec.ts new file mode 100644 index 000000000..d23293c59 --- /dev/null +++ b/src-ui/src/app/services/hot-key.service.spec.ts @@ -0,0 +1,99 @@ +import { TestBed } from '@angular/core/testing' +import { EventManager } from '@angular/platform-browser' +import { DOCUMENT } from '@angular/common' + +import { HotKeyService } from './hot-key.service' +import { NgbModal, NgbModalModule } from '@ng-bootstrap/ng-bootstrap' + +describe('HotKeyService', () => { + let service: HotKeyService + let eventManager: EventManager + let document: Document + let modalService: NgbModal + + beforeEach(() => { + TestBed.configureTestingModule({ + providers: [HotKeyService, EventManager], + imports: [NgbModalModule], + }) + service = TestBed.inject(HotKeyService) + eventManager = TestBed.inject(EventManager) + document = TestBed.inject(DOCUMENT) + modalService = TestBed.inject(NgbModal) + }) + + it('should support adding a shortcut', () => { + const callback = jest.fn() + const addEventListenerSpy = jest.spyOn(eventManager, 'addEventListener') + + const observable = service + .addShortcut({ keys: 'control.a' }) + .subscribe(() => { + callback() + }) + + expect(addEventListenerSpy).toHaveBeenCalled() + + document.dispatchEvent( + new KeyboardEvent('keydown', { key: 'a', ctrlKey: true }) + ) + expect(callback).toHaveBeenCalled() + + //coverage + observable.unsubscribe() + }) + + it('should support adding a shortcut with a description, show modal', () => { + const addEventListenerSpy = jest.spyOn(eventManager, 'addEventListener') + service + .addShortcut({ keys: 'control.a', description: 'Select all' }) + .subscribe() + expect(addEventListenerSpy).toHaveBeenCalled() + const modalSpy = jest.spyOn(modalService, 'open') + document.dispatchEvent( + new KeyboardEvent('keydown', { key: '?', shiftKey: true }) + ) + expect(modalSpy).toHaveBeenCalled() + }) + + it('should ignore keydown events from input elements that dont have a modifier key', () => { + // constructor adds a shortcut for shift.? + const modalSpy = jest.spyOn(modalService, 'open') + const input = document.createElement('input') + const textArea = document.createElement('textarea') + const event = new KeyboardEvent('keydown', { key: '?', shiftKey: true }) + jest.spyOn(event, 'target', 'get').mockReturnValue(input) + document.dispatchEvent(event) + jest.spyOn(event, 'target', 'get').mockReturnValue(textArea) + document.dispatchEvent(event) + expect(modalSpy).not.toHaveBeenCalled() + }) + + it('should dismiss all modals on escape and not fire event', () => { + const callback = jest.fn() + service + .addShortcut({ keys: 'escape', description: 'Escape' }) + .subscribe(callback) + const modalSpy = jest.spyOn(modalService, 'open') + document.dispatchEvent( + new KeyboardEvent('keydown', { key: '?', shiftKey: true }) + ) + expect(modalSpy).toHaveBeenCalled() + const dismissAllSpy = jest.spyOn(modalService, 'dismissAll') + document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape' })) + expect(dismissAllSpy).toHaveBeenCalled() + expect(callback).not.toHaveBeenCalled() + }) + + it('should not fire event on escape when open dropdowns ', () => { + const callback = jest.fn() + service + .addShortcut({ keys: 'escape', description: 'Escape' }) + .subscribe(callback) + const dropdown = document.createElement('div') + dropdown.classList.add('dropdown-menu', 'show') + document.body.appendChild(dropdown) + document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape' })) + expect(callback).not.toHaveBeenCalled() + }) +}) diff --git a/src-ui/src/app/services/hot-key.service.ts b/src-ui/src/app/services/hot-key.service.ts new file mode 100644 index 000000000..22a757581 --- /dev/null +++ b/src-ui/src/app/services/hot-key.service.ts @@ -0,0 +1,98 @@ +import { DOCUMENT } from '@angular/common' +import { Inject, Injectable } from '@angular/core' +import { EventManager } from '@angular/platform-browser' +import { NgbModal } from '@ng-bootstrap/ng-bootstrap' +import { Observable } from 'rxjs' +import { HotkeyDialogComponent } from '../components/common/hotkey-dialog/hotkey-dialog.component' + +export interface ShortcutOptions { + element?: any + keys: string + description?: string +} + +@Injectable({ + providedIn: 'root', +}) +export class HotKeyService { + private defaults: Partial = { + element: this.document, + } + + private hotkeys: Map = new Map() + + constructor( + private eventManager: EventManager, + @Inject(DOCUMENT) private document: Document, + private modalService: NgbModal + ) { + this.addShortcut({ keys: 'shift.?' }).subscribe(() => { + this.openHelpModal() + }) + } + + public addShortcut(options: ShortcutOptions) { + const optionsWithDefaults = { ...this.defaults, ...options } + const event = `keydown.${optionsWithDefaults.keys}` + + if (optionsWithDefaults.description) { + this.hotkeys.set( + optionsWithDefaults.keys, + optionsWithDefaults.description + ) + } + + return new Observable((observer) => { + const handler = (e: KeyboardEvent) => { + if ( + !(e.altKey || e.metaKey || e.ctrlKey) && + (e.target instanceof HTMLInputElement || + e.target instanceof HTMLTextAreaElement) + ) { + // Ignore keydown events from input elements that dont have a modifier key + return + } + + this.modalService.dismissAll() + if ( + e.key === 'Escape' && + (this.modalService.hasOpenModals() || + this.document.getElementsByClassName('dropdown-menu show').length > + 0) + ) { + // If there is a modal open or menu open, ignore the keydown event + return + } + + e.preventDefault() + observer.next(e) + } + + const dispose = this.eventManager.addEventListener( + optionsWithDefaults.element, + event, + handler + ) + + let disposeMeta + if (event.includes('control')) { + disposeMeta = this.eventManager.addEventListener( + optionsWithDefaults.element, + event.replace('control', 'meta'), + handler + ) + } + + return () => { + dispose() + if (disposeMeta) disposeMeta() + this.hotkeys.delete(optionsWithDefaults.keys) + } + }) + } + + private openHelpModal() { + const modal = this.modalService.open(HotkeyDialogComponent) + modal.componentInstance.hotkeys = this.hotkeys + } +} diff --git a/src-ui/src/app/services/open-documents.service.spec.ts b/src-ui/src/app/services/open-documents.service.spec.ts index 09341da62..21d5d91a8 100644 --- a/src-ui/src/app/services/open-documents.service.spec.ts +++ b/src-ui/src/app/services/open-documents.service.spec.ts @@ -135,6 +135,7 @@ describe('OpenDocumentsService', () => { expect(openDocumentsService.hasDirty()).toBeFalsy() openDocumentsService.setDirty(documents[0], true) expect(openDocumentsService.hasDirty()).toBeTruthy() + expect(openDocumentsService.isDirty(documents[0])).toBeTruthy() let openModal modalService.activeInstances.subscribe((instances) => { openModal = instances[0] diff --git a/src-ui/src/app/services/open-documents.service.ts b/src-ui/src/app/services/open-documents.service.ts index 363a51b03..33e98ce12 100644 --- a/src-ui/src/app/services/open-documents.service.ts +++ b/src-ui/src/app/services/open-documents.service.ts @@ -90,6 +90,10 @@ export class OpenDocumentsService { return this.dirtyDocuments.size > 0 } + isDirty(doc: Document): boolean { + return this.dirtyDocuments.has(doc.id) + } + closeDocument(doc: Document): Observable { let index = this.openDocuments.findIndex((d) => d.id == doc.id) if (index == -1) return of(true) diff --git a/src-ui/src/app/services/rest/search.service.spec.ts b/src-ui/src/app/services/rest/search.service.spec.ts index 7f42aa7da..346b8a092 100644 --- a/src-ui/src/app/services/rest/search.service.spec.ts +++ b/src-ui/src/app/services/rest/search.service.spec.ts @@ -6,10 +6,13 @@ import { Subscription } from 'rxjs' import { TestBed } from '@angular/core/testing' import { environment } from 'src/environments/environment' import { SearchService } from './search.service' +import { SettingsService } from '../settings.service' +import { SETTINGS_KEYS } from 'src/app/data/ui-settings' let httpTestingController: HttpTestingController let service: SearchService let subscription: Subscription +let settingsService: SettingsService const endpoint = 'search/autocomplete' describe('SearchService', () => { @@ -20,6 +23,7 @@ describe('SearchService', () => { }) httpTestingController = TestBed.inject(HttpTestingController) + settingsService = TestBed.inject(SettingsService) service = TestBed.inject(SearchService) }) @@ -36,4 +40,18 @@ describe('SearchService', () => { ) expect(req.request.method).toEqual('GET') }) + + it('should call correct api endpoint on globalSearch', () => { + const query = 'apple' + service.globalSearch(query).subscribe() + httpTestingController.expectOne( + `${environment.apiBaseUrl}search/?query=${query}` + ) + + settingsService.set(SETTINGS_KEYS.SEARCH_DB_ONLY, true) + subscription = service.globalSearch(query).subscribe() + httpTestingController.expectOne( + `${environment.apiBaseUrl}search/?query=${query}&db_only=true` + ) + }) }) diff --git a/src-ui/src/app/services/rest/search.service.ts b/src-ui/src/app/services/rest/search.service.ts index 4a75230d9..7a82d4f2f 100644 --- a/src-ui/src/app/services/rest/search.service.ts +++ b/src-ui/src/app/services/rest/search.service.ts @@ -1,15 +1,48 @@ import { HttpClient, HttpParams } from '@angular/common/http' import { Injectable } from '@angular/core' import { Observable } from 'rxjs' -import { map } from 'rxjs/operators' import { environment } from 'src/environments/environment' -import { DocumentService } from './document.service' +import { Document } from 'src/app/data/document' +import { DocumentType } from 'src/app/data/document-type' +import { Correspondent } from 'src/app/data/correspondent' +import { CustomField } from 'src/app/data/custom-field' +import { Group } from 'src/app/data/group' +import { MailAccount } from 'src/app/data/mail-account' +import { MailRule } from 'src/app/data/mail-rule' +import { StoragePath } from 'src/app/data/storage-path' +import { Tag } from 'src/app/data/tag' +import { User } from 'src/app/data/user' +import { Workflow } from 'src/app/data/workflow' +import { SettingsService } from '../settings.service' +import { SETTINGS_KEYS } from 'src/app/data/ui-settings' +import { SavedView } from 'src/app/data/saved-view' + +export interface GlobalSearchResult { + total: number + documents: Document[] + saved_views: SavedView[] + correspondents: Correspondent[] + document_types: DocumentType[] + storage_paths: StoragePath[] + tags: Tag[] + users: User[] + groups: Group[] + mail_accounts: MailAccount[] + mail_rules: MailRule[] + custom_fields: CustomField[] + workflows: Workflow[] +} @Injectable({ providedIn: 'root', }) export class SearchService { - constructor(private http: HttpClient) {} + public readonly searchResultObjectLimit: number = 3 // documents/views.py GlobalSearchView > OBJECT_LIMIT + + constructor( + private http: HttpClient, + private settingsService: SettingsService + ) {} autocomplete(term: string): Observable { return this.http.get( @@ -17,4 +50,19 @@ export class SearchService { { params: new HttpParams().set('term', term) } ) } + + globalSearch(query: string): Observable { + let params = new HttpParams().set('query', query) + if (this.searchDbOnly) { + params = params.set('db_only', true) + } + return this.http.get( + `${environment.apiBaseUrl}search/`, + { params } + ) + } + + public get searchDbOnly(): boolean { + return this.settingsService.get(SETTINGS_KEYS.SEARCH_DB_ONLY) + } } diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index 22e4b348b..04b908720 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -87,7 +87,7 @@ table .btn-link { color: var(--pngx-primary-text-contrast) !important; } -.navbar .dropdown .btn { +.navbar .dropdown > .btn { color: var(--pngx-primary-text-contrast) !important; } @@ -456,7 +456,7 @@ ul.pagination { color: var(--bs-body-color); &:hover, &:focus { - background-color: var(--pngx-bg-alt); + background-color: var(--pngx-bg-darker); color: var(--bs-body-color); } diff --git a/src-ui/src/theme.scss b/src-ui/src/theme.scss index 806966ec7..98261b8da 100644 --- a/src-ui/src/theme.scss +++ b/src-ui/src/theme.scss @@ -142,7 +142,7 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml,