/*
Copyright 2010-2012 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 "KissCount.hpp"
#include
#include
std::vector * KissCount::_importEngines;
std::vector * KissCount::_exportEngines;
KissCount::KissCount(int& argc, char** argv) : QApplication(argc, argv), _user(0)
{
QTextCodec::setCodecForLocale(QTextCodec::codecForName("utf8"));
_wxUI = new wxUI(this, "KissCount");
_wxUI->showMaximized();
_wxUI->setDisabled(true);
try
{
_db = new Database((argc == 2) ? argv[1] : 0, this);
}
catch (std::string s)
{
_wxUI->close();
throw s;
}
_wxUI->ChangeUser();
_wxUI->setDisabled(false);
}
KissCount::~KissCount()
{
delete _db;
delete _wxUI;
delete _importEngines;
delete _exportEngines;
if (_user) delete _user;
}
std::list KissCount::GetUsers()
{
return _db->GetUsers();
}
bool KissCount::IsValidUser(const QString& user, const QString& password)
{
return _db->IsValidUser(user, password) ;
}
void KissCount::LoadUser(const QString& user)
{
if (_user) delete _user;
_user = _db->LoadUser(user) ;
if (_user)
_wxUI->LoadUser();
}
void KissCount::LoadYear(int year, bool force)
{
if (!force && _user->_operations[year] != 0) return;
if (_user->_operations[year] != 0)
{
delete _user->_operations[year];
_user->_operations[year] = 0;
}
_db->LoadYear(_user, year);
}
User* KissCount::GetUser()
{
return _user;
}
int KissCount::GetAccountAmount(int id, int month, int year, bool* had_value, bool create_if_not_exsits)
{
return _db->GetAccountAmount(id, month, year, had_value, create_if_not_exsits);
}
int KissCount::CalcAccountAmount(int id, int month, int year, bool* had_values)
{
return _db->CalcAccountAmount(id, month, year, had_values);
}
void KissCount::UpdateOperation(Operation& op, bool checkTransfert)
{
// Unlink
if (checkTransfert)
{
op.transfert = 0;
_user->LinkOrUnlinkOperation(op);
}
_db->UpdateOperation(_user, op, checkTransfert);
// Link
if (checkTransfert)
_user->LinkOrUnlinkOperation(op);
}
int KissCount::AddOperation(Operation& op, bool checkTransfert)
{
int ret = _db->AddOperation(_user, op, checkTransfert);
if (checkTransfert && op.transfert)
_user->LinkOrUnlinkOperation(op);
return ret;
}
void KissCount::DeleteOperation(Operation& op)
{
if (op.transfert)
{
op.transfert = 0;
_user->LinkOrUnlinkOperation(op);
}
_db->DeleteOperation(_user, op);
}
void KissCount::DeleteOperations(int month, int year)
{
_db->DeleteOperations(_user, month, year);
if (month != -1)
(*_user->_operations[year]).erase(month);
if (month == -1 || !_user->_operations[year]->size())
{
delete _user->_operations[year];
_user->_operations.erase(year);
}
}
int KissCount::MetaAmount(int id)
{
return _db->MetaAmount(id);
}
int KissCount::MetaPositiveAmount(int id)
{
return _db->MetaPositiveAmount(id);
}
void KissCount::SetAccountAmount(int accountId, int month, int year, int amount)
{
_db->SetAccountAmount(accountId, month, year, amount);
}
int KissCount::AddAccount(Account& ac)
{
QDate curDate = QDate::currentDate();
ac.id = _db->AddAccount(_user, ac);
_user->AddAccount(ac);
SetAccountAmount(ac.id, curDate.month()-1, curDate.year(), 0.0);
return ac.id;
}
void KissCount::UpdateAccount(Account& ac)
{
_db->UpdateAccount(ac);
_user->UpdateAccount(ac);
}
void KissCount::DeleteAccount(Account& ac, int replacement)
{
std::map >* >::iterator it2;
_db->DeleteAccount(_user, ac, replacement);
_user->DeleteAccount(ac);
for (it2= _user->_operations.begin();
it2 != _user->_operations.end();
it2++)
LoadYear(it2->first, true);
}
void KissCount::AddSharedAccount(Account& ac, const QString& granted)
{
_db->AddSharedAccount(ac, granted);
}
void KissCount::RemoveSharedAccount(Account& ac, int granted)
{
_db->RemoveSharedAccount(ac, granted);
}
std::map KissCount::getSharedAccountOwners(int account)
{
return _db->getSharedAccountOwners(account);
}
QString KissCount::getSharedAccountOwner(int account)
{
return _db->getSharedAccountOwner(account);
}
int KissCount::AddCategory(Category& category)
{
category.id = _db->AddCategory(_user, category);
_user->AddCategory(category);
return category.id;
}
void KissCount::UpdateCategory(Category& category)
{
_db->UpdateCategory(category);
_user->UpdateCategory(category);
}
void KissCount::DeleteCategory(Category& category, int replacement)
{
std::map >* >::iterator it;
_db->DeleteCategory(_user, category, replacement);
_user->DeleteCategory(category);
for (it= _user->_operations.begin();
it != _user->_operations.end();
it++)
LoadYear(it->first, true);
}
int KissCount::AddTag(Tag& tag)
{
tag.id = _db->AddTag(_user, tag);
_user->AddTag(tag);
return tag.id;
}
void KissCount::UpdateTag(Tag& tag)
{
_db->UpdateTag(tag);
_user->UpdateTag(tag);
}
void KissCount::DeleteTag(Tag& tag, int replacement)
{
std::map >* >::iterator it;
_db->DeleteTag(_user, tag, replacement);
_user->DeleteTag(tag);
for (it= _user->_operations.begin();
it != _user->_operations.end();
it++)
LoadYear(it->first, true);
}
std::map > KissCount::GetAllOperations()
{
return _db->GetAllOperations(_user);
}
int KissCount::CopyGeneratedOperation(int monthTo, int yearTo, Operation& opOrig, Operation& opDest,
std::map &meta, QRegularExpression *regexp=0)
{
opDest = opOrig;
opDest.month = monthTo;
opDest.year = yearTo;
opDest.checked = false;
opDest.id = AddOperation(opDest);
opDest.childs.clear();
if (opOrig.meta)
meta[opOrig.id] = opDest.id;
if (regexp)
{
QRegularExpressionMatch match;
int first, second;
match = regexp->match(opOrig.description);
if (match.hasMatch())
{
first = match.captured(1).toInt();
second = match.captured(2).toInt();
if (first < second)
{
first++;
opDest.description = opOrig.description.left(match.capturedStart(1));
opDest.description += QString::number(first) + QString("/") + QString::number(second);
opDest.description += opOrig.description.right(opOrig.description.length()-match.capturedEnd(2));
UpdateOperation(opDest, false);
}
}
}
if (!QDate::isValid(opDest.year, opDest.month+1, opDest.day+1))
{
QDate d(opDest.year, opDest.month+1, 1);
opDest.day = d.daysInMonth()-1;
return 1;
}
return 0;
}
int KissCount::GenerateMonth(int monthFrom, int yearFrom, int monthTo, int yearTo)
{
std::vector::iterator it, it2;
std::vector::iterator childIt, copyIt;
std::map meta;
Operation op, op2, parentOp, childOp;
std::vector toCopy;
int nb_update_days = 0, first, second;
QRegularExpressionMatch match;
bool hasmatch;
/* Try to find something like XXX(1/5)XXX */
QRegularExpression regexp(".+\\((\\d+)/(\\d+)\\).*");
QString newDescription;
_db->GenerateMonth(_user, monthFrom, yearFrom, monthTo, yearTo);
if (!_user->_operations[yearTo])
_user->_operations[yearTo] = new std::map >();
if (monthFrom != -1 && yearFrom != -1)
{
LoadYear(yearFrom, false);
for(it = (*_user->_operations[yearFrom])[monthFrom].begin();
it != (*_user->_operations[yearFrom])[monthFrom].end();
it++)
{
hasmatch = false;
if (!it->fix_cost)
{
match = regexp.match(it->description);
if (!match.hasMatch())
continue;
hasmatch = true;
}
if (hasmatch)
{
first = match.captured(1).toInt();
second = match.captured(2).toInt();
if (first >= second)
continue;
if (!std::count(toCopy.begin(), toCopy.end(), it->id))
{
/* Copy only parent if has */
if (it->parent)
{
if (std::count(toCopy.begin(), toCopy.end(), it->parent))
continue;
toCopy.push_back(it->parent);
}
else
toCopy.push_back(it->id);
}
}
else
nb_update_days += CopyGeneratedOperation(monthTo, yearTo, *it, op, meta);
(*_user->_operations[yearTo])[monthTo].push_back(op);
}
/* Restart search and filter by copy */
for(it = (*_user->_operations[yearFrom])[monthFrom].begin();
it != (*_user->_operations[yearFrom])[monthFrom].end();
it++)
{
if (std::count(toCopy.begin(), toCopy.end(), it->id) ||
(it->parent && std::count(toCopy.begin(), toCopy.end(), it->parent)))
{
nb_update_days += CopyGeneratedOperation(monthTo, yearTo, *it, childOp, meta, ®exp);
(*_user->_operations[yearTo])[monthTo].push_back(childOp);
}
}
// Re Generate parents
for(it = (*_user->_operations[yearTo])[monthTo].begin();
it != (*_user->_operations[yearTo])[monthTo].end();
it++)
{
if (it->parent)
{
it->parent = meta[it->parent];
UpdateOperation(*it);
for(it2 = (*_user->_operations[yearTo])[monthTo].begin();
it2 != (*_user->_operations[yearTo])[monthTo].end();
it2++)
if (it2->id == it->parent)
{
it2->childs.push_back(it->id);
break;
}
}
}
}
_wxUI->GenerateMonth(monthTo, yearTo);
return nb_update_days;
}
void KissCount::ChangePassword(const QString& password)
{
_db->ChangePassword(_user, password);
}
bool KissCount::UserExists(const QString& name)
{
return _db->UserExists(name);
}
void KissCount::ChangeName(const QString& name)
{
_db->ChangeName(_user, name);
_user->_name = name;
}
// To enable translation during xgettext
QString default_cats[] = {
_("Fix"), _("Groceries"), _("Hobbies"), _("Car"),
_("Unexpected"), _("Other")
};
void KissCount::NewUser(const QString& name)
{
// QDate curDate = QDate::currentDate();
Account ac = {
/*.id = */0,
/*.name = */_("Account 1"),
/*.number = */0,
/*.shared = */false,
/*.blocked = */false,
/*._default = */true,
/*.is_owner = */true,
/* ._virtual = */false,
/* .hidden = */false,
/* .start_date = */QDate::currentDate(),
/* .end_date = */QDate::currentDate().addYears(50),
};
Category cat ;
_db->NewUser(name);
if (_user) delete _user;
_user = _db->LoadUser(name) ;
AddAccount(ac);
cat.parent = 0 ; cat.name = "Fix" ; cat.backcolor = view::OWN_YELLOW ; cat.forecolor = Qt::black; cat.fix_cost = true;
AddCategory(cat);
cat.parent = 0 ; cat.name = "Groceries" ; cat.backcolor = view::OWN_GREEN; cat.forecolor = Qt::black; cat.fix_cost = false;
AddCategory(cat);
cat.parent = 0 ; cat.name = "Hobbies" ; cat.backcolor = view::OWN_GREEN; cat.forecolor = Qt::black; cat.fix_cost = false;
AddCategory(cat);
cat.parent = 0 ; cat.name = "Car" ; cat.backcolor = view::OWN_GREEN; cat.forecolor = Qt::black; cat.fix_cost = false;
AddCategory(cat);
cat.parent = 0 ; cat.name = "Unexpected" ; cat.backcolor = view::OWN_GREEN; cat.forecolor = Qt::black; cat.fix_cost = false;
AddCategory(cat);
cat.parent = 0 ; cat.name = "Other" ; cat.backcolor = view::OWN_GREEN; cat.forecolor = Qt::black; cat.fix_cost = false;
AddCategory(cat);
SetOperationOrder("ASC");
}
void KissCount::KillMe()
{
_wxUI->KillMe();
_db->KillMe(_user);
delete _user;
_user = 0;
_wxUI->ChangeUser();
}
void KissCount::SetLanguage(QString language)
{
_user->SetLanguage(language);
}
QString KissCount::GetLanguage(void)
{
return _user->GetLanguage();
}
void KissCount::SetAccountLimitValue(int limit)
{
_user->SetAccountLimitValue(limit);
}
void KissCount::SetAccountLimitColor(QColor& color)
{
_user->SetAccountLimitColor(color);
}
int KissCount::GetAccountLimitValue(void)
{
return _user->GetAccountLimitValue();
}
QColor KissCount::GetAccountLimitColor(void)
{
return _user->GetAccountLimitColor();
}
/*
ASC (default) or DESC
*/
void KissCount::SetOperationOrder(const QString& order)
{
_user->_preferences["operation_order"] = order;
_db->UpdatePreference(_user, "operation_order");
}
const QString& KissCount::GetOperationOrder()
{
return _user->_preferences["operation_order"] ;
}
std::vector* KissCount::Search(QString* description, QDate* dateFrom, QDate* dateTo,
int* amountFrom, int* amountTo,
std::vector categories,
int types, std::vector accounts, std::vector tags)
{
return _db->Search(_user, description, dateFrom, dateTo, amountFrom, amountTo, categories, types, accounts, true, tags);
}
bool KissCount::SearchPreviousOperation(Operation* res, Operation& op, int month, int year, bool limitToType, int index)
{
std::vector* operations;
//wxDateSpan threeMonths(0, 3); Not working :(
std::vector v;
int i;
month -= 3;
if (month < 0)
{
year -= 1;
month += 12;
}
QDate date = QDate(year, month, 0);
if (limitToType)
operations = _db->Search(_user, &op.description, &date, 0, 0, 0, v, op.fix_cost ? +Database::FIX_OP : +Database::NON_FIX_OP, v, false, v);
else
operations = _db->Search(_user, &op.description, &date, 0, 0, 0, v, Database::ALL_OP, v, false, v);
if (!operations->size())
{
delete operations;
return false;
}
for(i=operations->size()-1; i>=0; i--)
if (!(*operations)[i].meta && index--)
{
*res = (*operations)[i];
delete operations;
return true;
}
delete operations;
return false;
}
void KissCount::GetHistory(int month, int year, QStringList& list)
{
month -= 3;
if (month < 0)
{
year -= 1;
month += 12;
}
_db->GetHistory(month, year, list);
}
void KissCount::GetStats(int monthFrom, int yearFrom, int monthTo, int yearTo,
std::map > >* accountAmounts,
std::map* categories, std::map* tags)
{
_db->GetStats(_user, monthFrom, yearFrom, monthTo, yearTo, accountAmounts, categories, tags);
}
void KissCount::GetMonthStats(int month, int year, int nbDays,
std::map >* operations,
std::map* categories, std::map* tags)
{
_db->GetMonthStats(_user, month, year, nbDays, operations, categories, tags);
}
void KissCount::UpdateStats()
{
_wxUI->UpdateStats();
}
std::map* KissCount::GetNotChecked(int month, int year)
{
return _db->GetNotChecked(_user, month, year);
}
std::map* KissCount::GetVirtualAmount(int month, int year)
{
return _db->GetVirtualAmount(_user, month, year);
}
QFont KissCount::ExtractFont(QString strFont)
{
long int pointSize, weight, family, style;
QString faceName;
if (!strFont.size())
{
DEFAULT_FONT(f);
return f;
}
QStringList list = strFont.split(";");
pointSize = list[0].toInt();
family = list[1].toInt();
style = list[2].toInt();
weight = list[3].toInt();
faceName = list[4];
(void) family;
(void) style;
return QFont(faceName, pointSize, weight);
}
QString KissCount::CompactFont(const QFont& font)
{
QString res ;
res = res.sprintf("%d;%d;%d;%d;", font.pointSize(), 0, 0, font.weight());
res += font.family();
return res;
}
void KissCount::UnRegisterImportEngine(ImportEngine* engine)
{
std::vector* importEngines = KissCount::GetImportEngines();
std::vector::iterator it = std::find(importEngines->begin(), importEngines->end(), engine);
if (it!=importEngines->end()) importEngines->erase(it);
}
void KissCount::RegisterImportEngine(ImportEngine* engine)
{
std::vector* importEngines = KissCount::GetImportEngines();
importEngines->push_back(engine);
}
QString KissCount::GetImportEngineExtensions()
{
QString res;
std::vector::iterator it;
int i;
std::vector* importEngines = KissCount::GetImportEngines();
for(i=0; i<(int)importEngines->size()-1; i++)
res = res + (*importEngines)[i]->GetFileExt() + ";;" ;
if (importEngines->size())
res = res + (*importEngines)[i]->GetFileExt();
return res;
}
ImportEngine* KissCount::GetImportEngine(QString path)
{
std::vector::iterator it;
std::vector* importEngines = KissCount::GetImportEngines();
for(it=importEngines->begin(); it!=importEngines->end(); it++)
{
if ((*it)->HandleFile(path, _user, _db, this))
return *it;
}
return 0;
}
void KissCount::UpdateImportPattern()
{
_db->UpdateImportPattern(_user);
}
void KissCount::UnRegisterExportEngine(ExportEngine* engine)
{
std::vector* exportEngines = KissCount::GetExportEngines();
std::vector::iterator it = std::find(exportEngines->begin(), exportEngines->end(), engine);
if (it!=exportEngines->end()) exportEngines->erase(it);
}
void KissCount::RegisterExportEngine(ExportEngine* engine)
{
std::vector* exportEngines = KissCount::GetExportEngines();
exportEngines->push_back(engine);
}
QString KissCount::GetExportEngineExtensions()
{
QString res;
std::vector::iterator it;
int i;
std::vector* exportEngines = KissCount::GetExportEngines();
for(i=0; i<(int)exportEngines->size()-1; i++)
res = res + (*exportEngines)[i]->GetFileExt() + ";;" ;
if (exportEngines->size())
res = res + (*exportEngines)[i]->GetFileExt();
return res;
}
ExportEngine* KissCount::GetExportEngine(QString path, QString filter)
{
std::vector::iterator it;
std::vector* exportEngines = KissCount::GetExportEngines();
QString tmp;
for(it=exportEngines->begin(); it!=exportEngines->end(); it++)
{
if (filter == (*it)->GetFileExt())
{
tmp = path;
if (!tmp.endsWith((*it)->GetShortExt()))
tmp += (*it)->GetShortExt();
if ((*it)->HandleFile(tmp, _user, _db, this))
return *it;
}
}
return 0;
}
std::vector* KissCount::GetImportEngines()
{
if (!_importEngines)
_importEngines = new std::vector;
return _importEngines;
}
std::vector* KissCount::GetExportEngines()
{
if (!_exportEngines)
_exportEngines = new std::vector;
return _exportEngines;
}
bool KissCount::ChangeDatabase(QString filename)
{
return _db->ChangeDatabase(filename);
}
QLocale* KissCount::GetLocale()
{
return _wxUI->GetLocale();
}
QString KissCount::GetDateLocalFormat()
{
return _wxUI->GetDateLocalFormat();
}
QString KissCount::FormatDate(int day, int month, int year)
{
return QDate(year, month, day).toString(_wxUI->GetDateLocalFormat());
}