/*
Copyright 2010-2011 Grégory Soutadé
This file is part of KissCount.
KissCount is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
KissCount 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with KissCount. If not, see .
*/
#include
#include
#include
#include
#include
#include "ImportPanel.hpp"
#include "grid/ChoiceDelegate.hpp"
ImportPanel::ImportPanel(KissCount* kiss, wxUI *parent, bool lowResolution) :
KissPanel(kiss, parent, lowResolution)
{
QVBoxLayout *vbox = new QVBoxLayout;
QVBoxLayout *vbox2 = new QVBoxLayout;
QHBoxLayout *hbox = new QHBoxLayout;
_hbox = new QHBoxLayout;
QPushButton* buttonOpen;
QGroupBox* staticAccount = new QGroupBox(_("Unresolved accounts"));
QGroupBox* staticCategory = new QGroupBox(_("Unresolved categories"));
QGroupBox* staticTag = new QGroupBox(_("Unresolved tags"));
_icons[KissPanel::LOW_RES_ICON] = IMPORT_LOW_ICON;
_icons[KissPanel::HIGH_RES_ICON] = IMPORT_ICON;
setLayout(vbox);
_fileTxt = new QLineEdit;
buttonOpen = new QPushButton("...");
connect(buttonOpen, SIGNAL(clicked()), this, SLOT(OnFile()));
connect(_fileTxt, SIGNAL(returnPressed()), this, SLOT(OnFileEnter()));
_buttonLoadOperations = new QPushButton(_("Load operations"));
_buttonLoadOperations->setEnabled(false);
connect(_buttonLoadOperations, SIGNAL(clicked()), this, SLOT(OnLoadOperations()));
_buttonIntegrate = new QPushButton(_("Integrate operations"));
_buttonIntegrate->setEnabled(false);
connect(_buttonIntegrate, SIGNAL(clicked()), this, SLOT(OnIntegrate()));
_checkSaveImportPatterns = new QCheckBox(_("Save import patterns"));
hbox->addWidget(_fileTxt);
hbox->addWidget(buttonOpen);
hbox->addWidget(_buttonLoadOperations);
hbox->addWidget(_buttonIntegrate);
hbox->addWidget(_checkSaveImportPatterns);
vbox->addLayout(hbox);
_accountsGrid = new QTableWidget(0, 3);
_accountsGrid->verticalHeader()->setHidden(true);
QStringList labels;
labels << _("File account") << _("Account name") << _("Internal account");
_accountsGrid->setHorizontalHeaderLabels(labels);
_accountsGrid->resizeColumnsToContents();
_categoriesGrid = new QTableWidget(0, 3);
_categoriesGrid->verticalHeader()->setHidden(true);
labels.clear();
labels << _("File category") << _("Category name") << _("Internal category");
_categoriesGrid->setHorizontalHeaderLabels(labels);
_categoriesGrid->resizeColumnsToContents();
_tagsGrid = new QTableWidget(0, 3);
_tagsGrid->verticalHeader()->setHidden(true);
labels.clear();
labels << _("File tag") << _("Tag name") << _("Internal tag");
_tagsGrid->setHorizontalHeaderLabels(labels);
_tagsGrid->resizeColumnsToContents();
{
QVBoxLayout *vbox = new QVBoxLayout;
vbox->addWidget(_accountsGrid);
staticAccount->setLayout(vbox);
}
vbox2->addWidget(staticAccount);
{
QVBoxLayout *vbox = new QVBoxLayout;
vbox->addWidget(_categoriesGrid);
staticCategory->setLayout(vbox);
}
{
QVBoxLayout *vbox = new QVBoxLayout;
vbox->addWidget(_tagsGrid);
staticTag->setLayout(vbox);
}
vbox2->addWidget(staticCategory);
vbox2->addWidget(staticTag);
_operationsGrid = new GridAccount(kiss, this, false, false, false);
connect(_operationsGrid, SIGNAL(cellChanged(int, int)), this, SLOT(OnOperationModified(int, int)));
_hbox->addLayout(vbox2);
_hbox->addWidget(_operationsGrid);
vbox->addLayout(_hbox);
layout();
}
KissPanel* ImportPanel::CreatePanel()
{
return new ImportPanel(_kiss, _wxUI, _lowResolution);
}
QString ImportPanel::GetToolTip()
{
return _("Import");
}
void ImportPanel::OnShow()
{
_wxUI->setWindowTitle(_("KissCount - Import"));
}
void ImportPanel::OnFile()
{
QString path;
path = QFileDialog::getOpenFileName(0, _("Choose a database to open"), "", _kiss->GetImportEngineExtensions());
if (!path.size())
return;
_fileTxt->setText(path);
ProcessFile();
}
void ImportPanel::OnFileEnter()
{
ProcessFile();
}
void ImportPanel::ProcessFile()
{
User* user = _kiss->GetUser();
int i;
QString* userAccounts;
std::map resolvedAccounts;
QString* userCategories;
std::map resolvedCategories;
QString* userTags;
std::map resolvedTags;
QTableWidgetItem* item;
QString path = _fileTxt->text();
_buttonLoadOperations->setEnabled(false);
_buttonIntegrate->setEnabled(false);
_accountsGrid->setRowCount(0);
_categoriesGrid->setRowCount(0);
_tagsGrid->setRowCount(0);
_operationsGrid->setRowCount(0);
_importEngine = _kiss->GetImportEngine(path);
if (!_importEngine)
{
QMessageBox::critical(0, _("Error"), _("Any engine can process this file !"));
return ;
}
_importEngine->ParseFile(_unresolvedAccounts, _unresolvedCategories, _unresolvedTags);
if (_unresolvedAccounts.size())
{
int nb_accounts = user->GetAccountsNumber();
userAccounts = new QString[nb_accounts+1];
userAccounts[0] = _("Create one");
for(i=0; i_accounts[i].name;
ChoiceDelegate* accountEditor = new ChoiceDelegate(this, userAccounts, nb_accounts+1);
_accountsGrid->setItemDelegateForColumn(2, accountEditor);
_buttonLoadOperations->setEnabled(true);
_accountsGrid->setRowCount(_unresolvedAccounts.size());
for (i=0; i<(int)_unresolvedAccounts.size(); i++)
{
item = new QTableWidgetItem(_unresolvedAccounts[i].number);
item->setFlags(item->flags() & ~Qt::ItemIsEditable);
_accountsGrid->setItem(i, 0, item);
_accountsGrid->setItem(i, 1, new QTableWidgetItem(_unresolvedAccounts[i].name));
_accountsGrid->setItem(i, 2, new QTableWidgetItem(userAccounts[0]));
}
_accountsGrid->resizeColumnsToContents();
_accountsGrid->layout();
}
if (_unresolvedCategories.size())
{
int nb_categories = user->GetCategoriesNumber();
userCategories = new QString[nb_categories+1];
userCategories[0] = _("Create one");
for(i=0; i_categories[i].name;
ChoiceDelegate* categoryEditor = new ChoiceDelegate(this, userCategories, nb_categories+1);
_categoriesGrid->setItemDelegateForColumn(2, categoryEditor);
_buttonLoadOperations->setEnabled(true);
_categoriesGrid->setRowCount(_unresolvedCategories.size());
for (i=0; i<(int)_unresolvedCategories.size(); i++)
{
item = new QTableWidgetItem(_unresolvedCategories[i].name);
item->setFlags(item->flags() & ~Qt::ItemIsEditable);
_categoriesGrid->setItem(i, 0, item);
_categoriesGrid->setItem(i, 1, new QTableWidgetItem(""));
_categoriesGrid->setItem(i, 2, new QTableWidgetItem(userCategories[0]));
}
_categoriesGrid->resizeColumnsToContents();
_categoriesGrid->layout();
}
if (_unresolvedTags.size())
{
int nb_tags = user->GetTagsNumber();
userTags = new QString[nb_tags+1];
userTags[0] = _("Create one");
for(i=0; i_tags[i].name;
ChoiceDelegate* tagEditor = new ChoiceDelegate(this, userTags, nb_tags+1);
_tagsGrid->setItemDelegateForColumn(2, tagEditor);
_buttonLoadOperations->setEnabled(true);
_tagsGrid->setRowCount(_unresolvedTags.size());
for (i=0; i<(int)_unresolvedTags.size(); i++)
{
item = new QTableWidgetItem(_unresolvedTags[i].name);
item->setFlags(item->flags() & ~Qt::ItemIsEditable);
_tagsGrid->setItem(i, 0, item);
_tagsGrid->setItem(i, 1, new QTableWidgetItem(""));
_tagsGrid->setItem(i, 2, new QTableWidgetItem(userTags[0]));
}
_tagsGrid->resizeColumnsToContents();
_tagsGrid->layout();
}
if (!_unresolvedAccounts.size() && !_unresolvedCategories.size() && !_unresolvedTags.size())
{
OnLoadOperations();
}
layout();
}
void ImportPanel::OnLoadOperations()
{
int i, nbAccounts=0, nbCategories=0, nbTags=0;
User* user = _kiss->GetUser();
Account account;
Category category;
Tag tag;
std::map accounts;
std::map categories;
std::map tags;
int oldid;
for(i=0; i<_accountsGrid->rowCount(); i++)
{
if (_accountsGrid->item(i, 2)->text() == _("Create one"))
nbAccounts++;
else
{
oldid = _unresolvedAccounts[i].id;
accounts[oldid] = user->GetAccountId(_accountsGrid->item(i, 2)->text());
}
}
for(i=0; i<_categoriesGrid->rowCount(); i++)
{
if (_categoriesGrid->item(i, 2)->text() == _("Create one"))
nbCategories++;
else
{
oldid = _unresolvedCategories[i].id;
categories[oldid] = user->GetCategoryId(_categoriesGrid->item(i, 2)->text());;
}
}
for(i=0; i<_tagsGrid->rowCount(); i++)
{
if (_tagsGrid->item(i, 2)->text() == _("Create one"))
nbTags++;
else
{
oldid = _unresolvedTags[i].id;
tags[oldid] = user->GetTagId(_tagsGrid->item(i, 2)->text());;
}
}
if (nbAccounts || nbCategories || nbTags)
{
QString message, v;
if (nbAccounts)
{
message += v.asprintf(_("%d accounts").toStdString().c_str(), nbAccounts);
if (nbCategories) message += _(" and ");
}
if (nbCategories)
message += v.asprintf(_("%d categories").toStdString().c_str(), nbCategories);
if (nbTags)
message += v.asprintf(_("%d tags").toStdString().c_str(), nbTags);
message += _(" will be created, is it ok ?");
if (QMessageBox::question(0, "KissCount", message, QMessageBox::Yes|QMessageBox::No) == QMessageBox::No)
return;
for(i=0; i<_accountsGrid->rowCount(); i++)
{
if (_accountsGrid->item(i, 2)->text() == _("Create one"))
{
account = _unresolvedAccounts[i] ;
if (_accountsGrid->item(i, 1)->text().length())
account.name = _accountsGrid->item(i, 1)->text();
else
account.name = _accountsGrid->item(i, 0)->text();
if (account.name.length() == 0)
{
QMessageBox::critical(0, _("Error"), _("Account ") + QString::number(i) + _(" must have a name"));
return;
}
account.number = _accountsGrid->item(i, 0)->text();
oldid = account.id;
accounts[oldid] = account.id = _kiss->AddAccount(account);
}
}
_accountsGrid->setRowCount(0);
for(i=0; i<_categoriesGrid->rowCount(); i++)
{
if (_categoriesGrid->item(i, 2)->text() == _("Create one"))
{
category = _unresolvedCategories[i] ;
if (_categoriesGrid->item(i, 1)->text().length())
category.name = _categoriesGrid->item(i, 1)->text();
else
category.name = _categoriesGrid->item(i, 0)->text();
if (category.name.length() == 0)
{
QMessageBox::critical(0, _("Error"), _("Category ") + QString::number(i) + _(" must have a name"));
return;
}
oldid = category.id;
categories[oldid] = category.id = _kiss->AddCategory(category);
}
}
_categoriesGrid->setRowCount(0);
for(i=0; i<_tagsGrid->rowCount(); i++)
{
if (_tagsGrid->item(i, 2)->text() == _("Create one"))
{
tag = _unresolvedTags[i] ;
if (_tagsGrid->item(i, 1)->text().length())
tag.name = _tagsGrid->item(i, 1)->text();
else
tag.name = _tagsGrid->item(i, 0)->text();
if (tag.name.length() == 0)
{
QMessageBox::critical(0, _("Error"), _("Tag ") + QString::number(i) + _(" must have a name"));
return;
}
oldid = tag.id;
tags[oldid] = tag.id = _kiss->AddTag(tag);
}
}
_tagsGrid->setRowCount(0);
NeedReload();
}
_operations = _importEngine->GetOperations(accounts, categories, tags);
if (_operations->size())
{
_hbox->removeWidget(_operationsGrid);
delete _operationsGrid;
_operationsGrid = new GridAccount(_kiss, this, false, false, false);
_hbox->addWidget(_operationsGrid);
_operationsGrid->LoadOperations(_operations, 0, 0);
_buttonIntegrate->setEnabled(true);
_buttonLoadOperations->setEnabled(false);
layout();
}
else
{
QMessageBox::information(0, "KissCount", _("No operation found into this file"));
}
}
void ImportPanel::OnIntegrate()
{
int i;
std::map mapid;
int oldid, account;
bool update;
std::map accountAmounts;
std::map::iterator it;
int amount;
if (!_operations->size()) return;
if (QMessageBox::question(0, "KissCount", _("Are you sure want to integrate these operations ?"), QMessageBox::Yes|QMessageBox::No) == QMessageBox::No)
return;
_buttonIntegrate->setEnabled(false);
_kiss->setOverrideCursor(QCursor(Qt::WaitCursor));
_wxUI->setEnabled(false);
_wxUI->repaint();
for(i=0; i<(int)_operations->size(); i++)
{
oldid = (*_operations)[i].id;
_kiss->AddOperation((*_operations)[i], false);
mapid[oldid] = (*_operations)[i].id;
}
for(i=0; i<(int)_operations->size(); i++)
{
update = false;
if ((*_operations)[i].parent)
{
(*_operations)[i].parent = mapid[(*_operations)[i].parent];
update = true;
}
if ((*_operations)[i].transfert)
{
(*_operations)[i].transfert = mapid[(*_operations)[i].transfert];
update = true;
}
if (update)
_kiss->UpdateOperation((*_operations)[i], false);
}
accountAmounts = _importEngine->GetAccountAmounts();
for(it=accountAmounts.begin(); it!=accountAmounts.end(); it++)
{
account = it->first.account;
amount = _kiss->GetAccountAmount(account, it->first.month, it->first.year);
if (!amount)
_kiss->SetAccountAmount(account, it->first.month, it->first.year, it->second);
}
if (_checkSaveImportPatterns->checkState() == Qt::Checked)
_kiss->UpdateImportPattern();
_operations->clear();
_operationsGrid->ClearGrid();
_wxUI->setEnabled(true);
_kiss->setOverrideCursor(QCursor(Qt::ArrowCursor));
QMessageBox::information(0, "KissCount", _("Operations successfully imported"));
NeedReload();
}
void ImportPanel::OnOperationModified(int row, int col)
{
static bool update = false;
if (col != GridAccount::DESCRIPTION &&
col != GridAccount::CATEGORY &&
col != GridAccount::TAG &&
col != GridAccount::ACCOUNT)
return ;
if (update) return;
update = true;
_operationsGrid->ClearGrid();
if (_importEngine->UpdatePattern(row-1) > 1)
_operationsGrid->LoadOperations(_operations, 0, 0);
layout();
update = false;
}