summaryrefslogtreecommitdiff
path: root/plugins/HistoryStats/src/column_inoutgraph.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/HistoryStats/src/column_inoutgraph.cpp')
-rw-r--r--plugins/HistoryStats/src/column_inoutgraph.cpp356
1 files changed, 356 insertions, 0 deletions
diff --git a/plugins/HistoryStats/src/column_inoutgraph.cpp b/plugins/HistoryStats/src/column_inoutgraph.cpp
new file mode 100644
index 0000000000..a28c4b7fd5
--- /dev/null
+++ b/plugins/HistoryStats/src/column_inoutgraph.cpp
@@ -0,0 +1,356 @@
+#include "_globals.h"
+#include "column_inoutgraph.h"
+
+/*
+ * ColInOutGraph
+ */
+
+ColInOutGraph::ColInOutGraph()
+ : m_nSource(0), m_bAbsolute(false), m_nAbsTime(1), m_bShowSum(true),
+ m_bDetail(true), m_bDetailPercent(false), m_bDetailInvert(false), m_bGraphPercent(true),
+ m_hSource(NULL), m_hAbsolute(NULL), m_hAbsTime(NULL), m_hShowSum(NULL),
+ m_hDetail(NULL), m_hDetailPercent(NULL), m_hDetailInvert(NULL), m_hGraphPercent(NULL)
+{
+}
+
+void ColInOutGraph::impl_copyConfig(const Column* pSource)
+{
+ const ColInOutGraph& src = *reinterpret_cast<const ColInOutGraph*>(pSource);
+
+ m_nSource = src.m_nSource;
+ m_bAbsolute = src.m_bAbsolute;
+ m_nAbsTime = src.m_nAbsTime;
+ m_bShowSum = src.m_bShowSum;
+ m_bDetail = src.m_bDetail;
+ m_bDetailPercent = src.m_bDetailPercent;
+ m_bDetailInvert = src.m_bDetailInvert;
+ m_bGraphPercent = src.m_bGraphPercent;
+}
+
+void ColInOutGraph::impl_configRead(const SettingsTree& settings)
+{
+ m_nSource = settings.readIntRanged(con::KeySource , 0, 0, 2);
+ m_bAbsolute = settings.readBool (con::KeyAbsolute , false);
+ m_nAbsTime = settings.readIntRanged(con::KeyAbsTime , 1, 0, 2);
+ m_bShowSum = settings.readBool (con::KeyShowSum , true);
+ m_bDetail = settings.readBool (con::KeyDetail , true);
+ m_bDetailPercent = settings.readBool (con::KeyDetailPercent, false);
+ m_bDetailInvert = settings.readBool (con::KeyDetailInvert , false);
+ m_bGraphPercent = settings.readBool (con::KeyGraphPercent , true);
+}
+
+void ColInOutGraph::impl_configWrite(SettingsTree& settings) const
+{
+ settings.writeInt (con::KeySource , m_nSource );
+ settings.writeBool(con::KeyAbsolute , m_bAbsolute );
+ settings.writeInt (con::KeyAbsTime , m_nAbsTime );
+ settings.writeBool(con::KeyShowSum , m_bShowSum );
+ settings.writeBool(con::KeyDetail , m_bDetail );
+ settings.writeBool(con::KeyDetailPercent, m_bDetailPercent);
+ settings.writeBool(con::KeyDetailInvert , m_bDetailInvert );
+ settings.writeBool(con::KeyGraphPercent , m_bGraphPercent );
+}
+
+void ColInOutGraph::impl_configToUI(OptionsCtrl& Opt, OptionsCtrl::Item hGroup)
+{
+ OptionsCtrl::Group hTemp;
+
+ /**/hTemp = Opt.insertGroup(hGroup, i18n(muT("Data source")));
+ /**/ m_hSource = Opt.insertRadio(hTemp, NULL, i18n(muT("Characters")));
+ /**/ Opt.insertRadio(hTemp, m_hSource, i18n(muT("Messages")));
+ /**/ Opt.insertRadio(hTemp, m_hSource, i18n(muT("Chats")));
+ /**/hTemp = Opt.insertGroup(hGroup, i18n(muT("Display as")));
+ /**/ m_hAbsolute = Opt.insertRadio(hTemp, NULL, i18n(muT("Absolute")));
+ /**/ m_hAbsolute = Opt.insertRadio(hTemp, m_hAbsolute, i18n(muT("Average")), OptionsCtrl::OCF_DISABLECHILDSONUNCHECK);
+ /**/ m_hAbsTime = Opt.insertRadio(m_hAbsolute, NULL, i18n(muT("Units per day")));
+ /**/ Opt.insertRadio(m_hAbsolute, m_hAbsTime, i18n(muT("Units per week")));
+ /**/ Opt.insertRadio(m_hAbsolute, m_hAbsTime, i18n(muT("Units per month (30 days)")));
+ /**/m_hShowSum = Opt.insertCheck(hGroup, i18n(muT("Show sum of incoming and outgoing")));
+ /**/m_hDetail = Opt.insertCheck(hGroup, i18n(muT("In/out details (tooltip)")), OptionsCtrl::OCF_DISABLECHILDSONUNCHECK);
+ /**/ m_hDetailPercent = Opt.insertCheck(m_hDetail, i18n(muT("Percentage in tooltip")));
+ /**/ m_hDetailInvert = Opt.insertCheck(m_hDetail, i18n(muT("Absolute in tooltip if average selected and vice versa")));
+ /**/m_hGraphPercent = Opt.insertCheck(hGroup, i18n(muT("Percentage in bar graph")));
+
+ Opt.setRadioChecked(m_hSource , m_nSource );
+ Opt.setRadioChecked(m_hAbsolute , m_bAbsolute ? 0 : 1);
+ Opt.setRadioChecked(m_hAbsTime , m_nAbsTime );
+ Opt.checkItem (m_hShowSum , m_bShowSum );
+ Opt.checkItem (m_hDetail , m_bDetail );
+ Opt.checkItem (m_hDetailPercent, m_bDetailPercent );
+ Opt.checkItem (m_hDetailInvert , m_bDetailInvert );
+ Opt.checkItem (m_hGraphPercent , m_bGraphPercent );
+}
+
+void ColInOutGraph::impl_configFromUI(OptionsCtrl& Opt)
+{
+ m_nSource = Opt.getRadioChecked(m_hSource);
+ m_bAbsolute = (Opt.getRadioChecked(m_hAbsolute) == 0);
+ m_nAbsTime = Opt.getRadioChecked(m_hAbsTime);
+ m_bShowSum = Opt.isItemChecked(m_hShowSum);
+ m_bDetail = Opt.isItemChecked(m_hDetail);
+ m_bDetailPercent = Opt.isItemChecked(m_hDetailPercent);
+ m_bDetailInvert = Opt.isItemChecked(m_hDetailInvert);
+ m_bGraphPercent = Opt.isItemChecked(m_hGraphPercent);
+}
+
+Column::StyleList ColInOutGraph::impl_outputGetAdditionalStyles(IDProvider& idp)
+{
+ StyleList l;
+
+ m_CSS = idp.getID();
+
+ l.push_back(StylePair(muT("td.") + m_CSS, muT("vertical-align: middle; padding: 2px 2px 2px 2px;")));
+
+ if (m_bShowSum)
+ {
+ l.push_back(StylePair(muT("td.") + m_CSS + muT(" div.n"), muT("text-align: center;")));
+ }
+
+ if (!usePNG())
+ {
+ l.push_back(StylePair(muT("div.") + m_CSS, muT("position: relative; left: 50%; margin-left: -50px; width: 100px; height: 15px; background-color: ") + utils::colorToHTML(con::ColorBack) + muT(";")));
+ l.push_back(StylePair(muT("div.") + m_CSS + muT(" div"), muT("position: absolute; top: 0px; height: 15px; overflow: hidden; font-size: 80%; color: ") + utils::colorToHTML(con::ColorBack) + muT(";")));
+ l.push_back(StylePair(muT("div.") + m_CSS + muT(" div.obar"), muT("left: 0px; background-color: ") + utils::colorToHTML(con::ColorOut) + muT(";")));
+ l.push_back(StylePair(muT("div.") + m_CSS + muT(" div.ibar"), muT("right: 0px; background-color: ") + utils::colorToHTML(con::ColorIn) + muT(";")));
+
+ if (m_bGraphPercent)
+ {
+ l.push_back(StylePair(muT("div.") + m_CSS + muT(" div.otext"), muT("left: 2px; width: 48px; text-align: left; z-index: 9;")));
+ l.push_back(StylePair(muT("div.") + m_CSS + muT(" div.itext"), muT("right: 2px; width: 48px; text-align: right; z-index: 9;")));
+ }
+ }
+
+ return l;
+}
+
+SIZE ColInOutGraph::impl_outputMeasureHeader() const
+{
+ SIZE colSize = { 2, 2 };
+
+ return colSize;
+}
+
+void ColInOutGraph::impl_outputRenderHeader(ext::ostream& tos, int row, int rowSpan) const
+{
+ static const mu_text* szSourceDesc[] = {
+ I18N(muT("Characters")),
+ I18N(muT("Messages")),
+ I18N(muT("Chats"))
+ };
+
+ static const mu_text* szSourceUnit[] = {
+ I18N(muT("day")),
+ I18N(muT("week")),
+ I18N(muT("month")),
+ };
+
+ if (row == 1)
+ {
+ ext::string strTitle;
+
+ if (m_bAbsolute)
+ {
+ strTitle = i18n(szSourceDesc[m_nSource]);
+ }
+ else
+ {
+ strTitle = str(ext::kformat(i18n(muT("#{data} per #{unit}")))
+ % muT("#{data}") * i18n(szSourceDesc[m_nSource])
+ % muT("#{unit}") * i18n(szSourceUnit[m_nAbsTime]));
+ }
+
+ writeRowspanTD(tos, getCustomTitle(i18n(szSourceDesc[m_nSource]), strTitle) + muT("<div style=\"width: 100px;\"></div>"), 1, 2, rowSpan, 2);
+ }
+ else if (row == 2)
+ {
+ writeRowspanTD(tos, i18n(muT("Outgoing")), 2, 2, rowSpan);
+ writeRowspanTD(tos, i18n(muT("Incoming")), 2, 2, rowSpan);
+ }
+}
+
+void ColInOutGraph::impl_outputRenderRow(ext::ostream& tos, const Contact& contact, DisplayType display)
+{
+ // fetch absolute values
+ static int (Contact::*getOut[3])() const = { &Contact::getOutBytes, &Contact::getOutMessages, &Contact::getOutChats };
+ static int (Contact::*getIn[3])() const = { &Contact::getInBytes, &Contact::getInMessages, &Contact::getInChats };
+
+ int numOut = (contact.*getOut[m_nSource])();
+ int numIn = (contact.*getIn[m_nSource])();
+ int numTotal = numOut + numIn;
+
+ // fetch average values
+ static double (Contact::*getAvgOut[3])() const = { &Contact::getOutBytesAvg, &Contact::getOutMessagesAvg, &Contact::getOutChatsAvg };
+ static double (Contact::*getAvgIn[3])() const = { &Contact::getInBytesAvg, &Contact::getInMessagesAvg, &Contact::getInChatsAvg };
+ static double (Contact::*getAvgTotal[3])() const = { &Contact::getTotalBytesAvg, &Contact::getTotalMessagesAvg, &Contact::getTotalChatsAvg };
+
+ static const double avgFactor[] = {
+ 60.0 * 60.0 * 24.0,
+ 60.0 * 60.0 * 24.0 * 7.0,
+ 60.0 * 60.0 * 24.0 * 30.0,
+ };
+
+ double avgOut = avgFactor[m_nAbsTime] * (contact.*getAvgOut[m_nSource])();
+ double avgIn = avgFactor[m_nAbsTime] * (contact.*getAvgIn[m_nSource])();
+ double avgTotal = avgFactor[m_nAbsTime] * (contact.*getAvgTotal[m_nSource])();
+
+ // begin output
+ tos << muT("<td colspan=\"2\" class=\"") << m_CSS;
+
+ if (m_bDetail)
+ {
+ ext::string strOut, strIn;
+
+ if ((m_bAbsolute && !m_bDetailInvert) || (!m_bAbsolute && m_bDetailInvert))
+ {
+ strOut = utils::intToGrouped(numOut);
+ strIn = utils::intToGrouped(numIn);
+ }
+ else
+ {
+ strOut = utils::floatToGrouped(avgOut, 1);
+ strIn = utils::floatToGrouped(avgIn, 1);
+ }
+
+ if (m_bDetailPercent)
+ {
+ tos << muT("\" title=\"")
+ << utils::htmlEscape(ext::str(ext::kformat(i18n(muT("[Out] #{out_amount} (#{out_ratio}) / [In] #{in_amount} (#{in_ratio})")))
+ % muT("#{out_amount}") * strOut
+ % muT("#{out_ratio}") * utils::ratioToPercent(numOut, numTotal)
+ % muT("#{in_amount}") * strIn
+ % muT("#{in_ratio}") * utils::ratioToPercent(numIn, numTotal)))
+ << muT("\">");
+ }
+ else
+ {
+ tos << muT("\" title=\"")
+ << utils::htmlEscape(ext::str(ext::kformat(i18n(muT("[Out] #{out_amount} / [In] #{in_amount}")))
+ % muT("#{out_amount}") * strOut
+ % muT("#{in_amount}") * strIn))
+ << muT("\">");
+ }
+ }
+ else
+ {
+ tos << muT("\">");
+ }
+
+ if (numOut + numIn == 0)
+ {
+ numOut = numIn = 1;
+ }
+
+ int allNum = numIn + numOut;
+ int outW = (int) (99.0 * numOut / allNum);
+ int inW = 99 - outW;
+
+ if (outW == 99)
+ {
+ outW++;
+ }
+
+ if (inW == 99)
+ {
+ inW++;
+ }
+
+ if (usePNG())
+ {
+ tos << muT("<div class=\"n\">");
+
+ if (m_bShowSum)
+ {
+ tos << (m_bAbsolute ? utils::intToGrouped(numTotal) : utils::floatToGrouped(avgTotal, 1))
+ << muT("<br/>");
+ }
+
+ // draw graph
+ Canvas canvas(100, 15);
+
+ canvas.fillBackground(con::ColorBack);
+
+ HDC hDC = canvas.beginDraw();
+
+ SetBkColor(hDC, con::ColorOut);
+ ExtTextOut(hDC, 0, 0, ETO_OPAQUE, &utils::rect(0, 0, outW, 15), NULL, 0, NULL);
+
+ SetBkColor(hDC, con::ColorIn);
+ ExtTextOut(hDC, 0, 0, ETO_OPAQUE, &utils::rect(100 - inW, 0, 100, 15), NULL, 0, NULL);
+
+ if (m_bGraphPercent)
+ {
+ SetBkMode(hDC, TRANSPARENT);
+ SetTextColor(hDC, con::ColorBack);
+
+ RECT r = { 2, 0, 98, 14 };
+
+ LOGFONT lf = {
+ -MulDiv(8, GetDeviceCaps(hDC, LOGPIXELSY), 72),
+ 0,
+ 0,
+ 0,
+ FW_NORMAL,
+ FALSE,
+ FALSE,
+ FALSE,
+ ANSI_CHARSET,
+ OUT_DEFAULT_PRECIS,
+ CLIP_DEFAULT_PRECIS,
+ DEFAULT_QUALITY,
+ DEFAULT_PITCH | FF_SWISS,
+ muT("Verdana")
+ };
+
+ HFONT hFont = CreateFontIndirect(&lf);
+ HGDIOBJ hOldFont = SelectObject(hDC, hFont);
+
+ DrawText(hDC, utils::ratioToPercent(numOut, allNum).c_str(), -1, &r, DT_LEFT | DT_VCENTER | DT_NOPREFIX | DT_SINGLELINE);
+ DrawText(hDC, utils::ratioToPercent(numIn, allNum).c_str(), -1, &r, DT_RIGHT | DT_VCENTER | DT_NOPREFIX | DT_SINGLELINE);
+
+ SelectObject(hDC, hOldFont);
+ }
+
+ canvas.endDraw();
+
+ // write PNG file
+ ext::string strFinalFile;
+
+ if (getStatistic()->newFilePNG(canvas, strFinalFile))
+ {
+ tos << muT("<img src=\"") << strFinalFile << muT("\"/>");
+ }
+
+ tos << muT("</div>");
+ }
+ else
+ {
+ if (m_bShowSum)
+ {
+ tos << muT("<div class=\"n\">")
+ << (m_bAbsolute ? utils::intToGrouped(numTotal) : utils::floatToGrouped(avgTotal, 1))
+ << muT("</div>");
+ }
+
+ tos << muT("<div class=\"") << m_CSS << muT("\">");
+
+ if (outW != 0)
+ {
+ tos << muT("<div class=\"obar\" style=\"width: ") << outW << muT("px;\"></div>");
+ }
+
+ if (inW != 0)
+ {
+ tos << muT("<div class=\"ibar\" style=\"width: ") << inW << muT("px;\"></div>");
+ }
+
+ if (m_bGraphPercent)
+ {
+ tos << muT("<div class=\"otext\">") << utils::ratioToPercent(numOut, allNum) << muT("</div>");
+ tos << muT("<div class=\"itext\">") << utils::ratioToPercent(numIn, allNum) << muT("</div>");
+ }
+
+ tos << muT("</div>");
+ }
+
+ tos << muT("</td>") << ext::endl;
+}