diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 3478d4d40..bdc803132 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -993,11 +993,11 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 82 + 96 src/app/components/document-list/filter-editor/filter-editor.component.html - 90 + 96 src/app/components/manage/mail/mail.component.html @@ -1287,7 +1287,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 88 + 102 src/app/components/manage/custom-fields/custom-fields.component.html @@ -1362,7 +1362,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 142 + 156 src/app/components/manage/custom-fields/custom-fields.component.html @@ -1798,12 +1798,12 @@ 37 - src/app/components/document-list/document-list.component.html - 236 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 11 - src/app/components/document-list/filter-editor/filter-editor.component.html - 76 + src/app/components/document-list/document-list.component.html + 236 src/app/data/document.ts @@ -2161,15 +2161,15 @@ src/app/components/document-detail/document-detail.component.ts - 769 + 773 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 591 + 711 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 630 + 750 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2204,27 +2204,27 @@ src/app/components/document-detail/document-detail.component.ts - 771 + 775 src/app/components/document-detail/document-detail.component.ts - 1064 + 1068 src/app/components/document-detail/document-detail.component.ts - 1102 + 1106 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 632 + 752 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 665 + 785 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 684 + 804 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2637,12 +2637,20 @@ 2 - src/app/components/common/date-dropdown/date-dropdown.component.html - 34 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 38 - src/app/components/common/date-dropdown/date-dropdown.component.html - 55 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 59 + + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 104 + + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 125 @@ -2671,19 +2679,23 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 367 + 398 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 407 + 438 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 445 + 476 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 483 + 514 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 576 @@ -2773,39 +2785,25 @@ 62 - - Create New Field + + Search fields src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html - 22 + 10 - - Add to document + + Create new field src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html - 25 - - - - Choose field - - src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts - 52 - - - - No unused fields found - - src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts - 56 + 21 Saved field "". src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts - 120 + 124 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2816,7 +2814,7 @@ Error saving field. src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts - 128 + 133 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2826,50 +2824,81 @@ now - src/app/components/common/date-dropdown/date-dropdown.component.html - 21 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 25 + + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 91 After - src/app/components/common/date-dropdown/date-dropdown.component.html - 30 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 34 + + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 100 Before - src/app/components/common/date-dropdown/date-dropdown.component.html - 51 + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 55 + + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 121 + + + + Added + + src/app/components/common/dates-dropdown/dates-dropdown.component.html + 76 + + + src/app/components/document-list/document-list.component.html + 245 + + + src/app/data/document.ts + 42 + + + src/app/data/document.ts + 93 Last 7 days - src/app/components/common/date-dropdown/date-dropdown.component.ts - 42 + src/app/components/common/dates-dropdown/dates-dropdown.component.ts + 45 Last month - src/app/components/common/date-dropdown/date-dropdown.component.ts - 47 + src/app/components/common/dates-dropdown/dates-dropdown.component.ts + 50 Last 3 months - src/app/components/common/date-dropdown/date-dropdown.component.ts - 52 + src/app/components/common/dates-dropdown/dates-dropdown.component.ts + 55 Last year - src/app/components/common/date-dropdown/date-dropdown.component.ts - 57 + src/app/components/common/dates-dropdown/dates-dropdown.component.ts + 60 @@ -4905,7 +4934,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 114 + 128 src/app/components/document-list/document-card-large/document-card-large.component.html @@ -5038,7 +5067,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 312 + 343 this string is used to separate processing, failed and added on the file upload widget @@ -5113,7 +5142,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 92 + 106 @@ -5142,7 +5171,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 95 + 109 @@ -5153,7 +5182,7 @@ src/app/components/document-detail/document-detail.component.ts - 1120 + 1124 src/app/guards/dirty-saved-view.guard.ts @@ -5186,7 +5215,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.ts - 115 + 121 src/app/data/document.ts @@ -5478,29 +5507,29 @@ Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 641 + 642 src/app/components/document-detail/document-detail.component.ts - 652 + 656 Error saving document src/app/components/document-detail/document-detail.component.ts - 656 + 660 src/app/components/document-detail/document-detail.component.ts - 697 + 701 Confirm delete src/app/components/document-detail/document-detail.component.ts - 724 + 728 src/app/components/manage/management-list/management-list.component.ts @@ -5515,138 +5544,138 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 725 + 729 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 726 + 730 Delete document src/app/components/document-detail/document-detail.component.ts - 728 + 732 Error deleting document src/app/components/document-detail/document-detail.component.ts - 747 + 751 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 767 + 771 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 628 + 748 This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 768 + 772 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 - 779 + 783 Error executing operation src/app/components/document-detail/document-detail.component.ts - 790 + 794 Page Fit src/app/components/document-detail/document-detail.component.ts - 859 + 863 Split confirm src/app/components/document-detail/document-detail.component.ts - 1062 + 1066 This operation will split the selected document(s) into new documents. src/app/components/document-detail/document-detail.component.ts - 1063 + 1067 Split operation will begin in the background. src/app/components/document-detail/document-detail.component.ts - 1078 + 1082 Error executing split operation src/app/components/document-detail/document-detail.component.ts - 1087 + 1091 Rotate confirm src/app/components/document-detail/document-detail.component.ts - 1099 + 1103 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 661 + 781 This operation will permanently rotate the original version of the current document. src/app/components/document-detail/document-detail.component.ts - 1100 + 1104 This will alter the original copy. src/app/components/document-detail/document-detail.component.ts - 1101 + 1105 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 663 + 783 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 - 1117 + 1121 Error executing rotate operation src/app/components/document-detail/document-detail.component.ts - 1129 + 1133 @@ -5714,64 +5743,90 @@ 65 + + Custom fields + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 78 + + + src/app/components/document-list/filter-editor/filter-editor.component.html + 75 + + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 129 + + + + Filter custom fields + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 79 + + + src/app/components/document-list/filter-editor/filter-editor.component.html + 76 + + Merge src/app/components/document-list/bulk-editor/bulk-editor.component.html - 98 + 112 Include: src/app/components/document-list/bulk-editor/bulk-editor.component.html - 120 + 134 Archived files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 124 + 138 Original files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 128 + 142 Use formatted filename src/app/components/document-list/bulk-editor/bulk-editor.component.html - 133 + 147 Error executing bulk operation src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 229 + 247 "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 304 + 335 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 310 + 341 "" and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 306 + 337 This is for messages like 'modify "tag1" and "tag2"' @@ -5779,7 +5834,7 @@ and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 314,316 + 345,347 this is for messages like 'modify "tag1", "tag2" and "tag3"' @@ -5787,14 +5842,14 @@ Confirm tags assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 331 + 362 This operation will add the tag "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 337 + 368 @@ -5803,14 +5858,14 @@ )"/> to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 342,344 + 373,375 This operation will remove the tag "" from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 350 + 381 @@ -5819,7 +5874,7 @@ )"/> from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 355,357 + 386,388 @@ -5830,126 +5885,176 @@ )"/> on selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 359,363 + 390,394 Confirm correspondent assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 400 + 431 This operation will assign the correspondent "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 402 + 433 This operation will remove the correspondent from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 404 + 435 Confirm document type assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 438 + 469 This operation will assign the document type "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 440 + 471 This operation will remove the document type from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 442 + 473 Confirm storage path assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 476 + 507 This operation will assign the storage path "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 478 + 509 This operation will remove the storage path from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 480 + 511 + + + + Confirm custom field assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 540 + + + + This operation will assign the custom field "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 546 + + + + This operation will assign the custom fields to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 551,553 + + + + This operation will remove the custom field "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 559 + + + + This operation will remove the custom fields from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 564,566 + + + + This operation will assign the custom fields and remove the custom fields on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 568,572 Delete confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 589 + 709 This operation will permanently delete selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 590 + 710 Delete document(s) src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 593 + 713 This operation will permanently redo OCR for selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 629 + 749 This operation will permanently rotate the original version of document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 662 + 782 Merge confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 682 + 802 This operation will merge selected documents into a new document. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 683 + 803 Merged document will be queued for consumption. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 696 + 816 @@ -6180,7 +6285,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.html - 96 + 102 @@ -6205,7 +6310,7 @@ src/app/components/document-list/filter-editor/filter-editor.component.ts - 120 + 126 src/app/data/document.ts @@ -6287,25 +6392,6 @@ 241 - - Added - - src/app/components/document-list/document-list.component.html - 245 - - - src/app/components/document-list/filter-editor/filter-editor.component.html - 82 - - - src/app/data/document.ts - 42 - - - src/app/data/document.ts - 93 - - Shared @@ -6356,67 +6442,67 @@ 285 + + Dates + + src/app/components/document-list/filter-editor/filter-editor.component.html + 86 + + Title & content src/app/components/document-list/filter-editor/filter-editor.component.ts - 118 - - - - Custom fields - - src/app/components/document-list/filter-editor/filter-editor.component.ts - 123 + 124 Advanced search src/app/components/document-list/filter-editor/filter-editor.component.ts - 127 + 133 More like src/app/components/document-list/filter-editor/filter-editor.component.ts - 133 + 139 equals src/app/components/document-list/filter-editor/filter-editor.component.ts - 139 + 145 is empty src/app/components/document-list/filter-editor/filter-editor.component.ts - 143 + 149 is not empty src/app/components/document-list/filter-editor/filter-editor.component.ts - 147 + 153 greater than src/app/components/document-list/filter-editor/filter-editor.component.ts - 151 + 157 less than src/app/components/document-list/filter-editor/filter-editor.component.ts - 155 + 161 @@ -6425,14 +6511,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 175,177 + 181,183 Without correspondent src/app/components/document-list/filter-editor/filter-editor.component.ts - 179 + 185 @@ -6441,14 +6527,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 185,187 + 191,193 Without document type src/app/components/document-list/filter-editor/filter-editor.component.ts - 189 + 195 @@ -6457,14 +6543,14 @@ )?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 195,197 + 201,203 Without storage path src/app/components/document-list/filter-editor/filter-editor.component.ts - 199 + 205 @@ -6472,49 +6558,65 @@ ?.name"/> src/app/components/document-list/filter-editor/filter-editor.component.ts - 203,204 + 209,210 Without any tag src/app/components/document-list/filter-editor/filter-editor.component.ts - 208 + 214 + + + + Custom fields: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 218,220 + + + + Without any custom field + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 224 Title: src/app/components/document-list/filter-editor/filter-editor.component.ts - 212 + 228 ASN: src/app/components/document-list/filter-editor/filter-editor.component.ts - 215 + 231 Owner: src/app/components/document-list/filter-editor/filter-editor.component.ts - 218 + 234 Owner not in: src/app/components/document-list/filter-editor/filter-editor.component.ts - 221 + 237 Without an owner src/app/components/document-list/filter-editor/filter-editor.component.ts - 224 + 240 diff --git a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html index 2489c995a..9111a4b29 100644 --- a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html +++ b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.html @@ -1,31 +1,27 @@ -
+
-
    -
  • - - -
  • -
+
+ @for (field of filteredFields; track field.id) { + + } + @if (!filterText?.length || filteredFields.length === 0) { + + } +
diff --git a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.scss b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.scss index 3240063aa..302dbfe77 100644 --- a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.scss +++ b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.scss @@ -1,5 +1,5 @@ .custom-fields-dropdown { - min-width: 380px; + min-width: 300px; // correct position on mobile @media (max-width: 575.98px) { @@ -8,13 +8,3 @@ } } } - -::ng-deep .custom-fields-dropdown .ng-select .ng-select-container .ng-value-container .ng-placeholder, -::ng-deep .custom-fields-dropdown .ng-select .ng-option, -::ng-deep .custom-fields-dropdown .ng-select .ng-select-container .ng-value-container .ng-value { - font-size: 0.875rem; -} - -::ng-deep .custom-fields-dropdown .paperless-input-select .ng-select .ng-select-container .ng-value-container .ng-input { - top: 4px; -} diff --git a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.spec.ts b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.spec.ts index 7c24578e6..121591ef1 100644 --- a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.spec.ts +++ b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.spec.ts @@ -1,5 +1,9 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing' - +import { + ComponentFixture, + TestBed, + fakeAsync, + tick, +} from '@angular/core/testing' import { CustomFieldsDropdownComponent } from './custom-fields-dropdown.component' import { HttpClientTestingModule } from '@angular/common/http/testing' import { ToastService } from 'src/app/services/toast.service' @@ -71,28 +75,33 @@ describe('CustomFieldsDropdownComponent', () => { let addedField component.added.subscribe((f) => (addedField = f)) component.documentId = 11 - component.field = fields[0].id - component.addField() + component.addField({ field: fields[0].id } as any) expect(addedField).not.toBeUndefined() }) - it('should clear field on open / close, updated unused fields', () => { - component.field = fields[1].id - component.onOpenClose() - expect(component.field).toBeUndefined() - - expect(component.unusedFields).toEqual(fields) - const updateSpy = jest.spyOn( - CustomFieldsDropdownComponent.prototype as any, - 'updateUnusedFields' - ) - component.existingFields = [{ field: fields[1].id } as any] - component.onOpenClose() - expect(updateSpy).toHaveBeenCalled() - expect(component.unusedFields).toEqual([fields[0]]) + it('should support filtering fields', () => { + const input = fixture.debugElement.query(By.css('input')) + input.nativeElement.value = 'Field 1' + input.triggerEventHandler('input', { target: input.nativeElement }) + fixture.detectChanges() + expect(component.filteredFields.length).toEqual(1) + expect(component.filteredFields[0].name).toEqual('Field 1') }) - it('should support creating field, show error if necessary', () => { + it('should support update unused fields', () => { + component.existingFields = [{ field: fields[0].id } as any] + component['updateUnusedFields']() + expect(component['unusedFields'].length).toEqual(1) + expect(component['unusedFields'][0].name).toEqual('Field 2') + }) + + it('should support getting data type label', () => { + expect(component.getDataTypeLabel(CustomFieldDataType.Integer)).toEqual( + 'Integer' + ) + }) + + it('should support creating field, show error if necessary, then add', fakeAsync(() => { let modal: NgbModalRef modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) const toastErrorSpy = jest.spyOn(toastService, 'showError') @@ -101,8 +110,9 @@ describe('CustomFieldsDropdownComponent', () => { CustomFieldsDropdownComponent.prototype as any, 'getFields' ) + const addFieldSpy = jest.spyOn(component, 'addField') - const createButton = fixture.debugElement.queryAll(By.css('button'))[1] + const createButton = fixture.debugElement.queryAll(By.css('button'))[3] createButton.triggerEventHandler('click') expect(modal).not.toBeUndefined() @@ -115,9 +125,11 @@ describe('CustomFieldsDropdownComponent', () => { // succeed editDialog.succeeded.emit(fields[0]) + tick(100) expect(toastInfoSpy).toHaveBeenCalled() expect(getFieldsSpy).toHaveBeenCalled() - }) + expect(addFieldSpy).toHaveBeenCalled() + })) it('should support creating field with name', () => { let modal: NgbModalRef @@ -128,4 +140,106 @@ describe('CustomFieldsDropdownComponent', () => { const editDialog = modal.componentInstance as CustomFieldEditDialogComponent expect(editDialog.object.name).toEqual('Foo bar') }) + + it('should support arrow keyboard navigation', fakeAsync(() => { + fixture.nativeElement + .querySelector('button') + .dispatchEvent(new MouseEvent('click')) // open + fixture.detectChanges() + tick(100) + const filterInputEl: HTMLInputElement = + component.listFilterTextInput.nativeElement + expect(document.activeElement).toEqual(filterInputEl) + const itemButtons = Array.from( + (fixture.nativeElement as HTMLDivElement).querySelectorAll( + '.custom-fields-dropdown button' + ) + ).filter((b) => b.textContent.includes('Field')) + filterInputEl.dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowDown', bubbles: true }) + ) + expect(document.activeElement).toEqual(itemButtons[0]) + itemButtons[0].dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowDown', bubbles: true }) + ) + expect(document.activeElement).toEqual(itemButtons[1]) + itemButtons[1].dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowUp', bubbles: true }) + ) + expect(document.activeElement).toEqual(itemButtons[0]) + itemButtons[0].dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowUp', bubbles: true }) + ) + expect(document.activeElement).toEqual(filterInputEl) + filterInputEl.value = 'foo' + component.filterText = 'foo' + + // dont move focus if we're traversing the field + filterInputEl.selectionStart = 1 + expect(document.activeElement).toEqual(filterInputEl) + + // now we're at end, so move focus + filterInputEl.selectionStart = 3 + filterInputEl.dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowDown', bubbles: true }) + ) + expect(document.activeElement).toEqual(itemButtons[0]) + })) + + it('should support arrow keyboard navigation after tab keyboard navigation', fakeAsync(() => { + fixture.nativeElement + .querySelector('button') + .dispatchEvent(new MouseEvent('click')) // open + fixture.detectChanges() + tick(100) + const filterInputEl: HTMLInputElement = + component.listFilterTextInput.nativeElement + expect(document.activeElement).toEqual(filterInputEl) + const itemButtons = Array.from( + (fixture.nativeElement as HTMLDivElement).querySelectorAll( + '.custom-fields-dropdown button' + ) + ).filter((b) => b.textContent.includes('Field')) + filterInputEl.dispatchEvent( + new KeyboardEvent('keydown', { key: 'Tab', bubbles: true }) + ) + itemButtons[0]['focus']() // normally handled by browser + itemButtons[0].dispatchEvent( + new KeyboardEvent('keydown', { key: 'Tab', bubbles: true }) + ) + itemButtons[1]['focus']() // normally handled by browser + itemButtons[1].dispatchEvent( + new KeyboardEvent('keydown', { + key: 'Tab', + shiftKey: true, + bubbles: true, + }) + ) + itemButtons[0]['focus']() // normally handled by browser + itemButtons[0].dispatchEvent( + new KeyboardEvent('keydown', { key: 'ArrowDown', bubbles: true }) + ) + expect(document.activeElement).toEqual(itemButtons[1]) + })) + + it('should support enter keyboard navigation', fakeAsync(() => { + jest.spyOn(component, 'canCreateFields', 'get').mockReturnValue(true) + const addFieldSpy = jest.spyOn(component, 'addField') + const createFieldSpy = jest.spyOn(component, 'createField') + component.filterText = 'Field 1' + component.listFilterEnter() + expect(addFieldSpy).toHaveBeenCalled() + + component.filterText = 'Field 3' + component.listFilterEnter() + expect(createFieldSpy).toHaveBeenCalledWith('Field 3') + + addFieldSpy.mockClear() + createFieldSpy.mockClear() + + component.filterText = undefined + component.listFilterEnter() + expect(createFieldSpy).not.toHaveBeenCalled() + expect(addFieldSpy).not.toHaveBeenCalled() + })) }) diff --git a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts index 79c0d1b58..652d7f3d8 100644 --- a/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts +++ b/src-ui/src/app/components/common/custom-fields-dropdown/custom-fields-dropdown.component.ts @@ -1,13 +1,17 @@ import { Component, + ElementRef, EventEmitter, Input, OnDestroy, Output, + QueryList, + ViewChild, + ViewChildren, } from '@angular/core' import { NgbModal } from '@ng-bootstrap/ng-bootstrap' import { Subject, first, takeUntil } from 'rxjs' -import { CustomField } 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 { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' import { ToastService } from 'src/app/services/toast.service' @@ -39,23 +43,25 @@ export class CustomFieldsDropdownComponent implements OnDestroy { @Output() created: EventEmitter = new EventEmitter() + @ViewChild('listFilterTextInput') listFilterTextInput: ElementRef + @ViewChildren('button') buttons: QueryList + private customFields: CustomField[] = [] - public unusedFields: CustomField[] + private unusedFields: CustomField[] = [] + private keyboardIndex: number - public name: string + public get filteredFields(): CustomField[] { + return this.unusedFields.filter( + (f) => + !this.filterText || + f.name.toLowerCase().includes(this.filterText.toLowerCase()) + ) + } - public field: number + public filterText: string private unsubscribeNotifier: Subject = new Subject() - get placeholderText(): string { - return $localize`Choose field` - } - - get notFoundText(): string { - return $localize`No unused fields found` - } - get canCreateFields(): boolean { return this.permissionsService.currentUserCan( PermissionAction.Add, @@ -87,28 +93,26 @@ export class CustomFieldsDropdownComponent implements OnDestroy { }) } - public getCustomFieldFromInstance( - instance: CustomFieldInstance - ): CustomField { - return this.customFields.find((f) => f.id === instance.field) - } - private updateUnusedFields() { this.unusedFields = this.customFields.filter( - (f) => - !this.existingFields?.find( - (e) => this.getCustomFieldFromInstance(e)?.id === f.id - ) + (f) => !this.existingFields?.find((e) => e.field === f.id) ) } - onOpenClose() { - this.field = undefined + onOpenClose(open: boolean) { + if (open) { + setTimeout(() => { + this.listFilterTextInput.nativeElement.focus() + }, 100) + } else { + this.filterText = undefined + } this.updateUnusedFields() } - addField() { - this.added.emit(this.customFields.find((f) => f.id === this.field)) + addField(field: CustomField) { + this.added.emit(field) + this.updateUnusedFields() } createField(newName: string = null) { @@ -121,6 +125,7 @@ export class CustomFieldsDropdownComponent implements OnDestroy { this.customFieldsService.clearCache() this.getFields() this.created.emit(newField) + setTimeout(() => this.addField(newField), 100) }) modal.componentInstance.failed .pipe(takeUntil(this.unsubscribeNotifier)) @@ -128,4 +133,82 @@ export class CustomFieldsDropdownComponent implements OnDestroy { this.toastService.showError($localize`Error saving field.`, e) }) } + + getDataTypeLabel(dataType: string) { + return DATA_TYPE_LABELS.find((l) => l.id === dataType)?.name + } + + public listFilterEnter() { + if (this.filteredFields.length === 1) { + this.addField(this.filteredFields[0]) + } else if ( + this.filterText && + this.filteredFields.length === 0 && + this.canCreateFields + ) { + this.createField(this.filterText) + } + } + + private focusNextButtonItem(setFocus: boolean = true) { + this.keyboardIndex = Math.min( + this.buttons.length - 1, + this.keyboardIndex + 1 + ) + if (setFocus) this.setButtonItemFocus() + } + + focusPreviousButtonItem(setFocus: boolean = true) { + this.keyboardIndex = Math.max(0, this.keyboardIndex - 1) + if (setFocus) this.setButtonItemFocus() + } + + setButtonItemFocus() { + this.buttons.get(this.keyboardIndex)?.nativeElement.focus() + } + + public listKeyDown(event: KeyboardEvent) { + switch (event.key) { + case 'ArrowDown': + if (event.target instanceof HTMLInputElement) { + if ( + !this.filterText || + event.target.selectionStart === this.filterText.length + ) { + this.keyboardIndex = -1 + this.focusNextButtonItem() + event.preventDefault() + } + } else if (event.target instanceof HTMLButtonElement) { + this.focusNextButtonItem() + event.preventDefault() + } + break + case 'ArrowUp': + if (event.target instanceof HTMLButtonElement) { + if (this.keyboardIndex === 0) { + this.listFilterTextInput.nativeElement.focus() + } else { + this.focusPreviousButtonItem() + } + event.preventDefault() + } + break + case 'Tab': + // just track the index in case user uses arrows + if (event.target instanceof HTMLInputElement) { + this.keyboardIndex = 0 + } else if (event.target instanceof HTMLButtonElement) { + if (event.shiftKey) { + if (this.keyboardIndex > 0) { + this.focusPreviousButtonItem(false) + } + } else { + this.focusNextButtonItem(false) + } + } + default: + break + } + } } diff --git a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts index b26ad9024..b439c770f 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts @@ -945,9 +945,9 @@ describe('DocumentDetailComponent', () => { fixture.detectChanges() expect(component.document.custom_fields).toHaveLength(initialLength - 1) expect(component.customFieldFormFields).toHaveLength(initialLength - 1) - expect(fixture.debugElement.nativeElement.textContent).not.toContain( - 'Field 1' - ) + expect( + fixture.debugElement.query(By.css('form')).nativeElement.textContent + ).not.toContain('Field 1') const updateSpy = jest.spyOn(documentService, 'update') component.save(true) expect(updateSpy.mock.lastCall[0].custom_fields).toHaveLength(