mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-22 12:11:19 +00:00 
			
		
		
		
	Expand create transaction form.
This commit is contained in:
		| @@ -118,7 +118,19 @@ class CreateController extends Controller | |||||||
|         $parts                = parse_url($previousUrl); |         $parts                = parse_url($previousUrl); | ||||||
|         $search               = sprintf('?%s', $parts['query'] ?? ''); |         $search               = sprintf('?%s', $parts['query'] ?? ''); | ||||||
|         $previousUrl          = str_replace($search, '', $previousUrl); |         $previousUrl          = str_replace($search, '', $previousUrl); | ||||||
| 
 |         // not really a fan of this, but meh.
 | ||||||
|  |         $optionalDateFields = [ | ||||||
|  |             'interest_date' => $optionalFields['interest_date'], | ||||||
|  |             'book_date'     => $optionalFields['book_date'], | ||||||
|  |             'process_date'  => $optionalFields['process_date'], | ||||||
|  |             'due_date'      => $optionalFields['due_date'], | ||||||
|  |             'payment_date'  => $optionalFields['payment_date'], | ||||||
|  |             'invoice_date'  => $optionalFields['invoice_date'], | ||||||
|  |         ]; | ||||||
|  | //        var_dump($optionalFields);
 | ||||||
|  | //        exit;
 | ||||||
|  |         $optionalFields['external_url'] = $optionalFields['external_url'] ?? false; | ||||||
|  |         $optionalFields['location'] = $optionalFields['location'] ?? false; | ||||||
|         session()->put('preFilled', $preFilled); |         session()->put('preFilled', $preFilled); | ||||||
| 
 | 
 | ||||||
|         return view( |         return view( | ||||||
| @@ -127,6 +139,7 @@ class CreateController extends Controller | |||||||
|                 'subTitleIcon', |                 'subTitleIcon', | ||||||
|                 'cash', |                 'cash', | ||||||
|                 'objectType', |                 'objectType', | ||||||
|  |                 'optionalDateFields', | ||||||
|                 'subTitle', |                 'subTitle', | ||||||
|                 'defaultCurrency', |                 'defaultCurrency', | ||||||
|                 'previousUrl', |                 'previousUrl', | ||||||
|   | |||||||
| @@ -26,15 +26,14 @@ import formatMoney from "../../util/format-money.js"; | |||||||
| import Autocomplete from "bootstrap5-autocomplete"; | import Autocomplete from "bootstrap5-autocomplete"; | ||||||
| import Post from "../../api/v2/model/transaction/post.js"; | import Post from "../../api/v2/model/transaction/post.js"; | ||||||
| import AttachmentPost from "../../api/v1/attachments/post.js"; | import AttachmentPost from "../../api/v1/attachments/post.js"; | ||||||
|  |  | ||||||
| import Get from "../../api/v2/model/currency/get.js"; |  | ||||||
| import BudgetGet from "../../api/v2/model/budget/get.js"; |  | ||||||
| import PiggyBankGet from "../../api/v2/model/piggy-bank/get.js"; |  | ||||||
| import SubscriptionGet from "../../api/v2/model/subscription/get.js"; |  | ||||||
| import {getVariable} from "../../store/get-variable.js"; | import {getVariable} from "../../store/get-variable.js"; | ||||||
| import {I18n} from "i18n-js"; | import {I18n} from "i18n-js"; | ||||||
| import {loadTranslations} from "../../support/load-translations.js"; | import {loadTranslations} from "../../support/load-translations.js"; | ||||||
| import Tags from "bootstrap5-tags"; | import Tags from "bootstrap5-tags"; | ||||||
|  | import {loadCurrencies} from "./shared/load-currencies.js"; | ||||||
|  | import {loadBudgets} from "./shared/load-budgets.js"; | ||||||
|  | import {loadPiggyBanks} from "./shared/load-piggy-banks.js"; | ||||||
|  | import {loadSubscriptions} from "./shared/load-subscriptions.js"; | ||||||
|  |  | ||||||
| import L from "leaflet"; | import L from "leaflet"; | ||||||
|  |  | ||||||
| @@ -172,71 +171,113 @@ let uploadFiles = function (fileData, id) { | |||||||
|  |  | ||||||
| let transactions = function () { | let transactions = function () { | ||||||
|     return { |     return { | ||||||
|         count: 0, |         // transactions are stored in "entries": | ||||||
|         totalAmount: 0, |         entries: [], | ||||||
|         transactionType: 'unknown', |  | ||||||
|         showSuccessMessage: false, |         // state of the form is stored in formState: | ||||||
|         showErrorMessage: false, |         formStates: { | ||||||
|         defaultCurrency: {}, |  | ||||||
|         entries: [], // loading things |  | ||||||
|             loadingCurrencies: true, |             loadingCurrencies: true, | ||||||
|             loadingBudgets: true, |             loadingBudgets: true, | ||||||
|             loadingPiggyBanks: true, |             loadingPiggyBanks: true, | ||||||
|             loadingSubscriptions: true, |             loadingSubscriptions: true, | ||||||
|  |             isSubmitting: false, | ||||||
|  |             returnHereButton: false, | ||||||
|  |             saveAsNewButton: false, // edit form only | ||||||
|  |             resetButton: true, | ||||||
|  |             rulesButton: true, | ||||||
|  |             webhooksButton: true, | ||||||
|  |         }, | ||||||
|  |  | ||||||
|         // data sets |         // form behaviour during transaction | ||||||
|  |         formBehaviour: { | ||||||
|  |             formType: 'create', foreignCurrencyEnabled: true, | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         // form data (except transactions) is stored in formData | ||||||
|  |         formData: { | ||||||
|  |             defaultCurrency: null, | ||||||
|             enabledCurrencies: [], |             enabledCurrencies: [], | ||||||
|             nativeCurrencies: [], |             nativeCurrencies: [], | ||||||
|             foreignCurrencies: [], |             foreignCurrencies: [], | ||||||
|             budgets: [], |             budgets: [], | ||||||
|         piggyBanks: {}, |             piggyBanks: [], | ||||||
|             subscriptions: [], |             subscriptions: [], | ||||||
|         dateFields: ['interest_date', 'book_date', 'process_date', 'due_date', 'payment_date', 'invoice_date'], |  | ||||||
|  |  | ||||||
|         foreignAmountEnabled: true, |  | ||||||
|         filters: { |  | ||||||
|             source: [], destination: [], |  | ||||||
|         }, |         }, | ||||||
|         errorMessageText: '', |  | ||||||
|         successMessageLink: '#', |  | ||||||
|         successMessageText: '', // error and success messages: |  | ||||||
|         showError: false, |  | ||||||
|         showSuccess: false, |  | ||||||
|         showWaitMessage: false, |  | ||||||
|  |  | ||||||
|         // four buttons |         // properties for the entire transaction group | ||||||
|         returnHereButton: false, |         groupProperties: { | ||||||
|         resetButton: false, |             transactionType: 'unknown', | ||||||
|         resetButtonEnabled: false, |             totalAmount: 0, | ||||||
|         rulesButton: true, |         }, | ||||||
|         webhookButton: true, |  | ||||||
|  |  | ||||||
|         // state of the form |         // notifications | ||||||
|         submitting: false, |         notifications: { | ||||||
|  |             error: { | ||||||
|  |                 show: false, | ||||||
|  |                 text: '', | ||||||
|  |                 url: '', | ||||||
|  |             }, | ||||||
|  |             success: { | ||||||
|  |                 show: false, | ||||||
|  |                 text: '', | ||||||
|  |                 url: '', | ||||||
|  |             }, | ||||||
|  |             wait: { | ||||||
|  |                 show: false,text: '', | ||||||
|  |                 url: '', | ||||||
|  |  | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // part of the account selection auto-complete | ||||||
|  |         filters: { | ||||||
|  |             source: [], | ||||||
|  |             destination: [], | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         // old properties, no longer used. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // data sets | ||||||
|  |  | ||||||
|         // used to display the success message |         // used to display the success message | ||||||
|         newGroupTitle: '', |         //newGroupTitle: '', | ||||||
|         newGroupId: 0, |         //newGroupId: 0, | ||||||
|  |  | ||||||
|         // map things: |         // map things: | ||||||
|         hasLocation: false, |         //hasLocation: false, | ||||||
|         latitude: 51.959659235274, |         //latitude: 51.959659235274, | ||||||
|         longitude: 5.756805887265858, |         //longitude: 5.756805887265858, | ||||||
|         zoomLevel: 13, |         //zoomLevel: 13, | ||||||
|  |  | ||||||
|  |         // events in the form | ||||||
|  |         changedDateTime(event) { | ||||||
|  |             console.log('changedDateTime'); | ||||||
|  |         }, | ||||||
|  |         changedDescription(event) { | ||||||
|  |             console.log('changedDescription'); | ||||||
|  |         }, | ||||||
|  |         changedDestinationAccount(event) { | ||||||
|  |             console.log('changedDestinationAccount') | ||||||
|  |         }, | ||||||
|  |         changedSourceAccount(event) { | ||||||
|  |             console.log('changedSourceAccount') | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |  | ||||||
|         detectTransactionType() { |         detectTransactionType() { | ||||||
|             const sourceType = this.entries[0].source_account.type ?? 'unknown'; |             const sourceType = this.entries[0].source_account.type ?? 'unknown'; | ||||||
|             const destType = this.entries[0].destination_account.type ?? 'unknown'; |             const destType = this.entries[0].destination_account.type ?? 'unknown'; | ||||||
|             if ('unknown' === sourceType && 'unknown' === destType) { |             if ('unknown' === sourceType && 'unknown' === destType) { | ||||||
|                 this.transactionType = 'unknown'; |                 this.groupProperties.transactionType = 'unknown'; | ||||||
|                 console.warn('Cannot infer transaction type from two unknown accounts.'); |                 console.warn('Cannot infer transaction type from two unknown accounts.'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             // transfer: both are the same and in strict set of account types |             // transfer: both are the same and in strict set of account types | ||||||
|             if (sourceType === destType && ['Asset account', 'Loan', 'Debt', 'Mortgage'].includes(sourceType)) { |             if (sourceType === destType && ['Asset account', 'Loan', 'Debt', 'Mortgage'].includes(sourceType)) { | ||||||
|                 this.transactionType = 'transfer'; |                 this.groupProperties.transactionType = 'transfer'; | ||||||
|                 console.log('Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|  |  | ||||||
|                 // this also locks the amount into the amount of the source account |                 // this also locks the amount into the amount of the source account | ||||||
|                 // and the foreign amount (if different) in that of the destination account. |                 // and the foreign amount (if different) in that of the destination account. | ||||||
| @@ -247,38 +288,39 @@ let transactions = function () { | |||||||
|             } |             } | ||||||
|             // withdrawals: |             // withdrawals: | ||||||
|             if ('Asset account' === sourceType && ['Expense account', 'Debt', 'Loan', 'Mortgage'].includes(destType)) { |             if ('Asset account' === sourceType && ['Expense account', 'Debt', 'Loan', 'Mortgage'].includes(destType)) { | ||||||
|                 this.transactionType = 'withdrawal'; |                 this.groupProperties.transactionType = 'withdrawal'; | ||||||
|                 console.log('[a] Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('[a] Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); |                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if ('Asset account' === sourceType && 'unknown' === destType) { |             if ('Asset account' === sourceType && 'unknown' === destType) { | ||||||
|                 this.transactionType = 'withdrawal'; |                 this.groupProperties.transactionType = 'withdrawal'; | ||||||
|                 console.log('[b] Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('[b] Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|                 console.log(this.entries[0].source_account); |                 console.log(this.entries[0].source_account); | ||||||
|                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); |                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (['Debt', 'Loan', 'Mortgage'].includes(sourceType) && 'Expense account' === destType) { |             if (['Debt', 'Loan', 'Mortgage'].includes(sourceType) && 'Expense account' === destType) { | ||||||
|                 this.transactionType = 'withdrawal'; |                 this.groupProperties.transactionType = 'withdrawal'; | ||||||
|                 console.log('[c] Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('[c] Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); |                 this.filterNativeCurrencies(this.entries[0].source_account.currency_code); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             // deposits: |             // deposits: | ||||||
|             if ('Revenue account' === sourceType && ['Asset account', 'Debt', 'Loan', 'Mortgage'].includes(destType)) { |             if ('Revenue account' === sourceType && ['Asset account', 'Debt', 'Loan', 'Mortgage'].includes(destType)) { | ||||||
|                 this.transactionType = 'deposit'; |                 this.groupProperties.transactionType = 'deposit'; | ||||||
|                 console.log('Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (['Debt', 'Loan', 'Mortgage'].includes(sourceType) && 'Asset account' === destType) { |             if (['Debt', 'Loan', 'Mortgage'].includes(sourceType) && 'Asset account' === destType) { | ||||||
|                 this.transactionType = 'deposit'; |                 this.groupProperties.transactionType = 'deposit'; | ||||||
|                 console.log('Transaction type is detected to be "' + this.transactionType + '".'); |                 console.log('Transaction type is detected to be "' + this.groupProperties.transactionType + '".'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             console.warn('Unknown account combination between "' + sourceType + '" and "' + destType + '".'); |             console.warn('Unknown account combination between "' + sourceType + '" and "' + destType + '".'); | ||||||
|         }, |         }, | ||||||
|  |  | ||||||
|         selectSourceAccount(item, ac) { |         selectSourceAccount(item, ac) { | ||||||
|             const index = parseInt(ac._searchInput.attributes['data-index'].value); |             const index = parseInt(ac._searchInput.attributes['data-index'].value); | ||||||
|             document.querySelector('#form')._x_dataStack[0].$data.entries[index].source_account = { |             document.querySelector('#form')._x_dataStack[0].$data.entries[index].source_account = { | ||||||
| @@ -366,160 +408,6 @@ let transactions = function () { | |||||||
|             console.log('Changed destination account into a known ' + item.type.toLowerCase()); |             console.log('Changed destination account into a known ' + item.type.toLowerCase()); | ||||||
|             document.querySelector('#form')._x_dataStack[0].detectTransactionType(); |             document.querySelector('#form')._x_dataStack[0].detectTransactionType(); | ||||||
|         }, |         }, | ||||||
|         loadCurrencies() { |  | ||||||
|             this.enabledCurrencies = []; |  | ||||||
|             this.nativeCurrencies = []; |  | ||||||
|             this.foreignCurrencies = []; |  | ||||||
|  |  | ||||||
|             this.foreignCurrencies.push({ |  | ||||||
|                 id: 0, name: '(no foreign currency)', code: '', default: false, symbol: '', decimal_places: 2, |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             console.log('Loading user currencies.'); |  | ||||||
|             let params = { |  | ||||||
|                 page: 1, limit: 1337 |  | ||||||
|             }; |  | ||||||
|             let getter = new Get(); |  | ||||||
|             getter.list({}).then((response) => { |  | ||||||
|                 for (let i in response.data.data) { |  | ||||||
|                     if (response.data.data.hasOwnProperty(i)) { |  | ||||||
|                         let current = response.data.data[i]; |  | ||||||
|                         if (current.attributes.enabled) { |  | ||||||
|                             let obj = |  | ||||||
|  |  | ||||||
|                                 { |  | ||||||
|                                     id: current.id, |  | ||||||
|                                     name: current.attributes.name, |  | ||||||
|                                     code: current.attributes.code, |  | ||||||
|                                     default: current.attributes.default, |  | ||||||
|                                     symbol: current.attributes.symbol, |  | ||||||
|                                     decimal_places: current.attributes.decimal_places, |  | ||||||
|  |  | ||||||
|                                 }; |  | ||||||
|                             if (obj.default) { |  | ||||||
|                                 this.defaultCurrency = obj; |  | ||||||
|                             } |  | ||||||
|                             this.enabledCurrencies.push(obj); |  | ||||||
|                             this.nativeCurrencies.push(obj); |  | ||||||
|                             this.foreignCurrencies.push(obj); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 this.loadingCurrencies = false; |  | ||||||
|             }); |  | ||||||
|         }, |  | ||||||
|         loadBudgets() { |  | ||||||
|             this.budgets = []; |  | ||||||
|  |  | ||||||
|             this.budgets.push({ |  | ||||||
|                 id: 0, name: '(no budget)', |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             console.log('Loading user budgets.'); |  | ||||||
|             let params = { |  | ||||||
|                 page: 1, limit: 1337 |  | ||||||
|             }; |  | ||||||
|             let getter = new BudgetGet(); |  | ||||||
|             getter.list({}).then((response) => { |  | ||||||
|                 for (let i in response.data.data) { |  | ||||||
|                     if (response.data.data.hasOwnProperty(i)) { |  | ||||||
|                         let current = response.data.data[i]; |  | ||||||
|                         let obj = { |  | ||||||
|                             id: current.id, name: current.attributes.name, |  | ||||||
|                         }; |  | ||||||
|                         this.budgets.push(obj); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 this.loadingBudgets = false; |  | ||||||
|                 console.log(this.budgets); |  | ||||||
|             }); |  | ||||||
|         }, |  | ||||||
|         loadPiggyBanks() { |  | ||||||
|             this.piggyBanks = {}; |  | ||||||
|             let tempObject = { |  | ||||||
|                 '0': { |  | ||||||
|                     id: 0, name: '(no group)', order: 0, piggyBanks: [{ |  | ||||||
|                         id: 0, name: '(no piggy bank)', order: 0, |  | ||||||
|                     }] |  | ||||||
|                 } |  | ||||||
|             }; |  | ||||||
|             console.log('Loading user piggy banks.'); |  | ||||||
|             let params = { |  | ||||||
|                 page: 1, limit: 1337 |  | ||||||
|             }; |  | ||||||
|             let getter = new PiggyBankGet(); |  | ||||||
|             getter.list({}).then((response) => { |  | ||||||
|                 for (let i in response.data.data) { |  | ||||||
|                     if (response.data.data.hasOwnProperty(i)) { |  | ||||||
|                         let current = response.data.data[i]; |  | ||||||
|                         let objectGroupId = current.attributes.object_group_id ?? '0'; |  | ||||||
|                         let objectGroupTitle = current.attributes.object_group_title ?? '(no group)'; |  | ||||||
|                         let piggyBank = { |  | ||||||
|                             id: current.id, name: current.attributes.name, order: current.attributes.order, |  | ||||||
|                         }; |  | ||||||
|                         if (!tempObject.hasOwnProperty(objectGroupId)) { |  | ||||||
|                             tempObject[objectGroupId] = { |  | ||||||
|                                 id: objectGroupId, |  | ||||||
|                                 name: objectGroupTitle, |  | ||||||
|                                 order: current.attributes.object_group_order ?? 0, |  | ||||||
|                                 piggyBanks: [] |  | ||||||
|                             }; |  | ||||||
|                         } |  | ||||||
|                         tempObject[objectGroupId].piggyBanks.push(piggyBank); |  | ||||||
|                         tempObject[objectGroupId].piggyBanks.sort((a, b) => a.order - b.order); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 //tempObject.sort((a,b) => a.order - b.order); |  | ||||||
|                 this.loadingPiggyBanks = false; |  | ||||||
|                 this.piggyBanks = Object.keys(tempObject).sort().reduce((obj, key) => { |  | ||||||
|                     obj[key] = tempObject[key]; |  | ||||||
|                     return obj; |  | ||||||
|                 }, {}); |  | ||||||
|             }); |  | ||||||
|         }, |  | ||||||
|         loadSubscriptions() { |  | ||||||
|             this.subscriptions = {}; |  | ||||||
|             let tempObject = { |  | ||||||
|                 '0': { |  | ||||||
|                     id: 0, name: '(no group)', order: 0, subscriptions: [{ |  | ||||||
|                         id: 0, name: '(no subscription)', order: 0, |  | ||||||
|                     }] |  | ||||||
|                 } |  | ||||||
|             }; |  | ||||||
|             console.log('Loading user suscriptions.'); |  | ||||||
|             let params = { |  | ||||||
|                 page: 1, limit: 1337 |  | ||||||
|             }; |  | ||||||
|             let getter = new SubscriptionGet(); |  | ||||||
|             getter.list({}).then((response) => { |  | ||||||
|                 for (let i in response.data.data) { |  | ||||||
|                     if (response.data.data.hasOwnProperty(i)) { |  | ||||||
|                         let current = response.data.data[i]; |  | ||||||
|                         let objectGroupId = current.attributes.object_group_id ?? '0'; |  | ||||||
|                         let objectGroupTitle = current.attributes.object_group_title ?? '(no group)'; |  | ||||||
|                         let piggyBank = { |  | ||||||
|                             id: current.id, name: current.attributes.name, order: current.attributes.order, |  | ||||||
|                         }; |  | ||||||
|                         if (!tempObject.hasOwnProperty(objectGroupId)) { |  | ||||||
|                             tempObject[objectGroupId] = { |  | ||||||
|                                 id: objectGroupId, |  | ||||||
|                                 name: objectGroupTitle, |  | ||||||
|                                 order: current.attributes.object_group_order ?? 0, |  | ||||||
|                                 subscriptions: [] |  | ||||||
|                             }; |  | ||||||
|                         } |  | ||||||
|                         tempObject[objectGroupId].subscriptions.push(piggyBank); |  | ||||||
|                         tempObject[objectGroupId].subscriptions.sort((a, b) => a.order - b.order); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 //tempObject.sort((a,b) => a.order - b.order); |  | ||||||
|                 this.loadingSubscriptions = false; |  | ||||||
|                 this.subscriptions = Object.keys(tempObject).sort().reduce((obj, key) => { |  | ||||||
|                     obj[key] = tempObject[key]; |  | ||||||
|                     return obj; |  | ||||||
|                 }, {}); |  | ||||||
|             }); |  | ||||||
|         }, |  | ||||||
|         changeSourceAccount(item, ac) { |         changeSourceAccount(item, ac) { | ||||||
|             console.log('changeSourceAccount'); |             console.log('changeSourceAccount'); | ||||||
|             if (typeof item === 'undefined') { |             if (typeof item === 'undefined') { | ||||||
| @@ -665,12 +553,28 @@ let transactions = function () { | |||||||
|                 loadTranslations(i18n, locale).then(() => { |                 loadTranslations(i18n, locale).then(() => { | ||||||
|                     this.addSplit(); |                     this.addSplit(); | ||||||
|                 }); |                 }); | ||||||
|  |  | ||||||
|             }); |             }); | ||||||
|             this.loadCurrencies(); |             // load currencies and save in form data. | ||||||
|             this.loadBudgets(); |             loadCurrencies().then(data => { | ||||||
|             this.loadPiggyBanks(); |                 this.formStates.loadingCurrencies = false; | ||||||
|             this.loadSubscriptions(); |                 this.formData.defaultCurrency = data.defaultCurrency; | ||||||
|  |                 this.formData.enabledCurrencies = data.enabledCurrencies; | ||||||
|  |                 this.formData.nativeCurrencies = data.nativeCurrencies; | ||||||
|  |                 this.formData.foreignCurrencies = data.foreignCurrencies; | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             loadBudgets().then(data => { | ||||||
|  |                 this.formData.budgets = data; | ||||||
|  |                 this.formStates.loadingBudgets = false; | ||||||
|  |             }); | ||||||
|  |             loadPiggyBanks().then(data => { | ||||||
|  |                 this.formData.piggyBanks = data; | ||||||
|  |                 this.formStates.loadingPiggyBanks = false; | ||||||
|  |             }); | ||||||
|  |             loadSubscriptions().then(data => { | ||||||
|  |                 this.formData.subscriptions = data; | ||||||
|  |                 this.formStates.loadingSubscriptions = false; | ||||||
|  |             }); | ||||||
|  |  | ||||||
|             document.addEventListener('upload-success', (event) => { |             document.addEventListener('upload-success', (event) => { | ||||||
|                 this.processUpload(event); |                 this.processUpload(event); | ||||||
| @@ -691,7 +595,7 @@ let transactions = function () { | |||||||
|             this.detectTransactionType(); |             this.detectTransactionType(); | ||||||
|  |  | ||||||
|             // parse transaction: |             // parse transaction: | ||||||
|             let transactions = parseFromEntries(this.entries, this.transactionType); |             let transactions = parseFromEntries(this.entries, this.groupProperties.transactionType); | ||||||
|             let submission = { |             let submission = { | ||||||
|                 // todo process all options |                 // todo process all options | ||||||
|                 group_title: null, fire_webhooks: false, apply_rules: false, transactions: transactions |                 group_title: null, fire_webhooks: false, apply_rules: false, transactions: transactions | ||||||
|   | |||||||
| @@ -37,7 +37,15 @@ let i18n; | |||||||
|  |  | ||||||
| let transactions = function () { | let transactions = function () { | ||||||
|     return { |     return { | ||||||
|  |         // transactions are stored in "entries": | ||||||
|         entries: [], |         entries: [], | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         init() { |         init() { | ||||||
|             Promise.all([getVariable('language', 'en_US')]).then((values) => { |             Promise.all([getVariable('language', 'en_US')]).then((values) => { | ||||||
|                 i18n = new I18n(); |                 i18n = new I18n(); | ||||||
|   | |||||||
| @@ -0,0 +1,46 @@ | |||||||
|  | /* | ||||||
|  |  * load-budgets.js | ||||||
|  |  * Copyright (c) 2024 james@firefly-iii.org | ||||||
|  |  * | ||||||
|  |  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU Affero General Public License as | ||||||
|  |  * published by the Free Software Foundation, either version 3 of the | ||||||
|  |  * License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU Affero General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU Affero General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import Get from "../../../api/v2/model/budget/get.js"; | ||||||
|  |  | ||||||
|  | export function loadBudgets() { | ||||||
|  |     let params = { | ||||||
|  |         page: 1, limit: 1337 | ||||||
|  |     }; | ||||||
|  |     let getter = new Get(); | ||||||
|  |     return getter.list(params).then((response) => { | ||||||
|  |         let returnData = [{ | ||||||
|  |             id: 0, name: '(no budget)', | ||||||
|  |         }]; | ||||||
|  |  | ||||||
|  |         for (let i in response.data.data) { | ||||||
|  |             if (response.data.data.hasOwnProperty(i)) { | ||||||
|  |                 let current = response.data.data[i]; | ||||||
|  |                 let obj = { | ||||||
|  |                     id: current.id, name: current.attributes.name, | ||||||
|  |                 }; | ||||||
|  |                 returnData.push(obj); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return returnData; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,65 @@ | |||||||
|  | /* | ||||||
|  |  * load-currencies.js | ||||||
|  |  * Copyright (c) 2024 james@firefly-iii.org | ||||||
|  |  * | ||||||
|  |  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU Affero General Public License as | ||||||
|  |  * published by the Free Software Foundation, either version 3 of the | ||||||
|  |  * License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU Affero General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU Affero General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import Get from "../../../api/v2/model/currency/get.js"; | ||||||
|  |  | ||||||
|  | export function loadCurrencies() { | ||||||
|  |     let params = { | ||||||
|  |         page: 1, limit: 1337 | ||||||
|  |     }; | ||||||
|  |     let getter = new Get(); | ||||||
|  |     return getter.list(params).then((response) => { | ||||||
|  |         let returnData = { | ||||||
|  |             defaultCurrency: {}, | ||||||
|  |             nativeCurrencies: [], | ||||||
|  |             foreignCurrencies: [], | ||||||
|  |             enabledCurrencies: [], | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         returnData.foreignCurrencies.push({ | ||||||
|  |             id: 0, name: '(no foreign currency)', code: '', default: false, symbol: '', decimal_places: 2, | ||||||
|  |         }); | ||||||
|  |         for (let i in response.data.data) { | ||||||
|  |             if (response.data.data.hasOwnProperty(i)) { | ||||||
|  |                 let current = response.data.data[i]; | ||||||
|  |                 if (current.attributes.enabled) { | ||||||
|  |                     let obj = | ||||||
|  |                         { | ||||||
|  |                             id: current.id, | ||||||
|  |                             name: current.attributes.name, | ||||||
|  |                             code: current.attributes.code, | ||||||
|  |                             default: current.attributes.default, | ||||||
|  |                             symbol: current.attributes.symbol, | ||||||
|  |                             decimal_places: current.attributes.decimal_places, | ||||||
|  |  | ||||||
|  |                         }; | ||||||
|  |                     if (obj.default) { | ||||||
|  |                         returnData.defaultCurrency = obj; | ||||||
|  |                     } | ||||||
|  |                     returnData.enabledCurrencies.push(obj); | ||||||
|  |                     returnData.nativeCurrencies.push(obj); | ||||||
|  |                     returnData.foreignCurrencies.push(obj); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return returnData; | ||||||
|  |     }); | ||||||
|  | } | ||||||
| @@ -0,0 +1,63 @@ | |||||||
|  | /* | ||||||
|  |  * load-piggy-banks.js | ||||||
|  |  * Copyright (c) 2024 james@firefly-iii.org | ||||||
|  |  * | ||||||
|  |  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU Affero General Public License as | ||||||
|  |  * published by the Free Software Foundation, either version 3 of the | ||||||
|  |  * License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU Affero General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU Affero General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | import Get from "../../../api/v2/model/piggy-bank/get.js"; | ||||||
|  |  | ||||||
|  | export function loadPiggyBanks() { | ||||||
|  |     let params = { | ||||||
|  |         page: 1, limit: 1337 | ||||||
|  |     }; | ||||||
|  |     let getter = new Get(); | ||||||
|  |     return getter.list(params).then((response) => { | ||||||
|  |         let piggyBanks = { | ||||||
|  |             '0': { | ||||||
|  |                 id: 0, name: '(no group)', order: 0, piggyBanks: [{ | ||||||
|  |                     id: 0, name: '(no piggy bank)', order: 0, | ||||||
|  |                 }] | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         for (let i in response.data.data) { | ||||||
|  |             if (response.data.data.hasOwnProperty(i)) { | ||||||
|  |                 let current = response.data.data[i]; | ||||||
|  |                 let objectGroupId = current.attributes.object_group_id ?? '0'; | ||||||
|  |                 let objectGroupTitle = current.attributes.object_group_title ?? '(no group)'; | ||||||
|  |                 let piggyBank = { | ||||||
|  |                     id: current.id, name: current.attributes.name, order: current.attributes.order, | ||||||
|  |                 }; | ||||||
|  |                 if (!piggyBanks.hasOwnProperty(objectGroupId)) { | ||||||
|  |                     piggyBanks[objectGroupId] = { | ||||||
|  |                         id: objectGroupId, | ||||||
|  |                         name: objectGroupTitle, | ||||||
|  |                         order: current.attributes.object_group_order ?? 0, | ||||||
|  |                         piggyBanks: [] | ||||||
|  |                     }; | ||||||
|  |                 } | ||||||
|  |                 piggyBanks[objectGroupId].piggyBanks.push(piggyBank); | ||||||
|  |                 piggyBanks[objectGroupId].piggyBanks.sort((a, b) => a.order - b.order); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         //tempObject.sort((a,b) => a.order - b.order); | ||||||
|  |         return Object.keys(piggyBanks).sort().reduce((obj, key) => { | ||||||
|  |             obj[key] = piggyBanks[key]; | ||||||
|  |             return obj; | ||||||
|  |         }, {}); | ||||||
|  |     }); | ||||||
|  | } | ||||||
| @@ -0,0 +1,61 @@ | |||||||
|  | /* | ||||||
|  |  * load-subscriptions.js | ||||||
|  |  * Copyright (c) 2024 james@firefly-iii.org | ||||||
|  |  * | ||||||
|  |  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU Affero General Public License as | ||||||
|  |  * published by the Free Software Foundation, either version 3 of the | ||||||
|  |  * License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU Affero General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU Affero General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | import SubscriptionGet from "../../../api/v2/model/subscription/get.js"; | ||||||
|  |  | ||||||
|  | export function loadSubscriptions() { | ||||||
|  |     let params = { | ||||||
|  |         page: 1, limit: 1337 | ||||||
|  |     }; | ||||||
|  |     let getter = new SubscriptionGet(); | ||||||
|  |     return getter.list(params).then((response) => { | ||||||
|  |         let subscriptions = { | ||||||
|  |             '0': { | ||||||
|  |                 id: 0, name: '(no group)', order: 0, subscriptions: [{ | ||||||
|  |                     id: 0, name: '(no subscription)', order: 0, | ||||||
|  |                 }] | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         for (let i in response.data.data) { | ||||||
|  |             if (response.data.data.hasOwnProperty(i)) { | ||||||
|  |                 let current = response.data.data[i]; | ||||||
|  |                 let objectGroupId = current.attributes.object_group_id ?? '0'; | ||||||
|  |                 let objectGroupTitle = current.attributes.object_group_title ?? '(no group)'; | ||||||
|  |                 let piggyBank = { | ||||||
|  |                     id: current.id, name: current.attributes.name, order: current.attributes.order, | ||||||
|  |                 }; | ||||||
|  |                 if (!subscriptions.hasOwnProperty(objectGroupId)) { | ||||||
|  |                     subscriptions[objectGroupId] = { | ||||||
|  |                         id: objectGroupId, | ||||||
|  |                         name: objectGroupTitle, | ||||||
|  |                         order: current.attributes.object_group_order ?? 0, | ||||||
|  |                         subscriptions: [] | ||||||
|  |                     }; | ||||||
|  |                 } | ||||||
|  |                 subscriptions[objectGroupId].subscriptions.push(piggyBank); | ||||||
|  |                 subscriptions[objectGroupId].subscriptions.sort((a, b) => a.order - b.order); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Object.keys(subscriptions).sort().reduce((obj, key) => { | ||||||
|  |             obj[key] = subscriptions[key]; | ||||||
|  |             return obj; | ||||||
|  |         }, {}); | ||||||
|  |     }); | ||||||
|  | } | ||||||
| @@ -1268,6 +1268,7 @@ return [ | |||||||
|     'sums_apply_to_range'                       => 'All sums apply to the selected range', |     'sums_apply_to_range'                       => 'All sums apply to the selected range', | ||||||
|     'mapbox_api_key'                            => 'To use map, get an API key from <a href="https://www.mapbox.com/">Mapbox</a>. Open your <code>.env</code> file and enter this code after <code>MAPBOX_API_KEY=</code>.', |     'mapbox_api_key'                            => 'To use map, get an API key from <a href="https://www.mapbox.com/">Mapbox</a>. Open your <code>.env</code> file and enter this code after <code>MAPBOX_API_KEY=</code>.', | ||||||
|     'press_object_location'                     => 'Right click or long press to set the object\'s location.', |     'press_object_location'                     => 'Right click or long press to set the object\'s location.', | ||||||
|  |     'click_tap_location'                        => 'Click or tap the map to add a location', | ||||||
|     'clear_location'                            => 'Clear location', |     'clear_location'                            => 'Clear location', | ||||||
|     'delete_all_selected_tags'                  => 'Delete all selected tags', |     'delete_all_selected_tags'                  => 'Delete all selected tags', | ||||||
|     'select_tags_to_delete'                     => 'Don\'t forget to select some tags.', |     'select_tags_to_delete'                     => 'Don\'t forget to select some tags.', | ||||||
| @@ -1895,6 +1896,8 @@ return [ | |||||||
|     // Ignore this comment
 |     // Ignore this comment
 | ||||||
| 
 | 
 | ||||||
|     // transactions:
 |     // transactions:
 | ||||||
|  |     'amount_foreign_if'                         => 'Amount in foreign currency, if any', | ||||||
|  |     'amount_destination_account'                => 'Amount in the currency of the destination account', | ||||||
|     'edit_transaction_title'                    => 'Edit transaction ":description"', |     'edit_transaction_title'                    => 'Edit transaction ":description"', | ||||||
|     'unreconcile'                               => 'Undo reconciliation', |     'unreconcile'                               => 'Undo reconciliation', | ||||||
|     'update_withdrawal'                         => 'Update withdrawal', |     'update_withdrawal'                         => 'Update withdrawal', | ||||||
|   | |||||||
| @@ -1,14 +1,14 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|  |     <!-- select for currency --> | ||||||
|     <div class="col-sm-3"> |     <div class="col-sm-3"> | ||||||
|         <template x-if="loadingCurrencies"> |         <!-- is loading currencies --> | ||||||
|                                                     <span class="form-control-plaintext"><em |         <template x-if="formStates.loadingCurrencies"> | ||||||
|                                                             class="fa-solid fa-spinner fa-spin"></em></span> |             <span class="form-control-plaintext"><em class="fa-solid fa-spinner fa-spin"></em></span> | ||||||
|         </template> |         </template> | ||||||
|         <template x-if="!loadingCurrencies"> |         <!-- is no longer loading currencies --> | ||||||
|             <select class="form-control" :id="'currency_code_' + index" |         <template x-if="!formStates.loadingCurrencies"> | ||||||
|                     x-model="transaction.currency_code" |             <select class="form-control" :id="'currency_code_' + index" x-model="transaction.currency_code"> | ||||||
|             > |                 <template x-for="currency in formData.nativeCurrencies"> | ||||||
|                 <template x-for="currency in nativeCurrencies"> |  | ||||||
|                     <option :selected="currency.id == defaultCurrency.id" |                     <option :selected="currency.id == defaultCurrency.id" | ||||||
|                             :label="currency.name" :value="currency.code" |                             :label="currency.name" :value="currency.code" | ||||||
|                             x-text="currency.name"></option> |                             x-text="currency.name"></option> | ||||||
| @@ -16,12 +16,13 @@ | |||||||
|             </select> |             </select> | ||||||
|         </template> |         </template> | ||||||
|     </div> |     </div> | ||||||
|  |     <!-- actual amount --> | ||||||
|     <div class="col-sm-9"> |     <div class="col-sm-9"> | ||||||
|         <input type="number" step="any" min="0" |         <input type="number" step="any" min="0" | ||||||
|                :id="'amount_' + index" |                :id="'amount_' + index" | ||||||
|                :data-index="index" |                :data-index="index" | ||||||
|                :class="{'is-invalid': transaction.errors.amount.length > 0, 'input-mask' : true, 'form-control': true}" |                :class="{'is-invalid': transaction.errors.amount.length > 0, 'input-mask' : true, 'form-control': true}" | ||||||
|                x-model="transaction.amount" data-inputmask="currency" |                x-model="transaction.amount" | ||||||
|                @change="changedAmount" |                @change="changedAmount" | ||||||
|                placeholder="0.00"> |                placeholder="0.00"> | ||||||
|         <template x-if="transaction.errors.amount.length > 0"> |         <template x-if="transaction.errors.amount.length > 0"> | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
|  | @if(true === $optionalFields['attachments']) | ||||||
|     <div class="row mb-3"> |     <div class="row mb-3"> | ||||||
|         <label :for="'attachments_' + index" |         <label :for="'attachments_' + index" | ||||||
|                class="col-sm-1 col-form-label d-none d-sm-block"> |                class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-file-import"></i> |             <em title="{{ __('firefly.attachments') }}" class="fa-solid fa-file-import"></em> | ||||||
|         </label> |         </label> | ||||||
|         <div class="col-sm-10"> |         <div class="col-sm-10"> | ||||||
|             <input type="file" multiple |             <input type="file" multiple | ||||||
| @@ -9,6 +10,7 @@ | |||||||
|                    :id="'attachments_' + index" |                    :id="'attachments_' + index" | ||||||
|                    :data-index="index" |                    :data-index="index" | ||||||
|                    name="attachments[]" |                    name="attachments[]" | ||||||
|                placeholder="{{ __('firefly.category')  }}"> |                    > | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  | @endif | ||||||
|   | |||||||
| @@ -1,20 +1,19 @@ | |||||||
| <template x-if="transactionType != 'deposit' && transactionType != 'transfer'"> | <template x-if="groupProperties.transactionType != 'deposit' && groupProperties.transactionType != 'transfer'"> | ||||||
|     <div class="row mb-3"> |     <div class="row mb-3"> | ||||||
|         <label :for="'budget_id_' + index" |         <label :for="'budget_id_' + index" | ||||||
|                class="col-sm-1 col-form-label d-none d-sm-block"> |                class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|             <i class="fa-solid fa-chart-pie"></i> |             <em title="{{ __('firefly.budget') }}" class="fa-solid fa-chart-pie"></em> | ||||||
|         </label> |         </label> | ||||||
|         <div class="col-sm-10"> |         <div class="col-sm-10"> | ||||||
|             <template x-if="loadingBudgets"> |             <template x-if="formStates.loadingBudgets"> | ||||||
|                                                         <span class="form-control-plaintext"><em |                 <span class="form-control-plaintext"><em class="fa-solid fa-spinner fa-spin"></em></span> | ||||||
|                                                                 class="fa-solid fa-spinner fa-spin"></em></span> |  | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="!loadingBudgets"> |             <template x-if="!formStates.loadingBudgets"> | ||||||
|                 <select class="form-control" |                 <select class="form-control" | ||||||
|                         :id="'budget_id_' + index" |                         :id="'budget_id_' + index" | ||||||
|                         x-model="transaction.budget_id" |                         x-model="transaction.budget_id" | ||||||
|                 > |                 > | ||||||
|                     <template x-for="budget in budgets"> |                     <template x-for="budget in formData.budgets"> | ||||||
|                         <option :label="budget.name" :value="budget.id" |                         <option :label="budget.name" :value="budget.id" | ||||||
|                                 x-text="budget.name"></option> |                                 x-text="budget.name"></option> | ||||||
|                     </template> |                     </template> | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'category_name_' + index" |     <label :for="'category_name_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |         <em title="{{ __('firefly.category') }}" class="fa-solid fa-bookmark"></em> | ||||||
|         <i class="fa-solid fa-bookmark"></i> |  | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="search" |         <input type="search" | ||||||
|   | |||||||
| @@ -1,16 +1,18 @@ | |||||||
| <template x-for="dateField in dateFields"> | @foreach($optionalDateFields as $name => $enabled) | ||||||
|  |     @if($enabled) | ||||||
|         <div class="row mb-1"> |         <div class="row mb-1"> | ||||||
|         <label :for="dateField + '_date_' + index" |             <label :for="'{{ $name }}_' + index" | ||||||
|                    class="col-sm-1 col-form-label d-none d-sm-block"> |                    class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|             <i class="fa-solid fa-calendar-alt" :title="dateField"></i> |                 <em class="fa-solid fa-calendar-alt" title="{{ __('firefly.pref_optional_tj_' . $name) }}"></em> | ||||||
|             </label> |             </label> | ||||||
|             <div class="col-sm-10"> |             <div class="col-sm-10"> | ||||||
|                 <input type="date" |                 <input type="date" | ||||||
|                        class="form-control" |                        class="form-control" | ||||||
|                    :id="dateField + '_date_' + index" |                        :id="'{{ $name }}_' + index" | ||||||
|                    x-model="transaction[dateField]" |                        x-model="transaction.{{ $name }}" | ||||||
|                        :data-index="index" |                        :data-index="index" | ||||||
|                        placeholder=""> |                        placeholder=""> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| </template> |     @endif | ||||||
|  | @endforeach | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label for="date_0" class="col-sm-1 col-form-label d-none d-sm-block"> |     <label :for="'date_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-calendar"></i> |         <em title="{{ __('firefly.date_and_time') }}" class="fa-solid fa-calendar"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="datetime-local" class="form-control" :id="'date_' + index" |         <input type="datetime-local" class="form-control" :id="'date_' + index" | ||||||
|                @change="detectTransactionType" |                @change="changedDateTime" | ||||||
|                x-model="transaction.date" |                x-model="transaction.date" | ||||||
|         > |         > | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -1,11 +1,11 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'description_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> |     <label :for="'description_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <em title="TODO explain me" class="fa-solid fa-font"></em> |         <em title="{{ __('firefly.description') }}" class="fa-solid fa-font"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="text" class="form-control ac-description" |         <input type="text" class="form-control ac-description" | ||||||
|                :id="'description_' + index" |                :id="'description_' + index" | ||||||
|                @change="detectTransactionType" |                @change="changedDescription" | ||||||
|                x-model="transaction.description" |                x-model="transaction.description" | ||||||
|                :data-index="index" |                :data-index="index" | ||||||
|                placeholder="{{ __('firefly.description')  }}"> |                placeholder="{{ __('firefly.description')  }}"> | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'dest_' + index" |     <label :for="'dest_' + index" | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |            class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-arrow-left"></i> |         <em title="{{ __('firefly.destination_account') }}" class="fa-solid fa-arrow-left"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="text" |         <input type="text" | ||||||
|                class="form-control ac-dest" |                class="form-control ac-dest" | ||||||
|                :id="'dest_' + index" |                :id="'dest_' + index" | ||||||
|  |                @changed="changedDestinationAccount" | ||||||
|                x-model="transaction.destination_account.alpine_name" |                x-model="transaction.destination_account.alpine_name" | ||||||
|                :data-index="index" |                :data-index="index" | ||||||
|                placeholder="{{ __('firefly.destination_account')  }}"> |                placeholder="{{ __('firefly.destination_account')  }}"> | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
|  | @if(true === $optionalFields['external_url']) | ||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'external_url_' + index" |     <label :for="'external_url_' + index" | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |            class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-link"></i> |         <em title="{{ __('firefly.external_url') }}" class="fa-solid fa-link"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="text" |         <input type="text" | ||||||
| @@ -19,3 +20,4 @@ | |||||||
|         </template> |         </template> | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  | @endif | ||||||
|   | |||||||
| @@ -1,17 +1,15 @@ | |||||||
| <template x-if="foreignAmountEnabled"> | <template x-if="formBehaviour.foreignCurrencyEnabled"> | ||||||
|     <div class="row mb-3"> |     <div class="row mb-3"> | ||||||
|         <div class="col-sm-3"> |         <div class="col-sm-3"> | ||||||
|             <label class="form-label"> </label> |             <label class="form-label"> </label> | ||||||
|             <template x-if="loadingCurrencies"> |             <template x-if="formStates.loadingCurrencies"> | ||||||
|                                                         <span class="form-control-plaintext"><em |                 <span class="form-control-plaintext"><em class="fa-solid fa-spinner fa-spin"></em></span> | ||||||
|                                                                 class="fa-solid fa-spinner fa-spin"></em></span> |  | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="!loadingCurrencies"> |             <template x-if="!formStates.loadingCurrencies"> | ||||||
|                 <select class="form-control" |                 <select class="form-control" | ||||||
|                         :id="'foreign_currency_code_' + index" |                         :id="'foreign_currency_code_' + index" | ||||||
|                         x-model="transaction.foreign_currency_code" |                         x-model="transaction.foreign_currency_code"> | ||||||
|                 > |                     <template x-for="currency in formData.foreignCurrencies"> | ||||||
|                     <template x-for="currency in foreignCurrencies"> |  | ||||||
|                         <option :label="currency.name" :value="currency.code" |                         <option :label="currency.name" :value="currency.code" | ||||||
|                                 x-text="currency.name"></option> |                                 x-text="currency.name"></option> | ||||||
|                     </template> |                     </template> | ||||||
| @@ -19,26 +17,21 @@ | |||||||
|             </template> |             </template> | ||||||
|         </div> |         </div> | ||||||
|         <div class="col-sm-9"> |         <div class="col-sm-9"> | ||||||
|             <template x-if="transactionType != 'transfer'"> |             <template x-if="groupProperties.transactionType != 'transfer'"> | ||||||
|                 <label class="small form-label">Amount in foreign amount, if |                 <label class="small form-label">{{ __('firefly.amount_foreign_if') }}</label> | ||||||
|                     any</label> |  | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="transactionType == 'transfer'"> |             <template x-if="groupProperties.transactionType == 'transfer'"> | ||||||
|                 <label class="small form-label">Amount in currency of |                 <label class="small form-label">{{ __('firefly.amount_destination_account') }}</label> | ||||||
|                     destination |  | ||||||
|                     account</label> |  | ||||||
|             </template> |             </template> | ||||||
|             <input type="number" step="any" min="0" |             <input type="number" step="any" min="0" | ||||||
|                    :id="'amount_' + index" |                    :id="'amount_' + index" | ||||||
|                    :data-index="index" |                    :data-index="index" | ||||||
|                    :class="{'is-invalid': transaction.errors.foreign_amount.length > 0, 'input-mask' : true, 'form-control': true}" |                    :class="{'is-invalid': transaction.errors.foreign_amount.length > 0, 'input-mask' : true, 'form-control': true}" | ||||||
|                    x-model="transaction.foreign_amount" |                    x-model="transaction.foreign_amount" | ||||||
|                    data-inputmask="currency" |  | ||||||
|                    @change="changedAmount" |                    @change="changedAmount" | ||||||
|                    placeholder="0.00"> |                    placeholder="0.00"> | ||||||
|             <template x-if="transaction.errors.foreign_amount.length > 0"> |             <template x-if="transaction.errors.foreign_amount.length > 0"> | ||||||
|                 <div class="invalid-feedback" |                 <div class="invalid-feedback" x-text="transaction.errors.foreign_amount[0]"></div> | ||||||
|                      x-text="transaction.errors.foreign_amount[0]"></div> |  | ||||||
|             </template> |             </template> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
|  | @if(true === $optionalFields['internal_reference']) | ||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'internal_reference_' + index" |     <label :for="'internal_reference_' + index" | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |            class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-anchor"></i> |         <em title="{{ __('firefly.internal_reference') }}" class="fa-solid fa-anchor"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="search" |         <input type="search" | ||||||
| @@ -12,3 +13,4 @@ | |||||||
|                placeholder="{{ __('firefly.internal_reference')  }}"> |                placeholder="{{ __('firefly.internal_reference')  }}"> | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  | @endif | ||||||
|   | |||||||
| @@ -1,18 +1,18 @@ | |||||||
|  | @if(true === $optionalFields['location']) | ||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'map_' + index" |     <label :for="'map_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |         <em title="{{ __('firefly.location') }}" class="fa-solid fa-earth-europe"></em> | ||||||
|         <i class="fa-solid fa-earth-europe"></i> |  | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <div :id="'location_map_' + index" style="height:300px;" :data-index="index"></div> |         <div :id="'location_map_' + index" style="height:300px;" :data-index="index"></div> | ||||||
|         <span class="muted small"> |         <span class="muted small"> | ||||||
|             <template x-if="!transaction.hasLocation"> |             <template x-if="!transaction.hasLocation"> | ||||||
|                                                         <span>Tap the map to add a location</span> |                 <span>{{ __('firefly.click_tap_location') }}</span> | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="transaction.hasLocation"> |             <template x-if="transaction.hasLocation"> | ||||||
|                                                         <a :data-index="index" href="#" @click="clearLocation">Clear point</a> |                 <a :data-index="index" href="#" @click="clearLocation">{{ __('firefly.clear_location') }}</a> | ||||||
|             </template> |             </template> | ||||||
|         </span> |         </span> | ||||||
|     </div> |     </div> | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
|  | @endif | ||||||
|   | |||||||
| @@ -1,11 +1,13 @@ | |||||||
|  | @if(true === $optionalFields['notes']) | ||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'notes_' + index" |     <label :for="'notes_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |         <em title="{{ __('firefly.notes') }}" class="fa-solid fa-font"></em> | ||||||
|         <i class="fa-solid fa-font"></i> |  | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|                                                 <textarea class="form-control" :id="'notes_' + index" |         <textarea class="form-control" | ||||||
|  |                   :id="'notes_' + index" | ||||||
|                   x-model="transaction.notes" |                   x-model="transaction.notes" | ||||||
|                   placeholder="{{ __('firefly.notes')  }}"></textarea> |                   placeholder="{{ __('firefly.notes')  }}"></textarea> | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  | @endif | ||||||
|   | |||||||
| @@ -1,20 +1,18 @@ | |||||||
| <template | <template x-if="groupProperties.transactionType != 'deposit' && groupProperties.transactionType != 'withdrawal'"> | ||||||
|     x-if="transactionType != 'deposit' && transactionType != 'withdrawal'"> |  | ||||||
|     <div class="row mb-3"> |     <div class="row mb-3"> | ||||||
|         <label :for="'piggy_bank_id_' + index" |         <label :for="'piggy_bank_id_' + index" | ||||||
|                class="col-sm-1 col-form-label d-none d-sm-block"> |                class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|             <i class="fa-solid fa-piggy-bank"></i> |             <em title="{{ __('firefly.piggy_bank') }}" class="fa-solid fa-piggy-bank"></em> | ||||||
|         </label> |         </label> | ||||||
|         <div class="col-sm-10"> |         <div class="col-sm-10"> | ||||||
|             <template x-if="loadingPiggyBanks"> |             <template x-if="formStates.loadingPiggyBanks"> | ||||||
|                                                         <span class="form-control-plaintext"><em |                 <span class="form-control-plaintext"><em class="fa-solid fa-spinner fa-spin"></em></span> | ||||||
|                                                                 class="fa-solid fa-spinner fa-spin"></em></span> |  | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="!loadingPiggyBanks"> |             <template x-if="!formStates.loadingPiggyBanks"> | ||||||
|                 <select class="form-control" |                 <select class="form-control" | ||||||
|                         :id="'piggy_bank_id_' + index" |                         :id="'piggy_bank_id_' + index" | ||||||
|                         x-model="transaction.piggy_bank_id"> |                         x-model="transaction.piggy_bank_id"> | ||||||
|                     <template x-for="group in piggyBanks"> |                     <template x-for="group in formData.piggyBanks"> | ||||||
|                         <optgroup :label="group.name"> |                         <optgroup :label="group.name"> | ||||||
|                             <template x-for="piggyBank in group.piggyBanks"> |                             <template x-for="piggyBank in group.piggyBanks"> | ||||||
|                                 <option :label="piggyBank.name" |                                 <option :label="piggyBank.name" | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'source_' + index" |     <label :for="'source_' + index" | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |            class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-arrow-right"></i> |         <em title="{{ __('firefly.source_account') }}" class="fa-solid fa-arrow-right"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <input type="text" |         <input type="text" | ||||||
| @@ -9,6 +9,7 @@ | |||||||
|                :id="'source_' + index" |                :id="'source_' + index" | ||||||
|                x-model="transaction.source_account.alpine_name" |                x-model="transaction.source_account.alpine_name" | ||||||
|                :data-index="index" |                :data-index="index" | ||||||
|  |                @changed="changedSourceAccount" | ||||||
|                placeholder="{{ __('firefly.source_account')  }}"> |                placeholder="{{ __('firefly.source_account')  }}"> | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,31 +1,39 @@ | |||||||
|  | <!-- RETURN HERE AFTER CREATE TRANSACTION --> | ||||||
|  | <template x-if="'create' === formBehaviour.formType"> | ||||||
|     <div class="form-check"> |     <div class="form-check"> | ||||||
|     <input class="form-check-input" x-model="returnHereButton" type="checkbox" |         <input class="form-check-input" x-model="formStates.returnHereButton" type="checkbox" id="returnButton"> | ||||||
|            id="returnButton"> |         <label class="form-check-label" for="returnButton">{{ __('firefly.create_another') }}</label> | ||||||
|     <label class="form-check-label" for="returnButton"> |     </div> | ||||||
|         Return here to create a new transaction | </template> | ||||||
|     </label> | 
 | ||||||
|  | <!-- RESET AFTER --> | ||||||
|  | <template x-if="'create' === formBehaviour.formType"> | ||||||
|  |     <div class="form-check"> | ||||||
|  |         <input class="form-check-input" x-model="formStates.resetButton" type="checkbox" id="resetButton" :disabled="!formStates.returnHereButton"> | ||||||
|  |         <label class="form-check-label" for="resetButton">{{ __('firefly.reset_after') }}</label> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  | <!-- RETURN HERE AFTER EDIT TRANSACTION --> | ||||||
|  | <template x-if="'edit' === formBehaviour.formType"> | ||||||
|  |     <div class="form-check"> | ||||||
|  |         <input class="form-check-input" x-model="formStates.returnHereButton" type="checkbox" id="returnButton"> | ||||||
|  |         <label class="form-check-label" for="returnButton">{{ __('firefly.after_update_create_another') }}</label> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  | <!-- CLONE INSTEAD OF EDIT CURRENT TRANSACTION --> | ||||||
|  | <template x-if="'edit' === formBehaviour.formType"> | ||||||
|  |     <div class="form-check"> | ||||||
|  |         <input class="form-check-input" x-model="formStates.saveAsNewButton" type="checkbox" id="saveAsNewButton"> | ||||||
|  |         <label class="form-check-label" for="saveAsNewButton">{{ __('firefly.store_as_new') }}</label> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <div class="form-check"> | ||||||
|  |     <input class="form-check-input" type="checkbox" id="rulesButton" :checked="formStates.rulesButton"> | ||||||
|  |     <label class="form-check-label" for="rulesButton">{{ __('firefly.apply_rules_checkbox') }}</label> | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| <div class="form-check"> | <div class="form-check"> | ||||||
|     <input class="form-check-input" x-model="resetButton" type="checkbox" |     <input class="form-check-input" type="checkbox" id="webhookButton" :checked="formStates.webhookButton"> | ||||||
|            id="resetButton" :disabled="!returnHereButton"> |     <label class="form-check-label" for="webhookButton">{{ __('firefly.fire_webhooks_checkbox') }}</label> | ||||||
|     <label class="form-check-label" for="resetButton"> |  | ||||||
|         Reset the form after returning |  | ||||||
|     </label> |  | ||||||
| </div> |  | ||||||
| 
 |  | ||||||
| <div class="form-check"> |  | ||||||
|     <input class="form-check-input" type="checkbox" id="rulesButton" |  | ||||||
|            :checked="rulesButton"> |  | ||||||
|     <label class="form-check-label" for="rulesButton"> |  | ||||||
|         Run rules on this transaction |  | ||||||
|     </label> |  | ||||||
| </div> |  | ||||||
| 
 |  | ||||||
| <div class="form-check"> |  | ||||||
|     <input class="form-check-input" type="checkbox" id="webhookButton" |  | ||||||
|            :checked="webhookButton"> |  | ||||||
|     <label class="form-check-label" for="webhookButton"> |  | ||||||
|         Run webhooks on this transaction |  | ||||||
|     </label> |  | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,19 +1,17 @@ | |||||||
| <template x-if="transactionType != 'transfer' && transactionType != 'deposit'"> | <template x-if="groupProperties.transactionType != 'transfer' && groupProperties.transactionType != 'deposit'"> | ||||||
|     <div class="row mb-3"> |     <div class="row mb-3"> | ||||||
|         <label :for="'bill_id_' + index" |         <label :for="'bill_id_' + index" class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|                class="col-sm-1 col-form-label d-none d-sm-block"> |             <em title="{{ __('firefly.subscription') }}" class="fa-solid fa-calendar"></em> | ||||||
|             <i class="fa-solid fa-calendar"></i> |  | ||||||
|         </label> |         </label> | ||||||
|         <div class="col-sm-10"> |         <div class="col-sm-10"> | ||||||
|             <template x-if="loadingSubscriptions"> |             <template x-if="formStates.loadingSubscriptions"> | ||||||
|                                                         <span class="form-control-plaintext"><em |                 <span class="form-control-plaintext"><em class="fa-solid fa-spinner fa-spin"></em></span> | ||||||
|                                                                 class="fa-solid fa-spinner fa-spin"></em></span> |  | ||||||
|             </template> |             </template> | ||||||
|             <template x-if="!loadingSubscriptions"> |             <template x-if="!formStates.loadingSubscriptions"> | ||||||
|                 <select class="form-control" |                 <select class="form-control" | ||||||
|                         :id="'bill_id_' + index" |                         :id="'bill_id_' + index" | ||||||
|                         x-model="transaction.bill_id"> |                         x-model="transaction.bill_id"> | ||||||
|                     <template x-for="group in subscriptions"> |                     <template x-for="group in formData.subscriptions"> | ||||||
|                         <optgroup :label="group.name"> |                         <optgroup :label="group.name"> | ||||||
|                             <template |                             <template | ||||||
|                                 x-for="subscription in group.subscriptions"> |                                 x-for="subscription in group.subscriptions"> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <div class="row mb-3"> | <div class="row mb-3"> | ||||||
|     <label :for="'tags_' + index" |     <label :for="'tags_' + index" | ||||||
|            class="col-sm-1 col-form-label d-none d-sm-block"> |            class="col-sm-1 col-form-label d-none d-sm-block"> | ||||||
|         <i class="fa-solid fa-tag"></i> |         <em title="{{ __('firefly.tags') }}" class="fa-solid fa-tag"></em> | ||||||
|     </label> |     </label> | ||||||
|     <div class="col-sm-10"> |     <div class="col-sm-10"> | ||||||
|         <select |         <select | ||||||
| @@ -10,7 +10,7 @@ | |||||||
|             x-model="transaction.tags" |             x-model="transaction.tags" | ||||||
|             :name="'tags['+index+'][]'" |             :name="'tags['+index+'][]'" | ||||||
|             multiple> |             multiple> | ||||||
|             <option value="">Type a tag...</option> |             <option value="">{{ __('firefly.select_tag') }}</option> | ||||||
|         </select> |         </select> | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ | |||||||
|             <x-transaction-tab-list></x-transaction-tab-list> |             <x-transaction-tab-list></x-transaction-tab-list> | ||||||
|             <div class="tab-content" id="splitTabsContent"> |             <div class="tab-content" id="splitTabsContent"> | ||||||
|                 <template x-for="transaction,index in entries"> |                 <template x-for="transaction,index in entries"> | ||||||
|                     <x-transaction-split></x-transaction-split> |                     <x-transaction-split :optionalFields="$optionalFields" :optionalDateFields="$optionalDateFields"></x-transaction-split> | ||||||
|                 </template> |                 </template> | ||||||
|             </div> |             </div> | ||||||
|             <div class="row"> |             <div class="row"> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user