[kmymoney/bug-420056] kmymoney/plugins/ofx/import: Add option to enable fixing of the buy/sell signs
Dawid Wróbel
null at kde.org
Wed Apr 15 23:51:20 BST 2020
Git commit 84d204b26fd449f501d4f7cb360adad7def73627 by Dawid Wróbel.
Committed on 15/04/2020 at 22:49.
Pushed by wrobelda into branch 'bug-420056'.
Add option to enable fixing of the buy/sell signs
Some providers do not follow the OFX spec and do not assign the correct
positive/negative signage to amount (TOTALs) and quantity (UNITs)
values assosciated with BUY/SELL investment transactions (see OFX2.2,
section 13.3.3). This change adds option to OFX import dialog and
account's online settings to force the signage to follow the the spec.
BUG: 419975
GUI:
M +1 -0 kmymoney/plugins/ofx/import/dialogs/konlinebankingstatus.cpp
M +19 -2 kmymoney/plugins/ofx/import/dialogs/konlinebankingstatusdecl.ui
M +18 -1 kmymoney/plugins/ofx/import/importoption.ui
M +96 -74 kmymoney/plugins/ofx/import/ofximporter.cpp
https://commits.kde.org/kmymoney/84d204b26fd449f501d4f7cb360adad7def73627
diff --git a/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatus.cpp b/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatus.cpp
index 1fb1d1afd..e5efc56b4 100644
--- a/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatus.cpp
+++ b/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatus.cpp
@@ -108,6 +108,7 @@ KOnlineBankingStatus::KOnlineBankingStatus(const MyMoneyAccount& acc, QWidget *p
m_timestampOffset->setTime(QTime::fromMSecsSinceStartOfDay(qAbs(offset)*60*1000));
m_invertAmount->setChecked(settings.value("kmmofx-invertamount").toLower() == QStringLiteral("yes"));
+ m_fixBuySellSignage->setChecked(settings.value("kmmofx-fixbuysellsignage").toLower() == QStringLiteral("yes"));
QString key = OFX_PASSWORD_KEY(settings.value("url"), settings.value("uniqueId"));
QString pwd;
diff --git a/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatusdecl.ui b/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatusdecl.ui
index 8c63bb82c..03d7b50d5 100644
--- a/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatusdecl.ui
+++ b/kmymoney/plugins/ofx/import/dialogs/konlinebankingstatusdecl.ui
@@ -7,7 +7,7 @@
<x>0</x>
<y>0</y>
<width>638</width>
- <height>392</height>
+ <height>435</height>
</rect>
</property>
<layout class="QVBoxLayout" name="verticalLayout_5">
@@ -383,7 +383,7 @@
<property name="title">
<string>Import options</string>
</property>
- <layout class="QGridLayout" name="gridLayout_3" rowstretch="2,0,0,0">
+ <layout class="QGridLayout" name="gridLayout_3" rowstretch="2,0,0,0,0">
<item row="0" column="1">
<widget class="QComboBox" name="m_preferredPayee">
<item>
@@ -485,6 +485,23 @@
</property>
</widget>
</item>
+ <item row="4" column="0">
+ <widget class="QLabel" name="label_4">
+ <property name="text">
+ <string>Fix sign of the investement transaction amount and quantity</string>
+ </property>
+ </widget>
+ </item>
+ <item row="4" column="1">
+ <widget class="QCheckBox" name="m_fixBuySellSignage">
+ <property name="toolTip">
+ <string><html><head/><body><p>Check this if the investment transactions after importing have incorrect Buy/Sell type assigned, which may be caused by your institution not applying the correct negative/positive sign to the share amount or quantity value.</p></body></html></string>
+ </property>
+ <property name="text">
+ <string/>
+ </property>
+ </widget>
+ </item>
</layout>
</widget>
</item>
diff --git a/kmymoney/plugins/ofx/import/importoption.ui b/kmymoney/plugins/ofx/import/importoption.ui
index ce9484229..d59558788 100644
--- a/kmymoney/plugins/ofx/import/importoption.ui
+++ b/kmymoney/plugins/ofx/import/importoption.ui
@@ -7,7 +7,7 @@
<x>0</x>
<y>0</y>
<width>598</width>
- <height>184</height>
+ <height>218</height>
</rect>
</property>
<layout class="QVBoxLayout" name="verticalLayout">
@@ -130,6 +130,23 @@
</property>
</widget>
</item>
+ <item row="4" column="0">
+ <widget class="QLabel" name="label_4">
+ <property name="text">
+ <string>Fix sign of the investement transaction amount and quantity</string>
+ </property>
+ </widget>
+ </item>
+ <item row="4" column="1">
+ <widget class="QCheckBox" name="m_fixBuySellSignage">
+ <property name="toolTip">
+ <string><html><head/><body><p>Check this if the investment transactions after importing have incorrect Buy/Sell type assigned, which may be caused by your institution not applying the correct negative/positive sign to the share amount or quantity value.</p></body></html></string>
+ </property>
+ <property name="text">
+ <string/>
+ </property>
+ </widget>
+ </item>
</layout>
</widget>
</item>
diff --git a/kmymoney/plugins/ofx/import/ofximporter.cpp b/kmymoney/plugins/ofx/import/ofximporter.cpp
index f5a120c56..33d36c26f 100644
--- a/kmymoney/plugins/ofx/import/ofximporter.cpp
+++ b/kmymoney/plugins/ofx/import/ofximporter.cpp
@@ -74,7 +74,7 @@ typedef enum {
class OFXImporter::Private
{
public:
- Private() : m_valid(false), m_preferName(PreferId), m_uniqueIdSource(UniqueIdUnknown), m_walletIsOpen(false), m_invertAmount(false), m_statusDlg(0), m_wallet(0),
+ Private() : m_valid(false), m_preferName(PreferId), m_uniqueIdSource(UniqueIdUnknown), m_walletIsOpen(false), m_invertAmount(false), m_fixBuySellSignage(false), m_statusDlg(0), m_wallet(0),
m_updateStartDate(QDate(1900,1,1)), m_timestampOffset(0) {}
bool m_valid;
@@ -86,6 +86,7 @@ public:
UniqueTransactionIdSource m_uniqueIdSource;
bool m_walletIsOpen;
bool m_invertAmount;
+ bool m_fixBuySellSignage;
QList<MyMoneyStatement> m_statementlist;
QList<MyMoneyStatement::Security> m_securitylist;
QString m_fatalerror;
@@ -185,6 +186,7 @@ void OFXImporter::slotImportFile()
d->m_uniqueIdSource = static_cast<UniqueTransactionIdSource>(option->m_uniqueIdSource->currentIndex());
d->m_timestampOffset = d->constructTimeOffset(option->m_timestampOffset, option->m_timestampOffsetSign);
d->m_invertAmount = option->m_invertAmount->isChecked();
+ d->m_fixBuySellSignage = option->m_fixBuySellSignage->isChecked();
if (url.isValid()) {
const QString filename(url.toLocalFile());
@@ -359,8 +361,95 @@ int OFXImporter::ofxTransactionCallback(struct OfxTransactionData data, void * p
}
}
+ bool unhandledtype = false;
+ QString type;
+
+ if (data.invtransactiontype_valid) {
+ switch (data.invtransactiontype) {
+ case OFX_BUYDEBT:
+ case OFX_BUYMF:
+ case OFX_BUYOPT:
+ case OFX_BUYOTHER:
+ case OFX_BUYSTOCK:
+ t.m_eAction = eMyMoney::Transaction::Action::Buy;
+ break;
+ case OFX_REINVEST:
+ t.m_eAction = eMyMoney::Transaction::Action::ReinvestDividend;
+ break;
+ case OFX_SELLDEBT:
+ case OFX_SELLMF:
+ case OFX_SELLOPT:
+ case OFX_SELLOTHER:
+ case OFX_SELLSTOCK:
+ t.m_eAction = eMyMoney::Transaction::Action::Sell;
+ break;
+ case OFX_INCOME:
+ t.m_eAction = eMyMoney::Transaction::Action::CashDividend;
+ // NOTE: With CashDividend, the amount of the dividend should
+ // be in data.amount. Since I've never seen an OFX file with
+ // cash dividends, this is an assumption on my part. (acejones)
+ break;
+
+ //
+ // These types are all not handled. We will generate a warning for them.
+ //
+ case OFX_CLOSUREOPT:
+ unhandledtype = true;
+ type = QStringLiteral("CLOSUREOPT (Close a position for an option)");
+ break;
+ case OFX_INVEXPENSE:
+ unhandledtype = true;
+ type = QStringLiteral("INVEXPENSE (Misc investment expense that is associated with a specific security)");
+ break;
+ case OFX_JRNLFUND:
+ unhandledtype = true;
+ type = QStringLiteral("JRNLFUND (Journaling cash holdings between subaccounts within the same investment account)");
+ break;
+ case OFX_MARGININTEREST:
+ unhandledtype = true;
+ type = QStringLiteral("MARGININTEREST (Margin interest expense)");
+ break;
+ case OFX_RETOFCAP:
+ unhandledtype = true;
+ type = QStringLiteral("RETOFCAP (Return of capital)");
+ break;
+ case OFX_SPLIT:
+ unhandledtype = true;
+ type = QStringLiteral("SPLIT (Stock or mutial fund split)");
+ break;
+ case OFX_TRANSFER:
+ unhandledtype = true;
+ type = QStringLiteral("TRANSFER (Transfer holdings in and out of the investment account)");
+ break;
+ default:
+ unhandledtype = true;
+ type = QString("UNKNOWN %1").arg(data.invtransactiontype);
+ break;
+ }
+ } else
+ t.m_eAction = eMyMoney::Transaction::Action::None;
+
+ t.m_shares = MyMoneyMoney();
+ if (data.units_valid) {
+ t.m_shares = MyMoneyMoney(data.units, 100000).reduce();
+ }
+
+ t.m_amount = MyMoneyMoney();
if (data.amount_valid) {
t.m_amount = MyMoneyMoney(data.amount, 1000);
+
+ if (pofx->d->m_fixBuySellSignage) {
+ if (t.m_eAction == eMyMoney::Transaction::Action::Buy
+ || t.m_eAction == eMyMoney::Transaction::Action::ReinvestDividend) {
+ t.m_amount = -t.m_amount.abs();
+ t.m_shares = t.m_shares.abs();
+ }
+ else if (t.m_eAction == eMyMoney::Transaction::Action::Sell) {
+ t.m_amount = t.m_amount.abs();
+ t.m_shares = -t.m_shares.abs();
+ }
+ }
+
if (pofx->d->m_invertAmount) {
t.m_amount = -t.m_amount;
}
@@ -498,11 +587,6 @@ int OFXImporter::ofxTransactionCallback(struct OfxTransactionData data, void * p
}
}
- t.m_shares = MyMoneyMoney();
- if (data.units_valid) {
- t.m_shares = MyMoneyMoney(data.units, 100000).reduce();
- }
-
t.m_price = MyMoneyMoney();
if (data.unitprice_valid) {
t.m_price = MyMoneyMoney(data.unitprice, 100000).reduce();
@@ -517,74 +601,6 @@ int OFXImporter::ofxTransactionCallback(struct OfxTransactionData data, void * p
t.m_fees += MyMoneyMoney(data.commission, 1000).reduce();
}
- bool unhandledtype = false;
- QString type;
-
- if (data.invtransactiontype_valid) {
- switch (data.invtransactiontype) {
- case OFX_BUYDEBT:
- case OFX_BUYMF:
- case OFX_BUYOPT:
- case OFX_BUYOTHER:
- case OFX_BUYSTOCK:
- t.m_eAction = eMyMoney::Transaction::Action::Buy;
- break;
- case OFX_REINVEST:
- t.m_eAction = eMyMoney::Transaction::Action::ReinvestDividend;
- break;
- case OFX_SELLDEBT:
- case OFX_SELLMF:
- case OFX_SELLOPT:
- case OFX_SELLOTHER:
- case OFX_SELLSTOCK:
- t.m_eAction = eMyMoney::Transaction::Action::Sell;
- break;
- case OFX_INCOME:
- t.m_eAction = eMyMoney::Transaction::Action::CashDividend;
- // NOTE: With CashDividend, the amount of the dividend should
- // be in data.amount. Since I've never seen an OFX file with
- // cash dividends, this is an assumption on my part. (acejones)
- break;
-
- //
- // These types are all not handled. We will generate a warning for them.
- //
- case OFX_CLOSUREOPT:
- unhandledtype = true;
- type = QStringLiteral("CLOSUREOPT (Close a position for an option)");
- break;
- case OFX_INVEXPENSE:
- unhandledtype = true;
- type = QStringLiteral("INVEXPENSE (Misc investment expense that is associated with a specific security)");
- break;
- case OFX_JRNLFUND:
- unhandledtype = true;
- type = QStringLiteral("JRNLFUND (Journaling cash holdings between subaccounts within the same investment account)");
- break;
- case OFX_MARGININTEREST:
- unhandledtype = true;
- type = QStringLiteral("MARGININTEREST (Margin interest expense)");
- break;
- case OFX_RETOFCAP:
- unhandledtype = true;
- type = QStringLiteral("RETOFCAP (Return of capital)");
- break;
- case OFX_SPLIT:
- unhandledtype = true;
- type = QStringLiteral("SPLIT (Stock or mutial fund split)");
- break;
- case OFX_TRANSFER:
- unhandledtype = true;
- type = QStringLiteral("TRANSFER (Transfer holdings in and out of the investment account)");
- break;
- default:
- unhandledtype = true;
- type = QString("UNKNOWN %1").arg(data.invtransactiontype);
- break;
- }
- } else
- t.m_eAction = eMyMoney::Transaction::Action::None;
-
// In the case of investment transactions, the 'total' is supposed to the total amount
// of the transaction. units * unitprice +/- commission. Easy, right? Sadly, it seems
// some ofx creators do not follow this in all circumstances. Therefore, we have to double-
@@ -857,6 +873,11 @@ MyMoneyKeyValueContainer OFXImporter::onlineBankingSettings(const MyMoneyKeyValu
} else {
kvp.deletePair(QStringLiteral("kmmofx-invertamount"));
}
+ if (d->m_statusDlg->m_fixBuySellSignage->isChecked()) {
+ kvp.setValue(QStringLiteral("kmmofx-fixbuysellsignage"), QStringLiteral("yes"));
+ } else {
+ kvp.deletePair(QStringLiteral("kmmofx-fixbuysellsignage"));
+ }
// get rid of pre 4.6 values
kvp.deletePair(QStringLiteral("kmmofx-preferPayeeid"));
}
@@ -918,6 +939,7 @@ bool OFXImporter::updateAccount(const MyMoneyAccount& acc, bool moreAccounts)
}
d->m_invertAmount = settings.value("kmmofx-invertamount").toLower() == QStringLiteral("yes");
+ d->m_fixBuySellSignage= settings.value("kmmofx-fixbuysellsignage").toLower() == QStringLiteral("yes");
}
d->m_timestampOffset = settings.value("kmmofx-timestampOffset").toInt();
//kDebug(0) << "ofx plugin: account" << acc.name() << "earliest transaction date to process =" << qPrintable(d->m_updateStartDate.toString(Qt::ISODate));
More information about the kde-doc-english
mailing list