////////////////////////////////////////////////////////////////////////////////
//
// Copyright 2016 - 2021, Thomas Lauf, Paul Beckingham, Federico Hernandez.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
//
// https://www.opensource.org/licenses/mit-license.php
//
////////////////////////////////////////////////////////////////////////////////
#include
#include
#include
#include
#include
#include
#include
// Implemented in CmdChart.cpp.
std::map createHolidayMap (Rules&, Interval&);
std::string renderHolidays (const std::map &);
////////////////////////////////////////////////////////////////////////////////
int CmdSummary (
const CLI& cli,
Rules& rules,
Database& database)
{
const bool verbose = rules.getBoolean ("verbose");
// Create a filter, and if empty, choose 'today'.
auto filter = cli.getFilter (Range { Datetime ("today"), Datetime ("tomorrow") });
// Load the data.
auto tracked = getTracked (database, rules, filter);
if (tracked.empty ())
{
if (verbose)
{
std::cout << "No filtered data found";
if (filter.is_started ())
{
std::cout << " in the range " << filter.start.toISOLocalExtended ();
if (filter.is_ended ())
std::cout << " - " << filter.end.toISOLocalExtended ();
}
if (! filter.tags ().empty ())
{
std::cout << " tagged with " << joinQuotedIfNeeded (", ", filter.tags ());
}
std::cout << ".\n";
}
return 0;
}
// Map tags to colors.
auto palette = createPalette (rules);
auto tag_colors = createTagColorMap (rules, palette, tracked);
Color colorID (rules.getBoolean ("color") ? rules.get ("theme.colors.ids") : "");
auto ids = findHint (cli, ":ids");
auto show_annotation = findHint (cli, ":annotations");
Table table;
table.width (1024);
table.colorHeader (Color ("underline"));
table.add ("Wk");
table.add ("Date");
table.add ("Day");
if (ids)
{
table.add ("ID");
}
table.add ("Tags");
auto offset = 0;
if (show_annotation)
{
table.add ("Annotation");
offset = 1;
}
table.add ("Start", false);
table.add ("End", false);
table.add ("Time", false);
table.add ("Total", false);
// Each day is rendered separately.
time_t grand_total = 0;
Datetime previous;
auto days_start = filter.is_started() ? filter.start : tracked.front ().start;
auto days_end = filter.is_ended() ? filter.end : tracked.back ().end;
if (days_end == 0)
{
days_end = Datetime ();
}
for (Datetime day = days_start.startOfDay (); day < days_end; ++day)
{
auto day_range = getFullDay (day);
time_t daily_total = 0;
int row = -1;
for (auto& track : subset (day_range, tracked))
{
// Make sure the track only represents one day.
if ((track.is_open () && day > Datetime ()))
continue;
row = table.addRow ();
if (day != previous)
{
table.set (row, 0, format ("W{1}", day.week ()));
table.set (row, 1, day.toString ("Y-M-D"));
table.set (row, 2, day.dayNameShort (day.dayOfWeek ()));
previous = day;
}
// Intersect track with day.
auto today = day_range.intersect (track);
if (track.is_open () && day <= Datetime () && today.end > Datetime ())
today.end = Datetime ();
std::string tags = join(", ", track.tags());
if (ids)
{
table.set (row, 3, format ("@{1}", track.id), colorID);
}
table.set (row, (ids ? 4 : 3), tags);
if (show_annotation)
{
auto annotation = track.getAnnotation ();
if (annotation.length () > 15)
annotation = annotation.substr (0, 12) + "...";
table.set (row, (ids ? 5 : 4), annotation);
}
table.set (row, (ids ? 5 : 4) + offset, today.start.toString ("h:N:S"));
table.set (row, (ids ? 6 : 5) + offset, (track.is_open () ? "-" : today.end.toString ("h:N:S")));
table.set (row, (ids ? 7 : 6) + offset, Duration (today.total ()).formatHours ());
daily_total += today.total ();
}
if (row != -1)
table.set (row, (ids ? 8 : 7) + offset, Duration (daily_total).formatHours ());
grand_total += daily_total;
}
// Add the total.
table.set (table.addRow (), (ids ? 8 : 7) + offset, " ", Color ("underline"));
table.set (table.addRow (), (ids ? 8 : 7) + offset, Duration (grand_total).formatHours ());
const auto with_holidays = rules.getBoolean ("reports.summary.holidays");
std::cout << '\n'
<< table.render ()
<< (with_holidays ? renderHolidays (createHolidayMap (rules, filter)) : "")
<< '\n';
return 0;
}
////////////////////////////////////////////////////////////////////////////////
std::string renderHolidays (const std::map &holidays)
{
std::stringstream out;
for (auto &entry : holidays)
{
out << entry.first.toString ("Y-M-D")
<< " "
<< entry.second
<< '\n';
}
return out.str ();
}
////////////////////////////////////////////////////////////////////////////////