mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-09-19 19:01:58 +00:00
Can now create transfers with different currencies.
This commit is contained in:
@@ -67,8 +67,10 @@ class JournalFormRequest extends Request
|
|||||||
'destination_account_name' => $this->string('destination_account_name'),
|
'destination_account_name' => $this->string('destination_account_name'),
|
||||||
'piggy_bank_id' => $this->integer('piggy_bank_id'),
|
'piggy_bank_id' => $this->integer('piggy_bank_id'),
|
||||||
|
|
||||||
// native amount
|
// native amount and stuff like that:
|
||||||
'native_amount' => $this->float('native_amount'),
|
'native_amount' => $this->float('native_amount'),
|
||||||
|
'source_amount' => $this->float('source_amount'),
|
||||||
|
'destination_amount' => $this->float('destination_amount'),
|
||||||
|
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -105,8 +107,10 @@ class JournalFormRequest extends Request
|
|||||||
'destination_account_name' => 'between:1,255',
|
'destination_account_name' => 'between:1,255',
|
||||||
'piggy_bank_id' => 'between:1,255',
|
'piggy_bank_id' => 'between:1,255',
|
||||||
|
|
||||||
// exchange rate data:
|
// foreign currency amounts
|
||||||
'native_amount' => 'numeric|more:0',
|
'native_amount' => 'numeric|more:0',
|
||||||
|
'source_amount' => 'numeric|more:0',
|
||||||
|
'destination_amount' => 'numeric|more:0',
|
||||||
];
|
];
|
||||||
|
|
||||||
// some rules get an upgrade depending on the type of data:
|
// some rules get an upgrade depending on the type of data:
|
||||||
|
@@ -778,29 +778,38 @@ class JournalRepository implements JournalRepositoryInterface
|
|||||||
/** @var TransactionType $transactionType */
|
/** @var TransactionType $transactionType */
|
||||||
$transactionType = TransactionType::where('type', ucfirst($data['what']))->first();
|
$transactionType = TransactionType::where('type', ucfirst($data['what']))->first();
|
||||||
$submittedCurrencyId = $data['currency_id'];
|
$submittedCurrencyId = $data['currency_id'];
|
||||||
$amount = strval($data['amount']);
|
|
||||||
|
|
||||||
// which account to check for what the native currency is?
|
// which account to check for what the native currency is?
|
||||||
$check = 'source';
|
$check = 'source';
|
||||||
if ($transactionType->type === TransactionType::DEPOSIT) {
|
if ($transactionType->type === TransactionType::DEPOSIT) {
|
||||||
$check = 'destination';
|
$check = 'destination';
|
||||||
}
|
}
|
||||||
if ($transactionType->type === TransactionType::TRANSFER) {
|
switch ($transactionType->type) {
|
||||||
throw new FireflyException('Cannot handle transfers in verifyNativeAmount()');
|
case TransactionType::DEPOSIT:
|
||||||
}
|
case TransactionType::WITHDRAWAL:
|
||||||
|
// continue:
|
||||||
|
$nativeCurrencyId = intval($accounts[$check]->getMeta('currency_id'));
|
||||||
|
|
||||||
// continue:
|
// does not match? Then user has submitted amount in a foreign currency:
|
||||||
$nativeCurrencyId = intval($accounts[$check]->getMeta('currency_id'));
|
if ($nativeCurrencyId !== $submittedCurrencyId) {
|
||||||
|
// store amount and submitted currency in "foreign currency" fields:
|
||||||
|
$data['foreign_amount'] = $data['amount'];
|
||||||
|
$data['foreign_currency_id'] = $submittedCurrencyId;
|
||||||
|
|
||||||
// does not match? Then user has submitted amount in a foreign currency:
|
// overrule the amount and currency ID fields to be the original again:
|
||||||
if ($nativeCurrencyId !== $submittedCurrencyId) {
|
$data['amount'] = strval($data['native_amount']);
|
||||||
// store amount and submitted currency in "foreign currency" fields:
|
$data['currency_id'] = $nativeCurrencyId;
|
||||||
$data['foreign_amount'] = $data['amount'];
|
}
|
||||||
$data['foreign_currency_id'] = $submittedCurrencyId;
|
break;
|
||||||
|
case TransactionType::TRANSFER:
|
||||||
// overrule the amount and currency ID fields to be the original again:
|
// source gets the original amount.
|
||||||
$data['amount'] = strval($data['native_amount']);
|
$data['amount'] = strval($data['source_amount']);
|
||||||
$data['currency_id'] = $nativeCurrencyId;
|
$data['currency_id'] = intval($accounts['source']->getMeta('currency_id'));
|
||||||
|
$data['foreign_amount'] = strval($data['destination_amount']);
|
||||||
|
$data['foreign_currency_id'] = intval($accounts['destination']->getMeta('currency_id'));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new FireflyException(sprintf('Cannot handle %s in verifyNativeAmount()', $transactionType->type));
|
||||||
}
|
}
|
||||||
|
|
||||||
return $data;
|
return $data;
|
||||||
|
@@ -11,27 +11,68 @@
|
|||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
|
// hide ALL exchange things and AMOUNT things
|
||||||
|
$('#exchange_rate_instruction_holder').hide();
|
||||||
|
$('#native_amount_holder').hide();
|
||||||
|
$('#amount_holder').hide();
|
||||||
|
$('#source_amount_holder').hide();
|
||||||
|
$('#destination_amount_holder').hide();
|
||||||
|
|
||||||
// respond to switch buttons (first time always triggers)
|
// respond to switch buttons (first time always triggers)
|
||||||
updateButtons();
|
updateButtons();
|
||||||
updateForm();
|
updateForm();
|
||||||
updateLayout();
|
updateLayout();
|
||||||
updateDescription();
|
updateDescription();
|
||||||
updateNativeCurrency(); // verify native currency by first account (may be different).
|
updateNativeCurrency();
|
||||||
|
|
||||||
// hide ALL exchange things
|
|
||||||
$('#exchange_rate_instruction_holder').hide();
|
|
||||||
$('#native_amount_holder').hide();
|
|
||||||
|
|
||||||
// when user changes source account, native currency may be different.
|
|
||||||
|
// when user changes source account or destination, native currency may be different.
|
||||||
$('select[name="source_account_id"]').on('change', updateNativeCurrency);
|
$('select[name="source_account_id"]').on('change', updateNativeCurrency);
|
||||||
|
$('select[name="destination_account_id"]').on('change', updateNativeCurrency);
|
||||||
|
|
||||||
// convert foreign currency to native currency.
|
// convert foreign currency to native currency (when input changes, exchange rate)
|
||||||
$('#ffInput_amount').on('change', convertForeignToNative);
|
$('#ffInput_amount').on('change', convertForeignToNative);
|
||||||
|
|
||||||
|
// convert source currency to destination currency (slightly different routine for transfers)
|
||||||
|
$('#ffInput_source_amount').on('change', convertSourceToDestination);
|
||||||
|
|
||||||
// when user selects different currency,
|
// when user selects different currency,
|
||||||
$('.currency-option').on('click', selectsForeignCurrency);
|
$('.currency-option').on('click', selectsForeignCurrency);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert from source amount currency to destination currency for transfers.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function convertSourceToDestination() {
|
||||||
|
var sourceAccount = $('select[name="source_account_id"]').val();
|
||||||
|
var destAccount = $('select[name="destination_account_id"]').val();
|
||||||
|
|
||||||
|
var sourceCurrency = accountInfo[sourceAccount].preferredCurrency;
|
||||||
|
var destinationCurrency = accountInfo[destAccount].preferredCurrency;
|
||||||
|
|
||||||
|
var sourceCurrencyCode = currencyInfo[sourceCurrency].code;
|
||||||
|
var destinationCurrencyCode = currencyInfo[destinationCurrency].code;
|
||||||
|
|
||||||
|
var date = $('#ffInput_date').val();
|
||||||
|
var amount = $('#ffInput_source_amount').val();
|
||||||
|
$('#ffInput_amount').val(amount);
|
||||||
|
var uri = 'json/rate/' + sourceCurrencyCode + '/' + destinationCurrencyCode + '/' + date + '?amount=' + amount;
|
||||||
|
console.log('Will grab ' + uri);
|
||||||
|
$.get(uri).done(updateDestinationAmount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Once the data has been grabbed will update the field (for transfers)
|
||||||
|
* @param data
|
||||||
|
*/
|
||||||
|
function updateDestinationAmount(data) {
|
||||||
|
console.log('Returned data:');
|
||||||
|
console.log(data);
|
||||||
|
$('#ffInput_destination_amount').val(data.amount);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This function generates a small helper text to explain the user
|
* This function generates a small helper text to explain the user
|
||||||
* that they have selected a foreign currency.
|
* that they have selected a foreign currency.
|
||||||
@@ -48,7 +89,21 @@ function getExchangeInstructions() {
|
|||||||
return text;
|
return text;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Same as above but for transfers
|
||||||
|
*/
|
||||||
|
function getTransferExchangeInstructions() {
|
||||||
|
var sourceAccount = $('select[name="source_account_id"]').val();
|
||||||
|
var destAccount = $('select[name="destination_account_id"]').val();
|
||||||
|
|
||||||
|
var sourceCurrency = accountInfo[sourceAccount].preferredCurrency;
|
||||||
|
var destinationCurrency = accountInfo[destAccount].preferredCurrency;
|
||||||
|
|
||||||
|
return transferInstructions.replace('@source_name', accountInfo[sourceAccount].name)
|
||||||
|
.replace('@dest_name', accountInfo[destAccount].name)
|
||||||
|
.replace(/@source_currency/g, currencyInfo[sourceCurrency].name)
|
||||||
|
.replace(/@dest_currency/g, currencyInfo[destinationCurrency].name);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* There is an input that shows the currency symbol that is native to the selected
|
* There is an input that shows the currency symbol that is native to the selected
|
||||||
@@ -63,6 +118,41 @@ function updateNativeCurrency() {
|
|||||||
$('.currency-option[data-id="' + nativeCurrencyId + '"]').click();
|
$('.currency-option[data-id="' + nativeCurrencyId + '"]').click();
|
||||||
$('[data-toggle="dropdown"]').parent().removeClass('open');
|
$('[data-toggle="dropdown"]').parent().removeClass('open');
|
||||||
$('select[name="source_account_id"]').focus();
|
$('select[name="source_account_id"]').focus();
|
||||||
|
|
||||||
|
validateCurrencyForTransfer();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When the transaction to create is a transfer some more checks are necessary.
|
||||||
|
*/
|
||||||
|
function validateCurrencyForTransfer() {
|
||||||
|
if (what !== "transfer") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$('#source_amount_holder').show();
|
||||||
|
var sourceAccount = $('select[name="source_account_id"]').val();
|
||||||
|
var destAccount = $('select[name="destination_account_id"]').val();
|
||||||
|
var sourceCurrency = accountInfo[sourceAccount].preferredCurrency;
|
||||||
|
var sourceSymbol = currencyInfo[sourceCurrency].symbol;
|
||||||
|
var destinationCurrency = accountInfo[destAccount].preferredCurrency;
|
||||||
|
var destinationSymbol = currencyInfo[destinationCurrency].symbol;
|
||||||
|
|
||||||
|
$('#source_amount_holder').show().find('.non-selectable-currency-symbol').text(sourceSymbol);
|
||||||
|
|
||||||
|
if (sourceCurrency === destinationCurrency) {
|
||||||
|
console.log('Both accounts accept ' + sourceCurrency);
|
||||||
|
$('#destination_amount_holder').hide();
|
||||||
|
$('#amount_holder').hide();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.log('Source accepts #' + sourceCurrency + ', destination #' + destinationCurrency);
|
||||||
|
$('#ffInput_exchange_rate_instruction').text(getTransferExchangeInstructions());
|
||||||
|
$('#exchange_rate_instruction_holder').show();
|
||||||
|
$('input[name="source_amount"]').val($('input[name="amount"]').val());
|
||||||
|
convertSourceToDestination();
|
||||||
|
|
||||||
|
$('#destination_amount_holder').show().find('.non-selectable-currency-symbol').text(destinationSymbol);
|
||||||
|
$('#amount_holder').hide();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -94,7 +184,7 @@ function updateForm() {
|
|||||||
$('input[name="what"]').val(what);
|
$('input[name="what"]').val(what);
|
||||||
switch (what) {
|
switch (what) {
|
||||||
case 'withdrawal':
|
case 'withdrawal':
|
||||||
// show source_id and dest_name:
|
// show source_id and dest_name
|
||||||
$('#source_account_id_holder').show();
|
$('#source_account_id_holder').show();
|
||||||
$('#destination_account_name_holder').show();
|
$('#destination_account_name_holder').show();
|
||||||
|
|
||||||
@@ -102,7 +192,7 @@ function updateForm() {
|
|||||||
$('#source_account_name_holder').hide();
|
$('#source_account_name_holder').hide();
|
||||||
$('#destination_account_id_holder').hide();
|
$('#destination_account_id_holder').hide();
|
||||||
|
|
||||||
// show budget:
|
//
|
||||||
$('#budget_id_holder').show();
|
$('#budget_id_holder').show();
|
||||||
|
|
||||||
// hide piggy bank:
|
// hide piggy bank:
|
||||||
@@ -114,6 +204,17 @@ function updateForm() {
|
|||||||
$('#ffInput_source_account_name').val($('#ffInput_destination_account_name').val());
|
$('#ffInput_source_account_name').val($('#ffInput_destination_account_name').val());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// exchange / foreign currencies:
|
||||||
|
// hide explanation, hide source and destination amounts:
|
||||||
|
$('#exchange_rate_instruction_holder').hide();
|
||||||
|
$('#source_amount_holder').hide();
|
||||||
|
$('#destination_amount_holder').hide();
|
||||||
|
// show normal amount:
|
||||||
|
$('#amount_holder').show();
|
||||||
|
|
||||||
|
// update the amount thing:
|
||||||
|
updateNativeCurrency();
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case 'deposit':
|
case 'deposit':
|
||||||
// show source_name and dest_id:
|
// show source_name and dest_id:
|
||||||
@@ -134,6 +235,17 @@ function updateForm() {
|
|||||||
$('#ffInput_destination_account_name').val($('#ffInput_source_account_name').val());
|
$('#ffInput_destination_account_name').val($('#ffInput_source_account_name').val());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// exchange / foreign currencies:
|
||||||
|
// hide explanation, hide source and destination amounts:
|
||||||
|
$('#exchange_rate_instruction_holder').hide();
|
||||||
|
$('#source_amount_holder').hide();
|
||||||
|
$('#destination_amount_holder').hide();
|
||||||
|
// show normal amount:
|
||||||
|
$('#amount_holder').show();
|
||||||
|
|
||||||
|
// update the amount thing:
|
||||||
|
updateNativeCurrency();
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case 'transfer':
|
case 'transfer':
|
||||||
// show source_id and dest_id:
|
// show source_id and dest_id:
|
||||||
@@ -144,7 +256,6 @@ function updateForm() {
|
|||||||
$('#source_account_name_holder').hide();
|
$('#source_account_name_holder').hide();
|
||||||
$('#destination_account_name_holder').hide();
|
$('#destination_account_name_holder').hide();
|
||||||
|
|
||||||
|
|
||||||
// hide budget
|
// hide budget
|
||||||
$('#budget_id_holder').hide();
|
$('#budget_id_holder').hide();
|
||||||
if (piggiesLength === 0) {
|
if (piggiesLength === 0) {
|
||||||
@@ -152,6 +263,10 @@ function updateForm() {
|
|||||||
} else {
|
} else {
|
||||||
$('#piggy_bank_id_holder').show();
|
$('#piggy_bank_id_holder').show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// update the amount thing:
|
||||||
|
updateNativeCurrency();
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
// no action.
|
// no action.
|
||||||
@@ -206,8 +321,7 @@ function getAccountId() {
|
|||||||
if (what === "withdrawal") {
|
if (what === "withdrawal") {
|
||||||
return $('select[name="source_account_id"]').val();
|
return $('select[name="source_account_id"]').val();
|
||||||
}
|
}
|
||||||
if (what === "deposit") {
|
if (what === "deposit" || what === "transfer") {
|
||||||
return $('select[name="destination_account_id"]').val();
|
return $('select[name="destination_account_id"]').val();
|
||||||
}
|
}
|
||||||
alert('Cannot handle ' + what);
|
|
||||||
}
|
}
|
||||||
|
@@ -136,6 +136,7 @@ return [
|
|||||||
'journals_in_period_for_category' => 'All transactions for category :name between :start and :end',
|
'journals_in_period_for_category' => 'All transactions for category :name between :start and :end',
|
||||||
'not_available_demo_user' => 'The feature you try to access is not available to demo users.',
|
'not_available_demo_user' => 'The feature you try to access is not available to demo users.',
|
||||||
'exchange_rate_instructions' => 'Asset account "@name" only accepts transactions in @native_currency. If you wish to use @foreign_currency instead, make sure that the amount in @native_currency is known as well:',
|
'exchange_rate_instructions' => 'Asset account "@name" only accepts transactions in @native_currency. If you wish to use @foreign_currency instead, make sure that the amount in @native_currency is known as well:',
|
||||||
|
'transfer_exchange_rate_instructions' => 'Source asset account "@source_name" only accepts transactions in @source_currency. Destination asset account "@dest_name" only accepts transactions in @dest_currency. You must provide the transferred amount correctly in both currencies.',
|
||||||
|
|
||||||
// repeat frequencies:
|
// repeat frequencies:
|
||||||
'repeat_freq_yearly' => 'yearly',
|
'repeat_freq_yearly' => 'yearly',
|
||||||
|
@@ -66,6 +66,8 @@ return [
|
|||||||
'decimal_places' => 'Decimal places',
|
'decimal_places' => 'Decimal places',
|
||||||
'exchange_rate_instruction' => 'Foreign currencies',
|
'exchange_rate_instruction' => 'Foreign currencies',
|
||||||
'exchanged_amount' => 'Exchanged amount',
|
'exchanged_amount' => 'Exchanged amount',
|
||||||
|
'source_amount' => 'Amount (source)',
|
||||||
|
'destination_amount' => 'Amount (destination)',
|
||||||
|
|
||||||
'revenue_account_source' => 'Revenue account (source)',
|
'revenue_account_source' => 'Revenue account (source)',
|
||||||
'source_account_asset' => 'Source account (asset account)',
|
'source_account_asset' => 'Source account (asset account)',
|
||||||
|
@@ -52,6 +52,10 @@
|
|||||||
|
|
||||||
{{ ExpandedForm.nonSelectableAmount('native_amount') }}
|
{{ ExpandedForm.nonSelectableAmount('native_amount') }}
|
||||||
|
|
||||||
|
{{ ExpandedForm.nonSelectableAmount('source_amount') }}
|
||||||
|
|
||||||
|
{{ ExpandedForm.nonSelectableAmount('destination_amount') }}
|
||||||
|
|
||||||
{# ALWAYS SHOW DATE #}
|
{# ALWAYS SHOW DATE #}
|
||||||
{{ ExpandedForm.date('date', preFilled.date|default(phpdate('Y-m-d'))) }}
|
{{ ExpandedForm.date('date', preFilled.date|default(phpdate('Y-m-d'))) }}
|
||||||
</div>
|
</div>
|
||||||
@@ -215,6 +219,8 @@
|
|||||||
var middleCrumbUrl = [];
|
var middleCrumbUrl = [];
|
||||||
var button = [];
|
var button = [];
|
||||||
var exchangeRateInstructions = "{{ 'exchange_rate_instructions'|_|escape('js') }}";
|
var exchangeRateInstructions = "{{ 'exchange_rate_instructions'|_|escape('js') }}";
|
||||||
|
var transferInstructions = "{{ 'transfer_exchange_rate_instructions'|_|escape('js') }}";
|
||||||
|
|
||||||
{% for type in {0:'withdrawal',1:'deposit',2:'transfer'} %}
|
{% for type in {0:'withdrawal',1:'deposit',2:'transfer'} %}
|
||||||
|
|
||||||
txt['{{ type }}'] = '{{ type|_ }}';
|
txt['{{ type }}'] = '{{ type|_ }}';
|
||||||
|
Reference in New Issue
Block a user