/***************************************************************************
* This file is part of the Lime Report project *
* Copyright (C) 2021 by Alexander Arin *
* arin_a@bk.ru *
* *
** GNU General Public License Usage **
* *
* This library 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. *
* You should have received a copy of the GNU General Public License *
* along with this program. If not, see . *
* *
** GNU Lesser General Public License **
* *
* This library is free software: you can redistribute it and/or modify *
* it under the terms of the GNU Lesser General Public License as *
* published by the Free Software Foundation, either version 3 of the *
* License, or (at your option) any later version. *
* You should have received a copy of the GNU Lesser General Public *
* License along with this library. *
* If not, see . *
* *
* This library 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. *
****************************************************************************/
#include "lrdatasourcemanager.h"
#include "lrdatadesignintf.h"
#include
#include
#if (QT_VERSION < QT_VERSION_CHECK(5, 15, 1))
#include
#endif
#include
#include
#include
#include
#ifdef BUILD_WITH_EASY_PROFILER
#include "easy/profiler.h"
#else
# define EASY_BLOCK(...)
# define EASY_END_BLOCK
#endif
namespace LimeReport{
DataNode::~DataNode()
{
for (int i = 0; im_childs.indexOf(const_cast(this));
}
return 0;
}
void DataNode::clear()
{
for (int i=0; ichild(row);
if (childNode){
return createIndex(row,column,childNode);
} else return QModelIndex();
}
QModelIndex DataSourceModel::parent(const QModelIndex& child) const
{
if (!child.isValid()) return QModelIndex();
DataNode* childNode = nodeFromIndex(child);
if (!childNode) return QModelIndex();
DataNode* parentNode = childNode->parent();
if ((parentNode == m_rootNode) || (!parentNode)) return QModelIndex();
return createIndex(parentNode->row(),0,parentNode);
}
int DataSourceModel::rowCount(const QModelIndex& parent) const
{
if (!m_rootNode) return 0;
DataNode* parentNode;
if (parent.isValid())
parentNode = nodeFromIndex(parent);
else
parentNode = m_rootNode;
return parentNode->childCount();
}
int DataSourceModel::columnCount(const QModelIndex& parent) const
{
Q_UNUSED(parent)
return 1;
}
QVariant DataSourceModel::data(const QModelIndex& index, int role) const
{
DataNode *node = nodeFromIndex(index);
switch (role) {
case Qt::DisplayRole:
if (!node) return QVariant();
return node->name();
break;
case Qt::DecorationRole :
if (!node) return QIcon();
return node->icon();
break;
default:
return QVariant();
}
}
void DataSourceModel::setDataSourceManager(DataSourceManager* dataManager)
{
m_dataManager = dataManager;
connect(m_dataManager,SIGNAL(datasourcesChanged()),this,SLOT(slotDatasourcesChanged()));
updateModel();
}
void DataSourceModel::slotDatasourcesChanged()
{
beginResetModel();
updateModel();
endResetModel();
}
DataNode* DataSourceModel::nodeFromIndex(const QModelIndex& index) const
{
if (index.isValid()){
return static_cast(index.internalPointer());
} else return m_rootNode;
}
void DataSourceModel::fillFields(DataNode* parent)
{
foreach(QString name, m_dataManager->fieldNames(parent->name())){
parent->addChild(name, DataNode::Field,QIcon(":/report/images/field"));
}
}
void DataSourceModel::updateModel()
{
QMap connections;
m_rootNode->clear();
DataNode* ds = m_rootNode->addChild(tr("Datasources"),DataNode::DataSources,QIcon(":/report/images/databases"));
foreach (QString name,m_dataManager->connectionNames()) {
DataNode* connection = ds->addChild(name,DataNode::Connection,QIcon(":/report/images/database"));
connections.insert(name,connection);
}
foreach (QString name, m_dataManager->dataSourceNames()) {
DataNode* datasource;
if (m_dataManager->isQuery(name)){
DataNode* connection = connections.value(m_dataManager->queryByName(name)->connectionName());
if (connection) datasource=connection->addChild(name,DataNode::Query,QIcon(":/report/images/table"));
else datasource=ds->addChild(name,DataNode::Query);
} else if (m_dataManager->isSubQuery(name)){
DataNode* connection = connections.value(m_dataManager->subQueryByName(name)->connectionName());
if (connection) datasource=connection->addChild(name,DataNode::SubQuery,QIcon(":/report/images/table"));
else datasource=ds->addChild(name,DataNode::SubQuery);
} else {
datasource=ds->addChild(name,DataNode::Model,QIcon(":/report/images/table"));
}
fillFields(datasource);
}
DataNode* vars = m_rootNode->addChild(tr("Variables"),DataNode::Variables,QIcon(":/report/images/folder"));
foreach (QString name, m_dataManager->variableNames()){
vars->addChild(name,DataNode::Variable,QIcon(":/report/images/value"));
}
vars = m_rootNode->addChild(tr("External variables"),DataNode::Variables,QIcon(":/report/images/folder"));
foreach (QString name, m_dataManager->userVariableNames()){
vars->addChild(name,DataNode::Variable,QIcon(":/report/images/value"));
}
}
DataSourceManager::DataSourceManager(QObject *parent) :
QObject(parent), m_lastError(""), m_designTime(false), m_needUpdate(false),
m_dbCredentialsProvider(0), m_hasChanges(false)
{
m_groupFunctionFactory.registerFunctionCreator(QLatin1String("COUNT"),new ConstructorGroupFunctionCreator);
m_groupFunctionFactory.registerFunctionCreator(QLatin1String("SUM"),new ConstructorGroupFunctionCreator);
m_groupFunctionFactory.registerFunctionCreator(QLatin1String("AVG"),new ConstructorGroupFunctionCreator);
m_groupFunctionFactory.registerFunctionCreator(QLatin1String("MIN"),new ConstructorGroupFunctionCreator);
m_groupFunctionFactory.registerFunctionCreator(QLatin1String("MAX"),new ConstructorGroupFunctionCreator);
setSystemVariable(QLatin1String("#PAGE"),1,SecondPass);
setSystemVariable(QLatin1String("#PAGE_COUNT"),0,SecondPass);
setSystemVariable(QLatin1String("#IS_LAST_PAGEFOOTER"),false,FirstPass);
setSystemVariable(QLatin1String("#IS_FIRST_PAGEFOOTER"),false,FirstPass);
m_datasourcesModel.setDataSourceManager(this);
connect(&m_reportVariables, SIGNAL(variableHasBeenAdded(QString)),
this, SLOT(slotVariableHasBeenAdded(QString)));
connect(&m_reportVariables, SIGNAL(variableHasBeenChanged(QString)),
this, SLOT(slotVariableHasBeenChanged(QString)));
connect(&m_userVariables, SIGNAL(variableHasBeenAdded(QString)),
this, SLOT(slotVariableHasBeenAdded(QString)));
connect(&m_userVariables, SIGNAL(variableHasBeenChanged(QString)),
this, SLOT(slotVariableHasBeenChanged(QString)));
}
QString DataSourceManager::defaultDatabasePath() const
{
return m_defaultDatabasePath;
}
void DataSourceManager::setDefaultDatabasePath(const QString &defaultDatabasePath)
{
m_defaultDatabasePath = defaultDatabasePath;
}
QString DataSourceManager::putGroupFunctionsExpressions(QString expression)
{
if (m_groupFunctionsExpressionsMap.contains(expression)){
return QString::number(m_groupFunctionsExpressionsMap.value(expression));
} else {
m_groupFunctionsExpressions.append(expression);
m_groupFunctionsExpressionsMap.insert(expression, m_groupFunctionsExpressions.size()-1);
return QString::number(m_groupFunctionsExpressions.size()-1);
}
}
void DataSourceManager::clearGroupFuntionsExpressions()
{
m_groupFunctionsExpressionsMap.clear();
m_groupFunctionsExpressions.clear();
}
QString DataSourceManager::getExpression(QString index)
{
bool ok = false;
int i = index.toInt(&ok);
if (ok && m_groupFunctionsExpressions.size()>i)
return m_groupFunctionsExpressions.at(index.toInt());
else return "";
}
bool DataSourceManager::designTime() const
{
return m_designTime;
}
void DataSourceManager::setDesignTime(bool designTime)
{
m_designTime = designTime;
}
DataSourceManager::~DataSourceManager()
{
clear(All);
clearGroupFunction();
}
void DataSourceManager::connectAllDatabases()
{
foreach(ConnectionDesc* conn,m_connections){
try{
connectConnection(conn);
} catch (ReportError &e){
putError(e.what());
setLastError(e.what());
qDebug()<(ds),true);
putHolder(name,holder);
emit datasourcesChanged();
m_needUpdate = true;
return ds;
}
void DataSourceManager::registerDbCredentialsProvider(IDbCredentialsProvider *provider)
{
m_dbCredentialsProvider = provider;
}
void DataSourceManager::addCallbackDatasource(ICallbackDatasource *datasource, const QString& name)
{
IDataSource* datasourceIntf = dynamic_cast(datasource);
if (datasourceIntf){
IDataSourceHolder* holder = new CallbackDatasourceHolder(datasourceIntf,true);
putHolder(name,holder);
emit datasourcesChanged();
}
}
QSharedPointerDataSourceManager::previewSQL(const QString &connectionName, const QString &sqlText, QString masterDatasource)
{
QSqlDatabase db = QSqlDatabase::database(connectionName);
if (db.isValid() && db.isOpen()){
QSqlQueryModel* model = new QSqlQueryModel();
QMap aliasesToParam;
QString queryText = replaceVariables(sqlText,aliasesToParam);
queryText = replaceFields(queryText,aliasesToParam,masterDatasource);
QSqlQuery query(db);
query.prepare(queryText);
foreach(QString param,aliasesToParam.keys()){
QVariant value;
if (param.contains(".")){
value = fieldData(aliasesToParam.value(param));
param=param.right(param.length()-param.indexOf('.')-1);
} else {
value = variable(aliasesToParam.value(param));
}
if (value.isValid() || m_designTime)
query.bindValue(':'+param,value);
}
query.exec();
#if QT_VERSION >= QT_VERSION_CHECK(6, 2, 0)
model->setQuery(std::move(query));
#else
model->setQuery(query);
#endif
m_lastError = model->lastError().text();
putError(m_lastError);
if (model->query().isActive())
return QSharedPointer(model);
else
return QSharedPointer(0);
}
if (!db.isOpen()){
m_lastError = tr("Connection \"%1\" is not open").arg(connectionName);
putError(m_lastError);
}
return QSharedPointer(0);
}
void DataSourceManager::updateDatasourceModel()
{
m_datasourcesModel.updateModel();
emit datasourcesChanged();
m_needUpdate = false;
}
QString DataSourceManager::extractField(QString source)
{
if (source.contains('.')) {
return source.right(source.length()-(source.indexOf('.')+1));
}
return source;
}
QString DataSourceManager::replaceVariables(QString value){
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 1)
QRegularExpression rx = getVariableRegEx();
QRegularExpressionMatchIterator iter = rx.globalMatch(value);
qsizetype pos = 0;
QString result;
while (iter.hasNext()) {
QRegularExpressionMatch match = iter.next();
QString var = match.captured(0);
var.remove("$V{");
var.remove("}");
result += value.mid(pos, match.capturedStart(0));
if (variable(var).isValid()){
result += variable(var).toString();
} else {
result += QString(tr("Variable \"%1\" not found!").arg(var));
}
pos = match.capturedEnd(0);
}
result += value.mid(pos);
return result;
#else
QRegExp rx(Const::VARIABLE_RX);
if (value.contains(rx)){
int pos = -1;
while ((pos=rx.indexIn(value))!=-1){
QString var=rx.cap(0);
var.remove("$V{");
var.remove("}");
if (variable(var).isValid()){
value.replace(pos,rx.cap(0).length(),variable(var).toString());
} else {
value.replace(pos,rx.cap(0).length(),QString(tr("Variable \"%1\" not found!").arg(var)));
}
}
}
return value;
#endif
return QString();
}
QString DataSourceManager::replaceVariables(QString query, QMap &aliasesToParam)
{
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 1)
QRegularExpression rx = getVariableRegEx();
int curentAliasIndex = 0;
if (query.contains(rx)){
int pos = -1;
QRegularExpressionMatch match = rx.match(query);
while ((pos=match.capturedStart())!=-1){
QString var=match.captured(0);
var.remove("$V{");
var.remove("}");
if (!match.captured(1).isEmpty()){
if (aliasesToParam.contains(var)){
curentAliasIndex++;
aliasesToParam.insert(var+"_v_alias"+QString::number(curentAliasIndex),var);
var += "_v_alias"+QString::number(curentAliasIndex);
} else {
aliasesToParam.insert(var,var);
}
query.replace(pos,match.captured(0).length(),":"+var);
} else {
QString varName = match.captured(2).trimmed();
QString varParam = match.captured(3).trimmed();
if (!varName.isEmpty()){
if (!varParam.isEmpty() && varParam.compare("nobind") == 0 ){
query.replace(pos,match.captured(0).length(), variable(varName).toString());
} else {
query.replace(pos,match.captured(0).length(),
QString(tr("Unknown parameter \"%1\" for variable \"%2\" found!")
.arg(varName)
.arg(varParam))
);
}
} else {
query.replace(pos,match.captured(0).length(),QString(tr("Variable \"%1\" not found!").arg(var)));
}
}
match = rx.match(query);
}
}
#else
QRegExp rx(Const::VARIABLE_RX);
int curentAliasIndex = 0;
if (query.contains(rx)){
int pos = -1;
while ((pos=rx.indexIn(query))!=-1){
QString var=rx.cap(0);
var.remove("$V{");
var.remove("}");
if (!rx.cap(1).isEmpty()){
if (aliasesToParam.contains(var)){
curentAliasIndex++;
aliasesToParam.insert(var+"_v_alias"+QString::number(curentAliasIndex),var);
var += "_v_alias"+QString::number(curentAliasIndex);
} else {
aliasesToParam.insert(var,var);
}
query.replace(pos,rx.cap(0).length(),":"+var);
} else {
QString varName = rx.cap(2).trimmed();
QString varParam = rx.cap(3).trimmed();
if (!varName.isEmpty()){
if (!varParam.isEmpty() && varParam.compare("nobind") == 0 ){
query.replace(pos,rx.cap(0).length(), variable(varName).toString());
} else {
query.replace(pos,rx.cap(0).length(),
QString(tr("Unknown parameter \"%1\" for variable \"%2\" found!")
.arg(varName)
.arg(varParam))
);
}
} else {
query.replace(pos,rx.cap(0).length(),QString(tr("Variable \"%1\" not found!").arg(var)));
}
}
}
}
#endif
return query;
}
QString DataSourceManager::replaceFields(QString query, QMap &aliasesToParam, QString masterDatasource)
{
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 1)
QRegularExpression rx = getFieldRegEx();
int curentAliasIndex = 0;
if (query.contains(rx)){
int pos = -1;
QRegularExpressionMatch match = rx.match(query);
while ((pos=match.capturedStart())!=-1){
QString field=match.captured(0);
field.remove("$D{");
field.remove("}");
if (!aliasesToParam.contains(field)){
if (field.contains("."))
aliasesToParam.insert(field, field);
else
aliasesToParam.insert(field, masterDatasource+"."+field);
} else {
curentAliasIndex++;
if (field.contains("."))
aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex), field);
else
aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex), masterDatasource+"."+field);
field+="_f_alias"+QString::number(curentAliasIndex);
}
query.replace(pos,match.capturedLength(),":"+extractField(field));
match = rx.match(query);
}
}
#else
QRegExp rx(Const::FIELD_RX);
if (query.contains(rx)){
int curentAliasIndex=0;
int pos;
while ((pos=rx.indexIn(query))!=-1){
QString field=rx.cap(0);
field.remove("$D{");
field.remove("}");
if (!aliasesToParam.contains(field)){
if (field.contains("."))
aliasesToParam.insert(field,field);
else
aliasesToParam.insert(field,masterDatasource+"."+field);
} else {
curentAliasIndex++;
if (field.contains("."))
aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex),field);
else
aliasesToParam.insert(field+"_f_alias"+QString::number(curentAliasIndex),masterDatasource+"."+field);
field+="_f_alias"+QString::number(curentAliasIndex);
}
query.replace(pos,rx.cap(0).length(),":"+extractField(field));
}
}
return query;
#endif
return query;
}
void DataSourceManager::setReportVariable(const QString &name, const QVariant &value)
{
if (!containsVariable(name)){
addVariable(name,value);
} else changeVariable(name,value);
}
void DataSourceManager::addQuery(const QString &name, const QString &sqlText, const QString &connectionName)
{
QueryDesc *queryDecs = new QueryDesc(name,sqlText,connectionName);
putQueryDesc(queryDecs);
putHolder(name,new QueryHolder(sqlText, connectionName, this));
m_hasChanges = true;
m_varToDataSource.clear();
emit datasourcesChanged();
}
void DataSourceManager::addSubQuery(const QString &name, const QString &sqlText, const QString &connectionName, const QString &masterDatasource)
{
SubQueryDesc *subQueryDesc = new SubQueryDesc(name.toLower(),sqlText,connectionName,masterDatasource);
putSubQueryDesc(subQueryDesc);
putHolder(name,new SubQueryHolder(sqlText, connectionName, masterDatasource, this));
m_hasChanges = true;
m_varToDataSource.clear();
emit datasourcesChanged();
}
void DataSourceManager::addProxy(const QString &name, const QString &master, const QString &detail, QList fields)
{
ProxyDesc *proxyDesc = new ProxyDesc();
proxyDesc->setName(name);
proxyDesc->setMaster(master);
proxyDesc->setDetail(detail);
foreach(FieldsCorrelation correlation,fields){
proxyDesc->addFieldsCorrelation(correlation);
}
putProxyDesc(proxyDesc);
putHolder(name,new ProxyHolder(proxyDesc, this));
m_hasChanges = true;
emit datasourcesChanged();
}
void DataSourceManager::addCSV(const QString& name, const QString& csvText, const QString &separator, bool firstRowIsHeader)
{
CSVDesc* csvDesc = new CSVDesc(name, csvText, separator, firstRowIsHeader);
putCSVDesc(csvDesc);
putHolder(name, new CSVHolder(*csvDesc, this));
m_hasChanges = true;
emit datasourcesChanged();
}
QString DataSourceManager::queryText(const QString &dataSourceName)
{
if (isQuery(dataSourceName)) return queryByName(dataSourceName)->queryText();
if (isSubQuery(dataSourceName)) return subQueryByName(dataSourceName)->queryText();
else return QString();
}
QueryDesc *DataSourceManager::queryByName(const QString &datasourceName)
{
int queryIndex = queryIndexByName(datasourceName);
if (queryIndex!=-1) return m_queries.at(queryIndex);
return 0;
}
SubQueryDesc* DataSourceManager::subQueryByName(const QString &datasourceName)
{
int queryIndex = subQueryIndexByName(datasourceName);
if (queryIndex!=-1) return m_subqueries.at(queryIndex);
return 0;
}
ConnectionDesc* DataSourceManager::connectionByName(const QString &connectionName)
{
int queryIndex = connectionIndexByName(connectionName);
if (queryIndex!=-1) return m_connections.at(queryIndex);
return 0;
}
int DataSourceManager::queryIndexByName(const QString &dataSourceName)
{
for(int i=0;iqueryName().compare(dataSourceName,Qt::CaseInsensitive)==0) return i;
}
return -1;
}
int DataSourceManager::subQueryIndexByName(const QString &dataSourceName)
{
for(int i=0;iqueryName().compare(dataSourceName,Qt::CaseInsensitive)==0) return i;
}
return -1;
}
int DataSourceManager::proxyIndexByName(const QString &dataSourceName)
{
for(int i=0;iname().compare(dataSourceName,Qt::CaseInsensitive)==0) return i;
}
return -1;
}
int DataSourceManager::csvIndexByName(const QString &dataSourceName)
{
for(int i=0; i < m_csvs.count();++i){
CSVDesc* desc=m_csvs.at(i);
if (desc->name().compare(dataSourceName,Qt::CaseInsensitive)==0) return i;
}
return -1;
}
int DataSourceManager::connectionIndexByName(const QString &connectionName)
{
for(int i=0;iname().compare(connectionName,Qt::CaseInsensitive)==0) return i;
}
return -1;
}
QList& DataSourceManager::conections()
{
return m_connections;
}
bool DataSourceManager::dataSourceIsValid(const QString &name)
{
if (m_datasources.value(name.toLower())) return !m_datasources.value(name.toLower())->isInvalid();
else throw ReportError(tr("Datasource \"%1\" not found!").arg(name));
}
bool DataSourceManager::isQuery(const QString &dataSourceName)
{
return (queryByName(dataSourceName));
}
QString DataSourceManager::connectionName(const QString &dataSourceName)
{
if (isQuery(dataSourceName)) return queryByName(dataSourceName)->connectionName();
if (isSubQuery(dataSourceName)) return subQueryByName(dataSourceName)->connectionName();
return QString();
}
ProxyDesc *DataSourceManager::proxyByName(const QString &datasourceName)
{
int proxyIndex = proxyIndexByName(datasourceName);
if (proxyIndex > -1) return m_proxies.at(proxyIndex);
else return 0;
}
CSVDesc *DataSourceManager::csvByName(const QString &datasourceName)
{
int csvIndex = csvIndexByName(datasourceName);
if (csvIndex > -1) return m_csvs.at(csvIndex);
else return 0;
}
void DataSourceManager::removeDatasource(const QString &name)
{
if (m_datasources.contains(name)){
IDataSourceHolder *holder;
holder=m_datasources.value(name);
m_datasources.remove(name);
delete holder;
}
if (isQuery(name)){
int queryIndex=queryIndexByName(name);
delete m_queries.at(queryIndex);
m_queries.removeAt(queryIndex);
}
if (isSubQuery(name)){
int queryIndex=subQueryIndexByName(name);
delete m_subqueries.at(queryIndex);
m_subqueries.removeAt(queryIndex);
}
if (isProxy(name)){
int proxyIndex=proxyIndexByName(name);
delete m_proxies.at(proxyIndex);
m_proxies.removeAt(proxyIndex);
}
if (isCSV(name)){
int csvIndex=csvIndexByName(name);
delete m_csvs.at(csvIndex);
m_csvs.removeAt(csvIndex);
}
invalidateLinkedDatasources(name);
m_hasChanges = true;
emit datasourcesChanged();
}
void DataSourceManager::removeConnection(const QString &connectionName)
{
QList::iterator cit = m_connections.begin();
while( cit != m_connections.end() ){
if ( ((*cit)->name().compare(connectionName) == 0) ){
if ((*cit)->isInternal())
{
QSqlDatabase db = QSqlDatabase::database(connectionName);
db.close();
}
QSqlDatabase::removeDatabase(connectionName);
delete (*cit);
cit = m_connections.erase(cit);
} else {
++cit;
}
}
m_hasChanges = true;
emit datasourcesChanged();
}
void DataSourceManager::addConnectionDesc(ConnectionDesc * connection)
{
if (!isConnection(connection->name())) {
connect(connection,SIGNAL(nameChanged(QString,QString)),this,SLOT(slotConnectionRenamed(QString,QString)));
m_connections.append(connection);
m_hasChanges = true;
if (connection->autoconnect()){
try{
connectConnection(connection);
} catch(ReportError &exception) {
qDebug()<name()));
}
}
bool DataSourceManager::checkConnectionDesc(ConnectionDesc *connection)
{
if (connectConnection(connection)){
if (connection->isInternal()){
QSqlDatabase::removeDatabase(connection->name());
if (designTime()) emit datasourcesChanged();
}
return true;
}
if (connection->isInternal())
QSqlDatabase::removeDatabase(connection->name());
return false;
}
void DataSourceManager::putHolder(const QString& name, IDataSourceHolder *dataSource)
{
if (!m_datasources.contains(name.toLower())){
m_datasources.insert(
name.toLower(),
dataSource
);
} else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(name));
}
void DataSourceManager::putQueryDesc(QueryDesc* queryDesc)
{
if (!containsDatasource(queryDesc->queryName())){
m_queries.append(queryDesc);
connect(queryDesc, SIGNAL(queryTextChanged(QString,QString)),
this, SLOT(slotQueryTextChanged(QString,QString)));
} else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(queryDesc->queryName()));
}
void DataSourceManager::putSubQueryDesc(SubQueryDesc *subQueryDesc)
{
if (!containsDatasource(subQueryDesc->queryName())){
m_subqueries.append(subQueryDesc);
connect(subQueryDesc, SIGNAL(queryTextChanged(QString,QString)),
this, SLOT(slotQueryTextChanged(QString,QString)));
} else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(subQueryDesc->queryName()));
}
void DataSourceManager::putProxyDesc(ProxyDesc *proxyDesc)
{
if (!containsDatasource(proxyDesc->name())){
m_proxies.append(proxyDesc);
} else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(proxyDesc->name()));
}
void DataSourceManager::putCSVDesc(CSVDesc *csvDesc)
{
if (!containsDatasource(csvDesc->name())){
m_csvs.append(csvDesc);
connect(csvDesc, SIGNAL(cvsTextChanged(QString, QString)),
this, SLOT(slotCSVTextChanged(QString, QString)));
} else throw ReportError(tr("Datasource with name \"%1\" already exists!").arg(csvDesc->name()));
}
bool DataSourceManager::initAndOpenDB(QSqlDatabase& db, ConnectionDesc& connectionDesc){
bool connected = false;
db.setHostName(replaceVariables(connectionDesc.host()));
db.setUserName(replaceVariables(connectionDesc.userName()));
db.setPassword(replaceVariables(connectionDesc.password()));
db.setDatabaseName(replaceVariables(connectionDesc.databaseName()));
if (connectionDesc.port()!="")
db.setPort(replaceVariables(connectionDesc.port()).toInt());
if (!connectionDesc.keepDBCredentials() && m_dbCredentialsProvider){
if (!m_dbCredentialsProvider->getUserName(connectionDesc.name()).isEmpty())
db.setUserName(m_dbCredentialsProvider->getUserName(connectionDesc.name()));
if (!m_dbCredentialsProvider->getPassword(connectionDesc.name()).isEmpty())
db.setPassword(m_dbCredentialsProvider->getPassword(connectionDesc.name()));
}
QString dbName = replaceVariables(connectionDesc.databaseName());
if (connectionDesc.driver().compare("QSQLITE")==0){
if (!defaultDatabasePath().isEmpty()){
dbName = !QFileInfo(dbName).exists() ?
defaultDatabasePath()+QFileInfo(dbName).fileName() :
dbName;
}
if (QFileInfo(dbName).exists()){
db.setDatabaseName(dbName);
} else {
setLastError(tr("Database \"%1\" not found").arg(dbName));
return false;
}
} else {
db.setDatabaseName(dbName);
}
connected=db.open();
if (!connected) setLastError(db.lastError().text());
return connected;
}
ReportSettings *DataSourceManager::reportSettings() const
{
return m_reportSettings;
}
void DataSourceManager::setReportSettings(ReportSettings *reportSettings)
{
m_reportSettings = reportSettings;
}
bool DataSourceManager::checkConnection(QSqlDatabase db){
QSqlQuery query("Select 1",db);
return query.first();
}
bool DataSourceManager::connectConnection(ConnectionDesc *connectionDesc)
{
bool connected = false;
clearErrors();
QString lastError ="";
foreach(QString datasourceName, dataSourceNames()){
dataSourceHolder(datasourceName)->clearErrors();
}
if (!QSqlDatabase::contains(connectionDesc->name())){
QString dbError;
{
QSqlDatabase db = QSqlDatabase::addDatabase(connectionDesc->driver(),connectionDesc->name());
connectionDesc->setInternal(true);
connected=initAndOpenDB(db, *connectionDesc);
dbError = db.lastError().text();
}
if (!connected){
if (!dbError.trimmed().isEmpty())
setLastError(dbError);
QSqlDatabase::removeDatabase(connectionDesc->name());
return false;
}
} else {
QSqlDatabase db = QSqlDatabase::database(connectionDesc->name());
if (!connectionDesc->isEqual(db) && connectionDesc->isInternal()){
db.close();
connected = initAndOpenDB(db, *connectionDesc);
} else {
connected = checkConnection(db);
if (!connected && connectionDesc->isInternal())
connected = initAndOpenDB(db, *connectionDesc);
}
}
if (!connected) {
if (connectionDesc->isInternal())
QSqlDatabase::removeDatabase(connectionDesc->name());
return false;
} else {
foreach(QString datasourceName, dataSourceNames()){
if (isQuery(datasourceName)){
QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName));
if (qh){
qh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
invalidateChildren(datasourceName);
}
}
}
foreach(QString datasourceName, dataSourceNames()){
if (isProxy(datasourceName)){
ProxyHolder* ph = dynamic_cast(dataSourceHolder(datasourceName));
if (ph){
ph->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
}
}
}
if (designTime()) emit datasourcesChanged();
}
return true;
}
void DataSourceManager::clearReportVariables()
{
m_reportVariables.clearUserVariables();
}
void DataSourceManager::connectAutoConnections()
{
foreach(ConnectionDesc* conn,m_connections){
if (conn->autoconnect()) {
try {
connectConnection(conn);
} catch(ReportError &e){
setLastError(e.what());
putError(e.what());
qDebug()< DataSourceManager::childDatasources(const QString &parentDatasourceName)
{
QList result;
foreach(QString datasourceName, dataSourceNames()){
if (isSubQuery(datasourceName)){
SubQueryHolder* sh = dynamic_cast(dataSourceHolder(datasourceName));
if (sh && sh->masterDatasource().compare(parentDatasourceName,Qt::CaseInsensitive)==0){
result.append(datasourceName);
}
}
}
return result;
}
void DataSourceManager::invalidateChildren(const QString &parentDatasourceName)
{
foreach(QString datasourceName, childDatasources(parentDatasourceName)){
SubQueryHolder* sh = dynamic_cast(dataSourceHolder(datasourceName));
if (sh)
sh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
invalidateChildren(datasourceName);
}
}
bool DataSourceManager::containsDatasource(const QString &dataSourceName)
{
return m_datasources.contains(dataSourceName.toLower());
}
bool DataSourceManager::isSubQuery(const QString &dataSourceName)
{
return subQueryIndexByName(dataSourceName) != -1;
}
bool DataSourceManager::isProxy(const QString &dataSourceName)
{
return proxyIndexByName(dataSourceName) != -1;
}
bool DataSourceManager::isCSV(const QString &datasourceName)
{
return csvIndexByName(datasourceName) != -1;
}
bool DataSourceManager::isConnection(const QString &connectionName)
{
return connectionIndexByName(connectionName) != -1;
}
bool DataSourceManager::isConnectionConnected(const QString &connectionName)
{
if (isConnection(connectionName) && QSqlDatabase::contains(connectionName)){
QSqlDatabase db = QSqlDatabase::database(connectionName);
return db.isValid() && QSqlDatabase::database(connectionName).isOpen();
}
return false;
}
bool DataSourceManager::connectConnection(const QString& connectionName)
{
return connectConnection(connectionByName(connectionName));
}
void DataSourceManager::disconnectConnection(const QString& connectionName)
{
foreach(QString datasourceName, dataSourceNames()){
if (isQuery(datasourceName) || isSubQuery(datasourceName)){
QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName));
if (qh && qh->connectionName().compare(connectionName, Qt::CaseInsensitive) == 0){
qh->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE, true);
qh->setLastError(tr("invalid connection"));
}
}
}
ConnectionDesc* connectionDesc = connectionByName(connectionName);
if (connectionDesc->isInternal()){
{
QSqlDatabase db = QSqlDatabase::database(connectionName);
if (db.isOpen()) db.close();
}
if (QSqlDatabase::contains(connectionName)) QSqlDatabase::removeDatabase(connectionName);
}
}
IDataSource *DataSourceManager::dataSource(const QString &name)
{
IDataSourceHolder* holder = m_datasources.value(name.toLower());
if (holder) {
if (holder->isInvalid()) {
setLastError(name+" : "+holder->lastError());
return 0;
} else {
return holder->dataSource(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
}
} else {
setLastError(tr("Datasource \"%1\" not found!").arg(name));
return 0;
}
}
IDataSourceHolder *DataSourceManager::dataSourceHolder(const QString &name)
{
if (m_datasources.value(name.toLower())) return m_datasources.value(name.toLower());
else return 0;
}
QStringList DataSourceManager::dataSourceNames()
{
QStringList result;
for (int i=0;iconnectionName().compare(connectionName,Qt::CaseInsensitive)==0){
result.append(query->queryName());
}
}
foreach (QueryDesc* query, m_subqueries) {
if (query->connectionName().compare(connectionName,Qt::CaseInsensitive)==0){
result.append(query->queryName());
}
}
return result;
}
QStringList DataSourceManager::connectionNames()
{
QStringList result;
foreach(ConnectionDesc* conDesc,m_connections){
result.append(conDesc->name());
}
return result;
}
QStringList DataSourceManager::fieldNames(const QString &datasourceName)
{
QStringList result;
IDataSource* ds = dataSource(datasourceName);
if (ds && !ds->isInvalid()){
for(int i=0; i < ds->columnCount(); i++){
result.append(ds->columnNameByIndex(i));
}
result.sort();
}
return result;
}
void DataSourceManager::addConnection(const QString &connectionName)
{
addConnectionDesc(new ConnectionDesc(QSqlDatabase::database(connectionName)));
m_hasChanges = true;
emit datasourcesChanged();
}
QObject *DataSourceManager::createElement(const QString& collectionName, const QString&)
{
if (collectionName=="connections"){
ConnectionDesc* connection = new ConnectionDesc;
m_connections.append(connection);
return connection;
}
if (collectionName=="queries"){
QueryDesc* queryDesc = new QueryDesc;
m_queries.append(queryDesc);
return queryDesc;
}
if (collectionName=="subqueries"){
SubQueryDesc* subQueryDesc = new SubQueryDesc;
m_subqueries.append(subQueryDesc);
return subQueryDesc;
}
if (collectionName=="subproxies"){
ProxyDesc* proxyDesc = new ProxyDesc;
m_proxies.append(proxyDesc);
return proxyDesc;
}
if (collectionName=="variables"){
VarDesc* var = new VarDesc;
m_tempVars.append(var);
return var;
}
if (collectionName=="csvs"){
CSVDesc* csvDesc = new CSVDesc;
m_csvs.append(csvDesc);
return csvDesc;
}
return 0;
}
int DataSourceManager::elementsCount(const QString &collectionName)
{
if (collectionName=="connections"){
return m_connections.count();
}
if (collectionName=="queries"){
return m_queries.count();
}
if (collectionName=="subqueries"){
return m_subqueries.count();
}
if (collectionName=="subproxies"){
return m_proxies.count();
}
if (collectionName=="variables"){
return m_reportVariables.variablesCount();
}
if (collectionName=="csvs"){
return m_csvs.count();
}
return 0;
}
QObject* DataSourceManager::elementAt(const QString &collectionName, int index)
{
if (collectionName=="connections"){
return m_connections.at(index);
}
if (collectionName=="queries"){
return m_queries.at(index);
}
if (collectionName=="subqueries"){
return m_subqueries.at(index);
}
if (collectionName=="subproxies"){
return m_proxies.at(index);
}
if (collectionName=="variables"){
return m_reportVariables.variableAt(index);
}
if (collectionName=="csvs"){
return m_csvs.at(index);
}
return 0;
}
void DataSourceManager::collectionLoadFinished(const QString &collectionName)
{
EASY_BLOCK("DataSourceManager::collectionLoadFinished");
if (collectionName.compare("connections",Qt::CaseInsensitive) == 0){
}
EASY_BLOCK("queryes");
if (collectionName.compare("queries",Qt::CaseInsensitive) == 0){
QMutableListIterator it(m_queries);
while (it.hasNext()){
it.next();
if (!m_datasources.contains(it.value()->queryName().toLower())){
connect(it.value(), SIGNAL(queryTextChanged(QString,QString)),
this, SLOT(slotQueryTextChanged(QString,QString)));
putHolder(it.value()->queryName(),new QueryHolder(it.value()->queryText(), it.value()->connectionName(), this));
} else {
delete it.value();
it.remove();
}
}
}
EASY_END_BLOCK;
EASY_BLOCK("subqueries")
if (collectionName.compare("subqueries",Qt::CaseInsensitive) == 0){
QMutableListIterator it(m_subqueries);
while (it.hasNext()){
it.next();
if (!m_datasources.contains(it.value()->queryName().toLower())){
connect(it.value(), SIGNAL(queryTextChanged(QString,QString)),
this, SLOT(slotQueryTextChanged(QString,QString)));
putHolder(it.value()->queryName(),new SubQueryHolder(
it.value()->queryText(),
it.value()->connectionName(),
it.value()->master(),
this)
);
} else {
delete it.value();
it.remove();
}
}
}
EASY_END_BLOCK;
EASY_BLOCK("subproxies");
if (collectionName.compare("subproxies",Qt::CaseInsensitive) == 0){
QMutableListIterator it(m_proxies);
while (it.hasNext()){
it.next();
if (!m_datasources.contains(it.value()->name().toLower())){
putHolder(it.value()->name(),new ProxyHolder(it.value(), this));
} else {
delete it.value();
it.remove();
}
}
}
EASY_END_BLOCK;
EASY_BLOCK("variables");
if (collectionName.compare("variables",Qt::CaseInsensitive) == 0){
foreach (VarDesc* item, m_tempVars) {
if (!m_reportVariables.containsVariable(item->name())){
m_reportVariables.addVariable(item->name(),item->value(),VarDesc::Report,FirstPass);
VarDesc* currentVar = m_reportVariables.variableByName(item->name());
currentVar->initFrom(item);
}
delete item;
}
m_tempVars.clear();
}
EASY_END_BLOCK;
if (collectionName.compare("csvs", Qt::CaseInsensitive) == 0){
QMutableListIterator it(m_csvs);
while (it.hasNext()){
it.next();
if (!m_datasources.contains(it.value()->name().toLower())){
connect(it.value(), SIGNAL(cvsTextChanged(QString,QString)),
this, SLOT(slotCSVTextChanged(QString,QString)));
putHolder(
it.value()->name(),
new CSVHolder(*it.value(), this)
);
} else {
delete it.value();
it.remove();
}
}
}
if (designTime()){
EASY_BLOCK("emit datasourcesChanged()");
emit datasourcesChanged();
EASY_END_BLOCK;
}
EASY_BLOCK("emit loadCollectionFinished(collectionName)");
emit loadCollectionFinished(collectionName);
EASY_END_BLOCK;
EASY_END_BLOCK;
}
void DataSourceManager::addVariable(const QString &name, const QVariant &value, VarDesc::VarType type, RenderPass pass)
{
if (type == VarDesc::User){
m_userVariables.addVariable(name, value, type, pass);
} else {
m_reportVariables.addVariable(name,value,type,pass);
}
if (designTime()){
EASY_BLOCK("DataSourceManager::addVariable emit ds changed");
emit datasourcesChanged();
EASY_END_BLOCK;
}
}
void DataSourceManager::deleteVariable(const QString& name)
{
m_userVariables.deleteVariable(name);
if (m_reportVariables.containsVariable(name)&&m_reportVariables.variableType(name)==VarDesc::Report){
m_reportVariables.deleteVariable(name);
if (designTime()){
m_hasChanges = true;
emit datasourcesChanged();
}
}
}
void DataSourceManager::changeVariable(const QString& name,const QVariant& value)
{
if (m_userVariables.containsVariable(name)){
m_userVariables.changeVariable(name,value);
}
if (m_reportVariables.containsVariable(name)){
m_reportVariables.changeVariable(name,value);
}
}
void DataSourceManager::setSystemVariable(const QString &name, const QVariant &value, RenderPass pass)
{
addVariable(name,value,VarDesc::System,pass);
}
void DataSourceManager::setLastError(const QString &value){
m_lastError = value;
if (!value.isEmpty() && !m_errorsList.contains(value)) {
m_errorsList.append(value);
}
}
void DataSourceManager::invalidateLinkedDatasources(QString datasourceName)
{
foreach(QString name, dataSourceNames()){
if (isSubQuery(name)){
if (subQueryByName(name)->master().compare(datasourceName) == 0)
dataSourceHolder(name)->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
}
if (isProxy(name)){
ProxyDesc* proxy = proxyByName(name);
if ((proxy->master().compare(datasourceName) == 0) || (proxy->child().compare(datasourceName) == 0))
dataSourceHolder(name)->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
}
}
}
void DataSourceManager::slotConnectionRenamed(const QString &oldName, const QString &newName)
{
foreach(QueryDesc* query, m_queries){
if (query->connectionName().compare(oldName,Qt::CaseInsensitive)==0) query->setConnectionName(newName);
}
foreach(SubQueryDesc* query, m_subqueries){
if (query->connectionName().compare(oldName,Qt::CaseInsensitive)==0) query->setConnectionName(newName);
}
}
void DataSourceManager::slotQueryTextChanged(const QString &queryName, const QString &queryText)
{
QueryHolder* holder = dynamic_cast(m_datasources.value(queryName));
if (holder){
holder->setQueryText(queryText);
}
m_varToDataSource.clear();
}
void DataSourceManager::invalidateQueriesContainsVariable(const QString& variableName)
{
if (!variableIsSystem(variableName)){
if (m_varToDataSource.contains(variableName)){
foreach(QString datasourceName, m_varToDataSource.value(variableName)){
QueryHolder* holder = dynamic_cast(m_datasources.value(datasourceName));
if (holder) holder->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE);
}
} else {
QVector datasources;
foreach (const QString& datasourceName, dataSourceNames()){
QueryHolder* holder = dynamic_cast(m_datasources.value(datasourceName));
if (holder){
#if (QT_VERSION < QT_VERSION_CHECK(5, 15, 1))
QRegExp rx(QString(Const::NAMED_VARIABLE_RX).arg(variableName));
#else
QRegularExpression rx = getNamedVariableRegEx(variableName);
#endif
if (holder->queryText().contains(rx)){
holder->invalidate(designTime() ? IDataSource::DESIGN_MODE : IDataSource::RENDER_MODE);
datasources.append(datasourceName);
}
}
}
m_varToDataSource.insert(variableName, datasources);
}
}
}
void DataSourceManager::slotVariableHasBeenAdded(const QString& variableName)
{
invalidateQueriesContainsVariable(variableName);
if (variableType(variableName) == VarDesc::Report)
m_hasChanges = true;
}
void DataSourceManager::slotVariableHasBeenChanged(const QString& variableName)
{
invalidateQueriesContainsVariable(variableName);
if (variableType(variableName) == VarDesc::Report)
m_hasChanges = true;
}
void DataSourceManager::slotCSVTextChanged(const QString &csvName, const QString &csvText)
{
CSVHolder* holder = dynamic_cast(m_datasources.value(csvName));
if (holder){
holder->setCSVText(csvText);
}
}
void DataSourceManager::clear(ClearMethod method)
{
m_varToDataSource.clear();
DataSourcesMap::iterator dit;
for( dit = m_datasources.begin(); dit != m_datasources.end(); ){
bool owned = (*dit)->isOwned() && (*dit)->isRemovable();
switch(method){
case All:
invalidateLinkedDatasources(dit.key());
delete dit.value();
dit = m_datasources.erase(dit);
break;
default:
if (owned){
invalidateLinkedDatasources(dit.key());
delete dit.value();
dit = m_datasources.erase(dit);
} else {
++dit;
}
}
}
QList::iterator cit = m_connections.begin();
while( cit != m_connections.end() ){
if ( (*cit)->isInternal() )
QSqlDatabase::removeDatabase( (*cit)->name() );
delete (*cit);
cit = m_connections.erase(cit);
}
//TODO: add smart pointes to collections
foreach(QueryDesc *desc, m_queries) delete desc;
foreach(SubQueryDesc* desc, m_subqueries) delete desc;
foreach(ProxyDesc* desc, m_proxies) delete desc;
m_queries.clear();
m_subqueries.clear();
m_proxies.clear();
// if (method == All)
// clearUserVariables();
clearReportVariables();
emit cleared();
}
void DataSourceManager::clearGroupFunction()
{
foreach(GroupFunction* gf ,m_groupFunctions.values()){
delete gf;
}
m_groupFunctions.clear();
}
void DataSourceManager::clearGroupFunctionValues(const QString& bandObjectName)
{
foreach(GroupFunction* gf, m_groupFunctions.values(bandObjectName)){
gf->values().clear();
}
}
GroupFunction* DataSourceManager::addGroupFunction(const QString &name, const QString &expression, const QString &band, const QString& dataBand)
{
GroupFunction* gf = m_groupFunctionFactory.createGroupFunction(name,expression,dataBand,this);
if (gf){
m_groupFunctions.insert(band,gf);
}
return gf;
}
GroupFunction *DataSourceManager::groupFunction(const QString &name, const QString &expression, const QString &band)
{
foreach(GroupFunction* gf,m_groupFunctions.values(band)){
if ((gf->name().compare(name,Qt::CaseInsensitive)==0)&&
(gf->data().compare(expression,Qt::CaseInsensitive)==0)
) return gf;
}
return 0;
}
QList DataSourceManager::groupFunctionsByBand(const QString &band)
{
return m_groupFunctions.values(band);
}
void DataSourceManager::updateChildrenData(const QString &datasourceName)
{
foreach(SubQueryDesc* subquery,m_subqueries){
if (subquery->master().compare(datasourceName,Qt::CaseInsensitive)==0){
SubQueryHolder* holder=dynamic_cast(dataSourceHolder(subquery->queryName()));
if (holder) holder->runQuery();
}
}
foreach(ProxyDesc* subproxy,m_proxies){
if(subproxy->master().compare(datasourceName,Qt::CaseInsensitive)==0){
ProxyHolder* holder = dynamic_cast(dataSourceHolder(subproxy->name()));
holder->filterModel();
}
}
}
QString DataSourceManager::extractDataSource(const QString &fieldName)
{
return fieldName.left(fieldName.indexOf('.'));
}
QString DataSourceManager::extractFieldName(const QString &fieldName)
{
return fieldName.right((fieldName.length()-fieldName.indexOf('.'))-1);
}
bool DataSourceManager::containsField(const QString &fieldName)
{
IDataSource* ds = dataSource(extractDataSource(fieldName));
if (ds) {
return ds->columnIndexByName(extractFieldName(fieldName))!=-1;
}
return false;
}
bool DataSourceManager::containsVariable(const QString& variableName)
{
if (m_userVariables.containsVariable(variableName)) return true;
return m_reportVariables.containsVariable(variableName);
}
void DataSourceManager::clearUserVariables()
{
m_userVariables.clearUserVariables();
m_reportVariables.clearUserVariables();
}
QVariant DataSourceManager::fieldData(const QString &fieldName)
{
if (containsField(fieldName)){
IDataSource* ds = dataSource(extractDataSource(fieldName));
if (ds) return ds->data(extractFieldName(fieldName));
}
return QVariant();
}
QVariant DataSourceManager::fieldDataByRowIndex(const QString &fieldName, int rowIndex)
{
if (containsField(fieldName)){
IDataSource* ds = dataSource(extractDataSource(fieldName));
if (ds){
return ds->dataByRowIndex(extractFieldName(fieldName), rowIndex);
}
}
return QVariant();
}
QVariant DataSourceManager::fieldDataByRowIndex(const QString &fieldName, int rowIndex, int role)
{
if(containsField(fieldName)) {
IDataSource *ds = dataSource(extractDataSource(fieldName));
if(ds) {
return ds->dataByRowIndex(extractFieldName(fieldName), rowIndex, role);
}
}
return QVariant();
}
QVariant DataSourceManager::fieldDataByRowIndex(const QString &fieldName, int rowIndex, const QString &roleName)
{
if(containsField(fieldName)) {
IDataSource *ds = dataSource(extractDataSource(fieldName));
if(ds) {
return ds->dataByRowIndex(extractFieldName(fieldName), rowIndex, roleName);
}
}
return QVariant();
}
QVariant DataSourceManager::fieldDataByKey(const QString& datasourceName, const QString& valueFieldName, const QString& keyFieldName, QVariant keyValue)
{
IDataSource* ds = dataSource(datasourceName);
if (ds){
return ds->dataByKeyField(valueFieldName, keyFieldName, keyValue);
}
return QVariant();
}
QVariant DataSourceManager::headerData(const QString &fieldName, const QString &roleName)
{
if(containsField(fieldName)) {
IDataSource *ds = dataSource(extractDataSource(fieldName));
if(ds) {
return ds->headerData(extractFieldName(fieldName), roleName);
}
}
return QVariant();
}
QString DataSourceManager::columnName(const QString &datasourceName, int index)
{
IDataSource *ds = dataSource(datasourceName);
if(ds && !ds->isInvalid() && ds->columnCount() > index) {
return ds->columnNameByIndex(index);
}
return QString("unknown");
}
int DataSourceManager::columnCount(const QString &datasourceName)
{
IDataSource *ds = dataSource(datasourceName);
if(ds && !ds->isInvalid()) {
return ds->columnCount();
}
return -1;
}
void DataSourceManager::reopenDatasource(const QString& datasourceName)
{
QueryHolder* qh = dynamic_cast(dataSourceHolder(datasourceName));
if (qh){
qh->invalidate(designTime()?IDataSource::DESIGN_MODE:IDataSource::RENDER_MODE);
invalidateChildren(datasourceName);
}
}
QVariant DataSourceManager::variable(const QString &variableName)
{
if (m_userVariables.containsVariable(variableName))
return m_userVariables.variable(variableName);
return m_reportVariables.variable(variableName);
}
RenderPass DataSourceManager::variablePass(const QString &name)
{
if (m_userVariables.containsVariable(name))
return m_userVariables.variablePass(name);
return m_reportVariables.variablePass(name);
}
bool DataSourceManager::variableIsSystem(const QString &name)
{
if (m_reportVariables.containsVariable(name))
return (m_reportVariables.variableType(name)==VarDesc::System);
return false;
}
bool DataSourceManager::variableIsMandatory(const QString& name)
{
if (m_reportVariables.containsVariable(name))
return m_reportVariables.variableByName(name)->isMandatory();
return false;
}
void DataSourceManager::setVarableMandatory(const QString& name, bool value)
{
if (m_reportVariables.containsVariable(name))
m_reportVariables.variableByName(name)->setMandatory(value);
}
QStringList DataSourceManager::variableNames()
{
return m_reportVariables.variableNames();
}
QStringList DataSourceManager::variableNamesByRenderPass(RenderPass pass)
{
QStringList result;
foreach(QString variableName, m_reportVariables.variableNames()){
if (m_reportVariables.variablePass(variableName) == pass){
result.append(variableName);
}
}
return result;
}
QStringList DataSourceManager::userVariableNames(){
return m_userVariables.variableNames();
}
VarDesc::VarType DataSourceManager::variableType(const QString &name)
{
if (m_reportVariables.containsVariable(name))
return m_reportVariables.variableType(name);
return VarDesc::User;
}
VariableDataType DataSourceManager::variableDataType(const QString& name)
{
if (m_reportVariables.containsVariable(name))
return m_reportVariables.variableByName(name)->dataType();
return Enums::Undefined;
}
void DataSourceManager::setVariableDataType(const QString& name, VariableDataType value)
{
if (m_reportVariables.containsVariable(name))
m_reportVariables.variableByName(name)->setDataType(value);
}
void DataSourceManager::setAllDatasourcesToFirst()
{
foreach(IDataSourceHolder* ds,m_datasources.values()) {
if (ds->dataSource()) ds->dataSource()->first();
}
}
} //namespace LimeReport