/* Copyright 2010 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 "GridAccount.h" #define SET_ROW_COLOR(row, backcolor, forecolor) for(int i=0; iGetUser(); std::vector::iterator accountIt; std::vector::iterator categoryIt; CreateGrid(1, NUMBER_COLS_OPS); SetColLabelSize(0); SetRowLabelSize(0); SetColSize (DESCRIPTION, GetColSize(DESCRIPTION)*3); SetDefaultCellFont(font); font.SetWeight(wxFONTWEIGHT_BOLD); wxString colsName[] = {wxT(""), _("Description"), _("Date"), _("Debit"), _("Credit"), _("Category"), _("Account"), wxT(""), wxT("")}; for(i=0; iGetAccountsNumber()]; for (i=0, accountIt = user->_accounts.begin(); accountIt != user->_accounts.end(); accountIt++, i++) _accounts[i] = accountIt->name; _categories = new wxString[user->GetCategoriesNumber()] ; for(i=0, categoryIt = user->_categories.begin(); categoryIt != user->_categories.end(); categoryIt++, i++) { _categories[i] = categoryIt->name ; } Connect(id, wxEVT_GRID_CELL_CHANGE, wxGridEventHandler(GridAccount::OnOperationModified), NULL, this); AutoSizeColumn(TREE, false); AutoSizeColumn(CATEGORY, false); AutoSizeColumn(DATE, false); AutoSizeColumn(ACCOUNT, false); AutoSizeColumn(DELETE, false); AutoSizeColumn(CHECKED, false); } GridAccount::~GridAccount() { delete[] _categories; delete[] _accounts; } wxPen GridAccount::GetColGridLinePen (int col) {return wxPen(*wxBLACK, 1, wxSOLID);} wxPen GridAccount::GetRowGridLinePen (int row) { if (row == 0 || row == _fixCosts || row == _week1 || row == _week2 || row == _week3 || row == _week4) return wxPen(*wxBLACK, 1, wxSOLID); return GetCellBackgroundColour(row, 0); } void GridAccount::ResetWeeks() { _week1 = _week2 = _week3 = _week4 = 0; } void GridAccount::SetWeek(int week, int line) { switch (week) { case 1: _week1 = line; break; case 2: _week2 = line; break; case 3: _week3 = line; break; case 4: _week4 = line; break; } } Operation& GridAccount::GetOperation(const wxString& id) { std::vector::iterator it; for(it=_operations->begin(); it!=_operations->end(); it++) if (it->id == id) return *it; } int GridAccount::GetDisplayedRow(const wxString& id) { for(int i=0; i<(int)_displayedOperations.size(); i++) if (_displayedOperations[i].id == id) return i; return -1; } void GridAccount::LoadOperations(std::vector* operations, bool canAddOperation, bool setWeek, int month, int year) { std::vector::iterator it; User* user = _kiss->GetUser(); int curLine = 0; Operation NULLop; NULLop.id = wxT(""); _operations = operations; _canAddOperation = canAddOperation; _curMonth = month; _curYear = year; _displayedOperations.clear(); _displayedOperations.push_back(NULLop); // Header _fixCosts = 0; it = _operations->begin(); for (;it != _operations->end() && it->fix_cost; it++) { if (it->parent.Length()) continue; if (setWeek) InsertOperationWithWeek(user, *it, ++curLine, true, it->month, it->year); else InsertOperation(user, *it, ++curLine, true, it->month, it->year); } if (canAddOperation) InsertOperation(user, NULLop, ++curLine, true, month, year); for (; it != _operations->end(); it++) { if (it->parent.Length()) continue; if (setWeek) InsertOperationWithWeek(user, *it, ++curLine, false, it->month, it->year); else InsertOperation(user, *it, ++curLine, false, it->month, it->year); } if (canAddOperation) InsertOperation(user, NULLop, ++curLine, false, month, year); AutoSizeColumn(TREE, false); AutoSizeColumn(CATEGORY, false); AutoSizeColumn(DATE, false); AutoSizeColumn(ACCOUNT, false); AutoSizeColumn(DELETE, false); AutoSizeColumn(CHECKED, false); } void GridAccount::InsertOperationWithWeek(User* user, Operation& op, int line, bool fix, int month, int year) { std::vector::iterator it; int curLine, curWeek, week, i; InsertOperation(user, op, line, fix, month, year); if (op.id.Length() && !fix) { for (it = _displayedOperations.begin(), curLine=0; it != _displayedOperations.end(); it++, curLine++) { if (it->id.Length() && !it->fix_cost) break; } if (it == _displayedOperations.end()) return; ResetWeeks(); curWeek = wxDateTime(it->day+1, (wxDateTime::Month)it->month, it->year).GetWeekOfMonth(); it++; for (i=1; it != _displayedOperations.end(); it++, curLine++) { if (!it->id.Length()) continue; week = wxDateTime(it->day+1, (wxDateTime::Month)it->month, it->year).GetWeekOfMonth(); if (week != curWeek) { SetWeek(i++, curLine); curWeek = week; } } } } void GridAccount::InsertOperation(User* user, Operation& op, int line, bool fix, int month, int year) { std::vector::iterator it; std::vector::iterator it2; int r, g, b, amount; wxColour color; wxDateTime curDate; wxString description; wxFont font; Category cat ; Operation op2; curDate.SetToCurrent(); // // First is header // if (op) _displayedOperations.insert(_displayedOperations.begin()+line, op); if (!user->_accounts.size()) return; InsertRows(line, 1); if (op.id.Length() && op.meta) { SetCellRenderer(line, TREE, new wxGridCellTreeButtonRenderer()); SetCellEditor(line, TREE, new wxGridCellTreeButtonEditor()); SetReadOnly(line, DATE, true); SetReadOnly(line, CREDIT, true); SetReadOnly(line, DEBIT, true); SetReadOnly(line, CATEGORY, true); SetReadOnly(line, ACCOUNT, true); } else SetReadOnly(line, TREE, true); SetCellEditor(line, DEBIT, new wxGridCellFloatEditor(wxID_ANY, 2)); SetCellEditor(line, CREDIT, new wxGridCellFloatEditor(wxID_ANY, 2)); wxGridCellChoiceEditor* accountEditor = new wxGridCellChoiceEditor(user->GetAccountsNumber(), _accounts, false); SetCellEditor(line, ACCOUNT, accountEditor); wxGridCellChoiceEditor* categoryEditor = new wxGridCellChoiceEditor(user->GetCategoriesNumber()-1, _categories+1, false); SetCellEditor(line, CATEGORY, categoryEditor); if (fix) { SetCellValue(line, CATEGORY, _("Fix")); SetReadOnly(line, CATEGORY); _fixCosts++; } if (op.id.Length()) { cat = user->GetCategory(op.category); SetCellEditor(line, DATE, new CalendarEditor(op.day, month, year)); description = op.description; UNESCAPE_CHARS(description); SetCellValue(line, DESCRIPTION, description); SetCellValue(line, DATE, wxString::Format(wxT("%02d/%02d/%d"), op.day+1, month+1, year)); if (op.amount < 0) SetCellValue(line, DEBIT, wxString::Format(wxT("%.2lf"), -op.amount)); else SetCellValue(line, CREDIT, wxString::Format(wxT("%.2lf"), op.amount)); if (!op.meta) SetCellValue(line, ACCOUNT, user->GetAccountName(op.account)); if (!fix && !op.meta) SetCellValue(line, CATEGORY, cat.name); SetCellRenderer(line, DELETE, new wxGridCellBoolRenderer ()); SetCellEditor(line, DELETE, new wxGridCellBoolEditor ()); SetCellRenderer(line, CHECKED, new wxGridCellBoolRenderer ()); SetCellEditor(line, CHECKED, new wxGridCellFastBoolEditor ()); color = cat.backcolor; if (op.checked) { r = ((color.Red()*1.5) >= 0xFF) ? 0xFF : color.Red()*1.5 ; g = ((color.Green()*1.5) >= 0xFF) ? 0xFF : color.Green()*1.5 ; b = ((color.Blue()*1.5) >= 0xFF) ? 0xFF : color.Blue()*1.5 ; color.Set(r, g, b, color.Alpha()); SetCellValue(line, CHECKED, wxT("1")); } SET_ROW_COLOR(line, color, cat.forecolor); if (cat.font.Length()) { font = user->GetCategoryFont(cat.id); SET_ROW_FONT(line, font); } if (op.meta && !op.amount) { amount = 0; for(it2=op.childs.begin(); it2!=op.childs.end(); it2++) { op2 = GetOperation(*it2); if (op2.amount > 0) amount += op2.amount; } SetCellValue(line, DEBIT, wxString::Format(wxT("%.2lf"), amount)); SetCellValue(line, CREDIT, wxString::Format(wxT("%.2lf"), amount)); } } else { SetCellEditor(line, DATE, new CalendarEditor(0, month, year)); if (!fix && curDate.GetMonth() == month && curDate.GetYear() == year) { SetCellValue(line, DATE, wxString::Format(wxT("%02d/%02d/%d"), curDate.GetDay(), month+1, year)); SetCellEditor(line, DATE, new CalendarEditor(curDate.GetDay()-1, month, year)); } else SetCellEditor(line, DATE, new CalendarEditor(0, month, year)); if (fix) { SET_ROW_COLOR(line, OWN_YELLOW, *wxBLACK); } else { SET_ROW_COLOR(line, OWN_GREEN, *wxBLACK); } SetReadOnly(line, CHECKED, true); SetReadOnly(line, DELETE, true); } SetCellAlignment(line, DATE, wxALIGN_CENTRE, wxALIGN_CENTRE); SetCellAlignment(line, DEBIT, wxALIGN_RIGHT, wxALIGN_CENTRE); SetCellAlignment(line, CREDIT, wxALIGN_RIGHT, wxALIGN_CENTRE); SetCellAlignment(line, DELETE, wxALIGN_CENTRE, wxALIGN_CENTRE); SetCellAlignment(line, CHECKED, wxALIGN_CENTRE, wxALIGN_CENTRE); AutoSizeRow(line); Layout(); SetMinSize(GetMinSize()); } // From http://nomadsync.cvs.sourceforge.net/nomadsync/nomadsync/src/ void GridAccount::OnCellLeftClick(wxGridEvent& evt) { if (evt.GetCol() != TREE && evt.GetCol() != DELETE && evt.GetCol() != CHECKED) { evt.Skip() ; return;} // This forces the cell to go into edit mode directly //m_waitForSlowClick = TRUE; SetGridCursor(evt.GetRow(), evt.GetCol()); // Store the click co-ordinates in the editor if possible // if an editor has created a ClientData area, we presume it's // a wxPoint and we store the click co-ordinates wxGridCellEditor* pEditor = GetCellEditor(evt.GetRow(), evt.GetCol()); wxPoint* pClickPoint = (wxPoint*)pEditor->GetClientData(); if (pClickPoint) { *pClickPoint = ClientToScreen(evt.GetPosition()); #ifndef __WINDOWS__ EnableCellEditControl(true); #endif } // hack to prevent selection from being lost when click combobox if (/*evt.GetCol() == 0 && */IsInSelection(evt.GetRow(), evt.GetCol())) { //m_selTemp = m_selection; m_selection = NULL; } pEditor->DecRef(); evt.Skip(); } void GridAccount::DeleteOperation(const Operation& op) { std::vector::iterator it; for (it=_operations->begin(); it!=_operations->end(); it++) if (it->id == op.id) { _operations->erase(it); break; } } void GridAccount::InsertIntoGrid(Operation& op) { int i; User* user = _kiss->GetUser(); for(i=0; i<(int)_displayedOperations.size(); i++) { if (!_displayedOperations[i].id.Length()) continue; if ((_displayedOperations)[i].fix_cost && !op.fix_cost) continue; if (!(_displayedOperations)[i].fix_cost && op.fix_cost) break; if (user->_preferences[wxT("operation_order")] == wxT("ASC")) { if ((_displayedOperations)[i].day > op.day) break; } else { if ((_displayedOperations)[i].day < op.day) break; } } if (i == (int)_displayedOperations.size() || i == _fixCosts) i--; else if (!(_displayedOperations)[i].fix_cost && op.fix_cost) i --; _operations->push_back(op); InsertOperationWithWeek(user, (*_operations)[_operations->size()-1], i, op.fix_cost, _curMonth, _curYear); } /* !!! op must not be a reference because we modify _displayedOerations !!! */ void GridAccount::RemoveMeta(Operation& op, int line, bool removeRoot, bool deleteOp) { std::vector::iterator it, it2; wxGridCellTreeButtonRenderer* treeRenderer; int i; Operation op2; treeRenderer = (wxGridCellTreeButtonRenderer*) GetCellRenderer(line, TREE); for(i=0; i<(int)op.childs.size(); i++) { op2 = GetOperation(op.childs[i]); if (op2.meta) RemoveMeta(op2, line+1, true, deleteOp); else { if (treeRenderer->IsCollapsed()) { DeleteRows(line+1, 1); if (_displayedOperations[line+1].fix_cost) _fixCosts--; _displayedOperations.erase(_displayedOperations.begin()+line+1); } if (deleteOp) { DeleteOperation(op2); _kiss->DeleteOperation(op2); } } } op.childs.clear(); if (removeRoot) { DeleteRows(line, 1); _displayedOperations.erase(_displayedOperations.begin()+line); if (deleteOp) { DeleteOperation(op); _kiss->DeleteOperation(op); } } treeRenderer->DecRef(); } void GridAccount::OnOperationModified(wxGridEvent& event) { User* user = _kiss->GetUser(); int row = event.GetRow(); int col = event.GetCol(); Operation new_op, cur_op, op_tmp; int op_complete = 6, i; wxString value ; wxDateTime date; bool need_insertion = false, fix_op=false; static bool inModification = false ; wxColour color ; unsigned char r, g, b; wxGridCellTreeButtonRenderer* treeRenderer; std::vector::iterator it; Operation op, op2; // Avoid recursives calls if (inModification) return; inModification = true ; if (col == TREE) { treeRenderer = (wxGridCellTreeButtonRenderer*) GetCellRenderer(row, col); op = _displayedOperations[row]; if (!treeRenderer->IsCollapsed()) { for (i=1, it=op.childs.begin(); it!=op.childs.end(); it++, i++) { op2 = GetOperation(*it); InsertOperation(user, op2, row+i, op2.fix_cost, _curMonth, _curYear); } } else { RemoveMeta(op, row, false, false); } treeRenderer->DecRef(); inModification = false; return; } if (col == DEBIT) SetCellValue(row, CREDIT, wxT("")); else if (col == CREDIT) SetCellValue(row, DEBIT, wxT("")); value = GetCellValue(row, DESCRIPTION); if (value.Length()) { new_op.description = value; op_complete--; } value = GetCellValue(row, DATE); if (value.Length()) { date.ParseFormat(value, wxT("%d/%m/%Y")); new_op.day = date.GetDay()-1; new_op.month = date.GetMonth(); new_op.year = date.GetYear(); op_complete--; } value = GetCellValue(row, DEBIT); if (value.Length()) { value.ToDouble(&new_op.amount); new_op.amount *= -1.0; op_complete--; } value = GetCellValue(row, CREDIT); if (value.Length()) { value.ToDouble(&new_op.amount); op_complete--; } value = GetCellValue(row, CATEGORY); if (value.Length()) { new_op.category = user->GetCategoryId(value); op_complete--; } value = GetCellValue(row, ACCOUNT); if (value.Length()) { new_op.account = user->GetAccountId(value); op_complete--; } value = GetCellValue(row, CHECKED); if (value.Length() && value != wxT("0")) new_op.checked = true; else new_op.checked = false; op_complete--; if (col == DESCRIPTION && (!GetCellValue(row, CATEGORY).Length() || !GetCellValue(row, ACCOUNT).Length())) { if (_kiss->SearchPreviousOperation(&op_tmp, new_op.description, _curMonth, _curYear)) { new_op.category = op_tmp.category; new_op.account = op_tmp.account; SetCellValue(row, CATEGORY, user->GetCategoryName(new_op.category)); SetCellValue(row, ACCOUNT, user->GetAccountName(new_op.account)); op_complete -= 2; } } cur_op = (_displayedOperations)[row] ; if (col == CHECKED || col == CATEGORY) { color = user->GetCategory(new_op.category).backcolor; if (new_op.checked) { r = ((color.Red()*1.5) >= 0xFF) ? 0xFF : color.Red()*1.5 ; g = ((color.Green()*1.5) >= 0xFF) ? 0xFF : color.Green()*1.5 ; b = ((color.Blue()*1.5) >= 0xFF) ? 0xFF : color.Blue()*1.5 ; color.Set(r, g, b, color.Alpha()); } SET_ROW_COLOR(row, color, user->GetCategory(new_op.category).forecolor); SET_ROW_FONT(row, user->GetCategoryFont(new_op.category)); } if (col == DELETE) { wxMessageDialog dialog(this, _("Are you sure want to delete : \n")+new_op.description, wxT("KissCount"), wxYES_NO); if (dialog.ShowModal() == wxID_NO) { SetCellValue(row, col, wxT("0")); inModification = false; return; } } // Modify a fix operation if (row < _fixCosts) { if (col == DELETE) { if (cur_op.parent.Length()) user->UnGroup(_displayedOperations[row]); if (cur_op.meta) RemoveMeta(_displayedOperations[row], row, true, true); else { DeleteRows(row, 1); DeleteOperation(cur_op); _kiss->DeleteOperation(cur_op); _displayedOperations.erase(_displayedOperations.begin()+row); } _fixCosts = _fixCosts--; inModification = false ; event.Skip(); return ; } new_op.id = cur_op.id; new_op.fix_cost = true; new_op.transfert = cur_op.transfert; new_op.meta = cur_op.meta; new_op.parent = cur_op.parent; new_op.childs = cur_op.childs; if (cur_op.day != new_op.day) { need_insertion = true; DeleteRows(row, 1); DeleteOperation(cur_op); _displayedOperations.erase(_displayedOperations.begin()+row); _fixCosts--; _kiss->UpdateOperation(new_op); } else { _kiss->UpdateOperation(new_op); (_displayedOperations)[row] = new_op; } fix_op = true; } // Add a fixCost else if (row == _fixCosts) { if (op_complete) { inModification = false ; return ; } need_insertion = true; fix_op = true; new_op.fix_cost = true; new_op.meta = false; for(i=0; iAddOperation(new_op); } // Modify an operation else if (row < (int)(_displayedOperations.size()-1)) { new_op.id = cur_op.id; new_op.fix_cost = false; new_op.transfert = cur_op.transfert; new_op.meta = cur_op.meta; new_op.parent = cur_op.parent; new_op.childs = cur_op.childs; if (col == DELETE) { if (cur_op.parent.Length()) user->UnGroup(_displayedOperations[row]); if (cur_op.meta) RemoveMeta(_displayedOperations[row], row, true, true); else { DeleteRows(row, 1); DeleteOperation(cur_op); _displayedOperations.erase(_displayedOperations.begin()+row); _kiss->DeleteOperation(cur_op); } inModification = false ; event.Skip(); return ; } if (cur_op.day != new_op.day) { need_insertion = true; DeleteRows(row, 1); DeleteOperation(cur_op); _displayedOperations.erase(_displayedOperations.begin()+row); _kiss->UpdateOperation(new_op); } else { _kiss->UpdateOperation(new_op); (_displayedOperations)[row] = new_op; } } // Add an operation else { if (op_complete) { inModification = false ; return ; } need_insertion = true; fix_op = false; new_op.fix_cost = false; new_op.meta = false; for(i=0; iAddOperation(new_op); } if (need_insertion) InsertIntoGrid(new_op); inModification = false ; if (!need_insertion && (col == CHECKED || col == CATEGORY) && new_op.parent.Length()) { op2 = GetOperation(new_op.parent); UpdateMeta(op2); int row2 = GetDisplayedRow(op2.id); wxGridEvent event2(-1, 0, NULL, row2, CATEGORY); OnOperationModified(event2); } event.Skip(); } void GridAccount::UpdateMeta(Operation& op) { std::vector::iterator it; Operation op_ ; wxString category = wxT(""); bool updateCat = false ; op.category = wxT(""); op.checked = true; op.amount = 0; for(it=op.childs.begin(); it!=op.childs.end(); it++) { op_ = GetOperation(*it); if (op_.year <= op.year && op_.month <= op.month && op_.day < op.day) { op.year = op_.year; op.month = op_.month; op.day = op_.day; } op.amount += op_.amount; op.checked &= op_.checked; if (!op.description.Length() && op_.description.Length()) op.description = op_.description; if (!category.Length()) { if (op_.category.Length()) { category = op_.category; updateCat = true; } } else { if (op_.category.Length() && op_.category != category) updateCat = false; } op_.parent = op.id; } if (updateCat) op.category = category; _kiss->UpdateOperation(op); } void GridAccount::Group() { std::vector rows; std::vector::iterator it; std::vector ops; std::vector::iterator it2; std::vector::iterator it3; wxString parent = wxT(""); Operation op, op2; int fix = -1, i; // Singly selected cells. const wxGridCellCoordsArray& cells(GetSelectedCells()); for (size_t i = 0; i < cells.size(); ++i) { const wxGridCellCoords& c = cells[i]; for (it=rows.begin(); it!=rows.end(); it++) if (*it == c.GetRow()) break; if (it != rows.end()) continue; op = _displayedOperations[c.GetRow()] ; if (op.id.Length()) { if (!parent.Length()) { if (op.parent.Length()) { parent = op.parent; continue; } else if(op.meta) parent = op.id; } else { if ((parent.Length() && op.parent.Length() && op.parent != parent)) { wxMessageBox(_("Cannot group these operations"), _("Error"), wxICON_ERROR | wxOK); return ; } } if (fix != -1 && ((!fix && op.fix_cost) || (fix && !op.fix_cost))) { wxMessageBox(_("Cannot group these operations"), _("Error"), wxICON_ERROR | wxOK); return ; } if (fix == -1) fix = op.fix_cost ? 1 : 0; ops.push_back(op); rows.push_back(c.GetRow()); } } if (!parent.Length()) { if (rows.size() < 2) return; op.parent = wxT(""); op.day = ops[0].day; op.month = ops[0].month; op.year = ops[0].year; op.amount = 0; op.description = wxT(""); op.category = wxT(""); op.fix_cost = fix; op.account = wxT(""); op.checked = false; op.transfert = wxT(""); op.formula = wxT(""); op.meta = true; op.childs.clear(); op.id = _kiss->AddOperation(op); } else { if (rows.size() < 1) return; for(i=0, it2=_displayedOperations.begin(); it2!=_displayedOperations.end(); it2++, i++) if (it2->id.Length() && it2->id == parent) { RemoveMeta(*it2, i, true, false); op = *it2; break; } if (op.id.Length()) return; } std::sort(rows.begin(), rows.end()); for(i=0; i<(int)rows.size(); i++) { if (rows[i] >= i) { if (ops[i].meta) RemoveMeta(ops[i], rows[i]-i, true, false); else { DeleteRows(rows[i]-i, 1); _displayedOperations.erase(_displayedOperations.begin()+rows[i]-i); } } else { if (ops[i].meta) RemoveMeta(ops[i], rows[i], true, false); else { DeleteRows(rows[i], 1); _displayedOperations.erase(_displayedOperations.begin()+rows[i]); } } } for(it2=ops.begin(); it2!=ops.end(); it2++) { for (i=0, it3=op.childs.begin(); it3!=op.childs.end(); it3++, i++) { op2 = GetOperation(*it3); if (*it3 == it2->id || op2.day > it2->day) break; } if (i) i--; if (it3 == op.childs.end()) op.childs.push_back(it2->id); else if (*it3 == it2->id) continue; else op.childs.insert(op.childs.begin()+i, it2->id); it2->parent = op.id; _kiss->UpdateOperation(*it2); } UpdateMeta(op); InsertIntoGrid(op); } void GridAccount::UnGroup() { }