diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 34096d3a2..8a5a8db60 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -525,7 +525,7 @@ src/app/components/document-detail/document-detail.component.html - 347 + 348 @@ -544,7 +544,7 @@ src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html - 36 + 41 src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html @@ -584,7 +584,7 @@ src/app/components/document-detail/document-detail.component.html - 339 + 340 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -718,7 +718,7 @@ src/app/components/document-detail/document-detail.component.html - 356 + 357 src/app/components/document-list/document-list.component.html @@ -1080,7 +1080,7 @@ src/app/components/document-detail/document-detail.component.html - 315 + 316 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1628,7 +1628,7 @@ src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html - 35 + 40 src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html @@ -3456,6 +3456,27 @@ 20 + + Default Currency + + src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html + 33 + + + + 3-character currency code + + src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html + 33 + + + + Use locale + + src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html + 33 + + Create new custom field @@ -5874,14 +5895,14 @@ Content src/app/components/document-detail/document-detail.component.html - 211 + 212 Metadata src/app/components/document-detail/document-detail.component.html - 220 + 221 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -5892,119 +5913,119 @@ Date modified src/app/components/document-detail/document-detail.component.html - 227 + 228 Date added src/app/components/document-detail/document-detail.component.html - 231 + 232 Media filename src/app/components/document-detail/document-detail.component.html - 235 + 236 Original filename src/app/components/document-detail/document-detail.component.html - 239 + 240 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 243 + 244 Original file size src/app/components/document-detail/document-detail.component.html - 247 + 248 Original mime type src/app/components/document-detail/document-detail.component.html - 251 + 252 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 256 + 257 Archive file size src/app/components/document-detail/document-detail.component.html - 262 + 263 Original document metadata src/app/components/document-detail/document-detail.component.html - 271 + 272 Archived document metadata src/app/components/document-detail/document-detail.component.html - 274 + 275 Preview src/app/components/document-detail/document-detail.component.html - 281 + 282 Notes src/app/components/document-detail/document-detail.component.html - 293,296 + 294,297 History src/app/components/document-detail/document-detail.component.html - 304 + 305 Save & next src/app/components/document-detail/document-detail.component.html - 341 + 342 Save & close src/app/components/document-detail/document-detail.component.html - 344 + 345 Enter Password src/app/components/document-detail/document-detail.component.html - 395 + 396 diff --git a/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.spec.ts b/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.spec.ts index 8706cfac1..ea60034e4 100644 --- a/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.spec.ts +++ b/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.spec.ts @@ -20,6 +20,12 @@ const customFields: CustomField[] = [ select_options: ['Option 1', 'Option 2', 'Option 3'], }, }, + { + id: 5, + name: 'Field 5', + data_type: CustomFieldDataType.Monetary, + extra_data: { default_currency: 'JPY' }, + }, ] const document: Document = { id: 1, @@ -112,6 +118,18 @@ describe('CustomFieldDisplayComponent', () => { expect(component.value).toEqual(100) }) + it('should respect explicit default currency', () => { + component['defaultCurrencyCode'] = 'EUR' // mock default locale injection + component.fieldId = 5 + component.document = { + id: 1, + title: 'Doc 1', + custom_fields: [{ field: 5, document: 1, created: null, value: '100' }], + } + expect(component.currency).toEqual('JPY') + expect(component.value).toEqual(100) + }) + it('should show select value', () => { expect(component.getSelectValue(customFields[3], 2)).toEqual('Option 3') }) diff --git a/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.ts b/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.ts index 7c97c660a..f565c95e2 100644 --- a/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.ts +++ b/src-ui/src/app/components/common/custom-field-display/custom-field-display.component.ts @@ -90,7 +90,9 @@ export class CustomFieldDisplayComponent implements OnInit, OnDestroy { )?.value if (this.value && this.field.data_type === CustomFieldDataType.Monetary) { this.currency = - this.value.match(/([A-Z]{3})/)?.[0] ?? this.defaultCurrencyCode + this.value.match(/([A-Z]{3})/)?.[0] ?? + this.field.extra_data?.default_currency ?? + this.defaultCurrencyCode this.value = parseFloat(this.value.replace(this.currency, '')) } else if ( this.value?.length && diff --git a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html index bc893d53a..953f66659 100644 --- a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html +++ b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html @@ -28,6 +28,11 @@ } } + @case (CustomFieldDataType.Monetary) { +
+ +
+ } } diff --git a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts index db6c7e218..48e5e53bb 100644 --- a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts +++ b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts @@ -90,6 +90,7 @@ export class CustomFieldEditDialogComponent data_type: new FormControl(null), extra_data: new FormGroup({ select_options: new FormArray([new FormControl(null)]), + default_currency: new FormControl(null), }), }) } diff --git a/src-ui/src/app/components/common/input/monetary/monetary.component.spec.ts b/src-ui/src/app/components/common/input/monetary/monetary.component.spec.ts index e0872be44..7a06d842e 100644 --- a/src-ui/src/app/components/common/input/monetary/monetary.component.spec.ts +++ b/src-ui/src/app/components/common/input/monetary/monetary.component.spec.ts @@ -52,6 +52,11 @@ describe('MonetaryComponent', () => { expect(component.defaultCurrencyCode).toEqual('BRL') }) + it('should support setting a default currency code', () => { + component.defaultCurrency = 'EUR' + expect(component.defaultCurrencyCode).toEqual('EUR') + }) + it('should parse monetary value correctly', () => { expect(component['parseMonetaryValue']('123.4')).toEqual('123.4') expect(component['parseMonetaryValue']('123.4', true)).toEqual('123.40') diff --git a/src-ui/src/app/components/common/input/monetary/monetary.component.ts b/src-ui/src/app/components/common/input/monetary/monetary.component.ts index 256dc8c18..c90042860 100644 --- a/src-ui/src/app/components/common/input/monetary/monetary.component.ts +++ b/src-ui/src/app/components/common/input/monetary/monetary.component.ts @@ -1,4 +1,4 @@ -import { Component, forwardRef, Inject, LOCALE_ID } from '@angular/core' +import { Component, forwardRef, Inject, Input, LOCALE_ID } from '@angular/core' import { NG_VALUE_ACCESSOR } from '@angular/forms' import { AbstractInputComponent } from '../abstract-input' import { getLocaleCurrencyCode } from '@angular/common' @@ -29,11 +29,16 @@ export class MonetaryComponent extends AbstractInputComponent { defaultCurrencyCode: string + @Input() + set defaultCurrency(currency: string) { + if (currency) this.defaultCurrencyCode = currency + } + constructor(@Inject(LOCALE_ID) currentLocale: string) { super() this.currency = this.defaultCurrencyCode = - getLocaleCurrencyCode(currentLocale) + this.defaultCurrency ?? getLocaleCurrencyCode(currentLocale) } writeValue(newValue: any): void { diff --git a/src-ui/src/app/components/common/input/text/text.component.html b/src-ui/src/app/components/common/input/text/text.component.html index b46ab23b7..29e5698ad 100644 --- a/src-ui/src/app/components/common/input/text/text.component.html +++ b/src-ui/src/app/components/common/input/text/text.component.html @@ -11,7 +11,7 @@ }
- + @if (hint) { } diff --git a/src-ui/src/app/components/common/input/text/text.component.ts b/src-ui/src/app/components/common/input/text/text.component.ts index a546e2e39..594f5f1d6 100644 --- a/src-ui/src/app/components/common/input/text/text.component.ts +++ b/src-ui/src/app/components/common/input/text/text.component.ts @@ -18,6 +18,9 @@ export class TextComponent extends AbstractInputComponent { @Input() autocomplete: string + @Input() + placeholder: string = '' + constructor() { super() } diff --git a/src-ui/src/app/components/document-detail/document-detail.component.html b/src-ui/src/app/components/document-detail/document-detail.component.html index 84ab680b1..124f4811d 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.html +++ b/src-ui/src/app/components/document-detail/document-detail.component.html @@ -157,6 +157,7 @@ @case (CustomFieldDataType.Monetary) { 0 + and len(attrs["extra_data"]["default_currency"]) != 3 + ) + ) + ): + raise serializers.ValidationError( + {"error": "extra_data.default_currency must be a 3-character string"}, + ) return super().validate(attrs) diff --git a/src/documents/tests/test_api_custom_fields.py b/src/documents/tests/test_api_custom_fields.py index edebf7f3c..6ffe14681 100644 --- a/src/documents/tests/test_api_custom_fields.py +++ b/src/documents/tests/test_api_custom_fields.py @@ -137,6 +137,66 @@ class TestCustomFieldsAPI(DirectoriesMixin, APITestCase): ) self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) + def test_create_custom_field_monetary_validation(self): + """ + GIVEN: + - Custom field does not exist + WHEN: + - API request to create custom field with invalid default currency option + - API request to create custom field with valid default currency option + THEN: + - HTTP 400 is returned + - HTTP 201 is returned + """ + + # not a string + resp = self.client.post( + self.ENDPOINT, + json.dumps( + { + "data_type": "monetary", + "name": "Monetary Field", + "extra_data": { + "default_currency": 123, + }, + }, + ), + content_type="application/json", + ) + self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) + + # not a 3-letter currency code + resp = self.client.post( + self.ENDPOINT, + json.dumps( + { + "data_type": "monetary", + "name": "Monetary Field", + "extra_data": { + "default_currency": "EU", + }, + }, + ), + content_type="application/json", + ) + self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) + + # valid currency code + resp = self.client.post( + self.ENDPOINT, + json.dumps( + { + "data_type": "monetary", + "name": "Monetary Field", + "extra_data": { + "default_currency": "EUR", + }, + }, + ), + content_type="application/json", + ) + self.assertEqual(resp.status_code, status.HTTP_201_CREATED) + def test_create_custom_field_instance(self): """ GIVEN: