Ref T310, added model statistics dialog

* UI for statistics
* renamed to "coverageSummaryForModel"
* model list "htmlStatistics"
This commit is contained in:
Klaus Basan
2018-08-26 19:36:17 +02:00
parent 35b70db67d
commit 54802e9ea9
11 changed files with 278 additions and 28 deletions

View File

@@ -47,6 +47,30 @@ namespace BlackMisc
return this->container().findFirstByOrDefault(&OBJ::getDbKey, key, notFound);
}
template<class OBJ, class CONTAINER, typename KEYTYPE>
CONTAINER IDatastoreObjectList<OBJ, CONTAINER, KEYTYPE>::findObjectsWithDbKey() const
{
CONTAINER objects;
for (const OBJ &obj : ITimestampObjectList<OBJ, CONTAINER>::container())
{
if (!obj.hasValidDbKey()) { continue; }
objects.push_back(obj);
}
return objects;
}
template<class OBJ, class CONTAINER, typename KEYTYPE>
CONTAINER IDatastoreObjectList<OBJ, CONTAINER, KEYTYPE>::findObjectsWithoutDbKey() const
{
CONTAINER objects;
for (const OBJ &obj : ITimestampObjectList<OBJ, CONTAINER>::container())
{
if (obj.hasValidDbKey()) { continue; }
objects.push_back(obj);
}
return objects;
}
template <class OBJ, class CONTAINER, typename KEYTYPE>
OBJ IDatastoreObjectList<OBJ, CONTAINER, KEYTYPE>::maxKeyObject() const
{
@@ -139,8 +163,8 @@ namespace BlackMisc
if (keys.contains(obj.getDbKey())) { continue; }
newValues.push_back(obj);
}
int delta = this->container().size() - newValues.size();
this->container() = newValues;
const int delta = this->container().size() - newValues.size();
if (delta > 0) { this->container() = newValues; }
return delta;
}
@@ -197,6 +221,16 @@ namespace BlackMisc
return count;
}
template<class OBJ, class CONTAINER, typename KEYTYPE>
bool IDatastoreObjectList<OBJ, CONTAINER, KEYTYPE>::containsAnyObjectWithoutKey() const
{
for (const OBJ &obj : ITimestampObjectList<OBJ, CONTAINER>::container())
{
if (!obj.hasValidDbKey()) { return true; }
}
return false;
}
template<class OBJ, class CONTAINER, typename KEYTYPE>
CONTAINER IDatastoreObjectList<OBJ, CONTAINER, KEYTYPE>::fromMultipleJsonFormats(const QJsonObject &jsonObject)
{

View File

@@ -30,6 +30,12 @@ namespace BlackMisc
//! Object with key, notFound otherwise
OBJ findByKey(KEYTYPE key, const OBJ &notFound = OBJ()) const;
//! Objects with DB key
CONTAINER findObjectsWithDbKey() const;
//! Objects without DB key
CONTAINER findObjectsWithoutDbKey() const;
//! Object with max.key
OBJ maxKeyObject() const;
@@ -66,6 +72,9 @@ namespace BlackMisc
//! Number of entries with valid DB key
int countWithValidDbKey() const;
//! Any object without key?
bool containsAnyObjectWithoutKey() const;
//! From multiple JSON formats
//! \remark supports native swift C++ format, DB format, and cache format
static CONTAINER fromMultipleJsonFormats(const QJsonObject &jsonObject);

View File

@@ -13,6 +13,7 @@
#include "blackmisc/aviation/airlineicaocode.h"
#include "blackmisc/aviation/callsign.h"
#include "blackmisc/aviation/livery.h"
#include "blackmisc/math/mathutils.h"
#include "blackmisc/compare.h"
#include "blackmisc/iterator.h"
#include "blackmisc/range.h"
@@ -26,6 +27,7 @@
#include <tuple>
using namespace BlackMisc::Network;
using namespace BlackMisc::Math;
using namespace BlackMisc::Aviation;
using namespace BlackMisc::PhysicalQuantities;
@@ -504,20 +506,18 @@ namespace BlackMisc
int CAircraftModelList::removeIfNotMatchingSimulator(const CSimulatorInfo &needToMatch)
{
if (this->isEmpty()) { return 0; }
int c = 0;
for (auto it = this->begin(); it != this->end();)
const int oldSize = this->size();
CAircraftModelList models;
for (const CAircraftModel &model : *this)
{
if (it->matchesSimulator(needToMatch))
if (model.matchesSimulator(needToMatch))
{
++it;
}
else
{
c++;
it = this->erase(it);
models.push_back(model);
}
}
return c;
const int diff = models.size() - oldSize;
if (diff > 0) { *this = models; }
return diff;
}
int CAircraftModelList::removeAllWithoutModelString()
@@ -848,7 +848,7 @@ namespace BlackMisc
// normally prefer colors if there is no airline
CMatchingUtils::addLogDetailsToList(log, remoteModel.getCallsign(), QString("Prefer color liveries: '%1', airline: '%2', ignore zero scores: '%3'").arg(boolToYesNo(preferColorLiveries), remoteModel.getAirlineIcaoCodeDesignator(), boolToYesNo(ignoreZeroScores)));
CMatchingUtils::addLogDetailsToList(log, remoteModel.getCallsign(), QString("--- Start scoring in list with %1 models").arg(this->size()));
CMatchingUtils::addLogDetailsToList(log, remoteModel.getCallsign(), this->extCoverageSummary(remoteModel));
CMatchingUtils::addLogDetailsToList(log, remoteModel.getCallsign(), this->coverageSummaryForModel(remoteModel));
int c = 1;
for (const CAircraftModel &model : *this)
@@ -1054,17 +1054,23 @@ namespace BlackMisc
QString html;
for (const CAircraftModel &model : *this)
{
if (!html.isEmpty()) { html += "<br>"; }
html += model.asHtmlSummary(" ");
html += html.isEmpty() ?
model.asHtmlSummary(" ") :
QStringLiteral("<br>") % model.asHtmlSummary(" ");
}
return html;
}
QString CAircraftModelList::coverageSummary(const QString &separator) const
{
if (this->isEmpty()) { return "no models"; } // avoid division by 0
const int dbEntries = this->countWithValidDbKey();
const double dbRatio = CMathUtils::round(static_cast<double>(100 * dbEntries) / this->size(), 1);
return
QStringLiteral("Entries: ") % QString::number(this->size()) %
QStringLiteral(" | valid DB keys: ") % QString::number(this->countWithValidDbKey()) % separator %
QStringLiteral(" | valid DB keys: ") % QString::number(dbEntries) %
QStringLiteral(" (") % QString::number(dbRatio) % QStringLiteral("%)") % separator %
QStringLiteral("color liveries: ") % QString::number(this->countModelsWithColorLivery()) %
QStringLiteral(" | airline liveries: ") % QString::number(this->countModelsWithAirlineLivery()) % separator %
QStringLiteral("VTOL: ") % QString::number(this->countVtolAircraft()) %
@@ -1075,7 +1081,7 @@ namespace BlackMisc
QStringLiteral("Simulators: ") % this->countPerSimulator().toQString();
}
QString CAircraftModelList::extCoverageSummary(const CAircraftModel &checkModel, const QString &separator) const
QString CAircraftModelList::coverageSummaryForModel(const CAircraftModel &checkModel, const QString &separator) const
{
const bool combinedCodeForModel = this->containsCombinedType(checkModel.getAircraftIcaoCode().getCombinedType());
const bool airlineForModel = checkModel.hasAirlineDesignator() && this->containsAirlineLivery(checkModel.getAirlineIcaoCode());
@@ -1087,5 +1093,18 @@ namespace BlackMisc
QStringLiteral("")
);
}
QString CAircraftModelList::htmlStatistics() const
{
const bool notOnlyDb = this->containsAnyObjectWithoutKey();
QString stats = this->coverageSummary("<br>");
if (notOnlyDb)
{
const CAircraftModelList dbModels = this->findObjectsWithDbKey();
stats += QStringLiteral("<br><br>DB objects:<br>---------<br>") %
dbModels.coverageSummary("<br>");
}
return stats;
}
} // namespace
} // namespace

View File

@@ -352,7 +352,10 @@ namespace BlackMisc
//! What kind of models are represented here?
//! \remark checking for some criteria in the given model
QString extCoverageSummary(const CAircraftModel &checkModel, const QString &separator = "\n") const;
QString coverageSummaryForModel(const CAircraftModel &checkModel, const QString &separator = "\n") const;
//! A HTML summary of the data in the list
QString htmlStatistics() const;
};
//! Model per callsign