/*************************************************************************** * This file is part of the Lime Report project * * Copyright (C) 2015 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 #include #include #include #include #include #include #include #include #include "time.h" #include "lrreportengine_p.h" #include "lrreportengine.h" #include "lrpagedesignintf.h" #include "lrdatasourcemanager.h" #ifdef HAVE_REPORT_DESIGNER #include "lrdatabrowser.h" #include "lrreportdesignwindow.h" #endif #include "serializators/lrxmlwriter.h" #include "serializators/lrxmlreader.h" #include "lrreportrender.h" #include "lrpreviewreportwindow.h" #include "lrpreviewreportwidget.h" #include "lrpreviewreportwidget_p.h" #include "lrexporterintf.h" #include "lrexportersfactory.h" #ifdef BUILD_WITH_EASY_PROFILER #include "easy/profiler.h" #else # define EASY_BLOCK(...) # define EASY_END_BLOCK #endif #ifdef HAVE_STATIC_BUILD #include "lrfactoryinitializer.h" #endif namespace LimeReport{ QSettings* ReportEngine::m_settings = 0; ReportEnginePrivate::ReportEnginePrivate(QObject *parent) : QObject(parent), m_fileName(""), m_settings(0), m_ownedSettings(false), m_printer(new QPrinter(QPrinter::HighResolution)), m_printerSelected(false), m_showProgressDialog(true), m_reportName(""), m_activePreview(0), m_previewWindowIcon(":/report/images/logo32"), m_previewWindowTitle(tr("Preview")), m_reportRendering(false), m_resultIsEditable(true), m_passPhrase("HjccbzHjlbyfCkjy"), m_fileWatcher( new QFileSystemWatcher( this ) ), m_reportLanguage(QLocale::AnyLanguage), m_previewLayoutDirection(Qt::LayoutDirectionAuto), m_designerFactory(0), m_previewScaleType(FitWidth), m_previewScalePercent(0), m_startTOCPage(0) { #ifdef HAVE_STATIC_BUILD initResources(); initReportItems(); #ifdef HAVE_REPORT_DESIGNER initObjectInspectorProperties(); #endif initSerializators(); #endif m_datasources = new DataSourceManager(this); m_datasources->setReportSettings(&m_reportSettings); scriptManager()->setDataManager(m_datasources); m_scriptEngineContext = new ScriptEngineContext(this); ICallbackDatasource* tableOfContents = m_datasources->createCallbackDatasource("tableofcontents"); connect(tableOfContents, SIGNAL(getCallbackData(LimeReport::CallbackInfo,QVariant&)), m_scriptEngineContext->tableOfContents(), SLOT(slotOneSlotDS(LimeReport::CallbackInfo,QVariant&))); m_datasources->setObjectName("datasources"); connect(m_datasources,SIGNAL(loadCollectionFinished(QString)),this,SLOT(slotDataSourceCollectionLoaded(QString))); connect(m_fileWatcher,SIGNAL(fileChanged(const QString &)),this,SLOT(slotLoadFromFile(const QString &))); QDir pluginsDir = QCoreApplication::applicationDirPath(); pluginsDir.cd("../lib" ); if (!pluginsDir.exists()){ pluginsDir.cd("./lib"); if (!pluginsDir.exists()) pluginsDir.setPath(QCoreApplication::applicationDirPath()); } foreach( const QString& pluginName, pluginsDir.entryList( QDir::Files ) ) { QPluginLoader loader( pluginsDir.absoluteFilePath( pluginName ) ); if( loader.load() ) { #ifndef HAVE_REPORT_DESIGNER if( LimeReportDesignerPluginInterface* designerPlugin = qobject_cast< LimeReportDesignerPluginInterface* >( loader.instance() ) ) { m_designerFactory = designerPlugin; break; } #endif } } } ReportEnginePrivate::~ReportEnginePrivate() { if (m_designerWindow) { m_designerWindow->close(); } if (m_activePreview){ m_activePreview->close(); } foreach(PageDesignIntf* page,m_pages) delete page; m_pages.clear(); foreach(ReportTranslation* translation, m_translations) delete translation; m_translations.clear(); if (m_ownedSettings&&m_settings) delete m_settings; } QObject* ReportEnginePrivate::createElement(const QString &, const QString &) { return appendPage(); } QObject *ReportEnginePrivate::elementAt(const QString &, int index) { return pageAt(index); } PageDesignIntf *ReportEnginePrivate::createPage(const QString &pageName) { PageDesignIntf* page =new PageDesignIntf(); page->setObjectName(pageName); page->pageItem()->setObjectName("Report"+pageName); page->setReportEditor(this); page->setReportSettings(&m_reportSettings); return page; } PageDesignIntf *ReportEnginePrivate::appendPage(const QString &pageName) { PageDesignIntf* page = createPage(pageName); m_pages.append(page); return page; } bool ReportEnginePrivate::deletePage(PageDesignIntf *page){ QList::iterator it = m_pages.begin(); while (it != m_pages.end()){ if (*it == page) { it = m_pages.erase(it); return true; } else ++it; } return false; } PageDesignIntf *ReportEnginePrivate::createPreviewPage() { return createPage(); } int ReportEnginePrivate::elementsCount(const QString &) { return m_pages.count(); } void ReportEnginePrivate::collectionLoadFinished(const QString &) { foreach (PageDesignIntf* page, m_pages) { page->setReportEditor(this); page->setReportSettings(&m_reportSettings); page->setSceneRect(-Const::SCENE_MARGIN,-Const::SCENE_MARGIN, page->pageItem()->width()+Const::SCENE_MARGIN*2, page->pageItem()->height()+Const::SCENE_MARGIN*2); } emit pagesLoadFinished(); } void ReportEnginePrivate::saveError(QString message) { m_lastError = message; } void ReportEnginePrivate::showError(QString message) { QMessageBox::critical(0,tr("Error"),message); } void ReportEnginePrivate::updateTranslations() { foreach(ReportTranslation* translation, m_translations.values()){ foreach(PageDesignIntf* page, m_pages){ translation->updatePageTranslation(page); } } } void ReportEnginePrivate::slotDataSourceCollectionLoaded(const QString &collectionName) { emit datasourceCollectionLoadFinished(collectionName); } void ReportEnginePrivate::slotPreviewWindowDestroyed(QObject* window) { if (m_activePreview == window){ m_activePreview = 0; } } void ReportEnginePrivate::slotDesignerWindowDestroyed(QObject *window) { Q_UNUSED(window) dataManager()->setDesignTime(false); } void ReportEnginePrivate::clearReport() { foreach(PageDesignIntf* page,m_pages) delete page; m_pages.clear(); foreach(ReportTranslation* reportTranslation, m_translations) delete reportTranslation; m_translations.clear(); m_datasources->clear(DataSourceManager::Owned); m_fileName=""; m_scriptEngineContext->clear(); m_reportSettings.setDefaultValues(); emit cleared(); } void ReportEnginePrivate::printReport(ItemsReaderIntf::Ptr reader, QPrinter& printer) { LimeReport::PageDesignIntf renderPage; renderPage.setItemMode(PrintMode); if (reader->first()){ reader->readItem(renderPage.pageItem()); printer.setFullPage(renderPage.pageItem()->fullPage()); printer.setOrientation((QPrinter::Orientation)renderPage.pageItem()->pageOrientation()); renderPage.setSceneRect(renderPage.pageItem()->mapToScene(renderPage.pageItem()->rect()).boundingRect()); if (renderPage.pageItem()->pageSize()==PageItemDesignIntf::Custom){ QSizeF pageSize = (renderPage.pageItem()->pageOrientation()==PageItemDesignIntf::Landscape)? QSizeF(renderPage.pageItem()->sizeMM().height(),renderPage.pageItem()->sizeMM().width()): renderPage.pageItem()->sizeMM(); printer.setPaperSize(pageSize,QPrinter::Millimeter); } else { printer.setPaperSize((QPrinter::PageSize)renderPage.pageItem()->pageSize()); } QPainter painter(&printer); renderPage.render(&painter); while (reader->next()){ printer.newPage(); renderPage.removeAllItems(); reader->readItem(renderPage.pageItem()); renderPage.setSceneRect(renderPage.pageItem()->mapToScene(renderPage.pageItem()->rect()).boundingRect()); renderPage.render(&painter); } } } void ReportEnginePrivate::printReport(ReportPages pages, QPrinter &printer) { int currenPage = 1; QMap> printProcessors; printProcessors.insert("default",QSharedPointer(new PrintProcessor(&printer))); foreach(PageItemDesignIntf::Ptr page, pages){ if ( (printer.printRange() == QPrinter::AllPages) || ( (printer.printRange()==QPrinter::PageRange) && (currenPage>=printer.fromPage()) && (currenPage<=printer.toPage()) ) ) { printProcessors["default"]->printPage(page); } currenPage++; } } void ReportEnginePrivate::printReport(ReportPages pages, QMap printers, bool printToAllPrinters) { if (printers.values().isEmpty()) return; int currenPage = 1; QMap> printProcessors; for (int i = 0; i < printers.keys().count(); ++i) { printProcessors.insert(printers.keys()[i],QSharedPointer(new PrintProcessor(printers[printers.keys()[i]]))); } PrintProcessor* defaultProcessor = 0; int currentPrinter = 0; if (printProcessors.contains("default")) defaultProcessor = printProcessors["default"].data(); else defaultProcessor = printProcessors.values().at(0).data(); foreach(PageItemDesignIntf::Ptr page, pages){ if (!printToAllPrinters){ if (printProcessors.contains(page->printerName())) printProcessors[page->printerName()]->printPage(page); else defaultProcessor->printPage(page); } else { printProcessors.values().at(currentPrinter)->printPage(page); if (currentPrinter < printers.values().count()-1) currentPrinter++; else currentPrinter = 0; } currenPage++; } } QStringList ReportEnginePrivate::aviableReportTranslations() { QStringList result; foreach (QLocale::Language language, aviableLanguages()){ result << QLocale::languageToString(language); } return result; } void ReportEnginePrivate::setReportTranslation(const QString &languageName) { foreach(QLocale::Language language, aviableLanguages()){ if (QLocale::languageToString(language).compare(languageName) == 0){ setReportLanguage(language); } } } bool ReportEnginePrivate::printReport(QPrinter* printer) { if (!printer&&!m_printerSelected){ QPrinterInfo pi; if (!pi.defaultPrinter().isNull()) #ifdef HAVE_QT4 m_printer.data()->setPrinterName(pi.defaultPrinter().printerName()); #endif #ifdef HAVE_QT5 #if (QT_VERSION >= QT_VERSION_CHECK(5, 3, 0)) m_printer.data()->setPrinterName(pi.defaultPrinterName()); #else m_printer.data()->setPrinterName(pi.defaultPrinter().printerName()); #endif #endif QPrintDialog dialog(m_printer.data(),QApplication::activeWindow()); m_printerSelected = dialog.exec()!=QDialog::Rejected; } if (!printer&&!m_printerSelected) return false; printer =(printer)?printer:m_printer.data(); if (printer&&printer->isValid()){ try{ bool designTime = dataManager()->designTime(); dataManager()->setDesignTime(false); ReportPages pages = renderToPages(); dataManager()->setDesignTime(designTime); if (pages.count()>0){ printReport(pages,*printer); } } catch(ReportError &exception){ saveError(exception.what()); } return true; } else return false; } bool ReportEnginePrivate::printReport(QMap printers, bool printToAllPrinters) { try{ bool designTime = dataManager()->designTime(); dataManager()->setDesignTime(false); ReportPages pages = renderToPages(); dataManager()->setDesignTime(designTime); if (pages.count()>0){ printReport(pages, printers, printToAllPrinters); } } catch(ReportError &exception){ saveError(exception.what()); return false; } return true; } bool ReportEnginePrivate::printPages(ReportPages pages, QPrinter *printer) { if (!printer&&!m_printerSelected){ QPrinterInfo pi; if (!pi.defaultPrinter().isNull()) #ifdef HAVE_QT4 m_printer.data()->setPrinterName(pi.defaultPrinter().printerName()); #endif #ifdef HAVE_QT5 #if (QT_VERSION >= QT_VERSION_CHECK(5, 3, 0)) m_printer.data()->setPrinterName(pi.defaultPrinterName()); #else m_printer.data()->setPrinterName(pi.defaultPrinter().printerName()); #endif #endif QPrintDialog dialog(m_printer.data(),QApplication::activeWindow()); m_printerSelected = dialog.exec()!=QDialog::Rejected; } if (!printer&&!m_printerSelected) return false; printer =(printer)?printer:m_printer.data(); if (printer&&printer->isValid()){ try{ if (pages.count()>0){ printReport( pages, *printer ); } } catch(ReportError &exception){ saveError(exception.what()); } return true; } else return false; } void ReportEnginePrivate::printToFile(const QString &fileName) { if (!fileName.isEmpty()){ QFile file(fileName); if (file.open(QIODevice::WriteOnly)){ QTextStream out(&file); try { dataManager()->setDesignTime(false); out<setDesignTime(true); } catch( ReportError &exception){ saveError(exception.what()); } } file.close(); } } bool ReportEnginePrivate::printToPDF(const QString &fileName) { return exportReport("PDF", fileName); } bool ReportEnginePrivate::exportReport(QString exporterName, const QString &fileName, const QMap ¶ms) { QString fn = fileName; if (ExportersFactory::instance().map().contains(exporterName)){ ReportExporterInterface* e = ExportersFactory::instance().objectCreator(exporterName)(this); if (fn.isEmpty()){ QString filter = QString("%1 (*.%2)").arg(e->exporterName()).arg(e->exporterFileExt()); QString fn = QFileDialog::getSaveFileName(0,tr("%1 file name").arg(e->exporterName()),"",filter); } if (!fn.isEmpty()){ QFileInfo fi(fn); if (fi.suffix().isEmpty()) fn += QString(".%1").arg(e->exporterFileExt()); bool designTime = dataManager()->designTime(); dataManager()->setDesignTime(false); ReportPages pages = renderToPages(); dataManager()->setDesignTime(designTime); bool result = e->exportPages(pages, fn, params); delete e; return result; } } return false; } void ReportEnginePrivate::previewReport(PreviewHints hints) { // QTime start = QTime::currentTime(); try{ dataManager()->setDesignTime(false); ReportPages pages = renderToPages(); dataManager()->setDesignTime(true); if (pages.count()>0){ Q_Q(ReportEngine); PreviewReportWindow* w = new PreviewReportWindow(q,0,settings()); w->setWindowFlags(Qt::Dialog|Qt::WindowMaximizeButtonHint|Qt::WindowCloseButtonHint| Qt::WindowMinMaxButtonsHint); w->setAttribute(Qt::WA_DeleteOnClose,true); w->setWindowModality(Qt::ApplicationModal); //w->setWindowIcon(QIcon(":/report/images/main.ico")); w->setWindowIcon(m_previewWindowIcon); w->setWindowTitle(m_previewWindowTitle); w->setSettings(settings()); w->setPages(pages); w->setLayoutDirection(m_previewLayoutDirection); if (!dataManager()->errorsList().isEmpty()){ w->setErrorMessages(dataManager()->errorsList()); } if (!hints.testFlag(PreviewBarsUserSetting)){ w->setMenuVisible(!hints.testFlag(HidePreviewMenuBar)); w->setStatusBarVisible(!hints.testFlag(HidePreviewStatusBar)); w->setToolBarVisible(!hints.testFlag(HidePreviewToolBar)); } w->setHideResultEditButton(resultIsEditable()); w->setStyleSheet(m_styleSheet); m_activePreview = w; w->setPreviewScaleType(m_previewScaleType, m_previewScalePercent); connect(w,SIGNAL(destroyed(QObject*)), this, SLOT(slotPreviewWindowDestroyed(QObject*))); w->exec(); } } catch (ReportError &exception){ saveError(exception.what()); showError(exception.what()); } } ReportDesignWindowInterface*ReportEnginePrivate::getDesignerWindow() { if (!m_designerWindow) { if (m_designerFactory){ m_designerWindow = m_designerFactory->getDesignerWindow(this,QApplication::activeWindow(),settings()); m_designerWindow->setAttribute(Qt::WA_DeleteOnClose,true); m_designerWindow->setWindowIcon(QIcon(":report/images/logo32")); m_designerWindow->setShowProgressDialog(m_showProgressDialog); } else { #ifdef HAVE_REPORT_DESIGNER m_designerWindow = new LimeReport::ReportDesignWindow(this,QApplication::activeWindow(),settings()); m_designerWindow->setAttribute(Qt::WA_DeleteOnClose,true); m_designerWindow->setWindowIcon(QIcon(":report/images/logo32")); m_designerWindow->setShowProgressDialog(m_showProgressDialog); #endif } } if (m_designerWindow){ m_datasources->updateDatasourceModel(); } return m_designerWindow; } PreviewReportWidget* ReportEnginePrivate::createPreviewWidget(QWidget* parent){ Q_Q(ReportEngine); PreviewReportWidget* widget = new PreviewReportWidget(q, parent); try{ dataManager()->setDesignTime(false); ReportPages pages = renderToPages(); dataManager()->setDesignTime(true); if (pages.count()>0) widget->d_ptr->setPages(pages); } catch (ReportError &exception){ saveError(exception.what()); showError(exception.what()); } return widget; } PageDesignIntf* ReportEnginePrivate::createPreviewScene(QObject* parent){ PageDesignIntf* result = 0; try { ReportPages pages = renderToPages(); result = new PageDesignIntf(parent); result->setPageItems(pages); } catch (ReportError &exception){ saveError(exception.what()); showError(exception.what()); } return result; } bool ReportEnginePrivate::emitSaveReport() { bool result = false; emit onSave(result); return result; } bool ReportEnginePrivate::emitSaveReportAs() { bool result = false; emit onSaveAs(result); return result; } bool ReportEnginePrivate::emitLoadReport() { bool result = false; emit onLoad(result); return result; } void ReportEnginePrivate::emitSaveFinished() { emit saveFinished(); } void ReportEnginePrivate::emitLoadFinished() { emit loadFinished(); } void ReportEnginePrivate::emitPrintedToPDF(QString fileName) { emit printedToPDF(fileName); } bool ReportEnginePrivate::isSaved() { foreach (PageDesignIntf* page, m_pages) { if (!page->isSaved()) return false; } return true; } void ReportEnginePrivate::setCurrentReportsDir(const QString &dirName) { if (QDir(dirName).exists()) m_reportsDir = dirName; } bool ReportEnginePrivate::slotLoadFromFile(const QString &fileName) { EASY_BLOCK("ReportEnginePrivate::slotLoadFromFile") PreviewReportWindow *currentPreview = qobject_cast(m_activePreview); if (!QFile::exists(fileName)) { if ( hasActivePreview() ) { QMessageBox::information( NULL, tr( "Report File Change" ), tr( "The report file \"%1\" has changed names or been deleted.\n\nThis preview is no longer valid." ).arg( fileName ) ); clearReport(); currentPreview->close(); } return false; } clearReport(); ItemsReaderIntf::Ptr reader = FileXMLReader::create(fileName); reader->setPassPhrase(m_passPhrase); if (reader->first()){ if (reader->readItem(this)){ m_fileName=fileName; QFileInfo fi(fileName); m_reportName = fi.fileName(); QString dbSettingFileName = fi.absolutePath()+"/"+fi.baseName()+".db"; if (QFile::exists(dbSettingFileName)){ QSettings dbcredentals(dbSettingFileName, QSettings::IniFormat); foreach (ConnectionDesc* connection, dataManager()->conections()) { if (!connection->keepDBCredentials()){ dbcredentals.beginGroup(connection->name()); connection->setUserName(dbcredentals.value("user").toString()); connection->setPassword(dbcredentals.value("password").toString()); dbcredentals.endGroup(); } } } EASY_BLOCK("Connect auto connections") dataManager()->connectAutoConnections(); EASY_END_BLOCK; dropChanges(); if ( hasActivePreview() ) { currentPreview->reloadPreview(); } EASY_END_BLOCK; return true; }; } m_lastError = reader->lastError(); EASY_END_BLOCK; return false; } void ReportEnginePrivate::cancelRender() { if (m_reportRender) m_reportRender->cancelRender(); m_reportRendering = false; } PageDesignIntf* ReportEngine::createPreviewScene(QObject* parent){ Q_D(ReportEngine); return d->createPreviewScene(parent); } void ReportEnginePrivate::designReport() { ReportDesignWindowInterface* designerWindow = getDesignerWindow(); if (designerWindow){ dataManager()->setDesignTime(true); connect(designerWindow, SIGNAL(destroyed(QObject*)), this, SLOT(slotDesignerWindowDestroyed(QObject*))); #ifdef Q_OS_WIN designerWindow->setWindowModality(Qt::ApplicationModal); #endif if (QApplication::activeWindow()==0){ designerWindow->show();; } else { designerWindow->showModal(); } } else { qDebug()<<(tr("Designer not found!")); } } void ReportEnginePrivate::setSettings(QSettings* value) { if (value){ if (m_ownedSettings&&m_settings) delete m_settings; m_settings = value; m_ownedSettings = false; } } QSettings*ReportEnginePrivate::settings() { if (m_settings){ return m_settings; } else { m_settings = new QSettings("LimeReport",QApplication::applicationName()); m_ownedSettings=true; return m_settings; } } bool ReportEnginePrivate::loadFromFile(const QString &fileName, bool autoLoadPreviewOnChange) { // only watch one file at a time if ( !m_fileWatcher->files().isEmpty() ) { m_fileWatcher->removePaths( m_fileWatcher->files() ); } if ( autoLoadPreviewOnChange ) { m_fileWatcher->addPath( fileName ); } bool result = slotLoadFromFile( fileName ); if (result) { emit loadFinished(); } return result; } bool ReportEnginePrivate::loadFromByteArray(QByteArray* data, const QString &name){ clearReport(); ItemsReaderIntf::Ptr reader = ByteArrayXMLReader::create(data); reader->setPassPhrase(m_passPhrase); if (reader->first()){ if (reader->readItem(this)){ m_fileName = ""; m_reportName = name; emit loadFinished(); return true; }; } return false; } bool ReportEnginePrivate::loadFromString(const QString &report, const QString &name) { clearReport(); ItemsReaderIntf::Ptr reader = StringXMLreader::create(report); reader->setPassPhrase(m_passPhrase); if (reader->first()){ if (reader->readItem(this)){ m_fileName = ""; m_reportName = name; emit loadFinished(); return true; }; } return false; } bool ReportEnginePrivate::saveToFile(const QString &fileName) { if (fileName.isEmpty() & m_fileName.isEmpty()) return false; QString fn = fileName.isEmpty() ? m_fileName : fileName; QFileInfo fi(fn); if (fi.suffix().isEmpty()) fn+=".lrxml"; QString dbSettingFileName = fi.absolutePath()+"/"+fi.baseName()+".db"; QSettings dbcredentals(dbSettingFileName, QSettings::IniFormat); foreach (ConnectionDesc* connection, dataManager()->conections()) { if (!connection->keepDBCredentials()){ dbcredentals.beginGroup(connection->name()); dbcredentals.setValue("user",connection->userName()); dbcredentals.setValue("password",connection->password()); dbcredentals.endGroup(); connection->setPassword(""); connection->setUserName(""); } } QScopedPointer< ItemsWriterIntf > writer(new XMLWriter()); writer->setPassPhrase(m_passPhrase); writer->putItem(this); m_fileName=fn; bool saved = writer->saveToFile(fn); foreach (ConnectionDesc* connection, dataManager()->conections()) { if (!connection->keepDBCredentials()){ dbcredentals.beginGroup(connection->name()); connection->setUserName(dbcredentals.value("user").toString()); connection->setPassword(dbcredentals.value("password").toString()); dbcredentals.endGroup(); } } if (saved){ foreach(PageDesignIntf* page, m_pages){ page->setToSaved(); } } dropChanges(); return saved; } QByteArray ReportEnginePrivate::saveToByteArray() { QScopedPointer< ItemsWriterIntf > writer(new XMLWriter()); writer->setPassPhrase(m_passPhrase); writer->putItem(this); QByteArray result = writer->saveToByteArray(); if (!result.isEmpty()){ foreach(PageDesignIntf* page, m_pages){ page->setToSaved(); } } dropChanges(); return result; } QString ReportEnginePrivate::saveToString(){ QScopedPointer< ItemsWriterIntf > writer(new XMLWriter()); writer->setPassPhrase(m_passPhrase); writer->putItem(this); QString result = writer->saveToString(); if (!result.isEmpty()){ foreach(PageDesignIntf* page, m_pages){ page->setToSaved(); } } dropChanges(); return result; } bool ReportEnginePrivate::isNeedToSave() { foreach(PageDesignIntf* page, m_pages){ if (page->isHasChanges()) return true; } if (dataManager()->hasChanges()){ return true; } if (scriptContext()->hasChanges()) return true; return false; } QString ReportEnginePrivate::renderToString() { LimeReport::ReportRender render; updateTranslations(); dataManager()->connectAllDatabases(); dataManager()->setDesignTime(false); if (m_pages.count()){ render.setDatasources(dataManager()); render.setScriptContext(scriptContext()); return render.renderPageToString(m_pages.at(0)->pageItem()); } else return QString(); } ScaleType ReportEnginePrivate::previewScaleType() { return m_previewScaleType; } int ReportEnginePrivate::previewScalePercent() { return m_previewScalePercent; } void ReportEnginePrivate::setPreviewScaleType(const ScaleType &scaleType, int percent) { m_previewScaleType = scaleType; m_previewScalePercent = percent; } PageItemDesignIntf* ReportEnginePrivate::getPageByName(const QString& pageName) { foreach(PageItemDesignIntf* page, m_renderingPages){ if ( page->objectName().compare(pageName, Qt::CaseInsensitive) == 0) return page; } return 0; } Qt::LayoutDirection ReportEnginePrivate::previewLayoutDirection() { return m_previewLayoutDirection; } void ReportEnginePrivate::setPreviewLayoutDirection(const Qt::LayoutDirection& layoutDirection) { m_previewLayoutDirection = layoutDirection; } void ReportEnginePrivate::setPassPhrase(const QString &passPhrase) { m_passPhrase = passPhrase; } void ReportEnginePrivate::reorderPages(const QList& reorderedPages) { m_pages.clear(); foreach(PageDesignIntf* page, reorderedPages){ m_pages.append(page); } } void ReportEnginePrivate::clearSelection() { foreach (PageDesignIntf* page, m_pages) { foreach(QGraphicsItem* item, page->selectedItems()){ item->setSelected(false); } } } bool ReportEnginePrivate::addTranslationLanguage(QLocale::Language language) { if (!m_translations.keys().contains(language)){ ReportTranslation* translation = 0; if (!m_translations.contains(QLocale::AnyLanguage)){ translation = new ReportTranslation(QLocale::AnyLanguage,m_pages); m_translations.insert(QLocale::AnyLanguage,translation); } translation = new ReportTranslation(language,m_pages); m_translations.insert(language, translation); return true; } else { m_lastError = tr("Language %1 already exists").arg(QLocale::languageToString(language)); return false; } } bool ReportEnginePrivate::removeTranslationLanguage(QLocale::Language language) { return m_translations.remove(language) != 0; } void ReportEnginePrivate::activateLanguage(QLocale::Language language) { if (!m_translations.keys().contains(language)) return; ReportTranslation* translation = m_translations.value(language); foreach(PageTranslation* pageTranslation, translation->pagesTranslation()){ PageItemDesignIntf* page = getPageByName(pageTranslation->pageName); if (page){ foreach(ItemTranslation* itemTranslation, pageTranslation->itemsTranslation){ BaseDesignIntf* item = page->childByName(itemTranslation->itemName); if (item) { foreach(PropertyTranslation* propertyTranslation, itemTranslation->propertyesTranslation){ if (propertyTranslation->checked) item->setProperty(propertyTranslation->propertyName.toLatin1(), propertyTranslation->value); } } } } } } QList ReportEnginePrivate::designerLanguages() { QList result; emit getAviableLanguages(&result); return result; } QLocale::Language ReportEnginePrivate::currentDesignerLanguage() { QLocale::Language result = emit getCurrentDefaultLanguage(); return result; } void ReportEnginePrivate::setCurrentDesignerLanguage(QLocale::Language language) { m_currentDesignerLanguage = language; emit currentDefaulLanguageChanged(language); } QString ReportEnginePrivate::styleSheet() const { return m_styleSheet; } void ReportEnginePrivate::setStyleSheet(const QString &styleSheet) { m_styleSheet = styleSheet; } bool ReportEnginePrivate::setReportLanguage(QLocale::Language language){ m_reportLanguage = language; if (!m_translations.keys().contains(language)) return false; // activateLanguage(language); return true; } QList ReportEnginePrivate::aviableLanguages() { return m_translations.keys(); } ReportTranslation*ReportEnginePrivate::reportTranslation(QLocale::Language language) { return m_translations.value(language); } bool ReportEnginePrivate::resultIsEditable() const { return m_resultIsEditable; } void ReportEnginePrivate::setResultEditable(bool value) { m_resultIsEditable = value; } bool ReportEnginePrivate::suppressFieldAndVarError() const { return m_reportSettings.suppressAbsentFieldsAndVarsWarnings(); } void ReportEnginePrivate::setSuppressFieldAndVarError(bool suppressFieldAndVarError) { m_reportSettings.setSuppressAbsentFieldsAndVarsWarnings(suppressFieldAndVarError); } bool ReportEnginePrivate::isBusy() { return m_reportRendering; } QString ReportEnginePrivate::previewWindowTitle() const { return m_previewWindowTitle; } void ReportEnginePrivate::setPreviewWindowTitle(const QString &previewWindowTitle) { m_previewWindowTitle = previewWindowTitle; } QIcon ReportEnginePrivate::previewWindowIcon() const { return m_previewWindowIcon; } void ReportEnginePrivate::setPreviewWindowIcon(const QIcon &previewWindowIcon) { m_previewWindowIcon = previewWindowIcon; } PageItemDesignIntf* ReportEnginePrivate::createRenderingPage(PageItemDesignIntf* page){ PageItemDesignIntf* result = dynamic_cast(page->cloneItem(page->itemMode())); ICollectionContainer* co = dynamic_cast(result); if (co) co->collectionLoadFinished("children"); return result; } void ReportEnginePrivate::initReport() { for(int index = 0; index < pageCount(); ++index){ PageDesignIntf* page = pageAt(index); if (page != 0){ foreach (BaseDesignIntf* item, page->pageItem()->childBaseItems()) { IPainterProxy *proxyItem = dynamic_cast(item); if (proxyItem){ proxyItem->setExternalPainter(this); } } } } } void ReportEnginePrivate::paintByExternalPainter(const QString& objectName, QPainter* painter, const QStyleOptionGraphicsItem* options) { emit externalPaint(objectName, painter, options); } ReportPages ReportEnginePrivate::renderToPages() { int startTOCPage = -1; int pageAfterTOCIndex = -1; if (m_reportRendering) return ReportPages(); initReport(); m_reportRender = ReportRender::Ptr(new ReportRender); updateTranslations(); connect(m_reportRender.data(),SIGNAL(pageRendered(int)), this, SIGNAL(renderPageFinished(int))); if (m_pages.count()){ #ifdef HAVE_UI_LOADER m_scriptEngineContext->initDialogs(); #endif ReportPages result; m_reportRendering = true; m_reportRender->setDatasources(dataManager()); m_reportRender->setScriptContext(scriptContext()); foreach (PageDesignIntf* page, m_pages) { PageItemDesignIntf* rp = createRenderingPage(page->pageItem()); m_renderingPages.append(rp); scriptContext()->baseDesignIntfToScript(rp->objectName(), rp); } scriptContext()->qobjectToScript("engine",this); if (m_scriptEngineContext->runInitScript()){ dataManager()->clearErrors(); dataManager()->connectAllDatabases(); dataManager()->setDesignTime(false); dataManager()->updateDatasourceModel(); activateLanguage(m_reportLanguage); emit renderStarted(); for(int i = 0; i < m_renderingPages.count(); ++i){ PageItemDesignIntf* page = m_renderingPages.at(i); if (!page->isTOC() && page->isPrintable()){ page->setReportSettings(&m_reportSettings); result.append(m_reportRender->renderPageToPages(page)); } else { startTOCPage = result.count(); pageAfterTOCIndex = i+1; m_reportRender->createTOCMarker(page->resetPageNumber()); } } for (int i=0; iisTOC()){ page->setReportSettings(&m_reportSettings); if (i < m_renderingPages.count()){ PageItemDesignIntf* secondPage = 0; if ( m_renderingPages.count() > (pageAfterTOCIndex)) secondPage = m_renderingPages.at(pageAfterTOCIndex); ReportPages pages = m_reportRender->renderTOC( page, true, secondPage && secondPage->resetPageNumber() ); for (int j=0; jrenderPageToPages(page)); } } } m_reportRender->secondRenderPass(result); emit renderFinished(); m_reportRender.clear(); //foreach(PageItemDesignIntf* page, m_renderingPages){ // delete page; //} m_renderingPages.clear(); } m_reportRendering = false; //activateLanguage(QLocale::AnyLanguage); return result; } else { return ReportPages(); } } QString ReportEnginePrivate::lastError() { return m_lastError; } ReportEngine::ReportEngine(QObject *parent) : QObject(parent), d_ptr(new ReportEnginePrivate()) { Q_D(ReportEngine); d->q_ptr=this; connect(d, SIGNAL(renderStarted()), this, SIGNAL(renderStarted())); connect(d, SIGNAL(renderPageFinished(int)), this, SIGNAL(renderPageFinished(int))); connect(d, SIGNAL(renderFinished()), this, SIGNAL(renderFinished())); connect(d, SIGNAL(onSave(bool&)), this, SIGNAL(onSave(bool&))); connect(d, SIGNAL(onSaveAs(bool&)), this, SIGNAL(onSaveAs(bool&))); connect(d, SIGNAL(onLoad(bool&)), this, SIGNAL(onLoad(bool&))); connect(d, SIGNAL(saveFinished()), this, SIGNAL(saveFinished())); connect(d, SIGNAL(loadFinished()), this, SIGNAL(loadFinished())); connect(d, SIGNAL(cleared()), this, SIGNAL(cleared())); connect(d, SIGNAL(printedToPDF(QString)), this, SIGNAL(printedToPDF(QString))); connect(d, SIGNAL(getAviableLanguages(QList*)), this, SIGNAL(getAviableLanguages(QList*))); connect(d, SIGNAL(currentDefaulLanguageChanged(QLocale::Language)), this, SIGNAL(currentDefaulLanguageChanged(QLocale::Language))); connect(d, SIGNAL(getCurrentDefaultLanguage()), this, SIGNAL(getCurrentDefaultLanguage())); connect(d, SIGNAL(externalPaint(const QString&, QPainter*, const QStyleOptionGraphicsItem*)), this, SIGNAL(externalPaint(const QString&, QPainter*, const QStyleOptionGraphicsItem*))); } ReportEngine::~ReportEngine() { delete d_ptr; } bool ReportEngine::printReport(QPrinter *printer) { Q_D(ReportEngine); return d->printReport(printer); } bool ReportEngine::printReport(QMap printers, bool printToAllPrinters) { Q_D(ReportEngine); return d->printReport(printers, printToAllPrinters); } bool ReportEngine::printPages(ReportPages pages, QPrinter *printer){ Q_D(ReportEngine); return d->printPages(pages,printer); } void ReportEngine::printToFile(const QString &fileName) { Q_D(ReportEngine); d->printToFile(fileName); } bool ReportEngine::printToPDF(const QString &fileName) { Q_D(ReportEngine); return d->printToPDF(fileName); } bool ReportEngine::exportReport(QString exporterName, const QString &fileName, const QMap ¶ms) { Q_D(ReportEngine); return d->exportReport(exporterName, fileName, params); } void ReportEngine::previewReport(PreviewHints hints) { Q_D(ReportEngine); if (m_settings) d->setSettings(m_settings); d->previewReport(hints); } void ReportEngine::designReport() { Q_D(ReportEngine); if (m_settings) d->setSettings(m_settings); d->designReport(); } ReportDesignWindowInterface* ReportEngine::getDesignerWindow() { Q_D(ReportEngine); return d->getDesignerWindow(); } PreviewReportWidget* ReportEngine::createPreviewWidget(QWidget *parent) { Q_D(ReportEngine); return d->createPreviewWidget(parent); } void ReportEngine::setPreviewWindowTitle(const QString &title) { Q_D(ReportEngine); d->setPreviewWindowTitle(title); } void ReportEngine::setPreviewWindowIcon(const QIcon &icon) { Q_D(ReportEngine); d->setPreviewWindowIcon(icon); } void ReportEngine::setResultEditable(bool value) { Q_D(ReportEngine); d->setResultEditable(value); } bool ReportEngine::resultIsEditable() { Q_D(ReportEngine); return d->resultIsEditable(); } bool ReportEngine::isBusy() { Q_D(ReportEngine); return d->isBusy(); } void ReportEngine::setPassPharse(QString &passPharse) { Q_D(ReportEngine); d->setPassPhrase(passPharse); } QList ReportEngine::aviableLanguages() { Q_D(ReportEngine); return d->aviableLanguages(); } bool ReportEngine::setReportLanguage(QLocale::Language language) { Q_D(ReportEngine); return d->setReportLanguage(language); } Qt::LayoutDirection ReportEngine::previewLayoutDirection() { Q_D(ReportEngine); return d->previewLayoutDirection(); } void ReportEngine::setPreviewLayoutDirection(const Qt::LayoutDirection& previewLayoutDirection) { Q_D(ReportEngine); d->setPreviewLayoutDirection(previewLayoutDirection); } QList ReportEngine::designerLanguages() { Q_D(ReportEngine); return d->designerLanguages(); } QLocale::Language ReportEngine::currentDesignerLanguage() { Q_D(ReportEngine); return d->currentDesignerLanguage(); } ScaleType ReportEngine::previewScaleType() { Q_D(ReportEngine); return d->previewScaleType(); } int ReportEngine::previewScalePercent() { Q_D(ReportEngine); return d->previewScalePercent(); } void ReportEngine::setPreviewScaleType(const ScaleType &previewScaleType, int percent) { Q_D(ReportEngine); d->setPreviewScaleType(previewScaleType, percent); } void ReportEngine::setShowProgressDialog(bool value) { Q_D(ReportEngine); d->setShowProgressDialog(value); } IDataSourceManager *ReportEngine::dataManager() { Q_D(ReportEngine); return d->dataManagerIntf(); } IScriptEngineManager *ReportEngine::scriptManager() { Q_D(ReportEngine); return d->scriptManagerIntf(); } bool ReportEngine::loadFromFile(const QString &fileName, bool autoLoadPreviewOnChange) { Q_D(ReportEngine); return d->loadFromFile(fileName, autoLoadPreviewOnChange); } bool ReportEngine::loadFromByteArray(QByteArray* data){ Q_D(ReportEngine); return d->loadFromByteArray(data); } bool ReportEngine::loadFromString(const QString &data) { Q_D(ReportEngine); return d->loadFromString(data); } QString ReportEngine::reportFileName() { Q_D(ReportEngine); return d->reportFileName(); } void ReportEngine::setReportFileName(const QString &fileName) { Q_D(ReportEngine); return d->setReportFileName(fileName); } bool ReportEngine::saveToFile(const QString &fileName) { Q_D(ReportEngine); return d->saveToFile(fileName); } QByteArray ReportEngine::saveToByteArray() { Q_D(ReportEngine); return d->saveToByteArray(); } QString ReportEngine::saveToString() { Q_D(ReportEngine); return d->saveToString(); } QString ReportEngine::lastError() { Q_D(ReportEngine); return d->lastError(); } void ReportEngine::setCurrentReportsDir(const QString &dirName) { Q_D(ReportEngine); return d->setCurrentReportsDir(dirName); } void ReportEngine::setReportName(const QString &name) { Q_D(ReportEngine); return d->setReportName(name); } QString ReportEngine::reportName() { Q_D(ReportEngine); return d->reportName(); } void ReportEngine::cancelRender() { Q_D(ReportEngine); d->cancelRender(); } ReportEngine::ReportEngine(ReportEnginePrivate &dd, QObject *parent) :QObject(parent),d_ptr(&dd) { Q_D(ReportEngine); d->q_ptr=this; connect(d, SIGNAL(renderStarted()), this, SIGNAL(renderStarted())); connect(d, SIGNAL(renderPageFinished(int)), this, SIGNAL(renderPageFinished(int))); connect(d, SIGNAL(renderFinished()), this, SIGNAL(renderFinished())); } ScriptEngineManager*LimeReport::ReportEnginePrivate::scriptManager(){ ScriptEngineManager::instance().setContext(scriptContext()); ScriptEngineManager::instance().setDataManager(dataManager()); return &ScriptEngineManager::instance(); } PrintProcessor::PrintProcessor(QPrinter* printer) : m_printer(printer), m_painter(new QPainter(m_printer)), m_firstPage(true) {} bool PrintProcessor::printPage(PageItemDesignIntf::Ptr page) { if (m_painter && !m_painter->isActive()) return false; LimeReport::PageDesignIntf renderPage; renderPage.setPageItem(page); renderPage.setItemMode(PrintMode); initPrinter(renderPage.pageItem()); QPointF backupPagePos = page->pos(); page->setPos(0,0); renderPage.setPageItem(page); renderPage.setSceneRect(renderPage.pageItem()->mapToScene(renderPage.pageItem()->rect()).boundingRect()); if (!m_firstPage){ m_printer->newPage(); } else { m_firstPage = false; } qreal leftMargin, topMargin, rightMargin, bottomMargin; m_printer->getPageMargins(&leftMargin, &topMargin, &rightMargin, &bottomMargin, QPrinter::Millimeter); QRectF printerPageRect = m_printer->pageRect(QPrinter::Millimeter); printerPageRect = QRectF(0,0,(printerPageRect.size().width() + rightMargin + leftMargin) * Const::mmFACTOR, (printerPageRect.size().height() + bottomMargin +topMargin) * Const::mmFACTOR); if (m_printer->pageSize() != static_cast(page->pageSize()) && printerPageRect.width() < page->geometry().width()) { qreal pageWidth = page->geometry().width(); QRectF currentPrintingRect = printerPageRect; while (pageWidth>0){ renderPage.render(m_painter, m_printer->pageRect(), currentPrintingRect); currentPrintingRect.adjust(printerPageRect.size().width(),0,printerPageRect.size().width(),0); pageWidth -= printerPageRect.size().width(); if (pageWidth>0) m_printer->newPage(); } } else { renderPage.render(m_painter); } page->setPos(backupPagePos); return true; } void PrintProcessor::initPrinter(PageItemDesignIntf* page) { if (page->oldPrintMode()){ m_printer->setPageMargins(page->leftMargin(), page->topMargin(), page->rightMargin(), page->bottomMargin(), QPrinter::Millimeter); m_printer->setOrientation(static_cast(page->pageOrientation())); QSizeF pageSize = (page->pageOrientation()==PageItemDesignIntf::Landscape)? QSizeF(page->sizeMM().height(),page->sizeMM().width()): page->sizeMM(); m_printer->setPaperSize(pageSize,QPrinter::Millimeter); } else { m_printer->setFullPage(page->fullPage()); m_printer->setOrientation(static_cast(page->pageOrientation())); if (page->pageSize()==PageItemDesignIntf::Custom){ QSizeF pageSize = (page->pageOrientation()==PageItemDesignIntf::Landscape)? QSizeF(page->sizeMM().height(),page->sizeMM().width()): page->sizeMM(); if (page->getSetPageSizeToPrinter() || m_printer->outputFormat() == QPrinter::PdfFormat) m_printer->setPaperSize(pageSize,QPrinter::Millimeter); } else { if (page->getSetPageSizeToPrinter() || m_printer->outputFormat() == QPrinter::PdfFormat) m_printer->setPaperSize(static_cast(page->pageSize())); } } } }// namespace LimeReport