From 599f7ffcf396d1cea75c712787ef23994caa5693 Mon Sep 17 00:00:00 2001 From: Felix Kauselmann <2039670+selmf@users.noreply.github.com> Date: Sun, 29 Jun 2014 21:38:00 +0200 Subject: [PATCH 01/34] Add a manga reading mode which displays image in reverse order in double page mode. The mode is triggered by selecting the "21" double page icon next to the double page icon. --- YACReader/configuration.cpp | 126 ++++++++++++---------- YACReader/configuration.h | 3 + YACReader/main_window_viewer.cpp | 19 +++- YACReader/main_window_viewer.h | 1 + YACReader/render.cpp | 31 ++++-- YACReader/render.h | 2 + YACReader/viewer.cpp | 8 ++ YACReader/viewer.h | 2 + YACReader/yacreader_images_win.qrc | 1 + common/yacreader_global.h | 1 + images/viewer_toolbar/doubleMangaPage.png | Bin 0 -> 514 bytes 11 files changed, 129 insertions(+), 65 deletions(-) create mode 100644 images/viewer_toolbar/doubleMangaPage.png diff --git a/YACReader/configuration.cpp b/YACReader/configuration.cpp index a89aab3d..69b3ffe6 100644 --- a/YACReader/configuration.cpp +++ b/YACReader/configuration.cpp @@ -74,6 +74,7 @@ void Configuration::load(const QString & path) windowSize = QSize(0,0); maximized = false; doublePage = false; + doubleMangaPage = false; adjustToFullSize = false; backgroundColor = QColor(40,40,40); alwaysOnTop = false; @@ -95,65 +96,74 @@ void Configuration::load(const QString & path) else { if(name==PATH) + { defaultPath = line.trimmed(); - else - if(name==MAG_GLASS_SIZE) - { - QStringList values = line.split(','); - magnifyingGlassSize = QSize(values[0].toInt(),values[1].toInt()); - } - else - if(name==ZOOM_LEVEL) - zoomLevel = line.toFloat(); - else - if(name==SLIDE_SIZE) - { - int height = line.toInt(); - gotoSlideSize = QSize(static_cast(height/SLIDE_ASPECT_RATIO),height); - } - else - if(name==FIT) - adjustToWidth = line.toInt(); - else - if(name==FLOW_TYPE) - flowType = (FlowType)line.toInt(); - else - if(name==FULLSCREEN) - fullScreen = line.toInt(); - else - if(name==FIT_TO_WIDTH_RATIO) - fitToWidthRatio = line.toFloat(); - else - if(name==Y_WINDOW_POS) - { - QStringList l = line.split(','); - windowPos = QPoint(l[0].toInt(),l[1].toInt()); - } - else - if(name==Y_WINDOW_SIZE) - { - QStringList l = line.split(','); - windowSize = QSize(l[0].toInt(),l[1].toInt()); - } - else - if(name==MAXIMIZED) - maximized = line.toInt(); - else - if(name==DOUBLE_PAGE) - doublePage = line.toInt(); - else - if(name==ADJUST_TO_FULL_SIZE) - adjustToFullSize = line.toInt(); - else - if(name==BACKGROUND_COLOR) - { - QStringList l = line.split(','); - backgroundColor = QColor(l[0].toInt(),l[1].toInt(),l[2].toInt()); - } - else - if(name==ALWAYS_ON_TOP) - alwaysOnTop = line.toInt(); - + } + if(name==MAG_GLASS_SIZE) + { + QStringList values = line.split(','); + magnifyingGlassSize = QSize(values[0].toInt(),values[1].toInt()); + } + if(name==ZOOM_LEVEL) + { + zoomLevel = line.toFloat(); + } + if(name==SLIDE_SIZE) + { + int height = line.toInt(); + gotoSlideSize = QSize(static_cast(height/SLIDE_ASPECT_RATIO),height); + } + if(name==FIT) + { + adjustToWidth = line.toInt(); + } + if(name==FLOW_TYPE) + { + flowType = (FlowType)line.toInt(); + } + if(name==FULLSCREEN) + { + fullScreen = line.toInt(); + } + if(name==FIT_TO_WIDTH_RATIO) + { + fitToWidthRatio = line.toFloat(); + } + if(name==Y_WINDOW_POS) + { + QStringList l = line.split(','); + windowPos = QPoint(l[0].toInt(),l[1].toInt()); + } + if(name==Y_WINDOW_SIZE) + { + QStringList l = line.split(','); + windowSize = QSize(l[0].toInt(),l[1].toInt()); + } + if(name==MAXIMIZED) + { + maximized = line.toInt(); + } + if(name==DOUBLE_PAGE) + { + doublePage = line.toInt(); + } + if(name==DOUBLE_MANGA_PAGE) + { + doubleMangaPage = line.toInt(); + } + if(name==ADJUST_TO_FULL_SIZE) + { + adjustToFullSize = line.toInt(); + } + if(name==BACKGROUND_COLOR) + { + QStringList l = line.split(','); + backgroundColor = QColor(l[0].toInt(),l[1].toInt(),l[2].toInt()); + } + if(name==ALWAYS_ON_TOP) + { + alwaysOnTop = line.toInt(); + } } diff --git a/YACReader/configuration.h b/YACReader/configuration.h index 7ef6b7ff..ba9e1531 100644 --- a/YACReader/configuration.h +++ b/YACReader/configuration.h @@ -35,6 +35,7 @@ using namespace YACReader; QSize windowSize; bool maximized; bool doublePage; + bool doubleMangaPage; bool alwaysOnTop; bool adjustToFullSize; QColor backgroundColor; @@ -75,6 +76,8 @@ using namespace YACReader; void setMaximized(bool b){settings->setValue(MAXIMIZED,b);} bool getDoublePage(){return settings->value(DOUBLE_PAGE).toBool();} void setDoublePage(bool b){settings->setValue(DOUBLE_PAGE,b);} + bool getDoubleMangaPage(){return settings->value(DOUBLE_MANGA_PAGE).toBool();} + void setDoubleMangaPage(bool b){settings->setValue(DOUBLE_MANGA_PAGE,b);} bool getAdjustToFullSize(){return settings->value(ADJUST_TO_FULL_SIZE).toBool();} void setAdjustToFullSize(bool b){settings->setValue(ADJUST_TO_FULL_SIZE,b);} QColor getBackgroundColor(){return settings->value(BACKGROUND_COLOR).value();} diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index ee3d148e..7498dde9 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -92,6 +92,7 @@ MainWindowViewer::~MainWindowViewer() delete leftRotationAction; delete rightRotationAction; delete doublePageAction; + delete doubleMangaPageAction; delete goToPage; delete optionsAction; delete helpAboutAction; @@ -326,7 +327,20 @@ void MainWindowViewer::createActions() doublePageAction->setCheckable(true); doublePageAction->setChecked(Configuration::getConfiguration().getDoublePage()); connect(doublePageAction, SIGNAL(triggered()),viewer,SLOT(doublePageSwitch())); - + + //tests for inversed mode! + doubleMangaPageAction = new QAction(tr("Double page manga mode"),this); + doubleMangaPageAction->setToolTip(tr("Reverse reading order in double page mode")); + //TODO: Find a good shortcut for this + //doubleMangaPageAction->setShortcut(tr("M")); + doubleMangaPageAction->setIcon(QIcon(":/images/viewer_toolbar/doubleMangaPage.png")); + doubleMangaPageAction->setDisabled(true); + doubleMangaPageAction->setCheckable(true); + //TODO: Configuration? + doubleMangaPageAction->setChecked(Configuration::getConfiguration().getDoubleMangaPage()); + connect(doubleMangaPageAction, SIGNAL(triggered()),viewer,SLOT(doubleMangaPageSwitch())); + // + goToPage = new QAction(tr("Go To"),this); goToPage->setShortcut(tr("G")); goToPage->setIcon(QIcon(":/images/viewer_toolbar/goto.png")); @@ -502,6 +516,7 @@ void MainWindowViewer::createToolBars() comicToolBar->addAction(leftRotationAction); comicToolBar->addAction(rightRotationAction); comicToolBar->addAction(doublePageAction); + comicToolBar->addAction(doubleMangaPageAction); #ifdef Q_OS_MAC comicToolBar->addWidget(new MacToolBarSeparator); @@ -748,6 +763,7 @@ void MainWindowViewer::enableActions() rightRotationAction->setDisabled(false); showMagnifyingGlass->setDisabled(false); doublePageAction->setDisabled(false); + doubleMangaPageAction->setDisabled(false); adjustToFullSizeAction->setDisabled(false); //setBookmark->setDisabled(false); showBookmarks->setDisabled(false); @@ -768,6 +784,7 @@ void MainWindowViewer::disableActions() rightRotationAction->setDisabled(true); showMagnifyingGlass->setDisabled(true); doublePageAction->setDisabled(true); + doubleMangaPageAction->setDisabled(false); adjustToFullSizeAction->setDisabled(true); setBookmark->setDisabled(true); showBookmarks->setDisabled(true); diff --git a/YACReader/main_window_viewer.h b/YACReader/main_window_viewer.h index 6b33c434..f757a4c8 100644 --- a/YACReader/main_window_viewer.h +++ b/YACReader/main_window_viewer.h @@ -96,6 +96,7 @@ class YACReaderSliderAction; QAction *showInfo; QAction *closeAction; QAction *doublePageAction; + QAction *doubleMangaPageAction; QAction *showShorcutsAction; QAction *showDictionaryAction; QAction *alwaysOnTopAction; diff --git a/YACReader/render.cpp b/YACReader/render.cpp index 81db733b..ce91ac89 100644 --- a/YACReader/render.cpp +++ b/YACReader/render.cpp @@ -453,11 +453,20 @@ void DoublePageRender::run() QImage auxImg(totalWidth,totalHeight,QImage::Format_RGB32); QPainter painter(&auxImg); - painter.drawImage(QRect(0,0,width1,totalHeight),img); - if(!img2.isNull()) - painter.drawImage(QRect(width1,0,width2,totalHeight),img2); - painter.end(); - + if (render->doubleMangaPage) { + qDebug() << "we are in the double Manga Page tree" << render->doubleMangaPage; + painter.drawImage(QRect(width2,0,width1,totalHeight),img); + if(!img2.isNull()) + painter.drawImage(QRect(0,0,width2,totalHeight),img2); + painter.end(); + } + else { + qDebug() << "no double Manga Page" << render->doubleMangaPage; + painter.drawImage(QRect(0,0,width1,totalHeight),img); + if(!img2.isNull()) + painter.drawImage(QRect(width1,0,width2,totalHeight),img2); + painter.end(); + } if(degrees > 0) { QMatrix m; @@ -479,7 +488,7 @@ void DoublePageRender::run() //----------------------------------------------------------------------------- Render::Render() -:currentIndex(0),doublePage(false),comic(0),loadedComic(false),imageRotation(0),numLeftPages(2),numRightPages(2) +:currentIndex(0),doublePage(false),doubleMangaPage(false),comic(0),loadedComic(false),imageRotation(0),numLeftPages(2),numRightPages(2) { int size = numLeftPages+numRightPages+1; currentPageBufferedIndex = numLeftPages; @@ -1030,6 +1039,16 @@ void Render::doublePageSwitch() } } +void Render::doubleMangaPageSwitch() +{ + doubleMangaPage = !doubleMangaPage; + if(comic&&doublePage) + { + invalidate(); + update(); + } +} + QString Render::getCurrentPagesInformation() { QString s = QString::number(currentIndex+1); diff --git a/YACReader/render.h b/YACReader/render.h index 423d7955..bb466762 100644 --- a/YACReader/render.h +++ b/YACReader/render.h @@ -126,6 +126,7 @@ public slots: QPixmap * getCurrentPage(); void goTo(int index); void doublePageSwitch(); + void doubleMangaPageSwitch(); void setRotation(int degrees); void setComic(Comic * c); void prepareAvailablePage(int page); @@ -182,6 +183,7 @@ signals: private: Comic * comic; bool doublePage; + bool doubleMangaPage; int previousIndex; int currentIndex; //QPixmap * currentPage; diff --git a/YACReader/viewer.cpp b/YACReader/viewer.cpp index d90c0f95..cb47cd05 100644 --- a/YACReader/viewer.cpp +++ b/YACReader/viewer.cpp @@ -23,6 +23,7 @@ fullscreen(false), information(false), adjustToWidthRatio(1), doublePage(false), +doubleMangaPage(false), wheelStop(false), direction(1), restoreMagnifyingGlass(false), @@ -688,6 +689,13 @@ void Viewer::doublePageSwitch() Configuration::getConfiguration().setDoublePage(doublePage); } +void Viewer::doubleMangaPageSwitch() +{ + doubleMangaPage = !doubleMangaPage; + render->doubleMangaPageSwitch(); + Configuration::getConfiguration().setDoubleMangaPage(doubleMangaPage); +} + void Viewer::resetContent() { configureContent(tr("Press 'O' to open comic.")); diff --git a/YACReader/viewer.h b/YACReader/viewer.h index 4abd7ec7..e2ff3291 100644 --- a/YACReader/viewer.h +++ b/YACReader/viewer.h @@ -64,6 +64,7 @@ class NotificationsLabelWidget; void setBookmark(bool); void save(); void doublePageSwitch(); + void doubleMangaPageSwitch(); void resetContent(); void setLoadingMessage(); void setPageUnavailableMessage(); @@ -93,6 +94,7 @@ virtual void mouseReleaseEvent ( QMouseEvent * event ); private: bool information; bool doublePage; + bool doubleMangaPage; PageLabelWidget * informationLabel; //QTimer * scroller; QPropertyAnimation * verticalScroller; diff --git a/YACReader/yacreader_images_win.qrc b/YACReader/yacreader_images_win.qrc index 3476e623..e7251da4 100644 --- a/YACReader/yacreader_images_win.qrc +++ b/YACReader/yacreader_images_win.qrc @@ -3,6 +3,7 @@ ../images/viewer_toolbar/bookmark.png ../images/viewer_toolbar/close.png ../images/viewer_toolbar/doublePage.png + ../images/viewer_toolbar/doubleMangaPage.png ../images/viewer_toolbar/flow.png ../images/viewer_toolbar/full.png ../images/viewer_toolbar/goto.png diff --git a/common/yacreader_global.h b/common/yacreader_global.h index f92952c4..ee782fa4 100644 --- a/common/yacreader_global.h +++ b/common/yacreader_global.h @@ -25,6 +25,7 @@ #define Y_WINDOW_SIZE "SIZE" #define MAXIMIZED "MAXIMIZED" #define DOUBLE_PAGE "DOUBLE_PAGE" +#define DOUBLE_MANGA_PAGE "DOUBLE_MANGA_PAGE" #define ADJUST_TO_FULL_SIZE "ADJUST_TO_FULL_SIZE" #define BACKGROUND_COLOR "BACKGROUND_COLOR" #define ALWAYS_ON_TOP "ALWAYS_ON_TOP" diff --git a/images/viewer_toolbar/doubleMangaPage.png b/images/viewer_toolbar/doubleMangaPage.png new file mode 100644 index 0000000000000000000000000000000000000000..99ee1720a393be9028e011626ce9eb16bfdbeba0 GIT binary patch literal 514 zcmeAS@N?(olHy`uVBq!ia0vp^{6Ngd!2%?sn)_0KltQvckS_y6l_~>6Lo)-z&;LOB zB?CjL0RzLU1O^7H84L{K`IF+0x&hTPC3(BMF#K=tKeHdm<1FxqEM{Qf76xHPhFNnY zK!Rljj_E)eY!5>fyAse+MrKbJ#}J9jwih;Li6#mh{doWT-PnM2u3F7g^(T1cT#9)( z$&H0u(E9cg56vG;W&2uu7j>-V2;^Xiwiad4mUivY2y8j=Zl1{QoyKPk+9l8Z7n!Me zOzO&29}}H7JG$P_t4<55t+D3R|3BW-72}nm79Ln9i#8(&KE}4`8J+$+qrAmXMvK()?d09d0+EweNn2h z+JXJbkF3*;%lq(!8dS;#O_yLZ@sj$DrB4DmO4Ss{&*5FY*b5JBT7;dOH!?p zi&B9UgOP!uiLQZ}u90Pkk%5(|rIm@9wt Date: Sun, 29 Jun 2014 20:00:34 +0000 Subject: [PATCH 02/34] main_window_viewer.cpp edited online with Bitbucket --- YACReader/main_window_viewer.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index 7498dde9..fa356772 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -784,7 +784,7 @@ void MainWindowViewer::disableActions() rightRotationAction->setDisabled(true); showMagnifyingGlass->setDisabled(true); doublePageAction->setDisabled(true); - doubleMangaPageAction->setDisabled(false); + doubleMangaPageAction->setDisabled(true); adjustToFullSizeAction->setDisabled(true); setBookmark->setDisabled(true); showBookmarks->setDisabled(true); @@ -917,7 +917,7 @@ void MainWindowViewer::checkNewVersion() QTimer * tT = new QTimer; tT->setSingleShot(true); connect(tT, SIGNAL(timeout()), versionChecker, SLOT(get())); - //versionChecker->get(); //TODÓ + //versionChecker->get(); //TOD� tT->start(100); conf.setLastVersionCheck(current); @@ -1159,7 +1159,7 @@ void MainWindowViewer::alwaysOnTopSwitch() void MainWindowViewer::adjustToFullSizeSwitch() { - Configuration::getConfiguration().setAdjustToFullSize(!Configuration::getConfiguration().getAdjustToFullSize()); + Configuration::getConfiguration().setAdjustToFullSize(!ConfdoubleMangaPageAction->setDisablediguration::getConfiguration().getAdjustToFullSize()); viewer->updatePage(); } From 0cfae101cadd13f9b4f3f5cee2d8289c1affba07 Mon Sep 17 00:00:00 2001 From: Felix Kauselmann <2039670+selmf@users.noreply.github.com> Date: Sun, 29 Jun 2014 20:24:25 +0000 Subject: [PATCH 03/34] main_window_viewer.cpp edited online with Bitbucket --- YACReader/main_window_viewer.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index fa356772..f04703a0 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -1159,7 +1159,7 @@ void MainWindowViewer::alwaysOnTopSwitch() void MainWindowViewer::adjustToFullSizeSwitch() { - Configuration::getConfiguration().setAdjustToFullSize(!ConfdoubleMangaPageAction->setDisablediguration::getConfiguration().getAdjustToFullSize()); + Configuration::getConfiguration().setAdjustToFullSize(!Configuration::getConfiguration().getAdjustToFullSize()); viewer->updatePage(); } From 5a933f577e2240d34b63089468e2ba0a69e7a86a Mon Sep 17 00:00:00 2001 From: Felix Kauselmann <2039670+selmf@users.noreply.github.com> Date: Sun, 29 Jun 2014 21:23:43 +0000 Subject: [PATCH 04/34] Sync doubleMangaPage with saved config when we create connections --- YACReader/viewer.cpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/YACReader/viewer.cpp b/YACReader/viewer.cpp index cb47cd05..e8c69861 100644 --- a/YACReader/viewer.cpp +++ b/YACReader/viewer.cpp @@ -97,6 +97,9 @@ shouldOpenPrevious(false) if(Configuration::getConfiguration().getDoublePage()) doublePageSwitch(); + + if(Configuration::getConfiguration().getDoubleMangaPage()) + doubleMangaPageSwitch(); createConnections(); From 664dac34010c946db4477065abd077a7f7145bcd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 14:50:28 +0200 Subject: [PATCH 05/34] code refactoring for adding support to multiple "comics views" two "comics views" added, classic (flow+table_view) and grid (based on qml/qtquick) TODO: the views are only interchangeable in compilation time, some more work is needed for doing it in run time. fixed reloadCovers --- YACReaderLibrary/YACReaderLibrary.pro | 16 + YACReaderLibrary/classic_comics_view.cpp | 233 ++++++++++++ YACReaderLibrary/classic_comics_view.h | 49 +++ YACReaderLibrary/comics_view.cpp | 11 + YACReaderLibrary/comics_view.h | 46 +++ YACReaderLibrary/db/tablemodel.cpp | 80 ++-- YACReaderLibrary/db/tablemodel.h | 9 +- YACReaderLibrary/db_helper.cpp | 1 + YACReaderLibrary/grid_comics_view.cpp | 200 ++++++++++ YACReaderLibrary/grid_comics_view.h | 58 +++ YACReaderLibrary/library_window.cpp | 377 ++++++++----------- YACReaderLibrary/library_window.h | 13 +- YACReaderLibrary/main.cpp | 5 +- YACReaderLibrary/qml/GridComicsView.qml | 288 ++++++++++++++ YACReaderLibrary/qml/YACReaderScrollView.qml | 336 +++++++++++++++++ YACReaderLibrary/qml/page.png | Bin 0 -> 155 bytes YACReaderLibrary/qml/reading.png | Bin 0 -> 374 bytes YACReaderLibrary/qml/star.png | Bin 0 -> 242 bytes YACReaderLibrary/qml/tick.png | Bin 0 -> 488 bytes YACReaderLibrary/server/startup.cpp | 13 +- YACReaderLibrary/yacreader_local_server.cpp | 7 +- YACReaderLibrary/yacreader_local_server.h | 1 + common/yacreader_global.cpp | 8 + common/yacreader_global.h | 1 + 24 files changed, 1473 insertions(+), 279 deletions(-) create mode 100644 YACReaderLibrary/classic_comics_view.cpp create mode 100644 YACReaderLibrary/classic_comics_view.h create mode 100644 YACReaderLibrary/comics_view.cpp create mode 100644 YACReaderLibrary/comics_view.h create mode 100644 YACReaderLibrary/grid_comics_view.cpp create mode 100644 YACReaderLibrary/grid_comics_view.h create mode 100644 YACReaderLibrary/qml/GridComicsView.qml create mode 100644 YACReaderLibrary/qml/YACReaderScrollView.qml create mode 100644 YACReaderLibrary/qml/page.png create mode 100644 YACReaderLibrary/qml/reading.png create mode 100644 YACReaderLibrary/qml/star.png create mode 100644 YACReaderLibrary/qml/tick.png diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index a97ade55..ccf4566e 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -113,6 +113,9 @@ HEADERS += comic_flow.h \ ../common/http_worker.h \ yacreader_libraries.h \ ../common/exit_check.h \ + comics_view.h \ + classic_comics_view.h + SOURCES += comic_flow.cpp \ create_library_dialog.cpp \ @@ -156,6 +159,9 @@ SOURCES += comic_flow.cpp \ ../common/yacreader_global.cpp \ yacreader_libraries.cpp \ ../common/exit_check.cpp \ + comics_view.cpp \ + classic_comics_view.cpp + include(./server/server.pri) @@ -186,6 +192,16 @@ TRANSLATIONS = yacreaderlibrary_es.ts \ isEqual(QT_MAJOR_VERSION, 5) { Release:DESTDIR = ../release5 Debug:DESTDIR = ../debug5 + +#QML/GridView +QT += quick qml + +HEADERS += grid_comics_view.h + +SOURCES += grid_comics_view.cpp + +RESOURCES += qml.qrc + } else { Release:DESTDIR = ../release Debug:DESTDIR = ../debug diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp new file mode 100644 index 00000000..26ee326f --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -0,0 +1,233 @@ +#include "classic_comics_view.h" + +#include "yacreader_table_view.h" + +#include "comic_flow_widget.h" +#include "QsLog.h" + +ClassicComicsView::ClassicComicsView(QWidget *parent) + :ComicsView(parent) +{ + QHBoxLayout * layout = new QHBoxLayout; + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + //FLOW----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + + if(QGLFormat::hasOpenGL() && (settings->value(USE_OPEN_GL).toBool() == true)) + comicFlow = new ComicFlowWidgetGL(0); + else + comicFlow = new ComicFlowWidgetSW(0); + + comicFlow->updateConfig(settings); + comicFlow->setFocusPolicy(Qt::StrongFocus); + comicFlow->setShowMarks(true); + setFocusProxy(comicFlow); + + comicFlow->setFocus(Qt::OtherFocusReason); + + comicFlow->setContextMenuPolicy(Qt::ActionsContextMenu); + + //TODO!!! set actions.... + //comicFlow->addAction(toggleFullScreenAction); + //comicFlow->addAction(openComicAction); + + //END FLOW---- + + + //layout----------------------------------------------- + sVertical = new QSplitter(Qt::Vertical); //spliter derecha + + sVertical->addWidget(comicFlow); + comics = new QWidget; + QVBoxLayout * comicsLayout = new QVBoxLayout; + comicsLayout->setSpacing(0); + comicsLayout->setContentsMargins(0,0,0,0); + //TODO ComicsView:(set toolbar) comicsLayout->addWidget(editInfoToolBar); + + tableView = new YACReaderTableView; + tableView->verticalHeader()->hide(); + comicsLayout->addWidget(tableView); + comics->setLayout(comicsLayout); + sVertical->addWidget(comics); + + //config-------------------------------------------------- + if(settings->contains(COMICS_VIEW_HEADERS)) + tableView->horizontalHeader()->restoreState(settings->value(COMICS_VIEW_HEADERS).toByteArray()); + + //connections--------------------------------------------- + connect(tableView, SIGNAL(clicked(QModelIndex)), this, SLOT(centerComicFlow(QModelIndex))); + connect(comicFlow, SIGNAL(centerIndexChanged(int)), this, SLOT(updateTableView(int))); + connect(tableView, SIGNAL(comicRated(int,QModelIndex)), this, SIGNAL(comicRated(int,QModelIndex))); + connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); + connect(tableView->horizontalHeader(), SIGNAL(sectionMoved(int,int,int)), this, SLOT(saveTableHeadersStatus())); + + layout->addWidget(sVertical); + setLayout(layout); + + layout->setMargin(0); +} + +void ClassicComicsView::setToolBar(QToolBar *toolBar) +{ + static_cast(comics->layout())->insertWidget(0,toolBar); +} + +void ClassicComicsView::setModel(TableModel *model) +{ + + ComicsView::setModel(model); + + if(model == NULL) + { + comicFlow->clear(); + } + else + { + connect(model, SIGNAL(dataChanged(QModelIndex,QModelIndex,QVector)), this, SLOT(applyModelChanges(QModelIndex,QModelIndex,QVector))); + connect(model, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(removeItemsFromFlow(QModelIndex,int,int))); + + tableView->setModel(model); + + tableView->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft); + #if QT_VERSION >= 0x050000 + tableView->horizontalHeader()->setSectionsMovable(true); + #else + tableView->horizontalHeader()->setMovable(true); + #endif + //TODO parametrizar la configuración de las columnas + for(int i = 0;ihorizontalHeader()->count();i++) + tableView->horizontalHeader()->hideSection(i); + + tableView->horizontalHeader()->showSection(TableModel::Number); + tableView->horizontalHeader()->showSection(TableModel::Title); + tableView->horizontalHeader()->showSection(TableModel::FileName); + tableView->horizontalHeader()->showSection(TableModel::NumPages); + tableView->horizontalHeader()->showSection(TableModel::Hash); //Size is part of the Hash...TODO add Columns::Size to Columns + tableView->horizontalHeader()->showSection(TableModel::ReadColumn); + tableView->horizontalHeader()->showSection(TableModel::CurrentPage); + tableView->horizontalHeader()->showSection(TableModel::Rating); + + //debido a un bug, qt4 no es capaz de ajustar el ancho teniendo en cuenta todas la filas (no sólo las visibles) + //así que se ecala la primera vez y después se deja el control al usuario. + //if(!settings->contains(COMICS_VIEW_HEADERS)) + tableView->resizeColumnsToContents(); + tableView->horizontalHeader()->setStretchLastSection(true); + + QStringList paths = model->getPaths(model->getCurrentPath());//TODO ComicsView: get currentpath from somewhere currentPath()); + comicFlow->setImagePaths(paths); + comicFlow->setMarks(model->getReadList()); + comicFlow->setFocus(Qt::OtherFocusReason); + } + + if(settings->contains(COMICS_VIEW_HEADERS)) + tableView->horizontalHeader()->restoreState(settings->value(COMICS_VIEW_HEADERS).toByteArray()); + + +} + +void ClassicComicsView::setCurrentIndex(const QModelIndex &index) +{ + tableView->setCurrentIndex(index); + //TODO ComicsView: scroll comicFlow to index +} + +QModelIndex ClassicComicsView::currentIndex() +{ + return tableView->currentIndex(); +} + +QItemSelectionModel *ClassicComicsView::selectionModel() +{ + return tableView->selectionModel(); +} + +void ClassicComicsView::scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint) +{ + comicFlow->setCenterIndex(mi.row()); +} + +void ClassicComicsView::toFullScreen() +{ + comicFlow->hide(); + comicFlow->setCenterIndex(comicFlow->centerIndex()); + comics->hide(); + + //showFullScreen() //parent windows + + comicFlow->show(); + comicFlow->setFocus(Qt::OtherFocusReason); +} + +void ClassicComicsView::toNormal() +{ + comicFlow->hide(); + comicFlow->setCenterIndex(comicFlow->centerIndex()); + comicFlow->render(); + comics->show(); + comicFlow->show(); +} + +void ClassicComicsView::updateConfig(QSettings *settings) +{ + comicFlow->updateConfig(settings); +} + +void ClassicComicsView::setItemActions(const QList &actions) +{ + tableView->addActions(actions); +} + +void ClassicComicsView::setViewActions(const QList &actions) +{ + comicFlow->addActions(actions); +} + +void ClassicComicsView::setShowMarks(bool show) +{ + comicFlow->setShowMarks(show); +} + +void ClassicComicsView::centerComicFlow(const QModelIndex & mi) +{ + comicFlow->showSlide(mi.row()); + comicFlow->setFocus(Qt::OtherFocusReason); +} + +void ClassicComicsView::updateTableView(int i) +{ + QModelIndex mi = model->index(i,2); + tableView->setCurrentIndex(mi); + tableView->scrollTo(mi,QAbstractItemView::EnsureVisible); +} + +void ClassicComicsView::saveTableHeadersStatus() +{ + settings->setValue(COMICS_VIEW_HEADERS,tableView->horizontalHeader()->saveState()); +} + +void ClassicComicsView::applyModelChanges(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QVector &roles) +{ + Q_UNUSED(topLeft); + Q_UNUSED(bottomRight); + if(roles.contains(TableModel::ReadColumnRole)) + { + comicFlow->setMarks(model->getReadList()); + comicFlow->updateMarks(); + } +} + +void ClassicComicsView::removeItemsFromFlow(const QModelIndex &parent, int from, int to) +{ + Q_UNUSED(parent); + for(int i = from; i<=to; i++) + comicFlow->remove(i); +} + +void ClassicComicsView::closeEvent(QCloseEvent *event) +{ + saveTableHeadersStatus(); + ComicsView::closeEvent(event); +} + diff --git a/YACReaderLibrary/classic_comics_view.h b/YACReaderLibrary/classic_comics_view.h new file mode 100644 index 00000000..9ecb9e3e --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.h @@ -0,0 +1,49 @@ +#ifndef CLASSIC_COMICS_VIEW_H +#define CLASSIC_COMICS_VIEW_H + +#include "comics_view.h" + +#include +#include + +class YACReaderTableView; +class QSplitter; +class ComicFlowWidget; +class QToolBar; +class TableModel; + +class ClassicComicsView : public ComicsView +{ + Q_OBJECT +public: + ClassicComicsView(QWidget *parent = 0); + void setToolBar(QToolBar * toolBar); + void setModel(TableModel *model); + void setCurrentIndex(const QModelIndex &index); + QModelIndex currentIndex(); + QItemSelectionModel * selectionModel(); + void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ); + void toFullScreen(); + void toNormal(); + void updateConfig(QSettings * settings); + void setItemActions(const QList & actions); + void setViewActions(const QList & actions); + +public slots: + void centerComicFlow(const QModelIndex & mi); + void updateTableView(int i); + void saveTableHeadersStatus(); + void applyModelChanges(const QModelIndex & topLeft,const QModelIndex & bottomRight,const QVector & roles); + void removeItemsFromFlow(const QModelIndex & parent, int from, int to); + //ComicsView + void setShowMarks(bool show); +private: + YACReaderTableView * tableView; + QWidget *comics; + QSplitter * sVertical; + ComicFlowWidget * comicFlow; + QSettings * settings; + void closeEvent ( QCloseEvent * event ); +}; + +#endif // CLASSIC_COMICS_VIEW_H diff --git a/YACReaderLibrary/comics_view.cpp b/YACReaderLibrary/comics_view.cpp new file mode 100644 index 00000000..5c5e7725 --- /dev/null +++ b/YACReaderLibrary/comics_view.cpp @@ -0,0 +1,11 @@ +#include "comics_view.h" + +ComicsView::ComicsView(QWidget *parent) : + QWidget(parent),model(NULL) +{ +} + +void ComicsView::setModel(TableModel *m) +{ + model = m; +} diff --git a/YACReaderLibrary/comics_view.h b/YACReaderLibrary/comics_view.h new file mode 100644 index 00000000..4076e114 --- /dev/null +++ b/YACReaderLibrary/comics_view.h @@ -0,0 +1,46 @@ +#ifndef COMICS_VIEW_H +#define COMICS_VIEW_H + +#include + +#include "tablemodel.h" +#include +#include +#include +#include + +class YACReaderTableView; +class QSplitter; +class ComicFlowWidget; +class QToolBar; +class TableModel; +class ComicsView : public QWidget +{ + Q_OBJECT +public: + explicit ComicsView(QWidget *parent = 0); + virtual void setToolBar(QToolBar * toolBar) = 0; + virtual void setModel(TableModel *model); + virtual void setCurrentIndex(const QModelIndex &index) = 0; + virtual QModelIndex currentIndex() = 0; + virtual QItemSelectionModel * selectionModel() = 0; + virtual void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ) = 0; + virtual void toFullScreen() = 0; + virtual void toNormal() = 0; + virtual void updateConfig(QSettings * settings) = 0; + //Actions for tableviews + virtual void setItemActions(const QList & actions) = 0; + //actions for visual-oriented views + virtual void setViewActions(const QList & actions) = 0; + //virtual selectItem(int index) = 0; +signals: + void selected(unsigned int); + void comicRated(int,QModelIndex); +public slots: + virtual void setShowMarks(bool show) = 0; +protected: + TableModel * model; + +}; + +#endif // COMICS_VIEW_H diff --git a/YACReaderLibrary/db/tablemodel.cpp b/YACReaderLibrary/db/tablemodel.cpp index e04bed06..08f996cc 100644 --- a/YACReaderLibrary/db/tablemodel.cpp +++ b/YACReaderLibrary/db/tablemodel.cpp @@ -15,7 +15,7 @@ TableModel::TableModel(QObject *parent) - : QAbstractItemModel(parent) + : QAbstractItemModel(parent) { connect(this,SIGNAL(beforeReset()),this,SIGNAL(modelAboutToBeReset())); connect(this,SIGNAL(reset()),this,SIGNAL(modelReset())); @@ -23,7 +23,7 @@ TableModel::TableModel(QObject *parent) //! [0] TableModel::TableModel( QSqlQuery &sqlquery, QObject *parent) - : QAbstractItemModel(parent) + : QAbstractItemModel(parent) { setupModelData(sqlquery); } @@ -46,6 +46,27 @@ int TableModel::columnCount(const QModelIndex &parent) const } //! [2] +QHash TableModel::roleNames() const { + QHash roles; + + roles[NumberRole] = "number"; + roles[TitleRole] = "title"; + roles[FileNameRole] = "file_name"; + roles[NumPagesRole] = "num_pages"; + roles[IdRole] = "id"; + roles[Parent_IdRole] = "parent_id"; + roles[PathRole] = "path"; + roles[HashRole] = "hash"; + roles[ReadColumnRole] = "read_column"; + roles[IsBisRole] = "is_bis"; + roles[CurrentPageRole] = "current_page"; + roles[RatingRole] = "rating"; + roles[HasBeenOpenedRole] = "has_been_opened"; + roles[CoverPathRole] = "cover_path"; + + return roles; +} + //! [3] QVariant TableModel::data(const QModelIndex &index, int role) const { @@ -84,10 +105,28 @@ QVariant TableModel::data(const QModelIndex &index, int role) const //TODO check here if any view is asking for TableModel::Roles //these roles will be used from QML/GridView - if (role != Qt::DisplayRole) - return QVariant(); - TableItem *item = static_cast(index.internalPointer()); + + if (role == NumberRole) + return item->data(Number); + else if (role == TitleRole) + return item->data(Title).isNull()?item->data(FileName):item->data(Title); + else if (role == RatingRole) + return item->data(Rating); + else if (role == CoverPathRole) + return "file:///"+_databasePath+"/covers/"+item->data(Hash).toString()+".jpg"; + else if (role == NumPagesRole) + return item->data(NumPages); + else if (role == CurrentPageRole) + return item->data(CurrentPage); + else if (role == ReadColumnRole) + return item->data(ReadColumn).toBool(); + else if (role == HasBeenOpenedRole) + return item->data(TableModel::HasBeenOpened); + + if (role != Qt::DisplayRole) + return QVariant(); + if(index.column() == TableModel::Hash) return QString::number(item->data(index.column()).toString().right(item->data(index.column()).toString().length()-40).toInt()/1024.0/1024.0,'f',2)+"Mb"; if(index.column() == TableModel::ReadColumn) @@ -467,7 +506,7 @@ QVector TableModel::setComicsRead(QList l db.close(); QSqlDatabase::removeDatabase(_databasePath); - emit dataChanged(index(list.first().row(),TableModel::ReadColumn),index(list.last().row(),TableModel::CurrentPage+1)); + emit dataChanged(index(list.first().row(),TableModel::ReadColumn),index(list.last().row(),TableModel::HasBeenOpened),QVector() = {ReadColumnRole,CurrentPageRole,HasBeenOpenedRole}); return getReadList(); } @@ -553,10 +592,10 @@ void TableModel::remove(ComicDB * comic, int row) endRemoveRows(); } -ComicDB TableModel::getComic(int row) +/*ComicDB TableModel::getComic(int row) { return getComic(index(row,0)); -} +}*/ void TableModel::remove(int row) { @@ -580,8 +619,8 @@ void TableModel::reload(const ComicDB & comic) } row++; } - if(found) - emit dataChanged(index(row,TableModel::CurrentPage),index(row,TableModel::CurrentPage)); + if(found) + emit dataChanged(index(row,ReadColumn),index(row,HasBeenOpened), QVector() = {ReadColumnRole,CurrentPageRole,HasBeenOpenedRole}); } void TableModel::resetComicRating(const QModelIndex &mi) @@ -600,27 +639,6 @@ void TableModel::resetComicRating(const QModelIndex &mi) QSqlDatabase::removeDatabase(_databasePath); } -QHash TableModel::roleNames() -{ - QHash roles; - - roles[NumberRole] = "number"; - roles[TitleRole] = "title"; - roles[FileNameRole] = "file_name"; - roles[NumPagesRole] = "num_pages"; - roles[IdRole] = "id"; - roles[Parent_IdRole] = "parent_id"; - roles[PathRole] = "path"; - roles[HashRole] = "hash"; - roles[ReadColumnRole] = "read"; - roles[IsBisRole] = "is_bis"; - roles[CurrentPageRole] = "current_page"; - roles[RatingRole] = "rating"; - roles[HasBeenOpenedRole] = "has_been_opened"; - roles[CoverPathRole] = "cover_path"; - - return roles; -} void TableModel::updateRating(int rating, QModelIndex mi) { diff --git a/YACReaderLibrary/db/tablemodel.h b/YACReaderLibrary/db/tablemodel.h index 88913bc4..7bfff95b 100644 --- a/YACReaderLibrary/db/tablemodel.h +++ b/YACReaderLibrary/db/tablemodel.h @@ -24,7 +24,7 @@ public: TableModel(QObject *parent = 0); TableModel( QSqlQuery &sqlquery, QObject *parent = 0); ~TableModel(); - + QVariant data(const QModelIndex &index, int role) const; Qt::ItemFlags flags(const QModelIndex &index) const; QVariant headerData(int section, Qt::Orientation orientation, @@ -39,8 +39,9 @@ public: //Métodos de conveniencia QStringList getPaths(const QString & _source); QString getComicPath(QModelIndex mi); + QString getCurrentPath(){return QString(_databasePath).remove("/.yacreaderlibrary");}; ComicDB getComic(const QModelIndex & mi); //--> para la edición - ComicDB getComic(int row); + //ComicDB getComic(int row); QVector getReadList(); QVector setAllComicsRead(YACReaderComicReadStatus readStatus); QList getComics(QList list); //--> recupera la información común a los comics seleccionados @@ -56,7 +57,7 @@ public: void reload(const ComicDB & comic); void resetComicRating(const QModelIndex & mi); - QHash roleNames(); + QHash roleNames() const; enum Columns { Number = 0, @@ -98,6 +99,8 @@ public slots: void finishTransaction(); void updateRating(int rating, QModelIndex mi); +protected: + private: void setupModelData( QSqlQuery &sqlquery); ComicDB _getComic(const QModelIndex & mi); diff --git a/YACReaderLibrary/db_helper.cpp b/YACReaderLibrary/db_helper.cpp index 648477d4..417d8d4d 100644 --- a/YACReaderLibrary/db_helper.cpp +++ b/YACReaderLibrary/db_helper.cpp @@ -253,6 +253,7 @@ void DBHelper::update(ComicInfo * comicInfo, QSqlDatabase & db) updateComicInfo.bindValue(":notes",comicInfo->notes); bool read = comicInfo->read || comicInfo->currentPage == comicInfo->numPages.toInt(); //if current page is the las page, the comic is read(completed) + comicInfo->read = read; updateComicInfo.bindValue(":read", read?1:0); updateComicInfo.bindValue(":id", comicInfo->id); updateComicInfo.bindValue(":edited", comicInfo->edited?1:0); diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp new file mode 100644 index 00000000..c8da3b20 --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -0,0 +1,200 @@ +#include "grid_comics_view.h" + +#include +#include + +#include "QsLog.h" + +GridComicsView::GridComicsView(QWidget *parent) : + ComicsView(parent),_selectionModel(NULL) +{ + qmlRegisterType("comicModel",1,0,"TableModel"); + + view = new QQuickView(); + container = QWidget::createWindowContainer(view, this); + + container->setMinimumSize(200, 200); + container->setFocusPolicy(Qt::TabFocus); + view->setSource(QUrl("qrc:/qml/GridComicsView.qml")); + + setShowMarks(true);//TODO save this in settings + + QVBoxLayout * l = new QVBoxLayout; + l->addWidget(container); + this->setLayout(l); + + setContentsMargins(0,0,0,0); + l->setContentsMargins(0,0,0,0); + + QLOG_INFO() << "GridComicsView"; +} + +GridComicsView::~GridComicsView() +{ + delete view; +} + +void GridComicsView::setToolBar(QToolBar *toolBar) +{ +QLOG_INFO() << "setToolBar"; +static_cast(this->layout())->insertWidget(1,toolBar); +} + +void GridComicsView::setModel(TableModel *model) +{ + QLOG_INFO() << "setModel"; + + QQmlContext *ctxt = view->rootContext(); + + //there is only one mothel in the system + ComicsView::setModel(model); + if(this->model != NULL) + { + QLOG_INFO() << "xxx"; + + if(_selectionModel != NULL) + delete _selectionModel; + _selectionModel = new QItemSelectionModel(this->model); + + ctxt->setContextProperty("comicsList", this->model); + ctxt->setContextProperty("comicsSelection", _selectionModel); + ctxt->setContextProperty("comicsSelectionHelper", this); + ctxt->setContextProperty("dummyValue", true); + } + +#ifdef Q_OS_MAC + ctxt->setContextProperty("backgroundColor", "#FAFAFA"); + ctxt->setContextProperty("cellColor", "#EDEDED"); + ctxt->setContextProperty("selectedColor", "#DDDDDD"); + ctxt->setContextProperty("titleColor", "#121212"); + ctxt->setContextProperty("textColor", "#636363"); + ctxt->setContextProperty("dropShadow",true); +#else + ctxt->setContextProperty("backgroundColor", "#2A2A2A"); + ctxt->setContextProperty("cellColor", "#212121"); + ctxt->setContextProperty("selectedColor", "#121212"); + ctxt->setContextProperty("titleColor", "#E6E6E6"); + ctxt->setContextProperty("textColor", "#E6E6E6"); + ctxt->setContextProperty("dropShadow",false); +#endif + + +} + +void GridComicsView::setCurrentIndex(const QModelIndex &index) +{ + QLOG_INFO() << "setCurrentIndex"; +} + +QModelIndex GridComicsView::currentIndex() +{ + QLOG_INFO() << "currentIndex"; + QModelIndexList indexes = _selectionModel->selectedRows(); + if(indexes.length()>0) + return indexes[0]; + + this->selectIndex(0); + return _selectionModel->selectedRows()[0]; +} + +QItemSelectionModel *GridComicsView::selectionModel() +{ + QLOG_INFO() << "selectionModel"; + QModelIndexList indexes = _selectionModel->selectedRows(); + if(indexes.length()==0) + this->selectIndex(0); + + return _selectionModel; +} + +void GridComicsView::scrollTo(const QModelIndex &mi, QAbstractItemView::ScrollHint hint) +{ + QLOG_INFO() << "scrollTo"; +} + +void GridComicsView::toFullScreen() +{ + QLOG_INFO() << "toFullScreen"; +} + +void GridComicsView::toNormal() +{ + QLOG_INFO() << "toNormal"; +} + +void GridComicsView::updateConfig(QSettings *settings) +{ + QLOG_INFO() << "updateConfig"; +} + +void GridComicsView::setItemActions(const QList &actions) +{ + QLOG_INFO() << "setItemActions"; +} + +void GridComicsView::setViewActions(const QList &actions) +{ + //TODO generate QML Menu from actions + QLOG_INFO() << "setViewActions"; + this->addActions(actions); +} + +QSize GridComicsView::sizeHint() +{ + QLOG_INFO() << "sizeHint"; + return QSize(1280,768); +} + +//helper +void GridComicsView::selectIndex(int index) +{ + QLOG_INFO() << "selectIndex" << index; + if(_selectionModel != NULL && model!=NULL) + _selectionModel->select(model->index(index,0),QItemSelectionModel::Select | QItemSelectionModel::Rows); +} + +bool GridComicsView::isSelectedIndex(int index) +{ + if(_selectionModel != NULL && model!=NULL) + { + QModelIndex mi = model->index(index,0); + return _selectionModel->isSelected(mi); + } + return false; +} + +void GridComicsView::clear() +{ + QLOG_INFO() << "clear"; + if(_selectionModel != NULL) + { + _selectionModel->clear(); + + QQmlContext *ctxt = view->rootContext(); + ctxt->setContextProperty("dummyValue", true); + } + //model->forceClear(); +} + +void GridComicsView::selectedItem(int index) +{ + emit doubleClicked(model->index(index,0)); +} + +void GridComicsView::setShowMarks(bool show) +{ + QLOG_INFO() << "setShowMarks"; + QQmlContext *ctxt = view->rootContext(); + ctxt->setContextProperty("show_marks", show); +} + +void GridComicsView::closeEvent(QCloseEvent *event) +{ + QLOG_INFO() << "closeEvent"; + QObject *object = view->rootObject(); + QMetaObject::invokeMethod(object, "exit"); + container->close(); + view->close(); + event->accept(); + ComicsView::closeEvent(event); +} diff --git a/YACReaderLibrary/grid_comics_view.h b/YACReaderLibrary/grid_comics_view.h new file mode 100644 index 00000000..1012679f --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.h @@ -0,0 +1,58 @@ +#ifndef GRID_COMICS_VIEW_H +#define GRID_COMICS_VIEW_H + +#include "comics_view.h" + +#include + +class QAbstractListModel; +class QItemSelectionModel; +class QQuickView; +class QQuickView; + + +class GridComicsView : public ComicsView +{ + Q_OBJECT +public: + explicit GridComicsView(QWidget *parent = 0); + virtual ~GridComicsView(); + void setToolBar(QToolBar * toolBar); + void setModel(TableModel *model); + void setCurrentIndex(const QModelIndex &index); + QModelIndex currentIndex(); + QItemSelectionModel * selectionModel(); + void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ); + void toFullScreen(); + void toNormal(); + void updateConfig(QSettings * settings); + void setItemActions(const QList & actions); + void setViewActions(const QList & actions); + + QSize sizeHint(); +signals: +signals: + void comicRated(int,QModelIndex); + void doubleClicked(QModelIndex); + +public slots: + //selection helper + void selectIndex(int index); + bool isSelectedIndex(int index); + void clear(); + //double clicked item + void selectedItem(int index); + + //ComicsView + void setShowMarks(bool show); + +private: + QItemSelectionModel * _selectionModel; + QQuickView *view; + QWidget *container; + bool dummy; + void closeEvent ( QCloseEvent * event ); + +}; + +#endif // GRID_COMICS_VIEW_H diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 66debab4..b6e32a6b 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -60,6 +60,9 @@ #include "comic_vine_dialog.h" //#include "yacreader_social_dialog.h" +#include "classic_comics_view.h" +#include "grid_comics_view.h" + #include "QsLog.h" #ifdef Q_OS_WIN @@ -118,10 +121,9 @@ void LibraryWindow::setupUI() else //if(settings->value(USE_OPEN_GL).toBool() == false) showMaximized(); - if(settings->contains(COMICS_VIEW_HEADERS)) - comicView->horizontalHeader()->restoreState(settings->value(COMICS_VIEW_HEADERS).toByteArray()); + /*if(settings->contains(COMICS_VIEW_HEADERS_GEOMETRY)) - comicView->horizontalHeader()->restoreGeometry(settings->value(COMICS_VIEW_HEADERS_GEOMETRY).toByteArray());*/ + comicsView->horizontalHeader()->restoreGeometry(settings->value(COMICS_VIEW_HEADERS_GEOMETRY).toByteArray());*/ /*socialDialog = new YACReaderSocialDialog(this); socialDialog->setHidden(true);*/ @@ -131,7 +133,7 @@ void LibraryWindow::doLayout() { //LAYOUT ELEMENTS------------------------------------------------------------ //--------------------------------------------------------------------------- - sVertical = new QSplitter(Qt::Vertical); //spliter derecha + QSplitter * sHorizontal = new QSplitter(Qt::Horizontal); //spliter principal #ifdef Q_OS_MAC sHorizontal->setStyleSheet("QSplitter::handle{image:none;background-color:#B8B8B8;} QSplitter::handle:vertical {height:1px;}"); @@ -142,49 +144,29 @@ void LibraryWindow::doLayout() //TOOLBARS------------------------------------------------------------------- //--------------------------------------------------------------------------- editInfoToolBar = new QToolBar(); + editInfoToolBar->setStyleSheet("QToolBar {border: none;}"); + #ifdef Q_OS_MAC libraryToolBar = addToolBar(tr("Library")); #else libraryToolBar = new YACReaderMainToolBar(this); #endif - //FLOW----------------------------------------------------------------------- - //--------------------------------------------------------------------------- - if(QGLFormat::hasOpenGL() && !settings->contains(USE_OPEN_GL)) - { - OnStartFlowSelectionDialog * flowSelDialog = new OnStartFlowSelectionDialog(); - flowSelDialog->exec(); - if(flowSelDialog->result() == QDialog::Accepted) - settings->setValue(USE_OPEN_GL,2); - else - settings->setValue(USE_OPEN_GL,0); + //FLOW----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + if(QGLFormat::hasOpenGL() && !settings->contains(USE_OPEN_GL)) + { + OnStartFlowSelectionDialog * flowSelDialog = new OnStartFlowSelectionDialog(); - delete flowSelDialog; - } + flowSelDialog->exec(); + if(flowSelDialog->result() == QDialog::Accepted) + settings->setValue(USE_OPEN_GL,2); + else + settings->setValue(USE_OPEN_GL,0); - if(QGLFormat::hasOpenGL() && (settings->value(USE_OPEN_GL).toBool() == true)) - comicFlow = new ComicFlowWidgetGL(0); - else - comicFlow = new ComicFlowWidgetSW(0); - - comicFlow->updateConfig(settings); - comicFlow->setFocusPolicy(Qt::StrongFocus); - comicFlow->setShowMarks(true); - setFocusProxy(comicFlow); - - fullScreenToolTip = new QLabel(comicFlow); - fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); - fullScreenToolTip->setPalette(QPalette(QColor(0,0,0))); - fullScreenToolTip->setFont(QFont("courier new",15,234)); - fullScreenToolTip->setAutoFillBackground(true); - fullScreenToolTip->hide(); - fullScreenToolTip->adjustSize(); - - comicFlow->setFocus(Qt::OtherFocusReason); - - comicFlow->addAction(toggleFullScreenAction); - comicFlow->addAction(openComicAction); + delete flowSelDialog; + } //SIDEBAR----------------------------------------------------------------------- //--------------------------------------------------------------------------- @@ -207,25 +189,23 @@ void LibraryWindow::doLayout() foldersTitle->addAction(colapseAllNodesAction); //FINAL LAYOUT------------------------------------------------------------- - sVertical->addWidget(comicFlow); - comics = new QWidget; - QVBoxLayout * comicsLayout = new QVBoxLayout; - comicsLayout->setSpacing(0); - comicsLayout->setContentsMargins(0,0,0,0); - comicsLayout->addWidget(editInfoToolBar); - editInfoToolBar->setStyleSheet("QToolBar {border: none;}"); - - comicView = new YACReaderTableView; - comicView->verticalHeader()->hide(); - comicsLayout->addWidget(comicView); - comics->setLayout(comicsLayout); - sVertical->addWidget(comics); + comicsView = new ClassicComicsView(); + comicsView->setToolBar(editInfoToolBar); + + fullScreenToolTip = new QLabel(comicsView); + fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); + fullScreenToolTip->setPalette(QPalette(QColor(0,0,0))); + fullScreenToolTip->setFont(QFont("courier new",15,234)); + fullScreenToolTip->setAutoFillBackground(true); + fullScreenToolTip->hide(); + fullScreenToolTip->adjustSize(); + sHorizontal->addWidget(sideBar); #ifndef Q_OS_MAC QVBoxLayout * rightLayout = new QVBoxLayout; rightLayout->addWidget(libraryToolBar); - rightLayout->addWidget(sVertical); + rightLayout->addWidget(comicsView); rightLayout->setMargin(0); rightLayout->setSpacing(0); @@ -257,10 +237,7 @@ void LibraryWindow::doLayout() connect(noLibrariesWidget,SIGNAL(createNewLibrary()),this,SLOT(createLibrary())); connect(noLibrariesWidget,SIGNAL(addExistingLibrary()),this,SLOT(showAddLibrary())); - comicFlow->addAction(toggleFullScreenAction); - comicFlow->addAction(openComicAction); - comicFlow->setContextMenuPolicy(Qt::ActionsContextMenu); //collapsible disabled in macosx (only temporaly) #ifdef Q_OS_MAC @@ -489,11 +466,11 @@ void LibraryWindow::createActions() deleteComicsAction->setText(tr("Delete selected comics")); deleteComicsAction->setIcon(QIcon(":/images/trash.png")); - hideComicViewAction = new QAction(this); - hideComicViewAction->setText(tr("Hide comic flow")); - hideComicViewAction->setIcon(QIcon(":/images/hideComicFlow.png")); - hideComicViewAction->setCheckable(true); - hideComicViewAction->setChecked(false); + hideComicViewAction = new QAction(this); + hideComicViewAction->setText(tr("Hide comic flow")); + hideComicViewAction->setIcon(QIcon(":/images/hideComicFlow.png")); + hideComicViewAction->setCheckable(true); + hideComicViewAction->setChecked(false); getInfoAction = new QAction(this); getInfoAction->setText(tr("Download tags from Comic Vine")); @@ -634,28 +611,49 @@ void LibraryWindow::createToolBars() editInfoToolBar->addAction(deleteComicsAction); editInfoToolBar->addWidget(new QToolBarStretch()); - editInfoToolBar->addAction(hideComicViewAction); + editInfoToolBar->addAction(hideComicViewAction); } void LibraryWindow::createMenus() { - comicView->addAction(openContainingFolderComicAction); - YACReader::addSperator(comicView); + QList itemActions; + itemActions << openContainingFolderComicAction + << YACReader::createSeparator() + << resetComicRatingAction + << YACReader::createSeparator() + << editSelectedComicsAction + << getInfoAction + << asignOrderActions + << YACReader::createSeparator() + << setAsReadAction + << setAsNonReadAction + << YACReader::createSeparator() + << deleteComicsAction; - comicView->addAction(resetComicRatingAction); - YACReader::addSperator(comicView); + QList viewActions; + viewActions << openComicAction + << YACReader::createSeparator() + << openContainingFolderComicAction + << YACReader::createSeparator() + << resetComicRatingAction + << YACReader::createSeparator() + << editSelectedComicsAction + << getInfoAction + << asignOrderActions + << YACReader::createSeparator() + << selectAllComicsAction + << YACReader::createSeparator() + << setAsReadAction + << setAsNonReadAction + << showHideMarksAction + << YACReader::createSeparator() + << deleteComicsAction + << YACReader::createSeparator() + << toggleFullScreenAction; - comicView->addAction(editSelectedComicsAction); - comicView->addAction(getInfoAction); - comicView->addAction(asignOrderActions); - YACReader::addSperator(comicView); - - comicView->addAction(setAsReadAction); - comicView->addAction(setAsNonReadAction); - YACReader::addSperator(comicView); - - comicView->addAction(deleteComicsAction); + comicsView->setItemActions(itemActions); + comicsView->setViewActions(viewActions); foldersView->addAction(openContainingFolderAction); YACReader::addSperator(foldersView); @@ -678,6 +676,9 @@ void LibraryWindow::createMenus() selectedLibrary->addAction(exportLibraryAction); selectedLibrary->addAction(importLibraryAction); + + + //MacOSX app menus #ifdef Q_OS_MACX @@ -776,9 +777,8 @@ void LibraryWindow::createConnections() connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(loadCovers(QModelIndex))); connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(updateHistory(QModelIndex))); - connect(comicView, SIGNAL(clicked(QModelIndex)), this, SLOT(centerComicFlow(QModelIndex))); - connect(comicFlow, SIGNAL(centerIndexChanged(int)), this, SLOT(updateComicView(int))); - connect(comicView, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + connect(comicsView, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + connect(showHideMarksAction,SIGNAL(toggled(bool)),comicsView,SLOT(setShowMarks(bool))); //actions connect(createLibraryAction,SIGNAL(triggered()),this,SLOT(createLibrary())); @@ -791,8 +791,7 @@ void LibraryWindow::createConnections() //connect(setAllAsReadAction,SIGNAL(triggered()),this,SLOT(setComicsReaded())); //connect(setAllAsNonReadAction,SIGNAL(triggered()),this,SLOT(setComicsUnreaded())); - connect(showHideMarksAction,SIGNAL(toggled(bool)),comicFlow,SLOT(setShowMarks(bool))); - + //comicsInfoManagement connect(exportComicsInfo,SIGNAL(triggered()),this,SLOT(showExportComicsInfo())); connect(importComicsInfo,SIGNAL(triggered()),this,SLOT(showImportComicsInfo())); @@ -819,8 +818,8 @@ void LibraryWindow::createConnections() #endif connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); //ComicFlow - connect(comicFlow,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); - connect(comicView,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + connect(comicsView,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); + connect(comicsView,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); //Folders filter //connect(clearFoldersFilter,SIGNAL(clicked()),foldersFilter,SLOT(clear())); connect(foldersFilter,SIGNAL(textChanged(QString)),this,SLOT(setFoldersFilter(QString))); @@ -838,13 +837,13 @@ void LibraryWindow::createConnections() //connect(dm,SIGNAL(directoryLoaded(QString)),foldersView,SLOT(expandAll())); //connect(dm,SIGNAL(directoryLoaded(QString)),this,SLOT(updateFoldersView(QString))); //Comicts edition - connect(selectAllComicsAction,SIGNAL(triggered()),comicView,SLOT(selectAll())); + connect(selectAllComicsAction,SIGNAL(triggered()),comicsView,SLOT(selectAll())); connect(editSelectedComicsAction,SIGNAL(triggered()),this,SLOT(showProperties())); connect(asignOrderActions,SIGNAL(triggered()),this,SLOT(asignNumbers())); connect(deleteComicsAction,SIGNAL(triggered()),this,SLOT(deleteComics())); - connect(hideComicViewAction, SIGNAL(toggled(bool)),this, SLOT(hideComicFlow(bool))); + connect(hideComicViewAction, SIGNAL(toggled(bool)),this, SLOT(hideComicFlow(bool))); connect(getInfoAction,SIGNAL(triggered()),this,SLOT(showComicVineScraper())); @@ -875,16 +874,14 @@ void LibraryWindow::loadLibrary(const QString & name) int ret = QMessageBox::question(this,tr("Update needed"),tr("This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now?"),QMessageBox::Yes,QMessageBox::No); if(ret == QMessageBox::Yes) { - //TODO update to new version updated = DataBaseManagement::updateToCurrentVersion(path+"/library.ydb"); if(!updated) QMessageBox::critical(this,tr("Update failed"), tr("The current library can't be udpated. Check for write write permissions on: ") + path+"/library.ydb"); } else { - comicView->setModel(NULL); + comicsView->setModel(NULL); foldersView->setModel(NULL); - comicFlow->clear(); disableAllActions();//TODO comprobar que se deben deshabilitar //será posible renombrar y borrar estas bibliotecas renameLibraryAction->setEnabled(true); @@ -935,9 +932,8 @@ void LibraryWindow::loadLibrary(const QString & name) if(ret == QMessageBox::Yes) QDesktopServices::openUrl(QUrl("http://www.yacreader.com")); - comicView->setModel(NULL); + comicsView->setModel(NULL); foldersView->setModel(NULL); - comicFlow->clear(); disableAllActions();//TODO comprobar que se deben deshabilitar //será posible renombrar y borrar estas bibliotecas renameLibraryAction->setEnabled(true); @@ -946,9 +942,8 @@ void LibraryWindow::loadLibrary(const QString & name) } else { - comicView->setModel(NULL); + comicsView->setModel(NULL); foldersView->setModel(NULL); - comicFlow->clear(); disableAllActions();//TODO comprobar que se deben deshabilitar //si la librería no existe en disco, se ofrece al usuario la posibiliad de eliminarla @@ -1035,45 +1030,15 @@ void LibraryWindow::loadCovers(const QModelIndex & mi) column = mi.column(); } - //comicView->setModel(NULL); + //comicsView->setModel(NULL); dmCV->setupModelData(folderId,dm->getDatabase()); - comicView->setModel(dmCV); - comicView->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft); -#if QT_VERSION >= 0x050000 - comicView->horizontalHeader()->setSectionsMovable(true); -#else - comicView->horizontalHeader()->setMovable(true); -#endif - //TODO parametrizar la configuración de las columnas - for(int i = 0;ihorizontalHeader()->count();i++) - comicView->horizontalHeader()->hideSection(i); - - comicView->horizontalHeader()->showSection(0); - comicView->horizontalHeader()->showSection(1); - comicView->horizontalHeader()->showSection(2); - comicView->horizontalHeader()->showSection(3); - comicView->horizontalHeader()->showSection(7); - comicView->horizontalHeader()->showSection(8); - comicView->horizontalHeader()->showSection(10); - comicView->horizontalHeader()->showSection(11); - - - //debido a un bug, qt4 no es capaz de ajustar el ancho teniendo en cuenta todas la filas (no sólo las visibles) - //así que se ecala la primera vez y después se deja el control al usuario. - //if(!settings->contains(COMICS_VIEW_HEADERS)) - comicView->resizeColumnsToContents(); - comicView->horizontalHeader()->setStretchLastSection(true); - - QStringList paths = dmCV->getPaths(currentPath()); - comicFlow->setImagePaths(paths); - comicFlow->setMarks(dmCV->getReadList()); - comicFlow->setFocus(Qt::OtherFocusReason); - + comicsView->setModel(dmCV); + QStringList paths = dmCV->getPaths(currentPath()); checkEmptyFolder(&paths); if(paths.size()>0) - comicView->setCurrentIndex(dmCV->index(0,0)); + comicsView->setCurrentIndex(dmCV->index(0,0)); } void LibraryWindow::checkEmptyFolder(QStringList * paths) @@ -1098,33 +1063,22 @@ void LibraryWindow::checkEmptyFolder(QStringList * paths) void LibraryWindow::reloadCovers() { - loadCovers(foldersView->currentIndex()); - + if(foldersView->selectionModel()->selectedRows().length()>0) + loadCovers(foldersView->currentIndex()); + else + loadCovers(QModelIndex()); +QLOG_INFO() << "reloaded covers at row : " << foldersView->currentIndex().row(); QModelIndex mi = dmCV->getIndexFromId(_comicIdEdited); - comicView->scrollTo(mi,QAbstractItemView::PositionAtCenter); - comicView->setCurrentIndex(mi); + comicsView->scrollTo(mi,QAbstractItemView::PositionAtCenter); + comicsView->setCurrentIndex(mi); //centerComicFlow(mi); - comicFlow->setCenterIndex(mi.row()); -} - -void LibraryWindow::centerComicFlow(const QModelIndex & mi) -{ - comicFlow->showSlide(mi.row()); - comicFlow->setFocus(Qt::OtherFocusReason); -} - -void LibraryWindow::updateComicView(int i) -{ - QModelIndex mi = dmCV->index(i,2); - comicView->setCurrentIndex(mi); - comicView->scrollTo(mi,QAbstractItemView::EnsureVisible); } void LibraryWindow::openComic() { if(!importedCovers) { - ComicDB comic = dmCV->getComic(comicView->currentIndex()); + ComicDB comic = dmCV->getComic(comicsView->currentIndex()); QString path = currentPath(); QList siblings = dmCV->getAllComics(); @@ -1159,42 +1113,24 @@ void LibraryWindow::openComic() } } -void LibraryWindow::setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus) -{ - - comicFlow->setMarks(dmCV->setComicsRead(getSelectedComics(),readStatus)); - comicFlow->updateMarks(); +void LibraryWindow::setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus) { + dmCV->setComicsRead(getSelectedComics(),readStatus); } -void LibraryWindow::setCurrentComicReaded() -{ +void LibraryWindow::setCurrentComicReaded() { this->setCurrentComicsStatusReaded(YACReader::Read); } void LibraryWindow::setCurrentComicOpened() { - + //TODO: remove? } -void LibraryWindow::setComicsReaded() -{ - comicFlow->setMarks(dmCV->setAllComicsRead(YACReader::Read)); - comicFlow->updateMarks(); -} - -void LibraryWindow::setCurrentComicUnreaded() -{ +void LibraryWindow::setCurrentComicUnreaded() { this->setCurrentComicsStatusReaded(YACReader::Unread); } -void LibraryWindow::setComicsUnreaded() -{ - comicFlow->setMarks(dmCV->setAllComicsRead(YACReader::Unread)); - comicFlow->updateMarks(); -} - -void LibraryWindow::createLibrary() -{ +void LibraryWindow::createLibrary() { createLibraryDialog->show(libraries); } @@ -1211,8 +1147,7 @@ void LibraryWindow::create(QString source, QString dest, QString name) } -void LibraryWindow::reloadCurrentLibrary() -{ +void LibraryWindow::reloadCurrentLibrary() { loadLibrary(selectedLibrary->currentText()); } @@ -1268,24 +1203,8 @@ void LibraryWindow::loadLibraries() } -void LibraryWindow::saveLibraries() -{ - +void LibraryWindow::saveLibraries() { libraries.save(); - /*QFile f(QCoreApplication::applicationDirPath()+"/libraries.yacr"); - if(!f.open(QIODevice::WriteOnly)) - { - QMessageBox::critical(NULL,tr("Saving libraries file...."),tr("There was a problem saving YACReaderLibrary libraries file. Please, check if you have enough permissions in the YACReader root folder.")); - } - else - { - QTextStream txtS(&f); - for(QMap::iterator i = libraries.begin();i!=libraries.end();i++) - { - txtS << i.key() << "\n"; - txtS << i.value() << "\n"; - } - }*/ } void LibraryWindow::updateLibrary() @@ -1313,9 +1232,9 @@ void LibraryWindow::deleteCurrentLibrary() d.rmdir(path); if(libraries.isEmpty())//no more libraries avaliable. { - comicView->setModel(NULL); + comicsView->setModel(NULL); foldersView->setModel(NULL); - comicFlow->clear(); + disableAllActions(); showNoLibrariesWidget(); } @@ -1335,9 +1254,9 @@ void LibraryWindow::removeLibrary() //selectedLibrary->setCurrentIndex(0); if(libraries.isEmpty())//no more libraries avaliable. { - comicView->setModel(NULL); + comicsView->setModel(NULL); foldersView->setModel(NULL); - comicFlow->clear(); + disableAllActions(); showNoLibrariesWidget(); } @@ -1408,11 +1327,10 @@ void LibraryWindow::setRootIndex() } else { - comicView->setModel(NULL); - comicFlow->clear(); + comicsView->setModel(NULL); } - foldersView->clearSelection(); + foldersView->selectionModel()->clear(); } setFolderAsNotCompletedAction->setVisible(false); @@ -1432,17 +1350,12 @@ void LibraryWindow::toFullScreen() { fromMaximized = this->isMaximized(); - comicFlow->hide(); - //comicFlow->setSlideSize(slideSizeF); - comicFlow->setCenterIndex(comicFlow->centerIndex()); - comics->hide(); - sideBar->hide(); + sideBar->hide(); libraryToolBar->hide(); - showFullScreen(); + comicsView->toFullScreen(); - comicFlow->show(); - comicFlow->setFocus(Qt::OtherFocusReason); + showFullScreen(); fullScreenToolTip->move((width()-fullScreenToolTip->width())/2,0); fullScreenToolTip->adjustSize(); @@ -1452,14 +1365,10 @@ void LibraryWindow::toFullScreen() void LibraryWindow::toNormal() { fullScreenToolTip->hide(); - comicFlow->hide(); - //comicFlow->setSlideSize(slideSizeW); - comicFlow->setCenterIndex(comicFlow->centerIndex()); - comicFlow->render(); - comics->show(); + sideBar->show(); - comicFlow->show(); + comicsView->toNormal(); if(fromMaximized) showMaximized(); @@ -1580,7 +1489,7 @@ void LibraryWindow::asignNumbers() void LibraryWindow::openContainingFolderComic() { -QModelIndex modelIndex = comicView->currentIndex(); +QModelIndex modelIndex = comicsView->currentIndex(); QFileInfo file = QDir::cleanPath(currentPath() + dmCV->getComicPath(modelIndex)); #ifdef Q_OS_LINUX QString path = file.absolutePath(); @@ -1656,7 +1565,7 @@ void LibraryWindow::importLibrary(QString clc,QString destPath,QString name) void LibraryWindow::reloadOptions() { //comicFlow->setFlowType(flowType); - comicFlow->updateConfig(settings); + comicsView->updateConfig(settings); } QString LibraryWindow::currentPath() @@ -1664,8 +1573,11 @@ QString LibraryWindow::currentPath() return libraries.getPath(selectedLibrary->currentText()); } +//TODO ComicsView: some actions in the comics toolbar can be relative to a certain view +//show/hide actions on show/hide widget void LibraryWindow::hideComicFlow(bool hide) { + /* if(hide) { QList sizes; @@ -1682,7 +1594,7 @@ void LibraryWindow::hideComicFlow(bool hide) sizes.append(total/3); sVertical->setSizes(sizes); } - +*/ } void LibraryWindow::showExportComicsInfo() @@ -1696,13 +1608,17 @@ void LibraryWindow::showImportComicsInfo() importComicsInfoDialog->dest = currentPath() + "/.yacreaderlibrary/library.ydb"; importComicsInfoDialog->show(); } - +#include "startup.h" +extern Startup * s; void LibraryWindow::closeEvent ( QCloseEvent * event ) { - settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry()); - settings->setValue(COMICS_VIEW_HEADERS,comicView->horizontalHeader()->saveState()); - event->accept(); - //settings->setValue(COMICS_VIEW_HEADERS_GEOMETRY,comicView->horizontalHeader()->saveGeometry()); + s->stop(); + settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry()); + + comicsView->close(); + QApplication::instance()->processEvents(); + event->accept(); + QMainWindow::closeEvent(event); } void LibraryWindow::showNoLibrariesWidget() @@ -1751,15 +1667,16 @@ bool lessThanModelIndexRow(const QModelIndex & m1, const QModelIndex & m2) QModelIndexList LibraryWindow::getSelectedComics() { //se fuerza a que haya almenos una fila seleccionada TODO comprobar se se puede forzar a la tabla a que lo haga automáticamente - QModelIndexList selection = comicView->selectionModel()->selectedRows(); - + //avoid selection.count()==0 forcing selection in comicsView + QModelIndexList selection = comicsView->selectionModel()->selectedRows(); + QLOG_INFO() << "selection count " << selection.length(); qSort(selection.begin(),selection.end(),lessThanModelIndexRow); - if(selection.count()==0) + /*if(selection.count()==0) { - comicView->selectRow(comicFlow->centerIndex()); - selection = comicView->selectionModel()->selectedRows(); - } + comicsView->selectRow(comicFlow->centerIndex()); + selection = comicsView->selectionModel()->selectedRows(); + }*/ return selection; } @@ -1778,19 +1695,21 @@ void LibraryWindow::deleteComics() QString libraryPath = currentPath(); foreach(ComicDB comic, comics) { - paths.append(libraryPath + comic.path); + paths.append(libraryPath + comic.path); + QLOG_INFO() << comic.path; + QLOG_INFO() << comic.id; + QLOG_INFO() << comic.parentId; } ComicsRemover * remover = new ComicsRemover(indexList,paths); - //comicView->showDeleteProgress(); + //comicsView->showDeleteProgress(); dmCV->startTransaction(); - connect(remover, SIGNAL(remove(int)), dmCV, SLOT(remove(int))); - connect(remover, SIGNAL(remove(int)), comicFlow, SLOT(remove(int))); + connect(remover, SIGNAL(remove(int)), dmCV, SLOT(remove(int))); connect(remover,SIGNAL(removeError()),this,SLOT(setRemoveError())); connect(remover, SIGNAL(finished()), dmCV, SLOT(finishTransaction())); - //connect(remover, SIGNAL(finished()), comicView, SLOT(hideDeleteProgress())); + //connect(remover, SIGNAL(finished()), comicsView, SLOT(hideDeleteProgress())); connect(remover, SIGNAL(finished()),this,SLOT(checkEmptyFolder())); connect(remover, SIGNAL(finished()),this,SLOT(checkRemoveError())); connect(remover, SIGNAL(finished()), remover, SLOT(deleteLater())); @@ -1895,9 +1814,7 @@ void LibraryWindow::importLibraryPackage() void LibraryWindow::updateComicsView(quint64 libraryId, const ComicDB & comic) { //TODO comprobar la biblioteca.... - if(libraryId == selectedLibrary->currentIndex()) - { + if(libraryId == selectedLibrary->currentIndex()) { dmCV->reload(comic); - comicFlow->setMarks(dmCV->getReadList()); } } diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 5925d897..d100212c 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -27,7 +27,6 @@ class HelpAboutDialog; class RenameLibraryDialog; class PropertiesDialog; class PackageManager; -class ComicFlowWidget; class QCheckBox; class QPushButton; class TableModel; @@ -50,6 +49,7 @@ class YACReaderLibraryListWidget; class YACReaderTreeView; class YACReaderMainToolBar; class ComicVineDialog; +class ComicsView; #include "comic_db.h" using namespace YACReader; @@ -59,7 +59,7 @@ class LibraryWindow : public QMainWindow Q_OBJECT private: YACReaderSideBar * sideBar; - QSplitter * sVertical; + CreateLibraryDialog * createLibraryDialog; ExportLibraryDialog * exportLibraryDialog; ImportLibraryDialog * importLibraryDialog; @@ -80,7 +80,6 @@ private: //YACReaderSortComics * proxySort; PackageManager * packageManager; - ComicFlowWidget * comicFlow; QSize slideSizeW; QSize slideSizeF; //search filter @@ -91,8 +90,8 @@ private: QPushButton * clearFoldersFilter; QCheckBox * includeComicsCheckBox; //------------- - QWidget *comics; - YACReaderTableView * comicView; + + ComicsView * comicsView; YACReaderTreeView * foldersView; YACReaderLibraryListWidget * selectedLibrary; TreeModel * dm; @@ -224,8 +223,6 @@ public: void loadCovers(const QModelIndex & mi); void checkEmptyFolder(QStringList * paths = 0); void reloadCovers(); - void centerComicFlow(const QModelIndex & mi); - void updateComicView(int i); void openComic(); void createLibrary(); void create(QString source,QString dest, QString name); @@ -261,8 +258,6 @@ public: void setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus); void setCurrentComicReaded(); void setCurrentComicUnreaded(); - void setComicsReaded(); - void setComicsUnreaded(); void hideComicFlow(bool hide); void showExportComicsInfo(); void showImportComicsInfo(); diff --git a/YACReaderLibrary/main.cpp b/YACReaderLibrary/main.cpp index 951356fb..1b90312a 100644 --- a/YACReaderLibrary/main.cpp +++ b/YACReaderLibrary/main.cpp @@ -213,9 +213,12 @@ int main( int argc, char ** argv ) YACReader::exitCheck(ret); - //server shutdown + //shutdown s->stop(); delete s; + localServer->close(); + delete localServer; + delete mw; QsLogging::Logger::destroyInstance(); diff --git a/YACReaderLibrary/qml/GridComicsView.qml b/YACReaderLibrary/qml/GridComicsView.qml new file mode 100644 index 00000000..c3e4e515 --- /dev/null +++ b/YACReaderLibrary/qml/GridComicsView.qml @@ -0,0 +1,288 @@ +import QtQuick 2.3 +import QtQuick.Controls 1.0 +import QtQuick.Controls 1.1 +import QtGraphicalEffects 1.0 +import comicModel 1.0 + +Rectangle { + id: main + color: backgroundColor + width: parent.width + height: parent.height + anchors.margins: 0 + + function selectAll(from,to) + { + for(var i = from+1;i ci) + selectAll(ci,index); + + mouse.accepted = true; + + comicsSelectionHelper.selectIndex(index) + grid.currentIndex = index; + + } + } + + + Menu { + id: myContextMenu + MenuItem { text: "Open containing folder..."; onTriggered: display.setRandomName() } + MenuSeparator{} + MenuItem { text: "Reset comic rating"; onTriggered: model1.removeRows(index, 1) } + MenuSeparator{} + MenuItem { text: "Edit"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Download tags from Comic Vine"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Asign current order to comics"; onTriggered: model1.removeRows(index, 1) } + MenuSeparator{} + MenuItem { text: "Set as read"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Set as unread"; onTriggered: model1.removeRows(index, 1) } + MenuSeparator{} + MenuItem { text: "Delete selected comics"; onTriggered: model1.removeRows(index, 1) } + //MenuItem { text: "Show details"; onTriggered: cell.state = 'Details'; + } + + + } + + DropShadow { + anchors.fill: source + horizontalOffset: 0 + verticalOffset: 0 + radius: 3 + samples: 24 + color: "#40000000" + transparentBorder: true; + source: realCell; + enabled: dropShadow; + visible: dropShadow; + } + + /**/ + + //cover + Image { + id: coverElement + width: 148 + height: 224 + anchors {horizontalCenter: parent.horizontalCenter; top: realCell.top; topMargin: 4} + source: cover_path + fillMode: Image.PreserveAspectCrop + //smooth: true + mipmap: true + //antialiasing: true + asynchronous : true + cache: false //TODO clear cache only when it is neede + } + //mark + Image { + id: mark + width: 23 + height: 23 + source: read_column&&show_marks?"tick.png":has_been_opened&&show_marks?"reading.png":"" + anchors {right: coverElement.right; top: coverElement.top; topMargin: 11; rightMargin: 11} + asynchronous : true + } + + //title + Text { + anchors { top: realCell.top; left: realCell.left; leftMargin: 4; rightMargin: 4; topMargin: 234; } + width: 148 + maximumLineCount: 2 + wrapMode: Text.WordWrap + text: title + elide: Text.ElideRight + color: titleColor + clip: true + font.letterSpacing: 0.5 + } + //number + Text { + anchors {bottom: realCell.bottom; left: realCell.left; margins: 4} + text: number?"#"+number:"" + color: textColor + font.letterSpacing: 0.5 + } + //page icon + Image { + id: pageImage + anchors {bottom: realCell.bottom; right: realCell.right; bottomMargin: 5; rightMargin: 4; leftMargin: 4} + source: "page.png" + } + //numPages + Text { + id: pages + anchors {bottom: realCell.bottom; right: pageImage.left; margins: 4} + text: has_been_opened?current_page+"/"+num_pages:num_pages + color: textColor + font.letterSpacing: 0.5 + } + //rating icon + Image { + id: ratingImage + anchors {bottom: realCell.bottom; right: pageImage.left; bottomMargin: 5; rightMargin: Math.floor(pages.width)+12} + source: "star.png" + } + //comic rating + Text { + id: comicRating + anchors {bottom: realCell.bottom; right: ratingImage.left; margins: 4} + text: rating>0?rating:"-" + color: textColor + } + } + } + + YACReaderScrollView{ + id: scrollView + anchors.fill: parent + anchors.margins: 0 + + + GridView { + id:grid + anchors.fill: parent + cellHeight: 295 + highlight: appHighlight + focus: true + model: comicsList + delegate: appDelegate + anchors.topMargin: 20 + anchors.bottomMargin: 20 + anchors.leftMargin: 10 + anchors.rightMargin: 10 + pixelAligned: true + //flickDeceleration: -2000 + snapMode: GridView.SnapToRow + currentIndex: 0 + cacheBuffer: 0 + + + function numCellsPerRow() { + return Math.floor(width / 190); + } + + onWidthChanged: { + var numCells = numCellsPerRow(); + var rest = width % 190; + + if(numCells > 0) + { + cellWidth = Math.floor(width / numCells) ; + //console.log("numCells=",numCells,"rest=",rest,"cellWidth=",cellWidth,"width=",width); + } + } + } + focus: true + Keys.onPressed: { + if (event.modifiers & Qt.ControlModifier || event.modifiers & Qt.ShiftModifier) + return; + var numCells = grid.numCellsPerRow(); + var ci + if (event.key === Qt.Key_Right) { + ci = Math.min(grid.currentIndex+1,grid.count); + } + else if (event.key === Qt.Key_Left) { + ci = Math.max(0,grid.currentIndex-1); + } + else if (event.key === Qt.Key_Up) { + ci = Math.max(0,grid.currentIndex-numCells); + } + else if (event.key === Qt.Key_Down) { + ci = Math.min(grid.currentIndex+numCells,grid.count); + } + + event.accepted = true; + //var ci = grid.currentIndex; + grid.currentIndex = -1 + comicsSelectionHelper.clear(); + comicsSelectionHelper.selectIndex(ci); + grid.currentIndex = ci; + } + //} + + /*MouseArea { + anchors.fill: parent + onClicked: { + clicked.accepted = false; + console.log("xx"); + } + + onWheel: { + var newValue = Math.max(0,scrollView.flickableItem.contentY - wheel.angleDelta.y) + scrollView.flickableItem.contentY = newValue + + } + }*/ + /*ScrollBar { + flickable: grid; + } + + PerformanceMeter { + anchors {top: parent.top; left: parent.left; margins: 4} + id: performanceMeter + width: 128 + height: 64 + enabled: (dummyValue || !dummyValue) + }*/ + } +} + + diff --git a/YACReaderLibrary/qml/YACReaderScrollView.qml b/YACReaderLibrary/qml/YACReaderScrollView.qml new file mode 100644 index 00000000..a8dc57ad --- /dev/null +++ b/YACReaderLibrary/qml/YACReaderScrollView.qml @@ -0,0 +1,336 @@ +/**************************************************************************** +** +** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). +** Contact: http://www.qt-project.org/legal +** +** This file is part of the Qt Quick Controls module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Digia Plc and its Subsidiary(-ies) nor the names +** of its contributors may be used to endorse or promote products derived +** from this software without specific prior written permission. +** +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +import QtQuick 2.2 +import QtQuick.Controls 1.2 +import QtQuick.Controls.Private 1.0 +import QtQuick.Controls.Styles 1.1 + +/*! + \qmltype ScrollView + \inqmlmodule QtQuick.Controls + \since 5.1 + \ingroup views + \brief Provides a scrolling view within another Item. + + A ScrollView can be used either to replace a \l Flickable or decorate an + existing \l Flickable. Depending on the platform, it will add scroll bars and + a content frame. + + Only one Item can be a direct child of the ScrollView and the child is implicitly anchored + to fill the scroll view. + + Example: + \code + ScrollView { + Image { source: "largeImage.png" } + } + \endcode + + In the previous example the Image item will implicitly get scroll behavior as if it was + used within a \l Flickable. The width and height of the child item will be used to + define the size of the content area. + + Example: + \code + ScrollView { + ListView { + ... + } + } + \endcode + + In this case the content size of the ScrollView will simply mirror that of its contained + \l flickableItem. + + You can create a custom appearance for a ScrollView by + assigning a \l {QtQuick.Controls.Styles::ScrollViewStyle}{ScrollViewStyle}. +*/ + +FocusScope { + id: root + + implicitWidth: 240 + implicitHeight: 150 + + /*! + This property tells the ScrollView if it should render + a frame around its content. + + The default value is \c false. + */ + property bool frameVisible: false + + /*! + This property controls if there should be a highlight + around the frame when the ScrollView has input focus. + + The default value is \c false. + + \note This property is only applicable on some platforms, such + as Mac OS. + */ + property bool highlightOnFocus: false + + /*! + \qmlproperty Item ScrollView::viewport + + The viewport determines the current "window" on the contentItem. + In other words, it clips it and the size of the viewport tells you + how much of the content area is visible. + */ + property alias viewport: viewportItem + + /*! + \qmlproperty Item ScrollView::flickableItem + + The flickableItem of the ScrollView. If the contentItem provided + to the ScrollView is a Flickable, it will be the \l contentItem. + */ + readonly property alias flickableItem: internal.flickableItem + + /*! + The contentItem of the ScrollView. This is set by the user. + + Note that the definition of contentItem is somewhat different to that + of a Flickable, where the contentItem is implicitly created. + */ + default property Item contentItem + + /*! \internal */ + property Item __scroller: scroller + /*! \internal */ + property alias __wheelAreaScrollSpeed: wheelArea.scrollSpeed + /*! \internal */ + property int __scrollBarTopMargin: 0 + /*! \internal */ + property int __viewTopMargin: 0 + /*! \internal */ + property alias __horizontalScrollBar: scroller.horizontalScrollBar + /*! \internal */ + property alias __verticalScrollBar: scroller.verticalScrollBar + /*! \qmlproperty Component ScrollView::style + + The style Component for this control. + \sa {Qt Quick Controls Styles QML Types} + + */ + property Component style: Qt.createComponent(Settings.style + "/ScrollViewStyle.qml", root) + + /*! \internal */ + property Style __style: styleLoader.item + + activeFocusOnTab: true + + onContentItemChanged: { +//console.log("onContentItemChanged"); + if (contentItem.hasOwnProperty("contentY") && // Check if flickable + contentItem.hasOwnProperty("contentHeight")) { + internal.flickableItem = contentItem // "Use content if it is a flickable + internal.flickableItem.parent = viewportItem + } else { + internal.flickableItem = flickableComponent.createObject(viewportItem) + contentItem.parent = internal.flickableItem.contentItem + } + internal.flickableItem.anchors.fill = viewportItem + if (!Settings.hasTouchScreen) + internal.flickableItem.interactive = false + } + + + children: Item { + id: internal + + property Flickable flickableItem + + Loader { + id: styleLoader + sourceComponent: style + onStatusChanged: { + if (status === Loader.Error) + console.error("Failed to load Style for", root) + } + property alias __control: root + } + + Binding { + target: flickableItem + property: "contentHeight" + when: contentItem !== flickableItem + value: contentItem ? contentItem.height : 0 + } + + Binding { + target: flickableItem + when: contentItem !== flickableItem + property: "contentWidth" + value: contentItem ? contentItem.width : 0 + } + + Connections { + target: flickableItem + + onContentYChanged: { + //console.log("onContentYChanged2"); + scroller.blockUpdates = true + scroller.verticalScrollBar.value = flickableItem.contentY + scroller.blockUpdates = false + } + + onContentXChanged: { + //console.log("onContentXChanged2"); + scroller.blockUpdates = true + scroller.horizontalScrollBar.value = flickableItem.contentX + scroller.blockUpdates = false + } + + } + + anchors.fill: parent + + Component { + id: flickableComponent + Flickable {} + } + + WheelArea { + id: wheelArea + parent: flickableItem + + // ### Note this is needed due to broken mousewheel behavior in Flickable. + + anchors.fill: parent + + property int stepSize: 295 + + property int acceleration: 40 + property int flickThreshold: Settings.dragThreshold + property real speedThreshold: 3 + property real ignored: 0.001 // ## flick() does not work with 0 yVelocity + property int maxFlick: 400 + + property bool horizontalRecursionGuard: false + property bool verticalRecursionGuard: false + + horizontalMinimumValue: flickableItem ? flickableItem.originX : 0 + horizontalMaximumValue: flickableItem ? flickableItem.originX + flickableItem.contentWidth - viewport.width : 0 + + verticalMinimumValue: flickableItem ? flickableItem.originY : 0 + verticalMaximumValue: flickableItem ? flickableItem.originY + flickableItem.contentHeight - viewport.height + __viewTopMargin : 0 + + Connections { + target: flickableItem + + onContentYChanged: { + //console.log("onContentYChanged"); + wheelArea.verticalRecursionGuard = true + wheelArea.verticalValue = flickableItem.contentY + wheelArea.verticalRecursionGuard = false + } + onContentXChanged: { + //console.log("onContentXChanged"); + wheelArea.horizontalRecursionGuard = true + wheelArea.horizontalValue = flickableItem.contentX + wheelArea.horizontalRecursionGuard = false + } + } + + onVerticalValueChanged: { + if (!verticalRecursionGuard) { + //console.log(verticalDelta); + + if (flickableItem.contentY < flickThreshold && verticalDelta > speedThreshold) { + flickableItem.flick(ignored, Math.min(maxFlick, acceleration * verticalDelta)) + } else if (flickableItem.contentY > flickableItem.contentHeight + - flickThreshold - viewport.height && verticalDelta < -speedThreshold) { + flickableItem.flick(ignored, Math.max(-maxFlick, acceleration * verticalDelta)) + } else { + var absDelta = Math.abs(verticalDelta); + + if(verticalDelta < 0) + flickableItem.contentY = verticalValue + Math.min(98,0.93*absDelta+4.5); + else + flickableItem.contentY = verticalValue - Math.min(98,0.93*absDelta+4.5); +} + + + //TODO: snap to row + + } + + } + + onHorizontalValueChanged: { + if (!horizontalRecursionGuard) + flickableItem.contentX = horizontalValue + } + } + + ScrollViewHelper { + id: scroller + anchors.fill: parent + active: wheelArea.active + property bool outerFrame: !frameVisible || !(__style ? __style.__externalScrollBars : 0) + property int scrollBarSpacing: outerFrame ? 0 : (__style ? __style.__scrollBarSpacing : 0) + property int verticalScrollbarOffset: verticalScrollBar.visible && !verticalScrollBar.isTransient ? + verticalScrollBar.width + scrollBarSpacing : 0 + property int horizontalScrollbarOffset: horizontalScrollBar.visible && !horizontalScrollBar.isTransient ? + horizontalScrollBar.height + scrollBarSpacing : 0 + Loader { + id: frameLoader + sourceComponent: __style ? __style.frame : null + anchors.fill: parent + anchors.rightMargin: scroller.outerFrame ? 0 : scroller.verticalScrollbarOffset + anchors.bottomMargin: scroller.outerFrame ? 0 : scroller.horizontalScrollbarOffset + } + + Item { + id: viewportItem + anchors.fill: frameLoader + anchors.topMargin: frameVisible ? __style.padding.top : 0 + anchors.leftMargin: frameVisible ? __style.padding.left : 0 + anchors.rightMargin: (frameVisible ? __style.padding.right : 0) + (scroller.outerFrame ? scroller.verticalScrollbarOffset : 0) + anchors.bottomMargin: (frameVisible ? __style.padding.bottom : 0) + (scroller.outerFrame ? scroller.horizontalScrollbarOffset : 0) + clip: true + } + } + FocusFrame { visible: highlightOnFocus && root.activeFocus } + } +} diff --git a/YACReaderLibrary/qml/page.png b/YACReaderLibrary/qml/page.png new file mode 100644 index 0000000000000000000000000000000000000000..100db8a07ce55af57e69d58fd70c3ec37690e1ed GIT binary patch literal 155 zcmeAS@N?(olHy`uVBq!ia0vp^96-#)!3HEdkIOdzDajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MYf(Ujv*Ddl795dN*Ekq=s4cMvViLzb0Jg5`9lq|Doh`1 z7AP^gO(=3xIFPEejA53HFf*eAJF8BM43|IxgRqEQ=BZ0eSwMprJYD@<);T3K0RU%2 BE8+kE literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/reading.png b/YACReaderLibrary/qml/reading.png new file mode 100644 index 0000000000000000000000000000000000000000..a26a81d63a321ff8c73407b041c96b3c22c3576b GIT binary patch literal 374 zcmV-+0g3*JP)+7*9lVSQEwMA4ff7z&p)DoQSkRc@A2T3PiBWd_$!B1)`!d-qGqS2G(IqLR zi?D!Q_7E=NlpSM#+PVK79MCDk2DZd!6>tE~=_4HA6@n+eR|M9f6Amy{I~Ttz0WYXO z)KkTMz@#~I9&kQmPw0`yyr%Mv(5pL7@pZE_!!6<3cmIAa4Ep2b$qQ(_6UFN1DJ6iGx9CTxW#|uyQ7I> zD^q~9RKpKj0d0quY%6v!>}vZQ@Vw#0R|gpeQT~Qc4?n%w_NLv7E1~dU<%&HFX6onI zXZEPEZOXLUBq2O&U$?^_##IrFTHhG8`hq9h2yi<%^NLk51r*Mi#nT`mV$R+eA?Ri8 naI}!EnyIhB!$FLh!+>GZiKSU*Paew!x{txr)z4*}Q$iB}nx9l& literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/tick.png b/YACReaderLibrary/qml/tick.png new file mode 100644 index 0000000000000000000000000000000000000000..78a20644c27b468c50767aa4a44597662f7253de GIT binary patch literal 488 zcmVP)wXM;%Yz~3&g? zC_;UD5u4#56c~&gN1%AQ2I4Rwg|QO@b$}DpJhgt^2k28-cbAj1Q&xen-6dqYD*IeUA1 zMJ6Vu&map(Fc<__K{>P+nl>?9e*E}xKTw#0_9*{be eYPCN=fB^su{-6g#-^-l<0000beginGroup("debugLogFile"); Logger* logger=new FileLogger(mainLogSettings,10000,app); - logger->installMsgHandler(); + logger->installMsgHandler(); // Configure template loader and cache QSettings* templateSettings=new QSettings(configFileName,QSettings::IniFormat,app); @@ -65,9 +65,14 @@ void Startup::start() { void Startup::stop() { - qDebug("ServiceHelper: Service has been stopped"); - // QCoreApplication destroys all objects that have been created in start(). - delete listener; + qDebug("ServiceHelper: Service has been stopped"); + // QCoreApplication destroys all objects that have been created in start(). + if(listener!=nullptr) + { + listener->close(); + delete listener; + listener = nullptr; + } } diff --git a/YACReaderLibrary/yacreader_local_server.cpp b/YACReaderLibrary/yacreader_local_server.cpp index a24c184f..8d6d7c82 100644 --- a/YACReaderLibrary/yacreader_local_server.cpp +++ b/YACReaderLibrary/yacreader_local_server.cpp @@ -64,7 +64,12 @@ bool YACReaderLocalServer::isRunning() socket.connectToServer(YACREADERLIBRARY_GUID); if (socket.waitForConnected(500)) return true; // Server is running (another instance of YACReaderLibrary has been launched) - return false; + return false; +} + +void YACReaderLocalServer::close() +{ + localServer->close(); } diff --git a/YACReaderLibrary/yacreader_local_server.h b/YACReaderLibrary/yacreader_local_server.h index d6c8832b..d5432e60 100644 --- a/YACReaderLibrary/yacreader_local_server.h +++ b/YACReaderLibrary/yacreader_local_server.h @@ -21,6 +21,7 @@ public slots: bool isListening(); void sendResponse(); static bool isRunning(); + void close(); private: //void run(); QLocalServer * localServer; diff --git a/common/yacreader_global.cpp b/common/yacreader_global.cpp index 0ac9c43e..c8ad06c3 100644 --- a/common/yacreader_global.cpp +++ b/common/yacreader_global.cpp @@ -19,3 +19,11 @@ void YACReader::addSperator(QWidget *w) separator->setSeparator(true); w->addAction(separator); } + + +QAction * YACReader::createSeparator() +{ + QAction * a = new QAction(0); + a->setSeparator(true); + return a; +} diff --git a/common/yacreader_global.h b/common/yacreader_global.h index f92952c4..4f3ac6d0 100644 --- a/common/yacreader_global.h +++ b/common/yacreader_global.h @@ -96,6 +96,7 @@ namespace YACReader QString getSettingsPath(); void addSperator(QWidget * w); +QAction * createSeparator(); } #endif From 3348311ac7157bb1c5d562a46534ae7e1840ad41 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 21:02:36 +0200 Subject: [PATCH 06/34] first version with comics views switching (flow<->grid) at run time --- YACReaderLibrary/images_win.qrc | 35 +++--- YACReaderLibrary/library_window.cpp | 112 +++++++++++++++++--- YACReaderLibrary/library_window.h | 20 ++++ YACReaderLibrary/yacreader_main_toolbar.cpp | 7 +- YACReaderLibrary/yacreader_main_toolbar.h | 3 + common/yacreader_global.h | 1 + 6 files changed, 147 insertions(+), 31 deletions(-) diff --git a/YACReaderLibrary/images_win.qrc b/YACReaderLibrary/images_win.qrc index 7b8ae9fb..5bc6d493 100644 --- a/YACReaderLibrary/images_win.qrc +++ b/YACReaderLibrary/images_win.qrc @@ -1,19 +1,20 @@ - - ../images/main_toolbar/back.png - ../images/main_toolbar/back_disabled.png - ../images/main_toolbar/forward.png - ../images/main_toolbar/forward_disabled.png - ../images/main_toolbar/settings.png - ../images/main_toolbar/server.png - ../images/main_toolbar/help.png - ../images/main_toolbar/fullscreen.png - - ../images/libraryIcon.png - ../images/setRoot.png - ../images/expand.png - ../images/colapse.png - ../images/newLibraryIcon.png - ../images/openLibraryIcon.png - + + ../images/main_toolbar/back.png + ../images/main_toolbar/back_disabled.png + ../images/main_toolbar/forward.png + ../images/main_toolbar/forward_disabled.png + ../images/main_toolbar/settings.png + ../images/main_toolbar/server.png + ../images/main_toolbar/help.png + ../images/main_toolbar/fullscreen.png + ../images/libraryIcon.png + ../images/setRoot.png + ../images/expand.png + ../images/colapse.png + ../images/newLibraryIcon.png + ../images/openLibraryIcon.png + ../images/main_toolbar/flow.png + ../images/main_toolbar/grid.png + diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index b6e32a6b..c01d75a9 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -189,9 +189,25 @@ void LibraryWindow::doLayout() foldersTitle->addAction(colapseAllNodesAction); //FINAL LAYOUT------------------------------------------------------------- + comicsViewStack = new QWidget(); + QHBoxLayout * l = new QHBoxLayout(); + + if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) { + comicsView = classicComicsView = new ClassicComicsView(); + comicsViewStatus = Flow; + //comicsViewStack->setCurrentIndex(Flow); + } else { + comicsView = gridComicsView = new GridComicsView(); + comicsViewStatus = Grid; + //comicsViewStack->setCurrentIndex(Grid); + } + + doComicsViewConnections(); - comicsView = new ClassicComicsView(); comicsView->setToolBar(editInfoToolBar); + l->addWidget(comicsView); + comicsViewStack->setLayout(l); + l->setContentsMargins(0,0,0,0); fullScreenToolTip = new QLabel(comicsView); fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); @@ -205,7 +221,7 @@ void LibraryWindow::doLayout() #ifndef Q_OS_MAC QVBoxLayout * rightLayout = new QVBoxLayout; rightLayout->addWidget(libraryToolBar); - rightLayout->addWidget(comicsView); + rightLayout->addWidget(comicsViewStack); rightLayout->setMargin(0); rightLayout->setSpacing(0); @@ -215,7 +231,7 @@ void LibraryWindow::doLayout() sHorizontal->addWidget(rightWidget); #else - sHorizontal->addWidget(sVertical); + sHorizontal->addWidget(comicsViewStack); #endif sHorizontal->setStretchFactor(0,0); @@ -286,7 +302,25 @@ void LibraryWindow::doModels() //comics dmCV = new TableModel(); - setFoldersFilter(""); + setFoldersFilter(""); +} + +void LibraryWindow::disconnectComicsViewConnections(ComicsView * widget) +{ + disconnect(widget, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + disconnect(showHideMarksAction,SIGNAL(toggled(bool)),widget,SLOT(setShowMarks(bool))); + disconnect(widget,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); + disconnect(widget,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + disconnect(selectAllComicsAction,SIGNAL(triggered()),widget,SLOT(selectAll())); +} + +void LibraryWindow::doComicsViewConnections() +{ + connect(comicsView, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + connect(showHideMarksAction,SIGNAL(toggled(bool)),comicsView,SLOT(setShowMarks(bool))); + connect(comicsView,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); + connect(comicsView,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + connect(selectAllComicsAction,SIGNAL(triggered()),comicsView,SLOT(selectAll())); } void LibraryWindow::createActions() @@ -416,6 +450,15 @@ void LibraryWindow::createActions() icoServerButton.addPixmap(QPixmap(":/images/main_toolbar/server.png"), QIcon::Normal); serverConfigAction->setIcon(icoServerButton); + toggleComicsViewAction = new QAction(tr("Change between comics views"),this); + toggleComicsViewAction->setShortcut(Qt::Key_V); + toggleComicsViewAction->setToolTip(tr("Change between comics views")); + QIcon icoViewsButton; + if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/grid.png"), QIcon::Normal); + else + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/flow.png"), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); //socialAction = new QAction(this); openContainingFolderAction = new QAction(this); @@ -569,6 +612,7 @@ void LibraryWindow::createToolBars() w2->setFixedWidth(10); libraryToolBar->addWidget(w2);} + libraryToolBar->addAction(toggleComicsViewAction); libraryToolBar->addAction(toggleFullScreenAction); libraryToolBar->addWidget(new QToolBarStretch()); @@ -583,6 +627,7 @@ void LibraryWindow::createToolBars() libraryToolBar->settingsButton->setDefaultAction(optionsAction); libraryToolBar->serverButton->setDefaultAction(serverConfigAction); libraryToolBar->helpButton->setDefaultAction(helpAboutAction); + libraryToolBar->toggleComicsViewButton->setDefaultAction(toggleComicsViewAction); libraryToolBar->fullscreenButton->setDefaultAction(toggleFullScreenAction); #endif @@ -777,9 +822,6 @@ void LibraryWindow::createConnections() connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(loadCovers(QModelIndex))); connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(updateHistory(QModelIndex))); - connect(comicsView, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); - connect(showHideMarksAction,SIGNAL(toggled(bool)),comicsView,SLOT(setShowMarks(bool))); - //actions connect(createLibraryAction,SIGNAL(triggered()),this,SLOT(createLibrary())); connect(exportLibraryAction,SIGNAL(triggered()),exportLibraryDialog,SLOT(show())); @@ -812,14 +854,13 @@ void LibraryWindow::createConnections() connect(expandAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(expandAll())); connect(colapseAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(collapseAll())); connect(toggleFullScreenAction,SIGNAL(triggered()),this,SLOT(toggleFullScreen())); + connect(toggleComicsViewAction,SIGNAL(triggered()),this,SLOT(toggleComicsView())); connect(optionsAction, SIGNAL(triggered()),optionsDialog,SLOT(show())); #ifdef SERVER_RELEASE connect(serverConfigAction, SIGNAL(triggered()), serverConfigDialog, SLOT(show())); #endif connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); - //ComicFlow - connect(comicsView,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); - connect(comicsView,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + //Folders filter //connect(clearFoldersFilter,SIGNAL(clicked()),foldersFilter,SLOT(clear())); connect(foldersFilter,SIGNAL(textChanged(QString)),this,SLOT(setFoldersFilter(QString))); @@ -837,7 +878,6 @@ void LibraryWindow::createConnections() //connect(dm,SIGNAL(directoryLoaded(QString)),foldersView,SLOT(expandAll())); //connect(dm,SIGNAL(directoryLoaded(QString)),this,SLOT(updateFoldersView(QString))); //Comicts edition - connect(selectAllComicsAction,SIGNAL(triggered()),comicsView,SLOT(selectAll())); connect(editSelectedComicsAction,SIGNAL(triggered()),this,SLOT(showProperties())); connect(asignOrderActions,SIGNAL(triggered()),this,SLOT(asignNumbers())); @@ -1069,8 +1109,11 @@ void LibraryWindow::reloadCovers() loadCovers(QModelIndex()); QLOG_INFO() << "reloaded covers at row : " << foldersView->currentIndex().row(); QModelIndex mi = dmCV->getIndexFromId(_comicIdEdited); - comicsView->scrollTo(mi,QAbstractItemView::PositionAtCenter); - comicsView->setCurrentIndex(mi); + if(mi.isValid()) + { + comicsView->scrollTo(mi,QAbstractItemView::PositionAtCenter); + comicsView->setCurrentIndex(mi); + } //centerComicFlow(mi); } @@ -1467,6 +1510,48 @@ void LibraryWindow::resetComicRating() dmCV->finishTransaction(); } +void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) +{ + disconnectComicsViewConnections(from); + from->close(); + + comicsView = to; + doComicsViewConnections(); + + comicsView->setToolBar(editInfoToolBar); + + QHBoxLayout * l = new QHBoxLayout(); + l->addWidget(to); + l->setContentsMargins(0,0,0,0); + delete comicsViewStack->layout(); + comicsViewStack->setLayout(l); + + delete from; + + reloadCovers(); +} + +//TODO recover the current comics selection and restore it in the destination +void LibraryWindow::toggleComicsView() +{ + if(comicsViewStatus == Flow){ + QIcon icoViewsButton; + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/flow.png"), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); + switchToComicsView(classicComicsView, gridComicsView = new GridComicsView()); + comicsViewStatus = Grid; + } + else{ + QIcon icoViewsButton; + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/grid.png"), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); + switchToComicsView(gridComicsView, classicComicsView = new ClassicComicsView()); + comicsViewStatus = Flow; + } + + settings->setValue(COMICS_VIEW_STATUS, comicsViewStatus); +} + void LibraryWindow::asignNumbers() { QModelIndexList indexList = getSelectedComics(); @@ -1616,6 +1701,7 @@ void LibraryWindow::closeEvent ( QCloseEvent * event ) settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry()); comicsView->close(); + QApplication::instance()->processEvents(); event->accept(); QMainWindow::closeEvent(event); diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index d100212c..9e10d130 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -50,6 +50,8 @@ class YACReaderTreeView; class YACReaderMainToolBar; class ComicVineDialog; class ComicsView; +class ClassicComicsView; +class GridComicsView; #include "comic_db.h" using namespace YACReader; @@ -92,6 +94,10 @@ private: //------------- ComicsView * comicsView; + ClassicComicsView * classicComicsView; + GridComicsView * gridComicsView; + QWidget * comicsViewStack; + YACReaderTreeView * foldersView; YACReaderLibraryListWidget * selectedLibrary; TreeModel * dm; @@ -128,6 +134,7 @@ private: QAction * toggleFullScreenAction; QAction * optionsAction; QAction * serverConfigAction; + QAction * toggleComicsViewAction; //QAction * socialAction; //tree actions @@ -191,6 +198,9 @@ private: void doLayout(); void doDialogs(); void doModels(); + void disconnectComicsViewConnections(ComicsView * widget); + void doComicsViewConnections(); + //ACTIONS MANAGEMENT void disableComicsActions(bool disabled); @@ -214,6 +224,14 @@ private: bool removeError; + enum ComicsViewStatus + { + Flow, + Grid + }; + + ComicsViewStatus comicsViewStatus; + protected: virtual void closeEvent ( QCloseEvent * event ); public: @@ -283,6 +301,8 @@ public: void setRemoveError(); void checkRemoveError(); void resetComicRating(); + void switchToComicsView(ComicsView *from, ComicsView *to); + void toggleComicsView(); }; #endif diff --git a/YACReaderLibrary/yacreader_main_toolbar.cpp b/YACReaderLibrary/yacreader_main_toolbar.cpp index 61f6a3da..c4e1dc82 100644 --- a/YACReaderLibrary/yacreader_main_toolbar.cpp +++ b/YACReaderLibrary/yacreader_main_toolbar.cpp @@ -44,11 +44,14 @@ YACReaderMainToolBar::YACReaderMainToolBar(QWidget *parent) : helpButton->setStyleSheet(qToolButtonStyleSheet); helpButton->setIconSize(QSize(14,25)); + toggleComicsViewButton = new QToolButton; + toggleComicsViewButton->setStyleSheet(qToolButtonStyleSheet); + toggleComicsViewButton->setIconSize(QSize(24,24)); + fullscreenButton = new QToolButton(); fullscreenButton->setStyleSheet(qToolButtonStyleSheet); fullscreenButton->setIconSize(QSize(24,24)); - mainLayout->setMargin(0); mainLayout->setSpacing(0); @@ -66,6 +69,8 @@ YACReaderMainToolBar::YACReaderMainToolBar(QWidget *parent) : mainLayout->addStretch(); + mainLayout->addWidget(toggleComicsViewButton); + addWideDivider(); mainLayout->addWidget(fullscreenButton); mainLayout->addSpacing(10); diff --git a/YACReaderLibrary/yacreader_main_toolbar.h b/YACReaderLibrary/yacreader_main_toolbar.h index b45dbf4d..a21fd3f5 100644 --- a/YACReaderLibrary/yacreader_main_toolbar.h +++ b/YACReaderLibrary/yacreader_main_toolbar.h @@ -9,6 +9,7 @@ class QResizeEvent; class QPaintEvent; class QHBoxLayout; +//TODO create methods for adding actions, separators and sctreches dynimically class YACReaderMainToolBar : public QWidget { Q_OBJECT @@ -21,8 +22,10 @@ public: QToolButton * settingsButton; QToolButton * serverButton; QToolButton * helpButton; + QToolButton * toggleComicsViewButton; QToolButton * fullscreenButton; + void setCurrentFolderName(const QString & name); signals: diff --git a/common/yacreader_global.h b/common/yacreader_global.h index 4f3ac6d0..47ca6203 100644 --- a/common/yacreader_global.h +++ b/common/yacreader_global.h @@ -55,6 +55,7 @@ #define MAIN_WINDOW_STATE "MAIN_WINDOW_STATE" #define COMICS_VIEW_HEADERS "COMICS_VIEW_HEADERS" #define COMICS_VIEW_HEADERS_GEOMETRY "COMICS_VIEW_HEADERS_GEOMETRY" +#define COMICS_VIEW_STATUS "COMICS_VIEW_STATUS" #define NUM_DAYS_BETWEEN_VERSION_CHECKS "NUM_DAYS_BETWEEN_VERSION_CHECKS" #define LAST_VERSION_CHECK "LAST_VERSION_CHECK" From ec62ed4bcf848ac9f6338e1a598838c3464419fa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 21:39:37 +0200 Subject: [PATCH 07/34] add missing file qml.qrc --- YACReaderLibrary/qml.qrc | 10 ++++++++++ 1 file changed, 10 insertions(+) create mode 100644 YACReaderLibrary/qml.qrc diff --git a/YACReaderLibrary/qml.qrc b/YACReaderLibrary/qml.qrc new file mode 100644 index 00000000..ec494823 --- /dev/null +++ b/YACReaderLibrary/qml.qrc @@ -0,0 +1,10 @@ + + + qml/GridComicsView.qml + qml/YACReaderScrollView.qml + qml/page.png + qml/star.png + qml/tick.png + qml/reading.png + + From 067b8ac0324af98d5b955b76c09f9f8b45760bb8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 22:03:54 +0200 Subject: [PATCH 08/34] fixed OSX compilation (Qt5) --- YACReaderLibrary/YACReaderLibrary.pro | 5 ++++- YACReaderLibrary/classic_comics_view.cpp | 4 ++++ YACReaderLibrary/db/data_base_management.cpp | 4 ++-- YACReaderLibrary/db/tablemodel.cpp | 4 ++-- YACReaderLibrary/library_window.cpp | 1 - 5 files changed, 12 insertions(+), 6 deletions(-) diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index ccf4566e..b34a5fc0 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -34,7 +34,6 @@ CONFIG -= embed_manifest_exe } unix:!macx{ -QMAKE_CXXFLAGS += -std=c++11 isEqual(QT_MAJOR_VERSION, 5) { INCLUDEPATH += /usr/include/poppler/qt5 @@ -67,6 +66,10 @@ CONFIG += objective_c } +unix{ +QMAKE_CXXFLAGS += -std=c++11 +} + #CONFIG += release CONFIG -= flat QT += sql network opengl script diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp index 26ee326f..e64106e9 100644 --- a/YACReaderLibrary/classic_comics_view.cpp +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -67,6 +67,10 @@ ClassicComicsView::ClassicComicsView(QWidget *parent) setLayout(layout); layout->setMargin(0); + +#ifdef Q_OS_MAC + sVertical->setCollapsible(1,false); +#endif } void ClassicComicsView::setToolBar(QToolBar *toolBar) diff --git a/YACReaderLibrary/db/data_base_management.cpp b/YACReaderLibrary/db/data_base_management.cpp index 244634db..1b0ce7c9 100644 --- a/YACReaderLibrary/db/data_base_management.cpp +++ b/YACReaderLibrary/db/data_base_management.cpp @@ -210,7 +210,7 @@ bool DataBaseManagement::createTables(QSqlDatabase & database) //queryDBInfo.finish(); QSqlQuery query("INSERT INTO db_info (version) " - "VALUES ('"VERSION"')",database); + "VALUES ('" VERSION "')",database); //query.finish(); } @@ -245,7 +245,7 @@ void DataBaseManagement::exportComicsInfo(QString source, QString dest) queryComicsInfo.exec();*/ QSqlQuery query("INSERT INTO dest.db_info (version) " - "VALUES ('"VERSION"')",destDB); + "VALUES ('" VERSION "')",destDB); //query.finish(); QSqlQuery exportData(destDB); diff --git a/YACReaderLibrary/db/tablemodel.cpp b/YACReaderLibrary/db/tablemodel.cpp index 08f996cc..1d9e51c2 100644 --- a/YACReaderLibrary/db/tablemodel.cpp +++ b/YACReaderLibrary/db/tablemodel.cpp @@ -506,7 +506,7 @@ QVector TableModel::setComicsRead(QList l db.close(); QSqlDatabase::removeDatabase(_databasePath); - emit dataChanged(index(list.first().row(),TableModel::ReadColumn),index(list.last().row(),TableModel::HasBeenOpened),QVector() = {ReadColumnRole,CurrentPageRole,HasBeenOpenedRole}); + emit dataChanged(index(list.first().row(),TableModel::ReadColumn),index(list.last().row(),TableModel::HasBeenOpened),QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); return getReadList(); } @@ -620,7 +620,7 @@ void TableModel::reload(const ComicDB & comic) row++; } if(found) - emit dataChanged(index(row,ReadColumn),index(row,HasBeenOpened), QVector() = {ReadColumnRole,CurrentPageRole,HasBeenOpenedRole}); + emit dataChanged(index(row,ReadColumn),index(row,HasBeenOpened), QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); } void TableModel::resetComicRating(const QModelIndex &mi) diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index c01d75a9..81fc2af4 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -258,7 +258,6 @@ void LibraryWindow::doLayout() //collapsible disabled in macosx (only temporaly) #ifdef Q_OS_MAC sHorizontal->setCollapsible(0,false); - sVertical->setCollapsible(1,false); #endif } From 863e993801eb1dd1c1f9b96fdea4c09ede3e6cae Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 22:05:33 +0200 Subject: [PATCH 09/34] fixed widgets layout in GridComicsView --- YACReaderLibrary/grid_comics_view.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp index c8da3b20..d8f5d0d8 100644 --- a/YACReaderLibrary/grid_comics_view.cpp +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -25,6 +25,7 @@ GridComicsView::GridComicsView(QWidget *parent) : setContentsMargins(0,0,0,0); l->setContentsMargins(0,0,0,0); + l->setSpacing(0); QLOG_INFO() << "GridComicsView"; } From 673deee27cfc1eaae02cc4d2fac8846379525dd4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 6 Jul 2014 22:40:46 +0200 Subject: [PATCH 10/34] reduced width of search edit in OSX --- custom_widgets/yacreader_search_line_edit.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/custom_widgets/yacreader_search_line_edit.cpp b/custom_widgets/yacreader_search_line_edit.cpp index 11f42622..26d00c31 100644 --- a/custom_widgets/yacreader_search_line_edit.cpp +++ b/custom_widgets/yacreader_search_line_edit.cpp @@ -34,7 +34,7 @@ YACReaderSearchLineEdit::YACReaderSearchLineEdit(QWidget *parent) qMax(msz.height(), clearButton->sizeHint().height() + frameWidth * 2 + 2)); #ifdef Q_OS_MAC - setMaximumWidth(300); + setMaximumWidth(212); #endif } From b860d8776a3d36c26024b53224b34813fcb49e95 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Mon, 7 Jul 2014 23:30:07 +0200 Subject: [PATCH 11/34] added full functional context menu to GridComcisView (ugly solution for the QAction/QML issue) --- YACReaderLibrary/grid_comics_view.cpp | 28 +++++++++++++++++++++++++ YACReaderLibrary/library_window.cpp | 20 +++++++++--------- YACReaderLibrary/library_window.h | 5 ++++- YACReaderLibrary/qml/GridComicsView.qml | 25 ++++++++++++++-------- 4 files changed, 58 insertions(+), 20 deletions(-) diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp index d8f5d0d8..8267e909 100644 --- a/YACReaderLibrary/grid_comics_view.cpp +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -138,6 +138,34 @@ void GridComicsView::setViewActions(const QList &actions) //TODO generate QML Menu from actions QLOG_INFO() << "setViewActions"; this->addActions(actions); + + //TODO this is completely unsafe, but QActions can't be used directly in QML + if(actions.length()>17) + { + QQmlContext *ctxt = view->rootContext(); + + ctxt->setContextProperty("openComicAction",actions[0]); + + ctxt->setContextProperty("openContainingFolderComicAction",actions[2]); + + ctxt->setContextProperty("resetComicRatingAction",actions[4]); + + ctxt->setContextProperty("editSelectedComicsAction",actions[6]); + ctxt->setContextProperty("getInfoAction",actions[7]); + ctxt->setContextProperty("asignOrderAction",actions[8]); + + ctxt->setContextProperty("selectAllComicsAction",actions[10]); + + ctxt->setContextProperty("setAsReadAction",actions[12]); + ctxt->setContextProperty("setAsNonReadAction",actions[13]); + ctxt->setContextProperty("showHideMarksAction",actions[14]); + + ctxt->setContextProperty("deleteComicsAction",actions[16]); + + ctxt->setContextProperty("toggleFullScreenAction",actions[18]); + } + else + QLOG_ERROR() << "setViewActions invoked with the wrong number of actions"; } QSize GridComicsView::sizeHint() diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 81fc2af4..84769b13 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -496,9 +496,9 @@ void LibraryWindow::createActions() editSelectedComicsAction->setText(tr("Edit")); editSelectedComicsAction->setIcon(QIcon(":/images/editComic.png")); - asignOrderActions = new QAction(this); - asignOrderActions->setText(tr("Asign current order to comics")); - asignOrderActions->setIcon(QIcon(":/images/asignNumber.png")); + asignOrderAction = new QAction(this); + asignOrderAction->setText(tr("Asign current order to comics")); + asignOrderAction->setIcon(QIcon(":/images/asignNumber.png")); forceConverExtractedAction = new QAction(this); forceConverExtractedAction->setText(tr("Update cover")); @@ -530,7 +530,7 @@ void LibraryWindow::disableComicsActions(bool disabled) openComicAction->setDisabled(disabled); editSelectedComicsAction->setDisabled(disabled); selectAllComicsAction->setDisabled(disabled); - asignOrderActions->setDisabled(disabled); + asignOrderAction->setDisabled(disabled); setAsReadAction->setDisabled(disabled); setAsNonReadAction->setDisabled(disabled); //setAllAsReadAction->setDisabled(disabled); @@ -635,7 +635,7 @@ void LibraryWindow::createToolBars() editInfoToolBar->addSeparator(); editInfoToolBar->addAction(editSelectedComicsAction); editInfoToolBar->addAction(getInfoAction); - editInfoToolBar->addAction(asignOrderActions); + editInfoToolBar->addAction(asignOrderAction); editInfoToolBar->addSeparator(); @@ -661,21 +661,19 @@ void LibraryWindow::createToolBars() void LibraryWindow::createMenus() { - QList itemActions; itemActions << openContainingFolderComicAction << YACReader::createSeparator() << resetComicRatingAction << YACReader::createSeparator() << editSelectedComicsAction << getInfoAction - << asignOrderActions + << asignOrderAction << YACReader::createSeparator() << setAsReadAction << setAsNonReadAction << YACReader::createSeparator() << deleteComicsAction; - QList viewActions; viewActions << openComicAction << YACReader::createSeparator() << openContainingFolderComicAction @@ -684,7 +682,7 @@ void LibraryWindow::createMenus() << YACReader::createSeparator() << editSelectedComicsAction << getInfoAction - << asignOrderActions + << asignOrderAction << YACReader::createSeparator() << selectAllComicsAction << YACReader::createSeparator() @@ -878,7 +876,7 @@ void LibraryWindow::createConnections() //connect(dm,SIGNAL(directoryLoaded(QString)),this,SLOT(updateFoldersView(QString))); //Comicts edition connect(editSelectedComicsAction,SIGNAL(triggered()),this,SLOT(showProperties())); - connect(asignOrderActions,SIGNAL(triggered()),this,SLOT(asignNumbers())); + connect(asignOrderAction,SIGNAL(triggered()),this,SLOT(asignNumbers())); connect(deleteComicsAction,SIGNAL(triggered()),this,SLOT(deleteComics())); @@ -1516,6 +1514,8 @@ void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) comicsView = to; doComicsViewConnections(); + to->setItemActions(itemActions); + to->setViewActions(viewActions); comicsView->setToolBar(editInfoToolBar); diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 9e10d130..6228fc62 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -162,11 +162,14 @@ private: //edit info actions QAction * selectAllComicsAction; QAction * editSelectedComicsAction; - QAction * asignOrderActions; + QAction * asignOrderAction; QAction * forceConverExtractedAction; QAction * deleteComicsAction; QAction * hideComicViewAction; + QList itemActions; + QList viewActions; + #ifdef Q_OS_MAC QToolBar * libraryToolBar; #else diff --git a/YACReaderLibrary/qml/GridComicsView.qml b/YACReaderLibrary/qml/GridComicsView.qml index c3e4e515..36ee0cfb 100644 --- a/YACReaderLibrary/qml/GridComicsView.qml +++ b/YACReaderLibrary/qml/GridComicsView.qml @@ -81,21 +81,28 @@ Rectangle { } } - + //Menu emits the 'main' signals Menu { id: myContextMenu - MenuItem { text: "Open containing folder..."; onTriggered: display.setRandomName() } + MenuItem { text: "Open comic"; enabled: true; iconSource:"qrc:///images/openInYACReader.png"; onTriggered: openComicAction.trigger() } MenuSeparator{} - MenuItem { text: "Reset comic rating"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Open containing folder..."; enabled: true; iconSource: "qrc:///images/open.png"; onTriggered: openContainingFolderComicAction.trigger() } MenuSeparator{} - MenuItem { text: "Edit"; onTriggered: model1.removeRows(index, 1) } - MenuItem { text: "Download tags from Comic Vine"; onTriggered: model1.removeRows(index, 1) } - MenuItem { text: "Asign current order to comics"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Reset comic rating"; onTriggered: resetComicRatingAction.trigger() } MenuSeparator{} - MenuItem { text: "Set as read"; onTriggered: model1.removeRows(index, 1) } - MenuItem { text: "Set as unread"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Edit"; enabled: true; iconSource:"qrc:///images/editComic.png"; onTriggered: editSelectedComicsAction.trigger() } + MenuItem { text: "Download tags from Comic Vine"; enabled: true; iconSource:"qrc:///images/getInfo.png"; onTriggered: getInfoAction.trigger() } + MenuItem { text: "Asign current order to comics"; enabled: true; iconSource:"qrc:///images/asignNumber.png"; onTriggered: asignOrderAction.trigger() } MenuSeparator{} - MenuItem { text: "Delete selected comics"; onTriggered: model1.removeRows(index, 1) } + MenuItem { text: "Select all comics"; enabled: true; iconSource:"qrc:///images/selectAll.png"; onTriggered: selectAllComicsAction.trigger() } + MenuSeparator{} + MenuItem { text: "Set as read"; enabled: true; iconSource:"qrc:///images/setReadButton.png"; onTriggered: setAsReadAction.trigger() } + MenuItem { text: "Set as unread"; enabled: true; iconSource:"qrc:///images/setUnread.png"; onTriggered: setAsNonReadAction.trigger() } + MenuItem { text: "Show or hide read marks"; enabled: true; iconSource:"qrc:///images/showMarks.png"; onTriggered: showHideMarksAction.trigger() } + MenuSeparator{} + MenuItem { text: "Delete selected comics"; enabled: true; iconSource:"qrc:///images/trash.png"; onTriggered: deleteComicsAction.trigger() } + MenuSeparator{} + MenuItem { text: "Fullscreen mode on/off"; onTriggered: toggleFullScreenAction.trigger() } //MenuItem { text: "Show details"; onTriggered: cell.state = 'Details'; } From 9f53ae6efcfb63c90308f86d4d30230c8750e66c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 9 Jul 2014 15:47:32 +0200 Subject: [PATCH 12/34] added an animation for the transition between flow and grid comics views --- YACReaderLibrary/YACReaderLibrary.pro | 6 +- YACReaderLibrary/comics_view_transition.cpp | 74 ++++++++++++++++++++ YACReaderLibrary/comics_view_transition.h | 31 ++++++++ YACReaderLibrary/images_osx.qrc | 2 + YACReaderLibrary/images_win.qrc | 2 + YACReaderLibrary/library_window.cpp | 39 +++++++---- YACReaderLibrary/library_window.h | 13 ++-- common/yacreader_global.h | 6 ++ images/flow_to_grid.gif | Bin 0 -> 124025 bytes images/grid_to_flow.gif | Bin 0 -> 129647 bytes 10 files changed, 152 insertions(+), 21 deletions(-) create mode 100644 YACReaderLibrary/comics_view_transition.cpp create mode 100644 YACReaderLibrary/comics_view_transition.h create mode 100644 images/flow_to_grid.gif create mode 100644 images/grid_to_flow.gif diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index b34a5fc0..1d19565d 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -199,9 +199,11 @@ isEqual(QT_MAJOR_VERSION, 5) { #QML/GridView QT += quick qml -HEADERS += grid_comics_view.h +HEADERS += grid_comics_view.h \ + comics_view_transition.h -SOURCES += grid_comics_view.cpp +SOURCES += grid_comics_view.cpp \ + comics_view_transition.cpp RESOURCES += qml.qrc diff --git a/YACReaderLibrary/comics_view_transition.cpp b/YACReaderLibrary/comics_view_transition.cpp new file mode 100644 index 00000000..3e2fdd12 --- /dev/null +++ b/YACReaderLibrary/comics_view_transition.cpp @@ -0,0 +1,74 @@ +#include "comics_view_transition.h" + +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +ComicsViewTransition::ComicsViewTransition(QWidget *parent) : + QWidget(parent),movie(0) +{ + QVBoxLayout * layout = new QVBoxLayout; + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + settings->beginGroup("libraryConfig"); + + movieLabel = new QLabel("Placeholder"); + movieLabel->setAlignment(Qt::AlignCenter); + QLabel * textLabel = new QLabel("Switching comics view"); + textLabel->setAlignment(Qt::AlignCenter); + textLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); + //movieLabel->setFixedSize(450,350); + + layout->addSpacing(100); + layout->addWidget(movieLabel); + layout->addSpacing(20); + layout->addWidget(textLabel); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet("QWidget {background:#2A2A2A}"); + + //QSizePolicy sp(); + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + //movieLabel->setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + setLayout(layout); +} + +QSize ComicsViewTransition::sizeHint() +{ + return QSize(450,350); +} + +void ComicsViewTransition::startMovie() +{ + if(movie) + delete movie; + + if(settings->value(COMICS_VIEW_STATUS) == YACReader::Flow) + movie = new QMovie(":/images/flow_to_grid.gif"); + else + movie = new QMovie(":/images/grid_to_flow.gif"); + + connect(movie,SIGNAL(finished()),this,SIGNAL(transitionFinished())); + //connect(movie,SIGNAL(finished()),movie,SLOT(deleteLater()); + movie->setSpeed(200); + movie->jumpToFrame(0); + movieLabel->setMovie(movie); + + QTimer::singleShot(100,movie,SLOT(start())); +} + +void ComicsViewTransition::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor("#2A2A2A")); +} diff --git a/YACReaderLibrary/comics_view_transition.h b/YACReaderLibrary/comics_view_transition.h new file mode 100644 index 00000000..ed8c55ba --- /dev/null +++ b/YACReaderLibrary/comics_view_transition.h @@ -0,0 +1,31 @@ +#ifndef COMICS_VIEW_TRANSITION_H +#define COMICS_VIEW_TRANSITION_H + +#include + +class QMovie; +class QSettings; +class QLabel; + +class ComicsViewTransition : public QWidget +{ + Q_OBJECT +public: + explicit ComicsViewTransition(QWidget *parent = 0); + QSize sizeHint(); + +signals: + void transitionFinished(); + +public slots: + void startMovie(); + +protected: + QMovie * movie; + QSettings * settings; + QLabel * movieLabel; + + void paintEvent(QPaintEvent *); +}; + +#endif // COMICS_VIEW_TRANSITION_H diff --git a/YACReaderLibrary/images_osx.qrc b/YACReaderLibrary/images_osx.qrc index e400dade..d5604ff7 100644 --- a/YACReaderLibrary/images_osx.qrc +++ b/YACReaderLibrary/images_osx.qrc @@ -17,5 +17,7 @@ ../images/colapse_osx.png ../images/newLibraryIcon_osx.png ../images/openLibraryIcon_osx.png + ../images/flow_to_grid.gif + ../images/grid_to_flow.gif diff --git a/YACReaderLibrary/images_win.qrc b/YACReaderLibrary/images_win.qrc index 5bc6d493..5d2bd0af 100644 --- a/YACReaderLibrary/images_win.qrc +++ b/YACReaderLibrary/images_win.qrc @@ -16,5 +16,7 @@ ../images/openLibraryIcon.png ../images/main_toolbar/flow.png ../images/main_toolbar/grid.png + ../images/flow_to_grid.gif + ../images/grid_to_flow.gif diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 84769b13..003bf740 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -62,6 +62,7 @@ #include "classic_comics_view.h" #include "grid_comics_view.h" +#include "comics_view_transition.h" #include "QsLog.h" @@ -189,8 +190,7 @@ void LibraryWindow::doLayout() foldersTitle->addAction(colapseAllNodesAction); //FINAL LAYOUT------------------------------------------------------------- - comicsViewStack = new QWidget(); - QHBoxLayout * l = new QHBoxLayout(); + comicsViewStack = new QStackedWidget(); if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) { comicsView = classicComicsView = new ClassicComicsView(); @@ -205,9 +205,8 @@ void LibraryWindow::doLayout() doComicsViewConnections(); comicsView->setToolBar(editInfoToolBar); - l->addWidget(comicsView); - comicsViewStack->setLayout(l); - l->setContentsMargins(0,0,0,0); + comicsViewStack->addWidget(comicsView); + comicsViewStack->addWidget(comicsViewTransition = new ComicsViewTransition()); fullScreenToolTip = new QLabel(comicsView); fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); @@ -886,6 +885,8 @@ void LibraryWindow::createConnections() //connect(socialAction,SIGNAL(triggered()),this,SLOT(showSocial())); + connect(comicsViewTransition,SIGNAL(transitionFinished()),this,SLOT(showComicsView())); + } void LibraryWindow::loadLibrary(const QString & name) @@ -1519,19 +1520,21 @@ void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) comicsView->setToolBar(editInfoToolBar); - QHBoxLayout * l = new QHBoxLayout(); - l->addWidget(to); - l->setContentsMargins(0,0,0,0); - delete comicsViewStack->layout(); - comicsViewStack->setLayout(l); + comicsViewStack->removeWidget(from); + comicsViewStack->insertWidget(0,comicsView); delete from; reloadCovers(); } -//TODO recover the current comics selection and restore it in the destination -void LibraryWindow::toggleComicsView() +void LibraryWindow::showComicsViewTransition() +{ + comicsViewStack->setCurrentIndex(1); + comicsViewTransition->startMovie(); +} + +void LibraryWindow::toggleComicsView_delayed() { if(comicsViewStatus == Flow){ QIcon icoViewsButton; @@ -1551,6 +1554,18 @@ void LibraryWindow::toggleComicsView() settings->setValue(COMICS_VIEW_STATUS, comicsViewStatus); } +void LibraryWindow::showComicsView() +{ + comicsViewStack->setCurrentIndex(0); +} + +//TODO recover the current comics selection and restore it in the destination +void LibraryWindow::toggleComicsView() +{ + QTimer::singleShot(0,this,SLOT(showComicsViewTransition())); + QTimer::singleShot(32,this,SLOT(toggleComicsView_delayed())); +} + void LibraryWindow::asignNumbers() { QModelIndexList indexList = getSelectedComics(); diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 6228fc62..5d098c12 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -52,6 +52,7 @@ class ComicVineDialog; class ComicsView; class ClassicComicsView; class GridComicsView; +class ComicsViewTransition; #include "comic_db.h" using namespace YACReader; @@ -96,7 +97,8 @@ private: ComicsView * comicsView; ClassicComicsView * classicComicsView; GridComicsView * gridComicsView; - QWidget * comicsViewStack; + QStackedWidget * comicsViewStack; + ComicsViewTransition * comicsViewTransition; YACReaderTreeView * foldersView; YACReaderLibraryListWidget * selectedLibrary; @@ -227,12 +229,6 @@ private: bool removeError; - enum ComicsViewStatus - { - Flow, - Grid - }; - ComicsViewStatus comicsViewStatus; protected: @@ -305,6 +301,9 @@ public: void checkRemoveError(); void resetComicRating(); void switchToComicsView(ComicsView *from, ComicsView *to); + void showComicsViewTransition(); + void toggleComicsView_delayed();//used in orther to avoid flickering; + void showComicsView(); void toggleComicsView(); }; diff --git a/common/yacreader_global.h b/common/yacreader_global.h index 47ca6203..09e9d95f 100644 --- a/common/yacreader_global.h +++ b/common/yacreader_global.h @@ -95,6 +95,12 @@ namespace YACReader SevenZNotFound = 700 }; + enum ComicsViewStatus + { + Flow, + Grid + }; + QString getSettingsPath(); void addSperator(QWidget * w); QAction * createSeparator(); diff --git a/images/flow_to_grid.gif b/images/flow_to_grid.gif new file mode 100644 index 0000000000000000000000000000000000000000..f78228cf5c685f3dc247eac77dddb8f9b84a5a73 GIT binary patch literal 124025 zcmbTdV|ZQNx4^p*?AW%^V8^!6CT)zyY0%g3xu>FDU_>FJr7nQ3Tf zSXo&C|F4Y(j;8xsom$q^&GheHRKm;2)cm8R8?~9Gm5qZK?M2sDT51~$FZZ2v8Ar3A+J^=x4HfnAt7cVE2i<66o zol8iVmsc1HrT+Jg_OCY=ix0x@Waad$JQ!__*H!<43k^b8d?4AD8 ztb;4ZzlOtM?r6{HW$MJq#R2_iNdF=D`t|?6sJ;DvO1rwfv;41k|33}8YI-|aa=x>4 zb#!+zxBM$k_m3+lVHp=oQ#VH!O-Dz&|9Fb(){btDuGWrD)G{*vdTP|nIyMd#j-IY8 z|I&H=T3E@!)y>qw+)_zajP|b;4jUT_VLonoIcXj)UMRne6c?8quaqD^pAZ*6luu58 zTS%Ut`#*GL9nIbCEgjtcL)YTJbou^Q-G9=--s!JrSxXli4@(Pq7e{;Qe+^sM=6{cc z`+v3f@46QMdn`Qvt1jo?U^xGIx&Q5D|Hq}j8T8NLf2Qu=lmAS9ONYN1@A5abf$*0< z&rgpJ_jk89zkgj{U0$4@ot_*Y9Ukoe+}qvR-rC$)Ut3*SURqq3pZhU8Gd=bF+vLRf z*yzac(BMFSU+>qRFWp_89qnzcpFg!UH#IiY*VWckS5;P&mz9dgar6_xH#BYm>B42s3;I*Bt!%-2nYbsJRl%| z{>~a~|I8f!%rO82ICM;@=8NFTw=TE|bG(I3zIAoI-ARuV+mxIArdR!i-vzw4&`FVv zG-0&VWz0^vV7k&xxlX@fsn%oS%;)P?qn9d2&eY*2pKlIbo0si=ZyHuF?z;n~E}q~) z!6Bhx;SrIrsOXs3xcG#`q~w&;wDgS3tn8fJy!?W~qT-U$vhs?`s_L5By84F3rgAU> z5+VT7(b?7grRQsJU;n`1(D2A;7XT3n0o)u8L_i#yUszmPURgp!08abQ&aLkK+&?(% zUEkdD03sb-UH|$$3_vy1bEtM?y!N?ITevE0k86M7VisvC`-w2Qi64~I!jfPPydL8io4lHjLsyaFqrIEh>F zIE|}5+~wzXWC8nc=ZXpLZ|K(Hf+!55vB3{nZGCPd&q}S zaHUu8bSaxJj|QoPoQKt^!zXWlu?f$R3fiM1Z$0jz4zX4ZdeM+tb3ixGndt&{X>rP> zS;-5(WoYW+IYPy3%xjP@PmxFZwjs)~}4;8ghq(0Ggoj_A}S!0y@AGEsr04y;w==cMkAn>Ap-1x57DY} zOfFf7#J3#kdQl2(l?LW^8fKY`dF|BHkc;TmsXYrHw|Tx5k?-Pc^1V<7G$dRqK8yQ) z9np|`#u+~NK6E`TX)OB-4bAA!#`o3SXT;f(HnF$sSWxt{3>t^cw7zyz_FehW&^RVUm+2{WT1l1se=ZI3m>h`Bq^w!h*2l0Oh_|Bk!c;A zap_k;?<0G#)v=MO=;a{xCVfA)8$z>5Ofi&4JMcCdoJqPtXlM!&K-;^i0c#Qb zc#eo>a50jmBA7O$AI6=zz3E~t^K_){D?JIoHHQwp(Y!)b-qXau^BQ1;b0@{L|1u@4 zZ%5&7rH+Z~FC^w8mYcYO0zX(tk#lh=5EMES;TY1LY#eq8EN~$P*2+?=IVtGb(P@PeJJwL}jRei&5qafwZ}E5YX`LDA-K zPNj4zgIF9v^g|?5OMla%OGSxl>G!5T)>w!HkxXb_&&Kr(=0dZqCN-)#vd|67ah$wI z(OM%iJwGzC6?jj!E4lit%>AUx>rk+ENz5hdWkjBff_PwZVlO^gDkLn32 z!7nT)eX!wAK$0qbaI;2zO)3#R>r@Jivyf-%c$`$t( zf_M`oI!~)QWHVlTdtFQ;Rt+w*MNNETQVFm;MyNvVFD>UPBY1zL%6_CJs;yD>dY)Av za9pf@qqjOR=oR^d?P%GsMy4}xS(YDcO9WQv9Tw{E*A9$y9`qaK zwOeh|Ozoqx4YDHAQ9|T?+FClVY!cqJ5dW!@k6YDem7>U32Vc}IB7^z*HS(PNuewZBez$d=oI+8RyCShh zZqsQQEuy(P%wF^Benm!#jDnM>qU$sc)eT;~VqeoH^PC@{ox5T)CLbhjAsh{uhbBm= zm?lmV*A2;DqlO!*u$JhKQFFMY(6L&Mn>`F3@^L%Y8Sb7&dv{r#b7Ym+?Xz<(egk_q zn=PYTHG4e3K7QZNusPij#^RZx+2Bc~$h9olSpFQ6&jT;&i(Tf85twN!<0z>+v6^r7 z9w6Z04WsaXM|<~cIzQ31bi(k+2Gp5X@p4g?fsTlKA~l!Cz*p|8d(0I|iu5plUE48v zlrAm2;J@K6V^TLqv#H-X!syui?dinqm-Y{$bN3(}4Eq%h5xl{dW6TEjO-(Y|)eq~= zpR?Fb1$}cGzAdztKBykKtY)pGL^#(Sd7o1byb0D(8o+sV?sX!(_oiOJORU~H zQARYIkcw-#?wd2>h~W?-(?j3<7u}S=?4Ko={Nvv6*H;Q)u`Q0fhw_+>qg)BX*85xb zAO8MsVUga)wZsoU;@-QWc>clH_q_V{dh1v;jKag5=xMR0`Z`{Cy;(Vdf5AWTq+sej zffc9s+*$o4UBjOdt%HtfV~*=qg`2b7&pss=9CKMIzJ~++&sr2zSDy}+u6^KM!VH}H zlM=KShF0B+a#$YUrwFdp7rTq#FWTdHxi=+%Kdq(`mQy;*w>p^5UkfNa9e0ctWZL9s zlY8#U?(!fxXr1|psa}0(xtFSn;;DDnhj$l9W$9IphrST$FxTYzd(rjJ9Rm6&TA?fO>HrNB_I^nRl^ij++eAQ_ z4j=d4Yg)k#e-te8%b&q1Sd%-1SvT~#%!T0)jcW-gwS-pDY)u6bzab2{hy)?%;>bV@ z%XM@&JUEwwKIN!F;wV+wFOz*beRjyV*8dD1~Ta-YYpqwJF}+IYNONqw1cnbtXX= zH~!Xz!T()6*g4)B2mm!kNkB1#n$S^b*&~`_ZWZymMzMXpsNx{(-jzI`)v>UZsBp80 zBk$wTQ80w>(U2a3&Wa;YLNLRelNBY=Vj)~KLNq(ll-wjfB-g}04#B6zY#b~!zi0s8 z8&i^|V>_r5wl3p1Dk&H*;qe`psl?8)f{?W2caauN@rqGtM1aID^`z;>c+I0Ua3+Qs z3flcr{0AE#3kmwNAhw$|#{1=Tf}jKlc3iV$0@OT%V>;YgFCzza`Qv;5!17F2;CEZKg2nPpW#uc&N? zE54d4koHmbhj-~sLLg$995dp-gt+VkcI1DOy=m0;D7T`Bw!fgQfc8*S^ZLZpVDM(;D4bQ3V;Go52XJ;qFA&LInw_`G2M?r;|`o) za7ZtplJ5L}pjenYA_xHi{LvvHQN-ylX+``Sisd3A#l|OX!0zbKXg0JZ$sv>T~f`HNycfa$Fh zHQ>U*<<-*K%;~Kx^6vH1a}UzZ9q5)gv9_Quv;ox%Z`27{Dh!*>gB8$NBu9*ls=PLO zr9Kox4cD+Z{flB$(KH&()7H&p6ChUe{ON<{4@zKe*9dHr7AqByaHxaVVGFYY04G%5?(g(Z!9B6Ru<`vUWe|94dVR6B<*Zfa{Hj=)pF`4~WMM`ZNU(Z8BD2l8aKX zJLe=yUoreh4#WadDJztS-b*cd~6XL{i?SB%#Ua zfEXCo!COZ2lcqx^D8gF~7ZBCU7$%4(J+BaDw69*89h_*hmIbyzsW6@YoE7ToiTMuM zYt(zuibmnsJ2W};#VjaGa4xGdNSidf-zh!?;7PNduBXy-!;cdy^pcd!-a)Ulj_t~t6{M^)-7 znPCPEUH2|eoNR4Il9I0ox$liq`92`dd;v}&%p~y0S2cz&VfH(syx1ng*Z@s16GF~$ zoZYc^!l!D32dpwzu}mYn+;nfXbkS{cb;KMg2#NwL{z2aYiguolvrHJDu@O za9#LUs4L@otJF11P;zMY>{+IyhX#ith-@DVgpdbGnZMY*SkFuc3DYu@o9JN-Y)ZyWR61dayp0o&u>HdK@s`fzRzN>D>-3O$T zUqHzR%{i>vCmRhcrDYY=;7v9G)IUc>{?93Z=fnd61jfSPPp?9NzqTY_*c&mPyyF35 zcv7^g*jNN^n}`F>;*!0kj^u16zi|(Pao*8{hJk_?L0n(YhoZx&oJ?OhocqZiv2dxn zWguTG23S%r>B-OwP`^YDVB)_;RS1*A1-B29@h5@RVx{q)rUwY&FPAt{vc1@48FI{a zNm%dQc8J!ohq(wFV$}q9DQ96reASNOI+FnMJZc58X=kvnoiy1mh@wbyQG$AonX@97 zqV(}hqNXH0%`}#ItEPKBjW_xDg+c^}$|x)ELVNH_ zx#ad$Nmev8Uhs>h(hkL0Ry5y>6O_E2Ka{4>o1tF&l%e(Kr#8y67PX&Nim6<`d{nkp z#_(lMA^O3aDt*laN`f*at;2U6*$WOv1Qp%{N9s!YIw>0jmBCs^ntChRB|QY?&S^&` zHu}0bF9fw|mi_uDtGZp{gjGpe#}--HdKHv})lu_DJHrpXJ)OUrt=a2v`Bj#`lis#? zVbmEZOz12#Zr2hrpI}+5=&$PEe$stir%p7X{gV`@wGZxIhcJ?@u>A;VZ}?rOU*#)z zatr7xO|3`nSe3oF26VUMJD!ck$o$HObT0GNe^{Q7zB>kd*>kJczEjq|(Z}!EorApH^!{}6$$dwV#mErR85Uk@!j5>w{g(@xT*YMSBGxS^Gu^PZ(zE{C(|k=5PnmwV)Ug6sG;gS}hI zhfb+hM@-1tCXER1xY{`|?Y-Sk_`YDP3MJGZkNM}hNykJKSqox|-v&?QGQ{APD?9l4 z=iztnc3VQ4oK1s+d!u_4Xa4KFqO}9S*Q}xLx@&mRpQC*V5t`{Yl)pX6#LTPG+xybu zEsM0e4{!K8dVG60%LLf;kQNtaM&zKjJaKKJcwBQ1q&GZ5wL%N&4@gQqpE0O0Y?a!) ze{eH?wtoHIQ1c0YNtyo^*X#8M-5iqT&U#?W;^{~68sU!Gi2hGEhW2k|R+j~qZfm+I zP8?InkarmA+U*lh+*MZRkHrP*E!j=2al3lQhp#@%v7N4O7|cApuz!I=O;r|``j>C* zB4}1G5RD(_P`;k^ORYNLQmqXVC^U75u!p&QomuC#ZtM2MxJXFoMxs6N%9dC^O^Ni| zs%^d-h@rUBR1<9@sc^0nt922+Fc<+^xfFN6*Ielby0H*9FQ$JQx=F*yC#8sZCaTtd zD>;fDJ8^iQ7yr4gS@i4CIPjPc{NpB)@JB^ojbhO%(WRE|aZ}b)q~uG!Yv{$=u*+BO zb*uNsS-Vds$UWX6C-rXaJYp9{99;!KiPcYz4rkofBD>z*Zc+v7tI-0yJ32-WipD)_ zSqH!OgSW2gLUYd)4rFExC~lntUzUro_%|cZJcHEov#K@+jz6Dy4NHk5B{PYhh-{uL zPWfKklz;xBn{&GQMDT!+@i4^2?uSms(m$i~a+V4AB}NFmeW~|f%fs4P68QQUbS}2p z2@kno`g-vrQt*)>z#k)YbNi~|;o06O5FW9)x#Jn>^yF!`KN7&9?qloela%4>)98D& z=yUbK?{qk5oYo%*bB0$qBUT1+1qCIH1aBw=du;_Jlm}Rw2c6J{cuIzZScJR|4Pn~~ zSVj&cQuHSwR${#uAZ+(*MG85~2xWtK3@*Ch!dxWVLsu6=IU;?giCk9{M3r^J-nRv- zFoP)%eZ>z!+=pSjqoJY#KJtrUijiUJ_u+e1Y6|yZrvv_anIUGCJ{Ii}Zy^zHn7nHE zgC_Pv*ubH$8pF*iBLF;+R)U_I%;7vv;U|j`T$vH?Gs8Z>A|F3QE+IvwT6v`_L~856 zGDpEMuaM}a$XG?+hda+yn6G0c+}D{Hmf#gSlKM2*BqR&X_XN!sa#>*2|FBe<8(O0`)>+9)Ch{_2nNGmFH@-MC;jt~@4Ru1`R^sG9;&^hL7g8{z73|Sjxzg=by?GKq zFagAzq$-(o)Q&e!6VEW3R9+EV4oHrq0UcU%l8!~K5<}M&K?^XZtpO^bDsbL?NP#ZM zaW;^4I)Oe}yc;d1nD!I7+zLPJVY4q3UL0DzD;V6rN= zSvOZt52Pvt9XN`bsS1_ClmE+s@+5f^KR`Tl99tccGe@}}uc}~M1Y)cbXMOPb2 zT_8mNafRP(H%&v4b|F~-Xp_kOiNZpI_oFT1a0o*hUioX+jBxDYNdb^}wU3O@D=ynt zIX*$1&TzS&KE9F_P|PtC)}vU3GDVeg{^=M-x-A=^KOc{@hFP-g_FYY7a#2fXL8lL; zQZ{DK3S(|kX$*HQJ9eQLw8W2xIF_Y$vKrW8L*gP$RnSRv<4mL^tTduuN6=eCZ$`N9 zQ-7FUf81Gb988GHQZFu4u*y=ns$WMb+)(3Ec-dKVFU&F5*$}$eu!+)0d{qZ}!bKnh zFZwhlB-fyHf$qi|8A$7~a>TA#n-q$w2z)^}F-`164P+ltiB6i3nrdlOz+=kI;?6}( zD%7df&HBle>>^+f(?;k7dIncZb1=F98P1e?gYX2f-K0fq0#$bQ%^VD*e(IpQ)sM(hz+Q#>|5@3L z@&h0uAbX~!F#-{M#s1GXoS7kE{30PLy>WPfh^iO3o5|q+5pp^;u zIyyF94=5d+QUfDS%*`*vAc3b>6^9o$x3(ckpV#)qy?2gJ#sQ$82Uh~fKTdD&x>~NE z_-60n06h}y;Rcf!@Cb<2aH$d?8j7tnZLZj~L*tLb6Ae+HDjrD)fJ1G`$E1ixQz#J= z4YQ;qrBk6qgww;#a@5et?{iskcQO;BTTs>c- z-1pl?E~RoRKA##TQ8uPR2iFoA8`^l2GSGsJ(v8oTPQ6B~zwmv1?v&%Jp+nxo=O)Bc z#-z{=jHNC}uVhP22`FZ+?OJf}A%`!$X2eC)>P8PNXnH89jY)u}Z;9w~vspr71Jk#&es!+z5k2 zg~^xh8#C`4-nmXUw&1&D%+UH5FkmTOm-1P8g zPfLuN;3GwTaSOA3d*%47kc5wn{biUgLme;^hjBnfUL-#Dq$9@_ic$U;N;SmQE7j(Z z<}Ge@zemjNawTe)+<0XPx1_BN%@_aj!}2iI=qi-#5MhWJn~B+RSxJxjadmLz-f>1F zEwyb8UAO)Tm0A7tUVZRd^hZc1XjLCgZ-Mo+kqRG;4PMQ7ielF|*}TQp_{GR<=~Ez^ z0Xy>IxepuatjErhE=#qSNeoB8OuE=cK#f%UrurdMgfnrb1JNgLGXQFDIDu-;LY5+0 zG!iYOw5w?O=w7FTLO+8asl|Yz;{Gb;61_6i!{NZ1t9<~2S zK*hh{iEM~=?MPt4BYig=BJm>A5jBb$vaY5a{~f*=xOMk^4O%K2*`+XT#h1zDv5b_M;cn?U<97on!^TlRC%Z z8>H`K(!OUiDuKY8FxSQB539c&w=Pip(HRw8GmESQqDYUO42rOz!yt;15&^x9 z$6L#xcT`Z4V%<){gYBUUbttMa0#k9Z%kWm0#^fo1X(;NJgnD|)I;6n#=2|-Tb!%k> zv*b{>=Mo0Jr3syv>##oAQuYz;Ndrorh-`ApE9S8wOI%>aSgJHF@^-H_UkhX@cAu4L zdEE5PRhBUR0o$0KimT6zo2}*`&1aj}qw^Y&ZNn1Y8HI1Qop^DVxh2BxV{eQ$cpwj{ z2aFYZ-`~q)=Sh=SLMePkLP0d}qIg|qiK>ThAEh;;mSg+*5of-=b$5dh%MFUk+0HoT z;TKTVnJJzr%#ilr7jdocsun=s**bqJ6IVQ<(N_Ku;&4??^v#CG^EfP@w5~w$Pq2DM z#anH@xk^q=R{E?Jgo^C53V5u!Bu~|pbMkDV>9-PXnK3O!FLwuZL|d$KeMs@OX8rNN z3Fcwv*J==UEuY+uzQ*zb$6iIx0cci zu~9>PW@1aaaz;0`SnF)uA}Ui;NZg+Ui0x7DZ00?XZgZ={Kj3CRL?QzAqhwq*`F=gwSQMMtCec5qQT3I{ZA?!3Mt%cX0lvg5YHFuOc zIM+QHm?H@k|G24j1zT@)6zg;=Hrgq1X&F;m=)|tx9NCcAhQn{Udew$beZJPf4(s`Q z>hdn0AG;Qxn$ye9!pXwjO-#D@)4I)98qL_zP0>}BBB|4*B5|JQfsVgXUzML3zc*ZE z9bBqKkUxo{dooHiYOh4=DTK+Xn1zooHKBfYw(TREiltos)OvI7N=CjxwaGUysp^dG zFloT>m2Ch`2icY&nbdEqFxtJ;zhL#WkE zncyxp@@#$f@n^Sq+ak(|Z)ROWyr@*hXJG$c&xQ_piPeG6+&Irfo{nUAhmpO*fc&G3PN$2O z(XX_=XMH;rzV#Lcp0clY_LLP_wekxB6$!`o1nN28&iC)Kh8`V{kh&*OJbAtZuU;Zc zbT8sqT@0=pIIkP|Es?<8`!t*`z3<3A-HJD~qD-7!mFMbhZE*Lbbva1qkd5b^w>wGX zpN1Rz?tV9XurPq6*QxT73$@uv5wfY{^mLQigoqsL`e>IFT(uPv>BX}Ur34~IbD=QT&CnVOgV zWcj<_8pi(cT)f;rT^|DfWCUsoIO%;3Oym#BjtE3s^4Y1d+9UGX5b!z_0A1?10f@~A zm_bDCo+MrjzZG0SFkd8QXS_;ph?ni1PSC=g18mELa?#IT$7k9z#OpSwV1f8gXz(?% zD;Wfoju`}5y}LqeIU)P<)%hsrsH zp+i1us7Jo1gTdMvJaq$n+l3{FMV#-EX?A?GzQsd>jpn7>LMX_ETHJicy3}F9E;0NQB&j!$w2Gps^`G zVhoa_)4c?!Q^31#(Qc&Bj0K|^mrPuPxcE{4%T3^;EtD7yP_sD)IFyi&Ck~PVRlz|j zR17j(LOETcL^9+0#TCc0MRPVPJ9!9=nd-2Or}B! zVL1|6r4x+7O;IsRiI|P1?Nty^^69;fVFVEKR3*a&Ql9C;$B9`4kF4ZS08fZ;Snq6V ziFPW}7&nhn+EY==hgtMdMK+Z&CY=Y@u34gA*y+yP>9ZPfNjp43iD(QY_+C5mb~en` z&ERUB3}Kp7qP-NYoix9t1e14BY_h54$r#3C^!Obt>PkwSHq7eWuow~++hrj`61=1= zV;>+k#!+I5H**vVr+5|yNk7LKF8%Emaq=aPEls8$4w4F20=WwpaVURWQCiPcrlBqn zk%t-t6JSPK5Q0o{*cZja#1^<%H@gkCjs zNwglt!lQWpu*iog@hy_T5RXAEhny$ggb`yzY0HI_)M=IN1*1iz$dAY+=0zEa1uvsC z*U3bDxP+o8#WX=h4SGcDf{ZpzgcBsBq}9cEmxcQw__|I1UfC3{Ch%dAXR$5G_)#P$ zl*wmXnwDDnc{Wq^7!+_+s%Vz}b_Il{To&>!Pp6t+-xN6QF19zRTvHO?oBEgM@vY|s>ntd zCCPPw)_l^hY4xdTXGOx`Ms8TC>3OW_C9Uljt{v2`9rmeB#s6SebY7gGate6^WQGnln`*&uoV!gxPS87caO}Dksr!H zgoLT(mG6L|xlP%z)!n^b0K)9w<`rb5pBI-+?I*tlk(aI>9#fExZs45fXJm?(Y7#*Z zJ%nDJ#-agXWISbFnysRt7-t04HRce&Z~`WL%9(u3w3j=P`No}YY&N7f6?!9QH^wbF zlqtB2Q@A|4S7`tWL8VasR5??kn8NxGOKOydt>lt>1RW?kr@q5zG}GEnDNTuC|J2K)CjO)T`Rj zT{<0yLh<0IyE2U_C?Ik4Ou*~+2wa)Gl7=$iZeFLr`I=8~i!7_%W6%hTL4nFRQwdUt z-t>o2(|LkGvsYVQ_s9D;Zi1~v7{tYgd8|^pKZ;|fw?5NifM(kjfQQf1-=lp(XlCexS%I5gQBh&D9>{G9wh@ZE-M`%tqn_0;t z$U*4E3UjWz;R14K4Tet@^%QP|W3X+%&+=|x&|&8_U_#sr0-0XR3-aZV%jCdNc2mb1 zZ~{OPsFu0(s)r2W)Hp#+)$j_1NT#UCf^P0cVNew&>Vm?M)6Gi7<~of@AJ4yiJwa%0 z712{0VntQ?Z_A*@5n$Ecmdm zMpCD0vT?}AxRv#h8>s?SPlx^{v9JqIKta%@H}*eTfuH#;oFN8E;~o+p)FiOA#UYS%r7uFRx9F#vA_ApHZ=^eb7AGuIkQTEK#*k%<);A)KD)<(n zz6QWODjcMNLuVXy(4`6a8i$13a0ECssnNedDROQCp@I3m>IJAc8#(zmyK2d09v$ADFC^Vu2-? zC9IYWOcljrz)~a;)sP3Kacca;)O`@u*}+VqoU^#dZIy<+X>xJN4PtYbRFM~+MQ$Wt zhPtY&8W3Xf^x>CtG>uG<2)p|9#_n@FPfxz|apm-_?PGL~^*fSsW}{$RIaV(Zd?-2GCh+o8gIV0^WF(en$$REPxW4Qv zGwA-H*}xAI!q|HwwDg1Vg*RO4{hqptt|pYyJyNM|PgQMsPF(ym?Csf}rm^0w#X9rt%XiGZg^g2}%&eX;Jo+q3 z?7LS9oZ8Z(<%|w|zsYkKJJee~Tyo*Z&&RSnrGI5GH6hR6l2S)!^ZZu6aZuo!Fv=Vz2zYVCUAt!#wr zrc8 zI_#6pm67ynmGc+>7`M@N_Lq;AmjT+{`uyj)-wdpNZ|Hq7+dt1=U$wdi>3#KPynq*; zt?E5f>i0&KUlji_um(uz_a*ILlwug#AQ3+Lp-04%adzu_LI;%{#5%eQ1gaT}G9xuaR{ zmB#+E9W#Y~z&o+*Y=>E20w>>-Y|&X3`*52u7H92>MY1lZeV4RJzAuCOQ)}Ta?IF)H(2-=#jj%nhDb_0Vgk;-KyS;z})7lq}bSEsmy=WQ3Ch`^OZtR=$ zb`~Rtm;$oB)P?pkp?I4F71E!%+8q^on6?=oN%u<$JF0BrZF6Es4{FmpYP{i?9}8+o z4?huh)elvY$_lWnI3YcrSnzJCda<*(Al|a{XmRVxL;kr=c5<}v*r~MP zkn)x6>`MEod&J1mu^R<_+Scwc_5-M1wu05} zcaS&k|NhF`PD;4nBQm;=UZuGFl;W3zV(dK-__CKZb-&NF@sM2ba#n@)aOnB{Hs1JU zw@K~cSitBhee>mKN5aFo&c;(V{AIsy>fyA_=sBMXelUjhcxkoqTr2@UoKbuH9clEZ z+!%heobY&?vGJ!m5PrNp^?1Fw@!0y6^6E(K>G9zEU1!GoZ&6tH_Z3Dj@OlHWt`n(v>dJv>aE$F@@%F7)5;X{bO=ijm| zqML??WTlWBRESbH7y}lfe#u~ME`ST*eNeFfx*vF;fDju7uulOh>3V2ahJSC>oxTO5 z0Z}AN8LVZ3Z>)m3Gx;KeLs3kGy=8ob4}C>p2;%L?Z`y%G_{h)w0c8i???w@%Mv>cG zWzl3{^vyx)kPz2HgqX}omXeT6uHbnm4kj-Y%Ox;VC{NLnPy#cYUdaq=n~QRf3!l1} zJYP3#!O&Ne&0wN z%M6Y9d>xg98$lZsnKTaR75NZ%+g5<0?Y^M(ewZ!Jf&pL-H6bBIw2>XlHx=bJ(jAaFo+G3PmO_2Z>It9(BS}YSeCmuV7BARQ3|BaJN!OX9viiMJVop z{uqKXM3VcS2j+sEM)!;G8j>(h0fb;RHbD9siK=jDV*!b2S?)| zm=(V2%|(gE%^EATpe@A8#yw;yYLzM?h{i#TE}~UWBtylcI4+{9E}~J!7EvmO0*aYz zaX^U$tUkbvN)o80~#D>X9krEonLXC4t3sxx-_*6KRD@ zcDb#7g@;dtS9XO@XNBKNg*VQBV>?;^JHQ`^@PA^vzbhaukubr*|9u7I2hg@tW#?v% zT@e64{Xg1n|LMB@2y}{#BLjlm!~bmsWUw1TOniQ<4r127RzM;l=M~hd0l`QWf7$M{ zF92j;*QJb{)cSV?WCmhx_lPXGc;GMF`B#n3iX(VTEiO$C{a6zKHZ5=Odzu~-LKzGMLS;jUN$)@mK9X+E zD2@qlEEas;o&1Q4a@0FfvASF5%AyZU7DX~GZ4&e?N6__nZLFnYsz5Rnl_J{}E$=O& zYUf`1HEul+n99M2L!hnduQ5a6qzpZYsf4ft6OTV@=_*rAjAQGK<*t}RS+;4P(uX9J zl}^ruKUUUkxSgPP{5fH{r2ZOa%FAvM)$$Hp_t_S? zT~)pdIO(luJYNRNK90561OU*!yNI;`OzD|5g9RP{Gyrg?Hr86pJPd&Fy8Ofqr|KSIEG9sS}$3W6p5ikpAuRsCj$_Y zeu#{HQ?Fi{=7Zl$i=O0+0!!3uZ7j?5D%4&{u-=jSkfq;>0?X!LV_^z*%Jx2>G750t z&v)F9>M!to)7o$5dy7+9$cXJz0b~!$tSpYL-a5>qEXUq2mD0}cFLMcq1Qq9=H&s^9 z2C*`gGIuLZhQfzdC<7lmTv@BBD^X(ua9-MSYe}`G4L_^!>cZl#lPBIUJTBx6=*wen;AzDAn~UeIC+^!j338K15EqCb#ESU|QA>{btV~1FV=hlU7ICTE;n71RN?T>y(9e}*4@w6@*y&>kH2w#Y7tMfui z=b3Q}36;rjJ1A(GY@aVmJNMeFtT2)~R6&^bfwUF8f&7&J(Ib0K1u(unZ#LiR!tM7= z^utT0>lOJv!qoJP^uw3J_sYECMfQfoDdFm7Ng*NZjwmru_SaMc+Y8RV&U66Ip}Aw< zqkOkS@E@5$P6@={M8Sa|I9kZdZHAci;HhT=SPBpfl?2@;K;Q;jSfuS=kuhmPQ0xp- zAi9#7uYsXip%hnwnE*6YU|1>MHj_CA#Cg(iZP z9D<@!DaLd^EhI!vgBG(CMt2K@bs7|bvqlk^u7S~{__QeAhu~N64ce%8wbTSS#L|c# zoV-j+w~3QH<$r!o@)k~|AoAUkiwJrfqr$s|ys0QAuCErQF;{|roQcdBpac!m_^VuyUjeFO@OyMT{9bBxC}@$^mgl{1*F? zSyJ5=lmtRb*2+%SbEjm`lY`eb+|b-KvI_XP-Y}ajPf~~=HQDM>u;c&3*;_@$6{yR) z-HkNvu8n)p;7$mdK#<@b+#$HTyK8WFcZc8*+-W4Z2PdH~Yo9yDS@&_@_xU*I81>Jp z`s%(f%}I>`el=r&k(w)UV}VZ0sBJPC{Z}S`Uy0Z@@~rzsLatqK8QXlgyg7*@ERTPZ zp1bvUR$9I(j`AOkw4dcKdgI$Lgl=LPI4Y%2p-l!we+)sy+I_~hU(I)A}FvWs-XIt>9wa;w;6At z+0cmq&}(vjjsk+2gIzmVL#MQowt;Rm?bm{cfrNj!qRw1b}pz#7y9dLf`IX0pBN07I3H`!im z6;<_h{Ck_!xdmVK>2?yEo6%Hrzcc2T@-B*I9(+kH6Q*u0{IX#`?ufnM-NZ6Tr>|Sk z-`-Y0R}YhNaY#{mCTpFYVxEYHH;8el4v8}t0I#|h6@ z)*r_whqW~=O+u#MzmVJhu9EZ^v8U%M#kPgYa=UdloV{b0MuAEf5)Tyia0VruK{fa? z7bsJmqmQGRg(d(qRbOw_-fu2wk`n(6`^06I7JZ7#Cp275(NfL#?~GpKW$f2y_WDYz zGclU=hH+Q6MpN_+YlDpC+#44(|4ji^8luH9wL3T5v|~%c^$qJDiL4N^6H&MIIsebD z)oD3q-lv&UM>WYEMtv6_=AS3w!{F2lDoH`qnndT^)9LVELd zieJB{w%Y2(YpibMPTxmriR!1hW$bpgxTkaGD#w7yHsA=|`fXvp76n4WvpZYYZE=qkHIb(C+N>YP&@5Gb?=WI&bv)Jwrz8RP~^v z@i&I8k@fo4MsC~U%&DvLpwGnj*8A+>CD+awvdiujx}DmshZGu7qv8;F)MdCDac=$6>*yW}AWp5gwv- z4Nx{TupQ?k3OZ7XJ$|^6@hsw?W_J(S?F$xSz@`C_>udUgExe13Jp)IW2Im+pk37=S zLfiP5sgM|V?m~|b18~Ltib0`9VPQodEFiP~_D5meh~aNhI#)+FMdR?}&~Pma(9v?Z z+e(;cI&f8kIpNVOA`q2NJUl{$qkPcMZ8u^o4~FI=D_OKcf8rF;&KJSl;Qt~OSV`?4 ziS6^1n)zigOm{A_nE~KZMibb^DLG2}xEpyZ4!9Y>nFWUQxABd-(^NiwkdbhaJCn0a@{;co`e9nx`3ufsr5FFE~H?bvOr#=;M0WQtNIY`9S(U8vhn)p z);zYoGIo+tcB~}kBxtxPSBy_x1ZY`EX2gg*4j8v=kR8b>5lQAq!fe(Q2aJStgqbFdSY?=jEvfCIauPm+VDADH$D2JR* z{tGMUi8eHXK0G5m@;W7I6}#XlojE=>fjBBz4ldOrgQh<%>Yz6=E@p~ApGr+ zHH-}Exe6TfKnaUSs+i4=H^;Sz&sl5F*;vil%0RM)W-STi?rY^9c;p^s~^|NkH`f++Ba@BzTS|AN4LZ!?6DRV}pr#pOpr3IG2>U;wuE&e1W* zt~z1=2Lj`V@P=@S%~b}vrT>q;1;~0kfn+oQ|6^| z{)x#ckJgc2Jo)3(OUsRKj@HJz{1xa<$y(*+0Ufw|_w+1f{qTwkcyfMsA9Q*BObY+- z3iD|?Cv`UxfA5cU350x%&+8Ay`X^!C#;!LMb$|0Em;q2I zE#ZBol+AhmPB2%59nJ|c^tYzcxS5kLj))03EI~$#f6`DkU1do|#Zgv0ygU@n3d>dZ zf4L@CwXB6R8D+jF0_XBR7_(!SLKZ|vd?^b%UKJni;#XjPA-X>)QGBsAkpOUcr8$BA zJnm4`Itl(!4t_M~j}Tv3ej`+TCn8xO#dKbNcyRW^4pLG+BT@tp*nKwtfQ5&Fd(xXfRHW%?=aC#D1dOweHUsenLW)> z8j;^qzMw)NjWI`y>;fE@`w{p1>$F|h(?>GP{0ypy0*}DvP{Kl-LsI($x<2hvGTrO_ zC?Df1b2Ia&S(37D@<3+f=+^)o`*|&+EDSDr6>1n#26Y1~_5cQpWId4SdXdUDh6AEA zuVNkEylmD2Zxp_--JkA18S$Zc-`Agw62K8qLp9^+U@iVc0krQTpX`+BRy-3NRiZ}P z8|nqvL+U8txxbV`uqU!E2KIQ#>e@dESA}(wT`|hGe&VQdG$G^Tbd3FHRmTO{dCs~X zBXMWlz~uuG4D`TB8F5YFujpRmXl3F{7JHwCI8uo(XLI5HqVR5_vLd;5p2tB#AD-}q zc$3a~G3wn+YJ&gq{LJ|M+zA|ulXdq^|g1Z zb>lN2Hv$@jr9ZT&NwXEatH=E=d~f$+$_=(*&Mx7xNx6{&K9CaMYT8%yo(@~42#*CE zOQvJ+oaDBedqC>IR6^gdCIrZgMIm9=I39L1N0s%-Z^EU$o# zeh8bgEFXoFYXaU5t$^0pfCx@dzS<5W%dDK3Z=zc{-YzrKth|gqPDp+7E-QnwfJGxl~)h(O|OT|qNT;<81-8UZn$$MN`8SN<p7(Q@PE@+q=a&D;#M|tdGA`m zuh-DH$Z^`n5-oXiU)+QpH7GxrpPvwM*MBvt6zx#1-;?B5X4yh1&Xb&!pZ0YwKkd+L zgIu_+Q+-ybHv=O=g>>M}EG*Bgi<*O}LMF-yw{Q!dkstq=&sB0E2`7Y32LJe1SRhXU zbqIspnE6~<0Z$2TPC16o_()O*SMjWFIjzUI%#XF3v^Km7#)xs*!e^C2xy5pl40*Xy z65R69n3N##Fx=6D zuYIlU!LQI-Wcl?>uX8-=82u&+IrS2 zZ)rdIY1E0J;KV&ubtG(&)FCfC+jzxCICu?08HDTWX{*`-_&emQieouq? zl_&AW*0LR5H@>>P>gtOcsto?XamrQbTgUKHpJ<4fz)@YmTOCe}sDGRs+u7hyS5#c1 zyn>dM9i3EYXpOz&bb*u8qe_SphB<1T^X|f`D~!~nt&l{YGeYNVr*DA2bE5bx@sp0} zg1$hH7X5i@rPu0ZC~p_@1#?vI3G`0^(YWQsX(nv#_k*v%fYH`PdZ+a!?zX^?NA6|r zsm|d`u0U_V)|`&0=2p~N(=7R|gXXIX$u-NQk`jJfM zQ(TuRY?!MW2#y5IH(a&pTCwz!OmL;%-K1B68G18{zkL0Zyt;8bw5ypaX-1!rW&RC` zniy=uR%L*Rs%9cayyBR8pIvUNZw>P->reetcdTp8zD+XOPVlE{?62PKOIlHW@6i8*97-n5qigjFwzB}#%&+5o`Vb}*)Qz0 z+aV@BM+F12IpXEpB7sc@B{8x^D#E)_xlJn_KeBlyjk|HJO?!=AvZbhbmphA1C+BUl z<@&D=H6Ad1=QM8QEzY!u$>U9DFBsXT#PY-Js;!9|CHan=#^W6EmSdP0`SusW$Aza& zJ8nO6s8QPE3i8(XAcJn?bmZ2S>;6-rri8!Q!kt0%si7B~f|S14C!CMU+O z&3@iC2JNxmDDp_VacXjq+lj;WIyHCvAL~wfO+!hoHF3AYHe}>oxw-JU*CyBqFO!> z;PyX-Ha5Z1;AG4$=$dqLZM0(PpcNT<|5n=UFgm6qY9*q#P8R-Kf?IusMh^+@ z>6)%VhFd_D&V&eV)FJrUF_fV(2p%WI=T|VdikCfg*mFMw6DfEfOs5h4USfsj-u&$` zcX*J8cThMCh%dr{FVPVE@h4RlAdJtD1@en5RFY(>fIDl?W0;CLl#3D?k2DehgAW@; z$oPWz8%Rcn#GFv?-;@X_Uk=j+d4<-oHZrn26Ec@Nf{W@AN(N}_l*v%}-OI6~LqX(y z0j#l((TcLf4%{JuhEd-E#4pQ!Q|^&0T;VjvG4GATeDlH|gFy!AJb~`9UUh-LkARjI zJUmkFsm2lKN5I=sLTEVojXTwoW*lyOh?hUrz6$sa-g0&nM<(u_fendN303+Phr|Rt zS$S_a>+5%ff$c$Ths0SDfJ^a|03lAKj=&Xk3=#z<5N7}YNzcg0Y|qF> z=E43i93>Y(0H6yz`(IxovicH-A&<}u{$f8`g^F9P?4BJB1L3}7`4S-fe#14=Y1HeFU zZf~>+5@Q6K_pM3aH$l%B`hMpONqG)uZp(WV2i0Z&M8SUSqp?!c*>42I z8RNTSvi=K1;Gw&E&*p`I3h~Yw%|jU*dJyZBwoU{?=^DM#B>?H=tF(%!?y0_i!CUeU z=nscV@qrFyj6q4Qt~p7A9ks!Ws6{+_&H*132!~suv)?x>?X?k3uvy}730DRK(Tls& znxYxJIJkNjQ2qcl(kZ7(A~-`AY!tI7M6~T>;l#|*r)0_EuiB!W=jnHnXNtCGvelCs29{;~bzb-XT@lwMz|dQO>06$! z0nmg1cZVPdN*_nMeqh|OIRwz9N#h@aNFtqYjKj;1jTRqb5f(;S&_@*%6C?d2!Ywbo z0GUPqXcGltz)%IruNxB6BDJ)}3C<%~E^ za{rRLf*EnsZ-G?J6&jrAuKUc4$kM_%B$cY7p^np~@>B`#r~X0>Q?vN&a1t;|&}@_L zPA8G#5@r{FN_Ks2b>8e@mh~al6KSG9&^uc5oT4XDNTP9#p46o2NTo8k+hvHw9(0@A zS>R8l*tv)Ex8u1M5y)_DOEwqmK;eDSQyJEK!y7#`^lEQk>8q1W8f}o{84?*MH{v zma2y4A!KaLb84!y-?26Q^4{aBy6N>Ve+WMBUy&QMzF+;yknfuKC28tz-2#)74{USk z)w=3|TK|CPqhqtZuKzP;2ugzYu4X-40m~gL+8p{g>gOgR0K-c0({)=XlzG>_EAzeP z&!Mjr#nXXUx*g9+=mWE0Uy;z1%rAUhyAP|C%NY0P$}%m@C*h43L8tA5ukJ?$q}y;e zct`s1m%}f)yjNN*aRS%wI<>+rI2_BW*Z19?6)LYpM&pUqrn|VTQKE8 zfN#C(64rc5FDdW4FtQD!5i56L25pQ;@gpd({85+%=J_u4E^Fssdb$LAE=IJR@CwnN ztDfy)b}tLvT%xqm5&9lZZ^w;XtaX_Y>db2w53(N!l(IvW|2RF~{!T(fG-cb#xjZ%h z1`vE29sVQ2<&$vRPphLj2CwhrmvPol@3A6>=j#-Zzu(VTx-vSP;}oQoygMfUG@wy} zW7dF1!scT!@gdP6LC41Q*b{jO<>ETfhi9K0d^CX%N=h6E-X4c)DiV*!hW`4#$8)VY z#i8#U<8HBcmnzV2S5j}#c)Eig%}-@Aoe<~3N+Te>qDU^9l(b5A$eUsLS->|bcBg4i zbi!i@(u13EeY+>@s5!&9m6ZJScT4ie%B-enGAw26`~HVo%`b8XPFg2TK%qC~!-V)eq!}wN<6Jk`p+|4-{us<_%tX(j`*Ne(q~1 zyS?HTyJ3}naFhR4Ao!zah1+YygKE@MZ@(0wcZ~5dL#TZP2>sH!H(NE z@X6UX;&t_(TP_&laL4bK8P*k*~N!684X&UpZa+3(2~wB~kuUm$l|l5lX)8jE|``J5<;i=~ym@qSyI?C%1P6 zA3Lz7Yl2&KcBgY6YiFlU!yk0^4z?ca1u)KHwKn!HULPCj6b>z%H?vC$7UL>fuMLpMQ@bMDHN~3LosIfdj(S1}+03n9NY?w7cypz5v{cv$*50Mw z2e^@&*^9j_i(>G`hv{7yr;%*xeSHX8-8|FpCEd8DcPl$)Gj?X70<`7^iI0485Fr&t z=96m4$fs^lL*3Z9*Owe3*8|3m{a$QHZF)zH4j%;EfTN#z?2@SbmI&+HbA`l#Z=~mP zcMMbhx$pYSJ~`EzjNSkA~er^|M}0*h9^q5=^ds= z=f^y*GXLyjjpeIElXxe=U`WeDC?;(h7PL{9nhhLVRY^@R=2VMyHJ0{snTWJiVmJrw( z`(yuzixT6XoWyd5fp>wJ0fro<#Hx{+PpJVV#uz)WzU+tlfyO^t%p~Q9$|8pA4}Mq2 zm Rx37dB^kfu%aYh?*2Eoja%U<;wy+1n9Pt5uv*E@s|&-n1gso=3|K_`T>dj{FKS=>rnB-;j#cEd3|032_%V$p6B_DOph$} zVb;q2MDaL5PKo3{mXS-0JY*pV)_sA{2Bz${oVyB9euF2OG7WJX>uagKSAI|kn4waF zDOZL}Vw9euEHLGoF@}M~?gm85PpHT3^L^Q`lA6WtF+`9mz_HF3nc5$RPl&wE@fZ}= zlz+QHME%I@+6I0f{uYsA^PLSy-zC-n`Nvy;NHxJ+{b3S2 z#DV-2zG)Fev*AnFgE;WwIKKmMe1G{y$wo3j!nVN>ETpIttjI=2ieQT5JO z6um=G?`S|AV>BfsYB(LsDlD1_;4$%tS$Gsp!V&Td2^&o|Ca5)Hc?DytEQV#)b8|HM zCsM3#Sk#^@8j?k9oVovT8%ifM;!HCJ4iIy>f`Z2%$5)5^5RP65iF*!bjD*JiBZ3g) z#0$qGN^8<15F`6TkmkeVgUpEyl@ka&5{NPq3?PUAriB0Z5)nwG)k>s)BKU7zF9Z+; zC;)f=*Sh`=Krz{|I%S$1eAH&B9p-bg8v&c?~Gt>X%nIdhDY%KAIy9d zATT&IMGEK;{~ye}4+30rYM~%7BKHk5pP5%w!?+gcS{^yEw^Xu`1GF!OIP1}WKFlna!2_QX7(1&k+@-@(w|A9VGK zg!NRROADmZ%1ocb))mRHw(@lfSu^6-@PN=9=dx6xXqSC8dc-{#ksbkIlpz@!3)I{+lGAOq}4X0aMCc-ZPY*i6J}u zJU3c=P1d}605?3METWuS9sB@(>N972Dmzk*i5Oa{kQ*01HzFCke#{}tWV)oHsdh2Z z?-Nk0DqpB>@p(>n?iblbxJX+2pQF_qH905pjF1v6RK*Q}$FPD=xZ_E|QDIEm^Ov`c zxq-IsVtEyef|AK9OL8+ybN2*rNfau4kqz$l;3H3l5u=s%{Sg6P}8 z4v4X{KZ5l8u_!}4#EFcMaE(_oJ;^*Dp`mo&84Ji!3!uvpy$P~`5v+Y0=BakTDt2+!`&77#3$BX5gE@YxzyB=cxrukNOrt_O0ZXI-!xByfV%YcRc>N z9M@T}0?GFrvqVadw-`l%)0r8fX;2^%WyyDdvSgyy2PSw|Z#64RP@bp!pqqc0C$Nx9 zln=gwrbfPqvS!Ku)SHIPKLxwjtn9SL=k}So3<-?}1S|K5*7#8GK3ZX3`O%>$4GF14 zb-O&7q7v9L^qS8IRz(bfZ)S^(0=9nyXoWRVEVP9+@YZ=1c3$3eyz`_w60*kuv5#G* zy;5L{tJV+*gS+QLy{>YR%~{yV(GO+nhHwqE9kB^dc9@&aDR%0}mOPPbv00qO8_2h} z#vOlF`Raa{AkB^zpM>5gI3@98UOSN!!id!NVwun=w@{eB#JkQ}!)EW-J%%<1H+YuM2RD^o^7$~m z-Ok)F{S6o%f0g^{{h5=c)xX?@QCGK2eURM(!WN&JsL`U5ZJM8}Dq!94bgV^Jy zg73@T7Sj1cH{3LP5cQeCrSQ=JUISaG&S^G9StP<2%YNK={6n5m4u` zkMI>M%xpft{xf)BLjv1Qr7CqVx~!W8cEA=z2M0w)uoy&Ut`Ah~%EOI2k`_j@15Z~P z;U_JR2z7k{JK7i%!@kO-2RIt2k(nSp5Djr-aK=dMnh+vqd?5Z>8%&8;gikysD${|P z=#)%L&Lk%$brBQ(bJrL>UUQ7>**;81ZJ+T)Gfe0J5@)qfizdJ{q(se?Tu}sM60#ga z+)~jfSfFD$hRT!maV0c(Q?rT4@hgM4)9#56$Ra(0Om?|4Y;?`og~H`TroZ^C@%%>X zB%ZK8naTc(dr0$zST-RaD~EZ59(mP6b@pmyDSw z>tsHMYSEBpSV}I$QcXzT1w)<=Th#=rW;XDpER}=>Z9Mar)X8neiY~2^M}(@~I8P$$ z=DzL^&!w^wmx^!I%*s9?OV&Ww>cD@@<0`}&Nf(P%?yObg7%!lfC6`?IV#ZIHR`bF7 zu0QtoEzF-*)Wu5h;t@8hq-AtQTh?x>Rcu)`C{MM^2JUMc*qCj_I%j7ulAC`ioLMG~ zue47)wPNeNIgS;JnFlUz&9L)RTl6fQiqw{rNNI6RzM`dZ6pez3;YuqVlnoVnzLK4( zeP>wvmVY2#7xgDwXMy!C_nSA|Vo0T9O2ozkGha_;&ZX6<*ZiLYp5FRRd)}8QmB$LT zE)YF?fQYqWXG!Z1Tza;kI6>X2@dpf@MO%+!ZKFw0dOO7R!d(Nrsv%Eck?ay+s(`lE zS>ZPPK*ni6bE3Xo|R&^WDb^3_XlX)B`z>HAV)|C;BxLNyo4nQ4@9Afm{aS$_lg{DfrE6SHCq z5t41XLuakEB8C-J+e#k6O*zN`C15FLrD-*v=$E0H{rbZOAH<(hLYtV8WQUeV6y(Lf zQrt21gZDkL1a_CQ3XymjiyFwOPzI1qJ!Ux>pDl>r2hgqv>3tYvl?j*IGj&31=yP-Kk@7gfaJ0{E~Zt-pBmW zSd3ts{Cm`Z{pX8AroQEs&s4pG1N#_G!Uk<3_A}=*Vgl|~kf&AgY zn&J9}h{P*lk!|78E8($^;qgQfiTn}C>B0Z4eWd^&07Af}|DE=wWzNGxOrI)K!z02% z0|1QwYhp=HTtb{*y!6U_7s6=q>pxs?hDro`r$A|S1fTy*ESVtM1;vSjL!;g%mOKFd z@mT^ukXzc@#F8!od_Z>jJMf$Ktxe9XXk>w_u4`-eLI5}Q(8G5O40^=${)B*rCMHb) zWuw3F;ieXsio565aI2QLrVwY>dw@yXhl8NKx$&cm4#4VZPm7lp!sFBP%ip(&rMH3s z8NYH2xhTTWlowu~KZPn>E~9Sv&R78O$1PS~*$JRTl?;~l?L!jBKE1QqK!zF>@{E5< zP)E)eu&T5G50CYLr8#}+XFM=jZo)%j#4HzcR7t$nNT~{X8GVc4iQ$>^?RPsNSFSAui^mk%I|P<=B$2W0PR9JElN2WK_xDE zssAu0Czcs$A98cnM*xe1gUX1af#yz}ZT48uf5I`=2c!*O3E>R)(3IBy>Hf^!aeCxv z(4Q0$%j!aoI~y)Cmw4Yp2)Iz_YsECPjtc$ToVE=^dOD8 ztVxeb2Vn#P@y^%#Qj)2~xj-{0X)g3PPEDIjNA@ep4-J$LZ~77`s5|wFwG? z+jbFYzcv(#Mx6iFB7JXsOXbFZ6t-YbXHk-h&wr2q)xzxY(F9R~OZJ}xpk)h%5Gl#nEVyw-1;W}Wb%ui2T4&j; zTa{*3Y1}P;qPoWhc}wLkvNrKJ*IlK&L`T zkn(*$NnX7D{W|$Ij%*XwT%7}W$l%|tn^?@IvK8^SEM(QIfstX3Fz$_fP!N9`{&6`iu9W= z3R932MpO);6Ae8=n=~|L4$**ipHvPyw!l=Iye?FXdYwM-p(PnI!*`RcNo)h89p9Fs4sK!(X`u`VR_wouap7qY z3O8NO)2G%j7}_WU6;ubOZ>q)5nN9UjR(aM^isA1n z*G02bdq!#-9cbBvkUKw3L}E7fQv_jOZIDL$QrS3IQ?wIb1ri{z91eMpL2G zS;5Y{4Ywep_~x}DDOZlrTxoK?SyHCL23QR2$T@Q4)H;)*S# z2h7_Mx*TwO)w`EV0sX`5vqj6xW3o`BtQ*s%IjsUFT?Bn0&cI9|hs3$$X!z8gvyKI0 zY;i}Vmw+^;gk#P)3Z9t5PnDjhn(v&}6l1=9&driK-&YdgBJz%Dz~T2fZEENsy5Dp` zwc8*{wkxM~UWKd{5$p#uM5=v@DB{@}r{!)$`hW2+^HqJH)1hFmAp@u6rmPEe!LpAJx2N>hn6@t~Xk!1pKM& zMV!~G4Xpk7O8P6m8`eh_6qg#P^A{d>44 zsWcCM)C{)w5v!uQST$G%*7TiTu;-%3BKcwUQKEB~!~R{x_5M9nf<3RpUKykQE90O5 zkOMFSC;xYP@NIIF6%i3G;Qvl;+9A)1I}2lwAt3+&{r@$&=_28Pl?0PAM{xh2$;}W% zcwhfmK`_WV;y;s{P6*L)+3(?#(%J`Jvi@852hi;H1$yY zc+-Os2;kmPQn=QkscC0~?y-4fxVOToBjV46dEn&g&NSlYeAVvJ4E*n2_~XlIz{Xiw zOV}I2=>O_L**Zv{_aO>?UfQ@nm8uU&)+Ah_fLJc{;LBfU@tyJ*Iv5Cka1eE$5*RF> zS(Orx;UCA6Ry9ClXgrAp)}F?8aE@pe686d+^#J?N(qf#m3Jc=tlBFv$$qECCtrp=l z*{sxeZl@;77?jtvmL)5tt1KGVZjD0y18VFR%9zBL{qaHW`9{rcoihSvSIg*Sm2 zq89m3I){*VdhMUDks)~rG60~ZgT$8Zvj22xfD&zSW7u^MPZm|@0H5JGT0+oHvm<#& zkkG0N&o9?{wAQ(`K%c`JGLxpOoIa@$0t|=+TER~kJ;OEI^Y$j)Uu_vtpP`$h5qH11 zi!Pz?>z_bFFwjRR=jK2pPfY)DNb$i1`ty+EM-NKlbv7kK1sRCAY+#p4bW1hD_Oyl>cB3H#$`=2(f2G5DT4nGxsYMB1}uvX ze1-ld`xxS&<16|2G*Up!JN`Av?uyBX>_)AEF@wb7WU%Nvv>jM#!-^kV{5es(ggpEt zkAeJ~Pn~X!c;6ag30me_P)>+zU3P`=<2Yg+9E$ZStkf?QYSU0S?%CfY^!AN3A{%Bg z7B6W-Z86rII>~-;C6m^)HDLr)kR7ZA*rA=gRZ83VyGTo$ZZN*ocH=<3f_i@8X#VW- z`HJe$dX6G-Rs5IYtabo^YBM-kPofAOVE?P*I4EKG*LsLl3GHHlz)P~e{$Hf$%}PR6M93!mOLnl&!rR z#}o$@!zKm3Ra;Ng*623Pi-m9I&6)Ogaj%tTcK5gI5_-8#UX1H8uXdlXoi1{gO>wRJ zbox9jdytCqZHuYdvOzhJrYZOApFc70v&;7I!**hHL>V?jEPD73Q&?X4kA%ZTS&l{c zU-?di%4(>7S9?(~o^h{>(wr}3z6x$(ocgj~uA)$~UKo*5a$Ik_{^Px(VwvGM?H2NT zBr`l@5!qB&w|#%J_97~DHyi~MxK*;B5pJLR8P<;4V9+UwQDcDd+INz)-Stpf_`~&; z?v$d7phXxqJmq~hNBM#^TK1xc=C6Yb?Lm6%AsAv00NP~T^_cg~N^n@XkU|j1%eW9R zlmc$8e|RvD2BKbIgz%MZKVcCH<9@XCPn4&F$iVc*&x%9(@WxQW6jq&q%_#oqngP@{f z0B#&nKX;^Zl+|BK(tOPz4ztK$k<(q$p2r`2^bE0#>3g^Fnu1g%*j^FN`;SCgLUbLN z@iMsIO6xQ^ge4n76O$<^Eula6VmLj#ioV^=TJ)-hvL_Q&nP6!N3@Q;|r?RV=J~=$e zXza45G$>NDf6Dj?hmjNuNj7BYHk)`);hfNKL(5jGsfY>U&TyTi=KdYuZ6G=y-PpCq zyEi)p*8dYZ9=w9&V>xZ#QXdxzk3p@JC+mRXoM~KThCg#G{kdX3X(O6e)Q@ONUi44m z8Ed|fj6jeIFKUwJ{JzcK3{L+@bv`l9uI_A@S%BTC3IV*nfJ?XgWh<``e5;tpSfnGhU{AXq0U9L7cwN}irx|ck6b===0w)N-FDkF`& ze|RKBPS<8TL+NYjR#>Fwe=qj|QsD#9tuVqm2Gh*mm2jgg81*}?VAB#*cHUNeTDJPs z`0;&}>fMpF1j@3UacZG4KAS3M=Gqj~QcQG6p~((-HD{@+G<&nu{3SyzSNnT&k?r}B zKI+U0g;NVbm94$~%Q_#=Vp%6_@z}{PRHt1qsV1m+`CFli7Pjtvl|8-!qvg!O@R(bn3NVHU4`BGqrEOI-_-)zx`-GtE2WEep0aq+2X-ugvO&&JMU1FG62W z8$uKy91f&`wCaeJOYM@f%H-~gKwB#RLBvJYFYQ)vU+Z_eN$5X)lT)6@J|+A#5rOB` zGguuw{80NxIv@f&&kz${d;k}b(@Lsyk#HHx<&MP#rnuX_!qKY6c(o*FBn#B>u^d*B zn2XTfSV8}DBtNstN$QCP`Z0{4NX(C|t?~_FWXMf(++i)D`3J zGvnRTFBI5u6c5fzsg*ttAT-z;e(^baXH7DHRfsKUl|#nD!l=ec1o&Y(XbpW76#tdi zAIB79riDE&GS$z882rx~r1W#LVKub6 zrqGGI`%%1H&czCDOba1t>$`V2LU~*zVBxaeoo|hXJgF42XY$p9t1KqWud@rp$bdcw z7nG5_p^Ly_5BGt!7?j#*ScD}6^8g=+n0_SPKUR+GSSXCe`e37b-01338W7|1f~IqN z1#nw-EU>OKpqVqdOk7Q1b53&eKA*JgoTx{CFs~ClXLI-3{PoMG%kTNr%TK!gpD(Q) zxQEz{;jeS7ue;IF$7^1Gsa4WiJT388M5VrH5=MJukt>Odv>AYU0W@ST>-DxwsT;3v zHgmS~^{y_i`^;SFWo`ZSzKy{f{=bx)8bAp^4D9>gDmMXSGNe$@f2C8a5rPaIkT_-r zCGh@Bxw(l47=C;|1-J57`YVGL_Y403F1}F55jyVks)dLa7f6du~rlx1-SI0}1vfHF(B5qQ$@{F>&vA8A)$Cq$43?4c1MczFV3hp>!sELOUn4duE6w5adG(_{h8vC}!o zpfXOZV*g)-G7&X29C?-*nY?s(Ls(9q4P&m^|d-c{wUYA0u!W|Iow( zzb6+R$xYbh06^|widl^T9QDUmEZh{U6_;A|GI~5&L>Z^s8l`GLNQ>Sahnh(>PpeqNy-T(tE z-!AlHysoT9xtcDJ^D$Cx7JtO1+a1Poy#j?}rx565n=bh01#y7~|IvJH+7{u#UN*wk zs)+Wm#^1|75t7m{6_;*YZE~Uz!)zt*Z@lW1??vdAG|jL>mW;v|uMJ#)4Zx2?a@dO) zs^5fvtK(wux>6dDYAbJXu_Xc)fjlE6MxLbK+mK@D=lXIKsaICMc<~Q?kcn!Nm`UE@SXV&0w~19urt15(I7LSVcK}%Q?YNlS=rb=pCTc8!Nwytr z*%H5q7c#h;;MkuKd`RM*|Wl*!8;65Lk1|0)SgmEExi_z zpYqO)2`i~b;s_D!B*$Equgzzr6XYZ>lP2A;b^;%9fprRwYmhk2aLTT7Y=l6jk>xN z^li^;G-A~{$NWEnf7vmqlsIm3$XlQ5(U=t8%5rRB&rWEZ3>?nYbY9*KHDzOhVXdM| zy2n!-`(501_y$q8sPalxwZFnd&m|Y&RCViy#Ry4xYE`&58YlFuU+(ISoP}GaOw+vX zo3xa=?|<2@zwowNCT;L^yU=()w0krPyDj=i{pITlm>1@c{o3k(HwzzR>99jY8Ed~Q zd}(_FGnEkTQQY_VJS~8g`%~h(r5g8Vfzeb{wDzS$XA#e)YL;k$c^#o{!O3O?veuQ% z7vav%Zfo!VhqAYRh_Y?lMQ4U7y1P52q(P(v=@2BPq!CGxcIfWzp}RXBLOP`z=>`!g zgT?cGYwz{^@~!>uzu~^G>pafmI^tJ-*=URS#Ft2)-z~#vtv+AJD}MV-M{+)wK+TQ% zJUd!a{`_oY?f7>W)~_7ad#bbq0mfGph#i;WQOV|8u>A^qCR z10llR(+@7M=)dXN2$ODM2d77F?m-VEDM2`)t&!PhHV4AAVt65_nY;^+#U4?C3m2)V zTr3?0xju6Ih?f#b0>UA=t%o<$A4sFVkti&_IP;x$*9^+}oz(Z4`1TyTMAqbLPxs46 z_Vh>nAS52Mp8Vz7w1;i~I#JWYAVh3BpcoDLQGg7F;xv$EtFkSy%{e_c-8_(u}$vPHD1c|Bul)fB67O(c&Q3R zMI^SvFARk=an!+Y8TiOh#uPu|hvar1(%N2H3tc9m`tx~z4k=4MBDW>&J%KMTRrvj8 zV!z21a2E|Ja@^O49rkYvvFcAK<;=uQN9{-m>rbk!|46tS+mW;{^AIwI@#4%ZY`J>`57N^vpl2!}BRi%`wz5GJE zrjEX#(edaZPl=dj$Lq!n(zyzaJEhrr3gViVI+a?7BU*hKMAdA`6-w-*uLi}4>olmN zwM9p@C%uU4O_MA2$VYW%x`~^dbxw`+mFLU(iJH@sPv5A{EeQ4Fe{!{8{9_O&+1!4_ zf4`#U!+ez2Q$kdAyk{THF;0H1}_jzSr-Q`$Y7xRbO#pr0CSwtWDfQKX9- zEwVAlCreMc)nw>>zeIFS-~$s2f9cd+2ATuuG??-}p@1 zEQG96e93KD_&hDel1v9q_CqT8Pa~nDthIIT7JuUjyAE_X(=-GZ1wLlHZastljmZ*y z6+s5i7Z-s}Jml7)S(BCqJp1#N--1`#8k6c?gz~S7Pv{RbT70RmDP(wW7*-FIL|fy} zHgV$(^T^l)<+*+C;)rA{QtQ&#IHdIjsPD6Ao)7!usHBz}t-PwK*}A5#odnp}@);HL2&}qMgUv<# zbaXKUN>_9~>SXpPU|np#wJlD{)}Y=;fa@FUKA3 zf7M@^zXh10aABQQc^~nVn*^a02?x!^2sy65UwJ#p%#jqfR1f|ha*0ov$_3k0UQ6*K z5TmJE1MH&aU+8)T>2v~yJO=W^8DH9Ph_dc0Ab=lt!tZ*ol>JN8vB~i8CW^+=70!yj zmpzp$7nnw1sPf|!1InOC8Hf8XQLlh6(APxgwZ{jleW_l!Gmq?&sJnl*DMSQ(jZwM9hbjEKg?npUOQubPK;Dk+-hQnHA9^5R!nKZ?aPtiZ)-3Z+-89iK7K7?l8wgl{P?PNyIWw2R7hL|_C zBD1O2Wy6gOiMN3yGdS5s2H=R<#@W3SlP;E&h*(1gfre$X-iAF|!YtkG_DjTSIYKJX zQ6qt6@pcOKIL!ta0oCFoBRe(z!D5vZMHI)35B0*}io<$a0cC=6bSeA5##VyN!UkSD z<+l`T{FKK{r4g9L^|T(DXf=!F;%F_i`DRbTa?0D$x=u4XI9rX53^_Y!T9sMb?~3C~ zuzr)AIM%@@Ie2TqA5AZ?FN(#T>v!h82fBFdDxGk#4S!#>;L~WObFt#m?aJHPteKO$g*L_~_pX;Ie}66Ygg*Q2{6*MT9OBf6yxwU{_5Hmv zT+-#^IKJZd=P=Bg``6;!&FTGGt##G?Rwb$5-;KllXMfjsb^reT_8lhq{HN9I-+-A^ zOp^n5;r8|KCs>lBCR_7^9oK?5{;{00J1I>3F!wV*pf>WBsYMj?9w#uD(Ddkfq36B{ zC!pq+-ib)77-2hZ@WHFR`CmI?=kK{^!HDsyhu#ac*VvJZGx^u8L(h2H0SNlp zO~CNti>L0`5mST(SIxqiXTJEcX4>0iwt8}u8Mq-Ag!zBN7Nr=C@#1e^?SLKw2xOOW z6UdWxpzI>z$~k!9>_Mv-qKcC0-FQiSM7vnzOR@@nc!|!lJ5=mTO0Q_}H5GSe*wXc+ zb^qd}<|S=%PtMATQ{YB#CT_#^_2p@E@KZ#!_i${NR7@lZlf$)l@e~!G8%h#n#?S6C zv<-izU&G7LPue3oT$Fz+iJz{iy-%{Vq#)c3h?!3M2SUj>b`xZ!6YVpl>nm8V6FqhqA4)13 zC?)z4q?Hlvaop+4R^$*=EX^JXl`X5*tS4Ye2OX(}ACX74LkrsOhj{6AR5E)vHEbaM zjeCt+)g)gleRUY%+S*W#=%lo+E9<1no#esI2nf|F2l?*>Q13l4nl}NOiM7{~0Z2%x zSO#S!h_}qCN?gaBeWFO53AD|&%d&V1 zpyrxbz^E>_D1cR6<|eNkaQm92??o3*D67ji+(}(l3?WrXCO%X9Bnu7pb(P?^Q)z2y zLUQR!QSS?aa@W9QJT!g-pIn@hJeWXT#)t^7<3x6F=%~^z8?MK6AIWn>fu^9wnSR_! z2(xTymtfg2bT$aUb+gxtwSK12sSH5-BKt&(Jk)|8vy9M$X}}DRyIEdi3tt)9YQy^@ z=NV%EvDnPxyYKf3q`K7ls6?m;roJ17OJ}NxM=|YfZ9^fm6uu7;9n~4UsZ>Gr$Q@Pi z1wQqo_?uhy>(B0UtZ=FDH`!^FeFos{;wFdRKK2_(gN4sdZZ)p}Hm1jVGkwK?!susY ztgMufdaGtZtKTQYDBa~jHkGDN%C_$~^YK@5{Hgs;9fSZ3tHM%BCL<2J!u`OdgEKfm zxzxGW%CB{;R+-M5@pHdlzcv84tteRc5&U1okOja1a0e*=&+!WvMQmN4l&Q^sE|U@#nu^Q|2Y|DRiD90P_{9rUQUWZmYHYGCs?E%8?(E8F3j%lbbpirVA7K5X zP0*Mjg^u?S5Sr1O#iiwy)wT7N2#onRO+bS_4DcQx;i_kriRohl&;Zj{jz13uKH$LF0LVhcM2MEw+x%b@ zB6X+oO=cP$VBl%1kXKsh9x?RYPC&O$S#S|BnN^vD*VI9wEwdnj8Sh(mh8KgTa{7i*Q?C}x^A7BvF~9WK69CtV5xr4!bZSOseA~l}SaQaJ>Gj~zkm#bt4suoNNu=z)Z`$&O?MM|`0dxFMY{eR8 z`z|#qLrpF4=jmhwJoK>P9pBq-Cgc|cNpj2Q-lfwP;fq9$$Yv^vB&y~JFf!j+e&w34 zcU#WusuI{Zmd*ohl$=~1KWRB$dRtpLJR6gWGwSG< zu}`Q{kmdw?8krPVr?Vk%1+|Bkf-$VZ&`j+SLp`5t-ChQUCB*Bchl|K9rb;`^CZa{z zYA7rOi0_g08UT(^1-ATJfSquet+2RwI7{1(p6VqdX%Y#`9eK2f_8~CdZtR6w62F>f zV5;34a*Gs$3|q7e&hlZpbf$KFw5;qQQI?9lAw8sn7i+PW@1EM9ZJcgXryoo<^6 z$@Ox|L4Rm>$~8FMbOZr7CAj;DkDZ(c2pm=FOi7-1+BCPQW!Vpk`b6>9SJw}oV{j{U zmW=e&uNDm_caOavD;1>ToXoxV;hCyMKjoS(qww8quHkUto@o#}1$EzR2{d2`{_<)V z;!Hf@o10qx%{I%g{Df!ez+eqDkLH*yvtR~|VrsLy85Qb!-vAI^7kf}tM_MUIK5g

A{+H)$o}su{Yc?g0`$RgX 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ði_h7+sjZvDE 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

whk9df+7_D!q@$EG+U0#1K<`k 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%0DLe_Zfpfv^}K)mCRo?k#%&8wWj>+ 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 literal 0 HcmV?d00001 From 1b2f33d0c68ab3eeefebe41ca1235a887feadfe7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 9 Jul 2014 17:38:12 +0200 Subject: [PATCH 13/34] added new view for empty folders in YACReaderLibrary that shows a list with all the subfolders TODO ?use a special case for the root folder? --- YACReaderLibrary/YACReaderLibrary.pro | 6 +- YACReaderLibrary/db/tablemodel.cpp | 4 +- YACReaderLibrary/db/tablemodel.h | 1 + YACReaderLibrary/db/treemodel.cpp | 70 ++++++--- YACReaderLibrary/db/treemodel.h | 2 + YACReaderLibrary/db_helper.cpp | 15 +- YACReaderLibrary/db_helper.h | 1 + YACReaderLibrary/empty_folder_widget.cpp | 89 +++++++++++ YACReaderLibrary/empty_folder_widget.h | 32 ++++ YACReaderLibrary/images_osx.qrc | 1 + YACReaderLibrary/images_win.qrc | 1 + YACReaderLibrary/library_window.cpp | 40 ++++- YACReaderLibrary/library_window.h | 182 ++++++++++++----------- images/empty_folder.png | Bin 0 -> 2515 bytes 14 files changed, 322 insertions(+), 122 deletions(-) create mode 100644 YACReaderLibrary/empty_folder_widget.cpp create mode 100644 YACReaderLibrary/empty_folder_widget.h create mode 100644 images/empty_folder.png diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index 1d19565d..3523e1ec 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -117,7 +117,8 @@ HEADERS += comic_flow.h \ yacreader_libraries.h \ ../common/exit_check.h \ comics_view.h \ - classic_comics_view.h + classic_comics_view.h \ + empty_folder_widget.h SOURCES += comic_flow.cpp \ @@ -163,7 +164,8 @@ SOURCES += comic_flow.cpp \ yacreader_libraries.cpp \ ../common/exit_check.cpp \ comics_view.cpp \ - classic_comics_view.cpp + classic_comics_view.cpp \ + empty_folder_widget.cpp diff --git a/YACReaderLibrary/db/tablemodel.cpp b/YACReaderLibrary/db/tablemodel.cpp index 1d9e51c2..8987176f 100644 --- a/YACReaderLibrary/db/tablemodel.cpp +++ b/YACReaderLibrary/db/tablemodel.cpp @@ -304,7 +304,9 @@ void TableModel::setupModelData(unsigned long long int folderId,const QString & db.close(); QSqlDatabase::removeDatabase(_databasePath); endResetModel(); - //f.close(); + + if(_data.length()==0) + emit isEmpty(); } QString TableModel::getComicPath(QModelIndex mi) diff --git a/YACReaderLibrary/db/tablemodel.h b/YACReaderLibrary/db/tablemodel.h index 7bfff95b..9df0126a 100644 --- a/YACReaderLibrary/db/tablemodel.h +++ b/YACReaderLibrary/db/tablemodel.h @@ -113,6 +113,7 @@ private: signals: void beforeReset(); void reset(); + void isEmpty(); }; //! [0] diff --git a/YACReaderLibrary/db/treemodel.cpp b/YACReaderLibrary/db/treemodel.cpp index fcf0ada8..f04b7bef 100644 --- a/YACReaderLibrary/db/treemodel.cpp +++ b/YACReaderLibrary/db/treemodel.cpp @@ -53,6 +53,7 @@ #include "data_base_management.h" #include "folder.h" #include "db_helper.h" +#include "qnaturalsorting.h" #ifdef Q_OS_MAC #include @@ -106,7 +107,7 @@ TreeModel::TreeModel(QObject *parent) TreeModel::TreeModel( QSqlQuery &sqlquery, QObject *parent) : QAbstractItemModel(parent),rootItem(0),rootBeforeFilter(0),filterEnabled(false),includeComics(false) { - //lo más probable es que el nodo raíz no necesite tener información + //lo m�s probable es que el nodo ra�z no necesite tener informaci�n QList rootData; rootData << "root"; //id 0, padre 0, title "root" (el id, y el id del padre van a ir en la clase TreeItem) rootItem = new TreeItem(rootData); @@ -265,7 +266,7 @@ void TreeModel::setupModelData(QString path) filterEnabled = false; rootItem = 0; rootBeforeFilter = 0; - //inicializar el nodo raíz + //inicializar el nodo ra�z QList rootData; rootData << "root"; //id 0, padre 0, title "root" (el id, y el id del padre van a ir en la clase TreeItem) rootItem = new TreeItem(rootData); @@ -291,10 +292,10 @@ void TreeModel::setupModelData(QString path) void TreeModel::setupModelData(QSqlQuery &sqlquery, TreeItem *parent) { - //64 bits para la primary key, es decir la misma precisión que soporta sqlit 2^64 - //el diccionario permitirá encontrar cualquier nodo del árbol rápidamente, de forma que añadir un hijo a un padre sea O(1) + //64 bits para la primary key, es decir la misma precisi�n que soporta sqlit 2^64 + //el diccionario permitir� encontrar cualquier nodo del �rbol r�pidamente, de forma que a�adir un hijo a un padre sea O(1) items.clear(); - //se añade el nodo 0 + //se a�ade el nodo 0 items.insert(parent->id,parent); while (sqlquery.next()) { @@ -308,11 +309,11 @@ void TreeModel::setupModelData(QSqlQuery &sqlquery, TreeItem *parent) TreeItem * item = new TreeItem(data); item->id = record.value("id").toULongLong(); - //la inserción de hijos se hace de forma ordenada + //la inserci�n de hijos se hace de forma ordenada TreeItem * parent = items.value(record.value("parentId").toULongLong()); if(parent !=0) //TODO if parent==0 the parent of item was removed from the DB and delete on cascade didn't work, ERROR. parent->appendChild(item); - //se añade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones + //se a�ade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones items.insert(item->id,item); } } @@ -323,12 +324,12 @@ void TreeModel::setupFilteredModelData() //TODO hay que liberar memoria de anteriores filtrados - //inicializar el nodo raíz + //inicializar el nodo ra�z if(rootBeforeFilter == 0) rootBeforeFilter = rootItem; else - delete rootItem;//los resultados de la búsqueda anterior deben ser borrados + delete rootItem;//los resultados de la b�squeda anterior deben ser borrados QList rootData; rootData << "root"; //id 1, padre 1, title "root" (el id, y el id del padre van a ir en la clase TreeItem) @@ -365,10 +366,10 @@ void TreeModel::setupFilteredModelData() void TreeModel::setupFilteredModelData(QSqlQuery &sqlquery, TreeItem *parent) { - //64 bits para la primary key, es decir la misma precisión que soporta sqlit 2^64 + //64 bits para la primary key, es decir la misma precisi�n que soporta sqlit 2^64 filteredItems.clear(); - //se añade el nodo 0 al modelo que representa el arbol de elementos que cumplen con el filtro + //se a�ade el nodo 0 al modelo que representa el arbol de elementos que cumplen con el filtro filteredItems.insert(parent->id,parent); while (sqlquery.next()) { //se procesan todos los folders que cumplen con el filtro @@ -387,39 +388,39 @@ void TreeModel::setupFilteredModelData(QSqlQuery &sqlquery, TreeItem *parent) //id del padre quint64 parentId = record.value("parentId").toULongLong(); - //se añade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones + //se a�ade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones if(!filteredItems.contains(item->id)) filteredItems.insert(item->id,item); - //es necesario conocer las coordenadas de origen para poder realizar scroll automático en la vista + //es necesario conocer las coordenadas de origen para poder realizar scroll autom�tico en la vista item->originalItem = items.value(item->id); - //si el padre ya existe en el modelo, el item se añade como hijo + //si el padre ya existe en el modelo, el item se a�ade como hijo if(filteredItems.contains(parentId)) filteredItems.value(parentId)->appendChild(item); - else//si el padre aún no se ha añadido, hay que añadirlo a él y todos los padres hasta el nodo raíz + else//si el padre a�n no se ha a�adido, hay que a�adirlo a �l y todos los padres hasta el nodo ra�z { - //comprobamos con esta variable si el último de los padres (antes del nodo raíz) ya existía en el modelo + //comprobamos con esta variable si el �ltimo de los padres (antes del nodo ra�z) ya exist�a en el modelo bool parentPreviousInserted = false; - //mientras no se alcance el nodo raíz se procesan todos los padres (de abajo a arriba) + //mientras no se alcance el nodo ra�z se procesan todos los padres (de abajo a arriba) while(parentId != ROOT ) { - //el padre no estaba en el modelo filtrado, así que se rescata del modelo original + //el padre no estaba en el modelo filtrado, as� que se rescata del modelo original TreeItem * parentItem = items.value(parentId); //se debe crear un nuevo nodo (para no compartir los hijos con el nodo original) - TreeItem * newparentItem = new TreeItem(parentItem->getData()); //padre que se añadirá a la estructura de directorios filtrados + TreeItem * newparentItem = new TreeItem(parentItem->getData()); //padre que se a�adir� a la estructura de directorios filtrados newparentItem->id = parentId; newparentItem->originalItem = parentItem; - //si el modelo contiene al padre, se añade el item actual como hijo + //si el modelo contiene al padre, se a�ade el item actual como hijo if(filteredItems.contains(parentId)) { filteredItems.value(parentId)->appendChild(item); parentPreviousInserted = true; } - //sino se registra el nodo para poder encontrarlo con posterioridad y se añade el item actual como hijo + //sino se registra el nodo para poder encontrarlo con posterioridad y se a�ade el item actual como hijo else { newparentItem->appendChild(item); @@ -432,7 +433,7 @@ void TreeModel::setupFilteredModelData(QSqlQuery &sqlquery, TreeItem *parent) parentId = parentItem->parentItem->id; } - //si el nodo es hijo de 1 y no había sido previamente insertado como hijo, se añade como tal + //si el nodo es hijo de 1 y no hab�a sido previamente insertado como hijo, se a�ade como tal if(!parentPreviousInserted) filteredItems.value(ROOT)->appendChild(item); } @@ -468,7 +469,7 @@ void TreeModel::resetFilter() //items.clear(); filteredItems.clear(); TreeItem * root = rootItem; - rootItem = rootBeforeFilter; //TODO si no se aplica el filtro previamente, esto invalidaría en modelo + rootItem = rootBeforeFilter; //TODO si no se aplica el filtro previamente, esto invalidar�a en modelo if(root !=0) delete root; @@ -518,3 +519,26 @@ void TreeModel::updateFolderFinishedStatus(const QModelIndexList &list, bool sta emit dataChanged(index(list.first().row(),TreeModel::Name),index(list.last().row(),TreeModel::Completed)); } + +QStringList TreeModel::getSubfoldersNames(const QModelIndex &mi) +{ + QStringList result; + qulonglong id = 1; + if(mi.isValid()){ + TreeItem * item = static_cast(mi.internalPointer()); + id = item->id; + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + + result = DBHelper::loadSubfoldersNames(id,db); + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + //TODO sort result)) + qSort(result.begin(),result.end(),naturalSortLessThanCI); + return result; +} diff --git a/YACReaderLibrary/db/treemodel.h b/YACReaderLibrary/db/treemodel.h index 02168cbb..d36dab40 100644 --- a/YACReaderLibrary/db/treemodel.h +++ b/YACReaderLibrary/db/treemodel.h @@ -85,6 +85,8 @@ public: void updateFolderCompletedStatus(const QModelIndexList & list, bool status); void updateFolderFinishedStatus(const QModelIndexList & list, bool status); + QStringList getSubfoldersNames(const QModelIndex & mi); + enum Columns { Name = 0, Path = 1, diff --git a/YACReaderLibrary/db_helper.cpp b/YACReaderLibrary/db_helper.cpp index 417d8d4d..39b18751 100644 --- a/YACReaderLibrary/db_helper.cpp +++ b/YACReaderLibrary/db_helper.cpp @@ -671,5 +671,18 @@ ComicInfo DBHelper::loadComicInfo(QString hash, QSqlDatabase & db) else comicInfo.existOnDb = false; - return comicInfo; + return comicInfo; +} + +QList DBHelper::loadSubfoldersNames(qulonglong folderId, QSqlDatabase &db) +{ + QList result; + QSqlQuery selectQuery(db); + selectQuery.prepare("SELECT name FROM folder WHERE parentId = :parentId AND id <> 1"); //do not select the root folder + selectQuery.bindValue(":parentId", folderId); + selectQuery.exec(); + while(selectQuery.next()){ + result << selectQuery.record().value("name").toString(); + } + return result; } diff --git a/YACReaderLibrary/db_helper.h b/YACReaderLibrary/db_helper.h index b4f88f89..014e65c9 100644 --- a/YACReaderLibrary/db_helper.h +++ b/YACReaderLibrary/db_helper.h @@ -51,6 +51,7 @@ public: static ComicDB loadComic(qulonglong id, QSqlDatabase & db); static ComicDB loadComic(QString cname, QString cpath, QString chash, QSqlDatabase & database); static ComicInfo loadComicInfo(QString hash, QSqlDatabase & db); + static QList loadSubfoldersNames(qulonglong folderId, QSqlDatabase & db); }; #endif diff --git a/YACReaderLibrary/empty_folder_widget.cpp b/YACReaderLibrary/empty_folder_widget.cpp new file mode 100644 index 00000000..6d440f51 --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.cpp @@ -0,0 +1,89 @@ +#include "empty_folder_widget.h" + +#include +#include +#include +#include + + +#include +void testListView(QListView * l) +{ + QStringListModel * slm = new QStringListModel(QStringList() = {"Lorem ipsum", "Hailer skualer", "Mumbaluba X", "Finger layden", "Pacum tactus filer", "Aposum", "En","Lorem ipsum", "Hailer skualer", "Mumbaluba X", "Finger layden", "Pacum tactus filer", "Aposum", "En" }); + l->setModel(slm); +} + +EmptyFolderWidget::EmptyFolderWidget(QWidget *parent) : + QWidget(parent),subfoldersModel(new QStringListModel()) +{ + QVBoxLayout * layout = new QVBoxLayout; + + iconLabel = new QLabel(); + iconLabel->setPixmap(QPixmap(":/images/empty_folder.png")); + iconLabel->setAlignment(Qt::AlignCenter); + + titleLabel = new QLabel("Subfolders in this folder"); + titleLabel->setAlignment(Qt::AlignCenter); + titleLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); + + foldersView = new QListView(); + foldersView->setMinimumWidth(282); + foldersView->setWrapping(true); + + foldersView->setStyleSheet("QListView {background-color:transparent; border: none; color:#858585; outline:0; font-size: 18px; font:bold; show-decoration-selected: 0; margin:0}" + "QListView::item:selected {background-color: #212121; color:#CCCCCC;}" + "QListView::item:hover {background-color:#212121; color:#CCCCCC; }" + + + "QScrollBar:vertical { border: none; background: #212121; width: 14px; margin: 0 10px 0 0; }" + "QScrollBar::handle:vertical { background: #858585; width: 14px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #212121; height: 0px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #212121; height: 0px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical {background: none; }" + "QScrollBar:horizontal{height:0px;}" + ); + + foldersView->setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + testListView(foldersView); + + layout->addSpacing(100); + layout->addWidget(iconLabel); + layout->addSpacing(30); + layout->addWidget(titleLabel); + layout->addSpacing(12); + layout->addWidget(foldersView,1,Qt::AlignHCenter); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet("QWidget {background:#2A2A2A}"); + + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + setLayout(layout); + + connect(foldersView,SIGNAL(clicked(QModelIndex)),this,SLOT(onItemClicked(QModelIndex))); +} + +void EmptyFolderWidget::setSubfolders(const QModelIndex &mi, const QStringList &foldersNames) +{ + parent = mi; + subfoldersModel->setStringList(foldersNames); + foldersView->setModel(subfoldersModel); +} + +void EmptyFolderWidget::onItemClicked(const QModelIndex &mi) +{ + emit subfolderSelected(parent,mi.row()); +} + +void EmptyFolderWidget::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor("#2A2A2A")); +} diff --git a/YACReaderLibrary/empty_folder_widget.h b/YACReaderLibrary/empty_folder_widget.h new file mode 100644 index 00000000..225948be --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.h @@ -0,0 +1,32 @@ +#ifndef EMPTY_FOLDER_WIDGET_H +#define EMPTY_FOLDER_WIDGET_H + +#include +#include + +class QLabel; +class QListView; +class QStringListModel; + +class EmptyFolderWidget : public QWidget +{ + Q_OBJECT +public: + explicit EmptyFolderWidget(QWidget *parent = 0); + void setSubfolders(const QModelIndex & mi, const QStringList & foldersNames); +signals: + void subfolderSelected(QModelIndex, int); + +public slots: + void onItemClicked(const QModelIndex & mi); + +protected: + QLabel * iconLabel; + QLabel * titleLabel; + QListView * foldersView; + QModelIndex parent; + QStringListModel * subfoldersModel; + void paintEvent(QPaintEvent *); +}; + +#endif // EMPTY_FOLDER_WIDGET_H diff --git a/YACReaderLibrary/images_osx.qrc b/YACReaderLibrary/images_osx.qrc index d5604ff7..64c41b0c 100644 --- a/YACReaderLibrary/images_osx.qrc +++ b/YACReaderLibrary/images_osx.qrc @@ -19,5 +19,6 @@ ../images/openLibraryIcon_osx.png ../images/flow_to_grid.gif ../images/grid_to_flow.gif + ../images/empty_folder.png diff --git a/YACReaderLibrary/images_win.qrc b/YACReaderLibrary/images_win.qrc index 5d2bd0af..ffa97d93 100644 --- a/YACReaderLibrary/images_win.qrc +++ b/YACReaderLibrary/images_win.qrc @@ -18,5 +18,6 @@ ../images/main_toolbar/grid.png ../images/flow_to_grid.gif ../images/grid_to_flow.gif + ../images/empty_folder.png diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 003bf740..6697f997 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -63,6 +63,7 @@ #include "classic_comics_view.h" #include "grid_comics_view.h" #include "comics_view_transition.h" +#include "empty_folder_widget.h" #include "QsLog.h" @@ -205,8 +206,11 @@ void LibraryWindow::doLayout() doComicsViewConnections(); comicsView->setToolBar(editInfoToolBar); - comicsViewStack->addWidget(comicsView); comicsViewStack->addWidget(comicsViewTransition = new ComicsViewTransition()); + comicsViewStack->addWidget(emptyFolderWidget = new EmptyFolderWidget()); + comicsViewStack->addWidget(comicsView); + + comicsViewStack->setCurrentWidget(comicsView); fullScreenToolTip = new QLabel(comicsView); fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); @@ -887,6 +891,8 @@ void LibraryWindow::createConnections() connect(comicsViewTransition,SIGNAL(transitionFinished()),this,SLOT(showComicsView())); + connect(dmCV,SIGNAL(isEmpty()),this,SLOT(showEmptyFolderView())); + connect(emptyFolderWidget,SIGNAL(subfolderSelected(QModelIndex,int)),this,SLOT(selectSubfolder(QModelIndex,int))); } void LibraryWindow::loadLibrary(const QString & name) @@ -1075,8 +1081,20 @@ void LibraryWindow::loadCovers(const QModelIndex & mi) QStringList paths = dmCV->getPaths(currentPath()); checkEmptyFolder(&paths); - if(paths.size()>0) + if(paths.size()>0) { comicsView->setCurrentIndex(dmCV->index(0,0)); + if(comicsViewStack->currentWidget() == emptyFolderWidget) + comicsViewStack->setCurrentWidget(comicsView); + } + else + emptyFolderWidget->setSubfolders(mi,dm->getSubfoldersNames(mi)); +} + +void LibraryWindow::selectSubfolder(const QModelIndex &mi, int child) +{ + QModelIndex dest = dm->index(child,0,mi); + foldersView->setCurrentIndex(dest); + loadCovers(dest); } void LibraryWindow::checkEmptyFolder(QStringList * paths) @@ -1521,7 +1539,7 @@ void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) comicsView->setToolBar(editInfoToolBar); comicsViewStack->removeWidget(from); - comicsViewStack->insertWidget(0,comicsView); + comicsViewStack->addWidget(comicsView); delete from; @@ -1530,7 +1548,7 @@ void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) void LibraryWindow::showComicsViewTransition() { - comicsViewStack->setCurrentIndex(1); + comicsViewStack->setCurrentWidget(comicsViewTransition); comicsViewTransition->startMovie(); } @@ -1556,14 +1574,22 @@ void LibraryWindow::toggleComicsView_delayed() void LibraryWindow::showComicsView() { - comicsViewStack->setCurrentIndex(0); + comicsViewStack->setCurrentWidget(comicsView); +} + +void LibraryWindow::showEmptyFolderView() +{ + comicsViewStack->setCurrentWidget(emptyFolderWidget); } //TODO recover the current comics selection and restore it in the destination void LibraryWindow::toggleComicsView() { - QTimer::singleShot(0,this,SLOT(showComicsViewTransition())); - QTimer::singleShot(32,this,SLOT(toggleComicsView_delayed())); + if(comicsViewStack->currentWidget()!=emptyFolderWidget) { + QTimer::singleShot(0,this,SLOT(showComicsViewTransition())); + QTimer::singleShot(32,this,SLOT(toggleComicsView_delayed())); + } else + toggleComicsView_delayed(); } void LibraryWindow::asignNumbers() diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 5d098c12..2f9026f6 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -53,6 +53,8 @@ class ComicsView; class ClassicComicsView; class GridComicsView; class ComicsViewTransition; +class EmptyFolderWidget; + #include "comic_db.h" using namespace YACReader; @@ -99,6 +101,7 @@ private: GridComicsView * gridComicsView; QStackedWidget * comicsViewStack; ComicsViewTransition * comicsViewTransition; + EmptyFolderWidget * emptyFolderWidget; YACReaderTreeView * foldersView; YACReaderLibraryListWidget * selectedLibrary; @@ -207,104 +210,107 @@ private: void doComicsViewConnections(); - //ACTIONS MANAGEMENT - void disableComicsActions(bool disabled); - void disableLibrariesActions(bool disabled); - void disableNoUpdatedLibrariesActions(bool disabled); - void disableFoldersActions(bool disabled); + //ACTIONS MANAGEMENT + void disableComicsActions(bool disabled); + void disableLibrariesActions(bool disabled); + void disableNoUpdatedLibrariesActions(bool disabled); + void disableFoldersActions(bool disabled); - void disableAllActions(); - //void disableActions(); - //void enableActions(); - //void enableLibraryActions(); + void disableAllActions(); + //void disableActions(); + //void enableActions(); + //void enableLibraryActions(); - QString currentPath(); + QString currentPath(); - //settings - QSettings * settings; + //settings + QSettings * settings; - //navigation backward and forward - int currentFolderNavigation; - QList history; + //navigation backward and forward + int currentFolderNavigation; + QList history; - bool removeError; + bool removeError; ComicsViewStatus comicsViewStatus; protected: - virtual void closeEvent ( QCloseEvent * event ); + virtual void closeEvent ( QCloseEvent * event ); public: - LibraryWindow(); - public slots: - void loadLibrary(const QString & path); - void loadCovers(const QModelIndex & mi); - void checkEmptyFolder(QStringList * paths = 0); - void reloadCovers(); - void openComic(); - void createLibrary(); - void create(QString source,QString dest, QString name); - void showAddLibrary(); - void openLibrary(QString path, QString name); - void loadLibraries(); - void saveLibraries(); - void reloadCurrentLibrary(); - void openLastCreated(); - void updateLibrary(); - //void deleteLibrary(); - void openContainingFolder(); - void setFolderAsNotCompleted(); - void setFolderAsCompleted(); - void setFolderAsFinished(); - void setFolderAsNotFinished(); - void openContainingFolderComic(); - void deleteCurrentLibrary(); - void removeLibrary(); - void renameLibrary(); - void rename(QString newName); - void cancelCreating(); - void stopLibraryCreator(); - void setRootIndex(); - void toggleFullScreen(); - void toNormal(); - void toFullScreen(); - void setFoldersFilter(QString filter); - void showProperties(); - void exportLibrary(QString destPath); - void importLibrary(QString clc,QString destPath,QString name); - void reloadOptions(); - void setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus); - void setCurrentComicReaded(); - void setCurrentComicUnreaded(); - void hideComicFlow(bool hide); - void showExportComicsInfo(); - void showImportComicsInfo(); - void asignNumbers(); - void showNoLibrariesWidget(); - void showRootWidget(); - void showImportingWidget(); - void manageCreatingError(const QString & error); - void manageUpdatingError(const QString & error); - void manageOpeningLibraryError(const QString & error); - QModelIndexList getSelectedComics(); - void deleteComics(); - //void showSocial(); - void backward(); - void forward(); - void updateHistory(const QModelIndex & mi); - void updateFoldersViewConextMenu(const QModelIndex & mi); - void libraryAlreadyExists(const QString & name); - void importLibraryPackage(); - void updateComicsView(quint64 libraryId, const ComicDB & comic); - void setCurrentComicOpened(); - void showComicVineScraper(); - void setRemoveError(); - void checkRemoveError(); - void resetComicRating(); - void switchToComicsView(ComicsView *from, ComicsView *to); - void showComicsViewTransition(); - void toggleComicsView_delayed();//used in orther to avoid flickering; - void showComicsView(); - void toggleComicsView(); + LibraryWindow(); + +public slots: + void loadLibrary(const QString & path); + void loadCovers(const QModelIndex & mi); + void selectSubfolder(const QModelIndex & mi, int child); + void checkEmptyFolder(QStringList * paths = 0); + void reloadCovers(); + void openComic(); + void createLibrary(); + void create(QString source,QString dest, QString name); + void showAddLibrary(); + void openLibrary(QString path, QString name); + void loadLibraries(); + void saveLibraries(); + void reloadCurrentLibrary(); + void openLastCreated(); + void updateLibrary(); + //void deleteLibrary(); + void openContainingFolder(); + void setFolderAsNotCompleted(); + void setFolderAsCompleted(); + void setFolderAsFinished(); + void setFolderAsNotFinished(); + void openContainingFolderComic(); + void deleteCurrentLibrary(); + void removeLibrary(); + void renameLibrary(); + void rename(QString newName); + void cancelCreating(); + void stopLibraryCreator(); + void setRootIndex(); + void toggleFullScreen(); + void toNormal(); + void toFullScreen(); + void setFoldersFilter(QString filter); + void showProperties(); + void exportLibrary(QString destPath); + void importLibrary(QString clc,QString destPath,QString name); + void reloadOptions(); + void setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus); + void setCurrentComicReaded(); + void setCurrentComicUnreaded(); + void hideComicFlow(bool hide); + void showExportComicsInfo(); + void showImportComicsInfo(); + void asignNumbers(); + void showNoLibrariesWidget(); + void showRootWidget(); + void showImportingWidget(); + void manageCreatingError(const QString & error); + void manageUpdatingError(const QString & error); + void manageOpeningLibraryError(const QString & error); + QModelIndexList getSelectedComics(); + void deleteComics(); + //void showSocial(); + void backward(); + void forward(); + void updateHistory(const QModelIndex & mi); + void updateFoldersViewConextMenu(const QModelIndex & mi); + void libraryAlreadyExists(const QString & name); + void importLibraryPackage(); + void updateComicsView(quint64 libraryId, const ComicDB & comic); + void setCurrentComicOpened(); + void showComicVineScraper(); + void setRemoveError(); + void checkRemoveError(); + void resetComicRating(); + void switchToComicsView(ComicsView *from, ComicsView *to); + void showComicsViewTransition(); + void toggleComicsView_delayed();//used in orther to avoid flickering; + void showComicsView(); + void showEmptyFolderView(); + void toggleComicsView(); }; #endif diff --git a/images/empty_folder.png b/images/empty_folder.png new file mode 100644 index 0000000000000000000000000000000000000000..fa3b13cde0239492769d94f53bf767b651edaf75 GIT binary patch literal 2515 zcmb7GdpHwp8{dYQr1FxIOnFCAbdW>FBw7wjuM%N1iX37QIgFtb5#C;Eyg8&^Bp*VW z(+oL_Eklbk?~p^W18uXDZ{PR*^BvynyRQEE-Ov4duHWyz@9X~KdG4P(i`b&9r3?T7 zwm6=#cLM-`aM^iPX_E}uqU}^!*cIjA8Fd~T7!~6e82~tW1?wN6>lo&DHNY*v?@C-m ze*g>s07pC8pS%z|@$EC&Yxl=p?SB1db5v0sPN?SC8o(wo)N$oC{RB4 z^0R7_Gcl5ImaAcJcpf@`3<3VCa2qo$emKRCTf?v*;Ov`830DJ6X{?66AkztF-+Sw| zPd2zqeACO1bq2-XZ^XOIuda6nmV6a%>0Lnzj}DhiuHK9~-G%*K2XbDi+)R8cFl0*`pO}v^NWfk@fyP+uRFO9L7;l81 zr#|YJDPHsiOZ}9Wvx;x;R`-+^O{$pAIV!@qiw6m2nhFnmAiwIvnkgp5?c#c*Lk0MH zlaF*L@Tq%_@07u6D8;~hf9Uk6<`R!?1OLEyu9|8FtjZU2$1a6y3ETXY^LSiqJtp5q z<6I5wd#`bJab$s$PJUr2pO5e^eA9(Rua0VVWNgOZw>4k9$3~&S>bPBsHbF4_y4@e; z#}l_2CruS9^@L0}Q`Hw;~U9eFk2d;@*Jr&O1FrAefg@ zTC?YnKA+zP47X^u)1eU;rz`l^P+H!+FBMp{1^$S&b{cw~&<^E+Cs~NgIlhay#U2uh ze;2ElnYNPNhEo39PhsII&%Rk$E9Sb9B3)D;SHdF3yFq2tWmUc1B7vgE*%NQ0fvr8N3@eM1!Gwl*#sP3EWL7rMKW0%WvBzoa#$rBaW#u9FT4o3HG8foG%|LaylW`JX;n&~0z`WIH<9mDYxhj}~ zCf4K9(I%C)xDY^h_6V>*Nq37p`#toP;1>j-=3q*&w~THESquTAO1zi0IWwF&&kq3$ z<^brATB8^-Y4so1ayyBZX^F`Z2i|q=D0xLowE^$Ln`(4eY~x;L@o2g{1-F6WQ-i*o z=Zl8yL}7qW{#w`1XZGUwD3y+Gm*7VUZ=VrKlaz4PuJzxP`NrzB*;8$HX zwmA@Lyu-?-od&cF2=&V%_F-GXe)cF@Ff0_MN`Xk=|Y~)7^q(Odg0$G*6 zYV`ke|1-ub6lZtlX)7}$Dz7>>PdpV{nfxXO$~4!2V+NV#hmNc-#~1&LsT5fD1}ECA zp@Hsw*1J5!uj|to1Q3H&LR1NgaBY)`J8KuWz6e^fGG>p~d#tV}cun7*d9~I=RE;fJ z{gmV6z`v*n(^qI)r1Sa1I7Qfvr40Oo&9LturmC3yiVF=i$TW3Q>2JH*-rlP0b&Z)Z zHx$e_ngT(gXA=ktzRXaTFb2)mPyA>kzGVuMP?TPR$aTHTbEATir{6BvKadID78EBadRx}8`vl{W zJ6~YpM70A88kr2)Qz^f*cuV!F+n6n}Dm>zdVcgq#d|}cv^|yKQGuj=H_zk!RHA-pH qYV58Bg5}KOp>1f*XfYY20;sS`^*(JHm@WII0URC9+E?3QZv7o7SoZM% literal 0 HcmV?d00001 From e423e7e2df05bed63667edf036d0f83effe5baea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 9 Jul 2014 19:42:23 +0200 Subject: [PATCH 14/34] fixed navigation history fixed selectAll action (only in classic view) --- YACReaderLibrary/classic_comics_view.cpp | 5 +++++ YACReaderLibrary/classic_comics_view.h | 2 ++ YACReaderLibrary/comics_view.h | 3 ++- YACReaderLibrary/grid_comics_view.cpp | 5 +++++ YACReaderLibrary/grid_comics_view.h | 1 + YACReaderLibrary/library_window.cpp | 1 + 6 files changed, 16 insertions(+), 1 deletion(-) diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp index e64106e9..c3ebed37 100644 --- a/YACReaderLibrary/classic_comics_view.cpp +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -188,6 +188,11 @@ void ClassicComicsView::setViewActions(const QList &actions) comicFlow->addActions(actions); } +void ClassicComicsView::selectAll() +{ + tableView->selectAll(); +} + void ClassicComicsView::setShowMarks(bool show) { comicFlow->setShowMarks(show); diff --git a/YACReaderLibrary/classic_comics_view.h b/YACReaderLibrary/classic_comics_view.h index 9ecb9e3e..9f8c71a6 100644 --- a/YACReaderLibrary/classic_comics_view.h +++ b/YACReaderLibrary/classic_comics_view.h @@ -37,6 +37,8 @@ public slots: void removeItemsFromFlow(const QModelIndex & parent, int from, int to); //ComicsView void setShowMarks(bool show); + void selectAll(); + private: YACReaderTableView * tableView; QWidget *comics; diff --git a/YACReaderLibrary/comics_view.h b/YACReaderLibrary/comics_view.h index 4076e114..5ae8e485 100644 --- a/YACReaderLibrary/comics_view.h +++ b/YACReaderLibrary/comics_view.h @@ -32,12 +32,13 @@ public: virtual void setItemActions(const QList & actions) = 0; //actions for visual-oriented views virtual void setViewActions(const QList & actions) = 0; - //virtual selectItem(int index) = 0; + signals: void selected(unsigned int); void comicRated(int,QModelIndex); public slots: virtual void setShowMarks(bool show) = 0; + virtual void selectAll() = 0; protected: TableModel * model; diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp index 8267e909..a8effdd9 100644 --- a/YACReaderLibrary/grid_comics_view.cpp +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -168,6 +168,11 @@ void GridComicsView::setViewActions(const QList &actions) QLOG_ERROR() << "setViewActions invoked with the wrong number of actions"; } +void GridComicsView::selectAll() +{ + QLOG_INFO() << "selectAll"; +} + QSize GridComicsView::sizeHint() { QLOG_INFO() << "sizeHint"; diff --git a/YACReaderLibrary/grid_comics_view.h b/YACReaderLibrary/grid_comics_view.h index 1012679f..3066ff68 100644 --- a/YACReaderLibrary/grid_comics_view.h +++ b/YACReaderLibrary/grid_comics_view.h @@ -45,6 +45,7 @@ public slots: //ComicsView void setShowMarks(bool show); + void selectAll(); private: QItemSelectionModel * _selectionModel; diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 6697f997..cc825d6c 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -1094,6 +1094,7 @@ void LibraryWindow::selectSubfolder(const QModelIndex &mi, int child) { QModelIndex dest = dm->index(child,0,mi); foldersView->setCurrentIndex(dest); + updateHistory(dest); loadCovers(dest); } From 0bbdbd715c0a8ede991e841b4b3c3686e477d743 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 13 Jul 2014 21:23:47 +0200 Subject: [PATCH 15/34] removed fullScreenToolTip --- YACReaderLibrary/library_window.cpp | 14 -------------- YACReaderLibrary/library_window.h | 1 - 2 files changed, 15 deletions(-) diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index cc825d6c..fe8621dc 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -212,14 +212,6 @@ void LibraryWindow::doLayout() comicsViewStack->setCurrentWidget(comicsView); - fullScreenToolTip = new QLabel(comicsView); - fullScreenToolTip->setText(tr(" press 'F' to close fullscreen mode ")); - fullScreenToolTip->setPalette(QPalette(QColor(0,0,0))); - fullScreenToolTip->setFont(QFont("courier new",15,234)); - fullScreenToolTip->setAutoFillBackground(true); - fullScreenToolTip->hide(); - fullScreenToolTip->adjustSize(); - sHorizontal->addWidget(sideBar); #ifndef Q_OS_MAC QVBoxLayout * rightLayout = new QVBoxLayout; @@ -1416,16 +1408,10 @@ void LibraryWindow::toFullScreen() comicsView->toFullScreen(); showFullScreen(); - - fullScreenToolTip->move((width()-fullScreenToolTip->width())/2,0); - fullScreenToolTip->adjustSize(); - fullScreenToolTip->show(); } void LibraryWindow::toNormal() { - fullScreenToolTip->hide(); - sideBar->show(); comicsView->toNormal(); diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 2f9026f6..8092a774 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -109,7 +109,6 @@ private: TableModel * dmCV; //QStringList paths; YACReaderLibraries libraries; - QLabel * fullScreenToolTip; QStackedWidget * mainWidget; NoLibrariesWidget * noLibrariesWidget; From 3c18599496d5e36dd30f40f7882b628a5cf1580b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sun, 13 Jul 2014 21:43:50 +0200 Subject: [PATCH 16/34] updated .desktop files from issue #16 --- YACReader.desktop | 2 +- YACReaderLibrary.desktop | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/YACReader.desktop b/YACReader.desktop index ddd79076..20c5f201 100644 --- a/YACReader.desktop +++ b/YACReader.desktop @@ -8,5 +8,5 @@ Terminal=false Type=Application StartupNotify=true Categories=Graphics;Viewer; -MimeType=application/x-cbz;application/x-cbr;application/x-cbt;application/x-cb7; +MimeType=application/x-cbz;application/x-cbr;application/x-cbt;application/x-cb7;application/x-pdf;application/x-zip;application/x-rar;application/x-7z;inode/directory; X-Desktop-File-Install-Version=0.22 diff --git a/YACReaderLibrary.desktop b/YACReaderLibrary.desktop index 38551ad7..76ee3432 100644 --- a/YACReaderLibrary.desktop +++ b/YACReaderLibrary.desktop @@ -8,5 +8,5 @@ Terminal=false Type=Application StartupNotify=true Categories=Graphics;Viewer; -MimeType=application/x-cbz;application/x-cbr;application/x-cbt;application/x-cb7; +MimeType= X-Desktop-File-Install-Version=0.22 From 50c8eca7c4440af43782dde012241ab84dcf4158 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Thu, 17 Jul 2014 22:30:03 +0200 Subject: [PATCH 17/34] added shortcuts management support TODO: integrate it in YACReader TODO: add icons for groups TODO: add conflicts detection TODO: fix any shortcut used in keyPressEvent TODO: choose new default shortcuts (F5 update, F11 fullscreen, etc...) --- YACReaderLibrary/YACReaderLibrary.pro | 1 + YACReaderLibrary/images.qrc | 2 + YACReaderLibrary/library_window.cpp | 226 ++++++++++++++---- YACReaderLibrary/library_window.h | 17 +- YACReaderLibrary/options_dialog.cpp | 64 +++-- YACReaderLibrary/options_dialog.h | 2 + images/accept_shortcut.png | Bin 0 -> 239 bytes images/clear_shortcut.png | Bin 0 -> 201 bytes shortcuts_management/actions_groups_model.cpp | 80 +++++++ shortcuts_management/actions_groups_model.h | 44 ++++ .../actions_shortcuts_model.cpp | 92 +++++++ .../actions_shortcuts_model.h | 38 +++ .../edit_shortcut_item_delegate.cpp | 145 +++++++++++ .../edit_shortcut_item_delegate.h | 48 ++++ shortcuts_management/shortcuts_dialog.cpp | 82 +++++++ shortcuts_management/shortcuts_dialog.h | 32 +++ shortcuts_management/shortcuts_management.pri | 16 ++ shortcuts_management/shortcuts_manager.cpp | 49 ++++ shortcuts_management/shortcuts_manager.h | 73 ++++++ 19 files changed, 933 insertions(+), 78 deletions(-) create mode 100644 images/accept_shortcut.png create mode 100644 images/clear_shortcut.png create mode 100644 shortcuts_management/actions_groups_model.cpp create mode 100644 shortcuts_management/actions_groups_model.h create mode 100644 shortcuts_management/actions_shortcuts_model.cpp create mode 100644 shortcuts_management/actions_shortcuts_model.h create mode 100644 shortcuts_management/edit_shortcut_item_delegate.cpp create mode 100644 shortcuts_management/edit_shortcut_item_delegate.h create mode 100644 shortcuts_management/shortcuts_dialog.cpp create mode 100644 shortcuts_management/shortcuts_dialog.h create mode 100644 shortcuts_management/shortcuts_management.pri create mode 100644 shortcuts_management/shortcuts_manager.cpp create mode 100644 shortcuts_management/shortcuts_manager.h diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index 3523e1ec..88275787 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -174,6 +174,7 @@ include(../custom_widgets/custom_widgets_yacreaderlibrary.pri) include(../compressed_archive/wrapper.pri) include(./comic_vine/comic_vine.pri) include(../QsLog/QsLog.pri) +include(../shortcuts_management/shortcuts_management.pri) RESOURCES += images.qrc files.qrc win32:RESOURCES += images_win.qrc diff --git a/YACReaderLibrary/images.qrc b/YACReaderLibrary/images.qrc index fb527aa9..d6551235 100644 --- a/YACReaderLibrary/images.qrc +++ b/YACReaderLibrary/images.qrc @@ -100,6 +100,8 @@ ../images/comic_vine/downArrow.png ../images/comic_vine/upArrow.png ../images/find_folder.png + ../images/clear_shortcut.png + ../images/accept_shortcut.png diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index fe8621dc..d61d796a 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -65,6 +65,9 @@ #include "comics_view_transition.h" #include "empty_folder_widget.h" +#include "shortcuts_dialog.h" +#include "shortcuts_manager.h" + #include "QsLog.h" #ifdef Q_OS_WIN @@ -270,6 +273,9 @@ void LibraryWindow::doDialogs() optionsDialog = new OptionsDialog(this); optionsDialog->restoreOptions(settings); + shortcutsDialog = new ShortcutsDialog(this); + setUpShortcutsManagement(); + #ifdef SERVER_RELEASE serverConfigDialog = new ServerConfigDialog(this); #endif @@ -289,6 +295,61 @@ void LibraryWindow::doDialogs() } +void LibraryWindow::setUpShortcutsManagement() +{ + shortcutsDialog->addActionsGroup("Comics",QIcon(":/images/openInYACReader.png"), + QList() + << openComicAction + << setAsReadAction + << setAsNonReadAction + << openContainingFolderComicAction + << resetComicRatingAction + << selectAllComicsAction + << editSelectedComicsAction + << asignOrderAction + << deleteComicsAction + << getInfoAction); + + shortcutsDialog->addActionsGroup("Folders",QIcon(), + QList() + << setRootIndexAction + << expandAllNodesAction + << colapseAllNodesAction + << openContainingFolderAction + << setFolderAsNotCompletedAction + << setFolderAsCompletedAction + << setFolderAsReadAction + << setFolderAsUnreadAction); + + shortcutsDialog->addActionsGroup("General",QIcon(), + QList() + << backAction + << forwardAction + << helpAboutAction + << optionsAction + << serverConfigAction); + + shortcutsDialog->addActionsGroup("Libraries",QIcon(), + QList() + << createLibraryAction + << openLibraryAction + << exportComicsInfoAction + << importComicsInfoAction + << exportLibraryAction + << importLibraryAction + << updateLibraryAction + << renameLibraryAction + << removeLibraryAction); + + shortcutsDialog->addActionsGroup("Visualization",QIcon(), + QList() + << showHideMarksAction + << toggleFullScreenAction + << toggleComicsViewAction + << hideComicViewAction); + +} + void LibraryWindow::doModels() { //folders @@ -323,67 +384,90 @@ void LibraryWindow::createActions() QIcon icoBackButton; icoBackButton.addPixmap(QPixmap(":/images/main_toolbar/back.png"), QIcon::Normal); //icoBackButton.addPixmap(QPixmap(":/images/main_toolbar/back_disabled.png"), QIcon::Disabled); - backAction->setIcon(icoBackButton); + backAction->setData(BACK_ACTION_YL); + backAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(BACK_ACTION_YL)); + backAction->setIcon(icoBackButton); backAction->setDisabled(true); forwardAction = new QAction(this); QIcon icoFordwardButton; icoFordwardButton.addPixmap(QPixmap(":/images/main_toolbar/forward.png"), QIcon::Normal); //icoFordwardButton.addPixmap(QPixmap(":/images/main_toolbar/forward_disabled.png"), QIcon::Disabled); + forwardAction->setData(FORWARD_ACTION_YL); + forwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(FORWARD_ACTION_YL)); forwardAction->setIcon(icoFordwardButton); forwardAction->setDisabled(true); createLibraryAction = new QAction(this); createLibraryAction->setToolTip(tr("Create a new library")); - createLibraryAction->setShortcut(Qt::Key_A); + createLibraryAction->setData(CREATE_LIBRARY_ACTION_YL); + createLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CREATE_LIBRARY_ACTION_YL)); createLibraryAction->setIcon(QIcon(":/images/newLibraryIcon.png")); openLibraryAction = new QAction(this); openLibraryAction->setToolTip(tr("Open an existing library")); - openLibraryAction->setShortcut(Qt::Key_O); + openLibraryAction->setData(OPEN_LIBRARY_ACTION_YL); + openLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_LIBRARY_ACTION_YL)); openLibraryAction->setIcon(QIcon(":/images/openLibraryIcon.png")); - exportComicsInfo = new QAction(tr("Export comics info"),this); - exportComicsInfo->setToolTip(tr("Export comics info")); - exportComicsInfo->setIcon(QIcon(":/images/exportComicsInfoIcon.png")); + exportComicsInfoAction = new QAction(tr("Export comics info"),this); + exportComicsInfoAction->setToolTip(tr("Export comics info")); + exportComicsInfoAction->setData(EXPORT_COMICS_INFO_YL); + exportComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_COMICS_INFO_YL)); + exportComicsInfoAction->setIcon(QIcon(":/images/exportComicsInfoIcon.png")); - importComicsInfo = new QAction(tr("Import comics info"),this); - importComicsInfo->setToolTip(tr("Import comics info")); - importComicsInfo->setIcon(QIcon(":/images/importComicsInfoIcon.png")); + importComicsInfoAction = new QAction(tr("Import comics info"),this); + importComicsInfoAction->setToolTip(tr("Import comics info")); + importComicsInfoAction->setData(IMPORT_COMICS_INFO_YL); + importComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_COMICS_INFO_YL)); + importComicsInfoAction->setIcon(QIcon(":/images/importComicsInfoIcon.png")); exportLibraryAction = new QAction(tr("Pack covers"),this); exportLibraryAction->setToolTip(tr("Pack the covers of the selected library")); + exportLibraryAction->setData(EXPORT_LIBRARY_ACTION_YL); + exportLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_LIBRARY_ACTION_YL)); exportLibraryAction->setIcon(QIcon(":/images/exportLibraryIcon.png")); importLibraryAction = new QAction(tr("Unpack covers"),this); importLibraryAction->setToolTip(tr("Unpack a catalog")); + importLibraryAction->setData(IMPORT_LIBRARY_ACTION_YL); + importLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_LIBRARY_ACTION_YL)); importLibraryAction->setIcon(QIcon(":/images/importLibraryIcon.png")); updateLibraryAction = new QAction(tr("Update library"),this); updateLibraryAction->setToolTip(tr("Update current library")); - updateLibraryAction->setShortcut(Qt::Key_U); + updateLibraryAction->setData(UPDATE_LIBRARY_ACTION_YL); + updateLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(UPDATE_LIBRARY_ACTION_YL)); updateLibraryAction->setIcon(QIcon(":/images/updateLibraryIcon.png")); renameLibraryAction = new QAction(tr("Rename library"),this); renameLibraryAction->setToolTip(tr("Rename current library")); - renameLibraryAction->setShortcut(Qt::Key_R); + renameLibraryAction->setData(RENAME_LIBRARY_ACTION_YL); + renameLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RENAME_LIBRARY_ACTION_YL)); renameLibraryAction->setIcon(QIcon(":/images/editIcon.png")); removeLibraryAction = new QAction(tr("Remove library"),this); removeLibraryAction->setToolTip(tr("Remove current library from your collection")); + removeLibraryAction->setData(REMOVE_LIBRARY_ACTION_YL); + removeLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(REMOVE_LIBRARY_ACTION_YL)); removeLibraryAction->setIcon(QIcon(":/images/removeLibraryIcon.png")); openComicAction = new QAction(tr("Open current comic"),this); openComicAction->setToolTip(tr("Open current comic on YACReader")); - openComicAction->setShortcut(Qt::Key_Return); + openComicAction->setData(OPEN_COMIC_ACTION_YL); + openComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_COMIC_ACTION_YL)); openComicAction->setIcon(QIcon(":/images/openInYACReader.png")); setAsReadAction = new QAction(tr("Set as read"),this); setAsReadAction->setToolTip(tr("Set comic as read")); + setAsReadAction->setData(SET_AS_READ_ACTION_YL); + setAsReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_AS_READ_ACTION_YL)); setAsReadAction->setIcon(QIcon(":/images/setReadButton.png")); setAsNonReadAction = new QAction(tr("Set as unread"),this); setAsNonReadAction->setToolTip(tr("Set comic as unread")); + setAsNonReadAction->setData(SET_AS_NON_READ_ACTION_YL); + setAsNonReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_AS_NON_READ_ACTION_YL)); setAsNonReadAction->setIcon(QIcon(":/images/setUnread.png")); /*setAllAsReadAction = new QAction(tr("Set all as read"),this); @@ -396,120 +480,157 @@ void LibraryWindow::createActions() showHideMarksAction = new QAction(tr("Show/Hide marks"),this); showHideMarksAction->setToolTip(tr("Show or hide readed marks")); - showHideMarksAction->setShortcut(Qt::Key_M); + showHideMarksAction->setData(SHOW_HIDE_MARKS_ACTION_YL); + showHideMarksAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_HIDE_MARKS_ACTION_YL)); showHideMarksAction->setCheckable(true); showHideMarksAction->setIcon(QIcon(":/images/showMarks.png")); showHideMarksAction->setChecked(true); toggleFullScreenAction = new QAction(tr("Fullscreen mode on/off"),this); - toggleFullScreenAction->setToolTip(tr("Fullscreen mode on/off (F)")); - toggleFullScreenAction->setShortcut(Qt::Key_F); + toggleFullScreenAction->setToolTip(tr("Fullscreen mode on/off")); + toggleFullScreenAction->setData(TOGGLE_FULL_SCREEN_ACTION_YL); + toggleFullScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_YL)); QIcon icoFullscreenButton; icoFullscreenButton.addPixmap(QPixmap(":/images/main_toolbar/fullscreen.png"), QIcon::Normal); toggleFullScreenAction->setIcon(icoFullscreenButton); helpAboutAction = new QAction(this); helpAboutAction->setToolTip(tr("Help, About YACReader")); - helpAboutAction->setShortcut(Qt::Key_F1); + helpAboutAction->setData(HELP_ABOUT_ACTION_YL); + helpAboutAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HELP_ABOUT_ACTION_YL)); QIcon icoHelpButton; icoHelpButton.addPixmap(QPixmap(":/images/main_toolbar/help.png"), QIcon::Normal); helpAboutAction->setIcon(icoHelpButton); setRootIndexAction = new QAction(this); - setRootIndexAction->setShortcut(Qt::Key_0); + setRootIndexAction->setData(SET_ROOT_INDEX_ACTION_YL); + setRootIndexAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_ROOT_INDEX_ACTION_YL)); setRootIndexAction->setToolTip(tr("Select root node")); setRootIndexAction->setIcon(QIcon(":/images/setRoot.png")); expandAllNodesAction = new QAction(this); - expandAllNodesAction->setShortcut(tr("+")); expandAllNodesAction->setToolTip(tr("Expand all nodes")); + expandAllNodesAction->setData(EXPAND_ALL_NODES_ACTION_YL); + expandAllNodesAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPAND_ALL_NODES_ACTION_YL)); expandAllNodesAction->setIcon(QIcon(":/images/expand.png")); colapseAllNodesAction = new QAction(this); - colapseAllNodesAction->setShortcut(tr("-")); colapseAllNodesAction->setToolTip(tr("Colapse all nodes")); + colapseAllNodesAction->setData(COLAPSE_ALL_NODES_ACTION_YL); + colapseAllNodesAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(COLAPSE_ALL_NODES_ACTION_YL)); colapseAllNodesAction->setIcon(QIcon(":/images/colapse.png")); optionsAction = new QAction(this); - optionsAction->setShortcut(Qt::Key_C); optionsAction->setToolTip(tr("Show options dialog")); + optionsAction->setData(OPTIONS_ACTION_YL); + optionsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPTIONS_ACTION_YL)); QIcon icoSettingsButton; icoSettingsButton.addPixmap(QPixmap(":/images/main_toolbar/settings.png"), QIcon::Normal); optionsAction->setIcon(icoSettingsButton); serverConfigAction = new QAction(this); - serverConfigAction->setShortcut(Qt::Key_S); serverConfigAction->setToolTip(tr("Show comics server options dialog")); + serverConfigAction->setData(SERVER_CONFIG_ACTION_YL); + serverConfigAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SERVER_CONFIG_ACTION_YL)); QIcon icoServerButton; icoServerButton.addPixmap(QPixmap(":/images/main_toolbar/server.png"), QIcon::Normal); serverConfigAction->setIcon(icoServerButton); toggleComicsViewAction = new QAction(tr("Change between comics views"),this); - toggleComicsViewAction->setShortcut(Qt::Key_V); toggleComicsViewAction->setToolTip(tr("Change between comics views")); QIcon icoViewsButton; if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/grid.png"), QIcon::Normal); else icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/flow.png"), QIcon::Normal); + toggleComicsViewAction->setData(TOGGLE_COMICS_VIEW_ACTION_YL); + toggleComicsViewAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_COMICS_VIEW_ACTION_YL)); toggleComicsViewAction->setIcon(icoViewsButton); //socialAction = new QAction(this); openContainingFolderAction = new QAction(this); openContainingFolderAction->setText(tr("Open folder...")); + openContainingFolderAction->setData(OPEN_CONTAINING_FOLDER_ACTION_YL); + openContainingFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_CONTAINING_FOLDER_ACTION_YL)); openContainingFolderAction->setIcon(QIcon(":/images/open.png")); setFolderAsNotCompletedAction = new QAction(this); setFolderAsNotCompletedAction->setText(tr("Set as uncompleted")); setFolderAsNotCompletedAction->setVisible(false); + setFolderAsNotCompletedAction->setData(SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL); + setFolderAsNotCompletedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL)); setFolderAsCompletedAction = new QAction(this); setFolderAsCompletedAction->setText(tr("Set as completed")); setFolderAsCompletedAction->setVisible(false); + setFolderAsCompletedAction->setData(SET_FOLDER_AS_COMPLETED_ACTION_YL); + setFolderAsCompletedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_COMPLETED_ACTION_YL)); - setFolderAsFinishedAction = new QAction(this); - setFolderAsFinishedAction->setText(tr("Set as read")); - setFolderAsFinishedAction->setVisible(false); + setFolderAsReadAction = new QAction(this); + setFolderAsReadAction->setText(tr("Set as read")); + setFolderAsReadAction->setVisible(false); + setFolderAsReadAction->setData(SET_FOLDER_AS_READ_ACTION_YL); + setFolderAsReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_READ_ACTION_YL)); - setFolderAsNotFinishedAction = new QAction(this); - setFolderAsNotFinishedAction->setText(tr("Set as unread")); - setFolderAsNotFinishedAction->setVisible(false); + setFolderAsUnreadAction = new QAction(this); + setFolderAsUnreadAction->setText(tr("Set as unread")); + setFolderAsUnreadAction->setVisible(false); + setFolderAsUnreadAction->setData(SET_FOLDER_AS_UNREAD_ACTION_YL); + setFolderAsUnreadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_UNREAD_ACTION_YL)); openContainingFolderComicAction = new QAction(this); openContainingFolderComicAction->setText(tr("Open containing folder...")); + openContainingFolderComicAction->setData(OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL); + openContainingFolderComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL)); openContainingFolderComicAction->setIcon(QIcon(":/images/open.png")); resetComicRatingAction = new QAction(this); resetComicRatingAction->setText(tr("Reset comic rating")); + resetComicRatingAction->setData(RESET_COMIC_RATING_ACTION_YL); + resetComicRatingAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RESET_COMIC_RATING_ACTION_YL)); //Edit comics actions------------------------------------------------------ selectAllComicsAction = new QAction(this); selectAllComicsAction->setText(tr("Select all comics")); - selectAllComicsAction->setIcon(QIcon(":/images/selectAll.png")); + selectAllComicsAction->setData(SELECT_ALL_COMICS_ACTION_YL); + selectAllComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SELECT_ALL_COMICS_ACTION_YL)); + selectAllComicsAction->setIcon(QIcon(":/images/selectAll.png")); editSelectedComicsAction = new QAction(this); editSelectedComicsAction->setText(tr("Edit")); + editSelectedComicsAction->setData(EDIT_SELECTED_COMICS_ACTION_YL); + editSelectedComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EDIT_SELECTED_COMICS_ACTION_YL)); editSelectedComicsAction->setIcon(QIcon(":/images/editComic.png")); asignOrderAction = new QAction(this); asignOrderAction->setText(tr("Asign current order to comics")); + asignOrderAction->setData(ASIGN_ORDER_ACTION_YL); + asignOrderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ASIGN_ORDER_ACTION_YL)); asignOrderAction->setIcon(QIcon(":/images/asignNumber.png")); - forceConverExtractedAction = new QAction(this); - forceConverExtractedAction->setText(tr("Update cover")); - forceConverExtractedAction->setIcon(QIcon(":/images/importCover.png")); + forceCoverExtractedAction = new QAction(this); + forceCoverExtractedAction->setText(tr("Update cover")); + forceCoverExtractedAction->setData(FORCE_COVER_EXTRACTED_ACTION_YL); + forceCoverExtractedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(FORCE_COVER_EXTRACTED_ACTION_YL)); + forceCoverExtractedAction->setIcon(QIcon(":/images/importCover.png")); deleteComicsAction = new QAction(this); deleteComicsAction->setText(tr("Delete selected comics")); + deleteComicsAction->setData(DELETE_COMICS_ACTION_YL); + deleteComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(DELETE_COMICS_ACTION_YL)); deleteComicsAction->setIcon(QIcon(":/images/trash.png")); hideComicViewAction = new QAction(this); hideComicViewAction->setText(tr("Hide comic flow")); + hideComicViewAction->setData(HIDE_COMIC_VIEW_ACTION_YL); + hideComicViewAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HIDE_COMIC_VIEW_ACTION_YL)); hideComicViewAction->setIcon(QIcon(":/images/hideComicFlow.png")); hideComicViewAction->setCheckable(true); hideComicViewAction->setChecked(false); getInfoAction = new QAction(this); + getInfoAction->setData(GET_INFO_ACTION_YL); + getInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GET_INFO_ACTION_YL)); getInfoAction->setText(tr("Download tags from Comic Vine")); getInfoAction->setIcon(QIcon(":/images/getInfo.png")); //------------------------------------------------------------------------- @@ -545,8 +666,8 @@ void LibraryWindow::disableLibrariesActions(bool disabled) updateLibraryAction->setDisabled(disabled); renameLibraryAction->setDisabled(disabled); removeLibraryAction->setDisabled(disabled); - exportComicsInfo->setDisabled(disabled); - importComicsInfo->setDisabled(disabled); + exportComicsInfoAction->setDisabled(disabled); + importComicsInfoAction->setDisabled(disabled); exportLibraryAction->setDisabled(disabled); //importLibraryAction->setDisabled(disabled); } @@ -554,8 +675,8 @@ void LibraryWindow::disableLibrariesActions(bool disabled) void LibraryWindow::disableNoUpdatedLibrariesActions(bool disabled) { updateLibraryAction->setDisabled(disabled); - exportComicsInfo->setDisabled(disabled); - importComicsInfo->setDisabled(disabled); + exportComicsInfoAction->setDisabled(disabled); + importComicsInfoAction->setDisabled(disabled); exportLibraryAction->setDisabled(disabled); } @@ -571,8 +692,8 @@ void LibraryWindow::disableFoldersActions(bool disabled) { setFolderAsNotCompletedAction->setVisible(false); setFolderAsCompletedAction->setVisible(false); - setFolderAsFinishedAction->setVisible(false); - setFolderAsNotFinishedAction->setVisible(false); + setFolderAsReadAction->setVisible(false); + setFolderAsUnreadAction->setVisible(false); } } @@ -699,16 +820,16 @@ void LibraryWindow::createMenus() foldersView->addAction(setFolderAsCompletedAction); YACReader::addSperator(foldersView); - foldersView->addAction(setFolderAsFinishedAction); - foldersView->addAction(setFolderAsNotFinishedAction); + foldersView->addAction(setFolderAsReadAction); + foldersView->addAction(setFolderAsUnreadAction); selectedLibrary->addAction(updateLibraryAction); selectedLibrary->addAction(renameLibraryAction); selectedLibrary->addAction(removeLibraryAction); YACReader::addSperator(selectedLibrary); - selectedLibrary->addAction(exportComicsInfo); - selectedLibrary->addAction(importComicsInfo); + selectedLibrary->addAction(exportComicsInfoAction); + selectedLibrary->addAction(importComicsInfoAction); YACReader::addSperator(selectedLibrary); selectedLibrary->addAction(exportLibraryAction); @@ -827,8 +948,8 @@ void LibraryWindow::createConnections() //comicsInfoManagement - connect(exportComicsInfo,SIGNAL(triggered()),this,SLOT(showExportComicsInfo())); - connect(importComicsInfo,SIGNAL(triggered()),this,SLOT(showImportComicsInfo())); + connect(exportComicsInfoAction,SIGNAL(triggered()),this,SLOT(showExportComicsInfo())); + connect(importComicsInfoAction,SIGNAL(triggered()),this,SLOT(showImportComicsInfo())); //properties & config connect(propertiesDialog,SIGNAL(accepted()),this,SLOT(reloadCovers())); @@ -852,6 +973,7 @@ void LibraryWindow::createConnections() connect(serverConfigAction, SIGNAL(triggered()), serverConfigDialog, SLOT(show())); #endif connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); + connect(optionsDialog, SIGNAL(editShortcuts()),shortcutsDialog,SLOT(show())); //Folders filter //connect(clearFoldersFilter,SIGNAL(clicked()),foldersFilter,SLOT(clear())); @@ -862,8 +984,8 @@ void LibraryWindow::createConnections() connect(openContainingFolderComicAction,SIGNAL(triggered()),this,SLOT(openContainingFolderComic())); connect(setFolderAsNotCompletedAction,SIGNAL(triggered()),this,SLOT(setFolderAsNotCompleted())); connect(setFolderAsCompletedAction,SIGNAL(triggered()),this,SLOT(setFolderAsCompleted())); - connect(setFolderAsFinishedAction,SIGNAL(triggered()),this,SLOT(setFolderAsFinished())); - connect(setFolderAsNotFinishedAction,SIGNAL(triggered()),this,SLOT(setFolderAsNotFinished())); + connect(setFolderAsReadAction,SIGNAL(triggered()),this,SLOT(setFolderAsRead())); + connect(setFolderAsUnreadAction,SIGNAL(triggered()),this,SLOT(setFolderAsUnread())); connect(openContainingFolderAction,SIGNAL(triggered()),this,SLOT(openContainingFolder())); connect(resetComicRatingAction,SIGNAL(triggered()),this,SLOT(resetComicRating())); @@ -1387,8 +1509,8 @@ void LibraryWindow::setRootIndex() setFolderAsNotCompletedAction->setVisible(false); setFolderAsCompletedAction->setVisible(false); - setFolderAsFinishedAction->setVisible(false); - setFolderAsNotFinishedAction->setVisible(false); + setFolderAsReadAction->setVisible(false); + setFolderAsUnreadAction->setVisible(false); } @@ -1650,12 +1772,12 @@ void LibraryWindow::setFolderAsCompleted() dm->updateFolderCompletedStatus(foldersView->selectionModel()->selectedRows(),true); } -void LibraryWindow::setFolderAsFinished() +void LibraryWindow::setFolderAsRead() { dm->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),true); } -void LibraryWindow::setFolderAsNotFinished() +void LibraryWindow::setFolderAsUnread() { dm->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),false); } @@ -1907,8 +2029,8 @@ void LibraryWindow::updateFoldersViewConextMenu(const QModelIndex &mi) bool isFinished = item->data(TreeModel::Finished).toBool(); bool isCompleted = item->data(TreeModel::Completed).toBool(); - setFolderAsFinishedAction->setVisible(!isFinished); - setFolderAsNotFinishedAction->setVisible(isFinished); + setFolderAsReadAction->setVisible(!isFinished); + setFolderAsUnreadAction->setVisible(isFinished); setFolderAsCompletedAction->setVisible(!isCompleted); setFolderAsNotCompletedAction->setVisible(isCompleted); diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index 8092a774..d44ec840 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -54,6 +54,7 @@ class ClassicComicsView; class GridComicsView; class ComicsViewTransition; class EmptyFolderWidget; +class ShortcutsDialog; #include "comic_db.h" @@ -76,6 +77,7 @@ private: RenameLibraryDialog * renameLibraryDialog; PropertiesDialog * propertiesDialog; ComicVineDialog * comicVineDialog; + ShortcutsDialog * shortcutsDialog; //YACReaderSocialDialog * socialDialog; bool fullscreen; bool importedCovers; //if true, the library is read only (not updates,open comic or properties) @@ -125,8 +127,8 @@ private: QAction * createLibraryAction; QAction * openLibraryAction; - QAction * exportComicsInfo; - QAction * importComicsInfo; + QAction * exportComicsInfoAction; + QAction * importComicsInfoAction; QAction * exportLibraryAction; QAction * importLibraryAction; @@ -151,8 +153,8 @@ private: QAction * setFolderAsNotCompletedAction; QAction * setFolderAsCompletedAction; //-- - QAction * setFolderAsFinishedAction; - QAction * setFolderAsNotFinishedAction; + QAction * setFolderAsReadAction; + QAction * setFolderAsUnreadAction; QAction * openContainingFolderComicAction; QAction * setAsReadAction; @@ -167,7 +169,7 @@ private: QAction * selectAllComicsAction; QAction * editSelectedComicsAction; QAction * asignOrderAction; - QAction * forceConverExtractedAction; + QAction * forceCoverExtractedAction; QAction * deleteComicsAction; QAction * hideComicViewAction; @@ -204,6 +206,7 @@ private: void createConnections(); void doLayout(); void doDialogs(); + void setUpShortcutsManagement(); void doModels(); void disconnectComicsViewConnections(ComicsView * widget); void doComicsViewConnections(); @@ -258,8 +261,8 @@ public slots: void openContainingFolder(); void setFolderAsNotCompleted(); void setFolderAsCompleted(); - void setFolderAsFinished(); - void setFolderAsNotFinished(); + void setFolderAsRead(); + void setFolderAsUnread(); void openContainingFolderComic(); void deleteCurrentLibrary(); void removeLibrary(); diff --git a/YACReaderLibrary/options_dialog.cpp b/YACReaderLibrary/options_dialog.cpp index d8b15a92..2ab02678 100644 --- a/YACReaderLibrary/options_dialog.cpp +++ b/YACReaderLibrary/options_dialog.cpp @@ -20,33 +20,59 @@ FlowType flowType = Strip; OptionsDialog::OptionsDialog(QWidget * parent) :YACReaderOptionsDialog(parent) { - QVBoxLayout * layout = new QVBoxLayout; + QTabWidget * tabWidget = new QTabWidget(); - QHBoxLayout * switchFlowType = new QHBoxLayout; - switchFlowType->addStretch(); - switchFlowType->addWidget(useGL); + QVBoxLayout * layout = new QVBoxLayout(this); + + QVBoxLayout * flowLayout = new QVBoxLayout; + QVBoxLayout * generalLayout = new QVBoxLayout(); + + QHBoxLayout * switchFlowType = new QHBoxLayout; + switchFlowType->addStretch(); + switchFlowType->addWidget(useGL); + + QHBoxLayout * buttons = new QHBoxLayout(); + buttons->addStretch(); + buttons->addWidget(accept); + buttons->addWidget(cancel); + + flowLayout->addWidget(sw); + flowLayout->addWidget(gl); + flowLayout->addLayout(switchFlowType); + + QVBoxLayout * shortcutsLayout = new QVBoxLayout(); + QPushButton * shortcutsButton = new QPushButton(tr("Edit shortcuts")); + shortcutsLayout->addWidget(shortcutsButton); + sw->hide(); - QHBoxLayout * buttons = new QHBoxLayout(); - buttons->addStretch(); - buttons->addWidget(accept); - buttons->addWidget(cancel); + QWidget * comicFlowW = new QWidget; + comicFlowW->setLayout(flowLayout); - layout->addWidget(sw); - layout->addWidget(gl); - layout->addLayout(switchFlowType); - layout->addLayout(buttons); + QGroupBox *generalBox = new QGroupBox(tr("Shortcuts")); + generalBox->setLayout(shortcutsLayout); + generalLayout->addWidget(generalBox); + generalLayout->addStretch(); - sw->hide(); + QWidget * generalW = new QWidget; + generalW->setLayout(generalLayout); - setLayout(layout); - //restoreOptions(settings); //load options - //resize(200,0); - setModal (true); - setWindowTitle(tr("Options")); + tabWidget->addTab(comicFlowW,tr("Comic Flow")); + tabWidget->addTab(generalW,tr("General")); - this->layout()->setSizeConstraint(QLayout::SetFixedSize); + layout->addWidget(tabWidget); + layout->addLayout(buttons); + setLayout(layout); + //restoreOptions(settings); //load options + //resize(200,0); + setModal (true); + setWindowTitle(tr("Options")); + + this->layout()->setSizeConstraint(QLayout::SetFixedSize); + + connect(shortcutsButton,SIGNAL(clicked()),this,SIGNAL(editShortcuts())); } + diff --git a/YACReaderLibrary/options_dialog.h b/YACReaderLibrary/options_dialog.h index d42a9ead..d597f686 100644 --- a/YACReaderLibrary/options_dialog.h +++ b/YACReaderLibrary/options_dialog.h @@ -12,6 +12,8 @@ class OptionsDialog : public YACReaderOptionsDialog Q_OBJECT public: OptionsDialog(QWidget * parent = 0); +signals: + void editShortcuts(); }; diff --git a/images/accept_shortcut.png b/images/accept_shortcut.png new file mode 100644 index 0000000000000000000000000000000000000000..4d2b852dbb11826d538a648f3f54978f42a34aa2 GIT binary patch literal 239 zcmeAS@N?(olHy`uVBq!ia0vp^{2desrQkZrc3qwyCW}}k0)(F2aY-^wQW4T&dvJw=O(A{p<9+w;u_(-I j@a2;<=8HJSz+u1;{43|ILXVFL&~Xf&u6{1-oD!M<=Sfdc literal 0 HcmV?d00001 diff --git a/images/clear_shortcut.png b/images/clear_shortcut.png new file mode 100644 index 0000000000000000000000000000000000000000..e7d2102090a0163f4d43f534146efb288749987e GIT binary patch literal 201 zcmeAS@N?(olHy`uVBq!ia0vp^{2}1{rUgjo_S{j>+FR3_;{%avpQ9##6^Uda62oW zNGrK>{q}EpdHFWU#B;}zr1coeS%oJ(@KM^!6@J`fD+9CD+y$Nv>c)p|`0$7@sMRc6 xeAy*)*@-ZAh#RIKn@O1TaS?83{1OR@5MT-Cc literal 0 HcmV?d00001 diff --git a/shortcuts_management/actions_groups_model.cpp b/shortcuts_management/actions_groups_model.cpp new file mode 100644 index 00000000..47ba2e91 --- /dev/null +++ b/shortcuts_management/actions_groups_model.cpp @@ -0,0 +1,80 @@ +#include "actions_groups_model.h" + +ActionsGroupsModel::ActionsGroupsModel(QObject *parent) : + QAbstractItemModel(parent) +{ +} + +int ActionsGroupsModel::rowCount(const QModelIndex &parent) const +{ + return groups.length(); +} + +int ActionsGroupsModel::columnCount(const QModelIndex &parent) const +{ + return 1; +} + +QModelIndex ActionsGroupsModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column, &groups.at(row).getActions()); +} + +QVariant ActionsGroupsModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + if (role == Qt::DecorationRole) + return QVariant(groups.at(index.row()).getIcon()); + + if (role != Qt::DisplayRole) + return QVariant(); + + return QVariant(groups[index.row()].getName()); +} + +QModelIndex ActionsGroupsModel::parent(const QModelIndex &index) const +{ + return QModelIndex(); +} + +void ActionsGroupsModel::addActionsGroup(const ActionsGroup &group) +{ + beginInsertRows(QModelIndex(),groups.length()-1,groups.length()); + groups.push_back(group); + endInsertRows(); +} + +QList ActionsGroupsModel::getActions(const QModelIndex &mi) +{ + if(mi.isValid()) + return groups[mi.row()].getActions(); + return QList(); +} + +//------------------------------------------------------------------- + +ActionsGroup::ActionsGroup(const QString &name, const QIcon &icon, QList &actions) + :name(name), icon(icon), actions(actions) +{ + +} + +QString ActionsGroup::getName() const +{ + return name; +} + +QIcon ActionsGroup::getIcon() const +{ + return icon; +} + +QList ActionsGroup::getActions() const +{ + return actions; +} diff --git a/shortcuts_management/actions_groups_model.h b/shortcuts_management/actions_groups_model.h new file mode 100644 index 00000000..91fb5a7c --- /dev/null +++ b/shortcuts_management/actions_groups_model.h @@ -0,0 +1,44 @@ +#ifndef ACTIONS_GROUPS_MODEL_H +#define ACTIONS_GROUPS_MODEL_H + +#include +#include + +class QAction; + +class ActionsGroup +{ +public: + ActionsGroup(const QString & name, const QIcon & icon, QList & actions); + QString getName() const; + QIcon getIcon() const; + QList getActions() const; +protected: + QString name; + QIcon icon; + QList actions; +}; + +class ActionsGroupsModel : public QAbstractItemModel +{ + Q_OBJECT +public: + explicit ActionsGroupsModel(QObject *parent = 0); + + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent = QModelIndex()) const; + QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; + QVariant data(const QModelIndex &index, int role) const; + QModelIndex parent(const QModelIndex &index) const; + + void addActionsGroup(const ActionsGroup & group); + QList getActions(const QModelIndex & mi); +signals: + +public slots: + +protected: + QList groups; +}; + +#endif // ACTIONS_GROUPS_MODEL_H diff --git a/shortcuts_management/actions_shortcuts_model.cpp b/shortcuts_management/actions_shortcuts_model.cpp new file mode 100644 index 00000000..0a9760d1 --- /dev/null +++ b/shortcuts_management/actions_shortcuts_model.cpp @@ -0,0 +1,92 @@ +#include "actions_shortcuts_model.h" +#include "shortcuts_manager.h" + +#include + +ActionsShortcutsModel::ActionsShortcutsModel(QObject *parent) : + QAbstractItemModel(parent) +{ + +} + +int ActionsShortcutsModel::rowCount(const QModelIndex &parent) const +{ + return actions.length(); +} + +int ActionsShortcutsModel::columnCount(const QModelIndex &parent) const +{ + return 3; +} + +QModelIndex ActionsShortcutsModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column, actions[row]); +} + +Qt::ItemFlags ActionsShortcutsModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + if(index.column() == KEYS) + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} + +QVariant ActionsShortcutsModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + if (role == Qt::DecorationRole && index.column() == ICON) + return QVariant(actions[index.row()]->icon()); + + if (role == Qt::TextAlignmentRole) + { + switch(index.column()) + { + case ICON: + return QVariant(Qt::AlignHCenter | Qt::AlignVCenter); + case NAME: + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + case KEYS: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + } + } + + if (role != Qt::DisplayRole) + return QVariant(); + + if (index.column() == NAME) + return QVariant(actions[index.row()]->toolTip()); + if (index.column() == KEYS) + return QVariant(actions[index.row()]->shortcut().toString()); + + return QVariant(); +} + +bool ActionsShortcutsModel::setData(const QModelIndex &index, const QVariant &value, int role) +{ + if(index.column() == KEYS) + { + actions[index.row()]->setShortcut(value.toString()); + ShortcutsManager::getShortcutsManager().saveShortcut(actions[index.row()]); + return true; + } + return false; +} + +QModelIndex ActionsShortcutsModel::parent(const QModelIndex &index) const +{ + return QModelIndex(); +} + +void ActionsShortcutsModel::addActions(const QList actions) +{ + beginResetModel(); + this->actions = actions; + endResetModel(); +} diff --git a/shortcuts_management/actions_shortcuts_model.h b/shortcuts_management/actions_shortcuts_model.h new file mode 100644 index 00000000..58ea16f5 --- /dev/null +++ b/shortcuts_management/actions_shortcuts_model.h @@ -0,0 +1,38 @@ +#ifndef ACTIONS_SHORTCUTS_MODEL_H +#define ACTIONS_SHORTCUTS_MODEL_H + +#include + +class QAction; + +class ActionsShortcutsModel : public QAbstractItemModel +{ + Q_OBJECT +public: + explicit ActionsShortcutsModel(QObject *parent = 0); + + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent = QModelIndex()) const; + QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; + QVariant data(const QModelIndex &index, int role) const; + bool setData(const QModelIndex &index, const QVariant &value, int role); + QModelIndex parent(const QModelIndex &index) const; + + void addActions(const QList actions); + Qt::ItemFlags flags(const QModelIndex &index) const; + + + enum Columns { + ICON = 0, + NAME, + KEYS + }; +signals: + +public slots: + +protected: + QList actions; +}; + +#endif // ACTIONS_SHORTCUTS_MODEL_H diff --git a/shortcuts_management/edit_shortcut_item_delegate.cpp b/shortcuts_management/edit_shortcut_item_delegate.cpp new file mode 100644 index 00000000..5b7d3627 --- /dev/null +++ b/shortcuts_management/edit_shortcut_item_delegate.cpp @@ -0,0 +1,145 @@ +#include "edit_shortcut_item_delegate.h" + +#include + +EditShortcutItemDelegate::EditShortcutItemDelegate(QObject *parent) : + QItemDelegate(parent) +{ +} + +QWidget *EditShortcutItemDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const +{ + KeySequenceLineEdit * editor = new KeySequenceLineEdit(parent); + connect(editor,SIGNAL(editingFinished()),this,SLOT(closeShortcutEditor())); + return editor; +} + +void EditShortcutItemDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const +{ + QString value = index.model()->data(index, Qt::DisplayRole).toString(); + + KeySequenceLineEdit * lineEdit = static_cast(editor); + lineEdit->setText(value); +} + +void EditShortcutItemDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const +{ + KeySequenceLineEdit *lineEdit = static_cast(editor); + + model->setData(index, lineEdit->text(), Qt::EditRole); +} + +void EditShortcutItemDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &mi) const +{ + editor->setGeometry(option.rect); +} + +bool EditShortcutItemDelegate::eventFilter(QObject* editor, QEvent* event) +{ + if(event->type()==QEvent::KeyPress) + return false; + return QItemDelegate::eventFilter(editor, event); +} + +void EditShortcutItemDelegate::closeShortcutEditor() +{ + emit commitData(static_cast(sender())); + emit closeEditor(static_cast(sender()),QAbstractItemDelegate::NoHint); +} + +//TODO uncoment commented code for enabling concatenated shortcuts +KeySequenceLineEdit::KeySequenceLineEdit(QWidget *parent) + :QLineEdit(parent)//,numKeys(0) +{ + //keys[0] = keys[1] = keys[2] = keys[3] = 0; + setAlignment(Qt::AlignRight); + + QPixmap clearPixmap(":/images/clear_shortcut.png"); + QPixmap acceptPixmap(":/images/accept_shortcut.png"); + + clearButton = new QToolButton(this); + acceptButton = new QToolButton(this); + QString buttonsStyle = "QToolButton { border: none; padding: 0px; }"; + + clearButton->setIcon(QIcon(clearPixmap)); + clearButton->setIconSize(clearPixmap.size()); + clearButton->setCursor(Qt::ArrowCursor); + clearButton->setStyleSheet(buttonsStyle); + + acceptButton->setIcon(QIcon(acceptPixmap)); + acceptButton->setIconSize(acceptPixmap.size()); + acceptButton->setCursor(Qt::ArrowCursor); + acceptButton->setStyleSheet(buttonsStyle); + + connect(clearButton, SIGNAL(clicked()), this, SLOT(clear())); + connect(acceptButton, SIGNAL(clicked()), this, SIGNAL(editingFinished())); +} + +void KeySequenceLineEdit::resizeEvent(QResizeEvent *) +{ + QSize szClear = clearButton->sizeHint(); + //int frameWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth); + int leftMargin = style()->pixelMetric(QStyle::PM_LayoutLeftMargin); + int topMargin = style()->pixelMetric(QStyle::PM_LayoutTopMargin); + clearButton->move(0 + leftMargin,topMargin-4); + + acceptButton->move( leftMargin + szClear.width(),topMargin-4); + +} + +void KeySequenceLineEdit::keyPressEvent(QKeyEvent * e) +{ + int key = e->key(); + + + //if ( numKeys > 3 || + if ( key == Qt::Key_Control || + key == Qt::Key_Shift || + key == Qt::Key_Meta || + key == Qt::Key_Alt ) + return; + + key |= translateModifiers(e->modifiers(), e->text()); + + /*switch (numKeys) { + case 0: + keys[0] = nextKey; + break; + case 1: + keys[1] = nextKey; + break; + case 2: + keys[2] = nextKey; + break; + case 3: + keys[3] = nextKey; + break; + default: + break; + }*/ + //numKeys++; + QKeySequence keySequence = QKeySequence(key); + setText(keySequence.toString(QKeySequence::NativeText)); + e->accept(); +} + +int KeySequenceLineEdit::translateModifiers(Qt::KeyboardModifiers state, + const QString &text) +{ + int result = 0; + // The shift modifier only counts when it is not used to type a symbol + // that is only reachable using the shift key anyway + if ((state & Qt::ShiftModifier) && (text.size() == 0 + || !text.at(0).isPrint() + || text.at(0).isLetterOrNumber() + || text.at(0).isSpace())) + result |= Qt::SHIFT; + if (state & Qt::ControlModifier) + result |= Qt::CTRL; + if (state & Qt::MetaModifier) + result |= Qt::META; + if (state & Qt::AltModifier) + result |= Qt::ALT; + return result; +} + diff --git a/shortcuts_management/edit_shortcut_item_delegate.h b/shortcuts_management/edit_shortcut_item_delegate.h new file mode 100644 index 00000000..7cc1e4e8 --- /dev/null +++ b/shortcuts_management/edit_shortcut_item_delegate.h @@ -0,0 +1,48 @@ +#ifndef EDIT_SHORTCUT_ITEM_DELEGATE_H +#define EDIT_SHORTCUT_ITEM_DELEGATE_H + +#include +#include +#include +#include +#include + +class KeySequenceLineEdit : public QLineEdit +{ + Q_OBJECT +public: + explicit KeySequenceLineEdit(QWidget *parent = 0); + +protected: + //int numKeys; + //int keys[4]; + void keyPressEvent(QKeyEvent *); + int translateModifiers(Qt::KeyboardModifiers state, const QString &text); + void resizeEvent(QResizeEvent *); + +private: + QToolButton *clearButton; + QToolButton *acceptButton; +}; + +class EditShortcutItemDelegate : public QItemDelegate +{ + Q_OBJECT +public: + explicit EditShortcutItemDelegate(QObject *parent = 0); + + QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &option, + const QModelIndex &index) const; + void setEditorData(QWidget *editor, const QModelIndex &index) const; + void setModelData(QWidget *editor, QAbstractItemModel *model, + const QModelIndex &index) const; + void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex & mi) const; + bool eventFilter(QObject *editor, QEvent *event); +signals: + +public slots: + void closeShortcutEditor(); + +}; + +#endif // EDIT_SHORTCUT_ITEM_DELEGATE_H diff --git a/shortcuts_management/shortcuts_dialog.cpp b/shortcuts_management/shortcuts_dialog.cpp new file mode 100644 index 00000000..d48c87d8 --- /dev/null +++ b/shortcuts_management/shortcuts_dialog.cpp @@ -0,0 +1,82 @@ +#include "shortcuts_dialog.h" + +#include "actions_groups_model.h" +#include "actions_shortcuts_model.h" +#include "edit_shortcut_item_delegate.h" + +#include +#include +#include +#include +#include +#include +#include + +#include "QsLog.h" + +ShortcutsDialog::ShortcutsDialog(QWidget *parent) : + QDialog(parent) +{ + QPushButton * resetButton = new QPushButton(tr("Restore defaults"),this); + QLabel * infoLabel = new QLabel(tr("To change a shortcut, select it, click in the key combination and type the new keys.")); + QVBoxLayout * layout = new QVBoxLayout(this); + QSplitter * splitter = new QSplitter(this); + actionsGroupsListView = new QListView(this); + + actionsTableView = new QTableView(this); + actionsTableView->verticalHeader()->setHidden(true); + actionsTableView->horizontalHeader()->setHidden(true); + splitter->addWidget(actionsGroupsListView); + splitter->addWidget(actionsTableView); + splitter->setStretchFactor(1,1); + splitter->setSizes(QList() << 200 << 400); + + layout->addWidget(infoLabel,0); + layout->addWidget(splitter,1); + layout->addWidget(resetButton,0,Qt::AlignRight); + + setLayout(layout); + + groupsModel = new ActionsGroupsModel(); + actionsModel = new ActionsShortcutsModel(); + actionsGroupsListView->setModel(groupsModel); + actionsTableView->setModel(actionsModel); + actionsTableView->setColumnWidth(0,30); + actionsTableView->setColumnWidth(1,270); + actionsTableView->horizontalHeader()->setStretchLastSection(true); + actionsTableView->setSelectionBehavior(QAbstractItemView::SelectRows); + actionsTableView->setShowGrid(false); + actionsTableView->setItemDelegateForColumn(ActionsShortcutsModel::KEYS,new EditShortcutItemDelegate(this)); + actionsTableView->installEventFilter(this); + /*actionsTableView->setStyleSheet("QTableView {outline: 0px;}" + "QTableView::item {outline: 0px;}"); + "QTableView {border:0px;}" + "QTableView::item:selected {outline: 0px; border: 0px;}" + "");*/ + + + connect(resetButton,SIGNAL(clicked()),this,SLOT(resetToDefaults())); + connect(actionsGroupsListView->selectionModel(),SIGNAL(currentChanged(QModelIndex,QModelIndex)),this,SLOT(loadShortcuts(QModelIndex,QModelIndex))); //clicked(QModelIndex) doesn't work :S + + setFixedSize(640,480); + setWindowTitle(tr("Shortcuts settings")); + + setModal(true); +} + +void ShortcutsDialog::addActionsGroup(const QString &name, const QIcon &ico, QList &group) +{ + //TODO + //groups model add + groupsModel->addActionsGroup(ActionsGroup(name,ico,group)); +} + +void ShortcutsDialog::resetToDefaults() +{ + +} + +void ShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex &mi2) +{ + actionsModel->addActions(groupsModel->getActions(mi)); +} diff --git a/shortcuts_management/shortcuts_dialog.h b/shortcuts_management/shortcuts_dialog.h new file mode 100644 index 00000000..d63c9bcd --- /dev/null +++ b/shortcuts_management/shortcuts_dialog.h @@ -0,0 +1,32 @@ +#ifndef SHORTCUTS_DIALOG_H +#define SHORTCUTS_DIALOG_H + +#include +#include + +class QListView; +class QTableView; + +class ActionsGroupsModel; +class ActionsShortcutsModel; + +class ShortcutsDialog : public QDialog +{ + Q_OBJECT +public: + explicit ShortcutsDialog(QWidget * parent = 0); + void addActionsGroup(const QString & name, const QIcon & ico, QList & group); +signals: + +public slots: + void resetToDefaults(); + void loadShortcuts(const QModelIndex & mi,const QModelIndex &mi2); + +protected: + QListView * actionsGroupsListView; + QTableView * actionsTableView; + ActionsGroupsModel * groupsModel; + ActionsShortcutsModel * actionsModel; +}; + +#endif // SHORTCUTS_DIALOG_H diff --git a/shortcuts_management/shortcuts_management.pri b/shortcuts_management/shortcuts_management.pri new file mode 100644 index 00000000..e85a6912 --- /dev/null +++ b/shortcuts_management/shortcuts_management.pri @@ -0,0 +1,16 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += \ + $$PWD/shortcuts_dialog.h \ + $$PWD/actions_groups_model.h \ + $$PWD/actions_shortcuts_model.h \ + $$PWD/edit_shortcut_item_delegate.h \ + $$PWD/shortcuts_manager.h + +SOURCES += \ + $$PWD/shortcuts_dialog.cpp \ + $$PWD/actions_groups_model.cpp \ + $$PWD/actions_shortcuts_model.cpp \ + $$PWD/edit_shortcut_item_delegate.cpp \ + $$PWD/shortcuts_manager.cpp diff --git a/shortcuts_management/shortcuts_manager.cpp b/shortcuts_management/shortcuts_manager.cpp new file mode 100644 index 00000000..eca6e45b --- /dev/null +++ b/shortcuts_management/shortcuts_manager.cpp @@ -0,0 +1,49 @@ +#include "shortcuts_manager.h" + +#include +#include +#include "yacreader_global.h" + +ShortcutsManager::ShortcutsManager() +{ + initDefaultShorcuts(); +} + +void ShortcutsManager::initDefaultShorcuts() +{ + defaultShorcuts.insert(CREATE_LIBRARY_ACTION_YL,Qt::Key_A); + defaultShorcuts.insert(OPEN_LIBRARY_ACTION_YL,Qt::Key_O); + defaultShorcuts.insert(UPDATE_LIBRARY_ACTION_YL,Qt::Key_U); + defaultShorcuts.insert(RENAME_LIBRARY_ACTION_YL,Qt::Key_R); + defaultShorcuts.insert(OPEN_COMIC_ACTION_YL,Qt::Key_Return); + defaultShorcuts.insert(SHOW_HIDE_MARKS_ACTION_YL,Qt::Key_M); + defaultShorcuts.insert(TOGGLE_FULL_SCREEN_ACTION_YL,Qt::Key_F); + defaultShorcuts.insert(HELP_ABOUT_ACTION_YL,Qt::Key_F1); + defaultShorcuts.insert(SET_ROOT_INDEX_ACTION_YL,Qt::Key_0); + defaultShorcuts.insert(EXPAND_ALL_NODES_ACTION_YL,Qt::Key_Plus); + defaultShorcuts.insert(COLAPSE_ALL_NODES_ACTION_YL,Qt::Key_Minus); + defaultShorcuts.insert(OPTIONS_ACTION_YL,Qt::Key_C); + defaultShorcuts.insert(SERVER_CONFIG_ACTION_YL,Qt::Key_S); + defaultShorcuts.insert(TOGGLE_COMICS_VIEW_ACTION_YL,Qt::Key_V); +} + +void ShortcutsManager::resetToDefaults() +{ + //TODO reset to defaults +} + +QString ShortcutsManager::getShortcut(const QString &name) +{ + QSettings s(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + s.beginGroup("shortcuts"); + + return s.value(name,defaultShorcuts.value(name)).toString(); +} + +void ShortcutsManager::saveShortcut(QAction *action) +{ + QSettings s(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + s.beginGroup("shortcuts"); + + return s.setValue(action->data().toString() , action->shortcut().toString()); +} diff --git a/shortcuts_management/shortcuts_manager.h b/shortcuts_management/shortcuts_manager.h new file mode 100644 index 00000000..d8cf0c69 --- /dev/null +++ b/shortcuts_management/shortcuts_manager.h @@ -0,0 +1,73 @@ +#ifndef SHORTCUTS_MANAGER_H +#define SHORTCUTS_MANAGER_H + +#include +#include +#include +#include + + +class QAction; + +//QAction: used setData() and data() for storing (userData) an identifier for each QAction. This value is ussed in QSettings + +class ShortcutsManager +{ +private: + ShortcutsManager(); + QMap defaultShorcuts; + void initDefaultShorcuts(); +public: + static ShortcutsManager & getShortcutsManager() + { + static ShortcutsManager manager; + return manager; + } + + void resetToDefaults(); + QString getShortcut(const QString & name); + void saveShortcut(QAction * action); +}; + +//ACTION NAMES YACReaderLibrary +#define BACK_ACTION_YL "BACK_ACTION_YL" +#define FORWARD_ACTION_YL "FORWARD_ACTION_YL" +#define CREATE_LIBRARY_ACTION_YL "CREATE_LIBRARY_ACTION_YL" +#define OPEN_LIBRARY_ACTION_YL "OPEN_LIBRARY_ACTION_YL" +#define EXPORT_COMICS_INFO_YL "EXPORT_COMICS_INFO_YL" +#define IMPORT_COMICS_INFO_YL "IMPORT_COMICS_INFO_YL" +#define EXPORT_LIBRARY_ACTION_YL "EXPORT_LIBRARY_ACTION_YL" +#define IMPORT_LIBRARY_ACTION_YL "IMPORT_LIBRARY_ACTION_YL" +#define UPDATE_LIBRARY_ACTION_YL "UPDATE_LIBRARY_ACTION_YL" +#define RENAME_LIBRARY_ACTION_YL "RENAME_LIBRARY_ACTION_YL" +#define REMOVE_LIBRARY_ACTION_YL "REMOVE_LIBRARY_ACTION_YL" +#define OPEN_COMIC_ACTION_YL "OPEN_COMIC_ACTION_YL" +#define SET_AS_READ_ACTION_YL "SET_AS_READ_ACTION_YL" +#define SET_AS_NON_READ_ACTION_YL "SET_AS_NON_READ_ACTION_YL" +#define SHOW_HIDE_MARKS_ACTION_YL "SHOW_HIDE_MARKS_ACTION_YL" +#define TOGGLE_FULL_SCREEN_ACTION_YL "TOGGLE_FULL_SCREEN_ACTION_YL" +#define HELP_ABOUT_ACTION_YL "HELP_ABOUT_ACTION_YL" +#define SET_ROOT_INDEX_ACTION_YL "SET_ROOT_INDEX_ACTION_YL" +#define EXPAND_ALL_NODES_ACTION_YL "EXPAND_ALL_NODES_ACTION_YL" +#define COLAPSE_ALL_NODES_ACTION_YL "COLAPSE_ALL_NODES_ACTION_YL" +#define OPTIONS_ACTION_YL "OPTIONS_ACTION_YL" +#define SERVER_CONFIG_ACTION_YL "SERVER_CONFIG_ACTION_YL" +#define TOGGLE_COMICS_VIEW_ACTION_YL "TOGGLE_COMICS_VIEW_ACTION_YL" +#define OPEN_CONTAINING_FOLDER_ACTION_YL "OPEN_CONTAINING_FOLDER_ACTION_YL" +#define SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL "SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL" +#define SET_FOLDER_AS_COMPLETED_ACTION_YL "SET_FOLDER_AS_COMPLETED_ACTION_YL" +#define SET_FOLDER_AS_READ_ACTION_YL "SET_FOLDER_AS_READ_ACTION_YL" +#define SET_FOLDER_AS_UNREAD_ACTION_YL "SET_FOLDER_AS_UNREAD_ACTION_YL" +#define OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL "OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL" +#define RESET_COMIC_RATING_ACTION_YL "RESET_COMIC_RATING_ACTION_YL" +#define SELECT_ALL_COMICS_ACTION_YL "SELECT_ALL_COMICS_ACTION_YL" +#define EDIT_SELECTED_COMICS_ACTION_YL "EDIT_SELECTED_COMICS_ACTION_YL" +#define ASIGN_ORDER_ACTION_YL "ASIGN_ORDER_ACTION_YL" +#define FORCE_COVER_EXTRACTED_ACTION_YL "FORCE_COVER_EXTRACTED_ACTION_YL" +#define DELETE_COMICS_ACTION_YL "DELETE_COMICS_ACTION_YL" +#define HIDE_COMIC_VIEW_ACTION_YL "HIDE_COMIC_VIEW_ACTION_YL" +#define GET_INFO_ACTION_YL "GET_INFO_ACTION_YL" + + + +#endif // SHORTCUTS_MANAGER_H From 30a717c5d2a58e6aad9c27eae56e694db99a4fd0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Thu, 17 Jul 2014 22:41:32 +0200 Subject: [PATCH 18/34] fixed a couple of names --- YACReaderLibrary/library_window.cpp | 8 ++++---- shortcuts_management/shortcuts_manager.h | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index d61d796a..1aa167fd 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -412,14 +412,14 @@ void LibraryWindow::createActions() exportComicsInfoAction = new QAction(tr("Export comics info"),this); exportComicsInfoAction->setToolTip(tr("Export comics info")); - exportComicsInfoAction->setData(EXPORT_COMICS_INFO_YL); - exportComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_COMICS_INFO_YL)); + exportComicsInfoAction->setData(EXPORT_COMICS_INFO_ACTION_YL); + exportComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_COMICS_INFO_ACTION_YL)); exportComicsInfoAction->setIcon(QIcon(":/images/exportComicsInfoIcon.png")); importComicsInfoAction = new QAction(tr("Import comics info"),this); importComicsInfoAction->setToolTip(tr("Import comics info")); - importComicsInfoAction->setData(IMPORT_COMICS_INFO_YL); - importComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_COMICS_INFO_YL)); + importComicsInfoAction->setData(IMPORT_COMICS_INFO_ACTION_YL); + importComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_COMICS_INFO_ACTION_YL)); importComicsInfoAction->setIcon(QIcon(":/images/importComicsInfoIcon.png")); exportLibraryAction = new QAction(tr("Pack covers"),this); diff --git a/shortcuts_management/shortcuts_manager.h b/shortcuts_management/shortcuts_manager.h index d8cf0c69..a44d641b 100644 --- a/shortcuts_management/shortcuts_manager.h +++ b/shortcuts_management/shortcuts_manager.h @@ -34,8 +34,8 @@ public: #define FORWARD_ACTION_YL "FORWARD_ACTION_YL" #define CREATE_LIBRARY_ACTION_YL "CREATE_LIBRARY_ACTION_YL" #define OPEN_LIBRARY_ACTION_YL "OPEN_LIBRARY_ACTION_YL" -#define EXPORT_COMICS_INFO_YL "EXPORT_COMICS_INFO_YL" -#define IMPORT_COMICS_INFO_YL "IMPORT_COMICS_INFO_YL" +#define EXPORT_COMICS_INFO_ACTION_YL "EXPORT_COMICS_INFO_ACTION_YL" +#define IMPORT_COMICS_INFO_ACTION_YL "IMPORT_COMICS_INFO_ACTION_YL" #define EXPORT_LIBRARY_ACTION_YL "EXPORT_LIBRARY_ACTION_YL" #define IMPORT_LIBRARY_ACTION_YL "IMPORT_LIBRARY_ACTION_YL" #define UPDATE_LIBRARY_ACTION_YL "UPDATE_LIBRARY_ACTION_YL" From 07b2cbb7d07a8d425772b6c7c285667024157f23 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Fri, 18 Jul 2014 20:03:37 +0200 Subject: [PATCH 19/34] added shortcuts management to YACReader added shortcuts management to keyPressEvent keys --- YACReader/YACReader.pri | 1 + YACReader/YACReader.pro | 2 +- YACReader/magnifying_glass.cpp | 63 ++- YACReader/main_window_viewer.cpp | 481 ++++++++++++------ YACReader/main_window_viewer.h | 23 +- YACReader/options_dialog.cpp | 1 + YACReader/viewer.cpp | 124 +++-- YACReader/yacreader_images.qrc | 2 + YACReaderLibrary/YACReaderLibrary.pro | 2 +- YACReaderLibrary/library_window.cpp | 27 +- YACReaderLibrary/library_window.h | 6 +- YACReaderLibrary/options_dialog.cpp | 13 +- YACReaderLibrary/options_dialog.h | 2 - custom_widgets/yacreader_options_dialog.cpp | 11 + custom_widgets/yacreader_options_dialog.h | 6 +- images/accept_shortcut.png | Bin 239 -> 204 bytes images/clear_shortcut.png | Bin 201 -> 200 bytes ...s_dialog.cpp => edit_shortcuts_dialog.cpp} | 10 +- ...tcuts_dialog.h => edit_shortcuts_dialog.h} | 10 +- shortcuts_management/shortcuts_management.pri | 4 +- shortcuts_management/shortcuts_manager.cpp | 46 ++ shortcuts_management/shortcuts_manager.h | 50 ++ 22 files changed, 607 insertions(+), 277 deletions(-) rename shortcuts_management/{shortcuts_dialog.cpp => edit_shortcuts_dialog.cpp} (86%) rename shortcuts_management/{shortcuts_dialog.h => edit_shortcuts_dialog.h} (70%) diff --git a/YACReader/YACReader.pri b/YACReader/YACReader.pri index f4c3788d..4d7da37b 100644 --- a/YACReader/YACReader.pri +++ b/YACReader/YACReader.pri @@ -134,6 +134,7 @@ SOURCES += $$PWD/../common/comic.cpp \ include($$PWD/../custom_widgets/custom_widgets_yacreader.pri) include($$PWD/../compressed_archive/wrapper.pri) +include($$PWD/../shortcuts_management/shortcuts_management.pri) RESOURCES += $$PWD/yacreader_images.qrc \ $$PWD/yacreader_files.qrc diff --git a/YACReader/YACReader.pro b/YACReader/YACReader.pro index 24a05ce0..e8746938 100644 --- a/YACReader/YACReader.pro +++ b/YACReader/YACReader.pro @@ -6,7 +6,7 @@ TARGET = YACReader DEPENDPATH += . \ release -DEFINES += NOMINMAX +DEFINES += NOMINMAX YACREADER unix:!macx{ QMAKE_CXXFLAGS += -std=c++11 diff --git a/YACReader/magnifying_glass.cpp b/YACReader/magnifying_glass.cpp index 0ff09aa7..c86f202e 100644 --- a/YACReader/magnifying_glass.cpp +++ b/YACReader/magnifying_glass.cpp @@ -1,6 +1,7 @@ #include "magnifying_glass.h" #include "viewer.h" #include "configuration.h" +#include "shortcuts_manager.h" #include @@ -244,26 +245,48 @@ void MagnifyingGlass::widthDown() void MagnifyingGlass::keyPressEvent(QKeyEvent *event) { bool validKey = false; - switch (event->key()) - { - case Qt::Key_Plus: - sizeUp(); - validKey = true; - break; - case Qt::Key_Minus: - sizeDown(); - validKey = true; - break; - case Qt::Key_Underscore: - zoomOut(); - validKey = true; - break; - case Qt::Key_Asterisk: - zoomIn(); - validKey = true; - break; - } - updateImage(); + + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y)) + { + sizeUp(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y)) + { + sizeDown(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y)) + { + zoomIn(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)) + { + zoomOut(); + validKey = true; + } + if(validKey) + { + updateImage(); event->setAccepted(true); + } } diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index ee3d148e..e262d31b 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -17,6 +17,8 @@ #include "yacreader_local_client.h" #include "yacreader_global.h" +#include "edit_shortcuts_dialog.h" +#include "shortcuts_manager.h" #include #include @@ -87,19 +89,19 @@ MainWindowViewer::~MainWindowViewer() delete openNextComicAction; delete prevAction; delete nextAction; - delete adjustHeight; - delete adjustWidth; + delete adjustHeightAction; + delete adjustWidthAction; delete leftRotationAction; delete rightRotationAction; delete doublePageAction; - delete goToPage; + delete goToPageAction; delete optionsAction; delete helpAboutAction; - delete showMagnifyingGlass; - delete setBookmark; - delete showBookmarks; + delete showMagnifyingGlassAction; + delete setBookmarkAction; + delete showBookmarksAction; delete showShorcutsAction; - delete showInfo; + delete showInfoAction; delete closeAction; delete showDictionaryAction; delete alwaysOnTopAction; @@ -163,8 +165,12 @@ void MainWindowViewer::setupUI() optionsDialog->restoreOptions(settings); shortcutsDialog = new ShortcutsDialog(this); + editShortcutsDialog = new EditShortcutsDialog(this); + connect(optionsDialog,SIGNAL(editShortcuts()),editShortcutsDialog,SLOT(show())); createActions(); + setUpShortcutsManagement(); + createToolBars(); setWindowTitle("YACReader"); @@ -240,181 +246,216 @@ void MainWindowViewer::openFromArgv() void MainWindowViewer::createActions() { - openAction = new QAction(tr("&Open"),this); - openAction->setShortcut(tr("O")); - openAction->setIcon(QIcon(":/images/viewer_toolbar/open.png")); - openAction->setToolTip(tr("Open a comic")); - connect(openAction, SIGNAL(triggered()), this, SLOT(open())); + openAction = new QAction(tr("&Open"),this); + openAction->setIcon(QIcon(":/images/viewer_toolbar/open.png")); + openAction->setToolTip(tr("Open a comic")); + openAction->setData(OPEN_ACTION_Y); + openAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_ACTION_Y)); + connect(openAction, SIGNAL(triggered()), this, SLOT(open())); - openFolderAction = new QAction(tr("Open Folder"),this); - openFolderAction->setShortcut(tr("Ctrl+O")); - openFolderAction->setIcon(QIcon(":/images/viewer_toolbar/openFolder.png")); - openFolderAction->setToolTip(tr("Open image folder")); - connect(openFolderAction, SIGNAL(triggered()), this, SLOT(openFolder())); + openFolderAction = new QAction(tr("Open Folder"),this); + openFolderAction->setIcon(QIcon(":/images/viewer_toolbar/openFolder.png")); + openFolderAction->setToolTip(tr("Open image folder")); + openFolderAction->setData(OPEN_FOLDER_ACTION_Y); + openFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_FOLDER_ACTION_Y)); + connect(openFolderAction, SIGNAL(triggered()), this, SLOT(openFolder())); - saveImageAction = new QAction(tr("Save"),this); - saveImageAction->setIcon(QIcon(":/images/viewer_toolbar/save.png")); - saveImageAction->setToolTip(tr("Save current page")); - saveImageAction->setDisabled(true); - connect(saveImageAction,SIGNAL(triggered()),this,SLOT(saveImage())); + saveImageAction = new QAction(tr("Save"),this); + saveImageAction->setIcon(QIcon(":/images/viewer_toolbar/save.png")); + saveImageAction->setToolTip(tr("Save current page")); + saveImageAction->setDisabled(true); + saveImageAction->setData(SAVE_IMAGE_ACTION_Y); + saveImageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SAVE_IMAGE_ACTION_Y)); + connect(saveImageAction,SIGNAL(triggered()),this,SLOT(saveImage())); - openPreviousComicAction = new QAction(tr("Previous Comic"),this); - openPreviousComicAction->setIcon(QIcon(":/images/viewer_toolbar/openPrevious.png")); - openPreviousComicAction->setShortcut(Qt::CTRL + Qt::Key_Left); - openPreviousComicAction->setToolTip(tr("Open previous comic")); - openPreviousComicAction->setDisabled(true); - connect(openPreviousComicAction,SIGNAL(triggered()),this,SLOT(openPreviousComic())); + openPreviousComicAction = new QAction(tr("Previous Comic"),this); + openPreviousComicAction->setIcon(QIcon(":/images/viewer_toolbar/openPrevious.png")); + openPreviousComicAction->setToolTip(tr("Open previous comic")); + openPreviousComicAction->setDisabled(true); + openPreviousComicAction->setData(OPEN_PREVIOUS_COMIC_ACTION_Y); + openPreviousComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_PREVIOUS_COMIC_ACTION_Y)); + connect(openPreviousComicAction,SIGNAL(triggered()),this,SLOT(openPreviousComic())); - openNextComicAction = new QAction(tr("Next Comic"),this); - openNextComicAction->setIcon(QIcon(":/images/viewer_toolbar/openNext.png")); - openNextComicAction->setShortcut(Qt::CTRL + Qt::Key_Right); - openNextComicAction->setToolTip(tr("Open next comic")); - openNextComicAction->setDisabled(true); - connect(openNextComicAction,SIGNAL(triggered()),this,SLOT(openNextComic())); + openNextComicAction = new QAction(tr("Next Comic"),this); + openNextComicAction->setIcon(QIcon(":/images/viewer_toolbar/openNext.png")); + openNextComicAction->setToolTip(tr("Open next comic")); + openNextComicAction->setDisabled(true); + openNextComicAction->setData(OPEN_NEXT_COMIC_ACTION_Y); + openNextComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_NEXT_COMIC_ACTION_Y)); + connect(openNextComicAction,SIGNAL(triggered()),this,SLOT(openNextComic())); - prevAction = new QAction(tr("&Previous"),this); - prevAction->setIcon(QIcon(":/images/viewer_toolbar/previous.png")); - prevAction->setShortcut(Qt::Key_Left); - prevAction->setShortcutContext(Qt::WidgetShortcut); - prevAction->setToolTip(tr("Go to previous page")); - prevAction->setDisabled(true); - connect(prevAction, SIGNAL(triggered()),viewer,SLOT(prev())); + prevAction = new QAction(tr("&Previous"),this); + prevAction->setIcon(QIcon(":/images/viewer_toolbar/previous.png")); + prevAction->setShortcutContext(Qt::WidgetShortcut); + prevAction->setToolTip(tr("Go to previous page")); + prevAction->setDisabled(true); + prevAction->setData(PREV_ACTION_Y); + prevAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(PREV_ACTION_Y)); + connect(prevAction, SIGNAL(triggered()),viewer,SLOT(prev())); - nextAction = new QAction(tr("&Next"),this); - nextAction->setIcon(QIcon(":/images/viewer_toolbar/next.png")); - nextAction->setShortcut(Qt::Key_Right); - nextAction->setShortcutContext(Qt::WidgetShortcut); - nextAction->setToolTip(tr("Go to next page")); + nextAction = new QAction(tr("&Next"),this); + nextAction->setIcon(QIcon(":/images/viewer_toolbar/next.png")); + nextAction->setShortcutContext(Qt::WidgetShortcut); + nextAction->setToolTip(tr("Go to next page")); nextAction->setDisabled(true); + nextAction->setData(NEXT_ACTION_Y); + nextAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(NEXT_ACTION_Y)); connect(nextAction, SIGNAL(triggered()),viewer,SLOT(next())); - adjustHeight = new QAction(tr("Fit Height"),this); - adjustHeight->setIcon(QIcon(":/images/viewer_toolbar/toHeight.png")); + adjustHeightAction = new QAction(tr("Fit Height"),this); + adjustHeightAction->setIcon(QIcon(":/images/viewer_toolbar/toHeight.png")); //adjustWidth->setCheckable(true); - adjustHeight->setDisabled(true); - adjustHeight->setChecked(Configuration::getConfiguration().getAdjustToWidth()); - adjustHeight->setToolTip(tr("Fit image to height")); + adjustHeightAction->setDisabled(true); + adjustHeightAction->setChecked(Configuration::getConfiguration().getAdjustToWidth()); + adjustHeightAction->setToolTip(tr("Fit image to height")); //adjustWidth->setIcon(QIcon(":/images/fitWidth.png")); - connect(adjustHeight, SIGNAL(triggered()),this,SLOT(fitToHeight())); + adjustHeightAction->setData(ADJUST_HEIGHT_ACTION_Y); + adjustHeightAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_HEIGHT_ACTION_Y)); + connect(adjustHeightAction, SIGNAL(triggered()),this,SLOT(fitToHeight())); - adjustWidth = new QAction(tr("Fit Width"),this); - adjustWidth->setIcon(QIcon(":/images/viewer_toolbar/toWidth.png")); + adjustWidthAction = new QAction(tr("Fit Width"),this); + adjustWidthAction->setIcon(QIcon(":/images/viewer_toolbar/toWidth.png")); //adjustWidth->setCheckable(true); - adjustWidth->setDisabled(true); - adjustWidth->setChecked(Configuration::getConfiguration().getAdjustToWidth()); - adjustWidth->setToolTip(tr("Fit image to width")); + adjustWidthAction->setDisabled(true); + adjustWidthAction->setChecked(Configuration::getConfiguration().getAdjustToWidth()); + adjustWidthAction->setToolTip(tr("Fit image to width")); //adjustWidth->setIcon(QIcon(":/images/fitWidth.png")); - connect(adjustWidth, SIGNAL(triggered()),this,SLOT(fitToWidth())); + adjustWidthAction->setData(ADJUST_WIDTH_ACTION_Y); + adjustWidthAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_WIDTH_ACTION_Y)); + connect(adjustWidthAction, SIGNAL(triggered()),this,SLOT(fitToWidth())); leftRotationAction = new QAction(tr("Rotate image to the left"),this); - leftRotationAction->setShortcut(tr("L")); leftRotationAction->setIcon(QIcon(":/images/viewer_toolbar/rotateL.png")); leftRotationAction->setDisabled(true); + leftRotationAction->setData(LEFT_ROTATION_ACTION_Y); + leftRotationAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(LEFT_ROTATION_ACTION_Y)); connect(leftRotationAction, SIGNAL(triggered()),viewer,SLOT(rotateLeft())); rightRotationAction = new QAction(tr("Rotate image to the right"),this); - rightRotationAction->setShortcut(tr("R")); rightRotationAction->setIcon(QIcon(":/images/viewer_toolbar/rotateR.png")); rightRotationAction->setDisabled(true); + rightRotationAction->setData(RIGHT_ROTATION_ACTION_Y); + rightRotationAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RIGHT_ROTATION_ACTION_Y)); connect(rightRotationAction, SIGNAL(triggered()),viewer,SLOT(rotateRight())); doublePageAction = new QAction(tr("Double page mode"),this); doublePageAction->setToolTip(tr("Switch to double page mode")); - doublePageAction->setShortcut(tr("D")); doublePageAction->setIcon(QIcon(":/images/viewer_toolbar/doublePage.png")); doublePageAction->setDisabled(true); doublePageAction->setCheckable(true); doublePageAction->setChecked(Configuration::getConfiguration().getDoublePage()); + doublePageAction->setData(DOUBLE_PAGE_ACTION_Y); + doublePageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(DOUBLE_PAGE_ACTION_Y)); connect(doublePageAction, SIGNAL(triggered()),viewer,SLOT(doublePageSwitch())); - goToPage = new QAction(tr("Go To"),this); - goToPage->setShortcut(tr("G")); - goToPage->setIcon(QIcon(":/images/viewer_toolbar/goto.png")); - goToPage->setDisabled(true); - goToPage->setToolTip(tr("Go to page ...")); - connect(goToPage, SIGNAL(triggered()),viewer,SLOT(showGoToDialog())); + goToPageAction = new QAction(tr("Go To"),this); + goToPageAction->setIcon(QIcon(":/images/viewer_toolbar/goto.png")); + goToPageAction->setDisabled(true); + goToPageAction->setToolTip(tr("Go to page ...")); + goToPageAction->setData(GO_TO_PAGE_ACTION_Y); + goToPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_PAGE_ACTION_Y)); + connect(goToPageAction, SIGNAL(triggered()),viewer,SLOT(showGoToDialog())); optionsAction = new QAction(tr("Options"),this); - optionsAction->setShortcut(tr("C")); optionsAction->setToolTip(tr("YACReader options")); + optionsAction->setData(OPTIONS_ACTION_Y); + optionsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPTIONS_ACTION_Y)); optionsAction->setIcon(QIcon(":/images/viewer_toolbar/options.png")); connect(optionsAction, SIGNAL(triggered()),optionsDialog,SLOT(show())); helpAboutAction = new QAction(tr("Help"),this); helpAboutAction->setToolTip(tr("Help, About YACReader")); - helpAboutAction->setShortcut(Qt::Key_F1); helpAboutAction->setIcon(QIcon(":/images/viewer_toolbar/help.png")); + helpAboutAction->setData(HELP_ABOUT_ACTION_Y); + helpAboutAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HELP_ABOUT_ACTION_Y)); connect(helpAboutAction, SIGNAL(triggered()),had,SLOT(show())); - showMagnifyingGlass = new QAction(tr("Magnifying glass"),this); - showMagnifyingGlass->setToolTip(tr("Switch Magnifying glass")); - showMagnifyingGlass->setShortcut(tr("Z")); - showMagnifyingGlass->setIcon(QIcon(":/images/viewer_toolbar/magnifyingGlass.png")); - showMagnifyingGlass->setDisabled(true); - showMagnifyingGlass->setCheckable(true); - connect(showMagnifyingGlass, SIGNAL(triggered()),viewer,SLOT(magnifyingGlassSwitch())); + showMagnifyingGlassAction = new QAction(tr("Magnifying glass"),this); + showMagnifyingGlassAction->setToolTip(tr("Switch Magnifying glass")); + showMagnifyingGlassAction->setIcon(QIcon(":/images/viewer_toolbar/magnifyingGlass.png")); + showMagnifyingGlassAction->setDisabled(true); + showMagnifyingGlassAction->setCheckable(true); + showMagnifyingGlassAction->setData(SHOW_MAGNIFYING_GLASS_ACTION_Y); + showMagnifyingGlassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_MAGNIFYING_GLASS_ACTION_Y)); + connect(showMagnifyingGlassAction, SIGNAL(triggered()),viewer,SLOT(magnifyingGlassSwitch())); - setBookmark = new QAction(tr("Set bookmark"),this); - setBookmark->setToolTip(tr("Set a bookmark on the current page")); - setBookmark->setShortcut(Qt::CTRL+Qt::Key_M); - setBookmark->setIcon(QIcon(":/images/viewer_toolbar/bookmark.png")); - setBookmark->setDisabled(true); - setBookmark->setCheckable(true); - connect(setBookmark,SIGNAL(triggered (bool)),viewer,SLOT(setBookmark(bool))); - connect(viewer,SIGNAL(pageAvailable(bool)),setBookmark,SLOT(setEnabled(bool))); - connect(viewer,SIGNAL(pageIsBookmark(bool)),setBookmark,SLOT(setChecked(bool))); + setBookmarkAction = new QAction(tr("Set bookmark"),this); + setBookmarkAction->setToolTip(tr("Set a bookmark on the current page")); + setBookmarkAction->setIcon(QIcon(":/images/viewer_toolbar/bookmark.png")); + setBookmarkAction->setDisabled(true); + setBookmarkAction->setCheckable(true); + setBookmarkAction->setData(SET_BOOKMARK_ACTION_Y); + setBookmarkAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_BOOKMARK_ACTION_Y)); + connect(setBookmarkAction,SIGNAL(triggered (bool)),viewer,SLOT(setBookmarkAction(bool))); + connect(viewer,SIGNAL(pageAvailable(bool)),setBookmarkAction,SLOT(setEnabled(bool))); + connect(viewer,SIGNAL(pageIsBookmark(bool)),setBookmarkAction,SLOT(setChecked(bool))); - showBookmarks = new QAction(tr("Show bookmarks"),this); - showBookmarks->setToolTip(tr("Show the bookmarks of the current comic")); - showBookmarks->setShortcut(tr("M")); - showBookmarks->setIcon(QIcon(":/images/viewer_toolbar/showBookmarks.png")); - showBookmarks->setDisabled(true); - connect(showBookmarks, SIGNAL(triggered()),viewer->getBookmarksDialog(),SLOT(show())); + showBookmarksAction = new QAction(tr("Show bookmarks"),this); + showBookmarksAction->setToolTip(tr("Show the bookmarks of the current comic")); + showBookmarksAction->setIcon(QIcon(":/images/viewer_toolbar/showBookmarks.png")); + showBookmarksAction->setDisabled(true); + showBookmarksAction->setData(SHOW_BOOKMARKS_ACTION_Y); + showBookmarksAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_BOOKMARKS_ACTION_Y)); + connect(showBookmarksAction, SIGNAL(triggered()),viewer->getBookmarksDialog(),SLOT(show())); showShorcutsAction = new QAction(tr("Show keyboard shortcuts"), this ); showShorcutsAction->setIcon(QIcon(":/images/viewer_toolbar/shortcuts.png")); + showShorcutsAction->setData(SHOW_SHORCUTS_ACTION_Y); + showShorcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_SHORCUTS_ACTION_Y)); connect(showShorcutsAction, SIGNAL(triggered()),shortcutsDialog,SLOT(show())); - showInfo = new QAction(tr("Show Info"),this); - showInfo->setShortcut(tr("I")); - showInfo->setIcon(QIcon(":/images/viewer_toolbar/info.png")); - showInfo->setDisabled(true); - connect(showInfo, SIGNAL(triggered()),viewer,SLOT(informationSwitch())); + showInfoAction = new QAction(tr("Show Info"),this); + showInfoAction->setIcon(QIcon(":/images/viewer_toolbar/info.png")); + showInfoAction->setDisabled(true); + showInfoAction->setData(SHOW_INFO_ACTION_Y); + showInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_INFO_ACTION_Y)); + connect(showInfoAction, SIGNAL(triggered()),viewer,SLOT(informationSwitch())); closeAction = new QAction(tr("Close"),this); - closeAction->setShortcut(Qt::Key_Escape); closeAction->setIcon(QIcon(":/images/viewer_toolbar/close.png")); + closeAction->setData(CLOSE_ACTION_Y); + closeAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CLOSE_ACTION_Y)); connect(closeAction,SIGNAL(triggered()),this,SLOT(close())); showDictionaryAction = new QAction(tr("Show Dictionary"),this); - showDictionaryAction->setShortcut(Qt::Key_T); showDictionaryAction->setIcon(QIcon(":/images/viewer_toolbar/translator.png")); //showDictionaryAction->setCheckable(true); showDictionaryAction->setDisabled(true); + showDictionaryAction->setData(SHOW_DICTIONARY_ACTION_Y); + showDictionaryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_DICTIONARY_ACTION_Y)); connect(showDictionaryAction,SIGNAL(triggered()),viewer,SLOT(translatorSwitch())); + //deprecated alwaysOnTopAction = new QAction(tr("Always on top"),this); - alwaysOnTopAction->setShortcut(Qt::Key_Q); alwaysOnTopAction->setIcon(QIcon(":/images/alwaysOnTop.png")); alwaysOnTopAction->setCheckable(true); alwaysOnTopAction->setDisabled(true); alwaysOnTopAction->setChecked(Configuration::getConfiguration().getAlwaysOnTop()); + alwaysOnTopAction->setData(ALWAYS_ON_TOP_ACTION_Y); + alwaysOnTopAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ALWAYS_ON_TOP_ACTION_Y)); connect(alwaysOnTopAction,SIGNAL(triggered()),this,SLOT(alwaysOnTopSwitch())); adjustToFullSizeAction = new QAction(tr("Show full size"),this); - adjustToFullSizeAction->setShortcut(Qt::Key_W); adjustToFullSizeAction->setIcon(QIcon(":/images/viewer_toolbar/full.png")); adjustToFullSizeAction->setCheckable(true); adjustToFullSizeAction->setDisabled(true); adjustToFullSizeAction->setChecked(Configuration::getConfiguration().getAdjustToFullSize()); + adjustToFullSizeAction->setData(ADJUST_TO_FULL_SIZE_ACTION_Y); + adjustToFullSizeAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_TO_FULL_SIZE_ACTION_Y)); connect(adjustToFullSizeAction,SIGNAL(triggered()),this,SLOT(adjustToFullSizeSwitch())); showFlowAction = new QAction(tr("Show go to flow"),this); - showFlowAction->setShortcut(Qt::Key_S); showFlowAction->setIcon(QIcon(":/images/viewer_toolbar/flow.png")); showFlowAction->setDisabled(true); + showFlowAction->setData(SHOW_FLOW_ACTION_Y); + showFlowAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_FLOW_ACTION_Y)); connect(showFlowAction,SIGNAL(triggered()),viewer,SLOT(goToFlowSwitch())); + + showEditShortcutsAction = new QAction(tr("Edit shortcuts"),this); + showEditShortcutsAction->setData(SHOW_EDIT_SHORTCUTS_ACTION_Y); + showEditShortcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_EDIT_SHORTCUTS_ACTION_Y)); + connect(showEditShortcutsAction,SIGNAL(triggered()),editShortcutsDialog,SLOT(show())); } void MainWindowViewer::createToolBars() @@ -445,7 +486,7 @@ void MainWindowViewer::createToolBars() #endif comicToolBar->addAction(prevAction); comicToolBar->addAction(nextAction); - comicToolBar->addAction(goToPage); + comicToolBar->addAction(goToPageAction); //#ifndef Q_OS_MAC // comicToolBar->addSeparator(); @@ -486,7 +527,7 @@ void MainWindowViewer::createToolBars() ); menu->addAction(sliderAction); QToolButton * tb2 = new QToolButton(); - tb2->addAction(adjustWidth); + tb2->addAction(adjustWidthAction); tb2->setMenu(menu); connect(sliderAction,SIGNAL(fitToWidthRatioChanged(float)),viewer,SLOT(updateFitToWidthRatio(float))); @@ -495,9 +536,9 @@ void MainWindowViewer::createToolBars() //tb2->addAction(); tb2->setPopupMode(QToolButton::MenuButtonPopup); - tb2->setDefaultAction(adjustWidth); + tb2->setDefaultAction(adjustWidthAction); comicToolBar->addWidget(tb2); - comicToolBar->addAction(adjustHeight); + comicToolBar->addAction(adjustHeightAction); comicToolBar->addAction(adjustToFullSizeAction); comicToolBar->addAction(leftRotationAction); comicToolBar->addAction(rightRotationAction); @@ -508,15 +549,15 @@ void MainWindowViewer::createToolBars() #else comicToolBar->addSeparator(); #endif - comicToolBar->addAction(showMagnifyingGlass); + comicToolBar->addAction(showMagnifyingGlassAction); #ifdef Q_OS_MAC comicToolBar->addWidget(new MacToolBarSeparator); #else comicToolBar->addSeparator(); #endif - comicToolBar->addAction(setBookmark); - comicToolBar->addAction(showBookmarks); + comicToolBar->addAction(setBookmarkAction); + comicToolBar->addAction(showBookmarksAction); #ifdef Q_OS_MAC comicToolBar->addWidget(new MacToolBarSeparator); @@ -525,7 +566,7 @@ void MainWindowViewer::createToolBars() #endif comicToolBar->addAction(showDictionaryAction); comicToolBar->addAction(showFlowAction); - comicToolBar->addAction(showInfo); + comicToolBar->addAction(showInfoAction); #ifdef Q_OS_MAC comicToolBar->addWidget(new MacToolBarSeparator); @@ -551,27 +592,28 @@ void MainWindowViewer::createToolBars() viewer->addAction(prevAction); viewer->addAction(nextAction); - viewer->addAction(goToPage); - viewer->addAction(adjustHeight); - viewer->addAction(adjustWidth); + viewer->addAction(goToPageAction); + viewer->addAction(adjustHeightAction); + viewer->addAction(adjustWidthAction); viewer->addAction(adjustToFullSizeAction); viewer->addAction(leftRotationAction); viewer->addAction(rightRotationAction); YACReader::addSperator(viewer); - viewer->addAction(showMagnifyingGlass); + viewer->addAction(showMagnifyingGlassAction); YACReader::addSperator(viewer); - viewer->addAction(setBookmark); - viewer->addAction(showBookmarks); + viewer->addAction(setBookmarkAction); + viewer->addAction(showBookmarksAction); YACReader::addSperator(viewer); viewer->addAction(showDictionaryAction); viewer->addAction(showFlowAction); - viewer->addAction(showInfo); + viewer->addAction(showInfoAction); YACReader::addSperator(viewer); viewer->addAction(showShorcutsAction); + viewer->addAction(showEditShortcutsAction); viewer->addAction(optionsAction); viewer->addAction(helpAboutAction); YACReader::addSperator(viewer); @@ -740,18 +782,18 @@ void MainWindowViewer::enableActions() saveImageAction->setDisabled(false); prevAction->setDisabled(false); nextAction->setDisabled(false); - adjustHeight->setDisabled(false); - adjustWidth->setDisabled(false); - goToPage->setDisabled(false); + adjustHeightAction->setDisabled(false); + adjustWidthAction->setDisabled(false); + goToPageAction->setDisabled(false); //alwaysOnTopAction->setDisabled(false); leftRotationAction->setDisabled(false); rightRotationAction->setDisabled(false); - showMagnifyingGlass->setDisabled(false); + showMagnifyingGlassAction->setDisabled(false); doublePageAction->setDisabled(false); adjustToFullSizeAction->setDisabled(false); //setBookmark->setDisabled(false); - showBookmarks->setDisabled(false); - showInfo->setDisabled(false); //TODO enable goTo and showInfo (or update) when numPages emited + showBookmarksAction->setDisabled(false); + showInfoAction->setDisabled(false); //TODO enable goTo and showInfo (or update) when numPages emited showDictionaryAction->setDisabled(false); showFlowAction->setDisabled(false); } @@ -760,18 +802,18 @@ void MainWindowViewer::disableActions() saveImageAction->setDisabled(true); prevAction->setDisabled(true); nextAction->setDisabled(true); - adjustHeight->setDisabled(true); - adjustWidth->setDisabled(true); - goToPage->setDisabled(true); + adjustHeightAction->setDisabled(true); + adjustWidthAction->setDisabled(true); + goToPageAction->setDisabled(true); //alwaysOnTopAction->setDisabled(true); leftRotationAction->setDisabled(true); rightRotationAction->setDisabled(true); - showMagnifyingGlass->setDisabled(true); + showMagnifyingGlassAction->setDisabled(true); doublePageAction->setDisabled(true); adjustToFullSizeAction->setDisabled(true); - setBookmark->setDisabled(true); - showBookmarks->setDisabled(true); - showInfo->setDisabled(true); //TODO enable goTo and showInfo (or update) when numPages emited + setBookmarkAction->setDisabled(true); + showBookmarksAction->setDisabled(true); + showInfoAction->setDisabled(true); //TODO enable goTo and showInfo (or update) when numPages emited openPreviousComicAction->setDisabled(true); openNextComicAction->setDisabled(true); showDictionaryAction->setDisabled(true); @@ -780,28 +822,38 @@ void MainWindowViewer::disableActions() void MainWindowViewer::keyPressEvent(QKeyEvent *event) { - //TODO remove unused keys - switch (event->key()) - { - case Qt::Key_Escape: - this->close(); - break; - case Qt::Key_F: - toggleFullScreen(); - break; - case Qt::Key_H: - toggleToolBars(); - break; - case Qt::Key_O: - open(); - break; - case Qt::Key_A: - changeFit(); - break; - default: - QWidget::keyPressEvent(event); - break; - } + //TODO remove unused keys + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_Y)) + { + toggleFullScreen(); + event->accept(); + } + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)) + { + toggleToolBars(); + event->accept(); + } + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)) + { + changeFit(); + event->accept(); + } + else + QWidget::keyPressEvent(event); } void MainWindowViewer::mouseDoubleClickEvent ( QMouseEvent * event ) @@ -900,7 +952,7 @@ void MainWindowViewer::checkNewVersion() QTimer * tT = new QTimer; tT->setSingleShot(true); connect(tT, SIGNAL(timeout()), versionChecker, SLOT(get())); - //versionChecker->get(); //TODÓ + //versionChecker->get(); //TOD� tT->start(100); conf.setLastVersionCheck(current); @@ -926,6 +978,131 @@ void MainWindowViewer::processReset() disableActions(); } +void MainWindowViewer::setUpShortcutsManagement() +{ + //actions holder + QObject * orphanActions = new QObject; + + editShortcutsDialog->addActionsGroup(tr("Comics"),QIcon(":/images/openInYACReader.png"), + QList() + << openAction + << openFolderAction + << saveImageAction + << openPreviousComicAction + << openNextComicAction); + + //keys without actions (General) + QAction * toggleFullScreenAction = new QAction(tr("Toggle fullscreen mode"),orphanActions); + toggleFullScreenAction->setData(TOGGLE_FULL_SCREEN_ACTION_Y); + toggleFullScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_Y)); + + QAction * toggleToolbarsAction = new QAction(tr("Hide/show toolbar"),orphanActions); + toggleToolbarsAction->setData(TOGGLE_TOOL_BARS_ACTION_Y); + toggleToolbarsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("General"),QIcon(), + QList() + << optionsAction + << helpAboutAction + << showShorcutsAction + << showInfoAction + << closeAction + << showDictionaryAction + << showFlowAction + << toggleFullScreenAction + << toggleToolbarsAction + << showEditShortcutsAction); + + //keys without actions (MGlass) + QAction * sizeUpMglassAction = new QAction(tr("Size up magnifying glass"),orphanActions); + sizeUpMglassAction->setData(SIZE_UP_MGLASS_ACTION_Y); + sizeUpMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y)); + + QAction * sizeDownMglassAction = new QAction(tr("Size down magnifying glass"),orphanActions); + sizeDownMglassAction->setData(SIZE_DOWN_MGLASS_ACTION_Y); + sizeDownMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y)); + + QAction * zoomInMglassAction = new QAction(tr("Zoom in magnifying glass"),orphanActions); + zoomInMglassAction->setData(ZOOM_IN_MGLASS_ACTION_Y); + zoomInMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y)); + + QAction * zoomOutMglassAction = new QAction(tr("Zoom out magnifying glass"),orphanActions); + zoomOutMglassAction->setData(ZOOM_OUT_MGLASS_ACTION_Y); + zoomOutMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Magnifiying glass"),QIcon(), + QList() + << showMagnifyingGlassAction + << sizeUpMglassAction + << sizeDownMglassAction + << zoomInMglassAction + << zoomOutMglassAction); + + //keys without actions + QAction * toggleFitToScreenAction = new QAction(tr("Toggle between fit to width and fit to height"),orphanActions); + toggleFitToScreenAction->setData(CHANGE_FIT_ACTION_Y); + toggleFitToScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Page adjustement"),QIcon(), + QList() + << adjustHeightAction + << adjustWidthAction + << toggleFitToScreenAction + << leftRotationAction + << rightRotationAction + << doublePageAction + << adjustToFullSizeAction); + + QAction * autoScrollForwardAction = new QAction(tr("Autoscroll down"),orphanActions); + autoScrollForwardAction->setData(AUTO_SCROLL_FORWARD_ACTION_Y); + autoScrollForwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_FORWARD_ACTION_Y)); + + QAction * autoScrollBackwardAction = new QAction(tr("Autoscroll up"),orphanActions); + autoScrollBackwardAction->setData(AUTO_SCROLL_BACKWARD_ACTION_Y); + autoScrollBackwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_BACKWARD_ACTION_Y)); + + QAction * moveDownAction = new QAction(tr("Move down"),orphanActions); + moveDownAction->setData(MOVE_DOWN_ACTION_Y); + moveDownAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_DOWN_ACTION_Y)); + + QAction * moveUpAction = new QAction(tr("Move up"),orphanActions); + moveUpAction->setData(MOVE_UP_ACTION_Y); + moveUpAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_UP_ACTION_Y)); + + QAction * moveLeftAction = new QAction(tr("Move left"),orphanActions); + moveLeftAction->setData(MOVE_LEFT_ACTION_Y); + moveLeftAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_LEFT_ACTION_Y)); + + QAction * moveRightAction = new QAction(tr("Move right"),orphanActions); + moveRightAction->setData(MOVE_RIGHT_ACTION_Y); + moveRightAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_RIGHT_ACTION_Y)); + + QAction * goToFirstPageAction = new QAction(tr("Go to the first page"),orphanActions); + goToFirstPageAction->setData(GO_TO_FIRST_PAGE_ACTION_Y); + goToFirstPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_FIRST_PAGE_ACTION_Y)); + + QAction * goToLastPageAction = new QAction(tr("Go to the last page"),orphanActions); + goToLastPageAction->setData(GO_TO_LAST_PAGE_ACTION_Y); + goToLastPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Reading"),QIcon(), + QList() + << nextAction + << prevAction + << setBookmarkAction + << showBookmarksAction + << autoScrollForwardAction + << autoScrollBackwardAction + << moveDownAction + << moveUpAction + << moveLeftAction + << moveRightAction + << goToFirstPageAction + << goToLastPageAction + << goToPageAction); + +} + void MainWindowViewer::changeFit() { Configuration & conf = Configuration::getConfiguration(); diff --git a/YACReader/main_window_viewer.h b/YACReader/main_window_viewer.h index 6b33c434..fedd1ca4 100644 --- a/YACReader/main_window_viewer.h +++ b/YACReader/main_window_viewer.h @@ -18,6 +18,7 @@ class HelpAboutDialog; class HttpVersionChecker; class ShortcutsDialog; class YACReaderSliderAction; +class EditShortcutsDialog; class MainWindowViewer : public QMainWindow { @@ -51,6 +52,7 @@ class YACReaderSliderAction; void fitToHeight(); void checkNewVersion(); void processReset(); + void setUpShortcutsManagement(); /*void viewComic(); void prev(); void next(); @@ -71,6 +73,7 @@ class YACReaderSliderAction; OptionsDialog * optionsDialog; HelpAboutDialog * had; ShortcutsDialog * shortcutsDialog; + EditShortcutsDialog * editShortcutsDialog; //! ToolBars QToolBar *comicToolBar; @@ -83,17 +86,17 @@ class YACReaderSliderAction; QAction *openNextComicAction; QAction *nextAction; QAction *prevAction; - QAction *adjustWidth; - QAction *adjustHeight; - QAction *goToPage; + QAction *adjustWidthAction; + QAction *adjustHeightAction; + QAction *goToPageAction; QAction *optionsAction; QAction *helpAboutAction; - QAction *showMagnifyingGlass; - QAction *setBookmark; - QAction *showBookmarks; + QAction *showMagnifyingGlassAction; + QAction *setBookmarkAction; + QAction *showBookmarksAction; QAction *leftRotationAction; QAction *rightRotationAction; - QAction *showInfo; + QAction *showInfoAction; QAction *closeAction; QAction *doublePageAction; QAction *showShorcutsAction; @@ -102,6 +105,8 @@ class YACReaderSliderAction; QAction *adjustToFullSizeAction; QAction *showFlowAction; + QAction *showEditShortcutsAction; + YACReaderSliderAction * sliderAction; HttpVersionChecker * versionChecker; @@ -115,8 +120,8 @@ class YACReaderSliderAction; void getSiblingComics(QString path,QString currentComic); //! Manejadores de evento: - void keyPressEvent(QKeyEvent *event); - //void resizeEvent(QResizeEvent * event); + void keyPressEvent(QKeyEvent *event); + //void resizeEvent(QResizeEvent * event); void mouseDoubleClickEvent ( QMouseEvent * event ); void dropEvent(QDropEvent *event); void dragEnterEvent(QDragEnterEvent *event); diff --git a/YACReader/options_dialog.cpp b/YACReader/options_dialog.cpp index 83a2de20..37d25792 100644 --- a/YACReader/options_dialog.cpp +++ b/YACReader/options_dialog.cpp @@ -113,6 +113,7 @@ OptionsDialog::OptionsDialog(QWidget * parent) layoutGeneral->addWidget(slideSizeBox); layoutGeneral->addWidget(fitBox); layoutGeneral->addWidget(colorBox); + layoutGeneral->addWidget(shortcutsBox); layoutGeneral->addStretch(); layoutFlow->addWidget(sw); layoutFlow->addWidget(gl); diff --git a/YACReader/viewer.cpp b/YACReader/viewer.cpp index d90c0f95..10812be8 100644 --- a/YACReader/viewer.cpp +++ b/YACReader/viewer.cpp @@ -12,6 +12,8 @@ #include "page_label_widget.h" #include "notifications_label_widget.h" #include "comic_db.h" +#include "shortcuts_manager.h" + #include @@ -367,62 +369,72 @@ void Viewer::scrollUp() void Viewer::keyPressEvent(QKeyEvent *event) { - if(render->hasLoadedComic()) - { - if(goToFlow->isVisible() && event->key()!=Qt::Key_S) - QCoreApplication::sendEvent(goToFlow,event); - else - switch (event->key()) - { - case Qt::Key_Space: - posByStep = height()/numScrollSteps; - nextPos=verticalScrollBar()->sliderPosition()+static_cast((height()*0.80)); - scrollDown(); - break; - case Qt::Key_B: - posByStep = height()/numScrollSteps; - nextPos=verticalScrollBar()->sliderPosition()-static_cast((height()*0.80)); - scrollUp(); - break; - case Qt::Key_S: - goToFlowSwitch(); - break; - case Qt::Key_T: - translatorSwitch(); - break; - case Qt::Key_Down: - /*if(verticalScrollBar()->sliderPosition()==verticalScrollBar()->maximum()) - next(); - else*/ - QAbstractScrollArea::keyPressEvent(event); - emit backgroundChanges(); - break; - case Qt::Key_Up: - /*if(verticalScrollBar()->sliderPosition()==verticalScrollBar()->minimum()) - prev(); - else*/ - QAbstractScrollArea::keyPressEvent(event); - emit backgroundChanges(); - break; - case Qt::Key_Home: - goTo(0); - break; - case Qt::Key_End: - goTo(this->render->numPages()-1); - break; - default: - QAbstractScrollArea::keyPressEvent(event); - break; - } - if(mglass->isVisible()) - switch(event->key()) - { - case Qt::Key_Plus: case Qt::Key_Minus: case Qt::Key_Underscore: case Qt::Key_Asterisk: - QCoreApplication::sendEvent(mglass,event); - } - } - else - QAbstractScrollArea::keyPressEvent(event); + if(render->hasLoadedComic()) + { + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + /*if(goToFlow->isVisible() && event->key()!=Qt::Key_S) + QCoreApplication::sendEvent(goToFlow,event); + else*/ + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_FORWARD_ACTION_Y)) + { + posByStep = height()/numScrollSteps; + nextPos=verticalScrollBar()->sliderPosition()+static_cast((height()*0.80)); + scrollDown(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_BACKWARD_ACTION_Y)) + { + posByStep = height()/numScrollSteps; + nextPos=verticalScrollBar()->sliderPosition()-static_cast((height()*0.80)); + scrollUp(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_DOWN_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_UP_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_LEFT_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_RIGHT_ACTION_Y)) + { + QAbstractScrollArea::keyPressEvent(event); + emit backgroundChanges(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_FIRST_PAGE_ACTION_Y)) + { + goTo(0); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)) + { + goTo(this->render->numPages()-1); + } + + else + QAbstractScrollArea::keyPressEvent(event); + + if(mglass->isVisible() && (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y))) + { + QCoreApplication::sendEvent(mglass,event); + } + + } + else + QAbstractScrollArea::keyPressEvent(event); } void Viewer::wheelEvent(QWheelEvent * event) diff --git a/YACReader/yacreader_images.qrc b/YACReader/yacreader_images.qrc index 3defdcfa..def5653a 100644 --- a/YACReader/yacreader_images.qrc +++ b/YACReader/yacreader_images.qrc @@ -73,5 +73,7 @@ ../images/dropDownArrow.png ../images/translatorSearch.png ../images/speaker.png + ../images/clear_shortcut.png + ../images/accept_shortcut.png diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index 88275787..7a396a70 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -13,7 +13,7 @@ INCLUDEPATH += ../common \ ./comic_vine \ ./comic_vine/model -DEFINES += SERVER_RELEASE NOMINMAX +DEFINES += SERVER_RELEASE NOMINMAX YACREADER_LIBRARY win32 { diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 1aa167fd..686afd9e 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -65,7 +65,7 @@ #include "comics_view_transition.h" #include "empty_folder_widget.h" -#include "shortcuts_dialog.h" +#include "edit_shortcuts_dialog.h" #include "shortcuts_manager.h" #include "QsLog.h" @@ -273,7 +273,7 @@ void LibraryWindow::doDialogs() optionsDialog = new OptionsDialog(this); optionsDialog->restoreOptions(settings); - shortcutsDialog = new ShortcutsDialog(this); + editShortcutsDialog = new EditShortcutsDialog(this); setUpShortcutsManagement(); #ifdef SERVER_RELEASE @@ -297,7 +297,7 @@ void LibraryWindow::doDialogs() void LibraryWindow::setUpShortcutsManagement() { - shortcutsDialog->addActionsGroup("Comics",QIcon(":/images/openInYACReader.png"), + editShortcutsDialog->addActionsGroup("Comics",QIcon(":/images/openInYACReader.png"), QList() << openComicAction << setAsReadAction @@ -310,7 +310,7 @@ void LibraryWindow::setUpShortcutsManagement() << deleteComicsAction << getInfoAction); - shortcutsDialog->addActionsGroup("Folders",QIcon(), + editShortcutsDialog->addActionsGroup("Folders",QIcon(), QList() << setRootIndexAction << expandAllNodesAction @@ -321,15 +321,16 @@ void LibraryWindow::setUpShortcutsManagement() << setFolderAsReadAction << setFolderAsUnreadAction); - shortcutsDialog->addActionsGroup("General",QIcon(), + editShortcutsDialog->addActionsGroup("General",QIcon(), QList() << backAction << forwardAction << helpAboutAction << optionsAction - << serverConfigAction); + << serverConfigAction + << showEditShortcutsAction); - shortcutsDialog->addActionsGroup("Libraries",QIcon(), + editShortcutsDialog->addActionsGroup("Libraries",QIcon(), QList() << createLibraryAction << openLibraryAction @@ -341,7 +342,7 @@ void LibraryWindow::setUpShortcutsManagement() << renameLibraryAction << removeLibraryAction); - shortcutsDialog->addActionsGroup("Visualization",QIcon(), + editShortcutsDialog->addActionsGroup("Visualization",QIcon(), QList() << showHideMarksAction << toggleFullScreenAction @@ -635,6 +636,11 @@ void LibraryWindow::createActions() getInfoAction->setIcon(QIcon(":/images/getInfo.png")); //------------------------------------------------------------------------- + showEditShortcutsAction = new QAction(tr("Edit shortcuts"),this); + showEditShortcutsAction->setData(SHOW_EDIT_SHORTCUTS_ACTION_YL); + showEditShortcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_EDIT_SHORTCUTS_ACTION_YL)); + showEditShortcutsAction->setShortcutContext(Qt::ApplicationShortcut); + addAction(showEditShortcutsAction); //disable actions disableAllActions(); } @@ -772,7 +778,6 @@ void LibraryWindow::createToolBars() editInfoToolBar->addWidget(new QToolBarStretch()); editInfoToolBar->addAction(hideComicViewAction); - } void LibraryWindow::createMenus() @@ -973,7 +978,7 @@ void LibraryWindow::createConnections() connect(serverConfigAction, SIGNAL(triggered()), serverConfigDialog, SLOT(show())); #endif connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); - connect(optionsDialog, SIGNAL(editShortcuts()),shortcutsDialog,SLOT(show())); + connect(optionsDialog, SIGNAL(editShortcuts()),editShortcutsDialog,SLOT(show())); //Folders filter //connect(clearFoldersFilter,SIGNAL(clicked()),foldersFilter,SLOT(clear())); @@ -1007,6 +1012,8 @@ void LibraryWindow::createConnections() connect(dmCV,SIGNAL(isEmpty()),this,SLOT(showEmptyFolderView())); connect(emptyFolderWidget,SIGNAL(subfolderSelected(QModelIndex,int)),this,SLOT(selectSubfolder(QModelIndex,int))); + + connect(showEditShortcutsAction,SIGNAL(triggered()),editShortcutsDialog,SLOT(show())); } void LibraryWindow::loadLibrary(const QString & name) diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h index d44ec840..b7761820 100644 --- a/YACReaderLibrary/library_window.h +++ b/YACReaderLibrary/library_window.h @@ -54,7 +54,7 @@ class ClassicComicsView; class GridComicsView; class ComicsViewTransition; class EmptyFolderWidget; -class ShortcutsDialog; +class EditShortcutsDialog; #include "comic_db.h" @@ -77,7 +77,7 @@ private: RenameLibraryDialog * renameLibraryDialog; PropertiesDialog * propertiesDialog; ComicVineDialog * comicVineDialog; - ShortcutsDialog * shortcutsDialog; + EditShortcutsDialog * editShortcutsDialog; //YACReaderSocialDialog * socialDialog; bool fullscreen; bool importedCovers; //if true, the library is read only (not updates,open comic or properties) @@ -173,6 +173,8 @@ private: QAction * deleteComicsAction; QAction * hideComicViewAction; + QAction *showEditShortcutsAction; + QList itemActions; QList viewActions; diff --git a/YACReaderLibrary/options_dialog.cpp b/YACReaderLibrary/options_dialog.cpp index 2ab02678..98794977 100644 --- a/YACReaderLibrary/options_dialog.cpp +++ b/YACReaderLibrary/options_dialog.cpp @@ -40,23 +40,15 @@ OptionsDialog::OptionsDialog(QWidget * parent) flowLayout->addWidget(gl); flowLayout->addLayout(switchFlowType); - QVBoxLayout * shortcutsLayout = new QVBoxLayout(); - QPushButton * shortcutsButton = new QPushButton(tr("Edit shortcuts")); - shortcutsLayout->addWidget(shortcutsButton); - - sw->hide(); QWidget * comicFlowW = new QWidget; comicFlowW->setLayout(flowLayout); - QGroupBox *generalBox = new QGroupBox(tr("Shortcuts")); - generalBox->setLayout(shortcutsLayout); - generalLayout->addWidget(generalBox); - generalLayout->addStretch(); - QWidget * generalW = new QWidget; generalW->setLayout(generalLayout); + generalLayout->addWidget(shortcutsBox); + generalLayout->addStretch(); tabWidget->addTab(comicFlowW,tr("Comic Flow")); tabWidget->addTab(generalW,tr("General")); @@ -71,7 +63,6 @@ OptionsDialog::OptionsDialog(QWidget * parent) this->layout()->setSizeConstraint(QLayout::SetFixedSize); - connect(shortcutsButton,SIGNAL(clicked()),this,SIGNAL(editShortcuts())); } diff --git a/YACReaderLibrary/options_dialog.h b/YACReaderLibrary/options_dialog.h index d597f686..d42a9ead 100644 --- a/YACReaderLibrary/options_dialog.h +++ b/YACReaderLibrary/options_dialog.h @@ -12,8 +12,6 @@ class OptionsDialog : public YACReaderOptionsDialog Q_OBJECT public: OptionsDialog(QWidget * parent = 0); -signals: - void editShortcuts(); }; diff --git a/custom_widgets/yacreader_options_dialog.cpp b/custom_widgets/yacreader_options_dialog.cpp index f9fcae64..c89b44dc 100644 --- a/custom_widgets/yacreader_options_dialog.cpp +++ b/custom_widgets/yacreader_options_dialog.cpp @@ -10,6 +10,7 @@ #include #include #include +#include YACReaderOptionsDialog::YACReaderOptionsDialog(QWidget * parent) :QDialog(parent) @@ -23,6 +24,16 @@ YACReaderOptionsDialog::YACReaderOptionsDialog(QWidget * parent) cancel->setDefault(true); + + QVBoxLayout * shortcutsLayout = new QVBoxLayout(); + QPushButton * shortcutsButton = new QPushButton(tr("Edit shortcuts")); + shortcutsLayout->addWidget(shortcutsButton); + + shortcutsBox = new QGroupBox(tr("Shortcuts")); + shortcutsBox->setLayout(shortcutsLayout); + + connect(shortcutsButton,SIGNAL(clicked()),this,SIGNAL(editShortcuts())); + connect(accept,SIGNAL(clicked()),this,SLOT(saveOptions())); connect(cancel,SIGNAL(clicked()),this,SLOT(restoreOptions())); //TODO fix this connect(cancel,SIGNAL(clicked()),this,SLOT(close())); diff --git a/custom_widgets/yacreader_options_dialog.h b/custom_widgets/yacreader_options_dialog.h index 67a08230..9b347ae8 100644 --- a/custom_widgets/yacreader_options_dialog.h +++ b/custom_widgets/yacreader_options_dialog.h @@ -8,6 +8,7 @@ class YACReaderGLFlowConfigWidget; class QCheckBox; class QPushButton; class QSettings; +class QGroupBox; class YACReaderOptionsDialog : public QDialog { @@ -20,6 +21,8 @@ protected: QPushButton * accept; QPushButton * cancel; + QGroupBox * shortcutsBox; + QSettings * settings; QSettings * previousSettings; @@ -56,6 +59,7 @@ protected slots: signals: void optionsChanged(); + void editShortcuts(); }; -#endif // YACREADER_OPTIONS_DIALOG_H \ No newline at end of file +#endif // YACREADER_OPTIONS_DIALOG_H diff --git a/images/accept_shortcut.png b/images/accept_shortcut.png index 4d2b852dbb11826d538a648f3f54978f42a34aa2..dc0017b95566704bdcb79d41406bb660df56f9ea 100644 GIT binary patch delta 139 zcmV;60CfNF0n7oANq=rhL_t(|+G70w|33q{fQf+uKwr)MeSHpOX$ERc08;y51~8GR zaXt_y78Vs9K+%ls90D2_pg0?<^L_t(|+G70w|33q{fQf+uK#^t^QWW?1^(6qQ0w8uMEGqiI zM54y|K&%hM2T+_1lvE%>BMe-CYX(XhfCPZ_0wOdn1UfkZ*=CS4_<=Yc*#LYRL4Xky z&)5_}4aBJtt{I0d4EQvn1Q<{fObiqN yCVZx&(?n^e$mYSX86}=6jjDwN^rFZE1Q-B6|4^ &group) +void EditShortcutsDialog::addActionsGroup(const QString &name, const QIcon &ico, QList &group) { //TODO //groups model add groupsModel->addActionsGroup(ActionsGroup(name,ico,group)); } -void ShortcutsDialog::resetToDefaults() +void EditShortcutsDialog::resetToDefaults() { } -void ShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex &mi2) +void EditShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex &mi2) { actionsModel->addActions(groupsModel->getActions(mi)); } diff --git a/shortcuts_management/shortcuts_dialog.h b/shortcuts_management/edit_shortcuts_dialog.h similarity index 70% rename from shortcuts_management/shortcuts_dialog.h rename to shortcuts_management/edit_shortcuts_dialog.h index d63c9bcd..0ba29413 100644 --- a/shortcuts_management/shortcuts_dialog.h +++ b/shortcuts_management/edit_shortcuts_dialog.h @@ -1,5 +1,5 @@ -#ifndef SHORTCUTS_DIALOG_H -#define SHORTCUTS_DIALOG_H +#ifndef EDIT_SHORTCUTS_DIALOG_H +#define EDIT_SHORTCUTS_DIALOG_H #include #include @@ -10,11 +10,11 @@ class QTableView; class ActionsGroupsModel; class ActionsShortcutsModel; -class ShortcutsDialog : public QDialog +class EditShortcutsDialog : public QDialog { Q_OBJECT public: - explicit ShortcutsDialog(QWidget * parent = 0); + explicit EditShortcutsDialog(QWidget * parent = 0); void addActionsGroup(const QString & name, const QIcon & ico, QList & group); signals: @@ -29,4 +29,4 @@ protected: ActionsShortcutsModel * actionsModel; }; -#endif // SHORTCUTS_DIALOG_H +#endif // EDIT_SHORTCUTS_DIALOG_H diff --git a/shortcuts_management/shortcuts_management.pri b/shortcuts_management/shortcuts_management.pri index e85a6912..d12f8fa0 100644 --- a/shortcuts_management/shortcuts_management.pri +++ b/shortcuts_management/shortcuts_management.pri @@ -2,14 +2,14 @@ INCLUDEPATH += $$PWD DEPENDPATH += $$PWD HEADERS += \ - $$PWD/shortcuts_dialog.h \ + $$PWD/edit_shortcuts_dialog.h \ $$PWD/actions_groups_model.h \ $$PWD/actions_shortcuts_model.h \ $$PWD/edit_shortcut_item_delegate.h \ $$PWD/shortcuts_manager.h SOURCES += \ - $$PWD/shortcuts_dialog.cpp \ + $$PWD/edit_shortcuts_dialog.cpp \ $$PWD/actions_groups_model.cpp \ $$PWD/actions_shortcuts_model.cpp \ $$PWD/edit_shortcut_item_delegate.cpp \ diff --git a/shortcuts_management/shortcuts_manager.cpp b/shortcuts_management/shortcuts_manager.cpp index eca6e45b..d7bb4375 100644 --- a/shortcuts_management/shortcuts_manager.cpp +++ b/shortcuts_management/shortcuts_manager.cpp @@ -11,6 +11,8 @@ ShortcutsManager::ShortcutsManager() void ShortcutsManager::initDefaultShorcuts() { +#ifdef YACREADER_LIBRARY + //ACTIONS defaultShorcuts.insert(CREATE_LIBRARY_ACTION_YL,Qt::Key_A); defaultShorcuts.insert(OPEN_LIBRARY_ACTION_YL,Qt::Key_O); defaultShorcuts.insert(UPDATE_LIBRARY_ACTION_YL,Qt::Key_U); @@ -25,6 +27,50 @@ void ShortcutsManager::initDefaultShorcuts() defaultShorcuts.insert(OPTIONS_ACTION_YL,Qt::Key_C); defaultShorcuts.insert(SERVER_CONFIG_ACTION_YL,Qt::Key_S); defaultShorcuts.insert(TOGGLE_COMICS_VIEW_ACTION_YL,Qt::Key_V); + + //COMMANDS (used in keypressevent) +#else + defaultShorcuts.insert(OPEN_ACTION_Y, Qt::Key_O); + defaultShorcuts.insert(OPEN_FOLDER_ACTION_Y, Qt::CTRL | Qt::Key_O); + defaultShorcuts.insert(OPEN_PREVIOUS_COMIC_ACTION_Y, Qt::CTRL | Qt::Key_Left); + defaultShorcuts.insert(OPEN_NEXT_COMIC_ACTION_Y, Qt::CTRL | Qt::Key_Right); + defaultShorcuts.insert(PREV_ACTION_Y, Qt::Key_Left); + defaultShorcuts.insert(NEXT_ACTION_Y, Qt::Key_Right); + defaultShorcuts.insert(LEFT_ROTATION_ACTION_Y, Qt::Key_L); + defaultShorcuts.insert(RIGHT_ROTATION_ACTION_Y, Qt::Key_R); + defaultShorcuts.insert(DOUBLE_PAGE_ACTION_Y, Qt::Key_D); + defaultShorcuts.insert(GO_TO_PAGE_ACTION_Y, Qt::Key_G); + defaultShorcuts.insert(OPTIONS_ACTION_Y, Qt::Key_C); + defaultShorcuts.insert(HELP_ABOUT_ACTION_Y, Qt::Key_F1); + defaultShorcuts.insert(SHOW_MAGNIFYING_GLASS_ACTION_Y, Qt::Key_Z); + defaultShorcuts.insert(SET_BOOKMARK_ACTION_Y, Qt::CTRL | Qt::Key_M); + defaultShorcuts.insert(SHOW_BOOKMARKS_ACTION_Y, Qt::Key_M); + defaultShorcuts.insert(SHOW_INFO_ACTION_Y, Qt::Key_I); + defaultShorcuts.insert(CLOSE_ACTION_Y, Qt::Key_Escape); + defaultShorcuts.insert(SHOW_DICTIONARY_ACTION_Y, Qt::Key_T); + defaultShorcuts.insert(ALWAYS_ON_TOP_ACTION_Y, Qt::Key_Q); //deprecated + defaultShorcuts.insert(ADJUST_TO_FULL_SIZE_ACTION_Y, Qt::Key_W); + defaultShorcuts.insert(SHOW_FLOW_ACTION_Y, Qt::Key_S); + + //main_window_viewer + defaultShorcuts.insert(TOGGLE_FULL_SCREEN_ACTION_Y, Qt::Key_F); + defaultShorcuts.insert(TOGGLE_TOOL_BARS_ACTION_Y, Qt::Key_H); + defaultShorcuts.insert(CHANGE_FIT_ACTION_Y, Qt::Key_A); + //viewer + defaultShorcuts.insert(AUTO_SCROLL_FORWARD_ACTION_Y, Qt::Key_Space); + defaultShorcuts.insert(AUTO_SCROLL_BACKWARD_ACTION_Y, Qt::Key_B); + defaultShorcuts.insert(MOVE_DOWN_ACTION_Y, Qt::Key_Down); + defaultShorcuts.insert(MOVE_UP_ACTION_Y, Qt::Key_Up); + defaultShorcuts.insert(GO_TO_FIRST_PAGE_ACTION_Y, Qt::Key_Home); + defaultShorcuts.insert(GO_TO_LAST_PAGE_ACTION_Y, Qt::Key_End); + //mglass + defaultShorcuts.insert(SIZE_UP_MGLASS_ACTION_Y, Qt::Key_Plus); + defaultShorcuts.insert(SIZE_DOWN_MGLASS_ACTION_Y, Qt::Key_Minus); + defaultShorcuts.insert(ZOOM_IN_MGLASS_ACTION_Y, Qt::Key_Asterisk); + defaultShorcuts.insert(ZOOM_OUT_MGLASS_ACTION_Y, Qt::Key_Underscore); + +#endif + } void ShortcutsManager::resetToDefaults() diff --git a/shortcuts_management/shortcuts_manager.h b/shortcuts_management/shortcuts_manager.h index a44d641b..6fd2f5c5 100644 --- a/shortcuts_management/shortcuts_manager.h +++ b/shortcuts_management/shortcuts_manager.h @@ -67,7 +67,57 @@ public: #define DELETE_COMICS_ACTION_YL "DELETE_COMICS_ACTION_YL" #define HIDE_COMIC_VIEW_ACTION_YL "HIDE_COMIC_VIEW_ACTION_YL" #define GET_INFO_ACTION_YL "GET_INFO_ACTION_YL" +#define SHOW_EDIT_SHORTCUTS_ACTION_YL "SHOW_EDIT_SHORTCUTS_ACTION_YL" + +//COMMANDS YACReaderLibrary +//ACTION NAMES YACReader +#define OPEN_ACTION_Y "OPEN_ACTION_Y" +#define OPEN_FOLDER_ACTION_Y "OPEN_FOLDER_ACTION_Y" +#define SAVE_IMAGE_ACTION_Y "SAVE_IMAGE_ACTION_Y" +#define OPEN_PREVIOUS_COMIC_ACTION_Y "OPEN_PREVIOUS_COMIC_ACTION_Y" +#define OPEN_NEXT_COMIC_ACTION_Y "OPEN_NEXT_COMIC_ACTION_Y" +#define PREV_ACTION_Y "PREV_ACTION_Y" +#define NEXT_ACTION_Y "NEXT_ACTION_Y" +#define ADJUST_HEIGHT_ACTION_Y "ADJUST_HEIGHT_Y" +#define ADJUST_WIDTH_ACTION_Y "ADJUST_WIDTH_Y" +#define LEFT_ROTATION_ACTION_Y "LEFT_ROTATION_ACTION_Y" +#define RIGHT_ROTATION_ACTION_Y "RIGHT_ROTATION_ACTION_Y" +#define DOUBLE_PAGE_ACTION_Y "DOUBLE_PAGE_ACTION_Y" +#define GO_TO_PAGE_ACTION_Y "GO_TO_PAGE_ACTION_Y" +#define OPTIONS_ACTION_Y "OPTIONS_ACTION_Y" +#define HELP_ABOUT_ACTION_Y "HELP_ABOUT_ACTION_Y" +#define SHOW_MAGNIFYING_GLASS_ACTION_Y "SHOW_MAGNIFYING_GLASS_ACTION_Y" +#define SET_BOOKMARK_ACTION_Y "SET_BOOKMARK_ACTION_Y" +#define SHOW_BOOKMARKS_ACTION_Y "SHOW_BOOKMARKS_ACTION_Y" +#define SHOW_SHORCUTS_ACTION_Y "SHOW_SHORCUTS_ACTION_Y" +#define SHOW_INFO_ACTION_Y "SHOW_INFO_ACTION_Y" +#define CLOSE_ACTION_Y "CLOSE_ACTION_Y" +#define SHOW_DICTIONARY_ACTION_Y "SHOW_DICTIONARY_ACTION_Y" +#define ALWAYS_ON_TOP_ACTION_Y "ALWAYS_ON_TOP_ACTION_Y" +#define ADJUST_TO_FULL_SIZE_ACTION_Y "ADJUST_TO_FULL_SIZE_ACTION_Y" +#define SHOW_FLOW_ACTION_Y "SHOW_FLOW_ACTION_Y" +#define SHOW_EDIT_SHORTCUTS_ACTION_Y "SHOW_EDIT_SHORTCUTS_ACTION_Y" + +//COMMANDS YACReader +//main_viewer_window +#define TOGGLE_FULL_SCREEN_ACTION_Y "TOGGLE_FULL_SCREEN_ACTION_Y" +#define TOGGLE_TOOL_BARS_ACTION_Y "TOGGLE_TOOL_BARS_ACTION_Y" +#define CHANGE_FIT_ACTION_Y "CHANGE_FIT_ACTION_Y" +//viewer +#define AUTO_SCROLL_FORWARD_ACTION_Y "AUTO_SCROLL_FORWARD_ACTION_Y" +#define AUTO_SCROLL_BACKWARD_ACTION_Y "AUTO_SCROLL_BACKWARD_ACTION_Y" +#define MOVE_DOWN_ACTION_Y "MOVE_DOWN_ACTION_Y" +#define MOVE_UP_ACTION_Y "MOVE_UP_ACTION_Y" +#define MOVE_LEFT_ACTION_Y "MOVE_LEFT_ACTION_Y" +#define MOVE_RIGHT_ACTION_Y "MOVE_RIGHT_ACTION_Y" +#define GO_TO_FIRST_PAGE_ACTION_Y "GO_TO_FIRST_PAGE_ACTION_Y" +#define GO_TO_LAST_PAGE_ACTION_Y "GO_TO_LAST_PAGE_ACTION_Y" +//mglass +#define SIZE_UP_MGLASS_ACTION_Y "SIZE_UP_MGLASS_ACTION_Y" +#define SIZE_DOWN_MGLASS_ACTION_Y "SIZE_DOWN_MGLASS_ACTION_Y" +#define ZOOM_IN_MGLASS_ACTION_Y "ZOOM_IN_MGLASS_ACTION_Y" +#define ZOOM_OUT_MGLASS_ACTION_Y "ZOOM_OUT_MGLASS_ACTION_Y" #endif // SHORTCUTS_MANAGER_H From 8e6347801baf2911eac9a88c661264096eaf334f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sat, 19 Jul 2014 08:57:48 +0200 Subject: [PATCH 20/34] updated grid UI design for OSX --- YACReaderLibrary/YACReaderLibrary.pro | 3 +++ YACReaderLibrary/grid_comics_view.cpp | 4 ++-- YACReaderLibrary/qml.qrc | 2 -- YACReaderLibrary/qml/page-macosx.png | Bin 0 -> 171 bytes YACReaderLibrary/qml/star-macosx.png | Bin 0 -> 288 bytes YACReaderLibrary/qml_osx.qrc | 6 ++++++ YACReaderLibrary/qml_win.qrc | 6 ++++++ 7 files changed, 17 insertions(+), 4 deletions(-) create mode 100644 YACReaderLibrary/qml/page-macosx.png create mode 100644 YACReaderLibrary/qml/star-macosx.png create mode 100644 YACReaderLibrary/qml_osx.qrc create mode 100644 YACReaderLibrary/qml_win.qrc diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index 7a396a70..bb4f8252 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -209,6 +209,9 @@ SOURCES += grid_comics_view.cpp \ comics_view_transition.cpp RESOURCES += qml.qrc +win32:RESOURCES += qml_win.qrc +unix:!macx:RESOURCES += qml_win.qrc +macx:RESOURCES += qml_osx.qrc } else { Release:DESTDIR = ../release diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp index a8effdd9..257c2d63 100644 --- a/YACReaderLibrary/grid_comics_view.cpp +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -64,8 +64,8 @@ void GridComicsView::setModel(TableModel *model) } #ifdef Q_OS_MAC - ctxt->setContextProperty("backgroundColor", "#FAFAFA"); - ctxt->setContextProperty("cellColor", "#EDEDED"); + ctxt->setContextProperty("backgroundColor", "#EDEDED"); + ctxt->setContextProperty("cellColor", "#FFFFFF"); ctxt->setContextProperty("selectedColor", "#DDDDDD"); ctxt->setContextProperty("titleColor", "#121212"); ctxt->setContextProperty("textColor", "#636363"); diff --git a/YACReaderLibrary/qml.qrc b/YACReaderLibrary/qml.qrc index ec494823..a02ccead 100644 --- a/YACReaderLibrary/qml.qrc +++ b/YACReaderLibrary/qml.qrc @@ -2,8 +2,6 @@ qml/GridComicsView.qml qml/YACReaderScrollView.qml - qml/page.png - qml/star.png qml/tick.png qml/reading.png diff --git a/YACReaderLibrary/qml/page-macosx.png b/YACReaderLibrary/qml/page-macosx.png new file mode 100644 index 0000000000000000000000000000000000000000..c8216591679ffb2e0dac358288275c4ce8d539ce GIT binary patch literal 171 zcmeAS@N?(olHy`uVBq!ia0vp^96-#)!3HEdkIOdzDajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MZTUcjv*Ddl6d&}|DSK*6bfGxv60~nlO_Wbn*`g(g%*rQ z(i0ExF*Q~=y12A3>9M|ESYq48wBT-!f!T$Y|2{k{46-5{4#+fnGdeUdGC1cdp31NN R!3#8)!PC{xWt~$(697>)GP3{x literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/star-macosx.png b/YACReaderLibrary/qml/star-macosx.png new file mode 100644 index 0000000000000000000000000000000000000000..37577a74c8a41fc0fefcab1b1fe49c750f2ee8ac GIT binary patch literal 288 zcmV+*0pI?KP)C24IAC14hUQ z{YySGLN_QQFan?XWQ4LoH)uNnY>?WLi7H4=I{CQR_rj!rre}xA-v7M^7fDX>;cf~LG0Dm%$F|N521bkdXAUS~WA?lm?MZ7#7tIvulm>39y z^0RoR(IeYhR2qTw+usVKMhmq=u>Go$FUqaTSFcB63jr5@3tQ!@qV0MlUxilrY|T!@ m`KRr){7>jM>~=Ii0R{k? + + qml/page-macosx.png + qml/star-macosx.png + + diff --git a/YACReaderLibrary/qml_win.qrc b/YACReaderLibrary/qml_win.qrc new file mode 100644 index 00000000..59e2872f --- /dev/null +++ b/YACReaderLibrary/qml_win.qrc @@ -0,0 +1,6 @@ + + + qml/page.png + qml/star.png + + From 5e7c2f72b50f9a5507bfe0bb17ed4e905b5b2647 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sat, 19 Jul 2014 09:47:10 +0200 Subject: [PATCH 21/34] fixed MacOSX-Qt5 build --- YACReaderLibrary/empty_folder_widget.cpp | 2 +- YACReaderLibrary/library_window.cpp | 8 ++++---- shortcuts_management/actions_groups_model.cpp | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/YACReaderLibrary/empty_folder_widget.cpp b/YACReaderLibrary/empty_folder_widget.cpp index 6d440f51..b05eddd1 100644 --- a/YACReaderLibrary/empty_folder_widget.cpp +++ b/YACReaderLibrary/empty_folder_widget.cpp @@ -9,7 +9,7 @@ #include void testListView(QListView * l) { - QStringListModel * slm = new QStringListModel(QStringList() = {"Lorem ipsum", "Hailer skualer", "Mumbaluba X", "Finger layden", "Pacum tactus filer", "Aposum", "En","Lorem ipsum", "Hailer skualer", "Mumbaluba X", "Finger layden", "Pacum tactus filer", "Aposum", "En" }); + QStringListModel * slm = new QStringListModel(QStringList() << "Lorem ipsum" << "Hailer skualer"<< "Mumbaluba X" << "Finger layden" << "Pacum tactus filer" << "Aposum" << "En" << "Lorem ipsum" << "Hailer skualer" << "Mumbaluba X" << "Finger layden" << "Pacum tactus filer" << "Aposum" << "En" ); l->setModel(slm); } diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 686afd9e..65ccb6da 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -857,8 +857,8 @@ void LibraryWindow::createMenus() libraryMenu->addAction(removeLibraryAction); libraryMenu->addSeparator(); - libraryMenu->addAction(exportComicsInfo); - libraryMenu->addAction(importComicsInfo); + libraryMenu->addAction(exportComicsInfoAction); + libraryMenu->addAction(importComicsInfoAction); libraryMenu->addSeparator(); @@ -872,8 +872,8 @@ void LibraryWindow::createMenus() folderMenu->addAction(setFolderAsNotCompletedAction); folderMenu->addAction(setFolderAsCompletedAction); folderMenu->addSeparator(); - folderMenu->addAction(setFolderAsFinishedAction); - folderMenu->addAction(setFolderAsNotFinishedAction); + folderMenu->addAction(setFolderAsReadAction); + folderMenu->addAction(setFolderAsUnreadAction); //comic QMenu * comicMenu = new QMenu(tr("Comic")); diff --git a/shortcuts_management/actions_groups_model.cpp b/shortcuts_management/actions_groups_model.cpp index 47ba2e91..a337cbff 100644 --- a/shortcuts_management/actions_groups_model.cpp +++ b/shortcuts_management/actions_groups_model.cpp @@ -20,7 +20,7 @@ QModelIndex ActionsGroupsModel::index(int row, int column, const QModelIndex &pa if (!hasIndex(row, column, parent)) return QModelIndex(); - return createIndex(row, column, &groups.at(row).getActions()); + return createIndex(row, column); } QVariant ActionsGroupsModel::data(const QModelIndex &index, int role) const From 1dc8942171c1bd9d62e51c257d7d42780a8fafd3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sat, 19 Jul 2014 10:55:52 +0200 Subject: [PATCH 22/34] shortcuts are now shown with their native representation, fixed edit shortcuts dialog layout --- YACReader/shortcuts_dialog.cpp | 2 +- shortcuts_management/actions_shortcuts_model.cpp | 2 +- shortcuts_management/edit_shortcut_item_delegate.cpp | 6 +++--- shortcuts_management/edit_shortcuts_dialog.cpp | 12 +++++++++--- 4 files changed, 14 insertions(+), 8 deletions(-) diff --git a/YACReader/shortcuts_dialog.cpp b/YACReader/shortcuts_dialog.cpp index 8c7fe908..dd7441cf 100644 --- a/YACReader/shortcuts_dialog.cpp +++ b/YACReader/shortcuts_dialog.cpp @@ -51,7 +51,7 @@ ShortcutsDialog::ShortcutsDialog(QWidget * parent) setLayout(imgMainLayout); - setFixedSize(QSize(700,500)); + setFixedSize(QSize(700,500)); QFile f(":/files/shortcuts.html"); f.open(QIODevice::ReadOnly); diff --git a/shortcuts_management/actions_shortcuts_model.cpp b/shortcuts_management/actions_shortcuts_model.cpp index 0a9760d1..14ed0346 100644 --- a/shortcuts_management/actions_shortcuts_model.cpp +++ b/shortcuts_management/actions_shortcuts_model.cpp @@ -63,7 +63,7 @@ QVariant ActionsShortcutsModel::data(const QModelIndex &index, int role) const if (index.column() == NAME) return QVariant(actions[index.row()]->toolTip()); if (index.column() == KEYS) - return QVariant(actions[index.row()]->shortcut().toString()); + return QVariant(actions[index.row()]->shortcut().toString(QKeySequence::NativeText)); return QVariant(); } diff --git a/shortcuts_management/edit_shortcut_item_delegate.cpp b/shortcuts_management/edit_shortcut_item_delegate.cpp index 5b7d3627..4c10b11a 100644 --- a/shortcuts_management/edit_shortcut_item_delegate.cpp +++ b/shortcuts_management/edit_shortcut_item_delegate.cpp @@ -75,15 +75,15 @@ KeySequenceLineEdit::KeySequenceLineEdit(QWidget *parent) connect(acceptButton, SIGNAL(clicked()), this, SIGNAL(editingFinished())); } -void KeySequenceLineEdit::resizeEvent(QResizeEvent *) +void KeySequenceLineEdit::resizeEvent(QResizeEvent * e) { QSize szClear = clearButton->sizeHint(); //int frameWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth); int leftMargin = style()->pixelMetric(QStyle::PM_LayoutLeftMargin); int topMargin = style()->pixelMetric(QStyle::PM_LayoutTopMargin); - clearButton->move(0 + leftMargin,topMargin-4); + clearButton->move(0 + leftMargin,(e->size().height()-19)/2); //16 is the icon height+1blank pixel - acceptButton->move( leftMargin + szClear.width(),topMargin-4); + acceptButton->move( leftMargin + szClear.width(),(e->size().height()-19)/2); } diff --git a/shortcuts_management/edit_shortcuts_dialog.cpp b/shortcuts_management/edit_shortcuts_dialog.cpp index 3ae1e7e1..cd8bdc5c 100644 --- a/shortcuts_management/edit_shortcuts_dialog.cpp +++ b/shortcuts_management/edit_shortcuts_dialog.cpp @@ -18,7 +18,7 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : QDialog(parent) { QPushButton * resetButton = new QPushButton(tr("Restore defaults"),this); - QLabel * infoLabel = new QLabel(tr("To change a shortcut, select it, click in the key combination and type the new keys.")); + QLabel * infoLabel = new QLabel(tr("To change a shortcut, double click in the key combination and type the new keys.")); QVBoxLayout * layout = new QVBoxLayout(this); QSplitter * splitter = new QSplitter(this); actionsGroupsListView = new QListView(this); @@ -42,7 +42,8 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : actionsGroupsListView->setModel(groupsModel); actionsTableView->setModel(actionsModel); actionsTableView->setColumnWidth(0,30); - actionsTableView->setColumnWidth(1,270); + actionsTableView->setColumnWidth(1,360); + //actionsTableView->horizontalHeader()->sectionResizeMode(QHeaderView::Custom); actionsTableView->horizontalHeader()->setStretchLastSection(true); actionsTableView->setSelectionBehavior(QAbstractItemView::SelectRows); actionsTableView->setShowGrid(false); @@ -58,7 +59,11 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : connect(resetButton,SIGNAL(clicked()),this,SLOT(resetToDefaults())); connect(actionsGroupsListView->selectionModel(),SIGNAL(currentChanged(QModelIndex,QModelIndex)),this,SLOT(loadShortcuts(QModelIndex,QModelIndex))); //clicked(QModelIndex) doesn't work :S - setFixedSize(640,480); +#ifdef Q_OS_MAC + setFixedSize(760,500); +#else + setFixedSize(804,500); //extra width for modifiers +#endif setWindowTitle(tr("Shortcuts settings")); setModal(true); @@ -79,4 +84,5 @@ void EditShortcutsDialog::resetToDefaults() void EditShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex &mi2) { actionsModel->addActions(groupsModel->getActions(mi)); + //actionsTableView->resizeColumnsToContents(); } From 40a4f408298a5e85aaa747884e5e49e2bb2d3133 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sat, 19 Jul 2014 12:45:27 +0200 Subject: [PATCH 23/34] added default value for empty shortcuts "None" autoload the actions in the first actions groups added to edit shortcuts dialog --- shortcuts_management/actions_shortcuts_model.cpp | 10 +++++++++- shortcuts_management/edit_shortcuts_dialog.cpp | 6 +++--- 2 files changed, 12 insertions(+), 4 deletions(-) diff --git a/shortcuts_management/actions_shortcuts_model.cpp b/shortcuts_management/actions_shortcuts_model.cpp index 14ed0346..86c7d7ba 100644 --- a/shortcuts_management/actions_shortcuts_model.cpp +++ b/shortcuts_management/actions_shortcuts_model.cpp @@ -57,13 +57,21 @@ QVariant ActionsShortcutsModel::data(const QModelIndex &index, int role) const } } + if(role == Qt::ForegroundRole && index.column() == KEYS && actions[index.row()]->shortcut().isEmpty()) + return QBrush(QColor("#AAAAAA")); + if (role != Qt::DisplayRole) return QVariant(); if (index.column() == NAME) return QVariant(actions[index.row()]->toolTip()); if (index.column() == KEYS) - return QVariant(actions[index.row()]->shortcut().toString(QKeySequence::NativeText)); + { + QKeySequence ks = actions[index.row()]->shortcut(); + if(ks.isEmpty()) + return tr("None"); + return QVariant(ks.toString(QKeySequence::NativeText)); + } return QVariant(); } diff --git a/shortcuts_management/edit_shortcuts_dialog.cpp b/shortcuts_management/edit_shortcuts_dialog.cpp index cd8bdc5c..5cfd9088 100644 --- a/shortcuts_management/edit_shortcuts_dialog.cpp +++ b/shortcuts_management/edit_shortcuts_dialog.cpp @@ -40,6 +40,7 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : groupsModel = new ActionsGroupsModel(); actionsModel = new ActionsShortcutsModel(); actionsGroupsListView->setModel(groupsModel); + actionsGroupsListView->setFocus(); actionsTableView->setModel(actionsModel); actionsTableView->setColumnWidth(0,30); actionsTableView->setColumnWidth(1,360); @@ -71,9 +72,9 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : void EditShortcutsDialog::addActionsGroup(const QString &name, const QIcon &ico, QList &group) { - //TODO - //groups model add groupsModel->addActionsGroup(ActionsGroup(name,ico,group)); + if(actionsTableView->model()->rowCount()==0)//first group added + actionsGroupsListView->selectionModel()->select(groupsModel->index(0,0),QItemSelectionModel::Select); } void EditShortcutsDialog::resetToDefaults() @@ -84,5 +85,4 @@ void EditShortcutsDialog::resetToDefaults() void EditShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex &mi2) { actionsModel->addActions(groupsModel->getActions(mi)); - //actionsTableView->resizeColumnsToContents(); } From 2712ef89b2de12fe31dd8a803e34901a77981c57 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Sat, 19 Jul 2014 16:29:19 +0200 Subject: [PATCH 24/34] added icons to actions groups (shortcuts) --- YACReader/main_window_viewer.cpp | 10 +++++----- YACReader/yacreader_images.qrc | 8 ++++++++ YACReaderLibrary/images.qrc | 10 +++++++++- YACReaderLibrary/library_window.cpp | 11 ++++++----- images/shortcuts_group_comics.png | Bin 0 -> 276 bytes images/shortcuts_group_folders.png | Bin 0 -> 157 bytes images/shortcuts_group_general.png | Bin 0 -> 319 bytes images/shortcuts_group_libraries.png | Bin 0 -> 164 bytes images/shortcuts_group_mglass.png | Bin 0 -> 351 bytes images/shortcuts_group_page.png | Bin 0 -> 162 bytes images/shortcuts_group_reading.png | Bin 0 -> 179 bytes images/shortcuts_group_visualization.png | Bin 0 -> 320 bytes 12 files changed, 28 insertions(+), 11 deletions(-) create mode 100644 images/shortcuts_group_comics.png create mode 100644 images/shortcuts_group_folders.png create mode 100644 images/shortcuts_group_general.png create mode 100644 images/shortcuts_group_libraries.png create mode 100644 images/shortcuts_group_mglass.png create mode 100644 images/shortcuts_group_page.png create mode 100644 images/shortcuts_group_reading.png create mode 100644 images/shortcuts_group_visualization.png diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index e262d31b..e77ded8e 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -983,7 +983,7 @@ void MainWindowViewer::setUpShortcutsManagement() //actions holder QObject * orphanActions = new QObject; - editShortcutsDialog->addActionsGroup(tr("Comics"),QIcon(":/images/openInYACReader.png"), + editShortcutsDialog->addActionsGroup(tr("Comics"),QIcon(":/images/shortcuts_group_comics.png"), QList() << openAction << openFolderAction @@ -1000,7 +1000,7 @@ void MainWindowViewer::setUpShortcutsManagement() toggleToolbarsAction->setData(TOGGLE_TOOL_BARS_ACTION_Y); toggleToolbarsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)); - editShortcutsDialog->addActionsGroup(tr("General"),QIcon(), + editShortcutsDialog->addActionsGroup(tr("General"),QIcon(":/images/shortcuts_group_general.png"), QList() << optionsAction << helpAboutAction @@ -1030,7 +1030,7 @@ void MainWindowViewer::setUpShortcutsManagement() zoomOutMglassAction->setData(ZOOM_OUT_MGLASS_ACTION_Y); zoomOutMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)); - editShortcutsDialog->addActionsGroup(tr("Magnifiying glass"),QIcon(), + editShortcutsDialog->addActionsGroup(tr("Magnifiying glass"),QIcon(":/images/shortcuts_group_mglass.png"), QList() << showMagnifyingGlassAction << sizeUpMglassAction @@ -1043,7 +1043,7 @@ void MainWindowViewer::setUpShortcutsManagement() toggleFitToScreenAction->setData(CHANGE_FIT_ACTION_Y); toggleFitToScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)); - editShortcutsDialog->addActionsGroup(tr("Page adjustement"),QIcon(), + editShortcutsDialog->addActionsGroup(tr("Page adjustement"),QIcon(":/images/shortcuts_group_page.png"), QList() << adjustHeightAction << adjustWidthAction @@ -1085,7 +1085,7 @@ void MainWindowViewer::setUpShortcutsManagement() goToLastPageAction->setData(GO_TO_LAST_PAGE_ACTION_Y); goToLastPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)); - editShortcutsDialog->addActionsGroup(tr("Reading"),QIcon(), + editShortcutsDialog->addActionsGroup(tr("Reading"),QIcon(":/images/shortcuts_group_reading.png"), QList() << nextAction << prevAction diff --git a/YACReader/yacreader_images.qrc b/YACReader/yacreader_images.qrc index def5653a..ca8f9714 100644 --- a/YACReader/yacreader_images.qrc +++ b/YACReader/yacreader_images.qrc @@ -75,5 +75,13 @@ ../images/speaker.png ../images/clear_shortcut.png ../images/accept_shortcut.png + ../images/shortcuts_group_comics.png + ../images/shortcuts_group_folders.png + ../images/shortcuts_group_general.png + ../images/shortcuts_group_libraries.png + ../images/shortcuts_group_mglass.png + ../images/shortcuts_group_page.png + ../images/shortcuts_group_reading.png + ../images/shortcuts_group_visualization.png diff --git a/YACReaderLibrary/images.qrc b/YACReaderLibrary/images.qrc index d6551235..750630a2 100644 --- a/YACReaderLibrary/images.qrc +++ b/YACReaderLibrary/images.qrc @@ -77,7 +77,7 @@ ../images/social_dialog/shadow.png ../images/social_dialog/twitter.png ../images/social_dialog/separator.png--> - ../images/main_toolbar/divider.png + ../images/main_toolbar/divider.png ../images/collapsed_branch_osx.png ../images/expanded_branch_osx.png ../images/folder_macosx.png @@ -102,6 +102,14 @@ ../images/find_folder.png ../images/clear_shortcut.png ../images/accept_shortcut.png + ../images/shortcuts_group_comics.png + ../images/shortcuts_group_folders.png + ../images/shortcuts_group_general.png + ../images/shortcuts_group_libraries.png + ../images/shortcuts_group_mglass.png + ../images/shortcuts_group_page.png + ../images/shortcuts_group_reading.png + ../images/shortcuts_group_visualization.png diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 65ccb6da..0d2e9a42 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -297,7 +297,8 @@ void LibraryWindow::doDialogs() void LibraryWindow::setUpShortcutsManagement() { - editShortcutsDialog->addActionsGroup("Comics",QIcon(":/images/openInYACReader.png"), + + editShortcutsDialog->addActionsGroup("Comics",QIcon(":/images/shortcuts_group_comics.png"), QList() << openComicAction << setAsReadAction @@ -310,7 +311,7 @@ void LibraryWindow::setUpShortcutsManagement() << deleteComicsAction << getInfoAction); - editShortcutsDialog->addActionsGroup("Folders",QIcon(), + editShortcutsDialog->addActionsGroup("Folders",QIcon(":/images/shortcuts_group_folders.png"), QList() << setRootIndexAction << expandAllNodesAction @@ -321,7 +322,7 @@ void LibraryWindow::setUpShortcutsManagement() << setFolderAsReadAction << setFolderAsUnreadAction); - editShortcutsDialog->addActionsGroup("General",QIcon(), + editShortcutsDialog->addActionsGroup("General",QIcon(":/images/shortcuts_group_general.png"), QList() << backAction << forwardAction @@ -330,7 +331,7 @@ void LibraryWindow::setUpShortcutsManagement() << serverConfigAction << showEditShortcutsAction); - editShortcutsDialog->addActionsGroup("Libraries",QIcon(), + editShortcutsDialog->addActionsGroup("Libraries",QIcon(":/images/shortcuts_group_libraries.png"), QList() << createLibraryAction << openLibraryAction @@ -342,7 +343,7 @@ void LibraryWindow::setUpShortcutsManagement() << renameLibraryAction << removeLibraryAction); - editShortcutsDialog->addActionsGroup("Visualization",QIcon(), + editShortcutsDialog->addActionsGroup("Visualization",QIcon(":/images/shortcuts_group_visualization.png"), QList() << showHideMarksAction << toggleFullScreenAction diff --git a/images/shortcuts_group_comics.png b/images/shortcuts_group_comics.png new file mode 100644 index 0000000000000000000000000000000000000000..b0c44cd98576820f55508a866011c4043384db2e GIT binary patch literal 276 zcmeAS@N?(olHy`uVBq!ia0vp^{2@)Z;PBq3c)UjmAa;4;HBn4T7r-89mOXEAUzvIp^Lv=U^p}z$EmO=LsXbn$q8n z%W14ertu7!{~S7>F{fnBqaJ_fHRHmmzh@_6V=e< z@fIp|@Ujq3*rvET`Oq@SS!MDLI~DX<7iAw=DarL}VdF}U(;P006E-)_NO;7^VZiXG W(cJ%`KX(|=2MnIBelF{r5}E+Dv0@AW literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_folders.png b/images/shortcuts_group_folders.png new file mode 100644 index 0000000000000000000000000000000000000000..968e5a20d07765c0d6ea94fc0c491f82e3aa6a84 GIT binary patch literal 157 zcmeAS@N?(olHy`uVBq!ia0vp^{2^)r^Ln>}1B}ix-;83!$wf(=?JLmqsxf65FwlEmBJ>EW3 zCOWa9Q)ss1EdB{AzcU;)lUE6Te^Bw&^G0R{MmC0=ZoWM0i}l_>qZmA0{an^LB{Ts5 Dm98>U literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_general.png b/images/shortcuts_group_general.png new file mode 100644 index 0000000000000000000000000000000000000000..81c8317a07480d5b651c0795b06e0b5c49bca34a GIT binary patch literal 319 zcmV-F0l@x=P)StlgkTj<7+q0gR9l>PXL$>o}KMvP_Jhe=qT_A@e-H$#kT7+qQ=l;}u{elSl{B zUHqDaKT)#$118{;%-U!q!*)nAl>|XlYylb6;d_9(07q(6J498Mi{lx*sDKw~q}Ix; z5$CE4kF74MV&tV@tP*^R){7iLIh@=k3(nPO!I*VKC8sKn6&Q2fm47EdFMzu-)^rma zlv1cT3CI^ivyf%mLxW$--&;H<(S^nBrDMHX$_h?P_KyHZB((4LKdD=s002ovPDHLkV1lZ?gg5{I literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_libraries.png b/images/shortcuts_group_libraries.png new file mode 100644 index 0000000000000000000000000000000000000000..7c7f9b197c9502858cb43c32c238142fec67a6b3 GIT binary patch literal 164 zcmeAS@N?(olHy`uVBq!ia0vp^{2}1B`jbtXyWkqKmUJ8)}4K|*2==FX2DBlHYV9m zew9C=UbD=M+ivniMziUG_tTR8+82EEF+I!QapiM^mL`wF0S3#*{9Bhv-DLrq$l&Sf K=d#Wzp$PyRxi$0v literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_mglass.png b/images/shortcuts_group_mglass.png new file mode 100644 index 0000000000000000000000000000000000000000..4d1dde7af6b24edca99c84d8ed359037f72a497f GIT binary patch literal 351 zcmV-l0igbgP)dg2b2h;!zYm=v6@7p zU@k4VWR+kyRw%|^S5zapqSUw?j-k|-MuLLZ*cOMX*H6~a`?mORa}`_oMSiSRx?BLU zj8h#o#u8qSwz%NaEkE8i_iq1uW0q9%RpgI>=y3A``ol~3%cZQDx_1bZBB xz?39cDi9?bV5SD!JHP%ODBQsNWP}v{M83h0U002ovPDHLkV1n)zjd=h7 literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_page.png b/images/shortcuts_group_page.png new file mode 100644 index 0000000000000000000000000000000000000000..a482cebbe2339e4c669cad19908bb42ad2cc01bc GIT binary patch literal 162 zcmeAS@N?(olHy`uVBq!ia0vp^{2}1B`jbtP%5^uwf(?>c~*q6fq|J}(oMmyftI)XfaWoHy85}S Ib4q9e05~f$rvLx| literal 0 HcmV?d00001 diff --git a/images/shortcuts_group_reading.png b/images/shortcuts_group_reading.png new file mode 100644 index 0000000000000000000000000000000000000000..fd520f183455508a70b05574f69c2f345a2da88a GIT binary patch literal 179 zcmeAS@N?(olHy`uVBq!ia0vp^{2}1B`jbtP%5^uwf(k7RCwCNQp*j3KoDiAzz(DY=m3tK z)C$xJ)Doz`SOHi8dg6#3KnG|Cl;C@0HkpjXE0etJ@^;>Q34*1}St8H#GmbqFEBXyQ zfTC%d@kb01oogWalZ?U(0#y76m9SMpIu$?$93?Yt1(fhVOqp>ZG020r1kroErMUOP zwWgJfnXg7ziLe#HApFg=P*S`#(L&R$>#`EoL_&h#_&tTw^y zof&B2)T12;Ge3npsku;?F}O3)Yt`$VvG1L)YVHr?4mW0ybiT_b#$a*#3NQfev`7kg S=w>|t0000 Date: Tue, 22 Jul 2014 22:02:06 +0200 Subject: [PATCH 25/34] added shortcuts conflicts detection --- YACReader/main_window_viewer.cpp | 26 +++++++++++++++---- YACReaderLibrary/library_window.cpp | 23 ++++++++++++---- .../actions_shortcuts_model.cpp | 12 ++++++--- .../actions_shortcuts_model.h | 2 +- .../edit_shortcuts_dialog.cpp | 7 +++++ shortcuts_management/edit_shortcuts_dialog.h | 1 + shortcuts_management/shortcuts_manager.cpp | 17 ++++++++++++ shortcuts_management/shortcuts_manager.h | 4 +++ 8 files changed, 78 insertions(+), 14 deletions(-) diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp index e77ded8e..3c16a805 100644 --- a/YACReader/main_window_viewer.cpp +++ b/YACReader/main_window_viewer.cpp @@ -983,14 +983,20 @@ void MainWindowViewer::setUpShortcutsManagement() //actions holder QObject * orphanActions = new QObject; + QList allActions; + QList tmpList; + + editShortcutsDialog->addActionsGroup(tr("Comics"),QIcon(":/images/shortcuts_group_comics.png"), - QList() + tmpList = QList() << openAction << openFolderAction << saveImageAction << openPreviousComicAction << openNextComicAction); + allActions << tmpList; + //keys without actions (General) QAction * toggleFullScreenAction = new QAction(tr("Toggle fullscreen mode"),orphanActions); toggleFullScreenAction->setData(TOGGLE_FULL_SCREEN_ACTION_Y); @@ -1001,7 +1007,7 @@ void MainWindowViewer::setUpShortcutsManagement() toggleToolbarsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)); editShortcutsDialog->addActionsGroup(tr("General"),QIcon(":/images/shortcuts_group_general.png"), - QList() + tmpList = QList() << optionsAction << helpAboutAction << showShorcutsAction @@ -1013,6 +1019,8 @@ void MainWindowViewer::setUpShortcutsManagement() << toggleToolbarsAction << showEditShortcutsAction); + allActions << tmpList; + //keys without actions (MGlass) QAction * sizeUpMglassAction = new QAction(tr("Size up magnifying glass"),orphanActions); sizeUpMglassAction->setData(SIZE_UP_MGLASS_ACTION_Y); @@ -1031,20 +1039,22 @@ void MainWindowViewer::setUpShortcutsManagement() zoomOutMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)); editShortcutsDialog->addActionsGroup(tr("Magnifiying glass"),QIcon(":/images/shortcuts_group_mglass.png"), - QList() + tmpList = QList() << showMagnifyingGlassAction << sizeUpMglassAction << sizeDownMglassAction << zoomInMglassAction << zoomOutMglassAction); + allActions << tmpList; + //keys without actions QAction * toggleFitToScreenAction = new QAction(tr("Toggle between fit to width and fit to height"),orphanActions); toggleFitToScreenAction->setData(CHANGE_FIT_ACTION_Y); toggleFitToScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)); editShortcutsDialog->addActionsGroup(tr("Page adjustement"),QIcon(":/images/shortcuts_group_page.png"), - QList() + tmpList = QList() << adjustHeightAction << adjustWidthAction << toggleFitToScreenAction @@ -1053,6 +1063,8 @@ void MainWindowViewer::setUpShortcutsManagement() << doublePageAction << adjustToFullSizeAction); + allActions << tmpList; + QAction * autoScrollForwardAction = new QAction(tr("Autoscroll down"),orphanActions); autoScrollForwardAction->setData(AUTO_SCROLL_FORWARD_ACTION_Y); autoScrollForwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_FORWARD_ACTION_Y)); @@ -1086,7 +1098,7 @@ void MainWindowViewer::setUpShortcutsManagement() goToLastPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)); editShortcutsDialog->addActionsGroup(tr("Reading"),QIcon(":/images/shortcuts_group_reading.png"), - QList() + tmpList = QList() << nextAction << prevAction << setBookmarkAction @@ -1101,6 +1113,10 @@ void MainWindowViewer::setUpShortcutsManagement() << goToLastPageAction << goToPageAction); + allActions << tmpList; + + ShortcutsManager::getShortcutsManager().registerActions(allActions); + } void MainWindowViewer::changeFit() diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp index 0d2e9a42..dc4a8ea5 100644 --- a/YACReaderLibrary/library_window.cpp +++ b/YACReaderLibrary/library_window.cpp @@ -298,8 +298,11 @@ void LibraryWindow::doDialogs() void LibraryWindow::setUpShortcutsManagement() { + QList allActions; + QList tmpList; + editShortcutsDialog->addActionsGroup("Comics",QIcon(":/images/shortcuts_group_comics.png"), - QList() + tmpList = QList() << openComicAction << setAsReadAction << setAsNonReadAction @@ -311,8 +314,10 @@ void LibraryWindow::setUpShortcutsManagement() << deleteComicsAction << getInfoAction); + allActions << tmpList; + editShortcutsDialog->addActionsGroup("Folders",QIcon(":/images/shortcuts_group_folders.png"), - QList() + tmpList = QList() << setRootIndexAction << expandAllNodesAction << colapseAllNodesAction @@ -321,9 +326,10 @@ void LibraryWindow::setUpShortcutsManagement() << setFolderAsCompletedAction << setFolderAsReadAction << setFolderAsUnreadAction); + allActions << tmpList; editShortcutsDialog->addActionsGroup("General",QIcon(":/images/shortcuts_group_general.png"), - QList() + tmpList = QList() << backAction << forwardAction << helpAboutAction @@ -331,8 +337,10 @@ void LibraryWindow::setUpShortcutsManagement() << serverConfigAction << showEditShortcutsAction); + allActions << tmpList; + editShortcutsDialog->addActionsGroup("Libraries",QIcon(":/images/shortcuts_group_libraries.png"), - QList() + tmpList = QList() << createLibraryAction << openLibraryAction << exportComicsInfoAction @@ -343,13 +351,18 @@ void LibraryWindow::setUpShortcutsManagement() << renameLibraryAction << removeLibraryAction); + allActions << tmpList; + editShortcutsDialog->addActionsGroup("Visualization",QIcon(":/images/shortcuts_group_visualization.png"), - QList() + tmpList = QList() << showHideMarksAction << toggleFullScreenAction << toggleComicsViewAction << hideComicViewAction); + allActions << tmpList; + + ShortcutsManager::getShortcutsManager().registerActions(allActions); } void LibraryWindow::doModels() diff --git a/shortcuts_management/actions_shortcuts_model.cpp b/shortcuts_management/actions_shortcuts_model.cpp index 86c7d7ba..ba9b98bc 100644 --- a/shortcuts_management/actions_shortcuts_model.cpp +++ b/shortcuts_management/actions_shortcuts_model.cpp @@ -80,9 +80,15 @@ bool ActionsShortcutsModel::setData(const QModelIndex &index, const QVariant &va { if(index.column() == KEYS) { - actions[index.row()]->setShortcut(value.toString()); - ShortcutsManager::getShortcutsManager().saveShortcut(actions[index.row()]); - return true; + ShortcutsManager sm = ShortcutsManager::getShortcutsManager(); + if(sm.checkConflicts(value.toString(), actions[index.row()])) + emit conflict(value.toString()); + else + { + actions[index.row()]->setShortcut(value.toString()); + ShortcutsManager::getShortcutsManager().saveShortcut(actions[index.row()]); + return true; + } } return false; } diff --git a/shortcuts_management/actions_shortcuts_model.h b/shortcuts_management/actions_shortcuts_model.h index 58ea16f5..03e3bde6 100644 --- a/shortcuts_management/actions_shortcuts_model.h +++ b/shortcuts_management/actions_shortcuts_model.h @@ -28,7 +28,7 @@ public: KEYS }; signals: - + void conflict(QString); public slots: protected: diff --git a/shortcuts_management/edit_shortcuts_dialog.cpp b/shortcuts_management/edit_shortcuts_dialog.cpp index 5cfd9088..46dea7a6 100644 --- a/shortcuts_management/edit_shortcuts_dialog.cpp +++ b/shortcuts_management/edit_shortcuts_dialog.cpp @@ -11,6 +11,7 @@ #include #include #include +#include #include "QsLog.h" @@ -59,6 +60,7 @@ EditShortcutsDialog::EditShortcutsDialog(QWidget *parent) : connect(resetButton,SIGNAL(clicked()),this,SLOT(resetToDefaults())); connect(actionsGroupsListView->selectionModel(),SIGNAL(currentChanged(QModelIndex,QModelIndex)),this,SLOT(loadShortcuts(QModelIndex,QModelIndex))); //clicked(QModelIndex) doesn't work :S + connect(actionsModel,SIGNAL(conflict(QString)),this,SLOT(processConflict(QString))); #ifdef Q_OS_MAC setFixedSize(760,500); @@ -86,3 +88,8 @@ void EditShortcutsDialog::loadShortcuts(const QModelIndex &mi,const QModelIndex { actionsModel->addActions(groupsModel->getActions(mi)); } + +void EditShortcutsDialog::processConflict(const QString &shortcutInConflict) +{ + QMessageBox::warning(this,tr("Shortcut in use"), QString(tr("The shortcut \"%1\" is already assigned to other function")).arg(shortcutInConflict)); +} diff --git a/shortcuts_management/edit_shortcuts_dialog.h b/shortcuts_management/edit_shortcuts_dialog.h index 0ba29413..e0debf98 100644 --- a/shortcuts_management/edit_shortcuts_dialog.h +++ b/shortcuts_management/edit_shortcuts_dialog.h @@ -21,6 +21,7 @@ signals: public slots: void resetToDefaults(); void loadShortcuts(const QModelIndex & mi,const QModelIndex &mi2); + void processConflict(const QString & shortcutInConflict); protected: QListView * actionsGroupsListView; diff --git a/shortcuts_management/shortcuts_manager.cpp b/shortcuts_management/shortcuts_manager.cpp index d7bb4375..df8d253d 100644 --- a/shortcuts_management/shortcuts_manager.cpp +++ b/shortcuts_management/shortcuts_manager.cpp @@ -93,3 +93,20 @@ void ShortcutsManager::saveShortcut(QAction *action) return s.setValue(action->data().toString() , action->shortcut().toString()); } + +void ShortcutsManager::registerActions(const QList &a) +{ + actions = a; +} + +bool ShortcutsManager::checkConflicts(const QKeySequence & shortcut, const QAction *dest) +{ + foreach(QAction * action, actions) + { + if(action != dest) //if the same shortcut is setted there is no conflict + if(action->shortcut() == shortcut) + return true; + } + + return false; +} diff --git a/shortcuts_management/shortcuts_manager.h b/shortcuts_management/shortcuts_manager.h index 6fd2f5c5..c6e3eae5 100644 --- a/shortcuts_management/shortcuts_manager.h +++ b/shortcuts_management/shortcuts_manager.h @@ -16,6 +16,8 @@ class ShortcutsManager private: ShortcutsManager(); QMap defaultShorcuts; + QList actions; //all actions registered, used for checking conflicts + void initDefaultShorcuts(); public: static ShortcutsManager & getShortcutsManager() @@ -27,6 +29,8 @@ public: void resetToDefaults(); QString getShortcut(const QString & name); void saveShortcut(QAction * action); + void registerActions(const QList & actions); + bool checkConflicts(const QKeySequence &shortcut, const QAction *dest); }; //ACTION NAMES YACReaderLibrary From 61e6478fc2b6638c5f5a750fa1d8b7edee601a43 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Tue, 22 Jul 2014 22:37:17 +0200 Subject: [PATCH 26/34] removed up key for opening comics in YACReaderLibrary //TODO: for some reason the key events are no captured by QTableView (comicsView) so navigation using up/down is not not workig --- YACReaderLibrary/classic_comics_view.cpp | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp index c3ebed37..00d5cfe6 100644 --- a/YACReaderLibrary/classic_comics_view.cpp +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -48,6 +48,7 @@ ClassicComicsView::ClassicComicsView(QWidget *parent) tableView = new YACReaderTableView; tableView->verticalHeader()->hide(); + tableView->setFocusPolicy(Qt::StrongFocus); comicsLayout->addWidget(tableView); comics->setLayout(comicsLayout); sVertical->addWidget(comics); @@ -60,7 +61,7 @@ ClassicComicsView::ClassicComicsView(QWidget *parent) connect(tableView, SIGNAL(clicked(QModelIndex)), this, SLOT(centerComicFlow(QModelIndex))); connect(comicFlow, SIGNAL(centerIndexChanged(int)), this, SLOT(updateTableView(int))); connect(tableView, SIGNAL(comicRated(int,QModelIndex)), this, SIGNAL(comicRated(int,QModelIndex))); - connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); + //connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); connect(tableView->horizontalHeader(), SIGNAL(sectionMoved(int,int,int)), this, SLOT(saveTableHeadersStatus())); layout->addWidget(sVertical); @@ -116,13 +117,13 @@ void ClassicComicsView::setModel(TableModel *model) //debido a un bug, qt4 no es capaz de ajustar el ancho teniendo en cuenta todas la filas (no sólo las visibles) //así que se ecala la primera vez y después se deja el control al usuario. //if(!settings->contains(COMICS_VIEW_HEADERS)) - tableView->resizeColumnsToContents(); + tableView->resizeColumnsToContents(); tableView->horizontalHeader()->setStretchLastSection(true); QStringList paths = model->getPaths(model->getCurrentPath());//TODO ComicsView: get currentpath from somewhere currentPath()); comicFlow->setImagePaths(paths); comicFlow->setMarks(model->getReadList()); - comicFlow->setFocus(Qt::OtherFocusReason); + //comicFlow->setFocus(Qt::OtherFocusReason); } if(settings->contains(COMICS_VIEW_HEADERS)) From 39d7ae6ce548ca14ade836fc88426c915b937472 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Tue, 22 Jul 2014 22:46:17 +0200 Subject: [PATCH 27/34] fixed double click for opening a comic --- YACReaderLibrary/classic_comics_view.cpp | 2 +- common/yacreader_flow_gl.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp index 00d5cfe6..48444658 100644 --- a/YACReaderLibrary/classic_comics_view.cpp +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -61,7 +61,7 @@ ClassicComicsView::ClassicComicsView(QWidget *parent) connect(tableView, SIGNAL(clicked(QModelIndex)), this, SLOT(centerComicFlow(QModelIndex))); connect(comicFlow, SIGNAL(centerIndexChanged(int)), this, SLOT(updateTableView(int))); connect(tableView, SIGNAL(comicRated(int,QModelIndex)), this, SIGNAL(comicRated(int,QModelIndex))); - //connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); + connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); connect(tableView->horizontalHeader(), SIGNAL(sectionMoved(int,int,int)), this, SLOT(saveTableHeadersStatus())); layout->addWidget(sVertical); diff --git a/common/yacreader_flow_gl.cpp b/common/yacreader_flow_gl.cpp index 58588251..5ce9e1dc 100644 --- a/common/yacreader_flow_gl.cpp +++ b/common/yacreader_flow_gl.cpp @@ -1059,7 +1059,7 @@ void YACReaderFlowGL::keyPressEvent(QKeyEvent *event) if(event->key() == Qt::Key_Up) { - emit selected(centerIndex()); + //emit selected(centerIndex()); return; } From 7020a48a2e4efcb6a8cbbf5eab5d111c495a1dc9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Tue, 29 Jul 2014 17:40:00 +0200 Subject: [PATCH 28/34] updated web library with new style and features added remote reading from iOS --- .../server/controllers/covercontroller.cpp | 22 +- .../server/controllers/foldercontroller.cpp | 122 ++-- .../controllers/librariescontroller.cpp | 73 ++- .../server/lib/bfHttpServer/httprequest.cpp | 4 +- .../server/lib/bfHttpServer/httpsession.cpp | 33 ++ .../server/lib/bfHttpServer/httpsession.h | 7 + .../lib/bfHttpServer/httpsessionstore.cpp | 2 +- .../lib/bfHttpServer/staticfilecontroller.cpp | 27 +- .../lib/bfHttpServer/staticfilecontroller.h | 1 + YACReaderLibrary/server/requestmapper.cpp | 18 +- release/server/docroot/css/styles_ipad.css | 515 ++++++++++------- release/server/docroot/css/styles_iphone.css | 539 +++++++++++------- .../images/bottomboxLibraries_ipad.jpg | Bin 1491 -> 0 bytes .../server/docroot/images/bottombox_ipad.jpg | Bin 2459 -> 0 bytes .../docroot/images/bottombox_iphone.jpg | Bin 1491 -> 0 bytes release/server/docroot/images/browse.png | Bin 0 -> 134 bytes release/server/docroot/images/browse@2x.png | Bin 0 -> 185 bytes .../server/docroot/images/bt_browse_ipad.jpg | Bin 716 -> 0 bytes .../docroot/images/bt_browse_iphone.jpg | Bin 701 -> 0 bytes .../server/docroot/images/bt_current_ipad.jpg | Bin 1113 -> 0 bytes .../docroot/images/bt_current_iphone.jpg | Bin 1113 -> 0 bytes .../server/docroot/images/bt_first_ipad.jpg | Bin 1651 -> 0 bytes .../server/docroot/images/bt_first_iphone.jpg | Bin 1651 -> 0 bytes .../server/docroot/images/bt_import_ipad.jpg | Bin 991 -> 0 bytes .../docroot/images/bt_import_iphone.jpg | Bin 957 -> 0 bytes .../docroot/images/bt_imported_ipad.jpg | Bin 738 -> 0 bytes .../docroot/images/bt_imported_iphone.jpg | Bin 727 -> 0 bytes .../server/docroot/images/bt_index_ipad.jpg | Bin 1149 -> 0 bytes .../server/docroot/images/bt_index_iphone.jpg | Bin 1149 -> 0 bytes .../server/docroot/images/bt_last_ipad.jpg | Bin 1665 -> 0 bytes .../server/docroot/images/bt_last_iphone.jpg | Bin 1665 -> 0 bytes .../docroot/images/bt_libraries_ipad.jpg | Bin 763 -> 0 bytes .../docroot/images/bt_libraries_iphone.jpg | Bin 763 -> 0 bytes .../docroot/images/bt_library_iphone.jpg | Bin 3317 -> 0 bytes .../server/docroot/images/bt_login_ipad.jpg | Bin 928 -> 0 bytes .../server/docroot/images/bt_login_iphone.jpg | Bin 928 -> 0 bytes .../server/docroot/images/bt_next_ipad.jpg | Bin 1560 -> 0 bytes .../server/docroot/images/bt_next_iphone.jpg | Bin 1560 -> 0 bytes .../docroot/images/bt_previous_ipad.jpg | Bin 1586 -> 0 bytes .../docroot/images/bt_previous_iphone.jpg | Bin 1586 -> 0 bytes .../server/docroot/images/bt_up_iphone.jpg | Bin 720 -> 0 bytes release/server/docroot/images/combo.png | Bin 0 -> 120 bytes release/server/docroot/images/combo@2x.png | Bin 0 -> 167 bytes release/server/docroot/images/download.png | Bin 0 -> 155 bytes release/server/docroot/images/download@2x.png | Bin 0 -> 203 bytes release/server/docroot/images/f.jpg | Bin 2802 -> 0 bytes release/server/docroot/images/f.png | Bin 7630 -> 0 bytes release/server/docroot/images/f@2x.png | Bin 0 -> 1262 bytes .../server/docroot/images/fnd_input_pass.jpg | Bin 546 -> 0 bytes .../docroot/images/fnd_input_username.jpg | Bin 604 -> 0 bytes release/server/docroot/images/fnd_inputs.jpg | Bin 1443 -> 0 bytes release/server/docroot/images/imported.png | Bin 0 -> 158 bytes release/server/docroot/images/imported@2x.png | Bin 0 -> 214 bytes release/server/docroot/images/indicator.png | Bin 0 -> 118 bytes .../server/docroot/images/indicator@2x.png | Bin 0 -> 220 bytes release/server/docroot/images/library.png | Bin 0 -> 201 bytes release/server/docroot/images/library@2x.png | Bin 0 -> 284 bytes release/server/docroot/images/next.png | Bin 0 -> 137 bytes release/server/docroot/images/next@2x.png | Bin 0 -> 339 bytes release/server/docroot/images/prev.png | Bin 0 -> 154 bytes release/server/docroot/images/prev@2x.png | Bin 0 -> 345 bytes release/server/docroot/images/read.png | Bin 0 -> 152 bytes release/server/docroot/images/read@2x.png | Bin 0 -> 201 bytes release/server/docroot/images/readMark.png | Bin 0 -> 196 bytes release/server/docroot/images/readMark@2x.png | Bin 0 -> 296 bytes release/server/docroot/images/readingMark.png | Bin 0 -> 206 bytes .../server/docroot/images/readingMark@2x.png | Bin 0 -> 296 bytes .../docroot/images/topboxLibraries_ipad.jpg | Bin 423 -> 0 bytes release/server/docroot/images/topbox_ipad.jpg | Bin 1480 -> 0 bytes .../server/docroot/images/topbox_iphone.jpg | Bin 423 -> 0 bytes release/server/docroot/images/up.png | Bin 0 -> 163 bytes release/server/docroot/images/up@2x.png | Bin 0 -> 271 bytes release/server/templates/folder_ipad.tpl | 155 +++-- release/server/templates/folder_iphone.tpl | 155 +++-- release/server/templates/libraries_ipad.tpl | 14 +- release/server/templates/libraries_iphone.tpl | 18 +- 76 files changed, 1077 insertions(+), 628 deletions(-) delete mode 100644 release/server/docroot/images/bottomboxLibraries_ipad.jpg delete mode 100644 release/server/docroot/images/bottombox_ipad.jpg delete mode 100644 release/server/docroot/images/bottombox_iphone.jpg create mode 100644 release/server/docroot/images/browse.png create mode 100644 release/server/docroot/images/browse@2x.png delete mode 100644 release/server/docroot/images/bt_browse_ipad.jpg delete mode 100644 release/server/docroot/images/bt_browse_iphone.jpg delete mode 100644 release/server/docroot/images/bt_current_ipad.jpg delete mode 100644 release/server/docroot/images/bt_current_iphone.jpg delete mode 100644 release/server/docroot/images/bt_first_ipad.jpg delete mode 100644 release/server/docroot/images/bt_first_iphone.jpg delete mode 100644 release/server/docroot/images/bt_import_ipad.jpg delete mode 100644 release/server/docroot/images/bt_import_iphone.jpg delete mode 100644 release/server/docroot/images/bt_imported_ipad.jpg delete mode 100644 release/server/docroot/images/bt_imported_iphone.jpg delete mode 100644 release/server/docroot/images/bt_index_ipad.jpg delete mode 100644 release/server/docroot/images/bt_index_iphone.jpg delete mode 100644 release/server/docroot/images/bt_last_ipad.jpg delete mode 100644 release/server/docroot/images/bt_last_iphone.jpg delete mode 100644 release/server/docroot/images/bt_libraries_ipad.jpg delete mode 100644 release/server/docroot/images/bt_libraries_iphone.jpg delete mode 100644 release/server/docroot/images/bt_library_iphone.jpg delete mode 100644 release/server/docroot/images/bt_login_ipad.jpg delete mode 100644 release/server/docroot/images/bt_login_iphone.jpg delete mode 100644 release/server/docroot/images/bt_next_ipad.jpg delete mode 100644 release/server/docroot/images/bt_next_iphone.jpg delete mode 100644 release/server/docroot/images/bt_previous_ipad.jpg delete mode 100644 release/server/docroot/images/bt_previous_iphone.jpg delete mode 100644 release/server/docroot/images/bt_up_iphone.jpg create mode 100644 release/server/docroot/images/combo.png create mode 100644 release/server/docroot/images/combo@2x.png create mode 100644 release/server/docroot/images/download.png create mode 100644 release/server/docroot/images/download@2x.png delete mode 100644 release/server/docroot/images/f.jpg delete mode 100644 release/server/docroot/images/f.png create mode 100644 release/server/docroot/images/f@2x.png delete mode 100644 release/server/docroot/images/fnd_input_pass.jpg delete mode 100644 release/server/docroot/images/fnd_input_username.jpg delete mode 100644 release/server/docroot/images/fnd_inputs.jpg create mode 100644 release/server/docroot/images/imported.png create mode 100644 release/server/docroot/images/imported@2x.png create mode 100644 release/server/docroot/images/indicator.png create mode 100644 release/server/docroot/images/indicator@2x.png create mode 100644 release/server/docroot/images/library.png create mode 100644 release/server/docroot/images/library@2x.png create mode 100644 release/server/docroot/images/next.png create mode 100644 release/server/docroot/images/next@2x.png create mode 100644 release/server/docroot/images/prev.png create mode 100644 release/server/docroot/images/prev@2x.png create mode 100644 release/server/docroot/images/read.png create mode 100644 release/server/docroot/images/read@2x.png create mode 100644 release/server/docroot/images/readMark.png create mode 100644 release/server/docroot/images/readMark@2x.png create mode 100644 release/server/docroot/images/readingMark.png create mode 100644 release/server/docroot/images/readingMark@2x.png delete mode 100644 release/server/docroot/images/topboxLibraries_ipad.jpg delete mode 100644 release/server/docroot/images/topbox_ipad.jpg delete mode 100644 release/server/docroot/images/topbox_iphone.jpg create mode 100644 release/server/docroot/images/up.png create mode 100644 release/server/docroot/images/up@2x.png diff --git a/YACReaderLibrary/server/controllers/covercontroller.cpp b/YACReaderLibrary/server/controllers/covercontroller.cpp index 4614ce59..80cc394c 100644 --- a/YACReaderLibrary/server/controllers/covercontroller.cpp +++ b/YACReaderLibrary/server/controllers/covercontroller.cpp @@ -44,14 +44,28 @@ void CoverController::service(HttpRequest& request, HttpResponse& response) QImage img(libraries.getPath(libraryName)+"/.yacreaderlibrary/covers/"+fileName); if (!img.isNull()) { - int width = 80; - if(session.getDisplayType()=="retina") + int width = 80, height = 120; + if(session.getDisplayType()=="@2x") + { width = 160; - img = img.scaledToWidth(width,Qt::SmoothTransformation); + height = 240; + } + + if(float(img.width())/img.height() < 0.66666) + img = img.scaledToWidth(width,Qt::SmoothTransformation); + else + img = img.scaledToHeight(height,Qt::SmoothTransformation); + + QImage destImg(width,height,QImage::Format_RGB32); + destImg.fill(Qt::black); + QPainter p(&destImg); + + p.drawImage((width-img.width())/2,(height-img.height())/2,img); + QByteArray ba; QBuffer buffer(&ba); buffer.open(QIODevice::WriteOnly); - img.save(&buffer, "JPG"); + destImg.save(&buffer, "JPG"); response.write(ba,true); } //DONE else, hay que devolver un 404 diff --git a/YACReaderLibrary/server/controllers/foldercontroller.cpp b/YACReaderLibrary/server/controllers/foldercontroller.cpp index d8ef8591..7300adf9 100644 --- a/YACReaderLibrary/server/controllers/foldercontroller.cpp +++ b/YACReaderLibrary/server/controllers/foldercontroller.cpp @@ -1,4 +1,6 @@ #include "foldercontroller.h" +#include "controllers/errorcontroller.h" + #include "db_helper.h" //get libraries #include "comic_db.h" @@ -36,7 +38,16 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) int libraryId = pathElements.at(2).toInt(); QString libraryName = DBHelper::getLibraryName(libraryId); qulonglong parentId = pathElements.at(4).toULongLong(); + + parentId = qMax(1,parentId); + QString folderName = DBHelper::getFolderName(libraryName,parentId); + if(folderName.isEmpty()) + { + ErrorController(300).service(request,response); + return; + } + if(parentId!=1) t.setVariable("folder.name",folderName); else @@ -70,7 +81,21 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) int upPage = 0; - + if(parentId == 1) + session.clearFoldersPath(); + else + { + if(fromUp) + session.popFolder(); + else + if(session.getFoldersPath().contains(parentId)) + { + while(session.topFolder()!=parentId) + session.popFolder(); + } + else + session.pushFolder(parentId); + } if(backId == 1 && parentId == 1) { @@ -83,7 +108,7 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) if(fromUp) { session.popPage(); - upPage = session.topPage(); + upPage = session.topPage(); page = upPage; } else //este nivel puede haberse cargado por primera vez ó puede que estemos navegando horizontalmente @@ -116,7 +141,7 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) //t.loop("element",folderContent.length()); - int elementsPerPage = 18; + int elementsPerPage = 24; int numFolders = folderContent.length(); //int numComics = folderComics.length(); @@ -135,9 +160,16 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) int indexCurrentPage = page*elementsPerPage; int numFoldersAtCurrentPage = qMax(0,qMin(numFolders - indexCurrentPage, elementsPerPage)); - //response.writeText(QString("indexCurrentPage : %1
").arg(indexCurrentPage)); - //response.writeText(QString("numFoldersAtCurrentPage : %1
").arg(numFoldersAtCurrentPage)); - //response.writeText(QString("foldersLength : %1
").arg(folderContent.length())); + //PATH + QStack foldersPath = session.getFoldersPath(); + t.setVariable(QString("library.name"),libraryName); + t.setVariable(QString("library.url"),QString("/library/%1/folder/1").arg(libraryId)); + t.loop("path",foldersPath.length()); + for(int i = 0; i < foldersPath.length(); i++){ + + t.setVariable(QString("path%1.url").arg(i),QString("/library/%1/folder/%2").arg(libraryId).arg(foldersPath[i])); + t.setVariable(QString("path%1.name").arg(i),DBHelper::getFolderName(libraryName,foldersPath[i])); + } t.loop("element",numFoldersAtCurrentPage); int i = 0; @@ -148,7 +180,6 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) if(item->isDir()) { t.setVariable(QString("element%1.class").arg(i),"folder"); - t.setVariable(QString("element%1.image.width").arg(i),"89px"); t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); t.setVariable(QString("element%1.browse").arg(i),QString("
browse").arg(QString("/library/%1/folder/%2").arg(libraryId).arg(item->id))); @@ -157,13 +188,17 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) //t.setVariable(QString("element%1.downloadurl").arg(i),"/library/"+libraryName+"/folder/"+QString("%1/info").arg(folderContent.at(i + (page*elementsPerPage))->id)); t.setVariable(QString("element%1.download").arg(i),QString("import").arg("/library/"+QString::number(libraryId)+"/folder/"+QString("%1/info").arg(folderContent.at(i + (page*elementsPerPage))->id))); - } + t.setVariable(QString("element%1.read").arg(i),""); + + t.setVariable(QString("element%1.size").arg(i),""); + t.setVariable(QString("element%1.pages").arg(i),""); + t.setVariable(QString("element%1.status").arg(i),""); + } else { t.setVariable(QString("element%1.class").arg(i),"cover"); const ComicDB * comic = (ComicDB *)item; t.setVariable(QString("element%1.browse").arg(i),""); - t.setVariable(QString("element%1.image.width").arg(i),"80px"); //t.setVariable(QString("element%1.downloadurl").arg(i),"/library/"+libraryName+"/comic/"+QString("%1").arg(comic->id)); if(!session.isComicOnDevice(comic->info.hash) && !session.isComicDownloaded(comic->info.hash)) t.setVariable(QString("element%1.download").arg(i),QString("import").arg("/library/"+QString::number(libraryId)+"/comic/"+QString("%1").arg(comic->id))); @@ -174,56 +209,28 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) //t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); + t.setVariable(QString("element%1.read").arg(i),QString("read").arg("/library/"+QString::number(libraryId)+"/comic/"+QString("%1").arg(comic->id)+"/remote")); + t.setVariable(QString("element%1.image.url").arg(i),QString("/library/%1/cover/%2.jpg").arg(libraryId).arg(comic->info.hash)); + + t.setVariable(QString("element%1.size").arg(i),"" + QString::number(comic->info.hash.right(comic->info.hash.length()-40).toInt()/1024.0/1024.0,'f',2)+"Mb"); + if(comic->info.hasBeenOpened) + t.setVariable(QString("element%1.pages").arg(i),QString("%1/%2 pages").arg(comic->info.currentPage).arg(comic->info.numPages.toInt())); + else + t.setVariable(QString("element%1.pages").arg(i),QString("%1").arg(comic->info.numPages.toInt())); + + if(comic->info.read) + t.setVariable(QString("element%1.status").arg(i), QString("
")); + else if(comic->info.hasBeenOpened) + t.setVariable(QString("element%1.status").arg(i), QString("
")); + else + t.setVariable(QString("element%1.status").arg(i),""); + + } i++; } - //int comicsOffset;// = qMax(0,((page - (numFolderPages - 1)) * 10) - (numFolders%10)); - - //int comicPage = numFolderPages!=0?page-(numFolderPages - 1):page; - - //if(comicPage > 0) - //{ - // comicsOffset = elementsPerPage - (numFolders%elementsPerPage); - // comicsOffset += (comicPage-1) *elementsPerPage; - //} - //else - // comicsOffset = 0; - - // - - //int globalComicsOffset = elementsPerPage - (numFolders%elementsPerPage); - //int numComicsAtCurrentPage = 0; - - //if(comicPage == 0) //primera página de los cómics - // numComicsAtCurrentPage = qMin(globalComicsOffset,numComics); - // else if (page == (numPages-1)) //última página de los cómics - // numComicsAtCurrentPage = elementsPerPage-globalComicsOffset + (numComics%elementsPerPage); - // else - // numComicsAtCurrentPage = elementsPerPage - numFoldersAtCurrentPage; - - //if(numComics == 0) - // numComicsAtCurrentPage = 0; - ////response.writeText(QString("numComicsAtCurrentPage : %1
").arg(numComicsAtCurrentPage)); - ////response.writeText(QString("comicsOffset : %1
").arg(comicsOffset)); - - //t.loop("elementcomic",numComicsAtCurrentPage); - //// - //int j = 0; - - //while(jinfo.title == 0 || comic->info.title->isEmpty()) - // t.setVariable(QString("elementcomic%1.name").arg(j),comic->name); - // //else - // // t.setVariable(QString("elementcomic%1.name").arg(i),*comic->info.title); - // t.setVariable(QString("elementcomic%1.url").arg(j),"/library/"+QUrl::toPercentEncoding(libraryName)+"/comic/"+QString("%1").arg(comic->id)); - // t.setVariable(QString("elementcomic%1.coverulr").arg(j),"/library/"+QUrl::toPercentEncoding(libraryName)+"/cover/"+QString("%1").arg(comic->info.hash + ".jpg")); - // j++; - //} - if(numPages > 1) { t.setCondition("pageIndex",true); @@ -267,7 +274,6 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) i++; count += indexCount.value(*itr); indexPage = count/elementsPerPage; - } } else @@ -295,12 +301,14 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) t.setVariable("page.previous",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg((page==0)?page:page-1)); t.setVariable("page.next",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg((page==numPages-1)?page:page+1)); t.setVariable("page.last",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg(numPages-1)); - + t.setCondition("index", true); } else { + t.loop("page",0); t.loop("index",0); + t.setCondition("index", false); t.setCondition("pageIndex",false); t.setCondition("alphaIndex",false); } @@ -310,4 +318,4 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) response.write(t.toLatin1(),true); -} \ No newline at end of file +} diff --git a/YACReaderLibrary/server/controllers/librariescontroller.cpp b/YACReaderLibrary/server/controllers/librariescontroller.cpp index fc6472f0..a154b7f8 100644 --- a/YACReaderLibrary/server/controllers/librariescontroller.cpp +++ b/YACReaderLibrary/server/controllers/librariescontroller.cpp @@ -10,32 +10,53 @@ LibrariesController::LibrariesController() {} void LibrariesController::service(HttpRequest& request, HttpResponse& response) { - HttpSession session=Static::sessionStore->getSession(request,response); - session.set("ySession","ok"); - session.clearNavigationPath(); - response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); - response.setHeader("Connection","close"); - - - QString postData = QString::fromUtf8(request.getBody()); - //response.writeText(postData); + HttpSession session=Static::sessionStore->getSession(request,response); + if(session.contains("ySession")) //session is already alive check if it is needed to update comics + { + QString postData = QString::fromUtf8(request.getBody()); + if(postData.length()>0) { + QList data = postData.split("\n"); + if(data.length() > 2) { + //ONLY COMICS ARE UPDATED, DEVICE CHARACTERISTICS ARE INMUTABLE + QList comics = data.at(2).split(":").at(1).split("\t"); + foreach(QString hash,comics) { + session.setComicOnDevice(hash); + } + } + } + } + else + { + session.set("ySession","ok"); - QList data = postData.split("\n"); - if(data.length() > 2) - { - session.setDeviceType(data.at(0).split(":").at(1)); - session.setDisplayType(data.at(1).split(":").at(1)); - QList comics = data.at(2).split(":").at(1).split("\t"); - foreach(QString hash,comics) - { - session.setComicOnDevice(hash); - } - } - else //valores por defecto, con propositos de depuración - { - session.setDeviceType("ipad"); - session.setDisplayType("retina"); - } + session.clearNavigationPath(); + session.clearFoldersPath(); + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.setHeader("Connection","close"); + + + QString postData = QString::fromUtf8(request.getBody()); + //response.writeText(postData); + + QList data = postData.split("\n"); + if(data.length() > 2) + { + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); + QList comics = data.at(2).split(":").at(1).split("\t"); + foreach(QString hash,comics) + { + session.setComicOnDevice(hash); + } + } + else //values by default, only for debug purposes. + { + session.setDeviceType("iphone"); + session.setDisplayType("@2x"); + } + + } Template t=Static::templateLoader->getTemplate("libraries_"+session.getDeviceType(),request.getHeader("Accept-Language")); t.enableWarnings(); @@ -54,6 +75,6 @@ void LibrariesController::service(HttpRequest& request, HttpResponse& response) i++; } + response.setStatus(200,"OK"); response.write(t.toLatin1(),true); - } diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp index b91cb817..8ed427b5 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp +++ b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp @@ -12,8 +12,8 @@ HttpRequest::HttpRequest(QSettings* settings) { status=waitForRequest; currentSize=0; expectedBodySize=0; - maxSize=settings->value("maxRequestSize","16000000").toInt(); - maxMultiPartSize=settings->value("maxMultiPartSize","1000000").toInt(); + maxSize=settings->value("maxRequestSize","32000000").toInt(); + maxMultiPartSize=settings->value("maxMultiPartSize","32000000").toInt(); } void HttpRequest::readRequest(QTcpSocket& socket) { diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp index 10e30596..3f1cbd21 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp @@ -307,3 +307,36 @@ int HttpSession::topPage() return dataPtr->yacreaderSessionData.navigationPath.top(); return 0; } + +void HttpSession::clearFoldersPath() +{ + if(dataPtr) + dataPtr->yacreaderSessionData.foldersPath.clear(); +} + +int HttpSession::popFolder() +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.foldersPath.isEmpty())) + return dataPtr->yacreaderSessionData.foldersPath.pop(); + return 0; +} + +void HttpSession::pushFolder(int page) +{ + if(dataPtr) + dataPtr->yacreaderSessionData.foldersPath.push(page); +} + +int HttpSession::topFolder() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.foldersPath.top(); + return 0; +} + +QStack HttpSession::getFoldersPath() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.foldersPath; + return QStack(); +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h index 829e077a..d2e6c6b7 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h @@ -119,6 +119,12 @@ public: void pushPage(int page); int topPage(); + void clearFoldersPath(); + int popFolder(); + void pushFolder(int page); + int topFolder(); + QStack getFoldersPath(); + private: struct YACReaderSessionData { @@ -132,6 +138,7 @@ private: qulonglong comicId; QStack navigationPath; + QStack foldersPath; Comic * comic; }; diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp index b37ee1d5..af7dc50c 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp @@ -14,7 +14,7 @@ HttpSessionStore::HttpSessionStore(QSettings* settings, QObject* parent) connect(&cleanupTimer,SIGNAL(timeout()),this,SLOT(timerEvent())); cleanupTimer.start(60000); cookieName=settings->value("cookieName","sessionid").toByteArray(); - expirationTime=settings->value("expirationTime",18000000).toInt(); + expirationTime=settings->value("expirationTime",86400000).toInt(); qDebug("HttpSessionStore: Sessions expire after %i milliseconds",expirationTime); } diff --git a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp index e01a39b7..fde1cf8d 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp +++ b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp @@ -56,7 +56,11 @@ void StaticFileController::service(HttpRequest& request, HttpResponse& response) stringPath.remove(fileName); HttpSession session=Static::sessionStore->getSession(request,response,false); QString device = session.getDeviceType(); - fileName = getDeviceAwareFileName(fileName, device, request.getHeader("Accept-Language"), stringPath); + QString display = session.getDisplayType(); + if(fileName.endsWith(".png")) + fileName = getDeviceAwareFileName(fileName, device, display, request.getHeader("Accept-Language"), stringPath); + else + fileName = getDeviceAwareFileName(fileName, device, request.getHeader("Accept-Language"), stringPath); QString newPath = stringPath.append(fileName); path = newPath.toLocal8Bit(); @@ -207,5 +211,24 @@ QString StaticFileController::getDeviceAwareFileName(QString fileName, QString d if(QFile(docroot+"/"+path+completeFileName).exists()) return completeFileName; //existe un archivo específico para este dispositivo y locales else - return getLocalizedFileName(fileName,locales,path); //no hay archivo específico para el dispositivo, pero puede haberlo para estas locales + return getLocalizedFileName(fileName,locales,path); //no hay archivo específico para el dispositivo, pero puede haberlo para estas locales +} + +QString StaticFileController::getDeviceAwareFileName(QString fileName, QString device, QString display, QString locales, QString path) const +{ + QFileInfo fi(fileName); + QString baseName = fi.baseName(); + QString extension = fi.completeSuffix(); + + QString completeFileName = completeFileName = baseName+display+"."+extension; + if(QFile(docroot+"/"+path+completeFileName).exists()) + return completeFileName; + else + { + completeFileName = baseName+"_"+device+display+"."+extension; + if((QFile(docroot+"/"+path+completeFileName).exists())) + return completeFileName; + } + + return fileName; } diff --git a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h index 1cbbdbf7..26413398 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h +++ b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h @@ -84,6 +84,7 @@ private: QString getLocalizedFileName(QString fileName, QString locales, QString path) const; QString getDeviceAwareFileName(QString fileName, QString device, QString locales, QString path) const; + QString getDeviceAwareFileName(QString fileName, QString device, QString display, QString locales, QString path) const; bool exists(QString localizedName, QString path) const; }; diff --git a/YACReaderLibrary/server/requestmapper.cpp b/YACReaderLibrary/server/requestmapper.cpp index 303ee8e0..5bbd9199 100644 --- a/YACReaderLibrary/server/requestmapper.cpp +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -50,10 +50,10 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { else { - //se comprueba que la sesión sea la correcta con el fin de evitar accesos no autorizados - HttpSession session=Static::sessionStore->getSession(request,response,false); - if(!session.isNull() && session.contains("ySession")) - { + //se comprueba que la sesión sea la correcta con el fin de evitar accesos no autorizados + HttpSession session=Static::sessionStore->getSession(request,response,false); + if(!session.isNull() && session.contains("ySession")) + { if(library.indexIn(path)!=-1 && DBHelper::getLibraries().contains(library.cap(1).toInt()) ) { //listar el contenido del folder @@ -83,11 +83,11 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { //response.writeText(library.cap(1)); Static::staticFileController->service(request, response); } - } - else //acceso no autorizado, redirección - { - ErrorController(300).service(request,response); - } + } + else //acceso no autorizado, redirección + { + ErrorController(300).service(request,response); + } } } diff --git a/release/server/docroot/css/styles_ipad.css b/release/server/docroot/css/styles_ipad.css index b54e9f71..b89a6bf8 100644 --- a/release/server/docroot/css/styles_ipad.css +++ b/release/server/docroot/css/styles_ipad.css @@ -1,246 +1,364 @@ body{ - background-color: #363636; + background-color: #F5F5F5; font-family: Arial, Helvetica, sans-serif; } -#content{ - width: 712px; - background: url("../images/topbox.jpg") no-repeat scroll 0 0 #FFF; - margin: 17px auto 0; - padding: 22px 0 22px 22px; - -} - +/* libraries */ #contentLibraries{ - width: 256px; - background: url("../images/topboxLibraries.jpg") no-repeat scroll 0 0 #FFF; - margin: 16px auto 0; - padding: 16px; + width: 300px; + border: 1px solid #C6C6C6; + background-color: white; + margin-left: auto; + margin-right: auto; + margin-top: 9px; } -.sombraLibraries{ - background: url("../images/bottomboxLibraries.jpg") no-repeat scroll bottom center #FFF; - width: 288px; - height: 14px; - margin: 0 auto; +#contentLibraries .library-icon +{ + float: left; + background-color: white; + height: 18px; + padding: 11px 19px 10px 19px; + display:block; } -.index{ - width: 718px; - background: url("../images/topbox.jpg") no-repeat scroll 0 0 #FFF; - margin: 7px auto 0; - padding: 16px 0px 0px 16px; +#contentLibraries li +{ + border-bottom: 1px solid #e2e2e2; + position: relative; + list-style: none; } +#contentLibraries li:last-child +{ + border: none; +} + +#contentLibraries .library-link +{ + width: 211px; + height: 28px; + border: none; + padding: 11px 0 0 0px; + background-color: white; + display: block; + float:left; + font-family: Arial; + font-size: 16px; + text-decoration: none; + color: #525252 ; +} + + #contentLibraries a +{ + position: absolute; + height: 39px; + width: 100%; + z-index: 10; + display: block; + top 0; + text-decoration: none; +} + +#contentLibraries .library-indicator +{ + float: left; + background-color: white; + height: 8px; + padding: 16px 16px 15px 16px; + display:block; +} + + +#content h1 +{ + color: #292929; + text-align: center; + font-size: 21px; +} + +#contentLibraries h1{ + color: #292929; + text-align: center; + border-bottom: 1px solid #C6C6C6; + font-size: 21px; + padding: 15px 0 16px 0; +} + +#folder-header +{ + position: fixed; + width: 100%; + height: 88px; + background-color: rgba(255,255,255,0.9); + border-bottom: 1px solid #C6C6C6; + z-index: 999; +} + +#folder-subheader1 +{ + width: 100%; + height: 40px; + margin-top: 18px; + +} + +#folder-subheader2 +{ + width: 100%; + padding-left: 16px; +} + +#topIndex +{ + position: absolute; + left: 16px; + top: 19px; +} + +#topIndex a +{ + float: left; +} + +.indicator { + margin: 0 9px; +} + +.path { + text-decoration: none; + color: #5C5C5C; + font-family: Arial, Helvetica; + font-size: 15px; + +} + +#header-combos +{ + position: absolute; + right: 15px; + top: 15px; + color: #a3a3a3; + width: 160px; +} + +#topIndex .next{ + width: 25px; + height: 19px; + border: none; + margin: 0 21px 4px 0; + padding: 5px 0 0 0; + display: block; + background: url("/images/next.png") no-repeat scroll 0 0 transparent; + background-size: 25px 19px; + padding: 0; + text-indent: -99999px; +} + +#topIndex .previous{ + width: 25px; + height: 19px; + border: none; + margin: 0 14px 4px 0; + padding: 5px 0 0 0; + display: block; + background: url("/images/prev.png") no-repeat scroll 0 0 transparent; + background-size: 25px 19px; + padding: 0; + text-indent: -99999px; +} + +#topIndex .up{ + width: 15px; + height: 19px; + border: none; + background: url("/images/up.png") no-repeat scroll 0 0 transparent; + background-size: 15px 19px; + color: #FFF; + display: block; + text-indent: -99999px; +} + +#itemContainer li +{ +float: left; +width: 242px; +height: 120px; +border: 1px solid #E2E2E2; +margin: 9px 9px 0px 0; +background-color: white; +overflow: hidden; +position: relative; +} + +.folderContent +{ + padding-top: 90px; + padding-left: 9px; +} +/* hasta aquí */ + .folder { float: left; + } .cover { float: left; +overflow: hidden; } -.cover img +.mark { - -webkit-box-shadow: 0px 0px 5px #333; + position: absolute; + top: 0px; + margin-left: 55px; } - .info { padding: 8px 0px 0px 0px; float: left; position: relative; height: 115px; -width: 135px; +width: 158px; } .buttons { -position:absolute; -bottom:0px; -left:0px; + position:absolute; + bottom:0px; + left:0px; + border-top: 1px solid #e2e2e2; + padding-top: 3px; + height: 25px; + width: 162px; + font-family: Arial; + color: #6e6e6e; + font-size: 10px; } -#itemContainer li +.elementInfo { -float: left; -width: 225px; -height: 120px; -margin: 20px 12px 20px 0; + position:absolute; + bottom:24px; + padding-top: 3px; + height: 25px; + width: 162px; + font-family: Arial; + color: #adadad; + font-size: 10px; } +.numPages +{ + float: left; + padding-left:8px; +} + +.comicSize +{ + float: right; + padding-right: 9px; +} + + #itemContainer a { text-decoration: none; - font-family: Arial; - color: #FFF; - font-size: 13px; + } .browseButton { - width: 57px; - height: 22px; - background: url("../images/bt_browse.jpg") no-repeat scroll 0 0 transparent; + width: 60px; + background: url("/images/browse.png") no-repeat scroll 0 0 transparent; + background-position: 16px 5px; + background-size: 7px 7px; border: none; - text-align:center; + text-align:right; display: block; - float: left; - padding: 4px 0 0 0; - margin: 0 0 0 4px; + float: right; + padding: 4px 10px 0 0; + color: #6e6e6e; } .importButton { - width: 57px; - height: 18px; - background: url("../images/bt_import.jpg") no-repeat scroll 0 0 transparent; + width: 60px; + background: url("/images/download.png") no-repeat scroll 0 0 transparent; + background-position: 3px 5px; + background-size: 7px 8px; border: none; - text-align:center; + text-align:left; display: block; float: left; margin: 0 0 0 4px; - padding: 4px 0 0 0; + padding: 4px 0 0 16px; + color: #6e6e6e; +} + +.readButton +{ + width: 60px; + background: url("/images/read.png") no-repeat scroll 0 0 transparent; + background-position: 24px 5px; + background-size: 7px 9px; + border: none; + text-align:right; + display: block; + float: right; + padding: 4px 10px 0 0; + color: #6e6e6e; } .importedButton { - width: 67px; - height: 18px; - background: url("../images/bt_import.jpg") no-repeat scroll 0 -22px transparent; + width: 60px; + background: url("/images/imported.png") no-repeat scroll 0 0 transparent; + background-position: 2px 6px; + background-size: 8px 6px; border: none; - text-align:center; + text-align:left; display: block; float: left; margin: 0 0 0 4px; - padding: 4px 0 0 0; - text-decoration: none; - font-family: Arial; - color: #FFF; - font-size: 13px; + padding: 4px 0 0 16px; + color: #6e6e6e; } -#librariesList a -{ - width: 256px; - height: 32px; - background: url("../images/bt_library_iphone.jpg") no-repeat scroll 0 0 transparent; - border: none; - display: block; - padding: 9px 0 0 40px; - text-decoration: none; - font-family: Arial; - color: #FFF; - font-size: 13px; + +#indexes{ + border-top: 1px solid #C6C6C6; + background-color: white; + padding: 0px; + margin: 9px 0 0 0; } -#topIndex -{ - width: 100%; - float: right; - margin: 0 24px 15px 0; -} - -#topIndex a -{ - float: right; -} - -#topIndex .next{ - width: 29px; - height: 29px; - border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - display: block; - background: url("../images/bt_next.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} - -#topIndex .previous{ - width: 29px; - height: 29px; - border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - display: block; - background: url("../images/bt_previous.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} - -#topIndex .up{ - width: 92px; - height: 29px; - border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - background: url("../images/bt_libraries.jpg") no-repeat scroll 0 0 transparent; - color: #FFF; - font-size: 17px; - display: block; - text-decoration: none; - font-family: Arial; - font-weight: bold; - text-align:center; -} - -#topIndex .libraries{ - width: 92px; - height: 29px; - border: none; - margin: 0 8px 4px 0; - padding: 6px 0 0 0; - background: url("../images/bt_libraries.jpg") no-repeat scroll 0 0 transparent; - color: #FFF; - font-size: 17px; - display: block; - text-decoration: none; - font-family: Arial; - font-weight: bold; - text-align:center; -} - -#libraryList li -{ - -} - -#libraryList a -{ - width: 256px; - height: 32px; - background: url("../images/bt_library.jpg") no-repeat scroll 0 0 transparent; - border: none; - padding: 8px 0 0 40px; - color: #FFF; - font-size: 14px; - display: block; - text-decoration: none; - font-family: Arial; +.index{ + background-color: white; + margin: 9px 0 9px 0; } #alphaIndex a, #pageIndex a{ width: 29px; - height: 29px; - background: url("../images/bt_index.jpg") no-repeat scroll 0 0 transparent; + height: 24px; border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - color: #FFF; + margin: 0 0 9px 9px; + padding: 5px 0 0 0px; + color: #5C5C5C; font-size: 20px; + text-align: center; display: block; text-decoration: none; font-family: Arial; - font-weight: bold; + border: 1px solid #E2E2E2; text-align:center; } @@ -249,36 +367,12 @@ margin: 20px 12px 20px 0; } #pageIndex .current{ - background: url("../images/bt_current.jpg") no-repeat scroll 0 0 transparent; - color: #EBEBEB; + color: white; + background-color: #A2A2A2; + border: 1px solid #A2A2A2; + } -#alphaIndex .next, #pageIndex .next { - background: url("../images/bt_next.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .previous, #pageIndex .previous{ - background: url("../images/bt_previous.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .first, #pageIndex .first{ - background: url("../images/bt_first.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .last, #pageIndex .last{ - background: url("../images/bt_last.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} - #content h1, #contentLibraries h1{ - color: #696969; - font-weight: bold; - font-size: 21px; - margin: 0 0 5px 0; - } #content h2, #contentLibraries h2{ color: #000; font-weight: bold; @@ -289,13 +383,13 @@ margin: 20px 12px 20px 0; .inputs_login{ width: 256px; height: 64px; - background: url("../images/fnd_inputs.jpg") no-repeat scroll 0 0 #FFF; + background: url("/images/fnd_inputs.jpg") no-repeat scroll 0 0 #FFF; margin: 0 0 18px 0; } .username{ width: 200px; height: 24px; - background: url("../images/fnd_input_username.jpg") no-repeat scroll 0 0 #2b2b2b; + background: url("/images/fnd_input_username.jpg") no-repeat scroll 0 0 #2b2b2b; border: none; padding: 0 0 0 44px; margin: 5px 0 6px 8px; @@ -305,7 +399,7 @@ margin: 20px 12px 20px 0; .pass{ width: 200px; height: 24px; - background: url("../images/fnd_input_pass.jpg") no-repeat scroll 0 0 #2b2b2b; + background: url("/images/fnd_input_pass.jpg") no-repeat scroll 0 0 #2b2b2b; border: none; padding: 0 0 0 44px; margin: 0 0 0 8px; @@ -315,7 +409,7 @@ margin: 20px 12px 20px 0; .button_sign{ width: 86px; height: 30px; - background: url("../images/bt_login.jpg") no-repeat scroll 0 0 transparent; + background: url("/images/bt_login.jpg") no-repeat scroll 0 0 transparent; border: none; margin: 0; padding: 0; @@ -331,12 +425,7 @@ margin: 20px 12px 20px 0; margin: 0 0 0 8px; line-height: 120%; } -.sombra{ - background: url("../images/bottombox.jpg") no-repeat scroll bottom center #FFF; - width: 734px; - height: 13px; - margin: 0 auto; -} + .clear{ height: 2px; clear: both; @@ -344,11 +433,33 @@ margin: 20px 12px 20px 0; .title{ font-family: Arial; - font-weight: bold; font-size: 12px; margin: 0 0 0 6px; - color: #333; + color: #555555 ; overflow: hidden; word-wrap: break-word; height: 80px; + text-decoration: none; } + +#indexalpha, #indexnumber{ + + -webkit-appearance: none; + background-color: rgba(255,255,255,0); + border-radius: 0px; + border: none; + color: #a3a3a3; + font-size: 16px; + font-family: Arial, Helvetica; + height: 30px; + margin: 0 0 0 10px; + padding:0; + float: right; +} + +.comboIndicator { + float: right; + padding: 14px 0 0 0; + margin: 0 0 0 4px; + width: 5px; +} \ No newline at end of file diff --git a/release/server/docroot/css/styles_iphone.css b/release/server/docroot/css/styles_iphone.css index dc5b1ee6..5a3cd9b1 100644 --- a/release/server/docroot/css/styles_iphone.css +++ b/release/server/docroot/css/styles_iphone.css @@ -1,138 +1,119 @@ body{ - background-color: #363636; + background-color: #F5F5F5; font-family: Arial, Helvetica, sans-serif; } -#content{ - width: 256px; - background: url("../images/topbox.jpg") no-repeat scroll 0 0 #FFF; - margin: 16px auto 0; - padding: 16px; +/* libraries */ +#contentLibraries{ + width: 300px; + border: 1px solid #C6C6C6; + background-color: white; + margin-left: auto; + margin-right: auto; + margin-top: 9px; } -.folder +#contentLibraries .library-icon { -float: left; -} - -.cover -{ -float: left; -} - -.cover img -{ - -webkit-box-shadow: 0px 0px 5px #333; -} - -.index{ - width: 272px; - background: url("../images/topbox.jpg") no-repeat scroll 0 0 #FFF; - margin: 7px auto 0; - padding: 16px 0px 0px 16px; -} - -.info -{ -padding: 8px 0px 0px 0px; -float: left; -position: relative; -height: 110px; -width: 160px; -} - -.buttons -{ -position:absolute; -bottom:0px; -left:0px; -} - -#itemContainer li -{ -float: left; -width: 250px; -height: 120px; -margin: 20px 0 15px 0; -} - -#itemContainer a -{ - text-decoration: none; - font-family: Arial; - color: #FFF; - font-size: 13px; - -} - -.browseButton -{ - width: 67px; - height: 24px; - background: url("../images/bt_browse.jpg") no-repeat scroll 0 0 transparent; - border: none; - text-align:center; - display: block; float: left; - padding: 5px 0 0 0; - margin: 0 0 0 8px; + background-color: white; + height: 18px; + padding: 11px 19px 10px 19px; + display:block; } -.importButton +#contentLibraries li { - width: 67px; - height: 20px; - background: url("../images/bt_import.jpg") no-repeat scroll 0 0 transparent; - border: none; - text-align:center; - display: block; - float: left; - margin: 0 0 0 8px; - padding: 4px 0 0 0; + border-bottom: 1px solid #e2e2e2; + position: relative; + list-style: none; } -.importedButton +#contentLibraries li:last-child { - width: 77px; - height: 20px; - background: url("../images/bt_import.jpg") no-repeat scroll 0 -24px transparent; border: none; - text-align:center; +} + +#contentLibraries .library-link +{ + width: 211px; + height: 28px; + border: none; + padding: 11px 0 0 0px; + background-color: white; display: block; - float: left; - margin: 0 0 0 4px; - padding: 4px 0 0 0; - text-decoration: none; + float:left; font-family: Arial; - color: #FFF; - font-size: 13px; -} - -#librariesList li -{ - - -} - -#librariesList a -{ - width: 256px; - height: 32px; - background: url("../images/bt_library_iphone.jpg") no-repeat scroll 0 0 transparent; - border: none; - display: block; - - padding: 9px 0 0 40px; + font-size: 16px; text-decoration: none; - font-family: Arial; - color: #FFF; - font-size: 13px; + color: #525252 ; +} + + #contentLibraries a +{ + position: absolute; + height: 39px; + width: 100%; + z-index: 10; + display: block; + top 0; + text-decoration: none; +} + +#contentLibraries .library-indicator +{ + float: left; + background-color: white; + height: 8px; + padding: 16px 16px 15px 16px; + display:block; +} + + +#content h1 +{ + color: #292929; + text-align: center; + font-size: 21px; +} + +#contentLibraries h1{ + color: #292929; + text-align: center; + border-bottom: 1px solid #C6C6C6; + font-size: 21px; + padding: 15px 0 16px 0; +} + +#folder-header +{ + position: fixed; + width: 100%; + height: 88px; + background-color: rgba(255,255,255,0.9); + border-bottom: 1px solid #C6C6C6; + z-index: 999; +} + +#folder-subheader1 +{ + width: 100%; + height: 40px; + margin-top: 18px; + +} + +#folder-subheader2 +{ + width: 100%; + padding-left: 16px; } #topIndex { - width: 100%; - margin: 7px 0px 7px 0px; + position: absolute; + left: 16px; + top: 19px; } #topIndex a @@ -140,95 +121,242 @@ margin: 20px 0 15px 0; float: left; } +.indicator { + margin: 0 5px; +} + +.path { + text-decoration: none; + color: #5C5C5C; + font-family: Arial, Helvetica; + font-size: 15px; + +} + +#header-combos +{ + position: absolute; + right: 10px; + top: 15px; + color: #a3a3a3; + width: 160px; +} + #topIndex .next{ - width: 29px; - height: 29px; + width: 25px; + height: 19px; border: none; - margin: 0 8px 4px 0; + margin: 0 21px 4px 0; padding: 5px 0 0 0; display: block; - background: url("../images/bt_next.jpg") no-repeat scroll 0 0 transparent; + background: url("/images/next.png") no-repeat scroll 0 0 transparent; + background-size: 25px 19px; padding: 0; - text-indent: -9999px; + text-indent: -99999px; } #topIndex .previous{ - width: 29px; - height: 29px; + width: 25px; + height: 19px; border: none; - margin: 0 8px 4px 0; + margin: 0 14px 4px 0; padding: 5px 0 0 0; display: block; - background: url("../images/bt_previous.jpg") no-repeat scroll 0 0 transparent; + background: url("/images/prev.png") no-repeat scroll 0 0 transparent; + background-size: 25px 19px; padding: 0; - text-indent: -9999px; + text-indent: -99999px; } #topIndex .up{ - width: 58px; - height: 29px; + width: 15px; + height: 19px; border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - background: url("../images/bt_up.jpg") no-repeat scroll 0 0 transparent; + background: url("/images/up.png") no-repeat scroll 0 0 transparent; + background-size: 15px 19px; color: #FFF; - font-size: 17px; display: block; - text-decoration: none; - font-family: Arial; - font-weight: bold; - text-align:center; + text-indent: -99999px; } -#topIndex .libraries{ - width: 92px; - height: 29px; - border: none; - margin: 0 8px 4px 0; - padding: 6px 0 0 0; - background: url("../images/bt_libraries.jpg") no-repeat scroll 0 0 transparent; - color: #FFF; - font-size: 17px; - display: block; - text-decoration: none; - font-family: Arial; - font-weight: bold; - text-align:center; -} - -#libraryList li +#itemContainer li { +width: 300px; +height: 120px; +border: 1px solid #E2E2E2; +margin: 9px auto 0px auto; +background-color: white; +overflow: hidden; +position: relative; +} + +.folderContent +{ + padding-top: 90px; +} +/* hasta aquí */ + +.folder +{ +float: left; } -#libraryList a +.cover { - width: 256px; - height: 32px; - background: url("../images/bt_library.jpg") no-repeat scroll 0 0 transparent; - border: none; - padding: 8px 0 0 40px; - color: #FFF; - font-size: 14px; - display: block; - text-decoration: none; +float: left; +overflow: hidden; +} + +.mark +{ + position: absolute; + top: 0px; + margin-left: 55px; +} + +.info +{ +padding: 8px 0px 0px 0px; +float: left; +position: relative; +height: 115px; +width: 212px; + +} + +.buttons +{ + position:absolute; + bottom:0px; + left:0px; + border-top: 1px solid #e2e2e2; + padding-top: 3px; + height: 25px; + width: 220px; font-family: Arial; + color: #6e6e6e; + font-size: 10px; +} + +.elementInfo +{ + position:absolute; + bottom:24px; + padding-top: 3px; + height: 25px; + width: 220px; + font-family: Arial; + color: #adadad; + font-size: 10px; +} + +.numPages +{ + float: left; + padding-left:8px; +} + +.comicSize +{ + float: right; + padding-right: 9px; +} + + + +#itemContainer a +{ + text-decoration: none; + + +} + +.browseButton +{ + width: 60px; + background: url("/images/browse.png") no-repeat scroll 0 0 transparent; + background-position: 16px 5px; + background-size: 7px 7px; + border: none; + text-align:right; + display: block; + float: right; + padding: 4px 10px 0 0; + color: #6e6e6e; +} + +.importButton +{ + width: 60px; + background: url("/images/download.png") no-repeat scroll 0 0 transparent; + background-position: 3px 5px; + background-size: 7px 8px; + border: none; + text-align:left; + display: block; + float: left; + margin: 0 0 0 4px; + padding: 4px 0 0 16px; + color: #6e6e6e; +} + +.readButton +{ + width: 60px; + background: url("/images/read.png") no-repeat scroll 0 0 transparent; + background-position: 24px 5px; + background-size: 7px 9px; + border: none; + text-align:right; + display: block; + float: right; + padding: 4px 10px 0 0; + color: #6e6e6e; +} + +.importedButton +{ + width: 60px; + background: url("/images/imported.png") no-repeat scroll 0 0 transparent; + background-position: 2px 6px; + background-size: 8px 6px; + border: none; + text-align:left; + display: block; + float: left; + margin: 0 0 0 4px; + padding: 4px 0 0 16px; + color: #6e6e6e; +} + + +#indexes{ + border-top: 1px solid #C6C6C6; + background-color: white; + padding: 0px; + margin: 9px 0 0 0; +} + +.index{ + background-color: white; + margin: 9px 0 9px 0; } #alphaIndex a, #pageIndex a{ width: 29px; - height: 29px; - background: url("../images/bt_index.jpg") no-repeat scroll 0 0 transparent; + height: 24px; border: none; - margin: 0 8px 4px 0; - padding: 5px 0 0 0; - color: #FFF; + margin: 0 0 9px 9px; + padding: 5px 0 0 0px; + color: #5C5C5C; font-size: 20px; + text-align: center; display: block; text-decoration: none; font-family: Arial; - font-weight: bold; + border: 1px solid #E2E2E2; text-align:center; } @@ -237,37 +365,13 @@ margin: 20px 0 15px 0; } #pageIndex .current{ - background: url("../images/bt_current.jpg") no-repeat scroll 0 0 transparent; - color: #EBEBEB; + color: white; + background-color: #A2A2A2; + border: 1px solid #A2A2A2; + } -#alphaIndex .next, #pageIndex .next{ - background: url("../images/bt_next.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .previous, #pageIndex .previous{ - background: url("../images/bt_previous.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .first, #pageIndex .first{ - background: url("../images/bt_first.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} -#alphaIndex .last, #pageIndex .last{ - background: url("../images/bt_last.jpg") no-repeat scroll 0 0 transparent; - padding: 0; - text-indent: -9999px; -} - #content h1{ - color: #696969; - font-weight: bold; - font-size: 21px; - margin: 0 0 5px 0; - } - #content h2{ + #content h2, #contentLibraries h2{ color: #000; font-weight: bold; font-size: 12px; @@ -277,13 +381,13 @@ margin: 20px 0 15px 0; .inputs_login{ width: 256px; height: 64px; - background: url("../images/fnd_inputs.jpg") no-repeat scroll 0 0 #FFF; + background: url("/images/fnd_inputs.jpg") no-repeat scroll 0 0 #FFF; margin: 0 0 18px 0; } .username{ width: 200px; height: 24px; - background: url("../images/fnd_input_username.jpg") no-repeat scroll 0 0 #2b2b2b; + background: url("/images/fnd_input_username.jpg") no-repeat scroll 0 0 #2b2b2b; border: none; padding: 0 0 0 44px; margin: 5px 0 6px 8px; @@ -293,7 +397,7 @@ margin: 20px 0 15px 0; .pass{ width: 200px; height: 24px; - background: url("../images/fnd_input_pass.jpg") no-repeat scroll 0 0 #2b2b2b; + background: url("/images/fnd_input_pass.jpg") no-repeat scroll 0 0 #2b2b2b; border: none; padding: 0 0 0 44px; margin: 0 0 0 8px; @@ -303,7 +407,7 @@ margin: 20px 0 15px 0; .button_sign{ width: 86px; height: 30px; - background: url("../images/bt_login.jpg") no-repeat scroll 0 0 transparent; + background: url("/images/bt_login.jpg") no-repeat scroll 0 0 transparent; border: none; margin: 0; padding: 0; @@ -319,12 +423,7 @@ margin: 20px 0 15px 0; margin: 0 0 0 8px; line-height: 120%; } -.sombra{ - background: url("../images/bottombox.jpg") no-repeat scroll bottom center #FFF; - width: 288px; - height: 14px; - margin: 0 auto; -} + .clear{ height: 2px; clear: both; @@ -332,11 +431,33 @@ margin: 20px 0 15px 0; .title{ font-family: Arial; - font-weight: bold; - font-size: 14px; + font-size: 12px; margin: 0 0 0 6px; - color: #333; + color: #555555 ; overflow: hidden; word-wrap: break-word; - height: 80px; + height: 65px; + text-decoration: none; } + +#indexalpha, #indexnumber{ + + -webkit-appearance: none; + background-color: rgba(255,255,255,0); + border-radius: 0px; + border: none; + color: #a3a3a3; + font-size: 16px; + font-family: Arial, Helvetica; + height: 30px; + margin: 0 0 0 10px; + padding:0; + float: right; +} + +.comboIndicator { + float: right; + padding: 14px 0 0 0; + margin: 0 0 0 4px; + width: 5px; +} \ No newline at end of file diff --git a/release/server/docroot/images/bottomboxLibraries_ipad.jpg b/release/server/docroot/images/bottomboxLibraries_ipad.jpg deleted file mode 100644 index 2df2df837a364207c96054c7556862123c2daba7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1491 zcma)4do?w_k8bH)vbC8NKOtr z8~{WC5U?mfH37)qg&jB)2_Qg&B@(En0MRbs=>9+u(6GorRWCRV;MW878OSI6*N%d4 zF}rv^CBD{CH2|_2AVL}f$pA%$2pOuHKoS5bWJx~*jY2RGhgHWz_?i6IBP~{AF*uwW zPW@kE@fpYn8mFee41-nQxO`7WH(uAy^%wdIZHoV~#N2D!cCTdazo_GR>)vsG{x=3f zhGP) z1&!G0h|}4rt4kE0_vTOwQ;Fu;mX_45x|Ip6A8F1aJ%TBA7q>izvQ)N7H%e1n;l(j6 zOf@llN7Q{jPohT?lme1wyc>gZEYAO?4b5^kW%dr7v`jb_J5-h9HPv+*c@hj;SpAnm zzcuCURslA4gKeuCZbStNYl;cjIV^RU1~<4V={7+BFowK_S-3x41BpJQ<%U0JmiB6G z8>tjYdEpBm>GfQStpUjfPh#$uSKsr6XV}#loB&z3b^LE1BgG9Li-W#Tmqa%&ocv}tA{;evBU^;pr6koi|k_6#q` zJ<+S`>dz`*Tz@xnLi>Yx2h@|enKSTJ18t3yePEYar!sIverHTGKA<(nkk#uMmF~_Q zw*^}o@j2FU62&^|lnE<=!klAs;C=JksDwD4i}umlLE+l#NjJb{6qpv7Cxm>E)7Cf7 z2?i1m%D5!0-~u8p`VHY~D=ez|C&bI^aDo!d_C$+r*Lg(?6yD^y=(i9AV!lxs*F|G# z-CyjH{7k*k-mddOnIm$e1Ag7Fu#kq&Hf0Z-42#X;=%`zyJaA1Cy=_slnq`%EGf9&{ z-P`7Ty{+@Fy)zt`gS{sn1!niz$+kyCz@WG2Ha=JTO_SEC-udy+)vGU^fA%}^=_^e< zdL32llbre3%HEWZEtLdOV<-W1Pi{?0emfWq(vA!4DJ0G0gs^PvrpdljJ;miwa-F*L zerlg9Cq_hj=hmy{r?mNrokewG5Lpxwe;Y4r^QyeDc1&sR8dF{%9s8c`6J7ftl`W1o zqF0_Fq>Vo3N`==?mCc5Y=G{ClPA44Pr&b(oBz$-~)R%uhF>mMS3*~~8;a<|ioc7pb zJiQL)g&WMUjJu|I7p31Ddn1_AD-|BcJ$Bc+({=z32@StA$))S19{aqUW^zo9+_@(0 zS=s8ttjR7bqN4;;24M>l=Q){f#-F)CtE77g9-<%UeSNh@4T57*?{uhoVJuJVWLpF4Sspq6#y ziE^s!wVV_dd$avsH`G7!xz_4T*>IOUR{2AL=Fs&aBU$7=)aA$)g*k{a89u9Eaff96 i5#d_?;e%}-?BoaAc!cg`u>N6gC1smcQY6V%)$=#1Wi)L7 diff --git a/release/server/docroot/images/bottombox_ipad.jpg b/release/server/docroot/images/bottombox_ipad.jpg deleted file mode 100644 index 22633b15ddb039ea581e221c92b0f2bf1650ed74..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2459 zcmbVM3sh5O9RKd!7<2NtgJFbB?gn{$u#F8j*kq2e4cRetB8Hi1bGyQ%?au8Y2nr+? zCm$eKR!372Ipr~xggSJD%121~icAY73$e&V5he84$@OipK_;C}|8wv6{lEYBfBgQB zdtKkVxCf)R4{0=72Kl)DBXfDXk;0Jyqn({m`wt`LcAPNC6EE+>R0 z(kd!2+C^evoCqYT3hYKxEp)Ztk0q~s4L zm7kw4%#Rn6j%<-wE|~L!$d&!Xt%Llp>YsK zigaX>Qyrug_oS?_jEyB>On<|+=CQFPj%h1`U_|c5j%u`51gFQ%`oe1lKe;Iup7 zSi5e6$(~^a=Fj2hgU~=2oyK4V0>DTH!2d-6m@5Go_m1ld*Z>%GI^B=X@bhDgW59Pj zlfht)4`8vzvseLGfQPVvpkORGC?GVH!wC(IoH%h}WR#B(e}DhLfWUAJ3y%n6heh}< z{ww1;4%kc(h+d^3Y=E*68XIw)1p5KvA&B2g2nw@65bx2uV*sVW9O*Q#08Bv8XdW$; zVP}W%F$0|qXwb-Sl*Y9gwLCQ%LQ6*2m_b}~EVL}-akutRfWm+bl;(?wuz9cGcZ^k- zSUrDLyS*kZ`NOVbuFD_*x<}bGHmvlo%Agbgrc8lQ{1_Vnr~~L5UxPjH?-s`dbR3{h z<8G9tr65>S-!qGIFR=~yDFcBakh2FZ(?QKqgaQp-iiU$y>{u!AuelryV=*Bb#1DN` zGVQkubtU2lm9)0INtMxXN$ZbpuDCn1t?#j|v0^(nRxaIgtM}LSNDo?e@2+s}?2(^y zzTMh?D>t%lu(|ktOW*3s^8*`rOf8R@0^W(ScFf{Q+T7IC z)O<8z=G-$Awr}rRU*FiU{#91-pDa#ntRQNw5p8U&FZOGy`B@}{l+=+?J!PYPsI6{< z4~E<7<3~OxY%2ik0jSo~O94G34X*|uL`{b)Kw{vE!CO5Jl0hi~wx-1ZeN46YDIEvU zQToTm?K7Yk~2jX580nb5xXn9XQ3tUU+s^2!cz=ii*S zbZvXTyPWqlpPk*uVdoFN3$?(}nU{l(MA z2WJ<5(OYrrKnM3qR>S#~gKO65cCj*dwEAavWOr1*_i)>ZC1K_=X~4i!X9!fGBdW%(SMXxM9il5e;Uzs#` k*qm=&+IIWi{;Y18%?w_k8bH)vbC8NKOtr z8~{WC5U?mfH37)qg&jB)2_Qg&B@(En0MRbs=>9+u(6GorRWCRV;MW878OSI6*N%d4 zF}rv^CBD{CH2|_2AVL}f$pA%$2pOuHKoS5bWJx~*jY2RGhgHWz_?i6IBP~{AF*uwW zPW@kE@fpYn8mFee41-nQxO`7WH(uAy^%wdIZHoV~#N2D!cCTdazo_GR>)vsG{x=3f zhGP) z1&!G0h|}4rt4kE0_vTOwQ;Fu;mX_45x|Ip6A8F1aJ%TBA7q>izvQ)N7H%e1n;l(j6 zOf@llN7Q{jPohT?lme1wyc>gZEYAO?4b5^kW%dr7v`jb_J5-h9HPv+*c@hj;SpAnm zzcuCURslA4gKeuCZbStNYl;cjIV^RU1~<4V={7+BFowK_S-3x41BpJQ<%U0JmiB6G z8>tjYdEpBm>GfQStpUjfPh#$uSKsr6XV}#loB&z3b^LE1BgG9Li-W#Tmqa%&ocv}tA{;evBU^;pr6koi|k_6#q` zJ<+S`>dz`*Tz@xnLi>Yx2h@|enKSTJ18t3yePEYar!sIverHTGKA<(nkk#uMmF~_Q zw*^}o@j2FU62&^|lnE<=!klAs;C=JksDwD4i}umlLE+l#NjJb{6qpv7Cxm>E)7Cf7 z2?i1m%D5!0-~u8p`VHY~D=ez|C&bI^aDo!d_C$+r*Lg(?6yD^y=(i9AV!lxs*F|G# z-CyjH{7k*k-mddOnIm$e1Ag7Fu#kq&Hf0Z-42#X;=%`zyJaA1Cy=_slnq`%EGf9&{ z-P`7Ty{+@Fy)zt`gS{sn1!niz$+kyCz@WG2Ha=JTO_SEC-udy+)vGU^fA%}^=_^e< zdL32llbre3%HEWZEtLdOV<-W1Pi{?0emfWq(vA!4DJ0G0gs^PvrpdljJ;miwa-F*L zerlg9Cq_hj=hmy{r?mNrokewG5Lpxwe;Y4r^QyeDc1&sR8dF{%9s8c`6J7ftl`W1o zqF0_Fq>Vo3N`==?mCc5Y=G{ClPA44Pr&b(oBz$-~)R%uhF>mMS3*~~8;a<|ioc7pb zJiQL)g&WMUjJu|I7p31Ddn1_AD-|BcJ$Bc+({=z32@StA$))S19{aqUW^zo9+_@(0 zS=s8ttjR7bqN4;;24M>l=Q){f#-F)CtE77g9-<%UeSNh@4T57*?{uhoVJuJVWLpF4Sspq6#y ziE^s!wVV_dd$avsH`G7!xz_4T*>IOUR{2AL=Fs&aBU$7=)aA$)g*k{a89u9Eaff96 i5#d_?;e%}-?BoaAc!cg`u>N6gC1smcQY6V%)$=#1Wi)L7 diff --git a/release/server/docroot/images/browse.png b/release/server/docroot/images/browse.png new file mode 100644 index 0000000000000000000000000000000000000000..9043aa9d9858598d9af07bdc6c95e402680c6a0f GIT binary patch literal 134 zcmeAS@N?(olHy`uVBq!ia0vp^>>$j+1|*LJg}1oj85^_xT291H%W55|cF45{<>wj&+>(Tf?xN eErqjzftle=sQ{~F>Xc-l9tKZWKbLh*2~7Yv@h24k literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/browse@2x.png b/release/server/docroot/images/browse@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..06d5f0580720bdc26a2e67ebe314626eabd6a77c GIT binary patch literal 185 zcmeAS@N?(olHy`uVBq!ia0vp^d?3uh1|;P@bT0xa$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWw1G(Vi}jAr-fhPMkjdoBt4x3X>X}(LseP9ojq{+XPos+Ivc= z`_4MTv4oFdBlD)SE;l)@yy?g@R<tDnm{r-UW|^CCGT literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/bt_browse_ipad.jpg b/release/server/docroot/images/bt_browse_ipad.jpg deleted file mode 100644 index 0657fe2e7ba488642904d9ddc3b9b0d0a6c27d0a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 716 zcmex=iF;o{=v z;^GnD0RsUZK7IjyJ|1CV5fNcw8EI*08F@HhWM^mR<>8eO5Ri}(6%>_%OAyQWe~3Yl zgF%eJl9^GEfk}{&S&;Gn5r#QH?=Uk0U5@~aOsp)7j6h+85Rk>l#LObVz{JGJf-Ekm z$mAfvD8#}nA__E=fteG@C`Kj)$B7IEf`&qqE-@RM2q8&B3 z(NMs^A@Re(kNf|W3Vl$-!FnGsATCN_H}cZev0$;b#4)j2L^ zIpgoV?JCSGJX>DNUz=TRaT{V8Go!tt3-6YvvGD>o&-~R-|I>4FHG}cND@^a@erZ4~ z11Y&4#}6?W%*k&I2J#So5%_l7W83ZA{|x^bZo-WfNO({iur)mVfD+r0n?f~@-)~}g z%)`K*z`Q?EpR>aIP@K?*X^l6wtmChazPKN5lfZRiF;o{=v z;^GnD0RsUZK7IjyJ|1CV5fNcw8EI*08F@HhWM^mR<>8eO5Ri}(6%>_%OAyQWe~3Yl zgF%A9nVC_Lfk}{&S&;Gn5rzp2j35AXJ{&MJv9MwY0)59K08+?|q!wt71G6y5I7Vhx zxPGAb1(^&c3Na`eHZ}<%DF>@?5N2Qo8iuOez`-z3pmE~EkN3;?=2WT6D!^Fg9&lC?4M=%)~fucqei)?0bD^0UCTkmnq(Idxk!AH&1r^{>WA$Bq| z+AC(UXPf6f_|I^Iwbt!T{P%0+X_;%}f6sZo=mta)6G+ijn7v?*Gf+3eM*>BjDnMy4 z+;%%3s*(|?QlO&1KDRY@djtFYUb}$j0*r_6G;VO4Rl!i%e)-iosJB4sF2;yJoDbrp SC~`OntZ4y}U}CQP|C;~C5UDGKfoZ!!63^Z%giXqz$D1XEXer(2;(B4 z1KF6E8G#fOuye4nvT!nUF)=c51BKa`fi}bB7=TjD>`WYtoD5vt0t`%y%q)z|%pkWg zv2m6%GBL9VvI?;&vI{FQa2Sd>8Yzo%20Do)su+t06*fvFO;mMWxbcIUNfDRi!HcHq z4};B`K2BQX5>o8yX6~M>5gHaQwdv5MN1sxfOD4CZF5Z0j@?%Ylw9lnlmR25~UJ;Q| z=^2?>W#tu>RjqC99i3fMrcRqaW9Fb2`P zZr*zG^x5+lFJFE6`tAFVpTB?}XJlq#VPR%vVPRooW6cFRS&&6hh}DpdK}py#kliRz zq>#g@QF)?h(844!W6q5#2QNAoeGq@xWTHALSR&cwe8)8uRi^%;U1o5QPwtX*|uZXo_$L%Z=b$=`|<0~ zzk5VvRCLT(vE#&z7e6dLBQq;IXRh3N^5)B*T2@|BSyf%LR_!`<>(#ISw-|VU(ZeLj zEXZKbaJ7HiZv9QWZ*Sb5pIi3t)A?`w$@z1R#+4Vp+j~0R&RcWR`ZZIRBtD&aGczMI zpkw9AM8V5J!Hb-^R6JJjIkk$&Sh1O$c@udy;)`rO|9^(e{|pu1ca_H0R{yB~?f?7u z{^0f5FMa)Ye9zT8d%EKbcByaI!m@wfWp8c5o*#U>_jq^@|F=!g z3})0mpSEoIob{6|jD!`;ifmtdtgZQVzbkY3hwFdiAHF>OhW!iB4!OIRFX_6k+jpSe z|9gMkrK)d>uj61}x(#v3_LY}%kzMur{^9=&x9acN{+e4IANNoG-{pTF-(Pus>zCAS zulXxiRx_8nUi#VlbR-r=oZ3uI+uiMJ5FsI-f0H zXLx1Gr;Y{JAA0MS^p;%x^m1kiCx27K@t6>XdMn{e%ztM8n*Z(ee+JR-IVJbYUZry2oa}G#BHN(Np75)AL;(xaG;Qcj|X;$}D?0 z*b%7 diff --git a/release/server/docroot/images/bt_current_iphone.jpg b/release/server/docroot/images/bt_current_iphone.jpg deleted file mode 100644 index 883d71828d279f80d893da255adf2ac7ff05ec8b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1113 zcmex=C5UDGKfoZ!!63^Z%giXqz$D1XEXer(2;(B4 z1KF6E8G#fOuye4nvT!nUF)=c51BKa`fi}bB7=TjD>`WYtoD5vt0t`%y%q)z|%pkWg zv2m6%GBL9VvI?;&vI{FQa2Sd>8Yzo%20Do)su+t06*fvFO;mMWxbcIUNfDRi!HcHq z4};B`K2BQX5>o8yX6~M>5gHaQwdv5MN1sxfOD4CZF5Z0j@?%Ylw9lnlmR25~UJ;Q| z=^2?>W#tu>RjqC99i3fMrcRqaW9Fb2`P zZr*zG^x5+lFJFE6`tAFVpTB?}XJlq#VPR%vVPRooW6cFRS&&6hh}DpdK}py#kliRz zq>#g@QF)?h(844!W6q5#2QNAoeGq@xWTHALSR&cwe8)8uRi^%;U1o5QPwtX*|uZXo_$L%Z=b$=`|<0~ zzk5VvRCLT(vE#&z7e6dLBQq;IXRh3N^5)B*T2@|BSyf%LR_!`<>(#ISw-|VU(ZeLj zEXZKbaJ7HiZv9QWZ*Sb5pIi3t)A?`w$@z1R#+4Vp+j~0R&RcWR`ZZIRBtD&aGczMI zpkw9AM8V5J!Hb-^R6JJjIkk$&Sh1O$c@udy;)`rO|9^(e{|pu1ca_H0R{yB~?f?7u z{^0f5FMa)Ye9zT8d%EKbcByaI!m@wfWp8c5o*#U>_jq^@|F=!g z3})0mpSEoIob{6|jD!`;ifmtdtgZQVzbkY3hwFdiAHF>OhW!iB4!OIRFX_6k+jpSe z|9gMkrK)d>uj61}x(#v3_LY}%kzMur{^9=&x9acN{+e4IANNoG-{pTF-(Pus>zCAS zulXxiRx_8nUi#VlbR-r=oZ3uI+uiMJ5FsI-f0H zXLx1Gr;Y{JAA0MS^p;%x^m1kiCx27K@t6>XdMn{e%ztM8n*Z(ee+JR-IVJbYUZry2oa}G#BHN(Np75)AL;(xaG;Qcj|X;$}D?0 z*b%7 diff --git a/release/server/docroot/images/bt_first_ipad.jpg b/release/server/docroot/images/bt_first_ipad.jpg deleted file mode 100644 index a6bfbcce76564679f4082c9b9af35a81aacc6798..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1651 zcma)(cU03?6vy8$31Lc&EJ=%zr4XbRWXci|&^W>n5io`gfn4vIG(fvO^#wjXl=>>-U~_-aX&%x#xb~d4fK{FxZH*IbZ`I7y#bF z0)oE*Y9HAvFcLrj0lu0*FaxA5z0Y|10PlLC0|^E}B7lBeu>S@Bu5aTo2ok2k^F8>{ zMoVn7Kf!66iYp&&R45;Q{hg@8yH91>pNBr1j!L5Ra8U=WZL+DN!?^M4r-MsQJ> z7$goPBsT&W1V=z{_*WJfQk)FI;0Tn+CZw{cj0zCbk+sxSl@qt$EAOGUOTpn>+-A=- zb*r56*=>5k5{fT6^tX>V8pO|0@9#SrVr^q+>!smz%o$zL&^bEqoe-Ly=#xu-*)^tV zkJ0C(KQ)zAM_Kv;N0WK>de%0)_Q#`Vms8#lA_@(T)SMa7j>j~UfZY8soG zTVA!k?q>G%_Vo{pf0~$_VzC#OmN_fjRpG-SI1GV+iy#mPBvK?uh>SugZxYc#0u>oc zdr@5v*>hrh<5bh+9CAG6cZrv)z38wCo>dr$*HfoDZuZ(YcfVqrzT!~>Ywzt6AqnX= zp5p)Qa%|ms63Hn?+Ar`&nGWsxG3ByD=N(9W>sP*|4RQn>vxpEJn(Ioy|%vDQbvbyNE&*R1JsUtne3 zC6ZBjr%lC6MoIpV9?R09F=jH=1r@DgtUiCjTJ6dvbcI75wiiR%)l)Dyvz3v`cIuKH zsC1@C78Kk`vCCO_m$hH)sY2)r`sY=?c2oM{^o+l%uZ$6^n%r*b+py|K7a~p^iJ?(+ zwjH@DQ~mi=V=-LM@p@nq!LT!%hYOFxaorZQ4i!_K^EN02-k6+4< zwJRtvtPfY{ZjfukY%t$x%{{-9P9X?D(hi9eW;+`s&}e@RT2T>yU#ERXrfH+@$6!WoU)ZN z`nun}pOT}b1t1mkna`QNO*GGp@G4ELB}cQTiHRTICXC}ri$kP8wJ5~`V7W5=rDC!y zGA!16I-*9hXrM&&#+psG5u4?$tG^){$K1>=sbJe=E8>Dm$EpXZ#zqq-SJ3u@bu0gwer?=G)BdQ-ovFXe+*_+-GN{2 zrY$)3C-5K4DFwFj>;!SA(Rh}k(<=wi(e-N1 z3oK2(%euu7p6B?1&NPYQaQzkqAktzEfBxOcX_m8;$-%WNR2r?7^rGJ?%tQI*$Vh|d zk>;SwXV>C5+YYKf!I+{uTm0b&E&UxmFYIo)D V$H!;nJPdU0$&k(0x>G0U{|DO$&fEY1 diff --git a/release/server/docroot/images/bt_first_iphone.jpg b/release/server/docroot/images/bt_first_iphone.jpg deleted file mode 100644 index a6bfbcce76564679f4082c9b9af35a81aacc6798..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1651 zcma)(cU03?6vy8$31Lc&EJ=%zr4XbRWXci|&^W>n5io`gfn4vIG(fvO^#wjXl=>>-U~_-aX&%x#xb~d4fK{FxZH*IbZ`I7y#bF z0)oE*Y9HAvFcLrj0lu0*FaxA5z0Y|10PlLC0|^E}B7lBeu>S@Bu5aTo2ok2k^F8>{ zMoVn7Kf!66iYp&&R45;Q{hg@8yH91>pNBr1j!L5Ra8U=WZL+DN!?^M4r-MsQJ> z7$goPBsT&W1V=z{_*WJfQk)FI;0Tn+CZw{cj0zCbk+sxSl@qt$EAOGUOTpn>+-A=- zb*r56*=>5k5{fT6^tX>V8pO|0@9#SrVr^q+>!smz%o$zL&^bEqoe-Ly=#xu-*)^tV zkJ0C(KQ)zAM_Kv;N0WK>de%0)_Q#`Vms8#lA_@(T)SMa7j>j~UfZY8soG zTVA!k?q>G%_Vo{pf0~$_VzC#OmN_fjRpG-SI1GV+iy#mPBvK?uh>SugZxYc#0u>oc zdr@5v*>hrh<5bh+9CAG6cZrv)z38wCo>dr$*HfoDZuZ(YcfVqrzT!~>Ywzt6AqnX= zp5p)Qa%|ms63Hn?+Ar`&nGWsxG3ByD=N(9W>sP*|4RQn>vxpEJn(Ioy|%vDQbvbyNE&*R1JsUtne3 zC6ZBjr%lC6MoIpV9?R09F=jH=1r@DgtUiCjTJ6dvbcI75wiiR%)l)Dyvz3v`cIuKH zsC1@C78Kk`vCCO_m$hH)sY2)r`sY=?c2oM{^o+l%uZ$6^n%r*b+py|K7a~p^iJ?(+ zwjH@DQ~mi=V=-LM@p@nq!LT!%hYOFxaorZQ4i!_K^EN02-k6+4< zwJRtvtPfY{ZjfukY%t$x%{{-9P9X?D(hi9eW;+`s&}e@RT2T>yU#ERXrfH+@$6!WoU)ZN z`nun}pOT}b1t1mkna`QNO*GGp@G4ELB}cQTiHRTICXC}ri$kP8wJ5~`V7W5=rDC!y zGA!16I-*9hXrM&&#+psG5u4?$tG^){$K1>=sbJe=E8>Dm$EpXZ#zqq-SJ3u@bu0gwer?=G)BdQ-ovFXe+*_+-GN{2 zrY$)3C-5K4DFwFj>;!SA(Rh}k(<=wi(e-N1 z3oK2(%euu7p6B?1&NPYQaQzkqAktzEfBxOcX_m8;$-%WNR2r?7^rGJ?%tQI*$Vh|d zk>;SwXV>C5+YYKf!I+{uTm0b&E&UxmFYIo)D V$H!;nJPdU0$&k(0x>G0U{|DO$&fEY1 diff --git a/release/server/docroot/images/bt_import_ipad.jpg b/release/server/docroot/images/bt_import_ipad.jpg deleted file mode 100644 index c7961f01028ea4ad1da0ef6025475c87ca4ed49f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 991 zcma)3dn}b<7=PaHc8=p5zRz)Fn2;$e>o7A)j!Rjwgly5V5htw)DQVW^5+f;!$!cbb z*}4uJUDjOY(z;NiC8wknTO*ufr@6$}d(LTxKYO0<`S#oIeV*Tb@B8qh`~JBqcxujSR7Y#!o>~ED(sKhz0l8yFOoSGcqj(e9`2Zf3uK<*7cwC4OAeyLK@ap$0Pq7}H@hnfa?N7`lW->oN z5=|Jt-nm`(V>>%OG&&1|*&@TwWp2~f`o3m^0-;y<{ndc6Ynu4ral&{&FbYCxAEj-Gtn zlvLyRvvKL|*wH+LA|jE(`Fzg3H|<|v>k3!Y;a=)giS!+>0{7Wje zjv+7I!&4=}-5>jWUyUd1zM$ZNhoKTpjjcSf`x0Z>Gj&&|FdEitE_WCm>#^>|%Fc_` zP3~b)&2tHuUVm*?nbvw+bz}Hf`+#1Khb7k=`?J+)Mj^d9HfhrDl&od4eKpVfo_5s7 zQJK+-rUO=OF0X7kb5&LAEj48o4Hld_-E1B|H0X^n-7?};(y!Iudq

gk}8XtlW=LD*gSH%8=mT;1HEpkXS7C zWqbo!HZUlF3kdQi{ul6rz)3(0$t2={;1J0neh8ZJIwX(?^WPAH7L$}nf;_SjT=xWt zC>9|g2#mZ54iQ>K3azI6qdA;IBq{Rd5JzZPbrhtMb&9Z#V#Q{a@JWgi5BaAb=BABe z)%v28)U2!gBrtdq!I2z3ud8!u2q4-#IG*X8Sg8WkdUDI;9ce7i`6hPsl*X0cox&uK zL_Eeve>ry$APG0XdjJv~3s%dQTloRXaKLkv!W2d9~{2sSlND_2BQS5=_CPWBF>+7cn^2!#~GuhU! zs|C+au}uw;8hKs*Py1L`V`pD)Md!k$W9RNQHJsIbw+Tc^6tAwFPG7bxx@ANMD~SPnbIn=x|b;Vri{efnImK$ahtxU3=n6Z*Lh{;9-6uH8!ktmqgr?r+3S; zjP0KVdg~pXF*Vy=DuJWteSX+cZFR=u$mVs~G=t_uVYlObVfUEMQP{1yZ_r>Ttc`FP zn=z8BPp>e$jM+X<9}Ue|mVd4GI4T6~QVm8Dp=Q$+&bh`|!c)8KyN#9oD!&FvggNwM znepv5HZ!X)T)k%P*P;jEOVlBIt=wIuj~Ot0NT~f{^^#?sv%|LP=N@tr{43$Pu?1~? hvu(ZZ(XnUZH5*GHWn5ZRu(o^RyIyHAm-*^W8>iF;o{=v z;^GnD0RsUZK7IjyJ|1CV5fNcw8EI*08F@HhWM^mR<>8eO5Ri}(6%>_%OAyQWe~3Yl zgF%eJnVC_Lfk}{&S&;Gn5r)}7?=Ud~U5@}v%#18-j39sz1F{&Im<1S^m{1i9Dl$1R z3NZ@{0)5ZSf@BV(Ad`@Rq9KFh#DgD=nuJh|VqkC(W?*JuWMN`th08Ds1_Ct{Iyfc@ zd}y4w@ZtYk3_Q$0#|bhE0xf@Hcn?SxP6ayTMuYtqKoCNy) zuC##Y?yOx~1RPle91nGa)i8n7oCY(2cEXrUOlx_>c&s^ z8u;%%o?mxAnJMU(US^GMWPz aQ$YO}_CYwzAPy*CgkOMzr|H7-|2F|Jp{xu5 diff --git a/release/server/docroot/images/bt_imported_iphone.jpg b/release/server/docroot/images/bt_imported_iphone.jpg deleted file mode 100644 index 50d99a9b4aa41b1693f207e05845f1347ff72900..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 727 zcmex=iF;o{=v z;^GnD0RsUZK7IjyJ|1CV5fNcw8EI*08F@HhWM^mR<>8eO5Ri}(6%>_%OAyQWe~3Yl zgF%A9mzhzJfk}{&S&;Gn5r&Bjj6eW%JpwSXuz|!-1sR!`1;C=n;(|;<4$Q(p(-@gq z5&D5)CTNf-#Gq*SQAr5Z6b1$dVFqSKU{J9lBp3w?4IC651q=fd8z=t1#lQnpz$C~l z2n@z2hWCKxMrHvClQj+u40`*4@*F@XFtRe;+q(CNi;GL?IwcS=|9s6<5D0$0o)1>S z1XA+79?WJ&Vly$Z*)x?xgb_?eMxdzPbG6))CI$0Z&Y8`&H@E$$zZ7-BzD3Hf=GbEf$=DnJ&YB+n&k(+*|FP$m{hd(TL8@OvbTY%)2#*Pz zRGAbp$x~(ae+CeQIFk{mP$11}Tkh3uH=EgbBn%P`bnEq%e`Rs{`|&@++nA>&g3D$? lC8nI2@H+VzJJ&w_L!Strn<$l(MdSnq&%!k7Qw1OQgMr04(u diff --git a/release/server/docroot/images/bt_index_ipad.jpg b/release/server/docroot/images/bt_index_ipad.jpg deleted file mode 100644 index 03c74de304e9cf1299b4d1f010a8e46f5572cdbf..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1149 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 diff --git a/release/server/docroot/images/bt_last_iphone.jpg b/release/server/docroot/images/bt_last_iphone.jpg deleted file mode 100644 index f78c7ee002c82d71144766ea41d901e90acf9c3b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1665 zcma))c{J2(7{}k={Ki;vF}4g%jwp$4DP!x#Ax5-Jma!EHO(7g*87>KtIAu*pjon1j zw3w)<$bN4NgF%YIm=R4)mcjh)bWgYYZ{PR4=XuZfd*1VYp7TD)Ao3n4I1ufL073!a zFJ1td0a%*|zo2LU0W|n-0%R7*S^0!FK|x@R*v7!(m;cLvID#cn zQjj!|S)%|@5R8Uk_&W;48YKPf!t{PD7TExjM~ zOetR@wwRgkwRdp$@H}xcATTnD8Xc3Iav}9%TJDv+tN8_mP~5f&a>BhPJdoImT0sw+gZ)ehF^Mj69MO9Z0oNpb0#T= z82+7?d_2O{3uI@!}pTg)9i17p$(s~fw9xI$zI$cn!Qg@v)e z5@FW{X4%qs!^KjV2+SE;1zw^S+1mB!r}&Q=8fC=tg$bGwA6#Rc(-ZZp*o1 zNNYoRAXY*_R*o@A$s2bqq-@n!&!%R_9G#!qA9>ZTWkBW01!lFO>E&Ub*R^4M)Qxm9 zQ9gWTmB(ZpPD_c6v>|3T*(u3+o1eO*cfT*fx<@utHv5f!YGt27UyXuY35Le^+I@yO z;wja`v&|z!SlGJ5NnCyFKQboz5FpTQYGwx_fKoGOydX3}z`>(4O{-P!SO|z|ZazAq zDacr`e2#z=1jvi#5C8&8bsjM!!ex>H99%SXr!EiNHNa@$OjTc?Ina}@nK z{9TvpiyVq~+M2mOB=^SY)wNbpmFup{1*|ms7?mmkzxR2vZrz8_@cgT+CBd3_WzB(A zo=E@GlM@vdI9~i}4&Rb4q9DK^G!0K0cBc`xy1fX^VB>2WPiiNa7`KkJR7IAD@46c7 z+LbX@ozz)LT`%~SJ5f^Lp_XwB(~#FMviCLPtCw}86|Jw}%nt{@^kvpJs0pc@*gg(U zlUt&;JE-8W55}#1WP%xbi@TJ(!Y*rHH_z&H8u)NGT;-j7ao2ctwO$FG ze)H+X@pf`6%YCSpVC1Yy>~Gwd$z78Wc6O8Jx!@H9^dg{UH@s~|=gy$_-`2M}AMz#9 zRMTa7x*6V)#J5_FA8j6Q`YYl_;td>UmuPc*-SWHoh*+V=*yZ~6C5@HwFiTYtI@W%1 zir;OrCx1z+U@?h5M2iaPA|^I#yj~p1*net3JwfqJ|GZbQ!Lw<-bw<<O32bIz9UgedBH`Yh1R_ zMA7)SQ}Y8STIku6!pg^_=`e1kWtt{WwD9R{9nHpDjcY8t5a2wA@09oW$fWqy^=@-F zKD#gOnBlH7RF=>F3dW~~?bqTh8}H=e!p|oc1e{jce((2+EgIH$8U>faSyKn`TQjZK zE!yPs($4*6;iIU(B*+zyfrvJzE7F@~bFV-7bLE~e{xNN_s`*4+q@X4RU$NTO>Ehzg z^qDDnv^{g|^XD-`T|eVC)3~S0IUBP_GH5u#+lw?cI{l>Y3xbB@gCloYx6);|dartW g1}qKKywQ}+CZ#)CSvedUAk+pq$T(ejc@$y)1L))6;s5{u diff --git a/release/server/docroot/images/bt_libraries_ipad.jpg b/release/server/docroot/images/bt_libraries_ipad.jpg deleted file mode 100644 index f3b10cd9904a38f5f8f005bde805e3a18a6abe16..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 763 zcma)1Ur19?82`R=?!8-gcY4n52Idwqq(xvfOA4&qW-+Rl_O6iWEs_YakdR&sggs>~ z=%F%3^&l7|_Fx4?AxV%ymO`asJ=9YpA_?ln+jsBA-FoYH&i9=kzw`aIKiHo^9LXNe z0x|%L4%j7-2M2pk4*`M#0Q*S#4vCD_-*W=2=cFKqh{82JNVEjN5@ZFkKlU<=fTk#l zs%WaJZP3U9eogZS3|$ZCx*-e~!lv16hQp>Ai$__w_Q$`n6}C74 diff --git a/release/server/docroot/images/bt_libraries_iphone.jpg b/release/server/docroot/images/bt_libraries_iphone.jpg deleted file mode 100644 index f3b10cd9904a38f5f8f005bde805e3a18a6abe16..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 763 zcma)1Ur19?82`R=?!8-gcY4n52Idwqq(xvfOA4&qW-+Rl_O6iWEs_YakdR&sggs>~ z=%F%3^&l7|_Fx4?AxV%ymO`asJ=9YpA_?ln+jsBA-FoYH&i9=kzw`aIKiHo^9LXNe z0x|%L4%j7-2M2pk4*`M#0Q*S#4vCD_-*W=2=cFKqh{82JNVEjN5@ZFkKlU<=fTk#l zs%WaJZP3U9eogZS3|$ZCx*-e~!lv16hQp>Ai$__w_Q$`n6}C74 diff --git a/release/server/docroot/images/bt_library_iphone.jpg b/release/server/docroot/images/bt_library_iphone.jpg deleted file mode 100644 index 8e2c312d1e1cd253aac134ba0c8353ef580832f7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3317 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 diff --git a/release/server/docroot/images/bt_login_ipad.jpg b/release/server/docroot/images/bt_login_ipad.jpg deleted file mode 100644 index a2879b08bdf93e1ba078ae25621f15748d00ee9e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 928 zcmex=5$jlQwN zYW44zwpEJL^%tJk`#a(2qNiUZ%ip+?OfCI7VQ={J`97P2gKw(ND2|ZZE$BMM<&uW1 z^T$si9jo&;PQB_DI_c%L&))ezPCvU}xy0gg?plr7E9Ow#@IXOBR^?Rd(rf)>ih%|$70*`6I=>A z=WTw%Di8ydVd~IJU-OgEMcnvWO~WGYnXl~}0$I;|ov)yQtU)pJYw6{8N-vdH%sKz^ zyWn}3N1ZZzj_@7ZZqE_gd(rEdUDMi*HL{4Yg-R|p8iH{F?Zv0{Tse+;>_pbH_YnM zW1q7>(NrRa-?Hvdmcb6WqQ5Oy5({je{^ne9=t1S_Z^D7C2e0U#Ie&Id^e^kKVl6p& zqXeUC^120oN~W~5%&F-)Gq+~WnXb&_j2DkHgW7h>E(z8Bem$gJx{$~0Hs8|gGD&Nn f{r+=ezUa!WAM0fAdA{>BU2$dI)a#$-|Gx5$jlQwN zYW44zwpEJL^%tJk`#a(2qNiUZ%ip+?OfCI7VQ={J`97P2gKw(ND2|ZZE$BMM<&uW1 z^T$si9jo&;PQB_DI_c%L&))ezPCvU}xy0gg?plr7E9Ow#@IXOBR^?Rd(rf)>ih%|$70*`6I=>A z=WTw%Di8ydVd~IJU-OgEMcnvWO~WGYnXl~}0$I;|ov)yQtU)pJYw6{8N-vdH%sKz^ zyWn}3N1ZZzj_@7ZZqE_gd(rEdUDMi*HL{4Yg-R|p8iH{F?Zv0{Tse+;>_pbH_YnM zW1q7>(NrRa-?Hvdmcb6WqQ5Oy5({je{^ne9=t1S_Z^D7C2e0U#Ie&Id^e^kKVl6p& zqXeUC^120oN~W~5%&F-)Gq+~WnXb&_j2DkHgW7h>E(z8Bem$gJx{$~0Hs8|gGD&Nn f{r+=ezUa!WAM0fAdA{>BU2$dI)a#$-|GxvvBIEn>4GS>W<`qKZ5k1wRm`|eba9_@r-^nKGv(HW z+#}>RLKj_-TeC9A<&(d*0`Kzvnsc=Q*$Fjc5?ytSzl90fYd6 zDlQ*Glq!0Y8j##2Yo@O*Cq+3Rf0|&h+2(wtf@_k7 z`JJ*kO~VTq#it#+wcZ~yikfF-S=jm=v@|~Cw$1Liy;AwJ*Fy``Xn%GLExY1*=Lg~* z_r(D1y(WjPtQ}4`dU$$y2L=U)Fk&w=uf)Y)yPlkqnwFlEo0ng3x3IE`Q~jW(_C-Tu z)63=-URQU|o8G=*{>bRq_=I5TH1F15?(+=|sbDQ5s8EUYODY#k8XO(O2R|*6YjM-;S~~>cDJUvcYHJW(~4md`u*A#^XPD6eu!PFjjCxA;0+ z&X@KL6~9u-_8&YS@^HJ>ex5}tTkBBVKtSb?9h6RNsd`qDKK@&nhG~87r7=G;zcVry zmkO&O@4G&BjM%EeiAmsQIteq^pUk?fqkw>&o|nH>zoQz3QE9Jb z)$CW`_vT)vZP;?8K049NV&X5-4Vik3N@VDxST@b~W*EJo&P~|uRytRM?IQFqdT!Te zoW9my^YBc?AGhtL6Bxqo=$&QdWR)o%o}w7i&RP8sq}WYVn-`EsA5AcRs79lrq`^d= zoV!QK=kjrvUJ6J_T&2wQZKTrGo`3B7);zLgumWWfkR|o3&Gwr-4%MA&a@rlPD>Gui zdjgS?(PwYj< zeZ%-OC%KfG<-E~?E&|h8;hv`UX1371uit@7NPN#3QBuVE^E%k{N|qtpO1H#o;~}K$ zp1JJIs)5CKM|zA2mBp)I0fa#7_XL=)yVO1o)tqahRP#)&GlFLhGq@Uc2FhV+jHeX~ zPLon`{Wltm5H!66pXQkUm9?&z%+?%&ZtE|JcVp86h)4V6VtaDvd8H=j#?M6IMxPOy zoNa;EB1m%K=H#z<2t*)+G*1rX+D=ct<7afP4U0g-zSVsft8<-qPVbm8vd+rZs6*2> z^u{n{m`tfSQ$^0M=QHSmG~MLkOG0km)UU#ZgXZS)#Dcy*3;D^#REN{v%M%NoT diff --git a/release/server/docroot/images/bt_next_iphone.jpg b/release/server/docroot/images/bt_next_iphone.jpg deleted file mode 100644 index 4686bc69be522634d3b9d0c9bbeef6a395e007da..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1560 zcma)(dovvBIEn>4GS>W<`qKZ5k1wRm`|eba9_@r-^nKGv(HW z+#}>RLKj_-TeC9A<&(d*0`Kzvnsc=Q*$Fjc5?ytSzl90fYd6 zDlQ*Glq!0Y8j##2Yo@O*Cq+3Rf0|&h+2(wtf@_k7 z`JJ*kO~VTq#it#+wcZ~yikfF-S=jm=v@|~Cw$1Liy;AwJ*Fy``Xn%GLExY1*=Lg~* z_r(D1y(WjPtQ}4`dU$$y2L=U)Fk&w=uf)Y)yPlkqnwFlEo0ng3x3IE`Q~jW(_C-Tu z)63=-URQU|o8G=*{>bRq_=I5TH1F15?(+=|sbDQ5s8EUYODY#k8XO(O2R|*6YjM-;S~~>cDJUvcYHJW(~4md`u*A#^XPD6eu!PFjjCxA;0+ z&X@KL6~9u-_8&YS@^HJ>ex5}tTkBBVKtSb?9h6RNsd`qDKK@&nhG~87r7=G;zcVry zmkO&O@4G&BjM%EeiAmsQIteq^pUk?fqkw>&o|nH>zoQz3QE9Jb z)$CW`_vT)vZP;?8K049NV&X5-4Vik3N@VDxST@b~W*EJo&P~|uRytRM?IQFqdT!Te zoW9my^YBc?AGhtL6Bxqo=$&QdWR)o%o}w7i&RP8sq}WYVn-`EsA5AcRs79lrq`^d= zoV!QK=kjrvUJ6J_T&2wQZKTrGo`3B7);zLgumWWfkR|o3&Gwr-4%MA&a@rlPD>Gui zdjgS?(PwYj< zeZ%-OC%KfG<-E~?E&|h8;hv`UX1371uit@7NPN#3QBuVE^E%k{N|qtpO1H#o;~}K$ zp1JJIs)5CKM|zA2mBp)I0fa#7_XL=)yVO1o)tqahRP#)&GlFLhGq@Uc2FhV+jHeX~ zPLon`{Wltm5H!66pXQkUm9?&z%+?%&ZtE|JcVp86h)4V6VtaDvd8H=j#?M6IMxPOy zoNa;EB1m%K=H#z<2t*)+G*1rX+D=ct<7afP4U0g-zSVsft8<-qPVbm8vd+rZs6*2> z^u{n{m`tfSQ$^0M=QHSmG~MLkOG0km)UU#ZgXZS)#Dcy*3;D^#REN{v%M%NoT diff --git a/release/server/docroot/images/bt_previous_ipad.jpg b/release/server/docroot/images/bt_previous_ipad.jpg deleted file mode 100644 index 1fa059d663736f756331d1f6ad02151839e68215..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1586 zcma)(cTkj96vf}SjUplo(u0KFQ9wWly3z%tcO%7JibxB+s0&gBM0$@R2+~6nR*C_o zBMebkX^MgrS&EdUDtuWp#^g`Wn>X*yIdkXz?p$~beh*mmb@X%qgaE*ab^yE#uv!5G z_YeR96!>Zacoi_KJ9*=sfm1upf#69H1)v`n;y)q3>svblf@mr2eh+@MfZG9<9`FG^ zB!mSBEQG{Da3@qq1H>SZ5bgOUI(iHm#eif)K;Q_?#vo}w|C<3?grq~zLkxiN2n#?! zNEC!beq}*m7y=*!5`{&xVg%^e1OdI=admkib_P=ojw`|noMt|^xbP_=ngz9MqNjZs zxu3mM6q`0zidauA)Uxo?)=}0ah+A4&^VB_mHM8Lq>7N$m{II@b_=Ch5m(2i471axR z`Zl(gT(7!a3k(Vl35|}4y&V^yo{^cAos(Ns{HUa~to(69Bk4&~b7xm~Pj6rU2>JEs z*!aZk-2B4FMGAFmduMlVpEfvzM4(VeGzx{nV9=g4WGqU66)lGWf^6!hbn;h@`_OCL z5=vn=E5LIoFw_b^d#UNW#yK5vS|rt+i=ee$SSPB;ZK0&?B*y3$nWp3amDkcaz)CqP z`XAs;37+(#$DJdy)QsYWF7n)#xURL!nZTILM~&UD=eJ#hVzWv}J);Xd7i_Ku-_9<5 z(mVEXS4Gd(EhH|dtf_B&aZgfJ|I)S4_}ud5{t3!HJOP+!>p@_VSbzgUEj_hC#*VAM zAMQ+jP7rHW_WIaw=@Nu_KR|x!Wh|9emHfig+R{`)dSJk#T>5snWcUL}K(Oc<7R|!M zTs!ZQIb)UUA|u6<8k`z>wYyPQA>~0QUP4N~$)n0sR+%%Q`_-UbN}rQ^$=%HOh=}+M zk+RCF%CZbXyg2#3u46gF&(~`m(MEuha6|!ZV!(DD^qq?=5Nio7sUe;X)FG5zH|FJ_ zBd)x$AL?6JOCfjJ>;_Hco%HsuB+51hdxL%xe2|QqWy(WUZHqd=mT2a|4mH1SkvpkT zxkRm`4gbQ`=kDBt4edq!$5R%|G@%Qr_-0#6`p$yV|?{(jRz-Y zmsD>U2I72H?5ihLnvF#iy>LR>GFHkmZ5tMT?B$hlQ)+kxv*E@zwK@#x?6Wz7C!f~& zzxG(e1`kgZ{b@5M&=42In|RW9rb3@3e4k(Goa>A5&9f#8n>E#IFi;3f!#Yr5P*A@v`!IO=+0}Ze z_&yh9zAb}))8j3Uj^wFq8yIc$Yv7H`Z*#mw-H&nR31m@P8#NpygBtrEi4^7WNhi*} zz!PdJOk7J4rScblsofWU@e61-iiqd|poB1cc zzGJqtBo{*U@JSmbqQT88a=-WmlN~P>*RC|3$_v9a*Q9g%M@8qj`Lds?mo1j$)u?Hv zCi;^XjQC`dHGbOF%3g{OKYGrQRceo#PFw9^{wJ|RiPdq+y{Ff|)bEG2=I@rYxmE{K zD`NPH54!s-ESyM=%LOf`lBd_!rsc#4O5JB}wtr3&Nqv))&rN+Bm(NvNdKKTL$ZOhg qxsF_!z$9aTV1LPNZ?xhSKU1o4f~mT?zR9R!mAgKpS;mk9JpMQ58kZgb diff --git a/release/server/docroot/images/bt_previous_iphone.jpg b/release/server/docroot/images/bt_previous_iphone.jpg deleted file mode 100644 index 1fa059d663736f756331d1f6ad02151839e68215..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1586 zcma)(cTkj96vf}SjUplo(u0KFQ9wWly3z%tcO%7JibxB+s0&gBM0$@R2+~6nR*C_o zBMebkX^MgrS&EdUDtuWp#^g`Wn>X*yIdkXz?p$~beh*mmb@X%qgaE*ab^yE#uv!5G z_YeR96!>Zacoi_KJ9*=sfm1upf#69H1)v`n;y)q3>svblf@mr2eh+@MfZG9<9`FG^ zB!mSBEQG{Da3@qq1H>SZ5bgOUI(iHm#eif)K;Q_?#vo}w|C<3?grq~zLkxiN2n#?! zNEC!beq}*m7y=*!5`{&xVg%^e1OdI=admkib_P=ojw`|noMt|^xbP_=ngz9MqNjZs zxu3mM6q`0zidauA)Uxo?)=}0ah+A4&^VB_mHM8Lq>7N$m{II@b_=Ch5m(2i471axR z`Zl(gT(7!a3k(Vl35|}4y&V^yo{^cAos(Ns{HUa~to(69Bk4&~b7xm~Pj6rU2>JEs z*!aZk-2B4FMGAFmduMlVpEfvzM4(VeGzx{nV9=g4WGqU66)lGWf^6!hbn;h@`_OCL z5=vn=E5LIoFw_b^d#UNW#yK5vS|rt+i=ee$SSPB;ZK0&?B*y3$nWp3amDkcaz)CqP z`XAs;37+(#$DJdy)QsYWF7n)#xURL!nZTILM~&UD=eJ#hVzWv}J);Xd7i_Ku-_9<5 z(mVEXS4Gd(EhH|dtf_B&aZgfJ|I)S4_}ud5{t3!HJOP+!>p@_VSbzgUEj_hC#*VAM zAMQ+jP7rHW_WIaw=@Nu_KR|x!Wh|9emHfig+R{`)dSJk#T>5snWcUL}K(Oc<7R|!M zTs!ZQIb)UUA|u6<8k`z>wYyPQA>~0QUP4N~$)n0sR+%%Q`_-UbN}rQ^$=%HOh=}+M zk+RCF%CZbXyg2#3u46gF&(~`m(MEuha6|!ZV!(DD^qq?=5Nio7sUe;X)FG5zH|FJ_ zBd)x$AL?6JOCfjJ>;_Hco%HsuB+51hdxL%xe2|QqWy(WUZHqd=mT2a|4mH1SkvpkT zxkRm`4gbQ`=kDBt4edq!$5R%|G@%Qr_-0#6`p$yV|?{(jRz-Y zmsD>U2I72H?5ihLnvF#iy>LR>GFHkmZ5tMT?B$hlQ)+kxv*E@zwK@#x?6Wz7C!f~& zzxG(e1`kgZ{b@5M&=42In|RW9rb3@3e4k(Goa>A5&9f#8n>E#IFi;3f!#Yr5P*A@v`!IO=+0}Ze z_&yh9zAb}))8j3Uj^wFq8yIc$Yv7H`Z*#mw-H&nR31m@P8#NpygBtrEi4^7WNhi*} zz!PdJOk7J4rScblsofWU@e61-iiqd|poB1cc zzGJqtBo{*U@JSmbqQT88a=-WmlN~P>*RC|3$_v9a*Q9g%M@8qj`Lds?mo1j$)u?Hv zCi;^XjQC`dHGbOF%3g{OKYGrQRceo#PFw9^{wJ|RiPdq+y{Ff|)bEG2=I@rYxmE{K zD`NPH54!s-ESyM=%LOf`lBd_!rsc#4O5JB}wtr3&Nqv))&rN+Bm(NvNdKKTL$ZOhg qxsF_!z$9aTV1LPNZ?xhSKU1o4f~mT?zR9R!mAgKpS;mk9JpMQ58kZgb diff --git a/release/server/docroot/images/bt_up_iphone.jpg b/release/server/docroot/images/bt_up_iphone.jpg deleted file mode 100644 index 94623e78c5b0c44d21ab6be834dafb75d35c2829..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 720 zcmex=iF;o{=v z;^GnD0RsUZK7IjyJ|1CV5fNcw8EI*08F@HhWM^mR<>8eO5Ri}(6%>_%OAyQWe~3Yl zgF%+TikVT6fk}{&S&;Gn5r)}7?=Ud~U5^0FER0O7j39sz1F{&ISpxtn+Er zs#U*hy?`VLtoiZzs*OKb0Ta;COgHSoY@nTRHWL$@JrgrT7{O#@1d3_}&g%844>ddW z@wQUVOn1o_$@Lre2K{HyN`_d;%xJHeTAppQeFrdoFszQ1)M~%wyZ*A8&rG9~XYb_B zL+k~q`oa&f0K~pt4sry-R{{wP?6-5b0ZoYi&+rp&vOv~z`!$uhbI(05-g)O;eSE>+ z8~+*pGprJi+xYF&{HXgM+%L$(EfBaa_E;Tmg+Qeuhm*kmNdhe(8b~<4KKTD803Y6; A=>Px# diff --git a/release/server/docroot/images/combo.png b/release/server/docroot/images/combo.png new file mode 100644 index 0000000000000000000000000000000000000000..54edc9fafbde3c37c4dde5aeff80382e495eccbd GIT binary patch literal 120 zcmeAS@N?(olHy`uVBq!ia0vp^tU%1n!3HFw&5r&6Qj#UE5hcO-X(i=}MX3yqDfvmM z3ZA)%>8U}fi7AzZCsS>JiljVU978H@B}MG1_*l=T!tngC#0#zf76v9B29|ujPR>k* QEkIQap00i_>zopr0NbA+CjbBd literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/combo@2x.png b/release/server/docroot/images/combo@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..39e5c700937a7e8ec1599760de64dc3bcc4e412c GIT binary patch literal 167 zcmeAS@N?(olHy`uVBq!ia0vp^AT}EZ8<70WD#W6a5s#(4CSmdKI;Vst0A>O+1ONa4 literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/download.png b/release/server/docroot/images/download.png new file mode 100644 index 0000000000000000000000000000000000000000..10b43d66be328c5f946556e772f54a078d35d982 GIT binary patch literal 155 zcmeAS@N?(olHy`uVBq!ia0vp^>_E)H!3HEvS)PI@$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWw1Gww^AIAr-fh5*(N+D=Ys`{OQ9}!N5A{#W#i9{Fh#KtYbJ; zp3+hz({QCurSiDvWHs3YmmuC?hG`6)j4Toi#?7LC94^ZQ01aaBboFyt=akR{0Kum( AbpQYW literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/download@2x.png b/release/server/docroot/images/download@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..98c6ae38f9d448b10db872feb999e0a3a341bcac GIT binary patch literal 203 zcmeAS@N?(olHy`uVBq!ia0vp^d_XL~!3HGNrubO_DajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MY*0Xjv*Ddk`fLue){z3U$cQ;r5c;j!4+;%87YlNuP*VN zEYW$9A^3cY%7*zIz6GW}0kioSHZt#x1XkcXcci*_{_;RZNpv4THu6{1-oD!M<1rbIJ literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/f.jpg b/release/server/docroot/images/f.jpg deleted file mode 100644 index ba93d1f45bb5ae2cf5c4718c38a7f074a7f61e97..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2802 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 Q2Q8H`)Iz^j&prPC0LUMYmH+?% diff --git a/release/server/docroot/images/f@2x.png b/release/server/docroot/images/f@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..e388d804f336a4d6c0f783ecbfc194d5e13f826c GIT binary patch literal 1262 zcmeAS@N?(olHy`uVBq!ia0vp^3xN0o2OE%Vcg(*9q$EpRBT9nv(@M${i&7aJQ}UBi z6+Ckj(^G>|6H_V+Po~;1FtAMUba4!+xb^1lMt_k=nFAjMHGOV0@p&Ek$9QLJwrfZB z>zgw*K3Z-4x*>L(Xl~=I>@6QNj!sy;;OmPV0q<>#7wn%cUv$#-p_7vF`?KesPxcAm zs@%Q*z1@MNBNKTZ-(>b`}!YApXR9F_ik>%qNY~9Bj(eVolmb`tEL_CVsH07 zOKa<39esPPs=aX+UzYS3J1<|p{OG@bbsM*StuCyw*_E?x|FeC}b=!ZJ?!KE;T)g;h<-Cg-~ z_rNQKY^Pk)($l-@_Q&7Z|KA%Jk8c`s1^2gn{_ybA&6oZwA8~)&n9RFz+LZkgDhH&F z_L)npNh|dTTBCaH$}x^;(FfP#o$s6}tIf_{f9{&b0YlN{#ZPN&V%Dt9RqxoUcEIBO zy9W;xewQAT(zl%L-{AZ0-`~EiTn2O1WT)L*)_gZwIN{X)Jjv%?%B#!z_YjF zWAwHA6OPfOgF8C)pz;*{4u=Dw|4-#)or#hz{N zh7C{MKe5gzyB--=S5o#c*nm(9!k)-jK%uPQKmrvZ)$8`|ug{V>uA3OHUpM!$6V^-* zOm9&C|M&F0dv%5qYi@B0zppWDo^ka5*;QVXP4C@%`>EvS3X{vmcgJ+ymv>AhWx7oR`=`SacFPvw`N&IYBkO0$Q1HP^GJ m*37>ni?vVy3L3=y6HH*(A!Ao{RJ*4Nq{`FP&t;ucLK6V+vM6`} literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/fnd_input_pass.jpg b/release/server/docroot/images/fnd_input_pass.jpg deleted file mode 100644 index a332678b3ea6c2130ff20ac17e75a6c6bf044494..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 546 zcmex=C5UDGKfoZ!!63mP&deytz$D1XEXer(2tzyr z12ZEd(3J?l%Er!xCIpleU|?coVPs}uhACuZVipu)VP#`b6m~QcVJ{RF6GSx!sPO+S z1|DWcpl!^84E799;}=*lFF0apuXy)qvE`3Og*5wXCw3jowec^J4`+? diff --git a/release/server/docroot/images/fnd_input_username.jpg b/release/server/docroot/images/fnd_input_username.jpg deleted file mode 100644 index c8a2fd9c748d8beaa1c6f5cca9505d30d42f578e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 604 zcmex=C5UDGKfoZ!!63mP!OSSgz$D1XEXer(2tyLc zVL(SB02>Dvng~!%fPs;jiItg&9j1<%k%>i+l|fNRn9a~pDNrOa$yp_oU0e{=ETHQD zw-|Vs8G#lu3o_Uwpw4aS3LC> zJ>c;(e6N*~s@8se*DpW590-@;o*0#17I86mO-=Z%#m71_)RjaICbKV4by^%Xi6up9 zm95~`kf=qe0k&uBMXeYjmPHrbtF5tl*o47aNO-1Ev;R~>b$-23(G0l|Oz#ec5w z-FI@&yPN6tK)hi8_3e7v6{QEZXIx^4XzR&%AUNYuo<+X&ob{pk%}2T~?co1^69B)? Bq^$q| diff --git a/release/server/docroot/images/fnd_inputs.jpg b/release/server/docroot/images/fnd_inputs.jpg deleted file mode 100644 index 4a53411d784f0fb842a82a5a0c8b5dd39f984315..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1443 zcmex=C5UDGKfoZ!!QjBiz|1Jfz$D1XEXer(2tx(P zVL&0EJ7Iv4gNd1i3n+>P1c07q;$&iCW8;R(FtIVRPGY{y5+Zt)T|8`>Acvx1VBtiO zg&Q{si*Y(8HXeLvlvLE@8Y+lk*8f`!JU~mC1epaH>=}N(|NQvb`I`R>2J;Q1yZ1`} z+__#|XHwmZxz}>9-8gvAWcBUd^|xP~4Uf8#W)d}Z+N_@y!Q=c_ie%%^w-tC$7=UeJe%I{jkQg(Bx{MUbSPnFK_ z*<@Tguxgf0L*Zg?rd1i67?1VpaIIW)f?djFgVwTc1(_iu&E%eQEAZ=`P+zfx+<3)* zIomIP`udr1d-UZE)pI`Odfa~3shalaZB6=Tud_Su%I-TG81*L2de+pybyM%Z`|KI} z=&SVB=TEcsC%t}kv3lpKRbRuyqb8hHK#O^zxgO4kFEVK0RL*>OiicI~;#<0FHTW{8 zc?SRKJ+{hzWx9-H3U*;ftEreINU!dsX~z?r`8(J!fOUhnR_bc2sLiZ!$gn zU}x3BUkopMmO0D&MDuDKzb!U7%kams&_^t%RAluoz7@MRg;`@l$!_;8(NO_nLNj@6 zecPtk$4L3^aDU7h!|-&PXKj3Y)IYiNhu;2Y;BSk5z;bFy;n!_Zh2OW_F8r0O&6H}S zHzTuba<&)aRF8$Pa<=UD+`hC?S#!ei%VMX!?7q#;vyWNB7O3#DtUqJhmIab7%}a0D zF1mH8AytTPdd9bIpgy?w2A?wKJ1i|V4hwL`4Fmt{h8+%>L47?foG>s@J>sp9!h`wLYBt*45Iz zO6xv`?b^Ynh}(iZekC~<^XKQ&{=9v5EjR6+<+8UcchB-ZA>BCXtkAIp!^tL2GUwWz lXL9U5ohM%1vh96ktE-E?#@_qyGkmYTnsxeYlAyr|6H_V+Po~-c6*+jiIEGZ*N=i7u7+1k!-H_oXGl|n7%TB=BNO85>WS=QF zHhyDJ4k@ztU|7Ybwv!=`b&?)iiYJe|Vl*o=10x&5_h*t{U7hXufo3syy85}Sb4q9e E0R1m4egFUf literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/imported@2x.png b/release/server/docroot/images/imported@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..41b9ad4e96532b047eac146c38c66e37dc9817ee GIT binary patch literal 214 zcmeAS@N?(olHy`uVBq!ia0vp^0zk~e!3HF=pW8M9DajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MHQYdjv*Ddk`fe{_k7?GV|c_SFnglRS%)lp0cnOpWo6b` z`V1SDl~re{maX}FfMse3U6Ni2|d@-Vg$+o0KU z>haYcq236dM1>`i%CZY24>@=kUu$78UBYl`ZiByZD1$=-BSXwAkG2i#7iR+9z~JfX K=d#Wzp$Pyl_d)dl literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/indicator.png b/release/server/docroot/images/indicator.png new file mode 100644 index 0000000000000000000000000000000000000000..138c77a84bb3339a5c1092bd83b6104481688dfc GIT binary patch literal 118 zcmeAS@N?(olHy`uVBq!ia0vp^tU%1k!3HFMt$j8RNJ*BsMwA5SrbP0l+XkK8S^5> literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/indicator@2x.png b/release/server/docroot/images/indicator@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..9f30e70415f7715ee848411d4ad9b6b5a82f3ee5 GIT binary patch literal 220 zcmeAS@N?(olHy`uVBq!ia0vp^Ahr+(8<0%e?(YqxBuiW)N`mv#O3D+9QW+dm@{>{( zJaZG%Q-e|yQz{EjrrH1%)p@!&hE&{2I&=E;?(+u@EMPdot>!yx%83SM1I9uHfqI5V zEDM(X+#zDr#Ib;hSz)Vu2W!F1GaShRL1#M799%v_@Ge(Om*hrEnb}EhG5cJ?&$M_e zUYNq|a7s|D#6mQQyI|28wfn{{to(tDejy@4Q#CpQS-9tYZ4qf?aA;s;2#WA!+@DuD Q9q1AUPgg&ebxsLQ0BK`La{vGU literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/library.png b/release/server/docroot/images/library.png new file mode 100644 index 0000000000000000000000000000000000000000..1a185057a328c0aa1c1dc75cd86edc7c57fd4449 GIT binary patch literal 201 zcmeAS@N?(olHy`uVBq!ia0vp^d_XM3!3HF=W8NDADajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_c!wVp1HAr-ftOmgIFFyLU(e7N_E?uSI)z{yz+LAF-Q#Ujcs zZ#l?moIhc~s3=UPDKl(yc`FJ7UWr@xi3r@U9dp;?^CGY#a--2Ix*DcPSV;QVE x>$v=)BoWiD-3w0*^-~1kIcOQ#E#f}N(&!?EHl96Qx+=389HzTS%zdO zichKTL#W}7xO~clq>vGWX>BPj;t;Q;EY!9LPnrN(`krP!#09A4Lqho$mQU&K!{Kis z(Q$}YR+1g3gV%?o)*d4_(Tt|j(NsEWA@x!zxp6w0O6liAQcISh<3nn-zG3ZCbn^zz ihcxI#4URYj2rvMR$xk-q0jcBw0000{( zJaZG%Q-e|yQz{EjrrH1%X?wajhE&{2N?5?KW!tuY=?mB;wq{&eB$yzmd^bmClcQPI i!4<5BPcStwFf(Md8T7mf@@N5SWAJqKb6Mw<&;$UgGbzdd literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/next@2x.png b/release/server/docroot/images/next@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..ec4feb220534dfd6107e93a2b65fb7ef17ad94b2 GIT binary patch literal 339 zcmV-Z0j&OsP)MwIvpFaV2?%kdd*&=&vz002ovPDHLkV1i%%fOr4^ literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/prev.png b/release/server/docroot/images/prev.png new file mode 100644 index 0000000000000000000000000000000000000000..43e4805e3916fdecdebc1a9dc8fc3a31521c50b9 GIT binary patch literal 154 zcmeAS@N?(olHy`uVBq!ia0vp^l0YoX!3HE#i{( zJaZG%Q-e|yQz{EjrrH1%*?77*hE&{2N>E^kijJ0T_+ihy_P`3(#bP~C3~W7HwlT0o zG~5vE_#Pe}ZhYyAS{2L96|RfLcoVxBn0Od&TlgCvd$uA8Xbyv?tDnm{r-UW|QRgn` literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/prev@2x.png b/release/server/docroot/images/prev@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..d505f4f51dff9bd246dd7948942d507848d9341e GIT binary patch literal 345 zcmV-f0jBYnD$VQWAX zL68vH5Hv&<1QC%8K}94%kP*=kbVL+{0dWXnLF_`95StJ-L=eJ=SV3433kWk}hOi?h zNDT-djpm&l@tF~km#H8cQ%RgOd(CcMT8?vR`5Ae95q_Txy3|_AJu)4?5;(cYX6E-p zB*!eTD8!icrI8d9zQz+`wjWkxnEj_N5$3q~K!S-Y2N7W63v}R^w2~<>CcTIaipgs$ r0AlhRJz$u!)*BGUmJWb|c?&QABjnXf3&W2300000NkvXXu0mjf*A#>Z literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/read.png b/release/server/docroot/images/read.png new file mode 100644 index 0000000000000000000000000000000000000000..9465a7d5f4727667be0605a7591744bf0dcc14c9 GIT binary patch literal 152 zcmeAS@N?(olHy`uVBq!ia0vp^>_E)P!3HG%MVKuHQj#UE5hcO-X(i=}MX3yqDfvmM z3ZA)%>8U}fi7AzZCsS>JimW_c978H@C7n5a`uF(;7B$~lM>w76b4UMKbLh*2~7aVH!tk~ literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/read@2x.png b/release/server/docroot/images/read@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..68ed871530b0ffc5740ec6f6336ab2690ad2dbec GIT binary patch literal 201 zcmeAS@N?(olHy`uVBq!ia0vp^d_XM3!3HF=W8NDADajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_c!wVp1HAr-fh&YV8|`+S3Z3`5d!&&g`F4q_=u3Oqaw74{dH z7*ibGX7oA;OHNds5L3jsO7du0Qcj7>C7$gw7r3q9m6+AJQE+vdnA5C+mR*d2W=YIC w6?2YvoM%nBZLw;im{cUk5`kB@IRp|IN*0AX-jET)X7KQ01>=#%OA)*d*Z(j?D)T5veqeAlU2&)F(Psyp2K@$y sPXVO~wme6&j-)X#Yf4?@W#BMiDDE)I+JE*$8qh`tPgg&ebxsLQ0Ful_p#T5? literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/readMark@2x.png b/release/server/docroot/images/readMark@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..1d9d0d3e3a3dab8d7259402b47f02d5172ec407e GIT binary patch literal 296 zcmeAS@N?(olHy`uVBq!ia0vp^azJdx!3HERoY(vbq$EpRBT9nv(@M${i&7aJQ}UBi z6+Ckj(^G>|6H_V+Po~-c6`k^QaSW-r_2$My&SnDv)(73fH+)}l+&v&@3tme#bd9y_{rM!qFj5E+OD_+S8&|VT6^}zU*3J!?nxxH n6qK4ADKy%4PbhI8s{}&@XA@`A!!uPtUom*P`njxgN@xNA$RK!! literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/readingMark.png b/release/server/docroot/images/readingMark.png new file mode 100644 index 0000000000000000000000000000000000000000..e37f3c80cd812942ac988e20df154626baba7ae7 GIT binary patch literal 206 zcmeAS@N?(olHy`uVBq!ia0vp^{6H+t!3HE3PrP~@NJ*BsMwA5SrL11q z;e{rivmB*lBz+~s8Zu(o3?5#rU_8=zDT3GG`X7c!WgZ2|4-C`OCNMdd^k%S0I9*+s zv5#TvN9KJOT2EOY;$^rd@w_pjVTuORO7@q`90m+?jwnqyx_IMSpxq3fu6{1-oD!M< D)h9`8 literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/readingMark@2x.png b/release/server/docroot/images/readingMark@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..589e2e9bf8474511263dd76023add1e2358d48e3 GIT binary patch literal 296 zcmeAS@N?(olHy`uVBq!ia0vp^azJdx!3HERoY(vbq$EpRBT9nv(@M${i&7aJQ}UBi z6+Ckj(^G>|6H_V+Po~-c6`k^QaSW-r_2$M~&K3h6wg)$QS;PyPr4LFU>^v7J^PtY6 zNqL9kdP^mb9h>E^z3SgCpk?|*QA}scru?>Q?DRqy2wTV(w^AtCwe m3XxTXbrn2&x2 literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/topboxLibraries_ipad.jpg b/release/server/docroot/images/topboxLibraries_ipad.jpg deleted file mode 100644 index c6b0f8a24cd03457c17b698028b7c56d621d0fbb..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 423 zcmex=C5UDGKfoZ!!NAI>z|1Jfz$D1XEXer(2ty3e z*~~ykA^~;|E=EQqKA0&0^f4<7Gm^N7sF1LjxCFDZ3acVJhpHfw6h!m?TMRrv4NQW} zf(-Tyx1XQB@jGE}Y?&~a!x7UyA9+CpU&wrO&-q2@x8 zI%!`-R`A6KFAsw3Kd?{Bg6_*Mxb+6RPl7MrSm=ZJpm=!|A)aKeE*;s!Iq*B*`F_9e zJKy;d|FORXMh>`H=K;Dn#(Yw zm^$yt7B6J7JeaI_maMEomtBUZHKWYE{o^*rYHFEVEY!1gYX+`phc|3Ezj0(i*;rLd zDpxtkPI{83TRL=Q*3-`zj^vfOP+USdF!LN6y11)lZYqe%9GK~Wn(>~1$ zrBW%-$mP<=BkimkuI!}^XQHi8g^prtmaCZt8z{=l=9*jPP^Qf*=vGH|+gkNF*6Uw( z)%BnNp{(Q1!EV0W3Ofty7UbukW3Jf>MD7y-*^*{#D7&V;V47#zRXnz0x~8*YTI|fc z$bP-38LHWE_J{uUx>Pe9S2h$_tCl&WkkK?%%8A)hacWu=zAse;;ZUwr$XBO|LScG# zwph$h=iA(>sjTVHaNAt9%Psch2Cbl5C~_6r+F7X1+NREiSC+K?x#aumwYX~kTnc@; zJW7TS9=7M9+a}aK0qnFb+H@Kp8mQZC)YkqRus3X0p#FtEAB+IQBtgXo0fpdfF#u;Ug2DWI;1a+vtous{P$c?a3jj3o z`phl0L_`?GrJ_A#1_Mjuax!&(b0{}QkVBs_kl#`g6^SHz%*@ib0Oa$VQ&)a_?>_|z zbcQnogNpg^AGSw%dwAm<;>1v5SOkB0`PvDlxyOV3&*2f#g*Z-C@KV#-h3Ezq{MD`0 z1nO3FPpcUvIJHczj>0p_+@a{k7&rMjrj=pH6k2L8X{Qy7t1p7F4`2YX_ diff --git a/release/server/docroot/images/topbox_iphone.jpg b/release/server/docroot/images/topbox_iphone.jpg deleted file mode 100644 index c6b0f8a24cd03457c17b698028b7c56d621d0fbb..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 423 zcmex=C5UDGKfoZ!!NAI>z|1Jfz$D1XEXer(2ty3e z*~~ykA^~;|E=EQqKA0&0^f4<7Gm^N7sF1LjxCFDZ3acVJhpHfw6h!m?TMRrv4NQW} zf(-Tyx1XQB@jGE}Y8U}fi7AzZCsS>Jid;Qi978H@B_$kSj*5={|9En`id%ANsi`vaMuugAbJe1h z?aTt}RE`|icAO%iC(POP_%rv;A{Whsqy-E;S#no>1Q=K(7;f2_>xnB$^Z^ZI@O1Ta JS?83{1OPAQFaiJo literal 0 HcmV?d00001 diff --git a/release/server/docroot/images/up@2x.png b/release/server/docroot/images/up@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..d990ce0d8fb2cf6386608432f9a5e8cf48161078 GIT binary patch literal 271 zcmV+q0r38bP)ug1}Ww%}+*tvEbT3l1+-i^CJu;P6JT z;)p=6;D|!ca73d2a73fOaMW)t6g|L4^C9w1pEcI~=x!>)gb5QSOqeiX!twPLU;r+0 Vp{p>?rG@|i002ovPDHLkV1h>9Zu - + Folder

- -

BROWSE AND IMPORT

-

{folder.name} {if pageIndex} - PAGE {page} OF {pages}{end pageIndex}

- -
{if pageIndex} {end pageIndex} up Libraries
- -
-
    - {loop element} -
  • {element.name}

    {element.download} {element.browse}
  • - {end element} +
    + +
    +
    {if pageIndex} {end pageIndex} up
    + + +
    +

    {folder.name}

    +
    + +
    + Libraries {library.name} {loop path} {path.name} {end path} +
    +
    + {if pageIndex} + + + {end pageIndex} + + {if alphaIndex} + + + {end alphaIndex} + +
    +
    + + +
    +
      + {loop element} +
    • +
      + +
      +
      +

      {element.name}

      +
      +
      {element.pages} {element.size} +
      +
      {element.download} {element.read} {element.browse} +
      +
      + {element.status} +
    • + {end element} +
    +
     
    +
    +
    + + {if index} +
    + {if alphaIndex} + +
    +
     
    -
    -
     
    - - {if alphaIndex} - -
    - -
     
    -
    -
     
    - - {end alphaIndex} - - - {if pageIndex} - -
    - -
     
    -
    -
     
    - {end pageIndex} - - - - + + {end alphaIndex} + + + {if pageIndex} + +
    + +
     
    +
    + {end pageIndex} +
+ {end index} + + + + + diff --git a/release/server/templates/folder_iphone.tpl b/release/server/templates/folder_iphone.tpl index 231a6843..b94e2c2c 100644 --- a/release/server/templates/folder_iphone.tpl +++ b/release/server/templates/folder_iphone.tpl @@ -1,62 +1,113 @@ - + Folder
- -

BROWSE AND IMPORT

-

{folder.name} {if pageIndex} - PAGE {page} OF {pages} {end pageIndex}

- -
Libraries up {if pageIndex} {end pageIndex}
- -
-
    - {loop element} -
  • {element.name}

    {element.download} {element.browse}
  • - {end element} +
    + +
    +
    {if pageIndex} {end pageIndex} up
    + + +
    +
    + +
    + Libraries {loop path} {path.name} {end path} +
    +
    + {if pageIndex} + + + {end pageIndex} + + {if alphaIndex} + + + {end alphaIndex} + +
    +
    + + +
    +
      + {loop element} +
    • +
      + +
      +
      +

      {element.name}

      +
      +
      {element.pages} {element.size} +
      +
      {element.download} {element.read} {element.browse} +
      +
      + {element.status} +
    • + {end element} +
    +
     
    +
    +
    + + {if index} +
    + {if alphaIndex} + +
    +
     
    -
    -
    -
     
    - - {if alphaIndex} -
    - -
     
    -
    -
     
    - - {end alphaIndex} - - - {if pageIndex} - -
    - -
     
    -
    -
     
    - - {end pageIndex} - - - +
+ + {end alphaIndex} + + + {if pageIndex} + +
+ +
 
+
+ {end pageIndex} +
+ {end index} + + + + + diff --git a/release/server/templates/libraries_ipad.tpl b/release/server/templates/libraries_ipad.tpl index e4bb6a35..e1c7b6eb 100644 --- a/release/server/templates/libraries_ipad.tpl +++ b/release/server/templates/libraries_ipad.tpl @@ -3,20 +3,24 @@ - Login + Libraries
-

LIBRARIES

-

Select a comic library

+

Libraries

-
 
\ No newline at end of file diff --git a/release/server/templates/libraries_iphone.tpl b/release/server/templates/libraries_iphone.tpl index a438be81..e1c7b6eb 100644 --- a/release/server/templates/libraries_iphone.tpl +++ b/release/server/templates/libraries_iphone.tpl @@ -3,20 +3,24 @@ - Login + Libraries -
-

LIBRARIES

-

Select a comic library

+
+

Libraries

-

    +

-
 
\ No newline at end of file From 3eecf65a8a3f2537a01d2ca21be677611ce8b59f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Tue, 29 Jul 2014 20:11:50 +0200 Subject: [PATCH 29/34] added support for opening next/previous comics from iOS remote viewer --- .../server/controllers/comiccontroller.cpp | 34 +++++++++++++++++-- .../controllers/librariescontroller.cpp | 5 +-- YACReaderLibrary/server/requestmapper.cpp | 4 +-- 3 files changed, 37 insertions(+), 6 deletions(-) diff --git a/YACReaderLibrary/server/controllers/comiccontroller.cpp b/YACReaderLibrary/server/controllers/comiccontroller.cpp index 188e5d52..26bfaed7 100644 --- a/YACReaderLibrary/server/controllers/comiccontroller.cpp +++ b/YACReaderLibrary/server/controllers/comiccontroller.cpp @@ -22,6 +22,8 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); qulonglong comicId = pathElements.at(4).toULongLong(); + bool remoteComic = path.contains("remote"); + //TODO //if(pathElements.size() == 6) //{ @@ -44,7 +46,8 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) ComicDB comic = DBHelper::getComicInfo(libraryName, comicId); - session.setDownloadedComic(comic.info.hash); + if(!remoteComic) + session.setDownloadedComic(comic.info.hash); Comic * comicFile = FactoryComic::newComic(libraries.getPath(libraryName)+comic.path); @@ -69,7 +72,34 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) response.setHeader("Content-Type", "plain/text; charset=ISO-8859-1"); //TODO this field is not used by the client! response.writeText(QString("library:%1\r\n").arg(libraryName)); - response.writeText(comic.toTXT(),true); + response.writeText(QString("libraryId:%1\r\n").arg(pathElements.at(2))); + if(remoteComic) //send previous and next comics id + { + QList siblings = DBHelper::getFolderComicsFromLibrary(libraryName, comic.parentId); + bool found = false; + int i; + for(i = 0; i < siblings.length(); i++) + { + if (siblings.at(i)->id == comic.id) + { + found = true; + break; + } + } + if(found) + { + if(i>0) + response.writeText(QString("previousComic:%1\r\n").arg(siblings.at(i-1)->id)); + if(iid)); + } + else + { + //ERROR + } + response.writeText(comic.toTXT(),true); + qDeleteAll(siblings); + } } else { diff --git a/YACReaderLibrary/server/controllers/librariescontroller.cpp b/YACReaderLibrary/server/controllers/librariescontroller.cpp index a154b7f8..658674cc 100644 --- a/YACReaderLibrary/server/controllers/librariescontroller.cpp +++ b/YACReaderLibrary/server/controllers/librariescontroller.cpp @@ -17,7 +17,8 @@ void LibrariesController::service(HttpRequest& request, HttpResponse& response) if(postData.length()>0) { QList data = postData.split("\n"); if(data.length() > 2) { - //ONLY COMICS ARE UPDATED, DEVICE CHARACTERISTICS ARE INMUTABLE + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); QList comics = data.at(2).split(":").at(1).split("\t"); foreach(QString hash,comics) { session.setComicOnDevice(hash); @@ -52,7 +53,7 @@ void LibrariesController::service(HttpRequest& request, HttpResponse& response) } else //values by default, only for debug purposes. { - session.setDeviceType("iphone"); + session.setDeviceType("ipad"); session.setDisplayType("@2x"); } diff --git a/YACReaderLibrary/server/requestmapper.cpp b/YACReaderLibrary/server/requestmapper.cpp index 5bbd9199..d010a4c1 100644 --- a/YACReaderLibrary/server/requestmapper.cpp +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -33,7 +33,7 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { QRegExp folder("/library/.+/folder/[0-9]+/?");//get comic content QRegExp folderInfo("/library/.+/folder/[0-9]+/info/?"); //get folder info QRegExp comic("/library/.+/comic/[0-9]+/?"); //get comic info - QRegExp comicOpen("/library/.+/comic/[0-9]+/open/?"); //the server will open for reading the comic + QRegExp comicOpen("/library/.+/comic/[0-9]+/remote/?"); //the server will open for reading the comic QRegExp comicClose("/library/.+/comic/[0-9]+/close/?"); //the server will close the comic and free memory QRegExp cover("/library/.+/cover/[0-9a-f]+.jpg"); //get comic cover (navigation) QRegExp comicPage("/library/.+/comic/[0-9]+/page/[0-9]+/?"); //get comic page @@ -69,7 +69,7 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { { CoverController().service(request, response); } - else if(comic.exactMatch(path)) + else if(comic.exactMatch(path) || comicOpen.exactMatch(path)) { ComicController().service(request, response); } From 20c596cec9e6c60562bf017da45817153f299277 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 30 Jul 2014 18:28:01 +0200 Subject: [PATCH 30/34] added sync back reading progress for remote reading added covers to remote folders (it could cause performance issues) added read next/previous to remote reading --- YACReaderLibrary/YACReaderLibrary.pro | 3 +- YACReaderLibrary/db_helper.cpp | 16 ++++++ YACReaderLibrary/db_helper.h | 1 + YACReaderLibrary/images.qrc | 2 + .../server/controllers/comiccontroller.cpp | 2 +- .../server/controllers/covercontroller.cpp | 10 ++++ .../server/controllers/foldercontroller.cpp | 12 ++++- .../controllers/updatecomiccontroller.cpp | 46 ++++++++++++++++++ .../controllers/updatecomiccontroller.h | 22 +++++++++ YACReaderLibrary/server/requestmapper.cpp | 8 ++- YACReaderLibrary/server/server.pri | 6 ++- images/f.png | Bin 0 -> 710 bytes images/f_retina.png | Bin 0 -> 1371 bytes 13 files changed, 120 insertions(+), 8 deletions(-) create mode 100644 YACReaderLibrary/server/controllers/updatecomiccontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/updatecomiccontroller.h create mode 100644 images/f.png create mode 100644 images/f_retina.png diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index bb4f8252..429511e0 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -28,8 +28,7 @@ LIBS += -lpoppler-qt4 INCLUDEPATH += ../dependencies/poppler/include/qt4 } -QMAKE_CXXFLAGS_RELEASE += /MP /Ob2 /Oi /Ot /GT /GL -QMAKE_LFLAGS_RELEASE += /LTCG + CONFIG -= embed_manifest_exe } diff --git a/YACReaderLibrary/db_helper.cpp b/YACReaderLibrary/db_helper.cpp index 39b18751..f7687689 100644 --- a/YACReaderLibrary/db_helper.cpp +++ b/YACReaderLibrary/db_helper.cpp @@ -22,6 +22,7 @@ #include "qnaturalsorting.h" +#include "QsLog.h" //server YACReaderLibraries DBHelper::getLibraries() @@ -297,6 +298,21 @@ void DBHelper::update(const Folder & folder, QSqlDatabase &db) updateFolderInfo.bindValue(":id", folder.id); updateFolderInfo.exec(); } + +void DBHelper::updateProgress(qulonglong libraryId, const ComicInfo &comicInfo) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + ComicDB comic = DBHelper::loadComic(comicInfo.id,db); + comic.info.currentPage = comicInfo.currentPage; + comic.info.hasBeenOpened = true; + + DBHelper::update(&comic.info,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); +} //inserts qulonglong DBHelper::insert(Folder * folder, QSqlDatabase & db) { diff --git a/YACReaderLibrary/db_helper.h b/YACReaderLibrary/db_helper.h index 014e65c9..146d60d5 100644 --- a/YACReaderLibrary/db_helper.h +++ b/YACReaderLibrary/db_helper.h @@ -42,6 +42,7 @@ public: static void update(ComicInfo * comicInfo, QSqlDatabase & db); static void updateRead(ComicInfo * comicInfo, QSqlDatabase & db); static void update(const Folder & folder, QSqlDatabase & db); + static void updateProgress(qulonglong libraryId,const ComicInfo & comicInfo); //TODO change libraryName by libraryId in all methods. //queries static QList getFoldersFromParent(qulonglong parentId, QSqlDatabase & db, bool sort = true); static QList getSortedComicsFromParent(qulonglong parentId, QSqlDatabase & db); diff --git a/YACReaderLibrary/images.qrc b/YACReaderLibrary/images.qrc index 750630a2..e444a4c2 100644 --- a/YACReaderLibrary/images.qrc +++ b/YACReaderLibrary/images.qrc @@ -110,6 +110,8 @@ ../images/shortcuts_group_page.png ../images/shortcuts_group_reading.png ../images/shortcuts_group_visualization.png + ../images/f.png + ../images/f_retina.png diff --git a/YACReaderLibrary/server/controllers/comiccontroller.cpp b/YACReaderLibrary/server/controllers/comiccontroller.cpp index 26bfaed7..43e9ea68 100644 --- a/YACReaderLibrary/server/controllers/comiccontroller.cpp +++ b/YACReaderLibrary/server/controllers/comiccontroller.cpp @@ -97,9 +97,9 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) { //ERROR } - response.writeText(comic.toTXT(),true); qDeleteAll(siblings); } + response.writeText(comic.toTXT(),true); } else { diff --git a/YACReaderLibrary/server/controllers/covercontroller.cpp b/YACReaderLibrary/server/controllers/covercontroller.cpp index 80cc394c..4821bfdd 100644 --- a/YACReaderLibrary/server/controllers/covercontroller.cpp +++ b/YACReaderLibrary/server/controllers/covercontroller.cpp @@ -23,6 +23,8 @@ void CoverController::service(HttpRequest& request, HttpResponse& response) QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); QString fileName = pathElements.at(4); + bool folderCover = request.getParameter("folderCover").length()>0; + //response.writeText(path+"
"); //response.writeText(libraryName+"
"); //response.writeText(libraries.value(libraryName)+"/.yacreaderlibrary/covers/"+fileName+"
"); @@ -62,6 +64,14 @@ void CoverController::service(HttpRequest& request, HttpResponse& response) p.drawImage((width-img.width())/2,(height-img.height())/2,img); + if(folderCover) + { + if(session.getDisplayType()=="@2x") + p.drawImage(0,0,QImage(":/images/f_retina.png")); + else + p.drawImage(0,0,QImage(":/images/f.png")); + } + QByteArray ba; QBuffer buffer(&ba); buffer.open(QIODevice::WriteOnly); diff --git a/YACReaderLibrary/server/controllers/foldercontroller.cpp b/YACReaderLibrary/server/controllers/foldercontroller.cpp index 7300adf9..b6239106 100644 --- a/YACReaderLibrary/server/controllers/foldercontroller.cpp +++ b/YACReaderLibrary/server/controllers/foldercontroller.cpp @@ -180,7 +180,15 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) if(item->isDir()) { t.setVariable(QString("element%1.class").arg(i),"folder"); - t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); + + QList children = DBHelper::getFolderComicsFromLibrary(libraryName, item->id); + if(children.length()>0) + { + const ComicDB * comic = static_cast(children.at(0)); + t.setVariable(QString("element%1.image.url").arg(i),QString("/library/%1/cover/%2.jpg?folderCover=true").arg(libraryId).arg(comic->info.hash)); + } + else + t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); t.setVariable(QString("element%1.browse").arg(i),QString("browse").arg(QString("/library/%1/folder/%2").arg(libraryId).arg(item->id))); @@ -217,7 +225,7 @@ void FolderController::service(HttpRequest& request, HttpResponse& response) if(comic->info.hasBeenOpened) t.setVariable(QString("element%1.pages").arg(i),QString("%1/%2 pages").arg(comic->info.currentPage).arg(comic->info.numPages.toInt())); else - t.setVariable(QString("element%1.pages").arg(i),QString("%1").arg(comic->info.numPages.toInt())); + t.setVariable(QString("element%1.pages").arg(i),QString("%1 pages").arg(comic->info.numPages.toInt())); if(comic->info.read) t.setVariable(QString("element%1.status").arg(i), QString("
")); diff --git a/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp b/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp new file mode 100644 index 00000000..793d1b72 --- /dev/null +++ b/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp @@ -0,0 +1,46 @@ +#include "updatecomiccontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +#include "template.h" +#include "../static.h" + +#include "comic_db.h" +#include "comic.h" + +#include "QsLog.h" + +UpdateComicController::UpdateComicController(){} + +void UpdateComicController::service(HttpRequest &request, HttpResponse &response) +{ + HttpSession session=Static::sessionStore->getSession(request,response,false); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + qulonglong libraryId = pathElements.at(2).toULongLong(); + QString libraryName = DBHelper::getLibraryName(libraryId); + qulonglong comicId = pathElements.at(4).toULongLong(); + + QString postData = QString::fromUtf8(request.getBody()); + + QLOG_INFO() << "POST DATA: " << postData; + + if(postData.length()>0) { + QList data = postData.split("\n"); + int currentPage = data.at(0).split(":").at(1).toInt(); + ComicInfo info; + info.currentPage = currentPage; + info.id = comicId; + DBHelper::updateProgress(libraryId,info); + } + else + { + response.setStatus(412,"No comic info received"); + response.writeText("",true); + return; + } + + response.write("OK",true); +} diff --git a/YACReaderLibrary/server/controllers/updatecomiccontroller.h b/YACReaderLibrary/server/controllers/updatecomiccontroller.h new file mode 100644 index 00000000..13ec4f58 --- /dev/null +++ b/YACReaderLibrary/server/controllers/updatecomiccontroller.h @@ -0,0 +1,22 @@ +#ifndef UPDATECOMICCONTROLLER_H +#define UPDATECOMICCONTROLLER_H + + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + + +class UpdateComicController : public HttpRequestHandler +{ + Q_OBJECT + Q_DISABLE_COPY(UpdateComicController); + +public: + UpdateComicController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // UPDATECOMICCONTROLLER_H diff --git a/YACReaderLibrary/server/requestmapper.cpp b/YACReaderLibrary/server/requestmapper.cpp index d010a4c1..5be29a92 100644 --- a/YACReaderLibrary/server/requestmapper.cpp +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -18,6 +18,7 @@ #include "controllers/comiccontroller.h" #include "controllers/folderinfocontroller.h" #include "controllers/pagecontroller.h" +#include "controllers/updatecomiccontroller.h" #include "controllers/errorcontroller.h" #include "db_helper.h" @@ -34,6 +35,7 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { QRegExp folderInfo("/library/.+/folder/[0-9]+/info/?"); //get folder info QRegExp comic("/library/.+/comic/[0-9]+/?"); //get comic info QRegExp comicOpen("/library/.+/comic/[0-9]+/remote/?"); //the server will open for reading the comic + QRegExp comicUpdate("/library/.+/comic/[0-9]+/update/?"); //get comic info QRegExp comicClose("/library/.+/comic/[0-9]+/close/?"); //the server will close the comic and free memory QRegExp cover("/library/.+/cover/[0-9a-f]+.jpg"); //get comic cover (navigation) QRegExp comicPage("/library/.+/comic/[0-9]+/page/[0-9]+/?"); //get comic page @@ -76,7 +78,11 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { else if(comicPage.exactMatch(path)) { PageController().service(request,response); - } + } + else if(comicUpdate.exactMatch(path)) + { + UpdateComicController().service(request, response); + } } else { diff --git a/YACReaderLibrary/server/server.pri b/YACReaderLibrary/server/server.pri index dc1f2466..9e3b2920 100644 --- a/YACReaderLibrary/server/server.pri +++ b/YACReaderLibrary/server/server.pri @@ -12,7 +12,8 @@ HEADERS += \ $$PWD/controllers/librariescontroller.h \ $$PWD/controllers/pagecontroller.h \ $$PWD/controllers/sessionmanager.h \ - $$PWD/controllers/covercontroller.h + $$PWD/controllers/covercontroller.h \ + server/controllers/updatecomiccontroller.h SOURCES += \ $$PWD/static.cpp \ @@ -25,7 +26,8 @@ SOURCES += \ $$PWD/controllers/librariescontroller.cpp \ $$PWD/controllers/pagecontroller.cpp \ $$PWD/controllers/sessionmanager.cpp \ - $$PWD/controllers/covercontroller.cpp + $$PWD/controllers/covercontroller.cpp \ + server/controllers/updatecomiccontroller.cpp include(lib/bfLogging/bfLogging.pri) include(lib/bfHttpServer/bfHttpServer.pri) diff --git a/images/f.png b/images/f.png new file mode 100644 index 0000000000000000000000000000000000000000..232a785d75767e8fae82aae15cf36213ecec1598 GIT binary patch literal 710 zcmeAS@N?(olHy`uVBq!ia0vp^0YF^A!3HE#-<5s?Qj#UE5hcO-X(i=}MX3yqDfvmM z3ZA)%>8U}fi7AzZCsS=07??6VT^vIyZoRp?F|Wx%=6In&AK%@WTIO07-ooXZUtUl* z=zZxkL;CU$mRgpXt6k(5?rU5v-CSLLea^|&5(Yl4@|t?i#EaQKf9x+RDEhkJ)h5W& zmV<#sz=46Yfq{ubff2}JQ~|LXSR5E~J@&ugUVnYHxa-DptGRR7B>RuY$++K3Fwpo? zwg1JEOIcrE+05(P5x4(pbfe`Smz>!qcV%N4xSM{4yqT87{omqR6+@{FI$`{63> zfOAt%wZ)##joXp&XA{5lsY`J>^X446{odYt|MD+e($;Nxs^T+e+3q8k_SY`7(hQS} zoYBX;_UPom{na(w8@HUTi0X^!oFVGSrf~fb*gY4HeR==2`Do|?k2iKl%+@B#eAj7H zets@w!GUUl-Ojuf^RgZ{QL&CicNKr?l_(xm9QL7*^%keCEkd%VK^x^JT5=>vb+yrXKE@%#ogm z5*nB>!^kp$0T@RN&lb=7vCV4P@r0|6H_V+Po~;1FtGgfba4!+xb^0)p?|bH!-0>=NnZ{;yDAvpV5zfC#%l7r zMg!NmD@02Dtld|nt}@y=VQ-=;=c(uuXT%*ozUn<#U8Nqk#H1-??T6jf@Bh|anyfnE z(dPH>tIb-3h=KzP2a`(!$rOhGO9z8v1Ca8tU!(V0e%|@#4}bg!wXpv$ zAlBWQzN?k3+407m8iBGhx%P(zJEr`Oh&<+II(_=|hc`FRZ@s=yfS)g}O<8#-yl>d`6m|J%Ax^1T4|>fp|J{n7{AgBxzkgcR2ip1f?|&OzZMe(cw^Tc+Mks%odF}hVyUkPQHyyoR z_ci9NX<@uCOGM}Tt%6~(93M`UY`YdH&NOG{@oTfT+BwWGi+;O^TU+66t$c;5gOxAP zV~kAKU(T|7%9G3ruoNN`+1_SfpKfj_4AW_9r!yj@pORrNYe z?eTejJ-ZC!`$n00-S(`@*V<11=%u*cWI2!vRJ3fywr?Vd+aGxs@9ME_p1vu*AnEX{ zyq4>ycvt6t>hoJn#HKo~fBl!-G}<3lIUhb%VzT_7ZNAmF#~ZI5X9%vj zfAoXilM8!(vBn=2@PG97cZ}dO8&>(Q^&JBGuQms~h;H!P`E*5<_=4{>t~+1EHi(o; z?GsYitKtAl0??EIq267|ikGjbwd|UI=<8Aqx0!8*+}EdT{j_~v-?@p&O#Xh^o}Uu^ z&m|vof300^ShME1$^p3}-&OAI;gkF?^oA`{XZ>%9hkNH8ogOylK9}61Epm$)th1tS zoGQQUWg(hi9=^Zi@9skGHmi4+Tkrnf!t_pRHAC?&voOYICPqNw&zEk7%{z^*GM!nr ze*ZI}gvhjGS_kI%^@KK9%1Q!N?&J5FuCO}F;L0k0@po@-zi;}^_1~o?wEtrJguZo6 zn-8g1tOr>rlUUXKf5nEW2E9)MZMP{LsB@{Xm)qaA$NtldtI0d=AOA7w-alWDzlUGO zzhc%_U}9lZau6Vea%q^ygf;N+33hQ57~Pbr{&jjnv0l--1|}VMedas;>-X>$&#(RJ zUA$rb>wn?1cP&1c0L+0$r^iQiAOHP>MdsM;S8v|rsUcY~i+>g&CN7P*mdKI;Vst0RKTg#Q*>R literal 0 HcmV?d00001 From c67be2325239d9dbaaf1492787b27bc1af129436 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 30 Jul 2014 18:30:45 +0200 Subject: [PATCH 31/34] fixed release config regresion --- YACReaderLibrary/YACReaderLibrary.pro | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro index 429511e0..bb4f8252 100644 --- a/YACReaderLibrary/YACReaderLibrary.pro +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -28,7 +28,8 @@ LIBS += -lpoppler-qt4 INCLUDEPATH += ../dependencies/poppler/include/qt4 } - +QMAKE_CXXFLAGS_RELEASE += /MP /Ob2 /Oi /Ot /GT /GL +QMAKE_LFLAGS_RELEASE += /LTCG CONFIG -= embed_manifest_exe } From 61e7bbc5d13c9a010ebfaa25a614532b4bc0cd12 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Wed, 30 Jul 2014 21:56:34 +0200 Subject: [PATCH 32/34] fixed simultaneously importing and remote reading --- .../server/controllers/comiccontroller.cpp | 16 ++++++- .../controllers/librariescontroller.cpp | 12 +++++ .../server/controllers/pagecontroller.cpp | 36 ++++++++++---- .../server/lib/bfHttpServer/httpsession.cpp | 48 +++++++++++++++++-- .../server/lib/bfHttpServer/httpsession.h | 10 +++- YACReaderLibrary/server/requestmapper.cpp | 8 ++-- 6 files changed, 113 insertions(+), 17 deletions(-) diff --git a/YACReaderLibrary/server/controllers/comiccontroller.cpp b/YACReaderLibrary/server/controllers/comiccontroller.cpp index 43e9ea68..88c39abf 100644 --- a/YACReaderLibrary/server/controllers/comiccontroller.cpp +++ b/YACReaderLibrary/server/controllers/comiccontroller.cpp @@ -9,6 +9,8 @@ #include "comic_db.h" #include "comic.h" +#include "QsLog.h" + #include ComicController::ComicController() {} @@ -22,7 +24,7 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); qulonglong comicId = pathElements.at(4).toULongLong(); - bool remoteComic = path.contains("remote"); + bool remoteComic = path.endsWith("remote"); //TODO //if(pathElements.size() == 6) @@ -67,7 +69,17 @@ void ComicController::service(HttpRequest& request, HttpResponse& response) if(thread != NULL) thread->start(); - session.setCurrentComic(comic.id, comicFile); + if(remoteComic) + { + QLOG_INFO() << "remote comic requested"; + session.setCurrentRemoteComic(comic.id, comicFile); + + } + else + { + QLOG_INFO() << "comic requested"; + session.setCurrentComic(comic.id, comicFile); + } response.setHeader("Content-Type", "plain/text; charset=ISO-8859-1"); //TODO this field is not used by the client! diff --git a/YACReaderLibrary/server/controllers/librariescontroller.cpp b/YACReaderLibrary/server/controllers/librariescontroller.cpp index 658674cc..6b555018 100644 --- a/YACReaderLibrary/server/controllers/librariescontroller.cpp +++ b/YACReaderLibrary/server/controllers/librariescontroller.cpp @@ -5,6 +5,7 @@ #include "template.h" #include "../static.h" +#include "QsLog.h" LibrariesController::LibrariesController() {} @@ -24,6 +25,14 @@ void LibrariesController::service(HttpRequest& request, HttpResponse& response) session.setComicOnDevice(hash); } } + else + { + if(data.length()>1) + { + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); + } + } } } else @@ -41,6 +50,9 @@ void LibrariesController::service(HttpRequest& request, HttpResponse& response) //response.writeText(postData); QList data = postData.split("\n"); + + QLOG_INFO() << "Data lenght : " << data.length(); + if(data.length() > 2) { session.setDeviceType(data.at(0).split(":").at(1)); diff --git a/YACReaderLibrary/server/controllers/pagecontroller.cpp b/YACReaderLibrary/server/controllers/pagecontroller.cpp index 9c2df45a..e6ccde83 100644 --- a/YACReaderLibrary/server/controllers/pagecontroller.cpp +++ b/YACReaderLibrary/server/controllers/pagecontroller.cpp @@ -7,6 +7,8 @@ #include #include +#include + #include "db_helper.h" PageController::PageController() {} @@ -16,6 +18,7 @@ void PageController::service(HttpRequest& request, HttpResponse& response) HttpSession session=Static::sessionStore->getSession(request,response,false); QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + bool remote = path.endsWith("remote"); //QByteArray path2=request.getPath(); //qDebug("PageController: request to -> %s ",path2.data()); @@ -27,10 +30,24 @@ void PageController::service(HttpRequest& request, HttpResponse& response) //qDebug("lib name : %s",pathElements.at(2).data()); - Comic * comicFile = session.getCurrentComic(); - if(session.getCurrentComicId() != 0 && !QPointer(comicFile).isNull()) + Comic * comicFile; + qulonglong currentComicId; + if(remote) + { + QLOG_INFO() << "se recupera comic remoto para servir páginas"; + comicFile = session.getCurrentRemoteComic(); + currentComicId = session.getCurrentRemoteComicId(); + } + else + { + QLOG_INFO() << "se recupera comic para servir páginas"; + comicFile = session.getCurrentComic(); + currentComicId = session.getCurrentComicId(); + } + + if(currentComicId != 0 && !QPointer(comicFile).isNull()) { - if(comicId == session.getCurrentComicId() && page < comicFile->numPages()) + if(comicId == currentComicId && page < comicFile->numPages()) { if(comicFile->pageIsLoaded(page)) { @@ -56,11 +73,14 @@ void PageController::service(HttpRequest& request, HttpResponse& response) } else { - if(comicId != session.getCurrentComicId()) + if(comicId != currentComicId) { - //delete comicFile; - session.dismissCurrentComic(); - } + //delete comicFile; + if(remote) + session.dismissCurrentRemoteComic(); + else + session.dismissCurrentComic(); + } response.setStatus(404,"not found"); //TODO qué mensaje enviar response.write("404 not found",true); } @@ -73,4 +93,4 @@ void PageController::service(HttpRequest& request, HttpResponse& response) //response.write(t.toLatin1(),true); -} \ No newline at end of file +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp index 3f1cbd21..c09cb553 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp @@ -16,6 +16,8 @@ HttpSession::HttpSession(bool canStore) { dataPtr->id=QUuid::createUuid().toString().toLatin1(); dataPtr->yacreaderSessionData.comic = 0; dataPtr->yacreaderSessionData.comicId = 0; + dataPtr->yacreaderSessionData.remoteComic = 0; + dataPtr->yacreaderSessionData.remoteComicId = 0; #ifdef SUPERVERBOSE qDebug("HttpSession: created new session data with id %s",dataPtr->id.data()); #endif @@ -210,7 +212,7 @@ QSet HttpSession::getDownloadedComics() else return QSet(); } -//current comic +//current comic (import) qulonglong HttpSession::getCurrentComicId() { if(dataPtr) @@ -249,6 +251,46 @@ void HttpSession::setCurrentComic(qulonglong id, Comic * comic) } } +//current comic (read) +qulonglong HttpSession::getCurrentRemoteComicId() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.remoteComicId ; + else + return 0; +} +Comic* HttpSession::getCurrentRemoteComic() +{ + if(dataPtr) + { + return dataPtr->yacreaderSessionData.remoteComic ; + } + else + return 0; +} +void HttpSession::dismissCurrentRemoteComic() +{ + if(dataPtr) + { + if(dataPtr->yacreaderSessionData.remoteComic != 0) + { + dataPtr->yacreaderSessionData.remoteComic->deleteLater(); + dataPtr->yacreaderSessionData.remoteComic = 0; + } + dataPtr->yacreaderSessionData.remoteComicId = 0; + } +} +void HttpSession::setCurrentRemoteComic(qulonglong id, Comic * comic) +{ + if(dataPtr) + { + dismissCurrentRemoteComic(); + dataPtr->yacreaderSessionData.remoteComicId = id; + dataPtr->yacreaderSessionData.remoteComic = comic; + } +} + + QString HttpSession::getDeviceType() { if(dataPtr) @@ -269,8 +311,8 @@ void HttpSession::setDeviceType(const QString & device) { if(dataPtr) { - dataPtr->yacreaderSessionData.comicsOnDevice.clear(); //TODO crear un método clear que limpie la sesión completamente - dataPtr->yacreaderSessionData.downloadedComics.clear(); + //dataPtr->yacreaderSessionData.comicsOnDevice.clear(); //TODO crear un método clear que limpie la sesión completamente + //dataPtr->yacreaderSessionData.downloadedComics.clear(); dataPtr->yacreaderSessionData.device = device; } } diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h index d2e6c6b7..1a0a42e1 100644 --- a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h @@ -102,11 +102,17 @@ public: QSet getComicsOnDevice(); QSet getDownloadedComics(); - //current comic + //current comic (import) qulonglong getCurrentComicId(); Comic * getCurrentComic(); void dismissCurrentComic(); void setCurrentComic(qulonglong id, Comic * comic); + + //current comic (read) + qulonglong getCurrentRemoteComicId(); + Comic * getCurrentRemoteComic(); + void dismissCurrentRemoteComic(); + void setCurrentRemoteComic(qulonglong id, Comic * comic); //device identification QString getDeviceType(); @@ -136,11 +142,13 @@ private: QString device; QString display; qulonglong comicId; + qulonglong remoteComicId; QStack navigationPath; QStack foldersPath; Comic * comic; + Comic * remoteComic; }; struct HttpSessionData { diff --git a/YACReaderLibrary/server/requestmapper.cpp b/YACReaderLibrary/server/requestmapper.cpp index 5be29a92..f867b944 100644 --- a/YACReaderLibrary/server/requestmapper.cpp +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -39,6 +39,7 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { QRegExp comicClose("/library/.+/comic/[0-9]+/close/?"); //the server will close the comic and free memory QRegExp cover("/library/.+/cover/[0-9a-f]+.jpg"); //get comic cover (navigation) QRegExp comicPage("/library/.+/comic/[0-9]+/page/[0-9]+/?"); //get comic page + QRegExp comicPageRemote("/library/.+/comic/[0-9]+/page/[0-9]+/remote?"); //get comic page (remote reading) QRegExp library("/library/([0-9]+)/.+"); //permite verificar que la biblioteca solicitada existe @@ -52,6 +53,7 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { else { + //se comprueba que la sesión sea la correcta con el fin de evitar accesos no autorizados HttpSession session=Static::sessionStore->getSession(request,response,false); if(!session.isNull() && session.contains("ySession")) @@ -73,11 +75,11 @@ void RequestMapper::service(HttpRequest& request, HttpResponse& response) { } else if(comic.exactMatch(path) || comicOpen.exactMatch(path)) { - ComicController().service(request, response); + ComicController().service(request, response); } - else if(comicPage.exactMatch(path)) + else if(comicPage.exactMatch(path) || comicPageRemote.exactMatch(path)) { - PageController().service(request,response); + PageController().service(request,response); } else if(comicUpdate.exactMatch(path)) { From bfc46dd0dfba5a27f212607ec82dd113f038433a Mon Sep 17 00:00:00 2001 From: Felix Kauselmann <2039670+selmf@users.noreply.github.com> Date: Sat, 2 Aug 2014 20:15:45 +0200 Subject: [PATCH 33/34] Invert double page info in manga mode --- YACReader/render.cpp | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/YACReader/render.cpp b/YACReader/render.cpp index ce91ac89..b8b3541e 100644 --- a/YACReader/render.cpp +++ b/YACReader/render.cpp @@ -1053,7 +1053,12 @@ QString Render::getCurrentPagesInformation() { QString s = QString::number(currentIndex+1); if (doublePage && (currentIndex+1 < (int)comic->numPages())) - s += "-"+QString::number(currentIndex+2); + { + if (doubleMangaPage) + s = QString::number(currentIndex+2) + "-" + s; + else + s += "-"+QString::number(currentIndex+2); + } s += "/"+QString::number(comic->numPages()); return s; } From 6e645838ec81d9dda385d98447dfa5b29b481fb3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Mon, 11 Aug 2014 20:46:44 +0200 Subject: [PATCH 34/34] added missing files (server) --- CHANGELOG.txt | 118 + COPYING.txt | 674 ++++++ INSTALL.txt | 31 + QsLog/QsLog.cpp | 249 ++ QsLog/QsLog.h | 137 ++ QsLog/QsLog.pri | 22 + QsLog/QsLogDest.cpp | 70 + QsLog/QsLogDest.h | 99 + QsLog/QsLogDestConsole.cpp | 55 + QsLog/QsLogDestConsole.h | 52 + QsLog/QsLogDestFile.cpp | 155 ++ QsLog/QsLogDestFile.h | 101 + QsLog/QsLogDestFunctor.cpp | 57 + QsLog/QsLogDestFunctor.h | 59 + QsLog/QsLogDisableForThisFile.h | 22 + QsLog/QsLogLevel.h | 45 + QsLog/QsLogSharedLibrary.pro | 35 + README.txt | 22 + YACReader.desktop | 12 + YACReader.pro | 3 + YACReader/YACReader.icns | Bin 0 -> 52094 bytes YACReader/YACReader.pri | 144 ++ YACReader/YACReader.pro | 98 + YACReader/bookmarks_dialog.cpp | 197 ++ YACReader/bookmarks_dialog.h | 45 + YACReader/configuration.cpp | 162 ++ YACReader/configuration.h | 94 + YACReader/goto_dialog.cpp | 84 + YACReader/goto_dialog.h | 32 + YACReader/goto_flow.cpp | 320 +++ YACReader/goto_flow.h | 110 + YACReader/goto_flow_decorationbar.cpp | 33 + YACReader/goto_flow_decorationbar.h | 13 + YACReader/goto_flow_gl.cpp | 158 ++ YACReader/goto_flow_gl.h | 39 + YACReader/goto_flow_toolbar.cpp | 122 + YACReader/goto_flow_toolbar.h | 33 + YACReader/goto_flow_widget.cpp | 75 + YACReader/goto_flow_widget.h | 41 + YACReader/icon.ico | Bin 0 -> 99678 bytes YACReader/icon.rc | 1 + YACReader/magnifying_glass.cpp | 292 +++ YACReader/magnifying_glass.h | 34 + YACReader/main.cpp | 118 + YACReader/main_window_viewer.cpp | 1352 +++++++++++ YACReader/main_window_viewer.h | 145 ++ YACReader/notifications_label_widget.cpp | 83 + YACReader/notifications_label_widget.h | 29 + YACReader/options_dialog.cpp | 307 +++ YACReader/options_dialog.h | 70 + YACReader/page_label_widget.cpp | 122 + YACReader/page_label_widget.h | 29 + YACReader/render.cpp | 1084 +++++++++ YACReader/render.h | 209 ++ YACReader/shortcuts_dialog.cpp | 67 + YACReader/shortcuts_dialog.h | 19 + YACReader/translator.cpp | 429 ++++ YACReader/translator.h | 102 + YACReader/viewer.cpp | 937 ++++++++ YACReader/viewer.h | 164 ++ YACReader/width_slider.cpp | 78 + YACReader/width_slider.h | 29 + YACReader/yacreader_es.qm | Bin 0 -> 13554 bytes YACReader/yacreader_es.ts | 792 +++++++ YACReader/yacreader_files.qrc | 12 + YACReader/yacreader_fr.ts | 791 +++++++ YACReader/yacreader_images.qrc | 87 + YACReader/yacreader_images_osx.qrc | 28 + YACReader/yacreader_images_win.qrc | 28 + YACReader/yacreader_local_client.cpp | 171 ++ YACReader/yacreader_local_client.h | 27 + YACReader/yacreader_nl.ts | 791 +++++++ YACReader/yacreader_pt.ts | 791 +++++++ YACReader/yacreader_ru.ts | 791 +++++++ YACReader/yacreader_source.ts | 791 +++++++ YACReader/yacreader_tr.ts | 725 ++++++ YACReaderLibrary.desktop | 12 + YACReaderLibrary/YACReaderLibrary.icns | Bin 0 -> 53727 bytes YACReaderLibrary/YACReaderLibrary.pro | 273 +++ YACReaderLibrary/add_library_dialog.cpp | 124 + YACReaderLibrary/add_library_dialog.h | 35 + YACReaderLibrary/bundle_creator.cpp | 13 + YACReaderLibrary/bundle_creator.h | 14 + YACReaderLibrary/classic_comics_view.cpp | 243 ++ YACReaderLibrary/classic_comics_view.h | 51 + YACReaderLibrary/comic_flow.cpp | 238 ++ YACReaderLibrary/comic_flow.h | 77 + YACReaderLibrary/comic_flow_widget.cpp | 345 +++ YACReaderLibrary/comic_flow_widget.h | 128 + YACReaderLibrary/comic_vine/comic_vine.pri | 44 + .../comic_vine/comic_vine_client.cpp | 164 ++ .../comic_vine/comic_vine_client.h | 41 + .../comic_vine/comic_vine_dialog.cpp | 722 ++++++ .../comic_vine/comic_vine_dialog.h | 130 ++ .../comic_vine/model/comics_model.cpp | 6 + .../comic_vine/model/comics_model.h | 18 + .../comic_vine/model/json_model.cpp | 6 + .../comic_vine/model/json_model.h | 19 + .../model/local_comic_list_model.cpp | 183 ++ .../comic_vine/model/local_comic_list_model.h | 42 + .../comic_vine/model/response_parser.cpp | 61 + .../comic_vine/model/response_parser.h | 27 + .../comic_vine/model/volume_comics_model.cpp | 172 ++ .../comic_vine/model/volume_comics_model.h | 41 + .../comic_vine/model/volumes_model.cpp | 161 ++ .../comic_vine/model/volumes_model.h | 50 + .../comic_vine/scraper_lineedit.cpp | 21 + .../comic_vine/scraper_lineedit.h | 19 + .../comic_vine/scraper_results_paginator.cpp | 75 + .../comic_vine/scraper_results_paginator.h | 34 + .../comic_vine/scraper_scroll_label.cpp | 53 + .../comic_vine/scraper_scroll_label.h | 25 + .../comic_vine/scraper_selector.cpp | 25 + .../comic_vine/scraper_selector.h | 28 + .../comic_vine/scraper_tableview.cpp | 61 + .../comic_vine/scraper_tableview.h | 18 + .../comic_vine/search_single_comic.cpp | 62 + .../comic_vine/search_single_comic.h | 22 + YACReaderLibrary/comic_vine/search_volume.cpp | 36 + YACReaderLibrary/comic_vine/search_volume.h | 21 + YACReaderLibrary/comic_vine/select_comic.cpp | 150 ++ YACReaderLibrary/comic_vine/select_comic.h | 34 + YACReaderLibrary/comic_vine/select_volume.cpp | 191 ++ YACReaderLibrary/comic_vine/select_volume.h | 39 + .../comic_vine/series_question.cpp | 46 + YACReaderLibrary/comic_vine/series_question.h | 23 + .../comic_vine/sort_volume_comics.cpp | 224 ++ .../comic_vine/sort_volume_comics.h | 99 + YACReaderLibrary/comic_vine/title_header.cpp | 53 + YACReaderLibrary/comic_vine/title_header.h | 22 + YACReaderLibrary/comics_remover.cpp | 29 + YACReaderLibrary/comics_remover.h | 28 + YACReaderLibrary/comics_view.cpp | 11 + YACReaderLibrary/comics_view.h | 47 + YACReaderLibrary/comics_view_transition.cpp | 74 + YACReaderLibrary/comics_view_transition.h | 31 + YACReaderLibrary/create_library_dialog.cpp | 206 ++ YACReaderLibrary/create_library_dialog.h | 61 + YACReaderLibrary/db/data_base_management.cpp | 686 ++++++ YACReaderLibrary/db/data_base_management.h | 61 + YACReaderLibrary/db/tableitem.cpp | 47 + YACReaderLibrary/db/tableitem.h | 27 + YACReaderLibrary/db/tablemodel.cpp | 660 ++++++ YACReaderLibrary/db/tablemodel.h | 120 + YACReaderLibrary/db/treeitem.cpp | 152 ++ YACReaderLibrary/db/treeitem.h | 79 + YACReaderLibrary/db/treemodel.cpp | 544 +++++ YACReaderLibrary/db/treemodel.h | 119 + YACReaderLibrary/db_helper.cpp | 704 ++++++ YACReaderLibrary/db_helper.h | 58 + YACReaderLibrary/empty_folder_widget.cpp | 89 + YACReaderLibrary/empty_folder_widget.h | 32 + .../export_comics_info_dialog.cpp | 92 + YACReaderLibrary/export_comics_info_dialog.h | 35 + YACReaderLibrary/export_library_dialog.cpp | 100 + YACReaderLibrary/export_library_dialog.h | 35 + YACReaderLibrary/files.qrc | 12 + YACReaderLibrary/grid_comics_view.cpp | 234 ++ YACReaderLibrary/grid_comics_view.h | 59 + YACReaderLibrary/icon.ico | Bin 0 -> 99678 bytes YACReaderLibrary/icon.rc | 3 + YACReaderLibrary/icon2.ico | Bin 0 -> 99678 bytes YACReaderLibrary/icon3.ico | Bin 0 -> 82726 bytes YACReaderLibrary/images.qrc | 117 + YACReaderLibrary/images_osx.qrc | 24 + YACReaderLibrary/images_win.qrc | 23 + .../import_comics_info_dialog.cpp | 111 + YACReaderLibrary/import_comics_info_dialog.h | 52 + YACReaderLibrary/import_library_dialog.cpp | 157 ++ YACReaderLibrary/import_library_dialog.h | 46 + YACReaderLibrary/import_widget.cpp | 385 +++ YACReaderLibrary/import_widget.h | 52 + YACReaderLibrary/library_creator.cpp | 635 +++++ YACReaderLibrary/library_creator.h | 86 + YACReaderLibrary/library_window.cpp | 2076 +++++++++++++++++ YACReaderLibrary/library_window.h | 323 +++ YACReaderLibrary/main.cpp | 226 ++ YACReaderLibrary/no_libraries_widget.cpp | 80 + YACReaderLibrary/no_libraries_widget.h | 19 + YACReaderLibrary/options_dialog.cpp | 69 + YACReaderLibrary/options_dialog.h | 18 + YACReaderLibrary/package_manager.cpp | 55 + YACReaderLibrary/package_manager.h | 24 + YACReaderLibrary/properties_dialog.cpp | 896 +++++++ YACReaderLibrary/properties_dialog.h | 141 ++ YACReaderLibrary/qml.qrc | 8 + YACReaderLibrary/qml/GridComicsView.qml | 295 +++ YACReaderLibrary/qml/YACReaderScrollView.qml | 336 +++ YACReaderLibrary/qml/page-macosx.png | Bin 0 -> 171 bytes YACReaderLibrary/qml/page.png | Bin 0 -> 155 bytes YACReaderLibrary/qml/reading.png | Bin 0 -> 374 bytes YACReaderLibrary/qml/star-macosx.png | Bin 0 -> 288 bytes YACReaderLibrary/qml/star.png | Bin 0 -> 242 bytes YACReaderLibrary/qml/tick.png | Bin 0 -> 488 bytes YACReaderLibrary/qml_osx.qrc | 6 + YACReaderLibrary/qml_win.qrc | 6 + YACReaderLibrary/rename_library_dialog.cpp | 76 + YACReaderLibrary/rename_library_dialog.h | 31 + .../server/controllers/comiccontroller.cpp | 124 + .../server/controllers/comiccontroller.h | 23 + .../server/controllers/covercontroller.cpp | 88 + .../server/controllers/covercontroller.h | 20 + .../server/controllers/dumpcontroller.cpp | 62 + .../server/controllers/dumpcontroller.h | 29 + .../server/controllers/errorcontroller.cpp | 26 + .../server/controllers/errorcontroller.h | 22 + .../controllers/fileuploadcontroller.cpp | 38 + .../server/controllers/fileuploadcontroller.h | 30 + .../server/controllers/foldercontroller.cpp | 329 +++ .../server/controllers/foldercontroller.h | 20 + .../controllers/folderinfocontroller.cpp | 39 + .../server/controllers/folderinfocontroller.h | 20 + .../server/controllers/formcontroller.cpp | 64 + .../server/controllers/formcontroller.h | 30 + .../controllers/librariescontroller.cpp | 93 + .../server/controllers/librariescontroller.h | 25 + .../server/controllers/pagecontroller.cpp | 96 + .../server/controllers/pagecontroller.h | 20 + .../server/controllers/sessioncontroller.cpp | 31 + .../server/controllers/sessioncontroller.h | 29 + .../server/controllers/sessionmanager.cpp | 0 .../server/controllers/sessionmanager.h | 0 .../server/controllers/templatecontroller.cpp | 31 + .../server/controllers/templatecontroller.h | 30 + .../controllers/updatecomiccontroller.cpp | 46 + .../controllers/updatecomiccontroller.h | 22 + YACReaderLibrary/server/documentcache.h | 4 + .../server/lib/bfHttpServer/bfHttpServer.pri | 12 + .../bfHttpServer/httpconnectionhandler.cpp | 164 ++ .../lib/bfHttpServer/httpconnectionhandler.h | 103 + .../httpconnectionhandlerpool.cpp | 64 + .../bfHttpServer/httpconnectionhandlerpool.h | 73 + .../server/lib/bfHttpServer/httpcookie.cpp | 199 ++ .../server/lib/bfHttpServer/httpcookie.h | 110 + .../server/lib/bfHttpServer/httplistener.cpp | 68 + .../server/lib/bfHttpServer/httplistener.h | 76 + .../server/lib/bfHttpServer/httprequest.cpp | 431 ++++ .../server/lib/bfHttpServer/httprequest.h | 212 ++ .../lib/bfHttpServer/httprequesthandler.cpp | 19 + .../lib/bfHttpServer/httprequesthandler.h | 45 + .../server/lib/bfHttpServer/httpresponse.cpp | 132 ++ .../server/lib/bfHttpServer/httpresponse.h | 135 ++ .../server/lib/bfHttpServer/httpsession.cpp | 384 +++ .../server/lib/bfHttpServer/httpsession.h | 180 ++ .../lib/bfHttpServer/httpsessionstore.cpp | 107 + .../lib/bfHttpServer/httpsessionstore.h | 104 + .../lib/bfHttpServer/staticfilecontroller.cpp | 234 ++ .../lib/bfHttpServer/staticfilecontroller.h | 92 + .../server/lib/bfLogging/bfLogging.pri | 5 + .../server/lib/bfLogging/dualfilelogger.cpp | 20 + .../server/lib/bfLogging/dualfilelogger.h | 58 + .../server/lib/bfLogging/filelogger.cpp | 174 ++ .../server/lib/bfLogging/filelogger.h | 122 + .../server/lib/bfLogging/logger.cpp | 172 ++ .../server/lib/bfLogging/logger.h | 181 ++ .../server/lib/bfLogging/logmessage.cpp | 75 + .../server/lib/bfLogging/logmessage.h | 91 + .../lib/bfTemplateEngine/bfTemplateEngine.pri | 7 + .../server/lib/bfTemplateEngine/template.cpp | 188 ++ .../server/lib/bfTemplateEngine/template.h | 167 ++ .../lib/bfTemplateEngine/templatecache.cpp | 30 + .../lib/bfTemplateEngine/templatecache.h | 77 + .../lib/bfTemplateEngine/templateloader.cpp | 109 + .../lib/bfTemplateEngine/templateloader.h | 85 + YACReaderLibrary/server/requestmapper.cpp | 101 + YACReaderLibrary/server/requestmapper.h | 36 + YACReaderLibrary/server/server.pri | 34 + YACReaderLibrary/server/startup.cpp | 89 + YACReaderLibrary/server/startup.h | 34 + YACReaderLibrary/server/static.cpp | 63 + YACReaderLibrary/server/static.h | 64 + YACReaderLibrary/server_config_dialog.cpp | 330 +++ YACReaderLibrary/server_config_dialog.h | 44 + YACReaderLibrary/yacreader_libraries.cpp | 147 ++ YACReaderLibrary/yacreader_libraries.h | 34 + YACReaderLibrary/yacreader_local_server.cpp | 218 ++ YACReaderLibrary/yacreader_local_server.h | 50 + YACReaderLibrary/yacreader_main_toolbar.cpp | 146 ++ YACReaderLibrary/yacreader_main_toolbar.h | 51 + YACReaderLibrary/yacreaderlibrary_es.qm | Bin 0 -> 35032 bytes YACReaderLibrary/yacreaderlibrary_es.ts | 1521 ++++++++++++ YACReaderLibrary/yacreaderlibrary_fr.ts | 1517 ++++++++++++ YACReaderLibrary/yacreaderlibrary_nl.ts | 1517 ++++++++++++ YACReaderLibrary/yacreaderlibrary_pt.ts | 1520 ++++++++++++ YACReaderLibrary/yacreaderlibrary_ru.ts | 1520 ++++++++++++ YACReaderLibrary/yacreaderlibrary_source.ts | 1517 ++++++++++++ YACReaderLibrary/yacreaderlibrary_tr.ts | 1308 +++++++++++ background.png | Bin 0 -> 3297 bytes cleanOSX.sh | 13 + common/bookmarks.cpp | 174 ++ common/bookmarks.h | 80 + common/check_new_version.cpp | 84 + common/check_new_version.h | 27 + common/comic.cpp | 743 ++++++ common/comic.h | 175 ++ common/comic_db.cpp | 477 ++++ common/comic_db.h | 154 ++ common/custom_widgets.cpp | 24 + common/custom_widgets.h | 12 + common/exit_check.cpp | 21 + common/exit_check.h | 9 + common/folder.cpp | 0 common/folder.h | 30 + common/http_worker.cpp | 65 + common/http_worker.h | 32 + common/library_item.cpp | 0 common/library_item.h | 16 + common/onstart_flow_selection_dialog.cpp | 54 + common/onstart_flow_selection_dialog.h | 13 + common/pdf_comic.h | 22 + common/pdf_comic.mm | 117 + common/pictureflow.cpp | 1380 +++++++++++ common/pictureflow.h | 227 ++ common/qnaturalsorting.cpp | 262 +++ common/qnaturalsorting.h | 15 + common/yacreader_flow_gl.cpp | 1580 +++++++++++++ common/yacreader_flow_gl.h | 402 ++++ common/yacreader_global.cpp | 29 + common/yacreader_global.h | 109 + compileOSX.sh | 47 + compileX11.sh | 43 + compressed_archive/7z_includes.h | 65 + compressed_archive/README_7zip.txt | 7 + compressed_archive/StdAfx.h | 9 + compressed_archive/StdAfx.h.cpp | 10 + compressed_archive/compressed_archive.cpp | 371 +++ compressed_archive/compressed_archive.h | 80 + compressed_archive/extract_callbacks.h | 328 +++ compressed_archive/extract_delegate.h | 14 + compressed_archive/libp7zip.patch | 11 + compressed_archive/open_callbacks.h | 54 + compressed_archive/wrapper.pri | 110 + create-dmg | 221 ++ custom_widgets/custom_widgets_yacreader.pri | 25 + .../custom_widgets_yacreaderlibrary.pri | 41 + custom_widgets/help_about_dialog.cpp | 75 + custom_widgets/help_about_dialog.h | 28 + custom_widgets/yacreader_busy_widget.cpp | 187 ++ custom_widgets/yacreader_busy_widget.h | 50 + custom_widgets/yacreader_dark_menu.cpp | 38 + custom_widgets/yacreader_dark_menu.h | 14 + .../yacreader_deleting_progress.cpp | 106 + custom_widgets/yacreader_deleting_progress.h | 26 + custom_widgets/yacreader_field_edit.cpp | 39 + custom_widgets/yacreader_field_edit.h | 23 + .../yacreader_field_plain_text_edit.cpp | 53 + .../yacreader_field_plain_text_edit.h | 25 + custom_widgets/yacreader_flow.cpp | 23 + custom_widgets/yacreader_flow.h | 21 + .../yacreader_flow_config_widget.cpp | 54 + custom_widgets/yacreader_flow_config_widget.h | 19 + .../yacreader_gl_flow_config_widget.cpp | 240 ++ .../yacreader_gl_flow_config_widget.h | 51 + .../yacreader_library_item_widget.cpp | 155 ++ .../yacreader_library_item_widget.h | 45 + .../yacreader_library_list_widget.cpp | 128 + .../yacreader_library_list_widget.h | 37 + custom_widgets/yacreader_options_dialog.cpp | 383 +++ custom_widgets/yacreader_options_dialog.h | 65 + custom_widgets/yacreader_search_line_edit.cpp | 55 + custom_widgets/yacreader_search_line_edit.h | 29 + custom_widgets/yacreader_sidebar.cpp | 130 ++ custom_widgets/yacreader_sidebar.h | 34 + custom_widgets/yacreader_social_dialog.cpp | 130 ++ custom_widgets/yacreader_social_dialog.h | 28 + .../yacreader_spin_slider_widget.cpp | 93 + custom_widgets/yacreader_spin_slider_widget.h | 35 + custom_widgets/yacreader_table_view.cpp | 422 ++++ custom_widgets/yacreader_table_view.h | 124 + custom_widgets/yacreader_titled_toolbar.cpp | 112 + custom_widgets/yacreader_titled_toolbar.h | 45 + custom_widgets/yacreader_tool_bar_stretch.cpp | 0 custom_widgets/yacreader_tool_bar_stretch.h | 18 + custom_widgets/yacreader_treeview.cpp | 80 + custom_widgets/yacreader_treeview.h | 27 + dependencies/poppler/bin/poppler-qt4.dll | Bin 0 -> 1671168 bytes .../poppler/bin/poppler-qt4.dll.manifest | 10 + dependencies/poppler/bin/poppler-qt5.dll | Bin 0 -> 1687552 bytes .../poppler/dependencies/bin/freetype6.dll | Bin 0 -> 410112 bytes .../dependencies/bin/freetype6.dll.manifest | 10 + .../poppler/dependencies/bin/openjpeg.dll | Bin 0 -> 87040 bytes .../dependencies/bin/openjpeg.dll.manifest | 10 + .../poppler/dependencies/lib/empty.txt | 0 .../include/qt4/poppler-annotation-helper.h | 198 ++ .../include/qt4/poppler-annotation-private.h | 111 + .../poppler/include/qt4/poppler-annotation.h | 920 ++++++++ .../include/qt4/poppler-converter-private.h | 49 + .../qt4/poppler-embeddedfile-private.h | 42 + .../poppler/include/qt4/poppler-export.h | 17 + .../poppler/include/qt4/poppler-form.h | 343 +++ .../qt4/poppler-link-extractor-private.h | 57 + .../poppler/include/qt4/poppler-link.h | 611 +++++ .../poppler/include/qt4/poppler-media.h | 97 + .../include/qt4/poppler-optcontent-private.h | 121 + .../poppler/include/qt4/poppler-optcontent.h | 76 + .../include/qt4/poppler-page-private.h | 54 + .../qt4/poppler-page-transition-private.h | 28 + .../include/qt4/poppler-page-transition.h | 148 ++ .../poppler/include/qt4/poppler-private.h | 311 +++ .../qt4/poppler-qiodeviceoutstream-private.h | 46 + .../poppler/include/qt4/poppler-qt4.h | 1809 ++++++++++++++ .../poppler/include/qt5/ArthurOutputDev.h | 170 ++ .../include/qt5/poppler-annotation-helper.h | 198 ++ .../include/qt5/poppler-annotation-private.h | 112 + .../poppler/include/qt5/poppler-annotation.h | 1030 ++++++++ .../include/qt5/poppler-converter-private.h | 49 + .../qt5/poppler-embeddedfile-private.h | 42 + .../poppler/include/qt5/poppler-export.h | 17 + .../poppler/include/qt5/poppler-form.h | 343 +++ .../qt5/poppler-link-extractor-private.h | 57 + .../poppler/include/qt5/poppler-link.h | 602 +++++ .../poppler/include/qt5/poppler-media.h | 100 + .../include/qt5/poppler-optcontent-private.h | 121 + .../poppler/include/qt5/poppler-optcontent.h | 77 + .../include/qt5/poppler-page-private.h | 54 + .../qt5/poppler-page-transition-private.h | 28 + .../include/qt5/poppler-page-transition.h | 148 ++ .../poppler/include/qt5/poppler-private.h | 240 ++ .../qt5/poppler-qiodeviceoutstream-private.h | 47 + .../poppler/include/qt5/poppler-qt5.h | 1771 ++++++++++++++ dependencies/poppler/lib/poppler-qt4.lib | Bin 0 -> 236940 bytes dependencies/poppler/lib/poppler-qt5.lib | Bin 0 -> 233116 bytes files/about.html | 93 + files/about_es_ES.html | 92 + files/helpYACReader.html | 145 ++ files/helpYACReaderLibrary.html | 94 + files/helpYACReaderLibrary_es_ES.html | 92 + files/helpYACReader_es_ES.html | 145 ++ files/shortcuts.html | 94 + files/shortcuts2.html | 38 + files/translator.html | 639 +++++ generateVS2010Projects.bat | 31 + icon.icns | Bin 0 -> 60104 bytes images/7z.png | Bin 0 -> 746 bytes images/accept_shortcut.png | Bin 0 -> 204 bytes images/adjustToFullSize.png | Bin 0 -> 21893 bytes images/alwaysOnTop.png | Bin 0 -> 28230 bytes images/asignNumber.png | Bin 0 -> 229 bytes images/bookmark.png | Bin 0 -> 24352 bytes images/branch-closed.png | Bin 0 -> 156 bytes images/branch-open.png | Bin 0 -> 145 bytes images/busy_background.png | Bin 0 -> 327 bytes images/center.png | Bin 0 -> 17966 bytes images/clearSearch.png | Bin 0 -> 1225 bytes images/clear_shortcut.png | Bin 0 -> 200 bytes images/close.png | Bin 0 -> 215 bytes images/colapse.png | Bin 0 -> 253 bytes images/colapse_osx.png | Bin 0 -> 255 bytes images/collapsed_branch_osx.png | Bin 0 -> 162 bytes images/collapsed_branch_selected.png | Bin 0 -> 139 bytes images/comic.png | Bin 0 -> 2867 bytes images/comic7z.png | Bin 0 -> 3045 bytes images/comicFolder.png | Bin 0 -> 26436 bytes images/comicRar.png | Bin 0 -> 3397 bytes images/comicTar.png | Bin 0 -> 2791 bytes images/comicZip.png | Bin 0 -> 3171 bytes images/comic_vine/downArrow.png | Bin 0 -> 139 bytes images/comic_vine/nextPage.png | Bin 0 -> 166 bytes images/comic_vine/previousPage.png | Bin 0 -> 167 bytes images/comic_vine/radioChecked.png | Bin 0 -> 236 bytes images/comic_vine/radioUnchecked.png | Bin 0 -> 189 bytes images/comic_vine/rowDown.png | Bin 0 -> 185 bytes images/comic_vine/rowUp.png | Bin 0 -> 186 bytes images/comic_vine/upArrow.png | Bin 0 -> 140 bytes images/coversPackage.png | Bin 0 -> 29501 bytes images/db.png | Bin 0 -> 16617 bytes images/defaultCover.png | Bin 0 -> 10871 bytes images/deleteLibrary.png | Bin 0 -> 21641 bytes images/deleting_progress/icon.png | Bin 0 -> 292 bytes images/deleting_progress/imgBottomLeft.png | Bin 0 -> 281 bytes images/deleting_progress/imgBottomMiddle.png | Bin 0 -> 124 bytes images/deleting_progress/imgBottomRight.png | Bin 0 -> 288 bytes images/deleting_progress/imgLeftMiddle.png | Bin 0 -> 114 bytes images/deleting_progress/imgRightMiddle.png | Bin 0 -> 114 bytes images/deleting_progress/imgTopLeft.png | Bin 0 -> 123 bytes images/deleting_progress/imgTopMiddle.png | Bin 0 -> 113 bytes images/deleting_progress/imgTopRight.png | Bin 0 -> 123 bytes images/dictionary.png | Bin 0 -> 23238 bytes images/doublePage.png | Bin 0 -> 35580 bytes images/down.png | Bin 0 -> 689 bytes images/dropDownArrow.png | Bin 0 -> 135 bytes images/edit.png | Bin 0 -> 1063 bytes images/editComic.png | Bin 0 -> 280 bytes images/editIcon.png | Bin 0 -> 269 bytes images/empty_folder.png | Bin 0 -> 2515 bytes images/expand.png | Bin 0 -> 158 bytes images/expand_osx.png | Bin 0 -> 199 bytes images/expanded_branch_osx.png | Bin 0 -> 169 bytes images/expanded_branch_selected.png | Bin 0 -> 133 bytes images/exportComicsInfo.png | Bin 0 -> 1448 bytes images/exportComicsInfoIcon.png | Bin 0 -> 289 bytes images/exportLibrary.png | Bin 0 -> 1233 bytes images/exportLibraryIcon.png | Bin 0 -> 241 bytes images/f.png | Bin 0 -> 710 bytes images/f_retina.png | Bin 0 -> 1371 bytes images/find_folder.png | Bin 0 -> 289 bytes images/fit.png | Bin 0 -> 6359 bytes images/flow1.png | Bin 0 -> 9112 bytes images/flow2.png | Bin 0 -> 13316 bytes images/flow3.png | Bin 0 -> 13026 bytes images/flow4.png | Bin 0 -> 8499 bytes images/flow5.png | Bin 0 -> 9135 bytes images/flow_to_grid.gif | Bin 0 -> 124025 bytes images/folder.png | Bin 0 -> 313 bytes images/folder_finished.png | Bin 0 -> 386 bytes images/folder_finished_macosx.png | Bin 0 -> 158 bytes images/folder_macosx.png | Bin 0 -> 1332 bytes images/fromTo.png | Bin 0 -> 171 bytes images/getInfo.png | Bin 0 -> 294 bytes images/glowLine.png | Bin 0 -> 460 bytes images/goto.png | Bin 0 -> 15572 bytes images/grid_to_flow.gif | Bin 0 -> 129647 bytes images/help.png | Bin 0 -> 16817 bytes images/helpImages/bookmark.png | Bin 0 -> 1801 bytes images/helpImages/center.png | Bin 0 -> 1645 bytes images/helpImages/colapse.png | Bin 0 -> 1379 bytes images/helpImages/comicFolder.png | Bin 0 -> 1826 bytes images/helpImages/coversPackage.png | Bin 0 -> 1670 bytes images/helpImages/deleteLibrary.png | Bin 0 -> 1601 bytes images/helpImages/doublePage.png | Bin 0 -> 1761 bytes images/helpImages/edit.png | Bin 0 -> 1399 bytes images/helpImages/expand.png | Bin 0 -> 1543 bytes images/helpImages/exportLibrary.png | Bin 0 -> 1633 bytes images/helpImages/fit.png | Bin 0 -> 1574 bytes images/helpImages/flow1.png | Bin 0 -> 766 bytes images/helpImages/flow2.png | Bin 0 -> 1039 bytes images/helpImages/flow3.png | Bin 0 -> 1038 bytes images/helpImages/folder.png | Bin 0 -> 1758 bytes images/helpImages/goto.png | Bin 0 -> 1369 bytes images/helpImages/help.png | Bin 0 -> 1615 bytes images/helpImages/icon.png | Bin 0 -> 1654 bytes images/helpImages/importLibrary.png | Bin 0 -> 1631 bytes images/helpImages/keyboard.png | Bin 0 -> 1502 bytes images/helpImages/mouse.png | Bin 0 -> 1550 bytes images/helpImages/new.png | Bin 0 -> 1685 bytes images/helpImages/next.png | Bin 0 -> 1665 bytes images/helpImages/nextComic.png | Bin 0 -> 1574 bytes images/helpImages/notCover.png | Bin 0 -> 583 bytes images/helpImages/open.png | Bin 0 -> 1790 bytes images/helpImages/openFolder.png | Bin 0 -> 1896 bytes images/helpImages/openLibrary.png | Bin 0 -> 1829 bytes images/helpImages/options.png | Bin 0 -> 1573 bytes images/helpImages/prev.png | Bin 0 -> 1694 bytes images/helpImages/previousComic.png | Bin 0 -> 1585 bytes images/helpImages/properties.png | Bin 0 -> 1721 bytes images/helpImages/removeLibrary.png | Bin 0 -> 1612 bytes images/helpImages/rotateL.png | Bin 0 -> 1757 bytes images/helpImages/rotateR.png | Bin 0 -> 1755 bytes images/helpImages/save.png | Bin 0 -> 1642 bytes images/helpImages/setBookmark.png | Bin 0 -> 1532 bytes images/helpImages/setRoot.png | Bin 0 -> 1810 bytes images/helpImages/shortcuts.png | Bin 0 -> 1703 bytes images/helpImages/speaker.png | Bin 0 -> 658 bytes images/helpImages/updateLibrary.png | Bin 0 -> 1589 bytes images/helpImages/zoom.png | Bin 0 -> 1618 bytes images/hiddenCovers.png | Bin 0 -> 446 bytes images/hideComicFlow.png | Bin 0 -> 227 bytes images/icon.png | Bin 0 -> 20829 bytes images/iconLibrary.png | Bin 0 -> 26640 bytes images/iconSearch.png | Bin 0 -> 1230 bytes images/imgBottomLeft.png | Bin 0 -> 221 bytes images/imgBottomMiddle.png | Bin 0 -> 142 bytes images/imgBottomRight.png | Bin 0 -> 210 bytes images/imgCenterSlide.png | Bin 0 -> 803 bytes images/imgCenterSlidePressed.png | Bin 0 -> 814 bytes images/imgEdit.png | Bin 0 -> 851 bytes images/imgGoToSlide.png | Bin 0 -> 947 bytes images/imgGoToSlidePressed.png | Bin 0 -> 940 bytes images/imgTopLeft.png | Bin 0 -> 188 bytes images/imgTopMiddle.png | Bin 0 -> 137 bytes images/imgTopRight.png | Bin 0 -> 195 bytes images/importBottomCoversDecoration.png | Bin 0 -> 138 bytes images/importComicsInfo.png | Bin 0 -> 1170 bytes images/importComicsInfoIcon.png | Bin 0 -> 207 bytes images/importCover.png | Bin 0 -> 25015 bytes images/importLibrary.png | Bin 0 -> 1279 bytes images/importLibraryIcon.png | Bin 0 -> 245 bytes images/importTopCoversDecoration.png | Bin 0 -> 132 bytes images/importingIcon.png | Bin 0 -> 3162 bytes images/iphoneConfig.png | Bin 0 -> 16872 bytes images/libraryIcon.png | Bin 0 -> 280 bytes images/libraryIconSelected.png | Bin 0 -> 152 bytes images/libraryIcon_osx.png | Bin 0 -> 258 bytes images/libraryOptions.png | Bin 0 -> 206 bytes images/main_toolbar/back.png | Bin 0 -> 225 bytes images/main_toolbar/back_disabled.png | Bin 0 -> 225 bytes images/main_toolbar/back_disabled_osx.png | Bin 0 -> 1231 bytes images/main_toolbar/back_osx.png | Bin 0 -> 1161 bytes images/main_toolbar/divider.png | Bin 0 -> 207 bytes images/main_toolbar/forward.png | Bin 0 -> 234 bytes images/main_toolbar/forward_disabled.png | Bin 0 -> 240 bytes images/main_toolbar/forward_disabled_osx.png | Bin 0 -> 1253 bytes images/main_toolbar/forward_osx.png | Bin 0 -> 1230 bytes images/main_toolbar/fullscreen.png | Bin 0 -> 259 bytes images/main_toolbar/fullscreen_osx.png | Bin 0 -> 1435 bytes images/main_toolbar/help.png | Bin 0 -> 384 bytes images/main_toolbar/help_osx.png | Bin 0 -> 1405 bytes images/main_toolbar/server.png | Bin 0 -> 196 bytes images/main_toolbar/server_osx.png | Bin 0 -> 1257 bytes images/main_toolbar/settings.png | Bin 0 -> 369 bytes images/main_toolbar/settings_osx.png | Bin 0 -> 1621 bytes images/new.png | Bin 0 -> 454 bytes images/newLibraryIcon.png | Bin 0 -> 212 bytes images/newLibraryIcon_osx.png | Bin 0 -> 217 bytes images/next.png | Bin 0 -> 21672 bytes images/nextComic.png | Bin 0 -> 23796 bytes images/nextCoverPage.png | Bin 0 -> 153 bytes images/noLibrariesIcon.png | Bin 0 -> 6047 bytes images/noLibrariesLine.png | Bin 0 -> 238 bytes images/notCover.png | Bin 0 -> 13927 bytes images/notificationsLabel.png | Bin 0 -> 3099 bytes images/numPagesLabel.png | Bin 0 -> 623 bytes images/numPagesLabelBig.png | Bin 0 -> 739 bytes images/numPagesLabelMedium.png | Bin 0 -> 702 bytes images/onStartFlowSelection.png | Bin 0 -> 22053 bytes images/onStartFlowSelection_es.png | Bin 0 -> 22971 bytes images/open.png | Bin 0 -> 193 bytes images/openFolder.png | Bin 0 -> 30976 bytes images/openInYACReader.png | Bin 0 -> 305 bytes images/openLibrary.png | Bin 0 -> 821 bytes images/openLibraryIcon.png | Bin 0 -> 298 bytes images/openLibraryIcon_osx.png | Bin 0 -> 289 bytes images/options.png | Bin 0 -> 20699 bytes images/pdf.png | Bin 0 -> 923 bytes images/prev.png | Bin 0 -> 24468 bytes images/previousComic.png | Bin 0 -> 23714 bytes images/previousCoverPage.png | Bin 0 -> 152 bytes images/properties.png | Bin 0 -> 21282 bytes images/qrMessage.png | Bin 0 -> 2781 bytes images/rar.png | Bin 0 -> 1091 bytes images/rating0.png | Bin 0 -> 402 bytes images/rating1.png | Bin 0 -> 479 bytes images/rating2.png | Bin 0 -> 514 bytes images/rating3.png | Bin 0 -> 514 bytes images/rating4.png | Bin 0 -> 498 bytes images/rating5.png | Bin 0 -> 404 bytes images/readRibbon.png | Bin 0 -> 7963 bytes images/readingRibbon.png | Bin 0 -> 6603 bytes images/removeLibrary.png | Bin 0 -> 19700 bytes images/removeLibraryIcon.png | Bin 0 -> 259 bytes images/rotateL.png | Bin 0 -> 27714 bytes images/rotateR.png | Bin 0 -> 27567 bytes images/save.png | Bin 0 -> 23937 bytes images/selectAll.png | Bin 0 -> 216 bytes images/server.png | Bin 0 -> 14333 bytes images/serverConfigBackground.png | Bin 0 -> 31622 bytes images/setAllRead.png | Bin 0 -> 255 bytes images/setAllUnread.png | Bin 0 -> 299 bytes images/setBookmark.png | Bin 0 -> 17421 bytes images/setRead.png | Bin 0 -> 18976 bytes images/setReadButton.png | Bin 0 -> 244 bytes images/setRoot.png | Bin 0 -> 389 bytes images/setRoot_osx.png | Bin 0 -> 385 bytes images/setUnread.png | Bin 0 -> 287 bytes images/shortcuts.png | Bin 0 -> 16124 bytes images/shortcuts_group_comics.png | Bin 0 -> 276 bytes images/shortcuts_group_folders.png | Bin 0 -> 157 bytes images/shortcuts_group_general.png | Bin 0 -> 319 bytes images/shortcuts_group_libraries.png | Bin 0 -> 164 bytes images/shortcuts_group_mglass.png | Bin 0 -> 351 bytes images/shortcuts_group_page.png | Bin 0 -> 162 bytes images/shortcuts_group_reading.png | Bin 0 -> 179 bytes images/shortcuts_group_visualization.png | Bin 0 -> 320 bytes images/showMarks.png | Bin 0 -> 283 bytes images/shownCovers.png | Bin 0 -> 445 bytes images/sliderAddPage.png | Bin 0 -> 186 bytes images/sliderBackground.png | Bin 0 -> 564 bytes images/sliderGround.png | Bin 0 -> 951 bytes images/sliderHandle.png | Bin 0 -> 1089 bytes images/sliderSubPage.png | Bin 0 -> 185 bytes images/social_dialog/close.png | Bin 0 -> 246 bytes images/social_dialog/facebook.png | Bin 0 -> 181 bytes images/social_dialog/google+.png | Bin 0 -> 344 bytes images/social_dialog/icon.png | Bin 0 -> 1324 bytes images/social_dialog/separator.png | Bin 0 -> 247 bytes images/social_dialog/shadow.png | Bin 0 -> 122 bytes images/social_dialog/twitter.png | Bin 0 -> 301 bytes images/speaker.png | Bin 0 -> 200 bytes images/tar.png | Bin 0 -> 480 bytes images/translatorSearch.png | Bin 0 -> 241 bytes images/trash.png | Bin 0 -> 192 bytes images/up.png | Bin 0 -> 702 bytes images/updateLibrary.png | Bin 0 -> 20542 bytes images/updateLibraryIcon.png | Bin 0 -> 306 bytes images/updatingIcon.png | Bin 0 -> 5746 bytes images/useNewFlowButton.png | Bin 0 -> 4174 bytes images/useOldFlowButton.png | Bin 0 -> 3998 bytes images/viewer_toolbar/bookmark.png | Bin 0 -> 199 bytes images/viewer_toolbar/bookmark_osx.png | Bin 0 -> 1234 bytes images/viewer_toolbar/close.png | Bin 0 -> 272 bytes images/viewer_toolbar/close_osx.png | Bin 0 -> 1517 bytes images/viewer_toolbar/doublePage.png | Bin 0 -> 149 bytes images/viewer_toolbar/doublePage_osx.png | Bin 0 -> 1186 bytes images/viewer_toolbar/flow.png | Bin 0 -> 153 bytes images/viewer_toolbar/flow_osx.png | Bin 0 -> 1056 bytes images/viewer_toolbar/full.png | Bin 0 -> 204 bytes images/viewer_toolbar/full_osx.png | Bin 0 -> 1491 bytes images/viewer_toolbar/goto.png | Bin 0 -> 1118 bytes images/viewer_toolbar/goto_osx.png | Bin 0 -> 1367 bytes images/viewer_toolbar/help.png | Bin 0 -> 287 bytes images/viewer_toolbar/help_osx.png | Bin 0 -> 1443 bytes images/viewer_toolbar/info.png | Bin 0 -> 225 bytes images/viewer_toolbar/info_osx.png | Bin 0 -> 1304 bytes images/viewer_toolbar/magnifyingGlass.png | Bin 0 -> 346 bytes images/viewer_toolbar/magnifyingGlass_osx.png | Bin 0 -> 1558 bytes images/viewer_toolbar/next.png | Bin 0 -> 194 bytes images/viewer_toolbar/next_osx.png | Bin 0 -> 1241 bytes images/viewer_toolbar/open.png | Bin 0 -> 304 bytes images/viewer_toolbar/openFolder.png | Bin 0 -> 162 bytes images/viewer_toolbar/openFolder_osx.png | Bin 0 -> 1155 bytes images/viewer_toolbar/openNext.png | Bin 0 -> 249 bytes images/viewer_toolbar/openNext_osx.png | Bin 0 -> 1376 bytes images/viewer_toolbar/openPrevious.png | Bin 0 -> 230 bytes images/viewer_toolbar/openPrevious_osx.png | Bin 0 -> 1382 bytes images/viewer_toolbar/open_osx.png | Bin 0 -> 1487 bytes images/viewer_toolbar/options.png | Bin 0 -> 331 bytes images/viewer_toolbar/options_osx.png | Bin 0 -> 1666 bytes images/viewer_toolbar/previous.png | Bin 0 -> 186 bytes images/viewer_toolbar/previous_osx.png | Bin 0 -> 1238 bytes images/viewer_toolbar/rotateL.png | Bin 0 -> 340 bytes images/viewer_toolbar/rotateL_osx.png | Bin 0 -> 1550 bytes images/viewer_toolbar/rotateR.png | Bin 0 -> 344 bytes images/viewer_toolbar/rotateR_osx.png | Bin 0 -> 1564 bytes images/viewer_toolbar/save.png | Bin 0 -> 208 bytes images/viewer_toolbar/save_osx.png | Bin 0 -> 1249 bytes images/viewer_toolbar/shortcuts.png | Bin 0 -> 284 bytes images/viewer_toolbar/shortcuts_osx.png | Bin 0 -> 1401 bytes images/viewer_toolbar/showBookmarks.png | Bin 0 -> 181 bytes images/viewer_toolbar/showBookmarks_osx.png | Bin 0 -> 1251 bytes images/viewer_toolbar/toHeight.png | Bin 0 -> 213 bytes images/viewer_toolbar/toHeight_osx.png | Bin 0 -> 1285 bytes images/viewer_toolbar/toWidth.png | Bin 0 -> 218 bytes images/viewer_toolbar/toWidth_osx.png | Bin 0 -> 1260 bytes images/viewer_toolbar/translator.png | Bin 0 -> 233 bytes images/viewer_toolbar/translator_osx.png | Bin 0 -> 1401 bytes images/zip.png | Bin 0 -> 843 bytes images/zoom.png | Bin 0 -> 17891 bytes release/languages/yacreader_es.qm | Bin 0 -> 13554 bytes release/languages/yacreader_fr.qm | Bin 0 -> 11991 bytes release/languages/yacreader_nl.qm | Bin 0 -> 11973 bytes release/languages/yacreader_pt.qm | Bin 0 -> 6959 bytes release/languages/yacreader_ru.qm | Bin 0 -> 11731 bytes release/languages/yacreader_tr.qm | Bin 0 -> 11234 bytes release/languages/yacreaderlibrary_es.qm | Bin 0 -> 35032 bytes release/languages/yacreaderlibrary_fr.qm | Bin 0 -> 26934 bytes release/languages/yacreaderlibrary_nl.qm | Bin 0 -> 26438 bytes release/languages/yacreaderlibrary_pt.qm | Bin 0 -> 6201 bytes release/languages/yacreaderlibrary_ru.qm | Bin 0 -> 16195 bytes release/languages/yacreaderlibrary_tr.qm | Bin 0 -> 24489 bytes release/server/docroot/css/reset.css | 46 + release/server/docroot/css/styles_ipad.css | 465 ++++ release/server/docroot/css/styles_iphone.css | 463 ++++ release/server/docroot/images/browse.png | Bin 0 -> 134 bytes release/server/docroot/images/browse@2x.png | Bin 0 -> 185 bytes release/server/docroot/images/combo.png | Bin 0 -> 120 bytes release/server/docroot/images/combo@2x.png | Bin 0 -> 167 bytes release/server/docroot/images/download.png | Bin 0 -> 155 bytes release/server/docroot/images/download@2x.png | Bin 0 -> 203 bytes release/server/docroot/images/f.png | Bin 0 -> 621 bytes release/server/docroot/images/f@2x.png | Bin 0 -> 1262 bytes release/server/docroot/images/imported.png | Bin 0 -> 158 bytes release/server/docroot/images/imported@2x.png | Bin 0 -> 214 bytes release/server/docroot/images/indicator.png | Bin 0 -> 118 bytes .../server/docroot/images/indicator@2x.png | Bin 0 -> 220 bytes release/server/docroot/images/library.png | Bin 0 -> 201 bytes release/server/docroot/images/library@2x.png | Bin 0 -> 284 bytes release/server/docroot/images/next.png | Bin 0 -> 137 bytes release/server/docroot/images/next@2x.png | Bin 0 -> 339 bytes release/server/docroot/images/prev.png | Bin 0 -> 154 bytes release/server/docroot/images/prev@2x.png | Bin 0 -> 345 bytes release/server/docroot/images/read.png | Bin 0 -> 152 bytes release/server/docroot/images/read@2x.png | Bin 0 -> 201 bytes release/server/docroot/images/readMark.png | Bin 0 -> 196 bytes release/server/docroot/images/readMark@2x.png | Bin 0 -> 296 bytes release/server/docroot/images/readingMark.png | Bin 0 -> 206 bytes .../server/docroot/images/readingMark@2x.png | Bin 0 -> 296 bytes release/server/docroot/images/up.png | Bin 0 -> 163 bytes release/server/docroot/images/up@2x.png | Bin 0 -> 271 bytes release/server/docroot/login.html | 26 + release/server/templates/folder_ipad.tpl | 114 + release/server/templates/folder_iphone.tpl | 113 + release/server/templates/libraries_ipad.tpl | 26 + release/server/templates/libraries_iphone.tpl | 26 + releaseOSX.sh | 17 + shortcuts_management/actions_groups_model.cpp | 80 + shortcuts_management/actions_groups_model.h | 44 + .../actions_shortcuts_model.cpp | 106 + .../actions_shortcuts_model.h | 38 + .../edit_shortcut_item_delegate.cpp | 145 ++ .../edit_shortcut_item_delegate.h | 48 + .../edit_shortcuts_dialog.cpp | 95 + shortcuts_management/edit_shortcuts_dialog.h | 33 + shortcuts_management/shortcuts_management.pri | 16 + shortcuts_management/shortcuts_manager.cpp | 112 + shortcuts_management/shortcuts_manager.h | 127 + 795 files changed, 73635 insertions(+) create mode 100644 CHANGELOG.txt create mode 100644 COPYING.txt create mode 100644 INSTALL.txt create mode 100644 QsLog/QsLog.cpp create mode 100644 QsLog/QsLog.h create mode 100644 QsLog/QsLog.pri create mode 100644 QsLog/QsLogDest.cpp create mode 100644 QsLog/QsLogDest.h create mode 100644 QsLog/QsLogDestConsole.cpp create mode 100644 QsLog/QsLogDestConsole.h create mode 100644 QsLog/QsLogDestFile.cpp create mode 100644 QsLog/QsLogDestFile.h create mode 100644 QsLog/QsLogDestFunctor.cpp create mode 100644 QsLog/QsLogDestFunctor.h create mode 100644 QsLog/QsLogDisableForThisFile.h create mode 100644 QsLog/QsLogLevel.h create mode 100644 QsLog/QsLogSharedLibrary.pro create mode 100644 README.txt create mode 100644 YACReader.desktop create mode 100644 YACReader.pro create mode 100644 YACReader/YACReader.icns create mode 100644 YACReader/YACReader.pri create mode 100644 YACReader/YACReader.pro create mode 100644 YACReader/bookmarks_dialog.cpp create mode 100644 YACReader/bookmarks_dialog.h create mode 100644 YACReader/configuration.cpp create mode 100644 YACReader/configuration.h create mode 100644 YACReader/goto_dialog.cpp create mode 100644 YACReader/goto_dialog.h create mode 100644 YACReader/goto_flow.cpp create mode 100644 YACReader/goto_flow.h create mode 100644 YACReader/goto_flow_decorationbar.cpp create mode 100644 YACReader/goto_flow_decorationbar.h create mode 100644 YACReader/goto_flow_gl.cpp create mode 100644 YACReader/goto_flow_gl.h create mode 100644 YACReader/goto_flow_toolbar.cpp create mode 100644 YACReader/goto_flow_toolbar.h create mode 100644 YACReader/goto_flow_widget.cpp create mode 100644 YACReader/goto_flow_widget.h create mode 100644 YACReader/icon.ico create mode 100644 YACReader/icon.rc create mode 100644 YACReader/magnifying_glass.cpp create mode 100644 YACReader/magnifying_glass.h create mode 100644 YACReader/main.cpp create mode 100644 YACReader/main_window_viewer.cpp create mode 100644 YACReader/main_window_viewer.h create mode 100644 YACReader/notifications_label_widget.cpp create mode 100644 YACReader/notifications_label_widget.h create mode 100644 YACReader/options_dialog.cpp create mode 100644 YACReader/options_dialog.h create mode 100644 YACReader/page_label_widget.cpp create mode 100644 YACReader/page_label_widget.h create mode 100644 YACReader/render.cpp create mode 100644 YACReader/render.h create mode 100644 YACReader/shortcuts_dialog.cpp create mode 100644 YACReader/shortcuts_dialog.h create mode 100644 YACReader/translator.cpp create mode 100644 YACReader/translator.h create mode 100644 YACReader/viewer.cpp create mode 100644 YACReader/viewer.h create mode 100644 YACReader/width_slider.cpp create mode 100644 YACReader/width_slider.h create mode 100644 YACReader/yacreader_es.qm create mode 100644 YACReader/yacreader_es.ts create mode 100644 YACReader/yacreader_files.qrc create mode 100644 YACReader/yacreader_fr.ts create mode 100644 YACReader/yacreader_images.qrc create mode 100644 YACReader/yacreader_images_osx.qrc create mode 100644 YACReader/yacreader_images_win.qrc create mode 100644 YACReader/yacreader_local_client.cpp create mode 100644 YACReader/yacreader_local_client.h create mode 100644 YACReader/yacreader_nl.ts create mode 100644 YACReader/yacreader_pt.ts create mode 100644 YACReader/yacreader_ru.ts create mode 100644 YACReader/yacreader_source.ts create mode 100644 YACReader/yacreader_tr.ts create mode 100644 YACReaderLibrary.desktop create mode 100644 YACReaderLibrary/YACReaderLibrary.icns create mode 100644 YACReaderLibrary/YACReaderLibrary.pro create mode 100644 YACReaderLibrary/add_library_dialog.cpp create mode 100644 YACReaderLibrary/add_library_dialog.h create mode 100644 YACReaderLibrary/bundle_creator.cpp create mode 100644 YACReaderLibrary/bundle_creator.h create mode 100644 YACReaderLibrary/classic_comics_view.cpp create mode 100644 YACReaderLibrary/classic_comics_view.h create mode 100644 YACReaderLibrary/comic_flow.cpp create mode 100644 YACReaderLibrary/comic_flow.h create mode 100644 YACReaderLibrary/comic_flow_widget.cpp create mode 100644 YACReaderLibrary/comic_flow_widget.h create mode 100644 YACReaderLibrary/comic_vine/comic_vine.pri create mode 100644 YACReaderLibrary/comic_vine/comic_vine_client.cpp create mode 100644 YACReaderLibrary/comic_vine/comic_vine_client.h create mode 100644 YACReaderLibrary/comic_vine/comic_vine_dialog.cpp create mode 100644 YACReaderLibrary/comic_vine/comic_vine_dialog.h create mode 100644 YACReaderLibrary/comic_vine/model/comics_model.cpp create mode 100644 YACReaderLibrary/comic_vine/model/comics_model.h create mode 100644 YACReaderLibrary/comic_vine/model/json_model.cpp create mode 100644 YACReaderLibrary/comic_vine/model/json_model.h create mode 100644 YACReaderLibrary/comic_vine/model/local_comic_list_model.cpp create mode 100644 YACReaderLibrary/comic_vine/model/local_comic_list_model.h create mode 100644 YACReaderLibrary/comic_vine/model/response_parser.cpp create mode 100644 YACReaderLibrary/comic_vine/model/response_parser.h create mode 100644 YACReaderLibrary/comic_vine/model/volume_comics_model.cpp create mode 100644 YACReaderLibrary/comic_vine/model/volume_comics_model.h create mode 100644 YACReaderLibrary/comic_vine/model/volumes_model.cpp create mode 100644 YACReaderLibrary/comic_vine/model/volumes_model.h create mode 100644 YACReaderLibrary/comic_vine/scraper_lineedit.cpp create mode 100644 YACReaderLibrary/comic_vine/scraper_lineedit.h create mode 100644 YACReaderLibrary/comic_vine/scraper_results_paginator.cpp create mode 100644 YACReaderLibrary/comic_vine/scraper_results_paginator.h create mode 100644 YACReaderLibrary/comic_vine/scraper_scroll_label.cpp create mode 100644 YACReaderLibrary/comic_vine/scraper_scroll_label.h create mode 100644 YACReaderLibrary/comic_vine/scraper_selector.cpp create mode 100644 YACReaderLibrary/comic_vine/scraper_selector.h create mode 100644 YACReaderLibrary/comic_vine/scraper_tableview.cpp create mode 100644 YACReaderLibrary/comic_vine/scraper_tableview.h create mode 100644 YACReaderLibrary/comic_vine/search_single_comic.cpp create mode 100644 YACReaderLibrary/comic_vine/search_single_comic.h create mode 100644 YACReaderLibrary/comic_vine/search_volume.cpp create mode 100644 YACReaderLibrary/comic_vine/search_volume.h create mode 100644 YACReaderLibrary/comic_vine/select_comic.cpp create mode 100644 YACReaderLibrary/comic_vine/select_comic.h create mode 100644 YACReaderLibrary/comic_vine/select_volume.cpp create mode 100644 YACReaderLibrary/comic_vine/select_volume.h create mode 100644 YACReaderLibrary/comic_vine/series_question.cpp create mode 100644 YACReaderLibrary/comic_vine/series_question.h create mode 100644 YACReaderLibrary/comic_vine/sort_volume_comics.cpp create mode 100644 YACReaderLibrary/comic_vine/sort_volume_comics.h create mode 100644 YACReaderLibrary/comic_vine/title_header.cpp create mode 100644 YACReaderLibrary/comic_vine/title_header.h create mode 100644 YACReaderLibrary/comics_remover.cpp create mode 100644 YACReaderLibrary/comics_remover.h create mode 100644 YACReaderLibrary/comics_view.cpp create mode 100644 YACReaderLibrary/comics_view.h create mode 100644 YACReaderLibrary/comics_view_transition.cpp create mode 100644 YACReaderLibrary/comics_view_transition.h create mode 100644 YACReaderLibrary/create_library_dialog.cpp create mode 100644 YACReaderLibrary/create_library_dialog.h create mode 100644 YACReaderLibrary/db/data_base_management.cpp create mode 100644 YACReaderLibrary/db/data_base_management.h create mode 100644 YACReaderLibrary/db/tableitem.cpp create mode 100644 YACReaderLibrary/db/tableitem.h create mode 100644 YACReaderLibrary/db/tablemodel.cpp create mode 100644 YACReaderLibrary/db/tablemodel.h create mode 100644 YACReaderLibrary/db/treeitem.cpp create mode 100644 YACReaderLibrary/db/treeitem.h create mode 100644 YACReaderLibrary/db/treemodel.cpp create mode 100644 YACReaderLibrary/db/treemodel.h create mode 100644 YACReaderLibrary/db_helper.cpp create mode 100644 YACReaderLibrary/db_helper.h create mode 100644 YACReaderLibrary/empty_folder_widget.cpp create mode 100644 YACReaderLibrary/empty_folder_widget.h create mode 100644 YACReaderLibrary/export_comics_info_dialog.cpp create mode 100644 YACReaderLibrary/export_comics_info_dialog.h create mode 100644 YACReaderLibrary/export_library_dialog.cpp create mode 100644 YACReaderLibrary/export_library_dialog.h create mode 100644 YACReaderLibrary/files.qrc create mode 100644 YACReaderLibrary/grid_comics_view.cpp create mode 100644 YACReaderLibrary/grid_comics_view.h create mode 100644 YACReaderLibrary/icon.ico create mode 100644 YACReaderLibrary/icon.rc create mode 100644 YACReaderLibrary/icon2.ico create mode 100644 YACReaderLibrary/icon3.ico create mode 100644 YACReaderLibrary/images.qrc create mode 100644 YACReaderLibrary/images_osx.qrc create mode 100644 YACReaderLibrary/images_win.qrc create mode 100644 YACReaderLibrary/import_comics_info_dialog.cpp create mode 100644 YACReaderLibrary/import_comics_info_dialog.h create mode 100644 YACReaderLibrary/import_library_dialog.cpp create mode 100644 YACReaderLibrary/import_library_dialog.h create mode 100644 YACReaderLibrary/import_widget.cpp create mode 100644 YACReaderLibrary/import_widget.h create mode 100644 YACReaderLibrary/library_creator.cpp create mode 100644 YACReaderLibrary/library_creator.h create mode 100644 YACReaderLibrary/library_window.cpp create mode 100644 YACReaderLibrary/library_window.h create mode 100644 YACReaderLibrary/main.cpp create mode 100644 YACReaderLibrary/no_libraries_widget.cpp create mode 100644 YACReaderLibrary/no_libraries_widget.h create mode 100644 YACReaderLibrary/options_dialog.cpp create mode 100644 YACReaderLibrary/options_dialog.h create mode 100644 YACReaderLibrary/package_manager.cpp create mode 100644 YACReaderLibrary/package_manager.h create mode 100644 YACReaderLibrary/properties_dialog.cpp create mode 100644 YACReaderLibrary/properties_dialog.h create mode 100644 YACReaderLibrary/qml.qrc create mode 100644 YACReaderLibrary/qml/GridComicsView.qml create mode 100644 YACReaderLibrary/qml/YACReaderScrollView.qml create mode 100644 YACReaderLibrary/qml/page-macosx.png create mode 100644 YACReaderLibrary/qml/page.png create mode 100644 YACReaderLibrary/qml/reading.png create mode 100644 YACReaderLibrary/qml/star-macosx.png create mode 100644 YACReaderLibrary/qml/star.png create mode 100644 YACReaderLibrary/qml/tick.png create mode 100644 YACReaderLibrary/qml_osx.qrc create mode 100644 YACReaderLibrary/qml_win.qrc create mode 100644 YACReaderLibrary/rename_library_dialog.cpp create mode 100644 YACReaderLibrary/rename_library_dialog.h create mode 100644 YACReaderLibrary/server/controllers/comiccontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/comiccontroller.h create mode 100644 YACReaderLibrary/server/controllers/covercontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/covercontroller.h create mode 100644 YACReaderLibrary/server/controllers/dumpcontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/dumpcontroller.h create mode 100644 YACReaderLibrary/server/controllers/errorcontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/errorcontroller.h create mode 100644 YACReaderLibrary/server/controllers/fileuploadcontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/fileuploadcontroller.h create mode 100644 YACReaderLibrary/server/controllers/foldercontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/foldercontroller.h create mode 100644 YACReaderLibrary/server/controllers/folderinfocontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/folderinfocontroller.h create mode 100644 YACReaderLibrary/server/controllers/formcontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/formcontroller.h create mode 100644 YACReaderLibrary/server/controllers/librariescontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/librariescontroller.h create mode 100644 YACReaderLibrary/server/controllers/pagecontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/pagecontroller.h create mode 100644 YACReaderLibrary/server/controllers/sessioncontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/sessioncontroller.h create mode 100644 YACReaderLibrary/server/controllers/sessionmanager.cpp create mode 100644 YACReaderLibrary/server/controllers/sessionmanager.h create mode 100644 YACReaderLibrary/server/controllers/templatecontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/templatecontroller.h create mode 100644 YACReaderLibrary/server/controllers/updatecomiccontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/updatecomiccontroller.h create mode 100644 YACReaderLibrary/server/documentcache.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/bfHttpServer.pri create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpcookie.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpcookie.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httplistener.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httplistener.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httprequest.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpresponse.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpresponse.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpsession.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.h create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp create mode 100644 YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h create mode 100644 YACReaderLibrary/server/lib/bfLogging/bfLogging.pri create mode 100644 YACReaderLibrary/server/lib/bfLogging/dualfilelogger.cpp create mode 100644 YACReaderLibrary/server/lib/bfLogging/dualfilelogger.h create mode 100644 YACReaderLibrary/server/lib/bfLogging/filelogger.cpp create mode 100644 YACReaderLibrary/server/lib/bfLogging/filelogger.h create mode 100644 YACReaderLibrary/server/lib/bfLogging/logger.cpp create mode 100644 YACReaderLibrary/server/lib/bfLogging/logger.h create mode 100644 YACReaderLibrary/server/lib/bfLogging/logmessage.cpp create mode 100644 YACReaderLibrary/server/lib/bfLogging/logmessage.h create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/bfTemplateEngine.pri create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/template.cpp create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/template.h create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.cpp create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.h create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.cpp create mode 100644 YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.h create mode 100644 YACReaderLibrary/server/requestmapper.cpp create mode 100644 YACReaderLibrary/server/requestmapper.h create mode 100644 YACReaderLibrary/server/server.pri create mode 100644 YACReaderLibrary/server/startup.cpp create mode 100644 YACReaderLibrary/server/startup.h create mode 100644 YACReaderLibrary/server/static.cpp create mode 100644 YACReaderLibrary/server/static.h create mode 100644 YACReaderLibrary/server_config_dialog.cpp create mode 100644 YACReaderLibrary/server_config_dialog.h create mode 100644 YACReaderLibrary/yacreader_libraries.cpp create mode 100644 YACReaderLibrary/yacreader_libraries.h create mode 100644 YACReaderLibrary/yacreader_local_server.cpp create mode 100644 YACReaderLibrary/yacreader_local_server.h create mode 100644 YACReaderLibrary/yacreader_main_toolbar.cpp create mode 100644 YACReaderLibrary/yacreader_main_toolbar.h create mode 100644 YACReaderLibrary/yacreaderlibrary_es.qm create mode 100644 YACReaderLibrary/yacreaderlibrary_es.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_fr.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_nl.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_pt.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_ru.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_source.ts create mode 100644 YACReaderLibrary/yacreaderlibrary_tr.ts create mode 100755 background.png create mode 100755 cleanOSX.sh create mode 100644 common/bookmarks.cpp create mode 100644 common/bookmarks.h create mode 100644 common/check_new_version.cpp create mode 100644 common/check_new_version.h create mode 100644 common/comic.cpp create mode 100644 common/comic.h create mode 100644 common/comic_db.cpp create mode 100644 common/comic_db.h create mode 100644 common/custom_widgets.cpp create mode 100644 common/custom_widgets.h create mode 100644 common/exit_check.cpp create mode 100644 common/exit_check.h create mode 100644 common/folder.cpp create mode 100644 common/folder.h create mode 100644 common/http_worker.cpp create mode 100644 common/http_worker.h create mode 100644 common/library_item.cpp create mode 100644 common/library_item.h create mode 100644 common/onstart_flow_selection_dialog.cpp create mode 100644 common/onstart_flow_selection_dialog.h create mode 100644 common/pdf_comic.h create mode 100644 common/pdf_comic.mm create mode 100644 common/pictureflow.cpp create mode 100644 common/pictureflow.h create mode 100644 common/qnaturalsorting.cpp create mode 100644 common/qnaturalsorting.h create mode 100644 common/yacreader_flow_gl.cpp create mode 100644 common/yacreader_flow_gl.h create mode 100644 common/yacreader_global.cpp create mode 100644 common/yacreader_global.h create mode 100755 compileOSX.sh create mode 100755 compileX11.sh create mode 100644 compressed_archive/7z_includes.h create mode 100644 compressed_archive/README_7zip.txt create mode 100644 compressed_archive/StdAfx.h create mode 100644 compressed_archive/StdAfx.h.cpp create mode 100644 compressed_archive/compressed_archive.cpp create mode 100644 compressed_archive/compressed_archive.h create mode 100644 compressed_archive/extract_callbacks.h create mode 100644 compressed_archive/extract_delegate.h create mode 100644 compressed_archive/libp7zip.patch create mode 100644 compressed_archive/open_callbacks.h create mode 100644 compressed_archive/wrapper.pri create mode 100755 create-dmg create mode 100644 custom_widgets/custom_widgets_yacreader.pri create mode 100644 custom_widgets/custom_widgets_yacreaderlibrary.pri create mode 100644 custom_widgets/help_about_dialog.cpp create mode 100644 custom_widgets/help_about_dialog.h create mode 100644 custom_widgets/yacreader_busy_widget.cpp create mode 100644 custom_widgets/yacreader_busy_widget.h create mode 100644 custom_widgets/yacreader_dark_menu.cpp create mode 100644 custom_widgets/yacreader_dark_menu.h create mode 100644 custom_widgets/yacreader_deleting_progress.cpp create mode 100644 custom_widgets/yacreader_deleting_progress.h create mode 100644 custom_widgets/yacreader_field_edit.cpp create mode 100644 custom_widgets/yacreader_field_edit.h create mode 100644 custom_widgets/yacreader_field_plain_text_edit.cpp create mode 100644 custom_widgets/yacreader_field_plain_text_edit.h create mode 100644 custom_widgets/yacreader_flow.cpp create mode 100644 custom_widgets/yacreader_flow.h create mode 100644 custom_widgets/yacreader_flow_config_widget.cpp create mode 100644 custom_widgets/yacreader_flow_config_widget.h create mode 100644 custom_widgets/yacreader_gl_flow_config_widget.cpp create mode 100644 custom_widgets/yacreader_gl_flow_config_widget.h create mode 100644 custom_widgets/yacreader_library_item_widget.cpp create mode 100644 custom_widgets/yacreader_library_item_widget.h create mode 100644 custom_widgets/yacreader_library_list_widget.cpp create mode 100644 custom_widgets/yacreader_library_list_widget.h create mode 100644 custom_widgets/yacreader_options_dialog.cpp create mode 100644 custom_widgets/yacreader_options_dialog.h create mode 100644 custom_widgets/yacreader_search_line_edit.cpp create mode 100644 custom_widgets/yacreader_search_line_edit.h create mode 100644 custom_widgets/yacreader_sidebar.cpp create mode 100644 custom_widgets/yacreader_sidebar.h create mode 100644 custom_widgets/yacreader_social_dialog.cpp create mode 100644 custom_widgets/yacreader_social_dialog.h create mode 100644 custom_widgets/yacreader_spin_slider_widget.cpp create mode 100644 custom_widgets/yacreader_spin_slider_widget.h create mode 100644 custom_widgets/yacreader_table_view.cpp create mode 100644 custom_widgets/yacreader_table_view.h create mode 100644 custom_widgets/yacreader_titled_toolbar.cpp create mode 100644 custom_widgets/yacreader_titled_toolbar.h create mode 100644 custom_widgets/yacreader_tool_bar_stretch.cpp create mode 100644 custom_widgets/yacreader_tool_bar_stretch.h create mode 100644 custom_widgets/yacreader_treeview.cpp create mode 100644 custom_widgets/yacreader_treeview.h create mode 100644 dependencies/poppler/bin/poppler-qt4.dll create mode 100644 dependencies/poppler/bin/poppler-qt4.dll.manifest create mode 100644 dependencies/poppler/bin/poppler-qt5.dll create mode 100644 dependencies/poppler/dependencies/bin/freetype6.dll create mode 100644 dependencies/poppler/dependencies/bin/freetype6.dll.manifest create mode 100644 dependencies/poppler/dependencies/bin/openjpeg.dll create mode 100644 dependencies/poppler/dependencies/bin/openjpeg.dll.manifest create mode 100644 dependencies/poppler/dependencies/lib/empty.txt create mode 100644 dependencies/poppler/include/qt4/poppler-annotation-helper.h create mode 100644 dependencies/poppler/include/qt4/poppler-annotation-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-annotation.h create mode 100644 dependencies/poppler/include/qt4/poppler-converter-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-embeddedfile-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-export.h create mode 100644 dependencies/poppler/include/qt4/poppler-form.h create mode 100644 dependencies/poppler/include/qt4/poppler-link-extractor-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-link.h create mode 100644 dependencies/poppler/include/qt4/poppler-media.h create mode 100644 dependencies/poppler/include/qt4/poppler-optcontent-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-optcontent.h create mode 100644 dependencies/poppler/include/qt4/poppler-page-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-page-transition-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-page-transition.h create mode 100644 dependencies/poppler/include/qt4/poppler-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-qiodeviceoutstream-private.h create mode 100644 dependencies/poppler/include/qt4/poppler-qt4.h create mode 100644 dependencies/poppler/include/qt5/ArthurOutputDev.h create mode 100644 dependencies/poppler/include/qt5/poppler-annotation-helper.h create mode 100644 dependencies/poppler/include/qt5/poppler-annotation-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-annotation.h create mode 100644 dependencies/poppler/include/qt5/poppler-converter-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-embeddedfile-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-export.h create mode 100644 dependencies/poppler/include/qt5/poppler-form.h create mode 100644 dependencies/poppler/include/qt5/poppler-link-extractor-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-link.h create mode 100644 dependencies/poppler/include/qt5/poppler-media.h create mode 100644 dependencies/poppler/include/qt5/poppler-optcontent-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-optcontent.h create mode 100644 dependencies/poppler/include/qt5/poppler-page-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-page-transition-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-page-transition.h create mode 100644 dependencies/poppler/include/qt5/poppler-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-qiodeviceoutstream-private.h create mode 100644 dependencies/poppler/include/qt5/poppler-qt5.h create mode 100644 dependencies/poppler/lib/poppler-qt4.lib create mode 100644 dependencies/poppler/lib/poppler-qt5.lib create mode 100644 files/about.html create mode 100644 files/about_es_ES.html create mode 100644 files/helpYACReader.html create mode 100644 files/helpYACReaderLibrary.html create mode 100644 files/helpYACReaderLibrary_es_ES.html create mode 100644 files/helpYACReader_es_ES.html create mode 100644 files/shortcuts.html create mode 100644 files/shortcuts2.html create mode 100644 files/translator.html create mode 100644 generateVS2010Projects.bat create mode 100755 icon.icns create mode 100644 images/7z.png create mode 100644 images/accept_shortcut.png create mode 100644 images/adjustToFullSize.png create mode 100644 images/alwaysOnTop.png create mode 100644 images/asignNumber.png create mode 100644 images/bookmark.png create mode 100644 images/branch-closed.png create mode 100644 images/branch-open.png create mode 100644 images/busy_background.png create mode 100644 images/center.png create mode 100644 images/clearSearch.png create mode 100644 images/clear_shortcut.png create mode 100644 images/close.png create mode 100644 images/colapse.png create mode 100644 images/colapse_osx.png create mode 100644 images/collapsed_branch_osx.png create mode 100644 images/collapsed_branch_selected.png create mode 100644 images/comic.png create mode 100644 images/comic7z.png create mode 100644 images/comicFolder.png create mode 100644 images/comicRar.png create mode 100644 images/comicTar.png create mode 100644 images/comicZip.png create mode 100644 images/comic_vine/downArrow.png create mode 100644 images/comic_vine/nextPage.png create mode 100644 images/comic_vine/previousPage.png create mode 100644 images/comic_vine/radioChecked.png create mode 100644 images/comic_vine/radioUnchecked.png create mode 100644 images/comic_vine/rowDown.png create mode 100644 images/comic_vine/rowUp.png create mode 100644 images/comic_vine/upArrow.png create mode 100644 images/coversPackage.png create mode 100644 images/db.png create mode 100644 images/defaultCover.png create mode 100644 images/deleteLibrary.png create mode 100644 images/deleting_progress/icon.png create mode 100644 images/deleting_progress/imgBottomLeft.png create mode 100644 images/deleting_progress/imgBottomMiddle.png create mode 100644 images/deleting_progress/imgBottomRight.png create mode 100644 images/deleting_progress/imgLeftMiddle.png create mode 100644 images/deleting_progress/imgRightMiddle.png create mode 100644 images/deleting_progress/imgTopLeft.png create mode 100644 images/deleting_progress/imgTopMiddle.png create mode 100644 images/deleting_progress/imgTopRight.png create mode 100644 images/dictionary.png create mode 100644 images/doublePage.png create mode 100644 images/down.png create mode 100644 images/dropDownArrow.png create mode 100644 images/edit.png create mode 100644 images/editComic.png create mode 100644 images/editIcon.png create mode 100644 images/empty_folder.png create mode 100644 images/expand.png create mode 100644 images/expand_osx.png create mode 100644 images/expanded_branch_osx.png create mode 100644 images/expanded_branch_selected.png create mode 100644 images/exportComicsInfo.png create mode 100644 images/exportComicsInfoIcon.png create mode 100644 images/exportLibrary.png create mode 100644 images/exportLibraryIcon.png create mode 100644 images/f.png create mode 100644 images/f_retina.png create mode 100644 images/find_folder.png create mode 100644 images/fit.png create mode 100644 images/flow1.png create mode 100644 images/flow2.png create mode 100644 images/flow3.png create mode 100644 images/flow4.png create mode 100644 images/flow5.png create mode 100644 images/flow_to_grid.gif create mode 100644 images/folder.png create mode 100644 images/folder_finished.png create mode 100644 images/folder_finished_macosx.png create mode 100644 images/folder_macosx.png create mode 100644 images/fromTo.png create mode 100644 images/getInfo.png create mode 100644 images/glowLine.png create mode 100644 images/goto.png create mode 100644 images/grid_to_flow.gif create mode 100644 images/help.png create mode 100644 images/helpImages/bookmark.png create mode 100644 images/helpImages/center.png create mode 100644 images/helpImages/colapse.png create mode 100644 images/helpImages/comicFolder.png create mode 100644 images/helpImages/coversPackage.png create mode 100644 images/helpImages/deleteLibrary.png create mode 100644 images/helpImages/doublePage.png create mode 100644 images/helpImages/edit.png create mode 100644 images/helpImages/expand.png create mode 100644 images/helpImages/exportLibrary.png create mode 100644 images/helpImages/fit.png create mode 100644 images/helpImages/flow1.png create mode 100644 images/helpImages/flow2.png create mode 100644 images/helpImages/flow3.png create mode 100644 images/helpImages/folder.png create mode 100644 images/helpImages/goto.png create mode 100644 images/helpImages/help.png create mode 100644 images/helpImages/icon.png create mode 100644 images/helpImages/importLibrary.png create mode 100644 images/helpImages/keyboard.png create mode 100644 images/helpImages/mouse.png create mode 100644 images/helpImages/new.png create mode 100644 images/helpImages/next.png create mode 100644 images/helpImages/nextComic.png create mode 100644 images/helpImages/notCover.png create mode 100644 images/helpImages/open.png create mode 100644 images/helpImages/openFolder.png create mode 100644 images/helpImages/openLibrary.png create mode 100644 images/helpImages/options.png create mode 100644 images/helpImages/prev.png create mode 100644 images/helpImages/previousComic.png create mode 100644 images/helpImages/properties.png create mode 100644 images/helpImages/removeLibrary.png create mode 100644 images/helpImages/rotateL.png create mode 100644 images/helpImages/rotateR.png create mode 100644 images/helpImages/save.png create mode 100644 images/helpImages/setBookmark.png create mode 100644 images/helpImages/setRoot.png create mode 100644 images/helpImages/shortcuts.png create mode 100644 images/helpImages/speaker.png create mode 100644 images/helpImages/updateLibrary.png create mode 100644 images/helpImages/zoom.png create mode 100644 images/hiddenCovers.png create mode 100644 images/hideComicFlow.png create mode 100644 images/icon.png create mode 100644 images/iconLibrary.png create mode 100644 images/iconSearch.png create mode 100644 images/imgBottomLeft.png create mode 100644 images/imgBottomMiddle.png create mode 100644 images/imgBottomRight.png create mode 100644 images/imgCenterSlide.png create mode 100644 images/imgCenterSlidePressed.png create mode 100644 images/imgEdit.png create mode 100644 images/imgGoToSlide.png create mode 100644 images/imgGoToSlidePressed.png create mode 100644 images/imgTopLeft.png create mode 100644 images/imgTopMiddle.png create mode 100644 images/imgTopRight.png create mode 100644 images/importBottomCoversDecoration.png create mode 100644 images/importComicsInfo.png create mode 100644 images/importComicsInfoIcon.png create mode 100644 images/importCover.png create mode 100644 images/importLibrary.png create mode 100644 images/importLibraryIcon.png create mode 100644 images/importTopCoversDecoration.png create mode 100644 images/importingIcon.png create mode 100644 images/iphoneConfig.png create mode 100644 images/libraryIcon.png create mode 100644 images/libraryIconSelected.png create mode 100644 images/libraryIcon_osx.png create mode 100644 images/libraryOptions.png create mode 100644 images/main_toolbar/back.png create mode 100644 images/main_toolbar/back_disabled.png create mode 100644 images/main_toolbar/back_disabled_osx.png create mode 100644 images/main_toolbar/back_osx.png create mode 100644 images/main_toolbar/divider.png create mode 100644 images/main_toolbar/forward.png create mode 100644 images/main_toolbar/forward_disabled.png create mode 100644 images/main_toolbar/forward_disabled_osx.png create mode 100644 images/main_toolbar/forward_osx.png create mode 100644 images/main_toolbar/fullscreen.png create mode 100644 images/main_toolbar/fullscreen_osx.png create mode 100644 images/main_toolbar/help.png create mode 100644 images/main_toolbar/help_osx.png create mode 100644 images/main_toolbar/server.png create mode 100644 images/main_toolbar/server_osx.png create mode 100644 images/main_toolbar/settings.png create mode 100644 images/main_toolbar/settings_osx.png create mode 100644 images/new.png create mode 100644 images/newLibraryIcon.png create mode 100644 images/newLibraryIcon_osx.png create mode 100644 images/next.png create mode 100644 images/nextComic.png create mode 100644 images/nextCoverPage.png create mode 100644 images/noLibrariesIcon.png create mode 100644 images/noLibrariesLine.png create mode 100644 images/notCover.png create mode 100644 images/notificationsLabel.png create mode 100644 images/numPagesLabel.png create mode 100644 images/numPagesLabelBig.png create mode 100644 images/numPagesLabelMedium.png create mode 100644 images/onStartFlowSelection.png create mode 100644 images/onStartFlowSelection_es.png create mode 100644 images/open.png create mode 100644 images/openFolder.png create mode 100644 images/openInYACReader.png create mode 100644 images/openLibrary.png create mode 100644 images/openLibraryIcon.png create mode 100644 images/openLibraryIcon_osx.png create mode 100644 images/options.png create mode 100644 images/pdf.png create mode 100644 images/prev.png create mode 100644 images/previousComic.png create mode 100644 images/previousCoverPage.png create mode 100644 images/properties.png create mode 100644 images/qrMessage.png create mode 100644 images/rar.png create mode 100644 images/rating0.png create mode 100644 images/rating1.png create mode 100644 images/rating2.png create mode 100644 images/rating3.png create mode 100644 images/rating4.png create mode 100644 images/rating5.png create mode 100644 images/readRibbon.png create mode 100644 images/readingRibbon.png create mode 100644 images/removeLibrary.png create mode 100644 images/removeLibraryIcon.png create mode 100644 images/rotateL.png create mode 100644 images/rotateR.png create mode 100644 images/save.png create mode 100644 images/selectAll.png create mode 100644 images/server.png create mode 100644 images/serverConfigBackground.png create mode 100644 images/setAllRead.png create mode 100644 images/setAllUnread.png create mode 100644 images/setBookmark.png create mode 100644 images/setRead.png create mode 100644 images/setReadButton.png create mode 100644 images/setRoot.png create mode 100644 images/setRoot_osx.png create mode 100644 images/setUnread.png create mode 100644 images/shortcuts.png create mode 100644 images/shortcuts_group_comics.png create mode 100644 images/shortcuts_group_folders.png create mode 100644 images/shortcuts_group_general.png create mode 100644 images/shortcuts_group_libraries.png create mode 100644 images/shortcuts_group_mglass.png create mode 100644 images/shortcuts_group_page.png create mode 100644 images/shortcuts_group_reading.png create mode 100644 images/shortcuts_group_visualization.png create mode 100644 images/showMarks.png create mode 100644 images/shownCovers.png create mode 100644 images/sliderAddPage.png create mode 100644 images/sliderBackground.png create mode 100644 images/sliderGround.png create mode 100644 images/sliderHandle.png create mode 100644 images/sliderSubPage.png create mode 100644 images/social_dialog/close.png create mode 100644 images/social_dialog/facebook.png create mode 100644 images/social_dialog/google+.png create mode 100644 images/social_dialog/icon.png create mode 100644 images/social_dialog/separator.png create mode 100644 images/social_dialog/shadow.png create mode 100644 images/social_dialog/twitter.png create mode 100644 images/speaker.png create mode 100644 images/tar.png create mode 100644 images/translatorSearch.png create mode 100644 images/trash.png create mode 100644 images/up.png create mode 100644 images/updateLibrary.png create mode 100644 images/updateLibraryIcon.png create mode 100644 images/updatingIcon.png create mode 100644 images/useNewFlowButton.png create mode 100644 images/useOldFlowButton.png create mode 100644 images/viewer_toolbar/bookmark.png create mode 100755 images/viewer_toolbar/bookmark_osx.png create mode 100644 images/viewer_toolbar/close.png create mode 100755 images/viewer_toolbar/close_osx.png create mode 100644 images/viewer_toolbar/doublePage.png create mode 100755 images/viewer_toolbar/doublePage_osx.png create mode 100644 images/viewer_toolbar/flow.png create mode 100644 images/viewer_toolbar/flow_osx.png create mode 100644 images/viewer_toolbar/full.png create mode 100755 images/viewer_toolbar/full_osx.png create mode 100644 images/viewer_toolbar/goto.png create mode 100644 images/viewer_toolbar/goto_osx.png create mode 100644 images/viewer_toolbar/help.png create mode 100755 images/viewer_toolbar/help_osx.png create mode 100644 images/viewer_toolbar/info.png create mode 100755 images/viewer_toolbar/info_osx.png create mode 100644 images/viewer_toolbar/magnifyingGlass.png create mode 100755 images/viewer_toolbar/magnifyingGlass_osx.png create mode 100644 images/viewer_toolbar/next.png create mode 100755 images/viewer_toolbar/next_osx.png create mode 100644 images/viewer_toolbar/open.png create mode 100644 images/viewer_toolbar/openFolder.png create mode 100644 images/viewer_toolbar/openFolder_osx.png create mode 100644 images/viewer_toolbar/openNext.png create mode 100644 images/viewer_toolbar/openNext_osx.png create mode 100644 images/viewer_toolbar/openPrevious.png create mode 100644 images/viewer_toolbar/openPrevious_osx.png create mode 100644 images/viewer_toolbar/open_osx.png create mode 100644 images/viewer_toolbar/options.png create mode 100755 images/viewer_toolbar/options_osx.png create mode 100644 images/viewer_toolbar/previous.png create mode 100755 images/viewer_toolbar/previous_osx.png create mode 100644 images/viewer_toolbar/rotateL.png create mode 100644 images/viewer_toolbar/rotateL_osx.png create mode 100644 images/viewer_toolbar/rotateR.png create mode 100644 images/viewer_toolbar/rotateR_osx.png create mode 100644 images/viewer_toolbar/save.png create mode 100644 images/viewer_toolbar/save_osx.png create mode 100644 images/viewer_toolbar/shortcuts.png create mode 100755 images/viewer_toolbar/shortcuts_osx.png create mode 100644 images/viewer_toolbar/showBookmarks.png create mode 100755 images/viewer_toolbar/showBookmarks_osx.png create mode 100644 images/viewer_toolbar/toHeight.png create mode 100755 images/viewer_toolbar/toHeight_osx.png create mode 100644 images/viewer_toolbar/toWidth.png create mode 100755 images/viewer_toolbar/toWidth_osx.png create mode 100644 images/viewer_toolbar/translator.png create mode 100755 images/viewer_toolbar/translator_osx.png create mode 100644 images/zip.png create mode 100644 images/zoom.png create mode 100644 release/languages/yacreader_es.qm create mode 100644 release/languages/yacreader_fr.qm create mode 100644 release/languages/yacreader_nl.qm create mode 100644 release/languages/yacreader_pt.qm create mode 100644 release/languages/yacreader_ru.qm create mode 100644 release/languages/yacreader_tr.qm create mode 100644 release/languages/yacreaderlibrary_es.qm create mode 100644 release/languages/yacreaderlibrary_fr.qm create mode 100644 release/languages/yacreaderlibrary_nl.qm create mode 100644 release/languages/yacreaderlibrary_pt.qm create mode 100644 release/languages/yacreaderlibrary_ru.qm create mode 100644 release/languages/yacreaderlibrary_tr.qm create mode 100644 release/server/docroot/css/reset.css create mode 100644 release/server/docroot/css/styles_ipad.css create mode 100644 release/server/docroot/css/styles_iphone.css create mode 100644 release/server/docroot/images/browse.png create mode 100644 release/server/docroot/images/browse@2x.png create mode 100644 release/server/docroot/images/combo.png create mode 100644 release/server/docroot/images/combo@2x.png create mode 100644 release/server/docroot/images/download.png create mode 100644 release/server/docroot/images/download@2x.png create mode 100644 release/server/docroot/images/f.png create mode 100644 release/server/docroot/images/f@2x.png create mode 100644 release/server/docroot/images/imported.png create mode 100644 release/server/docroot/images/imported@2x.png create mode 100644 release/server/docroot/images/indicator.png create mode 100644 release/server/docroot/images/indicator@2x.png create mode 100644 release/server/docroot/images/library.png create mode 100644 release/server/docroot/images/library@2x.png create mode 100644 release/server/docroot/images/next.png create mode 100644 release/server/docroot/images/next@2x.png create mode 100644 release/server/docroot/images/prev.png create mode 100644 release/server/docroot/images/prev@2x.png create mode 100644 release/server/docroot/images/read.png create mode 100644 release/server/docroot/images/read@2x.png create mode 100644 release/server/docroot/images/readMark.png create mode 100644 release/server/docroot/images/readMark@2x.png create mode 100644 release/server/docroot/images/readingMark.png create mode 100644 release/server/docroot/images/readingMark@2x.png create mode 100644 release/server/docroot/images/up.png create mode 100644 release/server/docroot/images/up@2x.png create mode 100644 release/server/docroot/login.html create mode 100644 release/server/templates/folder_ipad.tpl create mode 100644 release/server/templates/folder_iphone.tpl create mode 100644 release/server/templates/libraries_ipad.tpl create mode 100644 release/server/templates/libraries_iphone.tpl create mode 100755 releaseOSX.sh create mode 100644 shortcuts_management/actions_groups_model.cpp create mode 100644 shortcuts_management/actions_groups_model.h create mode 100644 shortcuts_management/actions_shortcuts_model.cpp create mode 100644 shortcuts_management/actions_shortcuts_model.h create mode 100644 shortcuts_management/edit_shortcut_item_delegate.cpp create mode 100644 shortcuts_management/edit_shortcut_item_delegate.h create mode 100644 shortcuts_management/edit_shortcuts_dialog.cpp create mode 100644 shortcuts_management/edit_shortcuts_dialog.h create mode 100644 shortcuts_management/shortcuts_management.pri create mode 100644 shortcuts_management/shortcuts_manager.cpp create mode 100644 shortcuts_management/shortcuts_manager.h diff --git a/CHANGELOG.txt b/CHANGELOG.txt new file mode 100644 index 00000000..3308f936 --- /dev/null +++ b/CHANGELOG.txt @@ -0,0 +1,118 @@ +7.1.0 +Añadida opción para resetear el rating de un comics +Corregidos bugs que afectaban a la información de página. +Corregido error que marcaba un comic terminado como empezado si se volvía a leer. +Añadidos 2 estados para las carpetas (Completo/Terminado) +Corregido bug en la comunicación YACReaderLibrary <-> YACReader +Añadidas las acciones relativas a los comics al menú contextual de la tabla de cómics. +Corrgido bug que provocaba el crecimiento ilimatado del log del servidor +Corregidos bugs menores + +7.0.2 (Sólo MacOSX) +Eliminado el uso de Poppler en la versión de MacOSX +Trabajo en traducciones. +Corregidos bugs menores + +7.0.1 +Añadido QsLog a YACReader +Corregido bug en la comunicación YACReaderLibrary <-> YACReader + +7.0 (Final) +Corregidos eventos de teclado en algunos diálogos +Corregido soporte para archivos Rar en sistemas Unix +Corregidos problemas borrando cómics +Mejorada la gestión de errores +Corregida la comunicación entre YACReader y YACReaderLibrary +Corregida la toolBar en MacOSX +Mejorada la compatabilidad de OpenGL en tarjetas NVIDIA +Corregidos bugs menores + +6.9 (No pública) +Añadida la apertura automática del siguiente/anterior cómic al llegar al final/portada del cómic actual +Corregido el comportamiento del diálogo de nueva versión detectada. Ahora avisa una vez al día o si el usuario lo elige cada 14 días. +Corregido el ajuste a lo ancho del título de la toolbar en YACReaderLibrary. +Añadido log a YACReaderLibrary (permitirá a los usuarios ofrecer más información sobre sus bugs) +Corregido bug en el historial de navegación (y al editar comics) después de usar el motor de búsqueda. + +6.8 (No pública) +Corregido bug que causaba un cierre inesperado después de cambiar el modo de sincronización vertical (flow) +Corregido bug que causaba que la toolbar en el visor no se pudiese ocultar/mostrar sin un cómic abierto +Mejorada la gestión de errores al abrir cómics +Corregidos algunos bugs relacionados con la apertura de cómics +Añadida función de rating +El visor ahora puede abrir archivos de imagen directamente. Si se abre un archivo de imagen se abre el directorio que lo contiene con todas las imágenes. +Corregida la ordenación de carpetas y cómics usada en la navegación desde dispositivos iOS + +6.7 (No pública) +Añadidos nuevos campos en la base de datos para almacenar información adicional sobre cómics: rating, página actual, bookmarks y configuración de imagen +Añadida comunicación entre YACReaderLibrary y YACReader para poder almacenar el progreso de los cómics e información adicional + +6.6 (No pública) +Modificado YACReader para que abra los archivos comprimidos usando 7z.dll (.so, .dylib) +YACReader abre ahora los cómics por la última página leída. +Corregido bug que causaba que algunos cómics no se pudiesen abrir desde YACReaderLibrary en YACReader +Corregido el modo en el que se actualizaba la "information label" + +6.5 +Nueva interfaz principal de YACReaderLibrary y YACReader +Corregido bug que causaba que el servidor no se activase en el primer arranque en MacOSX +Corregido bug que causaba un fallo al cerrar YACReaderLibrary cada vez que se usaba el servidor +Nuevo diseño para el diálogo de propiedades de los cómics. +Añadida navegación alante y atrás de las carpetas visitadas. +La edición del nombre de una biblioteca no fuerza ahora que se recargue la biblioteca +Corregido el color de fondo en la lupa +Nuevo botón para ajustar a lo alto +Eliminada la opción always on top +Mostrar en carpeta contenedora arreglado en Windows y MacOSX + +6.4 (No pública) +Normalizado el renderizado de páginas en modo doble página +Añadida la función de borrar cómics desde el disco duro +Nuevos iconos de la barra de herramientas de cómics + +6.3 (No pública) +Mejorada la gestión de errores relacionada con las bibliotecas +Añadido botón que permite ocultar las portadas en la pantalla de importación +Añadidos títulos "Bibliotecas" y "Carpetas" a la barra de navegación +Nuevos iconos para seleccionar la carpeta raíz, expandir y contraer todos. +Botón para cambiar el puerto del servidor por el usuario. +Ahora las columnas de la lista de cómics pueden reordenarse +Ahora YACReaderLibrary sólo permite una instancia ejecutandose. +Columna leído añadida. +Cambiado estilo de la lista de cómics +Corregidos bugs relacionados con realizar operaciones sobre cómics cuando no había ninguno seleccionado en la lista de cómics + +6.2 +Nueva ventana de "bienvenida" +Nueva ventana de importar/actualizar +Nuevo control para la búsqueda +Nueva imagen para las marcas de cómics leídos (sólo en OpenGL) +Cambiada la distribución de algunos iconos +Cambiado el modo de eliminar la metainformación (borrar base de datos/portadas de disco) +Ocultadas las opciones avanzadas de configuración de YACReader Flow, accesibles ahora tras pulsar un botón (diálogos de configuración más simples) + +6.0.1 (No pública) +Corregido bug al usar las teclas Inicio/fin +Corregido bug que al arrancar YACReaderLibrary por primera vez causaba que no se mostrasen las portadas (sólo bajo ciertas circunstancias) +Añadidos algunos atajos de teclado a YACReaderLibrary a los ya existentes + +6.0 + +Mejorada la velocidad de inicio gracias al uso de /LTCG como opción de compilación +Corregido bug relacionado con OpenGL que causaba consumo excesivo de CPU en tarjetas NVidia +Añadidos iconos para cada tipo de archivo soportado en YACReaderLibrary +Cambiado el icono "folder" en YACReaderLibrary +Añadida barra para ajustar el ancho de página en la toolbar de YACReader +Añadido widget para la information label +Añadido nuevo estilo visual a goToFlow +Añadidos filtros para controlar el brillo, el contraste y el valor gamma +Añadidas notificaciones de portada y última página +InformationLabel se muestra ahora en la esquina superiror derecha. +InformationLabel se muestra en 3 tamaños diferentes en función de la resolución +Corregido bug que causaba que las marcas de cómic leído no se dibujasen adecuadamente. +Se recuerda si se debe mostrar o no la "label" información. +Corregido bug que provocaba el fallo de YACReader al pasar muy rápido las páginas. +Añadida columna "Tamaño" a la lista de cómics en YACReaderLibrary +Añadida la ordinación "natural" de los comics que hay en directorio del cómic actual. +Corregido bug que causaba que se abriese el cómic erroneo en YACReaderLibrary. +Cambiado el modo en el que se cargan los lenguages, ahora se pueden añadir traducciones sin necesidad de recompilar. \ No newline at end of file diff --git a/COPYING.txt b/COPYING.txt new file mode 100644 index 00000000..94a9ed02 --- /dev/null +++ b/COPYING.txt @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This program 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. + + This program 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. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright (C) + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/INSTALL.txt b/INSTALL.txt new file mode 100644 index 00000000..8c65b479 --- /dev/null +++ b/INSTALL.txt @@ -0,0 +1,31 @@ +INSTALLATION GUIDE FOR LINUX USERS (BINARY PACKAGE) +********************************** +YACReader and YACReaderLibraries binaries are compiled under Ubuntu 13.10 and uses Qt5 and libpoppler-qt5. + + +COMPILATION GUIDE FOR LINUX/UNIX USERS +********************************** +YACReader and YACReaderLibrary are build using qmake. To build and install the program, run: + +qmake +make +make install + +from the source dir. For seperate builds of YACReader or YACReaderLibrary, enter their respective subfolders and run the commands from there. + +Build options: +--------------------- +You can adjust the installation prefix as well als the path make install uses to install the files. +Use "qmake PREFIX=DIR" to configure YACReader for your systems default prefix (for example "/", "/usr", "/usr/local"). + +For packaging purposes, you can use "make install INSTALL_ROOT=DIR" to install to a different location than the prefix. + +Default values: + +PREFIX=/usr +INSTALL_ROOT="" + + +DO YOU WANT TO HELP YACREADER? +****************************** +If you have experience creating packages, please help to create a package for your favourite distro! Send me an e-mail to: info@yacreader.com diff --git a/QsLog/QsLog.cpp b/QsLog/QsLog.cpp new file mode 100644 index 00000000..7eafd151 --- /dev/null +++ b/QsLog/QsLog.cpp @@ -0,0 +1,249 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "QsLog.h" +#include "QsLogDest.h" +#ifdef QS_LOG_SEPARATE_THREAD +#include +#include +#endif +#include +#include +#include +#include +#include +#include +#include + +namespace QsLogging +{ +typedef QVector DestinationList; + +static const char TraceString[] = "TRACE"; +static const char DebugString[] = "DEBUG"; +static const char InfoString[] = "INFO "; +static const char WarnString[] = "WARN "; +static const char ErrorString[] = "ERROR"; +static const char FatalString[] = "FATAL"; + +// not using Qt::ISODate because we need the milliseconds too +static const QString fmtDateTime("yyyy-MM-ddThh:mm:ss.zzz"); + +static Logger* sInstance = 0; + +static const char* LevelToText(Level theLevel) +{ + switch (theLevel) { + case TraceLevel: + return TraceString; + case DebugLevel: + return DebugString; + case InfoLevel: + return InfoString; + case WarnLevel: + return WarnString; + case ErrorLevel: + return ErrorString; + case FatalLevel: + return FatalString; + case OffLevel: + return ""; + default: { + assert(!"bad log level"); + return InfoString; + } + } +} + +#ifdef QS_LOG_SEPARATE_THREAD +class LogWriterRunnable : public QRunnable +{ +public: + LogWriterRunnable(QString message, Level level); + virtual void run(); + +private: + QString mMessage; + Level mLevel; +}; +#endif + +class LoggerImpl +{ +public: + LoggerImpl(); + +#ifdef QS_LOG_SEPARATE_THREAD + QThreadPool threadPool; +#endif + QMutex logMutex; + Level level; + DestinationList destList; +}; + +#ifdef QS_LOG_SEPARATE_THREAD +LogWriterRunnable::LogWriterRunnable(QString message, Level level) + : QRunnable() + , mMessage(message) + , mLevel(level) +{ +} + +void LogWriterRunnable::run() +{ + Logger::instance().write(mMessage, mLevel); +} +#endif + + +LoggerImpl::LoggerImpl() + : level(InfoLevel) +{ + // assume at least file + console + destList.reserve(2); +#ifdef QS_LOG_SEPARATE_THREAD + threadPool.setMaxThreadCount(1); + threadPool.setExpiryTimeout(-1); +#endif +} + + +Logger::Logger() + : d(new LoggerImpl) +{ +} + +Logger& Logger::instance() +{ + if (!sInstance) + sInstance = new Logger; + + return *sInstance; +} + +void Logger::destroyInstance() +{ + delete sInstance; + sInstance = 0; +} + +// tries to extract the level from a string log message. If available, conversionSucceeded will +// contain the conversion result. +Level Logger::levelFromLogMessage(const QString& logMessage, bool* conversionSucceeded) +{ + if (conversionSucceeded) + *conversionSucceeded = true; + + if (logMessage.startsWith(QLatin1String(TraceString))) + return TraceLevel; + if (logMessage.startsWith(QLatin1String(DebugString))) + return DebugLevel; + if (logMessage.startsWith(QLatin1String(InfoString))) + return InfoLevel; + if (logMessage.startsWith(QLatin1String(WarnString))) + return WarnLevel; + if (logMessage.startsWith(QLatin1String(ErrorString))) + return ErrorLevel; + if (logMessage.startsWith(QLatin1String(FatalString))) + return FatalLevel; + + if (conversionSucceeded) + *conversionSucceeded = false; + return OffLevel; +} + +Logger::~Logger() +{ +#ifdef QS_LOG_SEPARATE_THREAD + d->threadPool.waitForDone(); +#endif + delete d; + d = 0; +} + +void Logger::addDestination(DestinationPtr destination) +{ + assert(destination.data()); + d->destList.push_back(destination); +} + +void Logger::setLoggingLevel(Level newLevel) +{ + d->level = newLevel; +} + +Level Logger::loggingLevel() const +{ + return d->level; +} + +//! creates the complete log message and passes it to the logger +void Logger::Helper::writeToLog() +{ + const char* const levelName = LevelToText(level); + const QString completeMessage(QString("%1 %2 %3") + .arg(levelName) + .arg(QDateTime::currentDateTime().toString(fmtDateTime)) + .arg(buffer) + ); + + Logger::instance().enqueueWrite(completeMessage, level); +} + +Logger::Helper::~Helper() +{ + try { + writeToLog(); + } + catch(std::exception&) { + // you shouldn't throw exceptions from a sink + assert(!"exception in logger helper destructor"); + throw; + } +} + +//! directs the message to the task queue or writes it directly +void Logger::enqueueWrite(const QString& message, Level level) +{ +#ifdef QS_LOG_SEPARATE_THREAD + LogWriterRunnable *r = new LogWriterRunnable(message, level); + d->threadPool.start(r); +#else + write(message, level); +#endif +} + +//! Sends the message to all the destinations. The level for this message is passed in case +//! it's useful for processing in the destination. +void Logger::write(const QString& message, Level level) +{ + QMutexLocker lock(&d->logMutex); + for (DestinationList::iterator it = d->destList.begin(), + endIt = d->destList.end();it != endIt;++it) { + (*it)->write(message, level); + } +} + +} // end namespace diff --git a/QsLog/QsLog.h b/QsLog/QsLog.h new file mode 100644 index 00000000..f72e827c --- /dev/null +++ b/QsLog/QsLog.h @@ -0,0 +1,137 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOG_H +#define QSLOG_H + +#include "QsLogLevel.h" +#include "QsLogDest.h" +#include +#include + +#define QS_LOG_VERSION "2.0b3" + +namespace QsLogging +{ +class Destination; +class LoggerImpl; // d pointer + +class QSLOG_SHARED_OBJECT Logger +{ +public: + static Logger& instance(); + static void destroyInstance(); + static Level levelFromLogMessage(const QString& logMessage, bool* conversionSucceeded = 0); + + ~Logger(); + + //! Adds a log message destination. Don't add null destinations. + void addDestination(DestinationPtr destination); + //! Logging at a level < 'newLevel' will be ignored + void setLoggingLevel(Level newLevel); + //! The default level is INFO + Level loggingLevel() const; + + //! The helper forwards the streaming to QDebug and builds the final + //! log message. + class QSLOG_SHARED_OBJECT Helper + { + public: + explicit Helper(Level logLevel) : + level(logLevel), + qtDebug(&buffer) {} + ~Helper(); + QDebug& stream(){ return qtDebug; } + + private: + void writeToLog(); + + Level level; + QString buffer; + QDebug qtDebug; + }; + +private: + Logger(); + Logger(const Logger&); // not available + Logger& operator=(const Logger&); // not available + + void enqueueWrite(const QString& message, Level level); + void write(const QString& message, Level level); + + LoggerImpl* d; + + friend class LogWriterRunnable; +}; + +} // end namespace + +//! Logging macros: define QS_LOG_LINE_NUMBERS to get the file and line number +//! in the log output. +#ifndef QS_LOG_LINE_NUMBERS +#define QLOG_TRACE() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::TraceLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::TraceLevel).stream() +#define QLOG_DEBUG() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::DebugLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::DebugLevel).stream() +#define QLOG_INFO() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::InfoLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::InfoLevel).stream() +#define QLOG_WARN() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::WarnLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::WarnLevel).stream() +#define QLOG_ERROR() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::ErrorLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::ErrorLevel).stream() +#define QLOG_FATAL() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::FatalLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::FatalLevel).stream() +#else +#define QLOG_TRACE() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::TraceLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::TraceLevel).stream() << __FILE__ << '@' << __LINE__ +#define QLOG_DEBUG() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::DebugLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::DebugLevel).stream() << __FILE__ << '@' << __LINE__ +#define QLOG_INFO() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::InfoLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::InfoLevel).stream() << __FILE__ << '@' << __LINE__ +#define QLOG_WARN() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::WarnLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::WarnLevel).stream() << __FILE__ << '@' << __LINE__ +#define QLOG_ERROR() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::ErrorLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::ErrorLevel).stream() << __FILE__ << '@' << __LINE__ +#define QLOG_FATAL() \ + if (QsLogging::Logger::instance().loggingLevel() > QsLogging::FatalLevel) {} \ + else QsLogging::Logger::Helper(QsLogging::FatalLevel).stream() << __FILE__ << '@' << __LINE__ +#endif + +#ifdef QS_LOG_DISABLE +#include "QsLogDisableForThisFile.h" +#endif + +#endif // QSLOG_H diff --git a/QsLog/QsLog.pri b/QsLog/QsLog.pri new file mode 100644 index 00000000..4afc6b47 --- /dev/null +++ b/QsLog/QsLog.pri @@ -0,0 +1,22 @@ +INCLUDEPATH += $$PWD +#DEFINES += QS_LOG_LINE_NUMBERS # automatically writes the file and line for each log message +#DEFINES += QS_LOG_DISABLE # logging code is replaced with a no-op +#DEFINES += QS_LOG_SEPARATE_THREAD # messages are queued and written from a separate thread +SOURCES += $$PWD/QsLogDest.cpp \ + $$PWD/QsLog.cpp \ + $$PWD/QsLogDestConsole.cpp \ + $$PWD/QsLogDestFile.cpp \ + $$PWD/QsLogDestFunctor.cpp + +HEADERS += $$PWD/QSLogDest.h \ + $$PWD/QsLog.h \ + $$PWD/QsLogDestConsole.h \ + $$PWD/QsLogLevel.h \ + $$PWD/QsLogDestFile.h \ + $$PWD/QsLogDisableForThisFile.h \ + $$PWD/QsLogDestFunctor.h + +OTHER_FILES += \ + $$PWD/QsLogChanges.txt \ + $$PWD/QsLogReadme.txt \ + $$PWD/LICENSE.txt diff --git a/QsLog/QsLogDest.cpp b/QsLog/QsLogDest.cpp new file mode 100644 index 00000000..ae9f44bc --- /dev/null +++ b/QsLog/QsLogDest.cpp @@ -0,0 +1,70 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "QsLogDest.h" +#include "QsLogDestConsole.h" +#include "QsLogDestFile.h" +#include "QsLogDestFunctor.h" +#include + +namespace QsLogging +{ + +Destination::~Destination() +{ +} + +//! destination factory +DestinationPtr DestinationFactory::MakeFileDestination(const QString& filePath, + LogRotationOption rotation, const MaxSizeBytes &sizeInBytesToRotateAfter, + const MaxOldLogCount &oldLogsToKeep) +{ + if (EnableLogRotation == rotation) { + QScopedPointer logRotation(new SizeRotationStrategy); + logRotation->setMaximumSizeInBytes(sizeInBytesToRotateAfter.size); + logRotation->setBackupCount(oldLogsToKeep.count); + + return DestinationPtr(new FileDestination(filePath, RotationStrategyPtr(logRotation.take()))); + } + + return DestinationPtr(new FileDestination(filePath, RotationStrategyPtr(new NullRotationStrategy))); +} + +DestinationPtr DestinationFactory::MakeDebugOutputDestination() +{ + return DestinationPtr(new DebugOutputDestination); +} + +DestinationPtr DestinationFactory::MakeFunctorDestination(QsLogging::Destination::LogFunction f) +{ + return DestinationPtr(new FunctorDestination(f)); +} + +DestinationPtr DestinationFactory::MakeFunctorDestination(QObject *receiver, const char *member) +{ + return DestinationPtr(new FunctorDestination(receiver, member)); +} + +} // end namespace diff --git a/QsLog/QsLogDest.h b/QsLog/QsLogDest.h new file mode 100644 index 00000000..a404487b --- /dev/null +++ b/QsLog/QsLogDest.h @@ -0,0 +1,99 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOGDEST_H +#define QSLOGDEST_H + +#include "QsLogLevel.h" +#include +#include +class QString; +class QObject; + +#ifdef QSLOG_IS_SHARED_LIBRARY +#define QSLOG_SHARED_OBJECT Q_DECL_EXPORT +#elif QSLOG_IS_SHARED_LIBRARY_IMPORT +#define QSLOG_SHARED_OBJECT Q_DECL_IMPORT +#else +#define QSLOG_SHARED_OBJECT +#endif + +namespace QsLogging +{ + +class QSLOG_SHARED_OBJECT Destination +{ +public: + typedef void (*LogFunction)(const QString &message, Level level); + +public: + virtual ~Destination(); + virtual void write(const QString& message, Level level) = 0; + virtual bool isValid() = 0; // returns whether the destination was created correctly +}; +typedef QSharedPointer DestinationPtr; + + +// a series of "named" paramaters, to make the file destination creation more readable +enum LogRotationOption +{ + DisableLogRotation = 0, + EnableLogRotation = 1 +}; + +struct QSLOG_SHARED_OBJECT MaxSizeBytes +{ + MaxSizeBytes() : size(0) {} + explicit MaxSizeBytes(qint64 size_) : size(size_) {} + qint64 size; +}; + +struct QSLOG_SHARED_OBJECT MaxOldLogCount +{ + MaxOldLogCount() : count(0) {} + explicit MaxOldLogCount(int count_) : count(count_) {} + int count; +}; + + +//! Creates logging destinations/sinks. The caller shares ownership of the destinations with the logger. +//! After being added to a logger, the caller can discard the pointers. +class QSLOG_SHARED_OBJECT DestinationFactory +{ +public: + static DestinationPtr MakeFileDestination(const QString& filePath, + LogRotationOption rotation = DisableLogRotation, + const MaxSizeBytes &sizeInBytesToRotateAfter = MaxSizeBytes(), + const MaxOldLogCount &oldLogsToKeep = MaxOldLogCount()); + static DestinationPtr MakeDebugOutputDestination(); + // takes a pointer to a function + static DestinationPtr MakeFunctorDestination(Destination::LogFunction f); + // takes a QObject + signal/slot + static DestinationPtr MakeFunctorDestination(QObject *receiver, const char *member); +}; + +} // end namespace + +#endif // QSLOGDEST_H diff --git a/QsLog/QsLogDestConsole.cpp b/QsLog/QsLogDestConsole.cpp new file mode 100644 index 00000000..ed7fc53b --- /dev/null +++ b/QsLog/QsLogDestConsole.cpp @@ -0,0 +1,55 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "QsLogDestConsole.h" +#include +#include + +#if defined(Q_OS_WIN) +#define WIN32_LEAN_AND_MEAN +#include +void QsDebugOutput::output( const QString& message ) +{ + OutputDebugStringW(reinterpret_cast(message.utf16())); + OutputDebugStringW(L"\n"); +} +#elif defined(Q_OS_UNIX) +#include +void QsDebugOutput::output( const QString& message ) +{ + fprintf(stderr, "%s\n", qPrintable(message)); + fflush(stderr); +} +#endif + +void QsLogging::DebugOutputDestination::write(const QString& message, Level) +{ + QsDebugOutput::output(message); +} + +bool QsLogging::DebugOutputDestination::isValid() +{ + return true; +} diff --git a/QsLog/QsLogDestConsole.h b/QsLog/QsLogDestConsole.h new file mode 100644 index 00000000..f80f490e --- /dev/null +++ b/QsLog/QsLogDestConsole.h @@ -0,0 +1,52 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOGDESTCONSOLE_H +#define QSLOGDESTCONSOLE_H + +#include "QsLogDest.h" + +class QString; + +class QsDebugOutput +{ +public: + static void output(const QString& a_message); +}; + +namespace QsLogging +{ + +// debugger sink +class DebugOutputDestination : public Destination +{ +public: + virtual void write(const QString& message, Level level); + virtual bool isValid(); +}; + +} + +#endif // QSLOGDESTCONSOLE_H diff --git a/QsLog/QsLogDestFile.cpp b/QsLog/QsLogDestFile.cpp new file mode 100644 index 00000000..0f8f8048 --- /dev/null +++ b/QsLog/QsLogDestFile.cpp @@ -0,0 +1,155 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "QsLogDestFile.h" +#include +#include +#include +#include + +const int QsLogging::SizeRotationStrategy::MaxBackupCount = 10; + +QsLogging::RotationStrategy::~RotationStrategy() +{ +} + +QsLogging::SizeRotationStrategy::SizeRotationStrategy() + : mCurrentSizeInBytes(0) + , mMaxSizeInBytes(0) + , mBackupsCount(0) +{ +} + +void QsLogging::SizeRotationStrategy::setInitialInfo(const QFile &file) +{ + mFileName = file.fileName(); + mCurrentSizeInBytes = file.size(); +} + +void QsLogging::SizeRotationStrategy::includeMessageInCalculation(const QString &message) +{ + mCurrentSizeInBytes += message.toUtf8().size(); +} + +bool QsLogging::SizeRotationStrategy::shouldRotate() +{ + return mCurrentSizeInBytes > mMaxSizeInBytes; +} + +// Algorithm assumes backups will be named filename.X, where 1 <= X <= mBackupsCount. +// All X's will be shifted up. +void QsLogging::SizeRotationStrategy::rotate() +{ + if (!mBackupsCount) { + if (!QFile::remove(mFileName)) + std::cerr << "QsLog: backup delete failed " << qPrintable(mFileName); + return; + } + + // 1. find the last existing backup than can be shifted up + const QString logNamePattern = mFileName + QString::fromUtf8(".%1"); + int lastExistingBackupIndex = 0; + for (int i = 1;i <= mBackupsCount;++i) { + const QString backupFileName = logNamePattern.arg(i); + if (QFile::exists(backupFileName)) + lastExistingBackupIndex = qMin(i, mBackupsCount - 1); + else + break; + } + + // 2. shift up + for (int i = lastExistingBackupIndex;i >= 1;--i) { + const QString oldName = logNamePattern.arg(i); + const QString newName = logNamePattern.arg(i + 1); + QFile::remove(newName); + const bool renamed = QFile::rename(oldName, newName); + if (!renamed) { + std::cerr << "QsLog: could not rename backup " << qPrintable(oldName) + << " to " << qPrintable(newName); + } + } + + // 3. rename current log file + const QString newName = logNamePattern.arg(1); + if (QFile::exists(newName)) + QFile::remove(newName); + if (!QFile::rename(mFileName, newName)) { + std::cerr << "QsLog: could not rename log " << qPrintable(mFileName) + << " to " << qPrintable(newName); + } +} + +QIODevice::OpenMode QsLogging::SizeRotationStrategy::recommendedOpenModeFlag() +{ + return QIODevice::Append; +} + +void QsLogging::SizeRotationStrategy::setMaximumSizeInBytes(qint64 size) +{ + Q_ASSERT(size >= 0); + mMaxSizeInBytes = size; +} + +void QsLogging::SizeRotationStrategy::setBackupCount(int backups) +{ + Q_ASSERT(backups >= 0); + mBackupsCount = qMin(backups, SizeRotationStrategy::MaxBackupCount); +} + + +QsLogging::FileDestination::FileDestination(const QString& filePath, RotationStrategyPtr rotationStrategy) + : mRotationStrategy(rotationStrategy) +{ + mFile.setFileName(filePath); + if (!mFile.open(QFile::WriteOnly | QFile::Text | mRotationStrategy->recommendedOpenModeFlag())) + std::cerr << "QsLog: could not open log file " << qPrintable(filePath); + mOutputStream.setDevice(&mFile); + mOutputStream.setCodec(QTextCodec::codecForName("UTF-8")); + + mRotationStrategy->setInitialInfo(mFile); +} + +void QsLogging::FileDestination::write(const QString& message, Level) +{ + mRotationStrategy->includeMessageInCalculation(message); + if (mRotationStrategy->shouldRotate()) { + mOutputStream.setDevice(NULL); + mFile.close(); + mRotationStrategy->rotate(); + if (!mFile.open(QFile::WriteOnly | QFile::Text | mRotationStrategy->recommendedOpenModeFlag())) + std::cerr << "QsLog: could not reopen log file " << qPrintable(mFile.fileName()); + mRotationStrategy->setInitialInfo(mFile); + mOutputStream.setDevice(&mFile); + } + + mOutputStream << message << endl; + mOutputStream.flush(); +} + +bool QsLogging::FileDestination::isValid() +{ + return mFile.isOpen(); +} + diff --git a/QsLog/QsLogDestFile.h b/QsLog/QsLogDestFile.h new file mode 100644 index 00000000..ee7b5232 --- /dev/null +++ b/QsLog/QsLogDestFile.h @@ -0,0 +1,101 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOGDESTFILE_H +#define QSLOGDESTFILE_H + +#include "QsLogDest.h" +#include +#include +#include +#include + +namespace QsLogging +{ +class RotationStrategy +{ +public: + virtual ~RotationStrategy(); + + virtual void setInitialInfo(const QFile &file) = 0; + virtual void includeMessageInCalculation(const QString &message) = 0; + virtual bool shouldRotate() = 0; + virtual void rotate() = 0; + virtual QIODevice::OpenMode recommendedOpenModeFlag() = 0; +}; + +// Never rotates file, overwrites existing file. +class NullRotationStrategy : public RotationStrategy +{ +public: + virtual void setInitialInfo(const QFile &) {} + virtual void includeMessageInCalculation(const QString &) {} + virtual bool shouldRotate() { return false; } + virtual void rotate() {} + virtual QIODevice::OpenMode recommendedOpenModeFlag() { return QIODevice::Truncate; } +}; + +// Rotates after a size is reached, keeps a number of <= 10 backups, appends to existing file. +class SizeRotationStrategy : public RotationStrategy +{ +public: + SizeRotationStrategy(); + static const int MaxBackupCount; + + virtual void setInitialInfo(const QFile &file); + virtual void includeMessageInCalculation(const QString &message); + virtual bool shouldRotate(); + virtual void rotate(); + virtual QIODevice::OpenMode recommendedOpenModeFlag(); + + void setMaximumSizeInBytes(qint64 size); + void setBackupCount(int backups); + +private: + QString mFileName; + qint64 mCurrentSizeInBytes; + qint64 mMaxSizeInBytes; + int mBackupsCount; +}; + +typedef QSharedPointer RotationStrategyPtr; + +// file message sink +class FileDestination : public Destination +{ +public: + FileDestination(const QString& filePath, RotationStrategyPtr rotationStrategy); + virtual void write(const QString& message, Level level); + virtual bool isValid(); + +private: + QFile mFile; + QTextStream mOutputStream; + QSharedPointer mRotationStrategy; +}; + +} + +#endif // QSLOGDESTFILE_H diff --git a/QsLog/QsLogDestFunctor.cpp b/QsLog/QsLogDestFunctor.cpp new file mode 100644 index 00000000..601139d9 --- /dev/null +++ b/QsLog/QsLogDestFunctor.cpp @@ -0,0 +1,57 @@ +// Copyright (c) 2014, Razvan Petru +// Copyright (c) 2014, Omar Carey +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "QsLogDestFunctor.h" +#include +#include + +QsLogging::FunctorDestination::FunctorDestination(LogFunction f) + : QObject(NULL) + , mLogFunction(f) +{ +} + +QsLogging::FunctorDestination::FunctorDestination(QObject *receiver, const char *member) + : QObject(NULL) + , mLogFunction(NULL) +{ + connect(this, SIGNAL(logMessageReady(QString,int)), receiver, member, Qt::QueuedConnection); +} + + +void QsLogging::FunctorDestination::write(const QString &message, QsLogging::Level level) +{ + if (mLogFunction) + mLogFunction(message, level); + + if (level > QsLogging::TraceLevel) + emit logMessageReady(message, static_cast(level)); +} + +bool QsLogging::FunctorDestination::isValid() +{ + return true; +} diff --git a/QsLog/QsLogDestFunctor.h b/QsLog/QsLogDestFunctor.h new file mode 100644 index 00000000..e34631f0 --- /dev/null +++ b/QsLog/QsLogDestFunctor.h @@ -0,0 +1,59 @@ +// Copyright (c) 2014, Razvan Petru +// Copyright (c) 2014, Omar Carey +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOGDESTFUNCTOR_H +#define QSLOGDESTFUNCTOR_H + +#include "QsLogDest.h" +#include + +namespace QsLogging +{ +// Offers various types of function-like sinks. +// This is an advanced destination type. Depending on your configuration, LogFunction might be +// called from a different thread or even a different binary. You should not access QsLog from +// inside LogFunction and should not perform any time-consuming operations. +// logMessageReady is connected through a queued connection and trace messages are not included +class FunctorDestination : public QObject, public Destination +{ + Q_OBJECT +public: + explicit FunctorDestination(LogFunction f); + FunctorDestination(QObject *receiver, const char *member); + + virtual void write(const QString &message, Level level); + virtual bool isValid(); + +protected: + // int used to avoid registering a new enum type + Q_SIGNAL void logMessageReady(const QString &message, int level); + +private: + LogFunction mLogFunction; +}; +} + +#endif // QSLOGDESTFUNCTOR_H diff --git a/QsLog/QsLogDisableForThisFile.h b/QsLog/QsLogDisableForThisFile.h new file mode 100644 index 00000000..c70af103 --- /dev/null +++ b/QsLog/QsLogDisableForThisFile.h @@ -0,0 +1,22 @@ +#ifndef QSLOGDISABLEFORTHISFILE_H +#define QSLOGDISABLEFORTHISFILE_H + +#include +// When included AFTER QsLog.h, this file will disable logging in that C++ file. When included +// before, it will lead to compiler warnings or errors about macro redefinitions. + +#undef QLOG_TRACE +#undef QLOG_DEBUG +#undef QLOG_INFO +#undef QLOG_WARN +#undef QLOG_ERROR +#undef QLOG_FATAL + +#define QLOG_TRACE() if (1) {} else qDebug() +#define QLOG_DEBUG() if (1) {} else qDebug() +#define QLOG_INFO() if (1) {} else qDebug() +#define QLOG_WARN() if (1) {} else qDebug() +#define QLOG_ERROR() if (1) {} else qDebug() +#define QLOG_FATAL() if (1) {} else qDebug() + +#endif // QSLOGDISABLEFORTHISFILE_H diff --git a/QsLog/QsLogLevel.h b/QsLog/QsLogLevel.h new file mode 100644 index 00000000..62984732 --- /dev/null +++ b/QsLog/QsLogLevel.h @@ -0,0 +1,45 @@ +// Copyright (c) 2013, Razvan Petru +// All rights reserved. + +// Redistribution and use in source and binary forms, with or without modification, +// are permitted provided that the following conditions are met: + +// * Redistributions of source code must retain the above copyright notice, this +// list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright notice, this +// list of conditions and the following disclaimer in the documentation and/or other +// materials provided with the distribution. +// * The name of the contributors may not be used to endorse or promote products +// derived from this software without specific prior written permission. + +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +// IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE +// OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +// OF THE POSSIBILITY OF SUCH DAMAGE. + +#ifndef QSLOGLEVEL_H +#define QSLOGLEVEL_H + +namespace QsLogging +{ + +enum Level +{ + TraceLevel = 0, + DebugLevel, + InfoLevel, + WarnLevel, + ErrorLevel, + FatalLevel, + OffLevel +}; + +} + +#endif // QSLOGLEVEL_H diff --git a/QsLog/QsLogSharedLibrary.pro b/QsLog/QsLogSharedLibrary.pro new file mode 100644 index 00000000..51320684 --- /dev/null +++ b/QsLog/QsLogSharedLibrary.pro @@ -0,0 +1,35 @@ +# This pro file will build QsLog as a shared library +include(QsLog.pri) + +TARGET = QsLog +VERSION = "2.0.0" +QT -= gui +CONFIG -= console +CONFIG -= app_bundle +CONFIG += shared +TEMPLATE = lib + +DESTDIR = $$PWD/build-QsLogShared +OBJECTS_DIR = $$DESTDIR/obj +MOC_DIR = $$DESTDIR/moc + +win32 { + DEFINES += QSLOG_IS_SHARED_LIBRARY +} + +unix:!macx { + # make install will install the shared object in the appropriate folders + headers.files = QsLog.h QsLogDest.h QsLogLevel.h + headers.path = /usr/include/$(QMAKE_TARGET) + + other_files.files = *.txt + other_files.path = /usr/local/share/$(QMAKE_TARGET) + + contains(QT_ARCH, x86_64) { + target.path = /usr/lib64 + } else { + target.path = /usr/lib + } + + INSTALLS += headers target other_files +} diff --git a/README.txt b/README.txt new file mode 100644 index 00000000..b9d66e4a --- /dev/null +++ b/README.txt @@ -0,0 +1,22 @@ +LICENSE +******* +This software has been developed by Luis Ángel San Martín Rodríguez (luisangelsm@gmail.com) under GPL v3 license +(for more details read COPYING.txt). + +CONTACT +******* +Project home page : www.yacreader.com +e-mail: + info@yacreader.com + support@yacreader.com +Social: + Facebook - http://www.facebook.com/YACReader + Twitter - https://twitter.com/yacreader + YouTube - https://www.youtube.com/user/yacreader + +If you need help or have any suggestion, please, send me an e-mail. + +DONATIONS +********* +YACReader is free but it needs money to still be alive, so please, +if you like YACReader, visit the home page and make a donation. \ No newline at end of file diff --git a/YACReader.desktop b/YACReader.desktop new file mode 100644 index 00000000..20c5f201 --- /dev/null +++ b/YACReader.desktop @@ -0,0 +1,12 @@ +[Desktop Entry] +Name=YACReader +GenericName=Yet Another Comic Reader +Comment=Yet Another Comic Reader +Exec=YACReader %f +Icon=/usr/share/YACReader/icon.png +Terminal=false +Type=Application +StartupNotify=true +Categories=Graphics;Viewer; +MimeType=application/x-cbz;application/x-cbr;application/x-cbt;application/x-cb7;application/x-pdf;application/x-zip;application/x-rar;application/x-7z;inode/directory; +X-Desktop-File-Install-Version=0.22 diff --git a/YACReader.pro b/YACReader.pro new file mode 100644 index 00000000..bfc8dc47 --- /dev/null +++ b/YACReader.pro @@ -0,0 +1,3 @@ +TEMPLATE = subdirs +SUBDIRS = YACReader YACReaderLibrary +YACReaderLibrary.depends = YACReader diff --git a/YACReader/YACReader.icns b/YACReader/YACReader.icns new file mode 100644 index 0000000000000000000000000000000000000000..34f9b1a7779b4af6e7da2639cc58ba17d939b510 GIT binary patch literal 52094 zcma&O1z?j`_c;7iDAM9mcbC+g;tbswHiN+oCd@Y`jG^U8-KZyN+&xX&)ZN|P-6>G4 zFs%8`6Q=KP@ArQ%v`zBdd(S=Ro^$T8o3B3o{$~WK5BTb5A8!QFj?zaEEo}tRHS_Xt zTOEX4*Fr46kaf?D^(tRo29J;>=lfr%f4h6Lzpv&a19*S-;e*-1(b0kCoIIs3f|wOg z4zzYpwO8fHIq63c}O9sSoCDXH12jM6&Ax0VD%f2sQ$XFrWEDvQ{?PAeE$ zqNAPE@$>RvWQ7*2`ass{HHem02(mWB8HDI-A?9be zjRT$Kr0vTH$dx7E*4m=7Jzbl7{N6g4O~NG3;9G z4lQJBUV%iB%ZZN*x^^*gj~24GkWJzwhldg_UAc;GN02Sqk->3S0|?iyT)uMN4nZsv zVlReA--x+!BawPy1z@!+LLkr1%&%xo{lWy0(sMj``XoVB9lmWPhLkoET>F#m??~-T z|70aBJ`~aQ!*Ghk@cA7ngO^AxGjViwSg#v|m}=V}{_6aX7mxZotb#|#isPUE_{rge zM}0i)9oC@8H^)wVadzvT{rk3f+8SHIUY*?b-u4}b_q$ja8Cu!sA+86vJNj(3v2e09 zwRg8&ir{wJ+d0~p+SnVr`uaHOAZ|MhO{_LLxVw05*|rVFTkZDt*>m{#$q&Bx=JGxu z@DiU>gd~nAyP#3{F^1Jr``srFe;7~}=C2P}>1pYmO#DepD+u-q*5=Ln;&-&1P5uL| zpErN^4SWRbrmc+_yc)n1M!mHL0jAVOR-t2RXp8EhSFnekL+ZuF=g?lL)rSG0Y4uZh zg|(;ZLIep{KY95M8!@#GsZ#&`6nlTiM5pWT?*~^Pb$`8orj{TGS^eAM%L!gQQ}0I* ziy0V)ee&XknuNi3FPO*PK3C6t3n*b6w#wvd^)oeL6}DX1=MA>&MCupU^o*@nV;f?) z;{;hf8Fly=-fR>0ej{%8*{kQ?egA;l#^qRlErT^{3~gL&je$oHu72|1-u?S` z@63#Jws-f7j(D#}v;q<1C6>|C3wLkddvI_5*4*@XZ+Am&U3mldbx*xOWUG$FcQ?mp z=kMQ{o*Eq;AM76)Y_BaX$S#c7VGJm0Atqa^rYCRD&rJ^X4o-{=kM*~f=c>|Ea?{?m zBOo6vb+|M$I5s^o*xWcgcWbh*sk$^hRVEOrvOaTzRU;ey^G5mxdfOXnd;7;nnoIJ^ zveT4uE?t`Sh35)*ZvIY3_fU6hLqk_jYja7qDl1nZ6bfi0LHds#D^TQ!^$$HwUG25y zRShjo#hI#Xm5@(o)2PX8$$1Z;vbO!@vDS*xoSc%1+B{`ino=ZSFhCmqq1^Klj@J$Yc2XMW^v4Lbf1Er0eK+?86HYF$oFe zl%kUE>1O#icwbwcFl4>@ zZGyx4P0J9iBxLE5wc5J6FpP}oU>My(Lg6=xEYn)DbomPX70coEpZu=BcB7f2|2DrZ zn|*B7Ve0QCzppZK-2W51a-hGpwJI<9U7Wct_GH6Svt1VoW}ZHIcz62FWKVTv<~d&n zgHXg!d+kwr_x!_KljGxKqeGpwdAWu8K|674Kn!$ELOX~01|}v(hkCo)noIJPQW;-% zajWYRjMI(2?5OD;nHX+wt*vdWtf;Nblk=z~@+ZEVU?+eEe%bY{gQI;dm4%f}^>t0< z89Y*Kd^kC5pEYp0p4InF6%C!O)hcC8&p>B&hL9Z_9eO=5Hs*C_STC|3C#WtdFU?g- zOUmkNGX)fGQcPs{)n6l$4!Z%R5YxT6g*644N<~pohKiXKlSqyspo9zOBNN_sU4|kb zn7>z$T97N_i&E25S&6a4_^_Z0S1(__6dH2U6|{@C^_iMX0f$Ux^QFmAF*jnPLN8rH zFJHbEh@NuNgXgwy)Mv6OkrXaHJ|rUa+O>cy=dWD5eC67et4D2N58k&vSd~e-5gt#9 z4Gp<|HSpS{fXe|FE?vHS^@r`I7$=*(nv)t6cr`fg#`VjWL&8D=351ImL#|)C{-Nh; z?71PHDh|2)-T9!Xm!*#W1Ut#CXojJpvW7(rpMDP%G+i6dqZ zW4Xdp?!Yqy#K^?`0yT-nrm#}EJib7{t`nd>d;^ptZd?o5Te_u~$ zJ9p<58m@l##*GVKeR%Z9u{YlM@X$^_KfH^vljkaotBr$#5BVSY;-h274!*Yc!!4e! z&W=tFYaMX97*}un?!3*>pV4pk?A^Zom2F#IJA81@Hdib2H6Aaydi}N#kJqmSef!pC z@11-19yq$k+u7F1)!2HItvzt{600-syTA6$XD8j=-aLN#)UhqLwq|BV8;#5j&2g|| zWP|+|`~7$B*@bsH`pVW-^0_!+|yV+QnTG?2bn3>zU*%&)JdVBe7*@E}-Fw-zK{s@>u<|cNI#wK=tTk!a8JAAhu zJh1x}Z+AF1fvLTZ*_v5d*m(IkcpZIx|63 z&%XNM^DoYQ^F=^x$hmLgs0Z9Y`2$lMyB&G|h(s}duU!)2F^HJ>SH`jz|`=%Q|B%`kG9+JCs!{w zC*?GKVge!ue7wxy$NV7AO~BKjroG>HGgzzP=q0)a+SB3R@d(*)z zAFXxHn|E$?gbmUt^*^4$J9uh#n%OY(7!0RJeQD=iF+1wd1|0Kh#-l%sHqGDorHIb& z&$VFP=Ypu5NUEX>4JB3;F^cVvqi8i_g^aTQwcPcaf^0|9`e; z>Os&?ux3Q4e)5m8zxRBq{uI%LMSKQU<-fJ(Y9)r{vfjVU$A5Htp&r6;IP_m0140WX z03Z?!$A4{$>Zi6aJ_GE`e{GBE#}=^qs{j6cb|pNo{cq2ot1A&8ZQg%-{!AT?zzI$J z@9W!)=;$Jt;rnmxxw;sjYa^z&G`92q>GDi{5ZhrL*fAxAp-SN*!jOt zu=wnmIs;Q9(6kUe9tL3%V+u`M1Xfs#(giWqG~ka{Mg4zq*WwHH(+vFI7!*1nOU$>t zbylL8NB#JkHJpnt2fzHSgR!Cx5&@{f=`IrMm8?q=~Mmm?fEayd~m|c*wkF(EMWMradYrIcKVmVcuw+#)1MrF z^UYoRos7+l!MJ+aw9H-Y?eTjLz5UT=AD#T@{kISJ*_jxw#f0?trnT0>+QQt@#?IN@ z6X#-QZeqOpuf@OYqpiDi)w*??OiZDfZ1_v;zxon$R@N@WfaZYnQsZ>PFEXqPMOJIA z*kobr=;DfVb#=4ev_=nwj}wqJ*v{x~u=nu7;e2-<+~fZWx@+kxzD|xNVEM-(=J3?O z*4xX?=kR%U#o(=pwy~k!wx*Qtx8i+4ecr&lcx0u6x0~-7a^nMndhOH4Pae(BPY(}t z2o8I0vC_uOMr#mzYn<1sVeR1iJ$?A#!NZ657M|R?Gd?iX-<$W9x4*}-NX&j+<>2#4 z@twsN3lAPHpbs}ZcyRyr&6^YbU7g)MJvC7~eDRIaYKZ{1sXya3bP z2bXC6=I}sob8~BJOLGNruaDodP{aZ3A-5Y3r^n}R&)-|PfA8*{xtW>i@sWYf?&g}R zn!2*$+O)7ec!v8Z)l@u~5_zW$+s-j2qqg8aOKQvR=a z{PG~gRSQ||k8c^7n7Ij4-o6R;>tH{au!AF$;{#oF<;5js+490H$`L=X$O2(%FIV)| zz|`cO+qY+?hlhJR`zI&HhKEOoI-05r@>3)VSz3C5;u{~QKy2!bo42(Mjg3rB&W=y? zbu_k&&duGLf)&?S*W{_PWJ)PVsmNDe^!3&WLY&=j0n=T>Lqk&o!`%&)^}`dhGjkK& z^%Z61rOI@LOv>l*WSOZK{QZ^(BG*^o@i~*7{e1&H&8>AcO|X!W@&4MPg513FJY|+# zDdDk`CCaSJTkx2o(n5@U-|Xn?n(ppuZ>(=@=<9ClXsavA%TcA}mMT)kB88BdOcYAe zL$=_tV2u`H<8!XRp`)k2v$e6lxuK_{y`{b+H!Ce8FH#<25@+E0iVWU zCX?xG;>X@zz+W!lId9&cm76LM@tAZvPmz+Y1UHu_;Id`PEO3OuE|Rjp^V$mjWiVpD z#K-GonTRLmi3D6uNN1-a~J_fAak^KMB*B3_($L*vChFeYW};fkv|g^L~S)7as5L=j-d|kNf^+znH`4 zC+>6i+u~yd2VIOtpBif|U1Mlr?c#=a3j3{?DJ@DrK&{ota?jVN3R4DeU3T)r9CJDXF(W&ia5B5I?1Y^U_%bZSV5!;BTh*n( zo{m0X_^}cFRr|Udf)84|IxR=EILHdOf~c*gW)8++s&SB|dY(I%u35WEPk=1b)6-f3 zAtppifGt6dsF4ulUr&Q&TFcj)TH4s#JE9Ic<{MTm!Pb|En7~+_b=EGfj`prTJN!NT z@&5js-EFM_EjSEJ;pqx1S7$qyJzpheH4pdKceT|PXH!1*^}wx;#R!KOSi9P}ortNp z^<4e@x4SdbQ++*+rTL18-7el{+R>Vmz|7uxZ(#jn_0#$3$;qjik=cd5!N!W}(oEtz zZeC7HBN1CIWTmCs>HNw2_ogSOXQrkmCnv`HySkf-^YRLc^Vr|`x_hh)$5yk+@w4*D zk)GbMnK@0Tk-@&ic>JIvZ;XQ^eBbn559?&@0iAl+btGUA5s9aB%p(Ew8GwvZK7VSRs(tHg~l3wU=j$ zI6PKFTx3}I^(z6Pi4kvmx-Sbv-dpbJ%52UlEv_g@&z4J4Ym0Mg8cO6`MshNj9F-Uv z9dh;ZFQJL#6J8$rpozeB^w^$Pme*a7pOu=TOertQ$j+3}DWsSi$=s;epukAN#q;MQ z;u21KVSzqSM&`J;D-^j!WqFyY3YDTXFDE^PnVfVZE;%tcJoMb9%NH;Gav?Gz@>5R_ zP||KTZeLd_GczhurFdjSi0p3kfC!2mJa=NN~`VYlNTP z#5rmXfHj_Yo=g^b?aKE*T)Y;BUi|T=;D``HRAPL1KtOo#jrd5yuY|z!{+^(1z#h@_ zu>V0y2@4E3_u~~p+>P)^A~7K%DkdT}F^Wuuz*2Zv5+meIH;lDlC8l0JR2n^j6dw^9 z9>-;gctkplLQjroiZZw?aw3VZpnv4z1Fk9@0J?52uZrX2Nklr6mc-9aVG6`TH~<86 znNT1VNwacUXK`Nk;fUQzZ>J-PAxR=JU%=)`1fck_|1zmOEi((IU^lld~*A)?%4{hk{>Ur{`uZbwzkjG~6 zlvVXzBX^!Ye=^le+3oD+W2aI6R$9nfS1&gl&Tsp!{fA%KzVlEpDJ+g%Rdns-0T(x4 z_l;Vy;2W>AaX~#cxx3>$yqr#U)I`Nld2!q9-F`s>%lMiQ}x@~u&aYf!|LnyKI<{nnv9^J`j@)w>f`Ze9$&V?0%|W+zm*+%Abq^y zxH%5&X;A%o#(PHVTB31QI0$Zl>R;*K&=`Ht)ZP~Seh#wCK^(Nhc$3XW9Uih&*VRvV z1MB|1kfTtu?l^cX4uXu(vZ^Z=m~^!q;A7>4I~ywYIXb z#o;{s{r$WgEH?b5@GabMc9t%CKK~_*n&;)r<>sY` zsh8e)FLK{@Uw0=5UtbRoW3878-_*+aP_VE_k^J@X!-tN%;~&;@;Mnf1+kI`!?QmWW zFBQIlsoPf>+3`QVd+6Bl6Yril{`T7kcOTe=_wn}I>~3u0;^}4(ju?Y^Z2L{V;3R(Q zvC}`EeCO@A-#m8Y)o(tb*r7cEHw(>(Duu;4fr>Xd~^BY>D~MG?AZx{vaNpJyIwu= z`l~x!t*mV9R&I26!xX*&&L_uv`z!B7g%E!J`juU~w%~Wc?|moUKYZYoZQBkWwlK0W zTV?O{Qp~iREeV(G_Z|wncIouTo45J8Zr#6c@2dyj*pG9vH8b6`(F$T+HtS8ituzXM ztp|>a-@f_y2iHIN<~46u=M!g7oIQ2u$X+LBXKQmSQ*)zDW+v7fH+eW?3g6D*Sg_-^ zEnB|d^T8V)_O7Sj-u=oar(QqiX=iV1YH4O-YPiYJ$i{RX4!0DP;4)A1_kVKS=Cf~) z%YH9M+{e3Jx9{|E!@F5p+gqDifoHnO$kbx9&l;TjO9j5pchj%Oe6D+Xd3ZYGoZj8% z>hJAhYh`X~ZsTBMZ)9w0vdP5KdzafxILwB(ckuk#)yvh=(ZSjlVvDYBHsH!z+TncnxNh3G$=<`k#bwJjA7?jr zH%#5zIXeI3>EY;TYiVhR&2DXCVrF4v?6KK;gW)wmR8mlmc~XF=Ib^%d3*c%ZrkGJ2V86&gqYhnIezVfv$L_VFt@R@ zG&eOex3F?>b+7e_nUC;XIxyZ%#1C~j6ei8S(rO{?B2a&$FA+$_wVr8 za`5os{jVL^=HYIyQTMAoT|aYiGBLBav9@u=Ia*oT?>@F~`>XGrI{3=oH{bo_cL4iV9Qr z&Teksz4_*;v*)f~`xd1|UkZXPUqE*>sRH0pkXx0BcLGXW9tR61LbmQ_^R-qSfScV~X# z?){0VJ)2!TT|nL2f!zlpa^&#aAu&-m5~HIN#3eNYHy=aPX?~RCVT1Fs{X^ZaarSg| zbA`RcZ?RgvdZnZ9zAwW{OTRp@)85U;Z3C$LCE9DPoNzeQeFM(Z)7>fIN9VVHP6+je zIb2NsuI^W^Gd8oa!&yXdce`IK$a&Yr%XZ!0)xC{*+z&Qi4yg&Yp6mWn_j+!wS%)v& zdVX1>+qIa8fu+R{nc+7(kDFkzXH40fyOm`PB)VB(8DLQMtF0A9F^7$8Y(Uwwk>%Rv zKYhE?biI{E+v{1|=|S%KrMA}wZI7AvD&$K|@OSxNQ6S{^rlb3huo--+qYK>m?@ded zX#LjHAuLrn*rzhWTh3VM9*ed8-=+;8>-b$Ox%Kxr;PW2VTl#;CALgGzxH&o%MEMRkY&{~Z968%NJWQNc5a-~GM4fSBTwpZ*c61{gYs?z!7)==}Gt zf4qMFT%Ef9AK^n?#J)oP;-B&9KYGJo2vv94{B3|Pa%dil6#n=2RNV%9`8%>Y$QfAo zfA5Svd8)pF#agkZgPg$X82*iX{lCP6|JGX# zUFe9eW;F-?huNX>LfwG52Cy8-e+yUtWBx_;ZHz)7wsZd;ef~#p*aP(wJD6Mt+5bN+ z2l_w9L4RH3!vC}!tiPM4Kly*54z`}(Y&GkV{IC6Qf!7WrGXJ;r06nooLkCIu-{yY? z(RkP|U4-$!^?$BDgYm=vPj=%U8?dOpZv~5oA*cVR^*>iDuL7@&=~{=ekwsk#mekN(~?^82rf z=Or1SWU0i=q6%_mx`T@QYCiIg$?#|AE{=jr?50Q_A~{ekbxzkeJf zI!N^I`|)D&>1(hOy*~-_j{)F=XlrXjc53~^%l(Fxf1zh#W^A-^t-*huQNNGW(p|C) zx%hJTUM#-&W`%``shJrhVhrI_`j4i+$;#b-^A_*jvzp~US3msFzygdvXpq}6{EtN~ zvvTpUEVyHeh~;WeALj&$OsbGF9-bdBwY!K+?_|?e*5SLp^si}MEM8%kM8xh za{xPD)Be$4+uqgD@B9?l2FSj6&=&OGr(b;X@kgiMJF?9dTayW;wYnOj~@3oU39zS~Q z_`cn*IDj>8VhCj?e{V~yA)sVy<>`o&NOGPtTnB_@fiAZoyd?n{8UN@*k_m z2I_3Gb#QdFx3Ms@u(EZ+d3pOlApzL3MkZ#MiS#c`$6(F64I9m@Y;0_-t*tDvW?^n- zYGm*)ec_|u_fJc2`EsoG$qe?%)Wmqxs{fe$tMgd$>x>sR31%6okqpR8p{9sa3|`B0 zWb2WoGj59FYgdM z;sgoTm8KE+wSK}6dV57qD z%Y3}{|Cl%Y==mZ+y0uWrtajY9CW zqp7i>Tz27oUs%9O6@tJ7dfvOQH$Qv^!hdnXzs)ZQ=%fU)hXC1qPH<*01-JP8$&*Kqf_@`B4t`|*U>>SR z7w7NZoxgKuW^{N28$UGE(^Owu!T5GRMmHITLJ(fp$9wZn4FvTn82<3#qsPBJdGgz1 zAY9Nx!h+tzhYuE>KYtK-d-?2WALJ4te=vj&E^Du?Da%0L*n%x%C6t<)2H{uv?1>(@ z4JOXBrw{MlfA|>vP5Uv>Vj=V)`atjg-Me#hH*cb|hBG6By*=Gsot<61gF}M@-7rxW z@ADlRd_+KzK)jBRSIi?+z2fQpTX*k2TzCwdO?afWun+cuBF zr*Gc8efR#t!h;7e7xEZ>V7}n{y7z$Gx6!%PH>W4Y$3|erf!?l`rj|CeeN|f<1@I~>aKKP_X^^5uOk?EPayO0=#64r%>3y+>Ye7K;QRp%Z|O1PzW z_x}B{F^v362fEuD>g(#8(3VxLt<4Q}&27y^89Wi=4Q$$#OwF{0-q&Z}aa5uS09K8XM6j z0|@h%H?%b7rE&$7BN)FiG!w7%PExDy4D=2TPTskT3Chyj!0h*bdjh+I-qXkCotv4S zgnaQdI<;hUV4%ORgV3el3E9oMy4u>Bn(8{VVO4c$QDISCQ+6=}FA&H<(@byV z`@wHdn_Kz^hi2{o>)x8XedpobJNF(ux)(IBd*}A88JJaTVgiLf&Z7f;o$X!S-Cf;% z{eA6iO`6%Ot7;%ET2xq&UsPRFCTGUS27dtaLkXJ`0dKH%LfzZgH!uRz-oAy-Exo;P z_ujqx3$STRfT=*_11ELHN2jIgk5QfzeyHZr=h%ox8Pg=l=Y|2e*M`(HU(>+z=*o z$47@o0Ri0PXkY&zh$Z|7Ug-iDtqnCb03kO!E2{v5k)KssRZ%FU!~}ka0dazmBrxJi zpDXGI4NxcuqA)vm3%#xXl55Wnoe+_PE$FnbdyeRs#h z-yYt-`xM0Z&coS>@rj9X>}MF(JOWdV!fYVR1A{{&qqjlZj`sC*G*^}u=I7;AHg}D+ zrlNTUY0~t9^0Le>T}fQ36Z=cezYlgN1B znDz*$-u|Vd<6~pvW5a{!fb;O!_{97e$a+slOJj9uaZzDzR%UuyYTaCEPPSMeP059f zyoeruZ7W8sB-oKayz$O^k6XG&rf*J7PC(Y|@!h*OXBTeYo*KV5j*jXLkAQ0I=|%gN z4vdTr55f;R;09pECT0c)`+GXzfT}Ai!FD`Pu2MmDwnVASlw@b+7UULXvnkOhfMkgp zk}da^sE68n#;2zz(Fuc_P+KuI^KgE4eC+NdI=ppgczB?vy$uxK!05yjP;Yc_aCm$a z)H0keeO-_*YXpQeoR!O$3&kl)g&cme06nx&SI9{G4kO#j1Pzn<_`Uvcp}BK(auOCg zK70T6+|4_WZcmPm-JQlb2}&@8bAn3(ZZDl zxmhV}S*lW*BIhVXXy!6SN_jznf=0fAiDSG5DHA`Ly1#XB9J0n^W0OzkCq_r^&x3s5 znjarS2h4lA+dJAifO5C)-Jid8b7p21N-lsGFv_(w5^C0ih{*HvRH~dbz6OwtEoGsZ z%O#4E;w&YVOxTX?6MzEQ{MhgK!{4gAM<*wShhW1eM#qL9-T`bz=R4aw+S{92Y8&gC z1_ovy-kzHUH82Y&^8kizZ+lBaO;vSGb#+x`mLyG;mzO1$rUFtju8<*8W~A{|`6a3} zIx%>6AU;lm)hAoT>cPt4(TU-KkumIO9=v}Cn0erCdvj-7b6sOaRaImEz`fZSIA&(1 z@2YQ(YLt0LV|^VcfU2s>ipnC9ATu*vCF21knN&)bQN(hkRGOYun4@GR1-%Ar0kA+; zU3|}Us%sku$HxYSN3cU-;MT4Ffxf=^&c^PJrkeV)s`8eeiQ9uy01BNpoVhuHb?RxV zsj4h5D=Du8*({LCm1w4kLL>uNVhNwe;_yWhfl>u&{d7+BwPToSSQ!fv<+PbN`?$1g z04^N>Y5E8HXYcg)b$8Bnwe)p1RM(eQ)V6fb%=QgUPNOsK(_r9Cj1Bd))&VGhRa#b2 zQIub(lx1aRX2=Dp081dFNy!p6o5xhC(6l9mjF79xF<_>U$%pe*-|tk`Z*z4$0HYry z)v<5-ULTz8t#|r{nj5QXORJl^J8q75b@hx*PEW&ufEiP*uuW*WOKEX&Njcnys8lIZ zvNAH#`7(uEDwJ^re42<(Ve#Zx-G7lJ`0A-Y-~o&1`@|J2_O$o*b@lc46Z(utfX_Qx zTEK*?E3K}qZ|!KlIRX;a(K{A2rr@+?rR<#}?EN|lzD%1KQHSW>2p&SlUcohTNGQbZZ)841B> zeg_5&Y3m=)p1s}D(caPXcyX{7%(3pSj<%-8rl$JV)|$G8rjGgcj=*-^u8!8$MznNA zQ31fp$;wp8Qqf%d(y}6Xnm8SRad~M_S|Vo)B?$}~B|e@C#VUM7nk<77a`rE1ru)b# z&%0XMTRVG(#(RLC-9X9a768`R-ikJ9H;;}GTJ=FMf+f>f1DF+{xgObB8B%d}Zca{i zL2AFPd7L&_q1*swKUY#RhH!E0XkVKNgA57Iximq8?@Hum~HPT3TA0>+9+&iebxgvNGg~tel*@{QTSuC6`4Lu|T2)G$N0W@(iJPCyCAF zka%&)WHFl~;&MewYQiUf!dt#AfpM>=tgXGN8ExO#ICFovt^vSQmz5S57M7Nm*VNY4 zqYZYAjScm+%mZ+a0Y41hd-43I zEj>d6Esf1h0IZ^-q^!IQ6Rw=1($cb$lB$|ov~E>>eMKIK8z?D-GF<~OQ^unyRl*bw zo6qJ^D0Ci|12?VM@o<@o$Rs5aX^bQRpF(Fa6KTBHF=6-v;F&nalg|F(p|-}Rx>DG* zvZA8mELCPkR&Gg2DW-xeYY4SVDss|NQWR*4t}-<>BOO)Z%g@8a0;3C$pdyC5FP@_6Im8ztd4Mk9vnkhC3Rd01+6Zs4zc8k(#E;EGj82 zF9B5H5BP9$GzFFWD->|UM#!TG_(C3AF5$5FL^hwF%!_3)m~2)en;per-=NVbL^6#M z#b;5-2`qXvg^clw(@Xr8_#OZ0`s03d#Im!ox)62_thKZhr7{)rZLpCb767p%H%%@N zlr2k1QNpD;K9P&^-KboVfWl!V^Z6VqGoA_X66vhuWCkgYMg?}ECC2ke%-AG;QZyyR z8{?Ok@Yegi{lmq_{a_(>HdzW+_yfH+ ze)xHz4;@+7Q&(10Rsn}h5CBUrD8?vTn3Iv7E)SAz1qyQ+d;vd^FBEg((jO;T$WP{T znIuLOgU(lmouF6^cBm?^sI)9UD?Lr6Qpy6QTjePV9uY1~kr+^!NF{RxbOw*i=5dG&auO3X zG?7hLIqJ+wo!VyhGc|J*WE+49Og?tKwo6O|0iA+Wui^&A0r$v%TG0DkfVr&v6Ih2~n zAji-ck=J8B#?%Q|dIbC`|8IVI*fap%V@*+!DosU@8OUVOujZN@ekQWDXnZk?2Gwk<4VWZ&0bp5tJxG@|wgXA|)|0DIQpZMk53y zyoy=N1iXpg*((p~`uYck>IzgD8Q_|TMFKWUAW@{IqnSuLLHUY8!DUF~auze52D=ZJ zIN2Nuk4t58$ut%lfJZV}jHGxfH6kfKKAB1d5hGKg5@YGKa2ho+Anclkf-hm62$-ww z?CTq7$plMH2}itu!(xf~;#3uyv0sTfD+(5$CzCKKF>ENjr3gi07L&v0!kI>bGn>p| z5;=5gG=)MAOOB+#O;~CYF_{{f9L=JF1Ox@deW0P>OVj-5ZnVF-sSP3rnP7otq-cO~ zgk+XHEh9rASEMUa*ihUn~kxE4= zt4Tx}B_ckKnV3vZyb=@*$_kttINNmnFLl>e6I%A>rDuTG00aaRgUVqEW$7TgX{mA! zk13LHh|C18K**p;q#_oREflasLNb@fA+cDndP)q9N)9B&(op(ZFd8VdxacSn6QnNW z+Krdz+V8M@KkhCoAhf(&mX@JX(3xPJkwWR9+o&`qg9U}d$+U#z_%NUllSQF1`Ft{) z2)aWeuRSP7wVF3#FQmHCULFF*{ zAc(*lq-3^;$&typdljy*tptOwG(s=gGuExmd&ioM01(e6aNdX(~ag z3U0+`NJI>YlrKfO&UAqgj3hp+U70RX$!!G!s3#2rbsGM zisW*hge#HC z0s(``K$)2b{!t3=MtH~`O+=l61mR7$ejhj!O5*_Yvjq}Zh>RzYi9~D>7j7b;QhlKu zmMaF=8a#Pax(4VXf?bfXF(OK(A{jxsN}&M z5~&gn7pX$2$jmP&DguWCe1V$Ay0X%my7JQE%v1>_B0T6Y%!;h!Xee#wx8=QSSykt+ z6Zx`K;68AClyYeb=7=a2X#~}}jLfXejDorb@Ch608d}=A21n4L4a37zGqW?35c%sZ z%ag`ky>R}sZJJqm8YcJgKXN9#_DXPM0tNJnTm{Di#5d)l)YP=htc>iO+`Rmpyc+QR zz?JRl?e8C*g-|v+uM1)9If!R28z~hLFP{7K089yKDuHHBGe7@5Uloy~f}@itOg?O$ zSSDAdV5owrkVVL|%Z5NL%-G)1)eSLUxbp>Ri2HCm>=uN^272Vg>t~MmV{3F0BcXVG zU;G*3Pd^4kF$F>)DnP)y!?EGi62T>u)U;GZdIn}2S5-qKp}A*#_7;|XfU09~3aZPb zahKlQjO7GPrLeLv{03hi{Pw-4E=0r;$;8Bj_}J*ENc0kV(e~F10Z|l=NTGo0=A44U zg7S(c2%ilNjr3H=K;aR8KDya247NfJTO0+sKS(%k*?08#`(ONo1~^>)?z`{LefQ1R zUws>)6f0AO@nB~OILgAR=BB!0VerLEmwrC87mtlMRbol7IJ}FWAFO2S&fWWuo&4yt zZ+<#|iNel=?W4yNN#syAUzDB3kBj{3<9A=*<`<3;R)vit<1JyFuixfZ!=A zP09ET-op8T=MSn%mEepkWCCS6ok)$1z*4N`2x=OS-{{%+;?9eyiXymuBu$aXGQ>DqN~v9>fv41vbfhM}8tQ%&c6u-wuR7=(I0=+o!< zg_$USnY^X%=FOSe0PNZ7A`j0~u$TTYY&e+yK9}#R z+cOHpDFsb~Gqck}pSuToX?d=Irw~tEbD%=KkWrW+%WoWJN6Jc%SYf0p4_~}fsB-hi*LEQxCXgsIWC_>Hknyl1m`^&RT9q~wy}U> z0Ql$NXrViatXW}hdbDw&>g?goR))sLrsmeDjh6N5DP*J8`i+J)eKbeIO&d0DGBku* z0c)#uI@5^Wl8qbvf)81RZqV7daiz{IqH6%pH=;&bhKA@SosH`ZbZ;Tr%MA2UgQY9i zZ8S34V4ypXAbLT|wU;i@eE@av%k-A$XeuLL)&TtbZ^>U30{W;SVjJV3We8gf-}hih z_0YC8G=vHPeM4I=_F5k*1PpDZ9$L1BP$6I#?Xk|?-POh2=attEZiTx99v}+b9ofHy6J*zX%hm^1$&d&I}B!D9z1=C=b}OQm}h&yyM~;;(^#g zm4LosgvVNUoZHs7QC4w%dw1{D?Z?lbV^sprA@~0P?wF4EmC;|ty6Z!gfWB?G#|n2> z&$oU_Ya5w-v>2o|gN*;Phxcwl=zVe+?sgUzWx&xYV#Y+|Ms9Tr_CO4wOhDf@%ww6G z%l6OsHIt8?6V$6;JcFF{{pm@_ZT9tcHP)f^>mc4&SpsfYnnEHK)8aV4Y{vxw-uh4` zpl=)MfycQX3eFt8eGh7h7N0NNo}QSRo`uxl%~?=s1AU|ObImP~80hcnsDu1yahgOd zqKBS+3kNIEhdKe6LC>r1!rJg=WhCpf< zqQ=!#RmG`7AwTKkqu%a8KrAQ~FbwfnW!5`=K26*`s@q_?grsW+7KP9XN?h5g?PMGd?~tMCe944LfQfQlLqSLM*?$ ztN?=dG3XmU8U&$sz&6lB$IT^j0cr;p#yWZiMyGE=0yc0KnL#JfDFl+D=(x@3a32Jr zAu`<21d%?N4U3az=M}*bpTdIgKs)QNnGVVa^aDLioL@`qSHGBp3{y+jIGoy8J_r&S zptoQ~tx0Ukv2`Ou*pv`MZ>@(k3gm8c^3WV48_imt55_&66c+NsR@g{es39;Uc&u}K zM+W6k584}>n>z-_AyYm9d12kT$*JjS!lc#&q&-(bQg8^%aYA(rRvUws8y4rMr)6Yi zLsl5_7pk0sOh!^r;E%gJfsBS=<=CPg=5C&Gi|S`%wPp299gxNx86BILotm6|Fb%f= z#{u=AvDJ{pg2YpA@7UA~gpDggSzc5!JOqY;PFyWaCg$Gj|^@7oy`jSpEzhaUbdfszREst!;d4@_AXlCPki` zrT}@8${<7r=9x^uQ>3RRh5mZ^$DJ5cLOlUYzsfaEtsXDQDK2j4g)}v!k@P3WhbHFl zK?%j!I842K7_y0w4ryy|N86UR)`4+Ro?C$Co8{$YDlk(RtYbMCGh#NK#*n3^#9sUA ziw|GNC<7KTwtn2Zzx}3?=NFf>4nj_>r>C!PdUR-f`WD z0Uqkc@}5{V!YojKYHpwh?bgGR5UnjOXtP#hZBVO8V-?&WgfKJY+ThEQU^`P_^$<{j z@8u#;4JgShp)fh%+-D!2-s(<(9RRNc02tu1)xDzJ{F>h0{+>>>Ygx~&q4B|)>Ao&P zr#?`oxe2PM8tba+2(7qgi1Sv~KsK}z%N*usLu5k z^$fI6!?zp&kos*6HRWXxpRR{|ZaqjCK&feHY#M7Wg+zZ&x>Ce}m zfBx?5K0pX!5+y*Rk)b>qCG`7mJ`99Cf*=*9TX9Tv zc}i(rb3Hf;ZI8#AYin*!fFur0K=DzzWkF$1PDVBW8W^7(f%I}$R~wYBfm2agoRbch z-w9$1dT25!I))T;EgGB#xb+%(At5m#IGoLhkNE!U)0%w(P=M76_q}%>a&l^$8mg+$ zdK{D?)!ZJfs;I1=%*%qLTxz;hDOHvich3#>boF&~bVH?91C$cgRpdiZgdnztFQrQ) zL`Q?*1cqopT_aowj7qo>5JO=`NB{WMX?&mu1QY?OonBGu8YX;)S2e60&f%)^+0jbK z^GxNa^0NU9c;d>^vZ*eto}r^{LfzNa0=?==vXwH>0hl*s#XtiRlj7q-u4*8K{u25_ zaCmfZNL(TdGo>>LGhoHZ%e|UNVuFT9TJ4QE-G@ zRZ+`Gc}oY{=HK4lInV_KSTz-SSQ3Q8!oGxzmmCxl2~JleAruH05)yFj@}=v6!PleW z;u454mrlRC3zI2*s4MVT?s;zD0VTJrrnnTXM9Yl&MoUWya)%2tDhiaM6c%5ap4Zb` z47wq(&9AksrLnG}AQNmMEN=wLh6?G1U;q>!ejT_kI56zz&~w38;EHl=Y|M>l+J#dm z_G+XTvIF`U6CQ|O7#0>&m`|aW5Df3`j^+NeF~GqKn?a#(=5D#)W2# z=^X?}@2MbqFE-5v>) z=V&w0-P_&OTGxR(Yt~oj3(1D+qPj+1OMmmWiLTUDkyNt>tpKKl)C~Xh@Y(a{E)Wkh z5Pp030z%Noj~&E5%-G(eM~7XA-C7XD=dJcL7n`?b}8mxzXR(+t=K?W9Q_K zmKI%4Pj45YX{TCsLilXb9<`C>M#=mY@Ir3|yQ`fBxGg z;^G|2**T7ok@4Px=k^~xbmW;A&YV33< zo;bGf&{4o1J#zf{XHOC*EKi;V+OflXw~fKJL28#ltLYyk2Dl?*-Oal0{uWhbZmCkI zYiz7)t}CgmDXJ-~ZEkDV_jELO6{larBPGV-$o%c!@7R6v*!c?=$n$21sh>WM8112b z$6({(!^i)RIB9YA@_CA1a`51;F$}kZm~!<*w|c;-zq?i6O|+>=qAXWYr>Lziu2OEQ zuc<1kD6VblQg?J{Tic5=Vy0~)C0?S{`6s`CDCV&GY;DJ)hiXoafULRCXaZE=O7 ztxMJ3rfTje%!x7d0T2r$8^sxgFTOc;X^VB_*%@MJk9pFBE#>h#fLXb*?q{Pej4NO=FAiSg~D*r>tY2t|Cu9@-&n|BX_G z*waIHpcS^awRd%ETH9K6?YdSqSzTU8D3vNrg;rUtR+rT&$}5S^hNh-ELaQk#NT8-Q zI^y6MDjj`uYp)*Odkzw*Cr=-TYzwxYcTOUb0JYQ6iScpl1tL&K91OsI=14!vue+zG zsj;zjUM6j625pH#t*lg**Hmd+$wpmmm9DnDcBR3>X+&ml zj^gT(6AxcJeE7tPy%;biG2rglwsi>6USeYQcI@o#(;S#RIn>|VO?1&a0XQ(Y6=yrU zs&(B1U406Yl0wKQR7_G^rX`D2WQj^yQlr!oEv+g|xwgEvKI&&ULc$S;XV-iS1%OA7 zA3hGP9^gTDWdiNJZ*X*c0&+O;U)B3PCx`mFsUf$et+NMBW^ibrO{*9fZtGRn5ej8# zSv9IpqiE2SD(dU2RaKP=wN}yAQctL6mDUgeRP&_FBP5RE$n3Ij&Vh4v^6-%Z6XQE5 zMVoE?T|GVhTgJyB6T593fW-belLMWt%`HT;M{8?)cUN0mdwX+BZQoG8URMgpN@ann zp-!zX*N}yE^<~P2s(PimUW2AosaKWOD81DZjAOv#h@-PNetq$=r=L81_~5SXh);}- zj107(IrI&Wk5d7VZR0zLT{E_KkxfLiXG?1vv^dLEq`pomnmRYOe! zsVZ&O)F~>}s)E`Ys`5C~#@UN~Ba^Vdkm5ECQ|R) z*a&qCZL=CmB?O_ZE+$n~73yj=p(?4*Ew8C1R1iZgQ4uBR6BTu&ri9Sdk_C#U()`+W zhR#O;V!`VZ5&iE=Pd@eZg?$sFFH|H>l^VeN>ZsHG_qn{ zV+&DQS5sG}R#i2s6s0wpHP!WGg}PButg5VTXi!w3VHT40L~Q}7-@I8txytYzC@}Ac z$S==hW%BU8@zL!&z#nOCCEJ#?bZs3SM-;cKqqD0`-`Gg#grmejL<1d>MBYbRBBa?jwonoC~l~)udG!<8AMa5Ev_Xs*_+Cx>T6(2JR>5$IEw|! zBYU=wZW|fwBik0VwzcRxu*@3hB0Fbvz_aqx>04UWm9<(OQQE9k*VZ@b3H0TnMs0ls zu~|u~HC3dtrbt~BmP?5z7RfD=fP1a@PZKR5kgJ{1*+I`n zX~vQ87M;Gc85%=h1|3;}@k6cFs36Ev zpr|bc7+fMkm9MH+*Ow9Mg7pQlRGTz-dnTdFzl5Lq_~Uy+_VPvXZE(Rjakd5Q-^iK*v@=iw5IM z5t@E+9YJKSEB2)vW>S(R4vPO`AH_S}Ka8=XSp#_%b$y*$*V@t5gB40gCw!dN`s&8k zHbreAdQh2K-$;*Hh1j;~Nj9mfN~@7j717z-)UL%e+T22_nsv46 zMiR?3Wqm_KAsQqMBXw$}PFto>G}n`jwd*$)yQw8u-@xx$9{2d6N$@|9ZSBJ7t|#jo z2r>R28XQSMR zCNuac+D1iX1=&=tZ_3XoT|t>eE!)t+Hi&~C-`NMp=ja5$S}N=6>uV}A6?v7FaIov@ zo0`?tRfSbWSryqe8XZA4wzt>oYFgU#gTr0=jt;$Eqbfvwky;J9SX*ZYDw@#f$@01i zQd8ATSLikoL~(h3 zULATZ(blGJP^#OS8hZ!YdpkR{?fATwtZVIVt?zE`=|acv(2>|Ot8E|^>elKqm7%qB zKrWE16DK|NP$#0cqxyP;od{)hW?}JK1xeIsDx2Ed8`_k$WVMRSSL<}D)|Q^u2EC%G ziD<`-2t$L&M~k|(y`i~h0IO;+X!^Q)TeKRzrd-hlE`+*Vp};ngkuetXSZk~k`&ED5 zII(+TsHK)vG{SW+FUu{iQB|qQlIHHlu5N8xi>jH>clILcG~9`F8oOHBdb&ydmZ45` zawW_d8yi&jc5U6#)79Of(=|7#$nI?;9o>p5Ej0=nwkEI;m5Engp45Y!)z_*ao3vGm zhN_J1l5tY#QFRqpPEL z7~}=WDCzXgZ5>0CdnX2)YZ}r-ltmyHU<+imm+GSu2sw@SYx;ZiWRt#vXsE7L>FT;V z+XlA|40pDTKofY^_;7zufA7dhTVFeN0}v1=6wom0?H`{&9J_ZIa)LeG_(oI90Oait z>>uxIYFI;+8)tgoC`}yr>_f%f!#hD|+cJoySI0nGE%cn!-P&GADefMa*tuorfqf%` zEkoP3jP-VdS28l()j!nR(X?~__`b=5$Ik8_9-SN?>w<82_aMj{hB)_8>x2-e3$;w-u~{!F`OwmdGy@beLY*YU>g8>hdU?WqV9mi6L#OX^&A6%7pjWm z)JuEz?%p-NmS+*vJDuV+dBp( zjv|6`?m*YbuAPT=?A^OeP&QhW z$^gSoJ++alM))Q+^2K928|xKv!)`mM1(HRwm8;**BbvLh z92^=WCm16j#Y_xMfReNuyzku<-*)ezL*PSz@3@p&mw-XB3*w8rC>5;zM~)IlOio^Y z;!$i;ogGy*6dFPd9PD7D=p|zJl{L?2lr;ABkM08x^e_l2`}e|61*u>HDel=%kt+6s z2D2Y4@jZKw5c_6BQS0z=;;hB_vzIQNJbL)-6VG7t1d-V$g(^*~lVH;wMJ^P3MAu#} z&#K$9hgv}#oPgOl!mw*UxFKLc96k(MD>$_W4`N>(G}NQE4`B*_@-ggqJ@?#8uOi0s z(yPxs^W=py2e;s|l?WtjWDITrirQW58nkBoO!CGilEh=U&nw*V)Cz_11n;PmM; z58+J1#V22V{S6$;eD$w5O7!XHpOK#%{fqbt|NV2}htEE8NT0iIb&9(oRd5S9T8XP~ zVBY!lC3V`iuHG%XkDNG6@x%7-!nOm*kQXn3u=~hkFN5g!7S2C?@WF?l;}jzKEpgQZ z2k>wd@7aU>*@;o!NDqN65DU<27Kx=EX(!0avIc!y|2VOO3n|J&2Vn7;bLY>V2gUf2 zOOL+zCP<2Jzw;gfLtlLJ?bYwEes`7l-sl_Ri}{~^cxI?J%3n(5kOO7`>Pft&Giz<3 za%d8J$5;=7tc6AoX2Ahu^~BNBXCM31v(LVWK;N5xd;gzbUBOY=Mc;h$=|8Zwb$XyA zNJ>Q!=^z#$U}r26O5EiMrP^lfY!R9{q_SR-mrktVq@`_a=-j#;)S#V*K<)U`OMm(M z$J9x^&p&+Wq2osm3~vsSi1nxk5DVz7s0X1)BJ&H5T%NpkV@}T6aHwJ7{C8kPPES*J zZ*yhKmf_Kn<4?Yf13OReDlaN3%!&6DQ{Sh9SYXWQlJFsih7>$J-F*UrLL;K0;|eOf zy1Kh-OB)EHsBdKZ#WQV$afIf`qS@o8KKYY1wa=5L%V|3@}_~~7s?q!Ku@CmRA==4E}g<$yI zN3MRhcXwZF+txjkqlZs#?cLhe3@|!a1@vx-rOPuvyz`&G?wf2>H;?Ze-wPtv;7~9M z6QNzO3XDm_d7C5_ycd5ucIx6pn?}_+w)f1l=Rm?tMT8Q`fK@=JV}3K||MjP1Cw8~P ziP(AsI=>%2eJV$&ral2y0p*>0d^`mzTN|nGKk)h|Uw!@M8x=wfkj7vYNS2CnuY5nW zV_4rcIQiy>Uw!qjuawk#i@++tH_cr4O?}cm3BttbzkTo}Zh5H{km8vl!2<9G5aKZt zWqvSqX<(vz=jGQw`q#g|{3zR5&7a{yy^CEiQ{Ty{FZ;mzdid3kzWCQaUx}sO$4M@t)W43diSf0!pwZ|ITf%OJt;vT*7El$q z_+&ksX0g=V+`_^V@7S1vQ!sDN(xrkjA8Rws(m7^k=H|vTKQ@}TaH)lbxrL=AVL_PB zG+j9BGmr?>3+EY|np;{dT{sJzf;pP`Gw05p1x~?iPz&yoQ$RCfb%hfS07oEt6MX?BIZKB0OzLTTGcSXwD!SnBY{q+x^ z5PbE;r|9Fg`wOpb)THMf3Swe(>fkZ+`!iddlWMSHAk>@nd^HjvQ1j z6S_FEZ5a%@rMaoKQsKkbii~I!fq>O4nkV2ndX_)?IRg0HD<8i4%+ouv;zQhI0+Gm> z#pb!ViUbgkV6!<65NV*(_@wuSiNwOHW@V zCuEBx`4@HxUE^Y6mPds75xxs#jyATobb3&<5bzXxfTb7B;d6cS&LJ{B|GNuS(ZOMf ztJe{0iFD)DX(>K3*UYb`rjjDa6*Hnj2tQ+Ax!m0mLJ?MuJ_4k`0(F3<6IlxRQHgDj zK~vz$Q$uZOk;&`UuUoek1M1?mwAHKA(ozFOXQrmEX2g;!W{3I8eW{%I_(<)oZ5TFu z9#wi2cYvi4&F1qYxtGW(lPk}U1bIiT$;iyuke;?mlRP6Sg-D&7k`k+W<3qK2`MhvH zgNU#XAz$F*#X~xDHkZl}MH`@LL^Qr!|K_JSiT?bd47poG+J=nGwX4%srD&3gBqDKf z!pfDgF<}uAkr5CN^z|m)m$=E?Jbiq;z1{4rp$W*MDu)Hu0EgMU~?ZstS(JZZ9 zs00nX0g;7sME4t0R}ZE53j`j^Qdh4^S+_1Vg-o59lAKH;ck__D__){@q&OQX2Ky5} zGd$hYvL&vrQmL!Emxl}5kfjw*s6mpT4bVuDH9rBgfgfK^bL0!Xq7qY6l9F@Rrlczp zlT%WX)QPha5?00&amKMLmPNtj;J^TXU(#o`r<+tPAzcZnu|y(s7hBWJ&FM~5sx;n z13(IlcXcP-z&{hI1r``6L?R)d%i%ck#R8g{$)Y7}p&F$D{}yTU1j*n1Gl|J|C#74F2hqzwk+?kh#=&%ELw3$y*_v{JEdgVSkSUq{{S*jP8GbTmwx~d zLPJABf`fu0Bg40>bqCwiRV?Ij9XV_VDDPNX#cwawiB=(jG1X5R-S3poiY;;saNN{LW(u%N@_1@l|ZtzZp z0v?w`vgfd9)^rAwur^^@GtmN=5E+6&RH;$q0N4{F{rpV_2YEoCUqFCg=E>I5Z0&`% zWV!Fw93M!k`g+SG5}})ix1Ud7=yGZ{3Xh6ST^kKhGu7P&$QyQ*;XbdOd*PnW*`TM{P?P=Vr!Xipr72|Kj|NXvGU-Plg+i8blXx> zeZ76$+$2(=1dX?XBy(cp1&jGE1P}K%@to~w)@aT4bVyVIie_cOu(h+cv|(|qtxZkL z`D&4Y8z3?k`2VD#d&>RfUcUZ>A6xEudc;re6L#7|6eXANL{1p*JVQ=w3=9j2SRUCn zRkS*lh?<)m8{vUAO*$EKoj5caRsBqwEn&Ts$ymx<3S}=VD=5a=GtCz*MHd1RO)c_q zDF4aN$;)5v>Er9~>*M9#yT!}LEBL6tAR$D=6LEQ5v1|U0z#z2jh^VOK(v8U&kCw+T zix4>x=sZqbM|&%Vje~=|4cMiS(PDr{O0&WbYg;=zdmD>|rd+C3fEOTg7MdM>k1oUI zU!Fd4AGBNDu$z~MbRt5Q86$KSLReNL)^xb~1QEezA>omcF)>lm%VOeUqkXtsM@NEd z0p(j4sl>zG0On4}%%lgewP<&fZ=g zLarkh$wU7OwZXJQ3v`0I8&res=vFr9SLT*hjHOl#yvcPC1$hMr@fVmFzz~l?AueW` zRDa_umC4-mJG?NU5we-C5(z0Z76{#WB7v}=5_S>bWXlMtk$`ZV!C{dORQIuAIB)<4 zxiS={t+hFYWx=#$Vku!~$6`4NVGx{p*h_JW+-o z9&WA@LToIOikX%9}E{SAPQo^h( z0)<=b>KEuL5#xd&3`Z=g>De)?9Dv3`D{`KVyqJPPxu2MogTO{BM#iVG7AdlRN0wa9_Z zbzr%Df51Tma<COWZ~ zhjcL+3Hf{;1pOc-jZ2Wk61)hH1Ym@N8w${x0)tz{b!2fkE;26>!^&c*6~+uJdoIsW zDB`o6?CjhFeZ3t8LZOYxqMzaH1t!LCR5?nqsUdZB?- zo1!@as?Af4gKlF_*qJ(TxXv6NUo4e-NSIbwdy=R|Xb7@JVkeHHJx3nsBXV>R2pEga zMTV&bp=x7eb&FE_UWjcQ1SB{Mm5% zgpe;_3uS&0UQ+n_7UrPjTX6(J370DtLBP*Jaaj5b)WQVgZD8XGrTxJPV90UD=E@UqZA4&4e*c;+tbt4(-p2R{BmjtAw_JPg^NB9^O8dz z9sVf}NMJN)+fw4-aDn0NUE4u%w$oaB+Cb#o4r9Bui_|Y5IFQY-GPgoYv^1w#I*L%N z0)ZXZ#noNriT>>CF7j~oaN`Tvg6mNBObQAUnX6EMgBaAr!M4FT3dMJ9(>QUtxbcO@ z8Neo`~~@JH+g`+x0g)pCi8GdpFsh#rp=no zn7i=vFhV}dRp7!UoR>IqICeI+KVxBToP`p#)4?cPj2qO54$(;mJ0`=HWzDdnThL4` zE!jdBF=iznLDmNKVMmz- zbI@rWISzINleG4>V{sig_O=Ww8>X$51?FQb+||IeT1vz2jul@dbn}!zrP2*(@_+zu z46bfc54oFECjAwf_1t-%1`)pV-T6*P--*l7+AVTGUj~vR8>6kYhPl|@5$fxXPR=YS zp*vVv6LieYY!_?VQj5iwmX1=fo2N|T?n;%Pn_pm{hpVf*yQ{a{6~lyD1YZC~nb`rx zPwwmMfdvMWf_d~c+zFshhi%G){NHcA$?Fo(% zTWvqX&W2$6G8i@-HasM*vl9j&Ay4cfwx#1LQOx077miTk>LzuUQ8rO!hl>OPz0gm6 z+~9+Y4XbX8s` zmQr_DcZ>s`0RcgN*t+n>YS>4D-IXr5HVYQbdk<5-r?-GZ*c;i9%y|sBTNDhzvT$&q zcDrn`b;5Ck4~7nHFAxzzA+CdGIdYlyl7OH<4<1Z$MWNw`Odb&u6cZ5`=p7Um=Hu>z zeJX6$qC-(gCiGc<4sivP*sw$*nQ>M?qSj@w$Vuet>TCl~&lXM@E{o@IoJI2R&;Xf; z#}=cDddk632@Uc0kYG8FItq#k4UCLi9uO859kx8&+t(lWS;3DtGN$}%QZ%1CYu^SG zyhNn7Pq1Y$t?0M`!JGyaSC$KAO{o|IskI{u%P1!yk0%brQYb9K$H(8#+c#M5;q4n1 z6T3Vzz{4F|Jt2|d!J%>S;menWg-6ClM@RX2c*-;)tWAKkK`?hLi$etv*zo(YP@>zS zd2!e@hBZqd;Uiqavw|2b3+=-N>v&l-vh3|25fL4;3_FC8o^rWYaCm57NU*oaJ1k~p z{L0k$<*~8hp^-7Mk;`KIJmnbp3@a`;Y7EoS90nGHLMK4knA>1o4v*Q+(wvS#PsYd1 zRTgwRCj3h$Z~wrMWh<89q7UDo=(G(fF}NfoOzwmF4-5&7h>uTPi;Er8*Ael?5n;=4 z0Yg;WijXKz?BGyQebGwC=r?INO(%4*v$wUhM~~vzVX9>~@Nq=}Hhr9&Y0f-cvdeb& z3x*NPgM0&FLwGWF|R=5=TTYShhSiCN3oVHOI+dbiUOlTB9=vjhWG{ehsGwa&Dos0GBPSAkw|k{lNuhKmA`S# zs??R~>vBP;%_u0YD$U-Ixpr0B>a^qxg}$+&G$S#=gK{OP-2`KeXpPyH4VD~dFHe~Z z97>)uyjA$Mu5L1)*wj_Y32`XrwHdja@;4XcCC4YM%goH$keQK@zI^%Gy!;L8H|MP{ z(rNQcs})r_*l}IAcI}$%>ejx&zK({BmEP1I7!}ZmJ*%C)Ut7qK_yq>|#)e~~NG@>2 zeI~9Re(@{Q(sS~&S0v@+ZY-*;&d*9oSznN!nX!JudiB!NDoo zn4OiCnVC}!%FyWWmd=JX&J+rZ+T_uQRx$UqEp+vZN=;mqm9uGM&W80d7!T!1iBvh& z)n*qI3XuXx~a1b7iB>iVaHJ2#ta-+G@?=Q zd?4^l_v$@jQ<7FDC8uU&g%%3Qe$Hv2k#N^D} zO?jJ9u7QnOA}>eVKMXB~mfRA3XCDYsNV^?yl(I@!YfEcW_x|oiT*W-JeSF*4@RqGx zw;erw;?Ut8s{9-;YOXP+b~`nqxi<8Ro4vy0i6mA^?xvFBVnVA0@kiIFEYpl_A84p= z9qjM#15>GIaA<@W=0H-VuerIU8DuX_W6uzz4Gi)k5ATZ&Z6>8?bB*R zd;!Dq6>q=LxWwf2lG5txnmVEpcW`MMb$Wg8)}FpCTPfiTgidi{iyUE(jg9p5_Vsik zF51~QvTgegD*U_eJdVj+zIb*^X+iq*97zT9HKJHPJ?N6mXIbLf+7@K4w;N>*ZcKY; zFLX!17KQ=~4r<{a$SmQS%E@t@+1j}uXWXD|b@t-feK-jI=rd0}`Pk!^k1F$v;`keERQ1J4)4 zn{AL!0tMs9u~SedeE4YyD?{w`Z*RW){wH5u`5J=4-;h_0z9YU__}O!hA5p~kyZXrt zi6W5C(&C;LQ@=g#E3-Bgm6VpZZh>^l_A!*{_Q}16j-8~a0gpWT#A|QA^X_{ee1!9q z|NQ#rzs43W~rE13P69T4LKF z6aj_F(@;Zr`0}GKz5S1OA)fjP^r){)-$O$FX!HZ|&AcmLUEGpm!x1*3>h~DwMe~HK zT37h5%xPBl&&8RA{=u!o#3+4i9EuwUPzet|{p<^`y!JN`3_iQ^-Sq1CGY~IlCW&hTQwSO$Jj+& zefq{*@BIrx$zOl-@3)@5eDTysqQHfU0U~@)YZaMNi&c>;HgT}K7w9jUGnbfWKX=X? zGmcD(-4?gim9_P{{ZG6B9n2TUS0$$=MYuT$@O2s$y=V1cdkb+6v5;+JWns2x{=E4n zrgS>|G3TJjh~RvM`t;dd^+i_J_Ky5!sxYb0J!`ATn%Y$5EHX3SKaS`}z_RG54K-E6 zyBZst%D947B#y{EZA4^Aw+O%TsxTbtkgE zcv9@|6&9VTCKRIwI@-Fm6ci1Cd&E91o&WUBk?*vA0Ww)grLwO3;Lf3*Zhuq-;-`kd zJ%XuCh(^8eACV}`&qFF$S)|+fr!)KaY>h`G4ap#IPop7lVD9|Fj}lp+hfEx_vFG{s zKl|c^bD3~>@CgL&;fXBeKL3-O+|NT2klFILe?n#aoeGS!RN!7MGIMzUYd3$nTSAuV z@1I@0`tA2h>Mbg6k1v`DlBZr31^Q%c-u~97-(LOhyBeNaI31V=>{8n&ru6OsVGYOL z{OsHBzWXE#3vdL(dGif%dy%P2_|&_eLE(ibUj6**Z@+mxhO6Nj8NZG8flUwE(^D<} zY2=AlJ_Q}%h}c=po8e5oM~8Fk6fyPr^4i@{$9ntSH?;WH4CjRi%-fjE z<=5VL^Y!zwj(E!nG5Z;F*aXLfnK$)O+sS8Me)Y8%+htC8i-VB85sl!OI!NCCp|AI$ z=U;m6@G1um)-42YhQlnpIhWupVhR5;by9oa?8Rd>&McBM+mXw&rb6~igma}wrY>|J zKD<-!=b+;lEutd!@K0T`f4b1Qd$MN@hppwzU>PFz)=W&Osf9R1OVcF&)W{q8UWX8Z18&Hu`b z&*XQM0jF1f^__eA>xT|^kb&U;ez5(+%>HG;A7&u)znf_O5PJFpdeg+ixr{(>U9c)ei&Xa$?;io<^{~h?`U-Eh1e#^c+z8mzj?)!7;w)hKQzwh7x z_~y3$(C~wRpPDMTO#%4+p<91%@^|po__&hu!8{+|dS#OJV9%|9$|LiUBqaPgn)cdzs4(xb<_}|^UD4qN91H_*a-c$i`4-$SV{3iUd z2Z|qiBmRuPJW%}AH{x5}axdw=Xr#-+=yR(%a7f+7jLrv$bEylWx_Yr z|2js1V#m-iL!ro?48*l%%v*E~1ke}Cf=!0x`UI)0Od zk8Yg*+<)`-egEU78;aj3=)Ob#2H^0G_#wY}`@a8CawER~eTV!FfcK5~qTjrIzyJ8* zrUjVI18x6*y3u^2rFR=x_@9ah^qU)=x%Y9$>E^6P-GGiiALRJ|@!Xs7AK>C+>d?*b zMmYNWz@AmNz;AtE=u;mpz6E~5gTu!u%$py{gTwc_8Qtgs4ggQzIQLFx=YnE7Bfzuvs<_J836DS+FVZ)CLawFi&^C|=z5NxJ{| z|M9;qx{=Y$L-#!cU|j#sWB0{<&cHi2`CY%wP5S-CzxsQnZ^+4>`9G+zfZ~D=GI0eX4KP&0x2BrR2V3;K%wqgf}u$-iM{gztV4I z>TO;tyXR|>S46jgy7kL(_jVlmq4RFZaQ3I~VIz3O_tsYa-mA0H?$to>&#JpMemYv4O7fBTNv?Wd=YeW7c>U@(96P=7EijqhO&s&8xl z$NmQEfoRRN#xtD_{DgkbJ*x+cf8P%OeIF}BsZ;pY);4&ib705`{Wk0F-j>;hY~5HGT`fR>6YeynIjN#9jK!{hWbZE{0wfW0Yes; zR(IM5mN>b%Q09A4=+JnAd>K^(A(6{A-EtECD9IJNgAxG;0wlInr-Pc%wf0Bl!;l+m zh$-Az3!K5=O~VU=LqbC#Wk>1G1qOu1uS!pU>sIDJ8ygk^sT?;~u?sR_XJf-b=AEJW zbN#WNu7W$nX9ArvU*Hb-@R+2uHEY(aUX`9(SXMQ03-~KTY3Xa$tb#COLTq%fn8Sh+ z38i);a6(R|pXmayX53Nvn^D#qieDZQy=p@i&Om449QNiCa&Y+OzW3F!&1=`;S&J;D zrX(lE#RWm~g3{YT2B;GFRa2nfN&B1Scpcv*I4vtDb4}6;O15xyQF+tWqc@oU(?|PC zaF{zMCp$B9!@6`z#dFoF1nR_kkiU;i=)wm0G(Os+we=tDeoXht#RlsQ_OHs#PK^qq z^pewYy9S}|{cVuq%h!+eRc+iz!DnS`*svaYqHEW!Pv2BozByxcQo^zTnbUN=Q&~Vp zn5KW&>+yl<+NUZ&xHdOBl!6bBYU=D88W}$L$`N*pQvQaZ^6-iY_TB zDJsm*&CXbzS6y9IQJj~V6zb2vj&B%)8Fz%wzGgo%keUa4z=ucFcJ^)^8Gmtl;{EZ< zcV2n?(uH%U4onUYsFccv`Z`4o?ye{+F38K-kd~O3kbpM3I>N_K@H0NuDH%7Do^CFX zncSc8d69WJ@rDensO;#5O!cw1zy8;!|M%BFKl{v6Pdsw|$d0kmQ3}7luC}_Oq;PZ2 zhV-O_)Rh4~p6*hQ=!95*U*5I3hx)-}THT3fHj^{m{HR<=5+M(;62gY@xefZx!C~Ak zbKvmtGq`W;+?i7+cjC1Aa9eF{O;trjX>q}(oUDv>=!j{_D`S^I4>Kk%(g*F20;cAF z{Gi{O^>{-W*ztbOfU{HNrh9o-Px$8+@fsfpGmvNA4gT~1QO zvY3^t((s`DuU<{zM}&t2$1Rgk@Yg*NhzQ+D_cJmF{&Wr?g5)So*jQLtkeij0voR+( zH#<9bQ-Qi~d~)aH_QsmZlA_Ge(72rJ^cA7hBx#tV;}QbBpfD-*T^{ResC~@G1{aJ$ zzq5t#9Gsw?exQ8;X(ac!^_vTkhy2Z(Hy0EYl$7P?2O|l}RyP?lO;n$Q99E(~ZyIhC=L$ZhA-NBNF5cQ6{KPL8l4$Tt|UlSlpV8fGjMc zvQbh~-%zi>6=*f(S&+bm2yXo9Olk;Nza}khO{bVtRI7enDXY^%NBsmlPM|=H?m(AB;Yk+0+9mi?wUgQW91~ z1h_j-!$bOzf~V6K-MRBK!;ZSc8`e|iQzC~PN1j;bDfbT!k4ap$HghAC91PEEg$o-osz0>@*(M+T4uD1bqm%~>pS^YHS8 z41M&9gcPdDW58dPnhFKsq{PJd=s+pteIPMJjrMq;H)CcpANt+bQ@+EWGt4Lr$zRZ+ zYy}B27Rv$u;Er<_q0}7;n2i6?-BJEaFmiN8TT(eO~BuGc!A)!o$kjYTuh+@6xvG88`GDO{v~ z9sl}I1LUv&fXE_M@M!i{R+e`(;N1xP%ir_N7|>MW-~s%!b=RS8^%I{kpd02I;G5lf zlm7+trr&1Fnmu>kym`hn2)5bZTmZkyh=cufyQ4-tsNUT+(03EWoxLz)-eOCpLCfu0 zQBaw|e};c44M>B(X()RpgMJtN{m$U0KQ&|K%$c)h&zZm2+zMHle&BOckHOV6z+0Iw ixZ56gH+i`B*;(_T002n=`t^qzX=pTaQ{%f>>Ha^kqWY8o literal 0 HcmV?d00001 diff --git a/YACReader/YACReader.pri b/YACReader/YACReader.pri new file mode 100644 index 00000000..4d7da37b --- /dev/null +++ b/YACReader/YACReader.pri @@ -0,0 +1,144 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD +INCLUDEPATH += . +INCLUDEPATH += $$PWD/../common \ + $$PWD/../custom_widgets + +win32 { +LIBS += -L$$PWD/../dependencies/poppler/lib -loleaut32 -lole32 + +isEqual(QT_MAJOR_VERSION, 5) { +LIBS += -lpoppler-qt5 +INCLUDEPATH += ../dependencies/poppler/include/qt5 +} +else { +LIBS += -lpoppler-qt4 +INCLUDEPATH += ../dependencies/poppler/include/qt4 +} + +QMAKE_CXXFLAGS_RELEASE += /MP /Ob2 /Oi /Ot /GT /GL +QMAKE_LFLAGS_RELEASE += /LTCG +CONFIG -= embed_manifest_exe +} + +unix:!macx{ + +isEqual(QT_MAJOR_VERSION, 5) { +INCLUDEPATH += /usr/include/poppler/qt5 +LIBS += -L/usr/lib -lpoppler-qt5 +} +else { +INCLUDEPATH += /usr/include/poppler/qt4 +LIBS += -L/usr/lib -lpoppler-qt4 + +} +LIBS += -lGLU +} + +macx{ +#INCLUDEPATH += "/Volumes/Mac OS X Lion/usr/X11/include" +#isEqual(QT_MAJOR_VERSION, 5) { +#INCLUDEPATH += /usr/local/include/poppler/qt5 +#LIBS += -L/usr/local/lib -lpoppler-qt5 +#} +#else { +#INCLUDEPATH += /usr/local/include/poppler/qt4 +#LIBS += -L/usr/local/lib -lpoppler-qt4 +#} + +LIBS += -framework Foundation -framework ApplicationServices + +OBJECTIVE_SOURCES += $$PWD/../common/pdf_comic.mm +HEADERS += $$PWD/../common/pdf_comic.h +CONFIG += objective_c +} + +QT += network opengl +#CONFIG += release +CONFIG -= flat + +isEqual(QT_MAJOR_VERSION, 5) { + QT += multimedia +} else { + QT += phonon +} + +# Input +HEADERS += $$PWD/../common/comic.h \ + $$PWD/configuration.h \ + $$PWD/goto_dialog.h \ + $$PWD/magnifying_glass.h \ + $$PWD/main_window_viewer.h \ + $$PWD/viewer.h \ + $$PWD/goto_flow.h \ + $$PWD/options_dialog.h \ + $$PWD/../common/bookmarks.h \ + $$PWD/bookmarks_dialog.h \ + $$PWD/render.h \ + $$PWD/shortcuts_dialog.h \ + $$PWD/translator.h \ + $$PWD/goto_flow_gl.h \ + $$PWD/goto_flow_widget.h \ + $$PWD/page_label_widget.h \ + $$PWD/goto_flow_toolbar.h \ + $$PWD/goto_flow_decorationbar.h \ + $$PWD/width_slider.h \ + $$PWD/notifications_label_widget.h \ + $$PWD/../common/pictureflow.h \ + $$PWD/../common/custom_widgets.h \ + $$PWD/../common/check_new_version.h \ + $$PWD/../common/qnaturalsorting.h \ + $$PWD/../common/yacreader_flow_gl.h \ + $$PWD/../common/yacreader_global.h \ + $$PWD/../common/onstart_flow_selection_dialog.h \ + $$PWD/../common/comic_db.h \ + $$PWD/../common/folder.h \ + $$PWD/../common/library_item.h \ + $$PWD/yacreader_local_client.h \ + $$PWD/../common/http_worker.h \ + $$PWD/../common/exit_check.h \ + +SOURCES += $$PWD/../common/comic.cpp \ + $$PWD/configuration.cpp \ + $$PWD/goto_dialog.cpp \ + $$PWD/magnifying_glass.cpp \ + $$PWD/main_window_viewer.cpp \ + $$PWD/viewer.cpp \ + $$PWD/goto_flow.cpp \ + $$PWD/options_dialog.cpp \ + $$PWD/../common/bookmarks.cpp \ + $$PWD/bookmarks_dialog.cpp \ + $$PWD/render.cpp \ + $$PWD/shortcuts_dialog.cpp \ + $$PWD/translator.cpp \ + $$PWD/goto_flow_gl.cpp \ + $$PWD/goto_flow_widget.cpp \ + $$PWD/page_label_widget.cpp \ + $$PWD/goto_flow_toolbar.cpp \ + $$PWD/goto_flow_decorationbar.cpp \ + $$PWD/width_slider.cpp \ + $$PWD/notifications_label_widget.cpp \ + $$PWD/../common/pictureflow.cpp \ + $$PWD/../common/custom_widgets.cpp \ + $$PWD/../common/check_new_version.cpp \ + $$PWD/../common/qnaturalsorting.cpp \ + $$PWD/../common/yacreader_flow_gl.cpp \ + $$PWD/../common/onstart_flow_selection_dialog.cpp \ + $$PWD/../common/comic_db.cpp \ + $$PWD/../common/folder.cpp \ + $$PWD/../common/library_item.cpp \ + $$PWD/yacreader_local_client.cpp \ + $$PWD/../common/http_worker.cpp \ + $$PWD/../common/yacreader_global.cpp \ + $$PWD/../common/exit_check.cpp \ + +include($$PWD/../custom_widgets/custom_widgets_yacreader.pri) +include($$PWD/../compressed_archive/wrapper.pri) +include($$PWD/../shortcuts_management/shortcuts_management.pri) + +RESOURCES += $$PWD/yacreader_images.qrc \ + $$PWD/yacreader_files.qrc + +win32:RESOURCES += $$PWD/yacreader_images_win.qrc +unix:!macx:RESOURCES += $$PWD/yacreader_images_win.qrc +macx:RESOURCES += $$PWD/yacreader_images_osx.qrc diff --git a/YACReader/YACReader.pro b/YACReader/YACReader.pro new file mode 100644 index 00000000..e8746938 --- /dev/null +++ b/YACReader/YACReader.pro @@ -0,0 +1,98 @@ +# ##################################################################### +# Automatically generated by qmake (2.01a) mié 8. oct 20:54:05 2008 +# ##################################################################### +TEMPLATE = app +TARGET = YACReader +DEPENDPATH += . \ + release + +DEFINES += NOMINMAX YACREADER + + unix:!macx{ +QMAKE_CXXFLAGS += -std=c++11 +} + +isEqual(QT_MAJOR_VERSION, 5) { + Release:DESTDIR = ../release5 + Debug:DESTDIR = ../debug5 + +} else { + Release:DESTDIR = ../release + Debug:DESTDIR = ../debug +} + +SOURCES += main.cpp +include(YACReader.pri) +include(../QsLog/QsLog.pri) + +RC_FILE = icon.rc + +macx { + ICON = YACReader.icns +} + +TRANSLATIONS = yacreader_es.ts \ + yacreader_fr.ts \ + yacreader_ru.ts \ + yacreader_pt.ts \ + yacreader_nl.ts \ + yacreader_tr.ts \ + yacreader_source.ts + + +win32 { +!exists (../compressed_archive/lib7zip) { + error(You\'ll need 7zip source code to compile YACReader. \ + Please check the compressed_archive folder for further instructions.) +} +} + +unix { +exists (../compressed_archive/libp7zip) { + message(Found p7zip source code...) + system(patch -d ../compressed_archive -N -p0 -i libp7zip.patch) +} else { + error(You\'ll need 7zip source code to compile YACReader. \ + Please check the compressed_archive folder for further instructions.) +} +} + +unix:!macx { +#set install prefix if it's empty +isEmpty(PREFIX) { + PREFIX = /usr +} + +BINDIR = $$PREFIX/bin +LIBDIR = $$PREFIX/lib +DATADIR = $$PREFIX/share + +DEFINES += "LIBDIR=\\\"$$LIBDIR\\\"" "DATADIR=\\\"$$DATADIR\\\"" + + +#MAKE INSTALL + +INSTALLS += bin docs icon desktop translation + +bin.path = $$BINDIR +isEmpty(DESTDIR) { + bin.files = YACReader +} else { + bin.files = $$DESTDIR/YACReader +} + +docs.path = $$DATADIR/doc/YACReader +docs.files = ../*.txt + +icon.path = $$DATADIR/YACReader +icon.files = ../images/icon.png + +desktop.path = $$DATADIR/applications +desktop.extra = desktop-file-edit --set-icon=$$DATADIR/YACReader/icon.png $$PWD/../YACReader.desktop +desktop.files = ../YACReader.desktop + +#TODO: icons should be located at /usr/share/icons and have the same basename as their application + +translation.path = $$DATADIR/YACReader/languages +translation.files = ../release/languages/yacreader_* +} diff --git a/YACReader/bookmarks_dialog.cpp b/YACReader/bookmarks_dialog.cpp new file mode 100644 index 00000000..07df7222 --- /dev/null +++ b/YACReader/bookmarks_dialog.cpp @@ -0,0 +1,197 @@ +#include "bookmarks_dialog.h" + +#include +#include +#include +#include +#include +#include + +#include "bookmarks.h" + +BookmarksDialog::BookmarksDialog(QWidget * parent) + :QDialog(parent) +{ + setModal(true); + + //animation = new QPropertyAnimation(this,"windowOpacity"); + //animation->setDuration(150); + + QHBoxLayout * layout = new QHBoxLayout(); + + //bookmarks + QGridLayout * bookmarksL = new QGridLayout(); + + pages.push_back(new QLabel(tr("Lastest Page"))); + for(int i=0;i<3;i++) + pages.push_back(new QLabel("-")); + + QString labelsStyle = "QLabel {color:white;}"; + + foreach(QLabel * label,pages) + { + label->setStyleSheet(labelsStyle); + } + + int heightDesktopResolution = QApplication::desktop()->screenGeometry().height(); + int height,width; + height = heightDesktopResolution*0.50; + width = height*0.65; + + coverSize = QSize(width,height); + + for(int i=0;i<4;i++) + { + QLabel * l = new QLabel(); + l->setFixedSize(coverSize); + l->setScaledContents(false); + //l->setPixmap(QPixmap(":/images/notCover.png")); + l->installEventFilter(this); + images.push_back(l); + } + + for(int i=0;i<3;i++) + bookmarksL->addWidget(pages.at(i+1),0,i,Qt::AlignCenter); + + for(int i=0;i<3;i++) + bookmarksL->addWidget(images.at(i+1),1,i,Qt::AlignCenter); + + + //last page + QGridLayout * lp = new QGridLayout(); + lp->addWidget(pages.at(0),0,0,Qt::AlignCenter); + lp->addWidget(images.at(0),1,0,Qt::AlignCenter); + + layout->addLayout(bookmarksL); + QFrame *f = new QFrame( this ); + f->setFrameStyle( QFrame::VLine | QFrame::Sunken ); + layout->addWidget(f); + layout->addLayout(lp); + + QHBoxLayout * buttons = new QHBoxLayout(); + + cancel = new QPushButton(tr("Close")); + cancel->setFlat(true); + connect(cancel,SIGNAL(clicked()),this,SLOT(hide())); + buttons->addStretch(); + buttons->addWidget(cancel); + + cancel->setStyleSheet("QPushButton {border: 1px solid #242424; background: #2e2e2e; color:white; padding: 5px 26px 5px 26px; font-size:12px;font-family:Arial; font-weight:bold;}"); + + QVBoxLayout * l = new QVBoxLayout(); + + l->addWidget(new QLabel(""+tr("Click on any image to go to the bookmark")+""),0,Qt::AlignCenter); + l->addLayout(layout); +#ifdef Q_OS_MAC + l->addLayout(buttons); +#endif + + QPalette Pal(palette()); + // set black background + Pal.setColor(QPalette::Background, QColor("#454545")); + this->setAutoFillBackground(true); + this->setPalette(Pal); + + setLayout(l); +} + +void BookmarksDialog::setBookmarks(const Bookmarks & bm) +{ + lastPage = bm.getLastPage(); + if (lastPage > 0) + { + QPixmap p = QPixmap::fromImage(bm.getLastPagePixmap()); + if(p.isNull()) + { + images.at(0)->setAlignment(Qt::AlignHCenter|Qt::AlignVCenter); + images.at(0)->setText(tr("Loading...")); + } + else + { + images.at(0)->setAlignment(Qt::AlignHCenter|Qt::AlignBottom); + images.at(0)->setPixmap(p.scaled(coverSize,Qt::KeepAspectRatio,Qt::SmoothTransformation)); + } + } + else + { + images.at(0)->setAlignment(Qt::AlignHCenter|Qt::AlignVCenter); + images.at(0)->setPixmap(QPixmap(":/images/notCover.png").scaled(coverSize,Qt::KeepAspectRatio,Qt::SmoothTransformation)); + + } + + QList l = bm.getBookmarkPages(); + int s = l.count(); + for(int i=0;isetText(QString::number(l.at(i)+1)); + QPixmap p = QPixmap::fromImage(bm.getBookmarkPixmap(l.at(i))); + if(p.isNull()) + { + images.at(i+1)->setAlignment(Qt::AlignHCenter|Qt::AlignVCenter); + images.at(i+1)->setText(tr("Loading...")); + } + else + { + images.at(i+1)->setAlignment(Qt::AlignHCenter|Qt::AlignBottom); + images.at(i+1)->setPixmap(p.scaled(coverSize,Qt::KeepAspectRatio,Qt::SmoothTransformation)); + } + } + for(int i=s;i<3;i++) + { + pages.at(i+1)->setText("-"); + images.at(i+1)->setAlignment(Qt::AlignHCenter|Qt::AlignVCenter); + images.at(i+1)->setPixmap(QPixmap(":/images/notCover.png").scaled(coverSize,Qt::KeepAspectRatio,Qt::SmoothTransformation)); + } +} + +bool BookmarksDialog::eventFilter(QObject *obj, QEvent *event) +{ + if(event->type() == QEvent::MouseButtonPress) + { + if (obj == images.at(0)) + { + emit(goToPage(lastPage)); + close(); + event->accept(); + } + for(int i=1;i<=3;i++) + { + if(obj == images.at(i)) + { + bool b; + int page = pages.at(i)->text().toInt(&b)-1; + if(b) + { + emit(goToPage(page)); + close(); + } + event->accept(); + } + } + } + // pass the event on to the parent class + return QDialog::eventFilter(obj, event); +} + +void BookmarksDialog::keyPressEvent(QKeyEvent * event) +{ + if(event->key() == Qt::Key_M) + hide(); +} +/* +void BookmarksDialog::show() +{ + QDialog::show(); + disconnect(animation,SIGNAL(finished()),this,SLOT(close())); + animation->setStartValue(0); + animation->setEndValue(1); + animation->start(); +} + +void BookmarksDialog::hide() +{ + connect(animation,SIGNAL(finished()),this,SLOT(close())); + animation->setStartValue(1); + animation->setEndValue(0); + animation->start(); +}*/ diff --git a/YACReader/bookmarks_dialog.h b/YACReader/bookmarks_dialog.h new file mode 100644 index 00000000..c83a2c7e --- /dev/null +++ b/YACReader/bookmarks_dialog.h @@ -0,0 +1,45 @@ +#ifndef __BOOKMARKS_DIALOG_H +#define __BOOKMARKS_DIALOG_H + +#include +#include +#include +#include +#include +#include +#include + +#include "bookmarks.h" + + class BookmarksDialog : public QDialog + { + Q_OBJECT + + protected: + QList pages; + QList images; + + int lastPage; + + QPushButton * accept; + QPushButton * cancel; + + QSize coverSize; + + bool eventFilter(QObject *obj, QEvent *event); + void keyPressEvent(QKeyEvent * event); + //QPropertyAnimation * animation; + + public: + BookmarksDialog(QWidget * parent = 0); + + public slots: + void setBookmarks(const Bookmarks & bookmarks); + //void show(); + //void hide(); + + signals: + void goToPage(unsigned int page); + }; + +#endif // BOOKMARKS_DIALOG_H diff --git a/YACReader/configuration.cpp b/YACReader/configuration.cpp new file mode 100644 index 00000000..a89aab3d --- /dev/null +++ b/YACReader/configuration.cpp @@ -0,0 +1,162 @@ +#include "configuration.h" + +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +Configuration::Configuration() +{ + //read configuration + //load("/YACReader.conf"); +} + +/*Configuration::Configuration(const Configuration & conf) +{ + //nothing +}*/ + +void Configuration::load(QSettings * settings) +{ + this->settings = settings; + + //TODO set defaults + if(!settings->contains(PATH)) + settings->setValue(PATH,"."); + if(!settings->contains(GO_TO_FLOW_SIZE)) + settings->setValue(GO_TO_FLOW_SIZE,QSize(126,200)); + if(!settings->contains(MAG_GLASS_SIZE)) + settings->setValue(MAG_GLASS_SIZE,QSize(350,175)); + if(!settings->contains(ZOOM_LEVEL)) + settings->setValue(MAG_GLASS_SIZE,QSize(350,175)); + if(!settings->contains(FIT)) + settings->setValue(FIT,false); + if(!settings->contains(FLOW_TYPE)) + settings->setValue(FLOW_TYPE,0); + if(!settings->contains(FULLSCREEN)) + settings->setValue(FULLSCREEN,false); + if(!settings->contains(FIT_TO_WIDTH_RATIO)) + settings->setValue(FIT_TO_WIDTH_RATIO,1); + if(!settings->contains(Y_WINDOW_SIZE)) + settings->setValue(Y_WINDOW_SIZE,QSize(0,0)); + if(!settings->contains(MAXIMIZED)) + settings->setValue(MAXIMIZED,false); + if(!settings->contains(DOUBLE_PAGE)) + settings->setValue(DOUBLE_PAGE,false); + if(!settings->contains(ADJUST_TO_FULL_SIZE)) + settings->setValue(ADJUST_TO_FULL_SIZE,false); + if(!settings->contains(BACKGROUND_COLOR)) + settings->setValue(BACKGROUND_COLOR,QColor(40,40,40)); + if(!settings->contains(ALWAYS_ON_TOP)) + settings->setValue(ALWAYS_ON_TOP,false); + if(!settings->contains(SHOW_TOOLBARS)) + settings->setValue(SHOW_TOOLBARS, true); +} + +void Configuration::load(const QString & path) +{ + //load default configuration + defaultPath = "."; + magnifyingGlassSize = QSize(350,175); + gotoSlideSize = QSize(126,200); //normal + //gotoSlideSize = QSize(79,125); //small + //gotoSlideSize = QSize(173,275); //big + //gotoSlideSize = QSize(220,350); //huge + zoomLevel = 0.5; + adjustToWidth = true; + flowType = Strip; + fullScreen = false; + fitToWidthRatio = 1; + windowSize = QSize(0,0); + maximized = false; + doublePage = false; + adjustToFullSize = false; + backgroundColor = QColor(40,40,40); + alwaysOnTop = false; + + //load from file + QFile f(YACReader::getSettingsPath()+path); + f.open(QIODevice::ReadOnly); + QTextStream txtS(&f); + QString content = txtS.readAll(); + QStringList lines = content.split('\n'); + QString line,name; + int i=0; + foreach(line,lines) + { + if((i%2)==0) + { + name = line.trimmed(); + } + else + { + if(name==PATH) + defaultPath = line.trimmed(); + else + if(name==MAG_GLASS_SIZE) + { + QStringList values = line.split(','); + magnifyingGlassSize = QSize(values[0].toInt(),values[1].toInt()); + } + else + if(name==ZOOM_LEVEL) + zoomLevel = line.toFloat(); + else + if(name==SLIDE_SIZE) + { + int height = line.toInt(); + gotoSlideSize = QSize(static_cast(height/SLIDE_ASPECT_RATIO),height); + } + else + if(name==FIT) + adjustToWidth = line.toInt(); + else + if(name==FLOW_TYPE) + flowType = (FlowType)line.toInt(); + else + if(name==FULLSCREEN) + fullScreen = line.toInt(); + else + if(name==FIT_TO_WIDTH_RATIO) + fitToWidthRatio = line.toFloat(); + else + if(name==Y_WINDOW_POS) + { + QStringList l = line.split(','); + windowPos = QPoint(l[0].toInt(),l[1].toInt()); + } + else + if(name==Y_WINDOW_SIZE) + { + QStringList l = line.split(','); + windowSize = QSize(l[0].toInt(),l[1].toInt()); + } + else + if(name==MAXIMIZED) + maximized = line.toInt(); + else + if(name==DOUBLE_PAGE) + doublePage = line.toInt(); + else + if(name==ADJUST_TO_FULL_SIZE) + adjustToFullSize = line.toInt(); + else + if(name==BACKGROUND_COLOR) + { + QStringList l = line.split(','); + backgroundColor = QColor(l[0].toInt(),l[1].toInt(),l[2].toInt()); + } + else + if(name==ALWAYS_ON_TOP) + alwaysOnTop = line.toInt(); + + + + } + i++; + } +} diff --git a/YACReader/configuration.h b/YACReader/configuration.h new file mode 100644 index 00000000..7ef6b7ff --- /dev/null +++ b/YACReader/configuration.h @@ -0,0 +1,94 @@ +#ifndef __CONFIGURATION_H +#define __CONFIGURATION_H +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +#define CONF_FILE_PATH "." +#define SLIDE_ASPECT_RATIO 1.585 + +using namespace YACReader; + + class Configuration : public QObject + { + Q_OBJECT + + private: + QSettings * settings; + + QString defaultPath; + //configuration properties + QSize magnifyingGlassSize; + QSize gotoSlideSize; + float zoomLevel; + bool adjustToWidth; + bool fullScreen; + FlowType flowType; + float fitToWidthRatio; + QPoint windowPos; + QSize windowSize; + bool maximized; + bool doublePage; + bool alwaysOnTop; + bool adjustToFullSize; + QColor backgroundColor; + + Configuration(); + //Configuration(const Configuration & conf); + void load(const QString & path = CONF_FILE_PATH); + + + public: + static Configuration & getConfiguration() + { + static Configuration configuration; + return configuration; + }; + void load(QSettings * settings); + QString getDefaultPath() { return settings->value(PATH).toString(); } + void setDefaultPath(QString defaultPath){settings->setValue(PATH,defaultPath);} + QSize getMagnifyingGlassSize() { return settings->value(MAG_GLASS_SIZE).toSize();} + void setMagnifyingGlassSize(const QSize & mgs) { settings->setValue(MAG_GLASS_SIZE,mgs);} + QSize getGotoSlideSize() { return settings->value(GO_TO_FLOW_SIZE).toSize();} + void setGotoSlideSize(const QSize & gss) { settings->setValue(GO_TO_FLOW_SIZE,gss);} + float getZoomLevel() { return settings->value(ZOOM_LEVEL).toFloat();} + void setZoomLevel(float zl) { settings->setValue(ZOOM_LEVEL,zl);} + bool getAdjustToWidth() {return settings->value(FIT).toBool();} + void setAdjustToWidth(bool atw=true) {settings->setValue(FIT,atw);} + FlowType getFlowType(){return (FlowType)settings->value(FLOW_TYPE_SW).toInt();} + void setFlowType(FlowType type){settings->setValue(FLOW_TYPE_SW,type);} + bool getFullScreen(){return settings->value(FULLSCREEN).toBool();} + void setFullScreen(bool f){settings->setValue(FULLSCREEN,f);} + float getFitToWidthRatio(){return settings->value(FIT_TO_WIDTH_RATIO).toFloat();} + void setFitToWidthRatio(float r){settings->setValue(FIT_TO_WIDTH_RATIO,r);} + QPoint getPos(){return settings->value(Y_WINDOW_POS).toPoint();} + void setPos(QPoint p){settings->setValue(Y_WINDOW_POS,p);} + QSize getSize(){return settings->value(Y_WINDOW_SIZE).toSize();} + void setSize(QSize s){settings->setValue(Y_WINDOW_SIZE,s);} + bool getMaximized(){return settings->value(MAXIMIZED).toBool();} + void setMaximized(bool b){settings->setValue(MAXIMIZED,b);} + bool getDoublePage(){return settings->value(DOUBLE_PAGE).toBool();} + void setDoublePage(bool b){settings->setValue(DOUBLE_PAGE,b);} + bool getAdjustToFullSize(){return settings->value(ADJUST_TO_FULL_SIZE).toBool();} + void setAdjustToFullSize(bool b){settings->setValue(ADJUST_TO_FULL_SIZE,b);} + QColor getBackgroundColor(){return settings->value(BACKGROUND_COLOR).value();} + void setBackgroundColor(const QColor& color){settings->value(BACKGROUND_COLOR,color);} + bool getAlwaysOnTop(){return settings->value(ALWAYS_ON_TOP).toBool();} + void setAlwaysOnTop(bool b){ settings->setValue(ALWAYS_ON_TOP,b);} + bool getShowToolbars(){return settings->value(SHOW_TOOLBARS).toBool();} + void setShowToolbars(bool b){settings->setValue(SHOW_TOOLBARS,b);} + bool getShowInformation(){return settings->value(SHOW_INFO,false).toBool();} + void setShowInformation(bool b){settings->setValue(SHOW_INFO,b);} + QDate getLastVersionCheck(){return settings->value(LAST_VERSION_CHECK).toDate();} + void setLastVersionCheck(const QDate & date){ settings->setValue(LAST_VERSION_CHECK,date);} + int getNumDaysBetweenVersionChecks() {return settings->value(NUM_DAYS_BETWEEN_VERSION_CHECKS,1).toInt();} + void setNumDaysBetweenVersionChecks(int days) {return settings->setValue(NUM_DAYS_BETWEEN_VERSION_CHECKS,days);} + }; + +#endif diff --git a/YACReader/goto_dialog.cpp b/YACReader/goto_dialog.cpp new file mode 100644 index 00000000..b54805ca --- /dev/null +++ b/YACReader/goto_dialog.cpp @@ -0,0 +1,84 @@ +#include "goto_dialog.h" + +#include +#include +#include + + + +GoToDialog::GoToDialog(QWidget * parent) +:QDialog(parent) +{ + setupUI(); +} + +void GoToDialog::setupUI() +{ + textLabel = new QLabel(tr("Page : ")); + pageNumber = new QLineEdit; + v = new QIntValidator(this); + v->setBottom(1); + pageNumber->setValidator(v); + textLabel->setBuddy(pageNumber); + textLabel->setAlignment(Qt::AlignRight|Qt::AlignVCenter); + + accept = new QPushButton(tr("Go To")); + connect(accept,SIGNAL(clicked()),this,SLOT(goTo())); + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + QHBoxLayout *topLayout = new QHBoxLayout; + + topLayout->addWidget(textLabel); + topLayout->addWidget(pageNumber); + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addWidget(numPagesLabel = new QLabel(tr("Total pages : "))); + mainLayout->addLayout(topLayout); + mainLayout->addStretch(); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout *imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(); + QPixmap p(":/images/goto.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setWindowTitle(tr("Go to...")); + setModal (true); + + pageNumber->setFocusPolicy(Qt::StrongFocus); + pageNumber->setFocus(); +} + +void GoToDialog::goTo() +{ + unsigned int page = pageNumber->text().toInt(); + pageNumber->clear(); + + if(page >= 1) + emit(goToPage(page-1)); + + close(); + +} + +void GoToDialog::setNumPages(unsigned int numPages) +{ + numPagesLabel->setText(tr("Total pages : ")+QString::number(numPages)); + v->setTop(numPages); +} + +void GoToDialog::show() +{ + pageNumber->setFocus(); + QDialog::show(); +} \ No newline at end of file diff --git a/YACReader/goto_dialog.h b/YACReader/goto_dialog.h new file mode 100644 index 00000000..ef8a36b7 --- /dev/null +++ b/YACReader/goto_dialog.h @@ -0,0 +1,32 @@ +#ifndef __GOTODIALOG_H +#define __GOTODIALOG_H + +#include +#include +#include +#include +#include + + class GoToDialog : public QDialog + { + Q_OBJECT + public: + GoToDialog(QWidget * parent = 0); + private: + QLabel * numPagesLabel; + QLabel * textLabel; + QLineEdit * pageNumber; + QIntValidator * v; + QPushButton * accept; + QPushButton * cancel; + void setupUI(); + public slots: + void goTo(); + void setNumPages(unsigned int numPages); + void show(); + signals: + void goToPage(unsigned int page); + }; + +#endif + diff --git a/YACReader/goto_flow.cpp b/YACReader/goto_flow.cpp new file mode 100644 index 00000000..f6325b0f --- /dev/null +++ b/YACReader/goto_flow.cpp @@ -0,0 +1,320 @@ +#include "goto_flow.h" +#include "configuration.h" +#include "comic.h" + +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_flow.h" + +#include "goto_flow_toolbar.h" +#include "goto_flow_decorationbar.h" + + +GoToFlow::GoToFlow(QWidget *parent,FlowType flowType) +:GoToFlowWidget(parent),ready(false) +{ + updateTimer = new QTimer; + connect(updateTimer, SIGNAL(timeout()), this, SLOT(updateImageData())); + + worker = new PageLoader(&mutexGoToFlow); + + + flow = new YACReaderFlow(this,flowType); + flow->setReflectionEffect(PictureFlow::PlainReflection); + imageSize = Configuration::getConfiguration().getGotoSlideSize(); + + flow->setSlideSize(imageSize); + connect(flow,SIGNAL(centerIndexChanged(int)),this,SLOT(setPageNumber(int))); + connect(flow,SIGNAL(selected(unsigned int)),this,SIGNAL(goToPage(unsigned int))); + + connect(toolBar,SIGNAL(goTo(unsigned int)),this,SIGNAL(goToPage(unsigned int))); + connect(toolBar,SIGNAL(setCenter(unsigned int)),flow,SLOT(showSlide(unsigned int))); + + mainLayout->insertWidget(1,flow); + mainLayout->setStretchFactor(flow,1); + + resize(static_cast(5*imageSize.width()),static_cast(imageSize.height()*1.7)); + + //install eventFilter + //flow->installEventFilter(this); + /*edit->installEventFilter(this); + centerButton->installEventFilter(this); + goToButton->installEventFilter(this); + + connect(edit,SIGNAL(returnPressed()),goToButton,SIGNAL(clicked()));*/ + + this->setCursor(QCursor(Qt::ArrowCursor)); + + +} +GoToFlow::~GoToFlow() +{ + delete flow; + delete updateTimer; + worker->deleteLater(); +} + +void GoToFlow::keyPressEvent(QKeyEvent *event) +{ + switch (event->key()) + { + case Qt::Key_Left: case Qt::Key_Right: case Qt::Key_Up: + QApplication::sendEvent(flow,event); + return; + default: + break; + } + + GoToFlowWidget::keyPressEvent(event); +} + +void GoToFlow::centerSlide(int slide) +{ + if(flow->centerIndex()!=slide) + { + flow->setCenterIndex(slide); + if(ready)// load images if pages are loaded. + { + //worker->reset(); //BUG FIXED : image didn't load if worker was working + preload(); + } + } +} + +void GoToFlow::setNumSlides(unsigned int slides) +{ + // numPagesLabel->setText(tr("Total pages : ")+QString::number(slides)); + // numPagesLabel->adjustSize(); + imagesReady.clear(); + imagesReady.fill(false,slides); + + rawImages.clear(); + rawImages.resize(slides); + + toolBar->setTop(slides); + + SlideInitializer * si = new SlideInitializer(&mutexGoToFlow,flow,slides); + + imagesLoaded.clear(); + imagesLoaded.fill(false,slides); + + imagesSetted.clear(); + imagesSetted.fill(false,slides); + + numImagesLoaded = 0; + + connect(flow, SIGNAL(centerIndexChanged(int)), this, SLOT(preload())); + connect(flow, SIGNAL(centerIndexChangedSilent(int)), this, SLOT(preload())); + + ready = true; + worker->reset(); + + si->start(); +} + +void GoToFlow::reset() +{ + updateTimer->stop(); + /*imagesLoaded.clear(); + numImagesLoaded = 0; + imagesReady.clear(); + rawImages.clear();*/ + ready = false; +} + +void GoToFlow::setImageReady(int index,const QByteArray & image) +{ + rawImages[index]=image; + imagesReady[index]=true; + preload(); +} + +void GoToFlow::preload() +{ + if(numImagesLoaded < imagesLoaded.size()) + updateTimer->start(30); //TODO comprobar rendimiento, antes era 70 +} + +void GoToFlow::updateImageData() +{ + // can't do anything, wait for the next possibility + if(worker->busy()) + return; + + // set image of last one + int idx = worker->index(); + if( idx >= 0 && !worker->result().isNull()) + { + if(!imagesSetted[idx]) + { + flow->setSlide(idx, worker->result()); + imagesSetted[idx] = true; + numImagesLoaded++; + rawImages[idx].clear();; //release memory + imagesLoaded[idx]=true; + } + + } + + // try to load only few images on the left and right side + // i.e. all visible ones plus some extra +#define COUNT 8 + int indexes[2*COUNT+1]; + int center = flow->centerIndex(); + indexes[0] = center; + for(int j = 0; j < COUNT; j++) + { + indexes[j*2+1] = center+j+1; + indexes[j*2+2] = center-j-1; + } + for(int c = 0; c < 2*COUNT+1; c++) + { + int i = indexes[c]; + if((i >= 0) && (i < flow->slideCount())) + if(!imagesLoaded[i]&&imagesReady[i])//slide(i).isNull()) + { + // schedule thumbnail generation + + worker->generate(i, flow->slideSize(),rawImages[i]); + return; + } + + } + + // no need to generate anything? stop polling... + updateTimer->stop(); +} + +void GoToFlow::wheelEvent(QWheelEvent * event) +{ + if(event->delta()<0) + flow->showNext(); + else + flow->showPrevious(); + event->accept(); +} + +void GoToFlow::setFlowType(FlowType flowType) +{ + flow->setFlowType(flowType); +} + +void GoToFlow::updateSize() //TODO : fix. it doesn't work. +{ + imageSize = Configuration::getConfiguration().getGotoSlideSize(); + flow->setSlideSize(imageSize); + resize(static_cast(5*imageSize.width()),static_cast(imageSize.height()*1.7)); +} + +void GoToFlow::updateConfig(QSettings * settings) +{ + Q_UNUSED(settings) +} +//----------------------------------------------------------------------------- +//SlideInitializer +//----------------------------------------------------------------------------- +SlideInitializer::SlideInitializer(QMutex * m,PictureFlow * flow,int slides) +:QThread(),mutex(m),_flow(flow),_slides(slides) +{ + +} +void SlideInitializer::run() +{ + mutex->lock(); + + _flow->clear(); + for(int i=0;i<_slides;i++) + _flow->addSlide(QImage()); + _flow->setCenterIndex(0); + + mutex->unlock(); +} +//----------------------------------------------------------------------------- +//PageLoader +//----------------------------------------------------------------------------- + + +PageLoader::PageLoader(QMutex * m): +QThread(),mutex(m), restart(false), working(false), idx(-1) +{ +} + +PageLoader::~PageLoader() +{ + mutex->lock(); + condition.wakeOne(); + mutex->unlock(); + wait(); +} + +bool PageLoader::busy() const +{ + return isRunning() ? working : false; +} + +void PageLoader::generate(int index, QSize size,const QByteArray & rImage) +{ + mutex->lock(); + this->idx = index; + //this->img = QImage(); + this->size = size; + this->rawImage = rImage; + mutex->unlock(); + + if (!isRunning()) + start(); + else + { + // already running, wake up whenever ready + restart = true; + condition.wakeOne(); + } +} + +void PageLoader::run() +{ + for(;;) + { + // copy necessary data + mutex->lock(); + this->working = true; + //int idx = this->idx; + + + QImage image; + image.loadFromData(this->rawImage); + // let everyone knows it is ready + image = image.scaled(this->size,Qt::KeepAspectRatio,Qt::SmoothTransformation); + + mutex->unlock(); + + mutex->lock(); + this->working = false; + this->img = image; + mutex->unlock(); + + // put to sleep + mutex->lock(); + if (!this->restart) + condition.wait(mutex); + restart = false; + mutex->unlock(); + } +} diff --git a/YACReader/goto_flow.h b/YACReader/goto_flow.h new file mode 100644 index 00000000..eb2c2112 --- /dev/null +++ b/YACReader/goto_flow.h @@ -0,0 +1,110 @@ +#ifndef __GOTO_FLOW_H +#define __GOTO_FLOW_H + +#include "goto_flow_widget.h" +#include + +#include +#include + +class QLineEdit; +class QPushButton; +class QPixmap; +class QThread; +class QSize; +class QIntValidator; +class QWaitCondition; +class QEvent; +class QLabel; + + +class Comic; +class SlideInitializer; +class PageLoader; +class YACReaderFlow; +class PictureFlow; +class QKeyEvent; + +class GoToFlow : public GoToFlowWidget +{ + Q_OBJECT +public: + GoToFlow(QWidget* parent = 0,FlowType flowType = CoverFlowLike); + ~GoToFlow(); + bool ready; //comic is ready for read. +private: + YACReaderFlow * flow; + void keyPressEvent(QKeyEvent* event); + //Comic * comic; + QSize imageSize; + + QVector imagesLoaded; + QVector imagesSetted; + int numImagesLoaded; + QVector imagesReady; + QVector rawImages; + QTimer* updateTimer; + PageLoader* worker; + virtual void wheelEvent(QWheelEvent * event); + QMutex mutexGoToFlow; + + private slots: + void preload(); + void updateImageData(); + + public slots: + void centerSlide(int slide); + void reset(); + void setNumSlides(unsigned int slides); + void setImageReady(int index,const QByteArray & image); + void setFlowType(FlowType flowType); + void updateSize(); + void updateConfig(QSettings * settings); +signals: + void goToPage(unsigned int page); + +}; +//----------------------------------------------------------------------------- +//SlideInitializer +//----------------------------------------------------------------------------- +class SlideInitializer : public QThread +{ +public: + SlideInitializer(QMutex * m,PictureFlow * flow,int slides); +private: + QMutex * mutex; + PictureFlow * _flow; + int _slides; + void run(); +}; +//----------------------------------------------------------------------------- +//PageLoader +//----------------------------------------------------------------------------- + +class PageLoader : public QThread +{ +public: + PageLoader(QMutex * m); + ~PageLoader(); + // returns FALSE if worker is still busy and can't take the task + bool busy() const; + void generate(int index, QSize size,const QByteArray & rImage); + void reset(){idx = -1;}; + int index() const { return idx; } + QImage result() const { return img; } +protected: + void run(); +private: + QMutex * mutex; + QWaitCondition condition; + + bool restart; + bool working; + int idx; + + QSize size; + QImage img; + QByteArray rawImage; +}; + +#endif diff --git a/YACReader/goto_flow_decorationbar.cpp b/YACReader/goto_flow_decorationbar.cpp new file mode 100644 index 00000000..c9ef2fbb --- /dev/null +++ b/YACReader/goto_flow_decorationbar.cpp @@ -0,0 +1,33 @@ +#include "goto_flow_decorationbar.h" + +#include +#include + +GoToFlowDecorationBar::GoToFlowDecorationBar(QWidget * parent) +:QWidget(parent) +{ + QHBoxLayout * topBar = new QHBoxLayout(); + + QLabel * imgTopLeft = new QLabel(); + QLabel * imgTopRight = new QLabel(); + QLabel * imgTopMiddle = new QLabel(); + QPixmap pL(":/images/imgTopLeft.png"); + QPixmap pM(":/images/imgTopMiddle.png"); + QPixmap pR(":/images/imgTopRight.png"); + imgTopLeft->setPixmap(pL); + imgTopRight->setPixmap(pR); + imgTopMiddle->setPixmap(pM); + imgTopMiddle->setScaledContents(true); + + topBar->addWidget(imgTopLeft); + topBar->addWidget(imgTopMiddle); + topBar->addWidget(imgTopRight); + topBar->setStretchFactor(imgTopLeft,0); + topBar->setStretchFactor(imgTopMiddle,1); + topBar->setStretchFactor(imgTopRight,0); + + topBar->setMargin(0); + topBar->setSpacing(0); + + setLayout(topBar); +} \ No newline at end of file diff --git a/YACReader/goto_flow_decorationbar.h b/YACReader/goto_flow_decorationbar.h new file mode 100644 index 00000000..10397124 --- /dev/null +++ b/YACReader/goto_flow_decorationbar.h @@ -0,0 +1,13 @@ +#ifndef GOTO_FLOW_DECORATIONBAR_H +#define GOTO_FLOW_DECORATIONBAR_H + +#include + +class GoToFlowDecorationBar : public QWidget +{ + Q_OBJECT + public: + GoToFlowDecorationBar(QWidget * parent = 0); +}; + +#endif \ No newline at end of file diff --git a/YACReader/goto_flow_gl.cpp b/YACReader/goto_flow_gl.cpp new file mode 100644 index 00000000..8f315c7d --- /dev/null +++ b/YACReader/goto_flow_gl.cpp @@ -0,0 +1,158 @@ +#include "goto_flow_gl.h" + +#include +#include +#include +#include +#include +#include + +#include "configuration.h" + +#include "goto_flow_toolbar.h" +#include "goto_flow_decorationbar.h" + +GoToFlowGL::GoToFlowGL(QWidget* parent, FlowType flowType) + :GoToFlowWidget(parent) +{ + Q_UNUSED(flowType) + flow = new YACReaderPageFlowGL(this); + flow->setShowMarks(false); + + imageSize = Configuration::getConfiguration().getGotoSlideSize(); + + flow->setSlideSize(imageSize); + connect(flow,SIGNAL(centerIndexChanged(int)),this,SLOT(setPageNumber(int))); + connect(flow,SIGNAL(selected(unsigned int)),this,SIGNAL(goToPage(unsigned int))); + + connect(toolBar,SIGNAL(goTo(unsigned int)),this,SIGNAL(goToPage(unsigned int))); + connect(toolBar,SIGNAL(setCenter(unsigned int)),flow,SLOT(setCenterIndex(unsigned int))); + + mainLayout->insertWidget(1,flow); + mainLayout->setStretchFactor(flow,1); + + resize(static_cast(5*imageSize.width()),static_cast(imageSize.height()*1.7)); + + this->setCursor(QCursor(Qt::ArrowCursor)); +} + +GoToFlowGL::~GoToFlowGL() +{ + delete flow; +} + +void GoToFlowGL::reset() +{ + flow->reset(); +} + +void GoToFlowGL::centerSlide(int slide) +{ + if(flow->centerIndex()!=slide) + { + flow->setCenterIndex(slide); + } +} + +void GoToFlowGL::setFlowType(FlowType flowType) +{ + if(flowType == CoverFlowLike) + flow->setPreset(presetYACReaderFlowClassicConfig); + else if(flowType == Strip) + flow->setPreset(presetYACReaderFlowStripeConfig); + else if(flowType == StripOverlapped) + flow->setPreset(presetYACReaderFlowOverlappedStripeConfig); + else + flow->setPreset(defaultYACReaderFlowConfig); +} + +void GoToFlowGL::setNumSlides(unsigned int slides) +{ + flow->populate(slides); + toolBar->setTop(slides); +} +void GoToFlowGL::setImageReady(int index,const QByteArray & imageData) +{ + flow->rawImages[index] = imageData; + flow->imagesReady[index] = true; +} + +void GoToFlowGL::updateSize() +{ + +} + +void GoToFlowGL::updateConfig(QSettings * settings) +{ + Performance performance = medium; + + switch (settings->value(PERFORMANCE).toInt()) + { + case 0: + performance = low; + break; + case 1: + performance = medium; + break; + case 2: + performance = high; + break; + case 3: + performance = ultraHigh; + break; + } + + flow->setPerformance(performance); + + switch (settings->value(FLOW_TYPE_GL).toInt()) + { + case 0: + flow->setPreset(presetYACReaderFlowClassicConfig); + return; + case 1: + flow->setPreset(presetYACReaderFlowStripeConfig); + return; + case 2: + flow->setPreset(presetYACReaderFlowOverlappedStripeConfig); + return; + case 3: + flow->setPreset(defaultYACReaderFlowConfig); + return; + case 4: + flow->setPreset(pressetYACReaderFlowDownConfig); + return; + } + + + //custom config + + flow->setCF_RX(settings->value(X_ROTATION).toInt()); + flow->setCF_Y(settings->value(Y_POSITION).toInt()); + flow->setX_Distance(settings->value(COVER_DISTANCE).toInt()); + flow->setCenter_Distance(settings->value(CENTRAL_DISTANCE).toInt()); + flow->setCF_Z(settings->value(ZOOM_LEVEL).toInt()); + flow->setY_Distance(settings->value(Y_COVER_OFFSET).toInt()); + flow->setZ_Distance(settings->value(Z_COVER_OFFSET).toInt()); + flow->setRotation(settings->value(COVER_ROTATION).toInt()); + flow->setFadeOutDist(settings->value(FADE_OUT_DIST).toInt()); + flow->setLightStrenght(settings->value(LIGHT_STRENGTH).toInt()); + flow->setMaxAngle(settings->value(MAX_ANGLE).toInt()); + +/* flow->setVisibility(settings->value("visibilityDistance").toInt()); + flow->setLightStrenght(settings->value("lightStrength").toInt())*/; + +} + +void GoToFlowGL::keyPressEvent(QKeyEvent* event) +{ + switch (event->key()) + { + case Qt::Key_Left: case Qt::Key_Right: case Qt::Key_Up: + QApplication::sendEvent(flow,event); + return; + default: + break; + } + + GoToFlowWidget::keyPressEvent(event); +} diff --git a/YACReader/goto_flow_gl.h b/YACReader/goto_flow_gl.h new file mode 100644 index 00000000..5b6ed1b5 --- /dev/null +++ b/YACReader/goto_flow_gl.h @@ -0,0 +1,39 @@ +#ifndef __GOTO_FLOW_GL_H +#define __GOTO_FLOW_GL_H + +#include "yacreader_global.h" +#include "goto_flow_widget.h" +#include "yacreader_flow_gl.h" + +class QLineEdit; +class QIntValidator; +class QPushButton; +class QPushButton; +class QSize; +class QKeyEvent; + +class GoToFlowGL : public GoToFlowWidget +{ + Q_OBJECT +public: + GoToFlowGL(QWidget* parent = 0,FlowType flowType = CoverFlowLike); + ~GoToFlowGL(); + void reset(); + void centerSlide(int slide); + void setFlowType(FlowType flowType); + void setNumSlides(unsigned int slides); + void setImageReady(int index,const QByteArray & image); + void updateSize(); + + void updateConfig(QSettings * settings); + +signals: + void goToPage(unsigned int page); +private: + YACReaderPageFlowGL * flow; + void keyPressEvent(QKeyEvent* event); + //Comic * comic; + QSize imageSize; +}; + +#endif diff --git a/YACReader/goto_flow_toolbar.cpp b/YACReader/goto_flow_toolbar.cpp new file mode 100644 index 00000000..d1e4fe62 --- /dev/null +++ b/YACReader/goto_flow_toolbar.cpp @@ -0,0 +1,122 @@ +#include "goto_flow_toolbar.h" + +#include +#include +#include +#include +#include +#include +#include + +GoToFlowToolBar::GoToFlowToolBar(QWidget * parent) + :QWidget(parent) +{ + //fondo + QBoxLayout * background = new QHBoxLayout(this); + + QLabel * imgBottomLeft = new QLabel(this); + QLabel * imgBottomRight = new QLabel(this); + QLabel * imgBottomMiddle = new QLabel(this); + QPixmap pBL(":/images/imgBottomLeft.png"); + QPixmap pBM(":/images/imgBottomMiddle.png"); + QPixmap pBR(":/images/imgBottomRight.png"); + imgBottomLeft->setPixmap(pBL); + imgBottomRight->setPixmap(pBR); + imgBottomMiddle->setPixmap(pBM); + imgBottomMiddle->setScaledContents(true); + //imgTop->setStyleSheet("background-image: url(:/images/numPagesLabel.png); width: 100%; height:100%; background-repeat: none; border: none"); + + background->addWidget(imgBottomLeft); + background->addWidget(imgBottomMiddle); + background->addWidget(imgBottomRight); + background->setStretchFactor(imgBottomLeft,0); + background->setStretchFactor(imgBottomMiddle,1); + background->setStretchFactor(imgBottomRight,0); + + background->setMargin(0); + background->setSpacing(0); + + //elementos interactivos + //QVBoxLayout * mainLayout = new QVBoxLayout; + bar = new QWidget(this); + QHBoxLayout * bottom = new QHBoxLayout(bar); + bottom->addStretch(); + bottom->addWidget(new QLabel(tr("Page : "),bar)); + bottom->addWidget(edit = new QLineEdit(bar)); + v = new QIntValidator(bar); + v->setBottom(1); + edit->setValidator(v); + edit->setAlignment(Qt::AlignRight|Qt::AlignVCenter); + edit->setStyleSheet("background-image: url(:/images/imgEdit.png); width: 100%; height:100%; background-repeat: none; border: none; padding: 3px; color: white;"); + QPixmap p(":/images/imgEdit.png"); + edit->setFixedSize(54,50); + edit->setAttribute(Qt::WA_MacShowFocusRect,false); + edit->setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + //edit->resize(QSize(54,50)); + edit->setSizePolicy(QSizePolicy(QSizePolicy::Fixed,QSizePolicy::Fixed)); + edit->setAutoFillBackground(false); + connect(edit,SIGNAL(returnPressed()),this,SLOT(goTo())); + + QString centerButtonCSS = "QPushButton {background-image: url(:/images/imgCenterSlide.png); width: 100%; height:100%; background-repeat: none; border: none;} " + "QPushButton:focus { border: none; outline: none;}" + "QPushButton:pressed {background-image: url(:/images/imgCenterSlidePressed.png); width: 100%; height:100%; background-repeat: none; border: none;} "; + centerButton = new QPushButton(bar); + //centerButton->setIcon(QIcon(":/images/center.png")); + centerButton->setStyleSheet(centerButtonCSS); + centerButton->setFixedSize(26,50); + centerButton->setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + connect(centerButton,SIGNAL(clicked()),this,SLOT(centerSlide())); + bottom->addWidget(centerButton); + + QString goToButtonCSS = "QPushButton {background-image: url(:/images/imgGoToSlide.png); width: 100%; height:100%; background-repeat: none; border: none;} " + "QPushButton:focus { border: none; outline: none;}" + "QPushButton:pressed {background-image: url(:/images/imgGoToSlidePressed.png); width: 100%; height:100%; background-repeat: none; border: none;} "; + goToButton = new QPushButton(bar); + //goToButton->setIcon(QIcon(":/images/goto.png")); + goToButton->setStyleSheet(goToButtonCSS); + goToButton->setFixedSize(32,50); + goToButton->setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + + connect(goToButton,SIGNAL(clicked()),this,SLOT(goTo())); + bottom->addWidget(goToButton); + + bottom->addStretch(); + bottom->setMargin(0); + bottom->setSpacing(0); + + bar->setLayout(bottom); + //mainLayout->addWidget(bar); + setLayout(background); + bar->setGeometry(QRect(0,0,400,50)); + +} + +void GoToFlowToolBar::setPage(int pageNumber) +{ + edit->setText(QString::number(pageNumber+1)); +} + +void GoToFlowToolBar::setTop(int numPages) +{ + v->setTop(numPages); +} + +void GoToFlowToolBar::resizeEvent(QResizeEvent * event) +{ + + bar->setGeometry(QRect(0,(event->size().height()-50)+((50-bar->height())/2),event->size().width(),50)); + + QWidget::resizeEvent(event); +} + +void GoToFlowToolBar::goTo() +{ + if(edit->text().toInt()!=0) + emit(goTo(edit->text().toInt()-1)); +} + +void GoToFlowToolBar::centerSlide() +{ + if(edit->text().toInt()!=0) + emit(setCenter(edit->text().toInt()-1)); +} \ No newline at end of file diff --git a/YACReader/goto_flow_toolbar.h b/YACReader/goto_flow_toolbar.h new file mode 100644 index 00000000..0fe6694d --- /dev/null +++ b/YACReader/goto_flow_toolbar.h @@ -0,0 +1,33 @@ +#ifndef GOTO_FLOW_TOOLBAR_H +#define GOTO_FLOW_TOOLBAR_H + +#include + +class QLineEdit; +class QIntValidator; +class QPushButton; + +class GoToFlowToolBar : public QWidget +{ + Q_OBJECT + private: + QLineEdit * edit; + QIntValidator * v; + QPushButton * centerButton; + QPushButton * goToButton; + QWidget * bar; + void resizeEvent(QResizeEvent * event); + + public: + GoToFlowToolBar(QWidget * parent = 0); + public slots: + void setPage(int pageNumber); + void setTop(int numPages); + void goTo(); + void centerSlide(); + signals: + void setCenter(unsigned int); + void goTo(unsigned int); +}; + +#endif \ No newline at end of file diff --git a/YACReader/goto_flow_widget.cpp b/YACReader/goto_flow_widget.cpp new file mode 100644 index 00000000..1f0c98c8 --- /dev/null +++ b/YACReader/goto_flow_widget.cpp @@ -0,0 +1,75 @@ +#include "goto_flow_widget.h" + +#include +#include +#include +#include + +#include "goto_flow_toolbar.h" +#include "goto_flow_decorationbar.h" + +GoToFlowWidget::GoToFlowWidget(QWidget * parent) + :QWidget(parent) +{ + mainLayout = new QVBoxLayout; + + mainLayout->setMargin(0); + mainLayout->setSpacing(0); + + topBar = new GoToFlowDecorationBar(this); + toolBar = new GoToFlowToolBar(this); + + mainLayout->addWidget(topBar); + mainLayout->addWidget(toolBar); + + mainLayout->setMargin(0); + mainLayout->setSpacing(0); + + setLayout(mainLayout); + + //toolBar->installEventFilter(this); +} + +GoToFlowWidget::~GoToFlowWidget() { + delete topBar; + delete toolBar; + delete mainLayout; +} + +void GoToFlowWidget::setPageNumber(int page) +{ + toolBar->setPage(page); +} + +void GoToFlowWidget::keyPressEvent(QKeyEvent* event) +{ + switch (event->key()) + { + case Qt::Key_Return: case Qt::Key_Enter: + toolBar->goTo(); + toolBar->centerSlide(); + break; + case Qt::Key_Space: + toolBar->centerSlide(); + break; + case Qt::Key_S: + QCoreApplication::sendEvent(this->parent(),event); + break; + } + + event->accept(); +} + +/*bool GoToFlowWidget::eventFilter(QObject * target, QEvent * event) +{ + if(event->type() == QEvent::KeyPress) + { + QKeyEvent * e = static_cast(event); + if(e->key()==Qt::Key_S || e->key() == Qt::Key_Space) + { + this->keyPressEvent(e); + return true; + } + } + return QWidget::eventFilter(target,event); +}*/ diff --git a/YACReader/goto_flow_widget.h b/YACReader/goto_flow_widget.h new file mode 100644 index 00000000..e4f42b2c --- /dev/null +++ b/YACReader/goto_flow_widget.h @@ -0,0 +1,41 @@ +#ifndef __GOTO_FLOW_WIDGET_H +#define __GOTO_FLOW_WIDGET_H + +#include +#include +#include "yacreader_global.h" + +using namespace YACReader; + +class QSettings; +class GoToFlowDecorationBar; +class GoToFlowToolBar; +class QVBoxLayout; + +class GoToFlowWidget : public QWidget +{ + Q_OBJECT +protected: + QVBoxLayout * mainLayout; + GoToFlowDecorationBar * topBar; + GoToFlowToolBar * toolBar; +public: + GoToFlowWidget(QWidget * paret = 0); + virtual ~GoToFlowWidget() = 0; +public slots: + virtual void reset() = 0; + virtual void centerSlide(int slide) = 0; + virtual void setPageNumber(int page); + virtual void setFlowType(FlowType flowType) = 0; + virtual void setNumSlides(unsigned int slides) = 0; + virtual void setImageReady(int index,const QByteArray & image) = 0; + virtual void updateSize() = 0; + virtual void updateConfig(QSettings * settings) = 0; + +protected: + void keyPressEvent(QKeyEvent* event); + //bool eventFilter(QObject *, QEvent *); + +}; + +#endif diff --git a/YACReader/icon.ico b/YACReader/icon.ico new file mode 100644 index 0000000000000000000000000000000000000000..6c31de5a6f6c0c322fe853c1b53fe9c204f6330f GIT binary patch literal 99678 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 Y{QeF6{tf*84gCHM{QeF6zw{0K|6N)}g8%>k literal 0 HcmV?d00001 diff --git a/YACReader/icon.rc b/YACReader/icon.rc new file mode 100644 index 00000000..86e8e9b3 --- /dev/null +++ b/YACReader/icon.rc @@ -0,0 +1 @@ +IDI_ICON1 ICON DISCARDABLE "icon.ico" diff --git a/YACReader/magnifying_glass.cpp b/YACReader/magnifying_glass.cpp new file mode 100644 index 00000000..c86f202e --- /dev/null +++ b/YACReader/magnifying_glass.cpp @@ -0,0 +1,292 @@ +#include "magnifying_glass.h" +#include "viewer.h" +#include "configuration.h" +#include "shortcuts_manager.h" + +#include + +MagnifyingGlass::MagnifyingGlass(int w, int h, QWidget * parent) +:QLabel(parent),zoomLevel(0.5) +{ + setup(QSize(w,h)); +} + +MagnifyingGlass::MagnifyingGlass(const QSize & size, QWidget * parent) +:QLabel(parent),zoomLevel(0.5) +{ + setup(size); +} + +void MagnifyingGlass::setup(const QSize & size) +{ + resize(size); + setScaledContents(true); + setMouseTracking(true); + setCursor(QCursor(QBitmap(1,1),QBitmap(1,1))); +} + +void MagnifyingGlass::mouseMoveEvent(QMouseEvent * event) +{ + updateImage(); + event->accept(); +} + +void MagnifyingGlass::updateImage(int x, int y) +{ + //image section augmented + int zoomWidth = static_cast(width() * zoomLevel); + int zoomHeight = static_cast(height() * zoomLevel); + Viewer * p = (Viewer *)parent(); + int currentPos = p->verticalScrollBar()->sliderPosition(); + const QPixmap * image = p->pixmap(); + int iWidth = image->width(); + int iHeight = image->height(); + float wFactor = static_cast(iWidth) / p->widget()->width(); + float hFactor = static_cast(iHeight) / p->widget()->height(); + zoomWidth *= wFactor; + zoomHeight *= hFactor; + if(p->verticalScrollBar()->minimum()==p->verticalScrollBar()->maximum()) + { + int xp = static_cast(((x-p->widget()->pos().x())*wFactor)-zoomWidth/2); + int yp = static_cast((y-p->widget()->pos().y()+currentPos)*hFactor-zoomHeight/2); + int xOffset=0; + int yOffset=0; + int zw=zoomWidth; + int zh=zoomHeight; + //int wOffset,hOffset=0; + bool outImage = false; + if(xp<0) + { + xOffset = -xp; + xp=0; + zw = zw - xOffset; + outImage = true; + } + if(yp<0) + { + yOffset = -yp; + yp=0; + zh = zh - yOffset; + outImage = true; + } + + if(xp+zoomWidth >= image->width()) + { + zw -= xp+zw - image->width(); + outImage = true; + } + if(yp+zoomHeight >= image->height()) + { + zh -= yp+zh - image->height(); + outImage = true; + } + if(outImage) + { + QImage img(zoomWidth,zoomHeight,QImage::Format_RGB32); + img.fill(Configuration::getConfiguration().getBackgroundColor()); + if(zw>0&&zh>0) + { + QPainter painter(&img); + painter.drawPixmap(xOffset,yOffset,p->pixmap()->copy(xp,yp,zw,zh)); + } + setPixmap(QPixmap().fromImage(img)); + } + else + setPixmap(p->pixmap()->copy(xp,yp,zoomWidth,zoomHeight)); + } + else + { + int xp = static_cast(((x-p->widget()->pos().x())*wFactor)-zoomWidth/2); + int yp = static_cast((y+currentPos)*hFactor-zoomHeight/2); + int xOffset=0; + int yOffset=0; + int zw=zoomWidth; + int zh=zoomHeight; + //int wOffset,hOffset=0; + bool outImage = false; + if(xp<0) + { + xOffset = -xp; + xp=0; + zw = zw - xOffset; + outImage = true; + } + if(yp<0) + { + yOffset = -yp; + yp=0; + zh = zh - yOffset; + outImage = true; + } + + if(xp+zoomWidth >= image->width()) + { + zw -= xp+zw - image->width(); + outImage = true; + } + if(yp+zoomHeight >= image->height()) + { + zh -= yp+zh - image->height(); + outImage = true; + } + if(outImage) + { + QImage img(zoomWidth,zoomHeight,QImage::Format_RGB32); + img.fill(Configuration::getConfiguration().getBackgroundColor()); + if(zw>0&&zh>0) + { + QPainter painter(&img); + painter.drawPixmap(xOffset,yOffset,p->pixmap()->copy(xp,yp,zw,zh)); + } + setPixmap(QPixmap().fromImage(img)); + } + else + setPixmap(p->pixmap()->copy(xp,yp,zoomWidth,zoomHeight)); + } + move(static_cast(x-float(width())/2),static_cast(y-float(height())/2)); +} + +void MagnifyingGlass::updateImage() +{ + if(isVisible()) + { + QPoint p = QPoint(cursor().pos().x(),cursor().pos().y()); + p = this->parentWidget()->mapFromGlobal(p); + updateImage(p.x(),p.y()); + } +} +void MagnifyingGlass::wheelEvent(QWheelEvent * event) +{ + switch(event->modifiers()) + { + //size + case Qt::NoModifier: + if(event->delta()<0) + sizeUp(); + else + sizeDown(); + break; + //size height + case Qt::ControlModifier: + if(event->delta()<0) + heightUp(); + else + heightDown(); + break; + //size width + case Qt::AltModifier: + if(event->delta()<0) + widthUp(); + else + widthDown(); + break; + //zoom level + case Qt::ShiftModifier: + if(event->delta()<0) + zoomIn(); + else + zoomOut(); + break; + } + updateImage(); + event->setAccepted(true); +} +void MagnifyingGlass::zoomIn() +{ + if(zoomLevel>0.2f) + zoomLevel -= 0.025f; +} + +void MagnifyingGlass::zoomOut() +{ + if(zoomLevel<0.9f) + zoomLevel += 0.025f; +} + +void MagnifyingGlass::sizeUp() +{ + Viewer * p = (Viewer *)parent(); + if(width()<(p->width()*0.90f)) + resize(width()+30,height()+15); +} + +void MagnifyingGlass::sizeDown() +{ + if(width()>175) + resize(width()-30,height()-15); +} + +void MagnifyingGlass::heightUp() +{ + Viewer * p = (Viewer *)parent(); + if(height()<(p->height()*0.90f)) + resize(width(),height()+15); +} + +void MagnifyingGlass::heightDown() +{ + if(height()>80) + resize(width(),height()-15); +} + +void MagnifyingGlass::widthUp() +{ + Viewer * p = (Viewer *)parent(); + if(width()<(p->width()*0.90f)) + resize(width()+30,height()); +} + +void MagnifyingGlass::widthDown() +{ + if(width()>175) + resize(width()-30,height()); +} + +void MagnifyingGlass::keyPressEvent(QKeyEvent *event) +{ + bool validKey = false; + + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y)) + { + sizeUp(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y)) + { + sizeDown(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y)) + { + zoomIn(); + validKey = true; + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)) + { + zoomOut(); + validKey = true; + } + + if(validKey) + { + updateImage(); + event->setAccepted(true); + } +} diff --git a/YACReader/magnifying_glass.h b/YACReader/magnifying_glass.h new file mode 100644 index 00000000..519e3404 --- /dev/null +++ b/YACReader/magnifying_glass.h @@ -0,0 +1,34 @@ +#ifndef __MAGNIFYING_GLASS +#define __MAGNIFYING_GLASS + +#include +#include +#include +#include + + class MagnifyingGlass : public QLabel + { + Q_OBJECT + private: + float zoomLevel; + void setup(const QSize & size); + void keyPressEvent(QKeyEvent * event); + public: + MagnifyingGlass(int width,int height,QWidget * parent); + MagnifyingGlass(const QSize & size, QWidget * parent); + void mouseMoveEvent(QMouseEvent * event); + public slots: + void updateImage(int x, int y); + void updateImage(); + void wheelEvent(QWheelEvent * event); + void zoomIn(); + void zoomOut(); + void sizeUp(); + void sizeDown(); + void heightUp(); + void heightDown(); + void widthUp(); + void widthDown(); + }; + +#endif diff --git a/YACReader/main.cpp b/YACReader/main.cpp new file mode 100644 index 00000000..d5bbdfb5 --- /dev/null +++ b/YACReader/main.cpp @@ -0,0 +1,118 @@ +#include +#include +#include +//#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "main_window_viewer.h" +#include "configuration.h" +#include "exit_check.h" + +#include "QsLog.h" +#include "QsLogDest.h" + +using namespace QsLogging; + + #if defined(WIN32) && defined(_DEBUG) + #define _CRTDBG_MAP_ALLOC + #include + #include + #define DEBUG_NEW new( _NORMAL_BLOCK, __FILE__, __LINE__ ) + #define new DEBUG_NEW + #endif + +#ifdef Q_OS_MAC +#include +class YACReaderApplication: public QApplication +{ + public: + YACReaderApplication(int & argc, char ** argv) : QApplication(argc,argv) + {} + + void setWindow(MainWindowViewer * w) + { + window = w; + } + + protected: + bool event(QEvent * event) + { + switch(event->type()) + { + case QEvent::FileOpen: + window->openComicFromPath(static_cast(event)->file()); + return true; + default: + return QApplication::event(event); + } + } + private: + MainWindowViewer * window; +}; +#endif + +int main(int argc, char * argv[]) +{ + #if defined(_MSC_VER) && defined(_DEBUG) + _CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF ); +#endif + +//fix for misplaced text in Qt4.8 and Mavericks +#ifdef Q_OS_MAC + #if QT_VERSION < 0x050000 + if(QSysInfo::MacintoshVersion > QSysInfo::MV_10_8) + QFont::insertSubstitution(".Lucida Grande UI", "Lucida Grande"); + #endif +#endif + + +#ifdef Q_OS_MAC + YACReaderApplication app(argc,argv); +#else + QApplication app(argc, argv); +#endif + + app.setApplicationName("YACReader"); + app.setOrganizationName("YACReader"); + + QString destLog = YACReader::getSettingsPath()+"/yacreader.log"; + QDir().mkpath(YACReader::getSettingsPath()); + + Logger& logger = Logger::instance(); + logger.setLoggingLevel(QsLogging::TraceLevel); + + DestinationPtr fileDestination(DestinationFactory::MakeFileDestination( + destLog, EnableLogRotation, MaxSizeBytes(1048576), MaxOldLogCount(2))); + DestinationPtr debugDestination(DestinationFactory::MakeDebugOutputDestination()); + logger.addDestination(debugDestination); + logger.addDestination(fileDestination); + + QTranslator translator; + QString sufix = QLocale::system().name(); +#if defined Q_OS_UNIX && !defined Q_OS_MAC + translator.load(QString(DATADIR)+"/YACReader/languages/yacreader_"+sufix); +#else + translator.load(QCoreApplication::applicationDirPath()+"/languages/yacreader_"+sufix); +#endif + app.installTranslator(&translator); + + MainWindowViewer * mwv = new MainWindowViewer(); +#ifdef Q_OS_MAC + app.setWindow(mwv); +#endif + mwv->show(); + + int ret = app.exec(); + + //Configuration::getConfiguration().save(); + + YACReader::exitCheck(ret); + + return ret; +} diff --git a/YACReader/main_window_viewer.cpp b/YACReader/main_window_viewer.cpp new file mode 100644 index 00000000..3c16a805 --- /dev/null +++ b/YACReader/main_window_viewer.cpp @@ -0,0 +1,1352 @@ +#include "main_window_viewer.h" +#include "configuration.h" +#include "viewer.h" +#include "goto_dialog.h" +#include "custom_widgets.h" +#include "options_dialog.h" +#include "check_new_version.h" +#include "comic.h" +#include "bookmarks_dialog.h" +#include "shortcuts_dialog.h" +#include "width_slider.h" +#include "qnaturalsorting.h" +#include "help_about_dialog.h" +#include "yacreader_tool_bar_stretch.h" + +#include "comic_db.h" +#include "yacreader_local_client.h" + +#include "yacreader_global.h" +#include "edit_shortcuts_dialog.h" +#include "shortcuts_manager.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + + +#ifdef Q_OS_MAC +class MacToolBarSeparator : public QWidget +{ +public: + MacToolBarSeparator(QWidget * parent =0) + :QWidget(parent) + { + setFixedWidth(2); + } + + void paintEvent(QPaintEvent *event) + { + Q_UNUSED(event); + QPainter painter(this); + + QLinearGradient lG(0,0,0,height()); + + lG.setColorAt(0,QColor(128,128,128,0)); + lG.setColorAt(0.5,QColor(128,128,128,255)); + lG.setColorAt(1,QColor(128,128,128,0)); + + painter.fillRect(0,0,1,height(),lG); + + QLinearGradient lG2(1,0,1,height()); + + lG2.setColorAt(0,QColor(220,220,220,0)); + lG2.setColorAt(0.5,QColor(220,220,220,255)); + lG2.setColorAt(1,QColor(220,220,220,0)); + + painter.fillRect(1,0,1,height(),lG2); + } +}; +#endif + +MainWindowViewer::MainWindowViewer() +:QMainWindow(),fullscreen(false),toolbars(true),alwaysOnTop(false),currentDirectory("."),currentDirectoryImgDest("."),isClient(false) +{ + loadConfiguration(); + setupUI(); +} + +MainWindowViewer::~MainWindowViewer() +{ + delete settings; + delete viewer; + delete had; + + delete sliderAction; + delete openAction; + delete openFolderAction; + delete saveImageAction; + delete openPreviousComicAction; + delete openNextComicAction; + delete prevAction; + delete nextAction; + delete adjustHeightAction; + delete adjustWidthAction; + delete leftRotationAction; + delete rightRotationAction; + delete doublePageAction; + delete goToPageAction; + delete optionsAction; + delete helpAboutAction; + delete showMagnifyingGlassAction; + delete setBookmarkAction; + delete showBookmarksAction; + delete showShorcutsAction; + delete showInfoAction; + delete closeAction; + delete showDictionaryAction; + delete alwaysOnTopAction; + delete adjustToFullSizeAction; + delete showFlowAction; + +} +void MainWindowViewer::loadConfiguration() +{ + settings = new QSettings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + + Configuration & config = Configuration::getConfiguration(); + config.load(settings); + currentDirectory = config.getDefaultPath(); + fullscreen = config.getFullScreen(); +} + +void MainWindowViewer::setupUI() +{ + setWindowIcon(QIcon(":/images/icon.png")); + + setUnifiedTitleAndToolBarOnMac(true); + + viewer = new Viewer(this); + connect(viewer,SIGNAL(reset()),this,SLOT(processReset())); + //detected end of comic + connect(viewer,SIGNAL(openNextComic()),this,SLOT(openNextComic())); + //detected start of comic + connect(viewer,SIGNAL(openPreviousComic()),this,SLOT(openPreviousComic())); + + setCentralWidget(viewer); + int heightDesktopResolution = QApplication::desktop()->screenGeometry().height(); + int widthDesktopResolution = QApplication::desktop()->screenGeometry().width(); + int height,width; + height = static_cast(heightDesktopResolution*0.84); + width = static_cast(height*0.70); + Configuration & conf = Configuration::getConfiguration(); + QPoint p = conf.getPos(); + QSize s = conf.getSize(); + if(s.width()!=0) + { + move(p); + resize(s); + } + else + { + move(QPoint((widthDesktopResolution-width)/2,((heightDesktopResolution-height)-40)/2)); + resize(QSize(width,height)); + } + + had = new HelpAboutDialog(this); //TODO load data + + had->loadAboutInformation(":/files/about.html"); + had->loadHelp(":/files/helpYACReader.html"); + + optionsDialog = new OptionsDialog(this); + connect(optionsDialog,SIGNAL(accepted()),viewer,SLOT(updateOptions())); + connect(optionsDialog,SIGNAL(fitToWidthRatioChanged(float)),viewer,SLOT(updateFitToWidthRatio(float))); + connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); + connect(optionsDialog,SIGNAL(changedFilters(int,int,int)),viewer,SLOT(updateFilters(int,int,int))); + + optionsDialog->restoreOptions(settings); + shortcutsDialog = new ShortcutsDialog(this); + editShortcutsDialog = new EditShortcutsDialog(this); + connect(optionsDialog,SIGNAL(editShortcuts()),editShortcutsDialog,SLOT(show())); + + createActions(); + setUpShortcutsManagement(); + + createToolBars(); + + setWindowTitle("YACReader"); + + openFromArgv(); + + checkNewVersion(); + + viewer->setFocusPolicy(Qt::StrongFocus); + + + //if(Configuration::getConfiguration().getAlwaysOnTop()) + //{ + // setWindowFlags(this->windowFlags() | Qt::CustomizeWindowHint | Qt::WindowStaysOnTopHint); + //} + + if(fullscreen) + toFullScreen(); + if(conf.getMaximized()) + showMaximized(); + + setAcceptDrops(true); + + if(Configuration::getConfiguration().getShowToolbars() && !Configuration::getConfiguration().getFullScreen()) + showToolBars(); + else + hideToolBars(); +} + +void MainWindowViewer::openFromArgv() +{ + if(QCoreApplication::arguments().count() == 2) //only path... + { + isClient = false; + //TODO: new method open(QString) + QString pathFile = QCoreApplication::arguments().at(1); + QFileInfo fi(pathFile); + currentDirectory = fi.absoluteDir().path(); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + enableActions(); + viewer->open(pathFile); + } + else if(QCoreApplication::arguments().count() == 4) + { + + QString pathFile = QCoreApplication::arguments().at(1); + currentDirectory = pathFile; + quint64 comicId = QCoreApplication::arguments().at(2).toULongLong(); + libraryId = QCoreApplication::arguments().at(3).toULongLong(); + + enableActions(); + + currentComicDB.id = comicId; + YACReaderLocalClient client; + int tries = 1; + bool success = false; + while(!(success = client.requestComicInfo(libraryId,currentComicDB,siblingComics)) && tries != 0) + tries--; + + if(success) + { + isClient = true; + open(pathFile+currentComicDB.path,currentComicDB,siblingComics); + } + else + {isClient = false; QMessageBox::information(this,"Connection Error", "Unable to connect to YACReaderLibrary");/*error*/} + + optionsDialog->setFilters(currentComicDB.info.brightness, currentComicDB.info.contrast, currentComicDB.info.gamma); + } +} + +void MainWindowViewer::createActions() +{ + openAction = new QAction(tr("&Open"),this); + openAction->setIcon(QIcon(":/images/viewer_toolbar/open.png")); + openAction->setToolTip(tr("Open a comic")); + openAction->setData(OPEN_ACTION_Y); + openAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_ACTION_Y)); + connect(openAction, SIGNAL(triggered()), this, SLOT(open())); + + openFolderAction = new QAction(tr("Open Folder"),this); + openFolderAction->setIcon(QIcon(":/images/viewer_toolbar/openFolder.png")); + openFolderAction->setToolTip(tr("Open image folder")); + openFolderAction->setData(OPEN_FOLDER_ACTION_Y); + openFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_FOLDER_ACTION_Y)); + connect(openFolderAction, SIGNAL(triggered()), this, SLOT(openFolder())); + + saveImageAction = new QAction(tr("Save"),this); + saveImageAction->setIcon(QIcon(":/images/viewer_toolbar/save.png")); + saveImageAction->setToolTip(tr("Save current page")); + saveImageAction->setDisabled(true); + saveImageAction->setData(SAVE_IMAGE_ACTION_Y); + saveImageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SAVE_IMAGE_ACTION_Y)); + connect(saveImageAction,SIGNAL(triggered()),this,SLOT(saveImage())); + + openPreviousComicAction = new QAction(tr("Previous Comic"),this); + openPreviousComicAction->setIcon(QIcon(":/images/viewer_toolbar/openPrevious.png")); + openPreviousComicAction->setToolTip(tr("Open previous comic")); + openPreviousComicAction->setDisabled(true); + openPreviousComicAction->setData(OPEN_PREVIOUS_COMIC_ACTION_Y); + openPreviousComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_PREVIOUS_COMIC_ACTION_Y)); + connect(openPreviousComicAction,SIGNAL(triggered()),this,SLOT(openPreviousComic())); + + openNextComicAction = new QAction(tr("Next Comic"),this); + openNextComicAction->setIcon(QIcon(":/images/viewer_toolbar/openNext.png")); + openNextComicAction->setToolTip(tr("Open next comic")); + openNextComicAction->setDisabled(true); + openNextComicAction->setData(OPEN_NEXT_COMIC_ACTION_Y); + openNextComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_NEXT_COMIC_ACTION_Y)); + connect(openNextComicAction,SIGNAL(triggered()),this,SLOT(openNextComic())); + + prevAction = new QAction(tr("&Previous"),this); + prevAction->setIcon(QIcon(":/images/viewer_toolbar/previous.png")); + prevAction->setShortcutContext(Qt::WidgetShortcut); + prevAction->setToolTip(tr("Go to previous page")); + prevAction->setDisabled(true); + prevAction->setData(PREV_ACTION_Y); + prevAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(PREV_ACTION_Y)); + connect(prevAction, SIGNAL(triggered()),viewer,SLOT(prev())); + + nextAction = new QAction(tr("&Next"),this); + nextAction->setIcon(QIcon(":/images/viewer_toolbar/next.png")); + nextAction->setShortcutContext(Qt::WidgetShortcut); + nextAction->setToolTip(tr("Go to next page")); + nextAction->setDisabled(true); + nextAction->setData(NEXT_ACTION_Y); + nextAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(NEXT_ACTION_Y)); + connect(nextAction, SIGNAL(triggered()),viewer,SLOT(next())); + + adjustHeightAction = new QAction(tr("Fit Height"),this); + adjustHeightAction->setIcon(QIcon(":/images/viewer_toolbar/toHeight.png")); + //adjustWidth->setCheckable(true); + adjustHeightAction->setDisabled(true); + adjustHeightAction->setChecked(Configuration::getConfiguration().getAdjustToWidth()); + adjustHeightAction->setToolTip(tr("Fit image to height")); + //adjustWidth->setIcon(QIcon(":/images/fitWidth.png")); + adjustHeightAction->setData(ADJUST_HEIGHT_ACTION_Y); + adjustHeightAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_HEIGHT_ACTION_Y)); + connect(adjustHeightAction, SIGNAL(triggered()),this,SLOT(fitToHeight())); + + adjustWidthAction = new QAction(tr("Fit Width"),this); + adjustWidthAction->setIcon(QIcon(":/images/viewer_toolbar/toWidth.png")); + //adjustWidth->setCheckable(true); + adjustWidthAction->setDisabled(true); + adjustWidthAction->setChecked(Configuration::getConfiguration().getAdjustToWidth()); + adjustWidthAction->setToolTip(tr("Fit image to width")); + //adjustWidth->setIcon(QIcon(":/images/fitWidth.png")); + adjustWidthAction->setData(ADJUST_WIDTH_ACTION_Y); + adjustWidthAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_WIDTH_ACTION_Y)); + connect(adjustWidthAction, SIGNAL(triggered()),this,SLOT(fitToWidth())); + + leftRotationAction = new QAction(tr("Rotate image to the left"),this); + leftRotationAction->setIcon(QIcon(":/images/viewer_toolbar/rotateL.png")); + leftRotationAction->setDisabled(true); + leftRotationAction->setData(LEFT_ROTATION_ACTION_Y); + leftRotationAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(LEFT_ROTATION_ACTION_Y)); + connect(leftRotationAction, SIGNAL(triggered()),viewer,SLOT(rotateLeft())); + + rightRotationAction = new QAction(tr("Rotate image to the right"),this); + rightRotationAction->setIcon(QIcon(":/images/viewer_toolbar/rotateR.png")); + rightRotationAction->setDisabled(true); + rightRotationAction->setData(RIGHT_ROTATION_ACTION_Y); + rightRotationAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RIGHT_ROTATION_ACTION_Y)); + connect(rightRotationAction, SIGNAL(triggered()),viewer,SLOT(rotateRight())); + + doublePageAction = new QAction(tr("Double page mode"),this); + doublePageAction->setToolTip(tr("Switch to double page mode")); + doublePageAction->setIcon(QIcon(":/images/viewer_toolbar/doublePage.png")); + doublePageAction->setDisabled(true); + doublePageAction->setCheckable(true); + doublePageAction->setChecked(Configuration::getConfiguration().getDoublePage()); + doublePageAction->setData(DOUBLE_PAGE_ACTION_Y); + doublePageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(DOUBLE_PAGE_ACTION_Y)); + connect(doublePageAction, SIGNAL(triggered()),viewer,SLOT(doublePageSwitch())); + + goToPageAction = new QAction(tr("Go To"),this); + goToPageAction->setIcon(QIcon(":/images/viewer_toolbar/goto.png")); + goToPageAction->setDisabled(true); + goToPageAction->setToolTip(tr("Go to page ...")); + goToPageAction->setData(GO_TO_PAGE_ACTION_Y); + goToPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_PAGE_ACTION_Y)); + connect(goToPageAction, SIGNAL(triggered()),viewer,SLOT(showGoToDialog())); + + optionsAction = new QAction(tr("Options"),this); + optionsAction->setToolTip(tr("YACReader options")); + optionsAction->setData(OPTIONS_ACTION_Y); + optionsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPTIONS_ACTION_Y)); + optionsAction->setIcon(QIcon(":/images/viewer_toolbar/options.png")); + + connect(optionsAction, SIGNAL(triggered()),optionsDialog,SLOT(show())); + + helpAboutAction = new QAction(tr("Help"),this); + helpAboutAction->setToolTip(tr("Help, About YACReader")); + helpAboutAction->setIcon(QIcon(":/images/viewer_toolbar/help.png")); + helpAboutAction->setData(HELP_ABOUT_ACTION_Y); + helpAboutAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HELP_ABOUT_ACTION_Y)); + connect(helpAboutAction, SIGNAL(triggered()),had,SLOT(show())); + + showMagnifyingGlassAction = new QAction(tr("Magnifying glass"),this); + showMagnifyingGlassAction->setToolTip(tr("Switch Magnifying glass")); + showMagnifyingGlassAction->setIcon(QIcon(":/images/viewer_toolbar/magnifyingGlass.png")); + showMagnifyingGlassAction->setDisabled(true); + showMagnifyingGlassAction->setCheckable(true); + showMagnifyingGlassAction->setData(SHOW_MAGNIFYING_GLASS_ACTION_Y); + showMagnifyingGlassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_MAGNIFYING_GLASS_ACTION_Y)); + connect(showMagnifyingGlassAction, SIGNAL(triggered()),viewer,SLOT(magnifyingGlassSwitch())); + + setBookmarkAction = new QAction(tr("Set bookmark"),this); + setBookmarkAction->setToolTip(tr("Set a bookmark on the current page")); + setBookmarkAction->setIcon(QIcon(":/images/viewer_toolbar/bookmark.png")); + setBookmarkAction->setDisabled(true); + setBookmarkAction->setCheckable(true); + setBookmarkAction->setData(SET_BOOKMARK_ACTION_Y); + setBookmarkAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_BOOKMARK_ACTION_Y)); + connect(setBookmarkAction,SIGNAL(triggered (bool)),viewer,SLOT(setBookmarkAction(bool))); + connect(viewer,SIGNAL(pageAvailable(bool)),setBookmarkAction,SLOT(setEnabled(bool))); + connect(viewer,SIGNAL(pageIsBookmark(bool)),setBookmarkAction,SLOT(setChecked(bool))); + + showBookmarksAction = new QAction(tr("Show bookmarks"),this); + showBookmarksAction->setToolTip(tr("Show the bookmarks of the current comic")); + showBookmarksAction->setIcon(QIcon(":/images/viewer_toolbar/showBookmarks.png")); + showBookmarksAction->setDisabled(true); + showBookmarksAction->setData(SHOW_BOOKMARKS_ACTION_Y); + showBookmarksAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_BOOKMARKS_ACTION_Y)); + connect(showBookmarksAction, SIGNAL(triggered()),viewer->getBookmarksDialog(),SLOT(show())); + + showShorcutsAction = new QAction(tr("Show keyboard shortcuts"), this ); + showShorcutsAction->setIcon(QIcon(":/images/viewer_toolbar/shortcuts.png")); + showShorcutsAction->setData(SHOW_SHORCUTS_ACTION_Y); + showShorcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_SHORCUTS_ACTION_Y)); + connect(showShorcutsAction, SIGNAL(triggered()),shortcutsDialog,SLOT(show())); + + showInfoAction = new QAction(tr("Show Info"),this); + showInfoAction->setIcon(QIcon(":/images/viewer_toolbar/info.png")); + showInfoAction->setDisabled(true); + showInfoAction->setData(SHOW_INFO_ACTION_Y); + showInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_INFO_ACTION_Y)); + connect(showInfoAction, SIGNAL(triggered()),viewer,SLOT(informationSwitch())); + + closeAction = new QAction(tr("Close"),this); + closeAction->setIcon(QIcon(":/images/viewer_toolbar/close.png")); + closeAction->setData(CLOSE_ACTION_Y); + closeAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CLOSE_ACTION_Y)); + connect(closeAction,SIGNAL(triggered()),this,SLOT(close())); + + showDictionaryAction = new QAction(tr("Show Dictionary"),this); + showDictionaryAction->setIcon(QIcon(":/images/viewer_toolbar/translator.png")); + //showDictionaryAction->setCheckable(true); + showDictionaryAction->setDisabled(true); + showDictionaryAction->setData(SHOW_DICTIONARY_ACTION_Y); + showDictionaryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_DICTIONARY_ACTION_Y)); + connect(showDictionaryAction,SIGNAL(triggered()),viewer,SLOT(translatorSwitch())); + + //deprecated + alwaysOnTopAction = new QAction(tr("Always on top"),this); + alwaysOnTopAction->setIcon(QIcon(":/images/alwaysOnTop.png")); + alwaysOnTopAction->setCheckable(true); + alwaysOnTopAction->setDisabled(true); + alwaysOnTopAction->setChecked(Configuration::getConfiguration().getAlwaysOnTop()); + alwaysOnTopAction->setData(ALWAYS_ON_TOP_ACTION_Y); + alwaysOnTopAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ALWAYS_ON_TOP_ACTION_Y)); + connect(alwaysOnTopAction,SIGNAL(triggered()),this,SLOT(alwaysOnTopSwitch())); + + adjustToFullSizeAction = new QAction(tr("Show full size"),this); + adjustToFullSizeAction->setIcon(QIcon(":/images/viewer_toolbar/full.png")); + adjustToFullSizeAction->setCheckable(true); + adjustToFullSizeAction->setDisabled(true); + adjustToFullSizeAction->setChecked(Configuration::getConfiguration().getAdjustToFullSize()); + adjustToFullSizeAction->setData(ADJUST_TO_FULL_SIZE_ACTION_Y); + adjustToFullSizeAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADJUST_TO_FULL_SIZE_ACTION_Y)); + connect(adjustToFullSizeAction,SIGNAL(triggered()),this,SLOT(adjustToFullSizeSwitch())); + + showFlowAction = new QAction(tr("Show go to flow"),this); + showFlowAction->setIcon(QIcon(":/images/viewer_toolbar/flow.png")); + showFlowAction->setDisabled(true); + showFlowAction->setData(SHOW_FLOW_ACTION_Y); + showFlowAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_FLOW_ACTION_Y)); + connect(showFlowAction,SIGNAL(triggered()),viewer,SLOT(goToFlowSwitch())); + + showEditShortcutsAction = new QAction(tr("Edit shortcuts"),this); + showEditShortcutsAction->setData(SHOW_EDIT_SHORTCUTS_ACTION_Y); + showEditShortcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_EDIT_SHORTCUTS_ACTION_Y)); + connect(showEditShortcutsAction,SIGNAL(triggered()),editShortcutsDialog,SLOT(show())); +} + +void MainWindowViewer::createToolBars() +{ + comicToolBar = addToolBar(tr("&File")); + +#ifdef Q_OS_MAC + comicToolBar->setIconSize(QSize(16,16)); +#else + comicToolBar->setIconSize(QSize(18,18)); + comicToolBar->setStyleSheet("QToolBar{border:none;}"); +#endif + + QToolButton * tb = new QToolButton(); + tb->addAction(openAction); + tb->addAction(openFolderAction); + tb->setPopupMode(QToolButton::MenuButtonPopup); + tb->setDefaultAction(openAction); + + comicToolBar->addWidget(tb); + comicToolBar->addAction(saveImageAction); + comicToolBar->addAction(openPreviousComicAction); + comicToolBar->addAction(openNextComicAction); +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addSeparator(); +#endif + comicToolBar->addAction(prevAction); + comicToolBar->addAction(nextAction); + comicToolBar->addAction(goToPageAction); + +//#ifndef Q_OS_MAC +// comicToolBar->addSeparator(); +// comicToolBar->addAction(alwaysOnTopAction); +//#else +// alwaysOnTopAction->setEnabled(false); +//#endif + +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addSeparator(); +#endif + + //QWidget * widget = new QWidget(); + + //QToolButton * tbW = new QToolButton(widget); + //tbW->addAction(adjustWidth); + //tbW->setPopupMode(QToolButton::MenuButtonPopup); + //tbW->setDefaultAction(adjustWidth); + + //QHBoxLayout *layout = new QHBoxLayout; + //layout->addWidget(tbW); + //layout->setContentsMargins(0,0,0,0); + //widget->setLayout(layout); + //widget->setContentsMargins(0,0,0,0); + + //comicToolBar->addWidget(widget); + + //comicToolBar->addAction(adjustWidth); + + + + QMenu * menu = new QMenu(); + sliderAction = new YACReaderSliderAction(this); + menu->setAutoFillBackground(false); + menu->setStyleSheet(" QMenu {background:transparent; border: 0px;padding: 0px; }" + ); + menu->addAction(sliderAction); + QToolButton * tb2 = new QToolButton(); + tb2->addAction(adjustWidthAction); + tb2->setMenu(menu); + + connect(sliderAction,SIGNAL(fitToWidthRatioChanged(float)),viewer,SLOT(updateFitToWidthRatio(float))); + connect(optionsDialog,SIGNAL(fitToWidthRatioChanged(float)),sliderAction,SLOT(updateFitToWidthRatio(float))); + + + //tb2->addAction(); + tb2->setPopupMode(QToolButton::MenuButtonPopup); + tb2->setDefaultAction(adjustWidthAction); + comicToolBar->addWidget(tb2); + comicToolBar->addAction(adjustHeightAction); + comicToolBar->addAction(adjustToFullSizeAction); + comicToolBar->addAction(leftRotationAction); + comicToolBar->addAction(rightRotationAction); + comicToolBar->addAction(doublePageAction); + +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addSeparator(); +#endif + comicToolBar->addAction(showMagnifyingGlassAction); + +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addSeparator(); +#endif + comicToolBar->addAction(setBookmarkAction); + comicToolBar->addAction(showBookmarksAction); + +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addSeparator(); +#endif + comicToolBar->addAction(showDictionaryAction); + comicToolBar->addAction(showFlowAction); + comicToolBar->addAction(showInfoAction); + +#ifdef Q_OS_MAC + comicToolBar->addWidget(new MacToolBarSeparator); +#else + comicToolBar->addWidget(new QToolBarStretch()); +#endif + + + comicToolBar->addAction(showShorcutsAction); + comicToolBar->addAction(optionsAction); + comicToolBar->addAction(helpAboutAction); + //comicToolBar->addAction(closeAction); + + comicToolBar->setMovable(false); + + + viewer->addAction(openAction); + viewer->addAction(openFolderAction); + viewer->addAction(saveImageAction); + viewer->addAction(openPreviousComicAction); + viewer->addAction(openNextComicAction); + YACReader::addSperator(viewer); + + viewer->addAction(prevAction); + viewer->addAction(nextAction); + viewer->addAction(goToPageAction); + viewer->addAction(adjustHeightAction); + viewer->addAction(adjustWidthAction); + viewer->addAction(adjustToFullSizeAction); + viewer->addAction(leftRotationAction); + viewer->addAction(rightRotationAction); + YACReader::addSperator(viewer); + + viewer->addAction(showMagnifyingGlassAction); + YACReader::addSperator(viewer); + + viewer->addAction(setBookmarkAction); + viewer->addAction(showBookmarksAction); + YACReader::addSperator(viewer); + + viewer->addAction(showDictionaryAction); + viewer->addAction(showFlowAction); + viewer->addAction(showInfoAction); + YACReader::addSperator(viewer); + + viewer->addAction(showShorcutsAction); + viewer->addAction(showEditShortcutsAction); + viewer->addAction(optionsAction); + viewer->addAction(helpAboutAction); + YACReader::addSperator(viewer); + + viewer->addAction(closeAction); + + viewer->setContextMenuPolicy(Qt::ActionsContextMenu); + + //MacOSX app menus +#ifdef Q_OS_MAC + QMenuBar * menuBar = this->menuBar(); + //about / preferences + //TODO + + //file + QMenu * fileMenu = new QMenu(tr("File")); + + fileMenu->addAction(openAction); + fileMenu->addAction(openFolderAction); + fileMenu->addSeparator(); + fileMenu->addAction(saveImageAction); + + //tool bar + //QMenu * toolbarMenu = new QMenu(tr("Toolbar")); + //toolbarMenu->addAction(); + //TODO + + menuBar->addMenu(fileMenu); + //menu->addMenu(toolbarMenu); +#endif + +} + +void MainWindowViewer::reloadOptions() +{ + viewer->updateConfig(settings); +} + +void MainWindowViewer::open() +{ + QFileDialog openDialog; + QString pathFile = openDialog.getOpenFileName(this,tr("Open Comic"),currentDirectory,tr("Comic files") + "(*.cbr *.cbz *.rar *.zip *.tar *.pdf *.7z *.cb7 *.arj *.cbt)"); + if (!pathFile.isEmpty()) + { + openComicFromPath(pathFile); + } +} + +void MainWindowViewer::open(QString path, ComicDB & comic, QList & siblings) +{ + //currentComicDB = comic; + //siblingComics = siblings; + + QFileInfo fi(path); + + if(!comic.info.title.isNull() && !comic.info.title.toString().isEmpty()) + setWindowTitle("YACReader - " + comic.info.title.toString()); + else + setWindowTitle("YACReader - " + fi.fileName()); + + viewer->open(path,comic); + enableActions(); + int index = siblings.indexOf(comic); + + optionsDialog->setFilters(currentComicDB.info.brightness, currentComicDB.info.contrast, currentComicDB.info.gamma); + + if(index>0) + openPreviousComicAction->setDisabled(false); + else + openPreviousComicAction->setDisabled(true); + + if(index+1setDisabled(false); + else + openNextComicAction->setDisabled(true); +} + +void MainWindowViewer::openComicFromPath(QString pathFile) +{ + QFileInfo fi(pathFile); + currentDirectory = fi.dir().absolutePath(); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + + enableActions(); + + viewer->open(pathFile); + + isClient = false; + +} + +void MainWindowViewer::openFolder() +{ + QFileDialog openDialog; + QString pathDir = openDialog.getExistingDirectory(this,tr("Open folder"),currentDirectory); + if (!pathDir.isEmpty()) + { + openFolderFromPath(pathDir); + isClient = false; + } +} + +void MainWindowViewer::openFolderFromPath(QString pathDir) +{ + currentDirectory = pathDir; //TODO ?? + QFileInfo fi(pathDir); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + + enableActions(); + + viewer->open(pathDir); +} + +void MainWindowViewer::openFolderFromPath(QString pathDir, QString atFileName) +{ + currentDirectory = pathDir; //TODO ?? + QFileInfo fi(pathDir); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + + enableActions(); + + QDir d(pathDir); + d.setFilter(QDir::Files|QDir::NoDotAndDotDot); + d.setNameFilters(Comic::getSupportedImageFormats()); + d.setSorting(QDir::Name|QDir::IgnoreCase|QDir::LocaleAware); + QStringList list = d.entryList(); + + qSort(list.begin(),list.end(),naturalSortLessThanCI); + int i = 0; + foreach(QString path,list) + { + if(path.endsWith(atFileName)) + break; + i++; + } + + int index = 0; + if(i < list.count()) + index = i; + + viewer->open(pathDir,i); +} + +void MainWindowViewer::saveImage() +{ + QFileDialog saveDialog; + QString pathFile = saveDialog.getSaveFileName(this,tr("Save current page"),currentDirectoryImgDest+"/"+tr("page_%1.jpg").arg(viewer->getIndex()),tr("Image files (*.jpg)")); + if (!pathFile.isEmpty()) + { + QFileInfo fi(pathFile); + currentDirectoryImgDest = fi.absolutePath(); + const QPixmap * p = viewer->pixmap(); + if(p!=NULL) + p->save(pathFile); + } +} + +void MainWindowViewer::enableActions() +{ + saveImageAction->setDisabled(false); + prevAction->setDisabled(false); + nextAction->setDisabled(false); + adjustHeightAction->setDisabled(false); + adjustWidthAction->setDisabled(false); + goToPageAction->setDisabled(false); + //alwaysOnTopAction->setDisabled(false); + leftRotationAction->setDisabled(false); + rightRotationAction->setDisabled(false); + showMagnifyingGlassAction->setDisabled(false); + doublePageAction->setDisabled(false); + adjustToFullSizeAction->setDisabled(false); + //setBookmark->setDisabled(false); + showBookmarksAction->setDisabled(false); + showInfoAction->setDisabled(false); //TODO enable goTo and showInfo (or update) when numPages emited + showDictionaryAction->setDisabled(false); + showFlowAction->setDisabled(false); +} +void MainWindowViewer::disableActions() +{ + saveImageAction->setDisabled(true); + prevAction->setDisabled(true); + nextAction->setDisabled(true); + adjustHeightAction->setDisabled(true); + adjustWidthAction->setDisabled(true); + goToPageAction->setDisabled(true); + //alwaysOnTopAction->setDisabled(true); + leftRotationAction->setDisabled(true); + rightRotationAction->setDisabled(true); + showMagnifyingGlassAction->setDisabled(true); + doublePageAction->setDisabled(true); + adjustToFullSizeAction->setDisabled(true); + setBookmarkAction->setDisabled(true); + showBookmarksAction->setDisabled(true); + showInfoAction->setDisabled(true); //TODO enable goTo and showInfo (or update) when numPages emited + openPreviousComicAction->setDisabled(true); + openNextComicAction->setDisabled(true); + showDictionaryAction->setDisabled(true); + showFlowAction->setDisabled(true); +} + +void MainWindowViewer::keyPressEvent(QKeyEvent *event) +{ + //TODO remove unused keys + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_Y)) + { + toggleFullScreen(); + event->accept(); + } + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)) + { + toggleToolBars(); + event->accept(); + } + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)) + { + changeFit(); + event->accept(); + } + else + QWidget::keyPressEvent(event); +} + +void MainWindowViewer::mouseDoubleClickEvent ( QMouseEvent * event ) +{ + toggleFullScreen(); + event->accept(); +} + +void MainWindowViewer::toggleFullScreen() +{ + fullscreen?toNormal():toFullScreen(); + Configuration::getConfiguration().setFullScreen(fullscreen = !fullscreen); +} + +void MainWindowViewer::toFullScreen() +{ + fromMaximized = this->isMaximized(); + + hideToolBars(); + viewer->hide(); + viewer->fullscreen = true;//TODO, change by the right use of windowState(); + showFullScreen(); + viewer->show(); + if(viewer->magnifyingGlassIsVisible()) + viewer->showMagnifyingGlass(); +} + +void MainWindowViewer::toNormal() +{ + //show all + viewer->hide(); + viewer->fullscreen = false;//TODO, change by the right use of windowState(); + //viewer->hideMagnifyingGlass(); + if(fromMaximized) + showMaximized(); + else + showNormal(); + + if(Configuration::getConfiguration().getShowToolbars()) + showToolBars(); + viewer->show(); + if(viewer->magnifyingGlassIsVisible()) + viewer->showMagnifyingGlass(); +} +void MainWindowViewer::toggleToolBars() +{ + toolbars?hideToolBars():showToolBars(); + + Configuration::getConfiguration().setShowToolbars(toolbars); + + comicToolBar->setMovable(false); +} +void MainWindowViewer::hideToolBars() +{ + //hide all + this->comicToolBar->hide(); + toolbars = false; +} + +void MainWindowViewer::showToolBars() +{ + this->comicToolBar->show(); + toolbars = true; +} +void MainWindowViewer::fitToWidth() +{ + Configuration & conf = Configuration::getConfiguration(); + if(!conf.getAdjustToWidth()) + { + conf.setAdjustToWidth(true); + viewer->updatePage(); + } +} +void MainWindowViewer::fitToHeight() +{ + Configuration & conf = Configuration::getConfiguration(); + if(conf.getAdjustToWidth()) + { + conf.setAdjustToWidth(false); + viewer->updatePage(); + } +} + +void MainWindowViewer::checkNewVersion() +{ + Configuration & conf = Configuration::getConfiguration(); + QDate lastCheck = conf.getLastVersionCheck(); + QDate current = QDate::currentDate(); + if(lastCheck.isNull() || lastCheck.daysTo(current) >= conf.getNumDaysBetweenVersionChecks()) + { + versionChecker = new HttpVersionChecker(); + + connect(versionChecker,SIGNAL(newVersionDetected()), + this,SLOT(newVersion())); + + QTimer * tT = new QTimer; + tT->setSingleShot(true); + connect(tT, SIGNAL(timeout()), versionChecker, SLOT(get())); + //versionChecker->get(); //TOD� + tT->start(100); + + conf.setLastVersionCheck(current); + } +} + +void MainWindowViewer::processReset() +{ + if(isClient) + { + if(siblingComics.count()>1) + { + bool openNextB = openNextComicAction->isEnabled(); + bool openPrevB = openPreviousComicAction->isEnabled(); + disableActions(); + openNextComicAction->setEnabled(openNextB); + openPreviousComicAction->setEnabled(openPrevB); + } + else + disableActions(); + } + else + disableActions(); +} + +void MainWindowViewer::setUpShortcutsManagement() +{ + //actions holder + QObject * orphanActions = new QObject; + + QList allActions; + QList tmpList; + + + editShortcutsDialog->addActionsGroup(tr("Comics"),QIcon(":/images/shortcuts_group_comics.png"), + tmpList = QList() + << openAction + << openFolderAction + << saveImageAction + << openPreviousComicAction + << openNextComicAction); + + allActions << tmpList; + + //keys without actions (General) + QAction * toggleFullScreenAction = new QAction(tr("Toggle fullscreen mode"),orphanActions); + toggleFullScreenAction->setData(TOGGLE_FULL_SCREEN_ACTION_Y); + toggleFullScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_Y)); + + QAction * toggleToolbarsAction = new QAction(tr("Hide/show toolbar"),orphanActions); + toggleToolbarsAction->setData(TOGGLE_TOOL_BARS_ACTION_Y); + toggleToolbarsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_TOOL_BARS_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("General"),QIcon(":/images/shortcuts_group_general.png"), + tmpList = QList() + << optionsAction + << helpAboutAction + << showShorcutsAction + << showInfoAction + << closeAction + << showDictionaryAction + << showFlowAction + << toggleFullScreenAction + << toggleToolbarsAction + << showEditShortcutsAction); + + allActions << tmpList; + + //keys without actions (MGlass) + QAction * sizeUpMglassAction = new QAction(tr("Size up magnifying glass"),orphanActions); + sizeUpMglassAction->setData(SIZE_UP_MGLASS_ACTION_Y); + sizeUpMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y)); + + QAction * sizeDownMglassAction = new QAction(tr("Size down magnifying glass"),orphanActions); + sizeDownMglassAction->setData(SIZE_DOWN_MGLASS_ACTION_Y); + sizeDownMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y)); + + QAction * zoomInMglassAction = new QAction(tr("Zoom in magnifying glass"),orphanActions); + zoomInMglassAction->setData(ZOOM_IN_MGLASS_ACTION_Y); + zoomInMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y)); + + QAction * zoomOutMglassAction = new QAction(tr("Zoom out magnifying glass"),orphanActions); + zoomOutMglassAction->setData(ZOOM_OUT_MGLASS_ACTION_Y); + zoomOutMglassAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Magnifiying glass"),QIcon(":/images/shortcuts_group_mglass.png"), + tmpList = QList() + << showMagnifyingGlassAction + << sizeUpMglassAction + << sizeDownMglassAction + << zoomInMglassAction + << zoomOutMglassAction); + + allActions << tmpList; + + //keys without actions + QAction * toggleFitToScreenAction = new QAction(tr("Toggle between fit to width and fit to height"),orphanActions); + toggleFitToScreenAction->setData(CHANGE_FIT_ACTION_Y); + toggleFitToScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CHANGE_FIT_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Page adjustement"),QIcon(":/images/shortcuts_group_page.png"), + tmpList = QList() + << adjustHeightAction + << adjustWidthAction + << toggleFitToScreenAction + << leftRotationAction + << rightRotationAction + << doublePageAction + << adjustToFullSizeAction); + + allActions << tmpList; + + QAction * autoScrollForwardAction = new QAction(tr("Autoscroll down"),orphanActions); + autoScrollForwardAction->setData(AUTO_SCROLL_FORWARD_ACTION_Y); + autoScrollForwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_FORWARD_ACTION_Y)); + + QAction * autoScrollBackwardAction = new QAction(tr("Autoscroll up"),orphanActions); + autoScrollBackwardAction->setData(AUTO_SCROLL_BACKWARD_ACTION_Y); + autoScrollBackwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_BACKWARD_ACTION_Y)); + + QAction * moveDownAction = new QAction(tr("Move down"),orphanActions); + moveDownAction->setData(MOVE_DOWN_ACTION_Y); + moveDownAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_DOWN_ACTION_Y)); + + QAction * moveUpAction = new QAction(tr("Move up"),orphanActions); + moveUpAction->setData(MOVE_UP_ACTION_Y); + moveUpAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_UP_ACTION_Y)); + + QAction * moveLeftAction = new QAction(tr("Move left"),orphanActions); + moveLeftAction->setData(MOVE_LEFT_ACTION_Y); + moveLeftAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_LEFT_ACTION_Y)); + + QAction * moveRightAction = new QAction(tr("Move right"),orphanActions); + moveRightAction->setData(MOVE_RIGHT_ACTION_Y); + moveRightAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(MOVE_RIGHT_ACTION_Y)); + + QAction * goToFirstPageAction = new QAction(tr("Go to the first page"),orphanActions); + goToFirstPageAction->setData(GO_TO_FIRST_PAGE_ACTION_Y); + goToFirstPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_FIRST_PAGE_ACTION_Y)); + + QAction * goToLastPageAction = new QAction(tr("Go to the last page"),orphanActions); + goToLastPageAction->setData(GO_TO_LAST_PAGE_ACTION_Y); + goToLastPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)); + + editShortcutsDialog->addActionsGroup(tr("Reading"),QIcon(":/images/shortcuts_group_reading.png"), + tmpList = QList() + << nextAction + << prevAction + << setBookmarkAction + << showBookmarksAction + << autoScrollForwardAction + << autoScrollBackwardAction + << moveDownAction + << moveUpAction + << moveLeftAction + << moveRightAction + << goToFirstPageAction + << goToLastPageAction + << goToPageAction); + + allActions << tmpList; + + ShortcutsManager::getShortcutsManager().registerActions(allActions); + +} + +void MainWindowViewer::changeFit() +{ + Configuration & conf = Configuration::getConfiguration(); + conf.setAdjustToWidth(!conf.getAdjustToWidth()); + viewer->updatePage(); +} + +void MainWindowViewer::newVersion() +{ + QMessageBox msgBox; + msgBox.setText(tr("There is a new version avaliable")); + msgBox.setInformativeText(tr("Do you want to download the new version?")); + msgBox.setStandardButtons(QMessageBox::Yes | QMessageBox::Ignore | QMessageBox::No); + msgBox.setDefaultButton(QMessageBox::Yes); + msgBox.button(QMessageBox::Ignore)->setText(tr("Remind me in 14 days")); + msgBox.button(QMessageBox::No)->setText(tr("Not now")); + msgBox.setWindowFlags(Qt::WindowStaysOnTopHint); + msgBox.setModal(true); + int ret = msgBox.exec(); + + switch(ret) + { + case QMessageBox::Yes: + QDesktopServices::openUrl(QUrl("http://www.yacreader.com")); + break; + case QMessageBox::No: + Configuration::getConfiguration().setNumDaysBetweenVersionChecks(1); + break; + case QMessageBox::Ignore: + Configuration::getConfiguration().setNumDaysBetweenVersionChecks(14); + break; + } +} + +void MainWindowViewer::closeEvent ( QCloseEvent * event ) +{ + Q_UNUSED(event) + + if(isClient) + sendComic(); + + viewer->save(); + Configuration & conf = Configuration::getConfiguration(); + if(!fullscreen && !isMaximized()) + { + conf.setPos(pos()); + conf.setSize(size()); + } + conf.setMaximized(isMaximized()); + + emit (closed()); +} + +void MainWindowViewer::openPreviousComic() +{ + if(!siblingComics.isEmpty() && isClient) + { + sendComic(); + + int currentIndex = siblingComics.indexOf(currentComicDB); + if (currentIndex == -1) + return; + if(currentIndex-1 >= 0 && currentIndex-1 < siblingComics.count()) + { + siblingComics[currentIndex] = currentComicDB; //updated + currentComicDB = siblingComics.at(currentIndex-1); + open(currentDirectory+currentComicDB.path,currentComicDB,siblingComics); + } + return; + } + if(!previousComicPath.isEmpty()) + { + viewer->open(previousComicPath); + QFileInfo fi(previousComicPath); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + } +} + +void MainWindowViewer::openNextComic() +{ + if(!siblingComics.isEmpty() && isClient) + { + sendComic(); + + int currentIndex = siblingComics.indexOf(currentComicDB); + if (currentIndex == -1) + return; + if(currentIndex+1 > 0 && currentIndex+1 < siblingComics.count()) + { + siblingComics[currentIndex] = currentComicDB; //updated + currentComicDB = siblingComics.at(currentIndex+1); + open(currentDirectory+currentComicDB.path,currentComicDB,siblingComics); + } + return; + } + if(!nextComicPath.isEmpty()) + { + viewer->open(nextComicPath); + QFileInfo fi(nextComicPath); + getSiblingComics(fi.absolutePath(),fi.fileName()); + + setWindowTitle("YACReader - " + fi.fileName()); + } +} + +void MainWindowViewer::getSiblingComics(QString path,QString currentComic) +{ + QDir d(path); + d.setFilter(QDir::Files|QDir::NoDotAndDotDot); + d.setNameFilters(QStringList() << "*.cbr" << "*.cbz" << "*.rar" << "*.zip" << "*.tar" << "*.pdf" << "*.7z" << "*.cb7" << "*.arj" << "*.cbt"); + d.setSorting(QDir::Name|QDir::IgnoreCase|QDir::LocaleAware); + QStringList list = d.entryList(); + qSort(list.begin(),list.end(),naturalSortLessThanCI); + //std::sort(list.begin(),list.end(),naturalSortLessThanCI); + int index = list.indexOf(currentComic); + if(index == -1) //comic not found + { + /*QFile f(QCoreApplication::applicationDirPath()+"/errorLog.txt"); + if(!f.open(QIODevice::WriteOnly)) + { + QMessageBox::critical(NULL,tr("Saving error log file...."),tr("There was a problem saving YACReader error log file. Please, check if you have enough permissions in the YACReader root folder.")); + } + else + { + QTextStream txtS(&f); + txtS << "METHOD : MainWindowViewer::getSiblingComics" << '\n'; + txtS << "ERROR : current comic not found in its own path" << '\n'; + txtS << path << '\n'; + txtS << currentComic << '\n'; + txtS << "Comic list count : " + list.count() << '\n'; + foreach(QString s, list){ + txtS << s << '\n'; + } + f.close(); + }*/ + } + + previousComicPath = nextComicPath = ""; + if(index>0) + { + previousComicPath = path+"/"+list.at(index-1); + openPreviousComicAction->setDisabled(false); + } + else + openPreviousComicAction->setDisabled(true); + + if(index+1setDisabled(false); + } + else + openNextComicAction->setDisabled(true); +} + +void MainWindowViewer::dropEvent(QDropEvent *event) +{ + QList urlList; + QString fName; + QFileInfo info; + + if (event->mimeData()->hasUrls()) + { + urlList = event->mimeData()->urls(); + + if ( urlList.size() > 0 ) + { + fName = urlList[0].toLocalFile(); // convert first QUrl to local path + info.setFile( fName ); // information about file + if (info.isFile()) + { + QStringList imageSuffixs = Comic::getSupportedImageLiteralFormats(); + if(imageSuffixs.contains("."+info.suffix())) //image dropped + openFolderFromPath(info.absoluteDir().absolutePath(),info.fileName()); + else + openComicFromPath(fName); // if is file, setText + } + else + if(info.isDir()) + openFolderFromPath(fName); + + isClient = false; + } + } + + event->acceptProposedAction(); +} +void MainWindowViewer::dragEnterEvent(QDragEnterEvent *event) +{ + // accept just text/uri-list mime format + if (event->mimeData()->hasFormat("text/uri-list")) + { + event->acceptProposedAction(); + isClient = false; + } +} + +void MainWindowViewer::alwaysOnTopSwitch() +{ + if(!Configuration::getConfiguration().getAlwaysOnTop()) + { + setWindowFlags(this->windowFlags() | Qt::CustomizeWindowHint | Qt::WindowStaysOnTopHint); //always on top + show(); + } + else + { + setWindowFlags(this->windowFlags() ^ (Qt::CustomizeWindowHint | Qt::WindowStaysOnTopHint)); + show(); + } + Configuration::getConfiguration().setAlwaysOnTop(!Configuration::getConfiguration().getAlwaysOnTop()); +} + +void MainWindowViewer::adjustToFullSizeSwitch() +{ + Configuration::getConfiguration().setAdjustToFullSize(!Configuration::getConfiguration().getAdjustToFullSize()); + viewer->updatePage(); +} + +void MainWindowViewer::sendComic() +{ + YACReaderLocalClient * client = new YACReaderLocalClient; + currentComicDB.info.hasBeenOpened = true; + viewer->updateComic(currentComicDB); + int retries = 1; + while(!client->sendComicInfo(libraryId,currentComicDB) && retries!=0) + retries--; + connect(client,SIGNAL(finished()),client,SLOT(deleteLater())); + //delete client; +} diff --git a/YACReader/main_window_viewer.h b/YACReader/main_window_viewer.h new file mode 100644 index 00000000..fedd1ca4 --- /dev/null +++ b/YACReader/main_window_viewer.h @@ -0,0 +1,145 @@ +#ifndef __MAIN_WINDOW_VIEWER_H +#define __MAIN_WINDOW_VIEWER_H +#include +#include +#include +#include +#include +#include +#include +#include + +#include "comic_db.h" + +class Comic; +class Viewer; +class OptionsDialog; +class HelpAboutDialog; +class HttpVersionChecker; +class ShortcutsDialog; +class YACReaderSliderAction; +class EditShortcutsDialog; + + class MainWindowViewer : public QMainWindow + { + Q_OBJECT + + public slots: + void open(); + void open(QString path, ComicDB & comic, QList & siblings); + void openFolder(); + void saveImage(); + void toggleToolBars(); + void hideToolBars(); + void showToolBars(); + void changeFit(); + void enableActions(); + void disableActions(); + void toggleFullScreen(); + void toFullScreen(); + void toNormal(); + void loadConfiguration(); + void newVersion(); + void openPreviousComic(); + void openNextComic(); + void openComicFromPath(QString pathFile); + void openFolderFromPath(QString pathDir); + void openFolderFromPath(QString pathFile, QString atFileName); + void alwaysOnTopSwitch(); + void adjustToFullSizeSwitch(); + void reloadOptions(); + void fitToWidth(); + void fitToHeight(); + void checkNewVersion(); + void processReset(); + void setUpShortcutsManagement(); + /*void viewComic(); + void prev(); + void next(); + void updatePage();*/ + + private: + //!State + bool fullscreen; + bool toolbars; + bool alwaysOnTop; + bool fromMaximized; + + QString currentDirectory; + QString currentDirectoryImgDest; + //!Widgets + Viewer * viewer; + //GoToDialog * goToDialog; + OptionsDialog * optionsDialog; + HelpAboutDialog * had; + ShortcutsDialog * shortcutsDialog; + EditShortcutsDialog * editShortcutsDialog; + + //! ToolBars + QToolBar *comicToolBar; + + //! Actions + QAction *openAction; + QAction *openFolderAction; + QAction *saveImageAction; + QAction *openPreviousComicAction; + QAction *openNextComicAction; + QAction *nextAction; + QAction *prevAction; + QAction *adjustWidthAction; + QAction *adjustHeightAction; + QAction *goToPageAction; + QAction *optionsAction; + QAction *helpAboutAction; + QAction *showMagnifyingGlassAction; + QAction *setBookmarkAction; + QAction *showBookmarksAction; + QAction *leftRotationAction; + QAction *rightRotationAction; + QAction *showInfoAction; + QAction *closeAction; + QAction *doublePageAction; + QAction *showShorcutsAction; + QAction *showDictionaryAction; + QAction *alwaysOnTopAction; + QAction *adjustToFullSizeAction; + QAction *showFlowAction; + + QAction *showEditShortcutsAction; + + YACReaderSliderAction * sliderAction; + + HttpVersionChecker * versionChecker; + QString previousComicPath; + QString nextComicPath; + //! Método que inicializa el interfaz. + void setupUI(); + void openFromArgv(); + void createActions(); + void createToolBars(); + void getSiblingComics(QString path,QString currentComic); + + //! Manejadores de evento: + void keyPressEvent(QKeyEvent *event); + //void resizeEvent(QResizeEvent * event); + void mouseDoubleClickEvent ( QMouseEvent * event ); + void dropEvent(QDropEvent *event); + void dragEnterEvent(QDragEnterEvent *event); + + QSettings * settings; + + ComicDB currentComicDB; + QList siblingComics; + bool isClient; + quint64 libraryId; +signals: + void closed(); + protected: + virtual void closeEvent ( QCloseEvent * event ); + void sendComic(); + public: + MainWindowViewer(); + ~MainWindowViewer(); + +}; +#endif diff --git a/YACReader/notifications_label_widget.cpp b/YACReader/notifications_label_widget.cpp new file mode 100644 index 00000000..33810644 --- /dev/null +++ b/YACReader/notifications_label_widget.cpp @@ -0,0 +1,83 @@ +#include "notifications_label_widget.h" + +#include +#include +#include +#include + +NotificationsLabelWidget::NotificationsLabelWidget(QWidget * parent) + :QWidget(parent) +{ + setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + effect = new QGraphicsOpacityEffect(this); + effect->setOpacity(1.0); + + effect2= new QGraphicsOpacityEffect(this); + effect->setOpacity(1.0); + + anim = new QPropertyAnimation(effect,"opacity"); + anim->setDuration(500); + anim->setStartValue(1.0); + anim->setEndValue(0.0); + anim->setEasingCurve(QEasingCurve::InExpo); + + anim2 = new QPropertyAnimation(effect2,"opacity"); + anim2->setDuration(500); + anim2->setStartValue(1.0); + anim2->setEndValue(0.0); + anim2->setEasingCurve(QEasingCurve::InExpo); + anim2->start(); + + connect(anim,SIGNAL(finished()),this,SLOT(hide())); + + imgLabel = new QLabel(this); + QPixmap p(":/images/notificationsLabel.png"); + imgLabel->resize(p.size()); + imgLabel->setPixmap(p); + imgLabel->setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + + textLabel = new QLabel(this); + textLabel->setAlignment(Qt::AlignVCenter|Qt::AlignHCenter); + textLabel->setStyleSheet("QLabel { color : white; font-size:24px; }"); + textLabel->setAttribute(Qt::WA_LayoutUsesWidgetRect,true); + + textLabel->setGeometry(imgLabel->geometry()); +#ifndef Q_OS_MAC + imgLabel->setGraphicsEffect(effect); + textLabel->setGraphicsEffect(effect2); +#endif + resize(p.size()); + updatePosition(); + +} + +void NotificationsLabelWidget::flash() +{ + updatePosition(); + anim->stop(); + anim2->stop(); + anim->start(); + anim2->start(); + + setVisible(true); +} + +void NotificationsLabelWidget::setText(const QString & text) +{ + textLabel->setText(text); + QRect geom = imgLabel->geometry(); + QSize size = geom.size(); + size.setHeight(size.height() - 10); //TODO remove this amazing magic number + geom.setSize(size); + textLabel->setGeometry(geom); +} + +void NotificationsLabelWidget::updatePosition() +{ + QWidget * parent = dynamic_cast(this->parent()); + if(parent == 0) + { + return; + } + move(QPoint((parent->geometry().size().width()-this->width())/2,(parent->geometry().size().height()-this->height())/2)); +} \ No newline at end of file diff --git a/YACReader/notifications_label_widget.h b/YACReader/notifications_label_widget.h new file mode 100644 index 00000000..08265d4a --- /dev/null +++ b/YACReader/notifications_label_widget.h @@ -0,0 +1,29 @@ +#ifndef NOTIFICATIONS_LABEL_WIDGET_H +#define NOTIFICATIONS_LABEL_WIDGET_H + +#include + +class QLabel; +class QPropertyAnimation; +class QGraphicsOpacityEffect; + +class NotificationsLabelWidget : public QWidget +{ +Q_OBJECT +private: + QLabel * imgLabel; + QLabel * textLabel; + QPropertyAnimation * anim; + QPropertyAnimation * anim2; + QGraphicsOpacityEffect * effect; + QGraphicsOpacityEffect * effect2; +public: + NotificationsLabelWidget(QWidget * parent); + +public slots: + void flash(); + void setText(const QString & text); + void updatePosition(); +}; + +#endif \ No newline at end of file diff --git a/YACReader/options_dialog.cpp b/YACReader/options_dialog.cpp new file mode 100644 index 00000000..37d25792 --- /dev/null +++ b/YACReader/options_dialog.cpp @@ -0,0 +1,307 @@ +#include "options_dialog.h" +#include "configuration.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_spin_slider_widget.h" +#include "yacreader_flow_config_widget.h" +#include "yacreader_gl_flow_config_widget.h" + +OptionsDialog::OptionsDialog(QWidget * parent) +:YACReaderOptionsDialog(parent) +{ + + QTabWidget * tabWidget = new QTabWidget(); + + QVBoxLayout * layout = new QVBoxLayout(this); + + QWidget * pageGeneral = new QWidget(); + QWidget * pageFlow = new QWidget(); + QWidget * pageImage = new QWidget(); + QVBoxLayout * layoutGeneral = new QVBoxLayout(); + QVBoxLayout * layoutFlow = new QVBoxLayout(); + QVBoxLayout * layoutImageV = new QVBoxLayout(); + QGridLayout * layoutImage = new QGridLayout(); + + QGroupBox *slideSizeBox = new QGroupBox(tr("\"Go to flow\" size")); + //slideSizeLabel = new QLabel(,this); + slideSize = new QSlider(this); + slideSize->setMinimum(125); + slideSize->setMaximum(350); + slideSize->setPageStep(5); + slideSize->setOrientation(Qt::Horizontal); + QHBoxLayout * slideLayout = new QHBoxLayout(); + slideLayout->addWidget(slideSize); + slideSizeBox->setLayout(slideLayout); + + QGroupBox *pathBox = new QGroupBox(tr("My comics path")); + + QHBoxLayout * path = new QHBoxLayout(); + path->addWidget(pathEdit = new QLineEdit()); + path->addWidget(pathFindButton = new QPushButton(QIcon(":/images/comicFolder.png"),"")); + pathBox->setLayout(path); + + connect(pathFindButton,SIGNAL(clicked()),this,SLOT(findFolder())); + + //fitToWidthRatioLabel = new QLabel(tr("Page width stretch"),this); + QGroupBox *fitBox = new QGroupBox(tr("Page width stretch")); + fitToWidthRatioS = new QSlider(this); + fitToWidthRatioS->setMinimum(50); + fitToWidthRatioS->setMaximum(100); + fitToWidthRatioS->setPageStep(5); + fitToWidthRatioS->setOrientation(Qt::Horizontal); + //connect(fitToWidthRatioS,SIGNAL(valueChanged(int)),this,SLOT(fitToWidthRatio(int))); + QHBoxLayout * fitLayout = new QHBoxLayout; + fitLayout->addWidget(fitToWidthRatioS); + fitBox->setLayout(fitLayout); + + QHBoxLayout * colorSelection = new QHBoxLayout; + backgroundColor = new QLabel(); + QPalette pal = backgroundColor->palette(); + pal.setColor(backgroundColor->backgroundRole(), Qt::black); + backgroundColor->setPalette(pal); + backgroundColor->setAutoFillBackground(true); + + colorDialog = new QColorDialog(Qt::red,this); + connect(colorDialog,SIGNAL(colorSelected(QColor)),this,SLOT(updateColor(QColor))); + + QGroupBox *colorBox = new QGroupBox(tr("Background color")); + //backgroundColor->setMinimumWidth(100); + colorSelection->addWidget(backgroundColor); + colorSelection->addWidget(selectBackgroundColorButton = new QPushButton(tr("Choose"))); + colorSelection->setStretchFactor(backgroundColor,1); + colorSelection->setStretchFactor(selectBackgroundColorButton,0); + //colorSelection->addStretch(); + connect(selectBackgroundColorButton, SIGNAL(clicked()), colorDialog, SLOT(show())); + colorBox->setLayout(colorSelection); + + brightnessS = new YACReaderSpinSliderWidget(this,true); + brightnessS->setRange(0,100); + //brightnessS->setText(tr("Brightness")); + brightnessS->setTracking(false); + connect(brightnessS,SIGNAL(valueChanged(int)),this,SLOT(brightnessChanged(int))); + + contrastS = new YACReaderSpinSliderWidget(this,true); + contrastS->setRange(0,250); + //contrastS->setText(tr("Contrast")); + contrastS->setTracking(false); + connect(contrastS,SIGNAL(valueChanged(int)),this,SLOT(contrastChanged(int))); + + gammaS = new YACReaderSpinSliderWidget(this,true); + gammaS->setRange(0,250); + //gammaS->setText(tr("Gamma")); + gammaS->setTracking(false); + connect(gammaS,SIGNAL(valueChanged(int)),this,SLOT(gammaChanged(int))); + //connect(brightnessS,SIGNAL(valueChanged(int)),this,SIGNAL(changedOptions())); + + QHBoxLayout * buttons = new QHBoxLayout(); + buttons->addStretch(); + buttons->addWidget(new QLabel(tr("Restart is needed"))); + buttons->addWidget(accept); + buttons->addWidget(cancel); + + layoutGeneral->addWidget(pathBox); + layoutGeneral->addWidget(slideSizeBox); + layoutGeneral->addWidget(fitBox); + layoutGeneral->addWidget(colorBox); + layoutGeneral->addWidget(shortcutsBox); + layoutGeneral->addStretch(); + layoutFlow->addWidget(sw); + layoutFlow->addWidget(gl); + layoutFlow->addWidget(useGL); + layoutFlow->addStretch(); + layoutImage->addWidget(new QLabel(tr("Brightness")),0,0); + layoutImage->addWidget(new QLabel(tr("Contrast")),1,0); + layoutImage->addWidget(new QLabel(tr("Gamma")),2,0); + layoutImage->addWidget(brightnessS,0,1); + layoutImage->addWidget(contrastS,1,1); + layoutImage->addWidget(gammaS,2,1); + QPushButton * pushButton = new QPushButton(tr("Reset")); + connect(pushButton,SIGNAL(pressed()),this,SLOT(resetImageConfig())); + layoutImage->addWidget(pushButton,3,0); + layoutImage->setColumnStretch(1,1); + + + QGroupBox *imageBox = new QGroupBox(tr("Image options")); + imageBox->setLayout(layoutImage); + layoutImageV->addWidget(imageBox); + layoutImageV->addStretch(); + + + pageGeneral->setLayout(layoutGeneral); + pageFlow->setLayout(layoutFlow); + pageImage->setLayout(layoutImageV); + + tabWidget->addTab(pageGeneral,tr("General")); + tabWidget->addTab(pageFlow,tr("Page Flow")); + tabWidget->addTab(pageImage,tr("Image adjustment")); + + layout->addWidget(tabWidget); + layout->addLayout(buttons); + + setLayout(layout); + + //disable vSyncCheck + gl->vSyncCheck->hide(); + + //restoreOptions(); //load options + //resize(400,0); + setModal (true); + setWindowTitle(tr("Options")); + + this->layout()->setSizeConstraint(QLayout::SetFixedSize); +} + +void OptionsDialog::findFolder() +{ + QString s = QFileDialog::getExistingDirectory(0,tr("Comics directory"),"."); + if(!s.isEmpty()) + { + pathEdit->setText(s); + } +} + +void OptionsDialog::saveOptions() +{ + + settings->setValue(GO_TO_FLOW_SIZE,QSize(static_cast(slideSize->sliderPosition()/SLIDE_ASPECT_RATIO),slideSize->sliderPosition())); + + if(sw->radio1->isChecked()) + settings->setValue(FLOW_TYPE_SW,0); + if(sw->radio2->isChecked()) + settings->setValue(FLOW_TYPE_SW,1); + if(sw->radio3->isChecked()) + settings->setValue(FLOW_TYPE_SW,2); + + settings->setValue(PATH,pathEdit->text()); + + settings->setValue(BACKGROUND_COLOR,colorDialog->currentColor()); + settings->setValue(FIT_TO_WIDTH_RATIO,fitToWidthRatioS->sliderPosition()/100.0); + + YACReaderOptionsDialog::saveOptions(); +} + +void OptionsDialog::restoreOptions(QSettings * settings) +{ + YACReaderOptionsDialog::restoreOptions(settings); + + slideSize->setSliderPosition(settings->value(GO_TO_FLOW_SIZE).toSize().height()); + switch(settings->value(FLOW_TYPE_SW).toInt()) + { + case 0: + sw->radio1->setChecked(true); + break; + case 1: + sw->radio2->setChecked(true); + break; + case 2: + sw->radio3->setChecked(true); + break; + default: + sw->radio1->setChecked(true); + break; + } + + pathEdit->setText(settings->value(PATH).toString()); + + updateColor(settings->value(BACKGROUND_COLOR).value()); + fitToWidthRatioS->setSliderPosition(settings->value(FIT_TO_WIDTH_RATIO).toFloat()*100); + + brightnessS->setValue(settings->value(BRIGHTNESS,0).toInt()); + contrastS->setValue(settings->value(CONTRAST,100).toInt()); + gammaS->setValue(settings->value(GAMMA,100).toInt()); +} + + +void OptionsDialog::updateColor(const QColor & color) +{ + QPalette pal = backgroundColor->palette(); + pal.setColor(backgroundColor->backgroundRole(), color); + backgroundColor->setPalette(pal); + backgroundColor->setAutoFillBackground(true); + colorDialog->setCurrentColor(color); + + settings->setValue(BACKGROUND_COLOR,color); + + emit(changedOptions()); +} + +void OptionsDialog::fitToWidthRatio(int value) +{ + Configuration::getConfiguration().setFitToWidthRatio(value/100.0); + emit(fitToWidthRatioChanged(value/100.0)); +} + +void OptionsDialog::brightnessChanged(int value) +{ + QSettings settings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + settings.setValue(BRIGHTNESS,value); + emit changedFilters(brightnessS->getValue(), contrastS->getValue(), gammaS->getValue()); + //emit(changedImageOptions()); +} + +void OptionsDialog::contrastChanged(int value) +{ + QSettings settings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + settings.setValue(CONTRAST,value); + emit changedFilters(brightnessS->getValue(), contrastS->getValue(), gammaS->getValue()); + ///emit(changedImageOptions()); +} + +void OptionsDialog::gammaChanged(int value) +{ + QSettings settings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + settings.setValue(GAMMA,value); + emit changedFilters(brightnessS->getValue(), contrastS->getValue(), gammaS->getValue()); + //emit(changedImageOptions()); +} + +void OptionsDialog::resetImageConfig() +{ + brightnessS->setValue(0); + contrastS->setValue(100); + gammaS->setValue(100); + QSettings settings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + settings.setValue(BRIGHTNESS,0); + settings.setValue(CONTRAST,100); + settings.setValue(GAMMA,100); + emit changedFilters(brightnessS->getValue(), contrastS->getValue(), gammaS->getValue()); + //emit(changedImageOptions()); +} + +void OptionsDialog::show() +{ + //TODO solucionar el tema de las settings, esto sólo debería aparecer en una única línea de código + QSettings *s = new QSettings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + fitToWidthRatioS->disconnect(); + fitToWidthRatioS->setSliderPosition(settings->value(FIT_TO_WIDTH_RATIO).toFloat()*100); + connect(fitToWidthRatioS,SIGNAL(valueChanged(int)),this,SLOT(fitToWidthRatio(int))); + QDialog::show(); + delete s; +} + +void OptionsDialog::setFilters(int brightness, int contrast, int gamma) +{ + if(brightness != -1) + brightnessS->setValue(brightness); + else + brightnessS->setValue(0); + if(contrast != -1) + contrastS->setValue(contrast); + else + contrastS->setValue(100); + if(gamma != -1) + gammaS->setValue(gamma); + else + gammaS->setValue(100); + +} diff --git a/YACReader/options_dialog.h b/YACReader/options_dialog.h new file mode 100644 index 00000000..3bf0c5ec --- /dev/null +++ b/YACReader/options_dialog.h @@ -0,0 +1,70 @@ +#ifndef __OPTIONS_DIALOG_H +#define __OPTIONS_DIALOG_H + +#include "yacreader_options_dialog.h" + +class QDialog; +class QLabel; +class QLineEdit; +class QPushButton; +class QSlider; +class QPushButton; +class QRadioButton; +class QColorDialog; +class YACReaderSpinSliderWidget; + + +class OptionsDialog : public YACReaderOptionsDialog +{ +Q_OBJECT + public: + OptionsDialog(QWidget * parent = 0); + private: + //QLabel * pathLabel; + QLineEdit * pathEdit; + QPushButton * pathFindButton; + + QLabel * magGlassSizeLabel; + + QLabel * zoomLevel; + + //QLabel * slideSizeLabel; + QSlider * slideSize; + + //QLabel * fitToWidthRatioLabel; + QSlider * fitToWidthRatioS; + + QLabel * backgroundColor; + QPushButton * selectBackgroundColorButton; + + QColorDialog * colorDialog; + + YACReaderSpinSliderWidget * brightnessS; + + YACReaderSpinSliderWidget * contrastS; + + YACReaderSpinSliderWidget * gammaS; + + public slots: + void saveOptions(); + void restoreOptions(QSettings * settings); + void findFolder(); + void updateColor(const QColor & color); + void fitToWidthRatio(int value); + void brightnessChanged(int value); + void contrastChanged(int value); + void gammaChanged(int value); + void resetImageConfig(); + void show(); + void setFilters(int brightness, int contrast, int gamma); + +signals: + void changedOptions(); + void changedImageOptions(); + void changedFilters(int brightness, int contrast, int gamma); + void fitToWidthRatioChanged(float ratio); + +}; + + +#endif diff --git a/YACReader/page_label_widget.cpp b/YACReader/page_label_widget.cpp new file mode 100644 index 00000000..e7ec979c --- /dev/null +++ b/YACReader/page_label_widget.cpp @@ -0,0 +1,122 @@ +#include "page_label_widget.h" + +#include +#include +#include +#include +#include + +PageLabelWidget::PageLabelWidget(QWidget * parent) + :QWidget(parent) + { + animation = new QPropertyAnimation(this,"pos"); + animation->setDuration(150); + animation->setEndValue(QPoint((parent->geometry().size().width()-this->width()),-this->height())); + + int verticalRes = QApplication::desktop()->screenGeometry().height(); + + imgLabel = new QLabel(this); + QPixmap p; + if (verticalRes <= 1024) + p.load(":/images/numPagesLabel.png"); + else if (verticalRes <= 1200) + p.load(":/images/numPagesLabelMedium.png"); + else + p.load(":/images/numPagesLabelBig.png"); + imgLabel->resize(p.size()); + imgLabel->setPixmap(p); + + textLabel = new QLabel(this); + textLabel->setAlignment(Qt::AlignVCenter|Qt::AlignHCenter); + if(verticalRes <= 1024) + textLabel->setStyleSheet("QLabel { color : white; font-size:12px; padding-left:8px; }"); + else if (verticalRes <= 1200) + textLabel->setStyleSheet("QLabel { color : white; font-size:16px; padding-left:8px;}"); + else + textLabel->setStyleSheet("QLabel { color : white; font-size:20px; padding-left:8px; }"); + + //informationLabel->setAutoFillBackground(true); + //textLabel->setFont(QFont("courier new bold", 12)); + //textLabel->resize(100,25); + + resize(p.size()); + //por defecto aparece oculto + if(parent != 0) + move(QPoint((parent->geometry().size().width()-this->width()),-this->height())); + /*QSize size = textLabel->sizeHint(); + + int w = width(); // returns screen width + int h = height(); // returns screen height + int mw = size.width(); + int mh = size.height(); + int cw = (w-mw)/2; + int ch = 0; + textLabel->move(cw,ch);*/ + } + +void PageLabelWidget::show() +{ + if(this->pos().y() <= 0 && animation->state()!=QPropertyAnimation::Running) + { + QWidget * parent = dynamic_cast(this->parent()); + if(parent == 0) + { + return; + } + + QWidget::show(); + //connect(animation,SIGNAL(finished()),this,SLOT(QWidget::hide())); + animation->disconnect(); + + animation->setStartValue(QPoint((parent->geometry().size().width()-this->width()),-this->height())); + animation->setEndValue(QPoint((parent->geometry().size().width()-this->width()),0)); + animation->start(); + } +} + +void PageLabelWidget::hide() +{ + + if(this->pos().y() >= 0 && animation->state()!=QPropertyAnimation::Running) + { + QWidget * parent = dynamic_cast(this->parent()); + if(parent == 0) + { + return; + } + //connect(animation,SIGNAL(finished()),this,SLOT(setHidden())); + animation->setStartValue(QPoint((parent->geometry().size().width()-this->width()),0)); + animation->setEndValue(QPoint((parent->geometry().size().width()-this->width()),-this->height())); + animation->start(); + } +} + +void PageLabelWidget::setText(const QString & text) +{ + textLabel->setText(text); + QRect geom = imgLabel->geometry(); + QSize size = geom.size(); + size.setHeight(size.height() - 10);//TODO remove this amazing magic number + geom.setSize(size); + textLabel->setGeometry(geom); +} + +/*void PageLabelWidget::resizeEvent(QResizeEvent * event) +{ + move(QPoint((((QWidget *) parent())->geometry().size().width()-this->width())/2,0)); +}*/ + +void PageLabelWidget::updatePosition() +{ + QWidget * parent = dynamic_cast(this->parent()); + if(parent == 0) + { + return; + } + + animation->stop(); + if (animation->endValue().toPoint().y() == 0) + move(QPoint((parent->geometry().size().width()-this->width()),0)); + else + move(QPoint((parent->geometry().size().width()-this->width()),-this->height())); +} diff --git a/YACReader/page_label_widget.h b/YACReader/page_label_widget.h new file mode 100644 index 00000000..315a1e4c --- /dev/null +++ b/YACReader/page_label_widget.h @@ -0,0 +1,29 @@ +#ifndef PAGE_LABEL_WIDGET_H +#define PAGE_LABEL_WIDGET_H + +#include + +class QLabel; +class QPropertyAnimation; + +class PageLabelWidget : public QWidget +{ +Q_OBJECT +private: + QLabel * imgLabel; + QLabel * textLabel; + QPropertyAnimation * animation; + + //void resizeEvent(QResizeEvent * event); + +public: + PageLabelWidget(QWidget * parent); + +public slots: + void show(); + void hide(); + void setText(const QString & text); + void updatePosition(); +}; + +#endif \ No newline at end of file diff --git a/YACReader/render.cpp b/YACReader/render.cpp new file mode 100644 index 00000000..81db733b --- /dev/null +++ b/YACReader/render.cpp @@ -0,0 +1,1084 @@ +#include "render.h" +#include +#include +#include +#include +#include +#include +#include + +#include + +#include "comic_db.h" +#include "yacreader_global.h" + +template +inline const T& kClamp( const T& x, const T& low, const T& high ) +{ + if ( x < low ) return low; + else if ( high < x ) return high; + else return x; +} + +inline +int changeBrightness( int value, int brightness ) + { + return kClamp( value + brightness * 255 / 100, 0, 255 ); + } + +inline +int changeContrast( int value, int contrast ) + { + return kClamp((( value - 127 ) * contrast / 100 ) + 127, 0, 255 ); + } + +inline +int changeGamma( int value, int gamma ) + { + return kClamp( int( pow( value / 255.0, 100.0 / gamma ) * 255 ), 0, 255 ); + } + +inline +int changeUsingTable( int value, const int table[] ) + { + return table[ value ]; + } + +template< int operation( int, int ) > +static +QImage changeImage( const QImage& image, int value ) + { + QImage im = image; + im.detach(); + if( im.colorCount() == 0 ) /* truecolor */ + { + if( im.format() != QImage::Format_RGB32 ) /* just in case */ + im = im.convertToFormat( QImage::Format_RGB32 ); + int table[ 256 ]; + for( int i = 0; + i < 256; + ++i ) + table[ i ] = operation( i, value ); + if( im.hasAlphaChannel() ) + { + for( int y = 0; + y < im.height(); + ++y ) + { + QRgb* line = reinterpret_cast< QRgb* >( im.scanLine( y )); + for( int x = 0; + x < im.width(); + ++x ) + line[ x ] = qRgba( changeUsingTable( qRed( line[ x ] ), table ), + changeUsingTable( qGreen( line[ x ] ), table ), + changeUsingTable( qBlue( line[ x ] ), table ), + changeUsingTable( qAlpha( line[ x ] ), table )); + } + } + else + { + for( int y = 0; + y < im.height(); + ++y ) + { + QRgb* line = reinterpret_cast< QRgb* >( im.scanLine( y )); + for( int x = 0; + x < im.width(); + ++x ) + line[ x ] = qRgb( changeUsingTable( qRed( line[ x ] ), table ), + changeUsingTable( qGreen( line[ x ] ), table ), + changeUsingTable( qBlue( line[ x ] ), table )); + } + } + } + else + { + QVector colors = im.colorTable(); + for( int i = 0; + i < im.colorCount(); + ++i ) + colors[ i ] = qRgb( operation( qRed( colors[ i ] ), value ), + operation( qGreen( colors[ i ] ), value ), + operation( qBlue( colors[ i ] ), value )); + } + return im; + } + + +// brightness is multiplied by 100 in order to avoid floating point numbers +QImage changeBrightness( const QImage& image, int brightness ) + { + if( brightness == 0 ) // no change + return image; + return changeImage< changeBrightness >( image, brightness ); + } + + +// contrast is multiplied by 100 in order to avoid floating point numbers +QImage changeContrast( const QImage& image, int contrast ) + { + if( contrast == 100 ) // no change + return image; + return changeImage< changeContrast >( image, contrast ); + } + +// gamma is multiplied by 100 in order to avoid floating point numbers +QImage changeGamma( const QImage& image, int gamma ) + { + if( gamma == 100 ) // no change + return image; + return changeImage< changeGamma >( image, gamma ); + } + + + +//----------------------------------------------------------------------------- +// MeanNoiseReductionFilter +//----------------------------------------------------------------------------- + +MeanNoiseReductionFilter::MeanNoiseReductionFilter(enum NeighborghoodSize ns) +:neighborghoodSize(ns) +{ + +} + +QImage MeanNoiseReductionFilter::setFilter(const QImage & image) +{ + int width = image.width(); + int height = image.height(); + QImage result(width,height,image.format()); + int filterSize = sqrt((float)neighborghoodSize); + int bound = filterSize/2; + QRgb pix; + int r,g,b; + for(int j=bound;j redChannel; + QList greenChannel; + QList blueChannel; + for(int j=bound;j hist(256,0); + + for(int j=0;j 1; i--) + { + new_count += hist[i]; + percentage = new_count/count; + next_percentage = (new_count+hist[i-1])/count; + if(fabs (percentage - 0.006) < fabs (next_percentage - 0.006)) + { + max = i-1; + break; + } + } + QColor c; + int range = max - min; + for(int j=0;j f) +:QThread(), +render(r), +numPage(np), +data(rd), +page(p), +degrees(d), +filters(f) +{ +} + +void PageRender::run() +{ + QMutexLocker locker(&(render->mutex)); + + QImage img; + img.loadFromData(data); + if(degrees > 0) + { + QMatrix m; + m.rotate(degrees); + img = img.transformed(m,Qt::SmoothTransformation); + } + for(int i=0;isetFilter(img); + } + + + *page = img; + + emit pageReady(numPage); +} + +//----------------------------------------------------------------------------- +// DoublePageRender +//----------------------------------------------------------------------------- + +DoublePageRender::DoublePageRender(Render * r, int np, const QByteArray & rd, const QByteArray & rd2, QImage * p,unsigned int d, QVector f) +:PageRender(), +render(r), +numPage(np), +data(rd), +data2(rd2), +page(p), +degrees(d), +filters(f) +{ + +} + +void DoublePageRender::run() +{ + //QImage result; + QMutexLocker locker(&(render->mutex)); + QImage img, img2; + if(!data.isEmpty()) + img.loadFromData(data); + if(!data2.isEmpty()) + img2.loadFromData(data2); + /*if(img.isNull()) + img = QPixmap(img2.width(),img2.height()); + if(img2.isNull()) + img2 = QPixmap(img.width(),img.height());*/ + + if(img.isNull() && !img2.isNull()) + { + img = img2; + img2 = QImage(); + } + + + int totalWidth,totalHeight; + //x = img.width()+img2.width(); + totalHeight = qMax(img.height(),img2.height()); + + //widths fiting the normalized height + int width1, width2; + + //altura normalizada + if(!img2.isNull()) + { + if(img.height()!=totalHeight) + totalWidth = (width1 = ((img.width() * totalHeight) / img.height())) + (width2 = img2.width()); + else + totalWidth = (width1 = img.width()) + (width2 = ((img2.width() * totalHeight) / img2.height())); + } + else + totalWidth = width1 = img.width(); + + + + + QImage auxImg(totalWidth,totalHeight,QImage::Format_RGB32); + QPainter painter(&auxImg); + painter.drawImage(QRect(0,0,width1,totalHeight),img); + if(!img2.isNull()) + painter.drawImage(QRect(width1,0,width2,totalHeight),img2); + painter.end(); + + if(degrees > 0) + { + QMatrix m; + m.rotate(degrees); + auxImg = auxImg.transformed(m,Qt::SmoothTransformation); + } + for(int i=0;isetFilter(auxImg); + } + + *page = auxImg; + + emit pageReady(numPage); +} + +//----------------------------------------------------------------------------- +// Render +//----------------------------------------------------------------------------- + +Render::Render() +:currentIndex(0),doublePage(false),comic(0),loadedComic(false),imageRotation(0),numLeftPages(2),numRightPages(2) +{ + int size = numLeftPages+numRightPages+1; + currentPageBufferedIndex = numLeftPages; + for(int i = 0; imoveToThread(QApplication::instance()->thread()); + comic->deleteLater(); + } + + foreach(ImageFilter * filter, filters) + delete filter; + + foreach(PageRender * pr,pageRenders) + if(pr !=0) + { + if(pr->wait()) + delete pr; + } +} +//Este método se encarga de forzar el renderizado de las páginas. +//Actualiza el buffer según es necesario. +//si la pagina actual no está renderizada, se lanza un hilo que la renderize (double or single page mode) y se emite una señal que indica que se está renderizando. +void Render::render() +{ + updateBuffer(); + if(buffer[currentPageBufferedIndex]->isNull()) + { + if(pagesReady.size()>0) + { + if(doublePage) + { + if(pagesReady[currentIndex] && pagesReady[qMin(currentIndex+1,(int)comic->numPages()-1)]) + if(currentIndex+1 > (int)comic->numPages()-1) + pageRenders[currentPageBufferedIndex] = new DoublePageRender(this,currentIndex,comic->getRawData()->at(currentIndex),QByteArray(),buffer[currentPageBufferedIndex],imageRotation,filters); + else + pageRenders[currentPageBufferedIndex] = new DoublePageRender(this,currentIndex,comic->getRawData()->at(currentIndex),comic->getRawData()->at(currentIndex+1),buffer[currentPageBufferedIndex],imageRotation,filters); + else + //las páginas no están listas, y se están cargando en el cómic + emit processingPage(); //para evitar confusiones esta señal debería llamarse de otra forma + } + else + if(pagesReady[currentIndex]) + pageRenders[currentPageBufferedIndex] = new PageRender(this,currentIndex,comic->getRawData()->at(currentIndex),buffer[currentPageBufferedIndex],imageRotation,filters); + else + //las páginas no están listas, y se están cargando en el cómic + emit processingPage(); //para evitar confusiones esta señal debería llamarse de otra forma + + //si se ha creado un hilo para renderizar la página actual, se arranca + if(pageRenders[currentPageBufferedIndex]!=0) + { + //se conecta la señal pageReady del hilo, con el SLOT prepareAvailablePage + connect(pageRenders[currentPageBufferedIndex],SIGNAL(pageReady(int)),this,SLOT(prepareAvailablePage(int))); + //se emite la señal de procesando, debido a que los hilos se arrancan aquí + if(doublePage || filters.size()>0) + emit processingPage(); + pageRenders[currentPageBufferedIndex]->start(); + pageRenders[currentPageBufferedIndex]->setPriority(QThread::TimeCriticalPriority); + } + else + //en qué caso sería necesario hacer esto??? //TODO: IMPORTANTE, puede que no sea necesario. + emit processingPage(); + } + else + //no hay ninguna página lista para ser renderizada, es necesario esperar. + emit processingPage(); + } + else + // la página actual está lista + emit currentPageReady(); + + //se renderizan las páginas restantes para llenar el buffer. + if(doublePage) + fillBufferDoublePage(); + else + fillBuffer(); +} + +QPixmap * Render::getCurrentPage() +{ + QPixmap * page = new QPixmap(); + *page = page->fromImage(*buffer[currentPageBufferedIndex]); + return page; +} + +void Render::setRotation(int degrees) +{ + Q_UNUSED(degrees) +} + +void Render::setComic(Comic * c) +{ + if(comic !=0) + { + comic->moveToThread(QApplication::instance()->thread()); + comic->disconnect(); + comic->deleteLater(); + } + comic = c; +} + +void Render::prepareAvailablePage(int page) +{ + if(currentIndex == page) + emit currentPageReady(); +} + +void Render::update() +{ + render(); +} +//----------------------------------------------------------------------------- +// Comic interface +//----------------------------------------------------------------------------- +void Render::load(const QString & path, int atPage) +{ + createComic(path); + loadComic(path,atPage); + startLoad(); +} + +//----------------------------------------------------------------------------- +void Render::load(const QString & path, const ComicDB & comicDB) +{ + //TODO prepare filters + for(int i = 0; i < filters.count(); i++) + { + if(typeid(*filters[i]) == typeid(BrightnessFilter)) + if(comicDB.info.brightness == -1) + filters[i]->setLevel(0); + else + filters[i]->setLevel(comicDB.info.brightness); + if(typeid(*filters[i]) == typeid(ContrastFilter)) + if(comicDB.info.contrast == -1) + filters[i]->setLevel(100); + else + filters[i]->setLevel(comicDB.info.contrast); + if(typeid(*filters[i]) == typeid(GammaFilter)) + if(comicDB.info.gamma == -1) + filters[i]->setLevel(100); + else + filters[i]->setLevel(comicDB.info.gamma); + } + createComic(path); + loadComic(path,comicDB); + startLoad(); +} + +void Render::createComic(const QString & path) +{ + if(comic!=0) + { + //comic->moveToThread(QApplication::instance()->thread()); + comic->disconnect(); + comic->deleteLater(); + } + //comic->moveToThread(QApplication::instance()->thread()); + comic = FactoryComic::newComic(path); + + + if(comic == NULL)//archivo no encontrado o no válido + { + emit errorOpening(); + reset(); + return; + } + + previousIndex = currentIndex = 0; + + connect(comic,SIGNAL(errorOpening()),this,SIGNAL(errorOpening())); + connect(comic,SIGNAL(errorOpening(QString)),this,SIGNAL(errorOpening(QString))); + connect(comic,SIGNAL(crcErrorFound(QString)),this,SIGNAL(crcError(QString))); + connect(comic,SIGNAL(errorOpening()),this,SLOT(reset())); + connect(comic,SIGNAL(imageLoaded(int)),this,SIGNAL(imageLoaded(int))); + connect(comic,SIGNAL(imageLoaded(int)),this,SLOT(pageRawDataReady(int))); + connect(comic,SIGNAL(openAt(int)),this,SLOT(renderAt(int))); + connect(comic,SIGNAL(numPages(unsigned int)),this,SIGNAL(numPages(unsigned int))); + connect(comic,SIGNAL(numPages(unsigned int)),this,SLOT(setNumPages(unsigned int))); + connect(comic,SIGNAL(imageLoaded(int,QByteArray)),this,SIGNAL(imageLoaded(int,QByteArray))); + connect(comic,SIGNAL(isBookmark(bool)),this,SIGNAL(currentPageIsBookmark(bool))); + connect(comic,SIGNAL(isBookmark(bool)),this,SLOT(pageIsBookmark(bool))); + + connect(comic,SIGNAL(bookmarksUpdated()),this,SIGNAL(bookmarksUpdated())); + + //connect(comic,SIGNAL(isLast()),this,SIGNAL(isLast())); + //connect(comic,SIGNAL(isCover()),this,SIGNAL(isCover())); + + pagesReady.clear(); +} +void Render::loadComic(const QString & path,const ComicDB & comicDB) +{ + comic->load(path,comicDB); +} +void Render::loadComic(const QString & path, int atPage) +{ + comic->load(path,atPage); +} + +void Render::startLoad() +{ + QThread * thread = NULL; + + thread = new QThread(); + + comic->moveToThread(thread); + + connect(thread, SIGNAL(started()), comic, SLOT(process())); + connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater())); + + if(thread != NULL) + thread->start(); + + invalidate(); + loadedComic = true; + update(); +} + +void Render::renderAt(int page) +{ + previousIndex = currentIndex = page; + emit pageChanged(page); +} + +void Render::reset() +{ + loadedComic = false; + invalidate(); +} +//si se solicita la siguiente página, se calcula cuál debe ser en función de si se lee en modo a doble página o no. +//la página sólo se renderiza, si realmente ha cambiado. +void Render::nextPage() +{ + int nextPage; //indica cuál será la próxima página + if(doublePage) + { + nextPage = currentIndex; + if(currentIndex+2<(int)comic->numPages()) + { + nextPage = currentIndex+2; + if(currentIndex != nextPage) + comic->setIndex(nextPage); + } + } + else + { + nextPage = comic->nextPage(); + } + + //se fuerza renderizado si la página ha cambiado + if(currentIndex != nextPage) + { + previousIndex = currentIndex; + currentIndex = nextPage; + update(); + emit pageChanged(currentIndex); + } + else + emit isLast(); +} +//si se solicita la página anterior, se calcula cuál debe ser en función de si se lee en modo a doble página o no. +//la página sólo se renderiza, si realmente ha cambiado. +void Render::previousPage() +{ + int previousPage; //indica cuál será la próxima página + if(doublePage) + { + if(currentIndex == 1) + invalidate(); + previousPage = qMax(currentIndex-2,0); + if(currentIndex != previousPage) + { + comic->setIndex(previousPage); + } + } + else + { + previousPage = comic->previousPage(); + } + + //se fuerza renderizado si la página ha cambiado + if(currentIndex != previousPage) + { + previousIndex = currentIndex; + currentIndex = previousPage; + update(); + emit pageChanged(currentIndex); + } + else + emit isCover(); +} +unsigned int Render::getIndex() +{ + return comic->getIndex(); +} +unsigned int Render::numPages() +{ + return comic->numPages(); +} + +bool Render::hasLoadedComic() +{ + if(comic!=0) + return comic->loaded(); + return false; +} + +void Render::setNumPages(unsigned int numPages) +{ + pagesReady.fill(false,numPages); +} + +void Render::pageRawDataReady(int page) +{ + pagesEmited.push_back(page); + if(pageRenders.size()>0) + { + for(int i=0;i currentIndex-2*numLeftPages)) || + ((pagesEmited.at(i) > currentIndex+1) && (pagesEmited.at(i) < currentIndex+1+2*numRightPages)) ) + { + fillBufferDoublePage(); + } + } + else + { + if ( ((pagesEmited.at(i) < currentIndex) && (pagesEmited.at(i) > currentIndex-numLeftPages)) || + ((pagesEmited.at(i) > currentIndex) && (pagesEmited.at(i) < currentIndex+numRightPages)) ) + { + fillBuffer(); + } + } + } + pagesEmited.clear(); + } +} + +//sólo se renderiza la página, si ha habido un cambio de página +void Render::goTo(int index) +{ + + if(currentIndex != index) + { + comic->setIndex(index); + previousIndex = currentIndex; + currentIndex = index; + + //si cambia la paridad de las página en modo a doble página, se rellena el buffer. + //esto solo debería orcurrir al llegar al principio o al final + if(doublePage && ((previousIndex - index) % 2)!=0) + invalidate(); + + update(); + emit pageChanged(currentIndex); + } +} + +void Render::rotateRight() +{ + imageRotation = (imageRotation+90) % 360; + reload(); +} +void Render::rotateLeft() +{ + if(imageRotation == 0) + imageRotation = 270; + else + imageRotation = imageRotation - 90; + reload(); +} + +//Actualiza el buffer, añadiendo las imágenes (vacías) necesarias para su posterior renderizado y +//eliminado aquellas que ya no sean necesarias. También libera los hilos (no estoy seguro de que sea responsabilidad suya) +//Calcula el número de nuevas páginas que hay que buferear y si debe hacerlo por la izquierda o la derecha (según sea el sentido de la lectura) +void Render::updateBuffer() +{ + QMutexLocker locker(&mutex); + int windowSize = currentIndex - previousIndex; + if(doublePage) + { + windowSize = windowSize/2; + if(currentIndex == 0 && windowSize == 0 && previousIndex == 1) + windowSize = -1; + + } + if(windowSize > 0)//add pages to right pages and remove on the left + { + windowSize = qMin(windowSize,buffer.size()); + for(int i = 0; i < windowSize; i++) + { + //renders + PageRender * pr = pageRenders.front(); + pageRenders.pop_front(); + if(pr !=0) + { + if(pr->wait()) + delete pr; + } + pageRenders.push_back(0); + + //images + + if(buffer.front()!=0) + delete buffer.front(); + buffer.pop_front(); + buffer.push_back(new QImage()); + } + } + else //add pages to left pages and remove on the right + if(windowSize<0) + { + windowSize = -windowSize; + windowSize = qMin(windowSize,buffer.size()); + for(int i = 0; i < windowSize; i++) + { + //renders + PageRender * pr = pageRenders.back(); + pageRenders.pop_back(); + if(pr !=0) + { + if(pr->wait()) + delete pr; + } + pageRenders.push_front(0); + + //images + buffer.push_front(new QImage()); + QImage * p = buffer.back(); + if(p!=0) + delete p; + buffer.pop_back(); + } + } + previousIndex = currentIndex; +} + +void Render::fillBuffer() +{ + for(int i = 1; i <= qMax(numLeftPages,numRightPages); i++) + { + if ((currentIndex+i < (int)comic->numPages()) && + buffer[currentPageBufferedIndex+i]->isNull() && + i <= numRightPages && + pageRenders[currentPageBufferedIndex+i]==0 && + pagesReady[currentIndex+1]) //preload next pages + { + pageRenders[currentPageBufferedIndex+i] = new PageRender(this,currentIndex+i,comic->getRawData()->at(currentIndex+i),buffer[currentPageBufferedIndex+i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex],SIGNAL(pageReady(int)),this,SLOT(prepareAvailablePage(int))); + pageRenders[currentPageBufferedIndex+i]->start(); + } + + if ((currentIndex-i > 0) && + buffer[currentPageBufferedIndex-i]->isNull() && + i <= numLeftPages && + pageRenders[currentPageBufferedIndex-i]==0 && + pagesReady[currentIndex-1]) //preload previous pages + { + pageRenders[currentPageBufferedIndex-i] = new PageRender(this,currentIndex-i,comic->getRawData()->at(currentIndex-i),buffer[currentPageBufferedIndex-i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex],SIGNAL(pageReady(int)),this,SLOT(prepareAvailablePage(int))); + pageRenders[currentPageBufferedIndex-i]->start(); + } + } +} + +void Render::fillBufferDoublePage() +{ + for(int i = 1; i <= qMax(numLeftPages,numRightPages); i++) + { + if ((currentIndex+2*i < (int)comic->numPages()) && + buffer[currentPageBufferedIndex+i]->isNull() && + i <= numRightPages && + pageRenders[currentPageBufferedIndex+i]==0 && + (pagesReady[currentIndex+2*i] && pagesReady[qMin(currentIndex+(2*i)+1,(int)comic->numPages()-1)])) //preload next pages + { + if(currentIndex+(2*i)+1 > (int)comic->numPages()-1) + pageRenders[currentPageBufferedIndex+i] = new DoublePageRender(this,currentIndex+2*i,comic->getRawData()->at(currentIndex+(2*i)),QByteArray(),buffer[currentPageBufferedIndex+i],imageRotation,filters); + else + pageRenders[currentPageBufferedIndex+i] = new DoublePageRender(this,currentIndex+2*i,comic->getRawData()->at(currentIndex+(2*i)),comic->getRawData()->at(currentIndex+(2*i)+1),buffer[currentPageBufferedIndex+i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex],SIGNAL(pageReady(int)),this,SLOT(prepareAvailablePage(int))); + pageRenders[currentPageBufferedIndex+i]->start(); + } + + if ((currentIndex-2*i >= -1) && + buffer[currentPageBufferedIndex-i]->isNull() && + i <= numLeftPages && + pageRenders[currentPageBufferedIndex-i]==0 && + (pagesReady[qMax(currentIndex-2*i,0)] && pagesReady[qMin(currentIndex-(2*i)+1,(int)comic->numPages()-1)])) //preload previous pages + { + if(currentIndex-2*i == -1) + pageRenders[currentPageBufferedIndex-i] = new DoublePageRender(this,0,QByteArray(),comic->getRawData()->at(0),buffer[currentPageBufferedIndex-i],imageRotation,filters); + else + pageRenders[currentPageBufferedIndex-i] = new DoublePageRender(this,currentIndex-2*i,comic->getRawData()->at(currentIndex-(2*i)),comic->getRawData()->at(currentIndex-(2*i)+1),buffer[currentPageBufferedIndex-i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex],SIGNAL(pageReady(int)),this,SLOT(prepareAvailablePage(int))); + pageRenders[currentPageBufferedIndex-i]->start(); + } + } +} + +//Método que debe ser llamado cada vez que la estructura del buffer se vuelve inconsistente con el modo de lectura actual. +//se terminan todos los hilos en ejecución y se libera la memoria (de hilos e imágenes) +void Render::invalidate() +{ + for(int i=0;iwait(); + delete pageRenders[i]; + pageRenders[i] = 0; + } + } + + for(int i=0;inumPages())) + s += "-"+QString::number(currentIndex+2); + s += "/"+QString::number(comic->numPages()); + return s; +} + +void Render::setBookmark() +{ + comic->setBookmark(); +} + +void Render::removeBookmark() +{ + comic->removeBookmark(); +} + +void Render::save() +{ + comic->saveBookmarks(); +} + +Bookmarks * Render::getBookmarks() +{ + return comic->bm; +} + +void Render::reload() +{ + if(comic) + { + invalidate(); + update(); + } +} + +void Render::updateFilters(int brightness, int contrast, int gamma) +{ + for(int i = 0; i < filters.count(); i++) + { + if(typeid(*filters[i]) == typeid(BrightnessFilter)) + filters[i]->setLevel(brightness); + if(typeid(*filters[i]) == typeid(ContrastFilter)) + filters[i]->setLevel(contrast); + if(typeid(*filters[i]) == typeid(GammaFilter)) + filters[i]->setLevel(gamma); + } + + reload(); +} diff --git a/YACReader/render.h b/YACReader/render.h new file mode 100644 index 00000000..423d7955 --- /dev/null +++ b/YACReader/render.h @@ -0,0 +1,209 @@ + #ifndef RENDER_H +#define RENDER_H + +#include +#include +#include +#include +#include +#include +#include "comic.h" +//----------------------------------------------------------------------------- +// FILTERS +//----------------------------------------------------------------------------- + +#include + +class Comic; +class ComicDB; +class Render; + +class ImageFilter { +public: + ImageFilter(){}; + virtual QImage setFilter(const QImage & image) = 0; + inline int getLevel() {return level;}; + inline void setLevel(int l) {level = l;}; +protected: + int level; +}; + +class MeanNoiseReductionFilter : public ImageFilter { +public: + enum NeighborghoodSize{SMALL=9, LARGE=25 }; + MeanNoiseReductionFilter(enum NeighborghoodSize ns = SMALL); + virtual QImage setFilter(const QImage & image); +private: + enum NeighborghoodSize neighborghoodSize; +}; + +class MedianNoiseReductionFilter : public ImageFilter { +public: + enum NeighborghoodSize{SMALL=9, LARGE=25 }; + MedianNoiseReductionFilter(enum NeighborghoodSize ns = SMALL); + virtual QImage setFilter(const QImage & image); +private: + enum NeighborghoodSize neighborghoodSize; +}; + +class BrightnessFilter : public ImageFilter { +public: + BrightnessFilter(int l=-1); + virtual QImage setFilter(const QImage & image); +}; + +class ContrastFilter : public ImageFilter { +public: + ContrastFilter(int l=-1); + virtual QImage setFilter(const QImage & image); +}; + +class GammaFilter : public ImageFilter { +public: + GammaFilter(int l=-1); + virtual QImage setFilter(const QImage & image); +}; + +//----------------------------------------------------------------------------- +// RENDER +//----------------------------------------------------------------------------- + +class PageRender : public QThread +{ + Q_OBJECT +public: + PageRender(); + PageRender(Render * render,int numPage, const QByteArray & rawData, QImage * page,unsigned int degrees=0, QVector filters = QVector()); + int getNumPage(){return numPage;}; + void setData(const QByteArray & rawData){data = rawData;}; + void setPage(QImage * p){page = p;}; + void setRotation(unsigned int d){degrees = d;}; + void setFilters(QVector f){filters = f;}; +private: + int numPage; + QByteArray data; + QImage * page; + unsigned int degrees; + QVector filters; + void run(); + Render * render; +signals: + void pageReady(int); + +}; +//----------------------------------------------------------------------------- +// RENDER +//----------------------------------------------------------------------------- + +class DoublePageRender : public PageRender +{ + Q_OBJECT +public: + DoublePageRender(Render * render, int firstPage, const QByteArray & firstPageData,const QByteArray & secondPageData, QImage * page,unsigned int degrees=0, QVector filters = QVector()); +private: + int numPage; + QByteArray data; + QByteArray data2; + QImage * page; + unsigned int degrees; + QVector filters; + void run(); + Render * render; +signals: + void pageReady(int); + +}; + + +class Render : public QObject { +Q_OBJECT +public: + Render(); + ~Render(); + +public slots: + void render(); + QPixmap * getCurrentPage(); + void goTo(int index); + void doublePageSwitch(); + void setRotation(int degrees); + void setComic(Comic * c); + void prepareAvailablePage(int page); + void update(); + void setNumPages(unsigned int numPages); + void pageRawDataReady(int page); + //--comic interface + void nextPage(); + void previousPage(); + void load(const QString & path, const ComicDB & comic); + void load(const QString & path, int atPage); + void createComic(const QString & path); + void loadComic(const QString & path,const ComicDB & comic); + void loadComic(const QString & path, int atPage); + void startLoad(); + void rotateRight(); + void rotateLeft(); + unsigned int getIndex(); + unsigned int numPages(); + bool hasLoadedComic(); + void updateBuffer(); + void fillBuffer(); + void fillBufferDoublePage(); + void invalidate(); + QString getCurrentPagesInformation(); + void setBookmark(); + void removeBookmark(); + void save(); + void reset(); + void reload(); + void updateFilters(int brightness, int contrast, int gamma); + Bookmarks * getBookmarks(); + //sets the firt page to render + void renderAt(int page); + +signals: + void currentPageReady(); + void processingPage(); + void imagesLoaded(); + void imageLoaded(int index); + void imageLoaded(int index,const QByteArray & image); + void pageChanged(int index); + void numPages(unsigned int numPages); + void errorOpening(); + void errorOpening(QString); + void crcError(QString); + void currentPageIsBookmark(bool); + void isLast(); + void isCover(); + + void bookmarksUpdated(); + + +private: + Comic * comic; + bool doublePage; + int previousIndex; + int currentIndex; + //QPixmap * currentPage; + int currentPageBufferedIndex; + int numLeftPages; + int numRightPages; + QList pageRenders; + QList buffer; + void loadAll(); + void updateRightPages(); + void updateLeftPages(); + bool loadedComic; + QList pagesEmited; + QVector pagesReady; + int imageRotation; + QVector filters; + QMutex mutex; + + friend class PageRender; + friend class DoublePageRender; + +}; + + +#endif // RENDER_H diff --git a/YACReader/shortcuts_dialog.cpp b/YACReader/shortcuts_dialog.cpp new file mode 100644 index 00000000..dd7441cf --- /dev/null +++ b/YACReader/shortcuts_dialog.cpp @@ -0,0 +1,67 @@ +#include "shortcuts_dialog.h" +#include +#include +#include +#include +#include +#include +#include +#include + +ShortcutsDialog::ShortcutsDialog(QWidget * parent) + :QDialog(parent)//,Qt::FramelessWindowHint) +{ + setModal(true); + setWindowIcon(QIcon(":/images/shortcuts.png")); + setWindowTitle(tr("YACReader keyboard shortcuts")); + + QVBoxLayout * mainLayout = new QVBoxLayout; + + close = new QPushButton(tr("Close")); + connect(close,SIGNAL(clicked()),this,SLOT(close())); + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(close); + + QHBoxLayout * shortcutsLayout = new QHBoxLayout; + + shortcuts = new QTextEdit(); + shortcuts->setFrameStyle(QFrame::NoFrame); + + //"

General functions:


O : Open comic
Esc : Exit

" + shortcuts->setReadOnly(true); + shortcutsLayout->addWidget(shortcuts); + //shortcutsLayout->addWidget(shortcuts2); + shortcutsLayout->setSpacing(0); + mainLayout->addLayout(shortcutsLayout); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout *imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(); + QPixmap p(":/images/shortcuts.png"); + imgLabel->setPixmap(p); + + QVBoxLayout * imgLayout = new QVBoxLayout; + imgLayout->addWidget(imgLabel); + imgLayout->addStretch(); + + imgMainLayout->addLayout(imgLayout); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setFixedSize(QSize(700,500)); + + QFile f(":/files/shortcuts.html"); + f.open(QIODevice::ReadOnly); + QTextStream txtS(&f); + txtS.setCodec(QTextCodec::codecForName("UTF-8")); + QString content = txtS.readAll(); + + f.close(); + + shortcuts->setHtml(content); + + setWindowTitle(tr("Keyboard Shortcuts")); +} diff --git a/YACReader/shortcuts_dialog.h b/YACReader/shortcuts_dialog.h new file mode 100644 index 00000000..8f1b66b6 --- /dev/null +++ b/YACReader/shortcuts_dialog.h @@ -0,0 +1,19 @@ +#ifndef SHORTCUTS_DIALOG_H +#define SHORTCUTS_DIALOG_H + +#include +#include +#include + +class ShortcutsDialog : public QDialog +{ +Q_OBJECT + public: + ShortcutsDialog(QWidget * parent = 0); + private: + QTextEdit * shortcuts; + QPushButton * close; + public slots: +}; + +#endif // SHORTCUTS_DIALOG_H diff --git a/YACReader/translator.cpp b/YACReader/translator.cpp new file mode 100644 index 00000000..bff51e85 --- /dev/null +++ b/YACReader/translator.cpp @@ -0,0 +1,429 @@ +#include + +#if QT_VERSION >= 0x050000 +#include +#else +#include +#include +#endif + +#include +#include +#include +#include +#include +#include "translator.h" + +#include "yacreader_busy_widget.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +#define APPID "417CEAD93449502CC3C9B69FED26C54118E62BCC" + +YACReaderTranslator::YACReaderTranslator(QWidget * parent) +:QWidget(parent),drag(false) +{ + QString scrollBarStyle = "QScrollBar:vertical { border: none; background: #404040; width: 7px; margin: 0 3px 0 0; }" + "QScrollBar::handle:vertical { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical {background: none; }"; + + this->setCursor(QCursor(Qt::ArrowCursor)); + this->setAutoFillBackground(true); + this->setBackgroundRole(QPalette::Window); + QPalette p(this->palette()); + p.setColor(QPalette::Window, QColor("#404040")); + this->setPalette(p); + + QVBoxLayout *layout = new QVBoxLayout(this); + + //TITLE BAR + QHBoxLayout * titleBar = new QHBoxLayout(); + QPushButton * close = new QPushButton(QIcon(QPixmap(":/images/close.png")),""); + close->setFlat(true); + QLabel * title = new QLabel(tr("YACReader translator")); + title->setStyleSheet("QLabel {font-size:18px; font-family:Arial; color:white;}"); + titleBar->addWidget(title); + titleBar->addStretch(); + close->resize(14,14); + close->setStyleSheet("QPushButton {margin:0;padding:0;border:none;}"); + titleBar->addWidget(close); + titleBar->setContentsMargins(0,0,0,0); + titleBar->setSpacing(0); + connect(close,SIGNAL(clicked()),this->parent(),SLOT(animateHideTranslator())); + + layout->addLayout(titleBar); + + //INPUT TEXT + text = new QTextEdit(this); + text->setMinimumHeight(110); + text->setMaximumHeight(110); + layout->addSpacing(12); + layout->addWidget(text); + text->setStyleSheet("QTextEdit{border:none;background:#2a2a2a;color:white; font-size:12px; padding:6px;}"+scrollBarStyle); + + //COMBOBOXES + QHBoxLayout * combos = new QHBoxLayout(); + from = new QComboBox(this); + to = new QComboBox(this); + QString comboBoxStyle = "QComboBox {border:none;background:#2a2a2a;color:white;font-size:12px;font-family:Arial;padding-left:8px;}" + "QComboBox::down-arrow {image: url(:/images/dropDownArrow.png);}" + "QComboBox::drop-down {border:none; padding-right:10px;}" + "QComboBox QAbstractItemView {border: none; background:#272727; color:white; selection-background-color: #202020; outline:none;}" + "QComboBox QAbstractItemView::item {padding-left:8px;}" + scrollBarStyle + ; + from->setStyleSheet(comboBoxStyle); + to->setStyleSheet(comboBoxStyle); + from->setFixedHeight(22); + to->setFixedHeight(22); + QLabel * arrow = new QLabel(this); + QPixmap arrowPixmap(":/images/fromTo.png"); + arrow->setPixmap(arrowPixmap); + QPushButton * searchButton = new QPushButton(this); + searchButton->setIcon(QIcon(":/images/translatorSearch.png")); + searchButton->setStyleSheet("QPushButton {border:none; background:#2a2a2a;}"); + searchButton->setFixedSize(22,22); + combos->addWidget(from,1); + combos->addSpacing(9); + combos->addWidget(arrow,0); + combos->addSpacing(9); + combos->addWidget(to,1); + combos->addSpacing(9); + combos->addWidget(searchButton,0); + layout->addSpacing(12); + layout->addLayout(combos); + + + //RESULTS + QHBoxLayout * resultsTitleLayout = new QHBoxLayout(); + resultsTitle = new QLabel(tr("Translation")); + resultsTitle->setStyleSheet("QLabel {font-family:Arial;font-size:14px;color:#e3e3e3;}"); + speakButton = new QPushButton(this); + speakButton->setStyleSheet("QPushButton {border:none;}"); + speakButton->setIcon(QIcon(":/images/speaker.png")); + resultsTitleLayout->addWidget(resultsTitle,0,Qt::AlignVCenter); + resultsTitleLayout->addSpacing(10); + resultsTitleLayout->addWidget(speakButton,0,Qt::AlignVCenter); + resultsTitleLayout->addStretch(); + + layout->addSpacing(15); + layout->addLayout(resultsTitleLayout); + layout->addSpacing(12); + + resultText = new QLabel(); + resultText->setWordWrap(true); + resultText->setStyleSheet("QLabel {color:white;font-size:12px;}"); + resultText->setText("ñlkas lakj dflkaj lasd jflie lkajd fie kljads ijef lasei afsliej ljse f"); + layout->addWidget(resultText); + + layout->addStretch(); + + //CLEAR BUTTON + clearButton = new QPushButton(tr("clear")); + layout->addWidget(clearButton,0,Qt::AlignRight); + clearButton->setMinimumWidth(95); + clearButton->setStyleSheet("QPushButton {border:1px solid #212121; background:#2a2a2a; color:white; font-family:Arial; font-size:12px; padding-top:5px; padding-bottom:5px;}"); + + resize(400,479); + + layout->setMargin(0); + layout->setContentsMargins(18,12,18,12); + setContentsMargins(0,0,0,0); + layout->setSpacing(0); + + hideResults(); + populateCombos(); + + busyIndicator = new YACReaderBusyWidget(this); + busyIndicator->move((this->width()-busyIndicator->width())/2,(this->height()-busyIndicator->height())*2/3); + busyIndicator->hide(); + + show(); + + connect(searchButton,SIGNAL(pressed()),this,SLOT(translate())); + connect(speakButton,SIGNAL(pressed()),this,SLOT(play())); + connect(clearButton,SIGNAL(pressed()),this,SLOT(clear())); + + //multimedia/phonon +#if QT_VERSION >= 0x050000 + player = new QMediaPlayer; +#else + music = createPlayer(MusicCategory); +#endif + +} + +void YACReaderTranslator::hideResults() +{ + resultsTitle->setHidden(true); + speakButton->setHidden(true); + resultText->setHidden(true); +} + +void YACReaderTranslator::clear() +{ + hideResults(); + text->clear(); +} + +void YACReaderTranslator::translate() +{ + QString text = this->text->toPlainText(); + if(text.isEmpty()) + return; + QString from = this->from->itemData(this->from->currentIndex()).toString(); + QString to = this->to->itemData(this->to->currentIndex()).toString(); + + TranslationLoader * translationLoader = new TranslationLoader(text,from,to); + connect(translationLoader,SIGNAL(requestFinished(QString)),this,SLOT(setTranslation(QString))); + connect(translationLoader,SIGNAL(error()),this,SLOT(error())); + connect(translationLoader,SIGNAL(timeOut()),this,SLOT(error())); + connect(translationLoader,SIGNAL(finished()),translationLoader,SLOT(deleteLater())); + + TextToSpeachLoader * tts = new TextToSpeachLoader(text,from); + connect(tts,SIGNAL(requestFinished(QUrl)),this,SLOT(setSpeak(QUrl))); + connect(tts,SIGNAL(error()),this,SLOT(error())); + connect(tts,SIGNAL(timeOut()),this,SLOT(error())); + connect(tts,SIGNAL(finished()),tts,SLOT(deleteLater())); + + translationLoader->start(); + tts->start(); + + resultsTitle->setText(tr("Translation")); + + hideResults(); + + busyIndicator->show(); +} + +void YACReaderTranslator::error() +{ + resultsTitle->setText(tr("Service not available")); + resultsTitle->setHidden(false); + busyIndicator->hide(); +} + +void YACReaderTranslator::setSpeak(const QUrl & url) +{ + resultsTitle->setHidden(false); + speakButton->setHidden(false); + + ttsSource = url; +} + +void YACReaderTranslator::setTranslation(const QString & string) +{ + resultText->setText(string); + + resultsTitle->setHidden(false); + resultText->setHidden(false); + busyIndicator->hide(); +} + +void YACReaderTranslator::populateCombos() +{ + QList combos; + combos.append(from); + combos.append(to); + + for(int i=0;iaddItem("Arabic","ar"); + combo->addItem("Bulgarian","bg"); + combo->addItem("Catalan","ca"); + combo->addItem("Chinese Simplified","zh-CHS"); + combo->addItem("Chinese Traditional","zh-CHT"); + combo->addItem("Czech","cs"); + combo->addItem("Danish","da"); + combo->addItem("Dutch","nl"); + combo->addItem("English","en"); + combo->addItem("Estonian","et"); + combo->addItem("Finnish","fi"); + combo->addItem("French","fr"); + combo->addItem("German","de"); + combo->addItem("Greek","el"); + combo->addItem("Haitian Creole","ht"); + combo->addItem("Hebrew","he"); + combo->addItem("Hindi","hi"); + combo->addItem("Hungarian","hu"); + combo->addItem("Indonesian","id"); + combo->addItem("Italian","it"); + combo->addItem("Japanese","ja"); + combo->addItem("Korean","ko"); + combo->addItem("Latvian","lv"); + combo->addItem("Lithuanian","lt"); + combo->addItem("Norwegian","no"); + combo->addItem("Polish","pl"); + combo->addItem("Portuguese","pt"); + combo->addItem("Romanian","ro"); + combo->addItem("Russian","ru"); + combo->addItem("Slovak","sk"); + combo->addItem("Slovenian","sl"); + combo->addItem("Spanish","es"); + combo->addItem("Swedish","sv"); + combo->addItem("Thai","th"); + combo->addItem("Turkish","tr"); + combo->addItem("Ukrainian","uk"); + combo->addItem("Vietnamese","vi"); + } + from->setCurrentIndex(from->findText("English")); + to->setCurrentIndex(from->findText("Spanish")); +} + +void YACReaderTranslator::play() +{ + //QMessageBox::question(this,"xxx",ttsSource.toString()); +#if QT_VERSION >= 0x050000 + + player->setMedia(ttsSource); + player->play(); + +#else + MediaSource src(ttsSource); + src.setAutoDelete(true); + music->setCurrentSource(src); + music->play(); +#endif +} + +YACReaderTranslator::~YACReaderTranslator() +{ +#if QT_VERSION >= 0x050000 +#else + delete music; +#endif +} + +void YACReaderTranslator::mousePressEvent(QMouseEvent *event) +{ + QPoint p = mapTo(this,event->pos()); + if(p.y() < 40) + { + drag = true; + click = event->pos(); + } +} + +void YACReaderTranslator::mouseReleaseEvent(QMouseEvent *event) +{ + drag = false; + event->accept(); +} + +void YACReaderTranslator::mouseMoveEvent(QMouseEvent * event) +{ + if(drag) + this->move(QPoint(mapToParent(event->pos())-click)); + event->accept(); +} + +//--------------------------------------------------------------------------- +//--------------------------------------------------------------------------- +//--------------------------------------------------------------------------- + +TranslationLoader::TranslationLoader(QString text, QString from, QString to) + :QThread(),text(text),from(from),to(to) +{ +} + +void TranslationLoader::run() +{ + QNetworkAccessManager manager; + QEventLoop q; + QTimer tT; + + tT.setSingleShot(true); + connect(&tT, SIGNAL(timeout()), &q, SLOT(quit())); + connect(&manager, SIGNAL(finished(QNetworkReply*)),&q, SLOT(quit())); + + QString url = "http://api.microsofttranslator.com/V2/Ajax.svc/Translate?appid=%1&from=%2&to=%3&text=%4&contentType=text/plain"; + url = url.arg(APPID).arg(from).arg(to).arg(text); + + QNetworkReply *reply = manager.get(QNetworkRequest(QUrl(url))); + + tT.start(5000); // 5s timeout + q.exec(); + + if(tT.isActive()){ + // download complete + if(reply->error() == QNetworkReply::NoError) + { + QString utf8 = QString::fromUtf8(reply->readAll()); + utf8 = utf8.remove(0,1); + utf8 = utf8.remove(utf8.count()-1,1); + + QString translated(utf8); + emit(requestFinished(translated)); + } + else + emit(error()); + } else { + emit(timeOut()); + } +} + +//--------------------------------------------------------------------------- +//--------------------------------------------------------------------------- +//--------------------------------------------------------------------------- + + +TextToSpeachLoader::TextToSpeachLoader(QString text, QString language) + :QThread(),text(text),language(language) +{ +} + + +void TextToSpeachLoader::run() +{ + QNetworkAccessManager manager; + QEventLoop q; + QTimer tT; + + tT.setSingleShot(true); + connect(&tT, SIGNAL(timeout()), &q, SLOT(quit())); + connect(&manager, SIGNAL(finished(QNetworkReply*)),&q, SLOT(quit())); + + QString url = "http://api.microsofttranslator.com/V2/Ajax.svc/Speak?appid=%1&language=%2&text=%3&contentType=text/plain"; + url = url.arg(APPID).arg(language).arg(text); + + QNetworkReply *reply = manager.get(QNetworkRequest(QUrl(url))); + + tT.start(5000); // 5s timeout + q.exec(); + + if(tT.isActive()){ + // download complete + if(reply->error() == QNetworkReply::NoError) + { + QString utf8 = QString::fromUtf8(reply->readAll()); + utf8 = utf8.remove(0,1); + utf8 = utf8.remove(utf8.count()-1,1); + utf8 = utf8.replace("\\",""); + + emit(requestFinished(QUrl(utf8))); + } + else + emit(error()); + } else { + emit(timeOut()); + } +} diff --git a/YACReader/translator.h b/YACReader/translator.h new file mode 100644 index 00000000..1ce1bee0 --- /dev/null +++ b/YACReader/translator.h @@ -0,0 +1,102 @@ +#ifndef __TRANSLATOR_H +#define __TRANSLATOR_H + +class QUrl; +class QMouseEvent; +class QPoint; +class QTextEdit; +class QComboBox; +class QLabel; +class QPushButton; +class YACReaderBusyWidget; + +#include +#include +#include + +#if QT_VERSION >= 0x050000 + class QMediaPlayer; +#else + #include + using namespace Phonon; +#endif + + + +class YACReaderTranslator : public QWidget +{ + Q_OBJECT + public: + YACReaderTranslator(QWidget * parent = 0); + ~YACReaderTranslator(); + + public slots: + void play(); + + protected slots: + void translate(); + void setSpeak(const QUrl & url); + void setTranslation(const QString & string); + void error(); + void clear(); + +protected: + void mousePressEvent(QMouseEvent *event); + void mouseReleaseEvent(QMouseEvent *event); + void mouseMoveEvent ( QMouseEvent * event ); + void hideResults(); + + void populateCombos(); + bool drag; + QPoint click; +private: + +#if QT_VERSION >= 0x050000 + QMediaPlayer *player; +#else + MediaObject * music; +#endif + + QTextEdit * text; + QComboBox * from; + QComboBox * to; + QLabel * resultsTitle; + QPushButton * speakButton; + QLabel * resultText; + YACReaderBusyWidget * busyIndicator; + QUrl ttsSource; + QPushButton * clearButton; + +}; + +class TranslationLoader : public QThread +{ + Q_OBJECT +public: + TranslationLoader(QString text, QString from, QString to); +signals: + void requestFinished(QString); + void timeOut(); + void error(); +private: + QString text; + QString from; + QString to; + void run(); +}; + +class TextToSpeachLoader : public QThread +{ + Q_OBJECT +public: + TextToSpeachLoader(QString text, QString language); +signals: + void requestFinished(QUrl); + void timeOut(); + void error(); +private: + QString text; + QString language; + void run(); +}; +#endif diff --git a/YACReader/viewer.cpp b/YACReader/viewer.cpp new file mode 100644 index 00000000..10812be8 --- /dev/null +++ b/YACReader/viewer.cpp @@ -0,0 +1,937 @@ +#include "viewer.h" +#include "magnifying_glass.h" +#include "configuration.h" +#include "magnifying_glass.h" +#include "goto_flow.h" +#include "goto_flow_gl.h" +#include "bookmarks_dialog.h" +#include "render.h" +#include "goto_dialog.h" +#include "translator.h" +#include "onstart_flow_selection_dialog.h" +#include "page_label_widget.h" +#include "notifications_label_widget.h" +#include "comic_db.h" +#include "shortcuts_manager.h" + +#include + + +Viewer::Viewer(QWidget * parent) +:QScrollArea(parent), +currentPage(0), +magnifyingGlassShowed(false), +fullscreen(false), +information(false), +adjustToWidthRatio(1), +doublePage(false), +wheelStop(false), +direction(1), +restoreMagnifyingGlass(false), +drag(false), +numScrollSteps(22), +shouldOpenNext(false), +shouldOpenPrevious(false) +{ + translator = new YACReaderTranslator(this); + translator->hide(); + translatorAnimation = new QPropertyAnimation(translator,"pos"); + translatorAnimation->setDuration(150); + translatorXPos = -10000; + translator->move(-translator->width(),10); + //current comic page + content = new QLabel(this); + configureContent(tr("Press 'O' to open comic.")); + //scroll area configuration + setBackgroundRole(QPalette::Dark); + setWidget(content); + setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + setFrameStyle(QFrame::NoFrame); + setAlignment(Qt::AlignCenter); + + QPalette palette; + palette.setColor(backgroundRole(), Configuration::getConfiguration().getBackgroundColor()); + setPalette(palette); + //--------------------------------------- + mglass = new MagnifyingGlass(Configuration::getConfiguration().getMagnifyingGlassSize(),this); + mglass->hide(); + content->setMouseTracking(true); + setMouseTracking(true); + + showCursor(); + + goToDialog = new GoToDialog(this); + + QSettings * settings = new QSettings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + + //CONFIG GOTO_FLOW-------------------------------------------------------- + if(QGLFormat::hasOpenGL() && !settings->contains(USE_OPEN_GL)) + { + OnStartFlowSelectionDialog * flowSelDialog = new OnStartFlowSelectionDialog(); + + flowSelDialog->exec(); + if(flowSelDialog->result() == QDialog::Accepted) + settings->setValue(USE_OPEN_GL,2); + else + settings->setValue(USE_OPEN_GL,0); + + delete flowSelDialog; + } + + if(QGLFormat::hasOpenGL() && (settings->value(USE_OPEN_GL).toBool() == true)) + goToFlow = new GoToFlowGL(this,Configuration::getConfiguration().getFlowType()); + else + goToFlow = new GoToFlow(this,Configuration::getConfiguration().getFlowType()); + + goToFlow->setFocusPolicy(Qt::StrongFocus); + goToFlow->hide(); + showGoToFlowAnimation = new QPropertyAnimation(goToFlow,"pos"); + showGoToFlowAnimation->setDuration(150); + + bd = new BookmarksDialog(this->parentWidget()); + + render = new Render(); + + hideCursorTimer = new QTimer(); + hideCursorTimer->setSingleShot(true); + + if(Configuration::getConfiguration().getDoublePage()) + doublePageSwitch(); + + createConnections(); + + hideCursorTimer->start(2500); + + setMouseTracking(true); + + //animations + verticalScroller = new QPropertyAnimation(verticalScrollBar(), "sliderPosition"); + connect(verticalScroller,SIGNAL(valueChanged (const QVariant &)),this,SIGNAL(backgroundChanges())); + + notificationsLabel = new NotificationsLabelWidget(this); + notificationsLabel->hide(); + + informationLabel = new PageLabelWidget(this); + + setAcceptDrops(true); + +} + +Viewer::~Viewer() +{ + delete render; + delete goToFlow; + delete translator; + delete translatorAnimation; + delete content; + delete hideCursorTimer; + delete informationLabel; + delete verticalScroller; + delete bd; + delete notificationsLabel; + delete mglass; + if(currentPage != 0) + delete currentPage; +} + +void Viewer::createConnections() +{ + //magnifyingGlass (update mg after a background change + connect(this,SIGNAL(backgroundChanges()),mglass,SLOT(updateImage())); + + //goToDialog + connect(goToDialog,SIGNAL(goToPage(unsigned int)),this,SLOT(goTo(unsigned int))); + + //goToFlow goTo + connect(goToFlow,SIGNAL(goToPage(unsigned int)),this,SLOT(goTo(unsigned int))); + + //current time + QTimer * t = new QTimer(); + connect(t,SIGNAL(timeout()),this,SLOT(updateInformation())); + t->start(1000); + + //hide cursor + connect(hideCursorTimer,SIGNAL(timeout()),this,SLOT(hideCursor())); + + //bookmarks + connect(bd,SIGNAL(goToPage(unsigned int)),this,SLOT(goTo(unsigned int))); + + //render + connect(render,SIGNAL(errorOpening()),this,SLOT(resetContent())); + connect(render,SIGNAL(errorOpening()),this,SLOT(showMessageErrorOpening())); + connect(render,SIGNAL(errorOpening(QString)),this,SLOT(showMessageErrorOpening(QString))); + connect(render,SIGNAL(crcError(QString)),this,SLOT(processCRCError(QString))); + connect(render,SIGNAL(numPages(unsigned int)),goToFlow,SLOT(setNumSlides(unsigned int))); + connect(render,SIGNAL(numPages(unsigned int)),goToDialog,SLOT(setNumPages(unsigned int))); + //connect(render,SIGNAL(numPages(unsigned int)),this,SLOT(updateInformation())); + connect(render,SIGNAL(imageLoaded(int,QByteArray)),goToFlow,SLOT(setImageReady(int,QByteArray))); + connect(render,SIGNAL(currentPageReady()),this,SLOT(updatePage())); + connect(render,SIGNAL(processingPage()),this,SLOT(setLoadingMessage())); + connect(render,SIGNAL(currentPageIsBookmark(bool)),this,SIGNAL(pageIsBookmark(bool))); + connect(render,SIGNAL(pageChanged(int)),this,SLOT(updateInformation())); + //connect(render,SIGNAL(bookmarksLoaded(Bookmarks)),this,SLOT(setBookmarks(Bookmarks))); + + connect(render,SIGNAL(isLast()),this,SLOT(showIsLastMessage())); + connect(render,SIGNAL(isCover()),this,SLOT(showIsCoverMessage())); + + connect(render,SIGNAL(bookmarksUpdated()),this,SLOT(setBookmarks())); +} + +//Deprecated +void Viewer::prepareForOpening() +{ + if(render->hasLoadedComic()) + save(); + //bd->setBookmarks(*bm); + + goToFlow->reset(); + + //render->update(); + + verticalScrollBar()->setSliderPosition(verticalScrollBar()->minimum()); + + if(Configuration::getConfiguration().getShowInformation() && !information) + { + QTimer * timer = new QTimer(); + connect(timer,SIGNAL(timeout()),this,SLOT(informationSwitch())); + connect(timer,SIGNAL(timeout()),timer,SLOT(deleteLater())); + timer->start(); + } + + informationLabel->setText("..."); +} + +void Viewer::open(QString pathFile, int atPage) +{ + prepareForOpening(); + render->load(pathFile, atPage); +} + +void Viewer::open(QString pathFile, const ComicDB & comic) +{ + prepareForOpening(); + render->load(pathFile, comic); +} + +void Viewer::showMessageErrorOpening() +{ + QMessageBox::critical(this,tr("Not found"),tr("Comic not found")); + //resetContent(); --> not needed +} + +void Viewer::showMessageErrorOpening(QString message) +{ + QMessageBox::critical(this,tr("Error opening comic"),message); + resetContent(); +} + +void Viewer::processCRCError(QString message) +{ + QMessageBox::critical(this,tr("CRC Error"),message); +} + +void Viewer::next() +{ + direction = 1; + render->nextPage(); + updateInformation(); + shouldOpenPrevious = false; +} + +void Viewer::prev() +{ + direction = -1; + render->previousPage(); + updateInformation(); + shouldOpenNext = false; +} +void Viewer::showGoToDialog() +{ + goToDialog->show(); +} +void Viewer::goTo(unsigned int page) +{ + direction = 1; //in "go to" direction is always fordward + render->goTo(page); +} + +void Viewer::updatePage() +{ + QPixmap * previousPage = currentPage; + currentPage = render->getCurrentPage(); + content->setPixmap(*currentPage); + updateContentSize(); + updateVerticalScrollBar(); + emit backgroundChanges(); + emit(pageAvailable(true)); + + if(goToFlow->isHidden()) + setFocus(Qt::ShortcutFocusReason); + else + goToFlow->setFocus(Qt::OtherFocusReason); + delete previousPage; + + if(currentPage->isNull()) + setPageUnavailableMessage(); + + if(restoreMagnifyingGlass) + { + restoreMagnifyingGlass = false; + showMagnifyingGlass(); + } + +} + +void Viewer::updateContentSize() +{ + //there is an image to resize + if(currentPage !=0 && !currentPage->isNull()) + { + if(Configuration::getConfiguration().getAdjustToFullSize()) + { + content->resize(currentPage->width(),currentPage->height()); + } + else + { + float aspectRatio = (float)currentPage->width()/currentPage->height(); + //Fit to width + if(Configuration::getConfiguration().getAdjustToWidth()) + { + adjustToWidthRatio = Configuration::getConfiguration().getFitToWidthRatio(); + if(static_cast(width()*adjustToWidthRatio/aspectRatio)(height()*aspectRatio)>width()) + content->resize(width(),static_cast(width()/aspectRatio)); + else + content->resize(static_cast(height()*aspectRatio),height()); + else + content->resize(width()*adjustToWidthRatio,static_cast(width()*adjustToWidthRatio/aspectRatio)); + } + //Fit to height or fullsize/custom size + else + { + if(static_cast(height()*aspectRatio)>width()) //page width exceeds window width + content->resize(width(),static_cast(width()/aspectRatio)); + else + content->resize(static_cast(height()*aspectRatio),height()); + } + } + emit backgroundChanges(); + } + content->update(); //TODO, it shouldn't be neccesary +} + +void Viewer::updateVerticalScrollBar() +{ + if(direction > 0) + verticalScrollBar()->setSliderPosition(verticalScrollBar()->minimum()); + else + verticalScrollBar()->setSliderPosition(verticalScrollBar()->maximum()); +} + +void Viewer::scrollDown() +{ + if(verticalScrollBar()->sliderPosition()==verticalScrollBar()->maximum()) + { + next(); + } + else + { + int currentPos = verticalScrollBar()->sliderPosition(); + verticalScroller->setDuration(250); + verticalScroller->setStartValue(currentPos); + verticalScroller->setEndValue(nextPos); + + verticalScroller->start(); + + emit backgroundChanges(); + } +} + +void Viewer::scrollUp() +{ + if(verticalScrollBar()->sliderPosition()==verticalScrollBar()->minimum()) + { + prev(); + } + else + { + int currentPos = verticalScrollBar()->sliderPosition(); + verticalScroller->setDuration(250); + verticalScroller->setStartValue(currentPos); + verticalScroller->setEndValue(nextPos); + + verticalScroller->start(); + + emit backgroundChanges(); + } +} + +void Viewer::keyPressEvent(QKeyEvent *event) +{ + if(render->hasLoadedComic()) + { + int _key = event->key(); + Qt::KeyboardModifiers modifiers = event->modifiers(); + + if(modifiers & Qt::ShiftModifier) + _key |= Qt::SHIFT; + if (modifiers & Qt::ControlModifier) + _key |= Qt::CTRL; + if (modifiers & Qt::MetaModifier) + _key |= Qt::META; + if (modifiers & Qt::AltModifier) + _key |= Qt::ALT; + + QKeySequence key(_key); + /*if(goToFlow->isVisible() && event->key()!=Qt::Key_S) + QCoreApplication::sendEvent(goToFlow,event); + else*/ + + if (key == ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_FORWARD_ACTION_Y)) + { + posByStep = height()/numScrollSteps; + nextPos=verticalScrollBar()->sliderPosition()+static_cast((height()*0.80)); + scrollDown(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(AUTO_SCROLL_BACKWARD_ACTION_Y)) + { + posByStep = height()/numScrollSteps; + nextPos=verticalScrollBar()->sliderPosition()-static_cast((height()*0.80)); + scrollUp(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_DOWN_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_UP_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_LEFT_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(MOVE_RIGHT_ACTION_Y)) + { + QAbstractScrollArea::keyPressEvent(event); + emit backgroundChanges(); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_FIRST_PAGE_ACTION_Y)) + { + goTo(0); + } + + else if (key == ShortcutsManager::getShortcutsManager().getShortcut(GO_TO_LAST_PAGE_ACTION_Y)) + { + goTo(this->render->numPages()-1); + } + + else + QAbstractScrollArea::keyPressEvent(event); + + if(mglass->isVisible() && (key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_UP_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(SIZE_DOWN_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_IN_MGLASS_ACTION_Y) || + key == ShortcutsManager::getShortcutsManager().getShortcut(ZOOM_OUT_MGLASS_ACTION_Y))) + { + QCoreApplication::sendEvent(mglass,event); + } + + } + else + QAbstractScrollArea::keyPressEvent(event); +} + +void Viewer::wheelEvent(QWheelEvent * event) +{ + if(render->hasLoadedComic()) + { + if((event->delta()<0)&&(verticalScrollBar()->sliderPosition()==verticalScrollBar()->maximum())) + { + if(wheelStop) + { + next(); + verticalScroller->stop(); + event->accept(); + wheelStop = false; + return; + } + else + wheelStop = true; + } + else + { + if((event->delta()>0)&&(verticalScrollBar()->sliderPosition()==verticalScrollBar()->minimum())) + { + if(wheelStop) + { + prev(); + verticalScroller->stop(); + event->accept(); + wheelStop = false; + return; + } + else + wheelStop = true; + } + } + + int deltaNotFinished = 0; + if(verticalScroller->state() == QAbstractAnimation::Running) + { + deltaNotFinished = verticalScroller->startValue().toInt() - verticalScroller->endValue().toInt(); + verticalScroller->stop(); + } + + + int currentPos = verticalScrollBar()->sliderPosition(); + verticalScroller->setDuration(250); + verticalScroller->setStartValue(currentPos); + verticalScroller->setEndValue(currentPos - event->delta() - deltaNotFinished); + + verticalScroller->start(); + + //QAbstractScrollArea::wheelEvent(event); + } +} + +void Viewer::resizeEvent(QResizeEvent * event) +{ + updateContentSize(); + goToFlow->move(QPoint((width()-goToFlow->width())/2,height()-goToFlow->height())); + informationLabel->updatePosition(); + QScrollArea::resizeEvent(event); +} + +void Viewer::mouseMoveEvent(QMouseEvent * event) +{ + showCursor(); + hideCursorTimer->start(2500); + + if(magnifyingGlassShowed) + mglass->move(static_cast(event->x()-float(mglass->width())/2),static_cast(event->y()-float(mglass->height())/2)); + + if(render->hasLoadedComic()) + { + if(showGoToFlowAnimation->state()!=QPropertyAnimation::Running) + { + if(goToFlow->isVisible()) + { + QPoint gtfPos = goToFlow->mapFrom(this,event->pos()); + if(gtfPos.y() < 0 || gtfPos.x()<0 || gtfPos.x()>goToFlow->width())//TODO this extra check is for Mavericks (mouseMove over goToFlowGL seems to be broken) + animateHideGoToFlow(); + //goToFlow->hide(); + } + else + { + int umbral = (width()-goToFlow->width())/2; + if((event->y()>height()-15)&&(event->x()>umbral)&&(event->x()stop(); + } + } + } + + if(drag) + { + int currentPosY = verticalScrollBar()->sliderPosition(); + int currentPosX = horizontalScrollBar()->sliderPosition(); + verticalScrollBar()->setSliderPosition(currentPosY=currentPosY+(yDragOrigin-event->y())); + horizontalScrollBar()->setSliderPosition(currentPosX=currentPosX+(xDragOrigin-event->x())); + yDragOrigin = event->y(); + xDragOrigin = event->x(); + } + } + + +} + +const QPixmap * Viewer::pixmap() +{ + return content->pixmap(); +} + +void Viewer::magnifyingGlassSwitch() +{ + magnifyingGlassShowed?hideMagnifyingGlass():showMagnifyingGlass(); +} + +void Viewer::showMagnifyingGlass() +{ + if(render->hasLoadedComic()) + { + QPoint p = QPoint(cursor().pos().x(),cursor().pos().y()); + p = this->parentWidget()->mapFromGlobal(p); + mglass->move(static_cast(p.x()-float(mglass->width())/2) + ,static_cast(p.y()-float(mglass->height())/2)); + mglass->show(); + mglass->updateImage(mglass->x()+mglass->width()/2,mglass->y()+mglass->height()/2); + magnifyingGlassShowed = true; + } +} + +void Viewer::hideMagnifyingGlass() +{ + mglass->hide(); + magnifyingGlassShowed = false; +} + +void Viewer::informationSwitch() +{ + information?informationLabel->hide():informationLabel->show(); + //informationLabel->move(QPoint((width()-informationLabel->width())/2,0)); + information=!information; + Configuration::getConfiguration().setShowInformation(information); + //TODO it shouldn't be neccesary + informationLabel->adjustSize(); + informationLabel->update(); +} + +void Viewer::updateInformation() +{ + if(render->hasLoadedComic()) + { + informationLabel->setText(render->getCurrentPagesInformation()+" - "+QTime::currentTime().toString("HH:mm")); + informationLabel->adjustSize(); + informationLabel->update(); //TODO it shouldn't be neccesary + } +} + +void Viewer::goToFlowSwitch() +{ + goToFlow->isVisible()?animateHideGoToFlow():showGoToFlow(); +} + +void Viewer::translatorSwitch() +{ + translator->isVisible()?animateHideTranslator():animateShowTranslator(); +} + +void Viewer::showGoToFlow() +{ + if(render->hasLoadedComic()) + { + animateShowGoToFlow(); + } +} + +void Viewer::animateShowGoToFlow() +{ + if(goToFlow->isHidden() && showGoToFlowAnimation->state()!=QPropertyAnimation::Running) + { + disconnect(showGoToFlowAnimation,SIGNAL(finished()),goToFlow,SLOT(hide())); + connect(showGoToFlowAnimation,SIGNAL(finished()),this,SLOT(moveCursoToGoToFlow())); + showGoToFlowAnimation->setStartValue(QPoint((width()-goToFlow->width())/2,height()-10)); + showGoToFlowAnimation->setEndValue(QPoint((width()-goToFlow->width())/2,height()-goToFlow->height())); + showGoToFlowAnimation->start(); + goToFlow->centerSlide(render->getIndex()); + goToFlow->setPageNumber(render->getIndex()); + goToFlow->show(); + goToFlow->setFocus(Qt::OtherFocusReason); + } +} + +void Viewer::animateHideGoToFlow() +{ + if(goToFlow->isVisible() && showGoToFlowAnimation->state()!=QPropertyAnimation::Running) + { + connect(showGoToFlowAnimation,SIGNAL(finished()),goToFlow,SLOT(hide())); + disconnect(showGoToFlowAnimation,SIGNAL(finished()),this,SLOT(moveCursoToGoToFlow())); + showGoToFlowAnimation->setStartValue(QPoint((width()-goToFlow->width())/2,height()-goToFlow->height())); + showGoToFlowAnimation->setEndValue(QPoint((width()-goToFlow->width())/2,height())); + showGoToFlowAnimation->start(); + goToFlow->centerSlide(render->getIndex()); + goToFlow->setPageNumber(render->getIndex()); + this->setFocus(Qt::OtherFocusReason); + } +} + +void Viewer::moveCursoToGoToFlow() +{ + //Move cursor to goToFlow widget on show (this avoid hide when mouse is moved) + int y = goToFlow->pos().y(); + int x1 = goToFlow->pos().x(); + int x2 = x1 + goToFlow->width(); + QPoint cursorPos = mapFromGlobal(cursor().pos()); + int cursorX = cursorPos.x(); + int cursorY = cursorPos.y(); + + if(cursorY <= y) + cursorY = y + 10; + if(cursorX <= x1) + cursorX = x1 + 10; + if(cursorX >= x2) + cursorX = x2 - 10; + cursor().setPos(mapToGlobal(QPoint(cursorX,cursorY))); + hideCursorTimer->stop(); + showCursor(); +} + +void Viewer::rotateLeft() +{ + render->rotateLeft(); +} +void Viewer::rotateRight() +{ + render->rotateRight(); +} + +//TODO +void Viewer::setBookmark(bool set) +{ + render->setBookmark(); + if(set) //add bookmark + { + render->setBookmark(); + } + else //remove bookmark + { + render->removeBookmark(); + } +} + +void Viewer::save () +{ + if(render->hasLoadedComic()) + render->save(); +} + +void Viewer::doublePageSwitch() +{ + doublePage = !doublePage; + render->doublePageSwitch(); + Configuration::getConfiguration().setDoublePage(doublePage); +} + +void Viewer::resetContent() +{ + configureContent(tr("Press 'O' to open comic.")); + goToFlow->reset(); + emit reset(); +} + +void Viewer::setLoadingMessage() +{ + if(magnifyingGlassShowed) + { + hideMagnifyingGlass(); + restoreMagnifyingGlass = true; + } + emit(pageAvailable(false)); + configureContent(tr("Loading...please wait!")); +} + +void Viewer::setPageUnavailableMessage() +{ + if(magnifyingGlassShowed) + { + hideMagnifyingGlass(); + restoreMagnifyingGlass = true; + } + emit(pageAvailable(false)); + configureContent(tr("Page not available!")); +} + +void Viewer::configureContent(QString msg) +{ + content->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored); + content->setScaledContents(true); + content->setAlignment(Qt::AlignTop|Qt::AlignHCenter); + content->setText(msg); + content->setFont(QFont("courier new", 12)); + content->adjustSize(); + setFocus(Qt::ShortcutFocusReason); + //emit showingText(); +} + +void Viewer::hideCursor() +{ +#ifdef Q_OS_MAC + setCursor(QCursor(QBitmap(1,1),QBitmap(1,1))); +#else + setCursor(Qt::BlankCursor); +#endif +} +void Viewer::showCursor() +{ + if(drag) + setCursor(Qt::ClosedHandCursor); + else + setCursor(Qt::OpenHandCursor); +} + +void Viewer::updateOptions() +{ + + goToFlow->setFlowType(Configuration::getConfiguration().getFlowType()); + updateBackgroundColor(Configuration::getConfiguration().getBackgroundColor()); + updateContentSize(); + //goToFlow->updateSize(); +} + +void Viewer::updateBackgroundColor(const QColor & color) +{ + QPalette palette; + palette.setColor(backgroundRole(), color); + setPalette(palette); +} + +void Viewer::animateShowTranslator() +{ + if(translator->isHidden() && translatorAnimation->state()!=QPropertyAnimation::Running) + { + disconnect(translatorAnimation,SIGNAL(finished()),translator,SLOT(hide())); + if(translatorXPos == -10000) + translatorXPos = (width()-translator->width())/2; + int x = qMax(0,qMin(translatorXPos,width()-translator->width())); + if(translator->pos().x()<0) + { + translatorAnimation->setStartValue(QPoint(-translator->width(),translator->pos().y())); + } + else + { + translatorAnimation->setStartValue(QPoint(width()+translator->width(),translator->pos().y())); + } + translatorAnimation->setEndValue(QPoint(x,translator->pos().y())); + translatorAnimation->start(); + translator->show(); + translator->setFocus(Qt::OtherFocusReason); + } +} +void Viewer::animateHideTranslator() +{ + if(translator->isVisible() && translatorAnimation->state()!=QPropertyAnimation::Running) + { + connect(translatorAnimation,SIGNAL(finished()),translator,SLOT(hide())); + translatorAnimation->setStartValue(QPoint(translatorXPos = translator->pos().x(),translator->pos().y())); + if((translator->width()/2)+translator->pos().x() <= width()/2) + translatorAnimation->setEndValue(QPoint(-translator->width(),translator->pos().y())); + else + translatorAnimation->setEndValue(QPoint(width()+translator->width(),translator->pos().y())); + translatorAnimation->start(); + this->setFocus(Qt::OtherFocusReason); + } +} + +void Viewer::mousePressEvent ( QMouseEvent * event ) +{ + drag = true; + yDragOrigin = event->y(); + xDragOrigin = event->x(); + setCursor(Qt::ClosedHandCursor); + event->accept(); +} + +void Viewer::mouseReleaseEvent ( QMouseEvent * event ) +{ + drag = false; + setCursor(Qt::OpenHandCursor); + event->accept(); +} + +void Viewer::updateFitToWidthRatio(float ratio) +{ + Configuration::getConfiguration().setAdjustToWidth(true); + adjustToWidthRatio = ratio; + updateContentSize(); +} + +void Viewer::updateConfig(QSettings * settings) +{ + goToFlow->updateConfig(settings); + + QPalette palette; + palette.setColor(backgroundRole(), Configuration::getConfiguration().getBackgroundColor()); + setPalette(palette); +} + +//deprecated +void Viewer::updateImageOptions() +{ + render->reload(); +} + +void Viewer::updateFilters(int brightness, int contrast,int gamma) +{ + render->updateFilters(brightness,contrast,gamma); +} + +void Viewer::setBookmarks() +{ + bd->setBookmarks(*render->getBookmarks()); +} + +void Viewer::showIsCoverMessage() +{ + if(!shouldOpenPrevious) + { + notificationsLabel->setText(tr("Cover!")); + notificationsLabel->flash(); + shouldOpenPrevious = true; + } + else + { + shouldOpenPrevious = false; + emit (openPreviousComic()); + } + + shouldOpenNext = false; //single page comic +} + +void Viewer::showIsLastMessage() +{ + if(!shouldOpenNext) + { + notificationsLabel->setText(tr("Last page!")); + notificationsLabel->flash(); + shouldOpenNext = true; + } + else + { + shouldOpenNext = false; + emit (openNextComic()); + } + + shouldOpenPrevious = false; //single page comic +} + +unsigned int Viewer::getIndex() +{ + return render->getIndex()+1; +} + +int Viewer::getCurrentPageNumber() +{ + return render->getIndex(); +} + +void Viewer::updateComic(ComicDB & comic) +{ + if(render->hasLoadedComic()) + { + //set currentPage + comic.info.currentPage = render->getIndex()+1; + //set bookmarks + Bookmarks * boomarks = render->getBookmarks(); + QList boomarksList = boomarks->getBookmarkPages(); + int numBookmarks = boomarksList.size(); + if(numBookmarks > 0) + comic.info.bookmark1 = boomarksList[0]; + if(numBookmarks > 1) + comic.info.bookmark2 = boomarksList[1]; + if(numBookmarks > 2) + comic.info.bookmark3 = boomarksList[2]; + //set filters + //TODO: avoid use settings for this... + QSettings settings(YACReader::getSettingsPath()+"/YACReader.ini",QSettings::IniFormat); + int brightness = settings.value(BRIGHTNESS,0).toInt(); + int contrast = settings.value(CONTRAST,100).toInt(); + int gamma = settings.value(GAMMA,100).toInt(); + + if(brightness != 0 || comic.info.brightness!=-1) + comic.info.brightness = brightness; + if(contrast != 100 || comic.info.contrast!=-1) + comic.info.contrast = contrast; + if(gamma != 100 || comic.info.gamma!=-1) + comic.info.gamma = gamma; + } + + +} diff --git a/YACReader/viewer.h b/YACReader/viewer.h new file mode 100644 index 00000000..4abd7ec7 --- /dev/null +++ b/YACReader/viewer.h @@ -0,0 +1,164 @@ +#ifndef __VIEWER_H +#define __VIEWER_H + +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +class ComicDB; +class Comic; +class MagnifyingGlass; +class GoToFlow; +class BookmarksDialog; +class Render; +class GoToDialog; +class YACReaderTranslator; +class GoToFlowWidget; +class Bookmarks; +class PageLabelWidget; +class NotificationsLabelWidget; + + class Viewer : public QScrollArea + { + Q_OBJECT + public: + bool fullscreen; //TODO, change by the right use of windowState(); + public slots: + void prepareForOpening(); + void open(QString pathFile, int atPage = -1); + void open(QString pathFile, const ComicDB & comic); + void prev(); + void next(); + void showGoToDialog(); + void goTo(unsigned int page); + void updatePage(); + void updateContentSize(); + void updateVerticalScrollBar(); + void updateOptions(); + void scrollDown(); + void scrollUp(); + void magnifyingGlassSwitch(); + void showMagnifyingGlass(); + void hideMagnifyingGlass(); + void informationSwitch(); + void updateInformation(); + void goToFlowSwitch(); + void showGoToFlow(); + void moveCursoToGoToFlow(); + void animateShowGoToFlow(); + void animateHideGoToFlow(); + void rotateLeft(); + void rotateRight(); + bool magnifyingGlassIsVisible() {return magnifyingGlassShowed;} + void setBookmark(bool); + void save(); + void doublePageSwitch(); + void resetContent(); + void setLoadingMessage(); + void setPageUnavailableMessage(); + void configureContent(QString msg); + void hideCursor(); + void showCursor(); + void createConnections(); + void translatorSwitch(); + void animateShowTranslator(); + void animateHideTranslator(); +virtual void mousePressEvent ( QMouseEvent * event ); +virtual void mouseReleaseEvent ( QMouseEvent * event ); + void updateBackgroundColor(const QColor & color); + void updateFitToWidthRatio(float ratio); + void updateConfig(QSettings * settings); + void showMessageErrorOpening(); + void showMessageErrorOpening(QString); + void processCRCError(QString message); + void setBookmarks(); + //deprecated + void updateImageOptions(); + void updateFilters(int brightness, int contrast,int gamma); + void showIsCoverMessage(); + void showIsLastMessage(); + int getCurrentPageNumber(); + + private: + bool information; + bool doublePage; + PageLabelWidget * informationLabel; + //QTimer * scroller; + QPropertyAnimation * verticalScroller; + int posByStep; + int nextPos; + GoToFlowWidget * goToFlow; + QPropertyAnimation * showGoToFlowAnimation; + GoToDialog * goToDialog; + //!Image properties + float adjustToWidthRatio; + //! Comic + //Comic * comic; + int index; + QPixmap *currentPage; + BookmarksDialog * bd; + bool wheelStop; + Render * render; + QTimer * hideCursorTimer; + int direction; + bool drag; + int numScrollSteps; + + //!Widgets + QLabel *content; + + YACReaderTranslator * translator; + int translatorXPos; + QPropertyAnimation * translatorAnimation; + + int yDragOrigin; + int xDragOrigin; + + NotificationsLabelWidget * notificationsLabel; + + bool shouldOpenNext; + bool shouldOpenPrevious; + + private: + //!Magnifying glass + MagnifyingGlass *mglass; + bool magnifyingGlassShowed; + bool restoreMagnifyingGlass; + + //! Manejadores de evento: + void keyPressEvent(QKeyEvent * event); + void resizeEvent(QResizeEvent * event); + void wheelEvent(QWheelEvent * event); + void mouseMoveEvent(QMouseEvent * event); + + public: + Viewer(QWidget * parent = 0); + ~Viewer(); + void toggleFullScreen(); + const QPixmap * pixmap(); + //Comic * getComic(){return comic;} + const BookmarksDialog * getBookmarksDialog(){return bd;} + //returns the current index starting in 1 [1,nPages] + unsigned int getIndex(); + void updateComic(ComicDB & comic); + signals: + void backgroundChanges(); + void pageAvailable(bool); + void pageIsBookmark(bool); + void reset(); + void openNextComic(); + void openPreviousComic(); + }; + +#endif diff --git a/YACReader/width_slider.cpp b/YACReader/width_slider.cpp new file mode 100644 index 00000000..0ffbf101 --- /dev/null +++ b/YACReader/width_slider.cpp @@ -0,0 +1,78 @@ +#include "width_slider.h" + +#include +#include +#include +#include +#include "configuration.h" + +YACReaderSliderAction::YACReaderSliderAction (QWidget * parent) + :QWidgetAction (parent) { + + QWidget* pWidget = new QWidget (NULL); + QHBoxLayout* pLayout = new QHBoxLayout(); + + pLayout->addStretch(); + + percentageLabel = new QLabel ("100%"); + percentageLabel->setStyleSheet("QLabel { color : white; }"); + percentageLabel->setAlignment(Qt::AlignHCenter|Qt::AlignVCenter); + pLayout->addWidget (percentageLabel); + slider = new QSlider(NULL); + slider->setOrientation(Qt::Horizontal); + pLayout->addWidget (slider); + + QString sliderCSS = + + "QSlider::sub-page:horizontal {background-image: url(:/images/sliderSubPage.png); border: 0px; margin-left: 18px;}" + "QSlider::add-page:horizontal {background-image: url(:/images/sliderAddPage.png); border: 0px; margin-right: 25px;}" + "QSlider::handle:horizontal {image: url(:/images/sliderHandle.png); width: 31px;height:45px; }" + "QSlider::groove:horizontal {border-image:url(:/images/sliderGround.png); border-left:-2px; border-right:0;}" + ; + slider->setStyleSheet(sliderCSS); + slider->setFixedSize(218,45); + + QLabel* imgLabel = new QLabel(pWidget); + QPixmap p(":/images/sliderBackground.png"); + imgLabel->resize(p.size()); + imgLabel->setPixmap(p); + + pLayout->setMargin(0); + pLayout->setSpacing(0); + + pLayout->setStretchFactor(percentageLabel,1); + pLayout->setStretchFactor(slider,0); + + + pWidget->setLayout (pLayout); + pWidget->setAutoFillBackground(false); + + pWidget->setMinimumSize(276,45); + + setDefaultWidget(pWidget); + + slider->setMinimum(50); + slider->setMaximum(100); + slider->setPageStep(5); + + int value = Configuration::getConfiguration().getFitToWidthRatio()*100; + slider->setValue(value); + percentageLabel->setText(QString("%1 %").arg(value)); + connect(slider,SIGNAL(valueChanged(int)),this,SLOT(updateText(int))); + + +} + +void YACReaderSliderAction::updateText(int value) +{ + percentageLabel->setText(QString("%1 %").arg(value)); + Configuration::getConfiguration().setFitToWidthRatio(value/100.0); + emit(fitToWidthRatioChanged(value / 100.0f)); +} + +void YACReaderSliderAction::updateFitToWidthRatio(float v) +{ + int value = v*100; + slider->setValue(value); + percentageLabel->setText(QString("%1 %").arg(value)); +} \ No newline at end of file diff --git a/YACReader/width_slider.h b/YACReader/width_slider.h new file mode 100644 index 00000000..ea37fe3e --- /dev/null +++ b/YACReader/width_slider.h @@ -0,0 +1,29 @@ +#ifndef WIDTH_SLIDER_H +#define WIDTH_SLIDER_H + +#include + +class QLabel; +class QSlider; + +class YACReaderSliderAction : public QWidgetAction +{ + Q_OBJECT +private: + QLabel * percentageLabel; + QSlider * slider; + +public: + + YACReaderSliderAction (QWidget * parent = 0); + +public slots: + void updateText(int value); + void updateFitToWidthRatio(float v); + + +signals: + void fitToWidthRatioChanged(float value); +}; + +#endif diff --git a/YACReader/yacreader_es.qm b/YACReader/yacreader_es.qm new file mode 100644 index 0000000000000000000000000000000000000000..b2cee2ed4a0f86a1386c0169d06a5e03cb6a3aa2 GIT binary patch literal 13554 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 + + + + BookmarksDialog + + + Lastest Page + Última página + + + + Close + Cerrar + + + + Click on any image to go to the bookmark + Pulsa en cualquier imagen para ir al marcador + + + + + Loading... + Cargando... + + + + FileComic + + + Unknown error opening the file + Error desconocido abriendo el archivo + + + + 7z not found + 7z no encontrado + + + + Format not supported + Formato no soportado + + + + CRC error on page (%1): some of the pages will not be displayed correctly + Error CRC en la página (%1): algunas de las páginas no se mostrarán correctamente + + + + GoToDialog + + + Page : + Página : + + + + Go To + Ir a + + + + Cancel + Cancelar + + + + + Total pages : + Páginas totales: + + + + Go to... + Ir a... + + + + GoToFlowToolBar + + + Page : + Página : + + + + HelpAboutDialog + + + About + Acerca de + + + + Help + Ayuda + + + + MainWindowViewer + + + &Open + &Abrir + + + + O + O + + + + Open a comic + Abrir cómic + + + + Open Folder + Abrir carpeta + + + + Ctrl+O + Ctrl+O + + + + Open image folder + Open images in a folder + Abrir carpeta de imágenes + + + + Save + Guardar + + + + + Save current page + Guardar la página actual + + + + Previous Comic + Cómic anterior + + + + Open previous comic + Abrir cómic anterior + + + + Next Comic + Siguiente Cómic + + + + Open next comic + Abrir siguiente cómic + + + + &Previous + A&nterior + + + + Go to previous page + Ir a la página anterior + + + + &Next + Siguie&nte + + + + Go to next page + Ir a la página siguiente + + + + Fit Width + Ajustar anchura + + + + Fit image to height + Ajustar página a lo alto + + + + Fit Height + Ajustar altura + + + + Fit image to width + Ajustar página a lo ancho + + + + Rotate image to the left + Rotar imagen a la izquierda + + + + L + L + + + + Rotate image to the right + Rotar imagen a la derecha + + + + R + R + + + + Double page mode + Modo a doble página + + + + Switch to double page mode + Cambiar a modo de doble página + + + + D + D + + + + Go To + Ir a + + + + G + G + + + + Go to page ... + Ir a página... + + + + Options + Opciones + + + + C + C + + + + YACReader options + Opciones de YACReader + + + + Help + Ayuda + + + + Help, About YACReader + Ayuda, Sobre YACReader + + + + Magnifying glass + Lupa + + + + Switch Magnifying glass + Lupa On/Off + + + + Z + Z + + + + Set bookmark + Añadir marcador + + + + Set a bookmark on the current page + Añadir un marcador en la página actual + + + + Show bookmarks + Mostrar marcadores + + + + Show the bookmarks of the current comic + Mostrar los marcadores del cómic actual + + + + M + M + + + + Show keyboard shortcuts + Mostrar atajos de teclado + + + + Show Info + Mostrar información + + + + I + I + + + + Close + Cerrar + + + + Show Dictionary + Mostrar diccionario + + + + Always on top + Siempre visible + + + + Show full size + Mostrar a tamaño original + + + + Show go to flow + Mostrar flow ir a + + + + &File + &Archivo + + + + File + Archivo + + + + Open Comic + Abrir cómic + + + + Comic files + Archivos de cómic + + + + Remind me in 14 days + Recordar en 14 días + + + + Not now + Ahora no + + + + Open folder + Abrir carpeta + + + + Image files (*.jpg) + Archivos de imagen (*.jpg) + + + + page_%1.jpg + página_%1.jpg + + + + There is a new version avaliable + Hay una nueva versión disponible + + + + Do you want to download the new version? + ¿Desea descargar la nueva versión? + + + + OptionsDialog + + + "Go to flow" size + Tamaño de "Go to flow" + + + + My comics path + Ruta a mis cómics + + + + Page width stretch + Ajuste en anchura de la página + + + + Background color + Color de fondo + + + + Choose + Elegir + + + + Restart is needed + Es necesario reiniciar + + + + Brightness + Brillo + + + + Contrast + Contraste + + + + Gamma + Gamma + + + + Reset + Reset + + + + Image options + Opciones de imagen + + + + General + General + + + + Page Flow + Page Flow + + + + Image adjustment + Ajustes de imagen + + + + Options + Opciones + + + + Comics directory + Directorio de cómics + + + + QObject + + + 7z lib not found + 7z lib no encontrado + + + + unable to load 7z lib from ./utils + imposible cargar 7z lib de ./utils + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + Atajos de teclado de YACReader + + + + Close + Cerrar + + + + Keyboard Shortcuts + Atajos de teclado + + + + Viewer + + + + Press 'O' to open comic. + Pulsa 'O' para abrir un fichero. + + + + Not found + No encontrado + + + + Comic not found + Cómic no encontrado + + + + Error opening comic + Error abriendo cómic + + + + CRC Error + Error CRC + + + + Page not available! + ¡Página no disponible! + + + + Cover! + ¡Portada! + + + + Last page! + ¡Última página! + + + + Loading...please wait! + Cargando...espere, por favor! + + + + YACReaderFieldEdit + + + + Click to overwrite + Click para sobreescribir + + + + Restore to default + Restaurar valor por defecto + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Click para sobreescribir + + + + Restore to default + Restaurar valor por defecto + + + + YACReaderFlowConfigWidget + + + How to show covers: + Cómo mostrar las portadas: + + + + CoverFlow look + Tipo CoverFlow + + + + Stripe look + Tipo tira + + + + Overlapped Stripe look + Tipo tira solapada + + + + YACReaderGLFlowConfigWidget + + + Presets: + Predefinidos: + + + + Classic look + Tipo clásico + + + + Stripe look + Tipo tira + + + + Overlapped Stripe look + Tipo tira solapada + + + + Modern look + Tipo moderno + + + + Roulette look + Tipo ruleta + + + + Show advanced settings + Opciones avanzadas + + + + Custom: + Personalizado: + + + + View angle + Ãngulo de vista + + + + Position + Posición + + + + Cover gap + Hueco entre portadas + + + + Central gap + Hueco central + + + + Zoom + Zoom + + + + Y offset + Desplazamiento en Y + + + + Z offset + Desplazamiento en Z + + + + Cover Angle + Ãngulo de las portadas + + + + Visibility + Visibilidad + + + + Light + Luz + + + + Max angle + Ãngulo máximo + + + + Low Performance + Rendimiento bajo + + + + High Performance + Alto rendimiento + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Utilizar VSync (mejora la calidad de imagen en pantalla completa, peor rendimiento) + + + + Performance: + Rendimiento: + + + + YACReaderOptionsDialog + + + Save + Guardar + + + + Cancel + Cancelar + + + + Use hardware acceleration (restart needed) + Utilizar aceleración por hardware (necesario reiniciar) + + + + YACReaderTranslator + + + YACReader translator + Traductor YACReader + + + + + Translation + Traducción + + + + clear + limpiar + + + + Service not available + Servicio no disponible + + + diff --git a/YACReader/yacreader_files.qrc b/YACReader/yacreader_files.qrc new file mode 100644 index 00000000..68d07c60 --- /dev/null +++ b/YACReader/yacreader_files.qrc @@ -0,0 +1,12 @@ + + + ../files/about.html + ../files/helpYACReader.html + ../files/shortcuts.html + + + + ../files/about_es_ES.html + ../files/helpYACReader_es_ES.html + + diff --git a/YACReader/yacreader_fr.ts b/YACReader/yacreader_fr.ts new file mode 100644 index 00000000..29539a7d --- /dev/null +++ b/YACReader/yacreader_fr.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + Aller à la dernière page + + + + Close + Fermer + + + + Click on any image to go to the bookmark + Cliquez sur une image pour aller au marque-page + + + + + Loading... + Chargement... + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z introuvable + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + GoToDialog + + + Page : + Page : + + + + Go To + Aller à + + + + Cancel + Annuler + + + + + Total pages : + Nombre de pages : + + + + Go to... + Aller à... + + + + GoToFlowToolBar + + + Page : + Page : + + + + HelpAboutDialog + + + About + A propos + + + + Help + Aide + + + + MainWindowViewer + + + &Open + &Ouvrir + + + + O + O + + + + Open a comic + Ouvrir un comic + + + + Open Folder + Ouvrir un dossier + + + + Ctrl+O + Ctrl+O + + + + Open image folder + Ouvrir un dossier d'images + + + + Save + Sauvegarder + + + + + Save current page + Sauvegarder la page actuelle + + + + Previous Comic + Comic précédent + + + + Open previous comic + Ouvrir le comic précédent + + + + Next Comic + Comic suivant + + + + Open next comic + Ouvrir le livre suivant + + + + &Previous + &Précédent + + + + Go to previous page + Aller à la page précédente + + + + &Next + &Suivant + + + + Go to next page + Aller à la page suivante + + + + Fit Width + Ajuster la largeur + + + + Fit image to height + Ajuster l'image à la hauteur + + + + Fit Height + + + + + Fit image to width + Ajuster l'image à la largeur + + + + Rotate image to the left + Rotation sur la gauche + + + + L + L + + + + Rotate image to the right + Rotation sur la droite + + + + R + R + + + + Double page mode + Mode double page + + + + Switch to double page mode + Passer en mode double page + + + + D + D + + + + Go To + Aller à + + + + G + G + + + + Go to page ... + Aller à la page ... + + + + Options + Options + + + + C + C + + + + YACReader options + Options de YACReader + + + + Help + Aide + + + + Help, About YACReader + Aide, à propos de YACReader + + + + Magnifying glass + Loupe + + + + Switch Magnifying glass + Utiliser la loupe + + + + Z + Z + + + + Set bookmark + Placer un marque-page + + + + Set a bookmark on the current page + Placer un marque-page à la page actuelle + + + + Show bookmarks + Voir les marque-pages + + + + Show the bookmarks of the current comic + Voir les marque-pages de ce comic + + + + M + M + + + + Show keyboard shortcuts + Voir les raccourcis + + + + Show Info + Voir les infos + + + + I + I + + + + Close + Fermer + + + + Show Dictionary + Dictionnaire + + + + Always on top + Toujours au dessus + + + + Show full size + Plein écran + + + + Show go to flow + Afficher le go to flow + + + + &File + &Fichier + + + + File + + + + + Open Comic + Ouvrir le comic + + + + Comic files + Comic files + + + + Open folder + Ouvirir le dossier + + + + Image files (*.jpg) + Image files (*.jpg) + + + + page_%1.jpg + page_%1.jpg + + + + There is a new version avaliable + Une nouvelle version est disponible + + + + Do you want to download the new version? + Voulez-vous télécharger la nouvelle version? + + + + Remind me in 14 days + + + + + Not now + + + + + OptionsDialog + + + "Go to flow" size + Taille du "Go to flow" + + + + My comics path + Chemin de mes comics + + + + Page width stretch + Etirer la page + + + + Background color + Couleur d'arrière plan + + + + Choose + Choisir + + + + Restart is needed + Redémarrage nécessaire + + + + Brightness + Luminosité + + + + Contrast + Contraste + + + + Gamma + Gamma + + + + Reset + Reset + + + + Image options + Option de l'image + + + + General + Général + + + + Page Flow + Page Flow + + + + Image adjustment + Ajustement de l'image + + + + Options + Options + + + + Comics directory + Répertoire des comics + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + Raccourcis clavier de YACReader + + + + Close + Fermer + + + + Keyboard Shortcuts + Raccourcis clavier + + + + Viewer + + + + Press 'O' to open comic. + Appuyez sur "O" pour ouvrir un comic. + + + + Not found + Introuvable + + + + Comic not found + Comic introuvable + + + + Error opening comic + + + + + CRC Error + + + + + Loading...please wait! + Chargement...Patientez! + + + + Page not available! + + + + + Cover! + Couverture! + + + + Last page! + Dernière page! + + + + YACReaderFieldEdit + + + + Click to overwrite + Cliquez pour écraser + + + + Restore to default + Rétablir les paramètres par défaut + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Cliquez pour écraser + + + + Restore to default + Rétablir les paramètres par défaut + + + + YACReaderFlowConfigWidget + + + How to show covers: + Comment voir les couvertures: + + + + CoverFlow look + Vue CoverFlow + + + + Stripe look + Vue alignée + + + + Overlapped Stripe look + Vue superposée + + + + YACReaderGLFlowConfigWidget + + + Presets: + Réglages: + + + + Classic look + Vue classique + + + + Stripe look + Vue alignée + + + + Overlapped Stripe look + Vue superposée + + + + Modern look + Vue moderne + + + + Roulette look + Vue roulette + + + + Show advanced settings + Voir les paramètres avancés + + + + Custom: + Personnalisation: + + + + View angle + Angle de vue + + + + Position + Position + + + + Cover gap + Espace entre les couvertures + + + + Central gap + Espace couverture centrale + + + + Zoom + Zoom + + + + Y offset + Axe Y + + + + Z offset + Axe Z + + + + Cover Angle + Angle des couvertures + + + + Visibility + Visibilité + + + + Light + Lumière + + + + Max angle + Angle Maximum + + + + Low Performance + Faible performance + + + + High Performance + Haute performance + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Utiliser VSync (Améliore la qualité d'image en mode plein écran, ralentit la performance) + + + + Performance: + Performance: + + + + YACReaderOptionsDialog + + + Save + Sauvegarder + + + + Cancel + Annuler + + + + Use hardware acceleration (restart needed) + Utiliser accélération hardware (nécessite le redémarrage) + + + + YACReaderTranslator + + + YACReader translator + + + + + + Translation + + + + + clear + + + + + Service not available + + + + diff --git a/YACReader/yacreader_images.qrc b/YACReader/yacreader_images.qrc new file mode 100644 index 00000000..ca8f9714 --- /dev/null +++ b/YACReader/yacreader_images.qrc @@ -0,0 +1,87 @@ + + + ../images/icon.png + ../images/goto.png + ../images/comicFolder.png + ../images/flow1.png + ../images/flow2.png + ../images/flow3.png + ../images/flow4.png + ../images/flow5.png + ../images/notCover.png + ../images/shortcuts.png + ../images/close.png + ../images/up.png + ../images/down.png + ../images/numPagesLabel.png + ../images/numPagesLabelMedium.png + ../images/numPagesLabelBig.png + ../images/imgTopLeft.png + ../images/imgTopMiddle.png + ../images/imgTopRight.png + ../images/imgBottomLeft.png + ../images/imgBottomMiddle.png + ../images/imgBottomRight.png + ../images/imgEdit.png + ../images/imgCenterSlide.png + ../images/imgGoToSlide.png + ../images/imgCenterSlidePressed.png + ../images/imgGoToSlidePressed.png + ../images/sliderBackground.png + ../images/sliderGround.png + ../images/sliderSubPage.png + ../images/sliderAddPage.png + ../images/sliderHandle.png + + ../images/helpImages/open.png + ../images/helpImages/openFolder.png + ../images/helpImages/next.png + ../images/helpImages/prev.png + ../images/helpImages/icon.png + ../images/helpImages/zoom.png + ../images/helpImages/fit.png + ../images/helpImages/goto.png + ../images/helpImages/help.png + ../images/helpImages/center.png + ../images/helpImages/options.png + ../images/helpImages/comicFolder.png + ../images/helpImages/save.png + ../images/helpImages/rotateL.png + ../images/helpImages/rotateR.png + ../images/helpImages/flow1.png + ../images/helpImages/flow2.png + ../images/helpImages/flow3.png + ../images/helpImages/bookmark.png + ../images/helpImages/setBookmark.png + ../images/helpImages/notCover.png + ../images/helpImages/previousComic.png + ../images/helpImages/nextComic.png + ../images/helpImages/deleteLibrary.png + ../images/helpImages/properties.png + ../images/helpImages/doublePage.png + ../images/helpImages/shortcuts.png + ../images/helpImages/keyboard.png + ../images/helpImages/mouse.png + ../images/helpImages/speaker.png + ../images/defaultCover.png + ../images/onStartFlowSelection.png + ../images/onStartFlowSelection_es.png + ../images/useNewFlowButton.png + ../images/useOldFlowButton.png + ../images/notificationsLabel.png + ../images/fromTo.png + ../images/dropDownArrow.png + ../images/translatorSearch.png + ../images/speaker.png + ../images/clear_shortcut.png + ../images/accept_shortcut.png + ../images/shortcuts_group_comics.png + ../images/shortcuts_group_folders.png + ../images/shortcuts_group_general.png + ../images/shortcuts_group_libraries.png + ../images/shortcuts_group_mglass.png + ../images/shortcuts_group_page.png + ../images/shortcuts_group_reading.png + ../images/shortcuts_group_visualization.png + + diff --git a/YACReader/yacreader_images_osx.qrc b/YACReader/yacreader_images_osx.qrc new file mode 100644 index 00000000..6d589cf3 --- /dev/null +++ b/YACReader/yacreader_images_osx.qrc @@ -0,0 +1,28 @@ + + +../images/viewer_toolbar/bookmark_osx.png +../images/viewer_toolbar/close_osx.png +../images/viewer_toolbar/doublePage_osx.png +../images/viewer_toolbar/flow_osx.png +../images/viewer_toolbar/full_osx.png +../images/viewer_toolbar/goto_osx.png +../images/viewer_toolbar/help_osx.png +../images/viewer_toolbar/info_osx.png +../images/viewer_toolbar/magnifyingGlass_osx.png +../images/viewer_toolbar/next_osx.png +../images/viewer_toolbar/open_osx.png +../images/viewer_toolbar/openFolder_osx.png +../images/viewer_toolbar/openNext_osx.png +../images/viewer_toolbar/openPrevious_osx.png +../images/viewer_toolbar/options_osx.png +../images/viewer_toolbar/previous_osx.png +../images/viewer_toolbar/rotateL_osx.png +../images/viewer_toolbar/rotateR_osx.png +../images/viewer_toolbar/save_osx.png +../images/viewer_toolbar/shortcuts_osx.png +../images/viewer_toolbar/showBookmarks_osx.png +../images/viewer_toolbar/toHeight_osx.png +../images/viewer_toolbar/toWidth_osx.png +../images/viewer_toolbar/translator_osx.png + + diff --git a/YACReader/yacreader_images_win.qrc b/YACReader/yacreader_images_win.qrc new file mode 100644 index 00000000..3476e623 --- /dev/null +++ b/YACReader/yacreader_images_win.qrc @@ -0,0 +1,28 @@ + + + ../images/viewer_toolbar/bookmark.png + ../images/viewer_toolbar/close.png + ../images/viewer_toolbar/doublePage.png + ../images/viewer_toolbar/flow.png + ../images/viewer_toolbar/full.png + ../images/viewer_toolbar/goto.png + ../images/viewer_toolbar/help.png + ../images/viewer_toolbar/info.png + ../images/viewer_toolbar/magnifyingGlass.png + ../images/viewer_toolbar/next.png + ../images/viewer_toolbar/open.png + ../images/viewer_toolbar/openFolder.png + ../images/viewer_toolbar/openNext.png + ../images/viewer_toolbar/openPrevious.png + ../images/viewer_toolbar/options.png + ../images/viewer_toolbar/previous.png + ../images/viewer_toolbar/rotateL.png + ../images/viewer_toolbar/rotateR.png + ../images/viewer_toolbar/save.png + ../images/viewer_toolbar/shortcuts.png + ../images/viewer_toolbar/showBookmarks.png + ../images/viewer_toolbar/toHeight.png + ../images/viewer_toolbar/toWidth.png + ../images/viewer_toolbar/translator.png + + diff --git a/YACReader/yacreader_local_client.cpp b/YACReader/yacreader_local_client.cpp new file mode 100644 index 00000000..a6175b99 --- /dev/null +++ b/YACReader/yacreader_local_client.cpp @@ -0,0 +1,171 @@ +#include "yacreader_local_client.h" +#include "comic_db.h" +#include "yacreader_global.h" + +#include + +#include "QsLog.h" + +using namespace YACReader; + +YACReaderLocalClient::YACReaderLocalClient(QObject *parent) : + QObject(parent) +{ + localSocket = new QLocalSocket(this); + + //connect(localSocket, SIGNAL(readyRead()), this, SLOT(readMessage())); + + /*connect(socket, SIGNAL(error(QLocalSocket::LocalSocketError)), + this, SLOT(displayError(QLocalSocket::LocalSocketError)));*/ +} +YACReaderLocalClient::~YACReaderLocalClient() +{ + delete localSocket; +} +//información de comic recibida... +void YACReaderLocalClient::readMessage() +{ + +} +#include + +bool YACReaderLocalClient::requestComicInfo(quint64 libraryId, ComicDB & comic, QList & siblings) +{ + localSocket->connectToServer(YACREADERLIBRARY_GUID); + if(localSocket->isOpen()) + { + QByteArray block; + QDataStream out(&block, QIODevice::WriteOnly); + out.setVersion(QDataStream::Qt_4_8); + out << (quint32)0; + out << (quint8)YACReader::RequestComicInfo; + out << libraryId; + out << comic; + out.device()->seek(0); + out << (quint32)(block.size() - sizeof(quint32)); + + int written = 0; + int previousWritten = 0; + quint16 tries = 0; + while(written != block.size() && tries < 200) + { + written += localSocket->write(block); + localSocket->flush(); + if(written == previousWritten) //no bytes were written + tries++; + previousWritten = written; + } + if(tries == 200) + { + localSocket->close(); + QLOG_ERROR() << "Requesting Comic Info : unable to send request"; + return false; + } + + localSocket->waitForBytesWritten(2000); + + //QByteArray data; + tries = 0; + int dataAvailable = 0; + QByteArray packageSize; + localSocket->waitForReadyRead(1000); + while(packageSize.size() < sizeof(quint32) && tries < 20) + { + packageSize.append(localSocket->read(sizeof(quint32) - packageSize.size())); + localSocket->waitForReadyRead(100); + if(dataAvailable == packageSize.size()) + { + tries++; //TODO apply 'tries' fix + } + dataAvailable = packageSize.size(); + } + if(tries == 20) + { + localSocket->close(); + QLOG_ERROR() << "Requesting Comic Info : unable to read package size"; + return false; + } + QDataStream sizeStream(packageSize);//localSocket->read(sizeof(quint32))); + sizeStream.setVersion(QDataStream::Qt_4_8); + quint32 totalSize = 0; + sizeStream >> totalSize; + + QByteArray data; + + tries = 0; + int dataRead = 0; + localSocket->waitForReadyRead(1000); + while(data.length() < totalSize && tries < 20 ) + { + data.append(localSocket->readAll()); + if(data.length() < totalSize) + localSocket->waitForReadyRead(100); + if(data.length() == dataRead) + tries++; + dataRead = data.length(); + } + + if(tries == 20) + { + localSocket->close(); + QLOG_ERROR() << "Requesting Comic Info : unable to read data (" << data.length() << "," << totalSize << ")"; + return false; + } + + QDataStream dataStream(data); + dataStream >> comic; + dataStream >> siblings; + localSocket->close(); + return true; + } + else + { + QLOG_ERROR() << "Requesting Comic Info : unable to connect to the server"; + return false; + } +} + +bool YACReaderLocalClient::sendComicInfo(quint64 libraryId, ComicDB & comic) +{ + localSocket->connectToServer(YACREADERLIBRARY_GUID); + if(localSocket->isOpen()) + { + //QLOG_INFO() << "Connection opened for sending ComicInfo"; + QByteArray block; + QDataStream out(&block, QIODevice::WriteOnly); + out.setVersion(QDataStream::Qt_4_8); + out << (quint32)0; + out << (quint8)YACReader::SendComicInfo; + out << libraryId; + out << comic; + out.device()->seek(0); + out << (quint32)(block.size() - sizeof(quint32)); + + int written, previousWritten; + written = previousWritten = 0; + int tries = 0; + while(written != block.size() && tries < 100) + { + written += localSocket->write(block); + if(written == previousWritten) + tries++; + previousWritten = written; + } + localSocket->waitForBytesWritten(2000); + localSocket->close(); + //QLOG_INFO() << QString("Sending Comic Info : writen data (%1,%2)").arg(written).arg(block.size()); + if(tries == 100 && written != block.size()) + { + emit finished(); + QLOG_ERROR() << QString("Sending Comic Info : unable to write data (%1,%2)").arg(written).arg(block.size()); + return false; + } + emit finished(); + return true; + } + + emit finished(); + QLOG_ERROR() << "Sending Comic Info : unable to connect to the server"; + return false; + +} diff --git a/YACReader/yacreader_local_client.h b/YACReader/yacreader_local_client.h new file mode 100644 index 00000000..001cb1e7 --- /dev/null +++ b/YACReader/yacreader_local_client.h @@ -0,0 +1,27 @@ +#ifndef YACREADER_LOCAL_CLIENT_H +#define YACREADER_LOCAL_CLIENT_H + +#include + +class QLocalSocket; +class ComicDB; + +class YACReaderLocalClient : public QObject +{ + Q_OBJECT +public: + explicit YACReaderLocalClient(QObject *parent = 0); + ~YACReaderLocalClient(); +signals: + void finished(); +public slots: + void readMessage(); + bool requestComicInfo(quint64 libraryId, ComicDB & comic,QList & siblings); + bool sendComicInfo(quint64 libraryId, ComicDB & comic); + +private: + QLocalSocket * localSocket; + +}; + +#endif // YACREADER_LOCAL_CLIENT_H diff --git a/YACReader/yacreader_nl.ts b/YACReader/yacreader_nl.ts new file mode 100644 index 00000000..0404c9f5 --- /dev/null +++ b/YACReader/yacreader_nl.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + Laatste Pagina + + + + Close + Sluiten + + + + Click on any image to go to the bookmark + Klik op een afbeelding om naar de bladwijzer te gaan + + + + + Loading... + Inladen... + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7Z Archiefbestand niet gevonden + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + GoToDialog + + + Page : + Pagina : + + + + Go To + Ga Naar + + + + Cancel + Annuleren + + + + + Total pages : + Totaal aantal pagina's : + + + + Go to... + Ga naar... + + + + GoToFlowToolBar + + + Page : + Pagina : + + + + HelpAboutDialog + + + About + Over + + + + Help + Help + + + + MainWindowViewer + + + &Open + &Open + + + + O + O + + + + Open a comic + Open een strip + + + + Open Folder + Map Openen + + + + Ctrl+O + Ctrl+O + + + + Open image folder + Open afbeeldings map + + + + Save + Bewaar + + + + + Save current page + Bewaren huidige pagina + + + + Previous Comic + Vorige Strip + + + + Open previous comic + Open de vorige strip + + + + Next Comic + Volgende Strip + + + + Open next comic + Open volgende strip + + + + &Previous + &Vorige + + + + Go to previous page + Ga naar de vorige pagina + + + + &Next + &Volgende + + + + Go to next page + Ga naar de volgende pagina + + + + Fit Width + Vensterbreedte aanpassen + + + + Fit image to height + Afbeelding aanpassen aan hoogte + + + + Fit Height + + + + + Fit image to width + Afbeelding aanpassen aan breedte + + + + Rotate image to the left + Links omdraaien + + + + L + L + + + + Rotate image to the right + Rechts omdraaien + + + + R + R + + + + Double page mode + Dubbele bladzijde modus + + + + Switch to double page mode + Naar dubbele bladzijde modus + + + + D + D + + + + Go To + Ga Naar + + + + G + G + + + + Go to page ... + Ga naar bladzijde ... + + + + Options + Opties + + + + C + C + + + + YACReader options + YACReader opties + + + + Help + Help + + + + Help, About YACReader + Help, Over YACReader + + + + Magnifying glass + Vergrootglas + + + + Switch Magnifying glass + Overschakelen naar Vergrootglas + + + + Z + Z + + + + Set bookmark + Bladwijzer instellen + + + + Set a bookmark on the current page + Een bladwijzer toevoegen aan de huidige pagina + + + + Show bookmarks + Bladwijzers weergeven + + + + Show the bookmarks of the current comic + Toon de bladwijzers van de huidige strip + + + + M + M + + + + Show keyboard shortcuts + Toon de sneltoetsen + + + + Show Info + Info tonen + + + + I + I + + + + Close + Sluiten + + + + Show Dictionary + Woordenlijst weergeven + + + + Always on top + Altijd op voorgrond + + + + Show full size + Volledig Scherm + + + + Show go to flow + Toon ga naar de Omslagbrowser + + + + &File + &Bestand + + + + File + + + + + Open Comic + Open een Strip + + + + Comic files + Strip bestanden + + + + Open folder + Open een Map + + + + Image files (*.jpg) + Afbeelding bestanden (*.jpg) + + + + page_%1.jpg + pagina_%1.jpg + + + + There is a new version avaliable + Er is een nieuwe versie beschikbaar + + + + Do you want to download the new version? + Wilt u de nieuwe versie downloaden? + + + + Remind me in 14 days + + + + + Not now + + + + + OptionsDialog + + + "Go to flow" size + "Naar Omslagbrowser" afmetingen + + + + My comics path + Pad naar mijn strips + + + + Page width stretch + Pagina breedte + + + + Background color + Achtergrondkleur + + + + Choose + Kies + + + + Restart is needed + Herstart is nodig + + + + Brightness + Helderheid + + + + Contrast + Contrast + + + + Gamma + Gamma + + + + Reset + Standaardwaarden terugzetten + + + + Image options + Afbeelding opties + + + + General + Algemeen + + + + Page Flow + Omslagbrowser + + + + Image adjustment + Beeldaanpassing + + + + Options + Opties + + + + Comics directory + Strips map + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + YACReader sneltoetsen + + + + Close + Sluiten + + + + Keyboard Shortcuts + Sneltoetsen + + + + Viewer + + + + Press 'O' to open comic. + Druk 'O' om een strip te openen. + + + + Not found + Niet gevonden + + + + Comic not found + Strip niet gevonden + + + + Error opening comic + + + + + CRC Error + + + + + Loading...please wait! + Inladen...even wachten! + + + + Page not available! + + + + + Cover! + Omslag! + + + + Last page! + Laatste pagina! + + + + YACReaderFieldEdit + + + + Click to overwrite + Klik hier om te overschrijven + + + + Restore to default + Standaardwaarden herstellen + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Klik hier om te overschrijven + + + + Restore to default + Standaardwaarden herstellen + + + + YACReaderFlowConfigWidget + + + How to show covers: + Omslagbladen bekijken: + + + + CoverFlow look + Omslagbrowser uiterlijk + + + + Stripe look + Brede band + + + + Overlapped Stripe look + Overlappende band + + + + YACReaderGLFlowConfigWidget + + + Presets: + Voorinstellingen: + + + + Classic look + Klassiek + + + + Stripe look + Brede band + + + + Overlapped Stripe look + Overlappende band + + + + Modern look + Modern + + + + Roulette look + Roulette + + + + Show advanced settings + Toon geavanceerde instellingen + + + + Custom: + Aangepast: + + + + View angle + Kijkhoek + + + + Position + Positie + + + + Cover gap + Ruimte tss Omslag + + + + Central gap + Centrale ruimte + + + + Zoom + Zoom + + + + Y offset + Y-positie + + + + Z offset + Z- positie + + + + Cover Angle + Omslag hoek + + + + Visibility + Zichtbaarheid + + + + Light + Licht + + + + Max angle + Maximale hoek + + + + Low Performance + Lage Prestaties + + + + High Performance + Hoge Prestaties + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Gebruik VSync (verbetering van de beeldkwaliteit in de modus volledig scherm, slechtere prestatie) + + + + Performance: + Prestatie: + + + + YACReaderOptionsDialog + + + Save + Bewaar + + + + Cancel + Annuleren + + + + Use hardware acceleration (restart needed) + Gebruik hardware versnelling (opnieuw opstarten vereist) + + + + YACReaderTranslator + + + YACReader translator + + + + + + Translation + + + + + clear + + + + + Service not available + + + + diff --git a/YACReader/yacreader_pt.ts b/YACReader/yacreader_pt.ts new file mode 100644 index 00000000..bdd6ec65 --- /dev/null +++ b/YACReader/yacreader_pt.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + Última Página + + + + Close + Fechar + + + + Click on any image to go to the bookmark + Clique em qualquer imagem para ir para o marcador + + + + + Loading... + Carregando... + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z não encontrado + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + GoToDialog + + + Page : + Página : + + + + Go To + Ir Para + + + + Cancel + Cancelar + + + + + Total pages : + Total de páginas : + + + + Go to... + Ir para... + + + + GoToFlowToolBar + + + Page : + Página : + + + + HelpAboutDialog + + + About + + + + + Help + Ajuda + + + + MainWindowViewer + + + &Open + &Abrir + + + + O + O + + + + Open a comic + Abrir um quadrinho + + + + Open Folder + Abrir Pasta + + + + Ctrl+O + Ctrl+O + + + + Open image folder + + + + + Save + Salvar + + + + + Save current page + Salvar página atual + + + + Previous Comic + Quadrinho Anterior + + + + Open previous comic + Abrir quadrinho anterior + + + + Next Comic + Próximo Quadrinho + + + + Open next comic + Abrir próximo quadrinho + + + + &Previous + A&nterior + + + + Go to previous page + Ir para a página anterior + + + + &Next + &Próxima + + + + Go to next page + Ir para a próxima página + + + + Fit Width + Ajustar à Largura + + + + Fit image to height + + + + + Fit Height + + + + + Fit image to width + + + + + Rotate image to the left + Girar imagem à esquerda + + + + L + L + + + + Rotate image to the right + Girar imagem à direita + + + + R + R + + + + Double page mode + Modo dupla página + + + + Switch to double page mode + Alternar para o modo dupla página + + + + D + D + + + + Go To + Ir Para + + + + G + G + + + + Go to page ... + Ir para a página... + + + + Options + Opções + + + + C + C + + + + YACReader options + Opções do YACReader + + + + Help + Ajuda + + + + Help, About YACReader + Ajuda, Sobre o YACReader + + + + Magnifying glass + Lupa + + + + Switch Magnifying glass + Alternar Lupa + + + + Z + Z + + + + Set bookmark + Definir marcador + + + + Set a bookmark on the current page + Definir um marcador na página atual + + + + Show bookmarks + Mostrar marcadores + + + + Show the bookmarks of the current comic + Mostrar os marcadores do quadrinho atual + + + + M + M + + + + Show keyboard shortcuts + Mostrar teclas de atalhos + + + + Show Info + Mostrar Informações + + + + I + I + + + + Close + Fechar + + + + Show Dictionary + + + + + Always on top + + + + + Show full size + + + + + Show go to flow + + + + + &File + &Arquivo + + + + File + + + + + Open Comic + Abrir Quadrinho + + + + Comic files + + + + + Remind me in 14 days + + + + + Not now + + + + + Open folder + Abrir pasta + + + + Image files (*.jpg) + Arquivos de imagem (*.jpg) + + + + page_%1.jpg + + + + + There is a new version avaliable + Há uma nova versão disponível + + + + Do you want to download the new version? + Você deseja baixar a nova versão? + + + + OptionsDialog + + + "Go to flow" size + Tamanho do "Ir para cheia" + + + + My comics path + Meu caminho de quadrinhos + + + + Page width stretch + Trecho da largura da página + + + + Background color + + + + + Choose + + + + + Restart is needed + Reiniciar é necessário + + + + Brightness + + + + + Contrast + + + + + Gamma + + + + + Reset + + + + + Image options + + + + + General + + + + + Page Flow + + + + + Image adjustment + + + + + Options + Opções + + + + Comics directory + Diretório de quadrinhos + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + Teclas de atalhos do YACReader + + + + Close + Fechar + + + + Keyboard Shortcuts + + + + + Viewer + + + + Press 'O' to open comic. + Pressione 'O' para abrir um quadrinho. + + + + Not found + + + + + Comic not found + + + + + Error opening comic + + + + + CRC Error + + + + + Loading...please wait! + Carregando... por favor, aguarde! + + + + Page not available! + + + + + Cover! + + + + + Last page! + + + + + YACReaderFieldEdit + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFlowConfigWidget + + + How to show covers: + + + + + CoverFlow look + Olhar capa cheia + + + + Stripe look + Olhar lista + + + + Overlapped Stripe look + Olhar lista sobreposta + + + + YACReaderGLFlowConfigWidget + + + Presets: + + + + + Classic look + + + + + Stripe look + Olhar lista + + + + Overlapped Stripe look + Olhar lista sobreposta + + + + Modern look + + + + + Roulette look + + + + + Show advanced settings + + + + + Custom: + + + + + View angle + + + + + Position + + + + + Cover gap + + + + + Central gap + + + + + Zoom + + + + + Y offset + + + + + Z offset + + + + + Cover Angle + + + + + Visibility + + + + + Light + + + + + Max angle + + + + + Low Performance + + + + + High Performance + + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + + + + + Performance: + + + + + YACReaderOptionsDialog + + + Save + Salvar + + + + Cancel + Cancelar + + + + Use hardware acceleration (restart needed) + + + + + YACReaderTranslator + + + YACReader translator + + + + + + Translation + + + + + clear + + + + + Service not available + + + + diff --git a/YACReader/yacreader_ru.ts b/YACReader/yacreader_ru.ts new file mode 100644 index 00000000..183b89ab --- /dev/null +++ b/YACReader/yacreader_ru.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + ПоÑледнÑÑ Ð¡Ñ‚Ñ€Ð°Ð½Ð¸Ñ†Ð° + + + + Close + Закрыть + + + + Click on any image to go to the bookmark + Ðажмите на любое изображение, чтобы перейти к закладке + + + + + Loading... + Загрузка... + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z не найден + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + GoToDialog + + + Page : + Страница: + + + + Go To + Перейти к + + + + Cancel + Отмена + + + + + Total pages : + Общее количеÑтв Ñтраниц: + + + + Go to... + Перейти к... + + + + GoToFlowToolBar + + + Page : + Страница: + + + + HelpAboutDialog + + + About + О программе + + + + Help + Справка + + + + MainWindowViewer + + + &Open + &Открыть + + + + O + О + + + + Open a comic + Открыть ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + Open Folder + Открыть папку + + + + Ctrl+O + Ctrl+О + + + + Open image folder + Открыть папку Ñ Ð¸Ð·Ð¾Ð±Ñ€Ð°Ð¶ÐµÐ½Ð¸Ñми + + + + Save + Сохранить + + + + + Save current page + Сохранить нынешнюю Ñтраницу + + + + Previous Comic + Предыдущий ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + Open previous comic + Открыть предыдуший ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + Next Comic + Следующий ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + Open next comic + Открыть Ñледующий ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + &Previous + &Предыдущий + + + + Go to previous page + Перейти к предыдущей Ñтранице + + + + &Next + &Следующий + + + + Go to next page + Перейти к Ñледующей Ñтранице + + + + Fit Width + Подогнать ширину + + + + Fit image to height + + + + + Fit Height + + + + + Fit image to width + + + + + Rotate image to the left + Повернуть изображение против чаÑовой Ñтрелки + + + + L + L + + + + Rotate image to the right + Повернуть изображение по чаÑовой Ñтрелке + + + + R + R + + + + Double page mode + Двойной режим Ñтраницы + + + + Switch to double page mode + Переключить на двойной режим Ñтраницы + + + + D + D + + + + Go To + Перейти к + + + + G + G + + + + Go to page ... + Перейти к Ñтранице ... + + + + Options + ÐаÑтройки + + + + C + С + + + + YACReader options + ÐаÑтройки YACReader + + + + Help + Справка + + + + Help, About YACReader + Справка по YACReader + + + + Magnifying glass + Увеличительное Ñтекло + + + + Switch Magnifying glass + ПереключитьÑÑ Ð½Ð° увеличительное Ñтекло + + + + Z + Z + + + + Set bookmark + УÑтановить закладку + + + + Set a bookmark on the current page + УÑтановить закладку на текущей Ñтранице + + + + Show bookmarks + Показать закладки + + + + Show the bookmarks of the current comic + Показать закладки текущего комикÑа + + + + M + M + + + + Show keyboard shortcuts + Показать горÑчие клавиши + + + + Show Info + Показать информацию + + + + I + I + + + + Close + Закрыть + + + + Show Dictionary + Показать Ñловарь + + + + Always on top + Ð’Ñегда Ñверху + + + + Show full size + ПолноÑкранный режим + + + + Show go to flow + + + + + &File + &Файл + + + + File + + + + + Open Comic + Открыть ÐºÐ¾Ð¼Ð¸ÐºÑ + + + + Comic files + Файлы комикÑа + + + + Open folder + Открыть папку + + + + Image files (*.jpg) + Файлы изображений + + + + page_%1.jpg + + + + + There is a new version avaliable + ДоÑтупно новое обновление + + + + Do you want to download the new version? + Хотите загрузить новую верÑию ? + + + + Remind me in 14 days + + + + + Not now + + + + + OptionsDialog + + + "Go to flow" size + Перейти к иÑходному размеру + + + + My comics path + Путь комикÑа + + + + Page width stretch + РаÑÑ‚Ñнуть Ñтраницу в ширину + + + + Background color + Фоновый цвет + + + + Choose + Выбрать + + + + Restart is needed + Ðеобходима перезагрузка + + + + Brightness + ЯркоÑть + + + + Contrast + КонтраÑÑ‚ + + + + Gamma + Гамма + + + + Reset + ПерезапуÑк + + + + Image options + ÐаÑтройки Ð¸Ð·Ð¾Ð±Ñ€Ð°Ð¶ÐµÐ½Ð¸Ñ + + + + General + Общее + + + + Page Flow + Страница потока + + + + Image adjustment + Регулировки Ð¸Ð·Ð¾Ð±Ñ€Ð°Ð¶ÐµÐ½Ð¸Ñ + + + + Options + ÐаÑтройки + + + + Comics directory + Каталог комикÑов + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + Клавиатурные комбинации YACReader + + + + Close + Закрыть + + + + Keyboard Shortcuts + Клавиатурные комбинации + + + + Viewer + + + + Press 'O' to open comic. + Ðажмите "O" , чтобы открыть комикÑ. + + + + Not found + Ðе найдено + + + + Comic not found + ÐšÐ¾Ð¼Ð¸ÐºÑ Ð½Ðµ найден + + + + Error opening comic + + + + + CRC Error + + + + + Loading...please wait! + Загрузка ... ПожалуйÑта подождите! + + + + Page not available! + + + + + Cover! + + + + + Last page! + + + + + YACReaderFieldEdit + + + + Click to overwrite + Ðажмите, чтобы перепиÑать + + + + Restore to default + Вернуть начальные уÑтановки + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Ðажмите, чтобы перепиÑать + + + + Restore to default + Вернуть начальные уÑтановки + + + + YACReaderFlowConfigWidget + + + How to show covers: + Как показать обложки: + + + + CoverFlow look + ПредоÑмотр обложки + + + + Stripe look + Вид полоÑами + + + + Overlapped Stripe look + Вид перекрывающимиÑÑ Ð¿Ð¾Ð»Ð¾Ñами + + + + YACReaderGLFlowConfigWidget + + + Presets: + ПредуÑтановки: + + + + Classic look + КлаÑÑичеÑкий вид + + + + Stripe look + Вид полоÑами + + + + Overlapped Stripe look + Вид перекрывающимиÑÑ Ð¿Ð¾Ð»Ð¾Ñами + + + + Modern look + Современный вид + + + + Roulette look + Вид рулеткой + + + + Show advanced settings + + + + + Custom: + ПользовательÑкий: + + + + View angle + Угол Ð·Ñ€ÐµÐ½Ð¸Ñ + + + + Position + ÐŸÐ¾Ð·Ð¸Ñ†Ð¸Ñ + + + + Cover gap + ОÑветить разрыв + + + + Central gap + СфокуÑировать разрыв + + + + Zoom + МаÑштабировать + + + + Y offset + Смещение по Y + + + + Z offset + Смещение по Z + + + + Cover Angle + Охватить угол + + + + Visibility + ПрозрачноÑть + + + + Light + ОÑветить + + + + Max angle + МакÑимальный угол + + + + Low Performance + ÐœÐ¸Ð½Ð¸Ð¼Ð°Ð»ÑŒÐ½Ð°Ñ Ð¿Ñ€Ð¾Ð¸Ð·Ð²Ð¾Ð´Ð¸Ñ‚ÐµÐ»ÑŒÐ½Ð¾Ñть + + + + High Performance + МакÑÐ¸Ð¼Ð°Ð»ÑŒÐ½Ð°Ñ Ð¿Ñ€Ð¾Ð¸Ð·Ð²Ð¾Ð´Ð¸Ñ‚ÐµÐ»ÑŒÐ½Ð¾Ñть + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + ИÑпользовать VSync (повыÑить формат Ð¸Ð·Ð¾Ð±Ñ€Ð°Ð¶ÐµÐ½Ð¸Ñ Ð² полноÑкранном режиме , хуже производительноÑть) + + + + Performance: + ПроизводительноÑть: + + + + YACReaderOptionsDialog + + + Save + Сохранить + + + + Cancel + Отмена + + + + Use hardware acceleration (restart needed) + ИÑпользовать аппаратное уÑкорение (ТребуетÑÑ Ð¿ÐµÑ€ÐµÐ·Ð°Ð³Ñ€ÑƒÐ·ÐºÐ°) + + + + YACReaderTranslator + + + YACReader translator + + + + + + Translation + + + + + clear + + + + + Service not available + + + + diff --git a/YACReader/yacreader_source.ts b/YACReader/yacreader_source.ts new file mode 100644 index 00000000..1e8ee9f3 --- /dev/null +++ b/YACReader/yacreader_source.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + + + + + Close + + + + + Click on any image to go to the bookmark + + + + + + Loading... + + + + + FileComic + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + Unknown error opening the file + + + + + 7z not found + + + + + Format not supported + + + + + GoToDialog + + + Page : + + + + + Go To + + + + + Cancel + + + + + + Total pages : + + + + + Go to... + + + + + GoToFlowToolBar + + + Page : + + + + + HelpAboutDialog + + + About + + + + + Help + + + + + MainWindowViewer + + + &Open + + + + + O + + + + + Open a comic + + + + + Open Folder + + + + + Ctrl+O + + + + + Open image folder + + + + + Save + + + + + + Save current page + + + + + Previous Comic + + + + + Open previous comic + + + + + Next Comic + + + + + Open next comic + + + + + &Previous + + + + + Go to previous page + + + + + &Next + + + + + Go to next page + + + + + Fit Width + + + + + Fit image to height + + + + + Fit Height + + + + + Fit image to width + + + + + Rotate image to the left + + + + + L + + + + + Rotate image to the right + + + + + R + + + + + Double page mode + + + + + Switch to double page mode + + + + + D + + + + + Go To + + + + + G + + + + + Go to page ... + + + + + Options + + + + + C + + + + + YACReader options + + + + + Help + + + + + Help, About YACReader + + + + + Magnifying glass + + + + + Switch Magnifying glass + + + + + Z + + + + + Set bookmark + + + + + Set a bookmark on the current page + + + + + Show bookmarks + + + + + Show the bookmarks of the current comic + + + + + M + + + + + Show keyboard shortcuts + + + + + Show Info + + + + + I + + + + + Close + + + + + Show Dictionary + + + + + Always on top + + + + + Show full size + + + + + Show go to flow + + + + + &File + + + + + File + + + + + Open Comic + + + + + Comic files + + + + + Open folder + + + + + Image files (*.jpg) + + + + + page_%1.jpg + + + + + There is a new version avaliable + + + + + Do you want to download the new version? + + + + + Remind me in 14 days + + + + + Not now + + + + + OptionsDialog + + + "Go to flow" size + + + + + My comics path + + + + + Page width stretch + + + + + Background color + + + + + Choose + + + + + Restart is needed + + + + + Brightness + + + + + Contrast + + + + + Gamma + + + + + Reset + + + + + Image options + + + + + General + + + + + Page Flow + + + + + Image adjustment + + + + + Options + + + + + Comics directory + + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + + + + + Close + + + + + Keyboard Shortcuts + + + + + Viewer + + + + Press 'O' to open comic. + + + + + Not found + + + + + Comic not found + + + + + Error opening comic + + + + + CRC Error + + + + + Loading...please wait! + + + + + Page not available! + + + + + Cover! + + + + + Last page! + + + + + YACReaderFieldEdit + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFlowConfigWidget + + + How to show covers: + + + + + CoverFlow look + + + + + Stripe look + + + + + Overlapped Stripe look + + + + + YACReaderGLFlowConfigWidget + + + Presets: + + + + + Classic look + + + + + Stripe look + + + + + Overlapped Stripe look + + + + + Modern look + + + + + Roulette look + + + + + Show advanced settings + + + + + Custom: + + + + + View angle + + + + + Position + + + + + Cover gap + + + + + Central gap + + + + + Zoom + + + + + Y offset + + + + + Z offset + + + + + Cover Angle + + + + + Visibility + + + + + Light + + + + + Max angle + + + + + Low Performance + + + + + High Performance + + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + + + + + Performance: + + + + + YACReaderOptionsDialog + + + Save + + + + + Cancel + + + + + Use hardware acceleration (restart needed) + + + + + YACReaderTranslator + + + YACReader translator + + + + + + Translation + + + + + clear + + + + + Service not available + + + + diff --git a/YACReader/yacreader_tr.ts b/YACReader/yacreader_tr.ts new file mode 100644 index 00000000..56fb2aac --- /dev/null +++ b/YACReader/yacreader_tr.ts @@ -0,0 +1,725 @@ + + + + + BookmarksDialog + + Close + Kapat + + + Loading... + Yükleniyor... + + + Click on any image to go to the bookmark + Yer imine git + + + Lastest Page + Son Sayfa + + + + Configuration + + There was a problem saving YACReader configuration. Please, check if you have enough permissions in the YACReader root folder. + Yeni ayarlar kaydedilirken bir problem çıktı. Lütfen YACReader dosyasını açın. + + + Saving config file.... + Config dosyası kaydediliyor... + + + + FileComic + + File not found or not images in file + Dosya bulunamadı yada dosyada resim yok + + + 7z not found + 7z bulunamadı + + + Comic not found + Çizgi roman bulunamadı + + + Not found + Bulunamadı + + + File error + Dosya hatası + + + 7z problem + 7z Problemli + + + 7z reading + 7z Okunuyor + + + 7z crashed. + 7z BozulmuÅŸ. + + + problem reading from 7z + 7z Okunurken Problem OluÅŸtu + + + 7z crashed + 7z Bozulması + + + Unknown error 7z + Bilinmeyen 7z hatası + + + 7z wasn't found in your PATH. + 7z Yolu Bulunamadı. + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + Unknown error opening the file + + + + Format not supported + + + + + GoToDialog + + Go To + Git + + + Go to... + Git... + + + Total pages : + Toplam sayfa: + + + Cancel + Vazgeç + + + Page : + Sayfa : + + + + GoToFlowToolBar + + Page : + Sayfa : + + + + HelpAboutDialog + + Help + Yardım + + + About + Hakkında + + + + MainWindowViewer + + C + C + + + D + D + + + G + G + + + I + I + + + L + L + + + M + M + + + O + O + + + R + R + + + Z + Z + + + Help + Yardım + + + Save + Kaydet + + + &File + &Dosya + + + &Next + &İleri + + + &Open + &Aç + + + Close + Kapat + + + Open Comic + Çizgi Romanı Aç + + + Go To + Git + + + Open image folder + Resim dosyasınıaç + + + Set bookmark + Yer imi yap + + + page_%1.jpg + sayfa_%1.jpg + + + Switch to double page mode + Çift sayfa moduna geç + + + Save current page + Geçerli sayfayı kaydet + + + Double page mode + Çift sayfa modu + + + Switch Magnifying glass + Büyüteç + + + Open Folder + Dosyayı Aç + + + Ctrl+O + Ctrl+O + + + Comic files + Çizgi Roman Dosyaları + + + Go to previous page + Önceki sayfaya dön + + + Open a comic + Çizgi romanı aç + + + Image files (*.jpg) + Resim dosyaları (*.jpg) + + + Next Comic + Sırada ki çizgi roman + + + Saving error log file.... + Hata dosyasını kaydet... + + + Fit Width + Uygun GeniÅŸlik + + + Options + Ayarlar + + + Show Info + Bilgiyi göster + + + Open folder + Dosyayı aç + + + Go to page ... + Sayfata git... + + + Fit image to width + Görüntüyü sığdır + + + &Previous + &Geri + + + Go to next page + Sonra ki sayfaya geç + + + Show keyboard shortcuts + Kılavye kısayollarını göster + + + Open next comic + Sıradaki çizgi romanı aç + + + There is a new version avaliable + Yeni versiyon mevcut + + + Show bookmarks + Yer imlerini göster + + + Open previous comic + Önceki çizgi romanı aç + + + Rotate image to the left + Sayfayı sola yatır + + + Fit image to height + Uygun yüksekliÄŸe getir + + + Show the bookmarks of the current comic + Bu çizgi romanın yer imlerini göster + + + Show Dictionary + Sözlüğü göster + + + YACReader options + YACReader ayarları + + + Help, About YACReader + YACReader hakkında yardım ve bilgi + + + Show go to flow + Akışı göster + + + Previous Comic + Önce ki çizgi roman + + + Show full size + Tam erken + + + Magnifying glass + Büyüteç + + + Set a bookmark on the current page + Sayfayı yer imi olarak ayarla + + + Do you want to download the new version? + Yeni versiyonu indirmek ister misin ? + + + There was a problem saving YACReader error log file. Please, check if you have enough permissions in the YACReader root folder. + Kaydederken bir problem çıktı YACReader hata kayıt dosyası. Lütfen YACReader root dosyasını ziyaret edin. + + + Rotate image to the right + Sayfayı saÄŸa yator + + + Always on top + Her zaman üstte + + + Remind me in 14 days + + + + Not now + + + + Fit Height + + + + File + + + + + OptionsDialog + + Gamma + Gama + + + Reset + Yeniden baÅŸlat + + + My comics path + Çizgi Romanlarım + + + Image adjustment + Resim ayarları + + + Page width stretch + Sayfayı uzat + + + "Go to flow" size + Akış görünümüne git + + + Choose + Seç + + + Image options + Sayfa ayarları + + + Contrast + Kontrast + + + Options + Ayarlar + + + Comics directory + Çizgi roman konumu + + + Background color + Arka plan rengi + + + Page Flow + Sayfa akışı + + + General + Genel + + + Brightness + Parlaklık + + + Restart is needed + Yeniden baÅŸlatılmalı + + + + QObject + + 7z lib not found + + + + unable to load 7z lib from ./utils + + + + + ShortcutsDialog + + Close + Kapat + + + YACReader keyboard shortcuts + YACReader klavye kısayolları + + + Keyboard Shortcuts + Kılavye Kısayolları + + + + Viewer + + Press 'O' to open comic. + 'O'ya basarak aç. + + + Cover! + Kapak! + + + Comic not found + Çizgi roman bulunamadı + + + Not found + Bulunamadı + + + Last page! + Son sayfa! + + + Loading...please wait! + Yükleniyor... lütfen bekleyin! + + + Error opening comic + + + + CRC Error + + + + Page not available! + + + + + YACReaderDeletingProgress + + cancel + vazgeç + + + Please wait, deleting in progress... + Lütfen bekle silme iÅŸlemi gerçekleÅŸtiriliyor... + + + + YACReaderFieldEdit + + Restore to default + Varsayılana ayarla + + + Click to overwrite + Üzerine yazmak için tıkla + + + + YACReaderFieldPlainTextEdit + + Restore to default + Varsayılana ayarla + + + Click to overwrite + Üstüne yazmak için tıkla + + + + YACReaderFlowConfigWidget + + CoverFlow look + Kapak akışı görünümü + + + How to show covers: + Kapaklar nasıl gözüksün: + + + Stripe look + Åžerit görünüm + + + Overlapped Stripe look + Çakışan ÅŸerit görünüm + + + + YACReaderGLFlowConfigWidget + + Zoom + YakınlaÅŸ + + + Light + Işık + + + Show advanced settings + Daha fazla ayar göster + + + Roulette look + Rulet görünüm + + + Cover Angle + Kapak Açısı + + + Stripe look + Strip görünüm + + + Position + Pozisyon + + + Z offset + Z dengesi + + + Y offset + Y dengesi + + + Central gap + BoÅŸ merkaz + + + Presets: + Hazırlayan: + + + Overlapped Stripe look + Çakışan ÅŸerit görünüm + + + Modern look + Modern görünüm + + + View angle + Bakış açısı + + + Max angle + Maksimum açı + + + Custom: + KiÅŸisel: + + + Classic look + Klasik görünüm + + + Cover gap + Kapak + + + High Performance + Yüksek performans + + + Performance: + Performans: + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + VSync kullan + + + Visibility + Görünülebilirlik + + + Low Performance + Düşük Performans + + + + YACReaderOptionsDialog + + Save + Kaydet + + + Use hardware acceleration (restart needed) + Yüksek donanımlı kullan (yeniden baÅŸlatmak gerekli) + + + Cancel + Vazgeç + + + + YACReaderSideBar + + Search folders and comics + Dosyaları ve çizgi romanları ara + + + LIBRARIES + KÜTÜPHANELER + + + FOLDERS + DOSYALAR + + + + YACReaderTranslator + + YACReader translator + + + + Translation + + + + clear + + + + Service not available + + + + diff --git a/YACReaderLibrary.desktop b/YACReaderLibrary.desktop new file mode 100644 index 00000000..76ee3432 --- /dev/null +++ b/YACReaderLibrary.desktop @@ -0,0 +1,12 @@ +[Desktop Entry] +Name=YACReader Library +GenericName=Yet Another Comic Reader +Comment=Yet Another Comic Reader +Exec=YACReaderLibrary %f +Icon=/usr/share/YACReader/iconLibrary.png +Terminal=false +Type=Application +StartupNotify=true +Categories=Graphics;Viewer; +MimeType= +X-Desktop-File-Install-Version=0.22 diff --git a/YACReaderLibrary/YACReaderLibrary.icns b/YACReaderLibrary/YACReaderLibrary.icns new file mode 100644 index 0000000000000000000000000000000000000000..bca828394936567acc5791c88a62141b9bdc480c GIT binary patch literal 53727 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 I_-zaPKZgh27ytkO literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro new file mode 100644 index 00000000..bb4f8252 --- /dev/null +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -0,0 +1,273 @@ +###################################################################### +# Automatically generated by qmake (2.01a) dom 12. oct 20:47:48 2008 +###################################################################### + +TEMPLATE = app +TARGET = YACReaderLibrary +DEPENDPATH += . +INCLUDEPATH += . +INCLUDEPATH += ../common \ + ./server \ + ./db \ + ../custom_widgets \ + ./comic_vine \ + ./comic_vine/model + +DEFINES += SERVER_RELEASE NOMINMAX YACREADER_LIBRARY + +win32 { + +LIBS += -L../dependencies/poppler/lib -loleaut32 -lole32 -lshell32 + +isEqual(QT_MAJOR_VERSION, 5) { +LIBS += -lpoppler-qt5 +INCLUDEPATH += ../dependencies/poppler/include/qt5 +} +else { +LIBS += -lpoppler-qt4 +INCLUDEPATH += ../dependencies/poppler/include/qt4 +} + +QMAKE_CXXFLAGS_RELEASE += /MP /Ob2 /Oi /Ot /GT /GL +QMAKE_LFLAGS_RELEASE += /LTCG +CONFIG -= embed_manifest_exe +} + +unix:!macx{ + +isEqual(QT_MAJOR_VERSION, 5) { +INCLUDEPATH += /usr/include/poppler/qt5 +LIBS += -L/usr/lib -lpoppler-qt5 +} +else { +INCLUDEPATH += /usr/include/poppler/qt4 +LIBS += -L/usr/lib -lpoppler-qt4 +} +LIBS += -lGLU +} + +macx{ +#INCLUDEPATH += "/Volumes/Mac OS X Lion/usr/X11/include" +#isEqual(QT_MAJOR_VERSION, 5) { +#INCLUDEPATH += /usr/local/include/poppler/qt5 +#LIBS += -L/usr/local/lib -lpoppler-qt5 +#} +#else { +#INCLUDEPATH += /usr/local/include/poppler/qt4 +#LIBS += -L/usr/local/lib -lpoppler-qt4 +#} +#QT += macextras + +LIBS += -framework Foundation -framework ApplicationServices + +OBJECTIVE_SOURCES += $$PWD/../common/pdf_comic.mm +HEADERS += $$PWD/../common/pdf_comic.h +CONFIG += objective_c + +} + +unix{ +QMAKE_CXXFLAGS += -std=c++11 +} + +#CONFIG += release +CONFIG -= flat +QT += sql network opengl script + +# Input +HEADERS += comic_flow.h \ + create_library_dialog.h \ + library_creator.h \ + library_window.h \ + add_library_dialog.h \ + rename_library_dialog.h \ + properties_dialog.h \ + options_dialog.h \ + export_library_dialog.h \ + import_library_dialog.h \ + package_manager.h \ + bundle_creator.h \ + export_comics_info_dialog.h \ + import_comics_info_dialog.h \ + server_config_dialog.h \ + comic_flow_widget.h \ + db_helper.h \ + ./db/data_base_management.h \ + ./db/treeitem.h \ + ./db/treemodel.h \ + ./db/tablemodel.h \ + ./db/tableitem.h \ + ../common/comic_db.h \ + ../common/folder.h \ + ../common/library_item.h \ + ../common/comic.h \ + ../common/bookmarks.h \ + ../common/pictureflow.h \ + ../common/custom_widgets.h \ + ../common/qnaturalsorting.h \ + ../common/yacreader_flow_gl.h \ + ../common/yacreader_global.h \ + ../common/onstart_flow_selection_dialog.h \ + no_libraries_widget.h \ + import_widget.h \ + yacreader_local_server.h \ + yacreader_main_toolbar.h \ + comics_remover.h \ + ../common/http_worker.h \ + yacreader_libraries.h \ + ../common/exit_check.h \ + comics_view.h \ + classic_comics_view.h \ + empty_folder_widget.h + + +SOURCES += comic_flow.cpp \ + create_library_dialog.cpp \ + library_creator.cpp \ + library_window.cpp \ + main.cpp \ + add_library_dialog.cpp \ + rename_library_dialog.cpp \ + properties_dialog.cpp \ + options_dialog.cpp \ + export_library_dialog.cpp \ + import_library_dialog.cpp \ + package_manager.cpp \ + bundle_creator.cpp \ + export_comics_info_dialog.cpp \ + import_comics_info_dialog.cpp \ + server_config_dialog.cpp \ + comic_flow_widget.cpp \ + db_helper.cpp \ + ./db/data_base_management.cpp \ + ./db/treeitem.cpp \ + ./db/treemodel.cpp \ + ./db/tablemodel.cpp \ + ./db/tableitem.cpp \ + ../common/comic_db.cpp \ + ../common/folder.cpp \ + ../common/library_item.cpp \ + ../common/comic.cpp \ + ../common/bookmarks.cpp \ + ../common/pictureflow.cpp \ + ../common/custom_widgets.cpp \ + ../common/qnaturalsorting.cpp \ + ../common/yacreader_flow_gl.cpp \ + ../common/onstart_flow_selection_dialog.cpp \ + no_libraries_widget.cpp \ + import_widget.cpp \ + yacreader_local_server.cpp \ + yacreader_main_toolbar.cpp \ + comics_remover.cpp \ + ../common/http_worker.cpp \ +../common/yacreader_global.cpp \ + yacreader_libraries.cpp \ + ../common/exit_check.cpp \ + comics_view.cpp \ + classic_comics_view.cpp \ + empty_folder_widget.cpp + + + +include(./server/server.pri) +include(../custom_widgets/custom_widgets_yacreaderlibrary.pri) +include(../compressed_archive/wrapper.pri) +include(./comic_vine/comic_vine.pri) +include(../QsLog/QsLog.pri) +include(../shortcuts_management/shortcuts_management.pri) + +RESOURCES += images.qrc files.qrc +win32:RESOURCES += images_win.qrc +unix:!macx:RESOURCES += images_win.qrc +macx:RESOURCES += images_osx.qrc + +RC_FILE = icon.rc + +macx { + ICON = YACReaderLibrary.icns +} + +TRANSLATIONS = yacreaderlibrary_es.ts \ + yacreaderlibrary_ru.ts \ + yacreaderlibrary_pt.ts \ + yacreaderlibrary_fr.ts \ + yacreaderlibrary_nl.ts \ + yacreaderlibrary_tr.ts \ + yacreaderlibrary_source.ts + +isEqual(QT_MAJOR_VERSION, 5) { + Release:DESTDIR = ../release5 + Debug:DESTDIR = ../debug5 + +#QML/GridView +QT += quick qml + +HEADERS += grid_comics_view.h \ + comics_view_transition.h + +SOURCES += grid_comics_view.cpp \ + comics_view_transition.cpp + +RESOURCES += qml.qrc +win32:RESOURCES += qml_win.qrc +unix:!macx:RESOURCES += qml_win.qrc +macx:RESOURCES += qml_osx.qrc + +} else { + Release:DESTDIR = ../release + Debug:DESTDIR = ../debug +} + +win32 { +!exists(../compressed_archive/lib7zip){ + error(You\'ll need 7zip source code to compile YACReader. \ + Please check the compressed_archive folder for further instructions.) +} +} + +unix { +exists (../compressed_archive/libp7zip) { + message(Found p7zip source code...) + system(patch -d ../compressed_archive -N -p0 -i libp7zip.patch) +} else { + error(You\'ll need 7zip source code to compile YACReader. \ + Please check the compressed_archive folder for further instructions.) +} +} + +unix:!macx { +#set install prefix if it's empty +isEmpty(PREFIX) { + PREFIX = /usr +} + +BINDIR = $$PREFIX/bin +LIBDIR = $$PREFIX/lib +DATADIR = $$PREFIX/share + +DEFINES += "LIBDIR=\\\"$$LIBDIR\\\"" "DATADIR=\\\"$$DATADIR\\\"" "BINDIR=\\\"$$BINDIR\\\"" + +#MAKE INSTALL +INSTALLS += bin icon desktop server translation + +bin.path = $$BINDIR +isEmpty(DESTDIR) { + bin.files = YACReaderLibrary +} else { + bin.files = $$DESTDIR/YACReaderLibrary +} + +server.path = $$DATADIR/YACReader +server.files = ../release/server + +icon.path = $$DATADIR/YACReader +icon.files = ../images/iconLibrary.png ../images/db.png ../images/coversPackage.png + +desktop.path = $$DATADIR/applications +desktop.extra = desktop-file-edit --set-icon=$$DATADIR/YACReader/iconLibrary.png $$PWD/../YACReaderLibrary.desktop +desktop.files = ../YACReaderLibrary.desktop +#TODO: icons should be located at /usr/share/icons and have the same basename as their application + +translation.path = $$DATADIR/YACReader/languages +translation.files = ../release/languages/yacreaderlibrary_* +} diff --git a/YACReaderLibrary/add_library_dialog.cpp b/YACReaderLibrary/add_library_dialog.cpp new file mode 100644 index 00000000..2b7f666b --- /dev/null +++ b/YACReaderLibrary/add_library_dialog.cpp @@ -0,0 +1,124 @@ +#include "add_library_dialog.h" + +#include +#include +#include +#include + + +AddLibraryDialog::AddLibraryDialog(QWidget * parent) +:QDialog(parent) +{ + setupUI(); +} + +void AddLibraryDialog::setupUI() +{ + textLabel = new QLabel(tr("Comics folder : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + connect(path,SIGNAL(textChanged(QString)),this,SLOT(pathSetted(QString))); + + nameLabel = new QLabel(tr("Library Name : ")); + nameEdit = new QLineEdit; + nameLabel->setBuddy(nameEdit); + connect(nameEdit,SIGNAL(textChanged(QString)),this,SLOT(nameSetted(QString))); + + accept = new QPushButton(tr("Add")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(add())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + QGridLayout * content = new QGridLayout; + + content->addWidget(nameLabel,0,0); + content->addWidget(nameEdit,0,1); + + content->addWidget(textLabel,1,0); + content->addWidget(path,1,1); + content->addWidget(find,1,2); + content->setColumnStretch(2,0); + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(content); + mainLayout->addStretch(); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/openLibrary.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel);//,0,Qt::AlignTop); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Add an existing library")); +} + +void AddLibraryDialog::add() +{ + //accept->setEnabled(false); + emit(addLibrary(QDir::cleanPath(path->text()),nameEdit->text())); +} + +void AddLibraryDialog::nameSetted(const QString & text) +{ + if(!text.isEmpty()) + { + if(!path->text().isEmpty()) + { + QFileInfo fi(path->text()); + if(fi.isDir()) + accept->setEnabled(true); + else + accept->setEnabled(false); + } + } + else + accept->setEnabled(false); +} + +void AddLibraryDialog::pathSetted(const QString & text) +{ + QFileInfo fi(text); + if(fi.isDir()) + { + if(!nameEdit->text().isEmpty()) + accept->setEnabled(true); + } + else + accept->setEnabled(false); +} + +void AddLibraryDialog::findPath() +{ + QString s = QFileDialog::getExistingDirectory(0,"Comics directory","."); + if(!s.isEmpty()) + { + path->setText(s); + if(!nameEdit->text().isEmpty()) + accept->setEnabled(true); + } + else + accept->setEnabled(false); +} + +void AddLibraryDialog::close() +{ + path->clear(); + nameEdit->clear(); + accept->setEnabled(false); + QDialog::close(); +} diff --git a/YACReaderLibrary/add_library_dialog.h b/YACReaderLibrary/add_library_dialog.h new file mode 100644 index 00000000..4cbdd42b --- /dev/null +++ b/YACReaderLibrary/add_library_dialog.h @@ -0,0 +1,35 @@ +#ifndef __ADD_LIBRARY_DIALOG_H +#define __ADD_LIBRARY_DIALOG_H + +#include +#include +#include +#include +#include + + class AddLibraryDialog : public QDialog + { + Q_OBJECT + public: + AddLibraryDialog(QWidget * parent = 0); + private: + QLabel * nameLabel; + QLabel * textLabel; + QLineEdit * path; + QLineEdit * nameEdit; + QPushButton * find; + QPushButton * accept; + QPushButton * cancel; + void setupUI(); + public slots: + void add(); + void findPath(); + void close(); + void nameSetted(const QString & text); + void pathSetted(const QString & text); + signals: + void addLibrary(QString target, QString name); + }; + +#endif + diff --git a/YACReaderLibrary/bundle_creator.cpp b/YACReaderLibrary/bundle_creator.cpp new file mode 100644 index 00000000..8ea6eef2 --- /dev/null +++ b/YACReaderLibrary/bundle_creator.cpp @@ -0,0 +1,13 @@ +#include "bundle_creator.h" + + +BundleCreator::BundleCreator(void) + :QObject() +{ + +} + + +BundleCreator::~BundleCreator(void) +{ +} diff --git a/YACReaderLibrary/bundle_creator.h b/YACReaderLibrary/bundle_creator.h new file mode 100644 index 00000000..ff4dbd08 --- /dev/null +++ b/YACReaderLibrary/bundle_creator.h @@ -0,0 +1,14 @@ +#ifndef __BUNDLE_CREATOR_H +#define __BUNDLE_CREATOR_H + +#include + +class BundleCreator : public QObject +{ +Q_OBJECT +public: + BundleCreator(void); + ~BundleCreator(void); +}; + +#endif \ No newline at end of file diff --git a/YACReaderLibrary/classic_comics_view.cpp b/YACReaderLibrary/classic_comics_view.cpp new file mode 100644 index 00000000..48444658 --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -0,0 +1,243 @@ +#include "classic_comics_view.h" + +#include "yacreader_table_view.h" + +#include "comic_flow_widget.h" +#include "QsLog.h" + +ClassicComicsView::ClassicComicsView(QWidget *parent) + :ComicsView(parent) +{ + QHBoxLayout * layout = new QHBoxLayout; + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + //FLOW----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + + if(QGLFormat::hasOpenGL() && (settings->value(USE_OPEN_GL).toBool() == true)) + comicFlow = new ComicFlowWidgetGL(0); + else + comicFlow = new ComicFlowWidgetSW(0); + + comicFlow->updateConfig(settings); + comicFlow->setFocusPolicy(Qt::StrongFocus); + comicFlow->setShowMarks(true); + setFocusProxy(comicFlow); + + comicFlow->setFocus(Qt::OtherFocusReason); + + comicFlow->setContextMenuPolicy(Qt::ActionsContextMenu); + + //TODO!!! set actions.... + //comicFlow->addAction(toggleFullScreenAction); + //comicFlow->addAction(openComicAction); + + //END FLOW---- + + + //layout----------------------------------------------- + sVertical = new QSplitter(Qt::Vertical); //spliter derecha + + sVertical->addWidget(comicFlow); + comics = new QWidget; + QVBoxLayout * comicsLayout = new QVBoxLayout; + comicsLayout->setSpacing(0); + comicsLayout->setContentsMargins(0,0,0,0); + //TODO ComicsView:(set toolbar) comicsLayout->addWidget(editInfoToolBar); + + tableView = new YACReaderTableView; + tableView->verticalHeader()->hide(); + tableView->setFocusPolicy(Qt::StrongFocus); + comicsLayout->addWidget(tableView); + comics->setLayout(comicsLayout); + sVertical->addWidget(comics); + + //config-------------------------------------------------- + if(settings->contains(COMICS_VIEW_HEADERS)) + tableView->horizontalHeader()->restoreState(settings->value(COMICS_VIEW_HEADERS).toByteArray()); + + //connections--------------------------------------------- + connect(tableView, SIGNAL(clicked(QModelIndex)), this, SLOT(centerComicFlow(QModelIndex))); + connect(comicFlow, SIGNAL(centerIndexChanged(int)), this, SLOT(updateTableView(int))); + connect(tableView, SIGNAL(comicRated(int,QModelIndex)), this, SIGNAL(comicRated(int,QModelIndex))); + connect(comicFlow, SIGNAL(selected(uint)), this, SIGNAL(selected(uint))); + connect(tableView->horizontalHeader(), SIGNAL(sectionMoved(int,int,int)), this, SLOT(saveTableHeadersStatus())); + + layout->addWidget(sVertical); + setLayout(layout); + + layout->setMargin(0); + +#ifdef Q_OS_MAC + sVertical->setCollapsible(1,false); +#endif +} + +void ClassicComicsView::setToolBar(QToolBar *toolBar) +{ + static_cast(comics->layout())->insertWidget(0,toolBar); +} + +void ClassicComicsView::setModel(TableModel *model) +{ + + ComicsView::setModel(model); + + if(model == NULL) + { + comicFlow->clear(); + } + else + { + connect(model, SIGNAL(dataChanged(QModelIndex,QModelIndex,QVector)), this, SLOT(applyModelChanges(QModelIndex,QModelIndex,QVector))); + connect(model, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(removeItemsFromFlow(QModelIndex,int,int))); + + tableView->setModel(model); + + tableView->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft); + #if QT_VERSION >= 0x050000 + tableView->horizontalHeader()->setSectionsMovable(true); + #else + tableView->horizontalHeader()->setMovable(true); + #endif + //TODO parametrizar la configuración de las columnas + for(int i = 0;ihorizontalHeader()->count();i++) + tableView->horizontalHeader()->hideSection(i); + + tableView->horizontalHeader()->showSection(TableModel::Number); + tableView->horizontalHeader()->showSection(TableModel::Title); + tableView->horizontalHeader()->showSection(TableModel::FileName); + tableView->horizontalHeader()->showSection(TableModel::NumPages); + tableView->horizontalHeader()->showSection(TableModel::Hash); //Size is part of the Hash...TODO add Columns::Size to Columns + tableView->horizontalHeader()->showSection(TableModel::ReadColumn); + tableView->horizontalHeader()->showSection(TableModel::CurrentPage); + tableView->horizontalHeader()->showSection(TableModel::Rating); + + //debido a un bug, qt4 no es capaz de ajustar el ancho teniendo en cuenta todas la filas (no sólo las visibles) + //así que se ecala la primera vez y después se deja el control al usuario. + //if(!settings->contains(COMICS_VIEW_HEADERS)) + tableView->resizeColumnsToContents(); + tableView->horizontalHeader()->setStretchLastSection(true); + + QStringList paths = model->getPaths(model->getCurrentPath());//TODO ComicsView: get currentpath from somewhere currentPath()); + comicFlow->setImagePaths(paths); + comicFlow->setMarks(model->getReadList()); + //comicFlow->setFocus(Qt::OtherFocusReason); + } + + if(settings->contains(COMICS_VIEW_HEADERS)) + tableView->horizontalHeader()->restoreState(settings->value(COMICS_VIEW_HEADERS).toByteArray()); + + +} + +void ClassicComicsView::setCurrentIndex(const QModelIndex &index) +{ + tableView->setCurrentIndex(index); + //TODO ComicsView: scroll comicFlow to index +} + +QModelIndex ClassicComicsView::currentIndex() +{ + return tableView->currentIndex(); +} + +QItemSelectionModel *ClassicComicsView::selectionModel() +{ + return tableView->selectionModel(); +} + +void ClassicComicsView::scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint) +{ + comicFlow->setCenterIndex(mi.row()); +} + +void ClassicComicsView::toFullScreen() +{ + comicFlow->hide(); + comicFlow->setCenterIndex(comicFlow->centerIndex()); + comics->hide(); + + //showFullScreen() //parent windows + + comicFlow->show(); + comicFlow->setFocus(Qt::OtherFocusReason); +} + +void ClassicComicsView::toNormal() +{ + comicFlow->hide(); + comicFlow->setCenterIndex(comicFlow->centerIndex()); + comicFlow->render(); + comics->show(); + comicFlow->show(); +} + +void ClassicComicsView::updateConfig(QSettings *settings) +{ + comicFlow->updateConfig(settings); +} + +void ClassicComicsView::setItemActions(const QList &actions) +{ + tableView->addActions(actions); +} + +void ClassicComicsView::setViewActions(const QList &actions) +{ + comicFlow->addActions(actions); +} + +void ClassicComicsView::selectAll() +{ + tableView->selectAll(); +} + +void ClassicComicsView::setShowMarks(bool show) +{ + comicFlow->setShowMarks(show); +} + +void ClassicComicsView::centerComicFlow(const QModelIndex & mi) +{ + comicFlow->showSlide(mi.row()); + comicFlow->setFocus(Qt::OtherFocusReason); +} + +void ClassicComicsView::updateTableView(int i) +{ + QModelIndex mi = model->index(i,2); + tableView->setCurrentIndex(mi); + tableView->scrollTo(mi,QAbstractItemView::EnsureVisible); +} + +void ClassicComicsView::saveTableHeadersStatus() +{ + settings->setValue(COMICS_VIEW_HEADERS,tableView->horizontalHeader()->saveState()); +} + +void ClassicComicsView::applyModelChanges(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QVector &roles) +{ + Q_UNUSED(topLeft); + Q_UNUSED(bottomRight); + if(roles.contains(TableModel::ReadColumnRole)) + { + comicFlow->setMarks(model->getReadList()); + comicFlow->updateMarks(); + } +} + +void ClassicComicsView::removeItemsFromFlow(const QModelIndex &parent, int from, int to) +{ + Q_UNUSED(parent); + for(int i = from; i<=to; i++) + comicFlow->remove(i); +} + +void ClassicComicsView::closeEvent(QCloseEvent *event) +{ + saveTableHeadersStatus(); + ComicsView::closeEvent(event); +} + diff --git a/YACReaderLibrary/classic_comics_view.h b/YACReaderLibrary/classic_comics_view.h new file mode 100644 index 00000000..9f8c71a6 --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.h @@ -0,0 +1,51 @@ +#ifndef CLASSIC_COMICS_VIEW_H +#define CLASSIC_COMICS_VIEW_H + +#include "comics_view.h" + +#include +#include + +class YACReaderTableView; +class QSplitter; +class ComicFlowWidget; +class QToolBar; +class TableModel; + +class ClassicComicsView : public ComicsView +{ + Q_OBJECT +public: + ClassicComicsView(QWidget *parent = 0); + void setToolBar(QToolBar * toolBar); + void setModel(TableModel *model); + void setCurrentIndex(const QModelIndex &index); + QModelIndex currentIndex(); + QItemSelectionModel * selectionModel(); + void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ); + void toFullScreen(); + void toNormal(); + void updateConfig(QSettings * settings); + void setItemActions(const QList & actions); + void setViewActions(const QList & actions); + +public slots: + void centerComicFlow(const QModelIndex & mi); + void updateTableView(int i); + void saveTableHeadersStatus(); + void applyModelChanges(const QModelIndex & topLeft,const QModelIndex & bottomRight,const QVector & roles); + void removeItemsFromFlow(const QModelIndex & parent, int from, int to); + //ComicsView + void setShowMarks(bool show); + void selectAll(); + +private: + YACReaderTableView * tableView; + QWidget *comics; + QSplitter * sVertical; + ComicFlowWidget * comicFlow; + QSettings * settings; + void closeEvent ( QCloseEvent * event ); +}; + +#endif // CLASSIC_COMICS_VIEW_H diff --git a/YACReaderLibrary/comic_flow.cpp b/YACReaderLibrary/comic_flow.cpp new file mode 100644 index 00000000..6404e817 --- /dev/null +++ b/YACReaderLibrary/comic_flow.cpp @@ -0,0 +1,238 @@ +#include "comic_flow.h" +#include "qnaturalsorting.h" + +#include "yacreader_global.h" + +#include + +#include +#include +#include + +ComicFlow::ComicFlow(QWidget* parent,FlowType flowType) +:YACReaderFlow(parent,flowType) +{ + updateTimer = new QTimer; + connect(updateTimer, SIGNAL(timeout()), this, SLOT(updateImageData())); + + worker = new ImageLoader; + connect(this, SIGNAL(centerIndexChanged(int)), this, SLOT(preload())); + connect(this, SIGNAL(centerIndexChangedSilent(int)), this, SLOT(preload())); + + setReflectionEffect(PlainReflection); +} + +ComicFlow::~ComicFlow() +{ + delete worker; + delete updateTimer; +} + +void ComicFlow::setImagePaths(const QStringList& paths) +{ + clear(); + + //imagePath = path; + imageFiles = paths; + imagesLoaded.clear(); + imagesLoaded.fill(false,imageFiles.size()); + numImagesLoaded = 0; + + imagesSetted.clear(); + imagesSetted.fill(false,imageFiles.size()); + + // populate with empty images + QImage img; //TODO remove + QString s; + for(int i = 0; i < (int)imageFiles.size(); i++) + { + addSlide(img); + s = imageFiles.at(i); + s.remove(s.size()-4,4); + if(QFileInfo(s+".r").exists()) + markSlide(i); + } + + setCenterIndex(0); + worker->reset(); + preload(); +} + +void ComicFlow::preload() +{ + if(numImagesLoaded < imagesLoaded.size()) + updateTimer->start(30); //TODO comprobar rendimiento, originalmente era 70 +} + +void ComicFlow::updateImageData() +{ + // can't do anything, wait for the next possibility + if(worker->busy()) + return; + + // set image of last one + int idx = worker->index(); + if( idx >= 0 && !worker->result().isNull()) + { + if(!imagesSetted[idx]) + { + setSlide(idx, worker->result()); + imagesSetted[idx] = true; + numImagesLoaded++; + imagesLoaded[idx]=true; + } + } + + // try to load only few images on the left and right side + // i.e. all visible ones plus some extra +#define COUNT 8 + int indexes[2*COUNT+1]; + int center = centerIndex(); + indexes[0] = center; + for(int j = 0; j < COUNT; j++) + { + indexes[j*2+1] = center+j+1; + indexes[j*2+2] = center-j-1; + } + for(int c = 0; c < 2*COUNT+1; c++) + { + int i = indexes[c]; + if((i >= 0) && (i < slideCount())) + if(!imagesLoaded[i])//slide(i).isNull()) + { + // schedule thumbnail generation + QString fname = imageFiles[i]; + + + worker->generate(i, fname, slideSize()); + return; + } + } + + // no need to generate anything? stop polling... + updateTimer->stop(); +} + +void ComicFlow::keyPressEvent(QKeyEvent* event) +{ + PictureFlow::keyPressEvent(event); +} + +void ComicFlow::wheelEvent(QWheelEvent * event) +{ + if(event->delta()<0) + showNext(); + else + showPrevious(); + event->accept(); +} + +void ComicFlow::removeSlide(int cover) +{ + worker->lock(); + + worker->reset(); + + imageFiles.removeAt(cover); + if(imagesLoaded[cover]) + numImagesLoaded--; + imagesLoaded.remove(cover); + imagesSetted.remove(cover); + + YACReaderFlow::removeSlide(cover); + worker->unlock(); + + preload(); +} +//----------------------------------------------------------------------------- +//ImageLoader +//----------------------------------------------------------------------------- +static QImage loadImage(const QString& fileName) +{ + QImage image; + bool result = image.load(fileName); + + if(!result) + return QImage(); + + return image; +} + +ImageLoader::ImageLoader(): +QThread(), restart(false), working(false), idx(-1) +{ +} + +ImageLoader::~ImageLoader() +{ + mutex.lock(); + condition.wakeOne(); + mutex.unlock(); + wait(); +} + +bool ImageLoader::busy() const +{ + return isRunning() ? working : false; +} + +void ImageLoader::generate(int index, const QString& fileName, QSize size) +{ + mutex.lock(); + this->idx = index; + this->fileName = fileName; + this->size = size; + this->img = QImage(); + mutex.unlock(); + + if (!isRunning()) + start(); + else + { + // already running, wake up whenever ready + restart = true; + condition.wakeOne(); + } +} + +void ImageLoader::lock() +{ + mutex.lock(); +} + +void ImageLoader::unlock() +{ + mutex.unlock(); +} + +void ImageLoader::run() +{ + for(;;) + { + // copy necessary data + mutex.lock(); + this->working = true; + QString fileName = this->fileName; + mutex.unlock(); + + QImage image = loadImage(fileName); + + // let everyone knows it is ready + mutex.lock(); + this->working = false; + this->img = image; + mutex.unlock(); + + // put to sleep + mutex.lock(); + if (!this->restart) + condition.wait(&mutex); + restart = false; + mutex.unlock(); + } +} + +QImage ImageLoader::result() +{ + return img; +} diff --git a/YACReaderLibrary/comic_flow.h b/YACReaderLibrary/comic_flow.h new file mode 100644 index 00000000..632e4871 --- /dev/null +++ b/YACReaderLibrary/comic_flow.h @@ -0,0 +1,77 @@ +#ifndef __COMICFLOW_H +#define __COMICFLOW_H + +#include "yacreader_flow.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +class ImageLoader; +class ComicFlow : public YACReaderFlow +{ + Q_OBJECT +public: + ComicFlow(QWidget* parent = 0,FlowType flowType = CoverFlowLike); + virtual ~ComicFlow(); + + void setImagePaths(const QStringList& paths); + //bool eventFilter(QObject *target, QEvent *event); + void keyPressEvent(QKeyEvent* event); + void removeSlide(int cover); + +private slots: + void preload(); + void updateImageData(); + +private: + //QString imagePath; + QStringList imageFiles; + QVector imagesLoaded; + QVector imagesSetted; + int numImagesLoaded; + QTimer* updateTimer; + ImageLoader* worker; + virtual void wheelEvent(QWheelEvent * event); +}; + + +//----------------------------------------------------------------------------- +// Source code of ImageLoader class was modified from http://code.google.com/p/photoflow/ +//------------------------------------------------------------------------------ +class ImageLoader : public QThread +{ +public: + ImageLoader(); + ~ImageLoader(); + // returns FALSE if worker is still busy and can't take the task + bool busy() const; + void generate(int index, const QString& fileName, QSize size); + void reset(){idx = -1;}; + int index() const { return idx; }; + void lock(); + void unlock(); + QImage result(); + +protected: + void run(); + +private: + QMutex mutex; + QWaitCondition condition; + + bool restart; + bool working; + int idx; + QString fileName; + QSize size; + QImage img; +}; + + +#endif diff --git a/YACReaderLibrary/comic_flow_widget.cpp b/YACReaderLibrary/comic_flow_widget.cpp new file mode 100644 index 00000000..9b523d57 --- /dev/null +++ b/YACReaderLibrary/comic_flow_widget.cpp @@ -0,0 +1,345 @@ +#include "comic_flow_widget.h" + +ComicFlowWidget::ComicFlowWidget(QWidget * parent) + :QWidget(parent) +{ + +} + +ComicFlowWidgetSW::ComicFlowWidgetSW(QWidget * parent) + :ComicFlowWidget(parent) +{ + flow = new ComicFlow(parent); + + connect(flow,SIGNAL(centerIndexChanged(int)),this,SIGNAL(centerIndexChanged(int))); + connect(flow,SIGNAL(selected(unsigned int)),this,SIGNAL(selected(unsigned int))); + + QVBoxLayout * l = new QVBoxLayout; + l->addWidget(flow); + setLayout(l); + + //TODO eleminar "padding" + QPalette Pal(palette()); + // set black background + Pal.setColor(QPalette::Background, Qt::black); + setAutoFillBackground(true); + setPalette(Pal); + + //config + QMatrix m; + m.rotate(-90); + m.scale(-1,1); + QImage image(":/images/setRead.png"); + QImage imageTransformed = image.transformed(m,Qt::SmoothTransformation); + setMarkImage(imageTransformed); +} + +QSize ComicFlowWidgetSW::minimumSizeHint() const +{ + return flow->minimumSizeHint(); +} +QSize ComicFlowWidgetSW::sizeHint() const +{ + return flow->sizeHint(); +} + +void ComicFlowWidgetSW::setShowMarks(bool value) +{ + flow->setShowMarks(value); +} +void ComicFlowWidgetSW::setMarks(QVector marks) +{ + flow->setMarks(marks); +} +void ComicFlowWidgetSW::setMarkImage(QImage & image) +{ + flow->setMarkImage(image); +} +void ComicFlowWidgetSW::markSlide(int index, YACReaderComicReadStatus status) +{ + flow->markSlide(index,status); +} +void ComicFlowWidgetSW::unmarkSlide(int index) +{ + flow->unmarkSlide(index); +} +void ComicFlowWidgetSW::setSlideSize(QSize size) +{ + flow->setSlideSize(size); +} +void ComicFlowWidgetSW::clear() +{ + flow->clear(); +} +void ComicFlowWidgetSW::setImagePaths(QStringList paths) +{ + flow->setImagePaths(paths); +} +void ComicFlowWidgetSW::setCenterIndex(int index) +{ + flow->setCenterIndex(index); +} +void ComicFlowWidgetSW::showSlide(int index) +{ + flow->showSlide(index); +} +int ComicFlowWidgetSW::centerIndex() +{ + return flow->centerIndex(); +} +void ComicFlowWidgetSW::updateMarks() +{ + flow->updateMarks(); +} +void ComicFlowWidgetSW::setFlowType(FlowType flowType) +{ + flow->setFlowType(flowType); +} +void ComicFlowWidgetSW::render() +{ + flow->render(); +} +void ComicFlowWidgetSW::keyPressEvent(QKeyEvent* event) +{ + flow->keyPressEvent(event); +} +void ComicFlowWidgetSW::paintEvent(QPaintEvent *event) +{ + flow->paintEvent(event); +} +void ComicFlowWidgetSW::mousePressEvent(QMouseEvent* event) +{ + flow->mousePressEvent(event); +} +void ComicFlowWidgetSW::resizeEvent(QResizeEvent* event) +{ + flow->resizeEvent(event); +} +void ComicFlowWidgetSW::mouseDoubleClickEvent(QMouseEvent* event) +{ + flow->mouseDoubleClickEvent(event); +} +void ComicFlowWidgetSW::updateConfig(QSettings * settings) +{ + switch (settings->value(FLOW_TYPE_SW).toInt()) + { + case CoverFlowLike: + flow->setFlowType(CoverFlowLike); + return; + case Strip: + flow->setFlowType(Strip); + return; + case StripOverlapped: + flow->setFlowType(StripOverlapped); + return; + } +} + +void ComicFlowWidgetSW::remove(int cover) +{ + flow->removeSlide(cover); +} + + +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +///OpenGL ComicFlow +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// + +ComicFlowWidgetGL::ComicFlowWidgetGL(QWidget * parent) + :ComicFlowWidget(parent) +{ + flow = new YACReaderComicFlowGL(parent); + + connect(flow,SIGNAL(centerIndexChanged(int)),this,SIGNAL(centerIndexChanged(int))); + connect(flow,SIGNAL(selected(unsigned int)),this,SIGNAL(selected(unsigned int))); + + QVBoxLayout * l = new QVBoxLayout; + l->addWidget(flow); + l->setContentsMargins(0,0,0,0); + setLayout(l); + + //TODO eleminar "padding" + QPalette Pal(palette()); + // set black background + Pal.setColor(QPalette::Background, Qt::black); + setAutoFillBackground(true); + setPalette(Pal); +} + +QSize ComicFlowWidgetGL::minimumSizeHint() const +{ + return flow->minimumSizeHint(); +} +QSize ComicFlowWidgetGL::sizeHint() const +{ + return flow->sizeHint(); +} + +void ComicFlowWidgetGL::setShowMarks(bool value) +{ + flow->setShowMarks(value); +} +void ComicFlowWidgetGL::setMarks(QVector marks) +{ + flow->setMarks(marks); +} +void ComicFlowWidgetGL::setMarkImage(QImage & image) +{ + flow->setMarkImage(image); +} +void ComicFlowWidgetGL::markSlide(int index, YACReaderComicReadStatus status) +{ + flow->markSlide(index,status); +} +void ComicFlowWidgetGL::unmarkSlide(int index) +{ + flow->unmarkSlide(index); +} +void ComicFlowWidgetGL::setSlideSize(QSize size) +{ + flow->setSlideSize(size); +} +void ComicFlowWidgetGL::clear() +{ + flow->clear(); +} +void ComicFlowWidgetGL::setImagePaths(QStringList paths) +{ + flow->setImagePaths(paths); +} +void ComicFlowWidgetGL::setCenterIndex(int index) +{ + flow->setCenterIndex(index); +} +void ComicFlowWidgetGL::showSlide(int index) +{ + flow->showSlide(index); +} +int ComicFlowWidgetGL::centerIndex() +{ + return flow->centerIndex(); +} +void ComicFlowWidgetGL::updateMarks() +{ + flow->updateMarks(); +} +void ComicFlowWidgetGL::setFlowType(FlowType flowType) +{ + if(flowType == CoverFlowLike) + flow->setPreset(presetYACReaderFlowClassicConfig); + else if(flowType == Strip) + flow->setPreset(presetYACReaderFlowStripeConfig); + else if(flowType == StripOverlapped) + flow->setPreset(presetYACReaderFlowOverlappedStripeConfig); + else + flow->setPreset(defaultYACReaderFlowConfig); +} +void ComicFlowWidgetGL::render() +{ + flow->render(); +} +void ComicFlowWidgetGL::keyPressEvent(QKeyEvent* event) +{ + flow->keyPressEvent(event); +} +void ComicFlowWidgetGL::paintEvent(QPaintEvent *event) +{ + //flow->paintEvent(event); + ComicFlowWidget::paintEvent(event); +} +void ComicFlowWidgetGL::mousePressEvent(QMouseEvent* event) +{ + flow->mousePressEvent(event); +} +void ComicFlowWidgetGL::resizeEvent(QResizeEvent* event) +{ + flow->resizeGL(event->size().width(),event->size().height()); +} +void ComicFlowWidgetGL::mouseDoubleClickEvent(QMouseEvent* event) +{ + flow->mouseDoubleClickEvent(event); +} + +void ComicFlowWidgetGL::updateConfig(QSettings * settings) +{ + Performance performance = medium; + + switch (settings->value(PERFORMANCE).toInt()) + { + case 0: + performance = low; + break; + case 1: + performance = medium; + break; + case 2: + performance = high; + break; + case 3: + performance = ultraHigh; + break; + } + + flow->setPerformance(performance); + if(!settings->contains(V_SYNC)) + flow->useVSync(false); + else + flow->useVSync(settings->value(V_SYNC).toBool()); + + switch (settings->value(FLOW_TYPE_GL).toInt()) + { + case 0: + flow->setPreset(presetYACReaderFlowClassicConfig); + return; + case 1: + flow->setPreset(presetYACReaderFlowStripeConfig); + return; + case 2: + flow->setPreset(presetYACReaderFlowOverlappedStripeConfig); + return; + case 3: + flow->setPreset(defaultYACReaderFlowConfig); + return; + case 4: + flow->setPreset(pressetYACReaderFlowDownConfig); + return; + } + + + //custom config + + flow->setCF_RX(settings->value(X_ROTATION).toInt()); + flow->setCF_Y(settings->value(Y_POSITION).toInt()); + flow->setX_Distance(settings->value(COVER_DISTANCE).toInt()); + flow->setCenter_Distance(settings->value(CENTRAL_DISTANCE).toInt()); + flow->setCF_Z(settings->value(ZOOM_LEVEL).toInt()); + flow->setY_Distance(settings->value(Y_COVER_OFFSET).toInt()); + flow->setZ_Distance(settings->value(Z_COVER_OFFSET).toInt()); + flow->setRotation(settings->value(COVER_ROTATION).toInt()); + flow->setFadeOutDist(settings->value(FADE_OUT_DIST).toInt()); + flow->setLightStrenght(settings->value(LIGHT_STRENGTH).toInt()); + flow->setMaxAngle(settings->value(MAX_ANGLE).toInt()); + +/* flow->setVisibility(settings->value("visibilityDistance").toInt()); + flow->setLightStrenght(settings->value("lightStrength").toInt())*/; + +} + +void ComicFlowWidgetGL::remove(int cover) +{ + flow->remove(cover); +} + +//void ComicFlowWidgetGL::setCF_RX(int value){ flow->setCF_RX(value);} +//void ComicFlowWidgetGL::setCF_RY(int value){ flow->setCF_RY(value);} +//void ComicFlowWidgetGL::setCF_RZ(int value){ flow->setCF_RZ(value);} +//void ComicFlowWidgetGL::setZoom(int zoom){ flow->setZoom(zoom);} +//void ComicFlowWidgetGL::setRotation(int angle){ flow->setRotation(angle);} +//void ComicFlowWidgetGL::setX_Distance(int distance){ flow->setX_Distance(distance);} +//void ComicFlowWidgetGL::setCenter_Distance(int distance){ flow->setCenter_Distance(distance);} +//void ComicFlowWidgetGL::setZ_Distance(int distance){ flow->setZ_Distance(distance);} +//void ComicFlowWidgetGL::setCF_Y(int value){ flow->setCF_Y(value);} +//void ComicFlowWidgetGL::setY_Distance(int value){ flow->setY_Distance(value);} +//void ComicFlowWidgetGL::setPreset(const Preset & p){ flow->setPreset(p);} \ No newline at end of file diff --git a/YACReaderLibrary/comic_flow_widget.h b/YACReaderLibrary/comic_flow_widget.h new file mode 100644 index 00000000..028d9177 --- /dev/null +++ b/YACReaderLibrary/comic_flow_widget.h @@ -0,0 +1,128 @@ +#ifndef __COMIC_FLOW_WIDGET_H +#define __COMIC_FLOW_WIDGET_H + + +#include + +#include "pictureflow.h" +#include "comic_flow.h" +#include "yacreader_flow_gl.h" + +class ComicFlowWidget : public QWidget +{ + Q_OBJECT +public: + ComicFlowWidget(QWidget * paret = 0); + +public slots: + virtual void setShowMarks(bool value) = 0; + virtual void setMarks(QVector marks) = 0; + virtual void setMarkImage(QImage & image) = 0; + virtual void markSlide(int index, YACReaderComicReadStatus status) = 0; + virtual void unmarkSlide(int index) = 0; + virtual void setSlideSize(QSize size) = 0; + virtual void clear() = 0; + virtual void setImagePaths(QStringList paths) = 0; + virtual void setCenterIndex(int index) = 0; + virtual void showSlide(int index) = 0; + virtual int centerIndex() = 0; + virtual void updateMarks() = 0; + virtual void setFlowType(FlowType flowType) = 0; + virtual void render() = 0; + virtual void updateConfig(QSettings * settings) = 0; + virtual void remove(int cover) = 0; +signals: + void centerIndexChanged(int); + void selected(unsigned int); +}; + + +class ComicFlowWidgetSW : public ComicFlowWidget +{ + Q_OBJECT +private: + ComicFlow * flow; +public: + ComicFlowWidgetSW(QWidget * parent = 0); + + void setShowMarks(bool value); + void setMarks(QVector marks); + void setMarkImage(QImage & image); + void markSlide(int index, YACReaderComicReadStatus status); + void unmarkSlide(int index); + void setSlideSize(QSize size); + void clear(); + void setImagePaths(QStringList paths); + void setCenterIndex(int index); + void showSlide(int index); + int centerIndex(); + void updateMarks(); + void setFlowType(FlowType flowType); + void render(); + void updateConfig(QSettings * settings); + void remove(int cover); +protected: + void keyPressEvent(QKeyEvent* event); + void paintEvent(QPaintEvent *event); + void mousePressEvent(QMouseEvent* event); + void resizeEvent(QResizeEvent* event); + void mouseDoubleClickEvent(QMouseEvent* event); + QSize minimumSizeHint() const; + QSize sizeHint() const; + QSize slideSizeW; + QSize slideSizeF; +}; + +class ComicFlowWidgetGL : public ComicFlowWidget +{ + Q_OBJECT +private: + YACReaderComicFlowGL * flow; +public: + ComicFlowWidgetGL(QWidget * parent = 0); + + void setShowMarks(bool value); + void setMarks(QVector marks); + void setMarkImage(QImage & image); + void markSlide(int index, YACReaderComicReadStatus status); + void unmarkSlide(int index); + void setSlideSize(QSize size); + void clear(); + void setImagePaths(QStringList paths); + void setCenterIndex(int index); + void showSlide(int index); + int centerIndex(); + void updateMarks(); + void setFlowType(FlowType flowType); + void render(); + void updateConfig(QSettings * settings); + void remove(int cover); +//public slots: +// void setCF_RX(int value); +// //the Y Rotation of the Coverflow +// void setCF_RY(int value); +// //the Z Rotation of the Coverflow +// void setCF_RZ(int value); +// //perspective +// void setZoom(int zoom); +// void setRotation(int angle); +// //sets the distance between the covers +// void setX_Distance(int distance); +// //sets the distance between the centered and the non centered covers +// void setCenter_Distance(int distance); +// //sets the pushback amount +// void setZ_Distance(int distance); +// void setCF_Y(int value); +// void setY_Distance(int value); +// void setPreset(const Preset & p); +protected: + void keyPressEvent(QKeyEvent* event); + void paintEvent(QPaintEvent *event); + void mousePressEvent(QMouseEvent* event); + void resizeEvent(QResizeEvent* event); + void mouseDoubleClickEvent(QMouseEvent* event); + QSize minimumSizeHint() const; + QSize sizeHint() const; +}; + +#endif \ No newline at end of file diff --git a/YACReaderLibrary/comic_vine/comic_vine.pri b/YACReaderLibrary/comic_vine/comic_vine.pri new file mode 100644 index 00000000..823e536a --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine.pri @@ -0,0 +1,44 @@ + +HEADERS += \ + comic_vine/comic_vine_dialog.h \ + comic_vine/comic_vine_client.h \ + comic_vine/scraper_lineedit.h \ + comic_vine/title_header.h \ + comic_vine/series_question.h \ + comic_vine/search_single_comic.h \ + comic_vine/search_volume.h \ + comic_vine/select_comic.h \ + comic_vine/select_volume.h \ + comic_vine/model/volumes_model.h \ + comic_vine/model/comics_model.h \ + comic_vine/model/json_model.h \ + comic_vine/model/response_parser.h \ + comic_vine/scraper_tableview.h \ + comic_vine/sort_volume_comics.h \ + comic_vine/model/local_comic_list_model.h \ + comic_vine/model/volume_comics_model.h \ + comic_vine/scraper_scroll_label.h \ + comic_vine/scraper_results_paginator.h \ + comic_vine/scraper_selector.h + +SOURCES += \ + comic_vine/comic_vine_dialog.cpp \ + comic_vine/comic_vine_client.cpp \ + comic_vine/scraper_lineedit.cpp \ + comic_vine/title_header.cpp \ + comic_vine/series_question.cpp \ + comic_vine/search_single_comic.cpp \ + comic_vine/search_volume.cpp \ + comic_vine/select_comic.cpp \ + comic_vine/select_volume.cpp \ + comic_vine/model/volumes_model.cpp \ + comic_vine/model/comics_model.cpp \ + comic_vine/model/json_model.cpp \ + comic_vine/model/response_parser.cpp \ + comic_vine/scraper_tableview.cpp \ + comic_vine/sort_volume_comics.cpp \ + comic_vine/model/local_comic_list_model.cpp \ + comic_vine/model/volume_comics_model.cpp \ + comic_vine/scraper_scroll_label.cpp \ + comic_vine/scraper_results_paginator.cpp \ + comic_vine/scraper_selector.cpp diff --git a/YACReaderLibrary/comic_vine/comic_vine_client.cpp b/YACReaderLibrary/comic_vine/comic_vine_client.cpp new file mode 100644 index 00000000..de515f3a --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_client.cpp @@ -0,0 +1,164 @@ +#include "comic_vine_client.h" + +//this is the API key used by YACReader to access Comic Vine +//please, do not use it in your own software, get one for free at Comic Vine +static const QString CV_API_KEY = "46680bebb358f1de690a5a365e15d325f9649f91"; + +static const QString CV_WEB_ADDRESS = "http://www.comicvine.com/api"; + +//gets any volumen containing any comic matching 'query' +static const QString CV_SEARCH = CV_WEB_ADDRESS + "/search/?api_key=" + CV_API_KEY + + "&format=json&limit=100&resources=volume" + "&field_list=name,start_year,publisher,id,image,count_of_issues,deck" + "&sort=name:asc" + "&query=%1&page=%2"; +//http://www.comicvine.com/api/search/?api_key=46680bebb358f1de690a5a365e15d325f9649f91&format=json&limit=100&resources=volume&field_list=name,start_year,publisher,id,image,count_of_issues,deck&query=superman + +//gets the detail for a volume %1 +static const QString CV_SERIES_DETAIL = CV_WEB_ADDRESS + "/volume/4050-%1/?api_key=" + CV_API_KEY + + "&format=json&field_list=name,start_year,publisher,image,count_of_issues,id,description"; + +//gets info for comics in a volume id %1 +static const QString CV_COMICS_INFO = CV_WEB_ADDRESS + "/issues/?api_key=" + CV_API_KEY + + "&format=json&field_list=name,issue_number,id,image&filter=volume:%1" + "&sort=cover_date:asc" //sorting by cover_date, because comic vine doesn't use natural sorting (issue_number -> 1 10 11 ... 100 2 20 21....) + "&offset=%2"; + +//"http://www.comicvine.com/api/issues/?api_key=46680bebb358f1de690a5a365e15d325f9649f91&format=json&field_list=name,issue_number,id,image&filter=volume:%1&page=%2 + +//gets id for comic number %2 in a volume id %1 +static const QString CV_COMIC_ID = CV_WEB_ADDRESS + "/issues/?api_key=" + CV_API_KEY + + "&format=json&field_list=name,issue_number,id,image" + "&filter=volume:%1,issue_number:%2"; +//gets comic detail +static const QString CV_COMIC_DETAIL = CV_WEB_ADDRESS + "/issue/4000-%1/?api_key=" + CV_API_KEY + "&format=json"; +//http://www.comicvine.com/api/issue/4000-%1/?api_key=46680bebb358f1de690a5a365e15d325f9649f91&format=json + +//gets comic cover URL +static const QString CV_COVER_URL = CV_WEB_ADDRESS + "/issue/4000-%1/?api_key=" + CV_API_KEY + "&format=json&field_list=image"; + +//gets comics matching name %1 and number %2 +//http://comicvine.com/api/issues/?api_key=46680bebb358f1de690a5a365e15d325f9649f91&limit=20&filter=name:super,issue_number:15 + +ComicVineClient::ComicVineClient(QObject *parent) : + QObject(parent) +{ + +} + +//CV_SEARCH +void ComicVineClient::search(const QString & query, int page) +{ + HttpWorker * search = new HttpWorker(CV_SEARCH.arg(query).arg(page)); + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SLOT(proccessVolumesSearchData(const QByteArray &))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} +//CV_SEARCH result +void ComicVineClient::proccessVolumesSearchData(const QByteArray & data) +{ + QString json(data); + emit searchResult(json); + emit finished(); +} + +void ComicVineClient::proccessSeriesDetailData(const QByteArray &data) +{ + QString json(data); + emit seriesDetail(json); + emit finished(); +} + +void ComicVineClient::processVolumeComicsInfo(const QByteArray &data) +{ + QString json(data); + emit volumeComicsInfo(json); + emit finished(); +} + +void ComicVineClient::proccessComicDetailData(const QByteArray &data) +{ + QString json(data); + emit comicDetail(json); + emit finished(); +} + +//CV_SERIES_DETAIL +void ComicVineClient::getSeriesDetail(const QString & id) +{ + HttpWorker * search = new HttpWorker(CV_SERIES_DETAIL.arg(id)); + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SLOT(proccessSeriesDetailData(const QByteArray &))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} + +void ComicVineClient::getSeriesCover(const QString & url) +{ + HttpWorker * search = new HttpWorker(url); + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SIGNAL(seriesCover(const QByteArray &))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); //TODO + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} + +//CV_COMIC_IDS +void ComicVineClient::getVolumeComicsInfo(const QString & idVolume, int page) +{ + HttpWorker * search = new HttpWorker(CV_COMICS_INFO.arg(idVolume).arg((page-1)*100)); //page on works for search, using offset instead + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SLOT(processVolumeComicsInfo(const QByteArray &))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); //TODO + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} + +//CV_COMIC_ID +void ComicVineClient::getComicId(const QString & id, int comicNumber) +{ + +} + +//CV_COMIC_DETAIL +QByteArray ComicVineClient::getComicDetail(const QString & id, bool & outError, bool & outTimeout) +{ + HttpWorker * search = new HttpWorker(CV_COMIC_DETAIL.arg(id)); + + //connect(search,SIGNAL(dataReady(const QByteArray &)),this,SLOT(proccessComicDetailData(const QByteArray &))); + //connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); + //connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); + search->wait(); + outError = !(search->wasValid()); + outTimeout = search->wasTimeout(); + QByteArray result = search->getResult(); + delete search; + + return result; +} + +//CV_COMIC_DETAIL +void ComicVineClient::getComicDetailAsync(const QString & id) +{ + HttpWorker * search = new HttpWorker(CV_COMIC_DETAIL.arg(id)); + + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SLOT(proccessComicDetailData(const QByteArray &))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} + +void ComicVineClient::getComicCover(const QString &url) +{ + HttpWorker * search = new HttpWorker(url); + connect(search,SIGNAL(dataReady(const QByteArray &)),this,SIGNAL(comicCover(QByteArray))); + connect(search,SIGNAL(timeout()),this,SIGNAL(timeOut())); //TODO + connect(search,SIGNAL(finished()),search,SLOT(deleteLater())); + search->get(); +} + +//CV_COVER_DETAIL +void ComicVineClient::getCoverURL(const QString & id) +{ + +} diff --git a/YACReaderLibrary/comic_vine/comic_vine_client.h b/YACReaderLibrary/comic_vine/comic_vine_client.h new file mode 100644 index 00000000..6e1bc579 --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_client.h @@ -0,0 +1,41 @@ +#ifndef COMIC_VINE_CLIENT_H +#define COMIC_VINE_CLIENT_H + +#include "http_worker.h" + +#include + +class ComicVineClient : public QObject +{ + Q_OBJECT +public: + explicit ComicVineClient(QObject *parent = 0); + +signals: + void searchResult(QString); + void seriesDetail(QString);//JSON + void comicDetail(QString);//JSON + void seriesCover(const QByteArray &); + void comicCover(const QByteArray &); + void volumeComicsInfo(QString); + void timeOut(); + void finished(); +public slots: + void search(const QString & query, int page = 1); + void getSeriesDetail(const QString & id); + void getSeriesCover(const QString & url); + void getVolumeComicsInfo(const QString & idVolume, int page=1); + QByteArray getComicDetail(const QString & id, bool &outError, bool &outTimeout); + void getComicCover(const QString & url); + + void getComicId(const QString & id, int comicNumber); + void getCoverURL(const QString & id); + void getComicDetailAsync(const QString &id); +protected slots: + void proccessVolumesSearchData(const QByteArray & data); + void proccessSeriesDetailData(const QByteArray & data); + void processVolumeComicsInfo(const QByteArray & data); + void proccessComicDetailData(const QByteArray & data); + +}; +#endif // COMIC_VINE_CLIENT_H diff --git a/YACReaderLibrary/comic_vine/comic_vine_dialog.cpp b/YACReaderLibrary/comic_vine/comic_vine_dialog.cpp new file mode 100644 index 00000000..77e8311e --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_dialog.cpp @@ -0,0 +1,722 @@ +#include "comic_vine_dialog.h" +#include +#include +#include +#include +#include +#include +#include +#include +#if QT_VERSION >= 0x050000 + #include +#else + #include +#endif +#include +#include +#include "data_base_management.h" + +#include "yacreader_busy_widget.h" +#include "comic_vine_client.h" +#include "scraper_lineedit.h" +#include "title_header.h" +#include "series_question.h" +#include "search_single_comic.h" +#include "search_volume.h" +#include "select_comic.h" +#include "select_volume.h" +#include "sort_volume_comics.h" +#include "db_helper.h" +#include "response_parser.h" + +#include "QsLog.h" + + + +ComicVineDialog::ComicVineDialog(QWidget *parent) : + QDialog(parent) +{ + doLayout(); + doStackedWidgets(); + doConnections(); +} + +void ComicVineDialog::doLayout() +{ + setStyleSheet("" + "QDialog {background-color: #404040; }" + ""); + + QString dialogButtonsStyleSheet = "QPushButton {border: 1px solid #242424; background: #2e2e2e; color:white; padding: 5px 26px 5px 26px; font-size:12px;font-family:Arial; font-weight:bold;}"; + + skipButton = new QPushButton(tr("skip")); + backButton = new QPushButton(tr("back")); + nextButton = new QPushButton(tr("next")); + searchButton = new QPushButton(tr("search")); + closeButton = new QPushButton(tr("close")); + + skipButton->setStyleSheet(dialogButtonsStyleSheet); + backButton->setStyleSheet(dialogButtonsStyleSheet); + nextButton->setStyleSheet(dialogButtonsStyleSheet); + searchButton->setStyleSheet(dialogButtonsStyleSheet); + closeButton->setStyleSheet(dialogButtonsStyleSheet); + + content = new QStackedWidget(this); + + QVBoxLayout * mainLayout = new QVBoxLayout; + + QHBoxLayout * buttonLayout = new QHBoxLayout; + + buttonLayout->addStretch(); + buttonLayout->addWidget(skipButton); + buttonLayout->addWidget(backButton); + buttonLayout->addWidget(nextButton); + buttonLayout->addWidget(searchButton); + buttonLayout->addWidget(closeButton); + buttonLayout->setContentsMargins(0,0,0,0); + + mainLayout->addWidget(titleHeader = new TitleHeader); + mainLayout->addWidget(content); + mainLayout->addStretch(); + mainLayout->addLayout(buttonLayout); + + mainLayout->setContentsMargins(26,16,26,11); + + setLayout(mainLayout); + setFixedSize(872,529); + + setWindowTitle("Comic Vine Scraper (beta)"); +} + +void ComicVineDialog::doStackedWidgets() +{ + doLoading(); + content->addWidget(seriesQuestionWidget = new SeriesQuestion); + content->addWidget(searchSingleComicWidget = new SearchSingleComic); + content->addWidget(searchVolumeWidget = new SearchVolume); + content->addWidget(selectVolumeWidget = new SelectVolume); + content->addWidget(selectComicWidget = new SelectComic); + content->addWidget(sortVolumeComicsWidget = new SortVolumeComics); +} + +void ComicVineDialog::doConnections() +{ + connect(closeButton,SIGNAL(clicked()),this,SLOT(close())); + connect(nextButton,SIGNAL(clicked()),this,SLOT(goNext())); + connect(backButton,SIGNAL(clicked()),this,SLOT(goBack())); + connect(searchButton,SIGNAL(clicked()),this,SLOT(search())); + connect(skipButton,SIGNAL(clicked()),this,SLOT(goToNextComic())); + + connect(selectVolumeWidget,SIGNAL(loadPage(QString,int)),this,SLOT(searchVolume(QString,int))); + connect(selectComicWidget,SIGNAL(loadPage(QString,int)),this,SLOT(getVolumeComicsInfo(QString,int))); + connect(sortVolumeComicsWidget,SIGNAL(loadPage(QString,int)),this,SLOT(getVolumeComicsInfo(QString,int))); +} + +void ComicVineDialog::goNext() +{ + // + if(content->currentWidget() == seriesQuestionWidget) + { + if(seriesQuestionWidget->getYes()) + { + QString volumeSearchString = comics[0].getParentFolderName(); + mode = Volume; + + if(volumeSearchString.isEmpty()) + showSearchVolume(); + else + { + status = AutoSearching; + showLoading(tr("Looking for volume...")); + searchVolume(volumeSearchString); + } + } + else + { + status = AutoSearching; + mode = SingleComicInList; + ComicDB comic = comics[currentIndex]; + QString title = comic.getTitleOrFileName(); + titleHeader->setSubTitle(tr("comic %1 of %2 - %3").arg(currentIndex+1).arg(comics.length()).arg(title)); + + showLoading(tr("Looking for volume...")); + searchVolume(title); + } + } + else if (content->currentWidget() == selectVolumeWidget) { + currentVolumeId = selectVolumeWidget->getSelectedVolumeId(); + getVolumeComicsInfo(currentVolumeId); + + } else if (content->currentWidget() == sortVolumeComicsWidget) { + showLoading(); + + //ComicDB-ComicVineID + QList > matchingInfo = sortVolumeComicsWidget->getMatchingInfo(); + int count = selectVolumeWidget->getSelectedVolumeNumIssues(); + QString publisher = selectVolumeWidget->getSelectedVolumePublisher(); + QtConcurrent::run(this, &ComicVineDialog::getComicsInfo,matchingInfo,count,publisher); + } else if (content->currentWidget() == selectComicWidget) + { + showLoading(); + QString comicId = selectComicWidget->getSelectedComicId(); + int count = selectVolumeWidget->getSelectedVolumeNumIssues(); + QString publisher = selectVolumeWidget->getSelectedVolumePublisher(); + QtConcurrent::run(this, &ComicVineDialog::getComicInfo,comicId,count,publisher); + } +} + +void ComicVineDialog::goBack() +{ + switch (status) { + case SelectingSeries: + if(mode == Volume) + showSearchVolume(); + else + showSearchSingleComic(); + break; + case SortingComics: + showSelectVolume(); + break; + case SelectingComic: + if(mode == SingleComic) + showSelectVolume(); + break; + case AutoSearching: + if(mode == Volume) + showSearchVolume(); + else + showSearchSingleComic(); + default: + if(mode == Volume) + showSearchVolume(); + else + showSearchSingleComic(); + break; + } +} + +void ComicVineDialog::setComics(const QList & comics) +{ + this->comics = comics; +} + +void ComicVineDialog::show() +{ + QDialog::show(); + + currentIndex = 0; + + seriesQuestionWidget->setYes(true); + searchSingleComicWidget->clean(); + searchVolumeWidget->clean(); + + if(comics.length() == 1) + { + status = AutoSearching; + mode = SingleComic; + + ComicDB singleComic = comics[0]; + QString title = singleComic.getTitleOrFileName(); + titleHeader->setSubTitle(title); + showLoading(tr("Looking for volume...")); + + searchVolume(singleComic.getParentFolderName()); + QLOG_TRACE() << singleComic.getParentFolderName(); + }else if(comics.length()>1) + { + titleHeader->setSubTitle(tr("%1 comics selected").arg(comics.length())); + showSeriesQuestion(); + } +} + +void ComicVineDialog::doLoading() +{ + QWidget * w = new QWidget; + QVBoxLayout * l = new QVBoxLayout; + + YACReaderBusyWidget * bw = new YACReaderBusyWidget; + loadingMessage = new QLabel; + + loadingMessage->setStyleSheet("QLabel {color:white; font-size:12px;font-family:Arial;}"); + + l->addStretch(); + l->addWidget(bw,0,Qt::AlignHCenter); + l->addStretch(); + l->addWidget(loadingMessage); + + + l->setContentsMargins(0,0,0,0); + w->setLayout(l); + w->setContentsMargins(0,0,0,0); + + content->addWidget(w); +} + +void ComicVineDialog::debugClientResults(const QString & string) +{ + ResponseParser p; + p.loadJSONResponse(string); + //QMessageBox::information(0,"Result", QString("Number of results : %1").arg(p.getNumResults())); + if(p.responseError()) + { + QMessageBox::critical(0,tr("Error connecting to ComicVine"), tr("unknown error")); + goBack(); + } + else + { + switch(mode) + { + case SingleComic: case SingleComicInList: + if(p.getNumResults() == 0) + showSearchSingleComic(); + else + if(status == SearchingVolume) + showSelectVolume(string); + else + showSelectComic(string); + break; + case Volume: + if(p.getNumResults() == 0) + showSearchVolume(); + else + showSelectVolume(string); + break; + } + } +} + +void ComicVineDialog::showSeriesQuestion() +{ + status = AskingForInfo; + content->setCurrentWidget(seriesQuestionWidget); + backButton->setHidden(true); + skipButton->setHidden(true); + nextButton->setVisible(true); + searchButton->setHidden(true); + closeButton->setVisible(true); + + if(mode == SingleComicInList) + skipButton->setVisible(true); + else + skipButton->setHidden(true); +} + +void ComicVineDialog::showSearchSingleComic() +{ + status = AskingForInfo; + content->setCurrentWidget(searchSingleComicWidget); + backButton->setHidden(true); + skipButton->setHidden(true); + nextButton->setHidden(true); + searchButton->setVisible(true); + closeButton->setVisible(true); + + if(mode == SingleComicInList) + skipButton->setVisible(true); + else + skipButton->setHidden(true); +} + +void ComicVineDialog::showSearchVolume() +{ + status = AskingForInfo; + content->setCurrentWidget(searchVolumeWidget); + backButton->setHidden(true); + nextButton->setHidden(true); + searchButton->setVisible(true); + closeButton->setVisible(true); + toggleSkipButton(); +} + +void ComicVineDialog::showSelectVolume(const QString & json) +{ + showSelectVolume(); + selectVolumeWidget->load(json,currentVolumeSearchString); +} + +void ComicVineDialog::showSelectVolume() +{ + status = SelectingSeries; + + content->setCurrentWidget(selectVolumeWidget); + + backButton->setVisible(true); + nextButton->setVisible(true); + searchButton->setHidden(true); + closeButton->setVisible(true); + toggleSkipButton(); +} + +void ComicVineDialog::showSelectComic(const QString &json) +{ + status = SelectingComic; + + content->setCurrentWidget(selectComicWidget); + selectComicWidget->load(json,currentVolumeId); + + backButton->setVisible(true); + nextButton->setVisible(true); + searchButton->setHidden(true); + closeButton->setVisible(true); + toggleSkipButton(); +} + +void ComicVineDialog::showSortVolumeComics(const QString &json) +{ + status = SortingComics; + + content->setCurrentWidget(sortVolumeComicsWidget); + + sortVolumeComicsWidget->setData(comics, json, currentVolumeId); + + backButton->setVisible(true); + nextButton->setVisible(true); + searchButton->setHidden(true); + closeButton->setVisible(true); + toggleSkipButton(); +} + +void ComicVineDialog::queryTimeOut() +{ + QMessageBox::warning(this,"Comic Vine error", "Time out connecting to Comic Vine"); + + switch (status) { + case AutoSearching: + if(mode == Volume) + showSearchVolume(); + else + showSearchSingleComic(); + break; + case SearchingVolume: + if(mode == Volume) + showSearchVolume(); + else + showSearchSingleComic(); + break; + case SearchingSingleComic: + showSearchSingleComic(); + break; + case GettingVolumeComics: + showSelectVolume(); + break; + default: + break; + } +} + +void ComicVineDialog::getComicsInfo(QList > & matchingInfo, int count,const QString & publisher) +{ + QPair p; + QList comics; + foreach (p, matchingInfo) { + ComicVineClient * comicVineClient = new ComicVineClient; + //connect(comicVineClient,SIGNAL(searchResult(QString)),this,SLOT(debugClientResults(QString))); + //connect(comicVineClient,SIGNAL(timeOut()),this,SLOT(queryTimeOut())); + //connect(comicVineClient,SIGNAL(finished()),comicVineClient,SLOT(deleteLater())); + bool error; + bool timeout; + QByteArray result = comicVineClient->getComicDetail(p.second,error,timeout); //TODO check timeOut or Connection error + if(error || timeout) + continue; //TODO + ComicDB comic = parseComicInfo(p.first,result,count,publisher);//TODO check result error + comic.info.comicVineID = p.second; + comics.push_back(comic); + + setLoadingMessage(tr("Retrieving tags for : %1").arg(p.first.getFileName())); + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + db.open(); + db.transaction(); + foreach(ComicDB comic, comics) + { + DBHelper::update(&(comic.info),db); + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(databasePath); + + close(); + emit accepted(); +} + +void ComicVineDialog::getComicInfo(const QString &comicId, int count, const QString &publisher) +{ + + ComicVineClient * comicVineClient = new ComicVineClient; + bool error; + bool timeout; + QByteArray result = comicVineClient->getComicDetail(comicId,error,timeout); //TODO check timeOut or Connection error + if(error || timeout) + { + //TODO + if(mode == SingleComic || currentIndex == (comics.count()-1)) + { + close(); + emit accepted(); + } else + { + goToNextComic(); + } + } + + ComicDB comic = parseComicInfo(comics[currentIndex],result,count,publisher); //TODO check result error + comic.info.comicVineID = comicId; + setLoadingMessage(tr("Retrieving tags for : %1").arg(comics[currentIndex].getFileName())); + + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + db.open(); + db.transaction(); + + DBHelper::update(&(comic.info),db); + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(databasePath); + + if(mode == SingleComic || currentIndex == (comics.count()-1)) + { + close(); + emit accepted(); + } else + { + goToNextComic(); + } +} + +ComicDB ComicVineDialog::parseComicInfo(ComicDB & comic, const QString & json, int count, const QString & publisher) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + json + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + qDebug("Error detected"); + } + else + { + int numResults = sc.property("number_of_total_results").toString().toInt(); //fix to weird behaviour using hasNext + + if(numResults > 0) + { + QScriptValue result = sc.property("results"); + + QString title = result.property("name").toString(); + + QString number = result.property("issue_number").toString(); + //QString count; //get from select volume + + + QString volume = result.property("volume").property("name").toString(); + QString storyArc; //story_arc + QString arcNumber; //?? + QString arcCount; //count_of_issue_appearances -> NO + + QString genere; //no + + QMap authors = getAuthors(result.property("person_credits")); + + QString writer = QStringList(authors.values("writer")).join("\n"); + QString penciller = QStringList(authors.values("penciller")).join("\n"); + QString inker = QStringList(authors.values("inker")).join("\n"); + QString colorist = QStringList(authors.values("colorist")).join("\n"); + QString letterer = QStringList(authors.values("letterer")).join("\n"); + QString coverArtist = QStringList(authors.values("cover")).join("\n"); + + QString date = result.property("cover_date").toString(); + + //QString publisher; //get from select volume + QString format; //no + bool color; //no + QString ageRating; //no + + QString synopsis = result.property("description").toString().remove(QRegExp("<[^>]*>")); //description + QString characters = getCharacters(result.property("character_credits")); + + comic.info.title = title; + + comic.info.number = number; + comic.info.count = count; + + comic.info.writer = writer; + comic.info.penciller = penciller; + comic.info.inker = inker; + comic.info.colorist = colorist; + comic.info.letterer = letterer; + comic.info.coverArtist = coverArtist; + + QStringList tempList = date.split("-"); + std::reverse(tempList.begin(),tempList.end()); + comic.info.date = tempList.join("/"); + comic.info.volume = volume; + + comic.info.publisher = publisher; + + comic.info.synopsis = synopsis; + comic.info.characters = characters; + } + } + return comic; +} + +QString ComicVineDialog::getCharacters(const QScriptValue &json_characters) +{ + QString characters; + + QScriptValueIterator it(json_characters); + QScriptValue resultsValue; + while (it.hasNext()) { + it.next(); + if(it.flags() & QScriptValue::SkipInEnumeration) + continue; + resultsValue = it.value(); + + characters += resultsValue.property("name").toString() + "\n"; + } + + return characters; +} + +QMap ComicVineDialog::getAuthors(const QScriptValue &json_authors) +{ + QMap authors; + + QScriptValueIterator it(json_authors); + QScriptValue resultsValue; + while (it.hasNext()) { + it.next(); + if(it.flags() & QScriptValue::SkipInEnumeration) + continue; + resultsValue = it.value(); + + QString authorName = resultsValue.property("name").toString(); + + QStringList roles = resultsValue.property("role").toString().split(","); + foreach(QString role, roles) + { + if(role.trimmed() == "writer") + authors.insertMulti("writer",authorName); + else if(role.trimmed() == "inker") + authors.insertMulti("inker",authorName); + else if(role.trimmed() == "penciler" || role.trimmed() == "penciller") + authors.insertMulti("penciller",authorName); + else if(role.trimmed() == "colorist") + authors.insertMulti("colorist",authorName); + else if(role.trimmed() == "letterer") + authors.insertMulti("letterer",authorName); + else if(role.trimmed() == "cover") + authors.insertMulti("cover",authorName); + } + } + + return authors; +} + +void ComicVineDialog::toggleSkipButton() +{ + if (mode == SingleComicInList) + skipButton->setVisible(true); + else + skipButton->setHidden(true); +} + +void ComicVineDialog::goToNextComic() +{ + if(mode == SingleComic || currentIndex == (comics.count()-1)) + { + close(); + emit accepted(); + return; + } + + currentIndex++; + + showSearchSingleComic(); + + ComicDB comic = comics[currentIndex]; + QString title = comic.getTitleOrFileName(); + titleHeader->setSubTitle(tr("comic %1 of %2 - %3").arg(currentIndex+1).arg(comics.length()).arg(title)); +} + +void ComicVineDialog::showLoading(const QString &message) +{ + content->setCurrentIndex(0); + loadingMessage->setText(message); + backButton->setHidden(true); + skipButton->setHidden(true); + nextButton->setHidden(true); + searchButton->setHidden(true); + closeButton->setVisible(true); +} + +void ComicVineDialog::setLoadingMessage(const QString &message) +{ + loadingMessage->setText(message); +} + +void ComicVineDialog::search() +{ + switch (mode) { + case Volume: + launchSearchVolume(); + break; + default: + launchSearchComic(); + break; + } +} + +void ComicVineDialog::searchVolume(const QString &v, int page) +{ + showLoading(tr("Looking for volume...")); + + currentVolumeSearchString = v; + + ComicVineClient * comicVineClient = new ComicVineClient; + connect(comicVineClient,SIGNAL(searchResult(QString)),this,SLOT(debugClientResults(QString))); + connect(comicVineClient,SIGNAL(timeOut()),this,SLOT(queryTimeOut())); + connect(comicVineClient,SIGNAL(finished()),comicVineClient,SLOT(deleteLater())); + comicVineClient->search(v,page); + + status = SearchingVolume; +} + +void ComicVineDialog::getVolumeComicsInfo(const QString &vID, int page) +{ + showLoading(tr("Retrieving volume info...")); + + status = GettingVolumeComics; + + ComicVineClient * comicVineClient = new ComicVineClient; + if(mode == Volume) + connect(comicVineClient,SIGNAL(volumeComicsInfo(QString)),this,SLOT(showSortVolumeComics(QString))); + else + connect(comicVineClient,SIGNAL(volumeComicsInfo(QString)),this,SLOT(showSelectComic(QString))); + connect(comicVineClient,SIGNAL(timeOut()),this,SLOT(queryTimeOut())); + connect(comicVineClient,SIGNAL(finished()),comicVineClient,SLOT(deleteLater())); + + QLOG_TRACE() << vID; + + comicVineClient->getVolumeComicsInfo(vID,page); +} + +void ComicVineDialog::launchSearchVolume() +{ + showLoading(tr("Looking for volume...")); + //TODO: check if volume info is empty. + searchVolume(searchVolumeWidget->getVolumeInfo()); +} + +void ComicVineDialog::launchSearchComic() +{ + showLoading(tr("Looking for comic...")); + + QString volumeInfo = searchSingleComicWidget->getVolumeInfo(); + //QString comicInfo = searchSingleComicWidget->getComicInfo(); + //int comicNumber = searchSingleComicWidget->getComicNumber(); + + //if(comicInfo.isEmpty() && comicNumber == -1) + searchVolume(volumeInfo); +} + diff --git a/YACReaderLibrary/comic_vine/comic_vine_dialog.h b/YACReaderLibrary/comic_vine/comic_vine_dialog.h new file mode 100644 index 00000000..6474afa1 --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_dialog.h @@ -0,0 +1,130 @@ +#ifndef COMIC_VINE_DIALOG_H +#define COMIC_VINE_DIALOG_H + +#include + +#include "comic_db.h" + +class QPushButton; +class QStackedWidget; +class QLabel; +class QRadioButton; +class ComicVineClient; +class QTableView; +class TitleHeader; +class SeriesQuestion; +class SearchSingleComic; +class SearchVolume; +class SelectComic; +class SelectVolume; +class SortVolumeComics; +class QScriptValue; + +//TODO this should use a QStateMachine +//---------------------------------------- +class ComicVineDialog : public QDialog +{ + Q_OBJECT +public: + explicit ComicVineDialog(QWidget *parent = 0); + QString databasePath; + QString basePath; + void setComics(const QList & comics); + + +signals: + +public slots: + void show(); + +protected slots: + void goNext(); + void goBack(); + void debugClientResults(const QString & string); + //show widget methods + void showSeriesQuestion(); + void showSearchSingleComic(); + void showSearchVolume(); + void showLoading(const QString & message = ""); + void search(); + void searchVolume(const QString & v, int page = 1); + void getVolumeComicsInfo(const QString &vID, int page = 1); + void launchSearchVolume(); + void launchSearchComic(); + void showSelectVolume(const QString & json); + void showSelectVolume(); + void showSelectComic(const QString & json); + void showSortVolumeComics(const QString & json); + void queryTimeOut(); + void getComicsInfo(QList > & matchingInfo, int count, const QString & publisher); + void getComicInfo(const QString & comicId, int count, const QString & publisher); + ComicDB parseComicInfo(ComicDB &comic, const QString & json, int count, const QString &publisher); + void setLoadingMessage(const QString &message); + void goToNextComic(); + +private: + + QString getCharacters(const QScriptValue & json_characters); + QMap getAuthors(const QScriptValue & json_authors); + + void toggleSkipButton(); + + enum ScraperMode + { + SingleComic, //the scraper has been opened for a single comic + Volume, //the scraper is trying to get comics info for a whole volume + SingleComicInList //the scraper has been opened for a list of unrelated comics + }; + + enum ScraperStatus + { + AutoSearching, + AskingForInfo, + SelectingComic, + SelectingSeries, + SearchingSingleComic, + SearchingVolume, + SortingComics, + GettingVolumeComics + }; + + ScraperMode mode; + ScraperStatus status; + + int currentIndex; + + TitleHeader * titleHeader; + + QPushButton * skipButton; + QPushButton * backButton; + QPushButton * nextButton; + QPushButton * searchButton; + QPushButton * closeButton; + + //stacked widgets + QStackedWidget * content; + + QWidget * infoNotFound; + QWidget * singleComicBrowser; + + QLabel * loadingMessage; + + void doLayout(); + void doStackedWidgets(); + void doLoading(); + void doConnections(); + + QList comics; + + SeriesQuestion * seriesQuestionWidget; + SearchSingleComic * searchSingleComicWidget; + SearchVolume * searchVolumeWidget; + SelectVolume * selectVolumeWidget; + SelectComic * selectComicWidget; + SortVolumeComics * sortVolumeComicsWidget; + + QString currentVolumeSearchString; + QString currentVolumeId; +}; + +#endif // COMIC_VINE_DIALOG_H diff --git a/YACReaderLibrary/comic_vine/model/comics_model.cpp b/YACReaderLibrary/comic_vine/model/comics_model.cpp new file mode 100644 index 00000000..5b194f24 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/comics_model.cpp @@ -0,0 +1,6 @@ +#include "comics_model.h" + +ComicsModel::ComicsModel(QObject *parent) : + JSONModel(parent) +{ +} diff --git a/YACReaderLibrary/comic_vine/model/comics_model.h b/YACReaderLibrary/comic_vine/model/comics_model.h new file mode 100644 index 00000000..86bfb2e5 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/comics_model.h @@ -0,0 +1,18 @@ +#ifndef COMICS_MODEL_H +#define COMICS_MODEL_H + +#include "json_model.h" + +class ComicsModel : public JSONModel +{ + Q_OBJECT +public: + explicit ComicsModel(QObject *parent = 0); + +signals: + +public slots: + +}; + +#endif // COMICS_MODEL_H diff --git a/YACReaderLibrary/comic_vine/model/json_model.cpp b/YACReaderLibrary/comic_vine/model/json_model.cpp new file mode 100644 index 00000000..d0c4ce41 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/json_model.cpp @@ -0,0 +1,6 @@ +#include "json_model.h" + +JSONModel::JSONModel(QObject *parent) : + QAbstractItemModel(parent) +{ +} diff --git a/YACReaderLibrary/comic_vine/model/json_model.h b/YACReaderLibrary/comic_vine/model/json_model.h new file mode 100644 index 00000000..443e9a20 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/json_model.h @@ -0,0 +1,19 @@ +#ifndef JSON_MODEL_H +#define JSON_MODEL_H + +#include + +class JSONModel : public QAbstractItemModel +{ + Q_OBJECT +public: + explicit JSONModel(QObject *parent = 0); + virtual void load(const QString & json) = 0 ; + +signals: + +public slots: + +}; + +#endif // JSON_MODEL_H diff --git a/YACReaderLibrary/comic_vine/model/local_comic_list_model.cpp b/YACReaderLibrary/comic_vine/model/local_comic_list_model.cpp new file mode 100644 index 00000000..861e61f1 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/local_comic_list_model.cpp @@ -0,0 +1,183 @@ +#include "local_comic_list_model.h" + +LocalComicListModel::LocalComicListModel(QObject *parent) : + QAbstractItemModel(parent),numExtraRows(0) +{ +} + +void LocalComicListModel::load(QList &comics) +{ + _data = comics; +} + + +QModelIndex LocalComicListModel::parent(const QModelIndex &index) const +{ + Q_UNUSED(index) + return QModelIndex(); //no parent +} + +int LocalComicListModel::rowCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + return _data.count(); +} + +int LocalComicListModel::columnCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + if(_data.isEmpty()) + return 0; + else + return 1;//_data.at(0)->count(); +} + +QVariant LocalComicListModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + if (role == Qt::DecorationRole) + { + return QVariant(); + } + if (role == Qt::TextAlignmentRole) + { + //TODO + } + + if(role != Qt::DisplayRole) + return QVariant(); + + int row = index.row(); + + //if(row < _data.count()) + return _data[row].getFileName(); + //else + //return QVariant(); +} + +Qt::ItemFlags LocalComicListModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} + +QVariant LocalComicListModel::headerData(int section, Qt::Orientation orientation, int role) const +{ + + if ( role == Qt::TextAlignmentRole) + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + { + return QVariant(QString(tr("file name"))); + } + + return QVariant(); +} + +QModelIndex LocalComicListModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column); +} + +QList LocalComicListModel::getData() +{ + return _data; +} + +void LocalComicListModel::removeComics(const QList &selectedIndexes) +{ + QModelIndex mi = selectedIndexes.first(); + QModelIndex lastMi = selectedIndexes.last(); + int sourceRow = mi.row(); + int sourceLastRow = lastMi.row(); + + beginRemoveRows(QModelIndex(),selectedIndexes.first().row(),selectedIndexes.last().row()); + + for(int i = sourceLastRow;i>=sourceRow;i--) + { + _removed.push_front(_data.at(i)); + _data.removeAt(i); + } + + endRemoveRows(); + + beginInsertRows(QModelIndex(),_data.count()-_removed.count(),_data.count()-1); + for(int i = 0; i<_removed.count(); i++) + _data.append(ComicDB()); + endInsertRows(); +} + +void LocalComicListModel::restoreAll() +{ + int numItemsToRemove = 0; + for(int i = 0;numItemsToRemove<_removed.count();i++) + { + if(_data.at(i).getFileName().isEmpty()) + { + beginRemoveRows(QModelIndex(),i,i); + _data.removeAt(i); + endRemoveRows(); + + beginInsertRows(QModelIndex(),i,i); + _data.insert(i,_removed.at(numItemsToRemove)); + endInsertRows(); + + numItemsToRemove++; + } + } + + _removed.clear(); +} + +void LocalComicListModel::moveSelectionUp(const QList &selectedIndexes) +{ + QModelIndex mi = selectedIndexes.first(); + QModelIndex lastMi = selectedIndexes.last(); + int sourceRow = mi.row(); + int sourceLastRow = lastMi.row(); + int destRow = sourceRow - 1; + + if(destRow < 0) + return; + + beginMoveRows(mi.parent(),sourceRow,sourceLastRow,mi.parent(),destRow); + + for(int i = sourceRow; i <= sourceLastRow; i++) + _data.swap(i, i-1); + + endMoveRows(); +} + +void LocalComicListModel::moveSelectionDown(const QList &selectedIndexes) +{ + QModelIndex mi = selectedIndexes.first(); + QModelIndex lastMi = selectedIndexes.last(); + int sourceRow = mi.row(); + int sourceLastRow = lastMi.row(); + int destRow = sourceLastRow + 1; + + if(destRow >= _data.count()) + return; + + beginMoveRows(mi.parent(),sourceRow,sourceLastRow,mi.parent(),destRow+1); + + for(int i = sourceLastRow; i >= sourceRow; i--) + _data.swap(i, i+1); + + endMoveRows(); +} + +void LocalComicListModel::addExtraRows(int numRows) +{ + numExtraRows = numRows; + for(int i = 0; i + +#include "comic_db.h" + +class LocalComicListModel : public QAbstractItemModel +{ + Q_OBJECT +public: + explicit LocalComicListModel(QObject *parent = 0); + + void load(QList & comics); + + //QAbstractItemModel methods + QModelIndex parent(const QModelIndex &index) const; + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent) const; + QVariant data(const QModelIndex &index, int role) const; + Qt::ItemFlags flags(const QModelIndex &index) const; + QVariant headerData(int section, Qt::Orientation orientation, + int role = Qt::DisplayRole) const; + QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; + QList getData(); + + void removeComics(const QList & selectedIndexes); + void restoreAll(); +signals: + +public slots: + void moveSelectionUp(const QList & selectedIndexes); + void moveSelectionDown(const QList & selectedIndexes); + void addExtraRows(int numRows); + +private: + int numExtraRows; + QList _data; + QList _removed; +}; + +#endif // LOCAL_COMIC_LIST_MODEL_H diff --git a/YACReaderLibrary/comic_vine/model/response_parser.cpp b/YACReaderLibrary/comic_vine/model/response_parser.cpp new file mode 100644 index 00000000..1901def6 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/response_parser.cpp @@ -0,0 +1,61 @@ +#include "response_parser.h" + +#include +#include + +ResponseParser::ResponseParser(QObject *parent) : + QObject(parent),error(false),numResults(-1),currentPage(-1),totalPages(-1) +{ +} + +bool ResponseParser::responseError() +{ + return error; +} + +qint32 ResponseParser::getNumResults() +{ + return numResults; +} + +qint32 ResponseParser::getCurrentPage() +{ + return currentPage; +} + +qint32 ResponseParser::getTotalPages() +{ + return totalPages; +} + +void ResponseParser::loadJSONResponse(const QString &response) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + response + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + error = true; + qDebug("Error detected"); + } + else + { + error = false; + if(sc.property("number_of_total_results").isValid()) + numResults = sc.property("number_of_total_results").toString().toInt();// sc.property("number_of_total_results").toInt32(); + else + qDebug() << sc.property("oops").toString(); + + int limit = sc.property("limit").toInt32(); + int offset = sc.property("offset").toInt32(); + int total = sc.property("number_of_total_results").toInt32(); + if(limit > 0) + { + totalPages = (total / limit) + (total%limit>0?1:0); + currentPage = (offset / limit) + 1; + } + else + totalPages = currentPage = 1; + } +} diff --git a/YACReaderLibrary/comic_vine/model/response_parser.h b/YACReaderLibrary/comic_vine/model/response_parser.h new file mode 100644 index 00000000..9d325edf --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/response_parser.h @@ -0,0 +1,27 @@ +#ifndef RESPONSE_PARSER_H +#define RESPONSE_PARSER_H + +#include + +class ResponseParser : public QObject +{ + Q_OBJECT +public: + explicit ResponseParser(QObject *parent = 0); + bool responseError(); + qint32 getNumResults(); + qint32 getCurrentPage(); + qint32 getTotalPages(); +signals: + +public slots: + void loadJSONResponse(const QString & response); + +protected: + bool error; + qint32 numResults; + qint32 currentPage; + qint32 totalPages; +}; + +#endif // RESPONSE_PARSER_H diff --git a/YACReaderLibrary/comic_vine/model/volume_comics_model.cpp b/YACReaderLibrary/comic_vine/model/volume_comics_model.cpp new file mode 100644 index 00000000..0a7a7e1c --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/volume_comics_model.cpp @@ -0,0 +1,172 @@ +#include "volume_comics_model.h" +#include "qnaturalsorting.h" + + +#include + +bool lessThan(const QList & left, const QList & right) +{ + if ((left.count() > 0) && (right.count() > 0)) + return naturalSortLessThanCI(left.at(0),right.at(0)); + else + return true; +} + +VolumeComicsModel::VolumeComicsModel(QObject * parent) : + JSONModel(parent),numExtraRows(0) +{ +} + +void VolumeComicsModel::load(const QString & json) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + json + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + qDebug("Error detected"); + } + else + { + QScriptValueIterator it(sc.property("results")); + //bool test; + QScriptValue resultsValue; + while (it.hasNext()) { + it.next(); + if(it.flags() & QScriptValue::SkipInEnumeration) + continue; + resultsValue = it.value(); + QString issueNumber = resultsValue.property("issue_number").toString(); + QString name = resultsValue.property("name").toString(); + QString coverURL = resultsValue.property("image").property("medium_url").toString(); + QString id = resultsValue.property("id").toString(); + QStringList l; + l << issueNumber << name << coverURL << id; + _data.push_back(l); + } + + qSort(_data.begin(),_data.end(),lessThan); + } +} + +QModelIndex VolumeComicsModel::parent(const QModelIndex &index) const +{ + Q_UNUSED(index) + return QModelIndex(); //no parent +} + +int VolumeComicsModel::rowCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + return _data.count() + numExtraRows; +} + +int VolumeComicsModel::columnCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + if(_data.isEmpty()) + return 0; + else + return 2; +} + +QVariant VolumeComicsModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + int row = index.row(); + int column = index.column(); + + if (role == Qt::DecorationRole) + { + return QVariant(); + } + if (role == Qt::TextAlignmentRole) + { + switch(column)//TODO obtener esto de la query + { + case ISSUE: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TITLE: + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + } + } + + if(role != Qt::DisplayRole) + return QVariant(); + + if(row<_data.count()) + return _data[row][column]; + else + return QVariant(); +} + +Qt::ItemFlags VolumeComicsModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} + +QVariant VolumeComicsModel::headerData(int section, Qt::Orientation orientation, int role) const +{ + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + { + switch(section)//TODO obtener esto de la query + { + case ISSUE: + return QVariant(QString("issue")); + case TITLE: + return QVariant(QString(tr("title"))); + } + } + + if (orientation == Qt::Horizontal && role == Qt::TextAlignmentRole) + { + switch(section)//TODO obtener esto de la query + { + case ISSUE: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TITLE: + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + } + } + + return QVariant(); +} + +QModelIndex VolumeComicsModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column); +} + +QString VolumeComicsModel::getComicId(const QModelIndex &index) const +{ + int row = index.row(); + if(row >= _data.count()) + return ""; + return _data[row][ID]; +} + +QString VolumeComicsModel::getComicId(int row) const +{ + if(row >= _data.count()) + return ""; + return _data[row][ID]; +} + +QString VolumeComicsModel::getCoverURL(const QModelIndex &index) const +{ + return _data[index.row()][COVER_URL]; +} + +void VolumeComicsModel::addExtraRows(int numRows) +{ + numExtraRows = numRows; +} + diff --git a/YACReaderLibrary/comic_vine/model/volume_comics_model.h b/YACReaderLibrary/comic_vine/model/volume_comics_model.h new file mode 100644 index 00000000..d9007891 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/volume_comics_model.h @@ -0,0 +1,41 @@ +#ifndef VOLUME_COMICS_MODEL_H +#define VOLUME_COMICS_MODEL_H + +#include "json_model.h" + +class VolumeComicsModel : public JSONModel +{ + Q_OBJECT +public: + explicit VolumeComicsModel(QObject *parent = 0); + void load(const QString & json); + + QModelIndex parent(const QModelIndex &index) const; + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent) const; + QVariant data(const QModelIndex &index, int role) const; + Qt::ItemFlags flags(const QModelIndex &index) const; + QVariant headerData(int section, Qt::Orientation orientation, + int role = Qt::DisplayRole) const; + QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; +signals: + +public slots: + QString getComicId(const QModelIndex &index) const; + QString getComicId(int row) const; + QString getCoverURL(const QModelIndex &index) const; + void addExtraRows(int numRows); + +private: + int numExtraRows; + QList > _data; + + enum Column { + ISSUE = 0, + TITLE, + COVER_URL, + ID + }; +}; + +#endif // VOLUME_COMICS_MODEL_H diff --git a/YACReaderLibrary/comic_vine/model/volumes_model.cpp b/YACReaderLibrary/comic_vine/model/volumes_model.cpp new file mode 100644 index 00000000..e3bfff12 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/volumes_model.cpp @@ -0,0 +1,161 @@ +#include "volumes_model.h" + +#include + + +VolumesModel::VolumesModel(QObject *parent) : + JSONModel(parent) +{ +} + +VolumesModel::~VolumesModel() +{ + //std::for_each(_data.begin(), _data.end(), [](QList * ptr) { delete ptr; }); +} + +void VolumesModel::load(const QString &json) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + json + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + qDebug("Error detected"); + } + else + { + int numResults = sc.property("number_of_total_results").toString().toInt(); //fix to weird behaviour using hasNext + QScriptValueIterator it(sc.property("results")); + bool test; + QScriptValue resultsValue; + while (it.hasNext()) { + it.next(); + resultsValue = it.value(); + QString numIssues = resultsValue.property("count_of_issues").toString(); + QString year = resultsValue.property("start_year").toString(); + QString name = resultsValue.property("name").toString(); + QString publisher = resultsValue.property("publisher").property("name").toString(); + QString url = resultsValue.property("image").property("medium_url").toString(); + QString deck = resultsValue.property("deck").toString(); + QString id = resultsValue.property("id").toString(); + QStringList l; + l << name << year << numIssues << publisher << url << deck << id; + test = name.isEmpty() && year.isEmpty() && numIssues.isEmpty() && url.isEmpty(); + if(numResults>0 && !test) + _data.push_back(l); + numResults--; + } + } +} + +QModelIndex VolumesModel::parent(const QModelIndex &index) const +{ + Q_UNUSED(index) + return QModelIndex(); //no parent +} + +int VolumesModel::rowCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + return _data.count(); +} + +int VolumesModel::columnCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + if(_data.isEmpty()) + return 0; + else + return 4;//_data.at(0)->count(); +} + +QVariant VolumesModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + if (role == Qt::DecorationRole) + { + return QVariant(); + } + if (role == Qt::TextAlignmentRole) + { + //TODO + } + + if(role != Qt::DisplayRole) + return QVariant(); + + int row = index.row(); + int column = index.column(); + return _data[row][column]; +} + +Qt::ItemFlags VolumesModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} + +QVariant VolumesModel::headerData(int section, Qt::Orientation orientation, int role) const +{ + + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + { + switch(section)//TODO obtener esto de la query + { + case SERIES: + return QVariant(QString("series")); + case YEAR: + return QVariant(QString(tr("year"))); + case ISSUES: + return QVariant(QString(tr("issues"))); + case PUBLISHER: + return QVariant(QString(tr("publisher"))); + } + } + + if (orientation == Qt::Horizontal && role == Qt::TextAlignmentRole) + { + switch(section)//TODO obtener esto de la query + { + case YEAR: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ISSUES: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + } + } + + return QVariant(); +} + +QModelIndex VolumesModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column); +} + +QString VolumesModel::getVolumeId(const QModelIndex &index) const +{ + return _data[index.row()][ID]; +} + +int VolumesModel::getNumIssues(const QModelIndex &index) const +{ + return _data[index.row()][ISSUES].toInt(); +} + +QString VolumesModel::getPublisher(const QModelIndex &index) const +{ + return _data[index.row()][PUBLISHER]; +} + +QString VolumesModel::getCoverURL(const QModelIndex &index) const +{ + return _data[index.row()][COVER_URL]; +} + diff --git a/YACReaderLibrary/comic_vine/model/volumes_model.h b/YACReaderLibrary/comic_vine/model/volumes_model.h new file mode 100644 index 00000000..f8a2fe05 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/volumes_model.h @@ -0,0 +1,50 @@ +#ifndef VOLUMES_MODEL_H +#define VOLUMES_MODEL_H + +#include "json_model.h" + +class VolumesModel : public JSONModel +{ + Q_OBJECT +public: + explicit VolumesModel(QObject *parent = 0); + virtual ~VolumesModel(); + //receive a valid json with a list of volumes + void load(const QString & json); + + //QAbstractItemModel methods + QModelIndex parent(const QModelIndex &index) const; + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent) const; + QVariant data(const QModelIndex &index, int role) const; + Qt::ItemFlags flags(const QModelIndex &index) const; + QVariant headerData(int section, Qt::Orientation orientation, + int role = Qt::DisplayRole) const; + QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; + + QString getVolumeId(const QModelIndex & index) const; + int getNumIssues(const QModelIndex & index) const; + QString getPublisher(const QModelIndex & index) const; + QString getCoverURL(const QModelIndex & index) const; + +signals: + +public slots: + +private: + QList > _data; + +public: + enum Column { + SERIES = 0, + YEAR, + ISSUES, + PUBLISHER, + COVER_URL, + DECK, + ID + }; + +}; + +#endif // VOLUMES_MODEL_H diff --git a/YACReaderLibrary/comic_vine/scraper_lineedit.cpp b/YACReaderLibrary/comic_vine/scraper_lineedit.cpp new file mode 100644 index 00000000..94d03e95 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_lineedit.cpp @@ -0,0 +1,21 @@ +#include "scraper_lineedit.h" +#include + +ScraperLineEdit::ScraperLineEdit(const QString & title, QWidget * widget) + :QLineEdit(widget) +{ + titleLabel = new QLabel(title,this); + titleLabel->setStyleSheet("QLabel {color:white;}"); + + setStyleSheet(QString("QLineEdit {" + "border:none; background-color: #2E2E2E; color : white; padding-left: %1; padding-bottom: 1px; margin-bottom: 0px;" + "}").arg(titleLabel->sizeHint().width()+6)); + + setFixedHeight(22); +} + +void ScraperLineEdit::resizeEvent(QResizeEvent *) +{ + QSize szl = titleLabel->sizeHint(); + titleLabel->move(6,(rect().bottom() + 1 - szl.height())/2); +} diff --git a/YACReaderLibrary/comic_vine/scraper_lineedit.h b/YACReaderLibrary/comic_vine/scraper_lineedit.h new file mode 100644 index 00000000..30665b11 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_lineedit.h @@ -0,0 +1,19 @@ +#ifndef SCRAPPER_LINEEDIT_H +#define SCRAPPER_LINEEDIT_H + +#include + +class QLabel; + +class ScraperLineEdit : public QLineEdit +{ + Q_OBJECT +public: + ScraperLineEdit(const QString & title, QWidget * widget = 0); +protected: + void resizeEvent(QResizeEvent *); +private: + QLabel * titleLabel; +}; + +#endif // SCRAPPER_LINEEDIT_H diff --git a/YACReaderLibrary/comic_vine/scraper_results_paginator.cpp b/YACReaderLibrary/comic_vine/scraper_results_paginator.cpp new file mode 100644 index 00000000..f627d315 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_results_paginator.cpp @@ -0,0 +1,75 @@ +#include "scraper_results_paginator.h" +#include "response_parser.h" + +#include +#include +#include +#include + + +ScraperResultsPaginator::ScraperResultsPaginator(QWidget *parent) : + QWidget(parent),customLabel("items") +{ + QHBoxLayout * pagesButtonsLayout = new QHBoxLayout; + + QString labelStylesheet = "QLabel {color:white; font-size:12px;font-family:Arial;}"; + + nextPage = new QToolButton; + nextPage->setStyleSheet("QToolButton {border:none;}"); + QPixmap np(":/images/comic_vine/nextPage.png"); + nextPage->setIconSize(np.size()); + nextPage->setIcon(np); + + previousPage = new QToolButton; + previousPage->setStyleSheet("QToolButton {border:none;}"); + QPixmap pp(":/images/comic_vine/previousPage.png"); + previousPage->setIconSize(pp.size()); + previousPage->setIcon(pp); + + connect(nextPage,SIGNAL(clicked()),this,SIGNAL(loadNextPage())); + connect(previousPage,SIGNAL(clicked()),this,SIGNAL(loadPreviousPage())); + + numElements = new QLabel(tr("Number of volumes found : %1")); + numElements->setStyleSheet(labelStylesheet); + numPages = new QLabel(tr("page %1 of %2")); + numPages->setStyleSheet(labelStylesheet); + + pagesButtonsLayout->addSpacing(15); + pagesButtonsLayout->addWidget(numElements); + pagesButtonsLayout->addStretch(); + pagesButtonsLayout->addWidget(numPages); + pagesButtonsLayout->addWidget(previousPage); + pagesButtonsLayout->addWidget(nextPage); + + setContentsMargins(0,0,0,0); + pagesButtonsLayout->setContentsMargins(0,0,0,0); + + setLayout(pagesButtonsLayout); +} + +void ScraperResultsPaginator::update(const QString &json) +{ + ResponseParser rp; + rp.loadJSONResponse(json); + + currentPage = rp.getCurrentPage(); + numElements->setText(tr("Number of %1 found : %2").arg(customLabel).arg(rp.getNumResults())); + numPages->setText(tr("page %1 of %2").arg(currentPage).arg(rp.getTotalPages())); + + previousPage->setDisabled(currentPage == 1); + nextPage->setDisabled(currentPage == rp.getTotalPages()); + + numPages->setHidden(rp.getTotalPages()==1); + previousPage->setHidden(rp.getTotalPages()==1); + nextPage->setHidden(rp.getTotalPages()==1); +} + +int ScraperResultsPaginator::getCurrentPage() +{ + return currentPage; +} + +void ScraperResultsPaginator::setCustomLabel(const QString &label) +{ + customLabel = label; +} diff --git a/YACReaderLibrary/comic_vine/scraper_results_paginator.h b/YACReaderLibrary/comic_vine/scraper_results_paginator.h new file mode 100644 index 00000000..c371b7af --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_results_paginator.h @@ -0,0 +1,34 @@ +#ifndef SCRAPER_RESULTS_PAGINATOR_H +#define SCRAPER_RESULTS_PAGINATOR_H + +#include + +class QToolButton; +class QLabel; + +class ScraperResultsPaginator : public QWidget +{ + Q_OBJECT +public: + explicit ScraperResultsPaginator(QWidget *parent = 0); + void update(const QString & json); + int getCurrentPage(); + void setCustomLabel(const QString & label); +signals: + void loadNextPage(); + void loadPreviousPage(); + +public slots: + +private: + QToolButton * nextPage; + QToolButton * previousPage; + QLabel * numElements; + QLabel * numPages; + + int currentPage; + + QString customLabel; +}; + +#endif // SCRAPER_RESULTS_PAGINATOR_H diff --git a/YACReaderLibrary/comic_vine/scraper_scroll_label.cpp b/YACReaderLibrary/comic_vine/scraper_scroll_label.cpp new file mode 100644 index 00000000..82ce0bd1 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_scroll_label.cpp @@ -0,0 +1,53 @@ +#include "scraper_scroll_label.h" + +#include +#include +#include + +ScraperScrollLabel::ScraperScrollLabel(QWidget *parent) : + QScrollArea(parent) +{ + textLabel = new QLabel(this); + textLabel->setStyleSheet("QLabel {background-color: #2B2B2B; color:white; font-size:12px; font-family:Arial; }"); + + textLabel->setWordWrap(true); + textLabel->setMinimumSize(168,12); + + setWidget(textLabel); + setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + setStyleSheet( + "QScrollArea {background-color:#2B2B2B; border:none;}" + "QScrollBar:vertical { border: none; background: #2B2B2B; width: 3px; margin: 0; }" + "QScrollBar:horizontal { border: none; background: #2B2B2B; height: 3px; margin: 0; }" + "QScrollBar::handle:vertical { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::handle:horizontal { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::sub-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::add-line:horizontal { border: none; background: #404040; width: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 0 3px 0;}" + "QScrollBar::sub-line:horizontal { border: none; background: #404040; width: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 0 3px 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical, QScrollBar::add-page:horizontal, QScrollBar::sub-page:horizontal {background: none; }" + ); + + connect(textLabel,SIGNAL(linkActivated(QString)),this,SLOT(openLink(QString))); +} + +void ScraperScrollLabel::setAltText(const QString &text) +{ + textLabel->setAlignment(Qt::AlignTop|Qt::AlignHCenter); + textLabel->setText(text); + textLabel->adjustSize(); +} + +void ScraperScrollLabel::setText(const QString &text) +{ + textLabel->setAlignment(Qt::AlignTop|Qt::AlignLeft); + textLabel->setText(text); + textLabel->adjustSize(); +} + +void ScraperScrollLabel::openLink(const QString & link) +{ + QDesktopServices::openUrl(QUrl("http://www.comicvine.com"+link)); +} diff --git a/YACReaderLibrary/comic_vine/scraper_scroll_label.h b/YACReaderLibrary/comic_vine/scraper_scroll_label.h new file mode 100644 index 00000000..8b4c82be --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_scroll_label.h @@ -0,0 +1,25 @@ +#ifndef SCRAPER_SCROLL_LABEL_H +#define SCRAPER_SCROLL_LABEL_H + +#include + +class QLabel; + +class ScraperScrollLabel : public QScrollArea +{ + Q_OBJECT +public: + explicit ScraperScrollLabel(QWidget *parent = 0); + +signals: + +public slots: + void setText(const QString & text); + void setAltText(const QString &text); + + void openLink(const QString &link); +private: + QLabel * textLabel; +}; + +#endif // SCRAPER_SCROLL_LABEL_H diff --git a/YACReaderLibrary/comic_vine/scraper_selector.cpp b/YACReaderLibrary/comic_vine/scraper_selector.cpp new file mode 100644 index 00000000..e79117b9 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_selector.cpp @@ -0,0 +1,25 @@ +#include "scraper_selector.h" + +ScraperSelector::ScraperSelector(QWidget *parent) : + QWidget(parent) +{ + paginator = new ScraperResultsPaginator; + connect(paginator,SIGNAL(loadNextPage()),this,SLOT(loadNextPage())); + connect(paginator,SIGNAL(loadPreviousPage()),this,SLOT(loadPreviousPage())); +} + +void ScraperSelector::load(const QString &json, const QString &searchString) +{ + currentSearchString = searchString; + paginator->update(json); +} + +void ScraperSelector::loadNextPage() +{ + emit loadPage(currentSearchString,paginator->getCurrentPage()+1); +} + +void ScraperSelector::loadPreviousPage() +{ + emit loadPage(currentSearchString,paginator->getCurrentPage()-1); +} diff --git a/YACReaderLibrary/comic_vine/scraper_selector.h b/YACReaderLibrary/comic_vine/scraper_selector.h new file mode 100644 index 00000000..34ce409f --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_selector.h @@ -0,0 +1,28 @@ +#ifndef SCRAPER_SELECTOR_H +#define SCRAPER_SELECTOR_H + +#include + +#include "scraper_results_paginator.h" + +class ScraperSelector : public QWidget +{ + Q_OBJECT +public: + explicit ScraperSelector(QWidget *parent = 0); + virtual void load(const QString & json, const QString & searchString); +public slots: + +signals: + void loadPage(QString,int); + +private slots: + void loadNextPage(); + void loadPreviousPage(); + +protected: + QString currentSearchString; + ScraperResultsPaginator * paginator; +}; + +#endif // SCRAPER_SELECTOR_H diff --git a/YACReaderLibrary/comic_vine/scraper_tableview.cpp b/YACReaderLibrary/comic_vine/scraper_tableview.cpp new file mode 100644 index 00000000..22dbbed1 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_tableview.cpp @@ -0,0 +1,61 @@ +#include "scraper_tableview.h" + +#include + +ScraperTableView::ScraperTableView(QWidget *parent) : + QTableView(parent) +{ + QString tableStylesheet = "QTableView {color:white; border:0px;alternate-background-color: #2E2E2E;background-color: #2B2B2B; outline: 0px;}" + "QTableView::item {outline: 0px; border: 0px; color:#FFFFFF;}" + "QTableView::item:selected {outline: 0px; background-color: #555555; }" + "QHeaderView::section:horizontal {background-color:#292929; border-bottom:1px solid #1F1F1F; border-right:1px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #292929, stop: 1 #1F1F1F); border-left:none; border-top:none; padding:4px; color:#ebebeb;}" + "QHeaderView::section:vertical {border-bottom: 1px solid #DFDFDF;border-top: 1px solid #FEFEFE;}" + "QHeaderView::down-arrow {image: url(':/images/comic_vine/downArrow.png');}" + "QHeaderView::up-arrow {image: url(':/images/comic_vine/upArrow.png');}" + "QScrollBar:vertical { border: none; background: #2B2B2B; width: 3px; margin: 0; }" + "QScrollBar:horizontal { border: none; background: #2B2B2B; height: 3px; margin: 0; }" + "QScrollBar::handle:vertical { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::handle:horizontal { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::sub-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::add-line:horizontal { border: none; background: #404040; width: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 0 3px 0;}" + "QScrollBar::sub-line:horizontal { border: none; background: #404040; width: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 0 3px 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical, QScrollBar::add-page:horizontal, QScrollBar::sub-page:horizontal {background: none; }"; + + setStyleSheet(tableStylesheet); + + setShowGrid(false); +#if QT_VERSION >= 0x050000 + verticalHeader()->setSectionResizeMode(QHeaderView::Fixed); +#else + verticalHeader()->setResizeMode(QHeaderView::Fixed); +#endif + + //comicView->horizontalHeader()->setResizeMode(QHeaderView::ResizeToContents); + horizontalHeader()->setStretchLastSection(true); +#if QT_VERSION >= 0x050000 + horizontalHeader()->setSectionsClickable(false); +#else + horizontalHeader()->setClickable(false); +#endif + //comicView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents); + verticalHeader()->setDefaultSectionSize(24); +#if QT_VERSION >= 0x050000 + verticalHeader()->setSectionsClickable(false); //TODO comportamiento anómalo +#else + verticalHeader()->setClickable(false); //TODO comportamiento anómalo +#endif + + setCornerButtonEnabled(false); + + setSelectionBehavior(QAbstractItemView::SelectRows); + setSelectionMode(QAbstractItemView::ExtendedSelection); + + setAlternatingRowColors(true); + + verticalHeader()->hide(); + + setSelectionMode(QAbstractItemView::SingleSelection); +} diff --git a/YACReaderLibrary/comic_vine/scraper_tableview.h b/YACReaderLibrary/comic_vine/scraper_tableview.h new file mode 100644 index 00000000..deb151c7 --- /dev/null +++ b/YACReaderLibrary/comic_vine/scraper_tableview.h @@ -0,0 +1,18 @@ +#ifndef SCRAPPER_TABLEVIEW_H +#define SCRAPPER_TABLEVIEW_H + +#include + +class ScraperTableView : public QTableView +{ + Q_OBJECT +public: + explicit ScraperTableView(QWidget *parent = 0); + +signals: + +public slots: + +}; + +#endif // SCRAPPER_TABLEVIEW_H diff --git a/YACReaderLibrary/comic_vine/search_single_comic.cpp b/YACReaderLibrary/comic_vine/search_single_comic.cpp new file mode 100644 index 00000000..431b048a --- /dev/null +++ b/YACReaderLibrary/comic_vine/search_single_comic.cpp @@ -0,0 +1,62 @@ +#include "search_single_comic.h" + +#include "scraper_lineedit.h" + +#include +#include +#include + +SearchSingleComic::SearchSingleComic(QWidget * parent) + :QWidget(parent) +{ + + //QLabel * label = new QLabel(tr("Please provide some aditional information. At least one field is needed.")); + QLabel * label = new QLabel(tr("Please provide some aditional information.")); + label->setStyleSheet("QLabel {color:white; font-size:12px;font-family:Arial;}"); + + //titleEdit = new ScraperLineEdit(tr("Title:")); + //numberEdit = new ScraperLineEdit(tr("Number:")); + volumeEdit = new ScraperLineEdit(tr("Series:")); + + //numberEdit->setMaximumWidth(126); + + QVBoxLayout * l = new QVBoxLayout; + //QHBoxLayout * hl = new QHBoxLayout; + //hl->addWidget(titleEdit); + //hl->addWidget(numberEdit); + + l->addSpacing(35); + l->addWidget(label); + //l->addLayout(hl); + l->addWidget(volumeEdit); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); +} + +QString SearchSingleComic::getVolumeInfo() +{ + return volumeEdit->text(); +} + +QString SearchSingleComic::getComicInfo() +{ + //return titleEdit->text(); + return ""; +} + +int SearchSingleComic::getComicNumber() +{ + //QString numberText = numberEdit->text(); + //if(numberText.isEmpty()) + // return -1; + //return numberText.toInt(); + return 0; +} + +void SearchSingleComic::clean() +{ + volumeEdit->clear(); +} diff --git a/YACReaderLibrary/comic_vine/search_single_comic.h b/YACReaderLibrary/comic_vine/search_single_comic.h new file mode 100644 index 00000000..5045ee69 --- /dev/null +++ b/YACReaderLibrary/comic_vine/search_single_comic.h @@ -0,0 +1,22 @@ +#ifndef SEARCH_SINGLE_COMIC_H +#define SEARCH_SINGLE_COMIC_H + +#include + +class ScraperLineEdit; + +class SearchSingleComic : public QWidget +{ + Q_OBJECT +public: + SearchSingleComic(QWidget * parent = 0); + QString getVolumeInfo(); + QString getComicInfo(); + int getComicNumber(); + void clean(); +private: + ScraperLineEdit * titleEdit; + ScraperLineEdit * numberEdit; + ScraperLineEdit * volumeEdit; +}; +#endif // SEARCH_SINGLE_COMIC_H diff --git a/YACReaderLibrary/comic_vine/search_volume.cpp b/YACReaderLibrary/comic_vine/search_volume.cpp new file mode 100644 index 00000000..cb779b6f --- /dev/null +++ b/YACReaderLibrary/comic_vine/search_volume.cpp @@ -0,0 +1,36 @@ +#include "search_volume.h" + +#include "scraper_lineedit.h" + +#include +#include + +SearchVolume::SearchVolume(QWidget * parent) + :QWidget(parent) +{ + QLabel * label = new QLabel(tr("Please provide some aditional information.")); + label->setStyleSheet("QLabel {color:white; font-size:12px;font-family:Arial;}"); + + volumeEdit = new ScraperLineEdit(tr("Series:")); + + QVBoxLayout * l = new QVBoxLayout; + + l->addSpacing(35); + l->addWidget(label); + l->addWidget(volumeEdit); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); +} + +void SearchVolume::clean() +{ + volumeEdit->clear(); +} + +QString SearchVolume::getVolumeInfo() +{ + return volumeEdit->text(); +} diff --git a/YACReaderLibrary/comic_vine/search_volume.h b/YACReaderLibrary/comic_vine/search_volume.h new file mode 100644 index 00000000..627baebc --- /dev/null +++ b/YACReaderLibrary/comic_vine/search_volume.h @@ -0,0 +1,21 @@ +#ifndef SEARCH_VOLUME_H +#define SEARCH_VOLUME_H + +#include + +class ScraperLineEdit; + + +class SearchVolume : public QWidget +{ + Q_OBJECT +public: + SearchVolume(QWidget * parent = 0); + void clean(); +public slots: + QString getVolumeInfo(); +private: + ScraperLineEdit * volumeEdit; +}; + +#endif // SEARCH_VOLUME_H diff --git a/YACReaderLibrary/comic_vine/select_comic.cpp b/YACReaderLibrary/comic_vine/select_comic.cpp new file mode 100644 index 00000000..8105dfb1 --- /dev/null +++ b/YACReaderLibrary/comic_vine/select_comic.cpp @@ -0,0 +1,150 @@ +#include "select_comic.h" + +#include "comic_vine_client.h" +#include "scraper_scroll_label.h" +#include "scraper_tableview.h" +#include "volume_comics_model.h" + +#include +#include +#include + +SelectComic::SelectComic(QWidget *parent) + :ScraperSelector(parent),model(0) +{ + QString labelStylesheet = "QLabel {color:white; font-size:12px;font-family:Arial;}"; + + QLabel * label = new QLabel(tr("Please, select the right comic info.")); + label->setStyleSheet(labelStylesheet); + + QVBoxLayout * l = new QVBoxLayout; + QWidget * leftWidget = new QWidget; + QVBoxLayout * left = new QVBoxLayout; + QVBoxLayout * right = new QVBoxLayout; + QHBoxLayout * content = new QHBoxLayout; + + right->setContentsMargins(0,0,0,0); + + //widgets + cover = new QLabel(); + cover->setScaledContents(true); + cover->setAlignment(Qt::AlignTop|Qt::AlignHCenter); + cover->setMinimumSize(168,168*5.0/3); + cover->setStyleSheet("QLabel {background-color: #2B2B2B; color:white; font-size:12px; font-family:Arial; }"); + detailLabel = new ScraperScrollLabel(this); + + tableComics = new ScraperTableView(this); + //connections + connect(tableComics,SIGNAL(clicked(QModelIndex)),this,SLOT(loadComicInfo(QModelIndex))); + + paginator->setCustomLabel(tr("comics")); + + left->addWidget(cover); + left->addWidget(detailLabel,1); + left->addStretch(); + leftWidget->setMaximumWidth(180); + leftWidget->setLayout(left); + left->setContentsMargins(0,0,0,0); + leftWidget->setContentsMargins(0,0,0,0); + + right->addWidget(tableComics,0,Qt::AlignRight|Qt::AlignTop); + right->addWidget(paginator); + + content->addWidget(leftWidget); + content->addLayout(right); + + l->addSpacing(15); + l->addWidget(label); + l->addSpacing(5); + l->addLayout(content); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); +} + +void SelectComic::load(const QString &json, const QString & searchString) +{ + VolumeComicsModel * tempM = new VolumeComicsModel(); + tempM->load(json); + tableComics->setModel(tempM); + + tableComics->setFixedSize(619,341); + + if(model != 0) + delete model; + + model = tempM; + + if(model->rowCount()>0) + { + tableComics->selectRow(0); + loadComicInfo(model->index(0,0)); + } + + tableComics->resizeColumnToContents(0); + + ScraperSelector::load(json,searchString); +} + +SelectComic::~SelectComic() {} + +void SelectComic::loadComicInfo(const QModelIndex &mi) +{ + QString coverURL = model->getCoverURL(mi); + QString id = model->getComicId(mi); + + QString loadingStyle = "%1"; + cover->setText(loadingStyle.arg(tr("loading cover"))); + detailLabel->setAltText(loadingStyle.arg(tr("loading description"))); + + ComicVineClient * comicVineClient = new ComicVineClient; + connect(comicVineClient,SIGNAL(comicCover(const QByteArray &)),this,SLOT(setCover(const QByteArray &))); + connect(comicVineClient,SIGNAL(finished()),comicVineClient,SLOT(deleteLater())); + comicVineClient->getComicCover(coverURL); + + ComicVineClient * comicVineClient2 = new ComicVineClient; + connect(comicVineClient2,SIGNAL(comicDetail(QString)),this,SLOT(setDescription(QString))); + connect(comicVineClient2,SIGNAL(finished()),comicVineClient2,SLOT(deleteLater())); + comicVineClient2->getComicDetailAsync(id); +} + +void SelectComic::setCover(const QByteArray & data) +{ + QPixmap p; + p.loadFromData(data); + int w = p.width(); + int h = p.height(); + + cover->setPixmap(p); + float aspectRatio = static_cast(w)/h; + + cover->setFixedSize(180,static_cast(180/aspectRatio)); + + cover->update(); +} + +void SelectComic::setDescription(const QString &jsonDetail) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + jsonDetail + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + qDebug("Error detected"); + } + else + { + + QScriptValue descriptionValues = sc.property("results").property("description"); + bool valid = !descriptionValues.isNull() && descriptionValues.isValid(); + detailLabel->setText(valid?descriptionValues.toString().replace("getComicId(tableComics->currentIndex()); +} diff --git a/YACReaderLibrary/comic_vine/select_comic.h b/YACReaderLibrary/comic_vine/select_comic.h new file mode 100644 index 00000000..5d14a08b --- /dev/null +++ b/YACReaderLibrary/comic_vine/select_comic.h @@ -0,0 +1,34 @@ +#ifndef SELECT_COMIC_H +#define SELECT_COMIC_H + +#include "scraper_selector.h" + +class QLabel; +class VolumeComicsModel; +class QModelIndex; + +class ScraperScrollLabel; +class ScraperTableView; + +class SelectComic : public ScraperSelector +{ + Q_OBJECT +public: + SelectComic(QWidget * parent = 0); + void load(const QString & json, const QString & searchString); + virtual ~SelectComic(); + +public slots: + void loadComicInfo(const QModelIndex & mi); + void setCover(const QByteArray &); + void setDescription(const QString & jsonDetail); + QString getSelectedComicId(); + +private: + QLabel * cover; + ScraperScrollLabel * detailLabel; + ScraperTableView * tableComics; + VolumeComicsModel * model; +}; + +#endif // SELECT_COMIC_H diff --git a/YACReaderLibrary/comic_vine/select_volume.cpp b/YACReaderLibrary/comic_vine/select_volume.cpp new file mode 100644 index 00000000..9650a7f7 --- /dev/null +++ b/YACReaderLibrary/comic_vine/select_volume.cpp @@ -0,0 +1,191 @@ +#include "select_volume.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "scraper_tableview.h" + +#include + +#include "volumes_model.h" +#include "comic_vine_client.h" +#include "scraper_scroll_label.h" + +#include "response_parser.h" +#include "scraper_results_paginator.h" + +SelectVolume::SelectVolume(QWidget *parent) + :ScraperSelector(parent),model(0) +{ + proxyModel = new QSortFilterProxyModel; + + QString labelStylesheet = "QLabel {color:white; font-size:12px;font-family:Arial;}"; + + QLabel * label = new QLabel(tr("Please, select the right series for your comic.")); + label->setStyleSheet(labelStylesheet); + + QVBoxLayout * l = new QVBoxLayout; + QWidget * leftWidget = new QWidget; + QVBoxLayout * left = new QVBoxLayout; + QVBoxLayout * right = new QVBoxLayout; + QHBoxLayout * content = new QHBoxLayout; + + right->setContentsMargins(0,0,0,0); + + //widgets + cover = new QLabel(); + cover->setScaledContents(true); + cover->setAlignment(Qt::AlignTop|Qt::AlignHCenter); + cover->setMinimumSize(168,168*5.0/3); + cover->setStyleSheet("QLabel {background-color: #2B2B2B; color:white; font-size:12px; font-family:Arial; }"); + detailLabel = new ScraperScrollLabel(this); + + tableVolumes = new ScraperTableView(this); + tableVolumes->setSortingEnabled(true); +#if QT_VERSION >= 0x050000 + tableVolumes->horizontalHeader()->setSectionsClickable(true); +#else + tableVolumes->horizontalHeader()->setClickable(true); +#endif + //tableVolumes->horizontalHeader()->setSortIndicatorShown(false); + connect(tableVolumes->horizontalHeader(),SIGNAL(sectionClicked(int)), tableVolumes, SLOT(sortByColumn(int))); + //connections + connect(tableVolumes,SIGNAL(clicked(QModelIndex)),this,SLOT(loadVolumeInfo(QModelIndex))); + + paginator->setCustomLabel(tr("volumes")); + + left->addWidget(cover); + left->addWidget(detailLabel,1); + left->addStretch(); + leftWidget->setMaximumWidth(180); + leftWidget->setLayout(left); + left->setContentsMargins(0,0,0,0); + leftWidget->setContentsMargins(0,0,0,0); + + right->addWidget(tableVolumes,0,Qt::AlignRight|Qt::AlignTop); + right->addWidget(paginator); + + content->addWidget(leftWidget); + content->addLayout(right); + + l->addSpacing(15); + l->addWidget(label); + l->addSpacing(5); + l->addLayout(content); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); +} + +void SelectVolume::load(const QString & json, const QString & searchString) +{ + VolumesModel * tempM = new VolumesModel(); + tempM->load(json); + //tableVolumes->setModel(tempM); + + proxyModel->setSourceModel( tempM ); + tableVolumes->setModel(proxyModel); + tableVolumes->sortByColumn(0,Qt::AscendingOrder); + tableVolumes->resizeColumnsToContents(); + + tableVolumes->setFixedSize(619,341); + + if(model != 0) + delete model; + + model = tempM; + + if(model->rowCount()>0) + { + tableVolumes->selectRow(0); + loadVolumeInfo(proxyModel->index(0,0)); + } + + tableVolumes->setColumnWidth(0,350); + + ScraperSelector::load(json,searchString); +} + +SelectVolume::~SelectVolume() {} + +void SelectVolume::loadVolumeInfo(const QModelIndex & omi) +{ + QModelIndex mi = proxyModel->mapToSource(omi); + QString coverURL = model->getCoverURL(mi); + QString id = model->getVolumeId(mi); + + QString loadingStyle = "%1"; + cover->setText(loadingStyle.arg(tr("loading cover"))); + detailLabel->setAltText(loadingStyle.arg(tr("loading description"))); + + ComicVineClient * comicVineClient = new ComicVineClient; + connect(comicVineClient,SIGNAL(seriesCover(const QByteArray &)),this,SLOT(setCover(const QByteArray &))); + connect(comicVineClient,SIGNAL(finished()),comicVineClient,SLOT(deleteLater())); + comicVineClient->getSeriesCover(coverURL); + + ComicVineClient * comicVineClient2 = new ComicVineClient; + connect(comicVineClient2,SIGNAL(seriesDetail(QString)),this,SLOT(setDescription(QString))); + connect(comicVineClient2,SIGNAL(finished()),comicVineClient2,SLOT(deleteLater())); + comicVineClient2->getSeriesDetail(id); +} + +void SelectVolume::setCover(const QByteArray & data) +{ + QPixmap p; + p.loadFromData(data); + int w = p.width(); + int h = p.height(); + + cover->setPixmap(p); + float aspectRatio = static_cast(w)/h; + + cover->setFixedSize(180,static_cast(180/aspectRatio)); + + cover->update(); +} + +void SelectVolume::setDescription(const QString & jsonDetail) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + jsonDetail + ")"); + + if (!sc.property("error").isValid() && sc.property("error").toString() != "OK") + { + qDebug("Error detected"); + } + else + { + + QScriptValue descriptionValues = sc.property("results").property("description"); + bool valid = !descriptionValues.isNull() && descriptionValues.isValid(); + detailLabel->setText(valid?descriptionValues.toString().replace("getVolumeId(proxyModel->mapToSource(tableVolumes->currentIndex())); +} + +int SelectVolume::getSelectedVolumeNumIssues() +{ + return model->getNumIssues(proxyModel->mapToSource(tableVolumes->currentIndex())); +} + +QString SelectVolume::getSelectedVolumePublisher() +{ + return model->getPublisher(proxyModel->mapToSource(tableVolumes->currentIndex())); +} + + diff --git a/YACReaderLibrary/comic_vine/select_volume.h b/YACReaderLibrary/comic_vine/select_volume.h new file mode 100644 index 00000000..060933c2 --- /dev/null +++ b/YACReaderLibrary/comic_vine/select_volume.h @@ -0,0 +1,39 @@ +#ifndef SELECT_VOLUME_H +#define SELECT_VOLUME_H + +#include "scraper_selector.h" + +class QLabel; +class VolumesModel; +class QModelIndex; +class QToolButton; +class QSortFilterProxyModel; + +class ScraperScrollLabel; +class ScraperTableView; + +class SelectVolume : public ScraperSelector +{ + Q_OBJECT +public: + SelectVolume(QWidget * parent = 0); + void load(const QString & json, const QString & searchString); + virtual ~SelectVolume(); + +public slots: + void loadVolumeInfo(const QModelIndex & mi); + void setCover(const QByteArray &); + void setDescription(const QString & jsonDetail); + QString getSelectedVolumeId(); + int getSelectedVolumeNumIssues(); + QString getSelectedVolumePublisher(); + +private: + QLabel * cover; + ScraperScrollLabel * detailLabel; + ScraperTableView * tableVolumes; + VolumesModel * model; + QSortFilterProxyModel * proxyModel; +}; + +#endif // SELECT_VOLUME_H diff --git a/YACReaderLibrary/comic_vine/series_question.cpp b/YACReaderLibrary/comic_vine/series_question.cpp new file mode 100644 index 00000000..1fb93cb8 --- /dev/null +++ b/YACReaderLibrary/comic_vine/series_question.cpp @@ -0,0 +1,46 @@ +#include "series_question.h" + +#include +#include +#include + + +SeriesQuestion::SeriesQuestion(QWidget * parent) + :QWidget(parent) +{ + QVBoxLayout * l = new QVBoxLayout; + + QLabel * questionLabel = new QLabel(tr("You are trying to get information for various comics at once, are they part of the same series?")); + questionLabel->setStyleSheet("QLabel {color:white; font-size:12px;font-family:Arial;}"); + yes = new QRadioButton(tr("yes")); + no = new QRadioButton(tr("no")); + + QString rbStyle = "QRadioButton {margin-left:27px; margin-top:5px; color:white;font-size:12px;font-family:Arial;}" + "QRadioButton::indicator {width:11px;height:11px;}" + "QRadioButton::indicator::unchecked {image : url(:/images/comic_vine/radioUnchecked.png);}" + "QRadioButton::indicator::checked {image : url(:/images/comic_vine/radioChecked.png);}"; + yes->setStyleSheet(rbStyle); + no->setStyleSheet(rbStyle); + + yes->setChecked(true); + + l->addSpacing(35); + l->addWidget(questionLabel); + l->addWidget(yes); + l->addWidget(no); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); +} + +bool SeriesQuestion::getYes() +{ + return yes->isChecked(); +} + +void SeriesQuestion::setYes(bool y) +{ + yes->setChecked(y); +} diff --git a/YACReaderLibrary/comic_vine/series_question.h b/YACReaderLibrary/comic_vine/series_question.h new file mode 100644 index 00000000..c6620ecd --- /dev/null +++ b/YACReaderLibrary/comic_vine/series_question.h @@ -0,0 +1,23 @@ +#ifndef SERIES_QUESTION_H +#define SERIES_QUESTION_H + +#include + +class QRadioButton; + +class SeriesQuestion : public QWidget +{ + Q_OBJECT + +public: + SeriesQuestion(QWidget * parent = 0); + bool getYes(); + void setYes(bool yes = true); + +private: + QRadioButton * yes; + QRadioButton * no; +}; + + +#endif // SERIES_QUESTION_H diff --git a/YACReaderLibrary/comic_vine/sort_volume_comics.cpp b/YACReaderLibrary/comic_vine/sort_volume_comics.cpp new file mode 100644 index 00000000..6d7441f2 --- /dev/null +++ b/YACReaderLibrary/comic_vine/sort_volume_comics.cpp @@ -0,0 +1,224 @@ +#include "sort_volume_comics.h" + +#include +#include +#include +#include +#include + +#include "scraper_tableview.h" +#include "local_comic_list_model.h" +#include "volume_comics_model.h" + +SortVolumeComics::SortVolumeComics(QWidget *parent) : + ScraperSelector(parent) +{ + QString labelStylesheet = "QLabel {color:white; font-size:12px;font-family:Arial;}"; + + QLabel * label = new QLabel(tr("Please, sort the list of comics on the left until it matches the comics' information.")); + label->setStyleSheet(labelStylesheet); + + QLabel * sortLabel = new QLabel(tr("sort comics to match comic information")); + sortLabel->setStyleSheet(labelStylesheet); + + moveUpButtonCL = new ScrapperToolButton(ScrapperToolButton::LEFT); + moveUpButtonCL->setIcon(QIcon(":/images/comic_vine/rowUp.png")); + moveUpButtonCL->setAutoRepeat(true); + moveDownButtonCL = new ScrapperToolButton(ScrapperToolButton::RIGHT); + moveDownButtonCL->setIcon(QIcon(":/images/comic_vine/rowDown.png")); + moveDownButtonCL->setAutoRepeat(true); + //moveUpButtonIL = new ScrapperToolButton(ScrapperToolButton::LEFT); + //moveUpButtonIL->setIcon(QIcon(":/images/comic_vine/rowUp.png")); + //moveDownButtonIL = new ScrapperToolButton(ScrapperToolButton::RIGHT); + //moveDownButtonIL->setIcon(QIcon(":/images/comic_vine/rowDown.png")); + + connect(moveUpButtonCL,SIGNAL(clicked()),this,SLOT(moveUpCL())); + connect(moveDownButtonCL,SIGNAL(clicked()),this,SLOT(moveDownCL())); + //connect(moveUpButtonIL,SIGNAL(clicked()),this,SLOT(moveUpIL())); + //connect(moveUpButtonIL,SIGNAL(clicked()),this,SLOT(moveDownIL())); + + QVBoxLayout * l = new QVBoxLayout; + QHBoxLayout * content = new QHBoxLayout; + QHBoxLayout * sortButtonsLayout = new QHBoxLayout; + + tableFiles = new ScraperTableView(); + tableVolumeComics = new ScraperTableView(); + + tableFiles->setSelectionBehavior(QAbstractItemView::SelectRows); + tableFiles->setSelectionMode(QAbstractItemView::ContiguousSelection); + + tableFiles->setFixedSize(407,341); + tableVolumeComics->setFixedSize(407,341); + content->addWidget(tableFiles,0,Qt::AlignLeft|Qt::AlignTop); + content->addWidget(tableVolumeComics,0,Qt::AlignRight|Qt::AlignTop); + //content->addWidget(tableVolumes,0,Qt::AlignRight|Qt::AlignTop); + + connect(tableVolumeComics->verticalScrollBar(), SIGNAL(valueChanged(int)), this, SLOT(synchronizeScroll(int))); + connect(tableFiles->verticalScrollBar(), SIGNAL(valueChanged(int)), this, SLOT(synchronizeScroll(int))); + + //connect(tableVolumeComics, SIGNAL(pressed(QModelIndex)), tableFiles, SLOT(setCurrentIndex(QModelIndex))); + //connect(tableFiles, SIGNAL(pressed(QModelIndex)), tableVolumeComics, SLOT(setCurrentIndex(QModelIndex))); + + paginator->setCustomLabel(tr("issues")); + paginator->setMinimumWidth(422); + + sortButtonsLayout->addWidget(moveUpButtonCL); + sortButtonsLayout->addWidget(ScrapperToolButton::getSeparator()); + sortButtonsLayout->addWidget(moveDownButtonCL); + sortButtonsLayout->addSpacing(10); + //sortButtonsLayout->addStretch(); + sortButtonsLayout->addWidget(sortLabel); + sortButtonsLayout->addStretch(); + sortButtonsLayout->addWidget(paginator); + //sortButtonsLayout->addStretch(); + //sortButtonsLayout->addWidget(moveUpButtonIL); + //sortButtonsLayout->addWidget(ScrapperToolButton::getSeparator()); + //sortButtonsLayout->addWidget(moveDownButtonIL); + sortButtonsLayout->setSpacing(0); + + l->addSpacing(15); + l->addWidget(label); + l->addSpacing(5); + l->addLayout(content); + l->addLayout(sortButtonsLayout); + l->addStretch(); + + l->setContentsMargins(0,0,0,0); + setLayout(l); + setContentsMargins(0,0,0,0); + + //rows actions + QAction * removeItemFromList = new QAction(tr("remove selected comics"),this); + QAction * restoreAllItems = new QAction(tr("restore all removed comics"),this); + QAction * restoreItems = new QAction(tr("restore removed comics"),this); + + tableFiles->setContextMenuPolicy(Qt::ActionsContextMenu); + tableFiles->addAction(removeItemFromList); + tableFiles->addAction(restoreAllItems); + //tableFiles->addAction(restoreItems); + + connect(removeItemFromList,SIGNAL(triggered()),this,SLOT(removeSelectedComics())); + connect(restoreAllItems,SIGNAL(triggered()),this,SLOT(restoreAllComics())); + connect(restoreItems,SIGNAL(triggered()),this,SLOT(showRemovedComicsSelector())); +} + +void SortVolumeComics::setData(QList & comics, const QString &json, const QString &vID) +{ + //set up models + localComicsModel = new LocalComicListModel; + localComicsModel->load(comics); + + volumeComicsModel = new VolumeComicsModel; + volumeComicsModel->load(json); + + int numLocalComics = localComicsModel->rowCount(); + int numVolumeComics = volumeComicsModel->rowCount(); + + if(numLocalComics > numVolumeComics) + volumeComicsModel->addExtraRows(numLocalComics - numVolumeComics); + if(numLocalComics < numVolumeComics) + localComicsModel->addExtraRows(numVolumeComics - numLocalComics); + + tableFiles->setModel(localComicsModel); + tableVolumeComics->setModel(volumeComicsModel); + + tableVolumeComics->resizeColumnToContents(0); + + ScraperSelector::load(json,vID); +} + +void SortVolumeComics::synchronizeScroll(int pos) +{ + void * senderObject = sender(); + + if(senderObject == 0) //invalid call + return; + + QScrollBar * tableVolumeComicsScrollBar = tableVolumeComics->verticalScrollBar(); + QScrollBar * tableFilesScrollBar = tableFiles->verticalScrollBar(); + + if(senderObject == tableVolumeComicsScrollBar) + { + disconnect(tableFilesScrollBar,SIGNAL(valueChanged(int)),this,0); + tableFilesScrollBar->setValue(pos); + connect(tableFilesScrollBar, SIGNAL(valueChanged(int)), this, SLOT(synchronizeScroll(int))); + } + else + { + disconnect(tableVolumeComicsScrollBar,SIGNAL(valueChanged(int)),this,0); + tableVolumeComicsScrollBar->setValue(pos); + connect(tableVolumeComicsScrollBar, SIGNAL(valueChanged(int)), this, SLOT(synchronizeScroll(int))); + } +} + +void SortVolumeComics::moveUpCL() +{ + QList selection = tableFiles->selectionModel()->selectedIndexes(); + + if(selection.count() == 0) + return; + + localComicsModel->moveSelectionUp(selection); + + selection = tableFiles->selectionModel()->selectedIndexes(); + tableFiles->scrollTo(selection.first()); +} + +void SortVolumeComics::moveDownCL() +{ + QList selection = tableFiles->selectionModel()->selectedIndexes(); + + if(selection.count() > 0) + localComicsModel->moveSelectionDown(selection); + + selection = tableFiles->selectionModel()->selectedIndexes(); + tableFiles->scrollTo(selection.last()); +} + +void SortVolumeComics::moveUpIL() +{ + +} + +void SortVolumeComics::moveDownIL() +{ + +} + +void SortVolumeComics::removeSelectedComics() +{ + QList selection = tableFiles->selectionModel()->selectedIndexes(); + + localComicsModel->removeComics(selection); +} + +void SortVolumeComics::restoreAllComics() +{ + localComicsModel->restoreAll(); +} + +void SortVolumeComics::showRemovedComicsSelector() +{ + +} + +QList > SortVolumeComics::getMatchingInfo() +{ + QList comicList = localComicsModel->getData(); + QList > l; + + int index = 0; + + QString id; + foreach(ComicDB c, comicList) + { + id = volumeComicsModel->getComicId(index); + if(!c.getFileName().isEmpty() && !id.isEmpty()) //there is a valid comic, and valid comic ID + { + l.push_back(QPair(c,id)); + } + index++; + } + + return l; +} diff --git a/YACReaderLibrary/comic_vine/sort_volume_comics.h b/YACReaderLibrary/comic_vine/sort_volume_comics.h new file mode 100644 index 00000000..92955f90 --- /dev/null +++ b/YACReaderLibrary/comic_vine/sort_volume_comics.h @@ -0,0 +1,99 @@ +#ifndef SORT_VOLUME_COMICS_H +#define SORT_VOLUME_COMICS_H + +#include "scraper_selector.h" + +#include +#include +#include + +#include "comic_db.h" + +class ScraperTableView; +class LocalComicListModel; +class VolumeComicsModel; + +class ScrapperToolButton : public QPushButton +{ + Q_OBJECT +public: + enum Appearance { + DEFAULT, + LEFT, + RIGHT + }; + + ScrapperToolButton(ScrapperToolButton::Appearance appearance = DEFAULT, QWidget * parent=0):QPushButton(parent),appearance(appearance) { + setStyleSheet("QPushButton {border: none; background: #2e2e2e; color:white; border-radius:2px;}" + "QPushButton::pressed {border: none; background: #282828; color:white; border-radius:2px;}"); + setFixedSize(18,17); + } + static QWidget * getSeparator(){QWidget * w = new QWidget; w->setFixedWidth(1); w->setStyleSheet("QWidget {background:#282828;}"); return w;} + void setAppearance(ScrapperToolButton::Appearance appearance){this->appearance = appearance;} + virtual ~ScrapperToolButton() {} + + + +protected: + void paintEvent(QPaintEvent * e) + { + QPainter p(this); + + switch (appearance) { + case LEFT: + p.fillRect(16,0,2,18,QColor("#2E2E2E")); + break; + case RIGHT: + p.fillRect(0,0,2,18,QColor("#2E2E2E")); + break; + default: + break; + } + + QPushButton::paintEvent(e); + } + +private: + Appearance appearance; +}; + + +class SortVolumeComics : public ScraperSelector +{ + Q_OBJECT +public: + explicit SortVolumeComics(QWidget *parent = 0); + +signals: + +public slots: + void setData(QList & comics, const QString & json, const QString & vID); + QList > getMatchingInfo(); + +protected slots: + void synchronizeScroll(int pos); + void moveUpCL(); + void moveDownCL(); + void moveUpIL(); + void moveDownIL(); + + void removeSelectedComics(); + void restoreAllComics(); + void showRemovedComicsSelector(); + + +private: + ScraperTableView * tableFiles; + ScraperTableView * tableVolumeComics; + + LocalComicListModel * localComicsModel; + VolumeComicsModel * volumeComicsModel; + + ScrapperToolButton * moveUpButtonCL; + ScrapperToolButton * moveDownButtonCL; + ScrapperToolButton * moveUpButtonIL; + ScrapperToolButton * moveDownButtonIL; + +}; + +#endif // SORT_VOLUME_COMICS_H diff --git a/YACReaderLibrary/comic_vine/title_header.cpp b/YACReaderLibrary/comic_vine/title_header.cpp new file mode 100644 index 00000000..cebc0d6f --- /dev/null +++ b/YACReaderLibrary/comic_vine/title_header.cpp @@ -0,0 +1,53 @@ +#include "title_header.h" + +#include +#include +#include + +TitleHeader::TitleHeader(QWidget * parent ) + :QWidget(parent) +{ + mainTitleLabel = new QLabel(); + subTitleLabel = new QLabel(); + + mainTitleLabel->setStyleSheet("QLabel {color:white; font-size:18px;font-family:Arial;}"); + subTitleLabel->setStyleSheet("QLabel {color:white; font-size:12px;font-family:Arial;}"); + + QHBoxLayout * titleLayout = new QHBoxLayout; + QVBoxLayout * titleLabelsLayout = new QVBoxLayout; + + titleLabelsLayout->addWidget(mainTitleLabel); + titleLabelsLayout->addWidget(subTitleLabel); + titleLabelsLayout->setSpacing(0); + + titleLayout->addLayout(titleLabelsLayout); + titleLayout->setContentsMargins(0,0,0,0); + + setLayout(titleLayout); + + setContentsMargins(0,0,0,0); + + setTitle(tr("SEARCH")); +} + +void TitleHeader::setTitle(const QString & title) +{ + mainTitleLabel->setText(title); +} + +void TitleHeader::setSubTitle(const QString & title) +{ + subTitleLabel->setText(title); +} + +void TitleHeader::showButtons(bool show) +{ + if(show) + { + + } + else + { + + } +} diff --git a/YACReaderLibrary/comic_vine/title_header.h b/YACReaderLibrary/comic_vine/title_header.h new file mode 100644 index 00000000..a4e62e98 --- /dev/null +++ b/YACReaderLibrary/comic_vine/title_header.h @@ -0,0 +1,22 @@ +#ifndef TITLE_HEADER_H +#define TITLE_HEADER_H + +#include + +class QLabel; + +class TitleHeader : public QWidget +{ + Q_OBJECT +public: + TitleHeader(QWidget * parent = 0); +public slots: + void setTitle(const QString & title); + void setSubTitle(const QString & title); + void showButtons(bool show); +private: + QLabel * mainTitleLabel; + QLabel * subTitleLabel; +}; + +#endif // TITLE_HEADER_H diff --git a/YACReaderLibrary/comics_remover.cpp b/YACReaderLibrary/comics_remover.cpp new file mode 100644 index 00000000..d5d5676d --- /dev/null +++ b/YACReaderLibrary/comics_remover.cpp @@ -0,0 +1,29 @@ +#include "comics_remover.h" + +#include + +ComicsRemover::ComicsRemover(QModelIndexList & il, QList & ps, QObject *parent) : + QThread(parent),indexList(il), paths(ps) +{ +} + +void ComicsRemover::run() +{ + QString currentComicPath; + QListIterator i(indexList); + QListIterator i2(paths); + i.toBack(); + i2.toBack(); + + while (i.hasPrevious() && i2.hasPrevious()) + { + QModelIndex mi = i.previous(); + currentComicPath = i2.previous(); + if(QFile::remove(currentComicPath)) + emit remove(mi.row()); + else + emit removeError(); + } + + emit finished(); +} diff --git a/YACReaderLibrary/comics_remover.h b/YACReaderLibrary/comics_remover.h new file mode 100644 index 00000000..5a77a84b --- /dev/null +++ b/YACReaderLibrary/comics_remover.h @@ -0,0 +1,28 @@ +#ifndef COMICS_REMOVER_H +#define COMICS_REMOVER_H + +#include + +#include +#include + +class ComicsRemover : public QThread +{ + Q_OBJECT +public: + explicit ComicsRemover(QModelIndexList & indexList, QList & paths, QObject *parent = 0); + +signals: + void remove(int); + void removeError(); + void finished(); + +private: + void run(); + +private: + QModelIndexList indexList; + QList paths; +}; + +#endif // COMICS_REMOVER_H diff --git a/YACReaderLibrary/comics_view.cpp b/YACReaderLibrary/comics_view.cpp new file mode 100644 index 00000000..5c5e7725 --- /dev/null +++ b/YACReaderLibrary/comics_view.cpp @@ -0,0 +1,11 @@ +#include "comics_view.h" + +ComicsView::ComicsView(QWidget *parent) : + QWidget(parent),model(NULL) +{ +} + +void ComicsView::setModel(TableModel *m) +{ + model = m; +} diff --git a/YACReaderLibrary/comics_view.h b/YACReaderLibrary/comics_view.h new file mode 100644 index 00000000..5ae8e485 --- /dev/null +++ b/YACReaderLibrary/comics_view.h @@ -0,0 +1,47 @@ +#ifndef COMICS_VIEW_H +#define COMICS_VIEW_H + +#include + +#include "tablemodel.h" +#include +#include +#include +#include + +class YACReaderTableView; +class QSplitter; +class ComicFlowWidget; +class QToolBar; +class TableModel; +class ComicsView : public QWidget +{ + Q_OBJECT +public: + explicit ComicsView(QWidget *parent = 0); + virtual void setToolBar(QToolBar * toolBar) = 0; + virtual void setModel(TableModel *model); + virtual void setCurrentIndex(const QModelIndex &index) = 0; + virtual QModelIndex currentIndex() = 0; + virtual QItemSelectionModel * selectionModel() = 0; + virtual void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ) = 0; + virtual void toFullScreen() = 0; + virtual void toNormal() = 0; + virtual void updateConfig(QSettings * settings) = 0; + //Actions for tableviews + virtual void setItemActions(const QList & actions) = 0; + //actions for visual-oriented views + virtual void setViewActions(const QList & actions) = 0; + +signals: + void selected(unsigned int); + void comicRated(int,QModelIndex); +public slots: + virtual void setShowMarks(bool show) = 0; + virtual void selectAll() = 0; +protected: + TableModel * model; + +}; + +#endif // COMICS_VIEW_H diff --git a/YACReaderLibrary/comics_view_transition.cpp b/YACReaderLibrary/comics_view_transition.cpp new file mode 100644 index 00000000..3e2fdd12 --- /dev/null +++ b/YACReaderLibrary/comics_view_transition.cpp @@ -0,0 +1,74 @@ +#include "comics_view_transition.h" + +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +ComicsViewTransition::ComicsViewTransition(QWidget *parent) : + QWidget(parent),movie(0) +{ + QVBoxLayout * layout = new QVBoxLayout; + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + settings->beginGroup("libraryConfig"); + + movieLabel = new QLabel("Placeholder"); + movieLabel->setAlignment(Qt::AlignCenter); + QLabel * textLabel = new QLabel("Switching comics view"); + textLabel->setAlignment(Qt::AlignCenter); + textLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); + //movieLabel->setFixedSize(450,350); + + layout->addSpacing(100); + layout->addWidget(movieLabel); + layout->addSpacing(20); + layout->addWidget(textLabel); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet("QWidget {background:#2A2A2A}"); + + //QSizePolicy sp(); + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + //movieLabel->setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + setLayout(layout); +} + +QSize ComicsViewTransition::sizeHint() +{ + return QSize(450,350); +} + +void ComicsViewTransition::startMovie() +{ + if(movie) + delete movie; + + if(settings->value(COMICS_VIEW_STATUS) == YACReader::Flow) + movie = new QMovie(":/images/flow_to_grid.gif"); + else + movie = new QMovie(":/images/grid_to_flow.gif"); + + connect(movie,SIGNAL(finished()),this,SIGNAL(transitionFinished())); + //connect(movie,SIGNAL(finished()),movie,SLOT(deleteLater()); + movie->setSpeed(200); + movie->jumpToFrame(0); + movieLabel->setMovie(movie); + + QTimer::singleShot(100,movie,SLOT(start())); +} + +void ComicsViewTransition::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor("#2A2A2A")); +} diff --git a/YACReaderLibrary/comics_view_transition.h b/YACReaderLibrary/comics_view_transition.h new file mode 100644 index 00000000..ed8c55ba --- /dev/null +++ b/YACReaderLibrary/comics_view_transition.h @@ -0,0 +1,31 @@ +#ifndef COMICS_VIEW_TRANSITION_H +#define COMICS_VIEW_TRANSITION_H + +#include + +class QMovie; +class QSettings; +class QLabel; + +class ComicsViewTransition : public QWidget +{ + Q_OBJECT +public: + explicit ComicsViewTransition(QWidget *parent = 0); + QSize sizeHint(); + +signals: + void transitionFinished(); + +public slots: + void startMovie(); + +protected: + QMovie * movie; + QSettings * settings; + QLabel * movieLabel; + + void paintEvent(QPaintEvent *); +}; + +#endif // COMICS_VIEW_TRANSITION_H diff --git a/YACReaderLibrary/create_library_dialog.cpp b/YACReaderLibrary/create_library_dialog.cpp new file mode 100644 index 00000000..a86ddbfb --- /dev/null +++ b/YACReaderLibrary/create_library_dialog.cpp @@ -0,0 +1,206 @@ +#include "create_library_dialog.h" + +#include +#include +#include +#include +#include + +CreateLibraryDialog::CreateLibraryDialog(QWidget * parent) +:QDialog(parent) +{ + setupUI(); +} + +void CreateLibraryDialog::setupUI() +{ + textLabel = new QLabel(tr("Comics folder : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + connect(path,SIGNAL(textChanged(QString)),this,SLOT(pathSetted(QString))); + + nameLabel = new QLabel(tr("Library Name : ")); + nameEdit = new QLineEdit; + nameLabel->setBuddy(nameEdit); + connect(nameEdit,SIGNAL(textChanged(QString)),this,SLOT(nameSetted(QString))); + + accept = new QPushButton(tr("Create")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(create())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SIGNAL(cancelCreate())); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + QGridLayout * content = new QGridLayout; + + //QHBoxLayout *nameLayout = new QHBoxLayout; + + content->addWidget(nameLabel,0,0); + content->addWidget(nameEdit,0,1); + + //QHBoxLayout *libraryLayout = new QHBoxLayout; + + content->addWidget(textLabel,1,0); + content->addWidget(path,1,1); + content->addWidget(find,1,2); + content->setColumnMinimumWidth(2,0); //TODO + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addWidget(message = new QLabel(tr("Create a library could take several minutes. You can stop the process and update the library later for completing the task."))); + message->setWordWrap(true); + //message->hide(); + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(content); + + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/new.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Create new library")); +} + +void CreateLibraryDialog::show(const YACReaderLibraries & libs) +{ + libraries = libs; + QDialog::show(); +} + +void CreateLibraryDialog::create() +{ + + QFileInfo f(path->text()); + if(f.exists() && f.isDir() && f.isWritable()) + { + if(!libraries.contains(nameEdit->text())) + { + emit(createLibrary(QDir::cleanPath(path->text()),QDir::cleanPath(path->text())+"/.yacreaderlibrary",nameEdit->text())); + close(); + } + else + emit(libraryExists(nameEdit->text())); + } + else + QMessageBox::critical(NULL,tr("Path not found"),tr("The selected path does not exist or is not a valid path. Be sure that you have write access to this folder")); +} + +void CreateLibraryDialog::nameSetted(const QString & text) +{ + if(!text.isEmpty()) + { + if(!path->text().isEmpty()) + { + QFileInfo fi(path->text()); + if(fi.isDir()) + accept->setEnabled(true); + else + accept->setEnabled(false); + } + } + else + accept->setEnabled(false); +} + +void CreateLibraryDialog::pathSetted(const QString & text) +{ + QFileInfo fi(text); + if(fi.isDir()) + { + if(!nameEdit->text().isEmpty()) + accept->setEnabled(true); + } + else + accept->setEnabled(false); +} + +void CreateLibraryDialog::findPath() +{ + QString s = QFileDialog::getExistingDirectory(0,"Comics directory","."); + if(!s.isEmpty()) + { + path->setText(s); + if(!nameEdit->text().isEmpty()) + accept->setEnabled(true); + } + else + accept->setEnabled(false); +} + +void CreateLibraryDialog::close() +{ + path->clear(); + nameEdit->clear(); + accept->setEnabled(false); + QDialog::close(); +} + +void CreateLibraryDialog::setDataAndStart(QString name, QString path) +{ + this->path->setText(path); + this->nameEdit->setText(name); + QDialog::show(); + create(); +} +//----------------------------------------------------------------------------- +// UpdateLibraryDialog +//----------------------------------------------------------------------------- +UpdateLibraryDialog::UpdateLibraryDialog(QWidget * parent) +:QDialog(parent) +{ + QVBoxLayout * mainLayout = new QVBoxLayout; + mainLayout->addWidget(message = new QLabel(tr("Updating...."))); + mainLayout->addWidget(currentFileLabel = new QLabel("\n\n\n\n")); + currentFileLabel->setWordWrap(true); + + QHBoxLayout * bottom = new QHBoxLayout; + bottom->addStretch(); + bottom->addWidget(cancel = new QPushButton(tr("Cancel"))); + + connect(cancel,SIGNAL(clicked()),this,SIGNAL(cancelUpdate())); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + mainLayout->addStretch(); + + mainLayout->addLayout(bottom); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/updateLibrary.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Update library")); +} + +void UpdateLibraryDialog::showCurrentFile(QString file) +{ + currentFileLabel->setText(file); + currentFileLabel->update(); + this->update(); +} + +void UpdateLibraryDialog::close() +{ + currentFileLabel->setText(""); + this->adjustSize(); + QDialog::close(); +} diff --git a/YACReaderLibrary/create_library_dialog.h b/YACReaderLibrary/create_library_dialog.h new file mode 100644 index 00000000..55cbcb90 --- /dev/null +++ b/YACReaderLibrary/create_library_dialog.h @@ -0,0 +1,61 @@ +#ifndef __CREATE_LIBRARY_DIALOG_H +#define __CREATE_LIBRARY_DIALOG_H + +#include "yacreader_libraries.h" + +#include +#include +#include +#include +#include +#include + + class CreateLibraryDialog : public QDialog + { + Q_OBJECT + public: + CreateLibraryDialog(QWidget * parent = 0); + private: + QLabel * nameLabel; + QLabel * textLabel; + QLabel * message; + QProgressBar *progressBar; + QLineEdit * path; + QLineEdit * nameEdit; + QPushButton * find; + QPushButton * accept; + QPushButton * cancel; + YACReaderLibraries libraries; + void setupUI(); + public slots: + void create(); + void findPath(); + void close(); + void setDataAndStart(QString name, QString paht); + void nameSetted(const QString & text); + void pathSetted(const QString & text); + void show(const YACReaderLibraries &libraries); + signals: + void createLibrary(QString source, QString target, QString name); + void cancelCreate(); + void libraryExists(const QString & name); + }; + + class UpdateLibraryDialog : public QDialog + { + Q_OBJECT + public: + UpdateLibraryDialog(QWidget * parent = 0); + private: + QLabel * message; + QLabel * currentFileLabel; + QProgressBar *progressBar; + QPushButton * cancel; + public slots: + void showCurrentFile(QString file); + void close(); + signals: + void cancelUpdate(); + }; + +#endif diff --git a/YACReaderLibrary/db/data_base_management.cpp b/YACReaderLibrary/db/data_base_management.cpp new file mode 100644 index 00000000..1b0ce7c9 --- /dev/null +++ b/YACReaderLibrary/db/data_base_management.cpp @@ -0,0 +1,686 @@ +#include "data_base_management.h" + +#include +#include "library_creator.h" +#include "check_new_version.h" + +static QString fields = "title ," + + "coverPage," + "numPages," + + "number," + "isBis," + "count," + + "volume," + "storyArc," + "arcNumber," + "arcCount," + + "genere," + + "writer," + "penciller," + "inker," + "colorist," + "letterer," + "coverArtist," + + "date," + "publisher," + "format," + "color," + "ageRating," + + "synopsis," + "characters," + "notes," + + "comicVineID," + + "hash" + ; + +DataBaseManagement::DataBaseManagement() + :QObject(),dataBasesList() +{ + +} + +/*TreeModel * DataBaseManagement::newTreeModel(QString path) +{ + //la consulta se ejecuta... + QSqlQuery selectQuery(loadDatabase(path)); + selectQuery.setForwardOnly(true); + selectQuery.exec("select * from folder order by parentId,name"); + //selectQuery.finish(); + return new TreeModel(selectQuery); +}*/ + +QSqlDatabase DataBaseManagement::createDatabase(QString name, QString path) +{ + return createDatabase(QDir::cleanPath(path) + "/" + name + ".ydb"); +} + +QSqlDatabase DataBaseManagement::createDatabase(QString dest) +{ + QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE",dest); + db.setDatabaseName(dest); + if (!db.open()) + qDebug() << db.lastError(); + else { + qDebug() << db.tables(); + } + + { + QSqlQuery pragma("PRAGMA foreign_keys = ON",db); + //pragma.finish(); + DataBaseManagement::createTables(db); + + QSqlQuery query("INSERT INTO folder (parentId, name, path) " + "VALUES (1,'root', '/')",db); + } + //query.finish(); + //db.close(); + + return db; +} + +QSqlDatabase DataBaseManagement::loadDatabase(QString path) +{ + //TODO check path + QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE",path); + db.setDatabaseName(path+"/library.ydb"); + if (!db.open()) { + //se devuelve una base de datos vacía e inválida + + return QSqlDatabase(); + } + QSqlQuery pragma("PRAGMA foreign_keys = ON",db); + //pragma.finish(); + //devuelve la base de datos + return db; +} + +QSqlDatabase DataBaseManagement::loadDatabaseFromFile(QString filePath) +{ + //TODO check path + QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE",filePath); + db.setDatabaseName(filePath); + if (!db.open()) { + //se devuelve una base de datos vacía e inválida + + return QSqlDatabase(); + } + { + QSqlQuery pragma("PRAGMA foreign_keys = ON",db); + } + //pragma.finish(); + //devuelve la base de datos + return db; +} + +bool DataBaseManagement::createTables(QSqlDatabase & database) +{ + bool success = true; + + //FOLDER (representa una carpeta en disco) + { + QSqlQuery queryFolder(database); + queryFolder.prepare("CREATE TABLE folder (" + "id INTEGER PRIMARY KEY," + "parentId INTEGER NOT NULL," + "name TEXT NOT NULL," + "path TEXT NOT NULL," + //new 7.1 fields + "finished BOOLEAN DEFAULT 0," //reading + "completed BOOLEAN DEFAULT 1," //collecting + //-- + "FOREIGN KEY(parentId) REFERENCES folder(id) ON DELETE CASCADE)"); + success = success && queryFolder.exec(); + //queryFolder.finish(); + //COMIC INFO (representa la información de un cómic, cada cómic tendrá un idéntificador único formado por un hash sha1'de los primeros 512kb' + su tamaño en bytes) + QSqlQuery queryComicInfo(database); + queryComicInfo.prepare("CREATE TABLE comic_info (" + "id INTEGER PRIMARY KEY," + "title TEXT," + + "coverPage INTEGER DEFAULT 1," + "numPages INTEGER," + + "number INTEGER," + "isBis BOOLEAN," + "count INTEGER," + + "volume TEXT," + "storyArc TEXT," + "arcNumber INTEGER," + "arcCount INTEGER," + + "genere TEXT," + + "writer TEXT," + "penciller TEXT," + "inker TEXT," + "colorist TEXT," + "letterer TEXT," + "coverArtist TEXT," + + "date TEXT," //dd/mm/yyyy --> se mostrará en 3 campos diferentes + "publisher TEXT," + "format TEXT," + "color BOOLEAN," + "ageRating BOOLEAN," + + "synopsis TEXT," + "characters TEXT," + "notes TEXT," + + "hash TEXT UNIQUE NOT NULL," + "edited BOOLEAN DEFAULT 0," + "read BOOLEAN DEFAULT 0," +//new 7.0 fields + + "hasBeenOpened BOOLEAN DEFAULT 0," + "rating INTEGER DEFAULT 0," + "currentPage INTEGER DEFAULT 1, " + "bookmark1 INTEGER DEFAULT -1, " + "bookmark2 INTEGER DEFAULT -1, " + "bookmark3 INTEGER DEFAULT -1, " + "brightness INTEGER DEFAULT -1, " + "contrast INTEGER DEFAULT -1, " + "gamma INTEGER DEFAULT -1, " +//new 7.1 fields + "comicVineID TEXT" + + ")"); + success = success && queryComicInfo.exec(); + //queryComicInfo.finish(); + + //COMIC (representa un cómic en disco, contiene el nombre de fichero) + QSqlQuery queryComic(database); + queryComic.prepare("CREATE TABLE comic (id INTEGER PRIMARY KEY, parentId INTEGER NOT NULL, comicInfoId INTEGER NOT NULL, fileName TEXT NOT NULL, path TEXT, FOREIGN KEY(parentId) REFERENCES folder(id) ON DELETE CASCADE, FOREIGN KEY(comicInfoId) REFERENCES comic_info(id))"); + success = success && queryComic.exec(); + //queryComic.finish(); + //DB INFO + QSqlQuery queryDBInfo(database); + queryDBInfo.prepare("CREATE TABLE db_info (version TEXT NOT NULL)"); + success = success && queryDBInfo.exec(); + //queryDBInfo.finish(); + + QSqlQuery query("INSERT INTO db_info (version) " + "VALUES ('" VERSION "')",database); + //query.finish(); + } + + return success; +} +#include +void DataBaseManagement::exportComicsInfo(QString source, QString dest) +{ + //QSqlDatabase sourceDB = loadDatabase(source); + QSqlDatabase destDB = loadDatabaseFromFile(dest); + //sourceDB.open(); + { + QSqlQuery attach(destDB); + attach.prepare("ATTACH DATABASE '"+QDir().toNativeSeparators(dest) +"' AS dest;"); + //attach.bindValue(":dest",QDir().toNativeSeparators(dest)); + attach.exec(); + //attach.finish(); + + QSqlQuery attach2(destDB); + attach2.prepare("ATTACH DATABASE '"+QDir().toNativeSeparators(source) +"' AS source;"); + attach2.exec(); + //attach2.finish(); + + //sourceDB.close(); + QSqlQuery queryDBInfo(destDB); + queryDBInfo.prepare("CREATE TABLE dest.db_info (version TEXT NOT NULL)"); + queryDBInfo.exec(); + //queryDBInfo.finish(); + + /*QSqlQuery queryComicsInfo(sourceDB); + queryComicsInfo.prepare("CREATE TABLE dest.comic_info (id INTEGER PRIMARY KEY, hash TEXT NOT NULL, edited BOOLEAN DEFAULT FALSE, title TEXT, read BOOLEAN)"); + queryComicsInfo.exec();*/ + + QSqlQuery query("INSERT INTO dest.db_info (version) " + "VALUES ('" VERSION "')",destDB); + //query.finish(); + + QSqlQuery exportData(destDB); + exportData.prepare("create table dest.comic_info as select " + fields + + " from source.comic_info where source.comic_info.edited = 1"); + exportData.exec(); + //exportData.finish(); + } + + //sourceDB.close(); + destDB.close(); + QSqlDatabase::removeDatabase(dest); + +} + +bool DataBaseManagement::importComicsInfo(QString source, QString dest) +{ + QString error; + QString driver; + QStringList hashes; + + bool b = false; + + QSqlDatabase sourceDB = loadDatabaseFromFile(source); + QSqlDatabase destDB = loadDatabaseFromFile(dest); + + { + QSqlQuery pragma("PRAGMA synchronous=OFF",destDB); + + + QSqlQuery newInfo(sourceDB); + newInfo.prepare("SELECT * FROM comic_info"); + newInfo.exec(); + destDB.transaction(); + int cp; + while (newInfo.next()) //cada tupla deberá ser insertada o actualizada + { + QSqlQuery update(destDB); + update.prepare("UPDATE comic_info SET " + "title = :title," + + "coverPage = :coverPage," + "numPages = :numPages," + + "number = :number," + "isBis = :isBis," + "count = :count," + + "volume = :volume," + "storyArc = :storyArc," + "arcNumber = :arcNumber," + "arcCount = :arcCount," + + "genere = :genere," + + "writer = :writer," + "penciller = :penciller," + "inker = :inker," + "colorist = :colorist," + "letterer = :letterer," + "coverArtist = :coverArtist," + + "date = :date," + "publisher = :publisher," + "format = :format," + "color = :color," + "ageRating = :ageRating," + + "synopsis = :synopsis," + "characters = :characters," + "notes = :notes," + + "edited = :edited," + + "comicVineID = :comicVineID" + + " WHERE hash = :hash "); + + QSqlQuery insert(destDB); + insert.prepare("INSERT INTO comic_info " + "(title," + "coverPage," + "numPages," + "number," + "isBis," + "count," + "volume," + "storyArc," + "arcNumber," + "arcCount," + "genere," + "writer," + "penciller," + "inker," + "colorist," + "letterer," + "coverArtist," + "date," + "publisher," + "format," + "color," + "ageRating," + "synopsis," + "characters," + "notes," + "read," + "edited," + "comicVineID," + "hash)" + + "VALUES (:title," + ":coverPage," + ":numPages," + ":number," + ":isBis," + ":count," + + ":volume," + ":storyArc," + ":arcNumber," + ":arcCount," + + ":genere," + + ":writer," + ":penciller," + ":inker," + ":colorist," + ":letterer," + ":coverArtist," + + ":date," + ":publisher," + ":format," + ":color," + ":ageRating," + + ":synopsis," + ":characters," + ":notes," + + ":read," + ":edited," + ":comicVineID," + + ":hash )"); + + QSqlRecord record = newInfo.record(); + cp = record.value("coverPage").toInt(); + if(cp>1) + { + QSqlQuery checkCoverPage(destDB); + checkCoverPage.prepare("SELECT coverPage FROM comic_info where hash = :hash"); + checkCoverPage.bindValue(":hash",record.value("hash").toString()); + checkCoverPage.exec(); + bool extract = false; + if(checkCoverPage.next()) + { + extract = checkCoverPage.record().value("coverPage").toInt() != cp; + } + if(extract) + hashes.append(record.value("hash").toString()); + } + + bindValuesFromRecord(record,update); + + update.bindValue(":edited",1); + + + update.exec(); + + if(update.numRowsAffected() == 0) + { + + bindValuesFromRecord(record,insert); + insert.bindValue(":edited",1); + insert.bindValue(":read",0); + + insert.exec(); + + QString error1 = insert.lastError().databaseText(); + QString error2 = insert.lastError().driverText(); + + //QMessageBox::critical(NULL,"db",error1); + //QMessageBox::critical(NULL,"driver",error2); + } + //update.finish(); + //insert.finish(); + } + } + + destDB.commit(); + QString hash; + foreach(hash, hashes) + { + QSqlQuery getComic(destDB); + getComic.prepare("SELECT c.path,ci.coverPage FROM comic c INNER JOIN comic_info ci ON (c.comicInfoId = ci.id) where ci.hash = :hash"); + getComic.bindValue(":hash",hash); + getComic.exec(); + if(getComic.next()) + { + QString basePath = QString(dest).remove("/.yacreaderlibrary/library.ydb"); + QString path = basePath + getComic.record().value("path").toString(); + int coverPage = getComic.record().value("coverPage").toInt(); + ThumbnailCreator tc(path,basePath+"/.yacreaderlibrary/covers/"+hash+".jpg",coverPage); + tc.create(); + + } + } + + destDB.close(); + sourceDB.close(); + QSqlDatabase::removeDatabase(source); + QSqlDatabase::removeDatabase(dest); + return b; + +} +//TODO fix these bindings +void DataBaseManagement::bindValuesFromRecord(const QSqlRecord & record, QSqlQuery & query) +{ + bindString("title",record,query); + + bindInt("coverPage",record,query); + bindInt("numPages",record,query); + + bindInt("number",record,query); + bindInt("isBis",record,query); + bindInt("count",record,query); + + bindString("volume",record,query); + bindString("storyArc",record,query); + bindInt("arcNumber",record,query); + bindInt("arcCount",record,query); + + bindString("genere",record,query); + + bindString("writer",record,query); + bindString("penciller",record,query); + bindString("inker",record,query); + bindString("colorist",record,query); + bindString("letterer",record,query); + bindString("coverArtist",record,query); + + bindString("date",record,query); + bindString("publisher",record,query); + bindString("format",record,query); + bindInt("color",record,query); + bindString("ageRating",record,query); + + bindString("synopsis",record,query); + bindString("characters",record,query); + bindString("notes",record,query); + + bindString("comicVineID",record,query); + + bindString("hash",record,query); +} + +bool DataBaseManagement::addColumns(const QString &tableName, const QStringList &columnDefs, const QSqlDatabase &db) +{ + QString sql = "ALTER TABLE %1 ADD COLUMN %2"; + bool returnValue = true; + + foreach(QString columnDef, columnDefs) + { + QSqlQuery alterTable(db); + alterTable.prepare(sql.arg(tableName).arg(columnDef)); + //alterTableComicInfo.bindValue(":column_def",columnDef); + alterTable.exec(); + returnValue = returnValue && (alterTable.numRowsAffected() > 0); + } + + return returnValue; +} + +void DataBaseManagement::bindString(const QString & name, const QSqlRecord & record, QSqlQuery & query) +{ + if(!record.value(name).isNull()) + { + query.bindValue(":"+name,record.value(name).toString()); + } +} +void DataBaseManagement::bindInt(const QString & name, const QSqlRecord & record, QSqlQuery & query) +{ + if(!record.value(name).isNull()) + { + query.bindValue(":"+name,record.value(name).toInt()); + } +} + +QString DataBaseManagement::checkValidDB(const QString & fullPath) +{ + QSqlDatabase db = loadDatabaseFromFile(fullPath); + QString versionString = ""; + if(db.isValid() && db.isOpen()) + { + QSqlQuery version(db); + version.prepare("SELECT * FROM db_info"); + version.exec(); + + if(version.next()) + versionString = version.record().value("version").toString(); + } + + db.close(); + QSqlDatabase::removeDatabase(fullPath); + return versionString; +} + +int DataBaseManagement::compareVersions(const QString & v1, const QString v2) +{ + QStringList v1l = v1.split('.'); + QStringList v2l = v2.split('.'); + QList v1il; + QList v2il; + + foreach(QString s, v1l) + v1il.append(s.toInt()); + + foreach(QString s,v2l) + v2il.append(s.toInt()); + + for(int i=0;iv2il[i]) + return 1; + } + + if(v1il.length() < v2il.length()) + return -1; + if(v1il.length() == v2il.length()) + return 0; + if(v1il.length() > v2il.length()) + return 1; + + return 0; +} + +bool DataBaseManagement::updateToCurrentVersion(const QString & fullPath) +{ + bool pre7 = false; + bool pre7_1 = false; + if(compareVersions(DataBaseManagement::checkValidDB(fullPath),"7.0.0")<0) + pre7 = true; + if(compareVersions(DataBaseManagement::checkValidDB(fullPath),"7.0.3")<0) + pre7_1 = true; + + QSqlDatabase db = loadDatabaseFromFile(fullPath); + bool returnValue = false; + if(db.isValid() && db.isOpen()) + { + QSqlQuery updateVersion(db); + updateVersion.prepare("UPDATE db_info SET " + "version = :version"); + updateVersion.bindValue(":version",VERSION); + updateVersion.exec(); + + if(updateVersion.numRowsAffected() > 0) + returnValue = true; + + if(pre7) //TODO: execute only if previous version was < 7.0 + { + //new 7.0 fields + QStringList columnDefs; + columnDefs << "hasBeenOpened BOOLEAN DEFAULT 0" + << "rating INTEGER DEFAULT 0" + << "currentPage INTEGER DEFAULT 1" + << "bookmark1 INTEGER DEFAULT -1" + << "bookmark2 INTEGER DEFAULT -1" + << "bookmark3 INTEGER DEFAULT -1" + << "brightness INTEGER DEFAULT -1" + << "contrast INTEGER DEFAULT -1" + << "gamma INTEGER DEFAULT -1"; + + returnValue = returnValue && addColumns("comic_info", columnDefs, db); + } + //TODO update hasBeenOpened value + + if(pre7_1) + { + { + QStringList columnDefs; + columnDefs << "finished BOOLEAN DEFAULT 0" + << "completed BOOLEAN DEFAULT 1"; + returnValue = returnValue && addColumns("folder", columnDefs, db); + } + + {//comic_info + QStringList columnDefs; + columnDefs << "comicVineID TEXT DEFAULT NULL"; + returnValue = returnValue && addColumns("comic_info", columnDefs, db); + } + } + } + + db.close(); + QSqlDatabase::removeDatabase(fullPath); + return returnValue; +} + +//COMICS_INFO_EXPORTER +ComicsInfoExporter::ComicsInfoExporter() +:QThread() +{ +} + +void ComicsInfoExporter::exportComicsInfo(QSqlDatabase & source, QSqlDatabase & dest) +{ + Q_UNUSED(source) + Q_UNUSED(dest) + //TODO check this method +} + +void ComicsInfoExporter::run() +{ + +} + + +//COMICS_INFO_IMPORTER +ComicsInfoImporter::ComicsInfoImporter() +:QThread() +{ +} + +void ComicsInfoImporter::importComicsInfo(QSqlDatabase & source, QSqlDatabase & dest) +{ + Q_UNUSED(source) + Q_UNUSED(dest) + //TODO check this method +} + +void ComicsInfoImporter::run() +{ + +} diff --git a/YACReaderLibrary/db/data_base_management.h b/YACReaderLibrary/db/data_base_management.h new file mode 100644 index 00000000..ebb8aab4 --- /dev/null +++ b/YACReaderLibrary/db/data_base_management.h @@ -0,0 +1,61 @@ +#ifndef __DATA_BASE_MANAGEMENT_H +#define __DATA_BASE_MANAGEMENT_H + +#include +#include +#include + +#include "treemodel.h" + +class ComicsInfoExporter : public QThread +{ + Q_OBJECT +public: + ComicsInfoExporter(); + void exportComicsInfo(QSqlDatabase & source, QSqlDatabase & dest); +private: + void run(); +}; + +class ComicsInfoImporter : public QThread +{ + Q_OBJECT +public: + ComicsInfoImporter(); + void importComicsInfo(QSqlDatabase & source, QSqlDatabase & dest); +private: + void run(); + +}; + +class DataBaseManagement : public QObject +{ + Q_OBJECT +private: + QList dataBasesList; + static void bindString(const QString & name, const QSqlRecord & record, QSqlQuery & query); + static void bindInt(const QString & name, const QSqlRecord & record, QSqlQuery & query); + static void bindValuesFromRecord(const QSqlRecord & record, QSqlQuery & query); + + static bool addColumns(const QString & tableName, const QStringList & columnDefs, const QSqlDatabase & db); + +public: + DataBaseManagement(); + //TreeModel * newTreeModel(QString path); + //crea una base de datos y todas sus tablas + static QSqlDatabase createDatabase(QString name, QString path); + static QSqlDatabase createDatabase(QString dest); + //carga una base de datos desde la ruta path + static QSqlDatabase loadDatabase(QString path); + static QSqlDatabase loadDatabaseFromFile(QString path); + static bool createTables(QSqlDatabase & database); + + static void exportComicsInfo(QString source, QString dest); + static bool importComicsInfo(QString source, QString dest); + + static QString checkValidDB(const QString & fullPath); //retorna "" si la DB es inválida ó la versión si es válida. + static int compareVersions(const QString & v1, const QString v2); //retorna <0 si v1 < v2, 0 si v1 = v2 y >0 si v1 > v2 + static bool updateToCurrentVersion(const QString & path); +}; + +#endif diff --git a/YACReaderLibrary/db/tableitem.cpp b/YACReaderLibrary/db/tableitem.cpp new file mode 100644 index 00000000..842c47f4 --- /dev/null +++ b/YACReaderLibrary/db/tableitem.cpp @@ -0,0 +1,47 @@ + +#include + +#include "tableitem.h" + +//! [0] +TableItem::TableItem(const QList &data) + +{ + itemData = data; +} +//! [0] + +//! [1] +TableItem::~TableItem() +{ + +} +//! [1] + + +//! [5] +int TableItem::columnCount() const +{ + return itemData.count(); +} +//! [5] + +//! [6] +QVariant TableItem::data(int column) const +{ + return itemData.value(column); +} +//! [6] + +void TableItem::setData(int column,const QVariant & value) +{ + itemData[column] = value; +} + +//! [8] +int TableItem::row() const +{ + + return 0; +} +//! [8] diff --git a/YACReaderLibrary/db/tableitem.h b/YACReaderLibrary/db/tableitem.h new file mode 100644 index 00000000..f8679c46 --- /dev/null +++ b/YACReaderLibrary/db/tableitem.h @@ -0,0 +1,27 @@ +#ifndef TABLEITEM_H +#define TABLEITEM_H + +#include +#include + +//! [0] +class TableItem : public QObject +{ + Q_OBJECT +public: + TableItem(const QList &data); + ~TableItem(); + int columnCount() const; + QVariant data(int column) const; + void setData(int column,const QVariant & value); + int row() const; + //unsigned long long int id; //TODO sustituir por una clase adecuada + //Comic comic; +private: + QList itemData; + + +}; +//! [0] + +#endif \ No newline at end of file diff --git a/YACReaderLibrary/db/tablemodel.cpp b/YACReaderLibrary/db/tablemodel.cpp new file mode 100644 index 00000000..8987176f --- /dev/null +++ b/YACReaderLibrary/db/tablemodel.cpp @@ -0,0 +1,660 @@ + +#include +#include +#include + +#include "tableitem.h" +#include "tablemodel.h" +#include "data_base_management.h" +#include "qnaturalsorting.h" +#include "comic_db.h" +#include "db_helper.h" + +//ci.number,ci.title,c.fileName,ci.numPages,c.id,c.parentId,c.path,ci.hash,ci.read + + + +TableModel::TableModel(QObject *parent) + : QAbstractItemModel(parent) +{ + connect(this,SIGNAL(beforeReset()),this,SIGNAL(modelAboutToBeReset())); + connect(this,SIGNAL(reset()),this,SIGNAL(modelReset())); +} + +//! [0] +TableModel::TableModel( QSqlQuery &sqlquery, QObject *parent) + : QAbstractItemModel(parent) +{ + setupModelData(sqlquery); +} +//! [0] + +//! [1] +TableModel::~TableModel() +{ + qDeleteAll(_data); +} +//! [1] + +//! [2] +int TableModel::columnCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + if(_data.isEmpty()) + return 0; + return _data.first()->columnCount(); +} +//! [2] + +QHash TableModel::roleNames() const { + QHash roles; + + roles[NumberRole] = "number"; + roles[TitleRole] = "title"; + roles[FileNameRole] = "file_name"; + roles[NumPagesRole] = "num_pages"; + roles[IdRole] = "id"; + roles[Parent_IdRole] = "parent_id"; + roles[PathRole] = "path"; + roles[HashRole] = "hash"; + roles[ReadColumnRole] = "read_column"; + roles[IsBisRole] = "is_bis"; + roles[CurrentPageRole] = "current_page"; + roles[RatingRole] = "rating"; + roles[HasBeenOpenedRole] = "has_been_opened"; + roles[CoverPathRole] = "cover_path"; + + return roles; +} + +//! [3] +QVariant TableModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + /*if (index.column() == TableModel::Rating && role == Qt::DecorationRole) + { + TableItem *item = static_cast(index.internalPointer()); + return QPixmap(QString(":/images/rating%1.png").arg(item->data(index.column()).toInt())); + }*/ + + if (role == Qt::DecorationRole) + { + return QVariant(); + } + + if (role == Qt::TextAlignmentRole) + { + switch(index.column())//TODO obtener esto de la query + { + case TableModel::Number: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::NumPages: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::Hash: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::CurrentPage: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + default: + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + } + } + + + //TODO check here if any view is asking for TableModel::Roles + //these roles will be used from QML/GridView + + TableItem *item = static_cast(index.internalPointer()); + + if (role == NumberRole) + return item->data(Number); + else if (role == TitleRole) + return item->data(Title).isNull()?item->data(FileName):item->data(Title); + else if (role == RatingRole) + return item->data(Rating); + else if (role == CoverPathRole) + return "file:///"+_databasePath+"/covers/"+item->data(Hash).toString()+".jpg"; + else if (role == NumPagesRole) + return item->data(NumPages); + else if (role == CurrentPageRole) + return item->data(CurrentPage); + else if (role == ReadColumnRole) + return item->data(ReadColumn).toBool(); + else if (role == HasBeenOpenedRole) + return item->data(TableModel::HasBeenOpened); + + if (role != Qt::DisplayRole) + return QVariant(); + + if(index.column() == TableModel::Hash) + return QString::number(item->data(index.column()).toString().right(item->data(index.column()).toString().length()-40).toInt()/1024.0/1024.0,'f',2)+"Mb"; + if(index.column() == TableModel::ReadColumn) + return (item->data(TableModel::CurrentPage).toInt()==item->data(TableModel::NumPages).toInt() || item->data(TableModel::ReadColumn).toBool())?QVariant(tr("yes")):QVariant(tr("no")); + if(index.column() == TableModel::CurrentPage) + return item->data(TableModel::HasBeenOpened).toBool()?item->data(index.column()):QVariant("-"); + + if (index.column() == TableModel::Rating) + return QVariant(); + + return item->data(index.column()); +} +//! [3] + +//! [4] +Qt::ItemFlags TableModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + if(index.column() == TableModel::Rating) + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} +//! [4] + +//! [5] +QVariant TableModel::headerData(int section, Qt::Orientation orientation, + int role) const +{ + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + { + switch(section)//TODO obtener esto de la query + { + case TableModel::Number: + return QVariant(QString("#")); + case TableModel::Title: + return QVariant(QString(tr("Title"))); + case TableModel::FileName: + return QVariant(QString(tr("File Name"))); + case TableModel::NumPages: + return QVariant(QString(tr("Pages"))); + case TableModel::Hash: + return QVariant(QString(tr("Size"))); + case TableModel::ReadColumn: + return QVariant(QString(tr("Read"))); + case TableModel::CurrentPage: + return QVariant(QString(tr("Current Page"))); + case TableModel::Rating: + return QVariant(QString(tr("Rating"))); + } + } + + if (orientation == Qt::Horizontal && role == Qt::TextAlignmentRole) + { + switch(section)//TODO obtener esto de la query + { + case TableModel::Number: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::NumPages: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::Hash: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case TableModel::CurrentPage: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + default: + return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + } + } + + + if(orientation == Qt::Vertical && role == Qt::DecorationRole) + { + QString fileName = _data.value(section)->data(TableModel::FileName).toString(); + QFileInfo fi(fileName); + QString ext = fi.suffix(); + + if (ext.compare("cbr",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/comicRar.png")); + else if (ext.compare("cbz",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/comicZip.png")); + else if(ext.compare("pdf",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/pdf.png")); + else if (ext.compare("tar",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/tar.png")); + else if(ext.compare("zip",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/zip.png")); + else if(ext.compare("rar",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/rar.png")); + else if (ext.compare("7z",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/7z.png")); + else if (ext.compare("cb7",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/comic7z.png")); + else if (ext.compare("cb7",Qt::CaseInsensitive) == 0) + return QVariant(QIcon(":/images/comicTar.png")); + + } + + return QVariant(); +} +//! [5] + +//! [6] +QModelIndex TableModel::index(int row, int column, const QModelIndex &parent) + const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column, _data.at(row)); +} +//! [6] + +//! [7] +QModelIndex TableModel::parent(const QModelIndex &index) const +{ + Q_UNUSED(index) + return QModelIndex(); +} +//! [7] + +//! [8] +int TableModel::rowCount(const QModelIndex &parent) const +{ + if (parent.column() > 0) + return 0; + + if (!parent.isValid()) + return _data.count(); + + return 0; +} +//! [8] + +QStringList TableModel::getPaths(const QString & _source) +{ + QStringList paths; + QString source = _source + "/.yacreaderlibrary/covers/"; + QList::ConstIterator itr; + for(itr = _data.constBegin();itr != _data.constEnd();itr++) + { + QString hash = (*itr)->data(TableModel::Hash).toString(); + paths << source+ hash +".jpg"; + } + + return paths; +} + +void TableModel::setupModelData(unsigned long long int folderId,const QString & databasePath) +{ + //QFile f(QCoreApplication::applicationDirPath()+"/performance.txt"); + //f.open(QIODevice::Append); + beginResetModel(); + //QElapsedTimer timer; + //timer.start(); + qDeleteAll(_data); + _data.clear(); + + //QTextStream txtS(&f); + //txtS << "TABLEMODEL: Tiempo de borrado: " << timer.elapsed() << "ms\r\n"; + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + //crear la consulta + //timer.restart(); + QSqlQuery selectQuery(db); //TODO check + selectQuery.prepare("select ci.number,ci.title,c.fileName,ci.numPages,c.id,c.parentId,c.path,ci.hash,ci.read,ci.isBis,ci.currentPage,ci.rating,ci.hasBeenOpened from comic c inner join comic_info ci on (c.comicInfoId = ci.id) where c.parentId = :parentId"); + selectQuery.bindValue(":parentId", folderId); + selectQuery.exec(); + //txtS << "TABLEMODEL: Tiempo de consulta: " << timer.elapsed() << "ms\r\n"; + //timer.restart(); + setupModelData(selectQuery); + //txtS << "TABLEMODEL: Tiempo de creación del modelo: " << timer.elapsed() << "ms\r\n"; + //selectQuery.finish(); + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); + + if(_data.length()==0) + emit isEmpty(); +} + +QString TableModel::getComicPath(QModelIndex mi) +{ + if(mi.isValid()) + return _data.at(mi.row())->data(TableModel::Path).toString(); + return ""; +} + +void TableModel::setupModelData(QSqlQuery &sqlquery) +{ + TableItem * currentItem; + while (sqlquery.next()) + { + QList data; + QSqlRecord record = sqlquery.record(); + for(int i=0;idata(TableModel::FileName).toString(); + QString nameCurrent = currentItem->data(TableModel::FileName).toString(); + int numberLast,numberCurrent; + int max = (std::numeric_limits::max)(); + numberLast = numberCurrent = max; + + if(!last->data(TableModel::Number).isNull()) + numberLast = last->data(TableModel::Number).toInt(); + + if(!currentItem->data(TableModel::Number).isNull()) + numberCurrent = currentItem->data(TableModel::Number).toInt(); + + QList::iterator i; + i = _data.end(); + i--; + + if(numberCurrent != max) + { + while ((lessThan =numberCurrent < numberLast) && i != _data.begin()) + { + i--; + numberLast = max; + + if(!(*i)->data(TableModel::Number).isNull()) + numberLast = (*i)->data(TableModel::Number).toInt(); + } + } + else + { + while ((lessThan = naturalSortLessThanCI(nameCurrent,nameLast)) && i != _data.begin() && numberLast == max) + { + i--; + nameLast = (*i)->data(TableModel::FileName).toString(); + numberLast = max; + + if(!(*i)->data(TableModel::Number).isNull()) + numberLast = (*i)->data(TableModel::Number).toInt(); + } + + } + if(!lessThan) //si se ha encontrado un elemento menor que current, se inserta justo después + { + if(numberCurrent != max) + { + if(numberCurrent == numberLast) + if(currentItem->data(TableModel::IsBis).toBool()) + { + _data.insert(++i,currentItem); + } + else + _data.insert(i,currentItem); + else + _data.insert(++i,currentItem); + } + else + _data.insert(++i,currentItem); + } + else + { + _data.insert(i,currentItem); + } + + } + } +} + +ComicDB TableModel::getComic(const QModelIndex & mi) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + ComicDB c = DBHelper::loadComic(_data.at(mi.row())->data(TableModel::Id).toULongLong(),db); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return c; +} + +ComicDB TableModel::_getComic(const QModelIndex & mi) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + ComicDB c = DBHelper::loadComic(_data.at(mi.row())->data(TableModel::Id).toULongLong(),db); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return c; +} + + +QVector TableModel::getReadList() +{ + int numComics = _data.count(); + QVector readList(numComics); + for(int i=0;idata(TableModel::ReadColumn).toBool()) + readList[i] = YACReader::Read; + else if (_data.value(i)->data(TableModel::CurrentPage).toInt() == _data.value(i)->data(TableModel::NumPages).toInt()) + readList[i] = YACReader::Read; + else if (_data.value(i)->data(TableModel::HasBeenOpened).toBool()) + readList[i] = YACReader::Opened; + else + readList[i] = YACReader::Unread; + } + return readList; +} +//TODO untested, this method is no longer used +QVector TableModel::setAllComicsRead(YACReaderComicReadStatus read) +{ + return setComicsRead(persistentIndexList(),read); +} + +QList TableModel::getAllComics() +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + + QList comics; + int numComics = _data.count(); + for(int i=0;idata(TableModel::Id).toULongLong(),db)); + } + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return comics; +} + +QList TableModel::getComics(QList list) +{ + QList comics; + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + QList::const_iterator itr; + for(itr = list.constBegin(); itr!= list.constEnd();itr++) + { + comics.append(_getComic(*itr)); + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + return comics; +} +//TODO +QVector TableModel::setComicsRead(QList list,YACReaderComicReadStatus read) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + foreach (QModelIndex mi, list) + { + if(read == YACReader::Read) + { + _data.value(mi.row())->setData(TableModel::ReadColumn, QVariant(true)); + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(TableModel::Id).toULongLong(),db); + c.info.read = true; + DBHelper::update(&(c.info),db); + } + if(read == YACReader::Unread) + { + _data.value(mi.row())->setData(TableModel::ReadColumn, QVariant(false)); + _data.value(mi.row())->setData(TableModel::CurrentPage, QVariant(1)); + _data.value(mi.row())->setData(TableModel::HasBeenOpened, QVariant(false)); + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(TableModel::Id).toULongLong(),db); + c.info.read = false; + c.info.currentPage = 1; + c.info.hasBeenOpened = false; + DBHelper::update(&(c.info),db); + } + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + emit dataChanged(index(list.first().row(),TableModel::ReadColumn),index(list.last().row(),TableModel::HasBeenOpened),QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); + + return getReadList(); +} +qint64 TableModel::asignNumbers(QList list,int startingNumber) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + qint64 idFirst = _data.value(list[0].row())->data(TableModel::Id).toULongLong(); + int i = 0; + foreach (QModelIndex mi, list) + { + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(TableModel::Id).toULongLong(),db); + c.info.number = startingNumber+i; + c.info.edited = true; + DBHelper::update(&(c.info),db); + i++; + } + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + //emit dataChanged(index(list.first().row(),READ),index(list.last().row(),READ)); + + return idFirst; +} +QModelIndex TableModel::getIndexFromId(quint64 id) +{ + QList::ConstIterator itr; + int i=0; + for(itr = _data.constBegin();itr != _data.constEnd();itr++) + { + if((*itr)->data(TableModel::Id).toULongLong() == id) + break; + i++; + } + + return index(i,0); +} + +void TableModel::startTransaction() +{ + + dbTransaction = DataBaseManagement::loadDatabase(_databasePath); + dbTransaction.transaction(); +} + +void TableModel::finishTransaction() +{ + dbTransaction.commit(); + dbTransaction.close(); + QSqlDatabase::removeDatabase(_databasePath); + + +} + +void TableModel::removeInTransaction(int row) +{ + ComicDB c = DBHelper::loadComic(_data.at(row)->data(TableModel::Id).toULongLong(),dbTransaction); + + DBHelper::removeFromDB(&c,dbTransaction); + beginRemoveRows(QModelIndex(),row,row); + removeRow(row); + delete _data.at(row); + _data.removeAt(row); + + endRemoveRows(); +} + +void TableModel::remove(ComicDB * comic, int row) +{ + beginRemoveRows(QModelIndex(),row,row); + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::removeFromDB(comic,db); + + removeRow(row); + delete _data.at(row); + _data.removeAt(row); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endRemoveRows(); +} + +/*ComicDB TableModel::getComic(int row) +{ + return getComic(index(row,0)); +}*/ + +void TableModel::remove(int row) +{ + removeInTransaction(row); +} + +void TableModel::reload(const ComicDB & comic) +{ + int row = 0; + bool found = false; + foreach(TableItem * item,_data) + { + if(item->data(TableModel::Id).toULongLong() == comic.id) + { + found = true; + item->setData(TableModel::ReadColumn,comic.info.read); + item->setData(TableModel::CurrentPage,comic.info.currentPage); + item->setData(TableModel::HasBeenOpened,true); + break; + + } + row++; + } + if(found) + emit dataChanged(index(row,ReadColumn),index(row,HasBeenOpened), QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); +} + +void TableModel::resetComicRating(const QModelIndex &mi) +{ + ComicDB comic = getComic(mi); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + comic.info.rating = 0; + _data[mi.row()]->setData(TableModel::Rating,0); + DBHelper::update(&(comic.info),db); + + emit dataChanged(mi,mi); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + + +void TableModel::updateRating(int rating, QModelIndex mi) +{ + ComicDB comic = getComic(mi); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + //TODO optimize update + + comic.info.rating = rating; + _data[mi.row()]->setData(TableModel::Rating,rating); + DBHelper::update(&(comic.info),db); + + emit dataChanged(mi,mi); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} diff --git a/YACReaderLibrary/db/tablemodel.h b/YACReaderLibrary/db/tablemodel.h new file mode 100644 index 00000000..9df0126a --- /dev/null +++ b/YACReaderLibrary/db/tablemodel.h @@ -0,0 +1,120 @@ +#ifndef TABLEMODEL_H +#define TABLEMODEL_H + +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +class ComicDB; + +class TableItem; + +using namespace YACReader; + +//! [0] +class TableModel : public QAbstractItemModel +{ + Q_OBJECT + +public: + TableModel(QObject *parent = 0); + TableModel( QSqlQuery &sqlquery, QObject *parent = 0); + ~TableModel(); + + QVariant data(const QModelIndex &index, int role) const; + Qt::ItemFlags flags(const QModelIndex &index) const; + QVariant headerData(int section, Qt::Orientation orientation, + int role = Qt::DisplayRole) const; + QModelIndex index(int row, int column, + const QModelIndex &parent = QModelIndex()) const; + QModelIndex parent(const QModelIndex &index) const; + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent = QModelIndex()) const; + void setupModelData(unsigned long long int parentFolder,const QString & databasePath); + + //Métodos de conveniencia + QStringList getPaths(const QString & _source); + QString getComicPath(QModelIndex mi); + QString getCurrentPath(){return QString(_databasePath).remove("/.yacreaderlibrary");}; + ComicDB getComic(const QModelIndex & mi); //--> para la edición + //ComicDB getComic(int row); + QVector getReadList(); + QVector setAllComicsRead(YACReaderComicReadStatus readStatus); + QList getComics(QList list); //--> recupera la información común a los comics seleccionados + QList getAllComics(); + QModelIndex getIndexFromId(quint64 id); + //setcomicInfo(QModelIndex & mi); --> inserta en la base datos + //setComicInfoForAllComics(); --> inserta la información común a todos los cómics de una sola vez. + //setComicInfoForSelectedComis(QList list); -->inserta la información común para los comics seleccionados + QVector setComicsRead(QList list,YACReaderComicReadStatus read); + qint64 asignNumbers(QList list,int startingNumber); + void remove(ComicDB * comic, int row); + void removeInTransaction(int row); + void reload(const ComicDB & comic); + void resetComicRating(const QModelIndex & mi); + + QHash roleNames() const; + + enum Columns { + Number = 0, + Title = 1, + FileName = 2, + NumPages = 3, + Id = 4, + Parent_Id = 5, + Path = 6, + Hash = 7, + ReadColumn = 8, + IsBis = 9, + CurrentPage = 10, + Rating = 11, + HasBeenOpened = 12 +}; + + enum Roles { + NumberRole = Qt::UserRole + 1, + TitleRole, + FileNameRole, + NumPagesRole, + IdRole, + Parent_IdRole, + PathRole, + HashRole, + ReadColumnRole, + IsBisRole, + CurrentPageRole, + RatingRole, + HasBeenOpenedRole, + CoverPathRole + + }; + +public slots: + void remove(int row); + void startTransaction(); + void finishTransaction(); + void updateRating(int rating, QModelIndex mi); + +protected: + +private: + void setupModelData( QSqlQuery &sqlquery); + ComicDB _getComic(const QModelIndex & mi); + QList _data; + + QString _databasePath; + + QSqlDatabase dbTransaction; + +signals: + void beforeReset(); + void reset(); + void isEmpty(); +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db/treeitem.cpp b/YACReaderLibrary/db/treeitem.cpp new file mode 100644 index 00000000..8acf353c --- /dev/null +++ b/YACReaderLibrary/db/treeitem.cpp @@ -0,0 +1,152 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +/* + treeitem.cpp + + A container for items of data supplied by the simple tree model. +*/ + +#include + +#include "treeitem.h" +#include "qnaturalsorting.h" + +//! [0] +TreeItem::TreeItem(const QList &data, TreeItem *parent) +{ + parentItem = parent; + itemData = data; +} +//! [0] + +//! [1] +TreeItem::~TreeItem() +{ + qDeleteAll(childItems); +} +//! [1] + +//! [2] +void TreeItem::appendChild(TreeItem *item) +{ + item->parentItem = this; + + if(childItems.isEmpty()) + childItems.append(item); + else + { + TreeItem * last = childItems.back(); + QString nameLast = last->data(1).toString(); //TODO usar info name si está disponible, sino el nombre del fichero..... + QString nameCurrent = item->data(1).toString(); + QList::iterator i; + i = childItems.end(); + i--; + while (naturalSortLessThanCI(nameCurrent,nameLast) && i != childItems.begin()) + { + i--; + nameLast = (*i)->data(1).toString(); + } + if(!naturalSortLessThanCI(nameCurrent,nameLast)) //si se ha encontrado un elemento menor que current, se inserta justo después + childItems.insert(++i,item); + else + childItems.insert(i,item); + + } + + //childItems.append(item); +} +//! [2] + +//! [3] +TreeItem *TreeItem::child(int row) +{ + return childItems.value(row); +} +//! [3] + +//! [4] +int TreeItem::childCount() const +{ + return childItems.count(); +} +//! [4] + +//! [5] +int TreeItem::columnCount() const +{ + return itemData.count(); +} +//! [5] + +//! [6] +QVariant TreeItem::data(int column) const +{ + return itemData.value(column); +} +//! [6] + +void TreeItem::setData(int column, const QVariant & value) +{ + itemData[column] = value; +} + +//! [7] +TreeItem *TreeItem::parent() +{ + return parentItem; +} +//! [7] + +//! [8] +int TreeItem::row() const +{ + if (parentItem) + return parentItem->childItems.indexOf(const_cast(this)); + + return 0; +} +//! [8] + + +QList TreeItem::getData() const +{ + return itemData; +} diff --git a/YACReaderLibrary/db/treeitem.h b/YACReaderLibrary/db/treeitem.h new file mode 100644 index 00000000..fb2c0927 --- /dev/null +++ b/YACReaderLibrary/db/treeitem.h @@ -0,0 +1,79 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef TREEITEM_H +#define TREEITEM_H + +#include +#include +#include + +//! [0] +class TreeItem +{ +public: + TreeItem(const QList &data, TreeItem *parent = 0); + ~TreeItem(); + + void appendChild(TreeItem *child); + + TreeItem *child(int row); + int childCount() const; + int columnCount() const; + QVariant data(int column) const; + QList getData() const; + int row() const; + TreeItem *parent(); + TreeItem *parentItem; + unsigned long long int id; + QList comicNames; + TreeItem * originalItem; + void setData(int column, const QVariant &value); +private: + QList childItems; + QList itemData; + + + + +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db/treemodel.cpp b/YACReaderLibrary/db/treemodel.cpp new file mode 100644 index 00000000..f04b7bef --- /dev/null +++ b/YACReaderLibrary/db/treemodel.cpp @@ -0,0 +1,544 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +/* + treemodel.cpp + + Provides a simple tree model to show how to create and use hierarchical + models. +*/ + +#include + + +#include "treeitem.h" +#include "treemodel.h" +#include "data_base_management.h" +#include "folder.h" +#include "db_helper.h" +#include "qnaturalsorting.h" + +#ifdef Q_OS_MAC +#include +QIcon finishedFolderIcon; +void drawMacOSXFinishedFolderIcon() +{ + QIcon ico = QFileIconProvider().icon(QFileIconProvider::Folder); + QPixmap pixNormalOff = ico.pixmap(16,16, QIcon::Normal, QIcon::Off); + QPixmap pixNormalOn = ico.pixmap(16,16, QIcon::Normal, QIcon::On); + QPixmap pixSelectedOff = ico.pixmap(16,16, QIcon::Selected, QIcon::Off); + QPixmap pixSelectedOn = ico.pixmap(16,16, QIcon::Selected, QIcon::On); + QPixmap tick(":/images/folder_finished_macosx.png"); + + + { + QPainter p(&pixNormalOff); + p.drawPixmap(4,7,tick); + } + finishedFolderIcon.addPixmap(pixNormalOff, QIcon::Normal, QIcon::Off); + + { + QPainter p(&pixNormalOn); + p.drawPixmap(4,7,tick); + } + finishedFolderIcon.addPixmap(pixNormalOn, QIcon::Normal, QIcon::On); + + { + QPainter p(&pixSelectedOff); + p.drawPixmap(4,7,tick); + } + finishedFolderIcon.addPixmap(pixSelectedOff, QIcon::Selected, QIcon::Off); + + { + QPainter p(&pixSelectedOn); + p.drawPixmap(4,7,tick); + } + finishedFolderIcon.addPixmap(pixSelectedOn, QIcon::Selected, QIcon::On); +} +#endif + +#define ROOT 1 + +TreeModel::TreeModel(QObject *parent) + : QAbstractItemModel(parent),rootItem(0),rootBeforeFilter(0),filterEnabled(false),includeComics(false) +{ + connect(this,SIGNAL(beforeReset()),this,SIGNAL(modelAboutToBeReset())); + connect(this,SIGNAL(reset()),this,SIGNAL(modelReset())); +} + +//! [0] +TreeModel::TreeModel( QSqlQuery &sqlquery, QObject *parent) + : QAbstractItemModel(parent),rootItem(0),rootBeforeFilter(0),filterEnabled(false),includeComics(false) +{ + //lo m�s probable es que el nodo ra�z no necesite tener informaci�n + QList rootData; + rootData << "root"; //id 0, padre 0, title "root" (el id, y el id del padre van a ir en la clase TreeItem) + rootItem = new TreeItem(rootData); + rootItem->id = ROOT; + rootItem->parentItem = 0; + setupModelData(sqlquery, rootItem); + //sqlquery.finish(); +} +//! [0] + +//! [1] +TreeModel::~TreeModel() +{ + if(rootItem != 0) + delete rootItem; +} +//! [1] + +//! [2] +int TreeModel::columnCount(const QModelIndex &parent) const +{ + if (parent.isValid()) + return static_cast(parent.internalPointer())->columnCount(); + else + return rootItem->columnCount(); +} +//! [2] + +//! [3] +QVariant TreeModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + TreeItem *item = static_cast(index.internalPointer()); + + if (role == Qt::DecorationRole) + +#ifdef Q_OS_MAC + if(item->data(TreeModel::Finished).toBool()){ + if(finishedFolderIcon.isNull()){ + drawMacOSXFinishedFolderIcon(); + } + + return QVariant(finishedFolderIcon); + } + else { + return QVariant(QFileIconProvider().icon(QFileIconProvider::Folder)); + } +#else + if(item->data(TreeModel::Finished).toBool()) + return QVariant(QIcon(":/images/folder_finished.png")); + else + return QVariant(QIcon(":/images/folder.png")); +#endif + + if (role != Qt::DisplayRole) + return QVariant(); + + + + return item->data(index.column()); +} +//! [3] + +//! [4] +Qt::ItemFlags TreeModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} +//! [4] + +//! [5] +QVariant TreeModel::headerData(int section, Qt::Orientation orientation, + int role) const +{ + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + return rootItem->data(section); + + return QVariant(); +} +//! [5] + +//! [6] +QModelIndex TreeModel::index(int row, int column, const QModelIndex &parent) + const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + TreeItem *parentItem; + + if (!parent.isValid()) + parentItem = rootItem; + else + parentItem = static_cast(parent.internalPointer()); + + TreeItem *childItem = parentItem->child(row); + if (childItem) + return createIndex(row, column, childItem); + else + return QModelIndex(); +} +//! [6] + +//! [7] +QModelIndex TreeModel::parent(const QModelIndex &index) const +{ + if (!index.isValid()) + return QModelIndex(); + + TreeItem *childItem = static_cast(index.internalPointer()); + TreeItem *parentItem = childItem->parent(); + + if (parentItem == rootItem) + return QModelIndex(); + + return createIndex(parentItem->row(), 0, parentItem); +} +//! [7] + +QModelIndex TreeModel::indexFromItem(TreeItem * item,int column) +{ + //if(item->parent() != 0) + // return index(item->row(),column,parent(indexFromItem(item->parent(),column-1))); + //else + // return index(item->row(),0,QModelIndex()); + return createIndex(item->row(), column, item); +} + + +//! [8] +int TreeModel::rowCount(const QModelIndex &parent) const +{ + TreeItem *parentItem; + if (parent.column() > 0) + return 0; + + if (!parent.isValid()) + parentItem = rootItem; + else + parentItem = static_cast(parent.internalPointer()); + + return parentItem->childCount(); +} +//! [8] + +void TreeModel::setupModelData(QString path) +{ + beginResetModel(); + if(rootItem != 0) + delete rootItem; //TODO comprobar que se libera bien la memoria + filterEnabled = false; + rootItem = 0; + rootBeforeFilter = 0; + //inicializar el nodo ra�z + QList rootData; + rootData << "root"; //id 0, padre 0, title "root" (el id, y el id del padre van a ir en la clase TreeItem) + rootItem = new TreeItem(rootData); + rootItem->id = ROOT; + rootItem->parentItem = 0; + + //cargar la base de datos + _databasePath = path; + QSqlDatabase db = DataBaseManagement::loadDatabase(path); + //crear la consulta + { + QSqlQuery selectQuery("select * from folder where id <> 1 order by parentId,name",db); + + setupModelData(selectQuery,rootItem); + } + //selectQuery.finish(); + db.close(); + QSqlDatabase::removeDatabase(path); + endResetModel(); + +} + + +void TreeModel::setupModelData(QSqlQuery &sqlquery, TreeItem *parent) +{ + //64 bits para la primary key, es decir la misma precisi�n que soporta sqlit 2^64 + //el diccionario permitir� encontrar cualquier nodo del �rbol r�pidamente, de forma que a�adir un hijo a un padre sea O(1) + items.clear(); + //se a�ade el nodo 0 + items.insert(parent->id,parent); + + while (sqlquery.next()) { + QList data; + QSqlRecord record = sqlquery.record(); + + data << record.value("name").toString(); + data << record.value("path").toString(); + data << record.value("finished").toBool(); + data << record.value("completed").toBool(); + TreeItem * item = new TreeItem(data); + + item->id = record.value("id").toULongLong(); + //la inserci�n de hijos se hace de forma ordenada + TreeItem * parent = items.value(record.value("parentId").toULongLong()); + if(parent !=0) //TODO if parent==0 the parent of item was removed from the DB and delete on cascade didn't work, ERROR. + parent->appendChild(item); + //se a�ade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones + items.insert(item->id,item); + } +} + +void TreeModel::setupFilteredModelData() +{ + beginResetModel(); + + //TODO hay que liberar memoria de anteriores filtrados + + //inicializar el nodo ra�z + + if(rootBeforeFilter == 0) + rootBeforeFilter = rootItem; + else + delete rootItem;//los resultados de la b�squeda anterior deben ser borrados + + QList rootData; + rootData << "root"; //id 1, padre 1, title "root" (el id, y el id del padre van a ir en la clase TreeItem) + rootItem = new TreeItem(rootData); + rootItem->id = ROOT; + rootItem->parentItem = 0; + + //cargar la base de datos + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + //crear la consulta + { + QSqlQuery selectQuery(db); //TODO check + if(!includeComics) + { + selectQuery.prepare("select * from folder where id <> 1 and upper(name) like upper(:filter) order by parentId,name "); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + } + else + { + selectQuery.prepare("SELECT DISTINCT f.id, f.parentId, f.name, f.path, f.finished, f.completed FROM folder f INNER JOIN comic c ON (f.id = c.parentId) WHERE f.id <> 1 AND ((UPPER(c.fileName) like UPPER(:filter)) OR (UPPER(f.name) like UPPER(:filter2))) ORDER BY f.parentId,f.name"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":filter2", "%%"+filter+"%%"); + } + selectQuery.exec(); + + setupFilteredModelData(selectQuery,rootItem); + } + //selectQuery.finish(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + endResetModel(); +} + +void TreeModel::setupFilteredModelData(QSqlQuery &sqlquery, TreeItem *parent) +{ + //64 bits para la primary key, es decir la misma precisi�n que soporta sqlit 2^64 + filteredItems.clear(); + + //se a�ade el nodo 0 al modelo que representa el arbol de elementos que cumplen con el filtro + filteredItems.insert(parent->id,parent); + + while (sqlquery.next()) { //se procesan todos los folders que cumplen con el filtro + //datos de la base de datos + QList data; + QSqlRecord record = sqlquery.record(); + + data << record.value("name").toString(); + data << record.value("path").toString(); + data << record.value("finished").toBool(); + data << record.value("completed").toBool(); + + TreeItem * item = new TreeItem(data); + item->id = sqlquery.value(0).toULongLong(); + + //id del padre + quint64 parentId = record.value("parentId").toULongLong(); + + //se a�ade el item al map, de forma que se pueda encontrar como padre en siguientes iteraciones + if(!filteredItems.contains(item->id)) + filteredItems.insert(item->id,item); + + //es necesario conocer las coordenadas de origen para poder realizar scroll autom�tico en la vista + item->originalItem = items.value(item->id); + + //si el padre ya existe en el modelo, el item se a�ade como hijo + if(filteredItems.contains(parentId)) + filteredItems.value(parentId)->appendChild(item); + else//si el padre a�n no se ha a�adido, hay que a�adirlo a �l y todos los padres hasta el nodo ra�z + { + //comprobamos con esta variable si el �ltimo de los padres (antes del nodo ra�z) ya exist�a en el modelo + bool parentPreviousInserted = false; + + //mientras no se alcance el nodo ra�z se procesan todos los padres (de abajo a arriba) + while(parentId != ROOT ) + { + //el padre no estaba en el modelo filtrado, as� que se rescata del modelo original + TreeItem * parentItem = items.value(parentId); + //se debe crear un nuevo nodo (para no compartir los hijos con el nodo original) + TreeItem * newparentItem = new TreeItem(parentItem->getData()); //padre que se a�adir� a la estructura de directorios filtrados + newparentItem->id = parentId; + + newparentItem->originalItem = parentItem; + + //si el modelo contiene al padre, se a�ade el item actual como hijo + if(filteredItems.contains(parentId)) + { + filteredItems.value(parentId)->appendChild(item); + parentPreviousInserted = true; + } + //sino se registra el nodo para poder encontrarlo con posterioridad y se a�ade el item actual como hijo + else + { + newparentItem->appendChild(item); + filteredItems.insert(newparentItem->id,newparentItem); + parentPreviousInserted = false; + } + + //variables de control del bucle, se avanza hacia el nodo padre + item = newparentItem; + parentId = parentItem->parentItem->id; + } + + //si el nodo es hijo de 1 y no hab�a sido previamente insertado como hijo, se a�ade como tal + if(!parentPreviousInserted) + filteredItems.value(ROOT)->appendChild(item); + } + } +} + + + +QString TreeModel::getDatabase() +{ + return _databasePath; +} + +QString TreeModel::getFolderPath(const QModelIndex &folder) +{ + return static_cast(folder.internalPointer())->data(TreeModel::Path).toString(); +} + +void TreeModel::setFilter(QString filter, bool includeComics) +{ + this->filter = filter; + this->includeComics = includeComics; + filterEnabled = true; + setupFilteredModelData(); +} + +void TreeModel::resetFilter() +{ + beginResetModel(); + filter = ""; + includeComics = false; + //TODO hay que liberar la memoria reservada para el filtrado + //items.clear(); + filteredItems.clear(); + TreeItem * root = rootItem; + rootItem = rootBeforeFilter; //TODO si no se aplica el filtro previamente, esto invalidar�a en modelo + if(root !=0) + delete root; + + rootBeforeFilter = 0; + filterEnabled = false; + endResetModel(); + + +} + +void TreeModel::updateFolderCompletedStatus(const QModelIndexList &list, bool status) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + foreach (QModelIndex mi, list) + { + TreeItem * item = static_cast(mi.internalPointer()); + item->setData(TreeModel::Completed,status); + + Folder f = DBHelper::loadFolder(item->id,db); + f.setCompleted(status); + DBHelper::update(f,db); + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + emit dataChanged(index(list.first().row(),TreeModel::Name),index(list.last().row(),TreeModel::Completed)); +} + +void TreeModel::updateFolderFinishedStatus(const QModelIndexList &list, bool status) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + foreach (QModelIndex mi, list) + { + TreeItem * item = static_cast(mi.internalPointer()); + item->setData(TreeModel::Finished,status); + + Folder f = DBHelper::loadFolder(item->id,db); + f.setFinished(status); + DBHelper::update(f,db); + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + emit dataChanged(index(list.first().row(),TreeModel::Name),index(list.last().row(),TreeModel::Completed)); +} + +QStringList TreeModel::getSubfoldersNames(const QModelIndex &mi) +{ + QStringList result; + qulonglong id = 1; + if(mi.isValid()){ + TreeItem * item = static_cast(mi.internalPointer()); + id = item->id; + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + + result = DBHelper::loadSubfoldersNames(id,db); + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + //TODO sort result)) + qSort(result.begin(),result.end(),naturalSortLessThanCI); + return result; +} diff --git a/YACReaderLibrary/db/treemodel.h b/YACReaderLibrary/db/treemodel.h new file mode 100644 index 00000000..d36dab40 --- /dev/null +++ b/YACReaderLibrary/db/treemodel.h @@ -0,0 +1,119 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef TREEMODEL_H +#define TREEMODEL_H + +#include +#include +#include +#include +#include + +class TreeItem; + +//! [0] +class TreeModel : public QAbstractItemModel +{ + Q_OBJECT + +public: + TreeModel(QObject *parent = 0); + TreeModel( QSqlQuery &sqlquery, QObject *parent = 0); + ~TreeModel(); + + QVariant data(const QModelIndex &index, int role) const; + Qt::ItemFlags flags(const QModelIndex &index) const; + QVariant headerData(int section, Qt::Orientation orientation, + int role = Qt::DisplayRole) const; + QModelIndex index(int row, int column, + const QModelIndex &parent = QModelIndex()) const; + QModelIndex parent(const QModelIndex &index) const; + QModelIndex indexFromItem(TreeItem * item, int column); + /*QModelIndex _indexFromItem(TreeItem * item, int column); + int column;*/ + + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent = QModelIndex()) const; + void setupModelData(QString path); + QString getDatabase(); + + //Métodos de conveniencia + QString getFolderPath(const QModelIndex &folder); + + void setFilter(QString filter, bool includeComics); + void resetFilter(); + bool isFilterEnabled(){return filterEnabled;}; + + void updateFolderCompletedStatus(const QModelIndexList & list, bool status); + void updateFolderFinishedStatus(const QModelIndexList & list, bool status); + + QStringList getSubfoldersNames(const QModelIndex & mi); + + enum Columns { + Name = 0, + Path = 1, + Finished = 2, + Completed = 3 + };//id INTEGER PRIMARY KEY, parentId INTEGER NOT NULL, name TEXT NOT NULL, path TEXT NOT NULL + +private: + void setupModelData( QSqlQuery &sqlquery, TreeItem *parent); + void setupFilteredModelData( QSqlQuery &sqlquery, TreeItem *parent); + void setupFilteredModelData(); + + TreeItem *rootItem; //el árbol + QMap items; //relación entre folders + + TreeItem *rootBeforeFilter; + QMap filteredItems; //relación entre folders + + QString _databasePath; + + bool includeComics; + QString filter; + bool filterEnabled; +signals: + void beforeReset(); + void reset(); +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db_helper.cpp b/YACReaderLibrary/db_helper.cpp new file mode 100644 index 00000000..f7687689 --- /dev/null +++ b/YACReaderLibrary/db_helper.cpp @@ -0,0 +1,704 @@ +#include "db_helper.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +#include "library_item.h" +#include "comic_db.h" +#include "data_base_management.h" +#include "folder.h" +#include "yacreader_libraries.h" + +#include "qnaturalsorting.h" + +#include "QsLog.h" +//server + +YACReaderLibraries DBHelper::getLibraries() +{ + YACReaderLibraries libraries; + libraries.load(); + return libraries; +} +QList DBHelper::getFolderContentFromLibrary(const QString & libraryName, qulonglong folderId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList list = DBHelper::getFoldersFromParent(folderId,db,false); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return list; +} +QList DBHelper::getFolderComicsFromLibrary(const QString & libraryName, qulonglong folderId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList list = DBHelper::getComicsFromParent(folderId,db,false); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return list; +} +qulonglong DBHelper::getParentFromComicFolderId(const QString & libraryName, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + Folder f = DBHelper::loadFolder(id,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return f.parentId; +} +ComicDB DBHelper::getComicInfo(const QString & libraryName, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + ComicDB comic = DBHelper::loadComic(id,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return comic; +} + +QList DBHelper::getSiblings(const QString & libraryName, qulonglong parentId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList comics = DBHelper::getSortedComicsFromParent(parentId,db); + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return comics; +} + +QString DBHelper::getFolderName(const QString & libraryName, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QString name=""; + + { + QSqlQuery selectQuery(db); //TODO check + selectQuery.prepare("SELECT name FROM folder WHERE id = :id"); + selectQuery.bindValue(":id", id); + selectQuery.exec(); + + if(selectQuery.next()) + { + QSqlRecord record = selectQuery.record(); + name = record.value(0).toString(); + } + } + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return name; +} +QList DBHelper::getLibrariesNames() +{ + QStringList names = getLibraries().getNames(); + qSort(names.begin(),names.end(),naturalSortLessThanCI); + return names; +} +QString DBHelper::getLibraryName(int id) +{ + return getLibraries().getName(id); +} +//objects management +//deletes +void DBHelper::removeFromDB(LibraryItem * item, QSqlDatabase & db) +{ + if(item->isDir()) + DBHelper::removeFromDB(dynamic_cast(item),db); + else + DBHelper::removeFromDB(dynamic_cast(item),db); +} +void DBHelper::removeFromDB(Folder * folder, QSqlDatabase & db) +{ + QSqlQuery query(db); + query.prepare("DELETE FROM folder WHERE id = :id"); + query.bindValue(":id", folder->id); + query.exec(); +} +void DBHelper::removeFromDB(ComicDB * comic, QSqlDatabase & db) +{ + QSqlQuery query(db); + query.prepare("DELETE FROM comic WHERE id = :id"); + query.bindValue(":id", comic->id); + query.exec(); +} + +//updates +void DBHelper::update(ComicDB * comic, QSqlDatabase & db) +{ + Q_UNUSED(comic) + Q_UNUSED(db) + //do nothing +} + +void DBHelper::update(const QString & libraryName, ComicInfo & comicInfo) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryName); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + DBHelper::update(&comicInfo,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); +} + +void DBHelper::update(ComicInfo * comicInfo, QSqlDatabase & db) +{ + QSqlQuery updateComicInfo(db); + updateComicInfo.prepare("UPDATE comic_info SET " + "title = :title," + + "coverPage = :coverPage," + "numPages = :numPages," + + "number = :number," + "isBis = :isBis," + "count = :count," + + "volume = :volume," + "storyArc = :storyArc," + "arcNumber = :arcNumber," + "arcCount = :arcCount," + + "genere = :genere," + + "writer = :writer," + "penciller = :penciller," + "inker = :inker," + "colorist = :colorist," + "letterer = :letterer," + "coverArtist = :coverArtist," + + "date = :date," + "publisher = :publisher," + "format = :format," + "color = :color," + "ageRating = :ageRating," + + "synopsis = :synopsis," + "characters = :characters," + "notes = :notes," + + "read = :read," + "edited = :edited," + //new 7.0 fields + "hasBeenOpened = :hasBeenOpened," + + "currentPage = :currentPage," + "bookmark1 = :bookmark1," + "bookmark2 = :bookmark2," + "bookmark3 = :bookmark3," + "brightness = :brightness," + "contrast = :contrast, " + "gamma = :gamma," + "rating = :rating," + + //new 7.1 fields + "comicVineID = :comicVineID" + //-- + " WHERE id = :id "); + + updateComicInfo.bindValue(":title",comicInfo->title); + + updateComicInfo.bindValue(":coverPage", comicInfo->coverPage); + updateComicInfo.bindValue(":numPages", comicInfo->numPages); + + updateComicInfo.bindValue(":number", comicInfo->number); + updateComicInfo.bindValue(":isBis", comicInfo->isBis); + updateComicInfo.bindValue(":count", comicInfo->count); + + updateComicInfo.bindValue(":volume", comicInfo->volume); + updateComicInfo.bindValue(":storyArc", comicInfo->storyArc); + updateComicInfo.bindValue(":arcNumber",comicInfo->arcNumber); + updateComicInfo.bindValue(":arcCount",comicInfo->arcCount); + + updateComicInfo.bindValue(":genere",comicInfo->genere); + + updateComicInfo.bindValue(":writer",comicInfo->writer); + updateComicInfo.bindValue(":penciller",comicInfo->penciller); + updateComicInfo.bindValue(":inker",comicInfo->inker); + updateComicInfo.bindValue(":colorist",comicInfo->colorist); + updateComicInfo.bindValue(":letterer",comicInfo->letterer); + updateComicInfo.bindValue(":coverArtist",comicInfo->coverArtist); + + updateComicInfo.bindValue(":date",comicInfo->date); + updateComicInfo.bindValue(":publisher",comicInfo->publisher); + updateComicInfo.bindValue(":format",comicInfo->format); + updateComicInfo.bindValue(":color",comicInfo->color); + updateComicInfo.bindValue(":ageRating",comicInfo->ageRating); + + updateComicInfo.bindValue(":synopsis",comicInfo->synopsis); + updateComicInfo.bindValue(":characters",comicInfo->characters); + updateComicInfo.bindValue(":notes",comicInfo->notes); + + bool read = comicInfo->read || comicInfo->currentPage == comicInfo->numPages.toInt(); //if current page is the las page, the comic is read(completed) + comicInfo->read = read; + updateComicInfo.bindValue(":read", read?1:0); + updateComicInfo.bindValue(":id", comicInfo->id); + updateComicInfo.bindValue(":edited", comicInfo->edited?1:0); + + updateComicInfo.bindValue(":hasBeenOpened", comicInfo->hasBeenOpened?1:0); + updateComicInfo.bindValue(":currentPage", comicInfo->currentPage); + updateComicInfo.bindValue(":bookmark1", comicInfo->bookmark1); + updateComicInfo.bindValue(":bookmark2", comicInfo->bookmark2); + updateComicInfo.bindValue(":bookmark3", comicInfo->bookmark3); + updateComicInfo.bindValue(":brightness", comicInfo->brightness); + updateComicInfo.bindValue(":contrast", comicInfo->contrast); + updateComicInfo.bindValue(":gamma", comicInfo->gamma); + updateComicInfo.bindValue(":rating", comicInfo->rating); + + updateComicInfo.bindValue(":comicVineID", comicInfo->comicVineID); + + updateComicInfo.exec(); +} + +void DBHelper::updateRead(ComicInfo * comicInfo, QSqlDatabase & db) +{ + QSqlQuery updateComicInfo(db); + updateComicInfo.prepare("UPDATE comic_info SET " + "read = :read" + " WHERE id = :id "); + + updateComicInfo.bindValue(":read", comicInfo->read?1:0); + updateComicInfo.bindValue(":id", comicInfo->id); + updateComicInfo.exec(); +} + +void DBHelper::update(const Folder & folder, QSqlDatabase &db) +{ + QSqlQuery updateFolderInfo(db); + updateFolderInfo.prepare("UPDATE folder SET " + "finished = :finished, " + "completed = :completed " + "WHERE id = :id "); + updateFolderInfo.bindValue(":finished", folder.isFinished()?1:0); + updateFolderInfo.bindValue(":completed", folder.isCompleted()?1:0); + updateFolderInfo.bindValue(":id", folder.id); + updateFolderInfo.exec(); +} + +void DBHelper::updateProgress(qulonglong libraryId, const ComicInfo &comicInfo) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + ComicDB comic = DBHelper::loadComic(comicInfo.id,db); + comic.info.currentPage = comicInfo.currentPage; + comic.info.hasBeenOpened = true; + + DBHelper::update(&comic.info,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); +} +//inserts +qulonglong DBHelper::insert(Folder * folder, QSqlDatabase & db) +{ + QSqlQuery query(db); + query.prepare("INSERT INTO folder (parentId, name, path) " + "VALUES (:parentId, :name, :path)"); + query.bindValue(":parentId", folder->parentId); + query.bindValue(":name", folder->name); + query.bindValue(":path", folder->path); + query.exec(); + return query.lastInsertId().toULongLong(); +} + +qulonglong DBHelper::insert(ComicDB * comic, QSqlDatabase & db) +{ + if(!comic->info.existOnDb) + { + QSqlQuery comicInfoInsert(db); + comicInfoInsert.prepare("INSERT INTO comic_info (hash,numPages) " + "VALUES (:hash,:numPages)"); + comicInfoInsert.bindValue(":hash", comic->info.hash); + comicInfoInsert.bindValue(":numPages", comic->info.numPages); + comicInfoInsert.exec(); + comic->info.id =comicInfoInsert.lastInsertId().toULongLong(); + comic->_hasCover = false; + } + else + comic->_hasCover = true; + + QSqlQuery query(db); + query.prepare("INSERT INTO comic (parentId, comicInfoId, fileName, path) " + "VALUES (:parentId,:comicInfoId,:name, :path)"); + query.bindValue(":parentId", comic->parentId); + query.bindValue(":comicInfoId", comic->info.id); + query.bindValue(":name", comic->name); + query.bindValue(":path", comic->path); + query.exec(); + return query.lastInsertId().toULongLong(); +} +//queries +QList DBHelper::getFoldersFromParent(qulonglong parentId, QSqlDatabase & db, bool sort) +{ + QList list; + + QSqlQuery selectQuery(db); //TODO check + selectQuery.prepare("SELECT * FROM folder WHERE parentId = :parentId and id <> 1"); + selectQuery.bindValue(":parentId", parentId); + selectQuery.exec(); + + Folder * currentItem; + while (selectQuery.next()) + { + QList data; + QSqlRecord record = selectQuery.record(); + for(int i=0;i(list.back()); + QString nameLast = last->name; + QString nameCurrent = currentItem->name; + QList::iterator i; + i = list.end(); + i--; + while ((0 > (lessThan = naturalSortLessThanCI(nameCurrent,nameLast))) && i != list.begin()) + { + i--; + nameLast = (*i)->name; + } + if(lessThan>=0) //si se ha encontrado un elemento menor que current, se inserta justo después + list.insert(++i,currentItem); + else + list.insert(i,currentItem); + + } + } + + return list; +} + +QList DBHelper::getSortedComicsFromParent(qulonglong parentId, QSqlDatabase & db) +{ + + QList list; + + QSqlQuery selectQuery(db); + selectQuery.prepare("select c.id,c.parentId,c.fileName,c.path,ci.hash from comic c inner join comic_info ci on (c.comicInfoId = ci.id) where c.parentId = :parentId"); + selectQuery.bindValue(":parentId", parentId); + selectQuery.exec(); + + ComicDB currentItem; + while (selectQuery.next()) + { + QList data; + QSqlRecord record = selectQuery.record(); + for(int i=0;i(list.back()); + QString nameLast = last.name; + QString nameCurrent = currentItem.name; + + int numberLast,numberCurrent; + int max = (std::numeric_limits::max)(); + numberLast = numberCurrent = max; //TODO change by std limit + + if(!last.info.number.isNull()) + numberLast = last.info.number.toInt(); + + if(!currentItem.info.number.isNull()) + numberCurrent = currentItem.info.number.toInt(); + + QList::iterator i; + i = list.end(); + i--; + + if(numberCurrent != max) + { + while ((lessThan =numberCurrent < numberLast) && i != list.begin()) + { + i--; + numberLast = max; + + if(!(*i).info.number.isNull()) + numberLast = (*i).info.number.toInt(); + } + } + else + { + while ((lessThan = naturalSortLessThanCI(nameCurrent,nameLast)) && i != list.begin() && numberLast == max) + { + i--; + nameLast = (*i).name; + numberLast = max; + + if(!(*i).info.number.isNull()) + numberLast = (*i).info.number.toInt(); + } + + } + if(!lessThan) //si se ha encontrado un elemento menor que current, se inserta justo después + { + if(numberCurrent != max) + { + if(numberCurrent == numberLast) + if(currentItem.info.isBis.toBool()) + { + list.insert(++i,currentItem); + } + else + list.insert(i,currentItem); + else + list.insert(++i,currentItem); + } + else + list.insert(++i,currentItem); + } + else + { + list.insert(i,currentItem); + } + + } + } + //selectQuery.finish(); + return list; +} +QList DBHelper::getComicsFromParent(qulonglong parentId, QSqlDatabase & db, bool sort) +{ + QList list; + + QSqlQuery selectQuery(db); + selectQuery.prepare("select c.id,c.parentId,c.fileName,c.path,ci.hash from comic c inner join comic_info ci on (c.comicInfoId = ci.id) where c.parentId = :parentId"); + selectQuery.bindValue(":parentId", parentId); + selectQuery.exec(); + + ComicDB * currentItem; + while (selectQuery.next()) + { + QList data; + QSqlRecord record = selectQuery.record(); + for(int i=0;iid = record.value("id").toULongLong(); + currentItem->parentId = record.value(1).toULongLong(); + currentItem->name = record.value(2).toString(); + currentItem->path = record.value(3).toString(); + currentItem->info = DBHelper::loadComicInfo(record.value(4).toString(),db); + int lessThan = 0; + if(list.isEmpty() || !sort) + list.append(currentItem); + else + { + ComicDB * last = static_cast(list.back()); + QString nameLast = last->name; + QString nameCurrent = currentItem->name; + QList::iterator i; + i = list.end(); + i--; + while ((0 > (lessThan = nameCurrent.localeAwareCompare(nameLast))) && i != list.begin()) //se usa la misma ordenación que en QDir + { + i--; + nameLast = (*i)->name; + } + if(lessThan>0) //si se ha encontrado un elemento menor que current, se inserta justo después + list.insert(++i,currentItem); + else + list.insert(i,currentItem); + + } + } + //selectQuery.finish(); + return list; +} + +//loads +Folder DBHelper::loadFolder(qulonglong id, QSqlDatabase & db) +{ + Folder folder; + + QSqlQuery query(db); + query.prepare("SELECT * FROM folder WHERE id = :id"); + query.bindValue(":id",id); + query.exec(); + folder.id = id; + folder.parentId = 0; + if(query.next()) + { + QSqlRecord record = query.record(); + folder.parentId = record.value("parentId").toULongLong(); + folder.name = record.value("name").toString(); + folder.path = record.value("path").toString(); + //new 7.1 + folder.setFinished(record.value("finished").toBool()); + folder.setCompleted(record.value("completed").toBool()); + } + + return folder; +} + +ComicDB DBHelper::loadComic(qulonglong id, QSqlDatabase & db) +{ + ComicDB comic; + + QSqlQuery selectQuery(db); + selectQuery.prepare("select c.id,c.parentId,c.fileName,c.path,ci.hash from comic c inner join comic_info ci on (c.comicInfoId = ci.id) where c.id = :id"); + selectQuery.bindValue(":id", id); + selectQuery.exec(); + comic.id = id; + if(selectQuery.next()) + { + QSqlRecord record = selectQuery.record(); + //id = record.value("id").toULongLong(); + comic.parentId = record.value("parentId").toULongLong(); + comic.name = record.value("name").toString(); + comic.path = record.value("path").toString(); + comic.info = DBHelper::loadComicInfo(record.value("hash").toString(),db); + } + + return comic; +} + +ComicDB DBHelper::loadComic(QString cname, QString cpath, QString chash, QSqlDatabase & database) +{ + ComicDB comic; + + //comic.parentId = cparentId; + comic.name = cname; + comic.path = cpath; + + comic.info = DBHelper::loadComicInfo(chash,database); + + if(!comic.info.existOnDb) + { + comic.info.hash = chash; + comic.info.coverPage = 1; + comic._hasCover = false; + } + else + comic._hasCover = true; + + return comic; +} + +ComicInfo DBHelper::loadComicInfo(QString hash, QSqlDatabase & db) +{ + ComicInfo comicInfo; + + QSqlQuery findComicInfo(db); + findComicInfo.prepare("SELECT * FROM comic_info WHERE hash = :hash"); + findComicInfo.bindValue(":hash", hash); + findComicInfo.exec(); + + + if(findComicInfo.next()) + { + comicInfo.hash = hash; + QSqlRecord record = findComicInfo.record(); + + comicInfo.hash = hash; + comicInfo.id = record.value("id").toULongLong(); + comicInfo.read = record.value("read").toBool(); + comicInfo.edited = record.value("edited").toBool(); + + //new 7.0 fields + comicInfo.hasBeenOpened = record.value("hasBeenOpened").toBool(); + comicInfo.currentPage = record.value("currentPage").toInt(); + comicInfo.bookmark1 = record.value("bookmark1").toInt(); + comicInfo.bookmark2 = record.value("bookmark2").toInt(); + comicInfo.bookmark3 = record.value("bookmark3").toInt(); + comicInfo.brightness = record.value("brightness").toInt(); + comicInfo.contrast = record.value("contrast").toInt(); + comicInfo.gamma = record.value("gamma").toInt(); + comicInfo.rating = record.value("rating").toInt(); + //-- + + comicInfo.title = record.value("title"); + comicInfo.numPages = record.value("numPages"); + + comicInfo.coverPage = record.value("coverPage"); + + comicInfo.number = record.value("number"); + comicInfo.isBis = record.value("isBis"); + comicInfo.count = record.value("count"); + + comicInfo.volume = record.value("volume"); + comicInfo.storyArc = record.value("storyArc"); + comicInfo.arcNumber = record.value("arcNumber"); + comicInfo.arcCount = record.value("arcCount"); + + comicInfo.genere = record.value("genere"); + + comicInfo.writer = record.value("writer"); + comicInfo.penciller = record.value("penciller"); + comicInfo.inker = record.value("inker"); + comicInfo.colorist = record.value("colorist"); + comicInfo.letterer = record.value("letterer"); + comicInfo.coverArtist = record.value("coverArtist"); + + comicInfo.date = record.value("date"); + comicInfo.publisher = record.value("publisher"); + comicInfo.format = record.value("format"); + comicInfo.color = record.value("color"); + comicInfo.ageRating = record.value("ageRating"); + + comicInfo.synopsis = record.value("synopsis"); + comicInfo.characters = record.value("characters"); + comicInfo.notes = record.value("notes"); + + comicInfo.comicVineID = record.value("comicVineID"); + + comicInfo.existOnDb = true; + } + else + comicInfo.existOnDb = false; + + return comicInfo; +} + +QList DBHelper::loadSubfoldersNames(qulonglong folderId, QSqlDatabase &db) +{ + QList result; + QSqlQuery selectQuery(db); + selectQuery.prepare("SELECT name FROM folder WHERE parentId = :parentId AND id <> 1"); //do not select the root folder + selectQuery.bindValue(":parentId", folderId); + selectQuery.exec(); + while(selectQuery.next()){ + result << selectQuery.record().value("name").toString(); + } + return result; +} diff --git a/YACReaderLibrary/db_helper.h b/YACReaderLibrary/db_helper.h new file mode 100644 index 00000000..146d60d5 --- /dev/null +++ b/YACReaderLibrary/db_helper.h @@ -0,0 +1,58 @@ +#ifndef DB_HELPER_H +#define DB_HELPER_H + +class QString; +#include +#include + +class ComicDB; +class Folder; +class LibraryItem; +class QSqlDatabase; +class ComicInfo; +class QSqlRecord; +class QSqlQuery; +class YACReaderLibraries; + +class DBHelper +{ +public: + //server + static YACReaderLibraries getLibraries(); + static QList getFolderContentFromLibrary(const QString & libraryName, qulonglong folderId); + static QList getFolderComicsFromLibrary(const QString & libraryName, qulonglong folderId); + static qulonglong getParentFromComicFolderId(const QString & libraryName, qulonglong id); + static ComicDB getComicInfo(const QString & libraryName, qulonglong id); + static QList getSiblings(const QString & libraryName, qulonglong parentId); + static QString getFolderName(const QString & libraryName, qulonglong id); + static QList getLibrariesNames(); + static QString getLibraryName(int id); + + //objects management + //deletes + static void removeFromDB(LibraryItem * item, QSqlDatabase & db); + static void removeFromDB(Folder * folder, QSqlDatabase & db); + static void removeFromDB(ComicDB * comic, QSqlDatabase & db); + //inserts + static qulonglong insert(Folder * folder, QSqlDatabase & db); + static qulonglong insert(ComicDB * comic, QSqlDatabase & db); + //updates + static void update(const QString & libraryName, ComicInfo & comicInfo); + static void update(ComicDB * comics, QSqlDatabase & db); + static void update(ComicInfo * comicInfo, QSqlDatabase & db); + static void updateRead(ComicInfo * comicInfo, QSqlDatabase & db); + static void update(const Folder & folder, QSqlDatabase & db); + static void updateProgress(qulonglong libraryId,const ComicInfo & comicInfo); //TODO change libraryName by libraryId in all methods. + //queries + static QList getFoldersFromParent(qulonglong parentId, QSqlDatabase & db, bool sort = true); + static QList getSortedComicsFromParent(qulonglong parentId, QSqlDatabase & db); + static QList getComicsFromParent(qulonglong parentId, QSqlDatabase & db, bool sort = true); + //load + static Folder loadFolder(qulonglong id, QSqlDatabase & db); + static ComicDB loadComic(qulonglong id, QSqlDatabase & db); + static ComicDB loadComic(QString cname, QString cpath, QString chash, QSqlDatabase & database); + static ComicInfo loadComicInfo(QString hash, QSqlDatabase & db); + static QList loadSubfoldersNames(qulonglong folderId, QSqlDatabase & db); +}; + +#endif diff --git a/YACReaderLibrary/empty_folder_widget.cpp b/YACReaderLibrary/empty_folder_widget.cpp new file mode 100644 index 00000000..b05eddd1 --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.cpp @@ -0,0 +1,89 @@ +#include "empty_folder_widget.h" + +#include +#include +#include +#include + + +#include +void testListView(QListView * l) +{ + QStringListModel * slm = new QStringListModel(QStringList() << "Lorem ipsum" << "Hailer skualer"<< "Mumbaluba X" << "Finger layden" << "Pacum tactus filer" << "Aposum" << "En" << "Lorem ipsum" << "Hailer skualer" << "Mumbaluba X" << "Finger layden" << "Pacum tactus filer" << "Aposum" << "En" ); + l->setModel(slm); +} + +EmptyFolderWidget::EmptyFolderWidget(QWidget *parent) : + QWidget(parent),subfoldersModel(new QStringListModel()) +{ + QVBoxLayout * layout = new QVBoxLayout; + + iconLabel = new QLabel(); + iconLabel->setPixmap(QPixmap(":/images/empty_folder.png")); + iconLabel->setAlignment(Qt::AlignCenter); + + titleLabel = new QLabel("Subfolders in this folder"); + titleLabel->setAlignment(Qt::AlignCenter); + titleLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); + + foldersView = new QListView(); + foldersView->setMinimumWidth(282); + foldersView->setWrapping(true); + + foldersView->setStyleSheet("QListView {background-color:transparent; border: none; color:#858585; outline:0; font-size: 18px; font:bold; show-decoration-selected: 0; margin:0}" + "QListView::item:selected {background-color: #212121; color:#CCCCCC;}" + "QListView::item:hover {background-color:#212121; color:#CCCCCC; }" + + + "QScrollBar:vertical { border: none; background: #212121; width: 14px; margin: 0 10px 0 0; }" + "QScrollBar::handle:vertical { background: #858585; width: 14px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #212121; height: 0px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #212121; height: 0px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical {background: none; }" + "QScrollBar:horizontal{height:0px;}" + ); + + foldersView->setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + testListView(foldersView); + + layout->addSpacing(100); + layout->addWidget(iconLabel); + layout->addSpacing(30); + layout->addWidget(titleLabel); + layout->addSpacing(12); + layout->addWidget(foldersView,1,Qt::AlignHCenter); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet("QWidget {background:#2A2A2A}"); + + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + setLayout(layout); + + connect(foldersView,SIGNAL(clicked(QModelIndex)),this,SLOT(onItemClicked(QModelIndex))); +} + +void EmptyFolderWidget::setSubfolders(const QModelIndex &mi, const QStringList &foldersNames) +{ + parent = mi; + subfoldersModel->setStringList(foldersNames); + foldersView->setModel(subfoldersModel); +} + +void EmptyFolderWidget::onItemClicked(const QModelIndex &mi) +{ + emit subfolderSelected(parent,mi.row()); +} + +void EmptyFolderWidget::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor("#2A2A2A")); +} diff --git a/YACReaderLibrary/empty_folder_widget.h b/YACReaderLibrary/empty_folder_widget.h new file mode 100644 index 00000000..225948be --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.h @@ -0,0 +1,32 @@ +#ifndef EMPTY_FOLDER_WIDGET_H +#define EMPTY_FOLDER_WIDGET_H + +#include +#include + +class QLabel; +class QListView; +class QStringListModel; + +class EmptyFolderWidget : public QWidget +{ + Q_OBJECT +public: + explicit EmptyFolderWidget(QWidget *parent = 0); + void setSubfolders(const QModelIndex & mi, const QStringList & foldersNames); +signals: + void subfolderSelected(QModelIndex, int); + +public slots: + void onItemClicked(const QModelIndex & mi); + +protected: + QLabel * iconLabel; + QLabel * titleLabel; + QListView * foldersView; + QModelIndex parent; + QStringListModel * subfoldersModel; + void paintEvent(QPaintEvent *); +}; + +#endif // EMPTY_FOLDER_WIDGET_H diff --git a/YACReaderLibrary/export_comics_info_dialog.cpp b/YACReaderLibrary/export_comics_info_dialog.cpp new file mode 100644 index 00000000..3fb32267 --- /dev/null +++ b/YACReaderLibrary/export_comics_info_dialog.cpp @@ -0,0 +1,92 @@ +#include "export_comics_info_dialog.h" + +#include +#include +#include +#include +#include + +#include "data_base_management.h" + +ExportComicsInfoDialog::ExportComicsInfoDialog(QWidget *parent) + : QDialog(parent) +{ + textLabel = new QLabel(tr("Output file : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + + accept = new QPushButton(tr("Create")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(exportComicsInfo())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + connect(cancel,SIGNAL(clicked()),this,SIGNAL(rejected())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + QHBoxLayout *libraryLayout = new QHBoxLayout; + + libraryLayout->addWidget(textLabel); + libraryLayout->addWidget(path); + libraryLayout->addWidget(find); + libraryLayout->setStretchFactor(find,0); //TODO + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(libraryLayout); + mainLayout->addWidget(progress=new QLabel()); + mainLayout->addStretch(); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/exportComicsInfo.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Export comics info")); +} + +ExportComicsInfoDialog::~ExportComicsInfoDialog() +{ + +} + +void ExportComicsInfoDialog::findPath() +{ + QString s = QFileDialog::getSaveFileName(this,tr("Destination database name"),".","*.ydb"); + if(!s.isEmpty()) + { + path->setText(s); + accept->setEnabled(true); + } +} + +void ExportComicsInfoDialog::exportComicsInfo() +{ + QFileInfo f(path->text()); + QFileInfo fPath(f.absoluteDir().path()); + if(fPath.exists() && fPath.isDir() && fPath.isWritable()) + { + DataBaseManagement::exportComicsInfo(source,path->text()); + close(); + } + else + QMessageBox::critical(NULL,tr("Problem found while writing"),tr("The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder")); +} + +void ExportComicsInfoDialog::close() +{ + path->clear(); + QDialog::close(); +} diff --git a/YACReaderLibrary/export_comics_info_dialog.h b/YACReaderLibrary/export_comics_info_dialog.h new file mode 100644 index 00000000..07e3bf0d --- /dev/null +++ b/YACReaderLibrary/export_comics_info_dialog.h @@ -0,0 +1,35 @@ +#ifndef EXPORT_COMICS_INFO_DIALOG_H +#define EXPORT_COMICS_INFO_DIALOG_H + +#include +#include +#include +#include + + +class ExportComicsInfoDialog : public QDialog +{ + Q_OBJECT + +public: + ExportComicsInfoDialog(QWidget *parent = 0); + ~ExportComicsInfoDialog(); + QString source; + +public slots: + void findPath(); + void exportComicsInfo(); + void close(); + +private: + QLabel * progress; + QLabel * textLabel; + QLineEdit * path; + QPushButton * find; + QPushButton * accept; + QPushButton * cancel; + + +}; + +#endif // EXPORT_COMICS_INFO_DIALOG_H diff --git a/YACReaderLibrary/export_library_dialog.cpp b/YACReaderLibrary/export_library_dialog.cpp new file mode 100644 index 00000000..0d20fd2f --- /dev/null +++ b/YACReaderLibrary/export_library_dialog.cpp @@ -0,0 +1,100 @@ +#include "export_library_dialog.h" +#include +#include +#include +#include +#include + +ExportLibraryDialog::ExportLibraryDialog(QWidget * parent) +:QDialog(parent),progressCount(0) +{ + textLabel = new QLabel(tr("Output folder : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + + accept = new QPushButton(tr("Create")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(exportLibrary())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + connect(cancel,SIGNAL(clicked()),this,SIGNAL(rejected())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + QHBoxLayout *libraryLayout = new QHBoxLayout; + + libraryLayout->addWidget(textLabel); + libraryLayout->addWidget(path); + libraryLayout->addWidget(find); + libraryLayout->setStretchFactor(find,0); //TODO + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + progressBar = new QProgressBar(this); + progressBar->setMinimum(0); + progressBar->setMaximum(0); + progressBar->setTextVisible(false); + progressBar->hide(); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(libraryLayout); + mainLayout->addStretch(); + mainLayout->addWidget(progressBar); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/exportLibrary.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Create covers package")); +} + +void ExportLibraryDialog::exportLibrary() +{ + QFileInfo f(path->text()); + if(f.exists() && f.isDir() && f.isWritable()) + { + progressBar->show(); + accept->setEnabled(false); + emit exportPath(QDir::cleanPath(path->text())); + } + else + QMessageBox::critical(NULL,tr("Problem found while writing"),tr("The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder")); + +} + +void ExportLibraryDialog::findPath() +{ + QString s = QFileDialog::getExistingDirectory(0,tr("Destination directory"),"."); + if(!s.isEmpty()) + { + path->setText(s); + accept->setEnabled(true); + } +} + +void ExportLibraryDialog::close() +{ + path->clear(); + progressBar->hide(); + accept->setEnabled(false); + progressCount=0; + QDialog::close(); +} + +void ExportLibraryDialog::run() +{ + +} + diff --git a/YACReaderLibrary/export_library_dialog.h b/YACReaderLibrary/export_library_dialog.h new file mode 100644 index 00000000..86bd71b0 --- /dev/null +++ b/YACReaderLibrary/export_library_dialog.h @@ -0,0 +1,35 @@ +#ifndef EXPORT_LIBRARY_DIALOG_H +#define EXPORT_LIBRARY_DIALOG_H + +#include +#include +#include +#include +#include +#include +#include +#include + +class ExportLibraryDialog : public QDialog +{ + Q_OBJECT +public: + ExportLibraryDialog(QWidget * parent = 0); +public slots: + void exportLibrary(); + void findPath(); + void close(); +private: + int progressCount; + QProgressBar *progressBar; + QLabel * textLabel; + QLineEdit * path; + QPushButton * find; + QPushButton * accept; + QPushButton * cancel; + void run(); +signals: + void exportPath(QString); +}; + +#endif diff --git a/YACReaderLibrary/files.qrc b/YACReaderLibrary/files.qrc new file mode 100644 index 00000000..d436db6d --- /dev/null +++ b/YACReaderLibrary/files.qrc @@ -0,0 +1,12 @@ + + + ../files/about.html + ../files/helpYACReaderLibrary.html + + + + ../files/about_es_ES.html + ../files/helpYACReaderLibrary_es_ES.html + + + diff --git a/YACReaderLibrary/grid_comics_view.cpp b/YACReaderLibrary/grid_comics_view.cpp new file mode 100644 index 00000000..257c2d63 --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -0,0 +1,234 @@ +#include "grid_comics_view.h" + +#include +#include + +#include "QsLog.h" + +GridComicsView::GridComicsView(QWidget *parent) : + ComicsView(parent),_selectionModel(NULL) +{ + qmlRegisterType("comicModel",1,0,"TableModel"); + + view = new QQuickView(); + container = QWidget::createWindowContainer(view, this); + + container->setMinimumSize(200, 200); + container->setFocusPolicy(Qt::TabFocus); + view->setSource(QUrl("qrc:/qml/GridComicsView.qml")); + + setShowMarks(true);//TODO save this in settings + + QVBoxLayout * l = new QVBoxLayout; + l->addWidget(container); + this->setLayout(l); + + setContentsMargins(0,0,0,0); + l->setContentsMargins(0,0,0,0); + l->setSpacing(0); + + QLOG_INFO() << "GridComicsView"; +} + +GridComicsView::~GridComicsView() +{ + delete view; +} + +void GridComicsView::setToolBar(QToolBar *toolBar) +{ +QLOG_INFO() << "setToolBar"; +static_cast(this->layout())->insertWidget(1,toolBar); +} + +void GridComicsView::setModel(TableModel *model) +{ + QLOG_INFO() << "setModel"; + + QQmlContext *ctxt = view->rootContext(); + + //there is only one mothel in the system + ComicsView::setModel(model); + if(this->model != NULL) + { + QLOG_INFO() << "xxx"; + + if(_selectionModel != NULL) + delete _selectionModel; + _selectionModel = new QItemSelectionModel(this->model); + + ctxt->setContextProperty("comicsList", this->model); + ctxt->setContextProperty("comicsSelection", _selectionModel); + ctxt->setContextProperty("comicsSelectionHelper", this); + ctxt->setContextProperty("dummyValue", true); + } + +#ifdef Q_OS_MAC + ctxt->setContextProperty("backgroundColor", "#EDEDED"); + ctxt->setContextProperty("cellColor", "#FFFFFF"); + ctxt->setContextProperty("selectedColor", "#DDDDDD"); + ctxt->setContextProperty("titleColor", "#121212"); + ctxt->setContextProperty("textColor", "#636363"); + ctxt->setContextProperty("dropShadow",true); +#else + ctxt->setContextProperty("backgroundColor", "#2A2A2A"); + ctxt->setContextProperty("cellColor", "#212121"); + ctxt->setContextProperty("selectedColor", "#121212"); + ctxt->setContextProperty("titleColor", "#E6E6E6"); + ctxt->setContextProperty("textColor", "#E6E6E6"); + ctxt->setContextProperty("dropShadow",false); +#endif + + +} + +void GridComicsView::setCurrentIndex(const QModelIndex &index) +{ + QLOG_INFO() << "setCurrentIndex"; +} + +QModelIndex GridComicsView::currentIndex() +{ + QLOG_INFO() << "currentIndex"; + QModelIndexList indexes = _selectionModel->selectedRows(); + if(indexes.length()>0) + return indexes[0]; + + this->selectIndex(0); + return _selectionModel->selectedRows()[0]; +} + +QItemSelectionModel *GridComicsView::selectionModel() +{ + QLOG_INFO() << "selectionModel"; + QModelIndexList indexes = _selectionModel->selectedRows(); + if(indexes.length()==0) + this->selectIndex(0); + + return _selectionModel; +} + +void GridComicsView::scrollTo(const QModelIndex &mi, QAbstractItemView::ScrollHint hint) +{ + QLOG_INFO() << "scrollTo"; +} + +void GridComicsView::toFullScreen() +{ + QLOG_INFO() << "toFullScreen"; +} + +void GridComicsView::toNormal() +{ + QLOG_INFO() << "toNormal"; +} + +void GridComicsView::updateConfig(QSettings *settings) +{ + QLOG_INFO() << "updateConfig"; +} + +void GridComicsView::setItemActions(const QList &actions) +{ + QLOG_INFO() << "setItemActions"; +} + +void GridComicsView::setViewActions(const QList &actions) +{ + //TODO generate QML Menu from actions + QLOG_INFO() << "setViewActions"; + this->addActions(actions); + + //TODO this is completely unsafe, but QActions can't be used directly in QML + if(actions.length()>17) + { + QQmlContext *ctxt = view->rootContext(); + + ctxt->setContextProperty("openComicAction",actions[0]); + + ctxt->setContextProperty("openContainingFolderComicAction",actions[2]); + + ctxt->setContextProperty("resetComicRatingAction",actions[4]); + + ctxt->setContextProperty("editSelectedComicsAction",actions[6]); + ctxt->setContextProperty("getInfoAction",actions[7]); + ctxt->setContextProperty("asignOrderAction",actions[8]); + + ctxt->setContextProperty("selectAllComicsAction",actions[10]); + + ctxt->setContextProperty("setAsReadAction",actions[12]); + ctxt->setContextProperty("setAsNonReadAction",actions[13]); + ctxt->setContextProperty("showHideMarksAction",actions[14]); + + ctxt->setContextProperty("deleteComicsAction",actions[16]); + + ctxt->setContextProperty("toggleFullScreenAction",actions[18]); + } + else + QLOG_ERROR() << "setViewActions invoked with the wrong number of actions"; +} + +void GridComicsView::selectAll() +{ + QLOG_INFO() << "selectAll"; +} + +QSize GridComicsView::sizeHint() +{ + QLOG_INFO() << "sizeHint"; + return QSize(1280,768); +} + +//helper +void GridComicsView::selectIndex(int index) +{ + QLOG_INFO() << "selectIndex" << index; + if(_selectionModel != NULL && model!=NULL) + _selectionModel->select(model->index(index,0),QItemSelectionModel::Select | QItemSelectionModel::Rows); +} + +bool GridComicsView::isSelectedIndex(int index) +{ + if(_selectionModel != NULL && model!=NULL) + { + QModelIndex mi = model->index(index,0); + return _selectionModel->isSelected(mi); + } + return false; +} + +void GridComicsView::clear() +{ + QLOG_INFO() << "clear"; + if(_selectionModel != NULL) + { + _selectionModel->clear(); + + QQmlContext *ctxt = view->rootContext(); + ctxt->setContextProperty("dummyValue", true); + } + //model->forceClear(); +} + +void GridComicsView::selectedItem(int index) +{ + emit doubleClicked(model->index(index,0)); +} + +void GridComicsView::setShowMarks(bool show) +{ + QLOG_INFO() << "setShowMarks"; + QQmlContext *ctxt = view->rootContext(); + ctxt->setContextProperty("show_marks", show); +} + +void GridComicsView::closeEvent(QCloseEvent *event) +{ + QLOG_INFO() << "closeEvent"; + QObject *object = view->rootObject(); + QMetaObject::invokeMethod(object, "exit"); + container->close(); + view->close(); + event->accept(); + ComicsView::closeEvent(event); +} diff --git a/YACReaderLibrary/grid_comics_view.h b/YACReaderLibrary/grid_comics_view.h new file mode 100644 index 00000000..3066ff68 --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.h @@ -0,0 +1,59 @@ +#ifndef GRID_COMICS_VIEW_H +#define GRID_COMICS_VIEW_H + +#include "comics_view.h" + +#include + +class QAbstractListModel; +class QItemSelectionModel; +class QQuickView; +class QQuickView; + + +class GridComicsView : public ComicsView +{ + Q_OBJECT +public: + explicit GridComicsView(QWidget *parent = 0); + virtual ~GridComicsView(); + void setToolBar(QToolBar * toolBar); + void setModel(TableModel *model); + void setCurrentIndex(const QModelIndex &index); + QModelIndex currentIndex(); + QItemSelectionModel * selectionModel(); + void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ); + void toFullScreen(); + void toNormal(); + void updateConfig(QSettings * settings); + void setItemActions(const QList & actions); + void setViewActions(const QList & actions); + + QSize sizeHint(); +signals: +signals: + void comicRated(int,QModelIndex); + void doubleClicked(QModelIndex); + +public slots: + //selection helper + void selectIndex(int index); + bool isSelectedIndex(int index); + void clear(); + //double clicked item + void selectedItem(int index); + + //ComicsView + void setShowMarks(bool show); + void selectAll(); + +private: + QItemSelectionModel * _selectionModel; + QQuickView *view; + QWidget *container; + bool dummy; + void closeEvent ( QCloseEvent * event ); + +}; + +#endif // GRID_COMICS_VIEW_H diff --git a/YACReaderLibrary/icon.ico b/YACReaderLibrary/icon.ico new file mode 100644 index 0000000000000000000000000000000000000000..b2232648330a45c5639a6c13790191bbf0dd1a9a GIT binary patch literal 99678 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 bSt|Z2dZ3~QDte%z2P%4?q6hw$df@*7sMu_a literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/icon.rc b/YACReaderLibrary/icon.rc new file mode 100644 index 00000000..ef707930 --- /dev/null +++ b/YACReaderLibrary/icon.rc @@ -0,0 +1,3 @@ +IDI_ICON1 ICON DISCARDABLE "icon.ico" +IDI_ICON2 ICON DISCARDABLE "icon2.ico" +IDI_ICON3 ICON DISCARDABLE "icon3.ico" \ No newline at end of file diff --git a/YACReaderLibrary/icon2.ico b/YACReaderLibrary/icon2.ico new file mode 100644 index 0000000000000000000000000000000000000000..e87778428481086fb0b280ade6be220b1425b6a4 GIT binary patch literal 99678 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 N`wG}s;3->y{{v~kuC)LF literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/icon3.ico b/YACReaderLibrary/icon3.ico new file mode 100644 index 0000000000000000000000000000000000000000..bf3802dc518b0517de144e246a69de1d87559535 GIT binary patch literal 82726 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 literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/images.qrc b/YACReaderLibrary/images.qrc new file mode 100644 index 00000000..e444a4c2 --- /dev/null +++ b/YACReaderLibrary/images.qrc @@ -0,0 +1,117 @@ + + + ../images/folder.png + ../images/folder_finished.png + ../images/icon.png + ../images/iconLibrary.png + ../images/new.png + ../images/openLibrary.png + ../images/removeLibraryIcon.png + ../images/updateLibraryIcon.png + ../images/comicFolder.png + ../images/notCover.png + ../images/edit.png + ../images/editIcon.png + ../images/flow1.png + ../images/flow2.png + ../images/flow3.png + ../images/flow4.png + ../images/flow5.png + ../images/importLibrary.png + ../images/importLibraryIcon.png + ../images/exportLibrary.png + ../images/exportLibraryIcon.png + ../images/importLibraryIcon.png + ../images/open.png + ../images/coversPackage.png + ../images/setRead.png + + ../images/setUnread.png + + ../images/showMarks.png + ../images/editComic.png + ../images/selectAll.png + ../images/hideComicFlow.png + ../images/exportComicsInfo.png + ../images/importComicsInfo.png + ../images/exportComicsInfoIcon.png + ../images/importComicsInfoIcon.png + ../images/db.png + ../images/asignNumber.png + ../images/defaultCover.png + ../images/iphoneConfig.png + ../images/onStartFlowSelection.png + ../images/onStartFlowSelection_es.png + ../images/useNewFlowButton.png + ../images/useOldFlowButton.png + ../images/serverConfigBackground.png + ../images/noLibrariesIcon.png + ../images/noLibrariesLine.png + ../images/importingIcon.png + ../images/updatingIcon.png + ../images/importTopCoversDecoration.png + ../images/importBottomCoversDecoration.png + ../images/glowLine.png + ../images/clearSearch.png + ../images/iconSearch.png + ../images/readRibbon.png + ../images/readingRibbon.png + ../images/shownCovers.png + ../images/hiddenCovers.png + ../images/trash.png + ../images/setReadButton.png + ../images/openInYACReader.png + + ../images/main_toolbar/divider.png + ../images/collapsed_branch_osx.png + ../images/expanded_branch_osx.png + ../images/folder_macosx.png + ../images/libraryIconSelected.png + ../images/libraryOptions.png + ../images/branch-open.png + ../images/branch-closed.png + ../images/expanded_branch_selected.png + ../images/collapsed_branch_selected.png + ../images/previousCoverPage.png + ../images/nextCoverPage.png + ../images/getInfo.png + ../images/comic_vine/radioChecked.png + ../images/comic_vine/radioUnchecked.png + ../images/comic_vine/radioUnchecked.png + ../images/comic_vine/rowDown.png + ../images/comic_vine/rowUp.png + ../images/comic_vine/previousPage.png + ../images/comic_vine/nextPage.png + ../images/comic_vine/downArrow.png + ../images/comic_vine/upArrow.png + ../images/find_folder.png + ../images/clear_shortcut.png + ../images/accept_shortcut.png + ../images/shortcuts_group_comics.png + ../images/shortcuts_group_folders.png + ../images/shortcuts_group_general.png + ../images/shortcuts_group_libraries.png + ../images/shortcuts_group_mglass.png + ../images/shortcuts_group_page.png + ../images/shortcuts_group_reading.png + ../images/shortcuts_group_visualization.png + ../images/f.png + ../images/f_retina.png + + + diff --git a/YACReaderLibrary/images_osx.qrc b/YACReaderLibrary/images_osx.qrc new file mode 100644 index 00000000..64c41b0c --- /dev/null +++ b/YACReaderLibrary/images_osx.qrc @@ -0,0 +1,24 @@ + + + ../images/folder_finished_macosx.png + ../images/main_toolbar/back_osx.png + + ../images/main_toolbar/back_disabled_osx.png + ../images/main_toolbar/forward_osx.png + ../images/main_toolbar/forward_disabled_osx.png + ../images/main_toolbar/settings_osx.png + ../images/main_toolbar/server_osx.png + ../images/main_toolbar/help_osx.png + ../images/main_toolbar/fullscreen_osx.png + + ../images/libraryIcon_osx.png + ../images/setRoot_osx.png + ../images/expand_osx.png + ../images/colapse_osx.png + ../images/newLibraryIcon_osx.png + ../images/openLibraryIcon_osx.png + ../images/flow_to_grid.gif + ../images/grid_to_flow.gif + ../images/empty_folder.png + + diff --git a/YACReaderLibrary/images_win.qrc b/YACReaderLibrary/images_win.qrc new file mode 100644 index 00000000..ffa97d93 --- /dev/null +++ b/YACReaderLibrary/images_win.qrc @@ -0,0 +1,23 @@ + + + ../images/main_toolbar/back.png + ../images/main_toolbar/back_disabled.png + ../images/main_toolbar/forward.png + ../images/main_toolbar/forward_disabled.png + ../images/main_toolbar/settings.png + ../images/main_toolbar/server.png + ../images/main_toolbar/help.png + ../images/main_toolbar/fullscreen.png + ../images/libraryIcon.png + ../images/setRoot.png + ../images/expand.png + ../images/colapse.png + ../images/newLibraryIcon.png + ../images/openLibraryIcon.png + ../images/main_toolbar/flow.png + ../images/main_toolbar/grid.png + ../images/flow_to_grid.gif + ../images/grid_to_flow.gif + ../images/empty_folder.png + + diff --git a/YACReaderLibrary/import_comics_info_dialog.cpp b/YACReaderLibrary/import_comics_info_dialog.cpp new file mode 100644 index 00000000..e82ec33d --- /dev/null +++ b/YACReaderLibrary/import_comics_info_dialog.cpp @@ -0,0 +1,111 @@ +#include "import_comics_info_dialog.h" + +#include +#include +#include +#include + +#include "data_base_management.h" + +ImportComicsInfoDialog::ImportComicsInfoDialog(QWidget *parent) + : QDialog(parent) +{ + setModal(true); + setWindowTitle(tr("Import comics info")); + + + textLabel = new QLabel(tr("Info database location : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + + accept = new QPushButton(tr("Import")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(import())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + //connect(cancel,SIGNAL(clicked()),this,SIGNAL(rejected())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + QHBoxLayout *libraryLayout = new QHBoxLayout; + + libraryLayout->addWidget(textLabel); + libraryLayout->addWidget(path); + libraryLayout->addWidget(find); + libraryLayout->setStretchFactor(find,0); //TODO + + progressBar = new QProgressBar(this); + progressBar->setMinimum(0); + progressBar->setMaximum(0); + progressBar->setTextVisible(false); + progressBar->hide(); + connect(accept,SIGNAL(clicked()),progressBar,SLOT(show())); + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(libraryLayout); + mainLayout->addStretch(); + mainLayout->addWidget(progressBar); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/importComicsInfo.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); +} + +ImportComicsInfoDialog::~ImportComicsInfoDialog() +{ + +} + + +void ImportComicsInfoDialog::findPath() +{ + QString s = QFileDialog::getOpenFileName(0,"Comics Info",".",tr("Comics info file (*.ydb)")); + if(!s.isEmpty()) + { + path->setText(s); + accept->setEnabled(true); + } +} + +void ImportComicsInfoDialog::import() +{ + progressBar->show(); + + Importer * importer = new Importer(); + importer->source = path->text(); + importer->dest = dest; + connect(importer,SIGNAL(finished()),this,SLOT(close())); + connect(importer,SIGNAL(finished()),this,SLOT(hide())); + importer->start(); +} + +void ImportComicsInfoDialog::close() +{ + path->clear(); + progressBar->hide(); + accept->setDisabled(true); + QDialog::close(); + emit(finished(0)); +} + +void Importer::run() +{ + DataBaseManagement::importComicsInfo(source,dest); +} + + diff --git a/YACReaderLibrary/import_comics_info_dialog.h b/YACReaderLibrary/import_comics_info_dialog.h new file mode 100644 index 00000000..edc5e85e --- /dev/null +++ b/YACReaderLibrary/import_comics_info_dialog.h @@ -0,0 +1,52 @@ +#ifndef IMPORT_COMICS_INFO_DIALOG_H +#define IMPORT_COMICS_INFO_DIALOG_H + +#include +#include +#include +#include +#include +#include +#include + +class Importer : public QThread +{ +public: + QString source; + QString dest; +private: + void run(); +}; + +class ImportComicsInfoDialog : public QDialog +{ + Q_OBJECT + +public: + ImportComicsInfoDialog(QWidget *parent = 0); + ~ImportComicsInfoDialog(); + QString dest; + +private: + QLabel * nameLabel; + QLabel * textLabel; + QLabel * destLabel; + QLineEdit * path; + QLineEdit * destPath; + QLineEdit * nameEdit; + QPushButton * find; + QPushButton * findDest; + QPushButton * accept; + QPushButton * cancel; + QLabel * progress; + void setupUI(); + int progressCount; + QProgressBar *progressBar; + +public slots: + void findPath(); + void import(); + void close(); +}; + +#endif // IMPORT_COMICS_INFO_DIALOG_H diff --git a/YACReaderLibrary/import_library_dialog.cpp b/YACReaderLibrary/import_library_dialog.cpp new file mode 100644 index 00000000..02e2e6d1 --- /dev/null +++ b/YACReaderLibrary/import_library_dialog.cpp @@ -0,0 +1,157 @@ +#include "import_library_dialog.h" + +#include +#include +#include +#include +#include + +ImportLibraryDialog::ImportLibraryDialog(QWidget * parent) +:QDialog(parent),progressCount(0) +{ + setupUI(); +} + +void ImportLibraryDialog::setupUI() +{ + nameLabel = new QLabel(tr("Library Name : ")); + nameEdit = new QLineEdit; + nameLabel->setBuddy(nameEdit); + connect(nameEdit,SIGNAL(textChanged(QString)),this,SLOT(nameEntered())); + + textLabel = new QLabel(tr("Package location : ")); + path = new QLineEdit; + textLabel->setBuddy(path); + + destLabel = new QLabel(tr("Destination folder : ")); + destPath = new QLineEdit; + textLabel->setBuddy(destPath); + + accept = new QPushButton(tr("Unpack")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(add())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + //connect(cancel,SIGNAL(clicked()),this,SIGNAL(rejected())); + + find = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(find,SIGNAL(clicked()),this,SLOT(findPath())); + + findDest = new QPushButton(QIcon(":/images/find_folder.png"),""); + connect(findDest,SIGNAL(clicked()),this,SLOT(findDestination())); + + QGridLayout * content = new QGridLayout; + + content->addWidget(nameLabel,0,0); + content->addWidget(nameEdit,0,1); + + content->addWidget(textLabel,1,0); + content->addWidget(path,1,1); + content->addWidget(find,1,2); + content->setColumnStretch(2,0); //TODO + + content->addWidget(destLabel,2,0); + content->addWidget(destPath,2,1); + content->addWidget(findDest,2,2); + //destLayout->setStretchFactor(findDest,0); //TODO + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + progressBar = new QProgressBar(this); + progressBar->setMinimum(0); + progressBar->setMaximum(0); + progressBar->setTextVisible(false); + progressBar->hide(); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(content); + //mainLayout->addWidget(progress = new QLabel()); + mainLayout->addStretch(); + mainLayout->addWidget(progressBar); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/importLibrary.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Extract a catalog")); +} +void ImportLibraryDialog::show(const YACReaderLibraries &libs) +{ + libraries = libs; + QDialog::show(); +} + +void ImportLibraryDialog::add() +{ + if(!libraries.contains(nameEdit->text())) + { + accept->setEnabled(false); + progressBar->show(); + emit(unpackCLC(QDir::cleanPath(path->text()),QDir::cleanPath(destPath->text()),nameEdit->text())); + } + else + { + emit(libraryExists(nameEdit->text())); + } +} + +void ImportLibraryDialog::findPath() +{ + QString s = QFileDialog::getOpenFileName(0,"Covers Package",".",tr("Compresed library covers (*.clc)")); + if(!s.isEmpty()) + { + path->setText(s); + if(!destPath->text().isEmpty() && !nameEdit->text().isEmpty()) + accept->setEnabled(true); + } +} + + +void ImportLibraryDialog::findDestination() +{ + QString s = QFileDialog::getExistingDirectory(0,"Folder",".",QFileDialog::ShowDirsOnly); + if(!s.isEmpty()) + { + destPath->setText(s); + if(!path->text().isEmpty() && !nameEdit->text().isEmpty()) + accept->setEnabled(true); + } +} + +void ImportLibraryDialog::nameEntered() +{ + if(!nameEdit->text().isEmpty()) + { + if(!path->text().isEmpty() && !destPath->text().isEmpty()) + accept->setEnabled(true); + } + else + accept->setEnabled(false); +} + +void ImportLibraryDialog::close() +{ + path->clear(); + destPath->clear(); + nameEdit->clear(); + accept->setEnabled(false); + progressBar->hide(); + QDialog::hide(); +} + +void ImportLibraryDialog::closeEvent ( QCloseEvent * e ) +{ + close(); + e->accept(); +} diff --git a/YACReaderLibrary/import_library_dialog.h b/YACReaderLibrary/import_library_dialog.h new file mode 100644 index 00000000..d43efe1b --- /dev/null +++ b/YACReaderLibrary/import_library_dialog.h @@ -0,0 +1,46 @@ +#ifndef IMPORT_LIBRARY_DIALOG_H +#define IMPORT_LIBRARY_DIALOG_H +#include "yacreader_libraries.h" + +#include +#include +#include +#include +#include +#include + + class ImportLibraryDialog : public QDialog + { + Q_OBJECT + public: + ImportLibraryDialog(QWidget * parent = 0); + private: + QLabel * nameLabel; + QLabel * textLabel; + QLabel * destLabel; + QLineEdit * path; + QLineEdit * destPath; + QLineEdit * nameEdit; + QPushButton * find; + QPushButton * findDest; + QPushButton * accept; + QPushButton * cancel; + QProgressBar *progressBar; + void setupUI(); + int progressCount; + void closeEvent ( QCloseEvent * e ); + YACReaderLibraries libraries; + public slots: + void add(); + void findPath(); + void findDestination(); + void close(); + void nameEntered(); + void show(const YACReaderLibraries & libs); + + signals: + void unpackCLC(QString clc,QString targetFolder, QString name); + void libraryExists(const QString & name); + }; + +#endif diff --git a/YACReaderLibrary/import_widget.cpp b/YACReaderLibrary/import_widget.cpp new file mode 100644 index 00000000..f96dfc6a --- /dev/null +++ b/YACReaderLibrary/import_widget.cpp @@ -0,0 +1,385 @@ +#include "import_widget.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +class YACReaderActivityIndicatorWidget : public QWidget +{ +public: + YACReaderActivityIndicatorWidget(QWidget * parent = 0); +public slots: + +private: + QLabel * normal; + QLabel * glow; +}; + +YACReaderActivityIndicatorWidget::YACReaderActivityIndicatorWidget(QWidget * parent) + :QWidget(parent) +{ + QPixmap line(":/images/noLibrariesLine.png"); + QPixmap glowLine(":/images/glowLine.png"); + normal = new QLabel(this); + glow = new QLabel(this); + + normal->setPixmap(line); + glow->setPixmap(glowLine); + + + + QHBoxLayout * layout = new QHBoxLayout(); + + layout->addWidget(normal,0,Qt::AlignVCenter); + + setLayout(layout); + + layout->setMargin(4); + layout->setSpacing(0); + + //setFixedHeight(3); + //resize(579,3); + glow->setGeometry(4,4,glowLine.width(),glowLine.height()); + //normal->setGeometry(0,1,579,1); + + QGraphicsOpacityEffect * effect = new QGraphicsOpacityEffect(); + //effect->setOpacity(1.0); + + + QPropertyAnimation * animation = new QPropertyAnimation(effect,"opacity"); + + animation->setDuration(1000); + animation->setStartValue(1); + animation->setEndValue(0); + //animation->setEasingCurve(QEasingCurve::InQuint); + + QPropertyAnimation * animation2 = new QPropertyAnimation(effect,"opacity"); + + animation2->setDuration(1000); + animation2->setStartValue(0); + animation2->setEndValue(1); + //animation2->setEasingCurve(QEasingCurve::InQuint); + +#ifndef Q_OS_MAC + glow->setGraphicsEffect(effect); +#endif + + connect(animation,SIGNAL(finished()),animation2,SLOT(start())); + connect(animation2,SIGNAL(finished()),animation,SLOT(start())); + + animation->start(); +} + + + + +ImportWidget::ImportWidget(QWidget *parent) : + QWidget(parent) +{ + setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); + + QPalette p(palette()); + p.setColor(QPalette::Background, QColor(250,250,250)); + setAutoFillBackground(true); + setPalette(p); + + QPixmap icon(":/images/importingIcon.png"); + iconLabel = new QLabel(); + iconLabel->setPixmap(icon); + + /*QPixmap line(":/images/noLibrariesLine.png"); + QLabel * lineLabel = new QLabel(); + lineLabel->setPixmap(line);*/ + + YACReaderActivityIndicatorWidget * activityIndicator = new YACReaderActivityIndicatorWidget(); + + text = new QLabel();//""+tr("Importing comics")+""); + text->setStyleSheet("QLabel {font-size:25px;font-weight:bold;}"); + textDescription = new QLabel();//""+tr("Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.")+""); + textDescription->setWordWrap(true); + textDescription->setMaximumWidth(330); + currentComicLabel = new QLabel("..."); + + coversViewContainer = new QWidget(this); + QVBoxLayout * coversViewLayout = new QVBoxLayout; + coversViewContainer->setLayout(coversViewLayout); + coversView = new QGraphicsView(); + //coversView->setViewport(new QGLWidget(QGLFormat(QGL::SampleBuffers))); + coversView->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + coversView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + coversView->setMaximumHeight(300); + coversView->setStyleSheet("QGraphicsView {background-color: #E6E6E6;border:none;}"); + + coversScene = new QGraphicsScene(); + coversScene->setSceneRect(0,0,coversView->width(),coversView->height()); + coversView->setAlignment(Qt::AlignLeft); + coversView->setScene(coversScene); + + + QLabel * topDecorator = new QLabel(); + QLabel * bottomDecorator = new QLabel(); + QPixmap top(":/images/importTopCoversDecoration.png"); + QPixmap bottom(":/images/importBottomCoversDecoration.png"); + topDecorator->setPixmap(top); + bottomDecorator->setPixmap(bottom); + topDecorator->setScaledContents(true); + bottomDecorator->setScaledContents(true); + topDecorator->setFixedHeight(top.height()); + bottomDecorator->setFixedHeight(bottom.height()); + + coversViewLayout->addWidget(topDecorator,0); + coversViewLayout->addWidget(coversView,1); + coversViewLayout->addWidget(bottomDecorator,0); + coversViewLayout->setMargin(0); + coversViewLayout->setSpacing(0); + + QPushButton * stop = new QPushButton(tr("stop")); + stop->setSizePolicy(QSizePolicy::Maximum,QSizePolicy::Maximum); + + QVBoxLayout * layout = new QVBoxLayout(this); + QHBoxLayout * buttonLayout = new QHBoxLayout(); + QHBoxLayout * topLayout = new QHBoxLayout(); + QVBoxLayout * textLayout = new QVBoxLayout(); + + QWidget * topWidget = new QWidget(); + topWidget->setFixedWidth(650); + textLayout->addStretch(); + textLayout->addWidget(text); + textLayout->addSpacing(12); + textLayout->addWidget(textDescription); + textLayout->addStretch(); + + topLayout->addStretch(); + topLayout->addWidget(iconLabel,0,Qt::AlignVCenter); + topLayout->addSpacing(30); + topLayout->addLayout(textLayout,1); + topLayout->addStretch(); + topLayout->setMargin(0); + + topWidget->setLayout(topLayout); + + layout->setAlignment(Qt::AlignHCenter); + + buttonLayout->addSpacing(250); + buttonLayout->addWidget(stop); + buttonLayout->addSpacing(250); + + layout->addSpacing(50); + layout->addWidget(topWidget,0,Qt::AlignHCenter); + layout->addSpacing(20); + layout->addWidget(activityIndicator,0,Qt::AlignHCenter); + layout->addSpacing(10); + layout->addLayout(buttonLayout,0); + layout->addSpacing(10); + layout->addStretch(); + portadasLabel = new QLabel(""+tr("Some of the comics being added...")+""); + + hideButton = new QToolButton(this); + hideButton->setFixedSize(25,18); + hideButton->setStyleSheet("QToolButton {background: url(\":/images/shownCovers.png\"); border:none;}" + " QToolButton:checked {background:url(\":/images/hiddenCovers.png\"); border:none;}"); + hideButton->setCheckable(true); + + connect(hideButton,SIGNAL(toggled(bool)),this,SLOT(showCovers(bool))); + + layout->addWidget(portadasLabel,0,Qt::AlignHCenter); + layout->addWidget(coversViewContainer); + //layout->addStretch(); + layout->addWidget(currentComicLabel,0,Qt::AlignHCenter); + layout->setContentsMargins(0,layout->contentsMargins().top(),0,layout->contentsMargins().bottom()); + + connect(stop,SIGNAL(clicked()),this,SIGNAL(stop())); + //connect(stop,SIGNAL(clicked()),this,SLOT(addCoverTest())); + + previousWidth = 10; + updatingCovers = false; + elapsedTimer = new QElapsedTimer(); +} + +void ImportWidget::newComic(const QString & path, const QString & coverPath) +{ + currentComicLabel->setText(""+path+""); + + if(((elapsedTimer->elapsed()>=1000) || ((previousWidth < coversView->width()) && (elapsedTimer->elapsed()>=500))) && !updatingCovers)//todo elapsed time + { + + QPixmap p(coverPath); + p = p.scaledToHeight(300,Qt::SmoothTransformation); + QGraphicsPixmapItem * item = new QGraphicsPixmapItem(p); + item->setPos(previousWidth,0); + item->setZValue(i/10000.0); + previousWidth += 10 + p.width(); + coversScene->addItem(item); + + elapsedTimer->start(); + if(previousWidth >= coversView->width()+200 && !updatingCovers) + { + updatingCovers = true; + + foreach(QGraphicsItem * itemToRemove, coversScene->items()) + { + QGraphicsPixmapItem * last = dynamic_cast(itemToRemove); + + if((last->pos().x()+last->pixmap().width())<=0) + { + coversScene->removeItem(last); + delete last; + } + //else + // break; + } + + int width = p.width(); + + foreach(QGraphicsItem * itemToMove, coversScene->items()) + { + QTimeLine *timer = new QTimeLine(400); + timer->setFrameRange(0, 24); + timer->setUpdateInterval(17); + + QGraphicsItemAnimation *animation = new QGraphicsItemAnimation; + animation->setItem(itemToMove); + animation->setTimeLine(timer); + + QPointF point = itemToMove->scenePos(); + float step = (width+10)/24.0; + for (int i = 0; i < 24; ++i) + animation->setPosAt(i / 24.0, QPointF(point.x()-((i+1)*step), point.y())); + + timer->start(); + connect(timer,SIGNAL(finished()),timer,SLOT(deleteLater())); + connect(timer,SIGNAL(finished()),animation,SLOT(deleteLater())); + } + + QTimer::singleShot(400,this,SLOT(finishedUpdatingCover())); + + previousWidth -= 10+width; + } + + } +} + +void ImportWidget::finishedUpdatingCover() +{ + updatingCovers = false; +} + +void ImportWidget::newCover(const QPixmap & image) +{ + Q_UNUSED(image) +} +static int i = 1; +static int previousWidth = 10; +static int j = 0; +void ImportWidget::addCoverTest() +{ + QPixmap p(QString("c:/temp/%1.jpg").arg(i)); + p = p.scaledToHeight(300,Qt::SmoothTransformation); + QGraphicsPixmapItem * item = new QGraphicsPixmapItem(p); + item->setPos(previousWidth,0); + item->setZValue(i/10000.0); + previousWidth += 10 + p.width(); + coversScene->addItem(item); + if(previousWidth >= coversView->width()) + { + QGraphicsItem * last = coversScene->items().last(); + int width = p.width(); + if(j>=1) + { + coversScene->removeItem(last); + delete last; + } + else + j++; + + foreach(QGraphicsItem * itemToMove, coversScene->items()) + { + + QTimeLine *timer = new QTimeLine(/*350*/1000); + timer->setFrameRange(0, 60); + + QGraphicsItemAnimation *animation = new QGraphicsItemAnimation; + animation->setItem(itemToMove); + animation->setTimeLine(timer); + + QPointF point = itemToMove->scenePos(); + float step = (width+10)/60.0; + for (int i = 0; i < 60; ++i) + animation->setPosAt(i / 60.0, QPointF(point.x()-((i+1)*step), point.y())); + + timer->start(); + } + previousWidth -= 10+width; + } + + i++; +} + +void ImportWidget::clear() +{ + previousWidth = 10; + + //nos aseguramos de que las animaciones han finalizado antes de borrar + QList all = coversScene->items(); + for (int i = 0; i < all.size(); i++) + { + QGraphicsItem *gi = all[i]; + if(gi->parentItem()==NULL) + delete gi; + } + coversScene->clear(); + + updatingCovers = false; + + currentComicLabel->setText("..."); + + this->i = 0; +} + +void ImportWidget::setImportLook() +{ + iconLabel->setPixmap(QPixmap(":/images/importingIcon.png")); + text->setText(""+tr("Importing comics")+""); + textDescription->setText(""+tr("

YACReaderLibrary is now creating a new library.

Create a library could take several minutes. You can stop the process and update the library later for completing the task.

")+""); +} + +void ImportWidget::setUpdateLook() +{ + iconLabel->setPixmap(QPixmap(":/images/updatingIcon.png")); + text->setText(""+tr("Updating the library")+""); + textDescription->setText(""+tr("

The current library is being updated. For faster updates, please, update your libraries frequently.

You can stop the process and continue updating this library later.

")+"
"); +} + +void ImportWidget::clearScene() +{ + + +} + +void ImportWidget::showCovers(bool hide) +{ + portadasLabel->setHidden(hide); + coversViewContainer->setHidden(hide); +} + +void ImportWidget::resizeEvent(QResizeEvent * event) +{ + hideButton->move(event->size().width()-hideButton->width()- (currentComicLabel->height()/2),event->size().height()-hideButton->height()- (currentComicLabel->height()/2)); + + QWidget::resizeEvent(event); +} \ No newline at end of file diff --git a/YACReaderLibrary/import_widget.h b/YACReaderLibrary/import_widget.h new file mode 100644 index 00000000..5ff57814 --- /dev/null +++ b/YACReaderLibrary/import_widget.h @@ -0,0 +1,52 @@ +#ifndef IMPORT_WIDGET_H +#define IMPORT_WIDGET_H + +#include + +class QLabel; +class QGraphicsView; +class QGraphicsScene; +class QElapsedTimer; +class QVBoxLayout; +class QToolButton; +class QResizeEvent; + +class ImportWidget : public QWidget +{ + Q_OBJECT +public: + explicit ImportWidget(QWidget *parent = 0); + +signals: + void stop(); +public slots: + void newComic(const QString & path, const QString & coverPath); + void newCover(const QPixmap & image); + void clear(); + void addCoverTest(); + void finishedUpdatingCover(); + void clearScene(); + void setImportLook(); + void setUpdateLook(); + void showCovers(bool hide); +private: + QLabel * currentComicLabel; + QLabel * portadasLabel; + QLabel * iconLabel; + QLabel * text; + QLabel * textDescription; + QWidget * coversViewContainer; + QGraphicsView * coversView; + QGraphicsScene * coversScene; + int previousWidth; + bool updatingCovers; + QElapsedTimer * elapsedTimer; + quint64 i; + + QToolButton * hideButton; + + void resizeEvent(QResizeEvent * event); + +}; + +#endif // IMPORT_WIDGET_H diff --git a/YACReaderLibrary/library_creator.cpp b/YACReaderLibrary/library_creator.cpp new file mode 100644 index 00000000..505e83ea --- /dev/null +++ b/YACReaderLibrary/library_creator.cpp @@ -0,0 +1,635 @@ +#include "library_creator.h" +#include "custom_widgets.h" + +#include +#include +#include +#include +#include +#include + +#include "data_base_management.h" +#include "qnaturalsorting.h" +#include "db_helper.h" + +#include "compressed_archive.h" +#include "comic.h" + +#include "yacreader_global.h" + +#include "QsLog.h" + +#include +using namespace std; + +#ifdef Q_OS_MAC + #include "pdf_comic.h" +#else + +#if QT_VERSION >= 0x050000 + #include "poppler-qt5.h" +#else + #include "poppler-qt4.h" +#endif + +#endif + +//-------------------------------------------------------------------------------- +LibraryCreator::LibraryCreator() + :creation(false) +{ + _nameFilter << "*.cbr" << "*.cbz" << "*.rar" << "*.zip" << "*.tar" << "*.pdf" << "*.7z" << "*.cb7" << "*.arj" << "*.cbt"; +} + +void LibraryCreator::createLibrary(const QString &source, const QString &target) +{ + creation = true; + processLibrary(source,target); +} + +void LibraryCreator::updateLibrary(const QString &source, const QString &target) +{ + processLibrary(source,target); +} + +void LibraryCreator::processLibrary(const QString & source, const QString & target) +{ + _source = source; + _target = target; + if(DataBaseManagement::checkValidDB(target+"/library.ydb")=="") + { + //se limpia el directorio ./yacreaderlibrary + delTree(target); + _mode = CREATOR; + } + else // + _mode = UPDATER; +} + + +// +void LibraryCreator::run() +{ + stopRunning = false; + + //check for 7z lib +#if defined Q_OS_UNIX && !defined Q_OS_MAC + QLibrary *sevenzLib = new QLibrary(QString(LIBDIR)+"/p7zip/7z.so"); +#else + QLibrary *sevenzLib = new QLibrary(QApplication::applicationDirPath()+"/utils/7z"); +#endif + if(!sevenzLib->load()) + { + QLOG_ERROR() << "Loading 7z.dll : " + sevenzLib->errorString() << endl; + QApplication::exit(YACReader::SevenZNotFound); + exit(); + } + sevenzLib->deleteLater(); + + if(_mode == CREATOR) + { + QLOG_INFO() << "Starting to create new library ( " << _source << "," << _target << ")"; + _currentPathFolders.clear(); + _currentPathFolders.append(Folder(1,1,"root","/")); + //se crean los directorios .yacreaderlibrary y .yacreaderlibrary/covers + QDir dir; + dir.mkpath(_target+"/covers"); + + //se crea la base de datos .yacreaderlibrary/library.ydb + _database = DataBaseManagement::createDatabase("library",_target);// + if(!_database.isOpen()) + { + QLOG_ERROR() << "Unable to create data base" << _database.lastError().databaseText() + "-" + _database.lastError().driverText(); + emit failedCreatingDB(_database.lastError().databaseText() + "-" + _database.lastError().driverText()); + emit finished(); + creation = false; + return; + } + + /*QSqlQuery pragma("PRAGMA foreign_keys = ON",_database);*/ + _database.transaction(); + //se crea la librería + create(QDir(_source)); + _database.commit(); + _database.close(); + QSqlDatabase::removeDatabase(_database.connectionName()); + emit(created()); + QLOG_INFO() << "Create library END"; + } + else + { + QLOG_INFO() << "Starting to update library ( " << _source << "," << _target << ")"; + _currentPathFolders.clear(); + _currentPathFolders.append(Folder(1,1,"root","/")); + _database = DataBaseManagement::loadDatabase(_target); + //_database.setDatabaseName(_target+"/library.ydb"); + if(!_database.open()) + { + QLOG_ERROR() << "Unable to open data base" << _database.lastError().databaseText() + "-" + _database.lastError().driverText(); + emit failedOpeningDB(_database.lastError().databaseText() + "-" + _database.lastError().driverText()); + emit finished(); + creation = false; + return; + } + QSqlQuery pragma("PRAGMA foreign_keys = ON",_database); + _database.transaction(); + update(QDir(_source)); + _database.commit(); + _database.close(); + QSqlDatabase::removeDatabase(_target); + //si estabamos en modo creación, se está añadiendo una librería que ya existía y se ha actualizado antes de añadirse. + if(!creation) + emit(updated()); + else + emit(created()); + QLOG_INFO() << "Update library END"; + } + msleep(100);//TODO try to solve the problem with the udpate dialog (ya no se usa más...) + emit(finished()); + creation = false; +} + +void LibraryCreator::stop() +{ + _database.commit(); + stopRunning = true; +} + +//retorna el id del ultimo de los folders +qulonglong LibraryCreator::insertFolders() +{ + QList::iterator i; + int currentId = 0; + for (i = _currentPathFolders.begin(); i != _currentPathFolders.end(); ++i) + { + if(!(i->knownId)) + { + i->setFather(currentId); + currentId = DBHelper::insert(&(*i),_database);//insertFolder(currentId,*i); + i->setId(currentId); + } + else + { + currentId = i->id; + } + } + return currentId; +} + +void LibraryCreator::create(QDir dir) +{ + dir.setNameFilters(_nameFilter); + dir.setFilter(QDir::AllDirs|QDir::Files|QDir::NoDotAndDotDot); + QFileInfoList list = dir.entryInfoList(); + for (int i = 0; i < list.size(); ++i) + { + if(stopRunning) + return; + QFileInfo fileInfo = list.at(i); + QString fileName = fileInfo.fileName(); +#ifdef Q_OS_MAC + QStringList src = _source.split("/"); + QString filePath = fileInfo.absoluteFilePath(); + QStringList fp = filePath.split("/"); + for(int i = 0; i< src.count();i++) + { + fp.removeFirst(); + } + QString relativePath = "/" + fp.join("/"); +#else + QString relativePath = QDir::cleanPath(fileInfo.absoluteFilePath()).remove(_source); +#endif + if(fileInfo.isDir()) + { + //se añade al path actual el folder, aún no se sabe si habrá que añadirlo a la base de datos + _currentPathFolders.append(Folder(fileInfo.fileName(),relativePath)); + create(QDir(fileInfo.absoluteFilePath())); + //una vez importada la información del folder, se retira del path actual ya que no volverá a ser visitado + _currentPathFolders.pop_back(); + } + else + { + insertComic(relativePath,fileInfo); + } + } +} + +bool LibraryCreator::checkCover(const QString & hash) +{ + return QFile::exists(_target+"/covers/"+hash+".jpg"); +} + +void LibraryCreator::insertComic(const QString & relativePath,const QFileInfo & fileInfo) +{ + //Se calcula el hash del cómic + + QCryptographicHash crypto(QCryptographicHash::Sha1); + QFile file(fileInfo.absoluteFilePath()); + file.open(QFile::ReadOnly); + crypto.addData(file.read(524288)); + file.close(); + //hash Sha1 del primer 0.5MB + filesize + QString hash = QString(crypto.result().toHex().constData()) + QString::number(fileInfo.size()); + ComicDB comic = DBHelper::loadComic(fileInfo.fileName(),relativePath,hash,_database); + int numPages = 0; + bool exists = checkCover(hash); + if(! ( comic.hasCover() && exists)) + { + ThumbnailCreator tc(QDir::cleanPath(fileInfo.absoluteFilePath()),_target+"/covers/"+hash+".jpg",comic.info.coverPage.toInt()); + tc.create(); + numPages = tc.getNumPages(); + if (numPages > 0) + emit(comicAdded(relativePath,_target+"/covers/"+hash+".jpg")); + } + + if (numPages > 0 || exists) + { + //en este punto sabemos que todos los folders que hay en _currentPath, deberían estar añadidos a la base de datos + insertFolders(); + comic.info.numPages = numPages; + comic.parentId = _currentPathFolders.last().id; + DBHelper::insert(&comic,_database); + } +} + +void LibraryCreator::update(QDir dirS) +{ + //QLOG_TRACE() << "Updating" << dirS.absolutePath(); + //QLOG_TRACE() << "Getting info from dir" << dirS.absolutePath(); + dirS.setNameFilters(_nameFilter); + dirS.setFilter(QDir::AllDirs|QDir::NoDotAndDotDot); + dirS.setSorting(QDir::Name|QDir::IgnoreCase|QDir::LocaleAware); + QFileInfoList listSFolders = dirS.entryInfoList(); + dirS.setFilter(QDir::Files|QDir::NoDotAndDotDot); + dirS.setSorting(QDir::Name|QDir::IgnoreCase|QDir::LocaleAware); + QFileInfoList listSFiles = dirS.entryInfoList(); + + qSort(listSFolders.begin(),listSFolders.end(),naturalSortLessThanCIFileInfo); + qSort(listSFiles.begin(),listSFiles.end(),naturalSortLessThanCIFileInfo); + + QFileInfoList listS; + listS.append(listSFolders); + listS.append(listSFiles); + //QLOG_DEBUG() << "---------------------------------------------------------"; + //foreach(QFileInfo info,listS) + // QLOG_DEBUG() << info.fileName(); + + //QLOG_TRACE() << "END Getting info from dir" << dirS.absolutePath(); + + //QLOG_TRACE() << "Getting info from DB" << dirS.absolutePath(); + QList folders = DBHelper::getFoldersFromParent(_currentPathFolders.last().id,_database); + QList comics = DBHelper::getComicsFromParent(_currentPathFolders.last().id,_database); + //QLOG_TRACE() << "END Getting info from DB" << dirS.absolutePath(); + + QList listD; + qSort(folders.begin(),folders.end(),naturalSortLessThanCILibraryItem); + qSort(comics.begin(),comics.end(),naturalSortLessThanCILibraryItem); + listD.append(folders); + listD.append(comics); + //QLOG_DEBUG() << "---------------------------------------------------------"; + //foreach(LibraryItem * info,listD) + // QLOG_DEBUG() << info->name; + //QLOG_DEBUG() << "---------------------------------------------------------"; + int lenghtS = listS.size(); + int lenghtD = listD.size(); + //QLOG_DEBUG() << "S len" << lenghtS << "D len" << lenghtD; + //QLOG_DEBUG() << "---------------------------------------------------------"; + + bool updated; + int i,j; + for (i=0,j=0; (i < lenghtS)||(j < lenghtD);) + { + if(stopRunning) + return; + updated = false; + if(i>=lenghtS) //finished source files/dirs + { + //QLOG_WARN() << "finished source files/dirs" << dirS.absolutePath(); + //delete listD //from j + for(;j=lenghtD) //finished library files/dirs + { + //QLOG_WARN() << "finished library files/dirs" << dirS.absolutePath(); + //create listS //from i + for(;iname; + + int comparation = QString::localeAwareCompare(nameS,nameD); + if(fileInfoS.isDir()&&fileInfoD->isDir()) + if(comparation == 0)//same folder, update + { + _currentPathFolders.append(*static_cast(fileInfoD));//fileInfoD conoce su padre y su id + update(QDir(fileInfoS.absoluteFilePath())); + _currentPathFolders.pop_back(); + i++; + j++; + } + else + if(comparation < 0) //nameS doesn't exist on DB + { + + if(nameS!="/.yacreaderlibrary") + { + //QLOG_WARN() << "dir source < dest" << nameS << nameD; +#ifdef Q_OS_MAC + QStringList src = _source.split("/"); + QString filePath = fileInfoS.absoluteFilePath(); + QStringList fp = filePath.split("/"); + for(int i = 0; i< src.count();i++) + { + fp.removeFirst(); + } + QString path = "/" + fp.join("/"); +#else + QString path = QDir::cleanPath(fileInfoS.absoluteFilePath()).remove(_source); +#endif + _currentPathFolders.append(Folder(fileInfoS.fileName(),path)); + create(QDir(fileInfoS.absoluteFilePath())); + _currentPathFolders.pop_back(); + } + i++; + } + else //nameD no longer avaliable on Source folder... + { + if(nameS!="/.yacreaderlibrary") + { + //QLOG_WARN() << "dir source > dest" << nameS << nameD; + DBHelper::removeFromDB(fileInfoD,_database); + j++; + } + else + i++; //skip library directory + } + else // one of them(or both) is a file + if(fileInfoS.isDir()) //this folder doesn't exist on library + { + if(nameS!="/.yacreaderlibrary") //skip .yacreaderlibrary folder + { + //QLOG_WARN() << "one of them(or both) is a file" << nameS << nameD; +#ifdef Q_OS_MAC + QStringList src = _source.split("/"); + QString filePath = fileInfoS.absoluteFilePath(); + QStringList fp = filePath.split("/"); + for(int i = 0; i< src.count();i++) + { + fp.removeFirst(); + } + QString path = "/" + fp.join("/"); +#else + QString path = QDir::cleanPath(fileInfoS.absoluteFilePath()).remove(_source); +#endif + _currentPathFolders.append(Folder(fileInfoS.fileName(),path)); + create(QDir(fileInfoS.absoluteFilePath())); + _currentPathFolders.pop_back(); + } + i++; + } + else + if(fileInfoD->isDir()) //delete this folder from library + { + DBHelper::removeFromDB(fileInfoD,_database); + j++; + } + else //both are files //BUG on windows (no case sensitive) + { + //nameD.remove(nameD.size()-4,4); + int comparation = QString::localeAwareCompare(nameS,nameD); + if(comparation < 0) //create new thumbnail + { +#ifdef Q_OS_MAC + QStringList src = _source.split("/"); + QString filePath = fileInfoS.absoluteFilePath(); + QStringList fp = filePath.split("/"); + for(int i = 0; i< src.count();i++) + { + fp.removeFirst(); + } + QString path = "/" + fp.join("/"); +#else + QString path = QDir::cleanPath(fileInfoS.absoluteFilePath()).remove(_source); +#endif + insertComic(path,fileInfoS); + i++; + } + else + { + if(comparation > 0) //delete thumbnail + { + DBHelper::removeFromDB(fileInfoD,_database); + j++; + } + else //same file + { + if(fileInfoS.isFile() && !fileInfoD->isDir()) + { + //TODO comprobar fechas + tamaño + //if(fileInfoS.lastModified()>fileInfoD.lastModified()) + //{ + // dirD.mkpath(_target+(QDir::cleanPath(fileInfoS.absolutePath()).remove(_source))); + // emit(coverExtracted(QDir::cleanPath(fileInfoS.absoluteFilePath()).remove(_source))); + // ThumbnailCreator tc(QDir::cleanPath(fileInfoS.absoluteFilePath()),_target+(QDir::cleanPath(fileInfoS.absoluteFilePath()).remove(_source))+".jpg"); + // tc.create(); + //} + } + i++;j++; + } + } + } + } + } +} + +bool ThumbnailCreator::crash = false; + +ThumbnailCreator::ThumbnailCreator(QString fileSource, QString target, int coverPage) +:_fileSource(fileSource),_target(target),_numPages(0),_coverPage(coverPage) +{ +} + +void ThumbnailCreator::create() +{ + QFileInfo fi(_fileSource); + if(!fi.exists()) //TODO: error file not found. + { + _cover.load(":/images/notCover.png"); + QLOG_WARN() << "Extracting cover: file not found " << _fileSource; + return; + } + + if(fi.suffix().compare("pdf",Qt::CaseInsensitive) == 0) + { + +#ifdef Q_OS_MAC + MacOSXPDFComic * pdfComic = new MacOSXPDFComic(); + if(!pdfComic->openComic(_fileSource)) + { + delete pdfComic; + //QImage p; + //p.load(":/images/notCover.png"); + //p.save(_target); + return; + } +#else + Poppler::Document * pdfComic = Poppler::Document::load(_fileSource); +#endif + if (!pdfComic) + { + QLOG_WARN() << "Extracting cover: unable to open PDF file " << _fileSource; + //delete pdfComic; //TODO check if the delete is needed + pdfComic = 0; + //QImage p; + //p.load(":/images/notCover.png"); + //p.save(_target); + return; + } + _numPages = pdfComic->numPages(); + if(_numPages >= _coverPage) + { +#ifdef Q_OS_MAC + { + QImage p = pdfComic->getPage(_coverPage-1); //TODO check if the page is valid +#else + QImage p = pdfComic->page(_coverPage-1)->renderToImage(72,72); +#endif + _cover = QPixmap::fromImage(p); + if(_target!="") + { + QImage scaled; + if(p.width()>p.height()) //landscape?? + scaled = p.scaledToWidth(640,Qt::SmoothTransformation); + else + scaled = p.scaledToWidth(480,Qt::SmoothTransformation); + scaled.save(_target,0,75); + } +#ifdef Q_OS_MAC + } + pdfComic->releaseLastPageData(); +#endif + } + else if(_target!="") + { + QLOG_WARN() << "Extracting cover: requested cover index greater than numPages " << _fileSource; + //QImage p; + //p.load(":/images/notCover.png"); + //p.save(_target); + } + + delete pdfComic; + } + else + { + + if(crash) + return; + + CompressedArchive archive(_fileSource); + if(!archive.toolsLoaded()) + { + QLOG_WARN() << "Extracting cover: 7z lib not loaded"; + crash = true; + return; + } + if(!archive.isValid()) + QLOG_WARN() << "Extracting cover: file format not supported " << _fileSource; + //se filtran para obtener sólo los formatos soportados + QList order = archive.getFileNames(); + QList fileNames = FileComic::filter(order); + _numPages = fileNames.size(); + if(_numPages == 0) + { + QLOG_WARN() << "Extracting cover: empty comic " << _fileSource; + _cover.load(":/images/notCover.png"); + if(_target!="") + _cover.save(_target); + } + else + { + if(_coverPage > _numPages) + _coverPage = 1; + qSort(fileNames.begin(),fileNames.end(), naturalSortLessThanCI); + int index = order.indexOf(fileNames.at(_coverPage-1)); + + if(_target=="") + { + if(!_cover.loadFromData(archive.getRawDataAtIndex(index))) + { + QLOG_WARN() << "Extracting cover: unable to load image from extracted cover " << _fileSource; + _cover.load(":/images/notCover.png"); + } + } + else + { + QImage p; + if(p.loadFromData(archive.getRawDataAtIndex(index))) + { + QImage scaled; + if(p.width()>p.height()) //landscape?? + scaled = p.scaledToWidth(640,Qt::SmoothTransformation); + else + scaled = p.scaledToWidth(480,Qt::SmoothTransformation); + scaled.save(_target,0,75); + } + else + { + QLOG_WARN() << "Extracting cover: unable to load image from extracted cover " << _fileSource; + //p.load(":/images/notCover.png"); + //p.save(_target); + } + } + } + } +} diff --git a/YACReaderLibrary/library_creator.h b/YACReaderLibrary/library_creator.h new file mode 100644 index 00000000..26479fd2 --- /dev/null +++ b/YACReaderLibrary/library_creator.h @@ -0,0 +1,86 @@ +#ifndef __LIBRARY_CREATOR_H +#define __LIBRARY_CREATOR_H + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "folder.h" +#include "comic_db.h" + + + class LibraryCreator : public QThread + { + Q_OBJECT + public: + LibraryCreator(); + void createLibrary(const QString & source, const QString & target); + void updateLibrary(const QString & source, const QString & target); + void stop(); + private: + void processLibrary(const QString & source, const QString & target); + enum Mode {CREATOR,UPDATER}; + //atributos "globales" durante el proceso de creación y actualización + enum Mode _mode; + QString _source; + QString _target; + QStringList _nameFilter; + QSqlDatabase _database; + QList _currentPathFolders; //lista de folders en el orden en el que están siendo explorados, el último es el folder actual + //recursive method + void create(QDir currentDirectory); + void update(QDir currentDirectory); + void run(); + qulonglong insertFolders();//devuelve el id del último folder añadido (último en la ruta) + bool checkCover(const QString & hash); + void insertComic(const QString & relativePath,const QFileInfo & fileInfo); + //qulonglong insertFolder(qulonglong parentId,const Folder & folder); + //qulonglong insertComic(const Comic & comic); + bool stopRunning; + //LibraryCreator está en modo creación si creation == true; + bool creation; + signals: + void finished(); + void coverExtracted(QString); + void folderUpdated(QString); + void comicAdded(QString,QString); + void updated(); + void created(); + void failedCreatingDB(QString); + void failedOpeningDB(QString); + }; + + class ThumbnailCreator : public QObject + { + Q_OBJECT + + public: + ThumbnailCreator(QString fileSource, QString target="", int coverPage = 1); + private: + QString _fileSource; + QString _target; + QString _currentName; + int _numPages; + QPixmap _cover; + int _coverPage; + static bool crash; + + public slots: + void create(); + int getNumPages(){return _numPages;}; + QPixmap getCover(){return _cover;}; + signals: + void openingError(QProcess::ProcessError error); + + }; + +#endif diff --git a/YACReaderLibrary/library_window.cpp b/YACReaderLibrary/library_window.cpp new file mode 100644 index 00000000..dc4a8ea5 --- /dev/null +++ b/YACReaderLibrary/library_window.cpp @@ -0,0 +1,2076 @@ +#include "library_window.h" +#include "custom_widgets.h" +#include "treeitem.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include "data_base_management.h" +#include "yacreader_global.h" +#include "onstart_flow_selection_dialog.h" +#include "no_libraries_widget.h" +#include "import_widget.h" + +#include "yacreader_search_line_edit.h" +#include "comic_db.h" +#include "library_creator.h" +#include "package_manager.h" +#include "comic_flow_widget.h" +#include "create_library_dialog.h" +#include "rename_library_dialog.h" +#include "properties_dialog.h" +#include "export_library_dialog.h" +#include "import_library_dialog.h" +#include "export_comics_info_dialog.h" +#include "import_comics_info_dialog.h" +#include "add_library_dialog.h" +#include "options_dialog.h" +#include "help_about_dialog.h" +#include "server_config_dialog.h" +#include "tablemodel.h" +#include "yacreader_tool_bar_stretch.h" +#include "yacreader_table_view.h" + +#include "yacreader_dark_menu.h" +#include "yacreader_titled_toolbar.h" +#include "yacreader_main_toolbar.h" + +#include "yacreader_sidebar.h" + +#include "comics_remover.h" +#include "yacreader_library_list_widget.h" +#include "yacreader_treeview.h" + +#include "comic_vine_dialog.h" +//#include "yacreader_social_dialog.h" + +#include "classic_comics_view.h" +#include "grid_comics_view.h" +#include "comics_view_transition.h" +#include "empty_folder_widget.h" + +#include "edit_shortcuts_dialog.h" +#include "shortcuts_manager.h" + +#include "QsLog.h" + +#ifdef Q_OS_WIN + #include +#endif + +#ifdef Q_OS_MAC +//#include +#endif + +LibraryWindow::LibraryWindow() + :QMainWindow(),fullscreen(false),fetching(false),previousFilter(""),removeError(false) +{ + setupUI(); + loadLibraries(); + + if(libraries.isEmpty()) + { + showNoLibrariesWidget(); + } + else + { + showRootWidget(); + selectedLibrary->setCurrentIndex(0); + } +} + +void LibraryWindow::setupUI() +{ + setWindowIcon(QIcon(":/images/iconLibrary.png")); + + setUnifiedTitleAndToolBarOnMac(true); + + libraryCreator = new LibraryCreator(); + packageManager = new PackageManager(); + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + + createActions(); + doModels(); + + doLayout(); + createToolBars(); + doDialogs(); + createMenus(); + createConnections(); + + setWindowTitle(tr("YACReader Library")); + + setMinimumSize(800,480); + + //restore + if(settings->contains(MAIN_WINDOW_GEOMETRY)) + restoreGeometry(settings->value(MAIN_WINDOW_GEOMETRY).toByteArray()); + else + //if(settings->value(USE_OPEN_GL).toBool() == false) + showMaximized(); + + /*if(settings->contains(COMICS_VIEW_HEADERS_GEOMETRY)) + comicsView->horizontalHeader()->restoreGeometry(settings->value(COMICS_VIEW_HEADERS_GEOMETRY).toByteArray());*/ + + /*socialDialog = new YACReaderSocialDialog(this); + socialDialog->setHidden(true);*/ +} + +void LibraryWindow::doLayout() +{ + //LAYOUT ELEMENTS------------------------------------------------------------ + //--------------------------------------------------------------------------- + + QSplitter * sHorizontal = new QSplitter(Qt::Horizontal); //spliter principal +#ifdef Q_OS_MAC + sHorizontal->setStyleSheet("QSplitter::handle{image:none;background-color:#B8B8B8;} QSplitter::handle:vertical {height:1px;}"); +#else + sHorizontal->setStyleSheet("QSplitter::handle:vertical {height:4px;}"); +#endif + + //TOOLBARS------------------------------------------------------------------- + //--------------------------------------------------------------------------- + editInfoToolBar = new QToolBar(); + editInfoToolBar->setStyleSheet("QToolBar {border: none;}"); + +#ifdef Q_OS_MAC + libraryToolBar = addToolBar(tr("Library")); +#else + libraryToolBar = new YACReaderMainToolBar(this); +#endif + + + //FLOW----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + if(QGLFormat::hasOpenGL() && !settings->contains(USE_OPEN_GL)) + { + OnStartFlowSelectionDialog * flowSelDialog = new OnStartFlowSelectionDialog(); + + flowSelDialog->exec(); + if(flowSelDialog->result() == QDialog::Accepted) + settings->setValue(USE_OPEN_GL,2); + else + settings->setValue(USE_OPEN_GL,0); + + delete flowSelDialog; + } + + //SIDEBAR----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + sideBar = new YACReaderSideBar; + + foldersView = sideBar->foldersView; + selectedLibrary = sideBar->selectedLibrary; + foldersFilter = sideBar->foldersFilter; + + YACReaderTitledToolBar * librariesTitle = sideBar->librariesTitle; + + YACReaderTitledToolBar * foldersTitle = sideBar->foldersTitle; + + librariesTitle->addAction(createLibraryAction); + librariesTitle->addAction(openLibraryAction); + librariesTitle->addSpacing(3); + + foldersTitle->addAction(setRootIndexAction); + foldersTitle->addAction(expandAllNodesAction); + foldersTitle->addAction(colapseAllNodesAction); + + //FINAL LAYOUT------------------------------------------------------------- + comicsViewStack = new QStackedWidget(); + + if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) { + comicsView = classicComicsView = new ClassicComicsView(); + comicsViewStatus = Flow; + //comicsViewStack->setCurrentIndex(Flow); + } else { + comicsView = gridComicsView = new GridComicsView(); + comicsViewStatus = Grid; + //comicsViewStack->setCurrentIndex(Grid); + } + + doComicsViewConnections(); + + comicsView->setToolBar(editInfoToolBar); + comicsViewStack->addWidget(comicsViewTransition = new ComicsViewTransition()); + comicsViewStack->addWidget(emptyFolderWidget = new EmptyFolderWidget()); + comicsViewStack->addWidget(comicsView); + + comicsViewStack->setCurrentWidget(comicsView); + + sHorizontal->addWidget(sideBar); +#ifndef Q_OS_MAC + QVBoxLayout * rightLayout = new QVBoxLayout; + rightLayout->addWidget(libraryToolBar); + rightLayout->addWidget(comicsViewStack); + + rightLayout->setMargin(0); + rightLayout->setSpacing(0); + + QWidget * rightWidget = new QWidget(); + rightWidget->setLayout(rightLayout); + + sHorizontal->addWidget(rightWidget); +#else + sHorizontal->addWidget(comicsViewStack); +#endif + + sHorizontal->setStretchFactor(0,0); + sHorizontal->setStretchFactor(1,1); + mainWidget = new QStackedWidget(this); + mainWidget->addWidget(sHorizontal); + setCentralWidget(mainWidget); + //FINAL LAYOUT------------------------------------------------------------- + + + //OTHER---------------------------------------------------------------------- + //--------------------------------------------------------------------------- + noLibrariesWidget = new NoLibrariesWidget(); + mainWidget->addWidget(noLibrariesWidget); + + importWidget = new ImportWidget(); + mainWidget->addWidget(importWidget); + + connect(noLibrariesWidget,SIGNAL(createNewLibrary()),this,SLOT(createLibrary())); + connect(noLibrariesWidget,SIGNAL(addExistingLibrary()),this,SLOT(showAddLibrary())); + + + + //collapsible disabled in macosx (only temporaly) +#ifdef Q_OS_MAC + sHorizontal->setCollapsible(0,false); +#endif +} + +void LibraryWindow::doDialogs() +{ + createLibraryDialog = new CreateLibraryDialog(this); + renameLibraryDialog = new RenameLibraryDialog(this); + propertiesDialog = new PropertiesDialog(this); + comicVineDialog = new ComicVineDialog(this); + exportLibraryDialog = new ExportLibraryDialog(this); + importLibraryDialog = new ImportLibraryDialog(this); + exportComicsInfoDialog = new ExportComicsInfoDialog(this); + importComicsInfoDialog = new ImportComicsInfoDialog(this); + addLibraryDialog = new AddLibraryDialog(this); + optionsDialog = new OptionsDialog(this); + optionsDialog->restoreOptions(settings); + + editShortcutsDialog = new EditShortcutsDialog(this); + setUpShortcutsManagement(); + +#ifdef SERVER_RELEASE + serverConfigDialog = new ServerConfigDialog(this); +#endif + + had = new HelpAboutDialog(this); //TODO load data. + QString sufix = QLocale::system().name(); + if(QFile(":/files/about_"+sufix+".html").exists()) + had->loadAboutInformation(":/files/about_"+sufix+".html"); + else + had->loadAboutInformation(":/files/about.html"); + + if(QFile(":/files/helpYACReaderLibrary_"+sufix+".html").exists()) + had->loadHelp(":/files/helpYACReaderLibrary_"+sufix+".html"); + else + had->loadHelp(":/files/helpYACReaderLibrary.html"); + + +} + +void LibraryWindow::setUpShortcutsManagement() +{ + + QList allActions; + QList tmpList; + + editShortcutsDialog->addActionsGroup("Comics",QIcon(":/images/shortcuts_group_comics.png"), + tmpList = QList() + << openComicAction + << setAsReadAction + << setAsNonReadAction + << openContainingFolderComicAction + << resetComicRatingAction + << selectAllComicsAction + << editSelectedComicsAction + << asignOrderAction + << deleteComicsAction + << getInfoAction); + + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("Folders",QIcon(":/images/shortcuts_group_folders.png"), + tmpList = QList() + << setRootIndexAction + << expandAllNodesAction + << colapseAllNodesAction + << openContainingFolderAction + << setFolderAsNotCompletedAction + << setFolderAsCompletedAction + << setFolderAsReadAction + << setFolderAsUnreadAction); + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("General",QIcon(":/images/shortcuts_group_general.png"), + tmpList = QList() + << backAction + << forwardAction + << helpAboutAction + << optionsAction + << serverConfigAction + << showEditShortcutsAction); + + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("Libraries",QIcon(":/images/shortcuts_group_libraries.png"), + tmpList = QList() + << createLibraryAction + << openLibraryAction + << exportComicsInfoAction + << importComicsInfoAction + << exportLibraryAction + << importLibraryAction + << updateLibraryAction + << renameLibraryAction + << removeLibraryAction); + + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("Visualization",QIcon(":/images/shortcuts_group_visualization.png"), + tmpList = QList() + << showHideMarksAction + << toggleFullScreenAction + << toggleComicsViewAction + << hideComicViewAction); + + allActions << tmpList; + + ShortcutsManager::getShortcutsManager().registerActions(allActions); +} + +void LibraryWindow::doModels() +{ + //folders + dm = new TreeModel(); + //comics + dmCV = new TableModel(); + + setFoldersFilter(""); +} + +void LibraryWindow::disconnectComicsViewConnections(ComicsView * widget) +{ + disconnect(widget, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + disconnect(showHideMarksAction,SIGNAL(toggled(bool)),widget,SLOT(setShowMarks(bool))); + disconnect(widget,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); + disconnect(widget,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + disconnect(selectAllComicsAction,SIGNAL(triggered()),widget,SLOT(selectAll())); +} + +void LibraryWindow::doComicsViewConnections() +{ + connect(comicsView, SIGNAL(comicRated(int,QModelIndex)), dmCV, SLOT(updateRating(int,QModelIndex))); + connect(showHideMarksAction,SIGNAL(toggled(bool)),comicsView,SLOT(setShowMarks(bool))); + connect(comicsView,SIGNAL(selected(unsigned int)),this,SLOT(openComic())); + connect(comicsView,SIGNAL(doubleClicked(QModelIndex)),this,SLOT(openComic())); + connect(selectAllComicsAction,SIGNAL(triggered()),comicsView,SLOT(selectAll())); +} + +void LibraryWindow::createActions() +{ + backAction = new QAction(this); + QIcon icoBackButton; + icoBackButton.addPixmap(QPixmap(":/images/main_toolbar/back.png"), QIcon::Normal); + //icoBackButton.addPixmap(QPixmap(":/images/main_toolbar/back_disabled.png"), QIcon::Disabled); + backAction->setData(BACK_ACTION_YL); + backAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(BACK_ACTION_YL)); + backAction->setIcon(icoBackButton); + backAction->setDisabled(true); + + forwardAction = new QAction(this); + QIcon icoFordwardButton; + icoFordwardButton.addPixmap(QPixmap(":/images/main_toolbar/forward.png"), QIcon::Normal); + //icoFordwardButton.addPixmap(QPixmap(":/images/main_toolbar/forward_disabled.png"), QIcon::Disabled); + forwardAction->setData(FORWARD_ACTION_YL); + forwardAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(FORWARD_ACTION_YL)); + forwardAction->setIcon(icoFordwardButton); + forwardAction->setDisabled(true); + + createLibraryAction = new QAction(this); + createLibraryAction->setToolTip(tr("Create a new library")); + createLibraryAction->setData(CREATE_LIBRARY_ACTION_YL); + createLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(CREATE_LIBRARY_ACTION_YL)); + createLibraryAction->setIcon(QIcon(":/images/newLibraryIcon.png")); + + openLibraryAction = new QAction(this); + openLibraryAction->setToolTip(tr("Open an existing library")); + openLibraryAction->setData(OPEN_LIBRARY_ACTION_YL); + openLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_LIBRARY_ACTION_YL)); + openLibraryAction->setIcon(QIcon(":/images/openLibraryIcon.png")); + + exportComicsInfoAction = new QAction(tr("Export comics info"),this); + exportComicsInfoAction->setToolTip(tr("Export comics info")); + exportComicsInfoAction->setData(EXPORT_COMICS_INFO_ACTION_YL); + exportComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_COMICS_INFO_ACTION_YL)); + exportComicsInfoAction->setIcon(QIcon(":/images/exportComicsInfoIcon.png")); + + importComicsInfoAction = new QAction(tr("Import comics info"),this); + importComicsInfoAction->setToolTip(tr("Import comics info")); + importComicsInfoAction->setData(IMPORT_COMICS_INFO_ACTION_YL); + importComicsInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_COMICS_INFO_ACTION_YL)); + importComicsInfoAction->setIcon(QIcon(":/images/importComicsInfoIcon.png")); + + exportLibraryAction = new QAction(tr("Pack covers"),this); + exportLibraryAction->setToolTip(tr("Pack the covers of the selected library")); + exportLibraryAction->setData(EXPORT_LIBRARY_ACTION_YL); + exportLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPORT_LIBRARY_ACTION_YL)); + exportLibraryAction->setIcon(QIcon(":/images/exportLibraryIcon.png")); + + importLibraryAction = new QAction(tr("Unpack covers"),this); + importLibraryAction->setToolTip(tr("Unpack a catalog")); + importLibraryAction->setData(IMPORT_LIBRARY_ACTION_YL); + importLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(IMPORT_LIBRARY_ACTION_YL)); + importLibraryAction->setIcon(QIcon(":/images/importLibraryIcon.png")); + + updateLibraryAction = new QAction(tr("Update library"),this); + updateLibraryAction->setToolTip(tr("Update current library")); + updateLibraryAction->setData(UPDATE_LIBRARY_ACTION_YL); + updateLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(UPDATE_LIBRARY_ACTION_YL)); + updateLibraryAction->setIcon(QIcon(":/images/updateLibraryIcon.png")); + + renameLibraryAction = new QAction(tr("Rename library"),this); + renameLibraryAction->setToolTip(tr("Rename current library")); + renameLibraryAction->setData(RENAME_LIBRARY_ACTION_YL); + renameLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RENAME_LIBRARY_ACTION_YL)); + renameLibraryAction->setIcon(QIcon(":/images/editIcon.png")); + + removeLibraryAction = new QAction(tr("Remove library"),this); + removeLibraryAction->setToolTip(tr("Remove current library from your collection")); + removeLibraryAction->setData(REMOVE_LIBRARY_ACTION_YL); + removeLibraryAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(REMOVE_LIBRARY_ACTION_YL)); + removeLibraryAction->setIcon(QIcon(":/images/removeLibraryIcon.png")); + + openComicAction = new QAction(tr("Open current comic"),this); + openComicAction->setToolTip(tr("Open current comic on YACReader")); + openComicAction->setData(OPEN_COMIC_ACTION_YL); + openComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_COMIC_ACTION_YL)); + openComicAction->setIcon(QIcon(":/images/openInYACReader.png")); + + setAsReadAction = new QAction(tr("Set as read"),this); + setAsReadAction->setToolTip(tr("Set comic as read")); + setAsReadAction->setData(SET_AS_READ_ACTION_YL); + setAsReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_AS_READ_ACTION_YL)); + setAsReadAction->setIcon(QIcon(":/images/setReadButton.png")); + + setAsNonReadAction = new QAction(tr("Set as unread"),this); + setAsNonReadAction->setToolTip(tr("Set comic as unread")); + setAsNonReadAction->setData(SET_AS_NON_READ_ACTION_YL); + setAsNonReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_AS_NON_READ_ACTION_YL)); + setAsNonReadAction->setIcon(QIcon(":/images/setUnread.png")); + + /*setAllAsReadAction = new QAction(tr("Set all as read"),this); + setAllAsReadAction->setToolTip(tr("Set all comics as read")); + setAllAsReadAction->setIcon(QIcon(":/images/setAllRead.png")); + + setAllAsNonReadAction = new QAction(tr("Set all as unread"),this); + setAllAsNonReadAction->setToolTip(tr("Set all comics as unread")); + setAllAsNonReadAction->setIcon(QIcon(":/images/setAllUnread.png"));*/ + + showHideMarksAction = new QAction(tr("Show/Hide marks"),this); + showHideMarksAction->setToolTip(tr("Show or hide readed marks")); + showHideMarksAction->setData(SHOW_HIDE_MARKS_ACTION_YL); + showHideMarksAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_HIDE_MARKS_ACTION_YL)); + showHideMarksAction->setCheckable(true); + showHideMarksAction->setIcon(QIcon(":/images/showMarks.png")); + showHideMarksAction->setChecked(true); + + toggleFullScreenAction = new QAction(tr("Fullscreen mode on/off"),this); + toggleFullScreenAction->setToolTip(tr("Fullscreen mode on/off")); + toggleFullScreenAction->setData(TOGGLE_FULL_SCREEN_ACTION_YL); + toggleFullScreenAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_FULL_SCREEN_ACTION_YL)); + QIcon icoFullscreenButton; + icoFullscreenButton.addPixmap(QPixmap(":/images/main_toolbar/fullscreen.png"), QIcon::Normal); + toggleFullScreenAction->setIcon(icoFullscreenButton); + + helpAboutAction = new QAction(this); + helpAboutAction->setToolTip(tr("Help, About YACReader")); + helpAboutAction->setData(HELP_ABOUT_ACTION_YL); + helpAboutAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HELP_ABOUT_ACTION_YL)); + QIcon icoHelpButton; + icoHelpButton.addPixmap(QPixmap(":/images/main_toolbar/help.png"), QIcon::Normal); + helpAboutAction->setIcon(icoHelpButton); + + setRootIndexAction = new QAction(this); + setRootIndexAction->setData(SET_ROOT_INDEX_ACTION_YL); + setRootIndexAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_ROOT_INDEX_ACTION_YL)); + setRootIndexAction->setToolTip(tr("Select root node")); + setRootIndexAction->setIcon(QIcon(":/images/setRoot.png")); + + expandAllNodesAction = new QAction(this); + expandAllNodesAction->setToolTip(tr("Expand all nodes")); + expandAllNodesAction->setData(EXPAND_ALL_NODES_ACTION_YL); + expandAllNodesAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EXPAND_ALL_NODES_ACTION_YL)); + expandAllNodesAction->setIcon(QIcon(":/images/expand.png")); + + colapseAllNodesAction = new QAction(this); + colapseAllNodesAction->setToolTip(tr("Colapse all nodes")); + colapseAllNodesAction->setData(COLAPSE_ALL_NODES_ACTION_YL); + colapseAllNodesAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(COLAPSE_ALL_NODES_ACTION_YL)); + colapseAllNodesAction->setIcon(QIcon(":/images/colapse.png")); + + optionsAction = new QAction(this); + optionsAction->setToolTip(tr("Show options dialog")); + optionsAction->setData(OPTIONS_ACTION_YL); + optionsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPTIONS_ACTION_YL)); + QIcon icoSettingsButton; + icoSettingsButton.addPixmap(QPixmap(":/images/main_toolbar/settings.png"), QIcon::Normal); + optionsAction->setIcon(icoSettingsButton); + + serverConfigAction = new QAction(this); + serverConfigAction->setToolTip(tr("Show comics server options dialog")); + serverConfigAction->setData(SERVER_CONFIG_ACTION_YL); + serverConfigAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SERVER_CONFIG_ACTION_YL)); + QIcon icoServerButton; + icoServerButton.addPixmap(QPixmap(":/images/main_toolbar/server.png"), QIcon::Normal); + serverConfigAction->setIcon(icoServerButton); + + toggleComicsViewAction = new QAction(tr("Change between comics views"),this); + toggleComicsViewAction->setToolTip(tr("Change between comics views")); + QIcon icoViewsButton; + if(!settings->contains(COMICS_VIEW_STATUS) || settings->value(COMICS_VIEW_STATUS) == Flow) + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/grid.png"), QIcon::Normal); + else + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/flow.png"), QIcon::Normal); + toggleComicsViewAction->setData(TOGGLE_COMICS_VIEW_ACTION_YL); + toggleComicsViewAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(TOGGLE_COMICS_VIEW_ACTION_YL)); + toggleComicsViewAction->setIcon(icoViewsButton); + //socialAction = new QAction(this); + + openContainingFolderAction = new QAction(this); + openContainingFolderAction->setText(tr("Open folder...")); + openContainingFolderAction->setData(OPEN_CONTAINING_FOLDER_ACTION_YL); + openContainingFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_CONTAINING_FOLDER_ACTION_YL)); + openContainingFolderAction->setIcon(QIcon(":/images/open.png")); + + setFolderAsNotCompletedAction = new QAction(this); + setFolderAsNotCompletedAction->setText(tr("Set as uncompleted")); + setFolderAsNotCompletedAction->setVisible(false); + setFolderAsNotCompletedAction->setData(SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL); + setFolderAsNotCompletedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_NOT_COMPLETED_ACTION_YL)); + + setFolderAsCompletedAction = new QAction(this); + setFolderAsCompletedAction->setText(tr("Set as completed")); + setFolderAsCompletedAction->setVisible(false); + setFolderAsCompletedAction->setData(SET_FOLDER_AS_COMPLETED_ACTION_YL); + setFolderAsCompletedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_COMPLETED_ACTION_YL)); + + setFolderAsReadAction = new QAction(this); + setFolderAsReadAction->setText(tr("Set as read")); + setFolderAsReadAction->setVisible(false); + setFolderAsReadAction->setData(SET_FOLDER_AS_READ_ACTION_YL); + setFolderAsReadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_READ_ACTION_YL)); + + setFolderAsUnreadAction = new QAction(this); + setFolderAsUnreadAction->setText(tr("Set as unread")); + setFolderAsUnreadAction->setVisible(false); + setFolderAsUnreadAction->setData(SET_FOLDER_AS_UNREAD_ACTION_YL); + setFolderAsUnreadAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SET_FOLDER_AS_UNREAD_ACTION_YL)); + + openContainingFolderComicAction = new QAction(this); + openContainingFolderComicAction->setText(tr("Open containing folder...")); + openContainingFolderComicAction->setData(OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL); + openContainingFolderComicAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(OPEN_CONTAINING_FOLDER_COMIC_ACTION_YL)); + openContainingFolderComicAction->setIcon(QIcon(":/images/open.png")); + + resetComicRatingAction = new QAction(this); + resetComicRatingAction->setText(tr("Reset comic rating")); + resetComicRatingAction->setData(RESET_COMIC_RATING_ACTION_YL); + resetComicRatingAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RESET_COMIC_RATING_ACTION_YL)); + + //Edit comics actions------------------------------------------------------ + selectAllComicsAction = new QAction(this); + selectAllComicsAction->setText(tr("Select all comics")); + selectAllComicsAction->setData(SELECT_ALL_COMICS_ACTION_YL); + selectAllComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SELECT_ALL_COMICS_ACTION_YL)); + selectAllComicsAction->setIcon(QIcon(":/images/selectAll.png")); + + editSelectedComicsAction = new QAction(this); + editSelectedComicsAction->setText(tr("Edit")); + editSelectedComicsAction->setData(EDIT_SELECTED_COMICS_ACTION_YL); + editSelectedComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(EDIT_SELECTED_COMICS_ACTION_YL)); + editSelectedComicsAction->setIcon(QIcon(":/images/editComic.png")); + + asignOrderAction = new QAction(this); + asignOrderAction->setText(tr("Asign current order to comics")); + asignOrderAction->setData(ASIGN_ORDER_ACTION_YL); + asignOrderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ASIGN_ORDER_ACTION_YL)); + asignOrderAction->setIcon(QIcon(":/images/asignNumber.png")); + + forceCoverExtractedAction = new QAction(this); + forceCoverExtractedAction->setText(tr("Update cover")); + forceCoverExtractedAction->setData(FORCE_COVER_EXTRACTED_ACTION_YL); + forceCoverExtractedAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(FORCE_COVER_EXTRACTED_ACTION_YL)); + forceCoverExtractedAction->setIcon(QIcon(":/images/importCover.png")); + + deleteComicsAction = new QAction(this); + deleteComicsAction->setText(tr("Delete selected comics")); + deleteComicsAction->setData(DELETE_COMICS_ACTION_YL); + deleteComicsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(DELETE_COMICS_ACTION_YL)); + deleteComicsAction->setIcon(QIcon(":/images/trash.png")); + + hideComicViewAction = new QAction(this); + hideComicViewAction->setText(tr("Hide comic flow")); + hideComicViewAction->setData(HIDE_COMIC_VIEW_ACTION_YL); + hideComicViewAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(HIDE_COMIC_VIEW_ACTION_YL)); + hideComicViewAction->setIcon(QIcon(":/images/hideComicFlow.png")); + hideComicViewAction->setCheckable(true); + hideComicViewAction->setChecked(false); + + getInfoAction = new QAction(this); + getInfoAction->setData(GET_INFO_ACTION_YL); + getInfoAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(GET_INFO_ACTION_YL)); + getInfoAction->setText(tr("Download tags from Comic Vine")); + getInfoAction->setIcon(QIcon(":/images/getInfo.png")); + //------------------------------------------------------------------------- + + showEditShortcutsAction = new QAction(tr("Edit shortcuts"),this); + showEditShortcutsAction->setData(SHOW_EDIT_SHORTCUTS_ACTION_YL); + showEditShortcutsAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SHOW_EDIT_SHORTCUTS_ACTION_YL)); + showEditShortcutsAction->setShortcutContext(Qt::ApplicationShortcut); + addAction(showEditShortcutsAction); + //disable actions + disableAllActions(); +} +void LibraryWindow::disableComicsActions(bool disabled) +{ + //if there aren't comics, no fullscreen option will be available + toggleFullScreenAction->setDisabled(disabled); + //edit toolbar + openComicAction->setDisabled(disabled); + editSelectedComicsAction->setDisabled(disabled); + selectAllComicsAction->setDisabled(disabled); + asignOrderAction->setDisabled(disabled); + setAsReadAction->setDisabled(disabled); + setAsNonReadAction->setDisabled(disabled); + //setAllAsReadAction->setDisabled(disabled); + //setAllAsNonReadAction->setDisabled(disabled); + showHideMarksAction->setDisabled(disabled); + deleteComicsAction->setDisabled(disabled); + //context menu + openContainingFolderComicAction->setDisabled(disabled); + resetComicRatingAction->setDisabled(disabled); + + getInfoAction->setDisabled(disabled); + + +} +void LibraryWindow::disableLibrariesActions(bool disabled) +{ + updateLibraryAction->setDisabled(disabled); + renameLibraryAction->setDisabled(disabled); + removeLibraryAction->setDisabled(disabled); + exportComicsInfoAction->setDisabled(disabled); + importComicsInfoAction->setDisabled(disabled); + exportLibraryAction->setDisabled(disabled); + //importLibraryAction->setDisabled(disabled); +} + +void LibraryWindow::disableNoUpdatedLibrariesActions(bool disabled) +{ + updateLibraryAction->setDisabled(disabled); + exportComicsInfoAction->setDisabled(disabled); + importComicsInfoAction->setDisabled(disabled); + exportLibraryAction->setDisabled(disabled); +} + +void LibraryWindow::disableFoldersActions(bool disabled) +{ + setRootIndexAction->setDisabled(disabled); + expandAllNodesAction->setDisabled(disabled); + colapseAllNodesAction->setDisabled(disabled); + + openContainingFolderAction->setDisabled(disabled); + + if(disabled == false) + { + setFolderAsNotCompletedAction->setVisible(false); + setFolderAsCompletedAction->setVisible(false); + setFolderAsReadAction->setVisible(false); + setFolderAsUnreadAction->setVisible(false); + } +} + +void LibraryWindow::disableAllActions() +{ + disableComicsActions(true); + disableLibrariesActions(true); + disableFoldersActions(true); +} + +void LibraryWindow::createToolBars() +{ + +#ifdef Q_OS_MAC + libraryToolBar->setIconSize(QSize(16,16)); //TODO make icon size dynamic + + libraryToolBar->addAction(backAction); + libraryToolBar->addAction(forwardAction); + + {QWidget * w = new QWidget(); + w->setFixedWidth(10); + libraryToolBar->addWidget(w);} + +#ifdef SERVER_RELEASE + libraryToolBar->addAction(serverConfigAction); +#endif + libraryToolBar->addAction(optionsAction); + libraryToolBar->addAction(helpAboutAction); + + { QWidget * w2 = new QWidget(); + w2->setFixedWidth(10); + libraryToolBar->addWidget(w2);} + + libraryToolBar->addAction(toggleComicsViewAction); + libraryToolBar->addAction(toggleFullScreenAction); + + libraryToolBar->addWidget(new QToolBarStretch()); + libraryToolBar->addWidget(foldersFilter); + + libraryToolBar->setMovable(false); + + +#else + libraryToolBar->backButton->setDefaultAction(backAction); + libraryToolBar->forwardButton->setDefaultAction(forwardAction); + libraryToolBar->settingsButton->setDefaultAction(optionsAction); + libraryToolBar->serverButton->setDefaultAction(serverConfigAction); + libraryToolBar->helpButton->setDefaultAction(helpAboutAction); + libraryToolBar->toggleComicsViewButton->setDefaultAction(toggleComicsViewAction); + libraryToolBar->fullscreenButton->setDefaultAction(toggleFullScreenAction); +#endif + + editInfoToolBar->setIconSize(QSize(18,18)); + editInfoToolBar->addAction(openComicAction); + editInfoToolBar->addSeparator(); + editInfoToolBar->addAction(editSelectedComicsAction); + editInfoToolBar->addAction(getInfoAction); + editInfoToolBar->addAction(asignOrderAction); + + editInfoToolBar->addSeparator(); + + editInfoToolBar->addAction(selectAllComicsAction); + + editInfoToolBar->addSeparator(); + + editInfoToolBar->addAction(setAsReadAction); + //editInfoToolBar->addAction(setAllAsReadAction); + editInfoToolBar->addAction(setAsNonReadAction); + //editInfoToolBar->addAction(setAllAsNonReadAction); + + editInfoToolBar->addAction(showHideMarksAction); + + editInfoToolBar->addSeparator(); + + editInfoToolBar->addAction(deleteComicsAction); + + editInfoToolBar->addWidget(new QToolBarStretch()); + editInfoToolBar->addAction(hideComicViewAction); +} + +void LibraryWindow::createMenus() +{ + itemActions << openContainingFolderComicAction + << YACReader::createSeparator() + << resetComicRatingAction + << YACReader::createSeparator() + << editSelectedComicsAction + << getInfoAction + << asignOrderAction + << YACReader::createSeparator() + << setAsReadAction + << setAsNonReadAction + << YACReader::createSeparator() + << deleteComicsAction; + + viewActions << openComicAction + << YACReader::createSeparator() + << openContainingFolderComicAction + << YACReader::createSeparator() + << resetComicRatingAction + << YACReader::createSeparator() + << editSelectedComicsAction + << getInfoAction + << asignOrderAction + << YACReader::createSeparator() + << selectAllComicsAction + << YACReader::createSeparator() + << setAsReadAction + << setAsNonReadAction + << showHideMarksAction + << YACReader::createSeparator() + << deleteComicsAction + << YACReader::createSeparator() + << toggleFullScreenAction; + + comicsView->setItemActions(itemActions); + comicsView->setViewActions(viewActions); + + foldersView->addAction(openContainingFolderAction); + YACReader::addSperator(foldersView); + + foldersView->addAction(setFolderAsNotCompletedAction); + foldersView->addAction(setFolderAsCompletedAction); + YACReader::addSperator(foldersView); + + foldersView->addAction(setFolderAsReadAction); + foldersView->addAction(setFolderAsUnreadAction); + + selectedLibrary->addAction(updateLibraryAction); + selectedLibrary->addAction(renameLibraryAction); + selectedLibrary->addAction(removeLibraryAction); + YACReader::addSperator(selectedLibrary); + + selectedLibrary->addAction(exportComicsInfoAction); + selectedLibrary->addAction(importComicsInfoAction); + YACReader::addSperator(selectedLibrary); + + selectedLibrary->addAction(exportLibraryAction); + selectedLibrary->addAction(importLibraryAction); + + + + +//MacOSX app menus +#ifdef Q_OS_MACX + QMenuBar * menu = this->menuBar(); + //about / preferences + //TODO + + //library + QMenu * libraryMenu = new QMenu(tr("Library")); + + libraryMenu->addAction(updateLibraryAction); + libraryMenu->addAction(renameLibraryAction); + libraryMenu->addAction(removeLibraryAction); + libraryMenu->addSeparator(); + + libraryMenu->addAction(exportComicsInfoAction); + libraryMenu->addAction(importComicsInfoAction); + + libraryMenu->addSeparator(); + + libraryMenu->addAction(exportLibraryAction); + libraryMenu->addAction(importLibraryAction); + + //folder + QMenu * folderMenu = new QMenu(tr("Folder")); + folderMenu->addAction(openContainingFolderAction); + folderMenu->addSeparator(); + folderMenu->addAction(setFolderAsNotCompletedAction); + folderMenu->addAction(setFolderAsCompletedAction); + folderMenu->addSeparator(); + folderMenu->addAction(setFolderAsReadAction); + folderMenu->addAction(setFolderAsUnreadAction); + + //comic + QMenu * comicMenu = new QMenu(tr("Comic")); + comicMenu->addAction(openContainingFolderComicAction); + comicMenu->addSeparator(); + comicMenu->addAction(resetComicRatingAction); + + menu->addMenu(libraryMenu); + menu->addMenu(folderMenu); + menu->addMenu(comicMenu); +#endif +} + +void LibraryWindow::createConnections() +{ + //history navigation + connect(backAction,SIGNAL(triggered()),this,SLOT(backward())); + connect(forwardAction,SIGNAL(triggered()),this,SLOT(forward())); + + //libraryCreator connections + connect(createLibraryDialog,SIGNAL(createLibrary(QString,QString,QString)),this,SLOT(create(QString,QString,QString))); + connect(createLibraryDialog,SIGNAL(libraryExists(QString)),this,SLOT(libraryAlreadyExists(QString))); + connect(importComicsInfoDialog,SIGNAL(finished(int)),this,SLOT(reloadCurrentLibrary())); + + //connect(libraryCreator,SIGNAL(coverExtracted(QString)),createLibraryDialog,SLOT(showCurrentFile(QString))); + //connect(libraryCreator,SIGNAL(coverExtracted(QString)),updateLibraryDialog,SLOT(showCurrentFile(QString))); + connect(libraryCreator,SIGNAL(finished()),this,SLOT(showRootWidget())); + connect(libraryCreator,SIGNAL(updated()),this,SLOT(reloadCurrentLibrary())); + connect(libraryCreator,SIGNAL(created()),this,SLOT(openLastCreated())); + connect(libraryCreator,SIGNAL(comicAdded(QString,QString)),importWidget,SLOT(newComic(QString,QString))); + //libraryCreator errors + connect(libraryCreator,SIGNAL(failedCreatingDB(QString)),this,SLOT(manageCreatingError(QString))); + connect(libraryCreator,SIGNAL(failedUpdatingDB(QString)),this,SLOT(manageUpdatingError(QString))); //TODO: implement failedUpdatingDB + + //new import widget + connect(importWidget,SIGNAL(stop()),this,SLOT(stopLibraryCreator())); + + //packageManager connections + connect(exportLibraryDialog,SIGNAL(exportPath(QString)),this,SLOT(exportLibrary(QString))); + connect(exportLibraryDialog,SIGNAL(rejected()),packageManager,SLOT(cancel())); + connect(packageManager,SIGNAL(exported()),exportLibraryDialog,SLOT(close())); + connect(importLibraryDialog,SIGNAL(unpackCLC(QString,QString,QString)),this,SLOT(importLibrary(QString,QString,QString))); + connect(importLibraryDialog,SIGNAL(rejected()),packageManager,SLOT(cancel())); + connect(importLibraryDialog,SIGNAL(rejected()),this,SLOT(deleteCurrentLibrary())); + connect(importLibraryDialog,SIGNAL(libraryExists(QString)),this,SLOT(libraryAlreadyExists(QString))); + connect(packageManager,SIGNAL(imported()),importLibraryDialog,SLOT(hide())); + connect(packageManager,SIGNAL(imported()),this,SLOT(openLastCreated())); + + + //create and update dialogs + connect(createLibraryDialog,SIGNAL(cancelCreate()),this,SLOT(cancelCreating())); + + //open existing library from dialog. + connect(addLibraryDialog,SIGNAL(addLibrary(QString,QString)),this,SLOT(openLibrary(QString,QString))); + + //load library when selected library changes + connect(selectedLibrary,SIGNAL(currentIndexChanged(QString)),this,SLOT(loadLibrary(QString))); + + //rename library dialog + connect(renameLibraryDialog,SIGNAL(renameLibrary(QString)),this,SLOT(rename(QString))); + + //navigations between view modes (tree,list and flow) + connect(foldersView, SIGNAL(pressed(QModelIndex)), this, SLOT(updateFoldersViewConextMenu(QModelIndex))); + connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(loadCovers(QModelIndex))); + connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(updateHistory(QModelIndex))); + + //actions + connect(createLibraryAction,SIGNAL(triggered()),this,SLOT(createLibrary())); + connect(exportLibraryAction,SIGNAL(triggered()),exportLibraryDialog,SLOT(show())); + connect(importLibraryAction,SIGNAL(triggered()),this,SLOT(importLibraryPackage())); + + connect(openLibraryAction,SIGNAL(triggered()),this,SLOT(showAddLibrary())); + connect(setAsReadAction,SIGNAL(triggered()),this,SLOT(setCurrentComicReaded())); + connect(setAsNonReadAction,SIGNAL(triggered()),this,SLOT(setCurrentComicUnreaded())); + //connect(setAllAsReadAction,SIGNAL(triggered()),this,SLOT(setComicsReaded())); + //connect(setAllAsNonReadAction,SIGNAL(triggered()),this,SLOT(setComicsUnreaded())); + + + //comicsInfoManagement + connect(exportComicsInfoAction,SIGNAL(triggered()),this,SLOT(showExportComicsInfo())); + connect(importComicsInfoAction,SIGNAL(triggered()),this,SLOT(showImportComicsInfo())); + + //properties & config + connect(propertiesDialog,SIGNAL(accepted()),this,SLOT(reloadCovers())); + + //comic vine + connect(comicVineDialog,SIGNAL(accepted()),this,SLOT(reloadCovers())); + + connect(updateLibraryAction,SIGNAL(triggered()),this,SLOT(updateLibrary())); + connect(renameLibraryAction,SIGNAL(triggered()),this,SLOT(renameLibrary())); + //connect(deleteLibraryAction,SIGNAL(triggered()),this,SLOT(deleteLibrary())); + connect(removeLibraryAction,SIGNAL(triggered()),this,SLOT(removeLibrary())); + connect(openComicAction,SIGNAL(triggered()),this,SLOT(openComic())); + connect(helpAboutAction,SIGNAL(triggered()),had,SLOT(show())); + connect(setRootIndexAction,SIGNAL(triggered()),this,SLOT(setRootIndex())); + connect(expandAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(expandAll())); + connect(colapseAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(collapseAll())); + connect(toggleFullScreenAction,SIGNAL(triggered()),this,SLOT(toggleFullScreen())); + connect(toggleComicsViewAction,SIGNAL(triggered()),this,SLOT(toggleComicsView())); + connect(optionsAction, SIGNAL(triggered()),optionsDialog,SLOT(show())); +#ifdef SERVER_RELEASE + connect(serverConfigAction, SIGNAL(triggered()), serverConfigDialog, SLOT(show())); +#endif + connect(optionsDialog, SIGNAL(optionsChanged()),this,SLOT(reloadOptions())); + connect(optionsDialog, SIGNAL(editShortcuts()),editShortcutsDialog,SLOT(show())); + + //Folders filter + //connect(clearFoldersFilter,SIGNAL(clicked()),foldersFilter,SLOT(clear())); + connect(foldersFilter,SIGNAL(textChanged(QString)),this,SLOT(setFoldersFilter(QString))); + //connect(includeComicsCheckBox,SIGNAL(stateChanged(int)),this,SLOT(searchInFiles(int))); + + //ContextMenus + connect(openContainingFolderComicAction,SIGNAL(triggered()),this,SLOT(openContainingFolderComic())); + connect(setFolderAsNotCompletedAction,SIGNAL(triggered()),this,SLOT(setFolderAsNotCompleted())); + connect(setFolderAsCompletedAction,SIGNAL(triggered()),this,SLOT(setFolderAsCompleted())); + connect(setFolderAsReadAction,SIGNAL(triggered()),this,SLOT(setFolderAsRead())); + connect(setFolderAsUnreadAction,SIGNAL(triggered()),this,SLOT(setFolderAsUnread())); + connect(openContainingFolderAction,SIGNAL(triggered()),this,SLOT(openContainingFolder())); + connect(resetComicRatingAction,SIGNAL(triggered()),this,SLOT(resetComicRating())); + + //connect(dm,SIGNAL(directoryLoaded(QString)),foldersView,SLOT(expandAll())); + //connect(dm,SIGNAL(directoryLoaded(QString)),this,SLOT(updateFoldersView(QString))); + //Comicts edition + connect(editSelectedComicsAction,SIGNAL(triggered()),this,SLOT(showProperties())); + connect(asignOrderAction,SIGNAL(triggered()),this,SLOT(asignNumbers())); + + connect(deleteComicsAction,SIGNAL(triggered()),this,SLOT(deleteComics())); + + connect(hideComicViewAction, SIGNAL(toggled(bool)),this, SLOT(hideComicFlow(bool))); + + connect(getInfoAction,SIGNAL(triggered()),this,SLOT(showComicVineScraper())); + + //connect(socialAction,SIGNAL(triggered()),this,SLOT(showSocial())); + + connect(comicsViewTransition,SIGNAL(transitionFinished()),this,SLOT(showComicsView())); + + connect(dmCV,SIGNAL(isEmpty()),this,SLOT(showEmptyFolderView())); + connect(emptyFolderWidget,SIGNAL(subfolderSelected(QModelIndex,int)),this,SLOT(selectSubfolder(QModelIndex,int))); + + connect(showEditShortcutsAction,SIGNAL(triggered()),editShortcutsDialog,SLOT(show())); +} + +void LibraryWindow::loadLibrary(const QString & name) +{ + if(!libraries.isEmpty()) //si hay bibliotecas... + { + currentFolderNavigation=0; + backAction->setDisabled(true); + forwardAction->setDisabled(true); + history.clear(); + history.append(QModelIndex()); + + showRootWidget(); + QString path=libraries.getPath(name)+"/.yacreaderlibrary"; + QDir d; //TODO change this by static methods (utils class?? with delTree for example) + QString dbVersion; + if(d.exists(path) && d.exists(path+"/library.ydb") && (dbVersion = DataBaseManagement::checkValidDB(path+"/library.ydb")) != "") //si existe en disco la biblioteca seleccionada, y es válida.. + { + int comparation = DataBaseManagement::compareVersions(dbVersion,VERSION); + bool updated = false; + if(comparation < 0) + { + int ret = QMessageBox::question(this,tr("Update needed"),tr("This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now?"),QMessageBox::Yes,QMessageBox::No); + if(ret == QMessageBox::Yes) + { + updated = DataBaseManagement::updateToCurrentVersion(path+"/library.ydb"); + if(!updated) + QMessageBox::critical(this,tr("Update failed"), tr("The current library can't be udpated. Check for write write permissions on: ") + path+"/library.ydb"); + } + else + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + disableAllActions();//TODO comprobar que se deben deshabilitar + //será posible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + } + } + + if(comparation == 0 || updated) //en caso de que la versión se igual que la actual + { + index = 0; + + dm->setupModelData(path); + foldersView->setModel(dm); + + if(dm->rowCount(QModelIndex())>0) + disableFoldersActions(false); + else + disableFoldersActions(true); + + d.setCurrent(libraries.getPath(name)); + d.setFilter(QDir::AllDirs | QDir::Files | QDir::Hidden | QDir::NoSymLinks | QDir::NoDotAndDotDot); + if(d.count()<=1) //librería de sólo lectura + { + //QMessageBox::critical(NULL,QString::number(d.count()),QString::number(d.count())); + disableLibrariesActions(false); + updateLibraryAction->setDisabled(true); + openContainingFolderAction->setDisabled(true); + disableComicsActions(true); + toggleFullScreenAction->setEnabled(true); + + importedCovers = true; + } + else //librería normal abierta + { + disableLibrariesActions(false); + importedCovers = false; + } + + setRootIndex(); + //TODO encontrar el bug que provoca que no se carguen adecuadamente las carátulas en root. + setRootIndex(); + + foldersFilter->clear(); + } + else if(comparation > 0) + { + int ret = QMessageBox::question(this,tr("Download new version"),tr("This library was created with a newer version of YACReaderLibrary. Download the new version now?"),QMessageBox::Yes,QMessageBox::No); + if(ret == QMessageBox::Yes) + QDesktopServices::openUrl(QUrl("http://www.yacreader.com")); + + comicsView->setModel(NULL); + foldersView->setModel(NULL); + disableAllActions();//TODO comprobar que se deben deshabilitar + //será posible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + } + } + else + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + disableAllActions();//TODO comprobar que se deben deshabilitar + + //si la librería no existe en disco, se ofrece al usuario la posibiliad de eliminarla + if(!d.exists(path)) + { + QString currentLibrary = selectedLibrary->currentText(); + if(QMessageBox::question(this,tr("Library not available"),tr("Library '%1' is no longer available. Do you want to remove it?").arg(currentLibrary),QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes) + { + deleteCurrentLibrary(); + } + //será posible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + + } + else//si existe el path, puede ser que la librería sea alguna versión pre-5.0 ó que esté corrupta o que no haya drivers sql + { + + if(d.exists(path+"/library.ydb")) + { + QSqlDatabase db = DataBaseManagement::loadDatabase(path); + manageOpeningLibraryError(db.lastError().databaseText() + "-" + db.lastError().driverText()); + //será posible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + } + else + { + QString currentLibrary = selectedLibrary->currentText(); + QString path = libraries.getPath(selectedLibrary->currentText()); + if(QMessageBox::question(this,tr("Old library"),tr("Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now?").arg(currentLibrary),QMessageBox::Yes,QMessageBox::No)==QMessageBox::Yes) + { + QDir d(path+"/.yacreaderlibrary"); + delTree(d); + d.rmdir(path+"/.yacreaderlibrary"); + createLibraryDialog->setDataAndStart(currentLibrary,path); + //create(path,path+"/.yacreaderlibrary",currentLibrary); + } + //será posible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + } + } + } + } + else //en caso de que no exista ninguna biblioteca se desactivan los botones pertinentes + { + disableAllActions(); + showNoLibrariesWidget(); + } +} + +void LibraryWindow::loadCovers(const QModelIndex & mi) +{ + unsigned long long int folderId = 1; + if(mi.isValid()) + { + TreeItem *item = static_cast(mi.internalPointer()); + folderId = item->id; +#ifndef Q_OS_MAC + libraryToolBar->setCurrentFolderName(item->data(0).toString()); +#endif + } +#ifndef Q_OS_MAC + else libraryToolBar->setCurrentFolderName(selectedLibrary->currentText()); +#endif + + + + //cambiado de orden, ya que al llamar a foldersFilter->clear() se invalidan los model index + if(foldersFilter->text()!="") + { + //setFoldersFilter(""); + if(mi.isValid()) + { + index = static_cast(mi.internalPointer())->originalItem; + column = mi.column(); + foldersFilter->clear(); + } + } + else + { + index = static_cast(mi.internalPointer()); + column = mi.column(); + } + + //comicsView->setModel(NULL); + dmCV->setupModelData(folderId,dm->getDatabase()); + + comicsView->setModel(dmCV); + QStringList paths = dmCV->getPaths(currentPath()); + checkEmptyFolder(&paths); + + if(paths.size()>0) { + comicsView->setCurrentIndex(dmCV->index(0,0)); + if(comicsViewStack->currentWidget() == emptyFolderWidget) + comicsViewStack->setCurrentWidget(comicsView); + } + else + emptyFolderWidget->setSubfolders(mi,dm->getSubfoldersNames(mi)); +} + +void LibraryWindow::selectSubfolder(const QModelIndex &mi, int child) +{ + QModelIndex dest = dm->index(child,0,mi); + foldersView->setCurrentIndex(dest); + updateHistory(dest); + loadCovers(dest); +} + +void LibraryWindow::checkEmptyFolder(QStringList * paths) +{ + if(paths == 0) + { + QStringList pathList = dmCV->getPaths(currentPath()); + paths = &pathList; + } + + if(paths->size()>0 && !importedCovers) + { + disableComicsActions(false); + } + else + { + disableComicsActions(true); + if(paths->size()>0) + toggleFullScreenAction->setEnabled(true); + } +} + +void LibraryWindow::reloadCovers() +{ + if(foldersView->selectionModel()->selectedRows().length()>0) + loadCovers(foldersView->currentIndex()); + else + loadCovers(QModelIndex()); +QLOG_INFO() << "reloaded covers at row : " << foldersView->currentIndex().row(); + QModelIndex mi = dmCV->getIndexFromId(_comicIdEdited); + if(mi.isValid()) + { + comicsView->scrollTo(mi,QAbstractItemView::PositionAtCenter); + comicsView->setCurrentIndex(mi); + } + //centerComicFlow(mi); +} + +void LibraryWindow::openComic() +{ + if(!importedCovers) + { + ComicDB comic = dmCV->getComic(comicsView->currentIndex()); + QString path = currentPath(); + QList siblings = dmCV->getAllComics(); + + quint64 comicId = comic.id; + //TODO generate IDS for libraries... + quint64 libraryId = selectedLibrary->currentIndex(); + + // %1 %2 %3 NO-->%4 %5 %6 %7 %8 %9 %10 + //Invoke YACReader comicPath comicId libraryId NO-->currentPage bookmark1 bookmark2 bookmark3 brightness contrast gamma + bool yacreaderFound = false; +#ifdef Q_OS_MAC + QString comicIdS = QString("%1").arg(comicId); + QString libraryIdS = QString("%1").arg(libraryId); + QString yacreaderPath = QDir::cleanPath(QCoreApplication::applicationDirPath()+"/../../../YACReader.app"); + if(yacreaderFound = QFileInfo(yacreaderPath).exists()) + QProcess::startDetached("open", QStringList() << "-n" << yacreaderPath << "--args" << path << comicIdS << libraryIdS ); /*<< page << bookmark1 << bookmark2 << bookmark3 << brightness << contrast << gamma*///,QStringList() << path); + +#endif + +#ifdef Q_OS_WIN /* \"%4\" \"%5\" \"%6\" \"%7\" \"%8\" \"%9\" \"%10\" */ + yacreaderFound = QProcess::startDetached(QDir::cleanPath(QCoreApplication::applicationDirPath())+QString("/YACReader \"%1\" \"%2\" \"%3\"").arg(path).arg(comicId).arg(libraryId)/*.arg(page).arg(bookmark1).arg(bookmark2).arg(bookmark3).arg(brightness).arg(contrast).arg(gamma)*/,QStringList()); +#endif + +#if defined Q_OS_UNIX && !defined Q_OS_MAC + QStringList parameters = QStringList() << path << QString::number(comicId) << QString::number(libraryId); + yacreaderFound = QProcess::startDetached(QString("YACReader"),parameters); +#endif + if(!yacreaderFound) + QMessageBox::critical(this,tr("YACReader not found"),tr("YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary.")); + + setCurrentComicOpened(); + } +} + +void LibraryWindow::setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus) { + dmCV->setComicsRead(getSelectedComics(),readStatus); +} + +void LibraryWindow::setCurrentComicReaded() { + this->setCurrentComicsStatusReaded(YACReader::Read); +} + +void LibraryWindow::setCurrentComicOpened() +{ + //TODO: remove? +} + +void LibraryWindow::setCurrentComicUnreaded() { + this->setCurrentComicsStatusReaded(YACReader::Unread); +} + +void LibraryWindow::createLibrary() { + createLibraryDialog->show(libraries); +} + +void LibraryWindow::create(QString source, QString dest, QString name) +{ + QLOG_INFO() << QString("About to create a library from '%1' to '%2' with name '%3'").arg(source).arg(dest).arg(name); + libraryCreator->createLibrary(source,dest); + libraryCreator->start(); + _lastAdded = name; + _sourceLastAdded = source; + + importWidget->setImportLook(); + showImportingWidget(); + +} + +void LibraryWindow::reloadCurrentLibrary() { + loadLibrary(selectedLibrary->currentText()); +} + +void LibraryWindow::openLastCreated() +{ + + selectedLibrary->disconnect(); + + selectedLibrary->setCurrentIndex(selectedLibrary->findText(_lastAdded)); + libraries.addLibrary(_lastAdded,_sourceLastAdded); + selectedLibrary->addItem(_lastAdded,_sourceLastAdded); + selectedLibrary->setCurrentIndex(selectedLibrary->findText(_lastAdded)); + libraries.save(); + + connect(selectedLibrary,SIGNAL(currentIndexChanged(QString)),this,SLOT(loadLibrary(QString))); + + loadLibrary(_lastAdded); +} + +void LibraryWindow::showAddLibrary() +{ + addLibraryDialog->show(); +} + +void LibraryWindow::openLibrary(QString path, QString name) +{ + if(!libraries.contains(name)) + { + //TODO: fix bug, /a/b/c/.yacreaderlibrary/d/e + path.remove("/.yacreaderlibrary"); + QDir d; //TODO change this by static methods (utils class?? with delTree for example) + if(d.exists(path + "/.yacreaderlibrary")) + { + _lastAdded = name; + _sourceLastAdded = path; + openLastCreated(); + addLibraryDialog->close(); + } + else + QMessageBox::warning(this,tr("Library not found"),tr("The selected folder doesn't contain any library.")); + } + else + { + libraryAlreadyExists(name); + } +} + +void LibraryWindow::loadLibraries() +{ + libraries.load(); + foreach(QString name,libraries.getNames()) + selectedLibrary->addItem(name,libraries.getPath(name)); +} + + +void LibraryWindow::saveLibraries() { + libraries.save(); +} + +void LibraryWindow::updateLibrary() +{ + importWidget->setUpdateLook(); + showImportingWidget(); + + QString currentLibrary = selectedLibrary->currentText(); + QString path = libraries.getPath(currentLibrary); + _lastAdded = currentLibrary; + libraryCreator->updateLibrary(path,path+"/.yacreaderlibrary"); + libraryCreator->start(); +} + +void LibraryWindow::deleteCurrentLibrary() +{ + QString path = libraries.getPath(selectedLibrary->currentText()); + libraries.remove(selectedLibrary->currentText()); + selectedLibrary->removeItem(selectedLibrary->currentIndex()); + //selectedLibrary->setCurrentIndex(0); + path = path+"/.yacreaderlibrary"; + + QDir d(path); + delTree(d); + d.rmdir(path); + if(libraries.isEmpty())//no more libraries avaliable. + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + + disableAllActions(); + showNoLibrariesWidget(); + } + libraries.save(); +} + +void LibraryWindow::removeLibrary() +{ + QString currentLibrary = selectedLibrary->currentText(); + QMessageBox * messageBox = new QMessageBox(tr("Are you sure?"),tr("Do you want remove ")+currentLibrary+tr(" library?"),QMessageBox::Question,QMessageBox::Yes,QMessageBox::YesToAll,QMessageBox::No); + messageBox->button(QMessageBox::YesToAll)->setText(tr("Remove and delete metadata")); + int ret = messageBox->exec(); + if(ret == QMessageBox::Yes) + { + libraries.remove(currentLibrary); + selectedLibrary->removeItem(selectedLibrary->currentIndex()); + //selectedLibrary->setCurrentIndex(0); + if(libraries.isEmpty())//no more libraries avaliable. + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + + disableAllActions(); + showNoLibrariesWidget(); + } + libraries.save(); + } + else if(ret == QMessageBox::YesToAll) + { + deleteCurrentLibrary(); + } + +} + +void LibraryWindow::renameLibrary() +{ + renameLibraryDialog->show(); +} + +void LibraryWindow::rename(QString newName) //TODO replace +{ + QString currentLibrary = selectedLibrary->currentText(); + if(newName != currentLibrary) + { + if(!libraries.contains(newName)) + { + libraries.rename(currentLibrary,newName); + //selectedLibrary->removeItem(selectedLibrary->currentIndex()); + //libraries.addLibrary(newName,path); + selectedLibrary->renameCurrentLibrary(newName); + libraries.save(); + renameLibraryDialog->close(); +#ifndef Q_OS_MAC + if(!foldersView->currentIndex().isValid()) + libraryToolBar->setCurrentFolderName(selectedLibrary->currentText()); +#endif + } + else + { + libraryAlreadyExists(newName); + } + } + else + renameLibraryDialog->close(); + //selectedLibrary->setCurrentIndex(selectedLibrary->findText(newName)); +} + +void LibraryWindow::cancelCreating() +{ + stopLibraryCreator(); +} + +void LibraryWindow::stopLibraryCreator() +{ + libraryCreator->stop(); + libraryCreator->wait(); +} + +void LibraryWindow::setRootIndex() +{ + if(!libraries.isEmpty()) + { + QString path=libraries.getPath(selectedLibrary->currentText())+"/.yacreaderlibrary"; + QDir d; //TODO change this by static methods (utils class?? with delTree for example) + if(d.exists(path)) + { + loadCovers(QModelIndex()); + if(history.count()>1) + updateHistory(QModelIndex()); + } + else + { + comicsView->setModel(NULL); + } + + foldersView->selectionModel()->clear(); + } + + setFolderAsNotCompletedAction->setVisible(false); + setFolderAsCompletedAction->setVisible(false); + setFolderAsReadAction->setVisible(false); + setFolderAsUnreadAction->setVisible(false); +} + + +void LibraryWindow::toggleFullScreen() +{ + fullscreen?toNormal():toFullScreen(); + fullscreen = !fullscreen; +} + +void LibraryWindow::toFullScreen() +{ + fromMaximized = this->isMaximized(); + + sideBar->hide(); + libraryToolBar->hide(); + + comicsView->toFullScreen(); + + showFullScreen(); +} + +void LibraryWindow::toNormal() +{ + sideBar->show(); + + comicsView->toNormal(); + + if(fromMaximized) + showMaximized(); + else + showNormal(); + +#ifdef Q_OS_MAC + QTimer * timer = new QTimer(); + timer->setSingleShot(true); + timer->start(); + connect(timer,SIGNAL(timeout()),libraryToolBar,SLOT(show())); + connect(timer,SIGNAL(timeout()),timer,SLOT(deleteLater())); +#else + libraryToolBar->show(); +#endif + +} + +void LibraryWindow::setFoldersFilter(QString filter) +{ + if(filter.isEmpty() && dm->isFilterEnabled()) + { + dm->resetFilter(); + //foldersView->collapseAll(); + if(index != 0) + { + QModelIndex mi = dm->indexFromItem(index,column); + foldersView->scrollTo(mi,QAbstractItemView::PositionAtTop); + updateHistory(mi); + foldersView->setCurrentIndex(mi); + } + } + else + { + if(!filter.isEmpty()) + { + dm->setFilter(filter, true);//includeComicsCheckBox->isChecked()); + foldersView->expandAll(); + } + } +} + +void LibraryWindow::showProperties() +{ + QModelIndexList indexList = getSelectedComics(); + + QList comics = dmCV->getComics(indexList); + ComicDB c = comics[0]; + _comicIdEdited = c.id;//static_cast(indexList[0].internalPointer())->data(4).toULongLong(); + + propertiesDialog->databasePath = dm->getDatabase(); + propertiesDialog->basePath = currentPath(); + propertiesDialog->setComics(comics); + + propertiesDialog->show(); +} + +void LibraryWindow::showComicVineScraper() +{ + QModelIndexList indexList = getSelectedComics(); + + QList comics = dmCV->getComics(indexList); + ComicDB c = comics[0]; + _comicIdEdited = c.id;//static_cast(indexList[0].internalPointer())->data(4).toULongLong(); + + comicVineDialog->databasePath = dm->getDatabase(); + comicVineDialog->basePath = currentPath(); + comicVineDialog->setComics(comics); + + comicVineDialog->show(); +} + +void LibraryWindow::setRemoveError() +{ + removeError = true; +} + +void LibraryWindow::checkRemoveError() +{ + if(removeError) + { + QMessageBox::critical(this,tr("Unable to delete"),tr("There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder.")); + } + removeError = false; +} + +void LibraryWindow::resetComicRating() +{ + QModelIndexList indexList = getSelectedComics(); + + dmCV->startTransaction(); + for(auto & index:indexList) + { + dmCV->resetComicRating(index); + } + dmCV->finishTransaction(); +} + +void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) +{ + disconnectComicsViewConnections(from); + from->close(); + + comicsView = to; + doComicsViewConnections(); + to->setItemActions(itemActions); + to->setViewActions(viewActions); + + comicsView->setToolBar(editInfoToolBar); + + comicsViewStack->removeWidget(from); + comicsViewStack->addWidget(comicsView); + + delete from; + + reloadCovers(); +} + +void LibraryWindow::showComicsViewTransition() +{ + comicsViewStack->setCurrentWidget(comicsViewTransition); + comicsViewTransition->startMovie(); +} + +void LibraryWindow::toggleComicsView_delayed() +{ + if(comicsViewStatus == Flow){ + QIcon icoViewsButton; + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/flow.png"), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); + switchToComicsView(classicComicsView, gridComicsView = new GridComicsView()); + comicsViewStatus = Grid; + } + else{ + QIcon icoViewsButton; + icoViewsButton.addPixmap(QPixmap(":/images/main_toolbar/grid.png"), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); + switchToComicsView(gridComicsView, classicComicsView = new ClassicComicsView()); + comicsViewStatus = Flow; + } + + settings->setValue(COMICS_VIEW_STATUS, comicsViewStatus); +} + +void LibraryWindow::showComicsView() +{ + comicsViewStack->setCurrentWidget(comicsView); +} + +void LibraryWindow::showEmptyFolderView() +{ + comicsViewStack->setCurrentWidget(emptyFolderWidget); +} + +//TODO recover the current comics selection and restore it in the destination +void LibraryWindow::toggleComicsView() +{ + if(comicsViewStack->currentWidget()!=emptyFolderWidget) { + QTimer::singleShot(0,this,SLOT(showComicsViewTransition())); + QTimer::singleShot(32,this,SLOT(toggleComicsView_delayed())); + } else + toggleComicsView_delayed(); +} + +void LibraryWindow::asignNumbers() +{ + QModelIndexList indexList = getSelectedComics(); + + int startingNumber = indexList[0].row()+1; + if(indexList.count()>1) + { + bool ok; + int n = QInputDialog::getInt(this, tr("Asign comics numbers"), + tr("Asign numbers starting in:"), startingNumber,0,2147483647,1,&ok); + if (ok) + startingNumber = n; + else + return; + } + _comicIdEdited = dmCV->asignNumbers(indexList,startingNumber); + + reloadCovers(); +} + +void LibraryWindow::openContainingFolderComic() +{ +QModelIndex modelIndex = comicsView->currentIndex(); +QFileInfo file = QDir::cleanPath(currentPath() + dmCV->getComicPath(modelIndex)); +#ifdef Q_OS_LINUX + QString path = file.absolutePath(); + QDesktopServices::openUrl(QUrl("file:///"+path, QUrl::TolerantMode)); +#endif + +#ifdef Q_OS_MAC + QString filePath = file.absoluteFilePath(); + QStringList args; + args << "-e"; + args << "tell application \"Finder\""; + args << "-e"; + args << "activate"; + args << "-e"; + args << "select POSIX file \""+filePath+"\""; + args << "-e"; + args << "end tell"; + QProcess::startDetached("osascript", args); +#endif + +#ifdef Q_OS_WIN + QString filePath = file.absoluteFilePath(); + QString cmdArgs = QString("/select,\"") + QDir::toNativeSeparators(filePath) + QStringLiteral("\""); + ShellExecuteW(0, L"open", L"explorer.exe", reinterpret_cast(cmdArgs.utf16()), 0, SW_NORMAL); +#endif +} + +void LibraryWindow::openContainingFolder() +{ + QModelIndex modelIndex = foldersView->currentIndex(); + QString path; + if(modelIndex.isValid()) + path = QDir::cleanPath(currentPath() + dm->getFolderPath(modelIndex)); + else + path = QDir::cleanPath(currentPath()); + QDesktopServices::openUrl(QUrl("file:///"+path, QUrl::TolerantMode)); +} + +void LibraryWindow::setFolderAsNotCompleted() +{ + dm->updateFolderCompletedStatus(foldersView->selectionModel()->selectedRows(),false); +} + +void LibraryWindow::setFolderAsCompleted() +{ + dm->updateFolderCompletedStatus(foldersView->selectionModel()->selectedRows(),true); +} + +void LibraryWindow::setFolderAsRead() +{ + dm->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),true); +} + +void LibraryWindow::setFolderAsUnread() +{ + dm->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),false); +} + +void LibraryWindow::exportLibrary(QString destPath) +{ + QString currentLibrary = selectedLibrary->currentText(); + QString path = libraries.getPath(currentLibrary)+"/.yacreaderlibrary"; + packageManager->createPackage(path,destPath+"/"+currentLibrary); +} + +void LibraryWindow::importLibrary(QString clc,QString destPath,QString name) +{ + packageManager->extractPackage(clc,destPath+"/"+name); + _lastAdded = name; + _sourceLastAdded = destPath+"/"+name; +} + +void LibraryWindow::reloadOptions() +{ + //comicFlow->setFlowType(flowType); + comicsView->updateConfig(settings); +} + +QString LibraryWindow::currentPath() +{ + return libraries.getPath(selectedLibrary->currentText()); +} + +//TODO ComicsView: some actions in the comics toolbar can be relative to a certain view +//show/hide actions on show/hide widget +void LibraryWindow::hideComicFlow(bool hide) +{ + /* + if(hide) + { + QList sizes; + sizes.append(0); + int total = sVertical->sizes().at(0) + sVertical->sizes().at(1); + sizes.append(total); + sVertical->setSizes(sizes); + } + else + { + QList sizes; + int total = sVertical->sizes().at(0) + sVertical->sizes().at(1); + sizes.append(2*total/3); + sizes.append(total/3); + sVertical->setSizes(sizes); + } +*/ +} + +void LibraryWindow::showExportComicsInfo() +{ + exportComicsInfoDialog->source = currentPath() + "/.yacreaderlibrary/library.ydb"; + exportComicsInfoDialog->show(); +} + +void LibraryWindow::showImportComicsInfo() +{ + importComicsInfoDialog->dest = currentPath() + "/.yacreaderlibrary/library.ydb"; + importComicsInfoDialog->show(); +} +#include "startup.h" +extern Startup * s; +void LibraryWindow::closeEvent ( QCloseEvent * event ) +{ + s->stop(); + settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry()); + + comicsView->close(); + + QApplication::instance()->processEvents(); + event->accept(); + QMainWindow::closeEvent(event); +} + +void LibraryWindow::showNoLibrariesWidget() +{ + disableAllActions(); + foldersFilter->setDisabled(true); + mainWidget->setCurrentIndex(1); +} + +void LibraryWindow::showRootWidget() +{ + libraryToolBar->setDisabled(false); + foldersFilter->setEnabled(true); + mainWidget->setCurrentIndex(0); +} + +void LibraryWindow::showImportingWidget() +{ + disableAllActions(); + importWidget->clear(); + libraryToolBar->setDisabled(true); + foldersFilter->setDisabled(true); + mainWidget->setCurrentIndex(2); +} + +void LibraryWindow::manageCreatingError(const QString & error) +{ + QMessageBox::critical(this,tr("Error creating the library"),error); +} + +void LibraryWindow::manageUpdatingError(const QString & error) +{ + QMessageBox::critical(this,tr("Error updating the library"),error); +} + +void LibraryWindow::manageOpeningLibraryError(const QString & error) +{ + QMessageBox::critical(this,tr("Error opening the library"),error); +} + +bool lessThanModelIndexRow(const QModelIndex & m1, const QModelIndex & m2) +{ + return m1.row()selectionModel()->selectedRows(); + QLOG_INFO() << "selection count " << selection.length(); + qSort(selection.begin(),selection.end(),lessThanModelIndexRow); + + /*if(selection.count()==0) + { + comicsView->selectRow(comicFlow->centerIndex()); + selection = comicsView->selectionModel()->selectedRows(); + }*/ + return selection; +} + +void LibraryWindow::deleteComics() +{ + int ret = QMessageBox::question(this,tr("Delete comics"),tr("All the selected comics will be deleted from your disk. Are you sure?"),QMessageBox::Yes,QMessageBox::No); + + if(ret == QMessageBox::Yes) + { + + QModelIndexList indexList = getSelectedComics(); + + QList comics = dmCV->getComics(indexList); + + QList paths; + QString libraryPath = currentPath(); + foreach(ComicDB comic, comics) + { + paths.append(libraryPath + comic.path); + QLOG_INFO() << comic.path; + QLOG_INFO() << comic.id; + QLOG_INFO() << comic.parentId; + } + + ComicsRemover * remover = new ComicsRemover(indexList,paths); + + //comicsView->showDeleteProgress(); + dmCV->startTransaction(); + + connect(remover, SIGNAL(remove(int)), dmCV, SLOT(remove(int))); + connect(remover,SIGNAL(removeError()),this,SLOT(setRemoveError())); + connect(remover, SIGNAL(finished()), dmCV, SLOT(finishTransaction())); + //connect(remover, SIGNAL(finished()), comicsView, SLOT(hideDeleteProgress())); + connect(remover, SIGNAL(finished()),this,SLOT(checkEmptyFolder())); + connect(remover, SIGNAL(finished()),this,SLOT(checkRemoveError())); + connect(remover, SIGNAL(finished()), remover, SLOT(deleteLater())); + //connect(remover, SIGNAL(finished()), thread, SLOT(deleteLater())); + + remover->start(); + } +} + +/* +void LibraryWindow::showSocial() +{ + socialDialog->move(this->mapToGlobal(QPoint(width()-socialDialog->width()-10, centralWidget()->pos().y()+10))); + + QModelIndexList indexList = getSelectedComics(); + + ComicDB comic = dmCV->getComic(indexList.at(0)); + + socialDialog->setComic(comic,currentPath()); + socialDialog->setHidden(false); +}*/ + +void LibraryWindow::backward() +{ + if(currentFolderNavigation>0) + { + currentFolderNavigation--; + loadCovers(history.at(currentFolderNavigation)); + foldersView->setCurrentIndex(history.at(currentFolderNavigation)); + forwardAction->setEnabled(true); + } + if(currentFolderNavigation==0) + { + backAction->setEnabled(false); + } +} + +void LibraryWindow::forward() +{ + if(currentFolderNavigationsetCurrentIndex(history.at(currentFolderNavigation)); + backAction->setEnabled(true); + } + if(currentFolderNavigation==history.count()-1) + { + forwardAction->setEnabled(false); + } +} + +void LibraryWindow::updateHistory(const QModelIndex &mi) +{ + //remove history from current index + if(!mi.isValid()) + return; + int numElementsToRemove = history.count() - (currentFolderNavigation+1); + while(numElementsToRemove>0) + { + numElementsToRemove--; + history.removeLast(); + } + + if(mi!=history.at(currentFolderNavigation)) + { + history.append(mi); + + backAction->setEnabled(true); + currentFolderNavigation++; + } + + forwardAction->setEnabled(false); +} + +void LibraryWindow::updateFoldersViewConextMenu(const QModelIndex &mi) +{ + if(!mi.isValid()) + return; + + TreeItem * item = static_cast(mi.internalPointer()); + bool isFinished = item->data(TreeModel::Finished).toBool(); + bool isCompleted = item->data(TreeModel::Completed).toBool(); + + setFolderAsReadAction->setVisible(!isFinished); + setFolderAsUnreadAction->setVisible(isFinished); + + setFolderAsCompletedAction->setVisible(!isCompleted); + setFolderAsNotCompletedAction->setVisible(isCompleted); +} + +void LibraryWindow::libraryAlreadyExists(const QString & name) +{ + QMessageBox::information(this,tr("Library name already exists"),tr("There is another library with the name '%1'.").arg(name)); +} + +void LibraryWindow::importLibraryPackage() +{ + importLibraryDialog->show(libraries); +} + +void LibraryWindow::updateComicsView(quint64 libraryId, const ComicDB & comic) +{ + //TODO comprobar la biblioteca.... + if(libraryId == selectedLibrary->currentIndex()) { + dmCV->reload(comic); + } +} diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h new file mode 100644 index 00000000..b7761820 --- /dev/null +++ b/YACReaderLibrary/library_window.h @@ -0,0 +1,323 @@ +#ifndef __LIBRARYWINDOW_H +#define __LIBRARYWINDOW_H + +#include +#include +#include +#include +#include "yacreader_global.h" +#include + +class QTreeView; +class QDirModel; +class QAction; +class QToolBar; +class QComboBox; +class QThread; +class QStackedWidget; +class YACReaderSearchLineEdit; +class CreateLibraryDialog; +class ExportLibraryDialog; +class ImportLibraryDialog; +class ExportComicsInfoDialog; +class ImportComicsInfoDialog; +class AddLibraryDialog; +class LibraryCreator; +class HelpAboutDialog; +class RenameLibraryDialog; +class PropertiesDialog; +class PackageManager; +class QCheckBox; +class QPushButton; +class TableModel; +class QSplitter; +class TreeItem; +class TreeModel; +class QItemSelectionModel; +class QString; +class QLabel; +class NoLibrariesWidget; +class OptionsDialog; +class ServerConfigDialog; +class QCloseEvent; +class ImportWidget; +class QSettings; +class LibraryItem; +class YACReaderTableView; +class YACReaderSideBar; +class YACReaderLibraryListWidget; +class YACReaderTreeView; +class YACReaderMainToolBar; +class ComicVineDialog; +class ComicsView; +class ClassicComicsView; +class GridComicsView; +class ComicsViewTransition; +class EmptyFolderWidget; +class EditShortcutsDialog; + +#include "comic_db.h" + +using namespace YACReader; + +class LibraryWindow : public QMainWindow +{ + Q_OBJECT +private: + YACReaderSideBar * sideBar; + + CreateLibraryDialog * createLibraryDialog; + ExportLibraryDialog * exportLibraryDialog; + ImportLibraryDialog * importLibraryDialog; + ExportComicsInfoDialog * exportComicsInfoDialog; + ImportComicsInfoDialog * importComicsInfoDialog; + AddLibraryDialog * addLibraryDialog; + LibraryCreator * libraryCreator; + HelpAboutDialog * had; + RenameLibraryDialog * renameLibraryDialog; + PropertiesDialog * propertiesDialog; + ComicVineDialog * comicVineDialog; + EditShortcutsDialog * editShortcutsDialog; + //YACReaderSocialDialog * socialDialog; + bool fullscreen; + bool importedCovers; //if true, the library is read only (not updates,open comic or properties) + bool fromMaximized; + //Ya no se usan proxies, el rendimiento de la BD es suficiente + //YACReaderTreeSearch * proxyFilter; + //YACReaderSortComics * proxySort; + PackageManager * packageManager; + + QSize slideSizeW; + QSize slideSizeF; + //search filter + YACReaderSearchLineEdit * foldersFilter; + TreeItem * index; //index al que hay que hacer scroll después de pulsar sobre un folder filtrado + int column; + QString previousFilter; + QPushButton * clearFoldersFilter; + QCheckBox * includeComicsCheckBox; + //------------- + + ComicsView * comicsView; + ClassicComicsView * classicComicsView; + GridComicsView * gridComicsView; + QStackedWidget * comicsViewStack; + ComicsViewTransition * comicsViewTransition; + EmptyFolderWidget * emptyFolderWidget; + + YACReaderTreeView * foldersView; + YACReaderLibraryListWidget * selectedLibrary; + TreeModel * dm; + TableModel * dmCV; + //QStringList paths; + YACReaderLibraries libraries; + + QStackedWidget * mainWidget; + NoLibrariesWidget * noLibrariesWidget; + ImportWidget * importWidget; + + bool fetching; + + int i; + + QAction * backAction; + QAction * forwardAction; + + QAction * openComicAction; + QAction * createLibraryAction; + QAction * openLibraryAction; + + QAction * exportComicsInfoAction; + QAction * importComicsInfoAction; + + QAction * exportLibraryAction; + QAction * importLibraryAction; + + QAction * updateLibraryAction; + QAction * removeLibraryAction; + QAction * helpAboutAction; + QAction * renameLibraryAction; + QAction * toggleFullScreenAction; + QAction * optionsAction; + QAction * serverConfigAction; + QAction * toggleComicsViewAction; + //QAction * socialAction; + + //tree actions + QAction * setRootIndexAction; + QAction * expandAllNodesAction; + QAction * colapseAllNodesAction; + + QAction * openContainingFolderAction; + //-- + QAction * setFolderAsNotCompletedAction; + QAction * setFolderAsCompletedAction; + //-- + QAction * setFolderAsReadAction; + QAction * setFolderAsUnreadAction; + + QAction * openContainingFolderComicAction; + QAction * setAsReadAction; + QAction * setAsNonReadAction; + //QAction * setAllAsReadAction; + //QAction * setAllAsNonReadAction; + QAction * showHideMarksAction; + QAction * getInfoAction; //comic vine + QAction * resetComicRatingAction; + + //edit info actions + QAction * selectAllComicsAction; + QAction * editSelectedComicsAction; + QAction * asignOrderAction; + QAction * forceCoverExtractedAction; + QAction * deleteComicsAction; + QAction * hideComicViewAction; + + QAction *showEditShortcutsAction; + + QList itemActions; + QList viewActions; + +#ifdef Q_OS_MAC + QToolBar * libraryToolBar; +#else + YACReaderMainToolBar * libraryToolBar; +#endif + QToolBar * treeActions; + QToolBar * comicsToolBar; + QToolBar * editInfoToolBar; + + OptionsDialog * optionsDialog; + ServerConfigDialog * serverConfigDialog; + + QString libraryPath; + QString comicsPath; + + QString _lastAdded; + QString _sourceLastAdded; + + //QModelIndex _rootIndex; + //QModelIndex _rootIndexCV; + + quint64 _comicIdEdited; + + void setupUI(); + void createActions(); + void createToolBars(); + void createMenus(); + void createConnections(); + void doLayout(); + void doDialogs(); + void setUpShortcutsManagement(); + void doModels(); + void disconnectComicsViewConnections(ComicsView * widget); + void doComicsViewConnections(); + + + //ACTIONS MANAGEMENT + void disableComicsActions(bool disabled); + void disableLibrariesActions(bool disabled); + void disableNoUpdatedLibrariesActions(bool disabled); + void disableFoldersActions(bool disabled); + + void disableAllActions(); + //void disableActions(); + //void enableActions(); + //void enableLibraryActions(); + + QString currentPath(); + + //settings + QSettings * settings; + + //navigation backward and forward + int currentFolderNavigation; + QList history; + + bool removeError; + + ComicsViewStatus comicsViewStatus; + +protected: + virtual void closeEvent ( QCloseEvent * event ); +public: + LibraryWindow(); + +public slots: + void loadLibrary(const QString & path); + void loadCovers(const QModelIndex & mi); + void selectSubfolder(const QModelIndex & mi, int child); + void checkEmptyFolder(QStringList * paths = 0); + void reloadCovers(); + void openComic(); + void createLibrary(); + void create(QString source,QString dest, QString name); + void showAddLibrary(); + void openLibrary(QString path, QString name); + void loadLibraries(); + void saveLibraries(); + void reloadCurrentLibrary(); + void openLastCreated(); + void updateLibrary(); + //void deleteLibrary(); + void openContainingFolder(); + void setFolderAsNotCompleted(); + void setFolderAsCompleted(); + void setFolderAsRead(); + void setFolderAsUnread(); + void openContainingFolderComic(); + void deleteCurrentLibrary(); + void removeLibrary(); + void renameLibrary(); + void rename(QString newName); + void cancelCreating(); + void stopLibraryCreator(); + void setRootIndex(); + void toggleFullScreen(); + void toNormal(); + void toFullScreen(); + void setFoldersFilter(QString filter); + void showProperties(); + void exportLibrary(QString destPath); + void importLibrary(QString clc,QString destPath,QString name); + void reloadOptions(); + void setCurrentComicsStatusReaded(YACReaderComicReadStatus readStatus); + void setCurrentComicReaded(); + void setCurrentComicUnreaded(); + void hideComicFlow(bool hide); + void showExportComicsInfo(); + void showImportComicsInfo(); + void asignNumbers(); + void showNoLibrariesWidget(); + void showRootWidget(); + void showImportingWidget(); + void manageCreatingError(const QString & error); + void manageUpdatingError(const QString & error); + void manageOpeningLibraryError(const QString & error); + QModelIndexList getSelectedComics(); + void deleteComics(); + //void showSocial(); + void backward(); + void forward(); + void updateHistory(const QModelIndex & mi); + void updateFoldersViewConextMenu(const QModelIndex & mi); + void libraryAlreadyExists(const QString & name); + void importLibraryPackage(); + void updateComicsView(quint64 libraryId, const ComicDB & comic); + void setCurrentComicOpened(); + void showComicVineScraper(); + void setRemoveError(); + void checkRemoveError(); + void resetComicRating(); + void switchToComicsView(ComicsView *from, ComicsView *to); + void showComicsViewTransition(); + void toggleComicsView_delayed();//used in orther to avoid flickering; + void showComicsView(); + void showEmptyFolderView(); + void toggleComicsView(); +}; + +#endif + + + diff --git a/YACReaderLibrary/main.cpp b/YACReaderLibrary/main.cpp new file mode 100644 index 00000000..1b90312a --- /dev/null +++ b/YACReaderLibrary/main.cpp @@ -0,0 +1,226 @@ +#include "library_window.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" +#include "startup.h" +#include "yacreader_local_server.h" +#include "comic_db.h" +#include "db_helper.h" +#include "yacreader_libraries.h" +#include "exit_check.h" + +#include "QsLog.h" +#include "QsLogDest.h" + +#define PICTUREFLOW_QT4 1 + +//interfaz al servidor +Startup * s; + +using namespace QsLogging; + +void logSystemAndConfig() +{ + QLOG_INFO() << "---------- System & configuration ----------"; +#if defined(Q_OS_WIN) + switch (QSysInfo::windowsVersion()) + { + case QSysInfo::WV_NT: + QLOG_INFO() << "SO : Windows NT"; + break; + case QSysInfo::WV_2000: + QLOG_INFO() << "SO : Windows 2000"; + break; + case QSysInfo::WV_XP: + QLOG_INFO() << "SO : Windows XP"; + break; + case QSysInfo::WV_2003: + QLOG_INFO() << "SO : Windows 2003"; + break; + case QSysInfo::WV_VISTA: + QLOG_INFO() << "SO : Windows Vista"; + break; + case QSysInfo::WV_WINDOWS7: + QLOG_INFO() << "SO : Windows 7"; + break; + case QSysInfo::WV_WINDOWS8: + QLOG_INFO() << "SO : Windows 8"; + break; + default: + QLOG_INFO() << "Windows (unknown version)"; + break; + } + +#elif defined(Q_OS_MAC) + + switch (QSysInfo::MacVersion()) + { + case QSysInfo::MV_SNOWLEOPARD: + QLOG_INFO() << "SO : MacOSX Snow Leopard"; + break; + case QSysInfo::MV_LION: + QLOG_INFO() << "SO : MacOSX Lion"; + break; + case QSysInfo::MV_MOUNTAINLION: + QLOG_INFO() << "SO : MacOSX Mountain Lion"; + break; +#if QT_VERSION >= 0x050000 + case QSysInfo::MV_MAVERICKS: + QLOG_INFO() << "SO : MacOSX Maverics"; + break; +#endif + default: + QLOG_INFO() << "SO : MacOSX (unknown version)"; + break; + } + +#elif defined(Q_OS_LINUX) + QLOG_INFO() << "SO : Linux (unknown version)"; + +#else + QLOG_INFO() << "SO : Unknown"; +#endif + +#ifdef Q_OS_WIN + if(QLibrary::isLibrary(QApplication::applicationDirPath()+"/utils/7z.dll")) +#elif defined Q_OS_UNIX && !defined Q_OS_MAC + if(QLibrary::isLibrary(QString(LIBDIR)+"/p7zip/7z.so")) +#else + if(QLibrary::isLibrary(QApplication::applicationDirPath()+"/utils/7z.so")) +#endif + QLOG_INFO() << "7z : found"; + else + QLOG_ERROR() << "7z : not found"; +#if defined Q_OS_UNIX && !defined Q_OS_MAC + if(QFileInfo(QString(BINDIR)+"/qrencode").exists()) +#else + if(QFileInfo(QApplication::applicationDirPath()+"/utils/qrencode.exe").exists() || QFileInfo("./util/qrencode").exists()) +#endif + QLOG_INFO() << "qrencode : found"; + else + QLOG_INFO() << "qrencode : not found"; + + QSettings settings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + settings.beginGroup("libraryConfig"); + if(settings.value(SERVER_ON,true).toBool()) + QLOG_INFO() << "server : enabled"; + else + QLOG_INFO() << "server : disabled"; + + if(settings.value(USE_OPEN_GL).toBool()) + QLOG_INFO() << "OpenGL : enabled" << " - " << (settings.value(V_SYNC).toBool()?"VSync on":"VSync off"); + else + QLOG_INFO() << "OpenGL : disabled"; + + QLOG_INFO() << "Libraries: " << DBHelper::getLibraries().getLibraries(); + QLOG_INFO() << "--------------------------------------------"; +} + +int main( int argc, char ** argv ) +{ +//fix for misplaced text in Qt4.8 and Mavericks +#ifdef Q_OS_MAC + #if QT_VERSION < 0x050000 + if(QSysInfo::MacintoshVersion > QSysInfo::MV_10_8) + QFont::insertSubstitution(".Lucida Grande UI", "Lucida Grande"); + #endif +#endif + + QApplication app( argc, argv ); + + app.setApplicationName("YACReaderLibrary"); + app.setOrganizationName("YACReader"); + + QString destLog = YACReader::getSettingsPath()+"/yacreaderlibrary.log"; + QDir().mkpath(YACReader::getSettingsPath()); + + Logger& logger = Logger::instance(); + logger.setLoggingLevel(QsLogging::TraceLevel); + + DestinationPtr fileDestination(DestinationFactory::MakeFileDestination( + destLog, EnableLogRotation, MaxSizeBytes(1048576), MaxOldLogCount(2))); + DestinationPtr debugDestination(DestinationFactory::MakeDebugOutputDestination()); + logger.addDestination(debugDestination); + logger.addDestination(fileDestination); + + QTranslator translator; + QString sufix = QLocale::system().name(); +#if defined Q_OS_UNIX && !defined Q_OS_MAC + translator.load(QString(DATADIR) +"/YACReader/languages/yacreaderlibrary_"+sufix); +#else + translator.load(QCoreApplication::applicationDirPath()+"/languages/yacreaderlibrary_"+sufix); +#endif + app.installTranslator(&translator); + + QTranslator viewerTranslator; +#if defined Q_OS_UNIX && !defined Q_OS_MAC + viewerTranslator.load(QString(DATADIR)+"/YACReader/languages/yacreader_"+sufix); +#else + viewerTranslator.load(QCoreApplication::applicationDirPath()+"/languages/yacreader_"+sufix); +#endif + app.installTranslator(&viewerTranslator); + app.setApplicationName("YACReaderLibrary"); + + qRegisterMetaType("ComicDB"); + +#ifdef SERVER_RELEASE + QSettings * settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creaci�n del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + + s = new Startup(); + + if(settings->value(SERVER_ON,true).toBool()) + { + + s->start(); + } +#endif + QLOG_INFO() << "YACReaderLibrary attempting to start"; + + logSystemAndConfig(); + + if(YACReaderLocalServer::isRunning()) //s�lo se permite una instancia de YACReaderLibrary + { + QLOG_WARN() << "another instance of YACReaderLibrary is running"; + QsLogging::Logger::destroyInstance(); + return 0; + } + QLOG_INFO() << "YACReaderLibrary starting"; + + YACReaderLocalServer * localServer = new YACReaderLocalServer(); + + LibraryWindow * mw = new LibraryWindow(); + + mw->connect(localServer,SIGNAL(comicUpdated(quint64, const ComicDB &)),mw,SLOT(updateComicsView(quint64, const ComicDB &))); + + //connections to localServer + + mw->show(); + + int ret = app.exec(); + + QLOG_INFO() << "YACReaderLibrary closed with exit code :" << ret; + + YACReader::exitCheck(ret); + + //shutdown + s->stop(); + delete s; + localServer->close(); + delete localServer; + delete mw; + + QsLogging::Logger::destroyInstance(); + + return ret; +} diff --git a/YACReaderLibrary/no_libraries_widget.cpp b/YACReaderLibrary/no_libraries_widget.cpp new file mode 100644 index 00000000..21dfb166 --- /dev/null +++ b/YACReaderLibrary/no_libraries_widget.cpp @@ -0,0 +1,80 @@ +#include "no_libraries_widget.h" + +#include +#include +#include +#include + +NoLibrariesWidget::NoLibrariesWidget(QWidget *parent) : + QWidget(parent) +{ + setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); + + QPalette p(palette()); + p.setColor(QPalette::Background, QColor(250,250,250)); + setAutoFillBackground(true); + setPalette(p); + + QPixmap icon(":/images/noLibrariesIcon.png"); + QLabel * iconLabel = new QLabel(); + iconLabel->setPixmap(icon); + + QPixmap line(":/images/noLibrariesLine.png"); + QLabel * lineLabel = new QLabel(); + lineLabel->setPixmap(line); + + QLabel * text = new QLabel(""+tr("You don't have any librarires yet")+""); + text->setStyleSheet("QLabel {font-size:25px;font-weight:bold;}"); + QLabel * textDescription = new QLabel(""+tr("

You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.

Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.

")+"
"); + textDescription->setWordWrap(true); + textDescription->setMaximumWidth(330); + + QPushButton * createButton = new QPushButton(tr("create your first library")); + createButton->setSizePolicy(QSizePolicy::Preferred,QSizePolicy::Preferred); + QPushButton * addButton = new QPushButton(tr("add an existing one")); + addButton->setSizePolicy(QSizePolicy::Preferred,QSizePolicy::Preferred); + + QVBoxLayout * layout = new QVBoxLayout(this); + QHBoxLayout * buttonLayout = new QHBoxLayout(); + QHBoxLayout * topLayout = new QHBoxLayout(); + QVBoxLayout * textLayout = new QVBoxLayout(); + + QWidget * topWidget = new QWidget(); + topWidget->setFixedWidth(650); + textLayout->addStretch(); + textLayout->addWidget(text); + textLayout->addSpacing(12); + textLayout->addWidget(textDescription); + textLayout->addStretch(); + + topLayout->addStretch(); + topLayout->addWidget(iconLabel,0,Qt::AlignVCenter); + topLayout->addSpacing(30); + topLayout->addLayout(textLayout,1); + topLayout->addStretch(); + topLayout->setMargin(0); + + topWidget->setLayout(topLayout); + + layout->setAlignment(Qt::AlignHCenter); + + buttonLayout->addSpacing(125); + buttonLayout->addWidget(createButton); + layout->addSpacing(25); + buttonLayout->addWidget(addButton); + buttonLayout->addSpacing(125); + + layout->addStretch(); + layout->addWidget(topWidget); + layout->addSpacing(20); + layout->addWidget(lineLabel,0,Qt::AlignHCenter); + layout->addSpacing(10); + layout->addLayout(buttonLayout,0); + layout->addSpacing(150); + layout->addStretch(); + + connect(createButton,SIGNAL(clicked()),this,SIGNAL(createNewLibrary())); + connect(addButton,SIGNAL(clicked()),this,SIGNAL(addExistingLibrary())); + + +} diff --git a/YACReaderLibrary/no_libraries_widget.h b/YACReaderLibrary/no_libraries_widget.h new file mode 100644 index 00000000..c522944b --- /dev/null +++ b/YACReaderLibrary/no_libraries_widget.h @@ -0,0 +1,19 @@ +#ifndef NO_LIBRARIES_WIDGET_H +#define NO_LIBRARIES_WIDGET_H + +#include + +class NoLibrariesWidget : public QWidget +{ + Q_OBJECT +public: + explicit NoLibrariesWidget(QWidget *parent = 0); + +signals: + void createNewLibrary(); + void addExistingLibrary(); +public slots: + +}; + +#endif // NO_LIBRARIES_WIDGET_H diff --git a/YACReaderLibrary/options_dialog.cpp b/YACReaderLibrary/options_dialog.cpp new file mode 100644 index 00000000..98794977 --- /dev/null +++ b/YACReaderLibrary/options_dialog.cpp @@ -0,0 +1,69 @@ +#include "options_dialog.h" + +#include "yacreader_flow_gl.h" +#include "yacreader_flow_config_widget.h" +#include "yacreader_gl_flow_config_widget.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +FlowType flowType = Strip; + +OptionsDialog::OptionsDialog(QWidget * parent) +:YACReaderOptionsDialog(parent) +{ + QTabWidget * tabWidget = new QTabWidget(); + + QVBoxLayout * layout = new QVBoxLayout(this); + + QVBoxLayout * flowLayout = new QVBoxLayout; + QVBoxLayout * generalLayout = new QVBoxLayout(); + + QHBoxLayout * switchFlowType = new QHBoxLayout; + switchFlowType->addStretch(); + switchFlowType->addWidget(useGL); + + QHBoxLayout * buttons = new QHBoxLayout(); + buttons->addStretch(); + buttons->addWidget(accept); + buttons->addWidget(cancel); + + flowLayout->addWidget(sw); + flowLayout->addWidget(gl); + flowLayout->addLayout(switchFlowType); + + sw->hide(); + + QWidget * comicFlowW = new QWidget; + comicFlowW->setLayout(flowLayout); + + QWidget * generalW = new QWidget; + generalW->setLayout(generalLayout); + generalLayout->addWidget(shortcutsBox); + generalLayout->addStretch(); + + tabWidget->addTab(comicFlowW,tr("Comic Flow")); + tabWidget->addTab(generalW,tr("General")); + + layout->addWidget(tabWidget); + layout->addLayout(buttons); + setLayout(layout); + //restoreOptions(settings); //load options + //resize(200,0); + setModal (true); + setWindowTitle(tr("Options")); + + this->layout()->setSizeConstraint(QLayout::SetFixedSize); + +} + + + diff --git a/YACReaderLibrary/options_dialog.h b/YACReaderLibrary/options_dialog.h new file mode 100644 index 00000000..d42a9ead --- /dev/null +++ b/YACReaderLibrary/options_dialog.h @@ -0,0 +1,18 @@ +#ifndef __OPTIONS_DIALOG_H +#define __OPTIONS_DIALOG_H + +#include "yacreader_options_dialog.h" + +#include "yacreader_global.h" + +using namespace YACReader; + +class OptionsDialog : public YACReaderOptionsDialog +{ +Q_OBJECT + public: + OptionsDialog(QWidget * parent = 0); +}; + + +#endif diff --git a/YACReaderLibrary/package_manager.cpp b/YACReaderLibrary/package_manager.cpp new file mode 100644 index 00000000..d5f21ef9 --- /dev/null +++ b/YACReaderLibrary/package_manager.cpp @@ -0,0 +1,55 @@ +#include "package_manager.h" +#include + +PackageManager::PackageManager() +:_7z(0) +{ + +} + +void PackageManager::createPackage(const QString & libraryPath,const QString & dest) +{ + QStringList attributes; + attributes << "a" << "-y" << "-ttar" << dest+".clc" << libraryPath ; + _7z = new QProcess(); + connect(_7z,SIGNAL(error(QProcess::ProcessError)),this,SLOT(openingError(QProcess::ProcessError))); + connect(_7z,SIGNAL(finished(int,QProcess::ExitStatus)),this,SIGNAL(exported())); +#if defined Q_OS_UNIX && !defined Q_OS_MAC + _7z->start("7z",attributes); //TODO: use 7z.so +#else + _7z->start(QCoreApplication::applicationDirPath()+"/utils/7zip",attributes); //TODO: use 7z.dll +#endif +} + +void PackageManager::extractPackage(const QString & packagePath,const QString & destDir) +{ + QStringList attributes; + QString output = "-o"; + output += destDir; + attributes << "x" << "-y" << output << packagePath; + _7z = new QProcess(); + connect(_7z,SIGNAL(error(QProcess::ProcessError)),this,SLOT(openingError(QProcess::ProcessError))); + connect(_7z,SIGNAL(finished(int,QProcess::ExitStatus)),this,SIGNAL(imported())); +#if defined Q_OS_UNIX && !defined Q_OS_MAC + _7z->start("7z",attributes); //TODO: use 7z.so +#else + _7z->start(QCoreApplication::applicationDirPath()+"/utils/7zip",attributes); //TODO: use 7z.dll +#endif +} + +void PackageManager::cancel() +{ + if(_7z!=0) + { + _7z->disconnect(); + _7z->kill(); + if(creating) + { + //TODO remove dest+".clc" + } + else + { + //TODO fixed: is done by libraryWindow + } + } +} diff --git a/YACReaderLibrary/package_manager.h b/YACReaderLibrary/package_manager.h new file mode 100644 index 00000000..235651ef --- /dev/null +++ b/YACReaderLibrary/package_manager.h @@ -0,0 +1,24 @@ +#ifndef PACKAGE_MANAGER_H +#define PACKAGE_MANAGER_H + +#include + +class PackageManager : public QObject +{ + Q_OBJECT +public: + PackageManager(); + void createPackage(const QString & libraryPath,const QString & dest); + void extractPackage(const QString & packagePath,const QString & destDir); + public slots: + void cancel(); +private: + bool creating; + QProcess * _7z; + +signals: + void exported(); + void imported(); +}; + +#endif diff --git a/YACReaderLibrary/properties_dialog.cpp b/YACReaderLibrary/properties_dialog.cpp new file mode 100644 index 00000000..8101c2a6 --- /dev/null +++ b/YACReaderLibrary/properties_dialog.cpp @@ -0,0 +1,896 @@ +#include "properties_dialog.h" + +#include "data_base_management.h" +#include "library_creator.h" +#include "yacreader_field_edit.h" +#include "yacreader_field_plain_text_edit.h" +#include "db_helper.h" +//#include "yacreader_busy_widget.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +PropertiesDialog::PropertiesDialog(QWidget * parent) +:QDialog(parent) +{ + + createCoverBox(); + createGeneralInfoBox(); + createAuthorsBox(); + createPublishingBox(); + createButtonBox(); + createPlotBox(); + + createTabBar(); + + mainLayout = new QGridLayout; + //mainLayout->addWidget(coverBox,0,0); + mainLayout->addWidget(tabBar,0,1); + mainLayout->setColumnStretch(1,1); + /*mainLayout->addWidget(authorsBox,1,1); + mainLayout->addWidget(publishingBox,2,1);*/ + mainLayout->addWidget(buttonBox,1,1,Qt::AlignBottom); + + mainWidget = new QWidget(this); + mainWidget->setAutoFillBackground(true); + mainWidget->setFixedSize(470,444); + mainWidget->setLayout(mainLayout); + mainLayout->setSizeConstraint(QLayout::SetMinimumSize); + + int heightDesktopResolution = QApplication::desktop()->screenGeometry().height(); + int widthDesktopResolution = QApplication::desktop()->screenGeometry().width(); + int sHeight,sWidth; + sHeight = static_cast(heightDesktopResolution*0.65); + sWidth = static_cast(sHeight*1.4); + //setCover(QPixmap(":/images/notCover.png")); + + this->move(QPoint((widthDesktopResolution-sWidth)/2,((heightDesktopResolution-sHeight)-40)/2)); + setModal(true); + + setFixedSize( sizeHint() ); + mainWidget->move(280,0); +} + +QSize PropertiesDialog::sizeHint() +{ + return QSize(750,444); +} + +void PropertiesDialog::createTabBar() +{ + tabBar = new QTabWidget; + tabBar->addTab(generalInfoBox,tr("General info")); + tabBar->addTab(authorsBox,tr("Authors")); + tabBar->addTab(publishingBox,tr("Publishing")); + tabBar->addTab(plotBox,tr("Plot")); +} + +void PropertiesDialog::createCoverBox() +{ + coverBox = new QWidget(this); + + QHBoxLayout * layout = new QHBoxLayout; + + QLabel * label = new QLabel(tr("Cover page")); + label->setStyleSheet("QLabel {color: white; font-weight:bold; font-size:14px;}"); + layout->addWidget(label); + layout->addStretch(); + + coverPageEdit = new YACReaderFieldEdit(); + + showPreviousCoverPageButton = new QToolButton(); + showPreviousCoverPageButton->setIcon(QIcon(":/images/previousCoverPage.png")); + showPreviousCoverPageButton->setStyleSheet("QToolButton {border:none;}"); + showNextCoverPageButton = new QToolButton(); + showNextCoverPageButton->setIcon(QIcon(":/images/nextCoverPage.png")); + showNextCoverPageButton->setStyleSheet("QToolButton {border:none;}"); + + coverPageNumberLabel = new QLabel("-"); + + coverPageNumberLabel->setStyleSheet("QLabel {color: white; font-weight:bold; font-size:14px;}"); + + layout->addWidget(showPreviousCoverPageButton); + layout->addSpacing(5); + layout->addWidget(coverPageNumberLabel); + layout->addSpacing(5); + layout->addWidget(showNextCoverPageButton); + + coverPageEdit->setStyleSheet("QLineEdit {border:none;}"); + layout->setSpacing(0); + + coverBox->setLayout(layout); + + coverBox->setFixedWidth(280); + coverBox->move(0,444-28); + layout->setContentsMargins(5,4,5,0); + + //busyIndicator = new YACReaderBusyWidget(this); + //busyIndicator->move((280-busyIndicator->width())/2,(444-busyIndicator->height()-28)/2); + //busyIndicator->hide(); + + connect(showPreviousCoverPageButton,SIGNAL(clicked()),this,SLOT(loadPreviousCover())); + connect(showNextCoverPageButton,SIGNAL(clicked()),this,SLOT(loadNextCover())); + +} + +QFrame * createLine() +{ + QFrame * line = new QFrame(); + line->setObjectName(QString::fromUtf8("line")); + //line->setGeometry(QRect(320, 150, 118, 3)); + line->setFrameShape(QFrame::HLine); + line->setFrameShadow(QFrame::Sunken); + + return line; +} + +void PropertiesDialog::createGeneralInfoBox() +{ + generalInfoBox = new QWidget; + + QFormLayout *generalInfoLayout = new QFormLayout; + + generalInfoLayout->setFieldGrowthPolicy(QFormLayout::ExpandingFieldsGrow); + //generalInfoLayout->setRowWrapPolicy(QFormLayout::WrapAllRows); + generalInfoLayout->addRow(tr("Title:"), title = new YACReaderFieldEdit()); + + + QHBoxLayout * number = new QHBoxLayout; + number->addWidget(numberEdit = new YACReaderFieldEdit()); + numberValidator.setBottom(0); + numberEdit->setValidator(&numberValidator); + number->addWidget(new QLabel("Bis:")); + number->addWidget(isBisCheck = new QCheckBox()); + number->addWidget(new QLabel("of:")); + number->addWidget(countEdit = new YACReaderFieldEdit()); + countValidator.setBottom(0); + countEdit->setValidator(&countValidator); + number->addStretch(1); + /*generalInfoLayout->addRow(tr("&Issue number:"), ); + generalInfoLayout->addRow(tr("&Bis:"), );*/ + generalInfoLayout->addRow(tr("Issue number:"), number); + + generalInfoLayout->addRow(tr("Volume:"), volumeEdit = new YACReaderFieldEdit()); + + QHBoxLayout * arc = new QHBoxLayout; + arc->addWidget(storyArcEdit = new YACReaderFieldEdit()); + arc->addWidget(new QLabel("Arc number:")); + arc->addWidget(arcNumberEdit = new YACReaderFieldEdit()); + arcNumberValidator.setBottom(0); + arcNumberEdit->setValidator(&arcNumberValidator); + arc->addWidget(new QLabel("of:")); + arc->addWidget(arcCountEdit = new YACReaderFieldEdit()); + arcCountValidator.setBottom(0); + arcCountEdit->setValidator(&arcCountValidator); + arc->addStretch(1); + generalInfoLayout->addRow(tr("Story arc:"), arc); + + generalInfoLayout->addRow(tr("Genere:"), genereEdit = new YACReaderFieldEdit()); + + generalInfoLayout->addRow(tr("Size:"), size = new QLabel("size")); + + //generalInfoLayout->addRow(tr("Comic Vine link:"), comicVineLink = new QLabel("...")); + //generalInfoLayout->addRow(bottom); + + QVBoxLayout * main = new QVBoxLayout; + main->addLayout(generalInfoLayout); + main->addStretch(); + main->addWidget(comicVineLink = new QLabel("Comic Vine link : ...")); + comicVineLink->setOpenExternalLinks(true); + + generalInfoBox->setLayout(main); +} + +void PropertiesDialog::createAuthorsBox() +{ + authorsBox = new QWidget; + + QVBoxLayout *authorsLayout = new QVBoxLayout; + + //authorsLayout->setRowWrapPolicy(QFormLayout::WrapAllRows); + QHBoxLayout * h1 = new QHBoxLayout; + QVBoxLayout * vl1 = new QVBoxLayout; + QVBoxLayout * vr1 = new QVBoxLayout; + vl1->addWidget(new QLabel(tr("Writer(s):"))); + vl1->addWidget(writer = new YACReaderFieldPlainTextEdit()); + h1->addLayout(vl1); + vr1->addWidget(new QLabel(tr("Penciller(s):"))); + vr1->addWidget(penciller = new YACReaderFieldPlainTextEdit()); + h1->addLayout(vr1); + //authorsLayout->addRow(tr("Writer(s):"), new YACReaderFieldPlainTextEdit()); + //authorsLayout->addRow(tr("Penciller(s):"), new YACReaderFieldPlainTextEdit()); + QHBoxLayout * h2 = new QHBoxLayout; + QVBoxLayout * vl2 = new QVBoxLayout; + QVBoxLayout * vr2 = new QVBoxLayout; + vl2->addWidget(new QLabel(tr("Inker(s):"))); + vl2->addWidget(inker = new YACReaderFieldPlainTextEdit()); + h2->addLayout(vl2); + vr2->addWidget(new QLabel(tr("Colorist(s):"))); + vr2->addWidget(colorist = new YACReaderFieldPlainTextEdit()); + h2->addLayout(vr2); + + //authorsLayout->addRow(tr("Inker(s):"), new YACReaderFieldPlainTextEdit()); + //authorsLayout->addRow(tr("Colorist(s):"), new YACReaderFieldPlainTextEdit()); + + QHBoxLayout * h3 = new QHBoxLayout; + QVBoxLayout * vl3 = new QVBoxLayout; + QVBoxLayout * vr3 = new QVBoxLayout; + vl3->addWidget(new QLabel(tr("Letterer(s):"))); + vl3->addWidget(letterer = new YACReaderFieldPlainTextEdit()); + h3->addLayout(vl3); + vr3->addWidget(new QLabel(tr("Cover Artist(s):"))); + vr3->addWidget(coverArtist = new YACReaderFieldPlainTextEdit()); + h3->addLayout(vr3); + //authorsLayout->addRow(tr("Letterer(es):"), new YACReaderFieldPlainTextEdit()); + //authorsLayout->addRow(tr("Cover Artist(s):"), new YACReaderFieldPlainTextEdit()); + + authorsLayout->addLayout(h1); + authorsLayout->addLayout(h2); + authorsLayout->addLayout(h3); + authorsLayout->addStretch(1); + authorsBox->setLayout(authorsLayout); + +} + +void PropertiesDialog::createPublishingBox() +{ + publishingBox = new QWidget; + + QFormLayout *publishingLayout = new QFormLayout; + + publishingLayout->setFieldGrowthPolicy(QFormLayout::ExpandingFieldsGrow); + + QHBoxLayout * date = new QHBoxLayout; + date->addWidget(new QLabel(tr("Day:"))); + date->addWidget(dayEdit = new YACReaderFieldEdit()); + dayValidator.setRange(1,31); + dayEdit->setValidator(&dayValidator); + date->addWidget(new QLabel(tr("Month:"))); + date->addWidget(monthEdit = new YACReaderFieldEdit()); + monthValidator.setRange(1,12); + monthEdit->setValidator(&monthValidator); + date->addWidget(new QLabel(tr("Year:"))); + date->addWidget(yearEdit = new YACReaderFieldEdit()); + yearValidator.setRange(1,9999); + yearEdit->setValidator(&yearValidator); + date->addStretch(1); + + publishingLayout->setRowWrapPolicy(QFormLayout::WrapAllRows); + publishingLayout->addRow(date); + publishingLayout->addRow(tr("Publisher:"), publisherEdit = new YACReaderFieldEdit()); + publishingLayout->addRow(tr("Format:"), formatEdit = new YACReaderFieldEdit()); + publishingLayout->addRow(tr("Color/BW:"), colorCheck = new QCheckBox()); + publishingLayout->addRow(tr("Age rating:"), ageRatingEdit = new YACReaderFieldEdit()); + + publishingBox->setLayout(publishingLayout); +} + +void PropertiesDialog::createPlotBox() +{ + plotBox = new QWidget; + + QFormLayout *plotLayout = new QFormLayout; + plotLayout->setFieldGrowthPolicy(QFormLayout::ExpandingFieldsGrow); + + plotLayout->setRowWrapPolicy(QFormLayout::WrapAllRows); + plotLayout->addRow(tr("Synopsis:"), synopsis = new YACReaderFieldPlainTextEdit()); + plotLayout->addRow(tr("Characters:"), characters = new YACReaderFieldPlainTextEdit()); + plotLayout->addRow(tr("Notes:"), notes = new YACReaderFieldPlainTextEdit()); + + plotBox->setLayout(plotLayout); + +} + +void PropertiesDialog::createButtonBox() +{ + buttonBox = new QDialogButtonBox; + + closeButton = buttonBox->addButton(QDialogButtonBox::Close); + saveButton = buttonBox->addButton(QDialogButtonBox::Save); + //rotateWidgetsButton = buttonBox->addButton(tr("Rotate &Widgets"),QDialogButtonBox::ActionRole); + + //connect(rotateWidgetsButton, SIGNAL(clicked()), this, SLOT(rotateWidgets())); + connect(closeButton, SIGNAL(clicked()), this, SLOT(close())); + connect(saveButton, SIGNAL(clicked()), this, SLOT(save())); +} + +QImage blurred(const QImage& image, const QRect& rect, int radius, bool alphaOnly = false) +{ + int tab[] = { 14, 10, 8, 6, 5, 5, 4, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2 }; + int alpha = (radius < 1) ? 16 : (radius > 17) ? 1 : tab[radius-1]; + + QImage result = image.convertToFormat(QImage::Format_ARGB32_Premultiplied); + int r1 = rect.top(); + int r2 = rect.bottom(); + int c1 = rect.left(); + int c2 = rect.right(); + + int bpl = result.bytesPerLine(); + int rgba[4]; + unsigned char* p; + + int i1 = 0; + int i2 = 3; + + if (alphaOnly) + i1 = i2 = (QSysInfo::ByteOrder == QSysInfo::BigEndian ? 0 : 3); + + for (int col = c1; col <= c2; col++) { + p = result.scanLine(r1) + col * 4; + for (int i = i1; i <= i2; i++) + rgba[i] = p[i] << 4; + + p += bpl; + for (int j = r1; j < r2; j++, p += bpl) + for (int i = i1; i <= i2; i++) + p[i] = (rgba[i] += ((p[i] << 4) - rgba[i]) * alpha / 16) >> 4; + } + + for (int row = r1; row <= r2; row++) { + p = result.scanLine(row) + c1 * 4; + for (int i = i1; i <= i2; i++) + rgba[i] = p[i] << 4; + + p += 4; + for (int j = c1; j < c2; j++, p += 4) + for (int i = i1; i <= i2; i++) + p[i] = (rgba[i] += ((p[i] << 4) - rgba[i]) * alpha / 16) >> 4; + } + + for (int col = c1; col <= c2; col++) { + p = result.scanLine(r2) + col * 4; + for (int i = i1; i <= i2; i++) + rgba[i] = p[i] << 4; + + p -= bpl; + for (int j = r1; j < r2; j++, p -= bpl) + for (int i = i1; i <= i2; i++) + p[i] = (rgba[i] += ((p[i] << 4) - rgba[i]) * alpha / 16) >> 4; + } + + for (int row = r1; row <= r2; row++) { + p = result.scanLine(row) + c2 * 4; + for (int i = i1; i <= i2; i++) + rgba[i] = p[i] << 4; + + p -= 4; + for (int j = c1; j < c2; j++, p -= 4) + for (int i = i1; i <= i2; i++) + p[i] = (rgba[i] += ((p[i] << 4) - rgba[i]) * alpha / 16) >> 4; + } + + return result; +} + +void PropertiesDialog::setComics(QList comics) +{ + this->comics = comics; + + ComicDB comic = comics.at(0); + + if(!comic.info.title.isNull()) + title->setText(comic.info.title.toString()); + if(!comic.info.comicVineID.isNull()) + { + comicVineLink->setHidden(false); + comicVineLink->setText(QString(tr("Comic Vine link:
view ").arg(comic.info.comicVineID.toString()))); + } + else + comicVineLink->setHidden(true); + + if(comics.length()==1 && !comic.info.coverPage.isNull()) + { + coverPageEdit->setText(comic.info.coverPage.toString()); + coverPageValidator.setRange(1,comic.info.numPages.toInt()); + coverPageEdit->setValidator(&coverPageValidator); + //---------- + int coverPage = comic.info.coverPage.toInt(); + coverPageNumberLabel->setText(QString::number(coverPage)); + coverPageNumberLabel->adjustSize(); + + showPreviousCoverPageButton->setEnabled(true); + showNextCoverPageButton->setEnabled(true); + + if(coverPage == 1) + showPreviousCoverPageButton->setDisabled(true); + if(coverPage == comic.info.numPages.toInt()) + showNextCoverPageButton->setDisabled(true); + + coverChanged = false; + coverBox->show(); + + if(!QFileInfo(basePath+comics[0].path).exists()) + { + QMessageBox::warning(this,tr("Not found"),tr("Comic not found. You should update your library.")); + showPreviousCoverPageButton->setDisabled(true); + showNextCoverPageButton->setDisabled(true); + } + } + /*if(comic.info.numPages != NULL) + numPagesEdit->setText(QString::number(*comic.info.numPages));*/ + + + if(!comic.info.number.isNull()) + numberEdit->setText(comic.info.number.toString()); + if(!comic.info.isBis.isNull()) + isBisCheck->setChecked(comic.info.isBis.toBool()); + if(!comic.info.count.isNull()) + countEdit->setText(comic.info.count.toString()); + + if(!comic.info.volume.isNull()) + volumeEdit->setText(comic.info.volume.toString()); + if(!comic.info.storyArc.isNull()) + storyArcEdit->setText(comic.info.storyArc.toString()); + if(!comic.info.arcNumber.isNull()) + arcNumberEdit->setText(comic.info.arcNumber.toString()); + if(!comic.info.arcCount.isNull()) + arcCountEdit->setText(comic.info.arcCount.toString()); + + if(!comic.info.genere.isNull()) + genereEdit->setText(comic.info.genere.toString()); + + if(!comic.info.writer.isNull()) + writer->setPlainText(comic.info.writer.toString()); + if(!comic.info.penciller.isNull()) + penciller->setPlainText(comic.info.penciller.toString()); + if(!comic.info.inker.isNull()) + inker->setPlainText(comic.info.inker.toString()); + if(!comic.info.colorist.isNull()) + colorist->setPlainText(comic.info.colorist.toString()); + if(!comic.info.letterer.isNull()) + letterer->setPlainText(comic.info.letterer.toString()); + if(!comic.info.coverArtist.isNull()) + coverArtist->setPlainText(comic.info.coverArtist.toString()); + + size->setText(QString::number(comic.info.hash.right(comic.info.hash.length()-40).toInt()/1024.0/1024.0,'f',2)+"Mb"); + + if(!comic.info.date.isNull()) + { + QStringList date = (comic.info.date.toString()).split("/"); + dayEdit->setText(date[0]); + monthEdit->setText(date[1]); + yearEdit->setText(date[2]); + } + if(!comic.info.publisher.isNull()) + publisherEdit->setText(comic.info.publisher.toString()); + if(!comic.info.format.isNull()) + formatEdit->setText(comic.info.format.toString()); + if(!comic.info.color.isNull()) + colorCheck->setChecked(comic.info.color.toBool()); + else + colorCheck->setCheckState(Qt::PartiallyChecked); + + if(!comic.info.ageRating.isNull()) + ageRatingEdit->setText(comic.info.ageRating.toString()); + + if(!comic.info.synopsis.isNull()) + synopsis->setPlainText(comic.info.synopsis.toString()); + if(!comic.info.characters.isNull()) + characters->setPlainText(comic.info.characters.toString()); + if(!comic.info.notes.isNull()) + notes->setPlainText(comic.info.notes.toString()); + + + if(comics.length() > 1) + { + coverBox->hide(); + + setDisableUniqueValues(true); + this->setWindowTitle(tr("Edit selected comics information")); + setMultipleCover(); + + QList::iterator itr; + for(itr = ++comics.begin();itr!=comics.end();itr++) + { + if(itr->info.title.isNull() || itr->info.title.toString() != title->text()) + title->clear(); + + if(itr->info.count.isNull() || itr->info.count.toString() != countEdit->text()) + countEdit->clear(); + + if(itr->info.volume.isNull() || itr->info.volume.toString() != volumeEdit->text()) + volumeEdit->clear(); + if(itr->info.storyArc.isNull() || itr->info.storyArc.toString() != storyArcEdit->text()) + storyArcEdit->clear(); + if(itr->info.arcCount.isNull() || itr->info.arcCount.toString() != storyArcEdit->text()) + arcCountEdit->clear(); + + if(itr->info.genere.isNull() || itr->info.genere.toString() != genereEdit->text()) + genereEdit->clear(); + + if(itr->info.writer.isNull() || itr->info.writer.toString() != writer->toPlainText()) + writer->clear(); + if(itr->info.penciller.isNull() || itr->info.penciller.toString() != penciller->toPlainText()) + penciller->clear(); + if(itr->info.inker.isNull() || itr->info.inker.toString() != inker->toPlainText()) + inker->clear(); + if(itr->info.colorist.isNull() || itr->info.colorist.toString() != colorist->toPlainText()) + colorist->clear(); + if(itr->info.letterer.isNull() || itr->info.letterer.toString() != letterer->toPlainText()) + letterer->clear(); + if(itr->info.coverArtist.isNull() || itr->info.coverArtist.toString() != coverArtist->toPlainText()) + coverArtist->clear(); + + if(itr->info.date.isNull()) + { + dayEdit->clear(); + monthEdit->clear(); + yearEdit->clear(); + } + else + { + QStringList date = itr->info.date.toString().split("/"); + if(dayEdit->text() != date[0]) + dayEdit->clear(); + if(monthEdit->text() != date[1]) + monthEdit->clear(); + if(yearEdit->text() != date[2]) + yearEdit->clear(); + } + + if(itr->info.publisher.isNull() || itr->info.publisher.toString() != publisherEdit->text()) + publisherEdit->clear(); + if(itr->info.format.isNull() || itr->info.format.toString() != formatEdit->text()) + formatEdit->clear(); + if(itr->info.color.isNull() || itr->info.color.toBool() != colorCheck->isChecked()) + colorCheck->setCheckState(Qt::PartiallyChecked); + if(itr->info.ageRating.isNull() || itr->info.ageRating.toString() != ageRatingEdit->text()) + ageRatingEdit->clear(); + + if(itr->info.synopsis.isNull() || itr->info.synopsis.toString() != synopsis->toPlainText()) + synopsis->clear(); + if(itr->info.characters.isNull() || itr->info.characters.toString() != characters->toPlainText()) + characters->clear(); + if(itr->info.notes.isNull() || itr->info.notes.toString() != notes->toPlainText()) + notes->clear(); + } + } + else + { + this->setWindowTitle(tr("Edit comic information")); + setCover(comic.info.getCover(basePath)); + } + +} + +void PropertiesDialog::updateComics() +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + db.open(); + db.transaction(); + QList::iterator itr; + for(itr = comics.begin();itr!=comics.end();itr++) + { + if(itr->info.edited) + DBHelper::update(&(itr->info),db); + } + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(databasePath); +} + +void PropertiesDialog::setMultipleCover() +{ + ComicDB lastComic = comics.last(); + QPixmap last = lastComic.info.getCover(basePath); + last = last.scaledToHeight(444,Qt::SmoothTransformation); + + coverImage = QPixmap::fromImage(blurred(last.toImage(),QRect(0,0,last.width(),last.height()),15)); +} + +void PropertiesDialog::setCover(const QPixmap & coverI) +{ + coverImage = coverI.scaledToHeight(444,Qt::SmoothTransformation); +} + +void PropertiesDialog::setFilename(const QString & nameString) +{ + title->setText(nameString); +} +void PropertiesDialog::setNumpages(int pagesNum) +{ + numPagesEdit->setText(QString::number(pagesNum)); +} +void PropertiesDialog::setSize(float sizeFloat) +{ + + size->setText(QString::number(sizeFloat,'f',2) + " MB"); +} + +void PropertiesDialog::save() +{ + QList::iterator itr; + for(itr = comics.begin();itr!=comics.end();itr++) + { + //Comic & comic = comics[0]; + bool edited = false; + + if(title->isModified()) + { + itr->info.title = title->text(); + edited = true; + } + + if(comics.size()==1) + if(coverChanged) + { + itr->info.coverPage = coverPageNumberLabel->text(); + edited = true; + } + + /*if(comic.info.numPages != NULL) + numPagesEdit->setText(QString::number(*comic.info.numPages));*/ + if(comics.size()==1) + if(numberEdit->isModified()) + { + if (numberEdit->text().isEmpty()) + itr->info.number = QVariant(); + else + itr->info.number = numberEdit->text(); + edited = true; + } + if(comics.size()==1) + if(!itr->info.isBis.isNull() || isBisCheck->isChecked()) + { + itr->info.isBis = isBisCheck->isChecked(); + edited = true; + } + + if(countEdit->isModified()) + { + itr->info.count = countEdit->text(); + edited = true; + } + + if(volumeEdit->isModified()) + { + itr->info.volume = volumeEdit->text(); + edited = true; + } + if(storyArcEdit->isModified()) + { + itr->info.storyArc = storyArcEdit->text(); + edited = true; + } + if(comics.size()==1) + if(arcNumberEdit->isModified() && !arcNumberEdit->text().isEmpty()) + { + itr->info.arcNumber = arcNumberEdit->text(); + edited = true; + } + if(arcCountEdit->isModified()) + { + itr->info.arcCount = arcCountEdit->text(); + edited = true; + } + + if(genereEdit->isModified()) + { + itr->info.genere = genereEdit->text(); + edited = true; + } + + if(writer->document()->isModified()) + { + itr->info.writer = writer->toPlainText(); + edited = true; + } + if(penciller->document()->isModified()) + { + itr->info.penciller = penciller->toPlainText(); + edited = true; + } + if(inker->document()->isModified()) + { + itr->info.inker = inker->toPlainText(); + edited = true; + } + if(colorist->document()->isModified()) + { + itr->info.colorist = colorist->toPlainText(); + edited = true; + } + if(letterer->document()->isModified()) + { + itr->info.letterer = letterer->toPlainText(); + edited = true; + } + if(coverArtist->document()->isModified()) + { + itr->info.coverArtist = coverArtist->toPlainText(); + edited = true; + } + + if(dayEdit->isModified() || monthEdit->isModified() || yearEdit->isModified() ) + { + itr->info.date = dayEdit->text()+"/"+monthEdit->text()+"/"+yearEdit->text(); + edited = true; + } + if(publisherEdit->isModified()) + { + itr->info.publisher = publisherEdit->text(); + edited = true; + } + if(formatEdit->isModified()) + { + itr->info.format = formatEdit->text(); + edited = true; + } + if(colorCheck->checkState() != Qt::PartiallyChecked) + { + itr->info.color = colorCheck->isChecked(); + edited = true; + } + if(ageRatingEdit->isModified()) + { + itr->info.ageRating = ageRatingEdit->text(); + edited = true; + } + + if(synopsis->document()->isModified()) + { + itr->info.synopsis = synopsis->toPlainText(); + edited = true; + } + if(characters->document()->isModified()) + { + itr->info.characters = characters->toPlainText(); + edited = true; + } + if(notes->document()->isModified()) + { + itr->info.notes = notes->toPlainText(); + edited = true; + } + + itr->info.edited = edited; + } + updateComics(); + if(comics.count() == 1) + { + if(coverChanged)// && coverPageEdit->text().toInt() != *comics[0].info.coverPage) + { + ThumbnailCreator tc(basePath+comics[0].path,basePath+"/.yacreaderlibrary/covers/"+comics[0].info.hash+".jpg", comics[0].info.coverPage.toInt()); + tc.create(); + } + } + close(); + emit(accepted()); +} + +void PropertiesDialog::setDisableUniqueValues(bool disabled) +{ + coverPageEdit->setDisabled(disabled); + coverPageEdit->clear(); + numberEdit->setDisabled(disabled); + numberEdit->clear(); + isBisCheck->setDisabled(disabled); + isBisCheck->setChecked(false); + arcNumberEdit->setDisabled(disabled); + arcNumberEdit->clear(); +} + +void PropertiesDialog::closeEvent ( QCloseEvent * e ) +{ + + title->clear(); + title->setModified(false); + coverPageEdit->clear(); + // numPagesEdit->setText(QString::number(*comic.info.numPages)); + numberEdit->clear(); + isBisCheck->setChecked(false); + countEdit->clear(); + volumeEdit->clear(); + storyArcEdit->clear(); + arcNumberEdit->clear(); + arcCountEdit->clear(); + genereEdit->clear(); + writer->clear(); + penciller->clear(); + inker->clear(); + colorist->clear(); + letterer->clear(); + coverArtist->clear(); + dayEdit->clear(); + monthEdit->clear(); + yearEdit->clear(); + publisherEdit->clear(); + formatEdit->clear(); + colorCheck->setCheckState(Qt::PartiallyChecked); + ageRatingEdit->clear(); + synopsis->clear(); + characters->clear(); + notes->clear(); + + setDisableUniqueValues(false); + + tabBar->setCurrentIndex(0); + + coverPageEdit->setFocus(); + + QDialog::closeEvent(e); +} + +void PropertiesDialog::paintEvent(QPaintEvent * event) +{ + QDialog::paintEvent(event); + + QPainter p(this); + + p.drawPixmap(0,0,coverImage); + + //QPixmap shadow(":/images/social_dialog/shadow.png"); + //p.drawPixmap(280-shadow.width(),0,shadow.width(),444,shadow); + p.drawLine(279,0,279,444); + if(comics.length()==1) + p.fillRect(0,444-28,280,28,QColor(0,0,0,153)); +} + +void PropertiesDialog::updateCoverPageNumberLabel(int n) +{ + coverPageNumberLabel->setText(QString::number(n)); + coverPageNumberLabel->adjustSize(); +} + +void PropertiesDialog::loadNextCover() +{ + int current = coverPageNumberLabel->text().toInt(); + if(current < comics.at(0).info.numPages.toInt()) + { + updateCoverPageNumberLabel(current+1); + + ThumbnailCreator tc(basePath+comics[0].path,"",current+1); + tc.create(); + setCover(tc.getCover()); + repaint(); + + if((current+1) == comics.at(0).info.numPages.toInt()) + { + showNextCoverPageButton->setDisabled(true); + } + + showPreviousCoverPageButton->setEnabled(true); + //busyIndicator->show(); + if(current+1 != comics.at(0).info.coverPage) + coverChanged = true; + else + coverChanged = false; + } +} + +void PropertiesDialog::loadPreviousCover() +{ + int current = coverPageNumberLabel->text().toInt(); + if(current!=1) + { + updateCoverPageNumberLabel(current-1); + ThumbnailCreator tc(basePath+comics[0].path,"",current-1); + tc.create(); + setCover(tc.getCover()); + repaint(); + + if((current-1) == 1) + { + showPreviousCoverPageButton->setDisabled(true); + } + + showNextCoverPageButton->setEnabled(true); + //busyIndicator->show(); + if(current-1 != comics.at(0).info.coverPage.toInt()) + coverChanged = true; + else + coverChanged = false; + } +} diff --git a/YACReaderLibrary/properties_dialog.h b/YACReaderLibrary/properties_dialog.h new file mode 100644 index 00000000..a3088b1d --- /dev/null +++ b/YACReaderLibrary/properties_dialog.h @@ -0,0 +1,141 @@ +#ifndef __PROPERTIES_DIALOG_H +#define __PROPERTIES_DIALOG_H + +#include + +#include + +class QGridLayout; +class QTabWidget; +class QGroupBox; +class QLabel; +class QScrollArea; +class QWidget; +class YACReaderFieldEdit; +class YACReaderFieldPlainTextEdit; +class QDialogButtonBox; +class QCheckBox; +//class YACReaderBusyWidget; +class QToolButton; + +#include "comic_db.h" + + class PropertiesDialog : public QDialog + { + Q_OBJECT + private: + QWidget * mainWidget; + //YACReaderBusyWidget * busyIndicator; + + QGridLayout * mainLayout; + + QTabWidget * tabBar; + + QWidget * coverBox; + QLabel * cover; + QScrollArea * sa; + + QWidget * generalInfoBox; + YACReaderFieldEdit * title; + YACReaderFieldEdit * numPagesEdit; + QLabel * size; + QLabel * comicVineLink; + + YACReaderFieldEdit * coverPageEdit; + QIntValidator coverPageValidator; + + YACReaderFieldEdit * numberEdit; + QIntValidator numberValidator; + QCheckBox * isBisCheck; + YACReaderFieldEdit * countEdit; + QIntValidator countValidator; + + YACReaderFieldEdit * volumeEdit; + YACReaderFieldEdit * storyArcEdit; + YACReaderFieldEdit * arcNumberEdit; + QIntValidator arcNumberValidator; + YACReaderFieldEdit * arcCountEdit; + QIntValidator arcCountValidator; + + YACReaderFieldEdit * genereEdit; + + YACReaderFieldPlainTextEdit * writer; + YACReaderFieldPlainTextEdit * penciller; + YACReaderFieldPlainTextEdit * inker; + YACReaderFieldPlainTextEdit * colorist; + YACReaderFieldPlainTextEdit * letterer; + YACReaderFieldPlainTextEdit * coverArtist; + + YACReaderFieldEdit * dayEdit; + QIntValidator dayValidator; + YACReaderFieldEdit * monthEdit; + QIntValidator monthValidator; + YACReaderFieldEdit * yearEdit; + QIntValidator yearValidator; + YACReaderFieldEdit * publisherEdit; + YACReaderFieldEdit * formatEdit; + QCheckBox * colorCheck; + YACReaderFieldEdit * ageRatingEdit; + + YACReaderFieldPlainTextEdit * synopsis; + YACReaderFieldPlainTextEdit * characters; + YACReaderFieldPlainTextEdit * notes; + + QWidget * authorsBox; + + QWidget * publishingBox; + + QWidget * plotBox; + + QDialogButtonBox *buttonBox; + QPushButton *closeButton; + QPushButton *saveButton; + QPushButton *restoreButton; //?? + + QPixmap coverImage; + + QToolButton * showPreviousCoverPageButton; + QToolButton * showNextCoverPageButton; + QLabel * coverPageNumberLabel; + + void createTabBar(); + void createCoverBox(); + void createGeneralInfoBox(); + void createAuthorsBox(); + void createPublishingBox(); + void createPlotBox(); + + void createButtonBox(); + + void setDisableUniqueValues(bool disabled); + + QList comics; + void closeEvent ( QCloseEvent * e ); + void updateCoverPageNumberLabel(int n); + + bool coverChanged; + + public: + PropertiesDialog(QWidget * parent = 0); + QString databasePath; + QString basePath; + QSize sizeHint(); + void paintEvent(QPaintEvent * event); + + public slots: + void setComics(QList comics); + void updateComics(); + void save(); + //Deprecated + void setCover(const QPixmap & cover); + void setMultipleCover(); + void setFilename(const QString & name); + void setNumpages(int pages); + void setSize(float size); + void loadNextCover(); + void loadPreviousCover(); + + + }; +#endif + diff --git a/YACReaderLibrary/qml.qrc b/YACReaderLibrary/qml.qrc new file mode 100644 index 00000000..a02ccead --- /dev/null +++ b/YACReaderLibrary/qml.qrc @@ -0,0 +1,8 @@ + + + qml/GridComicsView.qml + qml/YACReaderScrollView.qml + qml/tick.png + qml/reading.png + + diff --git a/YACReaderLibrary/qml/GridComicsView.qml b/YACReaderLibrary/qml/GridComicsView.qml new file mode 100644 index 00000000..36ee0cfb --- /dev/null +++ b/YACReaderLibrary/qml/GridComicsView.qml @@ -0,0 +1,295 @@ +import QtQuick 2.3 +import QtQuick.Controls 1.0 +import QtQuick.Controls 1.1 +import QtGraphicalEffects 1.0 +import comicModel 1.0 + +Rectangle { + id: main + color: backgroundColor + width: parent.width + height: parent.height + anchors.margins: 0 + + function selectAll(from,to) + { + for(var i = from+1;i ci) + selectAll(ci,index); + + mouse.accepted = true; + + comicsSelectionHelper.selectIndex(index) + grid.currentIndex = index; + + } + } + + //Menu emits the 'main' signals + Menu { + id: myContextMenu + MenuItem { text: "Open comic"; enabled: true; iconSource:"qrc:///images/openInYACReader.png"; onTriggered: openComicAction.trigger() } + MenuSeparator{} + MenuItem { text: "Open containing folder..."; enabled: true; iconSource: "qrc:///images/open.png"; onTriggered: openContainingFolderComicAction.trigger() } + MenuSeparator{} + MenuItem { text: "Reset comic rating"; onTriggered: resetComicRatingAction.trigger() } + MenuSeparator{} + MenuItem { text: "Edit"; enabled: true; iconSource:"qrc:///images/editComic.png"; onTriggered: editSelectedComicsAction.trigger() } + MenuItem { text: "Download tags from Comic Vine"; enabled: true; iconSource:"qrc:///images/getInfo.png"; onTriggered: getInfoAction.trigger() } + MenuItem { text: "Asign current order to comics"; enabled: true; iconSource:"qrc:///images/asignNumber.png"; onTriggered: asignOrderAction.trigger() } + MenuSeparator{} + MenuItem { text: "Select all comics"; enabled: true; iconSource:"qrc:///images/selectAll.png"; onTriggered: selectAllComicsAction.trigger() } + MenuSeparator{} + MenuItem { text: "Set as read"; enabled: true; iconSource:"qrc:///images/setReadButton.png"; onTriggered: setAsReadAction.trigger() } + MenuItem { text: "Set as unread"; enabled: true; iconSource:"qrc:///images/setUnread.png"; onTriggered: setAsNonReadAction.trigger() } + MenuItem { text: "Show or hide read marks"; enabled: true; iconSource:"qrc:///images/showMarks.png"; onTriggered: showHideMarksAction.trigger() } + MenuSeparator{} + MenuItem { text: "Delete selected comics"; enabled: true; iconSource:"qrc:///images/trash.png"; onTriggered: deleteComicsAction.trigger() } + MenuSeparator{} + MenuItem { text: "Fullscreen mode on/off"; onTriggered: toggleFullScreenAction.trigger() } + //MenuItem { text: "Show details"; onTriggered: cell.state = 'Details'; + } + + + } + + DropShadow { + anchors.fill: source + horizontalOffset: 0 + verticalOffset: 0 + radius: 3 + samples: 24 + color: "#40000000" + transparentBorder: true; + source: realCell; + enabled: dropShadow; + visible: dropShadow; + } + + /**/ + + //cover + Image { + id: coverElement + width: 148 + height: 224 + anchors {horizontalCenter: parent.horizontalCenter; top: realCell.top; topMargin: 4} + source: cover_path + fillMode: Image.PreserveAspectCrop + //smooth: true + mipmap: true + //antialiasing: true + asynchronous : true + cache: false //TODO clear cache only when it is neede + } + //mark + Image { + id: mark + width: 23 + height: 23 + source: read_column&&show_marks?"tick.png":has_been_opened&&show_marks?"reading.png":"" + anchors {right: coverElement.right; top: coverElement.top; topMargin: 11; rightMargin: 11} + asynchronous : true + } + + //title + Text { + anchors { top: realCell.top; left: realCell.left; leftMargin: 4; rightMargin: 4; topMargin: 234; } + width: 148 + maximumLineCount: 2 + wrapMode: Text.WordWrap + text: title + elide: Text.ElideRight + color: titleColor + clip: true + font.letterSpacing: 0.5 + } + //number + Text { + anchors {bottom: realCell.bottom; left: realCell.left; margins: 4} + text: number?"#"+number:"" + color: textColor + font.letterSpacing: 0.5 + } + //page icon + Image { + id: pageImage + anchors {bottom: realCell.bottom; right: realCell.right; bottomMargin: 5; rightMargin: 4; leftMargin: 4} + source: "page.png" + } + //numPages + Text { + id: pages + anchors {bottom: realCell.bottom; right: pageImage.left; margins: 4} + text: has_been_opened?current_page+"/"+num_pages:num_pages + color: textColor + font.letterSpacing: 0.5 + } + //rating icon + Image { + id: ratingImage + anchors {bottom: realCell.bottom; right: pageImage.left; bottomMargin: 5; rightMargin: Math.floor(pages.width)+12} + source: "star.png" + } + //comic rating + Text { + id: comicRating + anchors {bottom: realCell.bottom; right: ratingImage.left; margins: 4} + text: rating>0?rating:"-" + color: textColor + } + } + } + + YACReaderScrollView{ + id: scrollView + anchors.fill: parent + anchors.margins: 0 + + + GridView { + id:grid + anchors.fill: parent + cellHeight: 295 + highlight: appHighlight + focus: true + model: comicsList + delegate: appDelegate + anchors.topMargin: 20 + anchors.bottomMargin: 20 + anchors.leftMargin: 10 + anchors.rightMargin: 10 + pixelAligned: true + //flickDeceleration: -2000 + snapMode: GridView.SnapToRow + currentIndex: 0 + cacheBuffer: 0 + + + function numCellsPerRow() { + return Math.floor(width / 190); + } + + onWidthChanged: { + var numCells = numCellsPerRow(); + var rest = width % 190; + + if(numCells > 0) + { + cellWidth = Math.floor(width / numCells) ; + //console.log("numCells=",numCells,"rest=",rest,"cellWidth=",cellWidth,"width=",width); + } + } + } + focus: true + Keys.onPressed: { + if (event.modifiers & Qt.ControlModifier || event.modifiers & Qt.ShiftModifier) + return; + var numCells = grid.numCellsPerRow(); + var ci + if (event.key === Qt.Key_Right) { + ci = Math.min(grid.currentIndex+1,grid.count); + } + else if (event.key === Qt.Key_Left) { + ci = Math.max(0,grid.currentIndex-1); + } + else if (event.key === Qt.Key_Up) { + ci = Math.max(0,grid.currentIndex-numCells); + } + else if (event.key === Qt.Key_Down) { + ci = Math.min(grid.currentIndex+numCells,grid.count); + } + + event.accepted = true; + //var ci = grid.currentIndex; + grid.currentIndex = -1 + comicsSelectionHelper.clear(); + comicsSelectionHelper.selectIndex(ci); + grid.currentIndex = ci; + } + //} + + /*MouseArea { + anchors.fill: parent + onClicked: { + clicked.accepted = false; + console.log("xx"); + } + + onWheel: { + var newValue = Math.max(0,scrollView.flickableItem.contentY - wheel.angleDelta.y) + scrollView.flickableItem.contentY = newValue + + } + }*/ + /*ScrollBar { + flickable: grid; + } + + PerformanceMeter { + anchors {top: parent.top; left: parent.left; margins: 4} + id: performanceMeter + width: 128 + height: 64 + enabled: (dummyValue || !dummyValue) + }*/ + } +} + + diff --git a/YACReaderLibrary/qml/YACReaderScrollView.qml b/YACReaderLibrary/qml/YACReaderScrollView.qml new file mode 100644 index 00000000..a8dc57ad --- /dev/null +++ b/YACReaderLibrary/qml/YACReaderScrollView.qml @@ -0,0 +1,336 @@ +/**************************************************************************** +** +** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). +** Contact: http://www.qt-project.org/legal +** +** This file is part of the Qt Quick Controls module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Digia Plc and its Subsidiary(-ies) nor the names +** of its contributors may be used to endorse or promote products derived +** from this software without specific prior written permission. +** +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +import QtQuick 2.2 +import QtQuick.Controls 1.2 +import QtQuick.Controls.Private 1.0 +import QtQuick.Controls.Styles 1.1 + +/*! + \qmltype ScrollView + \inqmlmodule QtQuick.Controls + \since 5.1 + \ingroup views + \brief Provides a scrolling view within another Item. + + A ScrollView can be used either to replace a \l Flickable or decorate an + existing \l Flickable. Depending on the platform, it will add scroll bars and + a content frame. + + Only one Item can be a direct child of the ScrollView and the child is implicitly anchored + to fill the scroll view. + + Example: + \code + ScrollView { + Image { source: "largeImage.png" } + } + \endcode + + In the previous example the Image item will implicitly get scroll behavior as if it was + used within a \l Flickable. The width and height of the child item will be used to + define the size of the content area. + + Example: + \code + ScrollView { + ListView { + ... + } + } + \endcode + + In this case the content size of the ScrollView will simply mirror that of its contained + \l flickableItem. + + You can create a custom appearance for a ScrollView by + assigning a \l {QtQuick.Controls.Styles::ScrollViewStyle}{ScrollViewStyle}. +*/ + +FocusScope { + id: root + + implicitWidth: 240 + implicitHeight: 150 + + /*! + This property tells the ScrollView if it should render + a frame around its content. + + The default value is \c false. + */ + property bool frameVisible: false + + /*! + This property controls if there should be a highlight + around the frame when the ScrollView has input focus. + + The default value is \c false. + + \note This property is only applicable on some platforms, such + as Mac OS. + */ + property bool highlightOnFocus: false + + /*! + \qmlproperty Item ScrollView::viewport + + The viewport determines the current "window" on the contentItem. + In other words, it clips it and the size of the viewport tells you + how much of the content area is visible. + */ + property alias viewport: viewportItem + + /*! + \qmlproperty Item ScrollView::flickableItem + + The flickableItem of the ScrollView. If the contentItem provided + to the ScrollView is a Flickable, it will be the \l contentItem. + */ + readonly property alias flickableItem: internal.flickableItem + + /*! + The contentItem of the ScrollView. This is set by the user. + + Note that the definition of contentItem is somewhat different to that + of a Flickable, where the contentItem is implicitly created. + */ + default property Item contentItem + + /*! \internal */ + property Item __scroller: scroller + /*! \internal */ + property alias __wheelAreaScrollSpeed: wheelArea.scrollSpeed + /*! \internal */ + property int __scrollBarTopMargin: 0 + /*! \internal */ + property int __viewTopMargin: 0 + /*! \internal */ + property alias __horizontalScrollBar: scroller.horizontalScrollBar + /*! \internal */ + property alias __verticalScrollBar: scroller.verticalScrollBar + /*! \qmlproperty Component ScrollView::style + + The style Component for this control. + \sa {Qt Quick Controls Styles QML Types} + + */ + property Component style: Qt.createComponent(Settings.style + "/ScrollViewStyle.qml", root) + + /*! \internal */ + property Style __style: styleLoader.item + + activeFocusOnTab: true + + onContentItemChanged: { +//console.log("onContentItemChanged"); + if (contentItem.hasOwnProperty("contentY") && // Check if flickable + contentItem.hasOwnProperty("contentHeight")) { + internal.flickableItem = contentItem // "Use content if it is a flickable + internal.flickableItem.parent = viewportItem + } else { + internal.flickableItem = flickableComponent.createObject(viewportItem) + contentItem.parent = internal.flickableItem.contentItem + } + internal.flickableItem.anchors.fill = viewportItem + if (!Settings.hasTouchScreen) + internal.flickableItem.interactive = false + } + + + children: Item { + id: internal + + property Flickable flickableItem + + Loader { + id: styleLoader + sourceComponent: style + onStatusChanged: { + if (status === Loader.Error) + console.error("Failed to load Style for", root) + } + property alias __control: root + } + + Binding { + target: flickableItem + property: "contentHeight" + when: contentItem !== flickableItem + value: contentItem ? contentItem.height : 0 + } + + Binding { + target: flickableItem + when: contentItem !== flickableItem + property: "contentWidth" + value: contentItem ? contentItem.width : 0 + } + + Connections { + target: flickableItem + + onContentYChanged: { + //console.log("onContentYChanged2"); + scroller.blockUpdates = true + scroller.verticalScrollBar.value = flickableItem.contentY + scroller.blockUpdates = false + } + + onContentXChanged: { + //console.log("onContentXChanged2"); + scroller.blockUpdates = true + scroller.horizontalScrollBar.value = flickableItem.contentX + scroller.blockUpdates = false + } + + } + + anchors.fill: parent + + Component { + id: flickableComponent + Flickable {} + } + + WheelArea { + id: wheelArea + parent: flickableItem + + // ### Note this is needed due to broken mousewheel behavior in Flickable. + + anchors.fill: parent + + property int stepSize: 295 + + property int acceleration: 40 + property int flickThreshold: Settings.dragThreshold + property real speedThreshold: 3 + property real ignored: 0.001 // ## flick() does not work with 0 yVelocity + property int maxFlick: 400 + + property bool horizontalRecursionGuard: false + property bool verticalRecursionGuard: false + + horizontalMinimumValue: flickableItem ? flickableItem.originX : 0 + horizontalMaximumValue: flickableItem ? flickableItem.originX + flickableItem.contentWidth - viewport.width : 0 + + verticalMinimumValue: flickableItem ? flickableItem.originY : 0 + verticalMaximumValue: flickableItem ? flickableItem.originY + flickableItem.contentHeight - viewport.height + __viewTopMargin : 0 + + Connections { + target: flickableItem + + onContentYChanged: { + //console.log("onContentYChanged"); + wheelArea.verticalRecursionGuard = true + wheelArea.verticalValue = flickableItem.contentY + wheelArea.verticalRecursionGuard = false + } + onContentXChanged: { + //console.log("onContentXChanged"); + wheelArea.horizontalRecursionGuard = true + wheelArea.horizontalValue = flickableItem.contentX + wheelArea.horizontalRecursionGuard = false + } + } + + onVerticalValueChanged: { + if (!verticalRecursionGuard) { + //console.log(verticalDelta); + + if (flickableItem.contentY < flickThreshold && verticalDelta > speedThreshold) { + flickableItem.flick(ignored, Math.min(maxFlick, acceleration * verticalDelta)) + } else if (flickableItem.contentY > flickableItem.contentHeight + - flickThreshold - viewport.height && verticalDelta < -speedThreshold) { + flickableItem.flick(ignored, Math.max(-maxFlick, acceleration * verticalDelta)) + } else { + var absDelta = Math.abs(verticalDelta); + + if(verticalDelta < 0) + flickableItem.contentY = verticalValue + Math.min(98,0.93*absDelta+4.5); + else + flickableItem.contentY = verticalValue - Math.min(98,0.93*absDelta+4.5); +} + + + //TODO: snap to row + + } + + } + + onHorizontalValueChanged: { + if (!horizontalRecursionGuard) + flickableItem.contentX = horizontalValue + } + } + + ScrollViewHelper { + id: scroller + anchors.fill: parent + active: wheelArea.active + property bool outerFrame: !frameVisible || !(__style ? __style.__externalScrollBars : 0) + property int scrollBarSpacing: outerFrame ? 0 : (__style ? __style.__scrollBarSpacing : 0) + property int verticalScrollbarOffset: verticalScrollBar.visible && !verticalScrollBar.isTransient ? + verticalScrollBar.width + scrollBarSpacing : 0 + property int horizontalScrollbarOffset: horizontalScrollBar.visible && !horizontalScrollBar.isTransient ? + horizontalScrollBar.height + scrollBarSpacing : 0 + Loader { + id: frameLoader + sourceComponent: __style ? __style.frame : null + anchors.fill: parent + anchors.rightMargin: scroller.outerFrame ? 0 : scroller.verticalScrollbarOffset + anchors.bottomMargin: scroller.outerFrame ? 0 : scroller.horizontalScrollbarOffset + } + + Item { + id: viewportItem + anchors.fill: frameLoader + anchors.topMargin: frameVisible ? __style.padding.top : 0 + anchors.leftMargin: frameVisible ? __style.padding.left : 0 + anchors.rightMargin: (frameVisible ? __style.padding.right : 0) + (scroller.outerFrame ? scroller.verticalScrollbarOffset : 0) + anchors.bottomMargin: (frameVisible ? __style.padding.bottom : 0) + (scroller.outerFrame ? scroller.horizontalScrollbarOffset : 0) + clip: true + } + } + FocusFrame { visible: highlightOnFocus && root.activeFocus } + } +} diff --git a/YACReaderLibrary/qml/page-macosx.png b/YACReaderLibrary/qml/page-macosx.png new file mode 100644 index 0000000000000000000000000000000000000000..c8216591679ffb2e0dac358288275c4ce8d539ce GIT binary patch literal 171 zcmeAS@N?(olHy`uVBq!ia0vp^96-#)!3HEdkIOdzDajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MZTUcjv*Ddl6d&}|DSK*6bfGxv60~nlO_Wbn*`g(g%*rQ z(i0ExF*Q~=y12A3>9M|ESYq48wBT-!f!T$Y|2{k{46-5{4#+fnGdeUdGC1cdp31NN R!3#8)!PC{xWt~$(697>)GP3{x literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/page.png b/YACReaderLibrary/qml/page.png new file mode 100644 index 0000000000000000000000000000000000000000..100db8a07ce55af57e69d58fd70c3ec37690e1ed GIT binary patch literal 155 zcmeAS@N?(olHy`uVBq!ia0vp^96-#)!3HEdkIOdzDajJoh?3y^w370~qErUQl>DSr z1<%~X^wgl##FWaylc_d9MYf(Ujv*Ddl795dN*Ekq=s4cMvViLzb0Jg5`9lq|Doh`1 z7AP^gO(=3xIFPEejA53HFf*eAJF8BM43|IxgRqEQ=BZ0eSwMprJYD@<);T3K0RU%2 BE8+kE literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/reading.png b/YACReaderLibrary/qml/reading.png new file mode 100644 index 0000000000000000000000000000000000000000..a26a81d63a321ff8c73407b041c96b3c22c3576b GIT binary patch literal 374 zcmV-+0g3*JP)+7*9lVSQEwMA4ff7z&p)DoQSkRc@A2T3PiBWd_$!B1)`!d-qGqS2G(IqLR zi?D!Q_7E=NlpSM#+PVK79MCDk2DZd!6>tE~=_4HA6@n+eR|M9f6Amy{I~Ttz0WYXO z)KkTMz@#~I9&kQmPw0`yyr%Mv(5pL7@pZE_!!6<3cmIAa4Ep2b$qQ(_6UFN1DJC24IAC14hUQ z{YySGLN_QQFan?XWQ4LoH)uNnY>?WLi7H4=I{CQR_rj!rre}xA-v7M^7fDX>;cf~LG0Dm%$F|N521bkdXAUS~WA?lm?MZ7#7tIvulm>39y z^0RoR(IeYhR2qTw+usVKMhmq=u>Go$FUqaTSFcB63jr5@3tQ!@qV0MlUxilrY|T!@ m`KRr){7>jM>~=Ii0R{k?6iGx9CTxW#|uyQ7I> zD^q~9RKpKj0d0quY%6v!>}vZQ@Vw#0R|gpeQT~Qc4?n%w_NLv7E1~dU<%&HFX6onI zXZEPEZOXLUBq2O&U$?^_##IrFTHhG8`hq9h2yi<%^NLk51r*Mi#nT`mV$R+eA?Ri8 naI}!EnyIhB!$FLh!+>GZiKSU*Paew!x{txr)z4*}Q$iB}nx9l& literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/qml/tick.png b/YACReaderLibrary/qml/tick.png new file mode 100644 index 0000000000000000000000000000000000000000..78a20644c27b468c50767aa4a44597662f7253de GIT binary patch literal 488 zcmVP)wXM;%Yz~3&g? zC_;UD5u4#56c~&gN1%AQ2I4Rwg|QO@b$}DpJhgt^2k28-cbAj1Q&xen-6dqYD*IeUA1 zMJ6Vu&map(Fc<__K{>P+nl>?9e*E}xKTw#0_9*{be eYPCN=fB^su{-6g#-^-l<0000 + + qml/page-macosx.png + qml/star-macosx.png + + diff --git a/YACReaderLibrary/qml_win.qrc b/YACReaderLibrary/qml_win.qrc new file mode 100644 index 00000000..59e2872f --- /dev/null +++ b/YACReaderLibrary/qml_win.qrc @@ -0,0 +1,6 @@ + + + qml/page.png + qml/star.png + + diff --git a/YACReaderLibrary/rename_library_dialog.cpp b/YACReaderLibrary/rename_library_dialog.cpp new file mode 100644 index 00000000..22202865 --- /dev/null +++ b/YACReaderLibrary/rename_library_dialog.cpp @@ -0,0 +1,76 @@ +#include "rename_library_dialog.h" + +#include +#include +#include + + + +RenameLibraryDialog::RenameLibraryDialog(QWidget * parent) +:QDialog(parent) +{ + setupUI(); +} + +void RenameLibraryDialog::setupUI() +{ + newNameLabel = new QLabel(tr("New Library Name : ")); + newNameEdit = new QLineEdit; + newNameLabel->setBuddy(newNameEdit); + connect(newNameEdit,SIGNAL(textChanged(QString)),this,SLOT(nameSetted(QString))); + + accept = new QPushButton(tr("Rename")); + accept->setDisabled(true); + connect(accept,SIGNAL(clicked()),this,SLOT(rename())); + + cancel = new QPushButton(tr("Cancel")); + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + QHBoxLayout *nameLayout = new QHBoxLayout; + + nameLayout->addWidget(newNameLabel); + nameLayout->addWidget(newNameEdit); + + QHBoxLayout *bottomLayout = new QHBoxLayout; + bottomLayout->addStretch(); + bottomLayout->addWidget(accept); + bottomLayout->addWidget(cancel); + + QVBoxLayout *mainLayout = new QVBoxLayout; + mainLayout->addLayout(nameLayout); + mainLayout->addStretch(); + mainLayout->addLayout(bottomLayout); + + QHBoxLayout * imgMainLayout = new QHBoxLayout; + QLabel * imgLabel = new QLabel(this); + QPixmap p(":/images/edit.png"); + imgLabel->setPixmap(p); + imgMainLayout->addWidget(imgLabel); + imgMainLayout->addLayout(mainLayout); + + setLayout(imgMainLayout); + + setModal(true); + setWindowTitle(tr("Rename current library")); +} + +void RenameLibraryDialog::rename() +{ + //accept->setEnabled(false); + emit(renameLibrary(newNameEdit->text())); +} + +void RenameLibraryDialog::nameSetted(const QString & text) +{ + if(!text.isEmpty()) + accept->setEnabled(true); + else + accept->setEnabled(false); +} + +void RenameLibraryDialog::close() +{ + newNameEdit->clear(); + //accept->setEnabled(false); + QDialog::close(); +} \ No newline at end of file diff --git a/YACReaderLibrary/rename_library_dialog.h b/YACReaderLibrary/rename_library_dialog.h new file mode 100644 index 00000000..abdd2e3e --- /dev/null +++ b/YACReaderLibrary/rename_library_dialog.h @@ -0,0 +1,31 @@ +#ifndef __RENAME_LIBRARY_DIALOG_H +#define __RENAME_LIBRARY_DIALOG_H + +#include +#include +#include +#include + + + class RenameLibraryDialog : public QDialog + { + Q_OBJECT + public: + RenameLibraryDialog(QWidget * parent = 0); + private: + QLabel * newNameLabel; + QLineEdit * newNameEdit; + QPushButton * accept; + QPushButton * cancel; + void setupUI(); + public slots: + void rename(); + void close(); + void nameSetted(const QString & name); +signals: + void renameLibrary(QString newName); + }; + + +#endif + diff --git a/YACReaderLibrary/server/controllers/comiccontroller.cpp b/YACReaderLibrary/server/controllers/comiccontroller.cpp new file mode 100644 index 00000000..88c39abf --- /dev/null +++ b/YACReaderLibrary/server/controllers/comiccontroller.cpp @@ -0,0 +1,124 @@ +#include "comiccontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +#include "template.h" +#include "../static.h" + +#include "comic_db.h" +#include "comic.h" + +#include "QsLog.h" + +#include + +ComicController::ComicController() {} + +void ComicController::service(HttpRequest& request, HttpResponse& response) +{ + HttpSession session=Static::sessionStore->getSession(request,response,false); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); + qulonglong comicId = pathElements.at(4).toULongLong(); + + bool remoteComic = path.endsWith("remote"); + + //TODO + //if(pathElements.size() == 6) + //{ + // QString action = pathElements.at(5); + // if(!action.isEmpty() && (action == "close")) + // { + // session.dismissCurrentComic(); + // response.write("",true); + // return; + // } + //} + + //Aplicar a todos los controladores + //TODO usar LibraryWindow para acceder a información de las bases de datos está mal, hay + //que crear una clase que se encargue de estas cosas + //¿Se está accediendo a la UI desde un hilo? + + YACReaderLibraries libraries = DBHelper::getLibraries(); + + + ComicDB comic = DBHelper::getComicInfo(libraryName, comicId); + + if(!remoteComic) + session.setDownloadedComic(comic.info.hash); + + Comic * comicFile = FactoryComic::newComic(libraries.getPath(libraryName)+comic.path); + + if(comicFile != NULL) + { + QThread * thread = NULL; + + thread = new QThread(); + + comicFile->moveToThread(thread); + + connect(thread, SIGNAL(started()), comicFile, SLOT(process())); + connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater())); + + comicFile->load(libraries.getPath(libraryName)+comic.path); + + if(thread != NULL) + thread->start(); + + if(remoteComic) + { + QLOG_INFO() << "remote comic requested"; + session.setCurrentRemoteComic(comic.id, comicFile); + + } + else + { + QLOG_INFO() << "comic requested"; + session.setCurrentComic(comic.id, comicFile); + } + + response.setHeader("Content-Type", "plain/text; charset=ISO-8859-1"); + //TODO this field is not used by the client! + response.writeText(QString("library:%1\r\n").arg(libraryName)); + response.writeText(QString("libraryId:%1\r\n").arg(pathElements.at(2))); + if(remoteComic) //send previous and next comics id + { + QList siblings = DBHelper::getFolderComicsFromLibrary(libraryName, comic.parentId); + bool found = false; + int i; + for(i = 0; i < siblings.length(); i++) + { + if (siblings.at(i)->id == comic.id) + { + found = true; + break; + } + } + if(found) + { + if(i>0) + response.writeText(QString("previousComic:%1\r\n").arg(siblings.at(i-1)->id)); + if(iid)); + } + else + { + //ERROR + } + qDeleteAll(siblings); + } + response.writeText(comic.toTXT(),true); + } + else + { + //delete comicFile; + response.setStatus(404,"not found"); + response.write("404 not found",true); + } + //response.write(t.toLatin1(),true); + +} diff --git a/YACReaderLibrary/server/controllers/comiccontroller.h b/YACReaderLibrary/server/controllers/comiccontroller.h new file mode 100644 index 00000000..71287b68 --- /dev/null +++ b/YACReaderLibrary/server/controllers/comiccontroller.h @@ -0,0 +1,23 @@ +#ifndef COMICCONTROLLER_H +#define COMICCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +#include +class Comic; +class QString; + +class ComicController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(ComicController); +public: + /** Constructor */ + ComicController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // COMICCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/covercontroller.cpp b/YACReaderLibrary/server/controllers/covercontroller.cpp new file mode 100644 index 00000000..4821bfdd --- /dev/null +++ b/YACReaderLibrary/server/controllers/covercontroller.cpp @@ -0,0 +1,88 @@ +#include "covercontroller.h" +#include "db_helper.h" //get libraries +#include "yacreader_libraries.h" + +#include "template.h" +#include "../static.h" + +CoverController::CoverController() {} + +void CoverController::service(HttpRequest& request, HttpResponse& response) +{ + + HttpSession session=Static::sessionStore->getSession(request,response,false); + + response.setHeader("Content-Type", "image/jpeg"); + response.setHeader("Connection","close"); + //response.setHeader("Content-Type", "plain/text; charset=ISO-8859-1"); + + YACReaderLibraries libraries = DBHelper::getLibraries(); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); + QString fileName = pathElements.at(4); + + bool folderCover = request.getParameter("folderCover").length()>0; + + //response.writeText(path+"
"); + //response.writeText(libraryName+"
"); + //response.writeText(libraries.value(libraryName)+"/.yacreaderlibrary/covers/"+fileName+"
"); + + //QFile file(libraries.value(libraryName)+"/.yacreaderlibrary/covers/"+fileName); + //if (file.exists()) { + // if (file.open(QIODevice::ReadOnly)) + // { + // qDebug("StaticFileController: Open file %s",qPrintable(file.fileName())); + // // Return the file content, do not store in cache + // while (!file.atEnd() && !file.error()) { + // response.write(file.read(131072)); + // } + // } + + // file.close(); + //} + + QImage img(libraries.getPath(libraryName)+"/.yacreaderlibrary/covers/"+fileName); + if (!img.isNull()) { + + int width = 80, height = 120; + if(session.getDisplayType()=="@2x") + { + width = 160; + height = 240; + } + + if(float(img.width())/img.height() < 0.66666) + img = img.scaledToWidth(width,Qt::SmoothTransformation); + else + img = img.scaledToHeight(height,Qt::SmoothTransformation); + + QImage destImg(width,height,QImage::Format_RGB32); + destImg.fill(Qt::black); + QPainter p(&destImg); + + p.drawImage((width-img.width())/2,(height-img.height())/2,img); + + if(folderCover) + { + if(session.getDisplayType()=="@2x") + p.drawImage(0,0,QImage(":/images/f_retina.png")); + else + p.drawImage(0,0,QImage(":/images/f.png")); + } + + QByteArray ba; + QBuffer buffer(&ba); + buffer.open(QIODevice::WriteOnly); + destImg.save(&buffer, "JPG"); + response.write(ba,true); + } + //DONE else, hay que devolver un 404 + else + { + response.setStatus(404,"not found"); + response.write("404 not found",true); + } +} + diff --git a/YACReaderLibrary/server/controllers/covercontroller.h b/YACReaderLibrary/server/controllers/covercontroller.h new file mode 100644 index 00000000..d4948f7c --- /dev/null +++ b/YACReaderLibrary/server/controllers/covercontroller.h @@ -0,0 +1,20 @@ +#ifndef COVERCONTROLLER_H +#define COVERCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class CoverController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(CoverController); +public: + + /** Constructor */ + CoverController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // COVERCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/dumpcontroller.cpp b/YACReaderLibrary/server/controllers/dumpcontroller.cpp new file mode 100644 index 00000000..2b67e536 --- /dev/null +++ b/YACReaderLibrary/server/controllers/dumpcontroller.cpp @@ -0,0 +1,62 @@ +/** + @file + @author Stefan Frings +*/ + +#include "dumpcontroller.h" +#include +#include + +DumpController::DumpController(){} + +void DumpController::service(HttpRequest& request, HttpResponse& response) { + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.setCookie(HttpCookie("firstCookie","hello",600)); + response.setCookie(HttpCookie("secondCookie","world",600)); + + QByteArray body(""); + body.append("Request:"); + body.append("
Method: "); + body.append(request.getMethod()); + body.append("
Path: "); + body.append(request.getPath()); + body.append("
Version: "); + body.append(request.getVersion()); + + body.append("

Headers:"); + QMapIterator i(request.getHeaderMap()); + while (i.hasNext()) { + i.next(); + body.append("
"); + body.append(i.key()); + body.append("="); + body.append(i.value()); + } + + body.append("

Parameters:"); + i=QMapIterator(request.getParameterMap()); + while (i.hasNext()) { + i.next(); + body.append("
"); + body.append(i.key()); + body.append("="); + body.append(i.value()); + } + + body.append("

Cookies:"); + i=QMapIterator(request.getCookieMap()); + while (i.hasNext()) { + i.next(); + body.append("
"); + body.append(i.key()); + body.append("="); + body.append(i.value()); + } + + body.append("

Body:
"); + body.append(request.getBody()); + + body.append(""); + response.write(body,true); +} diff --git a/YACReaderLibrary/server/controllers/dumpcontroller.h b/YACReaderLibrary/server/controllers/dumpcontroller.h new file mode 100644 index 00000000..a3787dbb --- /dev/null +++ b/YACReaderLibrary/server/controllers/dumpcontroller.h @@ -0,0 +1,29 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef DUMPCONTROLLER_H +#define DUMPCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller dumps the received HTTP request in the response. +*/ + +class DumpController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(DumpController); +public: + + /** Constructor */ + DumpController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // DUMPCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/errorcontroller.cpp b/YACReaderLibrary/server/controllers/errorcontroller.cpp new file mode 100644 index 00000000..4bff204b --- /dev/null +++ b/YACReaderLibrary/server/controllers/errorcontroller.cpp @@ -0,0 +1,26 @@ +#include "errorcontroller.h" + +#include "template.h" +#include "../static.h" + + +ErrorController::ErrorController(int errorCode) +:error(errorCode) +{} + +void ErrorController::service(HttpRequest& request, HttpResponse& response) +{ + Q_UNUSED(request) + switch(error) + { + case 300: + response.setStatus(300,"redirect"); + response.write(" ", true); + break; + case 404: + response.setStatus(404,"not found"); + response.write("404 not found",true); + break; + } + +} \ No newline at end of file diff --git a/YACReaderLibrary/server/controllers/errorcontroller.h b/YACReaderLibrary/server/controllers/errorcontroller.h new file mode 100644 index 00000000..82f997df --- /dev/null +++ b/YACReaderLibrary/server/controllers/errorcontroller.h @@ -0,0 +1,22 @@ +#ifndef ERRORCONTROLLER_H +#define ERRORCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class ErrorController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(ErrorController); +public: + + /** Constructor */ + ErrorController(int errorCode); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +private: + int error; +}; + +#endif // ERRORCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/fileuploadcontroller.cpp b/YACReaderLibrary/server/controllers/fileuploadcontroller.cpp new file mode 100644 index 00000000..30d76035 --- /dev/null +++ b/YACReaderLibrary/server/controllers/fileuploadcontroller.cpp @@ -0,0 +1,38 @@ +/** + @file + @author Stefan Frings +*/ + +#include "fileuploadcontroller.h" + +FileUploadController::FileUploadController() {} + +void FileUploadController::service(HttpRequest& request, HttpResponse& response) { + + if (request.getParameter("action")=="show") { + response.setHeader("Content-Type", "image/jpeg"); + QTemporaryFile* file=request.getUploadedFile("file1"); + if (file) { + while (!file->atEnd() && !file->error()) { + QByteArray buffer=file->read(65536); + response.write(buffer); + } + } + else { + response.write("upload failed"); + } + } + + else { + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.write(""); + response.write("Upload a JPEG image file

"); + response.write("

"); + response.write(" "); + response.write(" File:
"); + response.write(" "); + response.write("
"); + response.write("",true); + } +} + diff --git a/YACReaderLibrary/server/controllers/fileuploadcontroller.h b/YACReaderLibrary/server/controllers/fileuploadcontroller.h new file mode 100644 index 00000000..01865ea6 --- /dev/null +++ b/YACReaderLibrary/server/controllers/fileuploadcontroller.h @@ -0,0 +1,30 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef FILEUPLOADCONTROLLER_H +#define FILEUPLOADCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller displays a HTML form for file upload and recieved the file. +*/ + + +class FileUploadController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(FileUploadController); +public: + + /** Constructor */ + FileUploadController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // FILEUPLOADCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/foldercontroller.cpp b/YACReaderLibrary/server/controllers/foldercontroller.cpp new file mode 100644 index 00000000..b6239106 --- /dev/null +++ b/YACReaderLibrary/server/controllers/foldercontroller.cpp @@ -0,0 +1,329 @@ +#include "foldercontroller.h" +#include "controllers/errorcontroller.h" + +#include "db_helper.h" //get libraries +#include "comic_db.h" + +#include "folder.h" + +#include "template.h" +#include "../static.h" + +#include "qnaturalsorting.h" + +struct LibraryItemSorter +{ + bool operator()(const LibraryItem * a,const LibraryItem * b) const + { + return naturalSortLessThanCI(a->name,b->name); + } +}; + +FolderController::FolderController() {} + +void FolderController::service(HttpRequest& request, HttpResponse& response) +{ + HttpSession session=Static::sessionStore->getSession(request,response,false); + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.setHeader("Connection","close"); + + //QString y = session.get("xxx").toString(); + //response.writeText(QString("session xxx : %1
").arg(y)); + + Template t=Static::templateLoader->getTemplate("folder_"+session.getDeviceType(),request.getHeader("Accept-Language")); + t.enableWarnings(); + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + int libraryId = pathElements.at(2).toInt(); + QString libraryName = DBHelper::getLibraryName(libraryId); + qulonglong parentId = pathElements.at(4).toULongLong(); + + parentId = qMax(1,parentId); + + QString folderName = DBHelper::getFolderName(libraryName,parentId); + if(folderName.isEmpty()) + { + ErrorController(300).service(request,response); + return; + } + + if(parentId!=1) + t.setVariable("folder.name",folderName); + else + t.setVariable("folder.name",libraryName); + QList folderContent = DBHelper::getFolderContentFromLibrary(libraryName,parentId); + QList folderComics = DBHelper::getFolderComicsFromLibrary(libraryName,parentId); + + //response.writeText(libraryName); + + folderContent.append(folderComics); + + qSort(folderContent.begin(),folderContent.end(),LibraryItemSorter()); + folderComics.clear(); + + qulonglong backId = DBHelper::getParentFromComicFolderId(libraryName,parentId); + + int page = 0; + QByteArray p = request.getParameter("page"); + if(p.length() != 0) + page = p.toInt(); + + // /comicIdi/pagei/comicIdj/pagej/....../comicIdn/pagen + //QString currentPath = session.get("currentPath").toString(); + //QStringList pathSize = currentPath.split("/").last().toInt; + + bool fromUp = false; + + QMultiMap map = request.getParameterMap(); + if(map.contains("up")) + fromUp = true; + + int upPage = 0; + + if(parentId == 1) + session.clearFoldersPath(); + else + { + if(fromUp) + session.popFolder(); + else + if(session.getFoldersPath().contains(parentId)) + { + while(session.topFolder()!=parentId) + session.popFolder(); + } + else + session.pushFolder(parentId); + } + + if(backId == 1 && parentId == 1) + { + session.popPage(); + session.pushPage(page); + t.setVariable(QString("upurl"),"/?page=0"); + } + else + { + if(fromUp) + { + session.popPage(); + upPage = session.topPage(); + page = upPage; + } + else //este nivel puede haberse cargado por primera vez ó puede que estemos navegando horizontalmente + if(p.length() == 0) // acabamos de entrar + { + upPage = session.topPage(); + session.pushPage(page); + } + else //navegación horizontal + { + session.popPage(); + upPage = session.topPage(); + session.pushPage(page); + } + t.setVariable(QString("upurl"),"/library/" + QString::number(libraryId) + "/folder/" +QString("%1?page=%2&up=true").arg(backId).arg(upPage)); + } + + /*if(currentPath.length()>0) + { + if(currentPath.contains(QString("%1").arg(parentId)) + { + + } + else + { + session.set("currentPath",currentPath+QString("/%1/%2").arg(parentId).arg(page); + } + }*/ + + + //t.loop("element",folderContent.length()); + + int elementsPerPage = 24; + + int numFolders = folderContent.length(); + //int numComics = folderComics.length(); + int totalLength = folderContent.length() + folderComics.length(); + +// int numFolderPages = numFolders / elementsPerPage + ((numFolders%elementsPerPage)>0?1:0); + int numPages = totalLength / elementsPerPage + ((totalLength%elementsPerPage)>0?1:0); + + //response.writeText(QString("Number of pages : %1
").arg(numPages)); + + if(page < 0) + page = 0; + else if(page >= numPages) + page = numPages-1; + + int indexCurrentPage = page*elementsPerPage; + int numFoldersAtCurrentPage = qMax(0,qMin(numFolders - indexCurrentPage, elementsPerPage)); + + //PATH + QStack foldersPath = session.getFoldersPath(); + t.setVariable(QString("library.name"),libraryName); + t.setVariable(QString("library.url"),QString("/library/%1/folder/1").arg(libraryId)); + t.loop("path",foldersPath.length()); + for(int i = 0; i < foldersPath.length(); i++){ + + t.setVariable(QString("path%1.url").arg(i),QString("/library/%1/folder/%2").arg(libraryId).arg(foldersPath[i])); + t.setVariable(QString("path%1.name").arg(i),DBHelper::getFolderName(libraryName,foldersPath[i])); + } + + t.loop("element",numFoldersAtCurrentPage); + int i = 0; + while(iname); + if(item->isDir()) + { + t.setVariable(QString("element%1.class").arg(i),"folder"); + + QList children = DBHelper::getFolderComicsFromLibrary(libraryName, item->id); + if(children.length()>0) + { + const ComicDB * comic = static_cast(children.at(0)); + t.setVariable(QString("element%1.image.url").arg(i),QString("/library/%1/cover/%2.jpg?folderCover=true").arg(libraryId).arg(comic->info.hash)); + } + else + t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); + + t.setVariable(QString("element%1.browse").arg(i),QString("browse").arg(QString("/library/%1/folder/%2").arg(libraryId).arg(item->id))); + + //t.setVariable(QString("element%1.url").arg(i),"/library/"+libraryName+"/folder/"+QString("%1").arg(folderContent.at(i + (page*10))->id)); + //t.setVariable(QString("element%1.downloadurl").arg(i),"/library/"+libraryName+"/folder/"+QString("%1/info").arg(folderContent.at(i + (page*elementsPerPage))->id)); + + t.setVariable(QString("element%1.download").arg(i),QString("import").arg("/library/"+QString::number(libraryId)+"/folder/"+QString("%1/info").arg(folderContent.at(i + (page*elementsPerPage))->id))); + t.setVariable(QString("element%1.read").arg(i),""); + + t.setVariable(QString("element%1.size").arg(i),""); + t.setVariable(QString("element%1.pages").arg(i),""); + t.setVariable(QString("element%1.status").arg(i),""); + } + else + { + t.setVariable(QString("element%1.class").arg(i),"cover"); + const ComicDB * comic = (ComicDB *)item; + t.setVariable(QString("element%1.browse").arg(i),""); + //t.setVariable(QString("element%1.downloadurl").arg(i),"/library/"+libraryName+"/comic/"+QString("%1").arg(comic->id)); + if(!session.isComicOnDevice(comic->info.hash) && !session.isComicDownloaded(comic->info.hash)) + t.setVariable(QString("element%1.download").arg(i),QString("import").arg("/library/"+QString::number(libraryId)+"/comic/"+QString("%1").arg(comic->id))); + else if (!session.isComicDownloaded(comic->info.hash)) + t.setVariable(QString("element%1.download").arg(i),QString("
imported
")); + else + t.setVariable(QString("element%1.download").arg(i),QString("
importing
")); + + //t.setVariable(QString("element%1.image.url").arg(i),"/images/f.png"); + + t.setVariable(QString("element%1.read").arg(i),QString("read").arg("/library/"+QString::number(libraryId)+"/comic/"+QString("%1").arg(comic->id)+"/remote")); + + t.setVariable(QString("element%1.image.url").arg(i),QString("/library/%1/cover/%2.jpg").arg(libraryId).arg(comic->info.hash)); + + t.setVariable(QString("element%1.size").arg(i),"" + QString::number(comic->info.hash.right(comic->info.hash.length()-40).toInt()/1024.0/1024.0,'f',2)+"Mb"); + if(comic->info.hasBeenOpened) + t.setVariable(QString("element%1.pages").arg(i),QString("%1/%2 pages").arg(comic->info.currentPage).arg(comic->info.numPages.toInt())); + else + t.setVariable(QString("element%1.pages").arg(i),QString("%1 pages").arg(comic->info.numPages.toInt())); + + if(comic->info.read) + t.setVariable(QString("element%1.status").arg(i), QString("
")); + else if(comic->info.hasBeenOpened) + t.setVariable(QString("element%1.status").arg(i), QString("
")); + else + t.setVariable(QString("element%1.status").arg(i),""); + + + } + i++; + } + + if(numPages > 1) + { + t.setCondition("pageIndex",true); + + QMap indexCount; + + QString firstChar; + int xyz = 1; + for(QList::const_iterator itr=folderContent.constBegin();itr!=folderContent.constEnd();itr++) + { + firstChar = QString((*itr)->name[0]).toUpper(); + firstChar = firstChar.normalized(QString::NormalizationForm_D).at(0);//TODO _D or _KD?? + bool ok; + /*int dec = */firstChar.toInt(&ok, 10); + if(ok) + firstChar = "#"; + //response.writeText(QString("%1 - %2
").arg((*itr)->name).arg(xyz)); + if(indexCount.contains(firstChar)) + indexCount.insert(firstChar, indexCount.value(firstChar)+1); + else + indexCount.insert(firstChar, 1); + + xyz++; + } + + QList index = indexCount.keys(); + if(index.length()>1) + { + t.setCondition("alphaIndex",true); + + qSort(index.begin(),index.end(),naturalSortLessThanCI); + t.loop("index",index.length()); + int i=0; + int count=0; + int indexPage=0; + for(QList::const_iterator itr=index.constBegin();itr!=index.constEnd();itr++) + { + //response.writeText(QString("%1 - %2
").arg(*itr).arg(count)); + t.setVariable(QString("index%1.indexname").arg(i), *itr); + t.setVariable(QString("index%1.url").arg(i),QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg(indexPage)); + i++; + count += indexCount.value(*itr); + indexPage = count/elementsPerPage; + } + } + else + { + t.loop("index",0); + t.setCondition("alphaIndex",false); + + } + + t.loop("page",numPages); + int z = 0; + while(z < numPages) + { + + t.setVariable(QString("page%1.url").arg(z),QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg(z)); + t.setVariable(QString("page%1.number").arg(z),QString("%1").arg(z+1)); + if(page == z) + t.setVariable(QString("page%1.current").arg(z),"current"); + else + t.setVariable(QString("page%1.current").arg(z),""); + z++; + } + + t.setVariable("page.first",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg(0)); + t.setVariable("page.previous",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg((page==0)?page:page-1)); + t.setVariable("page.next",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg((page==numPages-1)?page:page+1)); + t.setVariable("page.last",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(parentId).arg(numPages-1)); + t.setCondition("index", true); + } + else + { + + t.loop("page",0); + t.loop("index",0); + t.setCondition("index", false); + t.setCondition("pageIndex",false); + t.setCondition("alphaIndex",false); + } + + t.setVariable("page",QString("%1").arg(page+1)); + t.setVariable("pages",QString("%1").arg(numPages)); + + response.write(t.toLatin1(),true); + +} diff --git a/YACReaderLibrary/server/controllers/foldercontroller.h b/YACReaderLibrary/server/controllers/foldercontroller.h new file mode 100644 index 00000000..4d757869 --- /dev/null +++ b/YACReaderLibrary/server/controllers/foldercontroller.h @@ -0,0 +1,20 @@ +#ifndef FOLDERCONTROLLER_H +#define FOLDERCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class FolderController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(FolderController); +public: + + /** Constructor */ + FolderController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // FOLDERCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/folderinfocontroller.cpp b/YACReaderLibrary/server/controllers/folderinfocontroller.cpp new file mode 100644 index 00000000..9dbb2103 --- /dev/null +++ b/YACReaderLibrary/server/controllers/folderinfocontroller.cpp @@ -0,0 +1,39 @@ +#include "folderinfocontroller.h" +#include "db_helper.h" //get libraries + +#include "folder.h" +#include "comic_db.h" + +#include "template.h" +#include "../static.h" + + +FolderInfoController::FolderInfoController() {} + +void FolderInfoController::service(HttpRequest& request, HttpResponse& response) +{ + response.setHeader("Content-Type", "plain/text; charset=ISO-8859-1"); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + int libraryId = pathElements.at(2).toInt(); + QString libraryName = DBHelper::getLibraryName(libraryId); + qulonglong parentId = pathElements.at(4).toULongLong(); + QList folderContent = DBHelper::getFolderContentFromLibrary(libraryName,parentId); + QList folderComics = DBHelper::getFolderComicsFromLibrary(libraryName,parentId); + + Folder * currentFolder; + for(QList::const_iterator itr = folderContent.constBegin();itr!=folderContent.constEnd();itr++) + { + currentFolder = (Folder *)(*itr); + response.writeText(QString("/library/%1/folder/%2/info\n").arg(libraryId).arg(currentFolder->id)); + } + + ComicDB * currentComic; + for(QList::const_iterator itr = folderComics.constBegin();itr!=folderComics.constEnd();itr++) + { + currentComic = (ComicDB *)(*itr); + response.writeText(QString("/library/%1/comic/%2\n").arg(libraryId).arg(currentComic->id)); + } + +} \ No newline at end of file diff --git a/YACReaderLibrary/server/controllers/folderinfocontroller.h b/YACReaderLibrary/server/controllers/folderinfocontroller.h new file mode 100644 index 00000000..69bdc6b6 --- /dev/null +++ b/YACReaderLibrary/server/controllers/folderinfocontroller.h @@ -0,0 +1,20 @@ +#ifndef FOLDERINFOCONTROLLER_H +#define FOLDERINFOCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class FolderInfoController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(FolderInfoController); +public: + + /** Constructor */ + FolderInfoController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // FOLDERINFOCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/formcontroller.cpp b/YACReaderLibrary/server/controllers/formcontroller.cpp new file mode 100644 index 00000000..7a0f2b27 --- /dev/null +++ b/YACReaderLibrary/server/controllers/formcontroller.cpp @@ -0,0 +1,64 @@ +/** + @file + @author Stefan Frings +*/ + +#include "formcontroller.h" +#include + +FormController::FormController() {} + +void FormController::service(HttpRequest& request, HttpResponse& response) { + + response.setHeader("Content-Type", "text/html; charset=utf-8"); + + QString data(request.getBody()); + + QStringList list = data.split("\n"); + + response.write(""); + response.writeText("á é í ó ú ñ -> \\ /Device type: "+list.first()); + + //test background proccesing + /*int i=0; + int j=0; + while(i<1000000000) + { + if(request.getBody().length()>1) + j++; + else + i++; + if(i%1000000 == 0) + response.write("

lista

"); + }*/ + + response.write("

lista

"); + + response.write("
    "); + + for(int i=1;i"+list.at(i)+""); + } + response.write("
",true); + + /*if (request.getParameter("action")=="show") { + response.write(""); + response.write("Name = "); + response.write(request.getParameter("name")); + response.write("
City = "); + response.write(request.getParameter("city")); + response.write("",true); + } + else { + response.write(""); + response.write("
"); + response.write(" "); + response.write(" Name:
"); + response.write(" City:
"); + response.write(" "); + response.write("
"); + response.write("",true); + }*/ +} + diff --git a/YACReaderLibrary/server/controllers/formcontroller.h b/YACReaderLibrary/server/controllers/formcontroller.h new file mode 100644 index 00000000..5ae709a8 --- /dev/null +++ b/YACReaderLibrary/server/controllers/formcontroller.h @@ -0,0 +1,30 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef FORMCONTROLLER_H +#define FORMCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller displays a HTML form and dumps the submitted input. +*/ + + +class FormController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(FormController); +public: + + /** Constructor */ + FormController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // FORMCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/librariescontroller.cpp b/YACReaderLibrary/server/controllers/librariescontroller.cpp new file mode 100644 index 00000000..6b555018 --- /dev/null +++ b/YACReaderLibrary/server/controllers/librariescontroller.cpp @@ -0,0 +1,93 @@ +#include "librariescontroller.h" +#include "db_helper.h" //get libraries +#include "yacreader_libraries.h" + +#include "template.h" +#include "../static.h" + +#include "QsLog.h" + +LibrariesController::LibrariesController() {} + +void LibrariesController::service(HttpRequest& request, HttpResponse& response) +{ + HttpSession session=Static::sessionStore->getSession(request,response); + if(session.contains("ySession")) //session is already alive check if it is needed to update comics + { + QString postData = QString::fromUtf8(request.getBody()); + if(postData.length()>0) { + QList data = postData.split("\n"); + if(data.length() > 2) { + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); + QList comics = data.at(2).split(":").at(1).split("\t"); + foreach(QString hash,comics) { + session.setComicOnDevice(hash); + } + } + else + { + if(data.length()>1) + { + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); + } + } + } + } + else + { + session.set("ySession","ok"); + + session.clearNavigationPath(); + session.clearFoldersPath(); + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.setHeader("Connection","close"); + + + QString postData = QString::fromUtf8(request.getBody()); + //response.writeText(postData); + + QList data = postData.split("\n"); + + QLOG_INFO() << "Data lenght : " << data.length(); + + if(data.length() > 2) + { + session.setDeviceType(data.at(0).split(":").at(1)); + session.setDisplayType(data.at(1).split(":").at(1)); + QList comics = data.at(2).split(":").at(1).split("\t"); + foreach(QString hash,comics) + { + session.setComicOnDevice(hash); + } + } + else //values by default, only for debug purposes. + { + session.setDeviceType("ipad"); + session.setDisplayType("@2x"); + } + + } + + Template t=Static::templateLoader->getTemplate("libraries_"+session.getDeviceType(),request.getHeader("Accept-Language")); + t.enableWarnings(); + + YACReaderLibraries libraries = DBHelper::getLibraries(); + QList names = DBHelper::getLibrariesNames(); + + t.loop("library",names.length()); + + int currentId = 0; + int i = 0; + foreach (QString name,names) { + currentId = libraries.getId(name); + t.setVariable(QString("library%1.name").arg(i),QString::number(currentId)); + t.setVariable(QString("library%1.label").arg(i),name); + i++; + } + + response.setStatus(200,"OK"); + response.write(t.toLatin1(),true); +} diff --git a/YACReaderLibrary/server/controllers/librariescontroller.h b/YACReaderLibrary/server/controllers/librariescontroller.h new file mode 100644 index 00000000..e61d873f --- /dev/null +++ b/YACReaderLibrary/server/controllers/librariescontroller.h @@ -0,0 +1,25 @@ +#ifndef LIBRARIESCONTROLLER_H +#define LIBRARIESCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller displays a HTML form and dumps the submitted input. +*/ + + +class LibrariesController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(LibrariesController); +public: + + /** Constructor */ + LibrariesController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // LIBRARIESCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/pagecontroller.cpp b/YACReaderLibrary/server/controllers/pagecontroller.cpp new file mode 100644 index 00000000..e6ccde83 --- /dev/null +++ b/YACReaderLibrary/server/controllers/pagecontroller.cpp @@ -0,0 +1,96 @@ +#include "pagecontroller.h" + +#include "../static.h" + +#include "comic.h" +#include "comiccontroller.h" +#include +#include + +#include + +#include "db_helper.h" + +PageController::PageController() {} + +void PageController::service(HttpRequest& request, HttpResponse& response) +{ + HttpSession session=Static::sessionStore->getSession(request,response,false); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + bool remote = path.endsWith("remote"); + + //QByteArray path2=request.getPath(); + //qDebug("PageController: request to -> %s ",path2.data()); + + QStringList pathElements = path.split('/'); + QString libraryName = DBHelper::getLibraryName(pathElements.at(2).toInt()); + qulonglong comicId = pathElements.at(4).toULongLong(); + unsigned int page = pathElements.at(6).toUInt(); + + //qDebug("lib name : %s",pathElements.at(2).data()); + + Comic * comicFile; + qulonglong currentComicId; + if(remote) + { + QLOG_INFO() << "se recupera comic remoto para servir páginas"; + comicFile = session.getCurrentRemoteComic(); + currentComicId = session.getCurrentRemoteComicId(); + } + else + { + QLOG_INFO() << "se recupera comic para servir páginas"; + comicFile = session.getCurrentComic(); + currentComicId = session.getCurrentComicId(); + } + + if(currentComicId != 0 && !QPointer(comicFile).isNull()) + { + if(comicId == currentComicId && page < comicFile->numPages()) + { + if(comicFile->pageIsLoaded(page)) + { + //qDebug("PageController: La página estaba cargada -> %s ",path.data()); + response.setHeader("Content-Type", "image/jpeg"); + response.setHeader("Transfer-Encoding","chunked"); + QByteArray pageData = comicFile->getRawPage(page); + QDataStream data(pageData); + char buffer[4096]; + while (!data.atEnd()) { + int len = data.readRawData(buffer,4096); + response.write(QByteArray(buffer,len)); + } + //response.write(pageData,true); + response.write(QByteArray(),true); + } + else + { + //qDebug("PageController: La página NO estaba cargada 404 -> %s ",path.data()); + response.setStatus(404,"not found"); //TODO qué mensaje enviar + response.write("404 not found",true); + } + } + else + { + if(comicId != currentComicId) + { + //delete comicFile; + if(remote) + session.dismissCurrentRemoteComic(); + else + session.dismissCurrentComic(); + } + response.setStatus(404,"not found"); //TODO qué mensaje enviar + response.write("404 not found",true); + } + } + else + { + response.setStatus(404,"not found"); + response.write("404 not found",true); + } + + //response.write(t.toLatin1(),true); + +} diff --git a/YACReaderLibrary/server/controllers/pagecontroller.h b/YACReaderLibrary/server/controllers/pagecontroller.h new file mode 100644 index 00000000..64540bc3 --- /dev/null +++ b/YACReaderLibrary/server/controllers/pagecontroller.h @@ -0,0 +1,20 @@ +#ifndef PAGECONTROLLER_H +#define PAGECONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class PageController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(PageController); +public: + + /** Constructor */ + PageController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // PAGECONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/sessioncontroller.cpp b/YACReaderLibrary/server/controllers/sessioncontroller.cpp new file mode 100644 index 00000000..34d56526 --- /dev/null +++ b/YACReaderLibrary/server/controllers/sessioncontroller.cpp @@ -0,0 +1,31 @@ +/** + @file + @author Stefan Frings +*/ + +#include "sessioncontroller.h" +#include "../static.h" +#include +#include + +SessionController::SessionController(){} + +void SessionController::service(HttpRequest& request, HttpResponse& response) { + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + + // Get current session, or create a new one + HttpSession session=Static::sessionStore->getSession(request,response); + if (!session.contains("startTime")) { + response.write("New session started. Reload this page now."); + session.set("startTime",QDateTime::currentDateTime()); + } + + else { + QDateTime startTime=session.get("startTime").toDateTime(); + response.write("Your session started "); + response.write(startTime.toString().toLatin1()); + response.write(""); + } + +} diff --git a/YACReaderLibrary/server/controllers/sessioncontroller.h b/YACReaderLibrary/server/controllers/sessioncontroller.h new file mode 100644 index 00000000..a13ee51f --- /dev/null +++ b/YACReaderLibrary/server/controllers/sessioncontroller.h @@ -0,0 +1,29 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef SESSIONCONTROLLER_H +#define SESSIONCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller demonstrates how to use sessions. +*/ + +class SessionController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(SessionController); +public: + + /** Constructor */ + SessionController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // SESSIONCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/sessionmanager.cpp b/YACReaderLibrary/server/controllers/sessionmanager.cpp new file mode 100644 index 00000000..e69de29b diff --git a/YACReaderLibrary/server/controllers/sessionmanager.h b/YACReaderLibrary/server/controllers/sessionmanager.h new file mode 100644 index 00000000..e69de29b diff --git a/YACReaderLibrary/server/controllers/templatecontroller.cpp b/YACReaderLibrary/server/controllers/templatecontroller.cpp new file mode 100644 index 00000000..d1816808 --- /dev/null +++ b/YACReaderLibrary/server/controllers/templatecontroller.cpp @@ -0,0 +1,31 @@ +/** + @file + @author Stefan Frings +*/ + +#include "templatecontroller.h" +#include "template.h" +#include "../static.h" + +TemplateController::TemplateController(){} + +void TemplateController::service(HttpRequest& request, HttpResponse& response) { + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + + Template t=Static::templateLoader->getTemplate("demo",request.getHeader("Accept-Language")); + t.enableWarnings(); + t.setVariable("path",request.getPath()); + QMap headers=request.getHeaderMap(); + QMapIterator iterator(headers); + t.loop("header",headers.size()); + int i=0; + while (iterator.hasNext()) { + iterator.next(); + t.setVariable(QString("header%1.name").arg(i),QString(iterator.key())); + t.setVariable(QString("header%1.value").arg(i),QString(iterator.value())); + ++i; + } + + response.write(t.toLatin1(),true); +} diff --git a/YACReaderLibrary/server/controllers/templatecontroller.h b/YACReaderLibrary/server/controllers/templatecontroller.h new file mode 100644 index 00000000..c5b0077d --- /dev/null +++ b/YACReaderLibrary/server/controllers/templatecontroller.h @@ -0,0 +1,30 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef TEMPLATECONTROLLER_H +#define TEMPLATECONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +/** + This controller generates a website using the template engine. + It generates a Latin1 (ISO-8859-1) encoded website from a UTF-8 encoded template file. +*/ + +class TemplateController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(TemplateController); + public: + + /** Constructor */ + TemplateController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); + }; + +#endif // TEMPLATECONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp b/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp new file mode 100644 index 00000000..793d1b72 --- /dev/null +++ b/YACReaderLibrary/server/controllers/updatecomiccontroller.cpp @@ -0,0 +1,46 @@ +#include "updatecomiccontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +#include "template.h" +#include "../static.h" + +#include "comic_db.h" +#include "comic.h" + +#include "QsLog.h" + +UpdateComicController::UpdateComicController(){} + +void UpdateComicController::service(HttpRequest &request, HttpResponse &response) +{ + HttpSession session=Static::sessionStore->getSession(request,response,false); + + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + qulonglong libraryId = pathElements.at(2).toULongLong(); + QString libraryName = DBHelper::getLibraryName(libraryId); + qulonglong comicId = pathElements.at(4).toULongLong(); + + QString postData = QString::fromUtf8(request.getBody()); + + QLOG_INFO() << "POST DATA: " << postData; + + if(postData.length()>0) { + QList data = postData.split("\n"); + int currentPage = data.at(0).split(":").at(1).toInt(); + ComicInfo info; + info.currentPage = currentPage; + info.id = comicId; + DBHelper::updateProgress(libraryId,info); + } + else + { + response.setStatus(412,"No comic info received"); + response.writeText("",true); + return; + } + + response.write("OK",true); +} diff --git a/YACReaderLibrary/server/controllers/updatecomiccontroller.h b/YACReaderLibrary/server/controllers/updatecomiccontroller.h new file mode 100644 index 00000000..13ec4f58 --- /dev/null +++ b/YACReaderLibrary/server/controllers/updatecomiccontroller.h @@ -0,0 +1,22 @@ +#ifndef UPDATECOMICCONTROLLER_H +#define UPDATECOMICCONTROLLER_H + + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + + +class UpdateComicController : public HttpRequestHandler +{ + Q_OBJECT + Q_DISABLE_COPY(UpdateComicController); + +public: + UpdateComicController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // UPDATECOMICCONTROLLER_H diff --git a/YACReaderLibrary/server/documentcache.h b/YACReaderLibrary/server/documentcache.h new file mode 100644 index 00000000..06ff67ac --- /dev/null +++ b/YACReaderLibrary/server/documentcache.h @@ -0,0 +1,4 @@ +#ifndef DOCUMENTCACHE_H +#define DOCUMENTCACHE_H + +#endif // DOCUMENTCACHE_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/bfHttpServer.pri b/YACReaderLibrary/server/lib/bfHttpServer/bfHttpServer.pri new file mode 100644 index 00000000..a109a573 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/bfHttpServer.pri @@ -0,0 +1,12 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += $$PWD/httplistener.h $$PWD/httpconnectionhandler.h $$PWD/httpconnectionhandlerpool.h $$PWD/httprequest.h $$PWD/httpresponse.h $$PWD/httpcookie.h $$PWD/httprequesthandler.h +HEADERS += $$PWD/httpsession.h $$PWD/httpsessionstore.h +HEADERS += $$PWD/staticfilecontroller.h + +SOURCES += $$PWD/httplistener.cpp $$PWD/httpconnectionhandler.cpp $$PWD/httpconnectionhandlerpool.cpp $$PWD/httprequest.cpp $$PWD/httpresponse.cpp $$PWD/httpcookie.cpp $$PWD/httprequesthandler.cpp +SOURCES += $$PWD/httpsession.cpp $$PWD/httpsessionstore.cpp +SOURCES += $$PWD/staticfilecontroller.cpp + +OTHER_FILES += $$PWD/../doc/readme.txt diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.cpp new file mode 100644 index 00000000..c085937d --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.cpp @@ -0,0 +1,164 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httpconnectionhandler.h" +#include "httpresponse.h" +#include +#include + +HttpConnectionHandler::HttpConnectionHandler(QSettings* settings, HttpRequestHandler* requestHandler) + : QThread() +{ + Q_ASSERT(settings!=0); + Q_ASSERT(requestHandler!=0); + this->settings=settings; + this->requestHandler=requestHandler; + currentRequest=0; + busy = false; + // execute signals in my own thread + moveToThread(this); + socket.moveToThread(this); + readTimer.moveToThread(this); + connect(&socket, SIGNAL(readyRead()), SLOT(read())); + connect(&socket, SIGNAL(disconnected()), SLOT(disconnected())); + connect(&readTimer, SIGNAL(timeout()), SLOT(readTimeout())); + readTimer.setSingleShot(true); + qDebug("HttpConnectionHandler (%p): constructed", this); + this->start(); +} + + +HttpConnectionHandler::~HttpConnectionHandler() { + socket.close(); + quit(); + wait(); + qDebug("HttpConnectionHandler (%p): destroyed", this); +} + + +void HttpConnectionHandler::run() { + qDebug("HttpConnectionHandler (%p): thread started", this); + try { + exec(); + } + catch (...) { + qCritical("HttpConnectionHandler (%p): an uncatched exception occured in the thread",this); + } + qDebug("HttpConnectionHandler (%p): thread stopped", this); + // Change to the main thread, otherwise deleteLater() would not work + moveToThread(QCoreApplication::instance()->thread()); +} + + +void HttpConnectionHandler::handleConnection(tSocketDescriptor socketDescriptor) { + qDebug("HttpConnectionHandler (%p): handle new connection", this); + busy = true; + Q_ASSERT(socket.isOpen()==false); // if not, then the handler is already busy + + if (!socket.setSocketDescriptor(socketDescriptor)) { + qCritical("HttpConnectionHandler (%p): cannot initialize socket: %s", this,qPrintable(socket.errorString())); + return; + } + // Start timer for read timeout + int readTimeout=settings->value("readTimeout",10000).toInt(); + readTimer.start(readTimeout); + // delete previous request + delete currentRequest; + currentRequest=0; +} + + +bool HttpConnectionHandler::isBusy() { + return busy; +} + +void HttpConnectionHandler::setBusy() { + this->busy = true; +} + + +void HttpConnectionHandler::readTimeout() { + qDebug("HttpConnectionHandler (%p): read timeout occured",this); + + //Commented out because QWebView cannot handle this. + //socket.write("HTTP/1.1 408 request timeout\r\nConnection: close\r\n\r\n408 request timeout\r\n"); + + socket.disconnectFromHost(); + delete currentRequest; + currentRequest=0; +} + + +void HttpConnectionHandler::disconnected() { + qDebug("HttpConnectionHandler (%p): disconnected", this); + socket.close(); + readTimer.stop(); + busy = false; +} + +void HttpConnectionHandler::read() { +#ifdef SUPERVERBOSE + qDebug("HttpConnectionHandler (%p): read input",this); +#endif + + // Create new HttpRequest object if necessary + if (!currentRequest) { + currentRequest=new HttpRequest(settings); + } + + // Collect data for the request object + while (socket.bytesAvailable() && currentRequest->getStatus()!=HttpRequest::complete && currentRequest->getStatus()!=HttpRequest::abort) { + currentRequest->readFromSocket(socket); + if (currentRequest->getStatus()==HttpRequest::waitForBody) { + // Restart timer for read timeout, otherwise it would + // expire during large file uploads. + int readTimeout=settings->value("readTimeout",10000).toInt(); + readTimer.start(readTimeout); + } + } + + // If the request is aborted, return error message and close the connection + if (currentRequest->getStatus()==HttpRequest::abort) { + socket.write("HTTP/1.1 413 entity too large\r\nConnection: close\r\n\r\n413 Entity too large\r\n"); + socket.disconnectFromHost(); + delete currentRequest; + currentRequest=0; + return; + } + + // If the request is complete, let the request mapper dispatch it + if (currentRequest->getStatus()==HttpRequest::complete) { + readTimer.stop(); + qDebug("HttpConnectionHandler (%p): received request",this); + HttpResponse response(&socket); + //response.setHeader("Connection","close"); No funciona bien con NSURLConnection + try { + requestHandler->service(*currentRequest, response); + } + catch (...) { + qCritical("HttpConnectionHandler (%p): An uncatched exception occured in the request handler",this); + } + + // Finalize sending the response if not already done + if (!response.hasSentLastPart()) { + response.write(QByteArray(),true); + } + + socket.disconnectFromHost(); //CAMBIADO sólo se van a soportar conexiones NO persistentes + + // Close the connection after delivering the response, if requested + //if (QString::compare(currentRequest->getHeader("Connection"),"close",Qt::CaseInsensitive)==0) { + // socket.disconnectFromHost(); + //} + //else { + // // Start timer for next request + // int readTimeout=settings->value("readTimeout",10000).toInt(); + // readTimer.start(readTimeout); + //} + // Prepare for next request + delete currentRequest; + currentRequest=0; + } +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.h b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.h new file mode 100644 index 00000000..0e8b4483 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandler.h @@ -0,0 +1,103 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPCONNECTIONHANDLER_H +#define HTTPCONNECTIONHANDLER_H + +#include +#include +#include +#include +#include "httprequest.h" +#include "httprequesthandler.h" + +/** + The connection handler accepts incoming connections and dispatches incoming requests to to a + request mapper. Since HTTP clients can send multiple requests before waiting for the response, + the incoming requests are queued and processed one after the other. +

+ Example for the required configuration settings: +

+  readTimeout=60000
+  maxRequestSize=16000
+  maxMultiPartSize=1000000
+  
+

+ The readTimeout value defines the maximum time to wait for a complete HTTP request. + @see HttpRequest for description of config settings maxRequestSize and maxMultiPartSize +*/ + +#if QT_VERSION >= 0x050000 + typedef qintptr tSocketDescriptor; +#else + typedef int tSocketDescriptor; +#endif + +class HttpConnectionHandler : public QThread { + Q_OBJECT + Q_DISABLE_COPY(HttpConnectionHandler) +public: + + /** + Constructor. + @param settings Configuration settings of the HTTP webserver + @param requestHandler handler that will process each incomin HTTP request + */ + HttpConnectionHandler(QSettings* settings, HttpRequestHandler* requestHandler); + + /** Destructor */ + virtual ~HttpConnectionHandler(); + + /** Returns true, if this handler is in use. */ + bool isBusy(); + + /** Mark this handler as busy */ + void setBusy(); + +private: + + /** Configuration settings */ + QSettings* settings; + + /** TCP socket of the current connection */ + QTcpSocket socket; + + /** Time for read timeout detection */ + QTimer readTimer; + + /** Storage for the current incoming HTTP request */ + HttpRequest* currentRequest; + + /** Dispatches received requests to services */ + HttpRequestHandler* requestHandler; + + /** This shows the busy-state from a very early time */ + bool busy; + + /** Executes the htreads own event loop */ + void run(); + +public slots: + + /** + Received from from the listener, when the handler shall start processing a new connection. + @param socketDescriptor references the accepted connection. + */ + void handleConnection(tSocketDescriptor socketDescriptor); + +private slots: + + /** Received from the socket when a read-timeout occured */ + void readTimeout(); + + /** Received from the socket when incoming data can be read */ + void read(); + + /** Received from the socket when a connection has been closed */ + void disconnected(); + +}; + +#endif // HTTPCONNECTIONHANDLER_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.cpp new file mode 100644 index 00000000..fbf70b49 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.cpp @@ -0,0 +1,64 @@ +#include "httpconnectionhandlerpool.h" + +HttpConnectionHandlerPool::HttpConnectionHandlerPool(QSettings* settings, HttpRequestHandler* requestHandler) + : QObject() +{ + Q_ASSERT(settings!=0); + this->settings=settings; + this->requestHandler=requestHandler; + cleanupTimer.start(settings->value("cleanupInterval",10000).toInt()); + connect(&cleanupTimer, SIGNAL(timeout()), SLOT(cleanup())); +} + + +HttpConnectionHandlerPool::~HttpConnectionHandlerPool() { + foreach(HttpConnectionHandler* handler, pool) { + connect(handler,SIGNAL(finished()),handler,SLOT(deleteLater())); + handler->quit(); + } +} + + +HttpConnectionHandler* HttpConnectionHandlerPool::getConnectionHandler() { + HttpConnectionHandler* freeHandler=0; + mutex.lock(); + // find a free handler in pool + foreach(HttpConnectionHandler* handler, pool) { + if (!handler->isBusy()) { + freeHandler=handler; + freeHandler->setBusy(); + break; + } + } + // create a new handler, if necessary + if (!freeHandler) { + int maxConnectionHandlers=settings->value("maxThreads",1000).toInt(); + if (pool.count()setBusy(); + pool.append(freeHandler); + } + } + mutex.unlock(); + return freeHandler; +} + + + +void HttpConnectionHandlerPool::cleanup() { + int maxIdleHandlers=settings->value("minThreads",50).toInt(); + int idleCounter=0; + mutex.lock(); + foreach(HttpConnectionHandler* handler, pool) { + if (!handler->isBusy()) { + if (++idleCounter > maxIdleHandlers) { + pool.removeOne(handler); + qDebug("HttpConnectionHandlerPool: Removed connection handler (%p), pool size is now %i",handler,pool.size()); + connect(handler,SIGNAL(finished()),handler,SLOT(deleteLater())); + handler->quit(); + break; // remove only one handler in each interval + } + } + } + mutex.unlock(); +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.h b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.h new file mode 100644 index 00000000..2dc94338 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpconnectionhandlerpool.h @@ -0,0 +1,73 @@ +#ifndef HTTPCONNECTIONHANDLERPOOL_H +#define HTTPCONNECTIONHANDLERPOOL_H + +#include +#include +#include +#include +#include "httpconnectionhandler.h" + +/** + Pool of http connection handlers. Connection handlers are created on demand and idle handlers are + cleaned up in regular time intervals. +

+ Example for the required configuration settings: +

+  minThreads=1
+  maxThreads=100
+  cleanupInterval=1000
+  maxRequestSize=16000
+  maxMultiPartSize=1000000
+  
+ The pool is empty initially and grows with the number of concurrent + connections. A timer removes one idle connection handler at each + interval, but it leaves some spare handlers in memory to improve + performance. + @see HttpConnectionHandler for description of config settings readTimeout + @see HttpRequest for description of config settings maxRequestSize and maxMultiPartSize +*/ + +class HttpConnectionHandlerPool : public QObject { + Q_OBJECT + Q_DISABLE_COPY(HttpConnectionHandlerPool) +public: + + /** + Constructor. + @param settings Configuration settings for the HTTP server. Must not be 0. + @param requestHandler The handler that will process each received HTTP request. + @warning The requestMapper gets deleted by the destructor of this pool + */ + HttpConnectionHandlerPool(QSettings* settings, HttpRequestHandler* requestHandler); + + /** Destructor */ + virtual ~HttpConnectionHandlerPool(); + + /** Get a free connection handler, or 0 if not available. */ + HttpConnectionHandler* getConnectionHandler(); + +private: + + /** Settings for this pool */ + QSettings* settings; + + /** Will be assigned to each Connectionhandler during their creation */ + HttpRequestHandler* requestHandler; + + /** Pool of connection handlers */ + QList pool; + + /** Timer to clean-up unused connection handler */ + QTimer cleanupTimer; + + /** Used to synchronize threads */ + QMutex mutex; + +private slots: + + /** Received from the clean-up timer. */ + void cleanup(); + +}; + +#endif // HTTPCONNECTIONHANDLERPOOL_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpcookie.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpcookie.cpp new file mode 100644 index 00000000..3f5be929 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpcookie.cpp @@ -0,0 +1,199 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httpcookie.h" + +HttpCookie::HttpCookie() { + version=1; + maxAge=0; + secure=false; +} + +HttpCookie::HttpCookie(const QByteArray name, const QByteArray value, const int maxAge, const QByteArray path, const QByteArray comment, const QByteArray domain, const bool secure) { + this->name=name; + this->value=value; + this->maxAge=maxAge; + this->path=path; + this->comment=comment; + this->domain=domain; + this->secure=secure; + this->version=1; +} + +HttpCookie::HttpCookie(const QByteArray source) { + version=1; + maxAge=0; + secure=false; + QList list=splitCSV(source); + foreach(QByteArray part, list) { + + // Split the part into name and value + QByteArray name; + QByteArray value; + int posi=part.indexOf('='); + if (posi) { + name=part.left(posi).trimmed(); + value=part.mid(posi+1).trimmed(); + } + else { + name=part.trimmed(); + value=""; + } + + // Set fields + if (name=="Comment") { + comment=value; + } + else if (name=="Domain") { + domain=value; + } + else if (name=="Max-Age") { + maxAge=value.toInt(); + } + else if (name=="Path") { + path=value; + } + else if (name=="Secure") { + secure=true; + } + else if (name=="Version") { + version=value.toInt(); + } + else { + if (this->name.isEmpty()) { + this->name=name; + this->value=value; + } + else { + qWarning("HttpCookie: Ignoring unknown %s=%s",name.data(),value.data()); + } + } + } +} + +QByteArray HttpCookie::toByteArray() const { + QByteArray buffer(name); + buffer.append('='); + buffer.append(value); + if (!comment.isEmpty()) { + buffer.append("; Comment="); + buffer.append(comment); + } + if (!domain.isEmpty()) { + buffer.append("; Domain="); + buffer.append(domain); + } + if (maxAge!=0) { + buffer.append("; Max-Age="); + buffer.append(QByteArray::number(maxAge)); + } + if (!path.isEmpty()) { + buffer.append("; Path="); + buffer.append(path); + } + if (secure) { + buffer.append("; Secure"); + } + buffer.append("; Version="); + buffer.append(QByteArray::number(version)); + return buffer; +} + +void HttpCookie::setName(const QByteArray name){ + this->name=name; +} + +void HttpCookie::setValue(const QByteArray value){ + this->value=value; +} + +void HttpCookie::setComment(const QByteArray comment){ + this->comment=comment; +} + +void HttpCookie::setDomain(const QByteArray domain){ + this->domain=domain; +} + +void HttpCookie::setMaxAge(const int maxAge){ + this->maxAge=maxAge; +} + +void HttpCookie::setPath(const QByteArray path){ + this->path=path; +} + +void HttpCookie::setSecure(const bool secure){ + this->secure=secure; +} + +QByteArray HttpCookie::getName() const { + return name; +} + +QByteArray HttpCookie::getValue() const { + return value; +} + +QByteArray HttpCookie::getComment() const { + return comment; +} + +QByteArray HttpCookie::getDomain() const { + return domain; +} + +int HttpCookie::getMaxAge() const { + return maxAge; +} + +QByteArray HttpCookie::getPath() const { + return path; +} + +bool HttpCookie::getSecure() const { + return secure; +} + +int HttpCookie::getVersion() const { + return version; +} + +QList HttpCookie::splitCSV(const QByteArray source) { + bool inString=false; + QList list; + QByteArray buffer; + for (int i=0; i +#include + +/** + HTTP cookie as defined in RFC 2109. This class can also parse + RFC 2965 cookies, but skips fields that are not defined in RFC + 2109. +*/ + +class HttpCookie +{ +public: + + /** Creates an empty cookie */ + HttpCookie(); + + /** + Create a cookie and set name/value pair. + @param name name of the cookie + @param value value of the cookie + @param maxAge maximum age of the cookie in seconds. 0=discard immediately + @param path Path for that the cookie will be sent, default="/" which means the whole domain + @param comment Optional comment, may be displayed by the web browser somewhere + @param domain Optional domain for that the cookie will be sent. Defaults to the current domain + @param secure If true, the cookie will only be sent on secure connections + */ + HttpCookie(const QByteArray name, const QByteArray value, const int maxAge, const QByteArray path="/", const QByteArray comment=QByteArray(), const QByteArray domain=QByteArray(), const bool secure=false); + + /** + Create a cookie from a string. + @param source String as received in a HTTP Cookie2 header. + */ + HttpCookie(const QByteArray source); + + /** Convert this cookie to a string that may be used in a Set-Cookie2 header. */ + QByteArray toByteArray() const ; + + /** + Split a string list into parts, where each part is delimited by semicolon. + Semicolons within double quotes are skipped. Double quotes are removed. + */ + static QList splitCSV(const QByteArray source); + + /** Set the name of this cookie */ + void setName(const QByteArray name); + + /** Set the value of this cookie */ + void setValue(const QByteArray value); + + /** Set the comment of this cookie */ + void setComment(const QByteArray comment); + + /** Set the domain of this cookie */ + void setDomain(const QByteArray domain); + + /** Set the maximum age of this cookie in seconds. 0=discard immediately */ + void setMaxAge(const int maxAge); + + /** Set the path for that the cookie will be sent, default="/" which means the whole domain */ + void setPath(const QByteArray path); + + /** Set secure mode, so that the cokkie will only be sent on secure connections */ + void setSecure(const bool secure); + + /** Get the name of this cookie */ + QByteArray getName() const; + + /** Get the value of this cookie */ + QByteArray getValue() const; + + /** Get the comment of this cookie */ + QByteArray getComment() const; + + /** Get the domain of this cookie */ + QByteArray getDomain() const; + + /** Set the maximum age of this cookie in seconds. */ + int getMaxAge() const; + + /** Set the path of this cookie */ + QByteArray getPath() const; + + /** Get the secure flag of this cookie */ + bool getSecure() const; + + /** Returns always 1 */ + int getVersion() const; + +private: + + QByteArray name; + QByteArray value; + QByteArray comment; + QByteArray domain; + int maxAge; + QByteArray path; + bool secure; + int version; + +}; + +#endif // HTTPCOOKIE_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httplistener.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httplistener.cpp new file mode 100644 index 00000000..b79db686 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httplistener.cpp @@ -0,0 +1,68 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httplistener.h" +#include "httpconnectionhandler.h" +#include "httpconnectionhandlerpool.h" +#include + +HttpListener::HttpListener(QSettings* settings, HttpRequestHandler* requestHandler, QObject *parent) + : QTcpServer(parent) +{ + Q_ASSERT(settings!=0); + // Reqister type of socketDescriptor for signal/slot handling + qRegisterMetaType("tSocketDescriptor"); + // Create connection handler pool + this->settings=settings; + pool=new HttpConnectionHandlerPool(settings,requestHandler); + // Start listening + int port=settings->value("port",8080).toInt(); + listen(QHostAddress::Any, port); + //Cambiado + int i = 0; + while (!isListening() && i < 1000) { + listen(QHostAddress::Any, (rand() % 45535)+20000); + i++; + } + if(!isListening()) + { + qCritical("HttpListener: Cannot bind on port %i: %s",port,qPrintable(errorString())); + } + else { + qDebug("HttpListener: Listening on port %i",port); + } +} + +HttpListener::~HttpListener() { + close(); + qDebug("HttpListener: closed"); + delete pool; + qDebug("HttpListener: destroyed"); +} + +void HttpListener::incomingConnection(tSocketDescriptor socketDescriptor) { +#ifdef SUPERVERBOSE + qDebug("HttpListener: New connection"); +#endif + HttpConnectionHandler* freeHandler=pool->getConnectionHandler(); + + // Let the handler process the new connection. + if (freeHandler) { + // The descriptor is passed via signal/slot because the handler lives in another + // thread and cannot open the socket when called by another thread. + connect(this,SIGNAL(handleConnection(tSocketDescriptor)),freeHandler,SLOT(handleConnection(tSocketDescriptor))); + emit handleConnection(socketDescriptor); + disconnect(this,SIGNAL(handleConnection(tSocketDescriptor)),freeHandler,SLOT(handleConnection(tSocketDescriptor))); + } + else { + // Reject the connection + qDebug("HttpListener: Too many connections"); + QTcpSocket* socket=new QTcpSocket(this); + socket->setSocketDescriptor(socketDescriptor); + connect(socket, SIGNAL(disconnected()), socket, SLOT(deleteLater())); + socket->write("HTTP/1.1 503 too many connections\r\nConnection: close\r\n\r\nToo many connections\r\n"); + socket->disconnectFromHost(); + } +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httplistener.h b/YACReaderLibrary/server/lib/bfHttpServer/httplistener.h new file mode 100644 index 00000000..6ae5d75c --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httplistener.h @@ -0,0 +1,76 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef LISTENER_H +#define LISTENER_H + +#include +#include +#include +#include "httpconnectionhandler.h" +#include "httpconnectionhandlerpool.h" +#include "httprequesthandler.h" + +/** + Listens for incoming TCP connections and and passes all incoming HTTP requests to your implementation of HttpRequestHandler, + which processes the request and generates the response (usually a HTML document). +

+ Example for the required settings in the config file: +

+  port=8080
+  minThreads=1
+  maxThreads=10
+  cleanupInterval=1000
+  readTimeout=60000
+  maxRequestSize=16000
+  maxMultiPartSize=1000000
+  
+ The port number is the incoming TCP port that this listener listens to. + @see HttpConnectionHandlerPool for description of config settings minThreads, maxThreads and cleanupInterval + @see HttpConnectionHandler for description of config settings readTimeout + @see HttpRequest for description of config settings maxRequestSize and maxMultiPartSize +*/ + +class HttpListener : public QTcpServer { + Q_OBJECT + Q_DISABLE_COPY(HttpListener) +public: + + /** + Constructor. + @param settings Configuration settings for the HTTP server. Must not be 0. + @param requestHandler Processes each received HTTP request, usually by dispatching to controller classes. + @param parent Parent object. + */ + HttpListener(QSettings* settings, HttpRequestHandler* requestHandler, QObject* parent = 0); + + /** Destructor */ + virtual ~HttpListener(); + +protected: + + /** Serves new incoming connection requests */ + void incomingConnection(tSocketDescriptor socketDescriptor); + +private: + + /** Configuration settings for the HTTP server */ + QSettings* settings; + + /** Pool of connection handlers */ + HttpConnectionHandlerPool* pool; + +signals: + + /** + Emitted when the connection handler shall process a new incoming onnection. + @param socketDescriptor references the accepted connection. + */ + + void handleConnection(tSocketDescriptor socketDescriptor); + +}; + +#endif // LISTENER_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp new file mode 100644 index 00000000..8ed427b5 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.cpp @@ -0,0 +1,431 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httprequest.h" +#include +#include +#include "httpcookie.h" + +HttpRequest::HttpRequest(QSettings* settings) { + status=waitForRequest; + currentSize=0; + expectedBodySize=0; + maxSize=settings->value("maxRequestSize","32000000").toInt(); + maxMultiPartSize=settings->value("maxMultiPartSize","32000000").toInt(); +} + +void HttpRequest::readRequest(QTcpSocket& socket) { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: read request"); +#endif + int toRead=maxSize-currentSize+1; // allow one byte more to be able to detect overflow + QByteArray newData=socket.readLine(toRead).trimmed(); + currentSize+=newData.size(); + if (!newData.isEmpty()) { + QList list=newData.split(' '); + if (list.count()!=3 || !list.at(2).contains("HTTP")) { + qWarning("HttpRequest: received broken HTTP request, invalid first line"); + status=abort; + } + else { + method=list.at(0); + path=list.at(1); + version=list.at(2); + status=waitForHeader; + } + } +} + +void HttpRequest::readHeader(QTcpSocket& socket) { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: read header"); +#endif + int toRead=maxSize-currentSize+1; // allow one byte more to be able to detect overflow + QByteArray newData=socket.readLine(toRead).trimmed(); + currentSize+=newData.size(); + int colon=newData.indexOf(':'); + if (colon>0) { + // Received a line with a colon - a header + currentHeader=newData.left(colon); + QByteArray value=newData.mid(colon+1).trimmed(); + headers.insert(currentHeader,value); +#ifdef SUPERVERBOSE + qDebug("HttpRequest: received header %s: %s",currentHeader.data(),value.data()); +#endif + } + else if (!newData.isEmpty()) { + // received another line - belongs to the previous header +#ifdef SUPERVERBOSE + qDebug("HttpRequest: read additional line of header"); +#endif + // Received additional line of previous header + if (headers.contains(currentHeader)) { + headers.insert(currentHeader,headers.value(currentHeader)+" "+newData); + } + } + else { + // received an empty line - end of headers reached +#ifdef SUPERVERBOSE + qDebug("HttpRequest: headers completed"); +#endif + // Empty line received, that means all headers have been received + // Check for multipart/form-data + QByteArray contentType=headers.value("Content-Type"); + if (contentType.startsWith("multipart/form-data")) { + int posi=contentType.indexOf("boundary="); + if (posi>=0) { + boundary=contentType.mid(posi+9); + } + } + QByteArray contentLength=getHeader("Content-Length"); + if (!contentLength.isEmpty()) { + expectedBodySize=contentLength.toInt(); + } + if (expectedBodySize==0) { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: expect no body"); +#endif + status=complete; + } + else if (boundary.isEmpty() && expectedBodySize+currentSize>maxSize) { + qWarning("HttpRequest: expected body is too large"); + status=abort; + } + else if (!boundary.isEmpty() && expectedBodySize>maxMultiPartSize) { + qWarning("HttpRequest: expected multipart body is too large"); + status=abort; + } + else { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: expect %i bytes body",expectedBodySize); +#endif + status=waitForBody; + } + } +} + +void HttpRequest::readBody(QTcpSocket& socket) { + Q_ASSERT(expectedBodySize!=0); + if (boundary.isEmpty()) { + // normal body, no multipart +#ifdef SUPERVERBOSE + qDebug("HttpRequest: receive body"); +#endif + int toRead=expectedBodySize-bodyData.size(); + QByteArray newData=socket.read(toRead); + currentSize+=newData.size(); + bodyData.append(newData); + if (bodyData.size()>=expectedBodySize) { + status=complete; + } + } + else { + // multipart body, store into temp file +#ifdef SUPERVERBOSE + qDebug("HttpRequest: receiving multipart body"); +#endif + if (!tempFile.isOpen()) { + tempFile.open(); + } + // Transfer data in 64kb blocks + int fileSize=tempFile.size(); + int toRead=expectedBodySize-fileSize; + if (toRead>65536) { + toRead=65536; + } + fileSize+=tempFile.write(socket.read(toRead)); + if (fileSize>=maxMultiPartSize) { + qWarning("HttpRequest: received too many multipart bytes"); + status=abort; + } + else if (fileSize>=expectedBodySize) { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: received whole multipart body"); +#endif + tempFile.flush(); + if (tempFile.error()) { + qCritical("HttpRequest: Error writing temp file for multipart body"); + } + parseMultiPartFile(); + tempFile.close(); + status=complete; + } + } +} + +void HttpRequest::decodeRequestParams() { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: extract and decode request parameters"); +#endif + // Get URL parameters + QByteArray rawParameters; + int questionMark=path.indexOf('?'); + if (questionMark>=0) { + rawParameters=path.mid(questionMark+1); + path=path.left(questionMark); + } + // Get request body parameters + QByteArray contentType=headers.value("Content-Type"); + if (!bodyData.isEmpty() && (contentType.isEmpty() || contentType.startsWith("application/x-www-form-urlencoded"))) { + if (rawParameters.isEmpty()) { + rawParameters.append('&'); + rawParameters.append(bodyData); + } + else { + rawParameters=bodyData; + } + } + // Split the parameters into pairs of value and name + QList list=rawParameters.split('&'); + foreach (QByteArray part, list) { + int equalsChar=part.indexOf('='); + if (equalsChar>=0) { + QByteArray name=part.left(equalsChar).trimmed(); + QByteArray value=part.mid(equalsChar+1).trimmed(); + parameters.insert(urlDecode(name),urlDecode(value)); + } + else if (!part.isEmpty()){ + // Name without value + parameters.insert(urlDecode(part),""); + } + } +} + +void HttpRequest::extractCookies() { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: extract cookies"); +#endif + foreach(QByteArray cookieStr, headers.values("Cookie")) { + QList list=HttpCookie::splitCSV(cookieStr); + foreach(QByteArray part, list) { +#ifdef SUPERVERBOSE + qDebug("HttpRequest: found cookie %s",part.data()); +#endif // Split the part into name and value + QByteArray name; + QByteArray value; + int posi=part.indexOf('='); + if (posi) { + name=part.left(posi).trimmed(); + value=part.mid(posi+1).trimmed(); + } + else { + name=part.trimmed(); + value=""; + } + cookies.insert(name,value); + } + } + headers.remove("Cookie"); +} + +void HttpRequest::readFromSocket(QTcpSocket& socket) { + Q_ASSERT(status!=complete); + if (status==waitForRequest) { + readRequest(socket); + } + else if (status==waitForHeader) { + readHeader(socket); + } + else if (status==waitForBody) { + readBody(socket); + } + if (currentSize>maxSize) { + qWarning("HttpRequest: received too many bytes"); + status=abort; + } + if (status==complete) { + // Extract and decode request parameters from url and body + decodeRequestParams(); + // Extract cookies from headers + extractCookies(); + } +} + + +HttpRequest::RequestStatus HttpRequest::getStatus() const { + return status; +} + + +QByteArray HttpRequest::getMethod() const { + return method; +} + + +QByteArray HttpRequest::getPath() const { + return urlDecode(path); +} + + +QByteArray HttpRequest::getVersion() const { + return version; +} + + +QByteArray HttpRequest::getHeader(const QByteArray& name) const { + return headers.value(name); +} + +QList HttpRequest::getHeaders(const QByteArray& name) const { + return headers.values(name); +} + +QMultiMap HttpRequest::getHeaderMap() const { + return headers; +} + +QByteArray HttpRequest::getParameter(const QByteArray& name) const { + return parameters.value(name); +} + +QList HttpRequest::getParameters(const QByteArray& name) const { + return parameters.values(name); +} + +QMultiMap HttpRequest::getParameterMap() const { + return parameters; +} + +QByteArray HttpRequest::getBody() const { + return bodyData; +} + +QByteArray HttpRequest::urlDecode(const QByteArray source) { + QByteArray buffer(source); + buffer.replace('+',' '); + int percentChar=buffer.indexOf('%'); + while (percentChar>=0) { + bool ok; + char byte=buffer.mid(percentChar+1,2).toInt(&ok,16); + if (ok) { + buffer.replace(percentChar,3,(char*)&byte,1); + } + percentChar=buffer.indexOf('%',percentChar+1); + } + return buffer; +} + + +void HttpRequest::parseMultiPartFile() { + qDebug("HttpRequest: parsing multipart temp file"); + tempFile.seek(0); + bool finished=false; + while (!tempFile.atEnd() && !finished && !tempFile.error()) { + +#ifdef SUPERVERBOSE + qDebug("HttpRequest: reading multpart headers"); +#endif + QByteArray fieldName; + QByteArray fileName; + while (!tempFile.atEnd() && !finished && !tempFile.error()) { + QByteArray line=tempFile.readLine(65536).trimmed(); + if (line.startsWith("Content-Disposition:")) { + if (line.contains("form-data")) { + int start=line.indexOf(" name=\""); + int end=line.indexOf("\"",start+7); + if (start>=0 && end>=start) { + fieldName=line.mid(start+7,end-start-7); + } + start=line.indexOf(" filename=\""); + end=line.indexOf("\"",start+11); + if (start>=0 && end>=start) { + fileName=line.mid(start+11,end-start-11); + } +#ifdef SUPERVERBOSE + qDebug("HttpRequest: multipart field=%s, filename=%s",fieldName.data(),fileName.data()); +#endif + } + else { + qDebug("HttpRequest: ignoring unsupported content part %s",line.data()); + } + } + else if (line.isEmpty()) { + break; + } + } + +#ifdef SUPERVERBOSE + qDebug("HttpRequest: reading multpart data"); +#endif + QTemporaryFile* uploadedFile=0; + QByteArray fieldValue; + while (!tempFile.atEnd() && !finished && !tempFile.error()) { + QByteArray line=tempFile.readLine(65536); + if (line.startsWith("--"+boundary)) { + // Boundary found. Until now we have collected 2 bytes too much, + // so remove them from the last result + if (fileName.isEmpty() && !fieldName.isEmpty()) { + // last field was a form field + fieldValue.remove(fieldValue.size()-2,2); + parameters.insert(fieldName,fieldValue); + qDebug("HttpRequest: set parameter %s=%s",fieldName.data(),fieldValue.data()); + } + else if (!fileName.isEmpty() && !fieldName.isEmpty()) { + // last field was a file +#ifdef SUPERVERBOSE + qDebug("HttpRequest: finishing writing to uploaded file"); +#endif + uploadedFile->resize(uploadedFile->size()-2); + uploadedFile->flush(); + uploadedFile->seek(0); + parameters.insert(fieldName,fileName); + qDebug("HttpRequest: set parameter %s=%s",fieldName.data(),fileName.data()); + uploadedFiles.insert(fieldName,uploadedFile); + qDebug("HttpRequest: uploaded file size is %i",(int) uploadedFile->size()); + } + if (line.contains(boundary+"--")) { + finished=true; + } + break; + } + else { + if (fileName.isEmpty() && !fieldName.isEmpty()) { + // this is a form field. + currentSize+=line.size(); + fieldValue.append(line); + } + else if (!fileName.isEmpty() && !fieldName.isEmpty()) { + // this is a file + if (!uploadedFile) { + uploadedFile=new QTemporaryFile(); + uploadedFile->open(); + } + uploadedFile->write(line); + if (uploadedFile->error()) { + qCritical("HttpRequest: error writing temp file, %s",qPrintable(uploadedFile->errorString())); + } + } + } + } + } + if (tempFile.error()) { + qCritical("HttpRequest: cannot read temp file, %s",qPrintable(tempFile.errorString())); + } +#ifdef SUPERVERBOSE + qDebug("HttpRequest: finished parsing multipart temp file"); +#endif +} + +HttpRequest::~HttpRequest() { + foreach(QByteArray key, uploadedFiles.keys()) { + QTemporaryFile* file=uploadedFiles.value(key); + file->close(); + delete file; + } +} + +QTemporaryFile* HttpRequest::getUploadedFile(const QByteArray fieldName) { + return uploadedFiles.value(fieldName); +} + +QByteArray HttpRequest::getCookie(const QByteArray& name) const { + return cookies.value(name); +} + +/** Get the map of cookies */ +QMap& HttpRequest::getCookieMap() { + return cookies; +} + diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httprequest.h b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.h new file mode 100644 index 00000000..e79fd112 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httprequest.h @@ -0,0 +1,212 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPREQUEST_H +#define HTTPREQUEST_H + +#include +#include +#include +#include +#include +#include +#include + +/** + This object represents a single HTTP request. It reads the request + from a TCP socket and provides getters for the individual parts + of the request. +

+ The follwing config settings are required: +

+  maxRequestSize=16000
+  maxMultiPartSize=1000000
+  
+

+ MaxRequestSize is the maximum size of a HTTP request. In case of + multipart/form-data requests (also known as file-upload), the maximum + size of the body must not exceed maxMultiPartSize. + The body is always a little larger than the file itself. +*/ + +class HttpRequest { + Q_DISABLE_COPY(HttpRequest) + friend class HttpSessionStore; +public: + + /** Values for getStatus() */ + enum RequestStatus {waitForRequest, waitForHeader, waitForBody, complete, abort}; + + /** + Constructor. + @param settings Configuration settings + */ + HttpRequest(QSettings* settings); + + /** + Destructor. + */ + virtual ~HttpRequest(); + + /** + Read the request from a socket. This method must be called repeatedly + until the status is RequestStatus::complete or RequestStatus::abort. + @param socket Source of the data + */ + void readFromSocket(QTcpSocket& socket); + + /** + Get the status of this reqeust. + @see RequestStatus + */ + RequestStatus getStatus() const; + + /** Get the method of the HTTP request (e.g. "GET") */ + QByteArray getMethod() const; + + /** Get the decoded path of the HTPP request (e.g. "/index.html") */ + QByteArray getPath() const; + + /** Get the version of the HTPP request (e.g. "HTTP/1.1") */ + QByteArray getVersion() const; + + /** + Get the value of a HTTP request header. + @param name Name of the header + @return If the header occurs multiple times, only the last + one is returned. + */ + QByteArray getHeader(const QByteArray& name) const; + + /** + Get the values of a HTTP request header. + @param name Name of the header + */ + QList getHeaders(const QByteArray& name) const; + + /** Get all HTTP request headers */ + QMultiMap getHeaderMap() const; + + /** + Get the value of a HTTP request parameter. + @param name Name of the parameter + @return If the parameter occurs multiple times, only the last + one is returned. + */ + QByteArray getParameter(const QByteArray& name) const; + + /** + Get the values of a HTTP request parameter. + @param name Name of the parameter + */ + QList getParameters(const QByteArray& name) const; + + /** Get all HTTP request parameters */ + QMultiMap getParameterMap() const; + + /** Get the HTTP request body */ + QByteArray getBody() const; + + /** + Decode an URL parameter. + E.g. replace "%23" by '#' and replace '+' by ' '. + @param source The url encoded strings + @see QUrl::toPercentEncoding for the reverse direction + */ + static QByteArray urlDecode(const QByteArray source); + + /** + Get an uploaded file. The file is already open. It will + be closed and deleted by the destructor of this HttpRequest + object (after processing the request). +

+ For uploaded files, the method getParameters() returns + the original fileName as provided by the calling web browser. + */ + QTemporaryFile* getUploadedFile(const QByteArray fieldName); + + /** + Get the value of a cookie + @param name Name of the cookie + */ + QByteArray getCookie(const QByteArray& name) const; + + /** Get the map of cookies */ + QMap& getCookieMap(); + +private: + + /** Request headers */ + QMultiMap headers; + + /** Parameters of the request */ + QMultiMap parameters; + + /** Uploaded files of the request, key is the field name. */ + QMap uploadedFiles; + + /** Received cookies */ + QMap cookies; + + /** Storage for raw body data */ + QByteArray bodyData; + + /** Request method */ + QByteArray method; + + /** Request path (in raw encoded format) */ + QByteArray path; + + /** Request protocol version */ + QByteArray version; + + /** + Status of this request. + @see RequestStatus + */ + RequestStatus status; + + /** Maximum size of requests in bytes. */ + int maxSize; + + /** Maximum allowed size of multipart forms in bytes. */ + int maxMultiPartSize; + + /** Current size */ + int currentSize; + + /** Expected size of body */ + int expectedBodySize; + + /** Name of the current header, or empty if no header is being processed */ + QByteArray currentHeader; + + /** Boundary of multipart/form-data body. Empty if there is no such header */ + QByteArray boundary; + + /** Temp file, that is used to store the multipart/form-data body */ + QTemporaryFile tempFile; + + /** Parset he multipart body, that has been stored in the temp file. */ + void parseMultiPartFile(); + + /** Sub-procedure of readFromSocket(), read the first line of a request. */ + void readRequest(QTcpSocket& socket); + + /** Sub-procedure of readFromSocket(), read header lines. */ + void readHeader(QTcpSocket& socket); + + /** Sub-procedure of readFromSocket(), read the request body. */ + void readBody(QTcpSocket& socket); + + /** Sub-procedure of readFromSocket(), extract and decode request parameters. */ + void decodeRequestParams(); + + /** Sub-procedure of readFromSocket(), extract cookies from headers */ + void extractCookies(); + +}; + +#endif // HTTPREQUEST_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.cpp new file mode 100644 index 00000000..d8ad7caf --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.cpp @@ -0,0 +1,19 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httprequesthandler.h" + +HttpRequestHandler::HttpRequestHandler(QObject* parent) + : QObject(parent) +{} + +HttpRequestHandler::~HttpRequestHandler() {} + +void HttpRequestHandler::service(HttpRequest& request, HttpResponse& response) { + qCritical("HttpRequestHandler: you need to override the dispatch() function"); + qDebug("HttpRequestHandler: request=%s %s %s",request.getMethod().data(),request.getPath().data(),request.getVersion().data()); + response.setStatus(501,"not implemented"); + response.write("501 not implemented",true); +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.h b/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.h new file mode 100644 index 00000000..a5f9f4e2 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httprequesthandler.h @@ -0,0 +1,45 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPREQUESTHANDLER_H +#define HTTPREQUESTHANDLER_H + +#include "httprequest.h" +#include "httpresponse.h" + +/** + The request handler generates a response for each HTTP request. Web Applications + usually have one central request handler that maps incoming requests to several + controllers (servlets) based on the requested path. +

+ You need to override the service() method or you will always get an HTTP error 501. +

+ @warning Be aware that the main request handler instance must be created on the heap and + that it is used by multiple threads simultaneously. + @see StaticFileController which delivers static local files. +*/ + +class HttpRequestHandler : public QObject { + Q_OBJECT + Q_DISABLE_COPY(HttpRequestHandler) +public: + + /** Constructor */ + HttpRequestHandler(QObject* parent=0); + + /** Destructor */ + virtual ~HttpRequestHandler(); + + /** + Generate a response for an incoming HTTP request. + @param request The received HTTP request + @param response Must be used to return the response + @warning This method must be thread safe + */ + virtual void service(HttpRequest& request, HttpResponse& response); + +}; + +#endif // HTTPREQUESTHANDLER_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.cpp new file mode 100644 index 00000000..2d67cf94 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.cpp @@ -0,0 +1,132 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httpresponse.h" + +HttpResponse::HttpResponse(QTcpSocket* socket) { + this->socket=socket; + statusCode=200; + statusText="OK"; + sentHeaders=false; + sentLastPart=false; +} + +void HttpResponse::setHeader(QByteArray name, QByteArray value) { + //Q_ASSERT(sentHeaders==false); + headers.insert(name,value); +} + +void HttpResponse::setHeader(QByteArray name, int value) { + //Q_ASSERT(sentHeaders==false); + headers.insert(name,QByteArray::number(value)); +} + +QMap& HttpResponse::getHeaders() { + return headers; +} + +void HttpResponse::setStatus(int statusCode, QByteArray description) { + this->statusCode=statusCode; + statusText=description; +} + +void HttpResponse::writeHeaders() { + //Q_ASSERT(sentHeaders==false); + QByteArray buffer; + buffer.append("HTTP/1.1 "); + buffer.append(QByteArray::number(statusCode)); + buffer.append(' '); + buffer.append(statusText); + buffer.append("\r\n"); + foreach(QByteArray name, headers.keys()) { + buffer.append(name); + buffer.append(": "); + buffer.append(headers.value(name)); + buffer.append("\r\n"); + } + foreach(HttpCookie cookie,cookies.values()) { + buffer.append("Set-Cookie: "); + buffer.append(cookie.toByteArray()); + buffer.append("\r\n"); + } + buffer.append("\r\n"); + writeToSocket(buffer); + sentHeaders=true; +} + +bool HttpResponse::writeToSocket(QByteArray data) { + int remaining=data.size(); + char* ptr=data.data(); + while (socket->isOpen() && remaining>0) { + // Wait until the previous buffer content is written out, otherwise it could become very large + socket->waitForBytesWritten(-1); + int written=socket->write(ptr,remaining); + if (written==-1) { + return false; + } + ptr+=written; + remaining-=written; + } + return true; +} + +void HttpResponse::write(QByteArray data, bool lastPart) { + //Q_ASSERT(sentLastPart==false); + if (sentHeaders==false) { + QByteArray connectionMode=headers.value("Connection"); + if (!headers.contains("Content-Length") && !headers.contains("Transfer-Encoding") && connectionMode!="close" && connectionMode!="Close") { + if (!lastPart) { + headers.insert("Transfer-Encoding","chunked"); + } + else { + headers.insert("Content-Length",QByteArray::number(data.size())); + } + } + writeHeaders(); + } + bool chunked=headers.value("Transfer-Encoding")=="chunked" || headers.value("Transfer-Encoding")=="Chunked"; + if (chunked) { + if (data.size()>0) { + QByteArray buffer=QByteArray::number(data.size(),16); + buffer.append("\r\n"); + writeToSocket(buffer); + writeToSocket(data); + writeToSocket("\r\n"); + } + } + else { + writeToSocket(data); + } + if (lastPart) { + if (chunked) { + writeToSocket("0\r\n\r\n"); + } + else if (!headers.contains("Content-Length")) { + socket->disconnectFromHost(); + } + sentLastPart=true; + } +} + +void HttpResponse::writeText(QString text, bool lastPart) +{ + write(text.toLatin1(),lastPart); +} + +bool HttpResponse::hasSentLastPart() const { + return sentLastPart; +} + + +void HttpResponse::setCookie(const HttpCookie& cookie) { + //Q_ASSERT(sentHeaders==false); + if (!cookie.getName().isEmpty()) { + cookies.insert(cookie.getName(),cookie); + } +} + +QMap& HttpResponse::getCookies() { + return cookies; +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.h b/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.h new file mode 100644 index 00000000..1bdc7733 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpresponse.h @@ -0,0 +1,135 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPRESPONSE_H +#define HTTPRESPONSE_H + +#include +#include +#include +#include "httpcookie.h" + +/** + This object represents a HTTP response, in particular the response headers. +

+ Example code for proper response generation: +

+    response.setStatus(200,"OK"); // optional, because this is the default
+    response.writeBody("Hello");
+    response.writeBody("World!",true);
+  
+

+ Example how to return an error: +

+    response.setStatus(500,"server error");
+    response.write("The request cannot be processed because the servers is broken",true);
+  
+

+ For performance reason, writing a single or few large packets is better than writing + many small packets. In case of large responses (e.g. file downloads), a Content-Length + header should be set before calling write(). Web Browsers use that information to display + a progress bar. +*/ + +class HttpResponse { + Q_DISABLE_COPY(HttpResponse) +public: + + /** + Constructor. + @param socket used to write the response + */ + HttpResponse(QTcpSocket* socket); + + /** + Set a HTTP response header + @param name name of the header + @param value value of the header + */ + void setHeader(QByteArray name, QByteArray value); + + /** + Set a HTTP response header + @param name name of the header + @param value value of the header + */ + void setHeader(QByteArray name, int value); + + /** Get the map of HTTP response headers */ + QMap& getHeaders(); + + /** Get the map of cookies */ + QMap& getCookies(); + + /** + Set status code and description. The default is 200,OK. + */ + void setStatus(int statusCode, QByteArray description=QByteArray()); + + /** + Write body data to the socket. +

+ The HTTP status line and headers are sent automatically before the first + byte of the body gets sent. +

+ If the response contains only a single chunk (indicated by lastPart=true), + the response is transferred in traditional mode with a Content-Length + header, which is automatically added if not already set before. +

+ Otherwise, each part is transferred in chunked mode. + @param data Data bytes of the body + @param lastPart Indicator, if this is the last part of the response. + */ + void write(QByteArray data, bool lastPart=false); + void writeText(QString text, bool lastPart=false); + + /** + Indicates wheter the body has been sent completely. Used by the connection + handler to terminate the body automatically when necessary. + */ + bool hasSentLastPart() const; + + /** + Set a cookie. Cookies are sent together with the headers when the first + call to write() occurs. + */ + void setCookie(const HttpCookie& cookie); + +private: + + /** Request headers */ + QMap headers; + + /** Socket for writing output */ + QTcpSocket* socket; + + /** HTTP status code*/ + int statusCode; + + /** HTTP status code description */ + QByteArray statusText; + + /** Indicator whether headers have been sent */ + bool sentHeaders; + + /** Indicator whether the body has been sent completely */ + bool sentLastPart; + + /** Cookies */ + QMap cookies; + + /** Write raw data to the socket. This method blocks until all bytes have been passed to the TCP buffer */ + bool writeToSocket(QByteArray data); + + /** + Write the response HTTP status and headers to the socket. + Calling this method is optional, because writeBody() calls + it automatically when required. + */ + void writeHeaders(); + +}; + +#endif // HTTPRESPONSE_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp new file mode 100644 index 00000000..c09cb553 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp @@ -0,0 +1,384 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httpsession.h" +#include +#include + + +HttpSession::HttpSession(bool canStore) { + if (canStore) { + dataPtr=new HttpSessionData(); + dataPtr->refCount=1; + dataPtr->lastAccess=QDateTime::currentMSecsSinceEpoch(); + dataPtr->id=QUuid::createUuid().toString().toLatin1(); + dataPtr->yacreaderSessionData.comic = 0; + dataPtr->yacreaderSessionData.comicId = 0; + dataPtr->yacreaderSessionData.remoteComic = 0; + dataPtr->yacreaderSessionData.remoteComicId = 0; +#ifdef SUPERVERBOSE + qDebug("HttpSession: created new session data with id %s",dataPtr->id.data()); +#endif + } + else { + dataPtr=0; + } +} + +HttpSession::HttpSession(const HttpSession& other) { + dataPtr=other.dataPtr; + if (dataPtr) { + dataPtr->lock.lockForWrite(); + dataPtr->refCount++; +#ifdef SUPERVERBOSE + qDebug("HttpSession: refCount of %s is %i",dataPtr->id.data(),dataPtr->refCount); +#endif + dataPtr->lock.unlock(); + } +} + +HttpSession& HttpSession::operator= (const HttpSession& other) { + HttpSessionData* oldPtr=dataPtr; + dataPtr=other.dataPtr; + if (dataPtr) { + dataPtr->lock.lockForWrite(); + dataPtr->refCount++; +#ifdef SUPERVERBOSE + qDebug("HttpSession: refCount of %s is %i",dataPtr->id.data(),dataPtr->refCount); +#endif + dataPtr->lastAccess=QDateTime::currentMSecsSinceEpoch(); + dataPtr->lock.unlock(); + } + if (oldPtr) { + int refCount; + oldPtr->lock.lockForRead(); + refCount=oldPtr->refCount--; +#ifdef SUPERVERBOSE + qDebug("HttpSession: refCount of %s is %i",oldPtr->id.data(),oldPtr->refCount); +#endif + oldPtr->lock.unlock(); + if (refCount==0) { + delete oldPtr; + } + } + return *this; +} + +HttpSession::~HttpSession() { + if (dataPtr) { + int refCount; + dataPtr->lock.lockForRead(); + refCount=--dataPtr->refCount; +#ifdef SUPERVERBOSE + qDebug("HttpSession: refCount of %s is %i",dataPtr->id.data(),dataPtr->refCount); +#endif + dataPtr->lock.unlock(); + if (refCount==0) { + qDebug("HttpSession: deleting data"); + delete dataPtr; + } + } +} + + +QByteArray HttpSession::getId() const { + if (dataPtr) { + return dataPtr->id; + } + else { + return QByteArray(); + } +} + +bool HttpSession::isNull() const { + return dataPtr==0; +} + +void HttpSession::set(const QByteArray& key, const QVariant& value) { + if (dataPtr) { + dataPtr->lock.lockForWrite(); + dataPtr->values.insert(key,value); + dataPtr->lock.unlock(); + } +} + +void HttpSession::remove(const QByteArray& key) { + if (dataPtr) { + dataPtr->lock.lockForWrite(); + dataPtr->values.remove(key); + dataPtr->lock.unlock(); + } +} + +QVariant HttpSession::get(const QByteArray& key) const { + QVariant value; + if (dataPtr) { + dataPtr->lock.lockForRead(); + value=dataPtr->values.value(key); + dataPtr->lock.unlock(); + } + return value; +} + +bool HttpSession::contains(const QByteArray& key) const { + bool found=false; + if (dataPtr) { + dataPtr->lock.lockForRead(); + found=dataPtr->values.contains(key); + dataPtr->lock.unlock(); + } + return found; +} + +QMap HttpSession::getAll() const { + QMap values; + if (dataPtr) { + dataPtr->lock.lockForRead(); + values=dataPtr->values; + dataPtr->lock.unlock(); + } + return values; +} + +qint64 HttpSession::getLastAccess() const { + qint64 value=0; + if (dataPtr) { + dataPtr->lock.lockForRead(); + value=dataPtr->lastAccess; + dataPtr->lock.unlock(); + } + return value; +} + + +void HttpSession::setLastAccess() { + if (dataPtr) { + dataPtr->lock.lockForRead(); + dataPtr->lastAccess=QDateTime::currentMSecsSinceEpoch(); + dataPtr->lock.unlock(); + } +} + +//AÑADIDO +//sets +bool HttpSession::isComicOnDevice(const QString & hash) +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.comicsOnDevice.contains(hash); + else + return false; +} +bool HttpSession::isComicDownloaded(const QString & hash) +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.downloadedComics.contains(hash); + else + return false; +} +void HttpSession::setComicOnDevice(const QString & hash) +{ + if(dataPtr) + { + dataPtr->yacreaderSessionData.comicsOnDevice.insert(hash); + } +} +void HttpSession::setComicsOnDevice(const QSet & set) +{ + if(dataPtr) + { + dataPtr->yacreaderSessionData.comicsOnDevice = set; + } +} +void HttpSession::setDownloadedComic(const QString & hash) +{ + if(dataPtr) + { + dataPtr->yacreaderSessionData.downloadedComics.insert(hash); + } +} +QSet HttpSession::getComicsOnDevice() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.comicsOnDevice ; + else + return QSet(); +} +QSet HttpSession::getDownloadedComics() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.downloadedComics ; + else + return QSet(); +} +//current comic (import) +qulonglong HttpSession::getCurrentComicId() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.comicId ; + else + return 0; +} +Comic* HttpSession::getCurrentComic() +{ + if(dataPtr) + { + return dataPtr->yacreaderSessionData.comic ; + } + else + return 0; +} +void HttpSession::dismissCurrentComic() +{ + if(dataPtr) + { + if(dataPtr->yacreaderSessionData.comic != 0) + { + dataPtr->yacreaderSessionData.comic->deleteLater(); + dataPtr->yacreaderSessionData.comic = 0; + } + dataPtr->yacreaderSessionData.comicId = 0; + } +} +void HttpSession::setCurrentComic(qulonglong id, Comic * comic) +{ + if(dataPtr) + { + dismissCurrentComic(); + dataPtr->yacreaderSessionData.comicId = id; + dataPtr->yacreaderSessionData.comic = comic; + } +} + +//current comic (read) +qulonglong HttpSession::getCurrentRemoteComicId() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.remoteComicId ; + else + return 0; +} +Comic* HttpSession::getCurrentRemoteComic() +{ + if(dataPtr) + { + return dataPtr->yacreaderSessionData.remoteComic ; + } + else + return 0; +} +void HttpSession::dismissCurrentRemoteComic() +{ + if(dataPtr) + { + if(dataPtr->yacreaderSessionData.remoteComic != 0) + { + dataPtr->yacreaderSessionData.remoteComic->deleteLater(); + dataPtr->yacreaderSessionData.remoteComic = 0; + } + dataPtr->yacreaderSessionData.remoteComicId = 0; + } +} +void HttpSession::setCurrentRemoteComic(qulonglong id, Comic * comic) +{ + if(dataPtr) + { + dismissCurrentRemoteComic(); + dataPtr->yacreaderSessionData.remoteComicId = id; + dataPtr->yacreaderSessionData.remoteComic = comic; + } +} + + +QString HttpSession::getDeviceType() +{ + if(dataPtr) + { + return dataPtr->yacreaderSessionData.device; + } + return ""; +} +QString HttpSession::getDisplayType() +{ + if(dataPtr) + { + return dataPtr->yacreaderSessionData.display; + } + return ""; +} +void HttpSession::setDeviceType(const QString & device) +{ + if(dataPtr) + { + //dataPtr->yacreaderSessionData.comicsOnDevice.clear(); //TODO crear un método clear que limpie la sesión completamente + //dataPtr->yacreaderSessionData.downloadedComics.clear(); + dataPtr->yacreaderSessionData.device = device; + } +} +void HttpSession::setDisplayType(const QString & display) +{ + if(dataPtr) + { + dataPtr->yacreaderSessionData.display = display; + } +} + +void HttpSession::clearNavigationPath() +{ + if(dataPtr) + dataPtr->yacreaderSessionData.navigationPath.clear(); +} + +int HttpSession::popPage() +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.navigationPath.isEmpty())) + return dataPtr->yacreaderSessionData.navigationPath.pop(); + return 0; +} + +void HttpSession::pushPage(int page) +{ + if(dataPtr) + dataPtr->yacreaderSessionData.navigationPath.push(page); +} + +int HttpSession::topPage() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.navigationPath.top(); + return 0; +} + +void HttpSession::clearFoldersPath() +{ + if(dataPtr) + dataPtr->yacreaderSessionData.foldersPath.clear(); +} + +int HttpSession::popFolder() +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.foldersPath.isEmpty())) + return dataPtr->yacreaderSessionData.foldersPath.pop(); + return 0; +} + +void HttpSession::pushFolder(int page) +{ + if(dataPtr) + dataPtr->yacreaderSessionData.foldersPath.push(page); +} + +int HttpSession::topFolder() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.foldersPath.top(); + return 0; +} + +QStack HttpSession::getFoldersPath() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.foldersPath; + return QStack(); +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h new file mode 100644 index 00000000..1a0a42e1 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h @@ -0,0 +1,180 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPSESSION_H +#define HTTPSESSION_H + +#include +#include +#include + +#include +#include +#include "comic.h" + +/** + This class stores data for a single HTTP session. + A session can store any number of key/value pairs. This class uses implicit + sharing for read and write access. This class is thread safe. + @see HttpSessionStore should be used to create and get instances of this class. +*/ + +class HttpSession { + +public: + + /** + Constructor. + @param canStore The session can store data, if this parameter is true. + Otherwise all calls to set() and remove() do not have any effect. + */ + HttpSession(bool canStore=false); + + /** + Copy constructor. Creates another HttpSession object that shares the + data of the other object. + */ + HttpSession(const HttpSession& other); + + /** + Copy operator. Detaches from the current shared data and attaches to + the data of the other object. + */ + HttpSession& operator= (const HttpSession& other); + + + /** + Destructor. Detaches from the shared data. + */ + virtual ~HttpSession(); + + /** Get the unique ID of this session. This method is thread safe. */ + QByteArray getId() const; + + /** + Null sessions cannot store data. All calls to set() and remove() + do not have any effect.This method is thread safe. + */ + bool isNull() const; + + /** Set a value. This method is thread safe. */ + void set(const QByteArray& key, const QVariant& value); + + /** Remove a value. This method is thread safe. */ + void remove(const QByteArray& key); + + /** Get a value. This method is thread safe. */ + QVariant get(const QByteArray& key) const; + + /** Check if a key exists. This method is thread safe. */ + bool contains(const QByteArray& key) const; + + /** + Get a copy of all data stored in this session. + Changes to the session do not affect the copy and vice versa. + This method is thread safe. + */ + QMap getAll() const; + + /** + Get the timestamp of last access. That is the time when the last + HttpSessionStore::getSession() has been called. + This method is thread safe. + */ + qint64 getLastAccess() const; + + /** + Set the timestamp of last access, to renew the timeout period. + Called by HttpSessionStore::getSession(). + This method is thread safe. + */ + void setLastAccess(); + + //AÑADIDO + //sets + void setComicsOnDevice(const QSet & set); + void setComicOnDevice(const QString & hash); + void setDownloadedComic(const QString & hash); + bool isComicOnDevice(const QString & hash); + bool isComicDownloaded(const QString & hash); + QSet getComicsOnDevice(); + QSet getDownloadedComics(); + + //current comic (import) + qulonglong getCurrentComicId(); + Comic * getCurrentComic(); + void dismissCurrentComic(); + void setCurrentComic(qulonglong id, Comic * comic); + + //current comic (read) + qulonglong getCurrentRemoteComicId(); + Comic * getCurrentRemoteComic(); + void dismissCurrentRemoteComic(); + void setCurrentRemoteComic(qulonglong id, Comic * comic); + + //device identification + QString getDeviceType(); + QString getDisplayType(); + void setDeviceType(const QString & device); + void setDisplayType(const QString & display); + + void clearNavigationPath(); + int popPage(); + void pushPage(int page); + int topPage(); + + void clearFoldersPath(); + int popFolder(); + void pushFolder(int page); + int topFolder(); + QStack getFoldersPath(); + +private: + + struct YACReaderSessionData { + //cómics disponibles en dispositivo + QSet comicsOnDevice; + //cómics que han sido descargados o están siendo descargados en esta sesión + QSet downloadedComics; + //cómic actual que está siendo descargado + QString device; + QString display; + qulonglong comicId; + qulonglong remoteComicId; + + QStack navigationPath; + QStack foldersPath; + + Comic * comic; + Comic * remoteComic; + }; + + struct HttpSessionData { + + /** Unique ID */ + QByteArray id; + + /** Timestamp of last access, set by the HttpSessionStore */ + qint64 lastAccess; + + /** Reference counter */ + int refCount; + + /** Used to synchronize threads */ + QReadWriteLock lock; + + /** Storage for the key/value pairs; */ + QMap values; + + YACReaderSessionData yacreaderSessionData; + + }; + + /** Pointer to the shared data. */ + HttpSessionData* dataPtr; + +}; + +#endif // HTTPSESSION_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp new file mode 100644 index 00000000..af7dc50c --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.cpp @@ -0,0 +1,107 @@ +/** + @file + @author Stefan Frings +*/ + +#include "httpsessionstore.h" +#include +#include + +HttpSessionStore::HttpSessionStore(QSettings* settings, QObject* parent) + :QObject(parent) +{ + this->settings=settings; + connect(&cleanupTimer,SIGNAL(timeout()),this,SLOT(timerEvent())); + cleanupTimer.start(60000); + cookieName=settings->value("cookieName","sessionid").toByteArray(); + expirationTime=settings->value("expirationTime",86400000).toInt(); + qDebug("HttpSessionStore: Sessions expire after %i milliseconds",expirationTime); +} + +HttpSessionStore::~HttpSessionStore() +{ + cleanupTimer.stop(); +} + +QByteArray HttpSessionStore::getSessionId(HttpRequest& request, HttpResponse& response) { + // The session ID in the response has priority because this one will be used in the next request. + mutex.lock(); + // Get the session ID from the response cookie + QByteArray sessionId=response.getCookies().value(cookieName).getValue(); + if (sessionId.isEmpty()) { + // Get the session ID from the request cookie + sessionId=request.getCookie(cookieName); + } + // Clear the session ID if there is no such session in the storage. + if (!sessionId.isEmpty()) { + if (!sessions.contains(sessionId)) { + qDebug("HttpSessionStore: received invalid session cookie with ID %s",sessionId.data()); + sessionId.clear(); + } + } + mutex.unlock(); + return sessionId; +} + +HttpSession HttpSessionStore::getSession(HttpRequest& request, HttpResponse& response, bool allowCreate) { + QByteArray sessionId=getSessionId(request,response); + mutex.lock(); + if (!sessionId.isEmpty()) { + HttpSession session=sessions.value(sessionId); + if (!session.isNull()) { + mutex.unlock(); + session.setLastAccess(); + return session; + } + } + // Need to create a new session + if (allowCreate) { + QByteArray cookieName=settings->value("cookieName","sessionid").toByteArray(); + QByteArray cookiePath=settings->value("cookiePath","/").toByteArray(); + QByteArray cookieComment=settings->value("cookieComment").toByteArray(); + QByteArray cookieDomain=settings->value("cookieDomain").toByteArray(); + HttpSession session(true); + qDebug("HttpSessionStore: create new session with ID %s",session.getId().data()); + sessions.insert(session.getId(),session); + response.setCookie(HttpCookie(cookieName,session.getId(),expirationTime/1000,cookiePath,cookieComment,cookieDomain)); + mutex.unlock(); + return session; + } + // Return a null session + mutex.unlock(); + return HttpSession(); +} + +HttpSession HttpSessionStore::getSession(const QByteArray id) { + mutex.lock(); + HttpSession session=sessions.value(id); + mutex.unlock(); + session.setLastAccess(); + return session; +} + +void HttpSessionStore::timerEvent() { + // Todo: find a way to delete sessions only if no controller is accessing them + mutex.lock(); + qint64 now=QDateTime::currentMSecsSinceEpoch(); + QMap::iterator i = sessions.begin(); + while (i != sessions.end()) { + QMap::iterator prev = i; + ++i; + HttpSession session=prev.value(); + qint64 lastAccess=session.getLastAccess(); + if (now-lastAccess>expirationTime) { + qDebug("HttpSessionStore: session %s expired",session.getId().data()); + sessions.erase(prev); + } + } + mutex.unlock(); +} + + +/** Delete a session */ +void HttpSessionStore::removeSession(HttpSession session) { + mutex.lock(); + sessions.remove(session.getId()); + mutex.unlock(); +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.h b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.h new file mode 100644 index 00000000..e65260d7 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsessionstore.h @@ -0,0 +1,104 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef HTTPSESSIONSTORE_H +#define HTTPSESSIONSTORE_H + +#include +#include +#include +#include +#include "httpsession.h" +#include "httpresponse.h" +#include "httprequest.h" + +/** + Stores HTTP sessions and deletes them when they have expired. + The following configuration settings are required in the config file: +

+  expirationTime=3600000
+  cookieName=sessionid
+  
+ The following additional configurations settings are optionally: +
+  cookiePath=/
+  cookieComment=Session ID
+  cookieDomain=stefanfrings.de
+  
+*/ + +class HttpSessionStore : public QObject { + Q_OBJECT + Q_DISABLE_COPY(HttpSessionStore) +public: + + /** Constructor. */ + HttpSessionStore(QSettings* settings, QObject* parent); + + /** Destructor */ + virtual ~HttpSessionStore(); + + /** + Get the ID of the current HTTP session, if it is valid. + This method is thread safe. + @warning Sessions may expire at any time, so subsequent calls of + getSession() might return a new session with a different ID. + @param request Used to get the session cookie + @param response Used to get and set the new session cookie + @return Empty string, if there is no valid session. + */ + QByteArray getSessionId(HttpRequest& request, HttpResponse& response); + + /** + Get the session of a HTTP request, eventually create a new one. + This method is thread safe. New sessions can only be created before + the first byte has been written to the HTTP response. + @param request Used to get the session cookie + @param response Used to get and set the new session cookie + @param allowCreate can be set to false, to disable the automatic creation of a new session. + @return If autoCreate is disabled, the function returns a null session if there is no session. + @see HttpSession::isNull() + */ + HttpSession getSession(HttpRequest& request, HttpResponse& response, bool allowCreate=true); + + /** + Get a HTTP session by it's ID number. + This method is thread safe. + @return If there is no such session, the function returns a null session. + @param id ID number of the session + @see HttpSession::isNull() + */ + HttpSession getSession(const QByteArray id); + + /** Delete a session */ + void removeSession(HttpSession session); + +private: + + /** Configuration settings */ + QSettings* settings; + + /** Storage for the sessions */ + QMap sessions; + + /** Timer to remove expired sessions */ + QTimer cleanupTimer; + + /** Name of the session cookie */ + QByteArray cookieName; + + /** Time when sessions expire (in ms)*/ + int expirationTime; + + /** Used to synchronize threads */ + QMutex mutex; + +private slots: + + /** Called every minute to cleanup expired sessions. */ + void timerEvent(); +}; + +#endif // HTTPSESSIONSTORE_H diff --git a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp new file mode 100644 index 00000000..fde1cf8d --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp @@ -0,0 +1,234 @@ +/** + @file + @author Stefan Frings +*/ + +#include "staticfilecontroller.h" +#include +#include +#include +#include "httpsession.h" +#include "static.h" +#include + +StaticFileController::StaticFileController(QSettings* settings, QObject* parent) + :HttpRequestHandler(parent) +{ + maxAge=settings->value("maxAge","60000").toInt(); + encoding=settings->value("encoding","UTF-8").toString(); + docroot=settings->value("path","./server/docroot").toString(); + // Convert relative path to absolute, based on the directory of the config file. +#ifdef Q_OS_WIN32 + if (QDir::isRelativePath(docroot) && settings->format()!=QSettings::NativeFormat) +#else + if (QDir::isRelativePath(docroot)) +#endif + { +#if defined Q_OS_UNIX && ! defined Q_OS_MAC + QFileInfo configFile(QString(DATADIR)+"/YACReader"); + docroot=QFileInfo(QString(DATADIR)+"/YACReader",docroot).absoluteFilePath(); +#else + QFileInfo configFile(QApplication::applicationDirPath()); + docroot=QFileInfo(QApplication::applicationDirPath(),docroot).absoluteFilePath(); +#endif + } + qDebug("StaticFileController: docroot=%s, encoding=%s, maxAge=%i",qPrintable(docroot),qPrintable(encoding),maxAge); + maxCachedFileSize=settings->value("maxCachedFileSize","65536").toInt(); + cache.setMaxCost(settings->value("cacheSize","1000000").toInt()); + cacheTimeout=settings->value("cacheTime","60000").toInt(); + qDebug("StaticFileController: cache timeout=%i, size=%i",cacheTimeout,cache.maxCost()); +} + + +void StaticFileController::service(HttpRequest& request, HttpResponse& response) { + QByteArray path=request.getPath(); + // Forbid access to files outside the docroot directory + if (path.startsWith("/..")) { + qWarning("StaticFileController: somebody attempted to access a file outside the docroot directory"); + response.setStatus(403,"forbidden"); + response.write("403 forbidden",true); + } + + //TODO(DONE) carga sensible al dispositivo y a la localización + QString stringPath = path; + QStringList paths = QString(path).split('/'); + QString fileName = paths.last(); + stringPath.remove(fileName); + HttpSession session=Static::sessionStore->getSession(request,response,false); + QString device = session.getDeviceType(); + QString display = session.getDisplayType(); + if(fileName.endsWith(".png")) + fileName = getDeviceAwareFileName(fileName, device, display, request.getHeader("Accept-Language"), stringPath); + else + fileName = getDeviceAwareFileName(fileName, device, request.getHeader("Accept-Language"), stringPath); + QString newPath = stringPath.append(fileName); + path = newPath.toLocal8Bit(); + + //CAMBIADO + response.setHeader("Connection","close"); + //END_TODO + + // Check if we have the file in cache + //qint64 now=QDateTime::currentMSecsSinceEpoch(); + // mutex.lock(); + // CacheEntry* entry=cache.object(path); + //if (entry && (cacheTimeout==0 || entry->created>now-cacheTimeout)) { + // QByteArray document=entry->document; //copy the cached document, because other threads may destroy the cached entry immediately after mutex unlock. + // mutex.unlock(); + // qDebug("StaticFileController: Cache hit for %s",path.data()); + // setContentType(path,response); + // response.setHeader("Cache-Control","max-age="+QByteArray::number(maxAge/1000)); + // response.write(document); + //} + //else { + + // mutex.unlock(); + //qDebug("StaticFileController: Cache miss for %s",path.data()); + // The file is not in cache. + // If the filename is a directory, append index.html. + if (QFileInfo(docroot+path).isDir()) { + path+="/index.html"; + } + + + QFile file(docroot+path); + if (file.exists()) { + qDebug("StaticFileController: Open file %s",qPrintable(file.fileName())); + if (file.open(QIODevice::ReadOnly)) { + setContentType(path,response); + //response.setHeader("Cache-Control","max-age="+QByteArray::number(maxAge/1000)); + //if (file.size()<=maxCachedFileSize) { + // // Return the file content and store it also in the cache + // entry=new CacheEntry(); + // while (!file.atEnd() && !file.error()) { + // QByteArray buffer=file.read(65536); + // response.write(buffer); + // entry->document.append(buffer); + // } + // entry->created=now; + // mutex.lock(); + // cache.insert(request.getPath(),entry,entry->document.size()); + // mutex.unlock(); + //} + //else { + // Return the file content, do not store in cache*/ + while (!file.atEnd() && !file.error()) { + response.write(file.read(131072)); + //} + } + file.close(); + } + else { + qWarning("StaticFileController: Cannot open existing file %s for reading",qPrintable(file.fileName())); + response.setStatus(403,"forbidden"); + response.write("403 forbidden",true); + } + } + else { + response.setStatus(404,"not found"); + response.write("404 not found",true); + } + //} +} + +void StaticFileController::setContentType(QString fileName, HttpResponse& response) const { + if (fileName.endsWith(".png")) { + response.setHeader("Content-Type", "image/png"); + } + else if (fileName.endsWith(".jpg")) { + response.setHeader("Content-Type", "image/jpeg"); + } + else if (fileName.endsWith(".gif")) { + response.setHeader("Content-Type", "image/gif"); + } + else if (fileName.endsWith(".pdf")) { + response.setHeader("Content-Type", "application/pdf"); + } + else if (fileName.endsWith(".txt")) { + response.setHeader("Content-Type", qPrintable("text/plain; charset="+encoding)); + } + else if (fileName.endsWith(".html") || fileName.endsWith(".htm")) { + response.setHeader("Content-Type", qPrintable("text/html; charset="+encoding)); + } + else if (fileName.endsWith(".css")) { + response.setHeader("Content-Type", "text/css"); + } + else if (fileName.endsWith(".js")) { + response.setHeader("Content-Type", "text/javascript"); + } + // Todo: add all of your content types +} + +bool StaticFileController::exists(QString localizedName, QString path) const +{ + QString fileName=docroot+"/"+path + localizedName; + QFile file(fileName); + return file.exists(); +} + +//retorna fileName si no se encontró alternativa traducida ó fileName-locale.extensión si se encontró +QString StaticFileController::getLocalizedFileName(QString fileName, QString locales, QString path) const +{ + QSet tried; // used to suppress duplicate attempts + QStringList locs=locales.split(',',QString::SkipEmptyParts); + QStringList fileNameParts = fileName.split('.'); + QString file = fileNameParts.first(); + QString extension = fileNameParts.last(); + // Search for exact match + foreach (QString loc,locs) { + loc.replace(QRegExp(";.*"),""); + loc.replace('-','_'); + QString localizedName=file+"-"+loc.trimmed()+"."+extension; + if (!tried.contains(localizedName)) { + if(exists(localizedName, path)) + return localizedName; + tried.insert(localizedName); + } + } + + // Search for correct language but any country + foreach (QString loc,locs) { + loc.replace(QRegExp("[;_-].*"),""); + QString localizedName=file+"-"+loc.trimmed()+"."+extension; + if (!tried.contains(localizedName)) { + if(exists(localizedName, path)) + return localizedName; + tried.insert(localizedName); + } + } + + return fileName; +} + +QString StaticFileController::getDeviceAwareFileName(QString fileName, QString device, QString locales, QString path) const +{ + QFileInfo fi(fileName); + QString baseName = fi.baseName(); + QString extension = fi.completeSuffix(); + + QString completeFileName = getLocalizedFileName(baseName+"_"+device+"."+extension,locales,path); + + if(QFile(docroot+"/"+path+completeFileName).exists()) + return completeFileName; //existe un archivo específico para este dispositivo y locales + else + return getLocalizedFileName(fileName,locales,path); //no hay archivo específico para el dispositivo, pero puede haberlo para estas locales +} + +QString StaticFileController::getDeviceAwareFileName(QString fileName, QString device, QString display, QString locales, QString path) const +{ + QFileInfo fi(fileName); + QString baseName = fi.baseName(); + QString extension = fi.completeSuffix(); + + QString completeFileName = completeFileName = baseName+display+"."+extension; + if(QFile(docroot+"/"+path+completeFileName).exists()) + return completeFileName; + else + { + completeFileName = baseName+"_"+device+display+"."+extension; + if((QFile(docroot+"/"+path+completeFileName).exists())) + return completeFileName; + } + + return fileName; +} diff --git a/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h new file mode 100644 index 00000000..26413398 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.h @@ -0,0 +1,92 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef STATICFILECONTROLLER_H +#define STATICFILECONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" +#include +#include + +/** + Delivers static files. It is usually called by the applications main request handler when + the caller request a path that is mapped to static files. +

+ The following settings are required in the config file: +

+  path=docroot
+  encoding=UTF-8
+  maxAge=60000
+  cacheTime=60000
+  cacheSize=1000000
+  maxCachedFileSize=65536
+  
+ The path is relative to the directory of the config file. In case of windows, if the + settings are in the registry, the path is relative to the current working directory. +

+ The encoding is sent to the web browser in case of text and html files. +

+ The cache improves performance of small files when loaded from a network + drive. Large files are not cached. Files are cached as long as possible, + when cacheTime=0. The maxAge value (in msec!) controls the remote browsers cache. +

+ Do not instantiate this class in each request, because this would make the file cache + useless. Better create one instance during start-up and call it when the application + received a related HTTP request. +*/ + +class StaticFileController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(StaticFileController); +public: + + /** Constructor */ + StaticFileController(QSettings* settings, QObject* parent = 0); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); + +private: + + /** Encoding of text files */ + QString encoding; + + /** Root directory of documents */ + QString docroot; + + /** Maximum age of files in the browser cache */ + int maxAge; + + struct CacheEntry { + QByteArray document; + qint64 created; + }; + + /** Timeout for each cached file */ + int cacheTimeout; + + + /** Maximum size of files in cache, larger files are not cached */ + int maxCachedFileSize; + + /** Cache storage */ + QCache cache; + + /** Used to synchronize cache access for threads */ + QMutex mutex; + + /** Set a content-type header in the response depending on the ending of the filename */ + void setContentType(QString file, HttpResponse& response) const; + + QString getLocalizedFileName(QString fileName, QString locales, QString path) const; + QString getDeviceAwareFileName(QString fileName, QString device, QString locales, QString path) const; + QString getDeviceAwareFileName(QString fileName, QString device, QString display, QString locales, QString path) const; + + bool exists(QString localizedName, QString path) const; +}; + +#endif // STATICFILECONTROLLER_H diff --git a/YACReaderLibrary/server/lib/bfLogging/bfLogging.pri b/YACReaderLibrary/server/lib/bfLogging/bfLogging.pri new file mode 100644 index 00000000..17eae35e --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/bfLogging.pri @@ -0,0 +1,5 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += $$PWD/logmessage.h $$PWD/logger.h $$PWD/filelogger.h $$PWD/dualfilelogger.h +SOURCES += $$PWD/logmessage.cpp $$PWD/logger.cpp $$PWD/filelogger.cpp $$PWD/dualfilelogger.cpp diff --git a/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.cpp b/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.cpp new file mode 100644 index 00000000..7329cae0 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.cpp @@ -0,0 +1,20 @@ +/** + @file + @author Stefan Frings +*/ + +#include "dualfilelogger.h" + + +DualFileLogger::DualFileLogger(QSettings* firstSettings, QSettings* secondSettings, const int refreshInterval, QObject* parent) + :Logger(parent) +{ + firstLogger=new FileLogger(firstSettings, refreshInterval, this); + secondLogger=new FileLogger(secondSettings, refreshInterval, this); +} + + +void DualFileLogger::log(const QtMsgType type, const QString& message) { + firstLogger->log(type, message); + secondLogger->log(type, message); +} diff --git a/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.h b/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.h new file mode 100644 index 00000000..39bec859 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/dualfilelogger.h @@ -0,0 +1,58 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef DUALFILELOGGER_H +#define DUALFILELOGGER_H + +#include "logger.h" +#include "filelogger.h" +#include +#include +#include + +/** + Logs messages into two log files simultaneously. + May be used to create two logfiles with different configuration settings. + @see FileLogger for a description of the two underlying loggers. +*/ + +class DualFileLogger : public Logger { + Q_OBJECT + Q_DISABLE_COPY(DualFileLogger) +public: + + /** + Constructor. + @param firstSettings Configuration settings for the first log file, usually stored in an INI file. + Must not be 0. + Settings are read from the current group, so the caller must have called settings->beginGroup(). + Because the group must not change during runtime, it is recommended to provide a + separate QSettings instance to the logger that is not used by other parts of the program. + @param secondSettings Same as firstSettings, but for the second log file. + @param refreshInterval Interval of checking for changed config settings in msec, or 0=disabled + @param parent Parent object. + */ + DualFileLogger(QSettings* firstSettings, QSettings* secondSettings, const int refreshInterval=10000, QObject *parent = 0); + + /** + Decorate and log a message. + This method is thread safe. + @param type Message type (level) + @param message Message text + @see LogMessage for a description of the message decoration. + */ + virtual void log(const QtMsgType type, const QString& message); + +private: + + /** First logger */ + FileLogger* firstLogger; + + /** Second logger */ + FileLogger* secondLogger; + +}; + +#endif // DUALFILELOGGER_H diff --git a/YACReaderLibrary/server/lib/bfLogging/filelogger.cpp b/YACReaderLibrary/server/lib/bfLogging/filelogger.cpp new file mode 100644 index 00000000..24e32a35 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/filelogger.cpp @@ -0,0 +1,174 @@ +/** + @file + @author Stefan Frings +*/ + +#include "filelogger.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "yacreader_global.h" + +void FileLogger::refreshSettings() { + mutex.lock(); + // Save old file name for later comparision with new settings + QString oldFileName=fileName; + + // Load new config settings + settings->sync(); + fileName=settings->value("fileName","server_log.log").toString(); + // Convert relative fileName to absolute, based on the directory of the config file. +#ifdef Q_OS_WIN32 + if (QDir::isRelativePath(fileName) && settings->format()!=QSettings::NativeFormat) +#else + if (QDir::isRelativePath(fileName)) +#endif + { + QFileInfo configFile(YACReader::getSettingsPath()); + fileName=QFileInfo(YACReader::getSettingsPath(),fileName).absoluteFilePath(); + } + maxSize=settings->value("maxSize",1048576).toLongLong(); + maxBackups=settings->value("maxBackups",1).toInt(); + msgFormat=settings->value("msgFormat","{timestamp} {type} {msg}").toString(); + timestampFormat=settings->value("timestampFormat","yyyy-MM-dd hh:mm:ss.zzz").toString(); + minLevel=static_cast(settings->value("minLevel",0).toInt()); + bufferSize=settings->value("bufferSize",0).toInt(); + + // Create new file if the filename has been changed + if (oldFileName!=fileName) { + fprintf(stderr,"Logging to %s\n",qPrintable(fileName)); + close(); + open(); + } + mutex.unlock(); +} + + +FileLogger::FileLogger(QSettings* settings, const int refreshInterval, QObject* parent) + : Logger(parent) +{ + Q_ASSERT(settings!=0); + Q_ASSERT(refreshInterval>=0); + this->settings=settings; + file=0; + if (refreshInterval>0) { + refreshTimer.start(refreshInterval,this); + } + flushTimer.start(1000,this); + refreshSettings(); +} + + +FileLogger::~FileLogger() { + close(); +} + + +void FileLogger::write(const LogMessage* logMessage) { + // Try to write to the file + if (file) { + + // Write the message + file->write(qPrintable(logMessage->toString(msgFormat,timestampFormat))); + + // Flush error messages immediately, to ensure that no important message + // gets lost when the program terinates abnormally. + if (logMessage->getType()>=QtCriticalMsg) { + file->flush(); + } + + // Check for success + if (file->error()) { + close(); + qWarning("Cannot write to log file %s: %s",qPrintable(fileName),qPrintable(file->errorString())); + } + + } + + // Fall-back to the super class method, if writing failed + if (!file) { + Logger::write(logMessage); + } + +} + +void FileLogger::open() { + if (fileName.isEmpty()) { + qWarning("Name of logFile is empty"); + } + else { + file=new QFile(fileName); + if (!file->open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text)) { + qWarning("Cannot open log file %s: %s",qPrintable(fileName),qPrintable(file->errorString())); + file=0; + } + } +} + + +void FileLogger::close() { + if (file) { + file->close(); + delete file; + file=0; + } +} + +void FileLogger::rotate() { + // count current number of existing backup files + int count=0; + forever { + QFile bakFile(QString("%1.%2").arg(fileName).arg(count+1)); + if (bakFile.exists()) { + ++count; + } + else { + break; + } + } + + // Remove all old backup files that exceed the maximum number + while (maxBackups>0 && count>=maxBackups) { + QFile::remove(QString("%1.%2").arg(fileName).arg(count)); + --count; + } + + // Rotate backup files + for (int i=count; i>0; --i) { + QFile::rename(QString("%1.%2").arg(fileName).arg(i),QString("%1.%2").arg(fileName).arg(i+1)); + } + + // Backup the current logfile + QFile::rename(fileName,fileName+".1"); +} + + +void FileLogger::timerEvent(QTimerEvent* event) { + if (!event) { + return; + } + else if (event->timerId()==refreshTimer.timerId()) { + refreshSettings(); + } + else if (event->timerId()==flushTimer.timerId() && file) { + mutex.lock(); + + // Flush the I/O buffer + file->flush(); + + // Rotate the file if it is too large + if (maxSize>0 && file->size()>=maxSize) { + close(); + rotate(); + open(); + } + + mutex.unlock(); + } +} diff --git a/YACReaderLibrary/server/lib/bfLogging/filelogger.h b/YACReaderLibrary/server/lib/bfLogging/filelogger.h new file mode 100644 index 00000000..617b5bff --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/filelogger.h @@ -0,0 +1,122 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef FILELOGGER_H +#define FILELOGGER_H + +#include +#include +#include +#include +#include +#include "logger.h" + +/** + Logger that uses a text file for output. Settings are read from a + config file using a QSettings object. Config settings can be changed at runtime. +

+ Example for the configuration settings: +

+  fileName=logs/QtWebApp.log
+  maxSize=1000000
+  maxBackups=2
+  minLevel=0
+  msgformat={timestamp} {typeNr} {type} thread={thread}: {msg}
+  timestampFormat=dd.MM.yyyy hh:mm:ss.zzz
+  bufferSize=0
+  
+ + - fileName is the name of the log file, relative to the directory of the settings file. + In case of windows, if the settings are in the registry, the path is relative to the current + working directory. + - maxSize is the maximum size of that file in bytes. The file will be backed up and + replaced by a new file if it becomes larger than this limit. Please note that + the actual file size may become a little bit larger than this limit. Default is 0=unlimited. + - maxBackups defines the number of backup files to keep. Default is 0=unlimited. + - minLevel defines the minimum type of messages that are written (together with buffered messages) into the file. Defaults is 0=debug. + - msgFormat defines the decoration of log messages, see LogMessage class. Default is "{timestamp} {type} {msg}". + - timestampFormat defines the format of timestamps, see QDateTime::toString(). Default is "yyyy-MM-dd hh:mm:ss.zzz". + - bufferSize defines the size of the buffer. Default is 0=disabled. + + @see set() describes how to set logger variables + @see LogMessage for a description of the message decoration. + @see Logger for a descrition of the buffer. +*/ + +class FileLogger : public Logger { + Q_OBJECT + Q_DISABLE_COPY(FileLogger) +public: + + /** + Constructor. + @param settings Configuration settings, usually stored in an INI file. Must not be 0. + Settings are read from the current group, so the caller must have called settings->beginGroup(). + Because the group must not change during runtime, it is recommended to provide a + separate QSettings instance to the logger that is not used by other parts of the program. + @param refreshInterval Interval of checking for changed config settings in msec, or 0=disabled + @param parent Parent object + */ + FileLogger(QSettings* settings, const int refreshInterval=10000, QObject* parent = 0); + + /** + Destructor. Closes the file. + */ + virtual ~FileLogger(); + + /** Write a message to the log file */ + virtual void write(const LogMessage* logMessage); + +protected: + + /** + Handler for timer events. + Refreshes config settings or synchronizes I/O buffer, depending on the event. + This method is thread-safe. + @param event used to distinguish between the two timers. + */ + void timerEvent(QTimerEvent* event); + +private: + + /** Configured name of the log file */ + QString fileName; + + /** Configured maximum size of the file in bytes, or 0=unlimited */ + long maxSize; + + /** Configured maximum number of backup files, or 0=unlimited */ + int maxBackups; + + /** Pointer to the configuration settings */ + QSettings* settings; + + /** Output file, or 0=disabled */ + QFile* file; + + /** Timer for refreshing configuration settings */ + QBasicTimer refreshTimer; + + /** Timer for flushing the file I/O buffer */ + QBasicTimer flushTimer; + + /** Open the output file */ + void open(); + + /** Close the output file */ + void close(); + + /** Rotate files and delete some backups if there are too many */ + void rotate(); + + /** + Refreshes the configuration settings. + This method is thread-safe. + */ + void refreshSettings(); + +}; + +#endif // FILELOGGER_H diff --git a/YACReaderLibrary/server/lib/bfLogging/logger.cpp b/YACReaderLibrary/server/lib/bfLogging/logger.cpp new file mode 100644 index 00000000..de3ab1a5 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/logger.cpp @@ -0,0 +1,172 @@ +/** + @file + @author Stefan Frings +*/ + +#include "logger.h" +#include +#include +#include +#include +#include + +Logger* Logger::defaultLogger=0; + + +QThreadStorage*> Logger::logVars; + + +QThreadStorage*> Logger::buffers; + + +QMutex Logger::mutex; + + +Logger::Logger(QObject* parent) + : QObject(parent), + msgFormat("{timestamp} {type} {msg}"), + timestampFormat("dd.MM.yyyy hh:mm:ss.zzz"), + minLevel(QtDebugMsg), + bufferSize(0) + {} + + +Logger::Logger(const QString msgFormat, const QString timestampFormat, const QtMsgType minLevel, const int bufferSize, QObject* parent) + :QObject(parent) { + this->msgFormat=msgFormat; + this->timestampFormat=timestampFormat; + this->minLevel=minLevel; + this->bufferSize=bufferSize; +} + + +void Logger::msgHandler(const QtMsgType type, const QString &message, const QString &file, const QString &function, const int line) { + static QMutex recursiveMutex(QMutex::Recursive); + static QMutex nonRecursiveMutex(QMutex::NonRecursive); + + // Prevent multiple threads from calling this method simultaneoulsy. + // But allow recursive calls, which is required to prevent a deadlock + // if the logger itself produces an error message. + recursiveMutex.lock(); + + // Fall back to stderr when this method has been called recursively. + if (defaultLogger && nonRecursiveMutex.tryLock()) { + defaultLogger->log(type, message, file, function, line); + nonRecursiveMutex.unlock(); + } + else { + fputs(qPrintable(message),stderr); + fflush(stderr); + } + + // Abort the program after logging a fatal message + if (type>=QtFatalMsg) { + //abort(); + } + + recursiveMutex.unlock(); +} + + +#if QT_VERSION >= 0x050000 + void Logger::msgHandler5(const QtMsgType type, const QMessageLogContext &context, const QString &message) { + (void)(context); // suppress "unused parameter" warning + msgHandler(type,message,context.file,context.function,context.line); + } +#else + void Logger::msgHandler4(const QtMsgType type, const char* message) { + msgHandler(type,message); + } +#endif + + +Logger::~Logger() { + if (defaultLogger==this) { +#if QT_VERSION >= 0x050000 + qInstallMessageHandler(0); +#else + qInstallMsgHandler(0); +#endif + defaultLogger=0; + } +} + + +void Logger::write(const LogMessage* logMessage) { + fputs(qPrintable(logMessage->toString(msgFormat,timestampFormat)),stderr); + fflush(stderr); +} + + +void Logger::installMsgHandler() { + defaultLogger=this; +#if QT_VERSION >= 0x050000 + qInstallMessageHandler(msgHandler5); +#else + qInstallMsgHandler(msgHandler4); +#endif +} + + +void Logger::set(const QString& name, const QString& value) { + mutex.lock(); + if (!logVars.hasLocalData()) { + logVars.setLocalData(new QHash); + } + logVars.localData()->insert(name,value); + mutex.unlock(); +} + + +void Logger::clear(const bool buffer, const bool variables) { + mutex.lock(); + if (buffer && buffers.hasLocalData()) { + QList* buffer=buffers.localData(); + while (buffer && !buffer->isEmpty()) { + LogMessage* logMessage=buffer->takeLast(); + delete logMessage; + } + } + if (variables && logVars.hasLocalData()) { + logVars.localData()->clear(); + } + mutex.unlock(); +} + + +void Logger::log(const QtMsgType type, const QString& message, const QString &file, const QString &function, const int line) { + mutex.lock(); + + // If the buffer is enabled, write the message into it + if (bufferSize>0) { + // Create new thread local buffer, if necessary + if (!buffers.hasLocalData()) { + buffers.setLocalData(new QList()); + } + QList* buffer=buffers.localData(); + // Append the decorated log message + LogMessage* logMessage=new LogMessage(type,message,logVars.localData(),file,function,line); + buffer->append(logMessage); + // Delete oldest message if the buffer became too large + if (buffer->size()>bufferSize) { + delete buffer->takeFirst(); + } + // If the type of the message is high enough, print the whole buffer + if (type>=minLevel) { + while (!buffer->isEmpty()) { + LogMessage* logMessage=buffer->takeFirst(); + write(logMessage); + delete logMessage; + } + } + } + + // Buffer is disabled, print the message if the type is high enough + else { + if (type>=minLevel) { + LogMessage logMessage(type,message,logVars.localData(),file,function,line); + write(&logMessage); + } + } + mutex.unlock(); +} diff --git a/YACReaderLibrary/server/lib/bfLogging/logger.h b/YACReaderLibrary/server/lib/bfLogging/logger.h new file mode 100644 index 00000000..bbd278ca --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/logger.h @@ -0,0 +1,181 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef LOGGER_H +#define LOGGER_H + +#include +#include +#include +#include +#include +#include "logmessage.h" + +/** + Decorates and writes log messages to the console, stderr. +

+ The decorator uses a predefined msgFormat string to enrich log messages + with additional information (e.g. timestamp). +

+ The msgFormat string and also the message text may contain additional + variable names in the form {name} that are filled by values + taken from a static thread local dictionary. +

+ The logger keeps a configurable number of messages in a ring-buffer. + A log message with a severity >= minLevel flushes the buffer, + so the stored messages get written out. If the buffer is disabled, then + only messages with severity >= minLevel are written out. +

+ If you enable the buffer and use minLevel=2, then the application logs + only errors together with some buffered debug messages. But as long no + error occurs, nothing gets written out. +

+ Each thread has it's own buffer. +

+ The logger can be registered to handle messages from + the static global functions qDebug(), qWarning(), qCritical() and qFatal(). + + @see set() describes how to set logger variables + @see LogMessage for a description of the message decoration. + @warning You should prefer a derived class, for example FileLogger, + because logging to the console is less useful. +*/ + +class Logger : public QObject { + Q_OBJECT + Q_DISABLE_COPY(Logger) +public: + + /** + Constructor. + Uses the same defaults as the other constructor. + @param parent Parent object + */ + Logger(QObject* parent); + + + /** + Constructor. + @param msgFormat Format of the decoration, e.g. "{timestamp} {type} thread={thread}: {msg}" + @param timestampFormat Format of timestamp, e.g. "dd.MM.yyyy hh:mm:ss.zzz" + @param minLevel Minimum severity that genertes an output (0=debug, 1=warning, 2=critical, 3=fatal). + @param bufferSize Size of the backtrace buffer, number of messages per thread. 0=disabled. + @param parent Parent object + @see LogMessage for a description of the message decoration. + */ + Logger(const QString msgFormat="{timestamp} {type} {msg}", const QString timestampFormat="dd.MM.yyyy hh:mm:ss.zzz", const QtMsgType minLevel=QtDebugMsg, const int bufferSize=0, QObject* parent = 0); + + /** Destructor */ + virtual ~Logger(); + + /** + Decorate and log the message, if type>=minLevel. + This method is thread safe. + @param type Message type (level) + @param message Message text + @param file Name of the source file where the message was generated (usually filled with the macro __FILE__) + @param function Name of the function where the message was generated (usually filled with the macro __LINE__) + @param line Line Number of the source file, where the message was generated (usually filles with the macro __func__ or __FUNCTION__) + @see LogMessage for a description of the message decoration. + */ + virtual void log(const QtMsgType type, const QString& message, const QString &file="", const QString &function="", const int line=0); + + /** + Installs this logger as the default message handler, so it + can be used through the global static logging functions (e.g. qDebug()). + */ + void installMsgHandler(); + + /** + Sets a thread-local variable that may be used to decorate log messages. + This method is thread safe. + @param name Name of the variable + @param value Value of the variable + */ + static void set(const QString& name, const QString& value); + + /** + Clear the thread-local data of the current thread. + @param buffer Whether to clear the backtrace buffer + @param variables Whether to clear the log variables + */ + static void clear(const bool buffer=true, const bool variables=true); + +protected: + + /** Format string for message decoration */ + QString msgFormat; + + /** Format string of timestamps */ + QString timestampFormat; + + /** Minimum level of message types that are written out */ + QtMsgType minLevel; + + /** Size of backtrace buffer, number of messages per thread. 0=disabled */ + int bufferSize; + + /** Used to synchronize access to the static members */ + static QMutex mutex; + + /** + Decorate and write a log message to stderr. Override this method + to provide a different output medium. + */ + virtual void write(const LogMessage* logMessage); + +private: + + /** Pointer to the default logger, used by msgHandler() */ + static Logger* defaultLogger; + + /** + Message Handler for the global static logging functions (e.g. qDebug()). + Forward calls to the default logger. +

+ In case of a fatal message, the program will abort. + Variables in the in the message are replaced by their values. + This method is thread safe. + @param type Message type (level) + @param message Message text + @param file Name of the source file where the message was generated (usually filled with the macro __FILE__) + @param function Name of the function where the message was generated (usually filled with the macro __LINE__) + @param line Line Number of the source file, where the message was generated (usually filles with the macro __func__ or __FUNCTION__) + */ + static void msgHandler(const QtMsgType type, const QString &message, const QString &file="", const QString &function="", const int line=0); + + +#if QT_VERSION >= 0x050000 + + /** + Wrapper for QT version 5. + @param type Message type (level) + @param context Message context + @param message Message text + @see msgHandler() + */ + static void msgHandler5(const QtMsgType type, const QMessageLogContext& context, const QString &message); + +#else + + /** + Wrapper for QT version 4. + @param type Message type (level) + @param message Message text + @see msgHandler() + */ + static void msgHandler4(const QtMsgType type, const char * message); + +#endif + + /** Thread local variables to be used in log messages */ + static QThreadStorage*> logVars; + + /** Thread local backtrace buffers */ + static QThreadStorage*> buffers; + +}; + +#endif // LOGGER_H diff --git a/YACReaderLibrary/server/lib/bfLogging/logmessage.cpp b/YACReaderLibrary/server/lib/bfLogging/logmessage.cpp new file mode 100644 index 00000000..4d610383 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/logmessage.cpp @@ -0,0 +1,75 @@ +/** + @file + @author Stefan Frings +*/ + +#include "logmessage.h" +#include + +LogMessage::LogMessage(const QtMsgType type, const QString& message, QHash* logVars, const QString &file, const QString &function, const int line) { + this->type=type; + this->message=message; + this->file=file; + this->function=function; + this->line=line; + timestamp=QDateTime::currentDateTime(); + threadId=QThread::currentThreadId(); + + // Copy the logVars if not null, + // so that later changes in the original do not affect the copy + if (logVars) { + this->logVars=*logVars; + } +} + +QString LogMessage::toString(const QString& msgFormat, const QString& timestampFormat) const { + QString decorated=msgFormat+"\n"; + decorated.replace("{msg}",message); + + if (decorated.contains("{timestamp}")) { + decorated.replace("{timestamp}",timestamp.toString(timestampFormat)); + } + + QString typeNr; + typeNr.setNum(type); + decorated.replace("{typeNr}",typeNr); + + switch (type) { + case QtDebugMsg: + decorated.replace("{type}","DEBUG"); + break; + case QtWarningMsg: + decorated.replace("{type}","WARNING"); + break; + case QtCriticalMsg: + decorated.replace("{type}","CRITICAL"); + break; + case QtFatalMsg: + decorated.replace("{type}","FATAL"); + break; + default: + decorated.replace("{type}",typeNr); + } + + decorated.replace("{file}",file); + decorated.replace("{function}",function); + decorated.replace("{line}",QString::number(line)); + + QString threadId; + threadId.setNum((quint64)QThread::currentThreadId()); // change to (qint64) for 64bit Mac OS + decorated.replace("{thread}",threadId); + + // Fill in variables + if (decorated.contains("{") && !logVars.isEmpty()) { + QList keys=logVars.keys(); + foreach (QString key, keys) { + decorated.replace("{"+key+"}",logVars.value(key)); + } + } + + return decorated; +} + +QtMsgType LogMessage::getType() const { + return type; +} diff --git a/YACReaderLibrary/server/lib/bfLogging/logmessage.h b/YACReaderLibrary/server/lib/bfLogging/logmessage.h new file mode 100644 index 00000000..433d949d --- /dev/null +++ b/YACReaderLibrary/server/lib/bfLogging/logmessage.h @@ -0,0 +1,91 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef LOGMESSAGE_H +#define LOGMESSAGE_H + +#include +#include +#include + +/** + Represents a single log message together with some data + that are used to decorate the log message. + + The following variables may be used in the message and in msgFormat: + + - {timestamp} Date and time of creation + - {typeNr} Type of the message in numeric format (0-3) + - {type} Type of the message in string format (DEBUG, WARNING, CRITICAL, FATAL) + - {thread} ID number of the thread + - {msg} Message text (only useable in msgFormat) + - {file} Filename where the message was generated # + - {function} Function where the message was generated # + - {line} Line number where the message was generated # + - {xxx} For any user-defined logger variable + + # The macros qDebug()...qFatal() dont fill these variable in case of QT versions before 5.0. +*/ + +class LogMessage +{ + Q_DISABLE_COPY(LogMessage) +public: + + /** + Constructor. All parameters are copied, so that later changes to them do not + affect this object. + @param type Type of the message + @param message Message text + @param logVars Logger variables, 0 is allowed + @param file Name of the source file where the message was generated + @param function Name of the function where the message was generated + @param line Line Number of the source file, where the message was generated + */ + LogMessage(const QtMsgType type, const QString& message, QHash* logVars, const QString &file, const QString &function, const int line); + + /** + Returns the log message as decorated string. + @param msgFormat Format of the decoration. May contain variables and static text, + e.g. "{timestamp} {type} thread={thread}: {msg}". + @param timestampFormat Format of timestamp, e.g. "dd.MM.yyyy hh:mm:ss.zzz", see QDateTime::toString(). + @see QDatetime for a description of the timestamp format pattern + */ + QString toString(const QString& msgFormat, const QString& timestampFormat) const; + + /** + Get the message type. + */ + QtMsgType getType() const; + +private: + + /** Logger variables */ + QHash logVars; + + /** Date and time of creation */ + QDateTime timestamp; + + /** Type of the message */ + QtMsgType type; + + /** ID number of the thread */ + Qt::HANDLE threadId; + + /** Message text */ + QString message; + + /** Filename where the message was generated */ + QString file; + + /** Function name where the message was generated */ + QString function; + + /** Line number where the message was generated */ + int line; + +}; + +#endif // LOGMESSAGE_H diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/bfTemplateEngine.pri b/YACReaderLibrary/server/lib/bfTemplateEngine/bfTemplateEngine.pri new file mode 100644 index 00000000..d3eba98b --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/bfTemplateEngine.pri @@ -0,0 +1,7 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += $$PWD/template.h $$PWD/templateloader.h $$PWD/templatecache.h +SOURCES += $$PWD/template.cpp $$PWD/templateloader.cpp $$PWD/templatecache.cpp + +OTHER_FILES += $$PWD/../doc/readme.txt diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/template.cpp b/YACReaderLibrary/server/lib/bfTemplateEngine/template.cpp new file mode 100644 index 00000000..23abac9e --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/template.cpp @@ -0,0 +1,188 @@ +/** + @file + @author Stefan Frings +*/ + +#include "template.h" +#include + +Template::Template(QString source, QString sourceName) + : QString(source) { + this->sourceName=sourceName; + this->warnings=false; +} + +Template::Template(QFile& file, QTextCodec* textCodec) { + this->warnings=false; + sourceName=QFileInfo(file.fileName()).baseName(); + if (!file.isOpen()) { + file.open(QFile::ReadOnly | QFile::Text); + } + QByteArray data=file.readAll(); + file.close(); + if (data.size()==0 || file.error()) { + qCritical("Template: cannot read from %s, %s",qPrintable(sourceName),qPrintable(file.errorString())); + append(textCodec->toUnicode(data)); + } +} + + +int Template::setVariable(QString name, QString value) { + int count=0; + QString variable="{"+name+"}"; + int start=indexOf(variable); + while (start>=0) { + replace(start, variable.length(), value); + count++; + start=indexOf(variable,start+value.length()); + } + if (count==0 && warnings) { + qWarning("Template: missing variable %s in %s",qPrintable(variable),qPrintable(sourceName)); + } + return count; +} + +int Template::setCondition(QString name, bool value) { + int count=0; + QString startTag=QString("{if %1}").arg(name); + QString elseTag=QString("{else %1}").arg(name); + QString endTag=QString("{end %1}").arg(name); + // search for if-else-end + int start=indexOf(startTag); + while (start>=0) { + int end=indexOf(endTag,start+startTag.length()); + if (end>=0) { + count++; + int ellse=indexOf(elseTag,start+startTag.length()); + if (ellse>start && ellse=0) { + int end=indexOf(endTag,start+startTag2.length()); + if (end>=0) { + count++; + int ellse=indexOf(elseTag,start+startTag2.length()); + if (ellse>start && ellse=0); + int count=0; + QString startTag="{loop "+name+"}"; + QString elseTag="{else "+name+"}"; + QString endTag="{end "+name+"}"; + // search for loop-else-end + int start=indexOf(startTag); + while (start>=0) { + int end=indexOf(endTag,start+startTag.length()); + if (end>=0) { + count++; + int ellse=indexOf(elseTag,start+startTag.length()); + if (ellse>start && ellse0) { + QString loopPart=mid(start+startTag.length(), ellse-start-startTag.length()); + QString insertMe; + for (int i=0; i0) { // and no else part + QString loopPart=mid(start+startTag.length(), end-start-startTag.length()); + QString insertMe; + for (int i=0; i +#include +#include +#include +#include +#include + +/** + Enhanced version of QString for template processing. Templates + are usually loaded from files, but may also be loaded from + prepared Strings. + Example template file: +

+ Hello {username}, how are you?
+
+ {if locked}
+     Your account is locked.
+ {else locked}
+     Welcome on our system.
+ {end locked}
+
+ The following users are on-line:
+     Username       Time
+ {loop user}
+     {user.name}    {user.time}
+ {end user}
+ 

+

+ Example code to fill this template: +

+ Template t(QFile("test.tpl"),QTextCode::codecForName("UTF-8"));
+ t.setVariable("user", "Stefan");
+ t.setCondition("locked",false);
+ t.loop("user",2);
+ t.setVariable("user0.name,"Markus");
+ t.setVariable("user0.time,"8:30");
+ t.setVariable("user1.name,"Roland");
+ t.setVariable("user1.time,"8:45");
+ 

+

+ The code example above shows how variable within loops are numbered. + Counting starts with 0. Loops can be nested, for example: +

+ <table>
+ {loop row}
+     <tr>
+     {loop row.column}
+         <td>{row.column.value}</td>
+     {end row.column}
+     </tr>
+ {end row}
+ </table>
+ 

+

+ Example code to fill this nested loop with 3 rows and 4 columns: +

+ t.loop("row",3);
+
+ t.loop("row0.column",4);
+ t.setVariable("row0.column0.value","a");
+ t.setVariable("row0.column1.value","b");
+ t.setVariable("row0.column2.value","c");
+ t.setVariable("row0.column3.value","d");
+
+ t.loop("row1.column",4);
+ t.setVariable("row1.column0.value","e");
+ t.setVariable("row1.column1.value","f");
+ t.setVariable("row1.column2.value","g");
+ t.setVariable("row1.column3.value","h");
+
+ t.loop("row2.column",4);
+ t.setVariable("row2.column0.value","i");
+ t.setVariable("row2.column1.value","j");
+ t.setVariable("row2.column2.value","k");
+ t.setVariable("row2.column3.value","l");
+ 

+ @see TemplateLoader + @see TemplateCache +*/ + +class Template : public QString { +public: + + /** + Constructor that reads the template from a string. + @param source The template source text + @param sourceName Name of the source file, used for logging + */ + Template(QString source, QString sourceName); + + /** + Constructor that reads the template from a file. Note that this class does not + cache template files by itself, so using this constructor is only recommended + to be used on local filesystem. + @param file File that provides the source text + @param textCodec Encoding of the source + @see TemplateLoader + @see TemplateCache + */ + Template(QFile& file, QTextCodec* textCodec); + + /** + Replace a variable by the given value. + Affects tags with the syntax + + - {name} + + After settings the + value of a variable, the variable does not exist anymore, + it it cannot be changed multiple times. + @param name name of the variable + @param value new value + @return The count of variables that have been processed + */ + int setVariable(QString name, QString value); + + /** + Set a condition. This affects tags with the syntax + + - {if name}...{end name} + - {if name}...{else name}...{end name} + - {ifnot name}...{end name} + - {ifnot name}...{else name}...{end name} + + @param name Name of the condition + @param value Value of the condition + @return The count of conditions that have been processed + */ + int setCondition(QString name, bool value); + + /** + Set number of repetitions of a loop. + This affects tags with the syntax + + - {loop name}...{end name} + - {loop name}...{else name}...{end name} + + @param name Name of the loop + @param repetitions The number of repetitions + @return The number of loops that have been processed + */ + int loop(QString name, int repetitions); + + /** + Enable warnings for missing tags + @param enable Warnings are enabled, if true + */ + void enableWarnings(bool enable=true); + +private: + + /** Name of the source file */ + QString sourceName; + + /** Enables warnings, if true */ + bool warnings; +}; + +#endif // TEMPLATE_H diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.cpp b/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.cpp new file mode 100644 index 00000000..823f4f24 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.cpp @@ -0,0 +1,30 @@ +#include "templatecache.h" +#include +#include +#include + +TemplateCache::TemplateCache(QSettings* settings, QObject* parent) + :TemplateLoader(settings,parent) +{ + cache.setMaxCost(settings->value("cacheSize","160000").toInt());//este tamaño antes era 1000000 + cacheTimeout=settings->value("cacheTime","60000").toInt(); + qDebug("TemplateCache: timeout=%i, size=%i",cacheTimeout,cache.maxCost()); +} + +QString TemplateCache::tryFile(QString localizedName) { + qint64 now=QDateTime::currentMSecsSinceEpoch(); + // search in cache + qDebug("TemplateCache: trying cached %s",qPrintable(localizedName)); + CacheEntry* entry=cache.object(localizedName); + if (entry && (cacheTimeout==0 || entry->created>now-cacheTimeout)) { + return entry->document; + } + // search on filesystem + entry=new CacheEntry(); + entry->created=now; + entry->document=TemplateLoader::tryFile(localizedName); + // Store in cache even when the file did not exist, to remember that there is no such file + cache.insert(localizedName,entry,entry->document.size()); + return entry->document; +} + diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.h b/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.h new file mode 100644 index 00000000..6e79f119 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/templatecache.h @@ -0,0 +1,77 @@ +#ifndef TEMPLATECACHE_H +#define TEMPLATECACHE_H + +#include "templateloader.h" +#include + +/** + Caching template loader, reduces the amount of I/O and improves performance + on remote file systems. The cache has a limited size, it prefers to keep + the last recently used files. Optionally, the maximum time of cached entries + can be defined to enforce a reload of the template file after a while. +

+ In case of local file system, the use of this cache is optionally, since + the operating system caches files already. +

+ Loads localized versions of template files. If the caller requests a file with the + name "index" and the suffix is ".tpl" and the requested locale is "de_DE, de, en-US", + then files are searched in the following order: + + - index-de_DE.tpl + - index-de.tpl + - index-en_US.tpl + - index-en.tpl + - index.tpl +

+ The following settings are required: +

+  path=.
+  suffix=.tpl
+  encoding=UTF-8
+  cacheSize=1000000
+  cacheTime=60000
+  
+ The path is relative to the directory of the config file. In case of windows, if the + settings are in the registry, the path is relative to the current working directory. +

+ Files are cached as long as possible, when cacheTime=0. + @see TemplateLoader +*/ + +class TemplateCache : public TemplateLoader { + Q_OBJECT + Q_DISABLE_COPY(TemplateCache); +public: + + /** + Constructor. + @param settings configurations settings + @param parent Parent object + */ + TemplateCache(QSettings* settings, QObject* parent=0); + +protected: + + /** + Try to get a file from cache or filesystem. + @param localizedName Name of the template with locale to find + @return The template document, or empty string if not found + */ + virtual QString tryFile(QString localizedName); + +private: + + struct CacheEntry { + QString document; + qint64 created; + }; + + /** Timeout for each cached file */ + int cacheTimeout; + + /** Cache storage */ + QCache cache; + +}; + +#endif // TEMPLATECACHE_H diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.cpp b/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.cpp new file mode 100644 index 00000000..5cb416e4 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.cpp @@ -0,0 +1,109 @@ +/** + @file + @author Stefan Frings +*/ + +#include "templateloader.h" +#include +#include +#include +#include +#include +#include + +TemplateLoader::TemplateLoader(QSettings* settings, QObject* parent) + : QObject(parent) +{ + templatePath=settings->value("path","./server/templates").toString(); + // Convert relative path to absolute, based on the directory of the config file. +#ifdef Q_OS_WIN32 + if (QDir::isRelativePath(templatePath) && settings->format()!=QSettings::NativeFormat) +#else + if (QDir::isRelativePath(templatePath)) +#endif + { +#if defined Q_OS_UNIX && !defined Q_OS_MAC + QFileInfo configFile(QString(DATADIR)+"/YACReader"); + templatePath=QFileInfo(QString(DATADIR)+"/YACReader",templatePath).absoluteFilePath(); +#else + QFileInfo configFile(QApplication::applicationDirPath()); + templatePath=QFileInfo(QApplication::applicationDirPath(),templatePath).absoluteFilePath(); +#endif + } + fileNameSuffix=settings->value("suffix",".tpl").toString(); + QString encoding=settings->value("encoding").toString(); + if (encoding.isEmpty()) { + textCodec=QTextCodec::codecForLocale(); + } + else { + textCodec=QTextCodec::codecForName(encoding.toLocal8Bit()); + } + qDebug("TemplateLoader: path=%s, codec=%s",qPrintable(templatePath),textCodec->name().data()); +} + +TemplateLoader::~TemplateLoader() {} + +QString TemplateLoader::tryFile(QString localizedName) { + QString fileName=templatePath+"/"+localizedName+fileNameSuffix; + qDebug("TemplateCache: trying file %s",qPrintable(fileName)); + QFile file(fileName); + if (file.exists()) { + file.open(QIODevice::ReadOnly); + QString document=textCodec->toUnicode(file.readAll()); + file.close(); + if (file.error()) { + qCritical("TemplateLoader: cannot load file %s, %s",qPrintable(fileName),qPrintable(file.errorString())); + return ""; + } + else { + return document; + } + } + return ""; +} + +Template TemplateLoader::getTemplate(QString templateName, QString locales) { + mutex.lock(); + QSet tried; // used to suppress duplicate attempts + QStringList locs=locales.split(',',QString::SkipEmptyParts); + + // Search for exact match + foreach (QString loc,locs) { + loc.replace(QRegExp(";.*"),""); + loc.replace('-','_'); + QString localizedName=templateName+"-"+loc.trimmed(); + if (!tried.contains(localizedName)) { + QString document=tryFile(localizedName); + if (!document.isEmpty()) { + mutex.unlock(); + return Template(document,localizedName); + } + tried.insert(localizedName); + } + } + + // Search for correct language but any country + foreach (QString loc,locs) { + loc.replace(QRegExp("[;_-].*"),""); + QString localizedName=templateName+"-"+loc.trimmed(); + if (!tried.contains(localizedName)) { + QString document=tryFile(localizedName); + if (!document.isEmpty()) { + mutex.unlock(); + return Template(document,localizedName); + } + tried.insert(localizedName); + } + } + + // Search for default file + QString document=tryFile(templateName); + if (!document.isEmpty()) { + mutex.unlock(); + return Template(document,templateName); + } + + qCritical("TemplateCache: cannot find template %s",qPrintable(templateName)); + mutex.unlock(); + return Template("",templateName); +} diff --git a/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.h b/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.h new file mode 100644 index 00000000..5635af40 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfTemplateEngine/templateloader.h @@ -0,0 +1,85 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef TEMPLATELOADER_H +#define TEMPLATELOADER_H + +#include +#include +#include +#include "template.h" +#include + +/** + Loads localized versions of template files. If the caller requests a file with the + name "index" and the suffix is ".tpl" and the requested locale is "de_DE, de, en-US", + then files are searched in the following order: + + - index-de_DE.tpl + - index-de.tpl + - index-en_US.tpl + - index-en.tpl + - index.tpl + + The following settings are required: +

+  path=.
+  suffix=.tpl
+  encoding=UTF-8
+  
+ The path is relative to the directory of the config file. In case of windows, if the + settings are in the registry, the path is relative to the current working directory. + @see TemplateCache +*/ + +class TemplateLoader : public QObject { + Q_OBJECT + Q_DISABLE_COPY(TemplateLoader); +public: + + /** + Constructor. + @param settings configurations settings + @param parent parent object + */ + TemplateLoader(QSettings* settings, QObject* parent=0); + + /** Destructor */ + virtual ~TemplateLoader(); + + /** + Get a template for a given locale. + This method is thread safe. + @param templateName base name of the template file, without suffix and without locale + @param locales Requested locale(s), e.g. "de_DE, en_EN". Strings in the format of + the HTTP header Accept-Locale may be used. Badly formatted parts in the string are silently + ignored. + @return If the template cannot be loaded, an error message is logged and an empty template is returned. + */ + Template getTemplate(QString templateName, QString locales=QString()); + +protected: + + /** + Try to get a file from cache or filesystem. + @param localizedName Name of the template with locale to find + @return The template document, or empty string if not found + */ + virtual QString tryFile(QString localizedName); + + /** Directory where the templates are searched */ + QString templatePath; + + /** Suffix to the filenames */ + QString fileNameSuffix; + + /** Codec for decoding the files */ + QTextCodec* textCodec; + + /** Used to synchronize threads */ + QMutex mutex; +}; + +#endif // TEMPLATELOADER_H diff --git a/YACReaderLibrary/server/requestmapper.cpp b/YACReaderLibrary/server/requestmapper.cpp new file mode 100644 index 00000000..f867b944 --- /dev/null +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -0,0 +1,101 @@ +/** + @file + @author Stefan Frings +*/ + +#include "requestmapper.h" +#include "static.h" +#include "staticfilecontroller.h" +#include "controllers/dumpcontroller.h" +#include "controllers/templatecontroller.h" +#include "controllers/formcontroller.h" +#include "controllers/fileuploadcontroller.h" +#include "controllers/sessioncontroller.h" + +#include "controllers/librariescontroller.h" +#include "controllers/foldercontroller.h" +#include "controllers/covercontroller.h" +#include "controllers/comiccontroller.h" +#include "controllers/folderinfocontroller.h" +#include "controllers/pagecontroller.h" +#include "controllers/updatecomiccontroller.h" +#include "controllers/errorcontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +RequestMapper::RequestMapper(QObject* parent) + :HttpRequestHandler(parent) {} + +void RequestMapper::service(HttpRequest& request, HttpResponse& response) { + QByteArray path=request.getPath(); + qDebug("RequestMapper: path=%s",path.data()); + + QRegExp folder("/library/.+/folder/[0-9]+/?");//get comic content + QRegExp folderInfo("/library/.+/folder/[0-9]+/info/?"); //get folder info + QRegExp comic("/library/.+/comic/[0-9]+/?"); //get comic info + QRegExp comicOpen("/library/.+/comic/[0-9]+/remote/?"); //the server will open for reading the comic + QRegExp comicUpdate("/library/.+/comic/[0-9]+/update/?"); //get comic info + QRegExp comicClose("/library/.+/comic/[0-9]+/close/?"); //the server will close the comic and free memory + QRegExp cover("/library/.+/cover/[0-9a-f]+.jpg"); //get comic cover (navigation) + QRegExp comicPage("/library/.+/comic/[0-9]+/page/[0-9]+/?"); //get comic page + QRegExp comicPageRemote("/library/.+/comic/[0-9]+/page/[0-9]+/remote?"); //get comic page (remote reading) + + QRegExp library("/library/([0-9]+)/.+"); //permite verificar que la biblioteca solicitada existe + + path = QUrl::fromPercentEncoding(path).toLatin1(); + + //primera petición, se ha hecho un post, se sirven las bibliotecas si la seguridad mediante login no está habilitada + if(path == "/") + { + LibrariesController().service(request, response); + } + + else + { + + //se comprueba que la sesión sea la correcta con el fin de evitar accesos no autorizados + HttpSession session=Static::sessionStore->getSession(request,response,false); + if(!session.isNull() && session.contains("ySession")) + { + if(library.indexIn(path)!=-1 && DBHelper::getLibraries().contains(library.cap(1).toInt()) ) + { + //listar el contenido del folder + if(folder.exactMatch(path)) + { + FolderController().service(request, response); + } + else if (folderInfo.exactMatch(path)) + { + FolderInfoController().service(request, response); + } + else if(cover.exactMatch(path)) + { + CoverController().service(request, response); + } + else if(comic.exactMatch(path) || comicOpen.exactMatch(path)) + { + ComicController().service(request, response); + } + else if(comicPage.exactMatch(path) || comicPageRemote.exactMatch(path)) + { + PageController().service(request,response); + } + else if(comicUpdate.exactMatch(path)) + { + UpdateComicController().service(request, response); + } + } + else + { + //response.writeText(library.cap(1)); + Static::staticFileController->service(request, response); + } + } + else //acceso no autorizado, redirección + { + ErrorController(300).service(request,response); + } + } + +} diff --git a/YACReaderLibrary/server/requestmapper.h b/YACReaderLibrary/server/requestmapper.h new file mode 100644 index 00000000..d44586f6 --- /dev/null +++ b/YACReaderLibrary/server/requestmapper.h @@ -0,0 +1,36 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef REQUESTMAPPER_H +#define REQUESTMAPPER_H + +#include "httprequesthandler.h" + +/** + The request mapper dispatches incoming HTTP requests to controller classes + depending on the requested path. +*/ + +class RequestMapper : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(RequestMapper) +public: + + /** + Constructor. + @param parent Parent object + */ + RequestMapper(QObject* parent=0); + + /** + Dispatch a request to a controller. + @param request The received HTTP request + @param response Must be used to return the response + */ + void service(HttpRequest& request, HttpResponse& response); + +}; + +#endif // REQUESTMAPPER_H diff --git a/YACReaderLibrary/server/server.pri b/YACReaderLibrary/server/server.pri new file mode 100644 index 00000000..9e3b2920 --- /dev/null +++ b/YACReaderLibrary/server/server.pri @@ -0,0 +1,34 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += \ + $$PWD/static.h \ + $$PWD/startup.h \ + $$PWD/requestmapper.h \ + $$PWD/controllers/comiccontroller.h \ + $$PWD/controllers/errorcontroller.h \ + $$PWD/controllers/foldercontroller.h \ + $$PWD/controllers/folderinfocontroller.h \ + $$PWD/controllers/librariescontroller.h \ + $$PWD/controllers/pagecontroller.h \ + $$PWD/controllers/sessionmanager.h \ + $$PWD/controllers/covercontroller.h \ + server/controllers/updatecomiccontroller.h + +SOURCES += \ + $$PWD/static.cpp \ + $$PWD/startup.cpp \ + $$PWD/requestmapper.cpp \ + $$PWD/controllers/comiccontroller.cpp \ + $$PWD/controllers/errorcontroller.cpp \ + $$PWD/controllers/foldercontroller.cpp \ + $$PWD/controllers/folderinfocontroller.cpp \ + $$PWD/controllers/librariescontroller.cpp \ + $$PWD/controllers/pagecontroller.cpp \ + $$PWD/controllers/sessionmanager.cpp \ + $$PWD/controllers/covercontroller.cpp \ + server/controllers/updatecomiccontroller.cpp + +include(lib/bfLogging/bfLogging.pri) +include(lib/bfHttpServer/bfHttpServer.pri) +include(lib/bfTemplateEngine/bfTemplateEngine.pri) diff --git a/YACReaderLibrary/server/startup.cpp b/YACReaderLibrary/server/startup.cpp new file mode 100644 index 00000000..7166e402 --- /dev/null +++ b/YACReaderLibrary/server/startup.cpp @@ -0,0 +1,89 @@ +/** + @file + @author Stefan Frings +*/ + +#include "static.h" +#include "startup.h" +#include "dualfilelogger.h" +#include "httplistener.h" +#include "requestmapper.h" +#include "staticfilecontroller.h" + +#include "yacreader_global.h" + +#include +#include + +/** Name of this application */ +#define APPNAME "YACReaderLibrary" + +/** Publisher of this application */ +#define ORGANISATION "YACReader" + +/** Short description of this application */ +#define DESCRIPTION "Comic reader and organizer" + +void Startup::start() { + // Initialize the core application + QCoreApplication* app = QApplication::instance(); + app->setApplicationName(APPNAME); + app->setOrganizationName(ORGANISATION); + QString configFileName=YACReader::getSettingsPath()+"/"+QCoreApplication::applicationName()+".ini"; + + // Configure logging into files + QSettings* mainLogSettings=new QSettings(configFileName,QSettings::IniFormat,app); + mainLogSettings->beginGroup("mainLogFile"); + //QSettings* debugLogSettings=new QSettings(configFileName,QSettings::IniFormat,app); + //debugLogSettings->beginGroup("debugLogFile"); + Logger* logger=new FileLogger(mainLogSettings,10000,app); + logger->installMsgHandler(); + + // Configure template loader and cache + QSettings* templateSettings=new QSettings(configFileName,QSettings::IniFormat,app); + templateSettings->beginGroup("templates"); + Static::templateLoader=new TemplateCache(templateSettings,app); + + // Configure session store + QSettings* sessionSettings=new QSettings(configFileName,QSettings::IniFormat,app); + sessionSettings->beginGroup("sessions"); + Static::sessionStore=new HttpSessionStore(sessionSettings,app); + + // Configure static file controller + QSettings* fileSettings=new QSettings(configFileName,QSettings::IniFormat,app); + fileSettings->beginGroup("docroot"); + Static::staticFileController=new StaticFileController(fileSettings,app); + + // Configure and start the TCP listener + qDebug("ServiceHelper: Starting service"); + QSettings* listenerSettings=new QSettings(configFileName,QSettings::IniFormat,app); + listenerSettings->beginGroup("listener"); + listener = new HttpListener(listenerSettings,new RequestMapper(app),app); + + qDebug("ServiceHelper: Service has started"); +} + + +void Startup::stop() { + qDebug("ServiceHelper: Service has been stopped"); + // QCoreApplication destroys all objects that have been created in start(). + if(listener!=nullptr) + { + listener->close(); + delete listener; + listener = nullptr; + } +} + + +Startup::Startup() +{ + +} + +QString Startup::getPort() +{ + return QString("%1").arg(listener->serverPort()); +} + + diff --git a/YACReaderLibrary/server/startup.h b/YACReaderLibrary/server/startup.h new file mode 100644 index 00000000..1ad5ebbe --- /dev/null +++ b/YACReaderLibrary/server/startup.h @@ -0,0 +1,34 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef STARTUP_H +#define STARTUP_H + +#include + +class HttpListener; +/** + Helper class to install and run the application as a windows + service. +*/ +class Startup +{ +private: + //QTcpServer + HttpListener * listener; +public: + + /** Constructor */ + Startup(); + /** Start the server */ + void start(); + /** Stop the server */ + void stop(); + + QString getPort(); +protected: +}; + +#endif // STARTUP_H diff --git a/YACReaderLibrary/server/static.cpp b/YACReaderLibrary/server/static.cpp new file mode 100644 index 00000000..38133b66 --- /dev/null +++ b/YACReaderLibrary/server/static.cpp @@ -0,0 +1,63 @@ +/** + @file + @author Stefan Frings +*/ + +#include "static.h" +#include +#include +#include +#include + +QString Static::configDir=0; + +TemplateLoader* Static::templateLoader=0; + +HttpSessionStore* Static::sessionStore=0; + +StaticFileController* Static::staticFileController=0; + +QString Static::getConfigFileName() { + return QString("%1/%2.ini").arg(getConfigDir()).arg(QCoreApplication::applicationName()); +} + +QString Static::getConfigDir() { + if (!configDir.isNull()) { + return configDir; + } + // Search config file + #if defined Q_OS_UNIX && !defined Q_OS_MAC + QString binDir=(QString(DATADIR) + "/YACReader"); + #else + QString binDir=QCoreApplication::applicationDirPath(); + #endif + QString organization=QCoreApplication::organizationName(); + QString configFileName=QCoreApplication::applicationName()+".ini"; + + QStringList searchList; + searchList.append(QDir::cleanPath(binDir)); + searchList.append(QDir::cleanPath(binDir+"/../etc")); + searchList.append(QDir::cleanPath(binDir+"/../../etc")); // for development under windows + searchList.append(QDir::rootPath()+"etc/xdg/"+organization); + searchList.append(QDir::rootPath()+"etc/opt"); + searchList.append(QDir::rootPath()+"etc"); + + foreach (QString dir, searchList) { + QFile file(dir+"/"+configFileName); + if (file.exists()) { + // found + configDir=dir; + qDebug("Using config file %s",qPrintable(file.fileName())); + return configDir; + } + } + + // not found + foreach (QString dir, searchList) { + qWarning("%s/%s not found",qPrintable(dir),qPrintable(configFileName)); + } + qWarning("Cannot find config file %s",qPrintable(configFileName)); //TODO establecer los valores por defecto + + return 0; +} + diff --git a/YACReaderLibrary/server/static.h b/YACReaderLibrary/server/static.h new file mode 100644 index 00000000..74abf55b --- /dev/null +++ b/YACReaderLibrary/server/static.h @@ -0,0 +1,64 @@ +/** + @file + @author Stefan Frings +*/ + +#ifndef STATIC_H +#define STATIC_H + +#include +#include "templatecache.h" +#include "httpsessionstore.h" +#include "staticfilecontroller.h" + +/** + This class contains some static resources that are used by the application. +*/ + +class Static +{ +public: + + /** + Search the main config file and return its full path. + On the first call, the INI file gets searched. If not found, + the application aborts with an error message. +

+ The filename is the applications name plus the ending ".ini". It is searched + in the following directories: + + - Same directory as the applications executable file + - In ../etc relative to the applications executable file + - In ../../etc relative to the applications executable file + - In /etc/xdg/{organisation name} on the root drive + - In /etc/opt on the root drive + - In /etc on the root drive + + */ + static QString getConfigFileName(); + + /** + Gets the directory where the main config file is located. + On the first call, the INI file gets searched. If not found, + the application aborts with an error message. + @see getConfigFileName() + */ + static QString getConfigDir(); + + /** Cache for template files */ + static TemplateLoader* templateLoader; + + /** Storage for session cookies */ + static HttpSessionStore* sessionStore; + + /** Controller for static files */ + static StaticFileController* staticFileController; + +private: + + /** Directory of the main config file */ + static QString configDir; + +}; + +#endif // STATIC_H diff --git a/YACReaderLibrary/server_config_dialog.cpp b/YACReaderLibrary/server_config_dialog.cpp new file mode 100644 index 00000000..571dcc7f --- /dev/null +++ b/YACReaderLibrary/server_config_dialog.cpp @@ -0,0 +1,330 @@ +#include "server_config_dialog.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "startup.h" +#include "yacreader_global.h" + +#ifndef Q_OS_WIN32 + +#include +#include +#include +#include +#include + +QList addresses() +{ + struct ifaddrs * ifAddrStruct=NULL; + struct ifaddrs * ifa=NULL; + void * tmpAddrPtr=NULL; + + QList localAddreses; + + getifaddrs(&ifAddrStruct); + + for (ifa = ifAddrStruct; ifa != NULL; ifa = ifa->ifa_next) { + if (ifa ->ifa_addr->sa_family==AF_INET) { // check it is IP4 + // is a valid IP4 Address + tmpAddrPtr=&((struct sockaddr_in *)ifa->ifa_addr)->sin_addr; + char addressBuffer[INET_ADDRSTRLEN]; + inet_ntop(AF_INET, tmpAddrPtr, addressBuffer, INET_ADDRSTRLEN); + QString add(addressBuffer); + localAddreses.push_back(QString(addressBuffer)); + //printf("%s IP Address %s\n", ifa->ifa_name, addressBuffer); + } else if (ifa->ifa_addr->sa_family==AF_INET6) { // check it is IP6 + // is a valid IP6 Address + tmpAddrPtr=&((struct sockaddr_in6 *)ifa->ifa_addr)->sin6_addr; + char addressBuffer[INET6_ADDRSTRLEN]; + inet_ntop(AF_INET6, tmpAddrPtr, addressBuffer, INET6_ADDRSTRLEN); + //printf("%s IP Address %s\n", ifa->ifa_name, addressBuffer); + } + } + if (ifAddrStruct!=NULL) freeifaddrs(ifAddrStruct); + return localAddreses; +} + +#endif + +extern Startup * s; + +ServerConfigDialog::ServerConfigDialog(QWidget * parent) + :QDialog(parent) +{ + accept = new QPushButton(tr("set port"),this); + qrCodeImage = new QPixmap(); + qrCode = new QLabel(this); + qrCode->move(196,73); + qrCode->setFixedSize(200,200); + qrCode->setScaledContents(true); + + QLabel * title1 = new QLabel(tr("EASY SERVER CONNECTION"),this); + title1->move(37,28); + title1->setStyleSheet("QLabel {color:#1F1F1F; font-size:18px; font-family: Arial; font-weight: bold;}"); + + QLabel * title2 = new QLabel(tr("SERVER ADDRESS"),this); + title2->move(451,28); + title2->setStyleSheet("QLabel {color:#1F1F1F; font-size:18px; font-family: Arial; font-weight: bold;}"); + + QLabel * qrMessage = new QLabel(tr("just scan the code with your device!!"),this); + qrMessage->move(194,290);//373,627); + qrMessage->setStyleSheet("QLabel {color:#1F1F1F; font-size:16px; font-family: Arial; font-style: italic;}"); + qrMessage->setWordWrap(true); + qrMessage->setFixedWidth(200); + + QLabel * propaganda = new QLabel(tr("YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. Discover it! "),this); + propaganda->move(36,375); + propaganda->setStyleSheet("QLabel {color:#1F1F1F; font-size:16px; font-family: Arial; font-style: italic;}" + "QLabel::a {color:#1A1A1A}"); + propaganda->setWordWrap(true); + propaganda->setFixedWidth(590); + propaganda->setOpenExternalLinks(true); + + //FORM--------------------------------------------------------------------- + QWidget * form = new QWidget(this); + QFormLayout * formLayout = new QFormLayout; + + /*QLabel * ipLabel = new QLabel(tr("IP address"),this); + ipLabel->move(452,75); + ipLabel->setStyleSheet("QLabel {color:#1F1F1F; font-size:13px; font-family: Arial; font-weight: bold;}"); + + QLabel * portLabel = new QLabel(tr("Port"),this); + portLabel->move(452, 114); + portLabel->setStyleSheet("QLabel {color:#1F1F1F; font-size:13px; font-family: Arial; font-weight: bold;}");*/ + + ip = new QComboBox(this); + connect(ip,SIGNAL(activated(const QString &)),this,SLOT(regenerateQR(const QString &))); + //ip->move(520,71); +#ifndef Q_OS_WIN32 + ip->setStyleSheet("QComboBox{font-size:10px;}"); +#endif + ip->setSizeAdjustPolicy(QComboBox::AdjustToContents); +#ifdef Q_OS_WIN32 + ip->setMinimumWidth(120); +#else + ip->setFixedSize(120,ip->height()); +#endif + + port = new QLineEdit("8080",this); + port->setReadOnly(false); + port->setMaximumWidth(50); +#ifndef Q_OS_WIN32 + port->setStyleSheet("QLineEdit{font-size:10px;}"); +#endif + //port->move(520,110); + QValidator *validator = new QIntValidator(1024, 65535, this); + port->setValidator(validator); + + //accept->move(514,149); + connect(accept,SIGNAL(pressed()),this,SLOT(updatePort())); + + formLayout->addRow(tr("IP address"),ip); + formLayout->addRow(tr("Port"),port); + formLayout->addRow("",accept); + + form->setLayout(formLayout); +#ifdef Q_OS_WIN32 + form->move(444,70); +#else + form->move(435,70); +#endif + //END FORM----------------------------------------------------------------- + + check = new QCheckBox(this); + check->move(453,314); + check->setText(tr("enable the server")); + check->setStyleSheet("QCheckBox {color:#1F1F1F; font-size:13px; font-family: Arial; font-weight: bold;}"); + + //check->setLayoutDirection(Qt::RightToLeft); + + //elementsLayout->setSpacing(40); + //elementsLayout->addWidget(iphone); + //elementsLayout->addStretch(); + //elementsLayout->addLayout(configLayout); + + //QVBoxLayout * mainLayout = new QVBoxLayout; + //mainLayout->addLayout(elementsLayout); + //mainLayout->addLayout(buttons); + //mainLayout->addWidget(qrCode,0,1); + + //this->setLayout(mainLayout); + + QPalette Pal(palette()); + // set black background + QPalette palette; + QImage image(":/images/serverConfigBackground.png"); + palette.setBrush(this->backgroundRole(), QBrush(image)); + + setPalette(palette); + + this->setFixedSize(image.size()); + + QSettings * settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + + if(settings->value(SERVER_ON,true).toBool()) + { + check->setChecked(true); + generateQR(); + } + else + check->setChecked(false); + + settings->endGroup(); + + connect(check,SIGNAL(stateChanged(int)),this,SLOT(enableServer(int))); +} + +void ServerConfigDialog::enableServer(int status) +{ + QSettings * settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + + if(status == Qt::Checked) + { + s->start(); + this->generateQR(); + settings->setValue(SERVER_ON,true); + } + else + { + s->stop(); + qrCode->setPixmap(QPixmap()); + ip->clear(); + port->setText(""); + settings->setValue(SERVER_ON,false); + } + settings->endGroup(); +} + +void ServerConfigDialog::generateQR() +{ + //QString items; + //foreach(QNetworkInterface interface, QNetworkInterface::allInterfaces()) + //{ + // if (~interface.flags() & QNetworkInterface::IsLoopBack)//interface.flags().testFlag(QNetworkInterface::IsRunning)) + // foreach (QNetworkAddressEntry entry, interface.addressEntries()) + // { + // if ( interface.hardwareAddress() != "00:00:00:00:00:00" && entry.ip().toString().contains(".")) + // items.append(interface.name() + entry.ip().toString()); + // } + //} + ip->clear(); + QString dir; +#ifdef Q_OS_WIN32 + QList list = QHostInfo::fromName( QHostInfo::localHostName() ).addresses(); + + QList otherAddresses; + foreach(QHostAddress add, list) + { + QString tmp = add.toString(); + if(tmp.contains(".") && !tmp.startsWith("127")) + { + if(dir.isEmpty() && tmp.startsWith("192.168.2.")) + dir = tmp; + else + otherAddresses.push_back(tmp); + + } + } + +#else + QList list = addresses(); + + QList otherAddresses; + foreach(QString add, list) + { + QString tmp = add; + if(tmp.contains(".") && !tmp.startsWith("127")) + { + if(dir.isEmpty() && tmp.startsWith("192.168.2.")) + dir = tmp; + else + otherAddresses.push_back(tmp); + + } + } +#endif + if(otherAddresses.length()>0 || !dir.isEmpty()) + { + if(!dir.isEmpty()) + { + generateQR(dir+":"+s->getPort()); + + ip->addItem(dir); + } + else + { + generateQR(otherAddresses.first()+":"+s->getPort()); + } + ip->addItems(otherAddresses); + port->setText(s->getPort()); + } + else + { + + } + //qrCode->setText(dir+":8080"); +} + +void ServerConfigDialog::generateQR(const QString & serverAddress) +{ + qrCode->clear(); + qrGenerator = new QProcess(); + QStringList attributes; + attributes << "-o" << "-" /*QCoreApplication::applicationDirPath()+"/utils/tmp.png"*/ << "-s" << "8" << "-l" << "H" << "-m" << "0" << serverAddress; + connect(qrGenerator,SIGNAL(finished(int,QProcess::ExitStatus)),this,SLOT(updateImage(void))); + connect(qrGenerator,SIGNAL(error(QProcess::ProcessError)),this,SLOT(openingError(QProcess::ProcessError))); //TODO: implement openingError +#if defined Q_OS_UNIX && !defined Q_OS_MAC + qrGenerator->start(QString("qrencode"),attributes); +#else + qrGenerator->start(QCoreApplication::applicationDirPath()+"/utils/qrencode",attributes); +#endif +} + +void ServerConfigDialog::updateImage() +{ + QByteArray imgBinary = qrGenerator->readAllStandardOutput(); + //imgBinary = imgBinary.replace(0x0D0A,0x0A); + + if(!qrCodeImage->loadFromData(imgBinary)) + qrCode->setText(tr("QR generator error!")); + else + qrCode->setPixmap(*qrCodeImage); + + delete qrGenerator; + + + +/* qrCodeImage->load(QCoreApplication::applicationDirPath()+"/utils/tmp.png"); + qrCode->setPixmap(*qrCodeImage); + + delete qrGenerator;*/ +} + +void ServerConfigDialog::regenerateQR(const QString & ip) +{ + generateQR(ip+":"+s->getPort()); +} + +void ServerConfigDialog::updatePort() +{ + + QSettings * settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("listener"); + settings->setValue("port",port->text().toInt()); + settings->endGroup(); + + s->stop(); + s->start(); + + generateQR(ip->currentText()+":"+port->text()); + +} diff --git a/YACReaderLibrary/server_config_dialog.h b/YACReaderLibrary/server_config_dialog.h new file mode 100644 index 00000000..b21c96ee --- /dev/null +++ b/YACReaderLibrary/server_config_dialog.h @@ -0,0 +1,44 @@ +#ifndef __SERVER_CONFIG_DIALOG_H +#define __SERVER_CONFIG_DIALOG_H + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +class ServerConfigDialog : public QDialog +{ +Q_OBJECT + public: + ServerConfigDialog(QWidget * parent = 0); + private: + QComboBox * ip; + QLineEdit * port; + + QCheckBox * check; + + QPushButton * close; + QPushButton * accept; + QLabel * qrCode; + QPixmap * qrCodeImage; + + QProcess * qrGenerator; + public slots: + void generateQR(); + void generateQR(const QString & serverAddress); + void regenerateQR(const QString & ip); + void updateImage(); + void enableServer(int status); + void updatePort(); +signals: + void portChanged(QString port); + +}; + + +#endif \ No newline at end of file diff --git a/YACReaderLibrary/yacreader_libraries.cpp b/YACReaderLibrary/yacreader_libraries.cpp new file mode 100644 index 00000000..2e5e3cd2 --- /dev/null +++ b/YACReaderLibrary/yacreader_libraries.cpp @@ -0,0 +1,147 @@ +#include "yacreader_libraries.h" +#include "yacreader_global.h" + + + +YACReaderLibraries::YACReaderLibraries() + :QObject() +{ + +} + +YACReaderLibraries::YACReaderLibraries(const YACReaderLibraries &source) + :QObject(),libraries(source.libraries) +{ + +} + +QList YACReaderLibraries::getNames() +{ + return libraries.keys(); +} + +QString YACReaderLibraries::getPath(const QString &name) +{ + return libraries.value(name).second; +} + +QString YACReaderLibraries::getPath(int id) +{ + foreach(QString name, libraries.keys()) + if(libraries.value(name).first == id) + return libraries.value(name).second; + return ""; +} + +QString YACReaderLibraries::getName(int id) +{ + foreach(QString name, libraries.keys()) + if(libraries.value(name).first == id) + return name; + return ""; +} + +bool YACReaderLibraries::isEmpty() +{ + return libraries.isEmpty(); +} + +bool YACReaderLibraries::contains(const QString &name) +{ + return libraries.contains(name); +} + +bool YACReaderLibraries::contains(int id) +{ + foreach(QString name, libraries.keys()) + if(libraries.value(name).first == id) + return true; + return false; +} + +void YACReaderLibraries::remove(const QString &name) +{ + libraries.remove(name); +} + +void YACReaderLibraries::rename(const QString &oldName, const QString &newName) +{ + if(libraries.contains(oldName)) + { + QPair value = libraries.value(oldName); + libraries.remove(oldName); + libraries.insert(newName,value); + } +} + +int YACReaderLibraries::getId(const QString &name) +{ + return libraries.value(name).first; +} + +YACReaderLibraries &YACReaderLibraries::operator=(const YACReaderLibraries &source) +{ + libraries = source.libraries; + return *this; +} + +QMap > YACReaderLibraries::getLibraries() +{ + return libraries; +} + + +void YACReaderLibraries::addLibrary(const QString &name, const QString &path) +{ + int newID=0; + foreach(QString lName, libraries.keys()) + newID = qMax(newID,libraries.value(lName).first); + newID++; + libraries.insert(name,QPair(newID,path)); +} + +void YACReaderLibraries::load() +{ + QSettings settings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + + if(settings.value(LIBRARIES).isValid()) + { + QByteArray data = settings.value(LIBRARIES).toByteArray(); + QDataStream in(&data, QIODevice::ReadOnly); + in >> libraries; + } + else //only for compatibility with old versions (<7.0) + { + QFile f(QCoreApplication::applicationDirPath()+"/libraries.yacr"); + f.open(QIODevice::ReadOnly); + QTextStream txtS(&f); + QString content = txtS.readAll(); + QStringList lines = content.split('\n'); + QString line,name; + int i=0; + + foreach(line,lines) + { + if((i%2)==0) + name = line; + else + addLibrary(name.trimmed(),line.trimmed()); + i++; + } + f.close(); + if(save()) + f.remove(); + } +} + +bool YACReaderLibraries::save() +{ + QSettings settings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); + + QByteArray data; + QDataStream out(&data, QIODevice::WriteOnly); + out << libraries; + settings.setValue(LIBRARIES, data); + + return settings.isWritable(); +} diff --git a/YACReaderLibrary/yacreader_libraries.h b/YACReaderLibrary/yacreader_libraries.h new file mode 100644 index 00000000..5cc32a82 --- /dev/null +++ b/YACReaderLibrary/yacreader_libraries.h @@ -0,0 +1,34 @@ +#ifndef YACREADER_LIBRARIES_H +#define YACREADER_LIBRARIES_H + +#include + +class YACReaderLibraries : public QObject +{ + Q_OBJECT + +public: + YACReaderLibraries(); + YACReaderLibraries(const YACReaderLibraries & source); + QList getNames(); + QString getPath(const QString & name); + QString getPath(int id); + QString getName(int id); + bool isEmpty(); + bool contains(const QString & name); + bool contains(int id); + void remove(const QString & name); + void rename(const QString & oldName, const QString & newName); + int getId(const QString & name); + YACReaderLibraries & operator=(const YACReaderLibraries & source); + QMap > getLibraries(); +public slots: + void addLibrary(const QString & name, const QString & path); + void load(); + bool save(); +private: + //name + QMap > libraries; +}; + +#endif // YACREADER_LIBRARIES_H diff --git a/YACReaderLibrary/yacreader_local_server.cpp b/YACReaderLibrary/yacreader_local_server.cpp new file mode 100644 index 00000000..8d6d7c82 --- /dev/null +++ b/YACReaderLibrary/yacreader_local_server.cpp @@ -0,0 +1,218 @@ +#include "yacreader_local_server.h" + +#include +#include +#include + +#include "yacreader_global.h" +#include "db_helper.h" + +#include "comic_db.h" + +#include "QsLog.h" + +using namespace YACReader; + +QMutex YACReaderClientConnectionWorker::dbMutex; +//int YACReaderClientConnectionWorker::count = 0; +YACReaderLocalServer::YACReaderLocalServer(QObject *parent) : + QObject(parent) +{ + localServer = new QLocalServer(this); + QLocalServer::removeServer(YACREADERLIBRARY_GUID); + if (!localServer->listen(YACREADERLIBRARY_GUID)) { + QLOG_ERROR() << "Unable to create local server"; + } + + connect(localServer, SIGNAL(newConnection()), this, SLOT(sendResponse())); +} + +bool YACReaderLocalServer::isListening() +{ + return localServer->isListening(); +} + +/*void YACReaderLocalServer::run() +{ + while(1) + exec(); +}*/ + +void YACReaderLocalServer::sendResponse() +{ + QLocalSocket *clientConnection = localServer->nextPendingConnection(); + //connect(clientConnection, SIGNAL(disconnected()),clientConnection, SLOT(deleteLater())); + clientConnection->setParent(0); + + YACReaderClientConnectionWorker * worker = new YACReaderClientConnectionWorker(clientConnection); + if(worker != 0) + { + clientConnection->moveToThread(worker); + connect(worker,SIGNAL(comicUpdated(quint64, ComicDB)),this,SIGNAL(comicUpdated(quint64, ComicDB))); + connect(worker,SIGNAL(finished()),worker,SLOT(deleteLater())); + worker->start(); + } + + QLOG_INFO() << "connection incoming"; + //clientConnection->waitForBytesWritten();*/ + //clientConnection->disconnectFromServer(); +} + +bool YACReaderLocalServer::isRunning() +{ + QLocalSocket socket; + socket.connectToServer(YACREADERLIBRARY_GUID); + if (socket.waitForConnected(500)) + return true; // Server is running (another instance of YACReaderLibrary has been launched) + return false; +} + +void YACReaderLocalServer::close() +{ + localServer->close(); +} + + +YACReaderClientConnectionWorker::YACReaderClientConnectionWorker( QLocalSocket *cc) + :QThread(),clientConnection(cc) +{ + +} + +YACReaderClientConnectionWorker::~YACReaderClientConnectionWorker() +{ + +} +/*#include +#include +#include */ +void YACReaderClientConnectionWorker::run() +{ + /*{ + QFile f(QString("c:/temp/thread%1.txt").arg(count)); + f.open(QIODevice::Append); + QTextStream out(&f); + out << QString("Thread%1 starts").arg(count) << endl; + f.close(); + } + uint t1 = QDateTime::currentMSecsSinceEpoch();*/ + + quint64 libraryId; + ComicDB comic; + int tries = 0; + int dataAvailable = 0; + QByteArray packageSize; + clientConnection->waitForReadyRead(1000); + while(packageSize.size() < sizeof(quint32) && tries < 20) + { + packageSize.append(clientConnection->read(sizeof(quint32) - packageSize.size())); + clientConnection->waitForReadyRead(100); + if(dataAvailable == packageSize.size()) + { + tries++; + } + dataAvailable = packageSize.size(); + } + if(tries == 20) + { + QLOG_ERROR() << "Local connection: unable to read the message size" << clientConnection->errorString(); + return; + } + + QDataStream sizeStream(packageSize); + sizeStream.setVersion(QDataStream::Qt_4_8); + quint32 totalSize = 0; + sizeStream >> totalSize; + + tries = 0; + QByteArray data; + int dataRead = 0; + while(data.size() < totalSize && tries < 200) + { + data.append(clientConnection->readAll()); + if(data.length() < totalSize) + clientConnection->waitForReadyRead(100); + if(dataRead == data.length()) //no bytes were read + tries++; + dataRead = data.length(); + } + if(tries == 200) + { + QLOG_ERROR() << QString("Local connection: unable to read message (%1,%2)").arg(data.size()).arg(totalSize); + return; + } + QDataStream dataStream(data); + quint8 msgType; + dataStream >> msgType; + dataStream >> libraryId; + dataStream >> comic; + + switch (msgType) + { + case YACReader::RequestComicInfo: + { + QList siblings; + getComicInfo(libraryId,comic,siblings); + + QByteArray block; + QDataStream out(&block, QIODevice::WriteOnly); + out.setVersion(QDataStream::Qt_4_8); + out << (quint32)0; + out << comic; + out << siblings; + out.device()->seek(0); + out << (quint32)(block.size() - sizeof(quint32)); + + int written = 0; + tries = 0; + while(written != block.size() && tries < 200) + { + int ret = clientConnection->write(block); + clientConnection->waitForBytesWritten(10); + if(ret != -1) + { + written += ret; + clientConnection->flush(); + } + else + tries++; + } + if(tries == 200 && written != block.size()) + QLOG_ERROR() << QString("Local connection (comic info requested): unable to send response (%1,%2)").arg(written).arg(block.size()); + break; + } + case YACReader::SendComicInfo: + { + updateComic(libraryId,comic); + //clientConnection->disconnectFromServer(); + break; + } + + } + + clientConnection->waitForDisconnected(); + clientConnection->deleteLater(); + /*count++; + uint t2 = QDateTime::currentMSecsSinceEpoch(); + { + QFile f(QString("c:/temp/thread%1.txt").arg(count)); + f.open(QIODevice::Append); + QTextStream out(&f); + out << QString("Thread%1 ends : time - %2").arg(count).arg(t2-t1) << endl; + f.close(); + }*/ +} + +void YACReaderClientConnectionWorker::getComicInfo(quint64 libraryId, ComicDB & comic, QList & siblings) +{ + QMutexLocker locker(&dbMutex); + comic = DBHelper::getComicInfo(DBHelper::getLibrariesNames().at(libraryId), comic.id); + siblings = DBHelper::getSiblings(DBHelper::getLibrariesNames().at(libraryId), comic.parentId); +} + +void YACReaderClientConnectionWorker::updateComic(quint64 libraryId, ComicDB & comic) +{ + QMutexLocker locker(&dbMutex); + DBHelper::update(DBHelper::getLibrariesNames().at(libraryId), comic.info); + emit comicUpdated(libraryId, comic); +} diff --git a/YACReaderLibrary/yacreader_local_server.h b/YACReaderLibrary/yacreader_local_server.h new file mode 100644 index 00000000..d5432e60 --- /dev/null +++ b/YACReaderLibrary/yacreader_local_server.h @@ -0,0 +1,50 @@ +#ifndef YACREADER_LOCAL_SERVER_H +#define YACREADER_LOCAL_SERVER_H + +#include +#include +#include + +class QLocalServer; +class QLocalSocket; +class ComicDB; + +class YACReaderLocalServer : public QObject +{ + Q_OBJECT +public: + explicit YACReaderLocalServer(QObject *parent = 0); + +signals: + void comicUpdated(quint64 libraryId, const ComicDB & comic); +public slots: + bool isListening(); + void sendResponse(); + static bool isRunning(); + void close(); +private: + //void run(); + QLocalServer * localServer; + +}; + +class YACReaderClientConnectionWorker : public QThread +{ + Q_OBJECT +public: + YACReaderClientConnectionWorker( QLocalSocket *clientConnection); + ~YACReaderClientConnectionWorker(); +signals: + void comicUpdated(quint64 libraryId, const ComicDB & comic); +private: + static QMutex dbMutex; + //static int count; + void run(); + + void getComicInfo(quint64 libraryId, ComicDB & comic, QList & sibling); + void updateComic(quint64 libraryId, ComicDB & comic); + + QLocalSocket *clientConnection; +}; + +#endif // YACREADER_LOCAL_SERVER_H diff --git a/YACReaderLibrary/yacreader_main_toolbar.cpp b/YACReaderLibrary/yacreader_main_toolbar.cpp new file mode 100644 index 00000000..c4e1dc82 --- /dev/null +++ b/YACReaderLibrary/yacreader_main_toolbar.cpp @@ -0,0 +1,146 @@ +#include "yacreader_main_toolbar.h" + +#include +#include +#include +#include +#include +#include +#include + +YACReaderMainToolBar::YACReaderMainToolBar(QWidget *parent) : + QWidget(parent) +{ + mainLayout = new QHBoxLayout; + + currentFolder = new QLabel(this); + //currentFolder->setAlignment(Qt::AlignCenter); + currentFolder->setStyleSheet(" QLabel {color:#404040; font-size:22px; font-weight:bold;}"); + + QFont f=currentFolder->font(); + f.setStyleStrategy(QFont::PreferAntialias); + currentFolder->setFont(f); + + QString qToolButtonStyleSheet = "QToolButton {border:none;}"; + + backButton = new QToolButton(); + backButton->setStyleSheet(qToolButtonStyleSheet); + + + forwardButton = new QToolButton(); + forwardButton->setStyleSheet(qToolButtonStyleSheet); + forwardButton->setDisabled(true); + + settingsButton = new QToolButton(); + settingsButton->setStyleSheet(qToolButtonStyleSheet); + settingsButton->setIconSize(QSize(24,24)); + + serverButton = new QToolButton(); + serverButton->setStyleSheet(qToolButtonStyleSheet); + serverButton->setIconSize(QSize(17,24)); + + + helpButton = new QToolButton(); + helpButton->setStyleSheet(qToolButtonStyleSheet); + helpButton->setIconSize(QSize(14,25)); + + toggleComicsViewButton = new QToolButton; + toggleComicsViewButton->setStyleSheet(qToolButtonStyleSheet); + toggleComicsViewButton->setIconSize(QSize(24,24)); + + fullscreenButton = new QToolButton(); + fullscreenButton->setStyleSheet(qToolButtonStyleSheet); + fullscreenButton->setIconSize(QSize(24,24)); + + mainLayout->setMargin(0); + mainLayout->setSpacing(0); + + mainLayout->addSpacing(12); + mainLayout->addWidget(backButton); + addDivider(); + mainLayout->addWidget(forwardButton); + + mainLayout->addSpacing(34); + mainLayout->addWidget(settingsButton); + addWideDivider(); + mainLayout->addWidget(serverButton); + addWideDivider(); + mainLayout->addWidget(helpButton); + + mainLayout->addStretch(); + + mainLayout->addWidget(toggleComicsViewButton); + addWideDivider(); + mainLayout->addWidget(fullscreenButton); + mainLayout->addSpacing(10); + + setLayout(mainLayout); + + setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Fixed); +} + + +QSize YACReaderMainToolBar::sizeHint() const +{ + return QSize(200,40); +} + +void YACReaderMainToolBar::paintEvent(QPaintEvent * event) +{ + Q_UNUSED(event); + + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor("#F0F0F0")); +} + +void YACReaderMainToolBar::resizeEvent(QResizeEvent * event) +{ + //210px x 2 = 420px + int freeWidth = event->size().width() - 420; + int maxLabelWidth = freeWidth>=0?freeWidth:0; + currentFolder->setMaximumWidth(maxLabelWidth); + currentFolder->adjustSize(); + + QFontMetrics metrix(currentFolder->font()); + QString clippedText = metrix.elidedText(currentFolderName, Qt::ElideRight, maxLabelWidth); + + currentFolder->setText(clippedText); + currentFolder->adjustSize(); + currentFolder->move((event->size().width()-currentFolder->width())/2,(event->size().height()-currentFolder->height())/2); +} + +void YACReaderMainToolBar::addDivider() +{ + QPixmap img(":/images/main_toolbar/divider.png"); + QLabel * divider = new QLabel(); + divider->setPixmap(img); + + mainLayout->addSpacing(5); + mainLayout->addWidget(divider); + mainLayout->addSpacing(5); +} + +void YACReaderMainToolBar::addWideDivider() +{ + mainLayout->addSpacing(3); + addDivider(); + mainLayout->addSpacing(3); +} + +void YACReaderMainToolBar::setCurrentFolderName(const QString & name) +{ + currentFolder->setText(name); + currentFolderName = name; + currentFolder->adjustSize(); + + int freeWidth = size().width() - 420; + int maxLabelWidth = freeWidth>=0?freeWidth:0; + currentFolder->setMaximumWidth(maxLabelWidth); + + QFontMetrics metrix(currentFolder->font()); + QString clippedText = metrix.elidedText(currentFolderName, Qt::ElideRight, maxLabelWidth); + + currentFolder->setText(clippedText); + currentFolder->adjustSize(); + currentFolder->move((width()-currentFolder->width())/2,(height()-currentFolder->height())/2); +} diff --git a/YACReaderLibrary/yacreader_main_toolbar.h b/YACReaderLibrary/yacreader_main_toolbar.h new file mode 100644 index 00000000..a21fd3f5 --- /dev/null +++ b/YACReaderLibrary/yacreader_main_toolbar.h @@ -0,0 +1,51 @@ +#ifndef YACREADER_MAIN_TOOLBAR_H +#define YACREADER_MAIN_TOOLBAR_H + +#include + +class QToolButton; +class QLabel; +class QResizeEvent; +class QPaintEvent; +class QHBoxLayout; + +//TODO create methods for adding actions, separators and sctreches dynimically +class YACReaderMainToolBar : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderMainToolBar(QWidget *parent = 0); + QSize sizeHint() const; + + QToolButton * backButton; + QToolButton * forwardButton; + QToolButton * settingsButton; + QToolButton * serverButton; + QToolButton * helpButton; + QToolButton * toggleComicsViewButton; + QToolButton * fullscreenButton; + + + void setCurrentFolderName(const QString & name); +signals: + +public slots: + +private: + void paintEvent(QPaintEvent *); + void resizeEvent(QResizeEvent *); + + + + QHBoxLayout * mainLayout; + + QLabel * currentFolder; + QString currentFolderName; + + void addDivider(); + void addWideDivider(); + + +}; + +#endif // YACREADER_MAIN_TOOLBAR_H diff --git a/YACReaderLibrary/yacreaderlibrary_es.qm b/YACReaderLibrary/yacreaderlibrary_es.qm new file mode 100644 index 0000000000000000000000000000000000000000..370934a55e10ff24e8cb4f7d6cce02461a553dfb GIT binary patch literal 35032 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 + + + + AddLibraryDialog + + + Comics folder : + Carpeta de cómics: + + + + Library Name : + Nombre de la biblioteca: + + + + Add + Añadir + + + + Cancel + Cancelar + + + + Add an existing library + Añadir una biblioteca existente + + + + ComicVineDialog + + + skip + omitir + + + + back + atrás + + + + next + siguiente + + + + search + buscar + + + + close + cerrar + + + + + + + + Looking for volume... + Buscando volumen... + + + + + comic %1 of %2 - %3 + cómic %1 de %2 - %3 + + + + %1 comics selected + %1 comics seleccionados + + + + Error connecting to ComicVine + Error conectando a ComicVine + + + + unknown error + error desconocido + + + + + Retrieving tags for : %1 + Recuperando etiquetas para : %1 + + + + Retrieving volume info... + Recuperando imformación del volumen... + + + + Looking for comic... + Buscando cómic... + + + + CreateLibraryDialog + + + Comics folder : + Carpeta de cómics: + + + + Library Name : + Nombre de la biblioteca: + + + + Create + Crear + + + + Cancel + Cancelar + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Crear una biblioteca puede llevar varios minutos. Puedes parar el proceso en cualquier momento y completar la tarea más tarde. + + + + Create new library + Crear la nueva biblioteca + + + + Path not found + Ruta no encontrada + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + La ruta seleccionada no existe o no es válida. Asegúrate de que tienes privilegios de escritura en esta carpeta + + + + ExportComicsInfoDialog + + + Output file : + Archivo de salida : + + + + Create + Crear + + + + Cancel + Cancelar + + + + Export comics info + Exportar información de los cómics + + + + Destination database name + Nombre de la base de datos de destino + + + + Problem found while writing + Problema encontrado mientras se escribía + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + La ruta seleccionada para el archivo de salida no existe o no es una ruta válida. Asegúrate de que tienes permisos de escritura en esta carpeta + + + + ExportLibraryDialog + + + Output folder : + Carpeta de destino: + + + + Create + Crear + + + + Cancel + Cancelar + + + + Create covers package + Crear paquete de portadas + + + + Problem found while writing + Problema encontrado mientras se escribía + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + La ruta seleccionada para el archivo de salida no existe o no es una ruta válida. Asegúrate de que tienes permisos de escritura en esta carpeta + + + + Destination directory + Carpeta de destino + + + + FileComic + + + Unknown error opening the file + Error desconocido abriendo el archivo + + + + 7z not found + 7z no encontrado + + + + Format not supported + Formato no soportado + + + + CRC error on page (%1): some of the pages will not be displayed correctly + Error CRC en la página (%1): algunas de las páginas no se mostrarán correctamente + + + + HelpAboutDialog + + + About + Acerca de + + + + Help + Ayuda + + + + ImportComicsInfoDialog + + + Import comics info + Importar información de cómics + + + + Info database location : + Ubicación de la base de datos de información : + + + + Import + Importar + + + + Cancel + Cancelar + + + + Comics info file (*.ydb) + Archivo de información de cómics (*.ydb) + + + + ImportLibraryDialog + + + Library Name : + Nombre de la biblioteca : + + + + Package location : + Ubicación del paquete: + + + + Destination folder : + Directorio de destino: + + + + Unpack + Desempaquetar + + + + Cancel + Cancelar + + + + Extract a catalog + Extraer un catálogo + + + + Compresed library covers (*.clc) + Compresed library covers (*.clc) + + + + ImportWidget + + + Importing comics + Importando cómics + + + + stop + parar + + + + Some of the comics being added... + Algunos de los cómics que estan siendo añadidos.... + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + <p>YACReaderLibrary está creando una nueva biblioteca.</p><p>Crear una biblioteca puede llevar varios minutos. Puedes parar el proceso en cualquier momento y actualizar la biblioteca más tarde para completar el proceso.</p> + + + + Updating the library + Actualizando la biblioteca + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later. + <p>La biblioteca actual está siendo actualizada. Para actualizaciones más rápidas, por favor, actualiza tus bibliotecas frecuentemente.</p><p>Puedes parar el proceso y continunar la actualización más tarde.</p> + + + + LibraryWindow + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> presiona 'F' para salir de pantalla completa </font> + + + + YACReader Library + YACReader Library + + + + Create a new library + Crear una nueva biblioteca + + + + Open an existing library + Abrir una biblioteca existente + + + + + Export comics info + Exportar información de los cómics + + + + + Import comics info + Importar información de cómics + + + + Pack covers + Empaquetar portadas + + + + Pack the covers of the selected library + Empaquetar las portadas de la biblioteca seleccionada + + + + Unpack covers + Desempaquetar portadas + + + + Unpack a catalog + Desempaquetar un catálogo + + + + Update library + Actualizar biblioteca + + + + Update current library + Actualizar la biblioteca seleccionada + + + + Rename library + Renombrar biblioteca + + + + Rename current library + Renombrar la biblioteca seleccionada + + + + Remove current library from your collection + Eliminar biblioteca de la colección + + + + Open current comic + Abrir cómic actual + + + + Open current comic on YACReader + Abrir el cómic actual en YACReader + + + + + Set as read + Marcar como leído + + + + Set comic as read + Marcar cómic como leído + + + + + Set as unread + Marcar como no leído + + + + Set comic as unread + Marcar cómic como no leído + + + + Show/Hide marks + Mostrar/Ocultar marcas + + + + Show or hide readed marks + Mostrar u ocultar marcas + + + + Fullscreen mode on/off + Modo a pantalla completa on/off + + + + Fullscreen mode on/off (F) + Activar/desactivar modo a pantalla completa (F) + + + + Help, About YACReader + Ayuda, A cerca de... YACReader + + + + Select root node + Seleccionar el nodo raíz + + + + + + + + + + + Expand all nodes + Expandir todos los nodos + + + + - + - + + + + Colapse all nodes + Contraer todos los nodos + + + + Show options dialog + Mostrar opciones + + + + Show comics server options dialog + + + + + Open folder... + Abrir carpeta... + + + + Set as uncompleted + Marcar como incompleto + + + + Set as completed + Marcar como completo + + + + Open containing folder... + Abrir carpeta contenedora... + + + + Reset comic rating + Reseteal cómic rating + + + + Select all comics + Seleccionar todos los cómics + + + + Edit + Editar + + + + Asign current order to comics + Asignar el orden actual a los cómics + + + + Update cover + Actualizar portada + + + + Delete selected comics + Borrar los cómics seleccionados + + + + Hide comic flow + Ocultar cómic flow + + + + Download tags from Comic Vine + Descargar etiquetas de Comic Vine + + + + Folder + Carpeta + + + + Comic + Cómic + + + + Library not available + Library ' + Biblioteca no disponible + + + + Library '%1' is no longer available. Do you want to remove it? + La biblioteca '%1' no está disponible. ¿Deseas eliminarla? + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + La biblioteca '%1' ha sido creada con una versión más antigua de YACReaderLibrary y debe ser creada de nuevo. ¿Deseas crear la biblioteca ahora? + + + + Old library + Biblioteca antigua + + + + YACReader not found + YACReader no encontrado + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + YACReader no encontrado, YACReader debe estar instalado en el mismo directorio que YACReaderLibrary. + + + + Unable to delete + No se ha podido borrar + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + Ha habido algún problema intentando borrar los cómics selecionados. Por favor, verifica los permisos de escritura en los arhicovs seleccionados o los directorios que los conienen. + + + + Error creating the library + Errar creando la biblioteca + + + + Error updating the library + Error actualizando la biblioteca + + + + Error opening the library + Error abriendo la biblioteca + + + + Delete comics + Borrar cómics + + + + All the selected comics will be deleted from your disk. Are you sure? + Todos los cómics seleccionados serán borrados de tu disco. ¿Estás seguro? + + + + Library name already exists + Ya existe el nombre de la biblioteca + + + + There is another library with the name '%1'. + Hay otra biblioteca con el nombre '%1'. + + + + + Library + Librería + + + + Remove library + Eliminar biblioteca + + + + Update needed + Se necesita actualizar + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Esta biblioteca fue creada con una versión anterior de YACReaderLibrary. Es necesario que se actualice. ¿Deseas hacerlo ahora? + + + + Update failed + La actualización ha fallado + + + + The current library can't be udpated. Check for write write permissions on: + La librería actual no ha podido ser actualizada. Verifica que posees permisos de escritura en: + + + + Download new version + Descargar la nueva versión + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Esta biblioteca fue creada con una versión más nueva de YACReaderLibrary. ¿Deseas descargar la nueva versión ahora? + + + + Library not found + Biblioteca no encontrada + + + + The selected folder doesn't contain any library. + La carpeta seleccionada no contiene ninguna biblioteca. + + + + Are you sure? + ¿Estás seguro? + + + + library? + ? + + + + Remove and delete metadata + Eliminar y borrar metadatos + + + + Do you want remove + ¿Deseas eliminar la biblioteca + + + + Asign comics numbers + Asignar números de cómic + + + + Asign numbers starting in: + Asignar números empezando en: + + + + LocalComicListModel + + + file name + nombre de archivo + + + + NoLibrariesWidget + + + You don't have any librarires yet + Aún no tienes ninguna biblioteca + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + <p>Puedes crear una biblioteca en cualquier carpeta, YACReaderLibrary importará todos las carpetas y cómics de esa carpeta. Si has creado alguna biblioteca anteriormente, puedes abrirla sin volver a crearla.</p><p>No olvides que puedes usar YACReader como una aplicación independiente para leer los cómics en tu ordenador.</p> + + + + create your first library + crea tu primera biblioteca + + + + add an existing one + añade una existente + + + + OptionsDialog + + + Options + Opciones + + + + PropertiesDialog + + + General info + Información general + + + + Authors + Autores + + + + Publishing + Publicación + + + + Plot + Argumento + + + + Cover page + Página de portada + + + + Title: + Título: + + + + Issue number: + Número: + + + + Volume: + Volumen: + + + + Story arc: + Arco argumental: + + + + Genere: + Género: + + + + Size: + Tamaño: + + + + Writer(s): + Guionista(s): + + + + Penciller(s): + Dibujant(es): + + + + Inker(s): + Entintador(es): + + + + Colorist(s): + Color: + + + + Letterer(s): + Letterer(es): + Rotulista(s): + + + + Cover Artist(s): + Artista(s) portada: + + + + Day: + Día: + + + + Month: + Mes: + + + + Year: + Año: + + + + Publisher: + Editorial: + + + + Format: + Formato: + + + + Color/BW: + Color/BN: + + + + Age rating: + Casificación edades: + + + + Synopsis: + Sinopsis: + + + + Characters: + Personajes: + + + + Notes: + Notas: + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> ver </a> + + + + Not found + No encontrado + + + + Comic not found. You should update your library. + Cómic no encontrado. Deberias actualizar tu biblioteca. + + + + Edit selected comics information + Editar la información de los cómics seleccionados + + + + Edit comic information + Editar la información del cócmic + + + + QObject + + + 7z lib not found + 7z lib no encontrado + + + + unable to load 7z lib from ./utils + imposible cargar 7z lib de ./utils + + + + RenameLibraryDialog + + + New Library Name : + Nuevo nombre de la biblioteca : + + + + Rename + Renombrar + + + + Cancel + Cancelar + + + + Rename current library + Renombrar la biblioteca seleccionada + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + Número de volúmenes encontrados : %1 + + + + + page %1 of %2 + página %1 de %2 + + + + Number of %1 found : %2 + Número de %1 encontrados : %2 + + + + SearchSingleComic + + + Please provide some aditional information. + Por favor, proporciona alguna información adicional. + + + + Series: + Series: + + + + SearchVolume + + + Please provide some aditional information. + Por favor, proporciona alguna informacion adicional. + + + + Series: + Series: + + + + SelectComic + + + Please, select the right comic info. + Por favor, selecciona la información correcta. + + + + comics + cómics + + + + loading cover + cargando portada + + + + loading description + cargando descripción + + + + description unavailable + descripción no disponible + + + + SelectVolume + + + Please, select the right series for your comic. + Por favor, seleciona la serie correcta para tu cómic. + + + + volumes + volúmenes + + + + loading cover + cargando portada + + + + loading description + cargando descripción + + + + description unavailable + descripción no disponible + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + Estás intentando obtener información de varios cómics a la vez, ¿son parte de la misma serie? + + + + yes + sí + + + + no + no + + + + ServerConfigDialog + + + set port + fijar puerto + + + + EASY SERVER CONNECTION + CONEXIÓN AL SERVIDOR FÃCILMENTE + + + + SERVER ADDRESS + DATOS SERVIDOR + + + + just scan the code with your device!! + ¡simplemente escanea el código con tu dispositivo! + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com'> Discover it! </a> + YACReader está ahora disponible para dispositivos iOS, la mejor experiencia de lectura de cómics ahora en tu iPad, iPhone o iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> ¡Descúbrelo! </a> + + + + IP address + IP usada + + + + Port + Puerto + + + + enable the server + activar el servidor + + + + QR generator error! + ¡Error del generador QR! + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + Por favor, ordena la lista de cómics en la izquiera hasta que coincida con la información adecuada. + + + + sort comics to match comic information + ordena los cómics para coincidir con la información + + + + issues + números + + + + remove selected comics + eliminar cómics seleccionados + + + + restore all removed comics + restaurar todos los cómics eliminados + + + + restore removed comics + restaurar cómics eliminados + + + + TableModel + + + yes + sí + + + + no + no + + + + Title + Título + + + + File Name + Nombre de archivo + + + + Pages + Páginas + + + + Size + Tamaño + + + + Read + Leído + + + + Current Page + Página Actual + + + + Rating + Nota + + + + TitleHeader + + + SEARCH + BUSCAR + + + + UpdateLibraryDialog + + + Updating.... + Actualizado... + + + + Cancel + Cancelar + + + + Update library + Actualizar biblioteca + + + + VolumeComicsModel + + + title + título + + + + VolumesModel + + + year + año + + + + issues + números + + + + publisher + editor + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + Borrando, por favor espera... + + + + cancel + cancelar + + + + YACReaderFieldEdit + + + + Click to overwrite + Click para sobreescribir + + + + Restore to default + Restaurar valor por defecto + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Click para sobreescribir + + + + Restore to default + Restaurar valor por defecto + + + + YACReaderFlowConfigWidget + + + How to show covers: + Cómo mostrar las portadas: + + + + CoverFlow look + Tipo CoverFlow + + + + Stripe look + Tipo tira + + + + Overlapped Stripe look + Tipo tira solapada + + + + YACReaderGLFlowConfigWidget + + + Presets: + Predeterminados: + + + + Classic look + Tipo clásico + + + + Stripe look + Tipo tira + + + + Overlapped Stripe look + Tipo tira solapada + + + + Modern look + Tipo moderno + + + + Roulette look + Tipo ruleta + + + + Show advanced settings + Opciones avanzadas + + + + Custom: + Personalizado: + + + + View angle + Ãngulo de vista + + + + Position + Posición + + + + Cover gap + Hueco entre portadas + + + + Central gap + Hueco central + + + + Zoom + Zoom + + + + Y offset + Desplazamiento en Y + + + + Z offset + Desplazamiento en Z + + + + Cover Angle + Ãngulo de las portadas + + + + Visibility + Visibilidad + + + + Light + Luz + + + + Max angle + Ãngulo máximo + + + + Low Performance + Rendimiento bajo + + + + High Performance + Alto rendimiento + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Usar VSync (mejora la calidad de imagen en pantalla completa, peor rendimiento) + + + + Performance: + Rendimiento: + + + + YACReaderOptionsDialog + + + Save + Guardar + + + + Cancel + Cancelar + + + + Use hardware acceleration (restart needed) + Usar aceleración por hardware (necesario reiniciar) + + + + YACReaderSideBar + + + LIBRARIES + BIBLIOTECAS + + + + FOLDERS + CARPETAS + + + + Search folders and comics + Buscar carpetas y cómics + + + diff --git a/YACReaderLibrary/yacreaderlibrary_fr.ts b/YACReaderLibrary/yacreaderlibrary_fr.ts new file mode 100644 index 00000000..2851cd6f --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_fr.ts @@ -0,0 +1,1517 @@ + + + + + AddLibraryDialog + + + Comics folder : + Dossier des comics : + + + + Library Name : + Nom de la librairie : + + + + Add + Ajouter + + + + Cancel + Annuler + + + + Add an existing library + Ajouter une librairie existante + + + + ComicVineDialog + + + skip + + + + + back + + + + + next + + + + + search + + + + + close + + + + + + + + + Looking for volume... + + + + + + comic %1 of %2 - %3 + + + + + %1 comics selected + + + + + Error connecting to ComicVine + + + + + unknown error + + + + + + Retrieving tags for : %1 + + + + + Retrieving volume info... + + + + + Looking for comic... + + + + + CreateLibraryDialog + + + Comics folder : + Dossier des comics : + + + + Library Name : + Nom de la librairie : + + + + Create + Créer + + + + Cancel + Annuler + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + La création d'une librairie peut prendre quelques minutes. Vous pouvez arrêter le processus et continuer plus tard. + + + + Create new library + Créer une nouvelle librairie + + + + Path not found + Chemin introuvable + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + Le chemin sélectionné n'existe pas ou contient un chemin invalide. Assurez-vous d'avoir les droits d'accès à ce dossier + + + + ExportComicsInfoDialog + + + Output file : + Fichier de sortie : + + + + Create + Créer + + + + Cancel + Annuler + + + + Export comics info + Exporter les infos des comics + + + + Destination database name + Nom de la base de données de destination + + + + Problem found while writing + Problème durant l'écriture + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Le chemin sélectionné pour le fichier n'existe pas ou contient un chemin invalide. Assurez-vous d'avoir les droits d'accès à ce dossier + + + + ExportLibraryDialog + + + Output folder : + Dossier de sortie : + + + + Create + Créer + + + + Cancel + Annuler + + + + Create covers package + Créer un pack de couvertures + + + + Problem found while writing + Problème durant l'écriture + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Le chemin sélectionné pour le fichier n'existe pas ou contient un chemin invalide. Assurez-vous d'avoir les droits d'accès à ce dossier + + + + Destination directory + Répertoire de destination + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z introuvable + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + HelpAboutDialog + + + About + A propos + + + + Help + Aide + + + + ImportComicsInfoDialog + + + Import comics info + Importer les infos des comics + + + + Info database location : + Emplacement des infos: + + + + Import + Importer + + + + Cancel + Annuler + + + + Comics info file (*.ydb) + Comics info file (*.ydb) + + + + ImportLibraryDialog + + + Library Name : + Nom de la librairie : + + + + Package location : + Emplacement : + + + + Destination folder : + Dossier de destination : + + + + Unpack + Désarchiver + + + + Cancel + Annuler + + + + Extract a catalog + Extraire un catalogue + + + + Compresed library covers (*.clc) + Compresed library covers (*.clc) + + + + ImportWidget + + + stop + Stop + + + + Some of the comics being added... + Ajout de comics... + + + + Importing comics + Importation de comics + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + <p>YACReaderLibrary est en train de créer une nouvelle librairie.</p><p>La création d'une librairie peut prendre quelques minutes. Vous pouvez arrêter le processus et poursuivre plus tard.</p> + + + + Updating the library + Mise à jour de la librairie + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <p>Mise à jour de la librairie. Pour plus de rapidité lors de la mise à jour, veuillez effectuer cette dernière régulièrement.</p><p>Vous pouvez arrêter le processus et poursuivre plus tard.</p> + + + + LibraryWindow + + + YACReader Library + Librairie de YACReader + + + + + Library + Librairie + + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> appuyez sur 'F' pour quitter le mode plein écran </font> + + + + Create a new library + Créer une nouvelle librairie + + + + Open an existing library + Ouvrir une librairie existante + + + + + Export comics info + Exporter les infos des comics + + + + + Import comics info + Importer les infos des comics + + + + Pack covers + Archiver les couvertures + + + + Pack the covers of the selected library + Archiver les couvertures de la librairie sélectionnée + + + + Unpack covers + Désarchiver les couvertures + + + + Unpack a catalog + Désarchiver un catalogue + + + + Update library + Mettre la librairie à jour + + + + Update current library + Mettre à jour la librairie actuelle + + + + Rename library + Renommer la librairie + + + + Rename current library + Renommer la librairie actuelle + + + + Remove library + Supprimer la librairie + + + + Remove current library from your collection + Enlever cette librairie de votre collection + + + + Open current comic + Ouvrir ce comic + + + + Open current comic on YACReader + Ouvrir ce comic dans YACReader + + + + + Set as read + Marquer comme lu + + + + Set comic as read + Marquer ce comic comme lu + + + + + Set as unread + Marquer comme non-lu + + + + Set comic as unread + Marquer ce comic comme non-lu + + + + Show/Hide marks + Afficher/Cacher les marqueurs + + + + Show or hide readed marks + Afficher ou cacher les marqueurs pour les livres lus + + + + Library not available + Library ' + Librairie non disponible + + + + Fullscreen mode on/off + Mode plein écran activé/désactivé + + + + Fullscreen mode on/off (F) + Mode plein écran activé/désactivé (F) + + + + Help, About YACReader + Aide, à propos de YACReader + + + + Select root node + Allerà la racine + + + + + + + + + + + Expand all nodes + Afficher tous les noeuds + + + + - + - + + + + Colapse all nodes + Masquer tous les noeuds + + + + Show options dialog + Ouvrir la boite de dialogue + + + + Show comics server options dialog + Ouvrir la boite de dialogue du serveur + + + + Open folder... + Ouvrir le dossier... + + + + Set as uncompleted + + + + + Set as completed + + + + + Open containing folder... + Ouvrir le dossier... + + + + Reset comic rating + + + + + Select all comics + Sélectionner tous les comics + + + + Edit + Editer + + + + Asign current order to comics + Assigner l'ordre actuel à vos comics + + + + Update cover + Mise à jour des couvertures + + + + Delete selected comics + Supprimer le comics sélectionné + + + + Hide comic flow + Cacher le comic flow + + + + Download tags from Comic Vine + + + + + Folder + + + + + Comic + + + + + Update needed + Mise à jour requise + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Cette librairie a été créée avec une ancienne version de YACReaderLibrary. Mise à jour necessaire. Mettre à jour? + + + + Update failed + Echec de la mise à jour + + + + The current library can't be udpated. Check for write write permissions on: + Cette librairie ne peut pas être mise à jour. Vérifiez les droits d'accès: + + + + Download new version + Téléchrger la nouvelle version + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Cette librairie a été créée avec une version plus récente de YACReaderLibrary. Télécharger la nouvelle version? + + + + Library '%1' is no longer available. Do you want to remove it? + La librarie '%1' n'est plus disponible. Voulez-vous la supprimer? + + + + Old library + Ancienne librairie + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + La librarie '%1' a été créée avec une ancienne version de YACReaderLibrary. Elle doit être re-créée. Voulez-vous créer la librairie? + + + + YACReader not found + + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + + Library not found + Librairie introuvable + + + + The selected folder doesn't contain any library. + Le dossier sélectionné ne contient aucune librairie. + + + + Are you sure? + Êtes-vous sûr? + + + + Do you want remove + Voulez-vous supprimer + + + + library? + la librairie? + + + + Remove and delete metadata + Supprimer les métadata + + + + Unable to delete + + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + + Asign comics numbers + Assigner les numéros aux comics + + + + Asign numbers starting in: + Assigner les numéros: + + + + Error creating the library + Erreur lors de la création de la librairie + + + + Error updating the library + Erreur lors de la mise à jour de la librairie + + + + Error opening the library + Erreur lors de l'ouverture de la librairie + + + + Delete comics + Supprimer les comics + + + + All the selected comics will be deleted from your disk. Are you sure? + Tous les comics sélectionnés vont être supprimés de votre disque. Êtes-vous sûr? + + + + Library name already exists + Le nom de la librairie existe déjà + + + + There is another library with the name '%1'. + Une autre librairie a le nom '%1'. + + + + LocalComicListModel + + + file name + + + + + NoLibrariesWidget + + + You don't have any librarires yet + Vous n'avez pas encore de librairie + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + <p>Vous pouvez creer une librairie dans n'importe quel dossierr, YACReaderLibrary importera les dossiers et les livres contenus dans ce dossier. Si vous avez déjà crer des librairies, vous pouvez les ouvrir.</p><p>N'oubliez pas que vous pouvez utiliser YACReader en tant que stand alone pour lire vos livres sur votre ordinateur.</p> + + + + create your first library + Créez votre première librairie + + + + add an existing one + Ajouter une librairie existante + + + + OptionsDialog + + + Options + Options + + + + PropertiesDialog + + + General info + Infos générales + + + + Authors + Auteurs + + + + Publishing + Publication + + + + Plot + Intrigue + + + + Cover page + Couverture + + + + Title: + Titre: + + + + Issue number: + Numéro: + + + + Volume: + Volume: + + + + Story arc: + Arc narratif: + + + + Genere: + Genre: + + + + Size: + Taille: + + + + Writer(s): + Scénariste(s): + + + + Penciller(s): + Dessinateur(s): + + + + Inker(s): + Encreur(s): + + + + Colorist(s): + Coloriste(s): + + + + Letterer(s): + Lettreur(s): + + + + Cover Artist(s): + Artiste(s) de couverture: + + + + Day: + Jour: + + + + Month: + Mois: + + + + Year: + Année: + + + + Publisher: + Editeur: + + + + Format: + Format: + + + + Color/BW: + Couleur/Noir et blanc: + + + + Age rating: + Limite d'âge: + + + + Synopsis: + Synopsis: + + + + Characters: + Personnages: + + + + Notes: + Notes: + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + Not found + Introuvable + + + + Comic not found. You should update your library. + Comic introuvable. Vous devriez mettre à jour votre librairie. + + + + Edit selected comics information + Editer les informations du comic sélectionné + + + + Edit comic information + Editer les informations du comic + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + + New Library Name : + Nouveau nom de librairie: + + + + Rename + Renommer + + + + Cancel + Annuler + + + + Rename current library + Renommer la librairie actuelle + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + + + + + + page %1 of %2 + + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + + Please provide some aditional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some aditional information. + + + + + Series: + + + + + SelectComic + + + Please, select the right comic info. + + + + + comics + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SelectVolume + + + Please, select the right series for your comic. + + + + + volumes + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + + + + + yes + oui + + + + no + non + + + + ServerConfigDialog + + + set port + Configurer le port + + + + EASY SERVER CONNECTION + CONNECTION AU SERVEUR + + + + SERVER ADDRESS + ADRESSE DU SERVEUR + + + + just scan the code with your device!! + Scannez simplement le code!! + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader est désormais disponible sur iOS, la meilleur manière de lire sur iPad, iPhone ou iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Essayez-le! </a> + + + + IP address + Adresse IP + + + + Port + Port + + + + enable the server + Autoriser le serveur + + + + QR generator error! + QR generator error! + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + + + + + sort comics to match comic information + + + + + issues + + + + + remove selected comics + + + + + restore all removed comics + + + + + restore removed comics + + + + + TableModel + + + yes + oui + + + + no + non + + + + Title + Titre + + + + File Name + Nom du fichier + + + + Pages + Pages + + + + Size + Taille + + + + Read + Lu + + + + Current Page + + + + + Rating + + + + + TitleHeader + + + SEARCH + + + + + UpdateLibraryDialog + + + Updating.... + Mise à jour... + + + + Cancel + Annuler + + + + Update library + Mettre la librairie à jour + + + + VolumeComicsModel + + + title + + + + + VolumesModel + + + year + + + + + issues + + + + + publisher + + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + Attendez, suppression en cours... + + + + cancel + Annuler + + + + YACReaderFieldEdit + + + + Click to overwrite + Cliquer pour modifier + + + + Restore to default + Restaurer les paramètres par défaut + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Cliquer pour modifier + + + + Restore to default + Restaurer les paramètres par défaut + + + + YACReaderFlowConfigWidget + + + How to show covers: + Comment voir les couvertures: + + + + CoverFlow look + Vue CoverFlow + + + + Stripe look + Vue alignée + + + + Overlapped Stripe look + Vue superposée + + + + YACReaderGLFlowConfigWidget + + + Presets: + Réglages: + + + + Classic look + Vue classique + + + + Stripe look + Vue alignée + + + + Overlapped Stripe look + Vue superposée + + + + Modern look + Vue moderne + + + + Roulette look + Vue roulette + + + + Show advanced settings + Réglages avancés + + + + Custom: + Personnalisation: + + + + View angle + Angle de vue + + + + Position + Position + + + + Cover gap + Espace entre les couvertures + + + + Central gap + Espace central + + + + Zoom + Zoom + + + + Y offset + Axe Y + + + + Z offset + Axe Z + + + + Cover Angle + Angle des couvertures + + + + Visibility + Visibilité + + + + Light + Lumière + + + + Max angle + Angle maximum + + + + Low Performance + Basse performance + + + + High Performance + Haute performance + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Utiliser VSync (améliore la qualité de l'image en plein écran, diminue la performance) + + + + Performance: + Performance: + + + + YACReaderOptionsDialog + + + Save + Sauvegarder + + + + Cancel + Annuler + + + + Use hardware acceleration (restart needed) + Utiliser l'accélération hardware (redémarrage nécessaire) + + + + YACReaderSideBar + + + LIBRARIES + LIBRAIRIES + + + + FOLDERS + DOSSIERS + + + + Search folders and comics + Recherche de dossiers et de comics + + + diff --git a/YACReaderLibrary/yacreaderlibrary_nl.ts b/YACReaderLibrary/yacreaderlibrary_nl.ts new file mode 100644 index 00000000..706c2aa4 --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_nl.ts @@ -0,0 +1,1517 @@ + + + + + AddLibraryDialog + + + Comics folder : + Strips map: + + + + Library Name : + Bibliotheek Naam : + + + + Add + Toevoegen + + + + Cancel + Annuleren + + + + Add an existing library + Voeg een bestaand bibliotheek toe + + + + ComicVineDialog + + + skip + + + + + back + + + + + next + + + + + search + + + + + close + + + + + + + + + Looking for volume... + + + + + + comic %1 of %2 - %3 + + + + + %1 comics selected + + + + + Error connecting to ComicVine + + + + + unknown error + + + + + + Retrieving tags for : %1 + + + + + Retrieving volume info... + + + + + Looking for comic... + + + + + CreateLibraryDialog + + + Comics folder : + Strips map: + + + + Library Name : + Bibliotheek Naam : + + + + Create + Aanmaken + + + + Cancel + Annuleren + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Een bibliotheek aanmaken kan enkele minuten duren. U kunt het proces stoppen en de bibliotheek later voltooien. + + + + Create new library + Een nieuwe Bibliotheek aanmaken + + + + Path not found + Pad niet gevonden + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + De geselecteerde pad bestaat niet of is geen geldig pad. Controleer of u schrijftoegang hebt tot deze map + + + + ExportComicsInfoDialog + + + Output file : + Uitvoerbestand: + + + + Create + Aanmaken + + + + Cancel + Annuleren + + + + Export comics info + Strip info exporteren + + + + Destination database name + Bestemmingsdatabase naam + + + + Problem found while writing + Probleem bij het schrijven + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Het gekozen pad voor het uitvoerbestand bestaat niet of is geen geldig pad. Controleer of u schrijftoegang hebt tot deze map + + + + ExportLibraryDialog + + + Output folder : + Uitvoermap : + + + + Create + Aanmaken + + + + Cancel + Annuleren + + + + Create covers package + Aanmaken omslag pakket + + + + Problem found while writing + Probleem bij het schrijven + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Het gekozen pad voor het uitvoerbestand bestaat niet of is geen geldig pad. Controleer of u schrijftoegang hebt tot deze map + + + + Destination directory + Doeldirectory + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7Z Archiefbestand niet gevonden + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + HelpAboutDialog + + + About + Over + + + + Help + Help + + + + ImportComicsInfoDialog + + + Import comics info + Strip info Importeren + + + + Info database location : + Info database locatie : + + + + Import + Importeren + + + + Cancel + Annuleren + + + + Comics info file (*.ydb) + Strips info bestand ( * .ydb) + + + + ImportLibraryDialog + + + Library Name : + Bibliotheek Naam : + + + + Package location : + Arrangement locatie : + + + + Destination folder : + Doelmap: + + + + Unpack + Uitpakken + + + + Cancel + Annuleren + + + + Extract a catalog + Een catalogus uitpakken + + + + Compresed library covers (*.clc) + Compresed omslag- bibliotheek ( * .clc) + + + + ImportWidget + + + stop + stop + + + + Some of the comics being added... + Enkele strips zijn toegevoegd ... + + + + Importing comics + Strips importeren + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + <P>YACReaderLibrary maak nu een nieuwe bibliotheek. < /p> <p>Een bibliotheek aanmaken kan enkele minuten duren. U kunt het proces stoppen en de bibliotheek later voltooien. < /p> + + + + Updating the library + Actualisering van de bibliotheek + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <P>De huidige bibliotheek wordt bijgewerkt. Voor snellere updates, update uw bibliotheken regelmatig. < /p> <p>u kunt het proces stoppen om later bij te werken. < /p> + + + + LibraryWindow + + + YACReader Library + YACReader Bibliotheek + + + + + Library + Bibliotheek + + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'>Druk op "F" om 'volledig scherm modus' te sluiten </font> + + + + Create a new library + Maak een nieuwe Bibliotheek + + + + Open an existing library + Open een bestaande Bibliotheek + + + + + Export comics info + Exporteren van strip info + + + + + Import comics info + Importeren van strip info + + + + Pack covers + Inpakken strip voorbladen + + + + Pack the covers of the selected library + Inpakken alle strip voorbladen van de geselecteerde Bibliotheek + + + + Unpack covers + Uitpakken voorbladen + + + + Unpack a catalog + Uitpaken van een catalogus + + + + Update library + Bibliotheek bijwerken + + + + Update current library + Huidige Bibliotheek bijwerken + + + + Rename library + Bibliotheek hernoemen + + + + Rename current library + Huidige Bibliotheek hernoemen + + + + Remove library + Bibliotheek verwijderen + + + + Remove current library from your collection + De huidige Bibliotheek verwijderen uit uw verzameling + + + + Open current comic + Huidige strip openen + + + + Open current comic on YACReader + Huidige strip openen in YACReader + + + + + Set as read + Instellen als gelezen + + + + Set comic as read + Strip Instellen als gelezen + + + + + Set as unread + Instellen als ongelezen + + + + Set comic as unread + Strip Instellen als ongelezen + + + + Show/Hide marks + Toon/Verberg markeringen + + + + Show or hide readed marks + Toon of Verberg gelezen markeringen + + + + Library not available + Library ' + Bibliotheek niet beschikbaar + + + + Fullscreen mode on/off + Volledig scherm modus aan/of + + + + Fullscreen mode on/off (F) + Volledig scherm modus aan/of (F) + + + + Help, About YACReader + Help, Over YACReader + + + + Select root node + Selecteer de hoofd categorie + + + + + + + + + + + Expand all nodes + Alle categorieën uitklappen + + + + - + - + + + + Colapse all nodes + Alle categorieën inklappen + + + + Show options dialog + Toon opties dialoog + + + + Show comics server options dialog + Toon strips-server opties dialoog + + + + Open folder... + Map openen ... + + + + Set as uncompleted + + + + + Set as completed + + + + + Open containing folder... + Open map ... + + + + Reset comic rating + + + + + Select all comics + Selecteer alle strips + + + + Edit + Bewerken + + + + Asign current order to comics + Nummeren van strips + + + + Update cover + Strip omslagen bijwerken + + + + Delete selected comics + Geselecteerde strips verwijderen + + + + Hide comic flow + Sluit de Omslagbrowser + + + + Download tags from Comic Vine + + + + + Folder + + + + + Comic + + + + + Update needed + Bijwerken is nodig + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Deze bibliotheek is gemaakt met een vorige versie van YACReaderLibrary. Het moet worden bijgewerkt. Nu bijwerken? + + + + Update failed + Bijwerken mislukt + + + + The current library can't be udpated. Check for write write permissions on: + De huidige bibliotheek kan niet worden bijgewerkt. Controleer bij of u schrijfbevoegdheid hebt: + + + + Download new version + Nieuwe versie ophalen + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Deze bibliotheek is gemaakt met een nieuwere versie van YACReaderLibrary. Download de nieuwe versie? + + + + Library '%1' is no longer available. Do you want to remove it? + Bibliotheek ' %1' is niet langer beschikbaar. Wilt u het verwijderen? + + + + Old library + Oude Bibliotheek + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + Bibliotheek ' %1' is gemaakt met een oudere versie van YACReaderLibrary. Zij moet opnieuw worden aangemaakt. Wilt u de bibliotheek nu aanmaken? + + + + YACReader not found + + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + + Library not found + Bibliotheek niet gevonden + + + + The selected folder doesn't contain any library. + De geselecteerde map bevat geen bibliotheek. + + + + Are you sure? + Weet u het zeker? + + + + Do you want remove + Wilt u verwijderen + + + + library? + Bibliotheek? + + + + Remove and delete metadata + Verwijder metagegevens + + + + Unable to delete + + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + + Asign comics numbers + Strips nummeren + + + + Asign numbers starting in: + Strips nummeren beginnen bij: + + + + Error creating the library + Fout bij aanmaken Bibliotheek + + + + Error updating the library + Fout bij bijwerken Bibliotheek + + + + Error opening the library + Fout bij openen Bibliotheek + + + + Delete comics + Strips verwijderen + + + + All the selected comics will be deleted from your disk. Are you sure? + Alle geselecteerde strips worden verwijderd van uw schijf. Weet u het zeker? + + + + Library name already exists + Bibliotheek naam bestaat al + + + + There is another library with the name '%1'. + Er is al een bibliotheek met de naam ' %1 '. + + + + LocalComicListModel + + + file name + + + + + NoLibrariesWidget + + + You don't have any librarires yet + Je hebt geen nog librarires + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + <P>u kunt een bibliotheek maken in een willekeurige map, YACReaderLibrary importeert alle strips en mappen uit deze map. Alle bibliotheek aangemaakt in het verleden kan je openen. < /p> <p>vergeet niet dat u YACReader kan gebruiken als stand-alone applicatie voor het lezen van de strips op de computer. < /p> + + + + create your first library + Maak uw eerste bibliotheek + + + + add an existing one + voeg een bestaande bibliotheek toe + + + + OptionsDialog + + + Options + Opties + + + + PropertiesDialog + + + General info + Algemene Info + + + + Authors + Auteurs + + + + Publishing + Uitgever + + + + Plot + Verhaal + + + + Cover page + Omslag + + + + Title: + Titel: + + + + Issue number: + Ids: + + + + Volume: + Inhoud: + + + + Story arc: + Verhaallijn: + + + + Genere: + Genre: + + + + Size: + Grootte(MB): + + + + Writer(s): + Schrijver(s): + + + + Penciller(s): + Tekenaar(s): + + + + Inker(s): + Inker(s): + + + + Colorist(s): + Inkleurder(s): + + + + Letterer(s): + Letterzetter(s): + + + + Cover Artist(s): + Omslag ontwikkelaar(s): + + + + Day: + Dag: + + + + Month: + Maand: + + + + Year: + Jaar: + + + + Publisher: + Uitgever: + + + + Format: + Formaat: + + + + Color/BW: + Kleur/ZW: + + + + Age rating: + Leeftijdsbeperking: + + + + Synopsis: + Synopsis: + + + + Characters: + Personages: + + + + Notes: + Opmerkingen: + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + Not found + Niet gevonden + + + + Comic not found. You should update your library. + Strip niet gevonden. U moet uw bibliotheek.bijwerken. + + + + Edit selected comics information + Geselecteerde strip informatie bijwerken + + + + Edit comic information + Strip informatie bijwerken + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + + New Library Name : + Nieuwe Bibliotheek Naam : + + + + Rename + Hernoem + + + + Cancel + Annuleren + + + + Rename current library + Hernoem de huidige bibiliotheek + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + + + + + + page %1 of %2 + + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + + Please provide some aditional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some aditional information. + + + + + Series: + + + + + SelectComic + + + Please, select the right comic info. + + + + + comics + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SelectVolume + + + Please, select the right series for your comic. + + + + + volumes + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + + + + + yes + Ja + + + + no + neen + + + + ServerConfigDialog + + + set port + Poort instellen + + + + EASY SERVER CONNECTION + GEMAKKELIJKE VERBINDING MET DE SERVER + + + + SERVER ADDRESS + SERVERADRES + + + + just scan the code with your device!! + Scan de code met uw apparaat! + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader is nu beschikbaar voor iOS apparaten, de beste strip leeservaring nu op uw iPad, iPhone of iPod touch. <A href= "http://ios.yacreader.com' style= "color:rgb(193, 148, 65) "> Ontdek het zelf! < /A> + + + + IP address + IP-adres + + + + Port + Poort + + + + enable the server + De server instellen + + + + QR generator error! + QR generator fout! + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + + + + + sort comics to match comic information + + + + + issues + + + + + remove selected comics + + + + + restore all removed comics + + + + + restore removed comics + + + + + TableModel + + + yes + Ja + + + + no + neen + + + + Title + Titel + + + + File Name + Bestandsnaam + + + + Pages + Pagina's + + + + Size + Grootte(MB) + + + + Read + Gelezen + + + + Current Page + + + + + Rating + + + + + TitleHeader + + + SEARCH + + + + + UpdateLibraryDialog + + + Updating.... + Bijwerken.... + + + + Cancel + Annuleren + + + + Update library + Bibliotheek bijwerken + + + + VolumeComicsModel + + + title + + + + + VolumesModel + + + year + + + + + issues + + + + + publisher + + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + Even geduld, verwijderen ... + + + + cancel + annuleren + + + + YACReaderFieldEdit + + + + Click to overwrite + Klik hier om te overschrijven + + + + Restore to default + Standaardwaarden herstellen + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Klik hier om te overschrijven + + + + Restore to default + Standaardwaarden herstellen + + + + YACReaderFlowConfigWidget + + + How to show covers: + Hoe omslagbladen bekijken: + + + + CoverFlow look + Omslagbrowser uiterlijk + + + + Stripe look + Brede band + + + + Overlapped Stripe look + Overlappende band + + + + YACReaderGLFlowConfigWidget + + + Presets: + Voorinstellingen: + + + + Classic look + Klassiek + + + + Stripe look + Brede band + + + + Overlapped Stripe look + Overlappende band + + + + Modern look + Modern + + + + Roulette look + Roulette + + + + Show advanced settings + Toon geavanceerde instellingen + + + + Custom: + Aangepast: + + + + View angle + Kijkhoek + + + + Position + Positie + + + + Cover gap + Ruimte tss Omslag + + + + Central gap + Centrale ruimte + + + + Zoom + Zoom + + + + Y offset + Y-positie + + + + Z offset + Z- positie + + + + Cover Angle + Omslag hoek + + + + Visibility + Zichtbaarheid + + + + Light + Licht + + + + Max angle + Maximale hoek + + + + Low Performance + Lage Prestaties + + + + High Performance + Hoge Prestaties + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Gebruik VSync (verbetering van de beeldkwaliteit in de modus volledig scherm, slechtere prestatie) + + + + Performance: + Prestatie: + + + + YACReaderOptionsDialog + + + Save + Bewaar + + + + Cancel + Annuleren + + + + Use hardware acceleration (restart needed) + Gebruik hardware versnelling (opnieuw opstarten vereist) + + + + YACReaderSideBar + + + LIBRARIES + BIBLIOTHEKEN + + + + FOLDERS + MAPPEN + + + + Search folders and comics + Zoeken in mappen en strips + + + diff --git a/YACReaderLibrary/yacreaderlibrary_pt.ts b/YACReaderLibrary/yacreaderlibrary_pt.ts new file mode 100644 index 00000000..6636dc77 --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_pt.ts @@ -0,0 +1,1520 @@ + + + + + AddLibraryDialog + + + Comics folder : + Pasta dos quadrinhos : + + + + Library Name : + Nome da Biblioteca : + + + + Add + Adicionar + + + + Cancel + Cancelar + + + + Add an existing library + Adicionar uma biblioteca existente + + + + ComicVineDialog + + + skip + + + + + back + + + + + next + + + + + search + + + + + close + + + + + + + + + Looking for volume... + + + + + + comic %1 of %2 - %3 + + + + + %1 comics selected + + + + + Error connecting to ComicVine + + + + + unknown error + + + + + + Retrieving tags for : %1 + + + + + Retrieving volume info... + + + + + Looking for comic... + + + + + CreateLibraryDialog + + + Comics folder : + Pasta dos quadrinhos : + + + + Library Name : + Nome da Biblioteca : + + + + Create + Criar + + + + Cancel + Cancelar + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + + + + + Create new library + Criar uma nova biblioteca + + + + Path not found + + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + ExportComicsInfoDialog + + + Output file : + + + + + Create + Criar + + + + Cancel + Cancelar + + + + Export comics info + + + + + Destination database name + + + + + Problem found while writing + + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + ExportLibraryDialog + + + Output folder : + Pasta de saída : + + + + Create + Criar + + + + Cancel + Cancelar + + + + Create covers package + Criar pacote de capas + + + + Problem found while writing + + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + Destination directory + Diretório de destino + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z não encontrado + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + HelpAboutDialog + + + About + + + + + Help + + + + + ImportComicsInfoDialog + + + Import comics info + + + + + Info database location : + + + + + Import + + + + + Cancel + Cancelar + + + + Comics info file (*.ydb) + + + + + ImportLibraryDialog + + + Library Name : + Nome da Biblioteca : + + + + Package location : + Local do pacote : + + + + Destination folder : + Pasta de destino : + + + + Unpack + Desempacotar + + + + Cancel + Cancelar + + + + Extract a catalog + Extrair um catálogo + + + + Compresed library covers (*.clc) + Capas da biblioteca compactada (*.clc) + + + + ImportWidget + + + Importing comics + + + + + stop + + + + + Some of the comics being added... + + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + + + + + Updating the library + + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later. + + + + + LibraryWindow + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> pressione 'F' para fechar o modo tela cheia </font> + + + + YACReader Library + Biblioteca YACReader + + + + Create a new library + Criar uma nova biblioteca + + + + Open an existing library + Abrir uma biblioteca existente + + + + + Export comics info + + + + + + Import comics info + + + + + Pack covers + + + + + Pack the covers of the selected library + Pacote de capas da biblioteca selecionada + + + + Unpack covers + + + + + Unpack a catalog + Desempacotar um catálogo + + + + Update current library + Atualizar biblioteca atual + + + + Rename library + + + + + Rename current library + Renomear biblioteca atual + + + + Remove current library from your collection + Remover biblioteca atual da sua coleção + + + + Open current comic + + + + + Open current comic on YACReader + Abrir quadrinho atual no YACReader + + + + + Set as read + + + + + Set comic as read + + + + + + Set as unread + + + + + Set comic as unread + + + + + Show/Hide marks + + + + + Show or hide readed marks + + + + + Fullscreen mode on/off + + + + + Fullscreen mode on/off (F) + Modo tela cheia ligar/desligar (F) + + + + Help, About YACReader + Ajuda, Sobre o YACReader + + + + Select root node + Selecionar raiz + + + + + + + + + + Expand all nodes + Expandir todos + + + + - + + + + + Colapse all nodes + Contrair todos + + + + Show options dialog + Mostrar opções + + + + Show comics server options dialog + + + + + Open folder... + Abrir pasta... + + + + Set as uncompleted + + + + + Set as completed + + + + + Open containing folder... + Abrir a pasta contendo... + + + + Reset comic rating + + + + + Select all comics + + + + + Edit + + + + + Asign current order to comics + + + + + Update cover + + + + + Delete selected comics + + + + + Hide comic flow + + + + + Download tags from Comic Vine + + + + + Folder + + + + + Comic + + + + + Library not available + Library ' + + + + + Old library + + + + + YACReader not found + + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + + Unable to delete + + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + + Error creating the library + + + + + Error updating the library + + + + + Error opening the library + + + + + Delete comics + + + + + All the selected comics will be deleted from your disk. Are you sure? + + + + + Library name already exists + + + + + There is another library with the name '%1'. + + + + + + Library + Biblioteca + + + + Update library + + + + + Remove library + + + + + Update needed + + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + + + + + Update failed + + + + + The current library can't be udpated. Check for write write permissions on: + + + + + Download new version + + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + + + + + Library '%1' is no longer available. Do you want to remove it? + + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + + + + + Library not found + + + + + The selected folder doesn't contain any library. + + + + + Are you sure? + Você tem certeza? + + + + library? + + + + + Remove and delete metadata + + + + + Do you want remove + Você deseja remover + + + + Asign comics numbers + + + + + Asign numbers starting in: + + + + + LocalComicListModel + + + file name + + + + + NoLibrariesWidget + + + You don't have any librarires yet + + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + + + + + create your first library + + + + + add an existing one + + + + + OptionsDialog + + + Options + + + + + PropertiesDialog + + + General info + + + + + Authors + + + + + Publishing + + + + + Plot + + + + + Cover page + + + + + Title: + + + + + Issue number: + + + + + Volume: + + + + + Story arc: + + + + + Genere: + + + + + Size: + + + + + Writer(s): + + + + + Penciller(s): + + + + + Inker(s): + + + + + Colorist(s): + + + + + Letterer(s): + + + + + Cover Artist(s): + + + + + Day: + + + + + Month: + + + + + Year: + + + + + Publisher: + + + + + Format: + + + + + Color/BW: + + + + + Age rating: + + + + + Synopsis: + + + + + Characters: + + + + + Notes: + + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + Not found + + + + + Comic not found. You should update your library. + + + + + Edit selected comics information + + + + + Edit comic information + + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + + New Library Name : + Novo nome da biblioteca : + + + + Rename + Renomear + + + + Cancel + Cancelar + + + + Rename current library + Renomear biblioteca atual + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + + + + + + page %1 of %2 + + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + + Please provide some aditional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some aditional information. + + + + + Series: + + + + + SelectComic + + + Please, select the right comic info. + + + + + comics + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SelectVolume + + + Please, select the right series for your comic. + + + + + volumes + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + + + + + yes + + + + + no + + + + + ServerConfigDialog + + + set port + + + + + EASY SERVER CONNECTION + + + + + SERVER ADDRESS + + + + + just scan the code with your device!! + + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com'> Discover it! </a> + + + + + IP address + + + + + Port + + + + + enable the server + + + + + QR generator error! + + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + + + + + sort comics to match comic information + + + + + issues + + + + + remove selected comics + + + + + restore all removed comics + + + + + restore removed comics + + + + + TableModel + + + yes + + + + + no + + + + + Title + + + + + File Name + + + + + Pages + + + + + Size + + + + + Read + + + + + Current Page + + + + + Rating + + + + + TitleHeader + + + SEARCH + + + + + UpdateLibraryDialog + + + Updating.... + Atualizando.... + + + + Cancel + Cancelar + + + + Update library + + + + + VolumeComicsModel + + + title + + + + + VolumesModel + + + year + + + + + issues + + + + + publisher + + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + + + + + cancel + + + + + YACReaderFieldEdit + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFlowConfigWidget + + + How to show covers: + Como mostrar capas: + + + + CoverFlow look + Olhar capa cheia + + + + Stripe look + Olhar lista + + + + Overlapped Stripe look + Olhar lista sobreposta + + + + YACReaderGLFlowConfigWidget + + + Presets: + + + + + Classic look + + + + + Stripe look + Olhar lista + + + + Overlapped Stripe look + Olhar lista sobreposta + + + + Modern look + + + + + Roulette look + + + + + Show advanced settings + + + + + Custom: + + + + + View angle + + + + + Position + + + + + Cover gap + + + + + Central gap + + + + + Zoom + + + + + Y offset + + + + + Z offset + + + + + Cover Angle + + + + + Visibility + + + + + Light + + + + + Max angle + + + + + Low Performance + + + + + High Performance + + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + + + + + Performance: + + + + + YACReaderOptionsDialog + + + Save + Salvar + + + + Cancel + Cancelar + + + + Use hardware acceleration (restart needed) + + + + + YACReaderSideBar + + + LIBRARIES + + + + + FOLDERS + + + + + Search folders and comics + + + + diff --git a/YACReaderLibrary/yacreaderlibrary_ru.ts b/YACReaderLibrary/yacreaderlibrary_ru.ts new file mode 100644 index 00000000..20cccb4f --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_ru.ts @@ -0,0 +1,1520 @@ + + + + + AddLibraryDialog + + + Comics folder : + Папка комикÑов: + + + + Library Name : + Ð˜Ð¼Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ¸: + + + + Add + Добавить + + + + Cancel + Отменить + + + + Add an existing library + Добавить в ÑущеÑтвующую библиотеку + + + + ComicVineDialog + + + skip + + + + + back + + + + + next + + + + + search + + + + + close + + + + + + + + + Looking for volume... + + + + + + comic %1 of %2 - %3 + + + + + %1 comics selected + + + + + Error connecting to ComicVine + + + + + unknown error + + + + + + Retrieving tags for : %1 + + + + + Retrieving volume info... + + + + + Looking for comic... + + + + + CreateLibraryDialog + + + Comics folder : + Папка комикÑов: + + + + Library Name : + Ð˜Ð¼Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ¸: + + + + Create + Создать + + + + Cancel + Отмена + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Создание библиотеки может занÑть неÑколько минут. Ð’Ñ‹ можете оÑтановить процеÑÑ Ð¸ обновить библиотеку позже Ð´Ð»Ñ Ð·Ð°Ð²ÐµÑ€ÑˆÐµÐ½Ð¸Ñ Ð·Ð°Ð´Ð°Ñ‡Ð¸. + + + + Create new library + Создать новую библиотеку + + + + Path not found + Путь не найден + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + Выбранный путь отÑутÑтвует, либо неверен. УбедитеÑÑŒ , что у Ð²Ð°Ñ ÐµÑть доÑтуп к Ñтой папке + + + + ExportComicsInfoDialog + + + Output file : + Файл вывода: + + + + Create + Создать + + + + Cancel + Отмена + + + + Export comics info + ЭкÑпортировать информацию комикÑа + + + + Destination database name + Ð˜Ð¼Ñ Ð½Ð°Ð·Ð½Ð°Ñ‡ÐµÐ½Ð½Ð¾Ð¹ базы данных + + + + Problem found while writing + Проблема при напиÑании + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Выбранный путь Ð´Ð»Ñ Ð¸Ð¼Ð¿Ð¾Ñ€Ñ‚Ð¸Ñ€ÑƒÐµÐ¼Ð¾Ð³Ð¾ файла отÑутÑтвует, либо неверен. УбедитеÑÑŒ , что у Ð²Ð°Ñ ÐµÑть доÑтуп к Ñтой папке + + + + ExportLibraryDialog + + + Output folder : + Файл вывода: + + + + Create + Создать + + + + Cancel + Отменить + + + + Create covers package + Создать комплект обложек + + + + Problem found while writing + Проблема при напиÑании + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Выбранный путь Ð´Ð»Ñ Ð¸Ð¼Ð¿Ð¾Ñ€Ñ‚Ð¸Ñ€ÑƒÐµÐ¼Ð¾Ð³Ð¾ файла отÑутÑтвует, либо неверен. УбедитеÑÑŒ , что у Ð²Ð°Ñ ÐµÑть доÑтуп к Ñтой папке + + + + Destination directory + Ðазначенное меÑтонахождение + + + + FileComic + + + Unknown error opening the file + + + + + 7z not found + 7z не найден + + + + Format not supported + + + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + HelpAboutDialog + + + About + О программе + + + + Help + ÐаÑтройки + + + + ImportComicsInfoDialog + + + Import comics info + Импортировать информаию комикÑа + + + + Info database location : + МеÑтонахождение базы данных: + + + + Import + Импортировать + + + + Cancel + Отмена + + + + Comics info file (*.ydb) + Ð˜Ð½Ñ„Ð¾Ñ€Ð¼Ð°Ñ†Ð¸Ñ Ñ„Ð°Ð¹Ð»Ð° комикÑа + + + + ImportLibraryDialog + + + Library Name : + Ð˜Ð¼Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ¸: + + + + Package location : + МеÑтоположение комплекта: + + + + Destination folder : + ÐÐ°Ð·Ð½Ð°Ñ‡ÐµÐ½Ð½Ð°Ñ Ð¿Ð°Ð¿ÐºÐ°: + + + + Unpack + РаÑпаковать + + + + Cancel + Отмена + + + + Extract a catalog + Извлечь каталог + + + + Compresed library covers (*.clc) + Ð¡Ð¶Ð°Ñ‚Ð°Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ° обложек + + + + ImportWidget + + + Importing comics + + + + + stop + + + + + Some of the comics being added... + + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Создание библиотеки может занÑть неÑколько минут. Ð’Ñ‹ можете оÑтановить процеÑÑ Ð¸ обновить библиотеку позже Ð´Ð»Ñ Ð·Ð°Ð²ÐµÑ€ÑˆÐµÐ½Ð¸Ñ Ð·Ð°Ð´Ð°Ñ‡Ð¸. + + + + Updating the library + + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later. + + + + + LibraryWindow + + + YACReader Library + Библиотека YACReader + + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> нажмите 'F' чтобы выйте из ПолноÑкранного режима </font> + + + + Create a new library + Создать новую библиотеку + + + + Open an existing library + Открыть ÑущеÑтвующую библиотеку + + + + + Export comics info + ЕкÑпорт комикÑа + + + + + Import comics info + Импорт комикÑа + + + + Pack covers + Запакавать обложки + + + + Pack the covers of the selected library + Запакавать обложки выбранной библиотеки + + + + Unpack covers + РаÑпокавать обложки + + + + Unpack a catalog + РаÑпакавать каталог + + + + Update library + Обновить библиотеку + + + + Update current library + Обновить текущую библиотеку + + + + Rename library + + + + + Rename current library + Переименовать текущую бибилиотеку + + + + Remove current library from your collection + Удалите текущую библиотеку из Ñвоей коллекции + + + + Open current comic + + + + + Open current comic on YACReader + + + + + + Set as read + + + + + Set comic as read + + + + + + Set as unread + + + + + Set comic as unread + + + + + Show/Hide marks + + + + + Show or hide readed marks + + + + + Fullscreen mode on/off + Полноекранный режим включить/выключить + + + + Fullscreen mode on/off (F) + полноекранный режим включить/выключить(F) + + + + Help, About YACReader + Справка, о программе YACReader + + + + Select root node + + + + + + + + + + + Expand all nodes + + + + + - + + + + + Colapse all nodes + + + + + Show options dialog + Показать наÑтройки диаога + + + + Show comics server options dialog + + + + + Open folder... + Открыть папку... + + + + Set as uncompleted + + + + + Set as completed + + + + + Open containing folder... + + + + + Reset comic rating + + + + + Select all comics + Выбрать вÑе комикÑÑ‹ + + + + Edit + Редактировать + + + + Asign current order to comics + + + + + Update cover + Обновить обложки + + + + Delete selected comics + + + + + Hide comic flow + Ðе показывать поток комикÑов + + + + Download tags from Comic Vine + + + + + Folder + + + + + Comic + + + + + Library not available + Library ' + Библиотека не доÑтупна + + + + Library '%1' is no longer available. Do you want to remove it? + + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + + + + + Old library + + + + + YACReader not found + + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + + Unable to delete + + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + + Error creating the library + + + + + Error updating the library + + + + + Error opening the library + + + + + Delete comics + + + + + All the selected comics will be deleted from your disk. Are you sure? + + + + + Library name already exists + + + + + There is another library with the name '%1'. + + + + + + Library + Библиотека + + + + Remove library + + + + + Update needed + Ðеобходимо обновление + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Эта библиотека была Ñоздана Ñ Ð¿Ñ€ÐµÐ´Ñ‹Ð´ÑƒÑ‰ÐµÐ¹ верÑией YACReaderLibrary. Она должна быть обновлена. Обновить ÑейчаÑ? + + + + Update failed + Обновить неудалоÑÑŒ + + + + The current library can't be udpated. Check for write write permissions on: + Ð’ наÑтоÑщее Ð²Ñ€ÐµÐ¼Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ° не может быть обновлена. Проверьте права на чтение/запиÑÑŒ: + + + + Download new version + Загрузить новую верÑию + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Эта библиотека был Ñоздан при новой верÑией YACReaderLibrary. Скачать новую верÑию ÑейчаÑ? + + + + Library not found + Библиотека не найдена + + + + The selected folder doesn't contain any library. + Ð’Ñ‹Ð±Ñ€Ð°Ð½Ð½Ð°Ñ Ð¿Ð°Ð¿ÐºÐ° не Ñодержит библиотеку. + + + + Are you sure? + Ð’Ñ‹ уверены? + + + + library? + + + + + Remove and delete metadata + + + + + Do you want remove + Ð’Ñ‹ хотите удалить + + + + Asign comics numbers + Ðазначение номеров комикÑа + + + + Asign numbers starting in: + Ðазначьте номера, начинающиеÑÑ Ð½Ð°: + + + + LocalComicListModel + + + file name + + + + + NoLibrariesWidget + + + You don't have any librarires yet + + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + + + + + create your first library + + + + + add an existing one + + + + + OptionsDialog + + + Options + ÐаÑтройки + + + + PropertiesDialog + + + General info + ÐžÐ±Ñ‰Ð¸Ñ Ð¸Ð½Ñ„Ð¾Ñ€Ð¼Ð°Ñ†Ð¸Ñ + + + + Authors + + + + + Publishing + + + + + Plot + + + + + Cover page + + + + + Title: + + + + + Issue number: + + + + + Volume: + Объём : + + + + Story arc: + + + + + Genere: + + + + + Size: + Размер: + + + + Writer(s): + + + + + Penciller(s): + + + + + Inker(s): + + + + + Colorist(s): + + + + + Letterer(s): + + + + + Cover Artist(s): + + + + + Day: + День: + + + + Month: + меÑÑц: + + + + Year: + Год: + + + + Publisher: + + + + + Format: + Формат: + + + + Color/BW: + + + + + Age rating: + + + + + Synopsis: + + + + + Characters: + + + + + Notes: + ПримичÑние: + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + Not found + Ðе найдено + + + + Comic not found. You should update your library. + + + + + Edit selected comics information + Редактировать информацию выбранного комикÑа + + + + Edit comic information + Реддактировать информацию + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + + New Library Name : + Ðовое Ð¸Ð¼Ñ Ð±Ð¸Ð±Ð»Ð¸Ð¾Ñ‚ÐµÐºÐ¸: + + + + Rename + Переименовать + + + + Cancel + Отмена + + + + Rename current library + Переименовать текущую бибилиотеку + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + + + + + + page %1 of %2 + + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + + Please provide some aditional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some aditional information. + + + + + Series: + + + + + SelectComic + + + Please, select the right comic info. + + + + + comics + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SelectVolume + + + Please, select the right series for your comic. + + + + + volumes + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + + + + + yes + + + + + no + + + + + ServerConfigDialog + + + set port + + + + + EASY SERVER CONNECTION + + + + + SERVER ADDRESS + + + + + just scan the code with your device!! + + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com'> Discover it! </a> + + + + + IP address + + + + + Port + Порт + + + + enable the server + + + + + QR generator error! + Ошибка QR генератора! + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + + + + + sort comics to match comic information + + + + + issues + + + + + remove selected comics + + + + + restore all removed comics + + + + + restore removed comics + + + + + TableModel + + + yes + + + + + no + + + + + Title + Заголовок + + + + File Name + Ð˜Ð¼Ñ Ñ„Ð°Ð¹Ð»Ð° + + + + Pages + Страницы + + + + Size + + + + + Read + + + + + Current Page + + + + + Rating + + + + + TitleHeader + + + SEARCH + + + + + UpdateLibraryDialog + + + Updating.... + Обновление... + + + + Cancel + Отмена + + + + Update library + Обновить библиотеку + + + + VolumeComicsModel + + + title + + + + + VolumesModel + + + year + + + + + issues + + + + + publisher + + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + + + + + cancel + + + + + YACReaderFieldEdit + + + + Click to overwrite + Ðажмите Ð´Ð»Ñ Ð¿ÐµÑ€ÐµÐ·Ð°Ð¿Ð¸Ñи + + + + Restore to default + Вернуть к первоначальным значениÑм + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Ðажмите Ð´Ð»Ñ Ð¿ÐµÑ€ÐµÐ·Ð°Ð¿Ð¸Ñи + + + + Restore to default + Вернуть к первоначальным значениÑм + + + + YACReaderFlowConfigWidget + + + How to show covers: + Как показать обложки: + + + + CoverFlow look + ПредоÑмотр обложки + + + + Stripe look + Вид полоÑами + + + + Overlapped Stripe look + Вид перекрывающимиÑÑ Ð¿Ð¾Ð»Ð¾Ñами + + + + YACReaderGLFlowConfigWidget + + + Presets: + ПредуÑтановки: + + + + Classic look + КлаÑÑичеÑкий вид + + + + Stripe look + Вид полоÑами + + + + Overlapped Stripe look + Вид перекрывающимиÑÑ Ð¿Ð¾Ð»Ð¾Ñами + + + + Modern look + Современный вид + + + + Roulette look + Вид рулеткой + + + + Show advanced settings + + + + + Custom: + ПользовательÑкий: + + + + View angle + Угол Ð·Ñ€ÐµÐ½Ð¸Ñ + + + + Position + ÐŸÐ¾Ð·Ð¸Ñ†Ð¸Ñ + + + + Cover gap + Охватить разрыв + + + + Central gap + СфокуÑировать разрыв + + + + Zoom + МаÑштабировать + + + + Y offset + Смещение по Y + + + + Z offset + Смещение по Z + + + + Cover Angle + Охватить угол + + + + Visibility + ПрозрачноÑть + + + + Light + ОÑветить + + + + Max angle + МакÑимальный угол + + + + Low Performance + ÐœÐ¸Ð½Ð¸Ð¼Ð°Ð»ÑŒÐ½Ð°Ñ Ð¿Ñ€Ð¾Ð¸Ð·Ð²Ð¾Ð´Ð¸Ñ‚ÐµÐ»ÑŒÐ½Ð¾Ñть + + + + High Performance + МакÑÐ¸Ð¼Ð°Ð»ÑŒÐ½Ð°Ñ Ð¿Ñ€Ð¾Ð¸Ð·Ð²Ð¾Ð´Ð¸Ñ‚ÐµÐ»ÑŒÐ½Ð¾Ñть + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + ИÑпользовать VSync (повыÑить формат Ð¸Ð·Ð¾Ð±Ñ€Ð°Ð¶ÐµÐ½Ð¸Ñ Ð² полноÑкранном режиме , хуже производительноÑть) + + + + Performance: + ПроизводительноÑть: + + + + YACReaderOptionsDialog + + + Save + Сохранить + + + + Cancel + Отмена + + + + Use hardware acceleration (restart needed) + ИÑпользовать аппаратное уÑкорение (необходима перезагрузка) + + + + YACReaderSideBar + + + LIBRARIES + + + + + FOLDERS + + + + + Search folders and comics + + + + diff --git a/YACReaderLibrary/yacreaderlibrary_source.ts b/YACReaderLibrary/yacreaderlibrary_source.ts new file mode 100644 index 00000000..8998aef1 --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_source.ts @@ -0,0 +1,1517 @@ + + + + + AddLibraryDialog + + + Comics folder : + + + + + Library Name : + + + + + Add + + + + + Cancel + + + + + Add an existing library + + + + + ComicVineDialog + + + skip + + + + + back + + + + + next + + + + + search + + + + + close + + + + + + + + + Looking for volume... + + + + + + comic %1 of %2 - %3 + + + + + %1 comics selected + + + + + Error connecting to ComicVine + + + + + unknown error + + + + + + Retrieving tags for : %1 + + + + + Retrieving volume info... + + + + + Looking for comic... + + + + + CreateLibraryDialog + + + Comics folder : + + + + + Library Name : + + + + + Create + + + + + Cancel + + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + + + + + Create new library + + + + + Path not found + + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + ExportComicsInfoDialog + + + Output file : + + + + + Create + + + + + Cancel + + + + + Export comics info + + + + + Destination database name + + + + + Problem found while writing + + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + ExportLibraryDialog + + + Output folder : + + + + + Create + + + + + Cancel + + + + + Create covers package + + + + + Problem found while writing + + + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + + + + + Destination directory + + + + + FileComic + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + + Unknown error opening the file + + + + + 7z not found + + + + + Format not supported + + + + + HelpAboutDialog + + + About + + + + + Help + + + + + ImportComicsInfoDialog + + + Import comics info + + + + + Info database location : + + + + + Import + + + + + Cancel + + + + + Comics info file (*.ydb) + + + + + ImportLibraryDialog + + + Library Name : + + + + + Package location : + + + + + Destination folder : + + + + + Unpack + + + + + Cancel + + + + + Extract a catalog + + + + + Compresed library covers (*.clc) + + + + + ImportWidget + + + stop + + + + + Some of the comics being added... + + + + + Importing comics + + + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + + + + + Updating the library + + + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + + + + + LibraryWindow + + + YACReader Library + + + + + + Library + + + + + <font color='white'> press 'F' to close fullscreen mode </font> + + + + + Create a new library + + + + + Open an existing library + + + + + + Export comics info + + + + + + Import comics info + + + + + Pack covers + + + + + Pack the covers of the selected library + + + + + Unpack covers + + + + + Unpack a catalog + + + + + Update library + + + + + Update current library + + + + + Rename library + + + + + Rename current library + + + + + Remove library + + + + + Remove current library from your collection + + + + + Open current comic + + + + + Open current comic on YACReader + + + + + + Set as read + + + + + Set comic as read + + + + + + Set as unread + + + + + Set comic as unread + + + + + Show/Hide marks + + + + + Show or hide readed marks + + + + + Library not available + Library ' + + + + + Fullscreen mode on/off + + + + + Fullscreen mode on/off (F) + + + + + Help, About YACReader + + + + + Select root node + + + + + + + + + + + Expand all nodes + + + + + - + + + + + Colapse all nodes + + + + + Show options dialog + + + + + Show comics server options dialog + + + + + Open folder... + + + + + Set as uncompleted + + + + + Set as completed + + + + + Open containing folder... + + + + + Reset comic rating + + + + + Select all comics + + + + + Edit + + + + + Asign current order to comics + + + + + Update cover + + + + + Delete selected comics + + + + + Hide comic flow + + + + + Download tags from Comic Vine + + + + + Folder + + + + + Comic + + + + + Update needed + + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + + + + + Update failed + + + + + The current library can't be udpated. Check for write write permissions on: + + + + + Download new version + + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + + + + + Library '%1' is no longer available. Do you want to remove it? + + + + + Old library + + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + + + + + YACReader not found + + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + + Library not found + + + + + The selected folder doesn't contain any library. + + + + + Are you sure? + + + + + Do you want remove + + + + + library? + + + + + Remove and delete metadata + + + + + Unable to delete + + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + + Asign comics numbers + + + + + Asign numbers starting in: + + + + + Error creating the library + + + + + Error updating the library + + + + + Error opening the library + + + + + Delete comics + + + + + All the selected comics will be deleted from your disk. Are you sure? + + + + + Library name already exists + + + + + There is another library with the name '%1'. + + + + + LocalComicListModel + + + file name + + + + + NoLibrariesWidget + + + You don't have any librarires yet + + + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + + + + + create your first library + + + + + add an existing one + + + + + OptionsDialog + + + Options + + + + + PropertiesDialog + + + General info + + + + + Authors + + + + + Publishing + + + + + Plot + + + + + Cover page + + + + + Title: + + + + + Issue number: + + + + + Volume: + + + + + Story arc: + + + + + Genere: + + + + + Size: + + + + + Writer(s): + + + + + Penciller(s): + + + + + Inker(s): + + + + + Colorist(s): + + + + + Letterer(s): + + + + + Cover Artist(s): + + + + + Day: + + + + + Month: + + + + + Year: + + + + + Publisher: + + + + + Format: + + + + + Color/BW: + + + + + Age rating: + + + + + Synopsis: + + + + + Characters: + + + + + Notes: + + + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + Not found + + + + + Comic not found. You should update your library. + + + + + Edit selected comics information + + + + + Edit comic information + + + + + QObject + + + 7z lib not found + + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + + New Library Name : + + + + + Rename + + + + + Cancel + + + + + Rename current library + + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + + + + + + page %1 of %2 + + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + + Please provide some aditional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some aditional information. + + + + + Series: + + + + + SelectComic + + + Please, select the right comic info. + + + + + comics + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SelectVolume + + + Please, select the right series for your comic. + + + + + volumes + + + + + loading cover + + + + + loading description + + + + + description unavailable + + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + + + + + yes + + + + + no + + + + + ServerConfigDialog + + + set port + + + + + EASY SERVER CONNECTION + + + + + SERVER ADDRESS + + + + + just scan the code with your device!! + + + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + + + + + IP address + + + + + Port + + + + + enable the server + + + + + QR generator error! + + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + + + + + sort comics to match comic information + + + + + issues + + + + + remove selected comics + + + + + restore all removed comics + + + + + restore removed comics + + + + + TableModel + + + yes + + + + + no + + + + + Title + + + + + File Name + + + + + Pages + + + + + Size + + + + + Read + + + + + Current Page + + + + + Rating + + + + + TitleHeader + + + SEARCH + + + + + UpdateLibraryDialog + + + Updating.... + + + + + Cancel + + + + + Update library + + + + + VolumeComicsModel + + + title + + + + + VolumesModel + + + year + + + + + issues + + + + + publisher + + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + + + + + cancel + + + + + YACReaderFieldEdit + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + + + + + Restore to default + + + + + YACReaderFlowConfigWidget + + + How to show covers: + + + + + CoverFlow look + + + + + Stripe look + + + + + Overlapped Stripe look + + + + + YACReaderGLFlowConfigWidget + + + Presets: + + + + + Classic look + + + + + Stripe look + + + + + Overlapped Stripe look + + + + + Modern look + + + + + Roulette look + + + + + Show advanced settings + + + + + Custom: + + + + + View angle + + + + + Position + + + + + Cover gap + + + + + Central gap + + + + + Zoom + + + + + Y offset + + + + + Z offset + + + + + Cover Angle + + + + + Visibility + + + + + Light + + + + + Max angle + + + + + Low Performance + + + + + High Performance + + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + + + + + Performance: + + + + + YACReaderOptionsDialog + + + Save + + + + + Cancel + + + + + Use hardware acceleration (restart needed) + + + + + YACReaderSideBar + + + LIBRARIES + + + + + FOLDERS + + + + + Search folders and comics + + + + diff --git a/YACReaderLibrary/yacreaderlibrary_tr.ts b/YACReaderLibrary/yacreaderlibrary_tr.ts new file mode 100644 index 00000000..6cbbbfbe --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_tr.ts @@ -0,0 +1,1308 @@ + + + + + AddLibraryDialog + + Add + Ekle + + + Add an existing library + Kütüphaneye ekle + + + Cancel + Vazgeç + + + Comics folder : + Çizfi roman dosyası : + + + Library Name : + Kütüphane adı : + + + + ComicVineDialog + + skip + + + + back + + + + next + + + + search + + + + close + + + + Looking for volume... + + + + comic %1 of %2 - %3 + + + + %1 comics selected + + + + Error connecting to ComicVine + + + + unknown error + + + + Retrieving tags for : %1 + + + + Retrieving volume info... + + + + Looking for comic... + + + + + CreateLibraryDialog + + Create new library + Yeni kütüphane oluÅŸtur + + + Cancel + Vazgeç + + + Create + OluÅŸtur + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Yeni kütüphanenin oluÅŸturulması birkaç dakika sürecek. + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + Seçilen dizine yazma iznimiz yok yazma izni olduÄŸundan emin ol + + + Comics folder : + Çizgi dosyası: + + + Library Name : + Kütüphane adı: + + + Path not found + Dizin bulunamadı + + + + ExportComicsInfoDialog + + Output file : + Çıkış dosyası : + + + Destination database name + Hedef adı + + + Cancel + Vazgeç + + + Create + OluÅŸtur + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Seçilen dizine yazma iznimiz yok yazma izni olduÄŸundan emin ol + + + Export comics info + Çizgi roman bilgilerini göster + + + Problem found while writing + Yazma sırasında bir problem oldu + + + + ExportLibraryDialog + + Cancel + Vazgeç + + + Create + Yeni bir tane yap + + + The selected path for the output file does not exist or is not a valid path. Be sure that you have write access to this folder + Seçilen konuma yeni bir kütüphane yazılamıyor + + + Output folder : + Çıkış klasörü: + + + Problem found while writing + Yazım aÅŸamasında bir problem bulundu + + + Create covers package + Kapak paketi oluÅŸtur + + + Destination directory + Hedef dizin + + + + FileComic + + File not found or not images in file + Dosya bulunamadı yada dosyada resim yok + + + 7z not found + 7z bulunamadı + + + Comic not found + Çizgi roman bulunamadı + + + Not found + Bulunamadı + + + File error + Dosya hatası + + + 7z problem + 7z Problemli + + + 7z reading + 7z Okuyor + + + 7z crashed. + 7z BozulmuÅŸ. + + + problem reading from 7z + 7z Dosyası Okunamıyor + + + 7z crashed + 7z BozulmuÅŸ + + + Unknown error 7z + Bilinmeyen 7z hatası + + + 7z wasn't found in your PATH. + 7z Dosya Yolu Bulunamadı. + + + CRC error on page (%1): some of the pages will not be displayed correctly + + + + Unknown error opening the file + + + + Format not supported + + + + + HelpAboutDialog + + Help + Yardım + + + About + Hakkında + + + + ImportComicsInfoDialog + + Cancel + Vazgeç + + + Import + Çıkart + + + Info database location : + Bilgi konumu : + + + Import comics info + Çizgi roman bilgilerini çıkart + + + Comics info file (*.ydb) + +Çizgi Roman bilgileri (*.ydb) + + + + ImportLibraryDialog + + Destination folder : + Hedef klasör: + + + Cancel + Vazgeç + + + Unpack + Paketten çıkar + + + Compresed library covers (*.clc) + Sıkıştırılmış kütüphane kapakları (*.clc) + + + Package location : + Paket konumu: + + + Library Name : + Kütüphane Adı : + + + Extract a catalog + Catalog'a çıkart + + + + ImportWidget + + stop + dur + + + Importing comics + önemli çizgi romanlar + + + <p>YACReaderLibrary is now creating a new library.</p><p>Create a library could take several minutes. You can stop the process and update the library later for completing the task.</p> + <p>YACReaderKütüphane ÅŸu anda yeni bir kütüphane oluÅŸturuyor</p><p>Kütüphanenin oluÅŸturulması birkaç dakika alacak.</p> + + + Some of the comics being added... + Bazı çizgi romanlar önceden eklenmiÅŸ... + + + Updating the library + Kütüphaneyi güncelle + + + <p>The current library is being updated. For faster updates, please, update your libraries frequently.</p><p>You can stop the process and continue updating this library later.</p> + <p>Kütüphane güncelleniyor</p><p>Güncellemeyi daha sonra iptal edebilirsin.</p> + + + + LibraryWindow + + + + + + + + - + - + + + Edit + Düzenle + + + The selected folder doesn't contain any library. + Seçilen dosya kütüphanede yok. + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Bu kütüphane YACReaderKütüphabenin bir önceki versiyonun oluÅŸturulmuÅŸ, güncellemeye ihtiyacın var. Åžimdi güncellemek ister misin ? + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> 'F'ye basarak tam ekran modundan çıkabilirsin </font> + + + Asign current order to comics + Asignar el orden actual a los cómics + + + Error opening the library + Haa kütüphanesini aç + + + Show/Hide marks + Altçizgileri aç/kapa + + + Show comics server options dialog + Çizgi romanların server ayarlarını göster + + + Remove current library from your collection + Kütüphaneyi koleksiyonundan kaldır + + + Set comic as read + Çizgi romanı okundu olarak iÅŸaretle + + + Remove and delete metadata + Metadata'yı kaldır ve sil + + + Old library + Eski kütüphane + + + Update cover + Kapağı güncelle + + + Library + Kütüphane + + + Rename current library + Kütüphaneyi adlandır + + + Fullscreen mode on/off + Tam ekran modu açık/kapalı + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Bu kütüphane YACRKütüphanenin üst bir versiyonunda oluÅŸturulmu. Yeni versiyonu indirmek ister misiniz ? + + + + Open current comic on YACReader + YACReader'ı geçerli çizgi roman okuyucsu seç + + + Update current library + Kütüphaneyi güncelle + + + Library '%1' is no longer available. Do you want to remove it? + Kütüphane '%1'ulaşılabilir deÄŸil. Kaldırmak ister misin? + + + Update library + Kütüphaneyi güncelle + + + Open folder... + Dosyayı aç... + + + Do you want remove + Kaldırmak ister misin + + + Error updating the library + Kütüphane güncelleme sorunu + + + Hide comic flow + Çizgi roman akışını gizle + + + Expand all nodes + Tüm düğümleri büyüt + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + Kütüphane '%1 YACRKütüphanenin eski bir sürümünde oluÅŸturulmuÅŸ, Kütüphaneyi yeniden oluÅŸturmak ister misin? + + + There was a problem saving YACReaderLibrary libraries file. Please, check if you have enough permissions in the YACReader root folder. + YACRKütüphane kütüphane dosyaları kaydedilirken bir sorun çıktı. Lütfen, YACReader root dosyalarını kontrol edin. + + + Pack covers + Paket kapakları + + + Set as read + Okundu olarak iÅŸaretle + + + Fullscreen mode on/off (F) + Tam ekran modunu aç/kapa(F) + + + Saving libraries file.... + Kütüphane dosyalarını kaydet... + + + Asign comics numbers + Çizgi roman numaralarını deÄŸiÅŸtir + + + Delete selected comics + Seçili çizgi romanları sil + + + Export comics info + Çizgi roman bilgilerini çıkart + + + Show options dialog + Ayarları göster + + + Create a new library + Yeni kütüphane oluÅŸtur + + + Library not available + Kütüphane ulaşılabilir deÄŸil + + + Import comics info + Çizgi roman bilgilerini içe aktar + + + The current library can't be udpated. Check for write write permissions on: + Kütüphane güncellenmemiÅŸ. Lütfen yazım izinlerini kontrol et: + + + Open current comic + Seçili çizgi romanı aç + + + Colapse all nodes + Tüm düğümleri daralt + + + YACReader Library + YACReader Kütüphane + + + Error creating the library + Kütüphane oluÅŸturma sorunu + + + Update failed + Güncelleme baÅŸarısız + + + Unpack covers + Kapakları aç + + + Update needed + Güncelleme gerekli + + + Open an existing library + Çıkış kütüphanesini aç + + + Library name already exists + Kütüphane ismi zaten alınmış + + + There is another library with the name '%1'. + Bu baÅŸka bir kütüphanenin adı '%1'. + + + Asign numbers starting in: + BaÅŸlangıç sayılarını düzenle: + + + Download new version + Yeni versiyonu indir + + + Delete comics + Çizgi romanları sil + + + Show or hide readed marks + OkunmuÅŸ iÅŸaretleri göster yada gizle + + + Select all comics + Tüm çizgi romanları seç + + + Set all comics as read + Tüm çizgi romanları okundu olarak ayarla + + + Pack the covers of the selected library + Kütüphanede ki kapakları paketle + + + Help, About YACReader + Yardım, Bigli, YACReader + + + Set comic as unread + Çizgi Romanı okunmadı olarak seç + + + Select root node + Kökü seçin + + + Unpack a catalog + KataloÄŸu çkart + + + All the selected comics will be deleted from your disk. Are you sure? + Seçilen tüm çizgi romanlar diskten silinecek emin misin ? + + + Set all as read + Hepsini okundu iÅŸaretle + + + Set as unread + Hepsini okunmadı iÅŸaretle + + + Library not found + Kütüphane bulunamadı + + + Rename library + Kütüphaneyi yeniden adlandır + + + Remove library + Kütüphaneyi sil + + + Open containing folder... + Klasör açılıyor... + + + Set all comics as unread + Tüm çizgiromanları okunmadı olarak iÅŸaretle + + + library? + kütüphane? + + + Set all as unread + Hepsini okunmadı olarak ayarla + + + Are you sure? + Emin misin? + + + Download tags from Comic Vine + + + + YACReader not found + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + + + + Unable to delete + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + + + + Set as uncompleted + + + + Set as completed + + + + Reset comic rating + + + + Folder + + + + Comic + + + + + LocalComicListModel + + file name + + + + + NoLibrariesWidget + + create your first library + İlk kütüphaneni oluÅŸtur + + + You don't have any librarires yet + Henüz bir kütüphaneye sahip deÄŸilsin + + + <p>You can create a library in any folder, YACReaderLibrary will import all comics and folders from this folder. If you have created any library in the past you can open them.</p><p>Don't forget that you can use YACReader as a stand alone application for reading the comics on your computer.</p> + <p>Yeni bir kütüphane oluÅŸturabilmeniçin kütüphane</p><p>No olvides que puedes usar YACReader como una aplicación independiente para leer los cómics en tu ordenador.</p> + + + add an existing one + Var olan bir tane ekle + + + + OptionsDialog + + Options + Ayarlar + + + + PropertiesDialog + + Day: + Gün: + + + Plot + Argumento + + + Size: + Boyut: + + + Year: + Yıl: + + + Inker(s): + Mürekkep(ler): + + + Publishing + Yayın + + + Publisher: + Yayıncı: + + + General info + Genel bilgi + + + Color/BW: + Renk/BW: + + + Edit selected comics information + Seçilen çizgi roman bilgilerini düzenle + + + Penciller(s): + Çizenler: + + + Colorist(s): + Renklendiren: + + + Issue number: + Yayın numarası: + + + Month: + Ay: + + + Notes: + Notlar: + + + Synopsis: + Özet: + + + Title: + BaÅŸlık: + + + Not found + Bulunamad + + + Characters: + Karakterler: + + + Authors + Yazarlar + + + Age rating: + YaÅŸ sınırı: + + + Story arc: + Hiakye: + + + Writer(s): + Yazarlar: + + + Comic not found. You should update your library. + Çizgi roman bulunamadı. Kütüphaneyi güncellemelisin. + + + Edit comic information + Çizgi roman bilgisini düzenle + + + Cover page + Kapak sayfası + + + Cover Artist(s): + Kapak artisti: + + + Volume: + Cilt: + + + Format: + Formato: + + + Genere: + Tür: + + + Letterer(s): + Mesaj(lar): + + + Comic Vine link: <a style='color: #FFCB00; text-decoration:none; font-weight:bold;' href="http://www.comicvine.com/comic/4000-%1/"> view </a> + + + + + QObject + + 7z lib not found + + + + unable to load 7z lib from ./utils + + + + + RenameLibraryDialog + + Rename current library + Kütüphaneyi yeniden adlandır + + + Cancel + Vazgeç + + + Rename + Yeniden adlandır + + + New Library Name : + Yeni Kütüphane Adı : + + + + ScraperResultsPaginator + + Number of volumes found : %1 + + + + page %1 of %2 + + + + Number of %1 found : %2 + + + + + SearchSingleComic + + Please provide some aditional information. + + + + Series: + + + + + SearchVolume + + Please provide some aditional information. + + + + Series: + + + + + SelectComic + + Please, select the right comic info. + + + + comics + + + + loading cover + + + + loading description + + + + description unavailable + + + + + SelectVolume + + Please, select the right series for your comic. + + + + volumes + + + + loading cover + + + + loading description + + + + description unavailable + + + + + SeriesQuestion + + You are trying to get information for various comics at once, are they part of the same series? + + + + yes + evet + + + no + hayır + + + + ServerConfigDialog + + Port + Port + + + EASY SERVER CONNECTION + KOLAY SERVER BAÄžLANTISI + + + just scan the code with your device!! + Sadece kodu cihaza tarat !! + + + enable the server + eriÅŸilebilir server + + + IP address + IP adres + + + YACReader is now available for iOS devices, the best comic reading experience now in your iPad, iPhone or iPod touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> Discover it! </a> + YACReader ÅŸimdi iOS cihazlarda Hemen iPad, iPhone veya iPod Touch'ına kapmak için tıkla (Çevirisi yapılmayacak) <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'> ¡Descúbrelo! </a> + + + QR generator error! + QR kod oluÅŸturma hatası! + + + set port + Port Ayarla + + + SERVER ADDRESS + Server Adres + + + + SortVolumeComics + + Please, sort the list of comics on the left until it matches the comics' information. + + + + sort comics to match comic information + + + + issues + + + + remove selected comics + + + + restore all removed comics + + + + restore removed comics + + + + + TableModel + + no + hayır + + + yes + evet + + + Read + Oku + + + Size + Boyut + + + Pages + Sayfalar + + + Title + BaÅŸlık + + + File Name + Dosya Adı + + + Current Page + + + + Rating + + + + + TitleHeader + + SEARCH + + + + + UpdateLibraryDialog + + Update library + Kütüphaneyi güncelle + + + Cancel + Vazgeç + + + Updating.... + Güncelleniyor... + + + + VolumeComicsModel + + title + + + + + VolumesModel + + year + + + + issues + + + + publisher + + + + + YACReaderDeletingProgress + + cancel + vazgeç + + + Please wait, deleting in progress... + Lütfen bekleyin, silme iÅŸlemi yapılıyor... + + + + YACReaderFieldEdit + + Restore to default + Varsayılana dön + + + Click to overwrite + Üstüne yazmak için tıkla + + + + YACReaderFieldPlainTextEdit + + Restore to default + Varsayılana dön + + + Click to overwrite + Üstüne yazmak için tıkla + + + + YACReaderFlowConfigWidget + + CoverFlow look + Kapak akışı görünümü + + + How to show covers: + Kapaklar nasıl gözüksün: + + + Stripe look + Åžerit görünüm + + + Overlapped Stripe look + Çakışan ÅŸerit görünüm + + + + YACReaderGLFlowConfigWidget + + Zoom + Zoom + + + Light + Işık + + + Show advanced settings + Daha fazla ayar göster + + + Roulette look + Rulet görünüm + + + Cover Angle + Kapak Açısı + + + Stripe look + Strip görünüm + + + Position + Pozisyon + + + Z offset + Z dengesi + + + Y offset + Y dengesi + + + Central gap + BoÅŸ merkez + + + Presets: + Hazırlayan: + + + Overlapped Stripe look + Çakışan ÅŸerit görünüm + + + Modern look + Modern görünüm + + + View angle + Bakış açısı + + + Max angle + Maksimum açı + + + Custom: + KiÅŸisel: + + + Classic look + Klasik görünüm + + + Cover gap + Kapak + + + High Performance + Yüksek Performans + + + Performance: + Performans: + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + VSync kullan + + + Visibility + Görünülebilirlik + + + Low Performance + Düşük Performans + + + + YACReaderOptionsDialog + + Save + Kaydet + + + Use hardware acceleration (restart needed) + Yüksek donanımlı kullan (yeniden baÅŸlatmak gerekli) + + + Cancel + Vazgeç + + + + YACReaderSideBar + + Search folders and comics + Klasörleri ve çizgi romanları ara + + + LIBRARIES + KÜTÜPHANELER + + + FOLDERS + DOSYALAR + + + + YACReaderSocialDialog + + I am reading %1 using YACReader. + YACReader ile okuyorum %1. + + + send to: + Gönder: + + + Follow YACReader! + YACReader'ı takip et ! + + + diff --git a/background.png b/background.png new file mode 100755 index 0000000000000000000000000000000000000000..b9a7e42ec0e3564f4b03bcb5bafdcf785f4a908b GIT binary patch literal 3297 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 literal 0 HcmV?d00001 diff --git a/cleanOSX.sh b/cleanOSX.sh new file mode 100755 index 00000000..0e55aa9e --- /dev/null +++ b/cleanOSX.sh @@ -0,0 +1,13 @@ +#!/bin/bash + +rm -R *.app +rm -R YACReader-* +rm -R *.dmg +cd YACReader +make clean +rm -R YACReader.app +cd .. +cd YACReaderLibrary +make clean +rm -R YACReaderLibrary.app +cd .. diff --git a/common/bookmarks.cpp b/common/bookmarks.cpp new file mode 100644 index 00000000..d60d8b60 --- /dev/null +++ b/common/bookmarks.cpp @@ -0,0 +1,174 @@ +#include "bookmarks.h" +#include +#include +#include +#include + +#include +#include + +#include "yacreader_global.h" + +Bookmarks::Bookmarks() +:lastPageIndex(0) +{ + list.load(); +} +void Bookmarks::setLastPage(int index,const QImage & page) +{ + lastPageIndex = index; + lastPage = page; +} +void Bookmarks::setBookmark(int index,const QImage & page) +{ + if(!bookmarks.contains(index)) + { + bookmarks.insert(index,page); + latestBookmarks.push_front(index); + if(latestBookmarks.count()>3) + { + bookmarks.remove(latestBookmarks.back()); + latestBookmarks.pop_back(); + } + } + else //udate de pixmap; + { + bookmarks[index]=page; + } +} + +void Bookmarks::removeBookmark(int index) +{ + bookmarks.remove(index); +} + +QList Bookmarks::getBookmarkPages() const +{ + return bookmarks.keys(); +} + +QImage Bookmarks::getBookmarkPixmap(int page) const +{ + return bookmarks.value(page); +} + +QImage Bookmarks::getLastPagePixmap() const +{ + return lastPage; +} + +int Bookmarks::getLastPage() const +{ + return lastPageIndex; +} + + +bool Bookmarks::isBookmark(int page) +{ + return bookmarks.contains(page); +} + +bool Bookmarks::imageLoaded(int page) +{ + return !bookmarks.value(page).isNull(); +} + +void Bookmarks::newComic(const QString & path) +{ + QFileInfo f(path); + QString comicID = f.fileName().toLower()+QString::number(f.size()); + clear(); + BookmarksList::Bookmark b = list.get(comicID); + comicPath=comicID; + lastPageIndex = b.lastPage; + latestBookmarks = b.bookmarks; + for(int i=0;i & bookmarkIndexes, int lastPage) +{ + lastPageIndex = lastPage; + foreach(int b, bookmarkIndexes) + if(b != -1) + { + latestBookmarks.push_back(b); + bookmarks.insert(b,QImage()); + } + + return true; +} + +void Bookmarks::save() +{ + BookmarksList::Bookmark b; + b.lastPage = lastPageIndex; + b.bookmarks = getBookmarkPages(); + + BookmarksList::Bookmark previousBookmarks; + bool updated = ((previousBookmarks.lastPage != b.lastPage) || (previousBookmarks.bookmarks != b.bookmarks)); + + if(b.added.isNull() || updated) + b.added = QDateTime::currentDateTime(); + list.add(comicPath,b); + list.save(); +} +//----------------------------------------------------------------------------- +void BookmarksList::load() +{ + QFile f(YACReader::getSettingsPath()+"/bookmarks.yacr"); + if(f.open(QIODevice::ReadOnly)) + { + QDataStream dataS(&f); + dataS >> list; + f.close(); + } +} + +void BookmarksList::save() +{ + QFile f(YACReader::getSettingsPath()+"/bookmarks.yacr"); + f.open(QIODevice::WriteOnly); + QDataStream dataS(&f); + if(list.count()>numMaxBookmarks) + deleteOldest(list.count()-numMaxBookmarks); + dataS << list; + f.close(); +} + + +void BookmarksList::deleteOldest(int num) +{ + Q_UNUSED(num) + QString comic; + QDateTime date(QDate(10000,1,1));//TODO MAX_DATE?? + for(QMap::const_iterator itr=list.begin();itr!=list.end();itr++) + { + if(itr->addedadded; + } + } + list.remove(comic); +} + +void BookmarksList::add(const QString & comicID, const Bookmark & b) +{ + list.insert(comicID,b); +} + +BookmarksList::Bookmark BookmarksList::get(const QString & comicID) +{ + //if(list.contains(comicID) + return list.value(comicID); +} diff --git a/common/bookmarks.h b/common/bookmarks.h new file mode 100644 index 00000000..e7d3c43b --- /dev/null +++ b/common/bookmarks.h @@ -0,0 +1,80 @@ +#ifndef BOOKMARKS_H +#define BOOKMARKS_H + +#include +#include +#include +#include +#include +#include +class BookmarksList +{ +public: + struct Bookmark { + int lastPage; + QList bookmarks; + QDateTime added; + Bookmark():lastPage(0){}; + friend QDataStream & operator<< ( QDataStream & out, const Bookmark & bm ) + { + out << bm.lastPage; + out << bm.bookmarks; + out << bm.added; + return out; + } + friend QDataStream & operator>> ( QDataStream & in, Bookmark & bm ) + { + in >> bm.lastPage; + in >> bm.bookmarks; + in >> bm.added; + return in; + } + + }; + BookmarksList():numMaxBookmarks(400){} + void load(); + void save(); + void add(const QString & comicID, const Bookmark & b); + Bookmark get(const QString & comicID); +protected: + QMap list; + void deleteOldest(int num); +private: + int numMaxBookmarks; + +}; + +class Bookmarks : public QObject +{ + Q_OBJECT + + protected: + QString comicPath; + //bookmarks setted by the user + QMap bookmarks; + QList latestBookmarks; + //last page readed + int lastPageIndex; + QImage lastPage; + BookmarksList list; + QDateTime added; + + public: + Bookmarks(); + void setLastPage(int index,const QImage & page); + void setBookmark(int index,const QImage & page); + void removeBookmark(int index); + QList getBookmarkPages() const; + QImage getBookmarkPixmap(int page) const; + QImage getLastPagePixmap() const; + int getLastPage() const; + bool isBookmark(int page); + bool imageLoaded(int page); + void newComic(const QString & path); + void clear(); + void save(); + bool load(const QList & bookmarkIndexes, int lastPage); + +}; + +#endif // BOOKMARKS_H diff --git a/common/check_new_version.cpp b/common/check_new_version.cpp new file mode 100644 index 00000000..e27c0562 --- /dev/null +++ b/common/check_new_version.cpp @@ -0,0 +1,84 @@ +#include "check_new_version.h" +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#define PREVIOUS_VERSION "6.0.0" + +HttpVersionChecker::HttpVersionChecker() + :HttpWorker("https://bitbucket.org/luisangelsm/yacreader/wiki/Home") +{ + connect(this,SIGNAL(dataReady(const QByteArray &)),this,SLOT(checkNewVersion(const QByteArray &))); +} + +void HttpVersionChecker::checkNewVersion(const QByteArray & data) +{ + checkNewVersion(QString(data)); +} + +bool HttpVersionChecker::checkNewVersion(QString sourceContent) +{ +#ifdef Q_OS_WIN32 + QRegExp rx(".*YACReader\\-([0-9]+).([0-9]+).([0-9]+)\\.?([0-9]+)?.{0,5}win32.*"); +#endif + +#ifdef Q_OS_LINUX + QRegExp rx(".*YACReader\\-([0-9]+).([0-9]+).([0-9]+)\\.?([0-9]+)?.{0,5}X11.*"); +#endif + +#ifdef Q_OS_MAC + QRegExp rx(".*YACReader\\-([0-9]+).([0-9]+).([0-9]+)\\.?([0-9]+)?.{0,5}Mac.*"); +#endif + + int index = 0; + bool newVersion = false; + bool sameVersion = true; + //bool currentVersionIsNewer = false; +#ifdef QT_DEBUG + QString version(PREVIOUS_VERSION); +#else + QString version(VERSION); +#endif + QStringList sl = version.split("."); + if((index = rx.indexIn(sourceContent))!=-1) + { + int length = qMin(sl.size(),(rx.cap(4)!="")?4:3); + for(int i=0;isl.at(i).toInt()){ + newVersion=true; + break; + } + else + sameVersion = sameVersion && rx.cap(i+1).toInt()==sl.at(i).toInt(); + } + if(!newVersion && sameVersion) + { + if((sl.size()==3)&&(rx.cap(4)!="")) + newVersion = true; + } + + + } + + if(newVersion == true) + { + emit newVersionDetected(); + return true; + } + else + { + return false; + } +} diff --git a/common/check_new_version.h b/common/check_new_version.h new file mode 100644 index 00000000..5c5e2fb5 --- /dev/null +++ b/common/check_new_version.h @@ -0,0 +1,27 @@ +#ifndef __CHECKUPDATE_H +#define __CHECKUPDATE_H + +#include "http_worker.h" +#include "yacreader_global.h" + +#include +#include +#include + + class HttpVersionChecker : public HttpWorker + { + Q_OBJECT + public: + HttpVersionChecker(); + public slots: + + private: + bool found; + private slots: + bool checkNewVersion(QString sourceContent); + void checkNewVersion(const QByteArray & data); + signals: + void newVersionDetected(); + }; + +#endif diff --git a/common/comic.cpp b/common/comic.cpp new file mode 100644 index 00000000..6c5b4821 --- /dev/null +++ b/common/comic.cpp @@ -0,0 +1,743 @@ +#include "comic.h" + +#include +#include +#include +#include +#include +#include +#include + +#include "bookmarks.h" //TODO desacoplar la dependencia con bookmarks +#include "qnaturalsorting.h" +#include "compressed_archive.h" +#include "comic_db.h" + +QStringList Comic::extensions = QStringList() << "*.jpg" << "*.jpeg" << "*.png" << "*.gif" << "*.tiff" << "*.tif" << "*.bmp"; +QStringList Comic::literalExtensions = QStringList() << ".jpg" << ".jpeg" << ".png" << ".gif" << ".tiff" << ".tif" << ".bmp"; + +//----------------------------------------------------------------------------- +Comic::Comic() +:_pages(),_index(0),_path(),_loaded(false),bm(new Bookmarks()),_loadedPages(),_isPDF(false) +{ + setup(); +} +//----------------------------------------------------------------------------- +Comic::Comic(const QString & pathFile, int atPage ) +:_pages(),_index(0),_path(pathFile),_loaded(false),bm(new Bookmarks()),_loadedPages(),_isPDF(false),_firstPage(atPage) +{ + setup(); +} +//----------------------------------------------------------------------------- +Comic::~Comic() +{ + delete bm; +} +//----------------------------------------------------------------------------- +void Comic::setup() +{ + connect(this,SIGNAL(pageChanged(int)),this,SLOT(checkIsBookmark(int))); + connect(this,SIGNAL(imageLoaded(int)),this,SLOT(updateBookmarkImage(int))); + connect(this,SIGNAL(imageLoaded(int)),this,SLOT(setPageLoaded(int))); +} +//----------------------------------------------------------------------------- +int Comic::nextPage() +{ + if(_index<_pages.size()-1) + { + _index++; + + emit pageChanged(_index); + } + else + emit isLast(); + return _index; +} +//--------------------------------------------------------------------------- +int Comic::previousPage() +{ + if(_index>0) + { + _index--; + + emit pageChanged(_index); + } + else + emit isCover(); + + return _index; +} +//----------------------------------------------------------------------------- +void Comic::setIndex(unsigned int index) +{ + int previousIndex = _index; + if(static_cast(index)<_pages.size()-1) + _index = index; + else + _index = _pages.size()-1; + + if(previousIndex != _index) + emit pageChanged(_index); +} +//----------------------------------------------------------------------------- +/*QPixmap * Comic::currentPage() +{ + QPixmap * p = new QPixmap(); + p->loadFromData(_pages[_index]); + return p; +} +//----------------------------------------------------------------------------- +QPixmap * Comic::operator[](unsigned int index) +{ + QPixmap * p = new QPixmap(); + p->loadFromData(_pages[index]); + return p; +}*/ +bool Comic::load(const QString & path, const ComicDB & comic) +{ + Q_UNUSED(path); + Q_UNUSED(comic); + return false; +}; +//----------------------------------------------------------------------------- +bool Comic::loaded() +{ + return _loaded; +} +//----------------------------------------------------------------------------- +void Comic::loadFinished() +{ + emit imagesLoaded(); +} +//----------------------------------------------------------------------------- +void Comic::setBookmark() +{ + QImage p; + p.loadFromData(_pages[_index]); + bm->setBookmark(_index,p); + //emit bookmarksLoaded(*bm); + emit bookmarksUpdated(); +} +//----------------------------------------------------------------------------- +void Comic::removeBookmark() +{ + bm->removeBookmark(_index); + //emit bookmarksLoaded(*bm); + emit bookmarksUpdated(); +} +//----------------------------------------------------------------------------- +void Comic::saveBookmarks() +{ + QImage p; + p.loadFromData(_pages[_index]); + bm->setLastPage(_index,p); + bm->save(); +} +//----------------------------------------------------------------------------- +void Comic::checkIsBookmark(int index) +{ + emit isBookmark(bm->isBookmark(index)); +} +//----------------------------------------------------------------------------- +void Comic::updateBookmarkImage(int index) +{ + if(bm->isBookmark(index)) + { + QImage p; + p.loadFromData(_pages[index]); + bm->setBookmark(index,p); + emit bookmarksUpdated(); + //emit bookmarksLoaded(*bm); + + } + if(bm->getLastPage() == index) + { + QImage p; + p.loadFromData(_pages[index]); + bm->setLastPage(index,p); + emit bookmarksUpdated(); + //emit bookmarksLoaded(*bm); + } + +} +//----------------------------------------------------------------------------- +void Comic::setPageLoaded(int page) +{ + _loadedPages[page] = true; +} +//----------------------------------------------------------------------------- +QByteArray Comic::getRawPage(int page) +{ + if(page < 0 || page >= _pages.size()) + return QByteArray(); + return _pages[page]; +} +//----------------------------------------------------------------------------- +bool Comic::pageIsLoaded(int page) +{ + if(page < 0 || page >= _pages.size()) + return false; + return _loadedPages[page]; +} + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + +FileComic::FileComic() + :Comic() +{ + +} + +FileComic::FileComic(const QString & path, int atPage ) + :Comic(path,atPage) +{ + load(path,atPage); +} + +FileComic::~FileComic() +{ + _pages.clear(); + _loadedPages.clear(); + _fileNames.clear(); + _newOrder.clear(); + _order.clear(); +} + +bool FileComic::load(const QString & path, int atPage) +{ + QFileInfo fi(path); + + if(fi.exists()) + { + if(atPage == -1) + { + bm->newComic(path); + emit bookmarksUpdated(); + } + _firstPage = atPage; + //emit bookmarksLoaded(*bm); + + _path = QDir::cleanPath(path); + //load files size + + return true; + } + else + { + //QMessageBox::critical(NULL,tr("Not found"),tr("Comic not found")+" : " + path); + emit errorOpening(); + return false; + } +} + +bool FileComic::load(const QString & path, const ComicDB & comic) +{ + QFileInfo fi(path); + + if(fi.exists()) + { + QList bookmarkIndexes; + bookmarkIndexes << comic.info.bookmark1 << comic.info.bookmark2 << comic.info.bookmark3; + if(bm->load(bookmarkIndexes,comic.info.currentPage-1)) + emit bookmarksUpdated(); + _firstPage = comic.info.currentPage-1; + _path = QDir::cleanPath(path); + return true; + } + else + { + //QMessageBox::critical(NULL,tr("Not found"),tr("Comic not found")+" : " + path); + emit errorOpening(); + return false; + } +} + +QList FileComic::filter(const QList & src) +{ + QList extensions = getSupportedImageLiteralFormats(); + QList filtered; + bool fileAccepted = false; + + foreach(QString fileName,src) + { + fileAccepted = false; + if(!fileName.contains("__MACOSX")) + { + foreach(QString extension,extensions) + { + if(fileName.endsWith(extension,Qt::CaseInsensitive)) + { + fileAccepted = true; + break; + } + } + } + if(fileAccepted) + filtered.append(fileName); + } + + return filtered; +} + +//DELEGATE methods +void FileComic::fileExtracted(int index, const QByteArray & rawData) +{ + /*QFile f("c:/temp/out2.txt"); + f.open(QIODevice::Append); + QTextStream out(&f);*/ + int sortedIndex = _fileNames.indexOf(_order.at(index)); + //out << sortedIndex << " , "; + //f.close(); + if(sortedIndex == -1) + return; + _pages[sortedIndex] = rawData; + emit imageLoaded(sortedIndex); + emit imageLoaded(sortedIndex,_pages[sortedIndex]); +} + +void FileComic::crcError(int index) +{ + emit crcErrorFound(tr("CRC error on page (%1): some of the pages will not be displayed correctly").arg(index+1)); +} + +//TODO: comprobar que si se produce uno de estos errores, la carga del cómic es irrecuperable +void FileComic::unknownError(int index) +{ + Q_UNUSED(index) + emit errorOpening(tr("Unknown error opening the file")); + //emit errorOpening(); +} + +//-------------------------------------- + +QList > FileComic::getSections(int & sectionIndex) +{ + QVector sortedIndexes; + foreach(QString name, _fileNames) + { + sortedIndexes.append(_order.indexOf(name)); + } + QList > sections; + quint32 previous = 0; + sectionIndex = -1; + int sectionCount = 0; + QVector section; + int idx = 0; + unsigned int realIdx; + foreach(quint32 i, sortedIndexes) + { + + if(_firstPage == idx) + { + sectionIndex = sectionCount; + realIdx = i; + } + if(previous <= i) + { + //out << "idx : " << i << endl; + section.append(i); + previous = i; + } + else + { + if(sectionIndex == sectionCount) //found + { + if(section.indexOf(realIdx)!=0) + { + QVector section1; + QVector section2; + foreach(quint32 si,section) + { + if(si (); + //out << "---------------" << endl; + section.append(i); + //out << "idx : " << i << endl; + previous = i; + sectionCount++; + } + + idx++; + } + if(sectionIndex == sectionCount) //found + { + if(section.indexOf(realIdx)!=0) + { + QVector section1; + QVector section2; + foreach(quint32 si,section) + { + if(si(_fileNames.size(),false); + + emit pageChanged(0); // this indicates new comic, index=0 + emit numPages(_pages.size()); + _loaded = true; + + _cfi=0; + qSort(_fileNames.begin(),_fileNames.end(), naturalSortLessThanCI); + + if(_firstPage == -1) + _firstPage = bm->getLastPage(); + _index = _firstPage; + emit(openAt(_index)); + + int sectionIndex; + QList > sections = getSections(sectionIndex); + + for(int i = sectionIndex; i(),this); + /* + foreach(QString name,_fileNames) + { + index = _order.indexOf(name); + sortedIndex = _fileNames.indexOf(name); + _pages[sortedIndex] = allData.at(index); + emit imageLoaded(sortedIndex); + emit imageLoaded(sortedIndex,_pages[sortedIndex]); + }*/ + + emit imagesLoaded(); + //moveToThread(QApplication::instance()->thread()); +} + + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + +FolderComic::FolderComic() + :Comic() +{ + +} + +FolderComic::FolderComic(const QString & path, int atPage ) + :Comic(path, atPage ) +{ + load(path, atPage ); +} + +FolderComic::~FolderComic() +{ + +} + +bool FolderComic::load(const QString & path, int atPage ) +{ + _path = path; + if(atPage == -1) + { + bm->newComic(_path); + emit bookmarksUpdated(); + } + _firstPage = atPage; + //emit bookmarksLoaded(*bm); + return true; +} + +void FolderComic::process() +{ + QDir d(_path); + + d.setNameFilters(getSupportedImageFormats()); + d.setFilter(QDir::Files|QDir::NoDotAndDotDot); + //d.setSorting(QDir::Name|QDir::IgnoreCase|QDir::LocaleAware); + QFileInfoList list = d.entryInfoList(); + + qSort(list.begin(),list.end(),naturalSortLessThanCIFileInfo); + + int nPages = list.size(); + _pages.clear(); + _pages.resize(nPages); + _loadedPages = QVector(nPages,false); + + if(nPages==0) + { + //TODO emitir este mensaje en otro sitio + //QMessageBox::critical(NULL,QObject::tr("No images found"),QObject::tr("There are not images on the selected folder")); + emit errorOpening(); + } + else + { + if(_firstPage == -1) + _firstPage = bm->getLastPage(); + + _index = _firstPage; + + emit(openAt(_index)); + + emit pageChanged(0); // this indicates new comic, index=0 + emit numPages(_pages.size()); + _loaded = true; + + int count=0; + int i=_firstPage; + while(countthread()); +} + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + +PDFComic::PDFComic() + :Comic() +{ + +} + +PDFComic::PDFComic(const QString & path, int atPage) + :Comic(path,atPage) +{ + load(path,atPage); +} + +PDFComic::~PDFComic() +{ + +} + +bool PDFComic::load(const QString & path, int atPage) +{ + QFileInfo fi(path); + + if(fi.exists()) + { + _path = path; + if(atPage == -1) + { + bm->newComic(_path); + emit bookmarksUpdated(); + } + _firstPage = atPage; + //emit bookmarksLoaded(*bm); + return true; + } + else + { + emit errorOpening(); + return false; + } +} + +bool PDFComic::load(const QString & path, const ComicDB & comic) +{ + QFileInfo fi(path); + + if(fi.exists()) + { + QList bookmarkIndexes; + bookmarkIndexes << comic.info.bookmark1 << comic.info.bookmark2 << comic.info.bookmark3; + if(bm->load(bookmarkIndexes,comic.info.currentPage-1)) + emit bookmarksUpdated(); + _firstPage = comic.info.currentPage-1; + _path = QDir::cleanPath(path); + return true; + } + else + { + //QMessageBox::critical(NULL,tr("Not found"),tr("Comic not found")+" : " + path); + emit errorOpening(); + return false; + } +} + +void PDFComic::process() +{ +#ifdef Q_OS_MAC + pdfComic = new MacOSXPDFComic(); + if(!pdfComic->openComic(_path)) + { + delete pdfComic; + emit errorOpening(); + return; + } +#else + + + pdfComic = Poppler::Document::load(_path); + if (!pdfComic) + { + //delete pdfComic; + //pdfComic = 0; + emit errorOpening(); + return; + } + + + + //pdfComic->setRenderHint(Poppler::Document::Antialiasing, true); + pdfComic->setRenderHint(Poppler::Document::TextAntialiasing, true); +#endif + + int nPages = pdfComic->numPages(); + emit pageChanged(0); // this indicates new comic, index=0 + emit numPages(nPages); + _loaded = true; + //QMessageBox::critical(NULL,QString("%1").arg(nPages),tr("Invalid PDF file")); + + _pages.clear(); + _pages.resize(nPages); + _loadedPages = QVector(nPages,false); + + if(_firstPage == -1) + _firstPage = bm->getLastPage(); + _index = _firstPage; + emit(openAt(_index)); + + for(int i=_index;ithread()); +} + +void PDFComic::renderPage(int page) +{ +#ifdef Q_OS_MAC + { + QImage img = pdfComic->getPage(page); + if(!img.isNull()) + { + QByteArray ba; + QBuffer buf(&ba); + img.save(&buf, "jpg"); + _pages[page] = ba; + emit imageLoaded(page); + emit imageLoaded(page,_pages[page]); + } + } + pdfComic->releaseLastPageData(); +#else + Poppler::Page* pdfpage = pdfComic->page(page); + if (pdfpage) + { + QImage img = pdfpage->renderToImage(150,150); + delete pdfpage; + QByteArray ba; + QBuffer buf(&ba); + img.save(&buf, "jpg"); + _pages[page] = ba; + emit imageLoaded(page); + emit imageLoaded(page,_pages[page]); + } +#endif +} + +Comic * FactoryComic::newComic(const QString & path) +{ + + QFileInfo fi(path); + if(fi.exists()) + { + if(fi.isFile()) + { + if(fi.suffix().compare("pdf",Qt::CaseInsensitive) == 0) + return new PDFComic(); + else + return new FileComic(); + } + else + { + if(fi.isDir()) + return new FolderComic(); + else + return NULL; + } + } + else + return NULL; + +} diff --git a/common/comic.h b/common/comic.h new file mode 100644 index 00000000..d034c470 --- /dev/null +++ b/common/comic.h @@ -0,0 +1,175 @@ +#ifndef __COMIC_H +#define __COMIC_H +#include +#include +#include +#include +#include + +#include "extract_delegate.h" + +#include "bookmarks.h" + +#ifdef Q_OS_MAC + +#include "pdf_comic.h" + +#else + +#if QT_VERSION >= 0x050000 + #include "poppler-qt5.h" +#else + #include "poppler-qt4.h" +#endif + +#endif + +class ComicDB; +//#define EXTENSIONS << "*.jpg" << "*.jpeg" << "*.png" << "*.gif" << "*.tiff" << "*.tif" << "*.bmp" Comic::getSupportedImageFormats() +//#define EXTENSIONS_LITERAL << ".jpg" << ".jpeg" << ".png" << ".gif" << ".tiff" << ".tif" << ".bmp" //Comic::getSupportedImageLiteralFormats() + class Comic : public QObject + { + Q_OBJECT + protected: + //Comic pages, one QPixmap for each file. + QVector _pages; + QVector _loadedPages; + //QVector _sizes; + QStringList _fileNames; + QMap _newOrder; + QList _order; + int _index; + QString _path; + bool _loaded; + + int _cfi; + + //open the comic at this point + int _firstPage; + + bool _isPDF; + + static QStringList extensions; + static QStringList literalExtensions; + public: + Bookmarks * bm; + + //Constructors + Comic(); + Comic(const QString & pathFile, int atPage = -1); + ~Comic(); + void setup(); + //Load pages from file + virtual bool load(const QString & path, int atPage = -1) = 0; + virtual bool load(const QString & path, const ComicDB & comic); + + /*void loadFromFile(const QString & pathFile); + void loadFromDir(const QString & pathDir); + void loadFromPDF(const QString & pathPDF);*/ + int nextPage(); + int previousPage(); + void setIndex(unsigned int index); + unsigned int getIndex(){return _index;}; + unsigned int numPages(){return _pages.size();} + //QPixmap * currentPage(); + bool loaded(); + //QPixmap * operator[](unsigned int index); + QVector * getRawData(){return &_pages;}; + QByteArray getRawPage(int page); + bool pageIsLoaded(int page); + + inline static QStringList getSupportedImageFormats() { return extensions;}; + inline static QStringList getSupportedImageLiteralFormats() { return literalExtensions;}; + + public slots: + void loadFinished(); + void setBookmark(); + void removeBookmark(); + void saveBookmarks(); + void checkIsBookmark(int index); + void updateBookmarkImage(int); + void setPageLoaded(int page); + signals: + void imagesLoaded(); + void imageLoaded(int index); + void imageLoaded(int index,const QByteArray & image); + void pageChanged(int index); + void openAt(int index); + void numPages(unsigned int numPages); + void errorOpening(); + void errorOpening(QString); + void crcErrorFound(QString); + void isBookmark(bool); + void bookmarksUpdated(); + void isCover(); + void isLast(); + + }; + + class FileComic : public Comic, public ExtractDelegate + { + Q_OBJECT + private: + QList > getSections(int & sectionIndex); + public: + FileComic(); + FileComic(const QString & path, int atPage = -1); + ~FileComic(); + void fileExtracted(int index, const QByteArray & rawData); + virtual bool load(const QString & path, int atPage = -1); + virtual bool load(const QString & path, const ComicDB & comic); + void crcError(int index); + void unknownError(int index); + static QList filter(const QList & src); + public slots: + void process(); + }; + + class FolderComic : public Comic + { + Q_OBJECT + private: + //void run(); + public: + FolderComic(); + FolderComic(const QString & path, int atPage = -1); + ~FolderComic(); + + virtual bool load(const QString & path, int atPage = -1); + public slots: + void process(); + + }; + + class PDFComic : public Comic + { + Q_OBJECT + private: + //pdf +#ifdef Q_OS_MAC + MacOSXPDFComic * pdfComic; +#else + Poppler::Document * pdfComic; +#endif + void renderPage(int page); + + //void run(); + public: + PDFComic(); + PDFComic(const QString & path, int atPage = -1); + ~PDFComic(); + + virtual bool load(const QString & path, int atPage = -1); + virtual bool load(const QString & path, const ComicDB & comic); + public slots: + void process(); + }; + + class FactoryComic + { + public: + static Comic * newComic(const QString & path); + }; + + +#endif diff --git a/common/comic_db.cpp b/common/comic_db.cpp new file mode 100644 index 00000000..119dd9a6 --- /dev/null +++ b/common/comic_db.cpp @@ -0,0 +1,477 @@ +#include "comic_db.h" + +#include +#include + +//----------------------------------------------------------------------------- +//COMIC------------------------------------------------------------------------ +//----------------------------------------------------------------------------- +ComicDB::ComicDB() +{ + +} + +bool ComicDB::isDir() +{ + return false; +} + +QString ComicDB::toTXT() +{ + QString txt; + + //Legacy info + txt.append(QString("comicid:%1\r\n").arg(id)); + txt.append(QString("hash:%1\r\n").arg(info.hash)); + txt.append(QString("path:%1\r\n").arg(path)); + txt.append(QString("numpages:%1\r\n").arg(info.numPages.toString())); + + //new 7.0 + txt.append(QString("rating:%1\r\n").arg(info.rating)); + txt.append(QString("currentPage:%1\r\n").arg(info.currentPage)); + txt.append(QString("contrast:%1\r\n").arg(info.contrast)); + + //Información general + if(!info.coverPage.isNull()) + txt.append(QString("coverPage:%1\r\n").arg(info.coverPage.toString())); + + if(!info.title.isNull()) + txt.append(QString("title:%1\r\n").arg(info.title.toString())); + + if(!info.number.isNull()) + txt.append(QString("number:%1\r\n").arg(info.number.toString())); + + if(!info.isBis.isNull()) + txt.append(QString("isBis:%1\r\n").arg(info.isBis.toBool()?"1":"0")); + + if(!info.count.isNull()) + txt.append(QString("count:%1\r\n").arg(info.count.toString())); + + if(!info.volume.isNull()) + txt.append(QString("volume:%1\r\n").arg(info.volume.toString())); + + if(!info.storyArc.isNull()) + txt.append(QString("storyArc:%1\r\n").arg(info.storyArc.toString())); + + if(!info.arcNumber.isNull()) + txt.append(QString("arcNumber:%1\r\n").arg(info.arcNumber.toString())); + + if(!info.arcCount.isNull()) + txt.append(QString("arcCount:%1\r\n").arg(info.arcCount.toString())); + + if(!info.genere.isNull()) + txt.append(QString("genere:%1\r\n").arg(info.genere.toString())); + + //Autores + if(!info.writer.isNull()) + txt.append(QString("writer:%1\r\n").arg(info.writer.toString())); + + if(!info.penciller.isNull()) + txt.append(QString("penciller:%1\r\n").arg(info.penciller.toString())); + + if(!info.inker.isNull()) + txt.append(QString("inker:%1\r\n").arg(info.inker.toString())); + + if(!info.colorist.isNull()) + txt.append(QString("colorist:%1\r\n").arg(info.colorist.toString())); + + if(!info.letterer.isNull()) + txt.append(QString("letterer:%1\r\n").arg(info.letterer.toString())); + + if(!info.coverArtist.isNull()) + txt.append(QString("coverArtist:%1\r\n").arg(info.coverArtist.toString())); + //Publicación + if(!info.date.isNull()) + txt.append(QString("date:%1\r\n").arg(info.date.toString())); + + if(!info.publisher.isNull()) + txt.append(QString("publisher:%1\r\n").arg(info.publisher.toString())); + + if(!info.format.isNull()) + txt.append(QString("format:%1\r\n").arg(info.format.toString())); + + if(!info.color.isNull()) + txt.append(QString("color:%1\r\n").arg(info.color.toString())); + + if(!info.ageRating.isNull()) + txt.append(QString("ageRating:%1\r\n").arg(info.ageRating.toString())); + //Argumento + if(!info.synopsis.isNull()) + txt.append(QString("synopsis:%1\r\n").arg(info.synopsis.toString())); + + if(!info.characters.isNull()) + txt.append(QString("characters:%1\r\n").arg(info.characters.toString())); + + if(!info.notes.isNull()) + txt.append(QString("notes:%1\r\n").arg(info.notes.toString())); + + return txt; +} + +QString ComicDB::getFileName() const +{ + return QFileInfo(path).fileName(); +} + +QString ComicDB::getTitleOrFileName() const +{ + if(!info.title.isNull() && info.title.toString().isEmpty()) + return info.title.toString(); + else + return QFileInfo(path).fileName(); +} + +QString ComicDB::getParentFolderName() const +{ + QStringList paths = path.split('/'); + if(paths.length()<2) + return ""; + else + return paths[paths.length()-2]; +} + +//----------------------------------------------------------------------------- +//COMIC_INFO------------------------------------------------------------------- +//----------------------------------------------------------------------------- +ComicInfo::ComicInfo() + :existOnDb(false), + rating(0), + hasBeenOpened(false), + currentPage(1), + bookmark1(-1), + bookmark2(-1), + bookmark3(-1), + brightness(-1), + contrast(-1), + gamma(-1) +{ + +} + +ComicInfo::ComicInfo(const ComicInfo & comicInfo) +{ + operator=(comicInfo); +} + +ComicInfo::~ComicInfo() +{ + +} +//the default operator= should work +ComicInfo & ComicInfo::operator=(const ComicInfo & comicInfo) +{ + hash = comicInfo.hash; + id = comicInfo.id; + existOnDb = comicInfo.existOnDb; + read = comicInfo.read; + edited = comicInfo.edited; + + hasBeenOpened = comicInfo.hasBeenOpened; + rating = comicInfo.rating; + currentPage = comicInfo.currentPage; + bookmark1 = comicInfo.bookmark1; + bookmark2 = comicInfo.bookmark2; + bookmark3 = comicInfo.bookmark3; + brightness = comicInfo.brightness; + contrast = comicInfo.contrast; + gamma = comicInfo.gamma; + + title = comicInfo.title; + coverPage = comicInfo.coverPage; + numPages = comicInfo.numPages; + number = comicInfo.number; + isBis = comicInfo.isBis; + count = comicInfo.count; + volume = comicInfo.volume; + storyArc = comicInfo.storyArc; + arcNumber = comicInfo.arcNumber; + arcCount = comicInfo.arcCount; + genere = comicInfo.genere; + writer = comicInfo.writer; + penciller = comicInfo.penciller; + inker = comicInfo.inker; + colorist = comicInfo.colorist; + letterer = comicInfo.letterer; + coverArtist = comicInfo.coverArtist; + date = comicInfo.date; + publisher = comicInfo.publisher; + format = comicInfo.format; + color = comicInfo.color; + ageRating = comicInfo.ageRating; + synopsis = comicInfo.synopsis; + characters = comicInfo.characters; + notes = comicInfo.notes; + comicVineID = comicInfo.comicVineID; + + return *this; +} + +//set fields +/* +void ComicInfo::setTitle(QString value) +{ + setValue(title,value); +} + +void ComicInfo::setCoverPage(int value) +{ + setValue(coverPage,value); +} +void ComicInfo::setNumPages(int value) +{ + setValue(numPages,value); +} + +void ComicInfo::setNumber(int value) +{ + setValue(number,value); +} + +void ComicInfo::setIsBis(bool value) +{ + setValue(isBis,value); +} + +void ComicInfo::setCount(int value) +{ + setValue(count,value); +} + +void ComicInfo::setVolume(QString value) +{ + setValue(volume,value); +} + +void ComicInfo::setStoryArc(QString value) +{ + setValue(storyArc,value); +} + +void ComicInfo::setArcNumber(int value) +{ + setValue(arcNumber,value); +} + +void ComicInfo::setArcCount(int value) +{ + setValue(arcCount,value); +} + +void ComicInfo::setGenere(QString value) +{ + setValue(genere,value); +} + +void ComicInfo::setWriter(QString value) +{ + setValue(writer,value); +} + +void ComicInfo::setPenciller(QString value) +{ + setValue(penciller,value); +} + +void ComicInfo::setInker(QString value) +{ + setValue(inker,value); +} + +void ComicInfo::setColorist(QString value) +{ + setValue(colorist,value); +} + +void ComicInfo::setLetterer(QString value) +{ + setValue(letterer,value); +} + +void ComicInfo::setCoverArtist(QString value) +{ + setValue(coverArtist,value); +} + +void ComicInfo::setDate(QString value) +{ + setValue(date,value); +} + +void ComicInfo::setPublisher(QString value) +{ + setValue(publisher,value); +} + +void ComicInfo::setFormat(QString value) +{ + setValue(format,value); +} + +void ComicInfo::setColor(bool value) +{ + setValue(color,value); +} + +void ComicInfo::setAgeRating(QString value) +{ + setValue(ageRating,value); +} + +void ComicInfo::setSynopsis(QString value) +{ + setValue(synopsis,value); +} + +void ComicInfo::setCharacters(QString value) +{ + setValue(characters,value); +} + +void ComicInfo::setNotes(QString value) +{ + setValue(notes,value); +}*/ + +QPixmap ComicInfo::getCover(const QString & basePath) +{ + if(cover.isNull()) + { + cover.load(basePath + "/.yacreaderlibrary/covers/" + hash + ".jpg"); + } + QPixmap c; + c.convertFromImage(cover); + return c; +} +QDataStream &operator<<(QDataStream & stream, const ComicDB & comic) +{ + stream << comic.id; + stream << comic.name; + stream << comic.parentId; + stream << comic.path; + stream << comic._hasCover; + stream << comic.info; + return stream; +} + +QDataStream &operator>>(QDataStream & stream, ComicDB & comic) +{ + stream >> comic.id; + stream >> comic.name; + stream >> comic.parentId; + stream >> comic.path; + stream >> comic._hasCover; + stream >> comic.info; + return stream; +} + +QDataStream &operator<<(QDataStream & stream, const ComicInfo & comicInfo) +{ + stream << comicInfo.id; + stream << comicInfo.read; + stream << comicInfo.edited; + stream << comicInfo.hash; + stream << comicInfo.existOnDb; + + stream << comicInfo.hasBeenOpened; + stream << comicInfo.rating; + stream << comicInfo.currentPage; + stream << comicInfo.bookmark1; + stream << comicInfo.bookmark2; + stream << comicInfo.bookmark3; + stream << comicInfo.brightness; + stream << comicInfo.contrast; + stream << comicInfo.gamma; + + stream << comicInfo.title; + + stream << comicInfo.coverPage; + stream << comicInfo.numPages; + + stream << comicInfo.number; + stream << comicInfo.isBis; + stream << comicInfo.count; + + stream << comicInfo.volume; + stream << comicInfo.storyArc; + stream << comicInfo.arcNumber; + stream << comicInfo.arcCount; + + stream << comicInfo.genere; + + stream << comicInfo.writer; + stream << comicInfo.penciller; + stream << comicInfo.inker; + stream << comicInfo.colorist; + stream << comicInfo.letterer; + stream << comicInfo.coverArtist; + + stream << comicInfo.date; + stream << comicInfo.publisher; + stream << comicInfo.format; + stream << comicInfo.color; + stream << comicInfo.ageRating; + + stream << comicInfo.synopsis; + stream << comicInfo.characters; + stream << comicInfo.notes; + + stream << comicInfo.comicVineID; + + return stream; +} + +QDataStream &operator>>(QDataStream & stream, ComicInfo & comicInfo) +{ + stream >> comicInfo.id; + stream >> comicInfo.read; + stream >> comicInfo.edited; + stream >> comicInfo.hash; + stream >> comicInfo.existOnDb; + + stream >> comicInfo.hasBeenOpened; + stream >> comicInfo.rating; + stream >> comicInfo.currentPage; + stream >> comicInfo.bookmark1; + stream >> comicInfo.bookmark2; + stream >> comicInfo.bookmark3; + stream >> comicInfo.brightness; + stream >> comicInfo.contrast; + stream >> comicInfo.gamma; + + stream >> comicInfo.title; + + stream >> comicInfo.coverPage; + stream >> comicInfo.numPages; + + stream >> comicInfo.number; + stream >> comicInfo.isBis; + stream >> comicInfo.count; + + stream >> comicInfo.volume; + stream >> comicInfo.storyArc; + stream >> comicInfo.arcNumber; + stream >> comicInfo.arcCount; + + stream >> comicInfo.genere; + + stream >> comicInfo.writer; + stream >> comicInfo.penciller; + stream >> comicInfo.inker; + stream >> comicInfo.colorist; + stream >> comicInfo.letterer; + stream >> comicInfo.coverArtist; + + stream >> comicInfo.date; + stream >> comicInfo.publisher; + stream >> comicInfo.format; + stream >> comicInfo.color; + stream >> comicInfo.ageRating; + + stream >> comicInfo.synopsis; + stream >> comicInfo.characters; + stream >> comicInfo.notes; + + stream >> comicInfo.comicVineID; + + return stream; +} diff --git a/common/comic_db.h b/common/comic_db.h new file mode 100644 index 00000000..f3c5cde6 --- /dev/null +++ b/common/comic_db.h @@ -0,0 +1,154 @@ +#ifndef __COMICDB_H +#define __COMICDB_H + +#include "library_item.h" +#include +#include +#include +#include +#include + +class ComicInfo +{ +public: + ComicInfo(); + ComicInfo(const ComicInfo & comicInfo); + ~ComicInfo(); + + ComicInfo & operator=(const ComicInfo & comicInfo); + + //mandatory fields + qulonglong id; + bool read; + bool edited; + QString hash; + bool existOnDb; + + int rating; + + bool hasBeenOpened; + + //viewer + int currentPage; + int bookmark1; + int bookmark2; + int bookmark3; + int brightness; + int contrast; + int gamma; + //----------------- + + + QVariant title;//string + + QVariant coverPage;//int + QVariant numPages;//int + + QVariant number;//int + QVariant isBis;//bool + QVariant count;//int + + QVariant volume;//string + QVariant storyArc;//string + QVariant arcNumber;//int + QVariant arcCount;//int + + QVariant genere;//string + + QVariant writer;//string + QVariant penciller;//string + QVariant inker;//string + QVariant colorist;//string + QVariant letterer;//string + QVariant coverArtist;//string + + QVariant date;//string + QVariant publisher;//string + QVariant format;//string + QVariant color;//bool + QVariant ageRating;//string + + QVariant synopsis;//string + QVariant characters;//string + QVariant notes;//string + + QVariant comicVineID;//string + + QImage cover; + + /*void setTitle(QVariant value); + + void setCoverPage(QVariant value); + void setNumPages(QVariant value); + + void setNumber(QVariant value); + void setIsBis(QVariant value); + void setCount(QVariant value); + + void setVolume(QVariant value); + void setStoryArc(QVariant value); + void setArcNumber(QVariant value); + void setArcCount(QVariant value); + + void setGenere(QVariant value); + + void setWriter(QVariant value); + void setPenciller(QVariant value); + void setInker(QVariant value); + void setColorist(QVariant value); + void setLetterer(QVariant value); + void setCoverArtist(QVariant value); + + void setDate(QVariant value); + void setPublisher(QVariant value); + void setFormat(QVariant value); + void setColor(QVariant value); + void setAgeRating(QVariant value); + + void setSynopsis(QVariant value); + void setCharacters(QVariant value); + void setNotes(QVariant value);*/ + + QPixmap getCover(const QString & basePath); + + friend QDataStream &operator<<(QDataStream & stream, const ComicInfo & comicInfo); + + friend QDataStream &operator>>(QDataStream & stream, ComicInfo & comicInfo); + +private: + +}; + +class ComicDB : public LibraryItem +{ +public: + ComicDB(); + + bool isDir(); + + bool _hasCover; + + bool hasCover() {return _hasCover;}; + + //return comic file name + QString getFileName() const; + + //returns comic title if it isn't null or empty, in other case returns fileName + QString getTitleOrFileName() const; + + //returns parent folder name + QString getParentFolderName() const; + + QString toTXT(); + + ComicInfo info; + + bool operator==(const ComicDB & other){return id == other.id;}; + + friend QDataStream &operator<<(QDataStream &, const ComicDB &); + friend QDataStream &operator>>(QDataStream &, ComicDB &); +}; + +Q_DECLARE_METATYPE(ComicDB); + +#endif diff --git a/common/custom_widgets.cpp b/common/custom_widgets.cpp new file mode 100644 index 00000000..b5b96cfb --- /dev/null +++ b/common/custom_widgets.cpp @@ -0,0 +1,24 @@ +#include "custom_widgets.h" + +#include +#include + +void delTree(QDir dir) +{ + dir.setFilter(QDir::AllDirs|QDir::Files|QDir::Hidden|QDir::NoDotAndDotDot); + QFileInfoList list = dir.entryInfoList(); + for (int i = 0; i < list.size(); ++i) + { + QFileInfo fileInfo = list.at(i); + QString path = fileInfo.filePath(); + if(fileInfo.isDir()) + { + delTree(QDir(fileInfo.absoluteFilePath())); + dir.rmdir(fileInfo.absoluteFilePath()); + } + else + { + dir.remove(fileInfo.absoluteFilePath()); + } + } +} diff --git a/common/custom_widgets.h b/common/custom_widgets.h new file mode 100644 index 00000000..3ca6d5f3 --- /dev/null +++ b/common/custom_widgets.h @@ -0,0 +1,12 @@ +#ifndef __CUSTOM_WIDGETS_H +#define __CUSTOM_WIDGETS_H + +class QDir; + +void delTree(QDir dir); + + + +#endif + + diff --git a/common/exit_check.cpp b/common/exit_check.cpp new file mode 100644 index 00000000..6d1cca54 --- /dev/null +++ b/common/exit_check.cpp @@ -0,0 +1,21 @@ +#include "exit_check.h" + +#include "yacreader_global.h" + +#include + +using namespace YACReader; + +void YACReader::exitCheck(int ret) +{ + switch(ret) + { + case YACReader::SevenZNotFound: + QMessageBox::critical(0,QObject::tr("7z lib not found"),QObject::tr("unable to load 7z lib from ./utils")); + break; + default: + break; + } + +} + diff --git a/common/exit_check.h b/common/exit_check.h new file mode 100644 index 00000000..a2c0b19a --- /dev/null +++ b/common/exit_check.h @@ -0,0 +1,9 @@ +#ifndef EXIT_CHECK_H +#define EXIT_CHECK_H + +namespace YACReader +{ + void exitCheck(int ret); +} + +#endif diff --git a/common/folder.cpp b/common/folder.cpp new file mode 100644 index 00000000..e69de29b diff --git a/common/folder.h b/common/folder.h new file mode 100644 index 00000000..862c05de --- /dev/null +++ b/common/folder.h @@ -0,0 +1,30 @@ +#ifndef __FOLDER_H +#define __FOLDER_H + +#include "library_item.h" + +#include + +class Folder : public LibraryItem +{ +public: + bool knownParent; + bool knownId; + + Folder():knownParent(false), knownId(false){}; + Folder(qulonglong sid, qulonglong pid,QString fn, QString fp):knownParent(true), knownId(true){id = sid; parentId = pid;name = fn; path = fp;}; + Folder(QString fn, QString fp):knownParent(false), knownId(false){name = fn; path = fp;}; + void setId(qulonglong sid){id = sid;knownId = true;}; + void setFather(qulonglong pid){parentId = pid;knownParent = true;}; + bool isDir() {return true;}; + bool isFinished() const {return finished;}; + bool isCompleted() const {return completed;}; + void setFinished(bool b) {finished = b;}; + void setCompleted(bool b) {completed = b;}; + +private: + bool finished; + bool completed; +}; + +#endif diff --git a/common/http_worker.cpp b/common/http_worker.cpp new file mode 100644 index 00000000..eabdcc6c --- /dev/null +++ b/common/http_worker.cpp @@ -0,0 +1,65 @@ +#include "http_worker.h" +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#define PREVIOUS_VERSION "6.0.0" + +HttpWorker::HttpWorker(const QString & urlString) + :QThread(),url(urlString),_error(false),_timeout(false) +{ + +} + +void HttpWorker::get() +{ + this->start(); +} + +QByteArray HttpWorker::getResult() +{ + return result; +} + +bool HttpWorker::wasValid() +{ + return !_error; +} + +bool HttpWorker::wasTimeout() +{ + return _timeout; +} + +void HttpWorker::run() +{ + QNetworkAccessManager manager; + QEventLoop q; + QTimer tT; + + tT.setSingleShot(true); + connect(&tT, SIGNAL(timeout()), &q, SLOT(quit())); + connect(&manager, SIGNAL(finished(QNetworkReply*)),&q, SLOT(quit())); + QNetworkReply *reply = manager.get(QNetworkRequest(url)); + + tT.start(5000); // 5s timeout + q.exec(); + + if(tT.isActive()){ + // download complete + _error = !(reply->error() == QNetworkReply::NoError); + result = reply->readAll(); + emit dataReady(result); + tT.stop(); + } else { + _timeout = true; + emit timeout(); + } +} diff --git a/common/http_worker.h b/common/http_worker.h new file mode 100644 index 00000000..10034717 --- /dev/null +++ b/common/http_worker.h @@ -0,0 +1,32 @@ +#ifndef __HTTP_WORKER_H +#define __HTTP_WORKER_H + +#include +#include +#include +#include +#include "yacreader_global.h" + + class HttpWorker : public QThread + { + Q_OBJECT + public: + HttpWorker(const QString & urlString); + public slots: + void get(); + QByteArray getResult(); + bool wasValid(); + bool wasTimeout(); + private: + void run(); + QUrl url; + int httpGetId; + QByteArray result; + bool _error; + bool _timeout; + signals: + void dataReady(const QByteArray &); + void timeout(); + }; + +#endif diff --git a/common/library_item.cpp b/common/library_item.cpp new file mode 100644 index 00000000..e69de29b diff --git a/common/library_item.h b/common/library_item.h new file mode 100644 index 00000000..2f6b8d9f --- /dev/null +++ b/common/library_item.h @@ -0,0 +1,16 @@ +#ifndef __LIBRARY_ITEM_H +#define __LIBRARY_ITEM_H + +#include + +class LibraryItem +{ +public: + virtual bool isDir() = 0; + QString name; + QString path; + qulonglong parentId; + qulonglong id; +}; + +#endif \ No newline at end of file diff --git a/common/onstart_flow_selection_dialog.cpp b/common/onstart_flow_selection_dialog.cpp new file mode 100644 index 00000000..57247ce0 --- /dev/null +++ b/common/onstart_flow_selection_dialog.cpp @@ -0,0 +1,54 @@ +#include "onstart_flow_selection_dialog.h" + +#include +#include +#include + +OnStartFlowSelectionDialog::OnStartFlowSelectionDialog(QWidget * parent) + :QDialog(parent) +{ + setModal(true); + QPushButton * acceptHW = new QPushButton(this); + connect(acceptHW,SIGNAL(clicked()),this,SLOT(accept())); + QPushButton * rejectHW = new QPushButton(this); //and use SW flow + connect(rejectHW,SIGNAL(clicked()),this,SLOT(reject())); + + acceptHW->setGeometry(90,165,110,118); + acceptHW->setFlat(true); + acceptHW->setAutoFillBackground(true); + rejectHW->setGeometry(464,165,110,118); + rejectHW->setFlat(true); + rejectHW->setAutoFillBackground(true); + + QPalette paletteHW; + QLocale locale = this->locale(); + QLocale::Language language = locale.language(); + + /*if(language == QLocale::Spanish) + paletteHW.setBrush(acceptHW->backgroundRole(), QBrush(QImage(":/images/useNewFlowButton_es.png"))); + else + paletteHW.setBrush(acceptHW->backgroundRole(), QBrush(QImage(":/images/useNewFlowButton.png")));*/ + + + paletteHW.setBrush(acceptHW->backgroundRole(), QBrush(QImage(":/images/nonexxx.png"))); + acceptHW->setPalette(paletteHW); + QPalette paletteSW; + paletteSW.setBrush(rejectHW->backgroundRole(), QBrush(QImage(":/images/nonexxx.png"))); + rejectHW->setPalette(paletteSW); + //QHBoxLayout * layout = new QHBoxLayout; + //layout->addWidget(acceptHW); + //layout->addWidget(rejectHW); + + QPalette palette; + if(language == QLocale::Spanish) + palette.setBrush(this->backgroundRole(), QBrush(QImage(":/images/onStartFlowSelection_es.png"))); + else + palette.setBrush(this->backgroundRole(), QBrush(QImage(":/images/onStartFlowSelection.png"))); + + setPalette(palette); + + + //setLayout(layout); + + resize(664,371); +} diff --git a/common/onstart_flow_selection_dialog.h b/common/onstart_flow_selection_dialog.h new file mode 100644 index 00000000..c333b48b --- /dev/null +++ b/common/onstart_flow_selection_dialog.h @@ -0,0 +1,13 @@ +#ifndef ONSTART_FLOW_SELECTION_DIALOG_H +#define ONSTART_FLOW_SELECTION_DIALOG_H + +#include + +class OnStartFlowSelectionDialog : public QDialog +{ + Q_OBJECT +public: + OnStartFlowSelectionDialog(QWidget * parent = 0); +}; + +#endif \ No newline at end of file diff --git a/common/pdf_comic.h b/common/pdf_comic.h new file mode 100644 index 00000000..7c5d7f48 --- /dev/null +++ b/common/pdf_comic.h @@ -0,0 +1,22 @@ +#ifndef PDF_COMIC_H +#define PDF_COMIC_H + +#include +#include + +class MacOSXPDFComic +{ +public: + MacOSXPDFComic(); + ~MacOSXPDFComic(); + bool openComic(const QString & path); + void closeComic(); + unsigned int numPages(); + QImage getPage(const int page); + void releaseLastPageData(); +private: + void * document; + void * lastPageData; +}; + +#endif // PDF_COMIC_H diff --git a/common/pdf_comic.mm b/common/pdf_comic.mm new file mode 100644 index 00000000..7f9b32ce --- /dev/null +++ b/common/pdf_comic.mm @@ -0,0 +1,117 @@ +#include "pdf_comic.h" + +#import +#import +#import + +#include "QsLog.h" +#include "QsLogDest.h" + + +MacOSXPDFComic::MacOSXPDFComic() +{ + +} + +MacOSXPDFComic::~MacOSXPDFComic() +{ + CGPDFDocumentRelease((CGPDFDocumentRef)document); +} + +bool MacOSXPDFComic::openComic(const QString &path) +{ + + CFURLRef pdfFileUrl; + CFStringRef str; + str=CFStringCreateWithCString( kCFAllocatorDefault,path.toUtf8().data(),kCFStringEncodingUTF8); + pdfFileUrl=CFURLCreateWithFileSystemPath( kCFAllocatorDefault,str,kCFURLPOSIXPathStyle,true ); + + CGPDFDocumentRef pdf = CGPDFDocumentCreateWithURL((CFURLRef)pdfFileUrl); + + document = pdf; + + CFRelease(str); + CFRelease(pdfFileUrl); + + return true; +} + +void MacOSXPDFComic::closeComic() +{ + //CGPDFDocumentRelease((CGPDFDocumentRef)document); +} + +unsigned int MacOSXPDFComic::numPages() +{ + return (int)CGPDFDocumentGetNumberOfPages((CGPDFDocumentRef)document); +} + +QImage MacOSXPDFComic::getPage(const int pageNum) +{ + CGPDFPageRef page = CGPDFDocumentGetPage((CGPDFDocumentRef)document, pageNum+1); + // Changed this line for the line above which is a generic line + //CGPDFPageRef page = [self getPage:page_number]; + + + + CGRect pageRect = CGPDFPageGetBoxRect(page, kCGPDFMediaBox); + int width = 1200; + + //NSLog(@"-----%f",pageRect.size.width); + CGFloat pdfScale = float(width)/pageRect.size.width; + + pageRect.size = CGSizeMake(pageRect.size.width*pdfScale, pageRect.size.height*pdfScale); + pageRect.origin = CGPointZero; + + CGColorSpaceRef genericColorSpace = CGColorSpaceCreateDeviceRGB(); + CGContextRef bitmapContext = CGBitmapContextCreate(NULL, + pageRect.size.width, + pageRect.size.height, + 8, 0, + genericColorSpace, + kCGImageAlphaPremultipliedFirst | kCGBitmapByteOrder32Little + ); + + CGContextSetInterpolationQuality(bitmapContext, kCGInterpolationHigh); + CGContextSetRenderingIntent(bitmapContext, kCGRenderingIntentDefault); + CGContextSetRGBFillColor( bitmapContext, 1.0, 1.0, 1.0, 1.0 ); + CGContextFillRect( bitmapContext, CGContextGetClipBoundingBox( bitmapContext )); + + //CGContextTranslateCTM( bitmapContext, 0, pageRect.size.height ); + //CGContextScaleCTM( bitmapContext, 1.0, -1.0 ); + + CGContextConcatCTM(bitmapContext, CGAffineTransformMakeScale(pdfScale, pdfScale)); + + + /*CGAffineTransform pdfXfm = CGPDFPageGetDrawingTransform( page, kCGPDFMediaBox, CGRectMake(pageRect.origin.x, pageRect.origin.y, pageRect.size.width, pageRect.size.height) , 0, true ); + */ + //CGContextConcatCTM( bitmapContext, pdfXfm ); + + CGContextDrawPDFPage(bitmapContext, page); + + CGImageRef image = CGBitmapContextCreateImage(bitmapContext); + + QImage qtImage; + + CFDataRef dataRef = CGDataProviderCopyData(CGImageGetDataProvider(image)); + + lastPageData = (void *)dataRef; + + const uchar *bytes = (const uchar *)CFDataGetBytePtr(dataRef); + + qtImage = QImage(bytes, pageRect.size.width, pageRect.size.height, QImage::Format_ARGB32); + + CGImageRelease(image); + //CFRelease(dataRef); + CGContextRelease(bitmapContext); + //CGPDFPageRelease(page); + CGColorSpaceRelease(genericColorSpace); + + return qtImage; +} + +void MacOSXPDFComic::releaseLastPageData() +{ + CFRelease((CFDataRef)lastPageData); +} + diff --git a/common/pictureflow.cpp b/common/pictureflow.cpp new file mode 100644 index 00000000..5aa4ba3f --- /dev/null +++ b/common/pictureflow.cpp @@ -0,0 +1,1380 @@ +/* + PictureFlow - animated image show widget + http://pictureflow.googlecode.com + + Copyright (C) 2008 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2007 Ariya Hidayat (ariya@kde.org) + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +#include "pictureflow.h" + +// detect Qt version +#if QT_VERSION >= 0x040000 +#define PICTUREFLOW_QT4 +#elif QT_VERSION >= 0x030000 +#define PICTUREFLOW_QT3 +#elif QT_VERSION >= 235 +#define PICTUREFLOW_QT2 +#else +#error PictureFlow widgets need Qt 2, Qt 3 or Qt 4 +#endif + +#ifdef PICTUREFLOW_QT4 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#endif + +#ifdef PICTUREFLOW_QT3 +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define qMax(x,y) ((x) > (y)) ? (x) : (y) +#define qMin(x,y) ((x) < (y)) ? (x) : (y) + +#define QVector QValueVector + +#define toImage convertToImage +#define contains find +#define modifiers state +#define ControlModifier ControlButton +#endif + +#ifdef PICTUREFLOW_QT2 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define qMax(x,y) ((x) > (y)) ? (x) : (y) +#define qMin(x,y) ((x) < (y)) ? (x) : (y) + +#define QVector QArray + +#define toImage convertToImage +#define contains find +#define modifiers state +#define ControlModifier ControlButton +#define flush flushX +#endif + +// for fixed-point arithmetic, we need minimum 32-bit long +// long long (64-bit) might be useful for multiplication and division +typedef long PFreal; +#define PFREAL_SHIFT 10 +#define PFREAL_ONE (1 << PFREAL_SHIFT) + +#define IANGLE_MAX 1024 +#define IANGLE_MASK 1023 + +inline PFreal fmul(PFreal a, PFreal b) +{ + return ((long long)(a))*((long long)(b)) >> PFREAL_SHIFT; +} + +inline PFreal fdiv(PFreal num, PFreal den) +{ + long long p = (long long)(num) << (PFREAL_SHIFT*2); + long long q = p / (long long)den; + long long r = q >> PFREAL_SHIFT; + + return r; +} + +inline PFreal fsin(int iangle) +{ + // warning: regenerate the table if IANGLE_MAX and PFREAL_SHIFT are changed! + static const PFreal tab[] = { + 3, 103, 202, 300, 394, 485, 571, 652, + 726, 793, 853, 904, 947, 980, 1004, 1019, + 1023, 1018, 1003, 978, 944, 901, 849, 789, + 721, 647, 566, 479, 388, 294, 196, 97, + -4, -104, -203, -301, -395, -486, -572, -653, + -727, -794, -854, -905, -948, -981, -1005, -1020, + -1024, -1019, -1004, -979, -945, -902, -850, -790, + -722, -648, -567, -480, -389, -295, -197, -98, + 3 + }; + + while(iangle < 0) + iangle += IANGLE_MAX; + iangle &= IANGLE_MASK; + + int i = (iangle >> 4); + PFreal p = tab[i]; + PFreal q = tab[(i+1)]; + PFreal g = (q - p); + return p + g * (iangle-i*16)/16; +} + +inline PFreal fcos(int iangle) +{ + return fsin(iangle + (IANGLE_MAX >> 2)); +} + +/* ---------------------------------------------------------- + +PictureFlowState stores the state of all slides, i.e. all the necessary +information to be able to render them. + +PictureFlowAnimator is responsible to move the slides during the +transition between slides, to achieve the effect similar to Cover Flow, +by changing the state. + +PictureFlowSoftwareRenderer (or PictureFlowOpenGLRenderer) is +the actual 3-d renderer. It should render all slides given the state +(an instance of PictureFlowState). + +Instances of all the above three classes are stored in +PictureFlowPrivate. + +------------------------------------------------------- */ + +struct SlideInfo +{ + int slideIndex; + int angle; + PFreal cx; + PFreal cy; + int blend; +}; + +class PictureFlowState +{ +public: + PictureFlowState(int angle=50, float spacingRatio=0); + ~PictureFlowState(); + + void reposition(); + void reset(); + + QRgb backgroundColor; + int slideWidth; + int slideHeight; + PictureFlow::ReflectionEffect reflectionEffect; + QVector slideImages; + + QVector marks; + bool showMarks; + QImage mark; + + int angle; + int rawAngle; + int spacing; + float spacingRatio; + PFreal offsetX; + PFreal offsetY; + + SlideInfo centerSlide; + QVector leftSlides; + QVector rightSlides; + int centerIndex; +}; + +class PictureFlowAnimator +{ +public: + PictureFlowAnimator(); + PictureFlowState* state; + + void start(int slide); + void stop(int slide); + void update(); + + int target; + int step; + int frame; + QTimer animateTimer; + bool animating; +}; + +class PictureFlowAbstractRenderer +{ +public: + PictureFlowAbstractRenderer(): state(0), dirty(false), widget(0) {} + virtual ~PictureFlowAbstractRenderer() {} + + PictureFlowState* state; + bool dirty; + QWidget* widget; + + virtual void init() = 0; + virtual void paint() = 0; +}; + +class PictureFlowSoftwareRenderer: public PictureFlowAbstractRenderer +{ +public: + PictureFlowSoftwareRenderer(); + ~PictureFlowSoftwareRenderer(); + + virtual void init(); + virtual void paint(); + void render(); + + +private: + QSize size; + QRgb bgcolor; + int effect; + QImage buffer; + QVector rays; + QImage* blankSurface; +#ifdef PICTUREFLOW_QT4 + QCache surfaceCache; + QHash imageHash; +#endif +#ifdef PICTUREFLOW_QT3 + QCache surfaceCache; + QMap imageHash; +#endif +#ifdef PICTUREFLOW_QT2 + QCache surfaceCache; + QIntDict imageHash; +#endif + + + void renderSlides(); + QRect renderSlide(const SlideInfo &slide, int col1 = -1, int col2 = -1); + QImage* surface(int slideIndex); +}; + +// ------------- PictureFlowState --------------------------------------- + +PictureFlowState::PictureFlowState(int a, float sr): +backgroundColor(0), slideWidth(150), slideHeight(200), +reflectionEffect(PictureFlow::BlurredReflection), centerIndex(0) , rawAngle(a), spacingRatio(sr) +{ +} + +PictureFlowState::~PictureFlowState() +{ + for(int i = 0; i < (int)slideImages.count(); i++) + delete slideImages[i]; +} + +// readjust the settings, call this when slide dimension is changed +void PictureFlowState::reposition() +{ + // angle = 70 * IANGLE_MAX / 360; // approx. 70 degrees tilted + angle = rawAngle * IANGLE_MAX / 360; + offsetX = slideWidth/2 * (PFREAL_ONE-fcos(angle)); + offsetY = slideWidth/2 * fsin(angle); + offsetX += slideWidth * PFREAL_ONE; + offsetY += slideWidth * PFREAL_ONE / 3; + if(rawAngle < 45) + offsetX += offsetX/4; + if(angle>0) + spacing = slideWidth * 0.35; + else + spacing = slideWidth*spacingRatio + slideWidth*(spacingRatio?0.10:0.2); +} + +// adjust slides so that they are in "steady state" position +void PictureFlowState::reset() +{ + centerSlide.angle = 0; + centerSlide.cx = 0; + centerSlide.cy = 0; + centerSlide.slideIndex = centerIndex; + centerSlide.blend = 256; + + if(angle == 0 && spacingRatio) + leftSlides.resize(4); + else + leftSlides.resize(6); + for(int i = 0; i < (int)leftSlides.count(); i++) + { + SlideInfo& si = leftSlides[i]; + si.angle = angle; + si.cx = -(offsetX + spacing*(i)*PFREAL_ONE); + si.cy = offsetY; + si.slideIndex = centerIndex-1-i; + si.blend = 200; + if(i == (int)leftSlides.count()-2) + si.blend = 128; + if(i == (int)leftSlides.count()-1) + si.blend = 0; + } + if(angle==0 && spacingRatio) + rightSlides.resize(4); + else + rightSlides.resize(6); + for(int i = 0; i < (int)rightSlides.count(); i++) + { + SlideInfo& si = rightSlides[i]; + si.angle = -angle; + si.cx = offsetX + spacing*(i)*PFREAL_ONE; + si.cy = offsetY; + si.slideIndex = centerIndex+1+i; + si.blend = 200; + if(i == (int)rightSlides.count()-2) + si.blend = 128; + if(i == (int)rightSlides.count()-1) + si.blend = 0; + } +} + +// ------------- PictureFlowAnimator --------------------------------------- + +PictureFlowAnimator::PictureFlowAnimator(): +state(0), target(0), step(0), frame(0), animating(false) +{ +} + +void PictureFlowAnimator::start(int slide) +{ + target = slide; + if(!animateTimer.isActive() && state) + { + step = (target < state->centerSlide.slideIndex) ? -1 : 1; + animateTimer.setSingleShot(true); + animateTimer.start(30); //TODO comprobar rendimiento, originalmente era 30 + animating = true; + } +} + +void PictureFlowAnimator::stop(int slide) +{ + step = 0; + target = slide; + frame = slide << 16; + animateTimer.stop(); + animating = false; +} + +void PictureFlowAnimator::update() +{ + /*if(!animateTimer.isActive()) + return;*/ + if(step == 0) + return; + if(!state) + return; + + int speed = 16384/4; //TODO comprobar rendimiento, originalmente era /4 + +#if 1 + // deaccelerate when approaching the target + const int max = 2 * 65536; //TODO cambiado de 2 * a 4 * comprobar rendimiento + + int fi = frame; + fi -= (target << 16); + if(fi < 0) + fi = -fi; + fi = qMin(fi, max); + + int ia = IANGLE_MAX * (fi-max/2) / (max*2); + speed = 512 + 16384 * (PFREAL_ONE+fsin(ia))/PFREAL_ONE; +#endif + + frame += speed*step; + + int index = frame >> 16; + int pos = frame & 0xffff; + int neg = 65536 - pos; + int tick = (step < 0) ? neg : pos; + PFreal ftick = (tick * PFREAL_ONE) >> 16; + + if(step < 0) + index++; + + if(state->centerIndex != index) + { + state->centerIndex = index; + frame = index << 16; + state->centerSlide.slideIndex = state->centerIndex; + for(int i = 0; i < (int)state->leftSlides.count(); i++) + state->leftSlides[i].slideIndex = state->centerIndex-1-i; + for(int i = 0; i < (int)state->rightSlides.count(); i++) + state->rightSlides[i].slideIndex = state->centerIndex+1+i; + } + + state->centerSlide.angle = (step * tick * state->angle) >> 16; + state->centerSlide.cx = -step * fmul(state->offsetX, ftick); + state->centerSlide.cy = fmul(state->offsetY, ftick); + + if(state->centerIndex == target) + { + stop(target); + state->reset(); + return; + } + + for(int i = 0; i < (int)state->leftSlides.count(); i++) + { + SlideInfo& si = state->leftSlides[i]; + si.angle = state->angle; + si.cx = -(state->offsetX + state->spacing*(i)*PFREAL_ONE + step*state->spacing*ftick); + si.cy = state->offsetY; + } + + for(int i = 0; i < (int)state->rightSlides.count(); i++) + { + SlideInfo& si = state->rightSlides[i]; + si.angle = -state->angle; + si.cx = state->offsetX + state->spacing*(i)*PFREAL_ONE - step*state->spacing*ftick; + si.cy = state->offsetY; + } + + if(step > 0) + { + PFreal ftick = (neg * PFREAL_ONE) >> 16; + state->rightSlides[0].angle = -(neg * state->angle) >> 16; + state->rightSlides[0].cx = fmul(state->offsetX, ftick); + state->rightSlides[0].cy = fmul(state->offsetY, ftick); + } + else + { + PFreal ftick = (pos * PFREAL_ONE) >> 16; + state->leftSlides[0].angle = (pos * state->angle) >> 16; + state->leftSlides[0].cx = -fmul(state->offsetX, ftick); + state->leftSlides[0].cy = fmul(state->offsetY, ftick); + } + + // must change direction ? + if(target < index) if(step > 0) + step = -1; + if(target > index) if(step < 0) + step = 1; + + // the first and last slide must fade in/fade out + int nleft = state->leftSlides.count(); + int nright = state->rightSlides.count(); + int fade = pos / 256; + + for(int index = 0; index < nleft; index++) + { + int blend = 200; + if(index == nleft-1) + blend = (step > 0) ? 0 : 128-fade/2; + if(index == nleft-2) + blend = (step > 0) ? 128-fade/2 : 200-(0.5625*fade/2); + if(index == nleft-3) + blend = (step > 0) ? 200-(0.5625*fade/2) : 200; + if(index == 0) + blend = (step > 0) ? 200 : 200 + 56-(0.4375*fade/2) ; + state->leftSlides[index].blend = blend; + } + for(int index = 0; index < nright; index++) + { + int blend = (index < nright-2) ? 200 : 128; + if(index == nright-1) + blend = (step > 0) ? fade/2 : 0; + if(index == nright-2) + blend = (step > 0) ? 128+(0.5625*fade/2) : (0.5625*fade/2); + if(index == nright-3) + blend = (step > 0) ? 200 : 128+(0.5625*fade/2); + if(index == 0) + blend = (step > 0) ? 200 + (0.4375*fade/2) : 200; + state->rightSlides[index].blend = blend; + } + + state->centerSlide.blend = (step > 0) ? 256 - (0.4375*fade/2) : 200 + (0.4375*fade/2); + +} + +// ------------- PictureFlowSoftwareRenderer --------------------------------------- + +PictureFlowSoftwareRenderer::PictureFlowSoftwareRenderer(): +PictureFlowAbstractRenderer(), size(0,0), bgcolor(0), effect(-1), blankSurface(0) +{ +#ifdef PICTUREFLOW_QT3 + surfaceCache.setAutoDelete(true); +#endif +} + +PictureFlowSoftwareRenderer::~PictureFlowSoftwareRenderer() +{ + surfaceCache.clear(); + buffer = QImage(); + delete blankSurface; +} + +void PictureFlowSoftwareRenderer::paint() +{ + if(!widget) + return; + + if(widget->size() != size) + init(); + + if(state->backgroundColor != bgcolor) + { + bgcolor = state->backgroundColor; + surfaceCache.clear(); + } + + if((int)(state->reflectionEffect) != effect) + { + effect = (int)state->reflectionEffect; + surfaceCache.clear(); + } + + if(dirty) + render(); + + QPainter painter(widget); + painter.drawImage(QPoint(0,0), buffer); +} + +void PictureFlowSoftwareRenderer::init() +{ + if(!widget) + return; + + surfaceCache.clear(); + blankSurface = 0; + + size = widget->size(); + int ww = size.width(); + int wh = size.height(); + int w = (ww+1)/2; + int h = (wh+1)/2; + if(h<10)//TODO a partir de qué h es seguro?? + return; + +#ifdef PICTUREFLOW_QT4 + buffer = QImage(ww, wh, QImage::Format_RGB32); +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + buffer.create(ww, wh, 32); +#endif + buffer.fill(bgcolor); + + rays.resize(w*2); + for(int i = 0; i < w; i++) + { + PFreal gg = ((PFREAL_ONE >> 1) + i * PFREAL_ONE) / (2*h); + rays[w-i-1] = -gg; + rays[w+i] = gg; + } + + dirty = true; +} + +// TODO: optimize this with lookup tables +static QRgb blendColor(QRgb c1, QRgb c2, int blend) +{ + int r = qRed(c1) * blend/256 + qRed(c2)*(256-blend)/256; + int g = qGreen(c1) * blend/256 + qGreen(c2)*(256-blend)/256; + int b = qBlue(c1) * blend/256 + qBlue(c2)*(256-blend)/256; + return qRgb(r, g, b); +} + + +static QImage* prepareSurface(const QImage* slideImage, int w, int h, QRgb bgcolor, +PictureFlow::ReflectionEffect reflectionEffect) +{ + + int iw,ih; + iw = slideImage->width(); + ih = slideImage->height(); + int psw,psh; + if(iw>ih) + { + psw = w; + psh = w * (1.0*ih/iw); + } + else + { + int h1=h; + psw = h1 * (1.0*iw/ih); + psh = h1; + + while(psw>w) + { + h1-=2; + psw = h1 * (1.0*iw/ih); + psh = h1; + } + } + w = psw; + +#ifdef PICTUREFLOW_QT4 + Qt::TransformationMode mode = Qt::SmoothTransformation; + QImage img = slideImage->scaled(psw, psh, Qt::IgnoreAspectRatio, mode); +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + QImage img = slideImage->smoothScale(w, h); +#endif + + // slightly larger, to accomodate for the reflection + int hs = h * 2; + int hofs = h / 3; + + // offscreen buffer: black is sweet +#ifdef PICTUREFLOW_QT4 + QImage* result = new QImage(hs, w, QImage::Format_RGB32); +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + QImage* result = new QImage; + result->create(hs, w, 32); +#endif + result->fill(bgcolor); + + // transpose the image, this is to speed-up the rendering + // because we process one column at a time + // (and much better and faster to work row-wise, i.e in one scanline) + int lhof = (h-psh); + //int lwof = (w-psw)/2; + for(int x = 0; x < psw; x++) + for(int y = 0; y < psh; y++) + + result->setPixel(hofs + y + lhof , x, img.pixel(x, y)); + + if(reflectionEffect != PictureFlow::NoReflection) + { + // create the reflection + int ht = hs - (h+hofs); + int hte = ht; + for(int x = 0; x < psw; x++) + for(int y = 0; y < ht; y++) + { + QRgb color; + if(ysetPixel(h+hofs + y, x,blendColor(color,bgcolor,80*(hte-y)/hte)); + } + + + } + + return result; +} + +QImage* PictureFlowSoftwareRenderer::surface(int slideIndex) +{ + if(!state) + return 0; + if(slideIndex < 0) + return 0; + if(slideIndex >= (int)state->slideImages.count()) + return 0; + +#ifdef PICTUREFLOW_QT4 + int key = slideIndex; +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + QString key = QString::number(slideIndex); +#endif + + QImage* img = state->slideImages.at(slideIndex); + + bool empty = img ? img->isNull() : true; + if(empty) + { + surfaceCache.remove(key); + imageHash.remove(slideIndex); + if(!blankSurface) + { + int sw = state->slideWidth; + int sh = state->slideHeight; + +#ifdef PICTUREFLOW_QT4 + QImage img = QImage(sw, sh, QImage::Format_RGB32); + + QPainter painter(&img); + QPoint p1(sw*4/10, 0); + QPoint p2(sw*6/10, sh); + QLinearGradient linearGrad(p1, p2); + linearGrad.setColorAt(0, Qt::black); + linearGrad.setColorAt(1, Qt::white); + painter.setBrush(linearGrad); + painter.fillRect(0, 0, sw, sh, QBrush(linearGrad)); + + + painter.end(); +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + QPixmap pixmap(sw, sh, 32); + QPainter painter(&pixmap); + painter.fillRect(pixmap.rect(), QColor(192,192,192)); + painter.fillRect(5, 5, sw-10, sh-10, QColor(64,64,64)); + painter.end(); + QImage img = pixmap.convertToImage(); +#endif + + blankSurface = prepareSurface(&img, sw, sh, bgcolor, state->reflectionEffect); + } + return blankSurface; + } + +#ifdef PICTUREFLOW_QT4 + bool exist = imageHash.contains(slideIndex); + if(exist) + if(img == imageHash.find(slideIndex).value()) +#endif +#ifdef PICTUREFLOW_QT3 + bool exist = imageHash.find(slideIndex) != imageHash.end(); + if(exist) + if(img == imageHash.find(slideIndex).data()) +#endif +#ifdef PICTUREFLOW_QT2 + if(img == imageHash[slideIndex]) +#endif + if(surfaceCache.contains(key)) + return surfaceCache[key]; + + + QImage* sr = prepareSurface(img, state->slideWidth, state->slideHeight, bgcolor, state->reflectionEffect); + //check if this slide must be marked + //if(marks[slideIndex]) + if(state->showMarks) + { + if(state->marks[slideIndex]) + { + QPainter painter(sr); + painter.setPen(QColor(255,0,0).rgb()); + int sh = sr->height(); + int jInit = sh*4/5; + int iInit = state->slideHeight+state->slideHeight/3; + /*for(int j = jInit; j < sh; j ++) + { + for(int i = iInit-(j-jInit); i < iInit; i ++) + { + + painter.drawPoint(i,j); + } + }*/ + painter.drawImage(QRect(iInit-(sh-jInit),jInit,sh-jInit,sh-jInit),state->mark); + } + } + surfaceCache.insert(key, sr); + imageHash.insert(slideIndex, img); + + return sr; +} + +// Renders a slide to offscreen buffer. Returns a rect of the rendered area. +// col1 and col2 limit the column for rendering. +QRect PictureFlowSoftwareRenderer::renderSlide(const SlideInfo &slide, int col1, int col2) +{ + int blend = slide.blend; + if(!blend) + return QRect(); + + QImage* src = surface(slide.slideIndex); + if(!src) + return QRect(); + + QRect rect(0, 0, 0, 0); + + int sw = src->height(); + int sh = src->width(); + int h = buffer.height(); + int w = buffer.width(); + + if(col1 > col2) + { + int c = col2; + col2 = col1; + col1 = c; + } + + col1 = (col1 >= 0) ? col1 : 0; + col2 = (col2 >= 0) ? col2 : w-1; + col1 = qMin(col1, w-1); + col2 = qMin(col2, w-1); + + int zoom = 100; + int distance = h * 100 / zoom; + PFreal sdx = fcos(slide.angle); + PFreal sdy = fsin(slide.angle); + PFreal xs = slide.cx - state->slideWidth * sdx/2; + PFreal ys = slide.cy - state->slideWidth * sdy/2; + PFreal dist = distance * PFREAL_ONE; + + int xi = qMax((PFreal)0, ((w*PFREAL_ONE/2) + fdiv(xs*h, dist+ys)) >> PFREAL_SHIFT); + if(xi >= w) + return rect; + + bool flag = false; + rect.setLeft(xi); + for(int x = qMax(xi, col1); x <= col2; x++) + { + PFreal hity = 0; + PFreal fk = rays[x]; + if(sdy) + { + fk = fk - fdiv(sdx,sdy); + hity = -fdiv((rays[x]*distance - slide.cx + slide.cy*sdx/sdy), fk); + } + + dist = distance*PFREAL_ONE + hity; + if(dist < 0) + continue; + + PFreal hitx = fmul(dist, rays[x]); + PFreal hitdist = fdiv(hitx - slide.cx, sdx); + + int column = sw/2 + (hitdist >> PFREAL_SHIFT); + if(column >= sw) + break; + if(column < 0) + continue; + + rect.setRight(x); + if(!flag) + rect.setLeft(x); + flag = true; + + int y1 = h/2; + int y2 = y1+ 1; + QRgb* pixel1 = (QRgb*)(buffer.scanLine(y1)) + x; + QRgb* pixel2 = (QRgb*)(buffer.scanLine(y2)) + x; + QRgb pixelstep = pixel2 - pixel1; + + int center = (sh/2); + int dy = dist / h; + int p1 = center*PFREAL_ONE - dy/2; + int p2 = center*PFREAL_ONE + dy/2; + + const QRgb *ptr = (const QRgb*)(src->scanLine(column)); + if(blend == 256) + while((y1 >= 0) && (y2 < h) && (p1 >= 0)) + { + *pixel1 = ptr[p1 >> PFREAL_SHIFT]; + *pixel2 = ptr[p2 >> PFREAL_SHIFT]; + p1 -= dy; + p2 += dy; + y1--; + y2++; + pixel1 -= pixelstep; + pixel2 += pixelstep; + } + else + while((y1 >= 0) && (y2 < h) && (p1 >= 0)) + { + QRgb c1 = ptr[p1 >> PFREAL_SHIFT]; + QRgb c2 = ptr[p2 >> PFREAL_SHIFT]; + *pixel1 = blendColor(c1, bgcolor, blend); + *pixel2 = blendColor(c2, bgcolor, blend); + p1 -= dy; + p2 += dy; + y1--; + y2++; + pixel1 -= pixelstep; + pixel2 += pixelstep; + } + } + + rect.setTop(0); + rect.setBottom(h-1); + return rect; +} + +void PictureFlowSoftwareRenderer::renderSlides() +{ + int nleft = state->leftSlides.count(); + int nright = state->rightSlides.count(); + + QRect r = renderSlide(state->centerSlide); + int c1 = r.left(); + int c2 = r.right(); + + for(int index = 0; index < nleft; index++) + { + QRect rs = renderSlide(state->leftSlides[index], 0, c1-1); + if(!rs.isEmpty()) + c1 = rs.left(); + } + for(int index = 0; index < nright; index++) + { + QRect rs = renderSlide(state->rightSlides[index], c2+1, buffer.width()); + if(!rs.isEmpty()) + c2 = rs.right(); + } +} + +// Render the slides. Updates only the offscreen buffer. +void PictureFlowSoftwareRenderer::render() +{ + buffer.fill(state->backgroundColor); + renderSlides(); + if(state->slideImages.size()>0) + { + int size = buffer.width() * 0.015; + int start = buffer.width() * 0.010; + + QPainter painter(&buffer); + painter.setPen(QColor(255,255,255).rgb()-state->backgroundColor); + painter.setFont(QFont("Arial", start+size*0.5)); + painter.drawText(start , start+size, QString().setNum(state->centerIndex+1)+"/"+QString().setNum(state->slideImages.size())); + } + dirty = false; +} + +// ----------------------------------------- + +class PictureFlowPrivate +{ +public: + PictureFlowState* state; + PictureFlowAnimator* animator; + PictureFlowAbstractRenderer* renderer; + QTimer triggerTimer; +}; + + +PictureFlow::PictureFlow(QWidget* parent,FlowType flowType): QWidget(parent) +{ + d = new PictureFlowPrivate; + + switch(flowType){ + case CoverFlowLike: + d->state = new PictureFlowState(50,0); + break; + case Strip: + d->state = new PictureFlowState(0,1); + break; + case StripOverlapped: + d->state = new PictureFlowState(0,0); + break; + } + + framesSkip = 0; + + d->state->reset(); + d->state->reposition(); + + d->renderer = new PictureFlowSoftwareRenderer; + d->renderer->state = d->state; + d->renderer->widget = this; + d->renderer->init(); + + d->animator = new PictureFlowAnimator; + d->animator->state = d->state; + QObject::connect(&d->animator->animateTimer, SIGNAL(timeout()), this, SLOT(updateAnimation())); + + QObject::connect(&d->triggerTimer, SIGNAL(timeout()), this, SLOT(render())); + +#ifdef PICTUREFLOW_QT4 + setAttribute(Qt::WA_StaticContents, true); + setAttribute(Qt::WA_OpaquePaintEvent, true); + setAttribute(Qt::WA_NoSystemBackground, true); +#endif +#ifdef PICTUREFLOW_QT3 + setWFlags(getWFlags() | Qt::WStaticContents); + setWFlags(getWFlags() | Qt::WNoAutoErase); +#endif +#ifdef PICTUREFLOW_QT2 + setWFlags(getWFlags() | Qt::WPaintClever); + setWFlags(getWFlags() | Qt::WRepaintNoErase); + setWFlags(getWFlags() | Qt::WResizeNoErase); +#endif +} + +PictureFlow::~PictureFlow() +{ + delete d->renderer; + delete d->animator; + delete d->state; + delete d; +} + +int PictureFlow::slideCount() const +{ + return d->state->slideImages.count(); +} + +QColor PictureFlow::backgroundColor() const +{ + return QColor(d->state->backgroundColor); +} + +void PictureFlow::setBackgroundColor(const QColor& c) +{ + d->state->backgroundColor = c.rgb(); + triggerRender(); +} + +QSize PictureFlow::slideSize() const +{ + return QSize(d->state->slideWidth, d->state->slideHeight); +} + +void PictureFlow::setSlideSize(QSize size) +{ + d->state->slideWidth = size.width(); + d->state->slideHeight = size.height(); + d->state->reposition(); + triggerRender(); +} + +PictureFlow::ReflectionEffect PictureFlow::reflectionEffect() const +{ + return d->state->reflectionEffect; +} + +void PictureFlow::setReflectionEffect(ReflectionEffect effect) +{ + d->state->reflectionEffect = effect; + triggerRender(); +} + +QImage PictureFlow::slide(int index) const +{ + QImage* i = 0; + if((index >= 0) && (index < slideCount())) + i = d->state->slideImages[index]; + return i ? QImage(*i) : QImage(); +} + +void PictureFlow::addSlide(const QImage& image) +{ + int c = d->state->slideImages.count(); + d->state->slideImages.resize(c+1); + d->state->slideImages[c] = new QImage(image); + d->state->marks.resize(c+1); + d->state->marks[c] = YACReader::Unread; + triggerRender(); +} + +void PictureFlow::addSlide(const QPixmap& pixmap) +{ + addSlide(pixmap.toImage()); +} + +void PictureFlow::removeSlide(int index) +{ + int c = d->state->slideImages.count(); + if (index >= 0 && index < c) + { + d->state->slideImages.remove(index); + d->state->marks.remove(index); + setCenterIndex(index); + } +} + +void PictureFlow::setSlide(int index, const QImage& image) +{ + if((index >= 0) && (index < slideCount())) + { + QImage* i = image.isNull() ? 0 : new QImage(image); + delete d->state->slideImages[index]; + d->state->slideImages[index] = i; + triggerRender(); + } +} + +void PictureFlow::setSlide(int index, const QPixmap& pixmap) +{ + setSlide(index, pixmap.toImage()); +} + +int PictureFlow::centerIndex() const +{ + return d->state->centerIndex; +} + +void PictureFlow::setCenterIndex(int index) +{ + index = qMin(index, slideCount()-1); + index = qMax(index, 0); + d->state->centerIndex = index; + d->state->reset(); + d->animator->stop(index); + triggerRender(); +} + +void PictureFlow::clear() +{ + int c = d->state->slideImages.count(); + for(int i = 0; i < c; i++) + delete d->state->slideImages[i]; + d->state->slideImages.resize(0); + + d->state->marks.resize(0); + + d->state->reset(); + triggerRender(); +} + +void PictureFlow::render() +{ + d->renderer->dirty = true; + update(); +} + +void PictureFlow::triggerRender() +{ +#ifdef PICTUREFLOW_QT4 + d->triggerTimer.setSingleShot(true); + d->triggerTimer.start(0); +#endif +#if defined(PICTUREFLOW_QT3) || defined(PICTUREFLOW_QT2) + d->triggerTimer.start(0, true); +#endif +} + +void PictureFlow::showPrevious() +{ + int step = d->animator->step; + int center = d->state->centerIndex; + + if(step > 0) + { + d->animator->start(center); + emit centerIndexChanged(center); + } + + if(step == 0) + if(center > 0) + { + d->animator->start(center - 1); + emit centerIndexChanged(center - 1); + } + + if(step < 0) + { + d->animator->target = qMax(0, center - 2); + emit centerIndexChanged(qMax(0, center - 2)); + } + +} + +void PictureFlow::showNext() +{ + int step = d->animator->step; + int center = d->state->centerIndex; + + + if(step < 0) + { + d->animator->start(center); + emit centerIndexChanged(center); + } + + if(step == 0) + if(center < slideCount()-1) + { + d->animator->start(center + 1); + emit centerIndexChanged(center + 1); + } + + if(step > 0) + { + d->animator->target = qMin(center + 2, slideCount()-1); + emit centerIndexChanged(qMin(center + 2, slideCount()-1)); + } + + +} + +void PictureFlow::showSlide(unsigned int index) +{ + index = qMax(index, 0); + index = qMin(slideCount()-1, index); + if(index == d->state->centerSlide.slideIndex) + return; + + int distance = centerIndex()-index; + + if(abs(distance)>10) + { + if(distance<0) + setCenterIndex(centerIndex()+(-distance)-10); + else + setCenterIndex(centerIndex()-distance+10); + } + + d->animator->start(index); +} + +void PictureFlow::keyPressEvent(QKeyEvent* event) +{ + if(event->key() == Qt::Key_Left) + { + /*if(event->modifiers() == Qt::ControlModifier) + showSlide(centerIndex()-10); + else*/ + showPrevious(); + event->accept(); + return; + } + + if(event->key() == Qt::Key_Right) + { + /*if(event->modifiers() == Qt::ControlModifier) + showSlide(centerIndex()+10); + else*/ + showNext(); + event->accept(); + return; + } + + if(event->key() == Qt::Key_Up) + { + //TODO emit selected signal + return; + } + + event->ignore(); +} + +void PictureFlow::mousePressEvent(QMouseEvent* event) +{ + if(event->x() > width()/2) + showNext(); + else + showPrevious(); +} + +void PictureFlow::paintEvent(QPaintEvent* event) +{ + Q_UNUSED(event); + d->renderer->paint(); +} + +void PictureFlow::resizeEvent(QResizeEvent* event) +{ + int heightWidget = event->size().height(); + int height,width; + height = heightWidget*0.55; + width = height*0.65; + setSlideSize(QSize(width,height)); + + render(); + d->animator->start(centerIndex()); + QWidget::resizeEvent(event); +} +#include +void PictureFlow::updateAnimation() //bucle principal +{ + QTime now; + now.start(); + bool frameSkiped = false; + + int old_center = d->state->centerIndex; + d->animator->update(); + if(framesSkip == 0) + render();//triggerRender(); + else + { + framesSkip--; + frameSkiped = true; + } + + + if(d->state->centerIndex != old_center) + emit centerIndexChangedSilent(d->state->centerIndex); + if(d->animator->animating == true) + { + int difference = 10-now.elapsed(); + if(difference >= 0 && !frameSkiped) + QTimer::singleShot(difference, this, SLOT(updateAnimation())); + else + { + QTimer::singleShot(0, this, SLOT(updateAnimation())); + if(!frameSkiped) + framesSkip = -( (difference - 10) / 10); + } + } + +} + +void PictureFlow::setFlowType(FlowType flowType) +{ + switch(flowType){ + case CoverFlowLike: + d->state->rawAngle = 50; + d->state->spacingRatio = 0, + d->state->reposition(); + break; + case Strip: + d->state->rawAngle = 0; + d->state->spacingRatio = 1; + d->state->reposition(); + break; + case StripOverlapped: + d->state->rawAngle = 0; + d->state->spacingRatio = 0; + d->state->reposition(); + break; + } + d->state->reset(); + d->renderer->init(); +} + +void PictureFlow::setMarkImage(const QImage & m) +{ + d->state->mark = m; +} + +void PictureFlow::markSlide(int index, YACReaderComicReadStatus readStatus) +{ + if(indexstate->marks.size()) + d->state->marks[index] = readStatus; +} + +void PictureFlow::updateMarks() +{ + d->renderer->init(); + d->renderer->paint(); + repaint(); +} + +void PictureFlow::unmarkSlide(int index) +{ + if(indexstate->marks.size()) + d->state->marks[index] = YACReader::Unread; +} + +void PictureFlow::setMarks(const QVector & m) +{ + d->state->marks = m; + updateMarks(); +} + +void PictureFlow::setShowMarks(bool enable) +{ + d->state->showMarks = enable; + updateMarks(); +} + +QVector PictureFlow::getMarks() +{ + return d->state->marks; +} + diff --git a/common/pictureflow.h b/common/pictureflow.h new file mode 100644 index 00000000..b12b0b25 --- /dev/null +++ b/common/pictureflow.h @@ -0,0 +1,227 @@ +/* + PictureFlow - animated image show widget + http://pictureflow.googlecode.com + + Copyright (C) 2008 Ariya Hidayat (ariya@kde.org) + Copyright (C) 2007 Ariya Hidayat (ariya@kde.org) + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +#ifndef PICTUREFLOW_H +#define PICTUREFLOW_H + +#include +#include "yacreader_global.h" //FlowType + +class PictureFlowPrivate; + +using namespace YACReader; + +/*! + Class PictureFlow implements an image show widget with animation effect + like Apple's CoverFlow (in iTunes and iPod). Images are arranged in form + of slides, one main slide is shown at the center with few slides on + the left and right sides of the center slide. When the next or previous + slide is brought to the front, the whole slides flow to the right or + the right with smooth animation effect; until the new slide is finally + placed at the center. + + */ +class PictureFlow : public QWidget +{ +Q_OBJECT + + Q_PROPERTY(QColor backgroundColor READ backgroundColor WRITE setBackgroundColor) + Q_PROPERTY(QSize slideSize READ slideSize WRITE setSlideSize) + Q_PROPERTY(int slideCount READ slideCount) + Q_PROPERTY(int centerIndex READ centerIndex WRITE setCenterIndex) + +public: + + enum ReflectionEffect + { + NoReflection, + PlainReflection, + BlurredReflection + }; + + + + /*! + Creates a new PictureFlow widget. + */ + PictureFlow(QWidget* parent = 0, FlowType flowType = CoverFlowLike); + + /*! + Destroys the widget. + */ + ~PictureFlow(); + + /*! + Returns the background color. + */ + QColor backgroundColor() const; + + /*! + Sets the background color. By default it is black. + */ + void setBackgroundColor(const QColor& c); + + /*! + Returns the dimension of each slide (in pixels). + */ + QSize slideSize() const; + + /*! + Sets the dimension of each slide (in pixels). + */ + void setSlideSize(QSize size); + + /*! + Returns the total number of slides. + */ + int slideCount() const; + + /*! + Returns QImage of specified slide. + */ + QImage slide(int index) const; + + /*! + Returns the index of slide currently shown in the middle of the viewport. + */ + int centerIndex() const; + + /*! + Returns the effect applied to the reflection. + */ + ReflectionEffect reflectionEffect() const; + + /*! + Sets the effect applied to the reflection. The default is PlainReflection. + */ + void setReflectionEffect(ReflectionEffect effect); + + +public slots: + + /*! + Adds a new slide. + */ + void addSlide(const QImage& image); + + /*! + Adds a new slide. + */ + void addSlide(const QPixmap& pixmap); + + /*! + Removes an existing slide. + */ + void removeSlide(int index); + + /*! + Sets an image for specified slide. If the slide already exists, + it will be replaced. + */ + void setSlide(int index, const QImage& image); + + /*! + Sets a pixmap for specified slide. If the slide already exists, + it will be replaced. + */ + void setSlide(int index, const QPixmap& pixmap); + + /*! + Sets slide to be shown in the middle of the viewport. No animation + effect will be produced, unlike using showSlide. + */ + void setCenterIndex(int index); + + /*! + Clears all slides. + */ + void clear(); + + /*! + Shows previous slide using animation effect. + */ + void showPrevious(); + + /*! + Shows next slide using animation effect. + */ + void showNext(); + + /*! + Go to specified slide using animation effect. + */ + void showSlide(unsigned int index); + + /*! + Rerender the widget. Normally this function will be automatically invoked + whenever necessary, e.g. during the transition animation. + */ + void render(); + + /*! + Schedules a rendering update. Unlike render(), this function does not cause + immediate rendering. + */ + void triggerRender(); + + void setFlowType(FlowType flowType); + + void setMarkImage(const QImage & mark); + + void markSlide(int index, YACReaderComicReadStatus readStatus = Read); + + void updateMarks(); + + void unmarkSlide(int index); + + void setMarks(const QVector & marks); + + void setShowMarks(bool enable); + + QVector getMarks(); + + +signals: + void centerIndexChanged(int index); + void centerIndexChangedSilent(int index); + +public: + void paintEvent(QPaintEvent *event); + void keyPressEvent(QKeyEvent* event); + void mousePressEvent(QMouseEvent* event); + void resizeEvent(QResizeEvent* event); + +private slots: + void updateAnimation(); + +private: + PictureFlowPrivate* d; + QImage mark; + int framesSkip; +}; + +#endif // PICTUREFLOW_H + diff --git a/common/qnaturalsorting.cpp b/common/qnaturalsorting.cpp new file mode 100644 index 00000000..97cbd5b0 --- /dev/null +++ b/common/qnaturalsorting.cpp @@ -0,0 +1,262 @@ +/* This file contains parts of the KDE libraries + Copyright (C) 1999 Ian Zepp (icszepp@islc.net) + Copyright (C) 2006 by Dominic Battre + Copyright (C) 2006 by Martin Pool + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include "qnaturalsorting.h" + +//from KDE +/* +int naturalCompare(const QString &_a, const QString &_b, Qt::CaseSensitivity caseSensitivity) +{ + // This method chops the input a and b into pieces of + // digits and non-digits (a1.05 becomes a | 1 | . | 05) + // and compares these pieces of a and b to each other + // (first with first, second with second, ...). + // + // This is based on the natural sort order code code by Martin Pool + // http://sourcefrog.net/projects/natsort/ + // Martin Pool agreed to license this under LGPL or GPL. + + // FIXME: Using toLower() to implement case insensitive comparison is + // sub-optimal, but is needed because we compare strings with + // localeAwareCompare(), which does not know about case sensitivity. + // A task has been filled for this in Qt Task Tracker with ID 205990. + // http://trolltech.com/developer/task-tracker/index_html?method=entry&id=205990 + QString a; + QString b; + if (caseSensitivity == Qt::CaseSensitive) { + a = _a; + b = _b; + } else { + a = _a.toLower(); + b = _b.toLower(); + } + + const QChar* currA = a.unicode(); // iterator over a + const QChar* currB = b.unicode(); // iterator over b + + if (currA == currB) { + return 0; + } + + const QChar* begSeqA = currA; // beginning of a new character sequence of a + const QChar* begSeqB = currB; + + while (!currA->isNull() && !currB->isNull()) { + if (currA->unicode() == QChar::ObjectReplacementCharacter) { + return 1; + } + + if (currB->unicode() == QChar::ObjectReplacementCharacter) { + return -1; + } + + if (currA->unicode() == QChar::ReplacementCharacter) { + return 1; + } + + if (currB->unicode() == QChar::ReplacementCharacter) { + return -1; + } + + // find sequence of characters ending at the first non-character + while (!currA->isNull() && !currA->isDigit() && !currA->isPunct() && !currA->isSpace()) { + ++currA; + } + + while (!currB->isNull() && !currB->isDigit() && !currB->isPunct() && !currB->isSpace()) { + ++currB; + } + + // compare these sequences + const QStringRef& subA(a.midRef(begSeqA - a.unicode(), currA - begSeqA)); + const QStringRef& subB(b.midRef(begSeqB - b.unicode(), currB - begSeqB)); + const int cmp = QStringRef::localeAwareCompare(subA, subB); + if (cmp != 0) { + return cmp < 0 ? -1 : +1; + } + + if (currA->isNull() || currB->isNull()) { + break; + } + + // find sequence of characters ending at the first non-character + while (currA->isPunct() || currA->isSpace() || currB->isPunct() || currB->isSpace()) { + if (*currA != *currB) { + return (*currA < *currB) ? -1 : +1; + } + ++currA; + ++currB; + } + + // now some digits follow... + if ((*currA == '0') || (*currB == '0')) { + // one digit-sequence starts with 0 -> assume we are in a fraction part + // do left aligned comparison (numbers are considered left aligned) + while (1) { + if (!currA->isDigit() && !currB->isDigit()) { + break; + } else if (!currA->isDigit()) { + return +1; + } else if (!currB->isDigit()) { + return -1; + } else if (*currA < *currB) { + return -1; + } else if (*currA > *currB) { + return + 1; + } + ++currA; + ++currB; + } + } else { + // No digit-sequence starts with 0 -> assume we are looking at some integer + // do right aligned comparison. + // + // The longest run of digits wins. That aside, the greatest + // value wins, but we can't know that it will until we've scanned + // both numbers to know that they have the same magnitude. + + bool isFirstRun = true; + int weight = 0; + while (1) { + if (!currA->isDigit() && !currB->isDigit()) { + if (weight != 0) { + return weight; + } + break; + } else if (!currA->isDigit()) { + if (isFirstRun) { + return *currA < *currB ? -1 : +1; + } else { + return -1; + } + } else if (!currB->isDigit()) { + if (isFirstRun) { + return *currA < *currB ? -1 : +1; + } else { + return +1; + } + } else if ((*currA < *currB) && (weight == 0)) { + weight = -1; + } else if ((*currA > *currB) && (weight == 0)) { + weight = + 1; + } + ++currA; + ++currB; + isFirstRun = false; + } + } + + begSeqA = currA; + begSeqB = currB; + } + + if (currA->isNull() && currB->isNull()) { + return 0; + } + + return currA->isNull() ? -1 : + 1; +} + +*/ +static inline QChar getNextChar(const QString &s, int location) +{ + return (location < s.length()) ? s.at(location) : QChar(); +} + +int naturalCompare(const QString &s1, const QString &s2, Qt::CaseSensitivity cs) +{ + for (int l1 = 0, l2 = 0; l1 <= s1.count() && l2 <= s2.count(); ++l1, ++l2) { + // skip spaces, tabs and 0's + QChar c1 = getNextChar(s1, l1); + while (c1.isSpace()) + c1 = getNextChar(s1, ++l1); + QChar c2 = getNextChar(s2, l2); + while (c2.isSpace()) + c2 = getNextChar(s2, ++l2); + + if (c1.isDigit() && c2.isDigit()) { + while (c1.digitValue() == 0) + c1 = getNextChar(s1, ++l1); + while (c2.digitValue() == 0) + c2 = getNextChar(s2, ++l2); + + int lookAheadLocation1 = l1; + int lookAheadLocation2 = l2; + int currentReturnValue = 0; + // find the last digit, setting currentReturnValue as we go if it isn't equal + for ( + QChar lookAhead1 = c1, lookAhead2 = c2; + (lookAheadLocation1 <= s1.length() && lookAheadLocation2 <= s2.length()); + lookAhead1 = getNextChar(s1, ++lookAheadLocation1), + lookAhead2 = getNextChar(s2, ++lookAheadLocation2) + ) { + bool is1ADigit = !lookAhead1.isNull() && lookAhead1.isDigit(); + bool is2ADigit = !lookAhead2.isNull() && lookAhead2.isDigit(); + if (!is1ADigit && !is2ADigit) + break; + if (!is1ADigit) + return -1; + if (!is2ADigit) + return 1; + if (currentReturnValue == 0) { + if (lookAhead1 < lookAhead2) { + currentReturnValue = -1; + } else if (lookAhead1 > lookAhead2) { + currentReturnValue = 1; + } + } + } + if (currentReturnValue != 0) + return currentReturnValue; + } + + if (cs == Qt::CaseInsensitive) { + if (!c1.isLower()) c1 = c1.toLower(); + if (!c2.isLower()) c2 = c2.toLower(); + } + int r = QString::localeAwareCompare(c1, c2); + if (r < 0) + return -1; + if (r > 0) + return 1; + } + // The two strings are the same (02 == 2) so fall back to the normal sort + return QString::compare(s1, s2, cs); +} +bool naturalSortLessThanCS( const QString &left, const QString &right ) +{ + return (naturalCompare( left, right, Qt::CaseSensitive ) < 0); +} + +bool naturalSortLessThanCI( const QString &left, const QString &right ) +{ + return (naturalCompare( left, right, Qt::CaseInsensitive ) < 0); +} + +bool naturalSortLessThanCIFileInfo(const QFileInfo & left,const QFileInfo & right) +{ + return naturalSortLessThanCI(left.fileName(),right.fileName()); +} + +bool naturalSortLessThanCILibraryItem(LibraryItem * left, LibraryItem * right) +{ + return naturalSortLessThanCI(left->name,right->name); +} diff --git a/common/qnaturalsorting.h b/common/qnaturalsorting.h new file mode 100644 index 00000000..9a84f96a --- /dev/null +++ b/common/qnaturalsorting.h @@ -0,0 +1,15 @@ + + +#ifndef __QNATURALSORTING_H +#define __QNATURALSORTING_H + +#include +#include +#include "library_item.h" + +bool naturalSortLessThanCS( const QString &left, const QString &right ); +bool naturalSortLessThanCI( const QString &left, const QString &right ); +bool naturalSortLessThanCIFileInfo(const QFileInfo & left,const QFileInfo & right); +bool naturalSortLessThanCILibraryItem(LibraryItem * left, LibraryItem * right); + +#endif diff --git a/common/yacreader_flow_gl.cpp b/common/yacreader_flow_gl.cpp new file mode 100644 index 00000000..5ce9e1dc --- /dev/null +++ b/common/yacreader_flow_gl.cpp @@ -0,0 +1,1580 @@ +#include "yacreader_flow_gl.h" + +#include +#include +//#include + +#ifdef Q_OS_MAC + #include +#else + #include +#endif + +#include +#include +#include + +/*** Animation Settings ***/ + +/*** Position Configuration ***/ + +int YACReaderFlowGL::updateInterval = 16; + +struct Preset defaultYACReaderFlowConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 3.f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 3.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.01f, //View_rotate_add sets the speed of the rotation + 0.02f, //View_rotate_sub sets the speed of reversing the rotation + 20.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + 0.f, //CF_Y the Y Position of the Coverflow + -8.f, //CF_Z the Z Position of the Coverflow + + 15.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + -50.f, //Rotation sets the rotation of each cover + 0.18f, //X_Distance sets the distance between the covers + 1.f, //Center_Distance sets the distance between the centered and the non centered covers + 0.1f, //Z_Distance sets the pushback amount + 0.0f, //Y_Distance sets the elevation amount + + 30.f //zoom level + +}; + +struct Preset presetYACReaderFlowClassicConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 2.f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 3.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.08f, //View_rotate_add sets the speed of the rotation + 0.08f, //View_rotate_sub sets the speed of reversing the rotation + 30.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + -0.2f, //CF_Y the Y Position of the Coverflow + -7.f, //CF_Z the Z Position of the Coverflow + + 0.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + -40.f, //Rotation sets the rotation of each cover + 0.18f, //X_Distance sets the distance between the covers + 1.f, //Center_Distance sets the distance between the centered and the non centered covers + 0.1f, //Z_Distance sets the pushback amount + 0.0f, //Y_Distance sets the elevation amount + + 22.f //zoom level + +}; + +struct Preset presetYACReaderFlowStripeConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 6.f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 4.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.08f, //View_rotate_add sets the speed of the rotation + 0.08f, //View_rotate_sub sets the speed of reversing the rotation + 30.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + -0.2f, //CF_Y the Y Position of the Coverflow + -7.f, //CF_Z the Z Position of the Coverflow + + 0.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + 0.f, //Rotation sets the rotation of each cover + 1.1f, //X_Distance sets the distance between the covers + 0.2f, //Center_Distance sets the distance between the centered and the non centered covers + 0.01f, //Z_Distance sets the pushback amount + 0.0f, //Y_Distance sets the elevation amount + + 22.f //zoom level + +}; + +struct Preset presetYACReaderFlowOverlappedStripeConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 2.f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 3.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.08f, //View_rotate_add sets the speed of the rotation + 0.08f, //View_rotate_sub sets the speed of reversing the rotation + 30.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + -0.2f, //CF_Y the Y Position of the Coverflow + -7.f, //CF_Z the Z Position of the Coverflow + + 0.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + 0.f, //Rotation sets the rotation of each cover + 0.18f, //X_Distance sets the distance between the covers + 1.f, //Center_Distance sets the distance between the centered and the non centered covers + 0.1f, //Z_Distance sets the pushback amount + 0.0f, //Y_Distance sets the elevation amount + + 22.f //zoom level + +}; + +struct Preset pressetYACReaderFlowUpConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 2.5f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 3.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.08f, //View_rotate_add sets the speed of the rotation + 0.08f, //View_rotate_sub sets the speed of reversing the rotation + 5.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + -0.2f, //CF_Y the Y Position of the Coverflow + -7.f, //CF_Z the Z Position of the Coverflow + + 0.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + -50.f, //Rotation sets the rotation of each cover + 0.18f, //X_Distance sets the distance between the covers + 1.f, //Center_Distance sets the distance between the centered and the non centered covers + 0.1f, //Z_Distance sets the pushback amount + -0.1f, //Y_Distance sets the elevation amount + + 22.f //zoom level + +}; + +struct Preset pressetYACReaderFlowDownConfig = { + 0.08f, //Animation_step sets the speed of the animation + 1.5f, //Animation_speedup sets the acceleration of the animation + 0.1f, //Animation_step_max sets the maximum speed of the animation + 2.5f, //Animation_Fade_out_dis sets the distance of view + + 1.5f, //pre_rotation sets the rotation increasion + 3.f, //View_rotate_light_strenght sets the light strenght on rotation + 0.08f, //View_rotate_add sets the speed of the rotation + 0.08f, //View_rotate_sub sets the speed of reversing the rotation + 5.f, //View_angle sets the maximum view angle + + 0.f, //CF_X the X Position of the Coverflow + -0.2f, //CF_Y the Y Position of the Coverflow + -7.f, //CF_Z the Z Position of the Coverflow + + 0.f, //CF_RX the X Rotation of the Coverflow + 0.f, //CF_RY the Y Rotation of the Coverflow + 0.f, //CF_RZ the Z Rotation of the Coverflow + + -50.f, //Rotation sets the rotation of each cover + 0.18f, //X_Distance sets the distance between the covers + 1.f, //Center_Distance sets the distance between the centered and the non centered covers + 0.1f, //Z_Distance sets the pushback amount + 0.1f, //Y_Distance sets the elevation amount + + 22.f //zoom level +}; +/*Constructor*/ +YACReaderFlowGL::YACReaderFlowGL(QWidget *parent,struct Preset p) + :QGLWidget(QGLFormat(QGL::SampleBuffers), parent),numObjects(0),lazyPopulateObjects(-1),bUseVSync(false),hasBeenInitialized(false) +{ + updateCount = 0; + config = p; + + currentSelected = 0; + + centerPos.x = 0.f; + centerPos.y = 0.f; + centerPos.z = 1.f; + centerPos.rot = 0.f; + + /*** Style ***/ + shadingTop = 0.8f; + shadingBottom = 0.02f; + reflectionUp = 0.f; + reflectionBottom = 0.6f; + + /*** System variables ***/ + numObjects = 0; + //CFImage Dummy; + viewRotate = 0.f; + viewRotateActive = 0; + stepBackup = config.animationStep/config.animationSpeedUp; + + /*QTimer * timer = new QTimer(); + connect(timer, SIGNAL(timeout()), this, SLOT(updateImageData())); + timer->start(70); + */ + + /*loader = new WidgetLoader(0,this); + loader->flow = this; + QThread * loaderThread = new QThread(parent); + + loader->moveToThread(loaderThread); + + loaderThread->start();*/ + + QGLFormat f = format(); + f.setVersion(2, 1); + f.setSwapInterval(0); + setFormat(f); + + timerId = startTimer(updateInterval); + +} + +void YACReaderFlowGL::timerEvent(QTimerEvent * event) +{ + if(timerId == event->timerId()) + updateGL(); + + //if(!worker->isRunning()) + //worker->start(); +} + +void YACReaderFlowGL::startAnimationTimer() +{ + if(timerId == -1) + timerId = startTimer(updateInterval); +} + +void YACReaderFlowGL::stopAnimationTimer() +{ + if(timerId != -1) + { + killTimer(timerId); + timerId = -1; + } +} + +YACReaderFlowGL::~YACReaderFlowGL() +{ + +} + +QSize YACReaderFlowGL::minimumSizeHint() const +{ + return QSize(320, 200); +} + +/*QSize YACReaderFlowGL::sizeHint() const +{ + return QSize(320, 200); +}*/ + +void YACReaderFlowGL::initializeGL() +{ + glEnable(GL_DEPTH_TEST); + glEnable(GL_CULL_FACE); + glEnable(GL_COLOR_MATERIAL); + glShadeModel(GL_SMOOTH); + glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + + defaultTexture = bindTexture(QImage(":/images/defaultCover.png"),GL_TEXTURE_2D,GL_RGBA,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); + markTexture = bindTexture(QImage(":/images/readRibbon.png"),GL_TEXTURE_2D,GL_RGBA,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); + readingTexture = bindTexture(QImage(":/images/readingRibbon.png"),GL_TEXTURE_2D,GL_RGBA,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); + if(lazyPopulateObjects!=-1) + populate(lazyPopulateObjects); + + hasBeenInitialized = true; +} + +void YACReaderFlowGL::paintGL() +{ + /*glClearDepth(1.0); + glClearColor(1,1,1,1);*/ + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + /*glLoadIdentity(); + glTranslatef(0.0, 0.0, -10.0); + glPopMatrix();*/ + if(numObjects>0) + { + updatePositions(); + udpatePerspective(width(),height()); + draw(); + } +} + +void YACReaderFlowGL::resizeGL(int width, int height) +{ + + fontSize = (width + height) * 0.010; + if(fontSize < 10) + fontSize = 10; + + //int side = qMin(width, height); + udpatePerspective(width,height); + + if(numObjects>0) + updatePositions(); +} + +void YACReaderFlowGL::udpatePerspective(int width, int height) +{ + glViewport(0, 0, width, height); + + glMatrixMode(GL_PROJECTION); + glLoadIdentity(); + + gluPerspective(20.0, GLdouble(width) / (float)height, 1.0, 200.0); + + glMatrixMode(GL_MODELVIEW); +} + +//----------------------------------------------------------------------------- +/*Private*/ +void YACReaderFlowGL::calcPos(CFImage *CF,int pos) +{ + if(pos == 0){ + CF->current = centerPos; + }else{ + if(pos > 0){ + CF->current.x = (config.centerDistance)+(config.xDistance*pos); + CF->current.y = config.yDistance*pos*-1; + CF->current.z = config.zDistance*pos*-1; + CF->current.rot = config.rotation; + }else{ + CF->current.x = (config.centerDistance)*-1+(config.xDistance*pos); + CF->current.y = config.yDistance*pos; + CF->current.z = config.zDistance*pos; + CF->current.rot = config.rotation*-1; + } + } + +} +void YACReaderFlowGL::calcRV(RVect *RV,int pos) +{ + calcPos(&dummy,pos); + + RV->x = dummy.current.x; + RV->y = dummy.current.y; + RV->z = dummy.current.z; + RV->rot = dummy.current.rot; + +} +bool YACReaderFlowGL::animate(RVect *Current,RVect to) +{ + float rotDiff = to.rot-Current->rot; + float xDiff = to.x-Current->x; + float yDiff = to.y-Current->y; + float zDiff = to.z-Current->z; + + if(fabs(rotDiff) < 0.01 + && fabs(xDiff) < 0.001 + && fabs(yDiff) < 0.001 + && fabs(zDiff) < 0.001) + return true; + + //calculate and apply positions + Current->x = Current->x+(xDiff)*config.animationStep; + Current->y = Current->y+(yDiff)*config.animationStep; + Current->z = Current->z+(zDiff)*config.animationStep; + + if(fabs(rotDiff) > 0.01){ + Current->rot = Current->rot+(rotDiff)*(config.animationStep*config.preRotation); + } + else + { + viewRotateActive = 0; + } + + return false; +} +void YACReaderFlowGL::drawCover(CFImage *CF) +{ + float w = CF->width; + float h = CF->height; + + //fadeout + float opacity = 1-1/(config.animationFadeOutDist+config.viewRotateLightStrenght*fabs(viewRotate))*fabs(0-CF->current.x); + + glLoadIdentity(); + glTranslatef(config.cfX,config.cfY,config.cfZ); + glRotatef(config.cfRX,1,0,0); + glRotatef(viewRotate*config.viewAngle+config.cfRY,0,1,0); + glRotatef(config.cfRZ,0,0,1); + + glTranslatef( CF->current.x, CF->current.y, CF->current.z ); + + glPushMatrix(); + glRotatef(CF->current.rot,0,1,0); + + glEnable(GL_TEXTURE_2D); + glBindTexture(GL_TEXTURE_2D, CF->img); + + //calculate shading + float LShading = ((config.rotation != 0 )?((CF->current.rot < 0)?1-1/config.rotation*CF->current.rot:1):1); + float RShading = ((config.rotation != 0 )?((CF->current.rot > 0)?1-1/(config.rotation*-1)*CF->current.rot:1):1); + float LUP = shadingTop+(1-shadingTop)*LShading; + float LDOWN = shadingBottom+(1-shadingBottom)*LShading; + float RUP = shadingTop+(1-shadingTop)*RShading; + float RDOWN = shadingBottom+(1-shadingBottom)*RShading;; + + + //DrawCover + glBegin(GL_QUADS); + + //esquina inferior izquierda + glColor4f(LDOWN*opacity,LDOWN*opacity,LDOWN*opacity,1); + glTexCoord2f(0.0f, 1.0f); + glVertex3f(w/2.f*-1.f, -0.5f, 0.f); + + //esquina inferior derecha + glColor4f(RDOWN*opacity,RDOWN*opacity,RDOWN*opacity,1); + glTexCoord2f(1.0f, 1.0f); + glVertex3f(w/2.f, -0.5f, 0.f); + + //esquina superior derecha + glColor4f(RUP*opacity,RUP*opacity,RUP*opacity,1); + glTexCoord2f(1.0f, 0.0f); + glVertex3f(w/2.f, -0.5f+h, 0.f); + + //esquina superior izquierda + glColor4f(LUP*opacity,LUP*opacity,LUP*opacity,1); + glTexCoord2f(0.0f, 0.0f); + glVertex3f(w/2.f*-1.f, -0.5f+h, 0.f); + + glEnd(); + + + + //Draw reflection + glBegin(GL_QUADS); + + //esquina inferior izquierda + glColor4f(LUP*opacity*reflectionUp,LUP*opacity*reflectionUp,LUP*opacity*reflectionUp,opacity*reflectionUp); + glTexCoord2f(0.0f, 0.0f); + glVertex3f(w/2.f*-1.f, -0.5f-h, 0.f); + + //esquina inferior derecha + glTexCoord2f(1.0f, 0.0f); + glVertex3f(w/2.f, -0.5f-h, 0.f); + + //esquina superior derecha + glColor4f(opacity*reflectionBottom,opacity*reflectionBottom,opacity*reflectionBottom,opacity*reflectionBottom); + glTexCoord2f(1.0f, 1.0f); + glVertex3f(w/2.f, -0.5f, 0.f); + + //esquina superior izquierda + glTexCoord2f(0.0f, 1.0f); + glVertex3f(w/2.f*-1.f, -0.5f, 0.f); + + glEnd(); + glDisable(GL_TEXTURE_2D); + + if(showMarks && loaded[CF->index] && marks[CF->index] != Unread) + { + glEnable(GL_TEXTURE_2D); + if(marks[CF->index] == Read) + glBindTexture(GL_TEXTURE_2D, markTexture); + else + glBindTexture(GL_TEXTURE_2D, readingTexture); + glBegin(GL_QUADS); + + //esquina inferior izquierda + glColor4f(RUP*opacity,RUP*opacity,RUP*opacity,1); + glTexCoord2f(0.0f, 1.0f); + glVertex3f(w/2.f-0.2, -0.685f+h, 0.001f); + + //esquina inferior derecha + glColor4f(RUP*opacity,RUP*opacity,RUP*opacity,1); + glTexCoord2f(1.0f, 1.0f); + glVertex3f(w/2.f-0.05, -0.685f+h, 0.001f); + + //esquina superior derecha + glColor4f(RUP*opacity,RUP*opacity,RUP*opacity,1); + glTexCoord2f(1.0f, 0.0f); + glVertex3f(w/2.f-0.05, -0.485f+h, 0.001f); + + //esquina superior izquierda + glColor4f(RUP*opacity,RUP*opacity,RUP*opacity,1); + glTexCoord2f(0.0f, 0.0f); + glVertex3f(w/2.f-0.2, -0.485f+h, 0.001f); + + glEnd(); + glDisable(GL_TEXTURE_2D); + } + + + glPopMatrix(); +} + +/*Public*/ +void YACReaderFlowGL::cleanupAnimation() +{ + config.animationStep = stepBackup; + viewRotateActive = 0; +} + +void YACReaderFlowGL::draw() +{ + int CS = currentSelected; + int count; + + + //Draw right Covers + for(count = numObjects-1;count > -1;count--){ + if(count > CS){ + drawCover(&cfImages[count]); + } + } + + //Draw left Covers + for(count = 0;count < numObjects-1;count++){ + if(count < CS){ + drawCover(&cfImages[count]); + } + } + + //Draw Center Cover + drawCover(&cfImages[CS]); + + //glDisable(GL_DEPTH_TEST); + glMatrixMode(GL_PROJECTION); + glPushMatrix(); + glLoadIdentity(); + glOrtho(-(float(width())/height())/2.0,(float(width())/height())/2.0, 0, 1, -10, 10); + glMatrixMode(GL_MODELVIEW); + glLoadIdentity(); + + glColor4f( 1.0f, 1.0f, 1.0f, 1.0f ); + + renderText(10, fontSize + 10,QString("%1/%2").arg(currentSelected+1).arg(numObjects),QFont("Arial", fontSize)); + + glEnable(GL_DEPTH_TEST); + glMatrixMode(GL_PROJECTION); + glPopMatrix(); + glMatrixMode(GL_MODELVIEW); +} + +void YACReaderFlowGL::showPrevious() +{ + startAnimationTimer(); + + if(currentSelected > 0){ + + currentSelected--; + emit centerIndexChanged(currentSelected); + config.animationStep *= config.animationSpeedUp; + + if(config.animationStep > config.animationStepMax){ + config.animationStep = config.animationStepMax; + } + + if(viewRotateActive && viewRotate > -1){ + viewRotate -= config.viewRotateAdd; + } + + viewRotateActive = 1; + + } +} + +void YACReaderFlowGL::showNext() +{ + startAnimationTimer(); + + if(currentSelected < numObjects-1){ + + currentSelected++; + emit centerIndexChanged(currentSelected); + config.animationStep *= config.animationSpeedUp; + + if(config.animationStep > config.animationStepMax){ + config.animationStep = config.animationStepMax; + } + + if(viewRotateActive && viewRotate < 1){ + viewRotate += config.viewRotateAdd; + } + + viewRotateActive = 1; + } +} + +void YACReaderFlowGL::setCurrentIndex(int pos) +{ + startAnimationTimer(); + + currentSelected = pos; + + config.animationStep *= config.animationSpeedUp; + + if(config.animationStep > config.animationStepMax){ + config.animationStep = config.animationStepMax; + } + + if(viewRotateActive && viewRotate < 1){ + viewRotate += config.viewRotateAdd; + } + + viewRotateActive = 1; +} + +void YACReaderFlowGL::updatePositions() +{ + int count; + + bool stopAnimation = true; + for(count = numObjects-1;count > -1;count--){ + calcRV(&cfImages[count].animEnd,count-currentSelected); + if(!animate(&cfImages[count].current,cfImages[count].animEnd)) + stopAnimation = false; + } + + //slowly reset view angle + if(!viewRotateActive){ + viewRotate += (0-viewRotate)*config.viewRotateSub; + } + + if(fabs (cfImages[currentSelected].current.x - cfImages[currentSelected].animEnd.x) < 1)//viewRotate < 0.2) + { + cleanupAnimation(); + if(updateCount >= 0) //TODO parametrizar + { + + updateCount = 0; + updateImageData(); + } + else + updateCount++; + } + else + updateCount++; + + if(stopAnimation) + stopAnimationTimer(); + +} + +void YACReaderFlowGL::insert(char *name, GLuint Tex, float x, float y,int item) +{ + startAnimationTimer(); + + Q_UNUSED(name) + //set a new entry + if(item == -1){ + + if(numObjects == 0){ + cfImages = (CFImage*)malloc(sizeof(CFImage)); + } + else + { + cfImages = (CFImage*)realloc(cfImages,(numObjects+1)*sizeof(CFImage)); + } + + item = numObjects; + numObjects++; + + calcRV(&cfImages[item].current,item); + cfImages[item].current.x += 1; + cfImages[item].current.rot = 90; + } + + cfImages[item].img = Tex; + cfImages[item].width = x; + cfImages[item].height = y; + cfImages[item].index = item; + //strcpy(cfImages[item].name,name); + + +} + +void YACReaderFlowGL::remove(int item) +{ + startAnimationTimer(); + + loaded.remove(item); + marks.remove(item); + paths.removeAt(item); + + //reposition current selection + if(item < currentSelected && currentSelected != 0){ + currentSelected--; + } + + int count = item; + while(count <= numObjects-2){ + cfImages[count] = cfImages[count+1]; + cfImages[count].index--; + count++; + } + + cfImages = (CFImage*)realloc(cfImages,numObjects*sizeof(CFImage)); + + numObjects--; +} + +/*Info*/ +CFImage YACReaderFlowGL::getCurrentSelected() +{ + return cfImages[currentSelected]; +} + +void YACReaderFlowGL::replace(char *name, GLuint Tex, float x, float y,int item) +{ + startAnimationTimer(); + + Q_UNUSED(name) + if(cfImages[item].index == item) + { + cfImages[item].img = Tex; + cfImages[item].width = x; + cfImages[item].height = y; + loaded[item]=true; + } + else + loaded[item]=false; +} + +void YACReaderFlowGL::populate(int n) +{ + emit centerIndexChanged(0); + float x = 1; + float y = 1 * (700.f/480.0f); + int i; + + for(i = 0;i(n,false); + //marks = QVector(n,false); + + + + //worker->start(); +} + +void YACReaderFlowGL::reset() +{ + startAnimationTimer(); + + currentSelected = 0; + loaded.clear(); + + for(int i = 0;i0) + delete[] cfImages; + numObjects = 0; + + if(!hasBeenInitialized) + lazyPopulateObjects = -1; +} + +void YACReaderFlowGL::reload() +{ + startAnimationTimer(); + + int n = numObjects; + reset(); + populate(n); +} + +//slots +void YACReaderFlowGL::setCF_RX(int value) +{ + startAnimationTimer(); + + config.cfRX = value; +} +void YACReaderFlowGL::setCF_RY(int value) +{ + startAnimationTimer(); + + config.cfRY = value; +} +void YACReaderFlowGL::setCF_RZ(int value) +{ + startAnimationTimer(); + + config.cfRZ = value; +} + +void YACReaderFlowGL::setZoom(int zoom) +{ + startAnimationTimer(); + + int width = this->width(); + int height = this->height(); + glViewport(0, 0, width, height); + + glMatrixMode(GL_PROJECTION); + glLoadIdentity(); + //float sideX = ((float(width)/height)/2)*1.5; + //float sideY = 0.5*1.5; + gluPerspective(zoom, (float)width / (float)height, 1.0, 200.0); + //glOrtho(-sideX, sideX, -sideY+0.2, +sideY+0.2, 4, 11.0); + + glMatrixMode(GL_MODELVIEW); + +} + +void YACReaderFlowGL::setRotation(int angle) +{ + startAnimationTimer(); + + config.rotation = -angle; +} +//sets the distance between the covers +void YACReaderFlowGL::setX_Distance(int distance) +{ + startAnimationTimer(); + + config.xDistance = distance/100.0; +} +//sets the distance between the centered and the non centered covers +void YACReaderFlowGL::setCenter_Distance(int distance) +{ + startAnimationTimer(); + + config.centerDistance = distance/100.0; +} +//sets the pushback amount +void YACReaderFlowGL::setZ_Distance(int distance) +{ + startAnimationTimer(); + + config.zDistance = distance/100.0; +} + +void YACReaderFlowGL::setCF_Y(int value) +{ + startAnimationTimer(); + + config.cfY = value/100.0; +} + +void YACReaderFlowGL::setCF_Z(int value) +{ + startAnimationTimer(); + + config.cfZ = value; +} + +void YACReaderFlowGL::setY_Distance(int value) +{ + startAnimationTimer(); + + config.yDistance = value / 100.0; +} + +void YACReaderFlowGL::setFadeOutDist(int value) +{ + startAnimationTimer(); + + config.animationFadeOutDist = value; +} + +void YACReaderFlowGL::setLightStrenght(int value) +{ + startAnimationTimer(); + + config.viewRotateLightStrenght = value; +} + +void YACReaderFlowGL::setMaxAngle(int value) +{ + startAnimationTimer(); + + config.viewAngle = value; +} + +void YACReaderFlowGL::setPreset(const Preset & p) +{ + startAnimationTimer(); + + config = p; +} + +void YACReaderFlowGL::setPerformance(Performance performance) +{ + if(this->performance != performance) + { + startAnimationTimer(); + + this->performance = performance; + reload(); + } +} + +void YACReaderFlowGL::useVSync(bool b) +{ + if(bUseVSync != b) + { + bUseVSync = b; + if(b) + { + QGLFormat f = format(); + f.setVersion(2, 1); + f.setSwapInterval(1); + setFormat(f); + } + else + { + QGLFormat f = format(); + f.setVersion(2, 1); + f.setSwapInterval(0); + setFormat(f); + } + reset(); + } +} +void YACReaderFlowGL::setShowMarks(bool value) +{ + startAnimationTimer(); + + showMarks = value; +} +void YACReaderFlowGL::setMarks(QVector marks) +{ + startAnimationTimer(); + + this->marks = marks; +} +void YACReaderFlowGL::setMarkImage(QImage & image) +{ + Q_UNUSED(image); + //qué pasa la primera vez?? + //deleteTexture(markTexture); + //markTexture = bindTexture(image,GL_TEXTURE_2D,GL_RGBA,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); +} +void YACReaderFlowGL::markSlide(int index, YACReaderComicReadStatus status) +{ + startAnimationTimer(); + + marks[index] = status; +} +void YACReaderFlowGL::unmarkSlide(int index) +{ + startAnimationTimer(); + + marks[index] = YACReader::Unread; +} +void YACReaderFlowGL::setSlideSize(QSize size) +{ + Q_UNUSED(size); + //TODO calcular el tamaño del widget +} +void YACReaderFlowGL::clear() +{ + reset(); +} + +void YACReaderFlowGL::setCenterIndex(unsigned int index) +{ + setCurrentIndex(index); +} +void YACReaderFlowGL::showSlide(int index) +{ + setCurrentIndex(index); +} +int YACReaderFlowGL::centerIndex() +{ + return currentSelected; +} +void YACReaderFlowGL::updateMarks() +{ + //do nothing +} +/*void YACReaderFlowGL::setFlowType(FlowType flowType) +{ + //TODO esperar a que se reimplemente flowtype +}*/ +void YACReaderFlowGL::render() +{ + //do nothing +} + +//EVENTOS +void YACReaderFlowGL::wheelEvent(QWheelEvent * event) +{ + if(event->delta()<0) + showNext(); + else + showPrevious(); + event->accept(); +} + +void YACReaderFlowGL::keyPressEvent(QKeyEvent *event) +{ + if(event->key() == Qt::Key_Left) + { + if(event->modifiers() == Qt::ControlModifier) + setCurrentIndex((currentSelected-10<0)?0:currentSelected-10); + else + showPrevious(); + event->accept(); + return; + } + + if(event->key() == Qt::Key_Right) + { + if(event->modifiers() == Qt::ControlModifier) + setCurrentIndex((currentSelected+10>=numObjects)?numObjects-1:currentSelected+10); + else + showNext(); + event->accept(); + return; + } + + if(event->key() == Qt::Key_Up) + { + //emit selected(centerIndex()); + return; + } + + event->ignore(); +} + +void YACReaderFlowGL::mousePressEvent(QMouseEvent *event) +{ + if(event->button() == Qt::LeftButton) + { + float x,y; + x = event->x(); + y = event->y(); + GLint viewport[4]; + GLdouble modelview[16]; + GLdouble projection[16]; + GLfloat winX, winY, winZ; + GLdouble posX, posY, posZ; + + glGetDoublev( GL_MODELVIEW_MATRIX, modelview ); + glGetDoublev( GL_PROJECTION_MATRIX, projection ); + glGetIntegerv( GL_VIEWPORT, viewport ); + + winX = (float)x; + winY = (float)viewport[3] - (float)y; + glReadPixels( x, int(winY), 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &winZ ); + + gluUnProject( winX, winY, winZ, modelview, projection, viewport, &posX, &posY, &posZ); + + if(posX >= 0.5) + { + //int index = currentSelected+1; + //while((cfImages[index].current.x-cfImages[index].width/(2.0*config.rotation)) < posX) + // index++; + //setCurrentIndex(index-1); + showNext(); + } + else if(posX <=-0.5) + showPrevious(); + } else + QGLWidget::mousePressEvent(event); +} + +void YACReaderFlowGL::mouseDoubleClickEvent(QMouseEvent* event) +{ + float x,y; + x = event->x(); + y = event->y(); + GLint viewport[4]; + GLdouble modelview[16]; + GLdouble projection[16]; + GLfloat winX, winY, winZ; + GLdouble posX, posY, posZ; + + glGetDoublev( GL_MODELVIEW_MATRIX, modelview ); + glGetDoublev( GL_PROJECTION_MATRIX, projection ); + glGetIntegerv( GL_VIEWPORT, viewport ); + + winX = (float)x; + winY = (float)viewport[3] - (float)y; + glReadPixels( x, int(winY), 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &winZ ); + + gluUnProject( winX, winY, winZ, modelview, projection, viewport, &posX, &posY, &posZ); + + if(posX <= 0.5 && posX >= -0.5) + { + emit selected(centerIndex()); + event->accept(); + } + +} + +YACReaderComicFlowGL::YACReaderComicFlowGL(QWidget *parent,struct Preset p ) + :YACReaderFlowGL(parent,p) +{ + worker = new ImageLoaderGL(this); + worker->flow = this; +} + +void YACReaderComicFlowGL::setImagePaths(QStringList paths) +{ + worker->reset(); + reset(); + numObjects = 0; + if(lazyPopulateObjects!=-1 || hasBeenInitialized) + YACReaderFlowGL::populate(paths.size()); + lazyPopulateObjects = paths.size(); + this->paths = paths; + //numObjects = paths.size(); +} + + +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// + +void YACReaderComicFlowGL::updateImageData() +{ + // can't do anything, wait for the next possibility + if(worker->busy()) + return; + + // set image of last one + int idx = worker->index(); + if( idx >= 0 && !worker->result().isNull()) + { + if(!loaded[idx]) + { + float x = 1; + QImage img = worker->result(); + GLuint cover; + if(performance == high || performance == ultraHigh) + cover = bindTexture(img, GL_TEXTURE_2D,GL_RGB,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); + else + cover = bindTexture(img, GL_TEXTURE_2D,GL_RGB,QGLContext::LinearFilteringBindOption); + float y = 1 * (float(img.height())/img.width()); + replace("cover", cover, x, y,idx); + /*CFImages[idx].width = x; + CFImages[idx].height = y; + CFImages[idx].img = worker->resultTexture; + strcpy(CFImages[idx].name,"cover");*/ + //loaded[idx] = true; + //numImagesLoaded++; + } + } + + // try to load only few images on the left and right side + // i.e. all visible ones plus some extra + int count=8; + switch(performance) + { + case low: + count = 8; + break; + case medium: + count = 10; + break; + case high: + count = 12; + break; + case ultraHigh: + count = 14; + break; + } + int * indexes = new int[2*count+1]; + int center = currentSelected; + indexes[0] = center; + for(int j = 0; j < count; j++) + { + indexes[j*2+1] = center+j+1; + indexes[j*2+2] = center-j-1; + } + for(int c = 0; c < 2*count+1; c++) + { + int i = indexes[c]; + if((i >= 0) && (i < numObjects)) + if(!loaded[i])//slide(i).isNull()) + { + //loader->loadTexture(i); + //loaded[i]=true; + // schedule thumbnail generation + if(paths.size()>0) + { + QString fname = paths.at(i); + //loaded[i]=true; + + worker->generate(i, fname); + } + delete[] indexes; + return; + } + } +} + +void YACReaderComicFlowGL::remove(int item) +{ + worker->lock(); + worker->reset(); + YACReaderFlowGL::remove(item); + worker->unlock(); +} + + +YACReaderPageFlowGL::YACReaderPageFlowGL(QWidget *parent,struct Preset p ) + :YACReaderFlowGL(parent,p) +{ + worker = new ImageLoaderByteArrayGL(this); + worker->flow = this; +} + +YACReaderPageFlowGL::~YACReaderPageFlowGL() +{ + this->killTimer(timerId); + //worker->deleteLater(); + rawImages.clear(); + free(cfImages); +} + +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// + +void YACReaderPageFlowGL::updateImageData() +{ + // can't do anything, wait for the next possibility + if(worker->busy()) + return; + + // set image of last one + int idx = worker->index(); + if( idx >= 0 && !worker->result().isNull()) + { + if(!loaded[idx]) + { + float x = 1; + QImage img = worker->result(); + GLuint cover; + if(performance == high || performance == ultraHigh) + cover = bindTexture(img, GL_TEXTURE_2D,GL_RGB,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); + else + cover = bindTexture(img, GL_TEXTURE_2D,GL_RGB,QGLContext::LinearFilteringBindOption); + float y = 1 * (float(img.height())/img.width()); + replace("cover", cover, x, y,idx); + /*CFImages[idx].width = x; + CFImages[idx].height = y; + CFImages[idx].img = worker->resultTexture; + strcpy(CFImages[idx].name,"cover");*/ + loaded[idx] = true; + //numImagesLoaded++; + } + } + + // try to load only few images on the left and right side + // i.e. all visible ones plus some extra + int count=8; + switch(performance) + { + case low: + count = 8; + break; + case medium: + count = 10; + break; + case high: + count = 12; + break; + case ultraHigh: + count = 14; + break; + } + int * indexes = new int[2*count+1]; + int center = currentSelected; + indexes[0] = center; + for(int j = 0; j < count; j++) + { + indexes[j*2+1] = center+j+1; + indexes[j*2+2] = center-j-1; + } + for(int c = 0; c < 2*count+1; c++) + { + int i = indexes[c]; + if((i >= 0) && (i < numObjects)) + if(rawImages.size()>0) + + if(!loaded[i]&&imagesReady[i])//slide(i).isNull()) + { + //loader->loadTexture(i); + //loaded[i]=true; + // schedule thumbnail generation + + //loaded[i]=true; + + worker->generate(i, rawImages.at(i)); + + delete[] indexes; + return; + } + } +} + +void YACReaderPageFlowGL::populate(int n) +{ + worker->reset(); + if(lazyPopulateObjects!=-1 || hasBeenInitialized) + YACReaderFlowGL::populate(n); + lazyPopulateObjects = n; + imagesReady = QVector (n,false); + rawImages = QVector (n); + imagesSetted = QVector (n,false); //puede sobrar +} + + +//----------------------------------------------------------------------------- +//ImageLoader +//----------------------------------------------------------------------------- +QImage ImageLoaderGL::loadImage(const QString& fileName) +{ + QImage image; + bool result = image.load(fileName); + + switch(flow->performance) + { + case low: + image = image.scaledToWidth(200,Qt::SmoothTransformation); + break; + case medium: + image = image.scaledToWidth(256,Qt::SmoothTransformation); + break; + case high: + image = image.scaledToWidth(320,Qt::SmoothTransformation); + break; + + } + + if(!result) + return QImage(); + + return image; +} + +ImageLoaderGL::ImageLoaderGL(YACReaderFlowGL * flow): +QThread(),flow(flow),restart(false), working(false), idx(-1) +{ + +} + +ImageLoaderGL::~ImageLoaderGL() +{ + mutex.lock(); + condition.wakeOne(); + mutex.unlock(); + wait(); +} + +bool ImageLoaderGL::busy() const +{ + return isRunning() ? working : false; +} + +void ImageLoaderGL::generate(int index, const QString& fileName) +{ + mutex.lock(); + this->idx = index; + this->fileName = fileName; + this->size = size; + this->img = QImage(); + mutex.unlock(); + + if (!isRunning()) + start(); + else + { + // already running, wake up whenever ready + restart = true; + condition.wakeOne(); + } +} + +void ImageLoaderGL::lock() +{ + mutex.lock(); +} + +void ImageLoaderGL::unlock() +{ + mutex.unlock(); +} + +void ImageLoaderGL::run() +{ + for(;;) + { + // copy necessary data + mutex.lock(); + this->working = true; + QString fileName = this->fileName; + mutex.unlock(); + + QImage image = loadImage(fileName); + + // let everyone knows it is ready + mutex.lock(); + this->working = false; + this->img = image; + mutex.unlock(); + + // put to sleep + mutex.lock(); + if (!this->restart) + condition.wait(&mutex); + restart = false; + mutex.unlock(); + } +} + +QImage ImageLoaderGL::result() +{ + return img; +} + +//----------------------------------------------------------------------------- +//ImageLoader +//----------------------------------------------------------------------------- +QImage ImageLoaderByteArrayGL::loadImage(const QByteArray& raw) +{ + QImage image; + bool result = image.loadFromData(raw); + + switch(flow->performance) + { + case low: + image = image.scaledToWidth(128,Qt::SmoothTransformation); + break; + case medium: + image = image.scaledToWidth(196,Qt::SmoothTransformation); + break; + case high: + image = image.scaledToWidth(256,Qt::SmoothTransformation); + break; + case ultraHigh: + image = image.scaledToWidth(320,Qt::SmoothTransformation); + break; + } + + if(!result) + return QImage(); + + return image; +} + +ImageLoaderByteArrayGL::ImageLoaderByteArrayGL(YACReaderFlowGL * flow): +QThread(),flow(flow),restart(false), working(false), idx(-1) +{ + +} + +ImageLoaderByteArrayGL::~ImageLoaderByteArrayGL() +{ + mutex.lock(); + condition.wakeOne(); + mutex.unlock(); + wait(); +} + +bool ImageLoaderByteArrayGL::busy() const +{ + return isRunning() ? working : false; +} + +void ImageLoaderByteArrayGL::generate(int index, const QByteArray& raw) +{ + mutex.lock(); + this->idx = index; + this->rawData = raw; + this->size = size; + this->img = QImage(); + mutex.unlock(); + + if (!isRunning()) + start(); + else + { + // already running, wake up whenever ready + restart = true; + condition.wakeOne(); + } +} + +void ImageLoaderByteArrayGL::run() +{ + for(;;) + { + // copy necessary data + mutex.lock(); + this->working = true; + QByteArray raw = this->rawData; + mutex.unlock(); + + QImage image = loadImage(raw); + + // let everyone knows it is ready + mutex.lock(); + this->working = false; + this->img = image; + mutex.unlock(); + + // put to sleep + mutex.lock(); + if (!this->restart) + condition.wait(&mutex); + restart = false; + mutex.unlock(); + } +} + +QImage ImageLoaderByteArrayGL::result() +{ + return img; +} + +//WidgetLoader::WidgetLoader(QWidget *parent, QGLWidget * shared) +// :QGLWidget(parent,shared) +//{ +//} +// +//void WidgetLoader::loadTexture(int index) +//{ +// QImage image; +// bool result = image.load(QString("./cover%1.jpg").arg(index+1)); +// //image = image.scaledToWidth(128,Qt::SmoothTransformation); //TODO parametrizar +// flow->cfImages[index].width = 0.5; +// flow->cfImages[index].height = 0.5 * (float(image.height())/image.width()); +// flow->cfImages[index].img = bindTexture(image, GL_TEXTURE_2D,GL_RGBA,QGLContext::LinearFilteringBindOption | QGLContext::MipmapBindOption); +//} diff --git a/common/yacreader_flow_gl.h b/common/yacreader_flow_gl.h new file mode 100644 index 00000000..a94453e4 --- /dev/null +++ b/common/yacreader_flow_gl.h @@ -0,0 +1,402 @@ +//OpenGL Coverflow API by J.Roth +#ifndef __YACREADER_FLOW_GL_H +#define __YACREADER_FLOW_GL_H + +#include +#include +#include +#include + +#include +#include +#include + +#include "pictureflow.h" //TODO mover los tipos de flow de sitio + +class ImageLoaderGL; +class QGLContext; +class WidgetLoader; +class ImageLoaderByteArrayGL; + +enum Performance +{ + low=0, + medium, + high, + ultraHigh +}; + +//Cover Vector +struct RVect{ + float x; + float y; + float z; + float rot; +}; + +//the cover info struct +struct CFImage{ + GLuint img; + //char name[256]; + + float width; + float height; + + int index; + + RVect current; + RVect animEnd; +}; + +struct Preset{ + /*** Animation Settings ***/ + //sets the speed of the animation + float animationStep; + //sets the acceleration of the animation + float animationSpeedUp; + //sets the maximum speed of the animation + float animationStepMax; + //sets the distance of view + float animationFadeOutDist; + //sets the rotation increasion + float preRotation; + //sets the light strenght on rotation + float viewRotateLightStrenght; + //sets the speed of the rotation + float viewRotateAdd; + //sets the speed of reversing the rotation + float viewRotateSub; + //sets the maximum view angle + float viewAngle; + + /*** Position Configuration ***/ + //the X Position of the Coverflow + float cfX; + //the Y Position of the Coverflow + float cfY; + //the Z Position of the Coverflow + float cfZ; + //the X Rotation of the Coverflow + float cfRX; + //the Y Rotation of the Coverflow + float cfRY; + //the Z Rotation of the Coverflow + float cfRZ; + //sets the rotation of each cover + float rotation; + //sets the distance between the covers + float xDistance; + //sets the distance between the centered and the non centered covers + float centerDistance; + //sets the pushback amount + float zDistance; + //sets the elevation amount + float yDistance; + + float zoom; +}; + +extern struct Preset defaultYACReaderFlowConfig; +extern struct Preset presetYACReaderFlowClassicConfig; +extern struct Preset presetYACReaderFlowStripeConfig; +extern struct Preset presetYACReaderFlowOverlappedStripeConfig; +extern struct Preset pressetYACReaderFlowUpConfig; +extern struct Preset pressetYACReaderFlowDownConfig; + +class YACReaderFlowGL : public QGLWidget +{ + Q_OBJECT +protected: + int timerId; + /*** System variables ***/ + CFImage dummy; + int viewRotateActive; + float stepBackup; + + /*functions*/ + void calcPos(CFImage *CF,int pos); + void calcRV(RVect *RV,int pos); + //returns true if the animation is finished for Current + bool animate(RVect *Current,RVect to); + void drawCover(CFImage *CF); + + void udpatePerspective(int width, int height); + + int updateCount; + WidgetLoader * loader; + int fontSize; + + GLuint defaultTexture; + GLuint markTexture; + GLuint readingTexture; + void initializeGL(); + void paintGL(); + void timerEvent(QTimerEvent *); + + //number of Covers + int numObjects; + int lazyPopulateObjects; + bool showMarks; + QVector loaded; + QVector marks; + QList paths; + CFImage * cfImages; + bool hasBeenInitialized; + + Performance performance; + bool bUseVSync; + + /*** Animation Settings ***/ + Preset config; + + //sets/returns the curent selected cover + int currentSelected; + + //defines the position of the centered cover + RVect centerPos; + + /*** Style ***/ + //sets the amount of shading of the covers in the back (0-1) + float shadingTop; + float shadingBottom; + + //sets the reflection strenght (0-1) + float reflectionUp; + float reflectionBottom; + + /*** System info ***/ + float viewRotate; + + //sets the updateInterval in ms + static int updateInterval; + + void startAnimationTimer(); + void stopAnimationTimer(); + +public: + + + /*Constructor*/ + YACReaderFlowGL(QWidget *parent = 0,struct Preset p = pressetYACReaderFlowDownConfig); + virtual ~YACReaderFlowGL(); + + //size; + QSize minimumSizeHint() const; + //QSize sizeHint() const; + + /*functions*/ + + //if called it moves the coverflow to the left + void showPrevious(); + //if called it moves the coverflow to the right + void showNext(); + //go to + void setCurrentIndex(int pos); + //must be called whenever the coverflow animation is stopped + void cleanupAnimation(); + //Draws the coverflow + void draw(); + //updates the coverflow + void updatePositions(); + //inserts a new item to the coverflow + //if item is set to a value > -1 it updates a already set value + //otherwise a new entry is set + void insert(char *name, GLuint Tex, float x, float y,int item = -1); + //removes a item + virtual void remove(int item); + //replaces the texture of the item 'item' with Tex + void replace(char *name, GLuint Tex, float x, float y,int item); + //create n covers with the default nu + void populate(int n); + /*Info*/ + //retuns the CFImage Struct of the current selected item + //to read title or textures + CFImage getCurrentSelected(); + + public slots: + void setCF_RX(int value); + //the Y Rotation of the Coverflow + void setCF_RY(int value); + //the Z Rotation of the Coverflow + void setCF_RZ(int value); + + //perspective + void setZoom(int zoom); + + void setRotation(int angle); + //sets the distance between the covers + void setX_Distance(int distance); + //sets the distance between the centered and the non centered covers + void setCenter_Distance(int distance); + //sets the pushback amount + void setZ_Distance(int distance); + + void setCF_Y(int value); + void setCF_Z(int value); + + void setY_Distance(int value); + + void setFadeOutDist(int value); + + void setLightStrenght(int value); + + void setMaxAngle(int value); + + void setPreset(const Preset & p); + + void setPerformance(Performance performance); + + void useVSync(bool b); + + virtual void updateImageData() = 0; + + void reset(); + void reload(); + + //interface with yacreaderlibrary, compatibility + void setShowMarks(bool value); + void setMarks(QVector marks); + void setMarkImage(QImage & image); + void markSlide(int index, YACReaderComicReadStatus status); + void unmarkSlide(int index); + void setSlideSize(QSize size); + void clear(); + void setCenterIndex(unsigned int index); + void showSlide(int index); + int centerIndex(); + void updateMarks(); + //void setFlowType(PictureFlow::FlowType flowType); + void render(); + + //void paintEvent(QPaintEvent *event); + void mouseDoubleClickEvent(QMouseEvent* event); + void mousePressEvent(QMouseEvent *event); + void wheelEvent(QWheelEvent * event); + void keyPressEvent(QKeyEvent *event); + void resizeGL(int width, int height); + friend class ImageLoaderGL; + friend class ImageLoaderByteArrayGL; + +signals: + void centerIndexChanged(int); + void selected(unsigned int); +}; + +//class WidgetLoader : public QGLWidget +//{ +// Q_OBJECT +//public: +// WidgetLoader(QWidget *parent, QGLWidget * shared); +// YACReaderFlowGL * flow; +//public slots: +// void loadTexture(int index); +// +//}; + +class YACReaderComicFlowGL : public YACReaderFlowGL +{ +public: + YACReaderComicFlowGL(QWidget *parent = 0,struct Preset p = defaultYACReaderFlowConfig); + void setImagePaths(QStringList paths); + void updateImageData(); + void remove(int item); + friend class ImageLoaderGL; +private: + ImageLoaderGL * worker; + +}; + +class YACReaderPageFlowGL : public YACReaderFlowGL +{ +public: + YACReaderPageFlowGL(QWidget *parent = 0,struct Preset p = defaultYACReaderFlowConfig); + ~YACReaderPageFlowGL(); + void updateImageData(); + void populate(int n); + QVector imagesReady; + QVector rawImages; + QVector imagesSetted; + friend class ImageLoaderByteArrayGL; +private: + ImageLoaderByteArrayGL * worker; +}; + +class ImageLoaderGL : public QThread +{ +public: + ImageLoaderGL(YACReaderFlowGL * flow); + ~ImageLoaderGL(); + // returns FALSE if worker is still busy and can't take the task + bool busy() const; + void generate(int index, const QString& fileName); + void reset(){idx = -1;fileName="";}; + int index() const { return idx; }; + void lock(); + void unlock(); + QImage result(); + YACReaderFlowGL * flow; + GLuint resultTexture; + QImage loadImage(const QString& fileName); + +protected: + void run(); + +private: + QMutex mutex; + QWaitCondition condition; + + + bool restart; + bool working; + int idx; + QString fileName; + QSize size; + QImage img; +}; + +class ImageLoaderByteArrayGL : public QThread +{ +public: + ImageLoaderByteArrayGL(YACReaderFlowGL * flow); + ~ImageLoaderByteArrayGL(); + // returns FALSE if worker is still busy and can't take the task + bool busy() const; + void generate(int index, const QByteArray& raw); + void reset(){idx = -1; rawData.clear();}; + int index() const { return idx; }; + QImage result(); + YACReaderFlowGL * flow; + GLuint resultTexture; + QImage loadImage(const QByteArray& rawData); + +protected: + void run(); + +private: + QMutex mutex; + QWaitCondition condition; + + + bool restart; + bool working; + int idx; + QByteArray rawData; + QSize size; + QImage img; +}; + +//class TextureLoader : public QThread +//{ +//public: +// TextureLoader(); +// ~TextureLoader(); +// // returns FALSE if worker is still busy and can't take the task +// +// YACReaderFlow * flow; +// ImageLoader * worker; +//protected: +// void run(); +// +//}; + +#endif diff --git a/common/yacreader_global.cpp b/common/yacreader_global.cpp new file mode 100644 index 00000000..c8ad06c3 --- /dev/null +++ b/common/yacreader_global.cpp @@ -0,0 +1,29 @@ +#include "yacreader_global.h" +#include + +using namespace YACReader; + +QString YACReader::getSettingsPath() +{ +#if QT_VERSION >= 0x050000 + return QStandardPaths::writableLocation(QStandardPaths::DataLocation); +#else + return QDesktopServices::storageLocation(QDesktopServices::DataLocation); +#endif + +} + +void YACReader::addSperator(QWidget *w) +{ + QAction * separator = new QAction(w); + separator->setSeparator(true); + w->addAction(separator); +} + + +QAction * YACReader::createSeparator() +{ + QAction * a = new QAction(0); + a->setSeparator(true); + return a; +} diff --git a/common/yacreader_global.h b/common/yacreader_global.h new file mode 100644 index 00000000..09e9d95f --- /dev/null +++ b/common/yacreader_global.h @@ -0,0 +1,109 @@ +#ifndef __YACREADER_GLOBAL_H +#define __YACREADER_GLOBAL_H + +#if QT_VERSION >= 0x050000 + #include +#else + #include +#endif + +#include + +#define VERSION "7.1.0" + +#define PATH "PATH" +#define MAG_GLASS_SIZE "MAG_GLASS_SIZE" +#define ZOOM_LEVEL "ZOOM_LEVEL" +#define SLIDE_SIZE "SLIDE_SIZE" +#define GO_TO_FLOW_SIZE "GO_TO_FLOW_SIZE" +#define FLOW_TYPE_SW "FLOW_TYPE_SW" +#define FIT "FIT" +#define FLOW_TYPE "FLOW_TYPE" +#define FULLSCREEN "FULLSCREEN" +#define FIT_TO_WIDTH_RATIO "FIT_TO_WIDTH_RATIO" +#define Y_WINDOW_POS "POS" +#define Y_WINDOW_SIZE "SIZE" +#define MAXIMIZED "MAXIMIZED" +#define DOUBLE_PAGE "DOUBLE_PAGE" +#define ADJUST_TO_FULL_SIZE "ADJUST_TO_FULL_SIZE" +#define BACKGROUND_COLOR "BACKGROUND_COLOR" +#define ALWAYS_ON_TOP "ALWAYS_ON_TOP" +#define SHOW_TOOLBARS "SHOW_TOOLBARS" +#define BRIGHTNESS "BRIGHTNESS" +#define CONTRAST "CONTRAST" +#define GAMMA "GAMMA" +#define SHOW_INFO "SHOW_INFO" + +#define FLOW_TYPE_GL "FLOW_TYPE_GL" +#define Y_POSITION "Y_POSITION" +#define COVER_DISTANCE "COVER_DISTANCE" +#define CENTRAL_DISTANCE "CENTRAL_DISTANCE" +#define ZOOM_LEVEL "ZOOM_LEVEL" +#define Z_COVER_OFFSET "Z_COVER_OFFSET" +#define COVER_ROTATION "COVER_ROTATION" +#define FADE_OUT_DIST "FADE_OUT_DIST" +#define LIGHT_STRENGTH "LIGHT_STRENGTH" +#define MAX_ANGLE "MAX_ANGLE" +#define PERFORMANCE "PERFORMANCE" +#define USE_OPEN_GL "USE_OPEN_GL" +#define X_ROTATION "X_ROTATION" +#define Y_COVER_OFFSET "Y_COVER_OFFSET" +#define V_SYNC "V_SYNC" +#define SERVER_ON "SERVER_ON" + +#define MAIN_WINDOW_GEOMETRY "MAIN_WINDOW_GEOMETRY" +#define MAIN_WINDOW_STATE "MAIN_WINDOW_STATE" +#define COMICS_VIEW_HEADERS "COMICS_VIEW_HEADERS" +#define COMICS_VIEW_HEADERS_GEOMETRY "COMICS_VIEW_HEADERS_GEOMETRY" +#define COMICS_VIEW_STATUS "COMICS_VIEW_STATUS" + +#define NUM_DAYS_BETWEEN_VERSION_CHECKS "NUM_DAYS_BETWEEN_VERSION_CHECKS" +#define LAST_VERSION_CHECK "LAST_VERSION_CHECK" + +#define YACREADERLIBRARY_GUID "ea343ff3-2005-4865-b212-7fa7c43999b8" + +#define LIBRARIES "LIBRARIES" + +namespace YACReader +{ + + enum FlowType + { + CoverFlowLike=0, + Strip, + StripOverlapped, + Modern, + Roulette, + Custom + }; + + enum YACReaderIPCMessages + { + RequestComicInfo = 0, + SendComicInfo, + }; + + enum YACReaderComicReadStatus + { + Unread = 0, + Read = 1, + Opened = 2 + }; + + enum YACReaderErrors + { + SevenZNotFound = 700 + }; + + enum ComicsViewStatus + { + Flow, + Grid + }; + +QString getSettingsPath(); +void addSperator(QWidget * w); +QAction * createSeparator(); +} +#endif + diff --git a/compileOSX.sh b/compileOSX.sh new file mode 100755 index 00000000..ec1974d7 --- /dev/null +++ b/compileOSX.sh @@ -0,0 +1,47 @@ +#! /bin/bash +if [ $2 == "clean" ]; then +./cleanOSX.sh +fi + +echo "Compiling YACReader" +cd ./YACReader +#/Developer/Qt5.1.1/5.1.1/clang_64/bin/qmake -spec macx-clang "CONFIG+=release" +qmake -spec macx-g++ "CONFIG+=release" +make +cd .. + +echo "Compiling YACReaderLibrary" +cd ./YACReaderLibrary +#/Developer/Qt5.1.1/5.1.1/clang_64/bin/qmake -spec macx-clang "CONFIG+=release" +qmake -spec macx-g++ "CONFIG+=release" +make +cd .. + +echo "Configuring release apps" + +cp -R ./YACReader/YACReader.app ./YACReader.app +cp -R ./YACReaderLibrary/YACReaderLibrary.app ./YACReaderLibrary.app + +./releaseOSX.sh + +#cp -R ./PlugInsYACReader ./YACReader.app/Contents/PlugIns +#cp -R ./PlugInsLibrary ./YACReaderLibrary.app/Contents/PlugIns + +echo "Copying to destination folder" +dest='YACReader-'$1' MacOSX-Intel' +mkdir "$dest" +cp -R ./YACReader.app "./${dest}/YACReader.app" +cp -R ./YACReaderLibrary.app "./${dest}/YACReaderLibrary.app" +cp ./COPYING.txt "./${dest}/" +cp ./README.txt "./${dest}/" + +mkdir "./${dest}/icons/" +cp ./images/db.png "./${dest}/icons/" +cp ./images/coversPackage.png "./${dest}/icons/" + +echo "Creating dmg package" +#tar -czf "${dest}".tar.gz "${dest}" +#hdiutil create "${dest}".dmg -srcfolder "./${dest}" -ov +./create-dmg --volname 'YACReader '$1' Installer' --volicon icon.icns --window-size 600 403 --icon-size 128 --app-drop-link 485 90 --background background.png --icon YACReader 80 90 --icon YACReaderLibrary 235 90 --eula COPYING.txt --icon icons 470 295 --icon README.txt 120 295 --icon COPYING.txt 290 295 "./${dest}.dmg" "./${dest}" + +echo "Done!" diff --git a/compileX11.sh b/compileX11.sh new file mode 100755 index 00000000..fe10596e --- /dev/null +++ b/compileX11.sh @@ -0,0 +1,43 @@ +#! /bin/bash + +cd ./compressed_archive +if [ ! -d "libp7zip" ]; then +echo "You need p7zip source code to compile YACReader. \ +Please check the compressed_archive folder for further instructions." +exit +fi +patch -p0 -i libp7zip.patch +cd .. + +echo "Compiling YACReader" +cd ./YACReader +qmake "CONFIG+=release" +make +cd .. + +echo "Compiling YACReaderLibrary" +cd ./YACReaderLibrary +qmake "CONFIG+=release" +make +cd .. + +echo "Copying to destination folder" +dest='YACReader-'$1'-X11-'$2'-qt5' +mkdir "$dest" +cp ./YACReader/YACReader "./${dest}/YACReader" +cp ./YACReaderLibrary/YACReaderLibrary "./${dest}/YACReaderLibrary" +cp ./COPYING.txt "./${dest}/" +cp ./README.txt "./${dest}/" +cp ./INSTALL.txt "./${dest}/" +cp ./images/icon.png "./${dest}/" +cp ./images/iconLibrary.png "./${dest}/" +cp ./images/db.png "./${dest}/" +cp ./images/coversPackage.png "./${dest}/" +cp -R ./utils "./${dest}/" +cp -R ./release/server "./${dest}/" +cp -R ./release/languages "./${dest}/" + +echo "Creating package" +tar -czf "${dest}".tar.gz "${dest}" + +echo "Done!" diff --git a/compressed_archive/7z_includes.h b/compressed_archive/7z_includes.h new file mode 100644 index 00000000..d80e8dc1 --- /dev/null +++ b/compressed_archive/7z_includes.h @@ -0,0 +1,65 @@ +#ifndef _7Z_INCLUDES_H +#define _7Z_INCLUDES_H + +//WIN includes +#ifdef Q_OS_WIN +#include "lib7zip/CPP/Common/StringConvert.h" +#include "lib7zip/CPP/Common/MyInitGuid.h" +#include "lib7zip/CPP/Common/MyCom.h" +#include "lib7zip/CPP/7zip/Common/FileStreams.h" +#include "lib7zip/CPP/7zip/Archive/IArchive.h" + +#include "lib7zip/CPP/7zip/IStream.h" + +#include "lib7zip/CPP/7zip/IPassword.h" +#include "lib7zip/CPP/7zip/MyVersion.h" + +#include "lib7zip/C/Types.h" + +#include "lib7zip/CPP/Windows/PropVariant.h" +#include "lib7zip/CPP/Windows/PropVariantConversions.h" + +#include "lib7zip/CPP/7zip/Common/StreamObjects.h" +#include "lib7zip/CPP/7zip/Common/StreamUtils.h" + +extern "C" +{ +#include "lib7zip/C/Alloc.h" +} +#else +//POSIX includes +#include "libp7zip/CPP/myWindows/myPrivate.h" +#include "libp7zip/CPP/myWindows/config.h" + +#include "libp7zip/CPP/Common/MyGuidDef.h" +#include "libp7zip/CPP/Common/MyWindows.h" + +#include "libp7zip/CPP/Common/StringConvert.h" +#include "libp7zip/CPP/Common/MyInitGuid.h" +#include "libp7zip/CPP/Common/MyCom.h" +#include "libp7zip/CPP/7zip/Common/FileStreams.h" +#include "libp7zip/CPP/7zip/Archive/IArchive.h" + +#include "libp7zip/CPP/7zip/IStream.h" + +#include "libp7zip/CPP/7zip/IPassword.h" +#include "libp7zip/CPP/7zip/MyVersion.h" + +#include "libp7zip/C/Types.h" + +#include "libp7zip/CPP/Windows/Defs.h" +#include "libp7zip/CPP/Windows/PropVariant.h" +#include "libp7zip/CPP/Windows/PropVariantConversions.h" + +#include "libp7zip/CPP/7zip/Common/StreamObjects.h" +#include "libp7zip/CPP/7zip/Common/StreamUtils.h" + +#include "libp7zip/CPP/7zip/ICoder.h" + +extern "C" +{ +#include "libp7zip/C/Alloc.h" +} +#endif + +#endif // _7Z_INCLUDES_H diff --git a/compressed_archive/README_7zip.txt b/compressed_archive/README_7zip.txt new file mode 100644 index 00000000..3e4a5864 --- /dev/null +++ b/compressed_archive/README_7zip.txt @@ -0,0 +1,7 @@ +If you are trying to compile YACReader, you need to donwload de source code of 7zip (Windows) or p7zip (Linux/MacOSX). + +Please, extract it and rename the folder to lib7zip (Windows) or libp7zip (Linux/MacOSX), then copy it to $YACREADER_SRC/compressed_archive/ (this +folder). If you are using a 64 bit Linux-System, please apply libp7zip.patch for successfull compilation or use compileX11.sh. + +YACReader is compiled using 7zip/p7zip 9.20.1 + diff --git a/compressed_archive/StdAfx.h b/compressed_archive/StdAfx.h new file mode 100644 index 00000000..2edddf4a --- /dev/null +++ b/compressed_archive/StdAfx.h @@ -0,0 +1,9 @@ +// StdAfx.h + +#ifndef __STDAFX_H +#define __STDAFX_H + +#include +#include + +#endif diff --git a/compressed_archive/StdAfx.h.cpp b/compressed_archive/StdAfx.h.cpp new file mode 100644 index 00000000..b86703cb --- /dev/null +++ b/compressed_archive/StdAfx.h.cpp @@ -0,0 +1,10 @@ +/*-------------------------------------------------------------------- +* +* Due to issues with the dependencies checker within the IDE, it +* sometimes fails to recompile the PCH file, if we force the IDE to +* This file is auto-generated by qmake since no PRECOMPILED_SOURCE was +* specified, and is used as the common stdafx.cpp. The file is only +* command line compilations by nmake. +* +--------------------------------------------------------------------*/ +#include "StdAfx.h" diff --git a/compressed_archive/compressed_archive.cpp b/compressed_archive/compressed_archive.cpp new file mode 100644 index 00000000..77fc9b1e --- /dev/null +++ b/compressed_archive/compressed_archive.cpp @@ -0,0 +1,371 @@ +#include "compressed_archive.h" +#include "extract_delegate.h" + +#include +#include +#include + +#include "open_callbacks.h" +#include "extract_callbacks.h" + +#include "yacreader_global.h" + +//DEFINE_GUID(CLSID_CFormat7z,0x23170F69, 0x40C1, 0x278A, 0x10, 0x00, 0x00, 0x01, 0x10, 0x07, 0x00, 0x00); +//DEFINE_GUID(IArchiveKK,0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x06, 0x00, 0x60, 0x00, 0x00); + +DEFINE_GUID(CLSID_CFormat7z, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x07, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatRar, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x03, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatZip, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x01, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatTar, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xee, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatArj, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x04, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatBZip2, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x02, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatCab, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x08, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatChm, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xe9, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatCompound,0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xe5, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatCpio, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xed, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatDeb, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xec, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatGZip, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xef, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatIso, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xe7, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatLzh, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x06, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatLzma, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x0a, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatNsis, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x09, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatRpm, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xeb, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatSplit, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xea, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatWim, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0xe6, 0x00, 0x00); +DEFINE_GUID(CLSID_CFormatZ, 0x23170f69, 0x40c1, 0x278a, 0x10, 0x00, 0x00, 0x01, 0x10, 0x05, 0x00, 0x00); + +#ifdef Q_OS_WIN +GUID _supportedFileFormats[] = {CLSID_CFormatRar,CLSID_CFormatZip,CLSID_CFormatTar,CLSID_CFormat7z,CLSID_CFormatArj}; +#else +GUID _supportedFileFormats[] = {CLSID_CFormatZip,CLSID_CFormatTar,CLSID_CFormat7z,CLSID_CFormatArj}; +#endif +std::vector supportedFileFormats (_supportedFileFormats, _supportedFileFormats + sizeof(_supportedFileFormats) / sizeof(_supportedFileFormats[0]) ); + +DEFINE_GUID(IID_InArchive, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x06, 0x00, 0x60, 0x00, 0x00); +DEFINE_GUID(IID_ISetCompressCodecsInfo, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x04, 0x00, 0x61, 0x00, 0x00); + +#ifdef Q_OS_UNIX +DEFINE_GUID(IID_IOutStream, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x03, 0x00, 0x04, 0x00, 0x00); +DEFINE_GUID(IID_IInStream, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x03, 0x00, 0x03, 0x00, 0x00); +DEFINE_GUID(IID_IStreamGetSize, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x03, 0x00, 0x06, 0x00, 0x00); +DEFINE_GUID(IID_ISequentialInStream, 0x23170F69, 0x40C1, 0x278A, 0x00, 0x00, 0x00, 0x03, 0x00, 0x01, 0x00, 0x00); +#endif + +struct SevenZipInterface { + CreateObjectFunc createObjectFunc; + GetMethodPropertyFunc getMethodPropertyFunc; + GetNumberOfMethodsFunc getNumberOfMethodsFunc; + GetNumberOfFormatsFunc getNumberOfFormatsFunc; + GetHandlerPropertyFunc getHandlerPropertyFunc; + GetHandlerPropertyFunc2 getHandlerPropertyFunc2; + SetLargePageModeFunc setLargePageModeFunc; + +#ifdef Q_OS_UNIX + CreateObjectFunc createObjectFuncRar; + GetMethodPropertyFunc getMethodPropertyFuncRar; + GetNumberOfMethodsFunc getNumberOfMethodsFuncRar; +#endif + + CMyComPtr archive; +}; + +//SevenZipInterface * szInterface; + +CompressedArchive::CompressedArchive(const QString & filePath, QObject *parent) : + QObject(parent),sevenzLib(0),valid(false),tools(false) +#ifdef Q_OS_UNIX + ,isRar(false) +#endif +{ + szInterface = new SevenZipInterface; + //load functions + if(!loadFunctions()) + return; + + tools = true; + //load file + if(szInterface->createObjectFunc != 0) + { + //QUuid CLSID_CFormat7z("23170f69-40c1-278a-1000-000110070000"); + //se crea el objeto Archivo: formato,tipo,objeto + bool formatFound = false; + CInFileStream *fileSpec = new CInFileStream; + CMyComPtr file = fileSpec; + + CArchiveOpenCallback *openCallbackSpec = new CArchiveOpenCallback; + CMyComPtr openCallback = openCallbackSpec; + openCallbackSpec->PasswordIsDefined = false; + // openCallbackSpec->PasswordIsDefined = true; + // openCallbackSpec->Password = L"1"; + + for(unsigned int i=0;icreateObjectFunc(&supportedFileFormats[i], &IID_InArchive, (void **)&szInterface->archive) != S_OK) + { + qDebug() << "wrong format"; + continue; + } +#ifdef UNICODE + if (!fileSpec->Open((LPCTSTR)filePath.toStdWString().c_str())) +#else + if (!fileSpec->Open((LPCTSTR)filePath.toStdString().c_str())) +#endif + { + qDebug() << "unable to load" + filePath; + continue; + } + //qDebug() << "Can not open archive file : " + filePath << endl; + + if (szInterface->archive->Open(file, 0, openCallback) == S_OK) + { + valid = formatFound = true; + break; + } + else + qDebug() << "Can not open archive file : " + filePath << endl; + } + if(!formatFound) + { +#ifdef Q_OS_WIN + qDebug() << "Can not open archive" << endl; +#else + if (szInterface->createObjectFunc(&CLSID_CFormatRar, &IID_InArchive, (void **)&szInterface->archive) != S_OK) + { + qDebug() << "Error creating rar archive :" + filePath; + return; + } + + CMyComPtr codecsInfo; + if (szInterface->archive->QueryInterface(IID_ISetCompressCodecsInfo,(void **)&codecsInfo) != S_OK) + { + qDebug() << "Error getting rar codec :" + filePath; + return; + } + + if (codecsInfo->SetCompressCodecsInfo(this) != S_OK) + { + qDebug() << "Error setting rar codec"; + return; + } + +#ifdef UNICODE + if (!fileSpec->Open((LPCTSTR)filePath.toStdWString().data())) +#else + if (!fileSpec->Open((LPCTSTR)filePath.toStdString().data())) +#endif + { + qDebug() << "Error opening rar file :" + filePath; + return; + } + //qDebug() << "Can not open archive file : " + filePath << endl; + + if (szInterface->archive->Open(file, 0, openCallback) == S_OK) + { + valid = formatFound = true; + isRar = true; + } + else + qDebug() << "Error opening rar archive"; + + +#endif + } + } +} + +CompressedArchive::~CompressedArchive() +{ +#ifdef Q_OS_UNIX + if(isRar) //TODO: fix this!!! Possible memory leak. If AddRef is not used, a crash occurs in "delete szInterface" + szInterface->archive->AddRef(); +#endif + if(valid) //TODO: fix this!!! Memory leak. + delete szInterface; +#ifdef Q_OS_UNIX + delete rarLib; +#endif + delete sevenzLib; +} + +bool CompressedArchive::loadFunctions() +{ + //LOAD library + //TODO check if this works in OSX (7z.so instead of 7z.dylib) + // fix1: try to load "7z.so" + // fix2: rename 7z.so to 7z.dylib + if(sevenzLib == 0) + { +#if defined Q_OS_UNIX + #if defined Q_OS_MAC + rarLib = new QLibrary(QApplication::applicationDirPath()+"/utils/Codecs/Rar29"); + #else + rarLib = new QLibrary(QString(LIBDIR)+"/p7zip/Codecs/Rar29.so"); + #endif + if(!rarLib->load()) + { + qDebug() << "Error Loading Rar29.so : " + rarLib->errorString() << endl; + QApplication::exit(YACReader::SevenZNotFound); + return false; + } +#endif +#if defined Q_OS_UNIX && !defined Q_OS_MAC + sevenzLib = new QLibrary(QString(LIBDIR)+"/p7zip/7z.so"); +#else + sevenzLib = new QLibrary(QApplication::applicationDirPath()+"/utils/7z"); +#endif + } + if(!sevenzLib->load()) + { + qDebug() << "Error Loading 7z.dll : " + sevenzLib->errorString() << endl; + QApplication::exit(YACReader::SevenZNotFound); + return false; + } + else + { + qDebug() << "Loading functions" << endl; + + if((szInterface->createObjectFunc = (CreateObjectFunc)sevenzLib->resolve("CreateObject")) == 0) + qDebug() << "fail loading function : CreateObject" << endl; + if((szInterface->getMethodPropertyFunc = (GetMethodPropertyFunc)sevenzLib->resolve("GetMethodProperty")) == 0) + qDebug() << "fail loading function : GetMethodProperty" << endl; + if((szInterface->getNumberOfMethodsFunc = (GetNumberOfMethodsFunc)sevenzLib->resolve("GetNumberOfMethods")) == 0) + qDebug() << "fail loading function : GetNumberOfMethods" << endl; + if((szInterface->getNumberOfFormatsFunc = (GetNumberOfFormatsFunc)sevenzLib->resolve("GetNumberOfFormats")) == 0) + qDebug() << "fail loading function : GetNumberOfFormats" << endl; + if((szInterface->getHandlerPropertyFunc = (GetHandlerPropertyFunc)sevenzLib->resolve("GetHandlerProperty")) == 0) + qDebug() << "fail loading function : GetHandlerProperty" << endl; + if((szInterface->getHandlerPropertyFunc2 = (GetHandlerPropertyFunc2)sevenzLib->resolve("GetHandlerProperty2")) == 0) + qDebug() << "fail loading function : GetHandlerProperty2" << endl; + if((szInterface->setLargePageModeFunc = (SetLargePageModeFunc)sevenzLib->resolve("SetLargePageMode")) == 0) + qDebug() << "fail loading function : SetLargePageMode" << endl; + +#ifdef Q_OS_UNIX + if((szInterface->createObjectFuncRar = (CreateObjectFunc)rarLib->resolve("CreateObject")) == 0) + qDebug() << "fail loading function (rar) : CreateObject" << endl; + if((szInterface->getMethodPropertyFuncRar = (GetMethodPropertyFunc)rarLib->resolve("GetMethodProperty")) == 0) + qDebug() << "fail loading function (rar) : GetMethodProperty" << endl; + if((szInterface->getNumberOfMethodsFuncRar = (GetNumberOfMethodsFunc)rarLib->resolve("GetNumberOfMethods")) == 0) + qDebug() << "fail loading function (rar) : GetNumberOfMethods" << endl; +#endif + } + + return true; +} + +QList CompressedArchive::getFileNames() +{ + QList files; + quint32 numItems = getNumFiles(); + for (quint32 i = 0; i < numItems; i++) + { + { + // Get name of file + NWindows::NCOM::CPropVariant prop; + /*szInterface->archive->GetProperty(i, kpidIsDir, &prop); + bool isDir; + if (prop.vt == VT_BOOL) + isDir = VARIANT_BOOLToBool(prop.boolVal); + else if (prop.vt == VT_EMPTY) + isDir = false; + + if(!isDir) + {*/ + szInterface->archive->GetProperty(i, kpidPath, &prop); + UString s = ConvertPropVariantToString(prop); + const wchar_t * chars = s.operator const wchar_t *(); + files.append(QString::fromWCharArray(chars)); + //} + } + } + return files; +} + +bool CompressedArchive::isValid() +{ + return valid; +} + +bool CompressedArchive::toolsLoaded() +{ + return tools; +} + +int CompressedArchive::getNumFiles() +{ + quint32 numItems = 0; + szInterface->archive->GetNumberOfItems(&numItems); + return numItems; +} + +QList CompressedArchive::getAllData(const QVector & indexes, ExtractDelegate * delegate) +{ + CArchiveExtractCallback *extractCallbackSpec = new CArchiveExtractCallback(true,delegate); + CMyComPtr extractCallback(extractCallbackSpec); + extractCallbackSpec->Init(szInterface->archive, L""); // second parameter is output folder path + extractCallbackSpec->PasswordIsDefined = false; + + HRESULT result; + if(indexes.isEmpty()) + result = szInterface->archive->Extract(NULL, -1, false, extractCallback); + else + result = szInterface->archive->Extract(indexes.data(), indexes.count(), false, extractCallback); + if (result != S_OK) + { + qDebug() << "Extract Error" << endl; + } + + return extractCallbackSpec->allFiles; +} + +QByteArray CompressedArchive::getRawDataAtIndex(int index) +{ + if(index>=0 && index < getNumFiles()) + { + CArchiveExtractCallback *extractCallbackSpec = new CArchiveExtractCallback; + CMyComPtr extractCallback(extractCallbackSpec); + extractCallbackSpec->Init(szInterface->archive, L""); // second parameter is output folder path + extractCallbackSpec->PasswordIsDefined = false; + + UInt32 indices[1]; + indices[0] = index; + HRESULT result = szInterface->archive->Extract(indices, 1, false, extractCallback); + if (result != S_OK) + { + qDebug() << "Extract Error" << endl; + } + + return QByteArray((char *)extractCallbackSpec->data,extractCallbackSpec->newFileSize); + } + return QByteArray(); +} + +#ifdef Q_OS_UNIX + +STDMETHODIMP CompressedArchive::GetNumberOfMethods(UInt32 *numMethods) +{ + return szInterface->getNumberOfMethodsFuncRar(numMethods); +} + +STDMETHODIMP CompressedArchive::GetProperty(UInt32 index, PROPID propID, PROPVARIANT *value) +{ + return szInterface->getMethodPropertyFuncRar(index,propID,value); +} + +int i = 0; +STDMETHODIMP CompressedArchive::CreateDecoder(UInt32 index, const GUID *interfaceID, void **coder) +{ + NCOM::CPropVariant propVariant; + szInterface->getMethodPropertyFuncRar(index,NMethodPropID::kDecoder,&propVariant); + return szInterface->createObjectFuncRar((const GUID *)propVariant.bstrVal,interfaceID,coder); +} + +STDMETHODIMP CompressedArchive::CreateEncoder(UInt32 index, const GUID *interfaceID, void **coder) +{ + return S_OK;//szInterface->createObjectFuncRar(&CLSID_CFormatRar,interfaceID,coder); +} + +#endif + + diff --git a/compressed_archive/compressed_archive.h b/compressed_archive/compressed_archive.h new file mode 100644 index 00000000..21d24e46 --- /dev/null +++ b/compressed_archive/compressed_archive.h @@ -0,0 +1,80 @@ +#ifndef COMPRESSED_ARCHIVE_H +#define COMPRESSED_ARCHIVE_H + +#include + +#ifdef Q_OS_UNIX + #include "libp7zip/CPP/7zip/ICoder.h" + #include "libp7zip/CPP/Common/MyCom.h" +#endif + +class ExtractDelegate; + +#ifdef Q_OS_WIN + #include "7z_includes.h" + #define _MY_WINAPI WINAPI +#else + #define _MY_WINAPI +#endif + +typedef quint32 (_MY_WINAPI * CreateObjectFunc)(const GUID *clsID,const GUID *interfaceID,void **outObject); +typedef quint32 (_MY_WINAPI *GetMethodPropertyFunc)(quint32 index, PROPID propID, PROPVARIANT *value); +typedef quint32 (_MY_WINAPI *GetNumberOfMethodsFunc)(quint32 *numMethods); +typedef quint32 (_MY_WINAPI *GetNumberOfFormatsFunc)(quint32 *numFormats); +typedef quint32 (_MY_WINAPI *GetHandlerPropertyFunc)(PROPID propID, PROPVARIANT *value); +typedef quint32 (_MY_WINAPI *GetHandlerPropertyFunc2)(quint32 index, PROPID propID, PROPVARIANT *value); +typedef quint32 (_MY_WINAPI *SetLargePageModeFunc)(); + +class QLibrary; +#include +#include + +struct SevenZipInterface; + +class MyCodecs; + +#ifdef Q_OS_UNIX + class CompressedArchive : public QObject, public ICompressCodecsInfo, public CMyUnknownImp +#else + class CompressedArchive : public QObject +#endif +{ + Q_OBJECT +public: + explicit CompressedArchive(const QString & filePath, QObject *parent = 0); + ~CompressedArchive(); + +#ifdef Q_OS_UNIX + MY_UNKNOWN_IMP + + STDMETHOD(GetNumberOfMethods)(UInt32 *numMethods); + STDMETHOD(GetProperty)(UInt32 index, PROPID propID, PROPVARIANT *value); + STDMETHOD(CreateDecoder)(UInt32 index, const GUID *iid, void **coder); + STDMETHOD(CreateEncoder)(UInt32 index, const GUID *iid, void **coder); + + bool isRar; +#endif + +signals: + +public slots: + int getNumFiles(); + QList getAllData(const QVector & indexes, ExtractDelegate * delegate = 0); + QByteArray getRawDataAtIndex(int index); + QList getFileNames(); + bool isValid(); + bool toolsLoaded(); +private: + SevenZipInterface * szInterface; + QLibrary * sevenzLib; +#ifdef Q_OS_UNIX + QLibrary * rarLib; +#endif + bool loadFunctions(); + bool tools; + bool valid; + + friend class MyCodecs; +}; + +#endif // COMPRESSED_ARCHIVE_H diff --git a/compressed_archive/extract_callbacks.h b/compressed_archive/extract_callbacks.h new file mode 100644 index 00000000..998b8734 --- /dev/null +++ b/compressed_archive/extract_callbacks.h @@ -0,0 +1,328 @@ +#ifndef EXTRACT_CALLBACKS_H +#define EXTRACT_CALLBACKS_H + +#include "7z_includes.h" +#include "extract_delegate.h" +#include + +using namespace NWindows; + +////////////////////////////////////////////////////////////// +// Archive Extracting callback class + +static const wchar_t *kCantDeleteOutputFile = L"ERROR: Can not delete output file "; + +static const char *kTestingString = "Testing "; +static const char *kExtractingString = "Extracting "; +static const char *kSkippingString = "Skipping "; + +static const char *kUnsupportedMethod = "Unsupported Method"; +static const char *kCRCFailed = "CRC Failed"; +static const char *kDataError = "Data Error"; +static const char *kUnknownError = "Unknown Error"; + +static const wchar_t *kEmptyFileAlias = L"[Content]"; + +static HRESULT IsArchiveItemProp(IInArchive *archive, UInt32 index, PROPID propID, bool &result) +{ + NCOM::CPropVariant prop; + RINOK(archive->GetProperty(index, propID, &prop)); + if (prop.vt == VT_BOOL) + result = VARIANT_BOOLToBool(prop.boolVal); + else if (prop.vt == VT_EMPTY) + result = false; + else + return E_FAIL; + return S_OK; +} +static HRESULT IsArchiveItemFolder(IInArchive *archive, UInt32 index, bool &result) +{ + return IsArchiveItemProp(archive, index, kpidIsDir, result); +} + +class CArchiveExtractCallback: + public IArchiveExtractCallback, + public ICryptoGetTextPassword, + public CMyUnknownImp +{ +public: + MY_UNKNOWN_IMP1(ICryptoGetTextPassword) + + // IProgress + STDMETHOD(SetTotal)(UInt64 size); + STDMETHOD(SetCompleted)(const UInt64 *completeValue); + + // IArchiveExtractCallback + STDMETHOD(GetStream)(UInt32 index, ISequentialOutStream **outStream, Int32 askExtractMode); + STDMETHOD(PrepareOperation)(Int32 askExtractMode); + STDMETHOD(SetOperationResult)(Int32 resultEOperationResult); + + // ICryptoGetTextPassword + STDMETHOD(CryptoGetTextPassword)(BSTR *aPassword); + +private: + CMyComPtr _archiveHandler; + UString _directoryPath; // Output directory + UString _filePath; // name inside arcvhive + UString _diskFilePath; // full path to file on disk + bool _extractMode; + bool all; + ExtractDelegate * delegate; + UInt32 _index; + struct CProcessedFileInfo + { + FILETIME MTime; + UInt32 Attrib; + bool isDir; + bool AttribDefined; + bool MTimeDefined; + } _processedFileInfo; + + COutFileStream *_outFileStreamSpec; + CMyComPtr _outFileStream; + +public: + void Init(IInArchive *archiveHandler, const UString &directoryPath); + + UInt64 NumErrors; + bool PasswordIsDefined; + QList allFiles; + UString Password; + Byte * data; + UInt64 newFileSize; + + CArchiveExtractCallback(bool c = false,ExtractDelegate * d = 0) : PasswordIsDefined(false),all(c),delegate(d) {} + ~CArchiveExtractCallback() {MidFree(data);} +}; + +void CArchiveExtractCallback::Init(IInArchive *archiveHandler, const UString &directoryPath) +{ + NumErrors = 0; + _archiveHandler = archiveHandler; + directoryPath;//unused +} + +STDMETHODIMP CArchiveExtractCallback::SetTotal(UInt64 /* size */) +{ + return S_OK; +} + +STDMETHODIMP CArchiveExtractCallback::SetCompleted(const UInt64 * /* completeValue */) +{ + return S_OK; +} + +STDMETHODIMP CArchiveExtractCallback::GetStream(UInt32 index, + ISequentialOutStream **outStream, Int32 askExtractMode) +{ + *outStream = 0; + _outFileStream.Release(); + _index = index; + + { + // Get Name + NCOM::CPropVariant prop; + RINOK(_archiveHandler->GetProperty(index, kpidPath, &prop)); + + UString fullPath; + if (prop.vt == VT_EMPTY) + fullPath = kEmptyFileAlias; + else + { + if (prop.vt != VT_BSTR) + return E_FAIL; + fullPath = prop.bstrVal; + } + _filePath = fullPath; + } + + askExtractMode;//unused + //if (askExtractMode != NArchive::NExtract::NAskMode::kExtract) + //return S_OK; + + { + // Get Attrib + NCOM::CPropVariant prop; + RINOK(_archiveHandler->GetProperty(index, kpidAttrib, &prop)); + if (prop.vt == VT_EMPTY) + { + _processedFileInfo.Attrib = 0; + _processedFileInfo.AttribDefined = false; + } + else + { + if (prop.vt != VT_UI4) + return E_FAIL; + _processedFileInfo.Attrib = prop.ulVal; + _processedFileInfo.AttribDefined = true; + } + } + + RINOK(IsArchiveItemFolder(_archiveHandler, index, _processedFileInfo.isDir)); + + { + // Get Modified Time + NCOM::CPropVariant prop; + RINOK(_archiveHandler->GetProperty(index, kpidMTime, &prop)); + _processedFileInfo.MTimeDefined = false; + switch(prop.vt) + { + case VT_EMPTY: + // _processedFileInfo.MTime = _utcMTimeDefault; + break; + case VT_FILETIME: + _processedFileInfo.MTime = prop.filetime; + _processedFileInfo.MTimeDefined = true; + break; + default: + return E_FAIL; + } + + } + + //se necesita conocer el tamaño del archivo para poder reservar suficiente memoria + bool newFileSizeDefined; + { + // Get Size + NCOM::CPropVariant prop; + RINOK(_archiveHandler->GetProperty(index, kpidSize, &prop)); + newFileSizeDefined = (prop.vt != VT_EMPTY); + if (newFileSizeDefined) + newFileSize = ConvertPropVariantToUInt64(prop); + } + + //No hay que crear ningún fichero, ni directorios intermedios + /*{ + // Create folders for file + int slashPos = _filePath.ReverseFind(WCHAR_PATH_SEPARATOR); + if (slashPos >= 0) + NFile::NDirectory::CreateComplexDirectory(_directoryPath + _filePath.Left(slashPos)); + } + + UString fullProcessedPath = _directoryPath + _filePath; + _diskFilePath = fullProcessedPath; + */ + if (_processedFileInfo.isDir) + { + //NFile::NDirectory::CreateComplexDirectory(fullProcessedPath); + } + else + { + /*NFile::NFind::CFileInfoW fi; + if (fi.Find(fullProcessedPath)) + { + if (!NFile::NDirectory::DeleteFileAlways(fullProcessedPath)) + { + qDebug() <<(UString(kCantDeleteOutputFile) + fullProcessedPath); + return E_ABORT; + } + }*/ + if(newFileSizeDefined) + { + CBufPtrSeqOutStream *outStreamSpec = new CBufPtrSeqOutStream; + CMyComPtr outStreamLocal(outStreamSpec); + data = (Byte *)MidAlloc(newFileSize); + outStreamSpec->Init(data, newFileSize); + *outStream = outStreamLocal.Detach(); + } + else + { + + } + + } + return S_OK; +} + +STDMETHODIMP CArchiveExtractCallback::PrepareOperation(Int32 askExtractMode) +{ + _extractMode = false; + switch (askExtractMode) + { + case NArchive::NExtract::NAskMode::kExtract: _extractMode = true; break; + }; + /* switch (askExtractMode) + { + case NArchive::NExtract::NAskMode::kExtract: qDebug() << (kExtractingString); break; + case NArchive::NExtract::NAskMode::kTest: qDebug() <<(kTestingString); break; + case NArchive::NExtract::NAskMode::kSkip: qDebug() <<(kSkippingString); break; + };*/ + //qDebug() << _filePath; + return S_OK; +} + +STDMETHODIMP CArchiveExtractCallback::SetOperationResult(Int32 operationResult) +{ + switch(operationResult) + { + case NArchive::NExtract::NOperationResult::kOK: + if(all && !_processedFileInfo.isDir) + { + QByteArray rawData((char *)data,newFileSize); + MidFree(data); + data = 0; + if(delegate != 0) + delegate->fileExtracted(_index,rawData); + else + { + allFiles.append(rawData); + } + } + break; + default: + { + NumErrors++; + qDebug() << " "; + switch(operationResult) + { + case NArchive::NExtract::NOperationResult::kUnSupportedMethod: + if(delegate != 0) + delegate->unknownError(_index); + qDebug() << kUnsupportedMethod; + break; + case NArchive::NExtract::NOperationResult::kCRCError: + if(delegate != 0) + delegate->crcError(_index); + qDebug() << kCRCFailed; + break; + case NArchive::NExtract::NOperationResult::kDataError: + if(delegate != 0) + delegate->unknownError(_index); + qDebug() << kDataError; + break; + default: + if(delegate != 0) + delegate->unknownError(_index); + qDebug() << kUnknownError; + } + } + } +/* + if (_outFileStream != NULL) + { + if (_processedFileInfo.MTimeDefined) + _outFileStreamSpec->SetMTime(&_processedFileInfo.MTime); + RINOK(_outFileStreamSpec->Close()); + } + _outFileStream.Release(); + if (_extractMode && _processedFileInfo.AttribDefined) + NFile::NDirectory::MySetFileAttributes(_diskFilePath, _processedFileInfo.Attrib);*/ + //qDebug() << endl; + return S_OK; +} + + +STDMETHODIMP CArchiveExtractCallback::CryptoGetTextPassword(BSTR *password) +{ + if (!PasswordIsDefined) + { + // You can ask real password here from user + // Password = GetPassword(OutStream); + // PasswordIsDefined = true; + qDebug() << "Password is not defined" << endl; + return E_ABORT; + } + return StringToBstr(Password, password); +} + +#endif diff --git a/compressed_archive/extract_delegate.h b/compressed_archive/extract_delegate.h new file mode 100644 index 00000000..888d886a --- /dev/null +++ b/compressed_archive/extract_delegate.h @@ -0,0 +1,14 @@ +#ifndef EXTRACT_DELEGATE_H +#define EXTRACT_DELEGATE_H + +#include + +class ExtractDelegate +{ + public: + virtual void fileExtracted(int index, const QByteArray & rawData) = 0; + virtual void crcError(int index) = 0; + virtual void unknownError(int index) = 0; +}; + +#endif //EXTRACT_DELEGATE_H \ No newline at end of file diff --git a/compressed_archive/libp7zip.patch b/compressed_archive/libp7zip.patch new file mode 100644 index 00000000..522c1202 --- /dev/null +++ b/compressed_archive/libp7zip.patch @@ -0,0 +1,11 @@ +--- libp7zip/CPP/myWindows/StdAfx.h 2014-06-06 23:52:13.397311952 +0200 ++++ libp7zip/CPP/myWindows/StdAfx.h 2014-06-06 23:53:20.353981756 +0200 +@@ -114,7 +114,7 @@ + + #if defined( __x86_64__ ) + +-#define _WIN64 1 ++//#define _WIN64 1 + + #endif + diff --git a/compressed_archive/open_callbacks.h b/compressed_archive/open_callbacks.h new file mode 100644 index 00000000..d696c1f6 --- /dev/null +++ b/compressed_archive/open_callbacks.h @@ -0,0 +1,54 @@ +#ifndef OPEN_CALLBACKS_H +#define OPEN_CALLBACKS_H + +#include "7z_includes.h" +#include +////////////////////////////////////////////////////////////// +// Archive Open callback class + + +class CArchiveOpenCallback: + public IArchiveOpenCallback, + public ICryptoGetTextPassword, + public CMyUnknownImp +{ +public: + MY_UNKNOWN_IMP1(ICryptoGetTextPassword) + + STDMETHOD(SetTotal)(const UInt64 *files, const UInt64 *bytes); + STDMETHOD(SetCompleted)(const UInt64 *files, const UInt64 *bytes); + + STDMETHOD(CryptoGetTextPassword)(BSTR *password); + + bool PasswordIsDefined; + UString Password; + + CArchiveOpenCallback() : PasswordIsDefined(false) {} +}; + +STDMETHODIMP CArchiveOpenCallback::SetTotal(const UInt64 * /* files */, const UInt64 * /* bytes */) +{ + return S_OK; +} + +STDMETHODIMP CArchiveOpenCallback::SetCompleted(const UInt64 * /* files */, const UInt64 * /* bytes */) +{ + return S_OK; +} + +STDMETHODIMP CArchiveOpenCallback::CryptoGetTextPassword(BSTR *password) +{ + if (!PasswordIsDefined) + { + // You can ask real password here from user + // Password = GetPassword(OutStream); + // PasswordIsDefined = true; + qDebug() << "Password is not defined" << endl; + return E_ABORT; + } + return StringToBstr(Password, password); +} + + + +#endif \ No newline at end of file diff --git a/compressed_archive/wrapper.pri b/compressed_archive/wrapper.pri new file mode 100644 index 00000000..933a27e7 --- /dev/null +++ b/compressed_archive/wrapper.pri @@ -0,0 +1,110 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +CONFIG += precompile_header + +win32 {PRECOMPILED_HEADER = $$PWD/StdAfx.h} +!win32 {PRECOMPILED_HEADER = $$PWD/libp7zip/CPP/myWindows/StdAfx.h} + +win32 { +INCLUDEPATH += $$PWD/lib7zip/CPP/ + +DEFINES += _UNICODE _WIN32 + +SOURCES += $$PWD/compressed_archive.cpp \ + $$PWD/lib7zip/CPP/Windows/FileIO.cpp \ + $$PWD/lib7zip/CPP/Windows/PropVariant.cpp \ + $$PWD/lib7zip/CPP/Windows/PropVariantConversions.cpp \ + $$PWD/lib7zip/CPP/Common/IntToString.cpp \ + $$PWD/lib7zip/CPP/Common/MyString.cpp \ + $$PWD/lib7zip/CPP/Common/MyVector.cpp \ + $$PWD/lib7zip/CPP/Common/StringConvert.cpp \ + $$PWD/lib7zip/CPP/Common/Wildcard.cpp \ + $$PWD/lib7zip/CPP/7zip/Common/FileStreams.cpp \ + $$PWD/lib7zip/CPP/7zip/Common/StreamUtils.cpp \ + $$PWD/lib7zip/C/Alloc.c \ + $$PWD/lib7zip/CPP/7zip/Common/StreamObjects.cpp + +HEADERS += $$PWD/compressed_archive.h \ + $$PWD/extract_delegate.h \ + $$PWD/7z_includes.h \ + $$PWD/open_callbacks.h \ + $$PWD/extract_callbacks.h\ + $$PWD/lib7zip/CPP/Windows/FileIO.h \ + $$PWD/lib7zip/CPP/Windows/PropVariant.h \ + $$PWD/lib7zip/CPP/Windows/PropVariantConversions.h \ + $$PWD/lib7zip/CPP/Common/IntToString.h \ + $$PWD/lib7zip/CPP/Common/MyString.h \ + $$PWD/lib7zip/CPP/Common/MyVector.h \ + $$PWD/lib7zip/CPP/Common/StringConvert.h \ + $$PWD/lib7zip/CPP/Common/Wildcard.h \ + $$PWD/lib7zip/CPP/7zip/Common/FileStreams.h \ + $$PWD/lib7zip/CPP/7zip/IStream.h \ + $$PWD/lib7zip/CPP/7zip/Common/StreamUtils.h \ + $$PWD/lib7zip/C/Alloc.h \ + $$PWD/lib7zip/CPP/7zip/Common/StreamObjects.h +} + +macx{ +LIBS += -framework IOKit -framework CoreFoundation + +DEFINES += UNICODE _UNICODE _FILE_OFFSET_BITS=64 _LARGEFILE_SOURCE \ + NDEBUG _REENTRANT ENV_UNIX \ + _7ZIP_LARGE_PAGES ENV_MACOSX _TCHAR_DEFINED +} + +unix:!macx{ +DEFINES += _FILE_OFFSET_BITS=64 _LARGEFILE_SOURCE \ + NDEBUG _REENTRANT ENV_UNIX \ + _7ZIP_LARGE_PAGES + } + +!win32 { +INCLUDEPATH += $$PWD/libp7zip/CPP/ \ + $$PWD/libp7zip/CPP/myWindows/ \ + $$PWD/libp7zip/CPP/include_windows/ + +SOURCES += $$PWD/compressed_archive.cpp \ + $$PWD/libp7zip/CPP/Windows/FileIO.cpp \ + $$PWD/libp7zip/CPP/Windows/FileFind.cpp \ + $$PWD/libp7zip/CPP/Windows/PropVariant.cpp \ + $$PWD/libp7zip/CPP/Windows/PropVariantConversions.cpp \ + $$PWD/libp7zip/CPP/Common/IntToString.cpp \ + $$PWD/libp7zip/CPP/Common/MyString.cpp \ + $$PWD/libp7zip/CPP/Common/MyVector.cpp \ + $$PWD/libp7zip/CPP/Common/StringConvert.cpp \ + $$PWD/libp7zip/CPP/Common/Wildcard.cpp \ + $$PWD/libp7zip/CPP/7zip/Common/FileStreams.cpp \ + $$PWD/libp7zip/CPP/7zip/Common/StreamUtils.cpp \ + $$PWD/libp7zip/C/Alloc.c \ + $$PWD/libp7zip/CPP/7zip/Common/StreamObjects.cpp \ + $$PWD/libp7zip/CPP/myWindows/wine_date_and_time.cpp \ + $$PWD/libp7zip/CPP/Common/MyWindows.cpp + +HEADERS += $$PWD/compressed_archive.h \ + $$PWD/7z_includes.h \ + $$PWD/open_callbacks.h \ + $$PWD/extract_callbacks.h\ + $$PWD/libp7zip/CPP/include_windows/windows.h \ + $$PWD/libp7zip/CPP/include_windows/tchar.h \ + $$PWD/libp7zip/CPP/include_windows/basetyps.h \ + $$PWD/libp7zip/CPP/Windows/FileFind.h \ + $$PWD/libp7zip/CPP/Windows/FileIO.h \ + $$PWD/libp7zip/CPP/Windows/PropVariant.h \ + $$PWD/libp7zip/CPP/Windows/PropVariantConversions.h \ + $$PWD/libp7zip/CPP/Common/IntToString.h \ + $$PWD/libp7zip/CPP/Common/MyString.h \ + $$PWD/libp7zip/CPP/Common/MyVector.h \ + $$PWD/libp7zip/CPP/Common/StringConvert.h \ + $$PWD/libp7zip/CPP/Common/Wildcard.h \ + $$PWD/libp7zip/CPP/7zip/Common/FileStreams.h \ + $$PWD/libp7zip/CPP/7zip/IStream.h \ + $$PWD/libp7zip/CPP/7zip/Common/StreamUtils.h \ + $$PWD/libp7zip/C/Alloc.h \ + $$PWD/libp7zip/CPP/7zip/Common/StreamObjects.h \ + $$PWD/libp7zip/CPP/Common/MyWindows.h \ + $$PWD/libp7zip/CPP/7zip/ICoder.h \ +} + + + diff --git a/create-dmg b/create-dmg new file mode 100755 index 00000000..b581913f --- /dev/null +++ b/create-dmg @@ -0,0 +1,221 @@ +#! /bin/bash + +# Create a read-only disk image of the contents of a folder + +set -e; + +function pure_version() { + echo '1.0.0.2' +} + +function version() { + echo "create-dmg $(pure_version)" +} + +function usage() { + version + echo "Creates a fancy DMG file." + echo "Usage: $(basename $0) options... image.dmg source_folder" + echo "All contents of source_folder will be copied into the disk image." + echo "Options:" + echo " --volname name" + echo " set volume name (displayed in the Finder sidebar and window title)" + echo " --volicon icon.icns" + echo " set volume icon" + echo " --background pic.png" + echo " set folder background image (provide png, gif, jpg)" + echo " --window-pos x y" + echo " set position the folder window" + echo " --window-size width height" + echo " set size of the folder window" + echo " --icon-size icon_size" + echo " set window icons size (up to 128)" + echo " --icon file_name x y" + echo " set position of the file's icon" + echo " --hide-extension file_name" + echo " hide the extension of file" + echo " --custom-icon file_name custom_icon_or_sample_file x y" + echo " set position and custom icon" + echo " --app-drop-link x y" + echo " make a drop link to Applications, at location x,y" + echo " --eula eula_file" + echo " attach a license file to the dmg" + echo " --no-internet-enable" + echo " disable automatic mount©" + echo " --version show tool version number" + echo " -h, --help display this help" + exit 0 +} + +WINX=10 +WINY=60 +WINW=500 +WINH=350 +ICON_SIZE=128 + +while test "${1:0:1}" = "-"; do + case $1 in + --volname) + VOLUME_NAME="$2" + shift; shift;; + --volicon) + VOLUME_ICON_FILE="$2" + shift; shift;; + --background) + BACKGROUND_FILE="$2" + BACKGROUND_FILE_NAME="$(basename $BACKGROUND_FILE)" + BACKGROUND_CLAUSE="set background picture of opts to file \".background:$BACKGROUND_FILE_NAME\"" + shift; shift;; + --icon-size) + ICON_SIZE="$2" + shift; shift;; + --window-pos) + WINX=$2; WINY=$3 + shift; shift; shift;; + --window-size) + WINW=$2; WINH=$3 + shift; shift; shift;; + --icon) + POSITION_CLAUSE="${POSITION_CLAUSE}set position of item \"$2\" to {$3, $4} +" + shift; shift; shift; shift;; + --hide-extension) + HIDING_CLAUSE="${HIDING_CLAUSE}set the extension hidden of item \"$2\" to true" + shift; shift;; + --custom-icon) + shift; shift; shift; shift; shift;; + -h | --help) + usage;; + --version) + version; exit 0;; + --pure-version) + pure_version; exit 0;; + --app-drop-link) + APPLICATION_LINK=$2 + APPLICATION_CLAUSE="set position of item \"Applications\" to {$2, $3} +" + shift; shift; shift;; + --eula) + EULA_RSRC=$2 + shift; shift;; + --no-internet-enable) + NOINTERNET=1 + shift;; + -*) + echo "Unknown option $1. Run with --help for help." + exit 1;; + esac +done + +test -z "$2" && { + echo "Not enough arguments. Invoke with --help for help." + exit 1 +} + +DMG_PATH="$1" +DMG_DIRNAME="$(dirname "$DMG_PATH")" +DMG_DIR="$(cd $DMG_DIRNAME > /dev/null; pwd)" +DMG_NAME="$(basename "$DMG_PATH")" +DMG_TEMP_NAME="$DMG_DIR/rw.${DMG_NAME}" +SRC_FOLDER="$(cd "$2" > /dev/null; pwd)" +test -z "$VOLUME_NAME" && VOLUME_NAME="$(basename "$DMG_PATH" .dmg)" + +AUX_PATH="$(dirname $0)/support" + +test -d "$AUX_PATH" || { + echo "Cannot find support directory: $AUX_PATH" + exit 1 +} + +if [ -f "$SRC_FOLDER/.DS_Store" ]; then + echo "Deleting any .DS_Store in source folder" + rm "$SRC_FOLDER/.DS_Store" +fi + +# Create the image +echo "Creating disk image..." +test -f "${DMG_TEMP_NAME}" && rm -f "${DMG_TEMP_NAME}" +ACTUAL_SIZE=`du -sm "$SRC_FOLDER" | sed -e 's/ .*//g'` +DISK_IMAGE_SIZE=$(expr $ACTUAL_SIZE + 20) +hdiutil create -srcfolder "$SRC_FOLDER" -volname "${VOLUME_NAME}" -fs HFS+ -fsargs "-c c=64,a=16,e=16" -format UDRW -size ${DISK_IMAGE_SIZE}m "${DMG_TEMP_NAME}" + +# mount it +echo "Mounting disk image..." +MOUNT_DIR="/Volumes/${VOLUME_NAME}" + +# try unmount dmg if it was mounted previously (e.g. developer mounted dmg, installed app and forgot to unmount it) +echo "Unmounting disk image..." +DEV_NAME=$(hdiutil info | egrep '^/dev/' | sed 1q | awk '{print $1}') +test -d "${MOUNT_DIR}" && hdiutil detach "${DEV_NAME}" + +echo "Mount directory: $MOUNT_DIR" +DEV_NAME=$(hdiutil attach -readwrite -noverify -noautoopen "${DMG_TEMP_NAME}" | egrep '^/dev/' | sed 1q | awk '{print $1}') +echo "Device name: $DEV_NAME" + +if ! test -z "$BACKGROUND_FILE"; then + echo "Copying background file..." + test -d "$MOUNT_DIR/.background" || mkdir "$MOUNT_DIR/.background" + cp "$BACKGROUND_FILE" "$MOUNT_DIR/.background/$BACKGROUND_FILE_NAME" +fi + +if ! test -z "$APPLICATION_LINK"; then + echo "making link to Applications dir" + echo $MOUNT_DIR + ln -s /Applications "$MOUNT_DIR/Applications" +fi + +if ! test -z "$VOLUME_ICON_FILE"; then + echo "Copying volume icon file '$VOLUME_ICON_FILE'..." + cp "$VOLUME_ICON_FILE" "$MOUNT_DIR/.VolumeIcon.icns" + SetFile -c icnC "$MOUNT_DIR/.VolumeIcon.icns" +fi + +# run applescript +APPLESCRIPT=$(mktemp -t createdmg) +cat "$AUX_PATH/template.applescript" | sed -e "s/WINX/$WINX/g" -e "s/WINY/$WINY/g" -e "s/WINW/$WINW/g" -e "s/WINH/$WINH/g" -e "s/BACKGROUND_CLAUSE/$BACKGROUND_CLAUSE/g" -e "s/ICON_SIZE/$ICON_SIZE/g" | perl -pe "s/POSITION_CLAUSE/$POSITION_CLAUSE/g" | perl -pe "s/APPLICATION_CLAUSE/$APPLICATION_CLAUSE/g" | perl -pe "s/HIDING_CLAUSE/$HIDING_CLAUSE/" >"$APPLESCRIPT" + +echo "Running Applescript: /usr/bin/osascript \"${APPLESCRIPT}\" \"${VOLUME_NAME}\"" +"/usr/bin/osascript" "${APPLESCRIPT}" "${VOLUME_NAME}" || true +echo "Done running the applescript..." +sleep 4 + +rm "$APPLESCRIPT" + +# make sure it's not world writeable +echo "Fixing permissions..." +chmod -Rf go-w "${MOUNT_DIR}" &> /dev/null || true +echo "Done fixing permissions." + +# make the top window open itself on mount: +echo "Blessing started" +bless --folder "${MOUNT_DIR}" --openfolder "${MOUNT_DIR}" +echo "Blessing finished" + +if ! test -z "$VOLUME_ICON_FILE"; then + # tell the volume that it has a special file attribute + SetFile -a C "$MOUNT_DIR" +fi + +# unmount +echo "Unmounting disk image..." +hdiutil detach "${DEV_NAME}" + +# compress image +echo "Compressing disk image..." +hdiutil convert "${DMG_TEMP_NAME}" -format UDZO -imagekey zlib-level=9 -o "${DMG_DIR}/${DMG_NAME}" +rm -f "${DMG_TEMP_NAME}" + +# adding EULA resources +if [ ! -z "${EULA_RSRC}" -a "${EULA_RSRC}" != "-null-" ]; then + echo "adding EULA resources" + "${AUX_PATH}/dmg-license.py" "${DMG_DIR}/${DMG_NAME}" "${EULA_RSRC}" +fi + +if [ ! -z "${NOINTERNET}" -a "${NOINTERNET}" == 1 ]; then + echo "not setting 'internet-enable' on the dmg" +else + hdiutil internet-enable -yes "${DMG_DIR}/${DMG_NAME}" +fi + +echo "Disk image done" +exit 0 diff --git a/custom_widgets/custom_widgets_yacreader.pri b/custom_widgets/custom_widgets_yacreader.pri new file mode 100644 index 00000000..a7493911 --- /dev/null +++ b/custom_widgets/custom_widgets_yacreader.pri @@ -0,0 +1,25 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += $$PWD/help_about_dialog.h \ + $$PWD/yacreader_field_edit.h \ + $$PWD/yacreader_field_plain_text_edit.h \ + $$PWD/yacreader_flow.h \ + $$PWD/yacreader_flow_config_widget.h \ + $$PWD/yacreader_gl_flow_config_widget.h \ + $$PWD/yacreader_options_dialog.h \ + $$PWD/yacreader_spin_slider_widget.h \ + $$PWD/yacreader_tool_bar_stretch.h \ + $$PWD/yacreader_busy_widget.h + + +SOURCES += $$PWD/help_about_dialog.cpp \ + $$PWD/yacreader_field_edit.cpp \ + $$PWD/yacreader_field_plain_text_edit.cpp \ + $$PWD/yacreader_flow.cpp \ + $$PWD/yacreader_flow_config_widget.cpp \ + $$PWD/yacreader_gl_flow_config_widget.cpp \ + $$PWD/yacreader_options_dialog.cpp \ + $$PWD/yacreader_spin_slider_widget.cpp \ + $$PWD/yacreader_tool_bar_stretch.cpp \ + $$PWD/yacreader_busy_widget.cpp \ No newline at end of file diff --git a/custom_widgets/custom_widgets_yacreaderlibrary.pri b/custom_widgets/custom_widgets_yacreaderlibrary.pri new file mode 100644 index 00000000..661463d2 --- /dev/null +++ b/custom_widgets/custom_widgets_yacreaderlibrary.pri @@ -0,0 +1,41 @@ +INCLUDEPATH += $$PWD +DEPENDPATH += $$PWD + +HEADERS += $$PWD/help_about_dialog.h \ + $$PWD/yacreader_field_edit.h \ + $$PWD/yacreader_field_plain_text_edit.h \ + $$PWD/yacreader_flow.h \ + $$PWD/yacreader_flow_config_widget.h \ + $$PWD/yacreader_gl_flow_config_widget.h \ + $$PWD/yacreader_options_dialog.h \ + $$PWD/yacreader_search_line_edit.h \ + $$PWD/yacreader_spin_slider_widget.h \ + $$PWD/yacreader_tool_bar_stretch.h \ + $$PWD/yacreader_titled_toolbar.h \ + $$PWD/yacreader_deleting_progress.h \ + $$PWD/yacreader_table_view.h \ + $$PWD/yacreader_sidebar.h \ + $$PWD/yacreader_library_list_widget.h \ + $$PWD/yacreader_library_item_widget.h \ + $$PWD/yacreader_treeview.h \ + $$PWD/yacreader_busy_widget.h + + +SOURCES += $$PWD/help_about_dialog.cpp \ + $$PWD/yacreader_field_edit.cpp \ + $$PWD/yacreader_field_plain_text_edit.cpp \ + $$PWD/yacreader_flow.cpp \ + $$PWD/yacreader_flow_config_widget.cpp \ + $$PWD/yacreader_gl_flow_config_widget.cpp \ + $$PWD/yacreader_options_dialog.cpp \ + $$PWD/yacreader_search_line_edit.cpp \ + $$PWD/yacreader_spin_slider_widget.cpp \ + $$PWD/yacreader_tool_bar_stretch.cpp \ + $$PWD/yacreader_titled_toolbar.cpp \ + $$PWD/yacreader_deleting_progress.cpp \ + $$PWD/yacreader_table_view.cpp \ + $$PWD/yacreader_sidebar.cpp \ + $$PWD/yacreader_library_list_widget.cpp \ + $$PWD/yacreader_library_item_widget.cpp \ + $$PWD/yacreader_treeview.cpp \ + $$PWD/yacreader_busy_widget.cpp \ No newline at end of file diff --git a/custom_widgets/help_about_dialog.cpp b/custom_widgets/help_about_dialog.cpp new file mode 100644 index 00000000..ff926a6b --- /dev/null +++ b/custom_widgets/help_about_dialog.cpp @@ -0,0 +1,75 @@ +#include "help_about_dialog.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +HelpAboutDialog::HelpAboutDialog(QWidget * parent) +:QDialog(parent) +{ + QVBoxLayout * layout = new QVBoxLayout(); + + tabWidget = new QTabWidget(); + + tabWidget->addTab(aboutText = new QTextBrowser(), tr("About")); + aboutText->setOpenExternalLinks(true); + //aboutText->setFont(QFont("Comic Sans MS", 10)); //purisa + tabWidget->addTab(helpText = new QTextBrowser(), tr("Help")); + helpText->setOpenExternalLinks(true); + //helpText->setFont(QFont("Comic Sans MS", 10)); + //helpText->setDisabled(true); + //tabWidget->addTab(,"About Qt"); + + layout->addWidget(tabWidget); + layout->setContentsMargins(1,3,1,1); + + setLayout(layout); + resize(500, QApplication::desktop()->availableGeometry().height()*0.83); +} + +HelpAboutDialog::~HelpAboutDialog() +{ + delete aboutText; + delete helpText; + delete tabWidget; +} + +HelpAboutDialog::HelpAboutDialog(const QString & pathAbout,const QString & pathHelp,QWidget * parent) +:QDialog(parent) +{ + loadAboutInformation(pathAbout); + loadHelp(pathHelp); +} + +void HelpAboutDialog::loadAboutInformation(const QString & path) +{ + aboutText->setHtml(fileToString(path).arg(VERSION)); + aboutText->moveCursor(QTextCursor::Start); +} + +void HelpAboutDialog::loadHelp(const QString & path) +{ + helpText->setHtml(fileToString(path)); + helpText->moveCursor(QTextCursor::Start); +} + +QString HelpAboutDialog::fileToString(const QString & path) +{ + QFile f(path); + f.open(QIODevice::ReadOnly); + QTextStream txtS(&f); + + txtS.setCodec(QTextCodec::codecForName("UTF-8")); + + QString content = txtS.readAll(); + f.close(); + + return content; +} \ No newline at end of file diff --git a/custom_widgets/help_about_dialog.h b/custom_widgets/help_about_dialog.h new file mode 100644 index 00000000..70a0a662 --- /dev/null +++ b/custom_widgets/help_about_dialog.h @@ -0,0 +1,28 @@ +#ifndef HELP_ABOUT_DIALOG_H +#define HELP_ABOUT_DIALOG_H + +#include + +class QTabWidget; +class QTextBrowser; + +class HelpAboutDialog : public QDialog +{ +Q_OBJECT +public: + HelpAboutDialog(QWidget * parent=0); + HelpAboutDialog(const QString & pathAbout,const QString & pathHelp,QWidget * parent =0); + ~HelpAboutDialog(); +public slots: + void loadAboutInformation(const QString & path); + void loadHelp(const QString & path); + +private: + QTabWidget *tabWidget; + QTextBrowser *aboutText; + QTextBrowser *helpText; + QString fileToString(const QString & path); +}; + + +#endif // HELP_ABOUT_DIALOG_H \ No newline at end of file diff --git a/custom_widgets/yacreader_busy_widget.cpp b/custom_widgets/yacreader_busy_widget.cpp new file mode 100644 index 00000000..94e93718 --- /dev/null +++ b/custom_widgets/yacreader_busy_widget.cpp @@ -0,0 +1,187 @@ +#include "yacreader_busy_widget.h" + +#include +#include +#include +#include + +YACReaderBusyWidget::YACReaderBusyWidget(QWidget *parent) + :QWidget(parent) +{ + setFixedSize(70,70); + BusyIndicator * busy = new BusyIndicator(this); + busy->setIndicatorStyle(BusyIndicator::StyleArc); + busy->setColor(Qt::white); + busy->move(20,20); +} + +void YACReaderBusyWidget::paintEvent(QPaintEvent * event) +{ + Q_UNUSED(event); + QPainter painter(this); + painter.setRenderHint(QPainter::Antialiasing); + painter.drawPixmap(0,0,width(),height(),QPixmap(":/images/busy_background.png")); +} + +BusyIndicator::BusyIndicator(QWidget *parent) : + QWidget(parent), + startAngle(0), + m_style(StyleArc) +{ + QSizePolicy policy(QSizePolicy::Preferred, QSizePolicy::Fixed); + policy.setHeightForWidth(true); + setSizePolicy(policy); + + fillColor = palette().color(QPalette::WindowText); + + timer.setInterval(16); + connect(&timer, SIGNAL(timeout()), this, SLOT(rotate())); + timer.start(); +} + +void BusyIndicator::rotate() +{ + startAngle += 9; + startAngle %= 360; + update(); +} + +void BusyIndicator::setIndicatorStyle(IndicatorStyle style) +{ + m_style = style; + update(); +} + +void BusyIndicator::setColor(QColor color) +{ + fillColor = color; +} + +const BusyIndicator::IndicatorStyle BusyIndicator::indicatorStyle() const +{ + return m_style; +} + + +QPixmap BusyIndicator::generatePixmap(int side) +{ + QPixmap pixmap(QSize(side, side)); + pixmap.fill(QColor(255, 255, 255, 0)); + + QPainter painter(&pixmap); + painter.setRenderHint(QPainter::Antialiasing); + + painter.translate(side / 2, side / 2); + painter.scale(side / 200.0, side / 200.0); + + switch (m_style) { + case StyleRect: + drawRectStyle(&painter); + break; + case StyleEllipse: + drawEllipseStyle(&painter); + break; + case StyleArc: + drawArcStyle(&painter); + break; + } + return pixmap; +} + +void BusyIndicator::drawRectStyle(QPainter *painter) +{ + // QColor color = palette().color(QPalette::WindowText); + QColor color = fillColor; + QBrush brush(color); + painter->setPen(Qt::NoPen); + + painter->rotate(startAngle); + + float angle = 0; + while (angle < 360) { + painter->setBrush(brush); + painter->drawRect(-8, -100, 16, 35); + + painter->rotate(30); + angle += 30; + + color.setAlphaF(angle / 360); + brush.setColor(color); + } +} + +void BusyIndicator::drawEllipseStyle(QPainter *painter) +{ + // QColor color = palette().color(QPalette::WindowText); + QColor color = fillColor; + QBrush brush(color); + painter->setPen(Qt::NoPen); + + painter->rotate(startAngle); + + float angle = 0; + while (angle < 360) { + painter->setBrush(brush); + painter->drawEllipse(-10, -100, 30, 30); + + painter->rotate(30); + angle += 30; + + color.setAlphaF(angle / 360); + brush.setColor(color); + } +} + +void BusyIndicator::drawArcStyle(QPainter *painter) +{ + // QColor color = palette().color(QPalette::WindowText); + QColor color = fillColor; + QConicalGradient gradient(0, 0, -startAngle); + gradient.setColorAt(0, color); + color.setAlpha(0); + gradient.setColorAt(0.8, color); + color.setAlpha(255); + gradient.setColorAt(1, color); + + QPen pen; + pen.setWidth(30); + pen.setBrush(QBrush(gradient)); + painter->setPen(pen); + + painter->drawArc(-85, -85, 170, 170, 0 * 16, 360 * 16); +} + +void BusyIndicator::paintEvent(QPaintEvent *) +{ + QString key = QString("%1:%2:%3:%4:%5") + .arg(metaObject()->className()) + .arg(width()) + .arg(height()) + .arg(startAngle) + .arg(m_style); + + QPixmap pixmap; + QPainter painter(this); + painter.setRenderHint(QPainter::Antialiasing); + + int side = qMin(width(), height()); + + if(!QPixmapCache::find(key, &pixmap)) { + pixmap = generatePixmap(side); + QPixmapCache::insert(key, pixmap); + } + + painter.translate(width() / 2 - side / 2, height() / 2 - side / 2); + + painter.drawPixmap(0, 0, side, side, pixmap); +} + +QSize BusyIndicator::minimumSizeHint() const +{ + return QSize(30, 30); +} + +QSize BusyIndicator::sizeHint() const +{ + return QSize(30, 30); +} diff --git a/custom_widgets/yacreader_busy_widget.h b/custom_widgets/yacreader_busy_widget.h new file mode 100644 index 00000000..c98dda07 --- /dev/null +++ b/custom_widgets/yacreader_busy_widget.h @@ -0,0 +1,50 @@ +#ifndef YACREADER_BUSYINDICATOR_H +#define YACREADER_BUSYINDICATOR_H + +#include +#include + +class YACReaderBusyWidget : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderBusyWidget(QWidget *parent = 0); + void paintEvent(QPaintEvent *); +}; + +class BusyIndicator : public QWidget +{ + Q_OBJECT +public: + enum IndicatorStyle{StyleRect, StyleEllipse, StyleArc}; + + explicit BusyIndicator(QWidget *parent = 0); + + void paintEvent(QPaintEvent *); + QSize minimumSizeHint() const; + QSize sizeHint() const; + + void setIndicatorStyle(IndicatorStyle); + void setColor(QColor color); + const IndicatorStyle indicatorStyle() const; + +signals: + +private slots: + void rotate(); + +private: + QPixmap generatePixmap(int sideLength); + void drawRectStyle(QPainter *painter); + void drawEllipseStyle(QPainter *painter); + void drawArcStyle(QPainter *painter); + + QTimer timer; + int startAngle; + + IndicatorStyle m_style; + + QColor fillColor; +}; + +#endif // BUSYINDICATOR_H diff --git a/custom_widgets/yacreader_dark_menu.cpp b/custom_widgets/yacreader_dark_menu.cpp new file mode 100644 index 00000000..0ed7118c --- /dev/null +++ b/custom_widgets/yacreader_dark_menu.cpp @@ -0,0 +1,38 @@ +#include "yacreader_dark_menu.h" + +#include +#include +#include + +YACReaderDarkMenu::YACReaderDarkMenu(QWidget * parent) + :QMenu(parent) +{ + //solid color: #454545 + QString style = "QMenu {background-color: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #6B6B6B, stop: 1 #424242); " + "border-left: 1px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #BCBCBC, stop: 1 #4C4C4C);" + "border-right: 1px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #BCBCBC, stop: 1 #4C4C4C);" + "border-top: 1px solid #BCBCBC;" + "border-bottom: 1px solid #4C4C4C;" + "padding-top:5px;padding-bottom:5px;}" + "QMenu::separator {height:0px;border-top: 1px solid #292929; border-bottom:1px solid #737373; margin-left:-1px; margin-right:-1px;}" + "QMenu::item {color:#CFD1D1;padding: 5px 25px 5px 32px;}" + "QMenu::item::selected {background-color:#242424;border-top: 1px solid #151515; border-bottom:1px solid #737373;}" + "QMenu::icon {padding-left:15px;}"; + + setStyleSheet(style); + + /* + QPixmap p(":/images/icon.png"); + QLabel * l = new QLabel(); + l->setPixmap(p); + l->move(0,-10); + + //test + YACReaderDarkMenu * customMenu = new YACReaderDarkMenu(this); + customMenu->addAction(toggleFullScreenAction); + customMenu->addAction(createLibraryAction); + customMenu->addSeparator(); + customMenu->addAction(openComicAction); + customMenu->show(); + */ +} \ No newline at end of file diff --git a/custom_widgets/yacreader_dark_menu.h b/custom_widgets/yacreader_dark_menu.h new file mode 100644 index 00000000..6d28749d --- /dev/null +++ b/custom_widgets/yacreader_dark_menu.h @@ -0,0 +1,14 @@ +#ifndef YACREADER_DARK_MENU_H +#define YACREADER_DARK_MENU_H + +#include + + +class YACReaderDarkMenu : public QMenu +{ + Q_OBJECT + public: + YACReaderDarkMenu(QWidget * parent = 0); +}; + +#endif // YACREADER_DARK_MENU_H \ No newline at end of file diff --git a/custom_widgets/yacreader_deleting_progress.cpp b/custom_widgets/yacreader_deleting_progress.cpp new file mode 100644 index 00000000..62a5a78f --- /dev/null +++ b/custom_widgets/yacreader_deleting_progress.cpp @@ -0,0 +1,106 @@ +#include "yacreader_deleting_progress.h" + +#include +#include +#include +#include +#include +#include + +YACReaderDeletingProgress::YACReaderDeletingProgress(QWidget *parent) : + QWidget(parent) +{ + QVBoxLayout * contentLayout = new QVBoxLayout(this); + + QLabel * iconLabel = new QLabel(); + QPixmap icon(":/images/deleting_progress/icon.png"); + iconLabel->setPixmap(icon); + iconLabel->setStyleSheet("QLabel {padding:0px; margin:0px;}"); + + textMessage = new QLabel(tr("Please wait, deleting in progress...")); + + textMessage->setStyleSheet("QLabel {color:#ABABAB; padding:0 0 0 0px; margin:0px; font-size:18px; font-weight:bold;}"); + + QProgressBar * progressBar = new QProgressBar(); + + progressBar->setTextVisible(false); + progressBar->setFixedHeight(6); + progressBar->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Fixed); + progressBar->setRange (0,10); + progressBar->setValue(5); + progressBar->setStyleSheet( + "QProgressBar { border: none; border-radius: 3px; background: #ABABAB; margin:0; margin-left:16; margin-right:16px;}" + "QProgressBar::chunk {background-color: #FFC745; border: none; border-radius: 3px;}"); + + QPushButton * button = new QPushButton(tr("cancel")); + + button->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum); + + contentLayout->addSpacing(16); + contentLayout->addWidget(iconLabel,0,Qt::AlignHCenter); + contentLayout->addSpacing(11); + contentLayout->addWidget(textMessage,0,Qt::AlignHCenter); + contentLayout->addSpacing(13); + contentLayout->addWidget(progressBar); + contentLayout->addSpacing(13); + contentLayout->addWidget(button,0,Qt::AlignHCenter); + contentLayout->addSpacing(18); + + contentLayout->setMargin(0); + + setLayout(contentLayout); + + setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum); + + resize( sizeHint() ); +} + +void YACReaderDeletingProgress::paintEvent(QPaintEvent * event) +{ + int borderTop, borderRight, borderBottom, borderLeft; + + QPixmap pL(":/images/deleting_progress/imgTopLeft.png"); + QPixmap pM(":/images/deleting_progress/imgTopMiddle.png"); + QPixmap pR(":/images/deleting_progress/imgTopRight.png"); + + QPixmap pLM(":/images/deleting_progress/imgLeftMiddle.png"); + + QPixmap pRM(":/images/deleting_progress/imgRightMiddle.png"); + + QPixmap pBL(":/images/deleting_progress/imgBottomLeft.png"); + QPixmap pBM(":/images/deleting_progress/imgBottomMiddle.png"); + QPixmap pBR(":/images/deleting_progress/imgBottomRight.png"); + + borderTop = pL.height(); + borderRight = pRM.width(); + borderBottom = pBM.height(); + borderLeft = pLM.width(); + + int width = this->width()-borderRight-borderLeft; + int height = this->height()-borderTop-borderBottom; + + QPainter painter(this); + + //corners + painter.drawPixmap(0,0,pL); + painter.drawPixmap(this->width()-borderRight,0,pR); + painter.drawPixmap(0,this->height()-pBL.height(),pBL); + painter.drawPixmap(this->width()-pBR.width(),this->height()-borderBottom,pBR); + + //middle + painter.drawPixmap(borderRight,0,width,borderTop,pM); + painter.drawPixmap(0,borderTop,borderLeft,height,pLM); + painter.drawPixmap(width+borderLeft,borderTop,borderRight,height,pRM); + painter.drawPixmap(pBR.width(),height+borderTop,this->width()-pBR.width()-pBL.width(),pBR.height(),pBM); + + //center + painter.fillRect(borderLeft,borderTop,width,height,QColor("#FAFAFA")); + + QWidget::paintEvent(event); +} + + +QSize YACReaderDeletingProgress::sizeHint() const +{ + return QSize(textMessage->sizeHint().width()+120,185); +} diff --git a/custom_widgets/yacreader_deleting_progress.h b/custom_widgets/yacreader_deleting_progress.h new file mode 100644 index 00000000..badf1e6a --- /dev/null +++ b/custom_widgets/yacreader_deleting_progress.h @@ -0,0 +1,26 @@ +#ifndef YACREADER_DELETING_PROGRESS_H +#define YACREADER_DELETING_PROGRESS_H + +#include + +class QLabel; + +class YACReaderDeletingProgress : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderDeletingProgress(QWidget *parent = 0); + QSize sizeHint() const; +signals: + +public slots: + +protected: + void paintEvent(QPaintEvent *); + +private: + QLabel * textMessage; + +}; + +#endif // YACREADER_DELETING_PROGRESS_H diff --git a/custom_widgets/yacreader_field_edit.cpp b/custom_widgets/yacreader_field_edit.cpp new file mode 100644 index 00000000..3169784d --- /dev/null +++ b/custom_widgets/yacreader_field_edit.cpp @@ -0,0 +1,39 @@ +#include "yacreader_field_edit.h" + +#include +#include + +YACReaderFieldEdit::YACReaderFieldEdit(QWidget * parent) + :QLineEdit(parent) +{ + setPlaceholderText(tr("Click to overwrite")); + setModified(false); + restore = new QAction(tr("Restore to default"),this); + this->addAction(restore); + //this->setContextMenuPolicy(Qt::ActionsContextMenu); +} + +void YACReaderFieldEdit::focusInEvent(QFocusEvent* e) +{ + if (e->reason() == Qt::MouseFocusReason) + { + setModified(true); + setPlaceholderText(""); + } + + QLineEdit::focusInEvent(e); +} + +void YACReaderFieldEdit::clear() +{ + setPlaceholderText(tr("Click to overwrite")); + QLineEdit::clear(); + QLineEdit::setModified(false); +} + +void YACReaderFieldEdit::setDisabled(bool disabled) +{ + if(disabled) + setPlaceholderText(""); + QLineEdit::setDisabled(disabled); +} \ No newline at end of file diff --git a/custom_widgets/yacreader_field_edit.h b/custom_widgets/yacreader_field_edit.h new file mode 100644 index 00000000..b7baf0f1 --- /dev/null +++ b/custom_widgets/yacreader_field_edit.h @@ -0,0 +1,23 @@ +#ifndef YACREADER_FIELD_EDIT_H +#define YACREADER_FIELD_EDIT_H + +#include + +class QAction; +class QFocusEvent; + +class YACReaderFieldEdit : public QLineEdit +{ + Q_OBJECT + public: + YACReaderFieldEdit(QWidget * parent = 0); + void clear(); + void setDisabled(bool disabled); + protected: + void focusInEvent(QFocusEvent* e); +private: + QAction * restore; + +}; + +#endif // YACREADER_FIELD_EDIT_H \ No newline at end of file diff --git a/custom_widgets/yacreader_field_plain_text_edit.cpp b/custom_widgets/yacreader_field_plain_text_edit.cpp new file mode 100644 index 00000000..c73cfc03 --- /dev/null +++ b/custom_widgets/yacreader_field_plain_text_edit.cpp @@ -0,0 +1,53 @@ +#include "yacreader_field_plain_text_edit.h" + +#include + +YACReaderFieldPlainTextEdit::YACReaderFieldPlainTextEdit(QWidget * parent) + :QPlainTextEdit(parent) +{ + document()->setModified(false); + setPlainText(tr("Click to overwrite")); + restore = new QAction(tr("Restore to default"),this); + this->addAction(restore); + //this->setContextMenuPolicy(Qt::ActionsContextMenu); +} + +void YACReaderFieldPlainTextEdit::focusInEvent(QFocusEvent* e) +{ + if (e->reason() == Qt::MouseFocusReason || e->reason() == Qt::TabFocusReason) + { + document()->setModified(true); + if(toPlainText()==tr("Click to overwrite")) + setPlainText(""); + } + + QPlainTextEdit::focusInEvent(e); +} + +void YACReaderFieldPlainTextEdit::focusOutEvent(QFocusEvent* e) +{ + /*if (e->reason() == Qt::MouseFocusReason || e->reason() == Qt::TabFocusReason) + { + if(toPlainText().isEmpty()) + { + setPlainText(tr("Click to overwrite")); + document()->setModified(false); + } + } + */ + QPlainTextEdit::focusOutEvent(e); +} + +void YACReaderFieldPlainTextEdit::clear() +{ + QPlainTextEdit::clear(); + document()->setModified(false); + setPlainText(tr("Click to overwrite")); +} + +void YACReaderFieldPlainTextEdit::setDisabled(bool disabled) +{ + if(disabled) + setPlainText(tr("Click to overwrite")); + QPlainTextEdit::setDisabled(disabled); +} diff --git a/custom_widgets/yacreader_field_plain_text_edit.h b/custom_widgets/yacreader_field_plain_text_edit.h new file mode 100644 index 00000000..0d02493c --- /dev/null +++ b/custom_widgets/yacreader_field_plain_text_edit.h @@ -0,0 +1,25 @@ +#ifndef YACREADER_FIELD_PLAIN_TEXT_EDIT_H +#define YACREADER_FIELD_PLAIN_TEXT_EDIT_H + +#include + +class QAction; +class QFocusEvent; + + +class YACReaderFieldPlainTextEdit : public QPlainTextEdit +{ + Q_OBJECT + public: + YACReaderFieldPlainTextEdit(QWidget * parent = 0); + void clear(); + void setDisabled(bool disabled); + protected: + void focusInEvent(QFocusEvent* e); + void focusOutEvent(QFocusEvent* e); +private: + QAction * restore; + +}; + +#endif // YACREADER_FIELD_PLAIN_TEXT_EDIT_H \ No newline at end of file diff --git a/custom_widgets/yacreader_flow.cpp b/custom_widgets/yacreader_flow.cpp new file mode 100644 index 00000000..e40507d3 --- /dev/null +++ b/custom_widgets/yacreader_flow.cpp @@ -0,0 +1,23 @@ +#include "yacreader_flow.h" + +#include + + +YACReaderFlow::YACReaderFlow(QWidget * parent,FlowType flowType) : PictureFlow(parent,flowType) {} + +void YACReaderFlow::mousePressEvent(QMouseEvent* event) +{ + if(event->x() > (width()+slideSize().width())/2) + showNext(); + else + if(event->x() < (width()-slideSize().width())/2) + showPrevious(); + //else (centered cover space) +} + +void YACReaderFlow::mouseDoubleClickEvent(QMouseEvent* event) +{ + if((event->x() > (width()-slideSize().width())/2)&&(event->x() < (width()+slideSize().width())/2)) + emit selected(centerIndex()); +} + diff --git a/custom_widgets/yacreader_flow.h b/custom_widgets/yacreader_flow.h new file mode 100644 index 00000000..7a478967 --- /dev/null +++ b/custom_widgets/yacreader_flow.h @@ -0,0 +1,21 @@ +#ifndef YACREADER_FLOW_H +#define YACREADER_FLOW_H + +#include "pictureflow.h" + +class QMouseEvent; + +class YACReaderFlow : public PictureFlow +{ +Q_OBJECT +public: + YACReaderFlow(QWidget * parent,FlowType flowType = CoverFlowLike); + + void mousePressEvent(QMouseEvent* event); + void mouseDoubleClickEvent(QMouseEvent* event); + +signals: + void selected(unsigned int centerIndex); +}; + +#endif // YACREADER_FLOW_H \ No newline at end of file diff --git a/custom_widgets/yacreader_flow_config_widget.cpp b/custom_widgets/yacreader_flow_config_widget.cpp new file mode 100644 index 00000000..8fd3ba94 --- /dev/null +++ b/custom_widgets/yacreader_flow_config_widget.cpp @@ -0,0 +1,54 @@ +#include "yacreader_flow_config_widget.h" + +#include +#include +#include +#include + +YACReaderFlowConfigWidget::YACReaderFlowConfigWidget(QWidget * parent ) + :QWidget(parent) +{ + QVBoxLayout * layout = new QVBoxLayout(this); + + QGroupBox *groupBox = new QGroupBox(tr("How to show covers:")); + + radio1 = new QRadioButton(tr("CoverFlow look")); + radio2 = new QRadioButton(tr("Stripe look")); + radio3 = new QRadioButton(tr("Overlapped Stripe look")); + + + QVBoxLayout *vbox = new QVBoxLayout; + QHBoxLayout * opt1 = new QHBoxLayout; + opt1->addWidget(radio1); + QLabel * lOpt1 = new QLabel(); + lOpt1->setPixmap(QPixmap(":/images/flow1.png")); + opt1->addStretch(); + opt1->addWidget(lOpt1); + vbox->addLayout(opt1); + + QHBoxLayout * opt2 = new QHBoxLayout; + opt2->addWidget(radio2); + QLabel * lOpt2 = new QLabel(); + lOpt2->setPixmap(QPixmap(":/images/flow2.png")); + opt2->addStretch(); + opt2->addWidget(lOpt2); + vbox->addLayout(opt2); + + QHBoxLayout * opt3 = new QHBoxLayout; + opt3->addWidget(radio3); + QLabel * lOpt3 = new QLabel(); + lOpt3->setPixmap(QPixmap(":/images/flow3.png")); + opt3->addStretch(); + opt3->addWidget(lOpt3); + vbox->addLayout(opt3); + + + //vbox->addStretch(1); + groupBox->setLayout(vbox); + + layout->addWidget(groupBox); + + layout->setContentsMargins(0,0,0,0); + + setLayout(layout); +} \ No newline at end of file diff --git a/custom_widgets/yacreader_flow_config_widget.h b/custom_widgets/yacreader_flow_config_widget.h new file mode 100644 index 00000000..b5dee55d --- /dev/null +++ b/custom_widgets/yacreader_flow_config_widget.h @@ -0,0 +1,19 @@ +#ifndef YACREADER_FLOW_CONFIG_WIDGET_H +#define YACREADER_FLOW_CONFIG_WIDGET_H + +#include + +class QRadioButton; + +class YACReaderFlowConfigWidget : public QWidget +{ + Q_OBJECT +public: + QRadioButton *radio1; + QRadioButton *radio2; + QRadioButton *radio3; + + YACReaderFlowConfigWidget(QWidget * parent = 0); +}; + +#endif // YACREADER_FLOW_CONFIG_WIDGET_H \ No newline at end of file diff --git a/custom_widgets/yacreader_gl_flow_config_widget.cpp b/custom_widgets/yacreader_gl_flow_config_widget.cpp new file mode 100644 index 00000000..35e09a2c --- /dev/null +++ b/custom_widgets/yacreader_gl_flow_config_widget.cpp @@ -0,0 +1,240 @@ +#include "yacreader_gl_flow_config_widget.h" + +#include "yacreader_spin_slider_widget.h" +#include "yacreader_flow_gl.h" //TODO + +#include +#include +#include +#include +#include + + +YACReaderGLFlowConfigWidget::YACReaderGLFlowConfigWidget(QWidget * parent /* = 0 */) + :QWidget(parent) +{ + QVBoxLayout * layout = new QVBoxLayout(this); + + //PRESETS------------------------------------------------------------------ + QGroupBox *groupBox = new QGroupBox(tr("Presets:")); + + radioClassic = new QRadioButton(tr("Classic look")); + //connect(radioClassic,SIGNAL(toggled(bool)),this,SLOT(setClassicConfig())); + + radioStripe = new QRadioButton(tr("Stripe look")); + //connect(radioStripe,SIGNAL(toggled(bool)),this,SLOT(setStripeConfig())); + + radioOver = new QRadioButton(tr("Overlapped Stripe look")); + //connect(radioOver,SIGNAL(toggled(bool)),this,SLOT(setOverlappedStripeConfig())); + + radionModern = new QRadioButton(tr("Modern look")); + //connect(radionModern,SIGNAL(toggled(bool)),this,SLOT(setModernConfig())); + + radioDown = new QRadioButton(tr("Roulette look")); + //connect(radioDown,SIGNAL(toggled(bool)),this,SLOT(setRouletteConfig())); + + QVBoxLayout *vbox = new QVBoxLayout; + QHBoxLayout * opt1 = new QHBoxLayout; + opt1->addWidget(radioClassic); + QLabel * lOpt1 = new QLabel(); + lOpt1->setPixmap(QPixmap(":/images/flow1.png")); + opt1->addStretch(); + opt1->addWidget(lOpt1); + vbox->addLayout(opt1); + + QHBoxLayout * opt2 = new QHBoxLayout; + opt2->addWidget(radioStripe); + QLabel * lOpt2 = new QLabel(); + lOpt2->setPixmap(QPixmap(":/images/flow2.png")); + opt2->addStretch(); + opt2->addWidget(lOpt2); + vbox->addLayout(opt2); + + QHBoxLayout * opt3 = new QHBoxLayout; + opt3->addWidget(radioOver); + QLabel * lOpt3 = new QLabel(); + lOpt3->setPixmap(QPixmap(":/images/flow3.png")); + opt3->addStretch(); + opt3->addWidget(lOpt3); + vbox->addLayout(opt3); + + QHBoxLayout * opt4 = new QHBoxLayout; + opt4->addWidget(radionModern); + QLabel * lOpt4 = new QLabel(); + lOpt4->setPixmap(QPixmap(":/images/flow4.png")); + opt4->addStretch(); + opt4->addWidget(lOpt4); + vbox->addLayout(opt4); + + QHBoxLayout * opt5 = new QHBoxLayout; + opt5->addWidget(radioDown); + QLabel * lOpt5 = new QLabel(); + lOpt5->setPixmap(QPixmap(":/images/flow5.png")); + opt5->addStretch(); + opt5->addWidget(lOpt5); + vbox->addLayout(opt5); + + showAdvancedOptions = new QPushButton(tr("Show advanced settings")); + showAdvancedOptions->setCheckable(true); + connect(showAdvancedOptions,SIGNAL(toggled(bool)),this,SLOT(avancedOptionToogled(bool))); + + vbox->addWidget(showAdvancedOptions,0,Qt::AlignRight); + + groupBox->setLayout(vbox); + + //OPTIONS------------------------------------------------------------------ + optionsGroupBox = new QGroupBox(tr("Custom:")); + + xRotation = new YACReaderSpinSliderWidget(this); + xRotation->setText(tr("View angle")); + xRotation->setRange(0,90); + //connect(xRotation,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(xRotation,SIGNAL(valueChanged(int)),this,SLOT(saveXRotation(int))); + + yPosition = new YACReaderSpinSliderWidget(this); + yPosition->setText(tr("Position")); + yPosition->setRange(-100,100); + //connect(yPosition,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(yPosition,SIGNAL(valueChanged(int)),this,SLOT(saveYPosition(int))); + + coverDistance = new YACReaderSpinSliderWidget(this); + coverDistance->setText(tr("Cover gap")); + coverDistance->setRange(0,150); + //connect(coverDistance,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(coverDistance,SIGNAL(valueChanged(int)),this,SLOT(saveCoverDistance(int))); + + centralDistance = new YACReaderSpinSliderWidget(this); + centralDistance->setText(tr("Central gap")); + centralDistance->setRange(0,150); + //connect(centralDistance,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(centralDistance,SIGNAL(valueChanged(int)),this,SLOT(saveCentralDistance(int))); + + zoomLevel = new YACReaderSpinSliderWidget(this); + zoomLevel->setText(tr("Zoom")); + zoomLevel->setRange(-20,0); + //connect(zoomLevel,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(zoomLevel,SIGNAL(valueChanged(int)),this,SLOT(saveZoomLevel(int))); + + yCoverOffset = new YACReaderSpinSliderWidget(this); + yCoverOffset->setText(tr("Y offset")); + yCoverOffset->setRange(-50,50); + //connect(yCoverOffset,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(yCoverOffset,SIGNAL(valueChanged(int)),this,SLOT(saveYCoverOffset(int))); + + zCoverOffset = new YACReaderSpinSliderWidget(this); + zCoverOffset->setText(tr("Z offset")); + zCoverOffset->setRange(-50,50); + //connect(zCoverOffset,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(zCoverOffset,SIGNAL(valueChanged(int)),this,SLOT(saveZCoverOffset(int))); + + coverRotation = new YACReaderSpinSliderWidget(this); + coverRotation->setText(tr("Cover Angle")); + coverRotation->setRange(0,360); + //connect(coverRotation,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(coverRotation,SIGNAL(valueChanged(int)),this,SLOT(saveCoverRotation(int))); + + fadeOutDist = new YACReaderSpinSliderWidget(this); + fadeOutDist->setText(tr("Visibility")); + fadeOutDist->setRange(0,10); + //connect(fadeOutDist,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(fadeOutDist,SIGNAL(valueChanged(int)),this,SLOT(saveFadeOutDist(int))); + + lightStrength = new YACReaderSpinSliderWidget(this); + lightStrength->setText(tr("Light")); + lightStrength->setRange(0,10); + //connect(lightStrength,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(lightStrength,SIGNAL(valueChanged(int)),this,SLOT(saveLightStrength(int))); + + maxAngle = new YACReaderSpinSliderWidget(this); + maxAngle->setText(tr("Max angle")); + maxAngle->setRange(0,90); + //connect(maxAngle,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + //connect(maxAngle,SIGNAL(valueChanged(int)),this,SLOT(saveMaxAngle(int))); + + QVBoxLayout *optionsLayoutStretch = new QVBoxLayout; + optionsLayoutStretch->setContentsMargins(0,0,0,0); + QGridLayout *optionsLayout = new QGridLayout; + optionsLayout->addWidget(xRotation,0,0); + optionsLayout->addWidget(yPosition,0,1); + optionsLayout->addWidget(coverDistance,1,0); + optionsLayout->addWidget(centralDistance,1,1); + optionsLayout->addWidget(zoomLevel,2,0); + optionsLayout->addWidget(yCoverOffset,2,1); + optionsLayout->addWidget(zCoverOffset,3,0); + optionsLayout->addWidget(coverRotation,3,1); + optionsLayout->addWidget(fadeOutDist,4,0); + optionsLayout->addWidget(lightStrength,4,1); + optionsLayout->addWidget(maxAngle,5,0); + + optionsLayoutStretch->addLayout(optionsLayout); + optionsLayoutStretch->addStretch(); + + optionsGroupBox->setLayout(optionsLayoutStretch); + + QHBoxLayout * groupBoxesLayout = new QHBoxLayout; + groupBoxesLayout->addWidget(groupBox); + groupBoxesLayout->addWidget(optionsGroupBox); + + optionsGroupBox->hide(); + + QHBoxLayout * performanceSliderLayout = new QHBoxLayout; + performanceSliderLayout->addWidget(new QLabel(tr("Low Performance"))); + performanceSliderLayout->addWidget(performanceSlider = new QSlider(Qt::Horizontal)); + performanceSliderLayout->addWidget(new QLabel(tr("High Performance"))); + + performanceSlider->setMinimum(0); + performanceSlider->setMaximum(3); + performanceSlider->setSingleStep(1); + performanceSlider->setPageStep(1); + performanceSlider->setTickInterval(1); + performanceSlider->setTickPosition(QSlider::TicksRight); + + QHBoxLayout * vSyncLayout = new QHBoxLayout; + + vSyncCheck = new QCheckBox(tr("Use VSync (improve the image quality in fullscreen mode, worse performance)")); + vSyncLayout->addStretch(); + vSyncLayout->addWidget(vSyncCheck); + + QVBoxLayout * performanceLayout = new QVBoxLayout; + performanceLayout->addLayout(performanceSliderLayout); + performanceLayout->addLayout(vSyncLayout); + + QGroupBox *performanceGroupBox = new QGroupBox(tr("Performance:")); + + //connect(performanceSlider, SIGNAL(valueChanged(int)),this,SLOT(savePerformance(int))); + //connect(performanceSlider, SIGNAL(valueChanged(int)),this,SLOT(optionsChanged())); + + performanceGroupBox->setLayout(performanceLayout); + + layout->addLayout(groupBoxesLayout); + layout->addWidget(performanceGroupBox); + + layout->setContentsMargins(0,0,0,0); + + setLayout(layout); + + +} + +void YACReaderGLFlowConfigWidget::avancedOptionToogled(bool show) +{ + if(show) + optionsGroupBox->show(); + else + optionsGroupBox->hide(); +} + +void YACReaderGLFlowConfigWidget::setValues(Preset preset) +{ + xRotation->setValue(preset.cfRX); + yPosition->setValue(preset.cfY*100); + coverDistance->setValue(preset.xDistance*100); + centralDistance->setValue(preset.centerDistance*100); + zoomLevel->setValue(preset.cfZ); + yCoverOffset->setValue(preset.yDistance*100); + zCoverOffset->setValue(preset.zDistance*100); + coverRotation->setValue(preset.rotation*-1); + fadeOutDist->setValue(preset.animationFadeOutDist); + lightStrength->setValue(preset.viewRotateLightStrenght); + maxAngle->setValue(preset.viewAngle); +} diff --git a/custom_widgets/yacreader_gl_flow_config_widget.h b/custom_widgets/yacreader_gl_flow_config_widget.h new file mode 100644 index 00000000..83ded28d --- /dev/null +++ b/custom_widgets/yacreader_gl_flow_config_widget.h @@ -0,0 +1,51 @@ +#ifndef YACREADER_GL_FLOW_CONFIG_WIDGET_H +#define YACREADER_GL_FLOW_CONFIG_WIDGET_H + +#include "yacreader_flow_gl.h" //TODO +#include + +class QRadioButton; +class YACReaderSpinSliderWidget; +class QSlider; +class QCheckBox; +class QPushButton; +class QGroupBox; + +class YACReaderGLFlowConfigWidget : public QWidget +{ + Q_OBJECT +public: + YACReaderGLFlowConfigWidget(QWidget * parent = 0); + + //GL......................... + QRadioButton *radioClassic; + QRadioButton *radioStripe; + QRadioButton *radioOver; + QRadioButton *radionModern; + QRadioButton *radioDown; + + YACReaderSpinSliderWidget * xRotation; + YACReaderSpinSliderWidget * yPosition; + YACReaderSpinSliderWidget * coverDistance; + YACReaderSpinSliderWidget * centralDistance; + YACReaderSpinSliderWidget * zoomLevel; + YACReaderSpinSliderWidget * yCoverOffset; + YACReaderSpinSliderWidget * zCoverOffset; + YACReaderSpinSliderWidget * coverRotation; + YACReaderSpinSliderWidget * fadeOutDist; + YACReaderSpinSliderWidget * lightStrength; + YACReaderSpinSliderWidget * maxAngle; + + QSlider * performanceSlider; + QCheckBox * vSyncCheck; + + QPushButton * showAdvancedOptions; + QGroupBox *optionsGroupBox; + +public slots: + void setValues(Preset preset); + void avancedOptionToogled(bool show); +}; + + +#endif // YACREADER_GL_FLOW_CONFIG_WIDGET_H \ No newline at end of file diff --git a/custom_widgets/yacreader_library_item_widget.cpp b/custom_widgets/yacreader_library_item_widget.cpp new file mode 100644 index 00000000..245b2a2f --- /dev/null +++ b/custom_widgets/yacreader_library_item_widget.cpp @@ -0,0 +1,155 @@ +#include "yacreader_library_item_widget.h" + +#include +#include +#include +#include + +YACReaderLibraryItemWidget::YACReaderLibraryItemWidget(QString n/*ame*/, QString p/*ath*/, QWidget *parent) : + QWidget(parent),name(n),path(p),isSelected(false) +{ + QHBoxLayout * mainLayout = new QHBoxLayout; + mainLayout->setMargin(0); + mainLayout->setSpacing(0); + + //installEventFilter(this); + + QPixmap iconPixmap(":/images/libraryIcon.png"); + icon = new QLabel(this); + icon->setPixmap(iconPixmap); + + nameLabel = new QLabel(name,this); + + options = new QToolButton(this); + options->setIcon(QIcon(":/images/libraryOptions.png")); + options->setHidden(true); + options->setFixedWidth(18); + options->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Minimum); + options->setStyleSheet("QToolButton {border:none;}"); + connect(options,SIGNAL(clicked()),this,SIGNAL(showOptions())); + /*up = new QToolButton(this); + up->setIcon(QIcon(":/images/libraryUp.png")); + up->setHidden(true); + up->setFixedWidth(18); + up->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Minimum); + + down = new QToolButton(this); + down->setIcon(QIcon(":/images/libraryDown.png")); + down->setHidden(true); + down->setFixedWidth(18); + down->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Minimum);*/ + + + mainLayout->addWidget(icon); + mainLayout->addWidget(nameLabel,Qt::AlignLeft); + mainLayout->addStretch(); + mainLayout->addWidget(options); + /*mainLayout->addWidget(up); + mainLayout->addWidget(down);*/ + + setLayout(mainLayout); +#ifndef Q_OS_MAC + QString styleSheet = "background-color:transparent; color:#DDDFDF;"; + setStyleSheet(styleSheet); +#endif + + + QString iconStyleSheet = "QLabel {padding:0 0 0 24px; margin:0px}"; + icon->setStyleSheet(iconStyleSheet); + + QString nameLabelStyleSheet = "QLabel {padding:0 0 0 3px; margin:0px;}"; + nameLabel->setStyleSheet(nameLabelStyleSheet); + + setMinimumHeight(20); +} + +void YACReaderLibraryItemWidget::showUpDownButtons(bool show) +{ + up->setHidden(!show); + down->setHidden(!show); +} + +/* +bool YACReaderLibraryItemWidget::eventFilter(QObject *object, QEvent *event){ + if(!isSelected && object==this && (event->type()==QEvent::Enter)) + { + QString styleSheet = "background-color:#5E5E5E; border-top: 1px solid #5E5E5E;border-bottom: 1px solid #5E5E5E; "; + setStyleSheet(styleSheet); + + up->setHidden(false); + down->setHidden(false); + options->setHidden(false); + + return true; + } + if(!isSelected && object==this && (event->type()==QEvent::Leave)) + { + QString styleSheet = "background-color:#454545; border-top: 1px solid #454545;border-bottom: 1px solid #454545;"; + setStyleSheet(styleSheet); + + up->setHidden(true); + down->setHidden(true); + options->setHidden(true); + + return true; + } + + if(object==this && (event->type()==QEvent::MouseButtonRelease)) + { + QString styleSheet = "background-color:#2E2E2E; border-top: 1px solid #1F1F1F;border-bottom: 1px solid #636363; padding-top:1px; padding-bottom:1px;"; + setStyleSheet(styleSheet); + emit(selected(name,path)); + isSelected = true; + return true; + } + + return false; +}*/ + + + +void YACReaderLibraryItemWidget::deselect() +{ + +#ifdef Q_OS_MAC + QString styleSheet = "background-color:transparent;"; + setStyleSheet(styleSheet); +#else + QString styleSheet = "background-color:transparent; color:#DDDFDF;"; + setStyleSheet(styleSheet); +#endif + + QPixmap iconPixmap(":/images/libraryIcon.png"); + icon->setPixmap(iconPixmap); + + /*up->setHidden(true); + down->setHidden(true);*/ + options->setHidden(true); + + isSelected = false; + + +} + +void YACReaderLibraryItemWidget::select() +{ +#ifdef Q_OS_MAC + QString styleSheet ="color: white; background-color:qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #6BAFE4, stop: 1 #3984D2); border-top: 2px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #5EA3DF, stop: 1 #73B8EA); border-left:none;border-right:none;border-bottom:1px solid #3577C2;"; +#else + QString styleSheet = "color: white; background-color:#2E2E2E; font-weight:bold;"; +#endif + setStyleSheet(styleSheet); + + options->setHidden(false); + + QPixmap iconPixmap(":/images/libraryIconSelected.png"); + icon->setPixmap(iconPixmap); + + isSelected = true; +} + +void YACReaderLibraryItemWidget::setName(const QString & name) +{ + this->name = name; + nameLabel->setText(name); +} diff --git a/custom_widgets/yacreader_library_item_widget.h b/custom_widgets/yacreader_library_item_widget.h new file mode 100644 index 00000000..74d90224 --- /dev/null +++ b/custom_widgets/yacreader_library_item_widget.h @@ -0,0 +1,45 @@ +#ifndef YACREADER_LIBRARY_ITEM_WIDGET_H +#define YACREADER_LIBRARY_ITEM_WIDGET_H + +#include + +class QLabel; +class QToolButton; +class QMouseEvent; +class QEvent; + +class YACReaderLibraryItemWidget : public QWidget +{ + Q_OBJECT + +public: + YACReaderLibraryItemWidget(QString name, QString path, QWidget *parent = 0); + QString name; + QString path; + +signals: + void selected(QString,QString); + void showOptions(); + +public slots: + void showUpDownButtons(bool show); + + //bool eventFilter(QObject *object, QEvent *event); + void select(); + void deselect(); + void setName(const QString & name); + +private: + + QLabel * icon; + QLabel * nameLabel; + + QToolButton * options; + QToolButton * up; + QToolButton * down; + + bool isSelected; + +}; + +#endif // YACREADER_LIBRARY_ITEM_WIDGET_H diff --git a/custom_widgets/yacreader_library_list_widget.cpp b/custom_widgets/yacreader_library_list_widget.cpp new file mode 100644 index 00000000..6e5cc676 --- /dev/null +++ b/custom_widgets/yacreader_library_list_widget.cpp @@ -0,0 +1,128 @@ +#include "yacreader_library_list_widget.h" + +#include "yacreader_library_item_widget.h" +#include +#include +#include +#include "qnaturalsorting.h" + +YACReaderLibraryListWidget::YACReaderLibraryListWidget(QWidget *parent) : + QWidget(parent),currentLibraryIndex(-1) +{ + QVBoxLayout * mainLayout = new QVBoxLayout; + mainLayout->setSpacing(0); + mainLayout->setMargin(0); + + this->setLayout(mainLayout); +} + +void YACReaderLibraryListWidget::addItem(QString name, QString path) +{ + QVBoxLayout * mainLayout = dynamic_cast(layout()); + + YACReaderLibraryItemWidget * library = new YACReaderLibraryItemWidget(name,path,this); + connect(library,SIGNAL(showOptions()),this,SLOT(showContextMenu())); + QList::iterator itr; + int i = 0; + for(itr = librariesList.begin(); itr!=librariesList.end() && !naturalSortLessThanCI(name,(*itr)->name);itr++) + i++; + + librariesList.insert(itr,library); + + //connect(library,SIGNAL(selected(QString,QString)),this,SIGNAL(librarySelected(QString,QString))); + //connect(library,SIGNAL(selected(QString,QString)),this,SLOT(updateLibraries(QString,QString))); + + mainLayout->insertWidget(i,library); +} + +QString YACReaderLibraryListWidget::currentText() +{ + return librariesList.at(currentLibraryIndex)->name; +} +int YACReaderLibraryListWidget::findText(QString text) +{ + for(int i=0;iname == text) + return i; + } + return -1; +} +void YACReaderLibraryListWidget::setCurrentIndex(int index) +{ + if(index>=0 && index < librariesList.count()) + { + librariesList.at(index)->select(); + currentLibraryIndex = index; + deselectAllBut(index); + emit currentIndexChanged(librariesList.at(currentLibraryIndex)->name); + } +} + +int YACReaderLibraryListWidget::currentIndex() +{ + return currentLibraryIndex; +} +void YACReaderLibraryListWidget::removeItem(int index) +{ + YACReaderLibraryItemWidget * itemWidget = librariesList.at(index); + this->layout()->removeWidget(itemWidget); + librariesList.removeAt(index); + if(librariesList.count()>0) + { + setCurrentIndex(0); + } + delete itemWidget; +} + +void YACReaderLibraryListWidget::mousePressEvent ( QMouseEvent * event ) +{ + if(librariesList.count()>0) + { + int h = librariesList.at(0)->height(); + int item = event->pos().y() / h; + if(item!=currentLibraryIndex) + { + setCurrentIndex(item); + } + } + +} + +void YACReaderLibraryListWidget::deselectAllBut(int index) +{ + for(int i=0;ideselect(); + } +} + +void YACReaderLibraryListWidget::showContextMenu() +{ + YACReaderLibraryItemWidget * itemWidget = librariesList.at(currentLibraryIndex); + QMenu::exec(actions(),itemWidget->mapToGlobal(QPoint(itemWidget->width()-8,itemWidget->height()/2))); +} + +void YACReaderLibraryListWidget::renameCurrentLibrary(QString newName) +{ + YACReaderLibraryItemWidget * itemWidget = librariesList.at(currentLibraryIndex); + + + this->layout()->removeWidget(itemWidget); + librariesList.removeOne(itemWidget); + + itemWidget->setName(newName); + + QList::iterator itr; + int i = 0; + for(itr = librariesList.begin(); itr!=librariesList.end() && !naturalSortLessThanCI(newName,(*itr)->name);itr++) + i++; + + librariesList.insert(itr,itemWidget); + + QVBoxLayout * mainLayout = dynamic_cast(layout()); + mainLayout->insertWidget(i,itemWidget); + + currentLibraryIndex = i; +} diff --git a/custom_widgets/yacreader_library_list_widget.h b/custom_widgets/yacreader_library_list_widget.h new file mode 100644 index 00000000..189dee1d --- /dev/null +++ b/custom_widgets/yacreader_library_list_widget.h @@ -0,0 +1,37 @@ +#ifndef YACREADER_LIBRARY_LIST_WIDGET_H +#define YACREADER_LIBRARY_LIST_WIDGET_H + +#include + +class YACReaderLibraryItemWidget; +class QMouseEvent; + +class YACReaderLibraryListWidget : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderLibraryListWidget(QWidget *parent = 0); + +signals: + void currentIndexChanged(QString text); + +public slots: + QString currentText(); + int findText(QString text); + void setCurrentIndex(int index); + void addItem(QString name, QString path); + int currentIndex(); + void removeItem(int index); + void showContextMenu(); + void renameCurrentLibrary(QString newName); +protected: + void mousePressEvent ( QMouseEvent * event ); +private: + int currentLibraryIndex; + QList < YACReaderLibraryItemWidget* > librariesList; + void deselectAllBut(int index); + +}; + +#endif // YACREADER_LIBRARY_LIST_WIDGET_H + diff --git a/custom_widgets/yacreader_options_dialog.cpp b/custom_widgets/yacreader_options_dialog.cpp new file mode 100644 index 00000000..c89b44dc --- /dev/null +++ b/custom_widgets/yacreader_options_dialog.cpp @@ -0,0 +1,383 @@ +#include "yacreader_options_dialog.h" + +#include "yacreader_flow_config_widget.h" +#include "yacreader_gl_flow_config_widget.h" +#include "yacreader_spin_slider_widget.h" +#include "yacreader_global.h" + +#include +#include +#include +#include +#include +#include + +YACReaderOptionsDialog::YACReaderOptionsDialog(QWidget * parent) + :QDialog(parent) +{ + + sw = new YACReaderFlowConfigWidget(this); + gl = new YACReaderGLFlowConfigWidget(this); + + accept = new QPushButton(tr("Save")); + cancel = new QPushButton(tr("Cancel")); + + cancel->setDefault(true); + + + QVBoxLayout * shortcutsLayout = new QVBoxLayout(); + QPushButton * shortcutsButton = new QPushButton(tr("Edit shortcuts")); + shortcutsLayout->addWidget(shortcutsButton); + + shortcutsBox = new QGroupBox(tr("Shortcuts")); + shortcutsBox->setLayout(shortcutsLayout); + + connect(shortcutsButton,SIGNAL(clicked()),this,SIGNAL(editShortcuts())); + + connect(accept,SIGNAL(clicked()),this,SLOT(saveOptions())); + connect(cancel,SIGNAL(clicked()),this,SLOT(restoreOptions())); //TODO fix this + connect(cancel,SIGNAL(clicked()),this,SLOT(close())); + + useGL = new QCheckBox(tr("Use hardware acceleration (restart needed)")); + connect(useGL,SIGNAL(stateChanged(int)),this,SLOT(saveUseGL(int))); + + //sw CONNECTIONS + connect(sw->radio1,SIGNAL(toggled(bool)),this,SLOT(setClassicConfigSW())); + connect(sw->radio2,SIGNAL(toggled(bool)),this,SLOT(setStripeConfigSW())); + connect(sw->radio3,SIGNAL(toggled(bool)),this,SLOT(setOverlappedStripeConfigSW())); + + //gl CONNECTIONS + connect(gl->radioClassic,SIGNAL(toggled(bool)),this,SLOT(setClassicConfig())); + connect(gl->radioStripe,SIGNAL(toggled(bool)),this,SLOT(setStripeConfig())); + connect(gl->radioOver,SIGNAL(toggled(bool)),this,SLOT(setOverlappedStripeConfig())); + connect(gl->radionModern,SIGNAL(toggled(bool)),this,SLOT(setModernConfig())); + connect(gl->radioDown,SIGNAL(toggled(bool)),this,SLOT(setRouletteConfig())); + + connect(gl->radioClassic,SIGNAL(toggled(bool)),this,SIGNAL(optionsChanged())); + connect(gl->radioStripe,SIGNAL(toggled(bool)),this,SIGNAL(optionsChanged())); + connect(gl->radioOver,SIGNAL(toggled(bool)),this,SIGNAL(optionsChanged())); + connect(gl->radionModern,SIGNAL(toggled(bool)),this,SIGNAL(optionsChanged())); + connect(gl->radioDown,SIGNAL(toggled(bool)),this,SIGNAL(optionsChanged())); + + connect(gl->xRotation,SIGNAL(valueChanged(int)),this,SLOT(saveXRotation(int))); + connect(gl->xRotation,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->yPosition,SIGNAL(valueChanged(int)),this,SLOT(saveYPosition(int))); + connect(gl->yPosition,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->coverDistance,SIGNAL(valueChanged(int)),this,SLOT(saveCoverDistance(int))); + connect(gl->coverDistance,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->centralDistance,SIGNAL(valueChanged(int)),this,SLOT(saveCentralDistance(int))); + connect(gl->centralDistance,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->zoomLevel,SIGNAL(valueChanged(int)),this,SLOT(saveZoomLevel(int))); + connect(gl->zoomLevel,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->yCoverOffset,SIGNAL(valueChanged(int)),this,SLOT(saveYCoverOffset(int))); + connect(gl->yCoverOffset,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->zCoverOffset,SIGNAL(valueChanged(int)),this,SLOT(saveZCoverOffset(int))); + connect(gl->zCoverOffset,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->coverRotation,SIGNAL(valueChanged(int)),this,SLOT(saveCoverRotation(int))); + connect(gl->coverRotation,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->fadeOutDist,SIGNAL(valueChanged(int)),this,SLOT(saveFadeOutDist(int))); + connect(gl->fadeOutDist,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->lightStrength,SIGNAL(valueChanged(int)),this,SLOT(saveLightStrength(int))); + connect(gl->lightStrength,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->maxAngle,SIGNAL(valueChanged(int)),this,SLOT(saveMaxAngle(int))); + connect(gl->maxAngle,SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->performanceSlider, SIGNAL(valueChanged(int)),this,SLOT(savePerformance(int))); + connect(gl->performanceSlider, SIGNAL(valueChanged(int)),this,SIGNAL(optionsChanged())); + + connect(gl->vSyncCheck,SIGNAL(stateChanged(int)),this,SLOT(saveUseVSync(int))); +} + +void YACReaderOptionsDialog::savePerformance(int value) +{ + settings->setValue(PERFORMANCE,value); +} + +void YACReaderOptionsDialog::saveUseVSync(int b) +{ + settings->setValue(V_SYNC,b); +} + +void YACReaderOptionsDialog::saveFlowParameters() +{ + settings->setValue(X_ROTATION,gl->xRotation->getValue()); + settings->setValue(Y_POSITION,gl->yPosition->getValue()); + settings->setValue(COVER_DISTANCE,gl->coverDistance->getValue()); + settings->setValue(CENTRAL_DISTANCE,gl->centralDistance->getValue()); + settings->setValue(ZOOM_LEVEL,gl->zoomLevel->getValue()); + settings->setValue(Y_COVER_OFFSET,gl->yCoverOffset->getValue()); + settings->setValue(Z_COVER_OFFSET,gl->zCoverOffset->getValue()); + settings->setValue(COVER_ROTATION,gl->coverRotation->getValue()); + settings->setValue(FADE_OUT_DIST,gl->fadeOutDist->getValue()); + settings->setValue(LIGHT_STRENGTH,gl->lightStrength->getValue()); + settings->setValue(MAX_ANGLE,gl->maxAngle->getValue()); +} + +void YACReaderOptionsDialog::saveOptions() +{ + emit(optionsChanged()); + close(); +} + +void YACReaderOptionsDialog::saveUseGL(int b) +{ + if(Qt::Checked == b) + { + sw->setVisible(false); + gl->setVisible(true); + } + else + { + gl->setVisible(false); + sw->setVisible(true); + } + resize(0,0); + + settings->setValue(USE_OPEN_GL,b); + +} + +void YACReaderOptionsDialog::saveXRotation(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(X_ROTATION,value); +} +void YACReaderOptionsDialog::saveYPosition(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(Y_POSITION,value); +} +void YACReaderOptionsDialog::saveCoverDistance(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(COVER_DISTANCE,value); +} +void YACReaderOptionsDialog::saveCentralDistance(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(CENTRAL_DISTANCE,value); +} +void YACReaderOptionsDialog::saveZoomLevel(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(ZOOM_LEVEL,value); +} +void YACReaderOptionsDialog::saveYCoverOffset(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(Y_COVER_OFFSET,value); +} +void YACReaderOptionsDialog::saveZCoverOffset(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(Z_COVER_OFFSET,value); +} +void YACReaderOptionsDialog::saveCoverRotation(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(COVER_ROTATION,value); +} +void YACReaderOptionsDialog::saveFadeOutDist(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(FADE_OUT_DIST,value); +} +void YACReaderOptionsDialog::saveLightStrength(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(LIGHT_STRENGTH,value); +} + +void YACReaderOptionsDialog::saveMaxAngle(int value) +{ + settings->setValue(FLOW_TYPE_GL,Custom); + settings->setValue(MAX_ANGLE,value); +} + +void YACReaderOptionsDialog::restoreOptions(QSettings * settings) +{ + this->settings = settings; + + //FLOW CONFIG + + if(settings->contains(USE_OPEN_GL) && settings->value(USE_OPEN_GL).toInt() == Qt::Checked) + { + sw->setVisible(false); + gl->setVisible(true); + useGL->setChecked(true); + } + else + { + gl->setVisible(false); + sw->setVisible(true); + useGL->setChecked(false); + } + + + if(!settings->contains(FLOW_TYPE_GL)) + { + setClassicConfig(); + gl->radioClassic->setChecked(true); + gl->performanceSlider->setValue(1); + return; + } + + if(settings->contains(V_SYNC) && settings->value(V_SYNC).toInt() == Qt::Checked) + gl->vSyncCheck->setChecked(true); + else + gl->vSyncCheck->setChecked(false); + + gl->performanceSlider->setValue(settings->value(PERFORMANCE).toInt()); + + FlowType flowType; + switch(settings->value(FLOW_TYPE_GL).toInt()) + { + case 0: + flowType = CoverFlowLike; + break; + case 1: + flowType = Strip; + break; + case 2: + flowType = StripOverlapped; + break; + case 3: + flowType = Modern; + break; + case 4: + flowType = Roulette; + break; + case 5: + flowType = Custom; + break; + } + + + if(flowType == Custom) + { + loadConfig(); + return; + } + + if(flowType == CoverFlowLike) + { + setClassicConfig(); + gl->radioClassic->setChecked(true); + return; + } + + if(flowType == Strip) + { + setStripeConfig(); + gl->radioStripe->setChecked(true); + return; + } + + if(flowType == StripOverlapped) + { + setOverlappedStripeConfig(); + gl->radioOver->setChecked(true); + return; + } + + if(flowType == Modern) + { + setModernConfig(); + gl->radionModern->setChecked(true); + return; + } + + if(flowType == Roulette) + { + setRouletteConfig(); + gl->radioDown->setChecked(true); + return; + } + + //END FLOW CONFIG +} + +void YACReaderOptionsDialog::loadConfig() +{ + gl->xRotation->setValue(settings->value(X_ROTATION).toInt()); + gl->yPosition->setValue(settings->value(Y_POSITION).toInt()); + gl->coverDistance->setValue(settings->value(COVER_DISTANCE).toInt()); + gl->centralDistance->setValue(settings->value(CENTRAL_DISTANCE).toInt()); + gl->zoomLevel->setValue(settings->value(ZOOM_LEVEL).toInt()); + gl->yCoverOffset->setValue(settings->value(Y_COVER_OFFSET).toInt()); + gl->zCoverOffset->setValue(settings->value(Z_COVER_OFFSET).toInt()); + gl->coverRotation->setValue(settings->value(COVER_ROTATION).toInt()); + gl->fadeOutDist->setValue(settings->value(FADE_OUT_DIST).toInt()); + gl->lightStrength->setValue(settings->value(LIGHT_STRENGTH).toInt()); + gl->maxAngle->setValue(settings->value(MAX_ANGLE).toInt()); +} + +void YACReaderOptionsDialog::setClassicConfigSW() +{ + settings->setValue(FLOW_TYPE_SW,CoverFlowLike); +} + +void YACReaderOptionsDialog::setStripeConfigSW() +{ + settings->setValue(FLOW_TYPE_SW,Strip); +} + +void YACReaderOptionsDialog::setOverlappedStripeConfigSW() +{ + settings->setValue(FLOW_TYPE_SW,StripOverlapped); +} + +void YACReaderOptionsDialog::setClassicConfig() +{ + settings->setValue(FLOW_TYPE_GL,CoverFlowLike); + + gl->setValues(presetYACReaderFlowClassicConfig); + + saveFlowParameters(); +} + +void YACReaderOptionsDialog::setStripeConfig() +{ + settings->setValue(FLOW_TYPE_GL,Strip); + + gl->setValues(presetYACReaderFlowStripeConfig); + + saveFlowParameters(); +} + +void YACReaderOptionsDialog::setOverlappedStripeConfig() +{ + settings->setValue(FLOW_TYPE_GL,StripOverlapped); + + gl->setValues(presetYACReaderFlowOverlappedStripeConfig); + + saveFlowParameters(); +} + +void YACReaderOptionsDialog::setModernConfig() +{ + settings->setValue(FLOW_TYPE_GL,Modern); + + gl->setValues(defaultYACReaderFlowConfig); + + saveFlowParameters(); +} + +void YACReaderOptionsDialog::setRouletteConfig() +{ + settings->setValue(FLOW_TYPE_GL,Roulette); + + gl->setValues(pressetYACReaderFlowDownConfig); + + saveFlowParameters(); +} diff --git a/custom_widgets/yacreader_options_dialog.h b/custom_widgets/yacreader_options_dialog.h new file mode 100644 index 00000000..9b347ae8 --- /dev/null +++ b/custom_widgets/yacreader_options_dialog.h @@ -0,0 +1,65 @@ +#ifndef YACREADER_OPTIONS_DIALOG_H +#define YACREADER_OPTIONS_DIALOG_H + +#include + +class YACReaderFlowConfigWidget; +class YACReaderGLFlowConfigWidget; +class QCheckBox; +class QPushButton; +class QSettings; +class QGroupBox; + +class YACReaderOptionsDialog : public QDialog +{ + Q_OBJECT +protected: + YACReaderFlowConfigWidget * sw; + YACReaderGLFlowConfigWidget * gl; + QCheckBox * useGL; + + QPushButton * accept; + QPushButton * cancel; + + QGroupBox * shortcutsBox; + + QSettings * settings; + QSettings * previousSettings; + +public: + YACReaderOptionsDialog(QWidget * parent); +public slots: + virtual void restoreOptions(QSettings * settings); + virtual void saveOptions(); +protected slots: + virtual void savePerformance(int value); + virtual void saveUseVSync(int b); + virtual void saveUseGL(int b); + virtual void saveXRotation(int value); + virtual void saveYPosition(int value); + virtual void saveCoverDistance(int value); + virtual void saveCentralDistance(int value); + virtual void saveZoomLevel(int value); + virtual void saveYCoverOffset(int value); + virtual void saveZCoverOffset(int value); + virtual void saveCoverRotation(int value); + virtual void saveFadeOutDist(int value); + virtual void saveLightStrength(int value); + virtual void saveMaxAngle(int value); + virtual void loadConfig(); + virtual void setClassicConfig(); + virtual void setStripeConfig(); + virtual void setOverlappedStripeConfig(); + virtual void setModernConfig(); + virtual void setRouletteConfig(); + virtual void setClassicConfigSW(); + virtual void setStripeConfigSW(); + virtual void setOverlappedStripeConfigSW(); + virtual void saveFlowParameters(); + +signals: + void optionsChanged(); + void editShortcuts(); +}; + +#endif // YACREADER_OPTIONS_DIALOG_H diff --git a/custom_widgets/yacreader_search_line_edit.cpp b/custom_widgets/yacreader_search_line_edit.cpp new file mode 100644 index 00000000..26d00c31 --- /dev/null +++ b/custom_widgets/yacreader_search_line_edit.cpp @@ -0,0 +1,55 @@ +#include "yacreader_search_line_edit.h" + +#include +#include +#include + +YACReaderSearchLineEdit::YACReaderSearchLineEdit(QWidget *parent) + : QLineEdit(parent) +{ + clearButton = new QToolButton(this); + searchLabel = new QLabel(this); + + QPixmap pixmap(":/images/clearSearch.png"); + QPixmap pixmapIcon(":/images/iconSearch.png"); + + searchLabel->setStyleSheet("QLabel { border: none; padding: 0px; }"); + searchLabel->setPixmap(pixmapIcon); + + clearButton->setIcon(QIcon(pixmap)); + clearButton->setIconSize(pixmap.size()); + clearButton->setCursor(Qt::ArrowCursor); + clearButton->setStyleSheet("QToolButton { border: none; padding: 0px; }"); + clearButton->hide(); + connect(clearButton, SIGNAL(clicked()), this, SLOT(clear())); + connect(this, SIGNAL(textChanged(const QString&)), this, SLOT(updateCloseButton(const QString&))); + int frameWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth); +#ifdef Q_OS_MAC + setStyleSheet(QString("QLineEdit {border-top:1px solid #9F9F9F; border-bottom:1px solid #ACACAC; border-right:1px solid #ACACAC; border-left:1px solid #ACACAC; border-radius: 10px; background-color:qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #CACACA, stop: 0.15 #FFFFFF); padding-left: %1px; padding-right: %2px; padding-bottom: 1px; margin-bottom: 1px;} ").arg(searchLabel->sizeHint().width() + frameWidth + 6).arg(clearButton->sizeHint().width() + frameWidth + 2)); +#else + setStyleSheet(QString("QLineEdit {border:none; background-color:qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #CACACA, stop: 0.15 #FFFFFF);; padding-left: %1px; padding-right: %2px; padding-bottom: 1px; margin-bottom: 0px;} ").arg(searchLabel->sizeHint().width() + frameWidth + 6).arg(clearButton->sizeHint().width() + frameWidth + 2)); +#endif + QSize msz = minimumSizeHint(); + setMinimumSize(qMax(msz.width(), clearButton->sizeHint().height() + frameWidth * 2 + 2), + qMax(msz.height(), clearButton->sizeHint().height() + frameWidth * 2 + 2)); + +#ifdef Q_OS_MAC + setMaximumWidth(212); +#endif +} + +void YACReaderSearchLineEdit::resizeEvent(QResizeEvent *) +{ + QSize sz = clearButton->sizeHint(); + int frameWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth); + clearButton->move(rect().right() - frameWidth - sz.width(), + (rect().bottom() + 1 - sz.height())/2); + + QSize szl = searchLabel->sizeHint(); + searchLabel->move(6,(rect().bottom() + 1 - szl.height())/2); +} + +void YACReaderSearchLineEdit::updateCloseButton(const QString& text) +{ + clearButton->setVisible(!text.isEmpty()); +} diff --git a/custom_widgets/yacreader_search_line_edit.h b/custom_widgets/yacreader_search_line_edit.h new file mode 100644 index 00000000..a9918594 --- /dev/null +++ b/custom_widgets/yacreader_search_line_edit.h @@ -0,0 +1,29 @@ +#ifndef YACREADER_SEARCH_LINE_EDIT_H +#define YACREADER_SEARCH_LINE_EDIT_H + +#include + +class QToolButton; +class QLabel; + +class YACReaderSearchLineEdit : public QLineEdit +{ + Q_OBJECT + +public: + YACReaderSearchLineEdit(QWidget *parent = 0); + +protected: + void resizeEvent(QResizeEvent *); + +private slots: + void updateCloseButton(const QString &text); + +private: + QToolButton *clearButton; + QLabel * searchLabel; +}; + + + +#endif // YACREADER_SEARCH_LINE_EDIT_H diff --git a/custom_widgets/yacreader_sidebar.cpp b/custom_widgets/yacreader_sidebar.cpp new file mode 100644 index 00000000..cb8c36b7 --- /dev/null +++ b/custom_widgets/yacreader_sidebar.cpp @@ -0,0 +1,130 @@ +#include "yacreader_sidebar.h" + +#include +#include + +#include "yacreader_treeview.h" +#include "yacreader_library_list_widget.h" +#include "yacreader_search_line_edit.h" +#include "yacreader_titled_toolbar.h" + +YACReaderSideBar::YACReaderSideBar(QWidget *parent) : + QWidget(parent) +{ + setSizePolicy(QSizePolicy::Preferred,QSizePolicy::Minimum); + + //widgets + foldersView = new YACReaderTreeView; + selectedLibrary = new YACReaderLibraryListWidget; + foldersFilter = new YACReaderSearchLineEdit(); + + librariesTitle = new YACReaderTitledToolBar(tr("LIBRARIES")); + + foldersTitle = new YACReaderTitledToolBar(tr("FOLDERS")); + + selectedLibrary->setContextMenuPolicy(Qt::ActionsContextMenu); + selectedLibrary->setAttribute(Qt::WA_MacShowFocusRect,false); + selectedLibrary->setFocusPolicy(Qt::NoFocus); + + foldersFilter->setAttribute(Qt::WA_MacShowFocusRect,false); + foldersFilter->setPlaceholderText(tr("Search folders and comics")); + + //layout + QVBoxLayout * l = new QVBoxLayout; + + l->setContentsMargins(0,0,0,0); +#ifndef Q_OS_MAC + l->addSpacing(5); +#endif + + l->addWidget(librariesTitle); + +#ifndef Q_OS_MAC + {QWidget * w = new QWidget(); + w->setStyleSheet("QWidget {border:none; border-bottom:1px solid #636363;border-top:1px solid #292929;}"); + w->setMinimumHeight(2); + + l->addSpacing(4); + + l->addWidget(w);} + + l->addSpacing(3); +#endif + + l->addWidget(selectedLibrary); + +#ifndef Q_OS_MAC + l->addSpacing(6); + + {QWidget * w = new QWidget(); + w->setStyleSheet("QWidget {border:none; border-bottom:1px solid #636363;border-top:1px solid #292929;}"); + w->setMinimumHeight(2); + + l->addSpacing(5); + + l->addWidget(w);} + + l->addSpacing(4); +#else + l->addSpacing(6); +#endif + + l->addWidget(foldersTitle); + +#ifndef Q_OS_MAC + {QWidget * w = new QWidget(); + w->setStyleSheet("QWidget {border:none; border-bottom:1px solid #636363;border-top:1px solid #292929;}"); + w->setMinimumHeight(2); + + l->addSpacing(4); + + l->addWidget(w);} + + + l->addSpacing(4); +#endif + + l->addWidget(foldersView); + + l->addWidget(foldersFilter); + + l->setSpacing(0); + setLayout(l); +} + + +void YACReaderSideBar::paintEvent(QPaintEvent * event) +{ + Q_UNUSED(event) + +#ifdef Q_OS_MAC + QPainter painter(this); + + QLinearGradient lG(0,0,0,height()); + + lG.setColorAt(0,QColor("#E8ECF1")); + lG.setColorAt(1,QColor("#D1D8E0")); + + painter.fillRect(0,0,width(),height(),lG); +#else + QPainter painter(this); + + painter.fillRect(0,0,width(),height(),QColor("#454545")); + //QWidget::paintEvent(event); +#endif + + + + //QPixmap shadow(":/images/side_bar/shadow.png"); + //painter.drawPixmap(width()-shadow.width(),0,shadow.width(),height(),shadow); + + // painter.setRenderHint(QPainter::Antialiasing); + // painter.drawLine(rect().topLeft(), rect().bottomRight()); + + //QWidget::paintEvent(event); +} + +QSize YACReaderSideBar::sizeHint() const +{ + return QSize(275,200); +} diff --git a/custom_widgets/yacreader_sidebar.h b/custom_widgets/yacreader_sidebar.h new file mode 100644 index 00000000..c9c9ff28 --- /dev/null +++ b/custom_widgets/yacreader_sidebar.h @@ -0,0 +1,34 @@ +#ifndef YACREADER_SIDEBAR_H +#define YACREADER_SIDEBAR_H + +#include + +class YACReaderTreeView; +class YACReaderLibraryListWidget; +class YACReaderSearchLineEdit; +class YACReaderTitledToolBar; +class YACReaderTitledToolBar; + +class YACReaderSideBar : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderSideBar(QWidget *parent = 0); + QSize sizeHint() const; + + YACReaderTreeView * foldersView; + YACReaderLibraryListWidget * selectedLibrary; + YACReaderSearchLineEdit * foldersFilter; + YACReaderTitledToolBar * librariesTitle; + YACReaderTitledToolBar * foldersTitle; + +signals: + +public slots: + +protected: + void paintEvent(QPaintEvent *); + +}; + +#endif // YACREADER_SIDEBAR_H diff --git a/custom_widgets/yacreader_social_dialog.cpp b/custom_widgets/yacreader_social_dialog.cpp new file mode 100644 index 00000000..32cc3d7c --- /dev/null +++ b/custom_widgets/yacreader_social_dialog.cpp @@ -0,0 +1,130 @@ +#include "yacreader_social_dialog.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "comic_db.h" + +YACReaderSocialDialog::YACReaderSocialDialog(QWidget *parent) : + QWidget(parent) +{ + + //setWindowFlags(Qt::Window | Qt::Dialog | Qt::FramelessWindowHint); + //setModal(true); + + + QToolButton * close = new QToolButton(this); + close->setIcon(QIcon(":/images/social_dialog/close.png")); + + QToolButton * facebook = new QToolButton(this); + facebook->setIcon(QIcon(":/images/social_dialog/facebook.png")); + + QToolButton * twitter = new QToolButton(this); + twitter->setIcon(QIcon(":/images/social_dialog/twitter.png")); + + QToolButton * google = new QToolButton(this); + google->setIcon(QIcon(":/images/social_dialog/google+.png")); + + QString styleSheet = "QToolButton {border:none; }"; + close->setStyleSheet(styleSheet); + facebook->setStyleSheet(styleSheet); + twitter->setStyleSheet(styleSheet); + google->setStyleSheet(styleSheet); + + QLabel * icon = new QLabel(this); + icon->setPixmap(QPixmap(":/images/social_dialog/icon.png")); + + plainText = new QTextEdit (this); + plainText->setStyleSheet("QTextEdit {border:none; padding:11px; font-size:12px; font-weight:bold; color:#525757;}"); + QTextCursor cursor(plainText->textCursor()); + QTextBlockFormat blockFormat = cursor.blockFormat(); + blockFormat.setLineHeight(12,QTextBlockFormat::SingleHeight); + cursor.setBlockFormat(blockFormat); + QLabel * sendTo = new QLabel(tr("send to:"),this); + sendTo->setStyleSheet("QLabel{color:#ABABAB; font-size:12px; font-weight:bold;}"); + + setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed); + + resize( sizeHint() ); + + close->move(437,5); + + QWidget * send = new QWidget(this); + QHBoxLayout * sendLayout = new QHBoxLayout; + + QPushButton * follow = new QPushButton(tr("Follow YACReader!"),this); + + follow->setStyleSheet("QPushButton{border:none; color:#FFFFFF;background:#404040; padding: 9px 25px 9px 25px; font-weight:bold; font-size:12px;}" + "QPushButton:hover{background:#E3B800;}"); + + sendLayout->setMargin(0); + sendLayout->setSpacing(0); + + sendLayout->addWidget(sendTo,1,Qt::AlignHCenter); + sendLayout->addSpacing(11); + sendLayout->addWidget(facebook,0,Qt::AlignHCenter); + sendLayout->addSpacing(6); + sendLayout->addWidget(twitter,0,Qt::AlignHCenter); + sendLayout->addSpacing(6); + sendLayout->addWidget(google,0,Qt::AlignHCenter); + + send->setLayout(sendLayout); + send->move(317,259); + + icon->move(279,14); + plainText->setFixedSize(291,155); + plainText->move(169,96); + + follow->move(230,307); + + connect(close,SIGNAL(released()),this,SLOT(close())); + + + +} + +void YACReaderSocialDialog::paintEvent(QPaintEvent * event) +{ + QPainter painter(this); + + //center + painter.fillRect(169,0,291,369,QColor("#F0F0F0")); + painter.fillRect(169,96,291,155,QColor("#FFFFFF")); + + + //QPixmap cover = QPixmap("c:/temp/6.jpg").scaledToHeight(369,Qt::SmoothTransformation); + painter.drawPixmap(0,0,169,369,cover,0,0, (169 * cover.height())/369 ,cover.height()); + + + QPixmap shadow(":/images/social_dialog/shadow.png"); + painter.drawPixmap(169-shadow.width(),0,shadow.width(),369,shadow); + + + QPixmap separtor(":/images/social_dialog/separator.png"); + painter.drawPixmap(169,96-separtor.height(),separtor); + + QPen pen("#C3CAD6"); + painter.setPen(pen); + painter.drawLine(169,251,460,251); + + QWidget::paintEvent(event); + +} + +QSize YACReaderSocialDialog::sizeHint() const +{ + return QSize(460,369); +} + +void YACReaderSocialDialog::setComic(ComicDB & comic, QString & basePath) +{ + this->cover = comic.info.getCover(basePath).scaledToHeight(369,Qt::SmoothTransformation); + plainText->setText(tr("I am reading %1 using YACReader.").arg(comic.path.split('/').last())); +} \ No newline at end of file diff --git a/custom_widgets/yacreader_social_dialog.h b/custom_widgets/yacreader_social_dialog.h new file mode 100644 index 00000000..b4340a92 --- /dev/null +++ b/custom_widgets/yacreader_social_dialog.h @@ -0,0 +1,28 @@ +#ifndef YACREADER_SOCIAL_DIALOG_H +#define YACREADER_SOCIAL_DIALOG_H + +#include + +class QPixmap; +class QTextEdit; +class ComicDB; + +class YACReaderSocialDialog : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderSocialDialog(QWidget *parent = 0); + QSize sizeHint() const; +signals: + +public slots: + void setComic(ComicDB & comic,QString & basePath); +protected: + void paintEvent(QPaintEvent *); + +private: + QPixmap cover; + QTextEdit * plainText; +}; + +#endif // YACREADER_SOCIAL_DIALOG_H diff --git a/custom_widgets/yacreader_spin_slider_widget.cpp b/custom_widgets/yacreader_spin_slider_widget.cpp new file mode 100644 index 00000000..56e8a9dd --- /dev/null +++ b/custom_widgets/yacreader_spin_slider_widget.cpp @@ -0,0 +1,93 @@ +#include "yacreader_spin_slider_widget.h" + +#include +#include +#include +#include + +YACReaderSpinSliderWidget::YACReaderSpinSliderWidget(QWidget * parent,bool strechableSlider) + :QWidget(parent),tracking(true) +{ + QHBoxLayout * layout = new QHBoxLayout; + layout->addWidget(label = new QLabel(this),1); + if(!strechableSlider) + layout->addStretch(); + spinBox = new QSpinBox(this); + layout->addWidget(spinBox); + slider = new QSlider(Qt::Horizontal,this); + layout->addWidget(slider); + if(strechableSlider) + { + layout->setStretchFactor(slider,0.85); + layout->setStretchFactor(spinBox,0); + layout->setStretchFactor(label,0.15); + } + + connect(spinBox, SIGNAL(valueChanged(int)), slider, SLOT(setValue(int))); + connect(slider, SIGNAL(valueChanged(int)), spinBox, SLOT(setValue(int))); + + connect(slider, SIGNAL(valueChanged(int)), this, SLOT(valueWillChange(int))); + connect(spinBox, SIGNAL(valueChanged(int)), this, SLOT(valueWillChangeFromSpinBox(int))); + + connect(slider, SIGNAL(sliderReleased()), this, SLOT(sliderRelease())); + + setLayout(layout); +} +void YACReaderSpinSliderWidget::valueWillChange(int v) +{ + Q_UNUSED(v) + if(tracking) + emit valueChanged(spinBox->value()); +} + +void YACReaderSpinSliderWidget::valueWillChangeFromSpinBox(int v) +{ + Q_UNUSED(v) + if(!tracking && !slider->isSliderDown()) + emit valueChanged(spinBox->value()); +} + +void YACReaderSpinSliderWidget::sliderRelease() +{ + if(!tracking) + emit valueChanged(spinBox->value()); +} + +void YACReaderSpinSliderWidget::setRange(int lowValue, int topValue, int step) +{ + spinBox->setMinimum(lowValue); + spinBox->setMaximum(topValue); + spinBox->setSingleStep(step); + + slider->setMinimum(lowValue); + slider->setMaximum(topValue); + slider->setSingleStep(step); +} + +void YACReaderSpinSliderWidget::setValue(int value) +{ + disconnect(spinBox, SIGNAL(valueChanged(int)), this, SLOT(valueWillChange(int))); + spinBox->setValue(value); + connect(spinBox, SIGNAL(valueChanged(int)), this, SLOT(valueWillChange(int))); +} + +void YACReaderSpinSliderWidget::setText(const QString & text) +{ + label->setText(text); +} + +int YACReaderSpinSliderWidget::getValue() +{ + return spinBox->value(); +} + +QSize YACReaderSpinSliderWidget::minimumSizeHint() const +{ + return QSize(270, 25); +} + +void YACReaderSpinSliderWidget::setTracking(bool b) +{ + tracking = b; + //slider->setTracking(b); +} diff --git a/custom_widgets/yacreader_spin_slider_widget.h b/custom_widgets/yacreader_spin_slider_widget.h new file mode 100644 index 00000000..8be271b0 --- /dev/null +++ b/custom_widgets/yacreader_spin_slider_widget.h @@ -0,0 +1,35 @@ +#ifndef YACREADER_SPIN_SLIDER_WIDGET_H +#define YACREADER_SPIN_SLIDER_WIDGET_H + +#include + +class QLabel; +class QSpinBox; +class QSlider; + +class YACReaderSpinSliderWidget : public QWidget +{ + Q_OBJECT +private: + QLabel * label; + QSpinBox * spinBox; + QSlider * slider; + bool tracking; +public: + YACReaderSpinSliderWidget(QWidget * parent = 0,bool strechableSlider = false); +public slots: + void setRange(int lowValue, int topValue, int step=1); + void setValue(int value); + void setText(const QString & text); + int getValue(); + QSize minimumSizeHint() const; + void setTracking(bool b); + void valueWillChange(int); + void valueWillChangeFromSpinBox(int); + void sliderRelease(); +signals: + void valueChanged(int); + +}; + +#endif // YACREADER_SPIN_SLIDER_WIDGET_H \ No newline at end of file diff --git a/custom_widgets/yacreader_table_view.cpp b/custom_widgets/yacreader_table_view.cpp new file mode 100644 index 00000000..11ca1186 --- /dev/null +++ b/custom_widgets/yacreader_table_view.cpp @@ -0,0 +1,422 @@ +#include "yacreader_table_view.h" + +#include +#include +#include +#include + +#include "tableitem.h" + +YACReaderTableView::YACReaderTableView(QWidget *parent) : + QTableView(parent),showDelete(false),editing(false),myeditor(0) +{ + setAlternatingRowColors(true); + verticalHeader()->setAlternatingRowColors(true); + setStyleSheet("QTableView {alternate-background-color: #F2F2F2;background-color: #FAFAFA; outline: 0px;}"// border: 1px solid #999999; border-right:none; border-bottom:none;}" + "QTableCornerButton::section {background-color:#F5F5F5; border:none; border-bottom:1px solid #B8BDC4; border-right:1px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #D1D1D1, stop: 1 #B8BDC4);}" + "QTableView::item {outline: 0px; border-bottom: 1px solid #DFDFDF;border-top: 1px solid #FEFEFE; padding-bottom:1px; color:#252626;}" + "QTableView {border-top:1px solid #B8B8B8;border-bottom:none;border-left:1px solid #B8B8B8;border-right:none;}" +#ifdef Q_OS_MAC + "QTableView {border-top:1px solid #B8B8B8;border-bottom:none;border-left:none;border-right:none;}" + "QTableView::item:selected {outline: 0px; border-bottom: 1px solid #3875D7;border-top: 1px solid #3875D7; padding-bottom:1px; background-color: #3875D7; color: #FFFFFF; }" + +#else + "QTableView::item:selected {outline: 0px; border-bottom: 1px solid #D4D4D4;border-top: 1px solid #D4D4D4; padding-bottom:1px; background-color: #D4D4D4; }" +#endif + "QHeaderView::section:horizontal {background-color:#F5F5F5; border-bottom:1px solid #B8BDC4; border-right:1px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #D1D1D1, stop: 1 #B8BDC4); border-left:none; border-top:none; padding:4px; color:#313232;}" + "QHeaderView::section:vertical {border-bottom: 1px solid #DFDFDF;border-top: 1px solid #FEFEFE;}" + //"QTableView::item:hover {border-bottom: 1px solid #A3A3A3;border-top: 1px solid #A3A3A3; padding-bottom:1px; background-color: #A3A3A3; color: #FFFFFF; }" + ""); + //comicView->setItemDelegate(new YACReaderComicViewDelegate()); + setContextMenuPolicy(Qt::ActionsContextMenu); + + setShowGrid(false); +#if QT_VERSION >= 0x050000 + verticalHeader()->setSectionResizeMode(QHeaderView::Fixed); +#else + verticalHeader()->setResizeMode(QHeaderView::Fixed); +#endif + + //comicView->horizontalHeader()->setResizeMode(QHeaderView::ResizeToContents); + horizontalHeader()->setStretchLastSection(true); +#if QT_VERSION >= 0x050000 + horizontalHeader()->setSectionsClickable(false); +#else + horizontalHeader()->setClickable(false); +#endif + //comicView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents); + verticalHeader()->setDefaultSectionSize(24); +#if QT_VERSION >= 0x050000 + verticalHeader()->setSectionsClickable(false); //TODO comportamiento anómalo +#else + verticalHeader()->setClickable(false); //TODO comportamiento anómalo +#endif + + setCornerButtonEnabled(false); + + setSelectionBehavior(QAbstractItemView::SelectRows); + setSelectionMode(QAbstractItemView::ExtendedSelection); + + setItemDelegateForColumn(11,new YACReaderRatingDelegate(this)); + setEditTriggers(QAbstractItemView::NoEditTriggers); + + setMouseTracking(true); + /*deletingProgress = new YACReaderDeletingProgress(this); + + showDeletingProgressAnimation = new QPropertyAnimation(deletingProgress,"pos"); + showDeletingProgressAnimation->setDuration(150);*/ +} + +void YACReaderTableView::mouseMoveEvent(QMouseEvent *event) +{ + QModelIndex mi = indexAt(event->pos()); + if(mi.isValid()) + { + QList selectedIndexes = this->selectedIndexes(); + if(selectedIndexes.contains(mi)) + { + if(mi.column() == 11) + { + if(!editing) + { + editing = true; + currentIndexEditing = mi; + edit(mi); + myeditor = indexWidget(mi); + } + else if(mi.row() != currentIndexEditing.row()) + closeRatingEditor(); + } + else + closeRatingEditor(); + } + else + closeRatingEditor(); + } + else + closeRatingEditor(); + + QTableView::mouseMoveEvent(event); +} +void YACReaderTableView::mousePressEvent(QMouseEvent * event) +{ + QTableView::mousePressEvent(event); + QModelIndex mi = indexAt(event->pos()); + if(mi.isValid()) + { + QList selectedIndexes = this->selectedIndexes(); + if(selectedIndexes.contains(mi)) + { + if(mi.column() == 11) + { + if(!editing) + { + editing = true; + currentIndexEditing = mi; + edit(mi); + myeditor = indexWidget(mi); + } + } + } + } +} +void YACReaderTableView::leaveEvent(QEvent * event) +{ + closeRatingEditor(); + event->accept(); +} + +void YACReaderTableView::closeRatingEditor() +{ + editing = false; + if(myeditor!=0) + closeEditor(myeditor,QAbstractItemDelegate::NoHint); + myeditor = 0; +} + +void YACReaderTableView::closeEditor ( QWidget * editor, QAbstractItemDelegate::EndEditHint hint ) +{ + editing = false; + myeditor = 0; + QTableView::closeEditor(editor,hint); +} +void YACReaderTableView::commitData ( QWidget * editor ) +{ + //TODO + StarEditor *starEditor = qobject_cast(editor); + if(starEditor->getShouldCommitData()) + emit comicRated(((StarEditor *)editor)->starRating().starCount(),currentIndexEditing); +} + +void YACReaderTableView::showDeleteProgress() +{ + /*showDelete = true; + + showDeletingProgressAnimation->setStartValue(deletingProgress->pos()); + showDeletingProgressAnimation->setEndValue(QPoint((width()-deletingProgress->width())/2 ,1)); + showDeletingProgressAnimation->start();*/ +} + +void YACReaderTableView::hideDeleteProgress() +{ + /*showDelete = false; + + if(showDeletingProgressAnimation->state()==QPropertyAnimation::Running) + showDeletingProgressAnimation->stop(); + + showDeletingProgressAnimation->setStartValue(deletingProgress->pos()); + showDeletingProgressAnimation->setEndValue(QPoint((width()-deletingProgress->width())/2 ,-deletingProgress->height())); + showDeletingProgressAnimation->start();*/ +} + +void YACReaderTableView::resizeEvent(QResizeEvent * event) +{ + /*event->size(); + + if(showDelete) + deletingProgress->move((event->size().width()-deletingProgress->width())/2 ,1); + else + deletingProgress->move((event->size().width()-deletingProgress->width())/2 ,-deletingProgress->height());*/ + + QTableView::resizeEvent(event); +} + +//------------------------------------------------------------------------------ +//YACReaderRatingDelegate------------------------------------------------------- +//------------------------------------------------------------------------------ +void YACReaderRatingDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, + const QModelIndex &index) const +{ + int rating = ((TableItem *)index.internalPointer())->data(11).toInt(); + + StarRating starRating(rating); + + QStyledItemDelegate::paint(painter, option, index); + + if(!(option.state & QStyle::State_Editing)) + { + if (option.state & QStyle::State_Selected) + starRating.paintSelected(painter, option.rect, option.palette, + StarRating::ReadOnly); + else + starRating.paint(painter, option.rect, option.palette, + StarRating::ReadOnly); + } +} + +QSize YACReaderRatingDelegate::sizeHint(const QStyleOptionViewItem &option, + const QModelIndex &index) const +{ + Q_UNUSED(option) + int rating = ((TableItem *)index.internalPointer())->data(11).toInt(); + StarRating starRating(rating); + return starRating.sizeHint(); +} + +QWidget *YACReaderRatingDelegate::createEditor(QWidget *parent, + const QStyleOptionViewItem &option, + const QModelIndex &index) const +{ + Q_UNUSED(option) + Q_UNUSED(index) + StarEditor *editor = new StarEditor(parent); + connect(editor, SIGNAL(editingFinished()), + this, SLOT(sendCloseEditor())); + connect(editor, SIGNAL(commitData()), + this, SLOT(sendCommitData())); + return editor; +} + +void YACReaderRatingDelegate::setEditorData(QWidget *editor, + const QModelIndex &index) const +{ + int rating = ((TableItem *)index.internalPointer())->data(11).toInt(); + + StarRating starRating(rating); + + StarEditor *starEditor = qobject_cast(editor); + starEditor->setStarRating(starRating); +} + +void YACReaderRatingDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, + const QModelIndex &index) const +{ + QStyledItemDelegate::setModelData(editor, model, index); +} + +void YACReaderRatingDelegate::sendCommitData() +{ + StarEditor *editor = qobject_cast(sender()); + emit commitData(editor); +} +void YACReaderRatingDelegate::sendCloseEditor() +{ + StarEditor *editor = qobject_cast(sender()); + emit closeEditor(editor); +} + +//------------------------------------------------------------------------------- +//StarRating--------------------------------------------------------------------- +//------------------------------------------------------------------------------- + +const int PaintingScaleFactor = 20; + +StarRating::StarRating(int starCount, int maxStarCount) +{ + myStarCount = starCount; + myMaxStarCount = maxStarCount; + + int numVertex = 5; + double pi = 3.14159265359; + double angle = 3.14159265359 / numVertex; + + float rOuter = 0.3f; + float rInner = 0.15f; + for (int i = 0; i < 2 * numVertex; i++) + { + double r = (i & 1) == 0 ? rOuter : rInner; + starPolygon << QPointF(0.5 + cos((i * angle)-pi/2) * r, 0.5 + sin((i * angle)-pi/2) * r); + } + + diamondPolygon << QPointF(0.4, 0.5) << QPointF(0.5, 0.4) + << QPointF(0.6, 0.5) << QPointF(0.5, 0.6) + << QPointF(0.4, 0.5); +} + +QSize StarRating::sizeHint() const +{ + return PaintingScaleFactor * QSize(myMaxStarCount, 1); +} + +void StarRating::paint(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode) const +{ + Q_UNUSED(palette) + painter->save(); + + painter->setRenderHint(QPainter::Antialiasing, true); + painter->setPen(Qt::NoPen); + + //if (mode == Editable) { + // painter->setBrush(palette.highlight()); + //} else { + QBrush brush(QColor("#e9be0f")); + painter->setBrush(brush); + //} + + int yOffset = (rect.height() - PaintingScaleFactor) / 2; + painter->translate(rect.x(), rect.y() + yOffset); + painter->scale(PaintingScaleFactor, PaintingScaleFactor); + + for (int i = 0; i < myMaxStarCount; ++i) { + if (i < myStarCount) { + painter->drawPolygon(starPolygon, Qt::WindingFill); + } else if (mode == Editable) { + painter->drawEllipse(QPointF(0.5,0.5),0.08,0.08);//(diamondPolygon, Qt::WindingFill); + } + painter->translate(1.0, 0.0); + } + + painter->restore(); +} + +void StarRating::paintSelected(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode, QColor color) const +{ + Q_UNUSED(palette) + Q_UNUSED(mode) + painter->save(); + + painter->setRenderHint(QPainter::Antialiasing, true); + painter->setPen(Qt::NoPen); + + QBrush star(color); + QBrush dot(QColor("#ffffff")); + + int yOffset = (rect.height() - PaintingScaleFactor) / 2; + painter->translate(rect.x(), rect.y() + yOffset); + painter->scale(PaintingScaleFactor, PaintingScaleFactor); + + for (int i = 0; i < myMaxStarCount; ++i) { + if (i < myStarCount) { + painter->setBrush(star); + painter->drawPolygon(starPolygon, Qt::WindingFill); + } else { + painter->setBrush(dot); + painter->drawEllipse(QPointF(0.5,0.5),0.08,0.08); + } + painter->translate(1.0, 0.0); + } + + painter->restore(); +} + +void StarRating::paintSelected(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode) const +{ + paintSelected(painter,rect, palette,mode,QColor("#ffffff")); +} + + +//------------------------------------------------------------------------------- +//StarEditor--------------------------------------------------------------------- +//------------------------------------------------------------------------------- + +StarEditor::StarEditor(QWidget *parent) + : QWidget(parent),shouldCommitData(false) +{ + //setMouseTracking(true); + //setAutoFillBackground(true); +} + +QSize StarEditor::sizeHint() const +{ + return myStarRating.sizeHint(); +} + +void StarEditor::paintEvent(QPaintEvent *) +{ + /* + QPainter painter(this); + myStarRating.paintSelected(&painter, rect(), this->palette(), + StarRating::Editable,QColor("#615f59"));*/ +} + +void StarEditor::mouseMoveEvent(QMouseEvent *event) +{ + Q_UNUSED(event) + /*int star = starAtPosition(event->x()); + + if (star != myStarRating.starCount() && star != -1) { + myStarRating.setStarCount(star); + update(); + }*/ +} +void StarEditor::leaveEvent(QEvent * event){ + emit editingFinished(); + QWidget::leaveEvent(event); +} + +void StarEditor::mousePressEvent(QMouseEvent * event ) +{ + if(event->button() == Qt::LeftButton) + { + int star = starAtPosition(event->x()); + + if (star != myStarRating.starCount() && star != -1) { + myStarRating.setStarCount(star); + shouldCommitData = true; + emit commitData(); + } + } +} + +int StarEditor::starAtPosition(int x) +{ + int star = (x / (myStarRating.sizeHint().width() + / myStarRating.maxStarCount())) + 1; + if (star <= 0 || star > myStarRating.maxStarCount()) + return -1; + + return star; +} diff --git a/custom_widgets/yacreader_table_view.h b/custom_widgets/yacreader_table_view.h new file mode 100644 index 00000000..9ce2fc96 --- /dev/null +++ b/custom_widgets/yacreader_table_view.h @@ -0,0 +1,124 @@ +#ifndef YACREADER_TABLE_VIEW_H +#define YACREADER_TABLE_VIEW_H + +#include +#include + +class YACReaderDeletingProgress; +class QResizeEvent; +class QPropertyAnimation; + +class YACReaderTableView : public QTableView +{ + Q_OBJECT +public: + explicit YACReaderTableView(QWidget *parent = 0); + +signals: + void comicRated(int,QModelIndex); +public slots: + void showDeleteProgress(); + void hideDeleteProgress(); + void closeRatingEditor(); +protected slots: + +virtual void closeEditor ( QWidget * editor, QAbstractItemDelegate::EndEditHint hint ); +virtual void commitData ( QWidget * editor ); +private: + YACReaderDeletingProgress * deletingProgress; + bool showDelete; + QPropertyAnimation * showDeletingProgressAnimation; + + void resizeEvent(QResizeEvent * event); + void mouseMoveEvent(QMouseEvent *event); + void mousePressEvent(QMouseEvent * event); + void leaveEvent(QEvent * event); + + bool editing; + QModelIndex currentIndexEditing; + QWidget * myeditor; +}; + +//--- + +class YACReaderRatingDelegate : public QStyledItemDelegate +{ + Q_OBJECT + +public: + YACReaderRatingDelegate(QWidget *parent = 0) : QStyledItemDelegate(parent) {} + + void paint(QPainter *painter, const QStyleOptionViewItem &option, + const QModelIndex &index) const; + QSize sizeHint(const QStyleOptionViewItem &option, + const QModelIndex &index) const; + QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &option, + const QModelIndex &index) const; + void setEditorData(QWidget *editor, const QModelIndex &index) const; + void setModelData(QWidget *editor, QAbstractItemModel *model, + const QModelIndex &index) const; + +private slots: + void sendCloseEditor(); + void sendCommitData(); +}; + +//--- + +class StarRating +{ +public: + enum EditMode { Editable, ReadOnly }; + + StarRating(int starCount = 1, int maxStarCount = 5); + + void paint(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode) const; + void paintSelected(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode, QColor color) const; + void paintSelected(QPainter *painter, const QRect &rect, + const QPalette &palette, EditMode mode) const; + QSize sizeHint() const; + int starCount() const { return myStarCount; } + int maxStarCount() const { return myMaxStarCount; } + void setStarCount(int starCount) { myStarCount = starCount; } + void setMaxStarCount(int maxStarCount) { myMaxStarCount = maxStarCount; } +private: + QPolygonF starPolygon; + QPolygonF diamondPolygon; + int myStarCount; + int myMaxStarCount; +}; +Q_DECLARE_METATYPE(StarRating); +//--- + +class StarEditor : public QWidget +{ + Q_OBJECT + +public: + StarEditor(QWidget *parent = 0); + + QSize sizeHint() const; + void setStarRating(const StarRating &starRating) { + myStarRating = starRating; + } + StarRating starRating() { return myStarRating; } + bool getShouldCommitData() {return shouldCommitData;}; + +signals: + void editingFinished(); + void commitData(); + +protected: + void paintEvent(QPaintEvent *event); + void mouseMoveEvent(QMouseEvent *event); + void mousePressEvent(QMouseEvent *event); + void leaveEvent(QEvent * event); + +private: + int starAtPosition(int x); + StarRating myStarRating; + bool shouldCommitData; +}; +#endif // YACREADER_TABLE_VIEW_H diff --git a/custom_widgets/yacreader_titled_toolbar.cpp b/custom_widgets/yacreader_titled_toolbar.cpp new file mode 100644 index 00000000..9015ff29 --- /dev/null +++ b/custom_widgets/yacreader_titled_toolbar.cpp @@ -0,0 +1,112 @@ +#include "yacreader_titled_toolbar.h" + +#include +#include +#include +#include +#include +#include + + + +DropShadowLabel::DropShadowLabel(QWidget* parent) : + QLabel(parent) +{ } + +void DropShadowLabel::drawText(QPainter *painter, + QPoint offset) +{ + Q_ASSERT(painter != 0); + + // Draw shadow. + painter->setPen(QPen(textColor)); + painter->drawText(rect().translated(offset), + alignment(), text()); +} +void DropShadowLabel::drawTextEffect(QPainter *painter, + QPoint offset) +{ + Q_ASSERT(painter != 0); + + // Draw shadow. + painter->setPen(QPen(dropShadowColor)); + painter->drawText(rect().translated(offset), + alignment(), text()); +} + +void DropShadowLabel::paintEvent(QPaintEvent *event) +{ + Q_UNUSED(event); + + QPainter painter(this); + painter.setFont(font()); + //TODO find where is the '3' comming from? + drawTextEffect(&painter, QPoint(contentsMargins().left(), 1)); + drawText(&painter, QPoint(contentsMargins().left(), 0)); +} + +void DropShadowLabel::setColor(const QColor & color) +{ + textColor = color; +} + +void DropShadowLabel::setDropShadowColor(const QColor & color) +{ + dropShadowColor = color; +} + + + +YACReaderTitledToolBar::YACReaderTitledToolBar(const QString & title, QWidget *parent) : + QWidget(parent) +{ + QHBoxLayout * mainLayout = new QHBoxLayout; + mainLayout->setMargin(0); + mainLayout->setSpacing(0); + + QString styleSheet = "QWidget {border:0px;}"; + setStyleSheet(styleSheet); + + nameLabel = new DropShadowLabel(this); + nameLabel->setText(title); +#ifdef Q_OS_MAC + QString nameLabelStyleSheet = "QLabel {padding:0 0 0 10px; margin:0px; font-size:11px; font-weight:bold;}"; + nameLabel->setColor(QColor("#707E8C")); + nameLabel->setDropShadowColor(QColor("#F9FAFB")); +#else + QString nameLabelStyleSheet = "QLabel {padding:0 0 0 10px; margin:0px; font-size:11px; font-weight:bold;}"; + nameLabel->setColor(QColor("#BDBFBF")); + nameLabel->setDropShadowColor(QColor("#000000")); +#endif + nameLabel->setStyleSheet(nameLabelStyleSheet); + + mainLayout->addWidget(nameLabel,Qt::AlignLeft); + mainLayout->addStretch(); + + setLayout(mainLayout); + + setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Minimum); + + setMinimumHeight(25); +} + + +void YACReaderTitledToolBar::addAction(QAction * action) +{ + QHBoxLayout * mainLayout = dynamic_cast(layout()); + + QToolButton * tb = new QToolButton(this); + tb->setDefaultAction(action); + tb->setIconSize(QSize(16,16)); + tb->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); + //tb->setStyleSheet("QToolButton:hover {background-color:#C5C5C5;}"); + + mainLayout->addWidget(tb); +} + +void YACReaderTitledToolBar::addSpacing(int spacing) +{ + QHBoxLayout * mainLayout = dynamic_cast(layout()); + + mainLayout->addSpacing(spacing); +} diff --git a/custom_widgets/yacreader_titled_toolbar.h b/custom_widgets/yacreader_titled_toolbar.h new file mode 100644 index 00000000..0b4a03c1 --- /dev/null +++ b/custom_widgets/yacreader_titled_toolbar.h @@ -0,0 +1,45 @@ +#ifndef YACREADER_TITLED_TOOLBAR_H +#define YACREADER_TITLED_TOOLBAR_H + +#include +#include +#include +#include +#include + +class QIcon; + +class DropShadowLabel : public QLabel +{ + Q_OBJECT + +public: + + DropShadowLabel(QWidget* parent = 0); + void paintEvent(QPaintEvent *event); + void setColor(const QColor & color); + void setDropShadowColor(const QColor & color); +private: + + QColor dropShadowColor; + QColor textColor; + void drawText(QPainter *painter, QPoint offset); + void drawTextEffect(QPainter* painter, QPoint offset); +}; + +class YACReaderTitledToolBar : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderTitledToolBar(const QString & title, QWidget *parent = 0); + +signals: + +public slots: + void addAction(QAction * action); + void addSpacing(int space); +private: + DropShadowLabel * nameLabel; +}; + +#endif // YACREADER_TITLED_TOOLBAR_H diff --git a/custom_widgets/yacreader_tool_bar_stretch.cpp b/custom_widgets/yacreader_tool_bar_stretch.cpp new file mode 100644 index 00000000..e69de29b diff --git a/custom_widgets/yacreader_tool_bar_stretch.h b/custom_widgets/yacreader_tool_bar_stretch.h new file mode 100644 index 00000000..d4817176 --- /dev/null +++ b/custom_widgets/yacreader_tool_bar_stretch.h @@ -0,0 +1,18 @@ +#ifndef YACREADER_TOOL_BAR_STRETCH_H +#define YACREADER_TOOL_BAR_STRETCH_H + +#include +#include + +class QToolBarStretch : public QWidget +{ +public: + QToolBarStretch(QWidget * parent=0):QWidget(parent) + { + QHBoxLayout * l= new QHBoxLayout(); + l->addStretch(); + setLayout(l); + } +}; + +#endif // YACREADER_TOOL_BAR_STRETCH_H diff --git a/custom_widgets/yacreader_treeview.cpp b/custom_widgets/yacreader_treeview.cpp new file mode 100644 index 00000000..8b60c330 --- /dev/null +++ b/custom_widgets/yacreader_treeview.cpp @@ -0,0 +1,80 @@ +#include "yacreader_treeview.h" +#include "treeitem.h" +#include "treemodel.h" + +#include +#include + +YACReaderTreeView::YACReaderTreeView(QWidget *parent) : + QTreeView(parent) +{ + setContextMenuPolicy(Qt::ActionsContextMenu); + setContextMenuPolicy(Qt::ActionsContextMenu); + header()->hide(); + setUniformRowHeights(true); + setSelectionBehavior(QAbstractItemView::SelectRows); + setAttribute(Qt::WA_MacShowFocusRect,false); + + setItemDelegate(new YACReaderTreeViewItemDeletegate(this)); + +#ifdef Q_OS_MAC + setStyleSheet("QTreeView {background-color:transparent; border: none;}" + "QTreeView::item:selected {background-color:qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #6BAFE4, stop: 1 #3984D2); border-top: 2px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #5EA3DF, stop: 1 #73B8EA); border-left:none;border-right:none;border-bottom:1px solid #3577C2;}" + "QTreeView::branch:selected {background-color:qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #6BAFE4, stop: 1 #3984D2); border-top: 2px solid qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1, stop: 0 #5EA3DF, stop: 1 #73B8EA); border-left:none;border-right:none;border-bottom:1px solid #3577C2;}" + "QTreeView::branch:open:selected:has-children {image: url(':/images/expanded_branch_osx.png');}" + "QTreeView::branch:closed:selected:has-children {image: url(':/images/collapsed_branch_osx.png');}"); +#else + setStyleSheet("QTreeView {background-color:transparent; border: none; color:#DDDFDF; outline:0; show-decoration-selected: 0;}" + "QTreeView::item:selected {background-color: #2E2E2E; color:white; font:bold;}" + "QTreeView::item:hover {background-color:#2E2E2E; color:white; font:bold;}" + "QTreeView::branch:selected {background-color:#2E2E2E;}" + + + "QScrollBar:vertical { border: none; background: #404040; width: 7px; margin: 0 3px 0 0; }" + "QScrollBar::handle:vertical { background: #DDDDDD; width: 7px; min-height: 20px; }" + "QScrollBar::add-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #404040; height: 10px; subcontrol-position: top; subcontrol-origin: margin; margin: 0 3px 0 0;}" + "QScrollBar::up-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-up.png') center top no-repeat;}" + "QScrollBar::down-arrow:vertical {border:none;width: 9px;height: 6px;background: url(':/images/folders_view/line-down.png') center top no-repeat;}" + + "QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical {background: none; }" + + "QTreeView::branch:has-children:!has-siblings:closed,QTreeView::branch:closed:has-children:has-siblings {border-image: none;image: url(':/images/branch-closed.png');}" + "QTreeView::branch:has-children:selected:!has-siblings:closed,QTreeView::branch:closed:selected:has-children:has-siblings {border-image: none;image: url(':/images/collapsed_branch_selected.png');}" + + "QTreeView::branch:open:has-children:!has-siblings,QTreeView::branch:open:has-children:has-siblings {border-image: none;image: url(':/images/branch-open.png');}" + "QTreeView::branch:open:has-children:selected:!has-siblings,QTreeView::branch:open:has-children:selected:has-siblings {border-image: none;image: url(':/images/expanded_branch_selected.png');}" + + + ); +#endif +} + + +YACReaderTreeViewItemDeletegate::YACReaderTreeViewItemDeletegate(QObject *parent) + :QStyledItemDelegate(parent) +{ + +} + +void YACReaderTreeViewItemDeletegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const +{ + TreeItem * item = static_cast(index.internalPointer()); + + if(!item->data(TreeModel::Completed).toBool()) + { + painter->save(); +#ifdef Q_OS_MAC + painter->setBrush(QBrush(QColor(85,95,127))); +#else + painter->setBrush(QBrush(QColor(237,197,24))); +#endif + painter->setPen(QPen(QBrush(),0)); + painter->drawRect(0,option.rect.y(),2,option.rect.height()); + painter->restore(); + } + + QStyledItemDelegate::paint(painter, option, index); +} + diff --git a/custom_widgets/yacreader_treeview.h b/custom_widgets/yacreader_treeview.h new file mode 100644 index 00000000..bb71448d --- /dev/null +++ b/custom_widgets/yacreader_treeview.h @@ -0,0 +1,27 @@ +#ifndef YACREADER_TREEVIEW_H +#define YACREADER_TREEVIEW_H + +#include +#include + +class YACReaderTreeView : public QTreeView +{ + Q_OBJECT +public: + explicit YACReaderTreeView(QWidget *parent = 0); + +signals: + +public slots: + +}; + +class YACReaderTreeViewItemDeletegate: public QStyledItemDelegate +{ + Q_OBJECT +public: + explicit YACReaderTreeViewItemDeletegate(QObject *parent = 0); + void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const; +}; + +#endif // YACREADER_TREEVIEW_H diff --git a/dependencies/poppler/bin/poppler-qt4.dll b/dependencies/poppler/bin/poppler-qt4.dll new file mode 100644 index 0000000000000000000000000000000000000000..f5b21d16386323d3a4499d9a5942421f0fd24361 GIT binary patch literal 1671168 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

XMY$S7!5 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 UlE9YNqxV@mBLPcBe~ zQ%o(}`6EICq+ax!{2!_{Y)6Iw4_;+(d>9Iz#0wDCaJ*W4;ryV)K?vzbN5^{x)cwo- zqFT~7yn#z=ct?jKCNEx7J8xRcZdIDTWyGe<@dBOf==oUF=r%P}f`+2s@oZCJ_~2IF zHA=dfk|tDm>eaMA^E7N2TG$k%1?*NMJm4%g4CcC$VqvwNX&CQmdDwSI&AYjwOPX|dOa~y z{_?P8UTZ;<0duv5oud;!yR{e6Jp_OFf8aO1OZch{`}EHgoM2_pa*FGx+aE9~YB+}A z8W5O+djPNJlVd0~{V?8czf`hZGTsvXUZ5ri%p_;T{9}jYUvLYwB@Ft&D5${+yDIO} zd|0B{7JF*ekebv~{~LlytX9eMZ1)@$cRq%B_SWsHY>25uQL?Th-|O(-dA)pg<*n#l zOqG>D^M7Y+!4n2;?>wc$I9a|BLi%jAF%SryWD$XwIfu9szd$TaATk>FBx=ctK|<>x zT%zFr1C4@kBI7)4Bx*^Cm!cxXT3qaFD*&v8nE|nd8@mH@AV&c6Be0o0HnsZ;ISVn4 zC(N6H`5%m!{})5@FBlKZw<8{7?|HHwOL>(O%iEt`9IWq#1G;bQAe_7GC$a~7jUXHX z$Z2gYFyOd^Uc&T?=3HI~VL9ew&e8q3L|kE%=bP^#heEqmDNwAZ4J&W4_r zOgIzmLTP>t=y@b*@+n#A_72J`kd;d`X^K7!q6ZvzrN^=J>M zXa9lWs9uV$-O6r6O-iT;LLBY0VUTw<)k7|W=4#wk?8b3q1!-p}1 z=eYi}*4XCAco;#Z3md>V+#{755yDZ`MIS1 z!y&4XnP`Z;^zkLZnwC`gLLQ5v9r_;vYonIH()zMESXzLkEQwr1{rKjD4ndB9w-W$Q1bcGMzpHvB zyQdiREWCe-p7~rH^v)6$CXdJT#2Tf1oNvJ%kNYfr(3BkY#F3E=p4rY8_gU@tei7~v(WnKQu8o{%Yd2w& zY&Svyyq+%23_CZH!lJCrE>yv_Ev#+=ioppGc8F!{_xKU0-9E~Dt*0MTyc%b_#k_5( zwKuZ=#DxN6&G*0X-Fq6)25L8^d(%DJkTrS~eonI;`kG0B7Bdu{r1*y)`-ugLg)v10 z*v7^>Ik8a`f!!z{vTvmWl&$;#1%Gk+u$3ZWN$#z!bLjwOJ1|dNi8SA_pfxmz+h-+W z1aU`TBjVmt3IiJ81|eJE%m-c*c5f_YwNTyBS>hz_La)d6f%@{^-$h>z;~NmDQ8&WQ zEei8uzy1TtIk(2NyEiT1u!9ff+oqV-QD~v)%_1s6=g_DGoha4Gk7KSwDbGfZONj1T z4mOb4&dRCb5p*X8vd4n%RMyCg4RWqQUTl_g&GKTaoNJXAXUe&m^5QHxH%neTQ_jnb z%rnS&*2p|_Pj46NIfXZ!`{czra-O+!U)KqFaV}EDQ<0o!pfs19H#0KN%FAZ;^wO(E za$Xjy;0cDF-X6SKN(ZdUF%saB^B~(%4GJ^ShB zQpuSiPp;fORKEt35ZPf6N{gqZv8VfDPoCJ*^4Qag z*i()A)Km}Ri~aB?{vcT)ma;DKM^oYt3LX;6wlV$#@1&;Z@tecgv^kz4HIZJ0ml5_U zH@57asg8wm{_3xtrt#hik0&#A)xdjdw**?!mD}FnZ1{@Z3ae8be-BU=NF#$s7Lo-Z zei-Qou$U=oC3hr$^HxfpQr@QSD1chG$wbJd*!*4hx4StuOL?BQJP&ZHiTR7+ z{5l$s4>7X6qg3ZntXY11B`ovVO+|eEBkD|{coH^pR4 zMcAFnldxwd&K}YF5Og@g#(hHDsS6?O6(a0EJ_Of93>$u#eSw;993}k?IycIfhmC!N ztwcujhno2bJ@V@|W2!Fd!I|$S_5g8PJR7M8{|1dc?*Yk~N)N`6zYJAvh(1JJ_z88v zLv`=^;dz2b>Mu?1!f(eYUErY2S3v_j5hj0-&0g3S_zm<@H89&5!b80#K>_umHicz(g>Ia1z{f@+mywR`vztxqu`x>|j_ z3pd5e`A+X}p$Il${Jt)+-~7k%JjM`BFV5_J3zWX2sSQR)tqJ?pe_Y;(VQHI7;Ax_D zJBkDZ|3W+wBWSEGH9q@5l+f!23<*3x}4UL*<`Bd4)x+p~vBUH^s^kqb4?&MSTxK z70&w}=6xUD>G!T5+Uc76Ijc_S@E*EOHO^-^uX~bfzb-&VB=}B668iV_2-%lXc49Ty z+(p=K%8sFY1K)9rQmjw3UMB9JFJW&_LKyd%%5GT}a-<;e6GhwLA@{} zAFlI$R;>nhDr5%G&z{OvHo6IZfG`8-CpFIEqhhZMD}F_zfN=ioRX92lT7HLbu@6ta zFgiQdhS==l0_eRoxW6OyG}$Tj@WDl#x1oCUv)3t|gq=7i>9eMF6x*2W+^eB)K1l1n za^W(IKp&DY%owSI)l-Y4Y-a%Tp z4>M_Kh14n~JR~ZdKeWP=h$jn#vnP0kSVa%AXHg-Bo6xh49=SnSGI}g(CrL#i2GV~u z&_h=6G*5q|%+uuKs-ADFsIi|+rIYICcF^Ci&hEH_u(=y|@%;G69m_)HDb%+45^Zmc z)#5ftw;93^ggrZHY?koGQ@y1G@hI=*fdPYef^-<_FdO^gqw|A7N2YW*`n^_1Cd~MT zYBcp4&sJ2zPhoE6Ef`zVsBIgsZ}ggA zn(`pz$yj>`QqWc?8Z5Mi3TK20XUP>Vc}c9ng-b&V?~|A01Pd3m6)FuD7K93mLWNnO z!kIxF_wxp?qDZAM_W-wm5l^~rB0L7OSRT%pDK;D3c@j)lpQzIb|G+;W+@9)o=qrvF zFFpSA``9hF(oWA1(snxx%B$<>mD`Be&M7eNO)_B={O7>m$C>Z@4f^3z z4|{nIpW~3nO*^I_tMr95@WrpO?;v-tho?ay`CX{I){0z7^808yIw(Dytnwn6hs4);?O>vad%w^2!9V2{~WXRP_;PT0=6*ni$ad=Me% zlRa>OFQA)({W%P_cHeak`+P9^Fc?sMG*yS42&`?%2D{n}_sJjGs17681OgEPC~5qz z6M7wZwi!|7DHM7MU7rAxeSkUWFeZBrh(Shs*c~<;SvehM__mJPY2}E9Z^S(V(&y86 z@A{iwcL-Iw!X@U?cr?PqdZ;|%qha65Z`q2BIG%GGY0qcg^?_bjgb1VEf4K<5e9#9h zxWIw%uT4YBGKEeO%3G=YM3h&Hp1`%B-%yly0Ob)OgM*M9%>w}4Llstv+Q{^lJ&6iQ zzgmIq8urob$+u|%?WYAOS=Bn}YQe|ENd*kTk$LVBY*(E9><)AYQAfyk+W|*pC!yCq`8q zKUU?u24M=yNZE2@_l+&B_D|uerhlY*aN3`OPp6qBZMc>}NAEI1sbOaiwhW>}sTw~n zPaZHw3n;mtlFgLdFIAovIS>Hh!EVa>;J#Ra)qAMe>Rn3VUFXfa&YFYHPa(QBx<6eK ztkPz`9ZUM-B*wFIEpYvw4b+Bgd1w;DDA9-*z1dp zE@gopjgtOVMnol(!aUvW7k!uP^$#B5sNlF0{a?MCW8thB7dflgToN1?#Kr246uJlJ z0`9^2&mV9sVEA!ZQWD<4t3AMS>0q2+DMEK7-^prX**NL*KY$`*tz5{!tL`rpdHYJYg@kfb$Gr3 zl3WV}E4)EGm`o65K8%yM$^I6+D3$MoEWSndx8q0HzELjAbw%#Yfu4Fh-H{~US;UTS z`g%~d!^C+fO!GnRbkr^*=UvmxvV9AufBPm#J}%}uh|2d2;nV!0peJq?2r}X-5LXhz zY||3dy$$0PD&gZvs<{{vE_B-|yeX7vXTuHmVL)Bsuoqt?G$}Fr!V3-En3WK{T(@E}RnjY8C^$SOxsqD+$H@h&2nS!L3w~ z56O0>!keyeEZLD8smgi(#F@Q$zDcqpN45`izVrSGxr<)Rh*af9Ey$Aec9K#c^|5jM zpi8Z(6OmtHj81`C(Bu3#J+AIsmpB?jNz#mU0scntT{>G9`jZC;X4pIpwx!>NVU^dc zz&w_hELF%)UbB+__Q-4e{9%Q>CdeOF%4?eF;UzyBlhdaLtrAYchy#D%KUf7K%a)R= zeVIpgtnA#64x`ad=;>w0ipY8b8INuQes%5XImkgj0Cl_Q!iT^Ha&&zlrxNgDe7in9 zvsdf8p=)>72c5^20e@_)eEh~1xwA^{ti+Lj*Z#;@(&+b`>S~GR%Uy&OQ5DC`hn-y< z52pGm;FmpXGXO(?mJR?S>j^7R`F|LD{dQ~N^ZDA6}$X9NVlwkEw$S!cgbDbwD{Wz zTCS3(D7n!X(et&E!{y)Z=D8>1ctXjIyt!3$)M3%cUD1>nock3x2~m%A?N=eZ1`r+* z5Kf{dkIS9gv?_#Zivl&iD|w3ozKPUgCuQca@%DO@d{;^3F8><>vdaJt^&FjrXhkql zDIU5^HN%Ke$p44=wGU@%ZRs_M^Ggt|m|msw5>ITt4Vz|#D`+~2DMV8X(}{*N(oVF} zvr9}6B0hC;j4&%a-1hkrv*OvANwb2_2Xo|ZPb5U`5VhUCG!pXT8D~}@PUfG|KoM=_ zJvLPgl$PRH&}er?<`qTqauVnWq)(JFRVo4g#W+n+^5URO%_xQ`^|2{Dn;52Xlr1k; z%|cOIEp*;j0vicjzc_nNLqW;q<~loE$9M=Q$e7c6rnf@qq9EDL#%$9o`?s(5`~6?_ z>fQnG-*pa=x_7ZFb-Qs^6V!jd&|#aifvTuETy>`!ArJM#gA^?UMD>t3`AS?@$^E^| z*iGS1o9S)S4~OgnxN!`jBJ2Z=#+=40P0goF%|}hId@kDjRQGOTgQstRaoUBH13}|0 zu#3qp1e7*?n796BqF=buKkQ{@XNdmQpnoJklOQ?$BJW>g8oKv`IsS*gWoqs~FLnE3 z{fxI6{f?=*WpytJNN>=e>6geEu1}$4AV{Dk-(kxTXxW$uSbV(!awvG}ECJqkfERKf z1$gG^ozsB}9dx?;A4+N5A2g<;uZ~>XEYZ=ExJy-yORi2Z^!J~t?~#ALKBAAIqqi7a zCK58q!__$b?DSKOY9GGQ9q_}{4E_AsE1QQztK^g0fNU}n7%uxH(@v&c{VXwYmO$}ZBKI$_0x1g zh|@SnU97S?;;60D51~Se9F)%EHP|mSW`HulU=Jj79_&oxJ;Y48p)+M_VKC$SF)q6| z&Sllv&)bKv&Br}5HfMs_T!GC&Wb8~l_3Y|~KsGjuKeFuUSf|>h^U|K#m2{Vw-55=+ z(}AXE+s*XT95o6&&VtaTL#F0Rqph=U zG!%4|aF8-YCnHLM(_lMZ2mWf6_C!bF{=UN!bU{aLpVv)otTf-S`%Dy-O#d8x58|B2 z5VW?Y(-mixd!T)4P0_S!HR_A_SR~SeF)*Mvt>vyTxEBv$n12nGh27Xl*OUcElM11A zu5OT6s?pSp<)|v9!qi-cV}08|tsY8YeO-#wX6wT90bF6%iC=rdwHGUZ>*mqn+QAB! zsd;XSt8uOtyWPtfjt>V-&9#@Kr}%c`Ub}%%C2r)iblV5yR3eMWKFl{cmFPm4({$S{ zaw^dVenJpp6R$_moaSOJNE6F4o>EW%)3j$leLGK!y#_toPxtIYfe*7lP)_r7`{{5= zN@=)6i^$PVi>|#yHvkWmGIkI5KA)j5u$`OJ%=9^=eD>p8#g`NF#~3)P;>!+vh1qY0 z;VV~%z7=W9%1zBHQcTUTTIsO013~X_WbLwjj^}rQtbQP?6?hFhFP0E9A?5vG2@sZ6 z>S}brwBT~q3fSWKa;o@p{tNi}=sLnzIq)?OSfW-GJaK#hZ`__J0bl=(0^fu$T$&8! zsEV&X&Ja{$V$Vi&9F$ zE^Vm`Oj6hC(p{|3^^~zfsPVnVX2W`)I7O2C+kfq)`ui?YzM;Pr zd|MssuMx1*XR7jt+SZbAGsKjaIYDvK{O0Dll$a`zn6rPs89U z)AswY1mMgGCrPc29F4ygzFxTRZl>uKvSOYSMfi=Gf!*o5&cS>P!5R=Es(8+J122O! zs9ZJP0=Am(m|p479ijhL!wo4#Q#bmZ0px(q%c>KIce-7pv1oYXwUbK1`ma$10_+5? zqO%4APP3v`&qiz>DJUcRWHc$tVRcY!2c=EucEppUQxOehJh37IwC9 z4KT8X^h?T%pYj)c`zdmS@j`x)R1->$t^CEN_=^Ml#g_Ps<1k(-^4?zLeYnUwMNZvR z1(xnM5Ur#>HYZZaRY9qb%?i9p?P|6|*$xNE)EB=Gq)Y}4Q|Flu(4*NdPp(Ucoo zOwH?8qe4^jGk$!*iHheeWxqmJtoam_sPH!_{K~b^nigc4{pLyhty>gczZc&S93TQn zQxNC+3Ms$H#Tu^!&CFj=!dBy zfQBQc4Um4|+2DK)H%F|5jplZKd&KNgttN0_5WAhw8mcZjF}o${w+_EN?MPwgrgCUS z6VyF4ks+<8QsHN4PLyIHU&XJ$IEr%|^85(#o(UpB6YO!2iNxQ@8)+5-;n_0c8-(KZFa10t@fRCF{vvNQkt`X6ef~8$$(_h*;EdiXwyYdHK0nAnvz%wn=nxdb}T4$ z_3%X#T%StM^3!kAm(aYYK9;Wh z&a>^Wff`!SOKCVvc}k=w`9O3D0GDLHUB%`pASU_)pvzJd#Tym9E@g{QAOSx-y(1w# zYgoGM?~J7@GuN*LZeVv#QP@*|oCCsE6`|kS*}Y$;CQcBV6M6`B33EogN`Q?X z#rp9N?9WvQGeGzP$n-ZjtQC5MB?ioo7@}jixg|y3|G5IYI6PJli2=XVDmo?3kVV2d zGmhgiN%cYZsGxncPv2@EGYC$#V)F2=PF1?P@^n&H)zH03omF3o@sgVAX}t6sI$$xS zGJvaAo#!DIlQ4q(e179RjY`0HfoM>niGRbXthD|DULbPvWJuDHmnfIez>g9|;1(l1 zB}BFH`sh4wNPVW^^--=^ee`2^eIO1~bJIquj^d2?+FKRl?bBHNW3nI?`>jMDh_@C&n^dx~7 z-TNYj=#)I`Y zUATbi#-l<}vjR@IfdfK~5O)5b8zGo4-H51OAX;E6vQL4(YhOYSM9)U(hk%`W1JP3V zg&3;U7r`tvrgigxKDo1(?iTolu@u9g^jjHAf_U)AaOl*r6tH#e6C*hsy2coi0Tw|A zG{iv#;^`fr8w&Bgv~+bwugh);)~8aJ=!<9pU%XT*FSX}~)WMqWa0BImVFu+Xk@%z$ zlQq17@|Pk5T5++L$MMBi*uyNhi1jzNxPgVVuI3U*OaQ_B#Cj4ru9cKI#Q8U zn)96G?wegczO>|u|`w?T^0XZ>$cNsXytDNN_2wG7X#Yk6g`=N;|2YYEJH zW+N%E5-}APdj6Xr_Hr*&5=EHZAyB+3pu`NFfK9c!X z?%2V-3-`cf1GC!+xR|Hwz89d7@%F@%;b%-*x zBDH4^tR(VtU>(4X7E&M6%W~ADSTOkXayMCqsMZR!Wvt9VGPiZ0yq!qy;y}#cKy<#{ zxtprqN3Fk2t^b)?ubkHT7ol<(se1GPz*D7Uwv+n!1jRf7%^#=cDLp^4^RR*l01vqL zqKAO{?apIr#Ss7|>>AX9T?fSrg7i32dDc9nMntuJ2!a@ihTNIm_RP)*n(xM0x~YB@ zwnj2sGm2+KJ`Oo~A`CF(<2>;mBq9oV&xg9*Av;e~tWQ!+(ruJhKC1^uWTH~#+}$M2 zLQ0@+fyPvyM|n7TNaeYvKfth{;I?aUcKlA8J&Onkw_ewT^&9As?u)8gVEV&d)|7dx zKN?&GHAk0JyukOPKWhABjD43OMS7A5ZJ9I*8JC5Oxae$w^b)<&wTBSg*qNpL!pUOs z99?lyXudHzYT00cAtk&3r7N@0e14z>`jISKSItu(KIvI1F0G|aZ)mPrG$1`oVo<2W zBInN(z{`uQB1z8AsfHVHW?+>U#Q*Qe#$Gik?1r4x@q)IQR{$^2?qQp#rLAH$9S5|t z7v^gDR+VqRRR+YHW7DkaOxi${bJ86LENkSXW5(-yQ_#FeA0iZ}t z0KnY1LbN^BN5W^TJyXLT8--$gCybwi%=R!LpHiz0LW98WOKcaS8MTYOekLJsd9Sk9 z=X)iWQa$@MV1X2ULt;tr7EzJ|!fqV79b`(2|znS=0@iF5=_K9VK9fNE2 z7}?0ZFngP+%SweWn-9IjNjJowE zPZ0NBeUYMrAMGbbKBR5D2uk4$FSgY3`bd7KosGk6a~OT8WW=;b8pp2fyoe|%-hSBT z78~lWptFC)->zJLqd6~l`|QD7Td|?0f61W3OlD9C=gnA8o=;p)R>rp0$@)5=5DnYS zvb#faPlJ*ODwMN0?2z3jLe3^AilkyheSTWP(Z%zUtAMOHO${axcXqd2&|?;Rmch#| zY%`gmsU~s!b09lEa6YZ(eJNeyI|x1%`K;+EDVGjIx?D>Wwj1l_^VQQLo=kaBrt~C# zm8Xw*vgAcsaEO;HH_7mvuiOH^csTY~Zj&8Z^u1kn%#1joYiNOiSOos`mF?`m#we5K zd8x7~&JHF>Zd^iCtt}(f)8AoZPLm`)J-DSZdkKyX{@M%>h}HCT3xZx}2JQY(WqrsQ z4Effv1+l78FL4OO!JaBTOEXZqO)pI-uh!!KYAx3SX5Tszry+5T=@aWA=74%|{$8<4 zdXm!1Ij8z)5Sz|^n#LOuG)9?ct@e$GcSd~2fX6}h25yeva@O<6pl}GxRK&SxXS@F@ zW-4s&mC}f}o~8-BMeG)AYm;dM4kKY#U`2TDGR5T9<7&Y<)*tz`NtK-`~tf5x+I1zOm=be6KhQ5>xZ7r^(e zP`0(Lq4ww>NOp}N*)aUNDk2b_4#{_Bb|VZrvTs7J2w2sfS;*(SAmnU4suMaxrN0@m z8F+zuD<*sfU76^x`X*G}h#Pddg6shNR!|TOpn_ck-wSCV`-ttFZ{m`8mMv-)eqjKv zcLV2`U2ZVdzXn+OnSTdBf>w83YnmB1gp^xM>q&4{Sq5eO1s;)w;t3WT=9-!(2d$>&#oFK^ZR5O@U=c3*(iPguEj8V}55@~s4XM8h zx2#KvWl=>+a8XL*JYBFzHy6?Xuj5$nwJ6A8&cFj;-b65Wb3jJpjkw=ePcf`3*hlxm z*qz*2`-GB5SIc6mAN^$9{bZ>*+ItL02w^utU1`TN_a`D(mI_Aqp^C^|qP*lD5N-&l zXow~W>ao5d-Uz|@S~w&G%TjQ`jSf2=Lfq!4N3>P96kbbYY+k{>MXfmdCHHC7$6-KJ zPOT)kIdITpz7D}LP*U;wwZVKCx%9Q>>ju#hqAK#1HVrxuABt1gHtn5kbDAKNm-nWb z>C>rv#{N0Ye7ExX6~5^+4k(S!K7}5q#px0JpKL(*Ok0=%da^A9NiLx~&^SrvC~Zlm zGS>7CNV2n`+c)vfc!tZB3;}G6qzXO!58(|HqN^WHVgRuJW2W_Y(_n6%M!&P+Nb?!V zpjS%^jgUWKYh|^KuQA(7^sC0;78!%hexl~%DAaVHAt)PG74KuxLNkmr9=RV@2HV=x zU@|bK(V}g1euj2}lN>=qZ!hY_+pxY!x2w?*97k3M&d=a7(wLh9uMy z7}f`alNyV(K{7aTrNFnQ)u9{21*1y61}@^^-eFFjeiQ_G?nUeHZQ-uitn6;0r-&HT z7$v4f5$qgZwhQ%TY&^=m&kSPWY%a!uhj>bEq__7%pad-^9UUi!WW=inpiDVFfa<)i7$yDohRic$SLH z5KmxX7C5W(-io2}&JNVUHOGY=zkxE>EyoqM#JU|;U7iJ+`!Uk~Mr-iO!rXbjub^#2 ze+)KD&iB>e3($lA?(fR z8pRN34jejXVM3Lr^$*kANvO>9N{Vi-6&Ko$mEHp^lk;+K^L^DY=+if9qPoW1L2h+L z#-bdugM$w166F%PJ-l;HbQrr268JPGelRQKGYRlQ12TJxY5hw`Y61=^VhJv?En_Qk z06O?KluU??cG?9nTxD1{95`iGt)WXH)FW*=Y-ReRso-Dse$cBy*q37ccK0a;uBKP^ zNFBS*8DReFM?<>?4F3ItH*^GXZJVJm&nB#&Yg%9fNe++}jw>lm>o!1LalvLatrPb( z3;_89K)z|+T+&{+%?(Gq02VhY0J?>?HXpqFZ>-Z6@lH<$I+Q-w%~1L~PSPRq=8lIt~s8FK9$nMx6n6rC(5feG47NsuaauTJ&k16+KiBJC39Lr z5ZIhsLhkzPZXQqK5|ZivhS((}Q;3(<5OyI!LcnZS1k4Uaz=WfN5HL@u0w!Y)ts--6 zRdrjFiO)TP&F>bR0sG6j_*Z{msEBC~vL77;!?f3{es5byn$Gh=+N2d3;WpQW>>D9m zhn$;2m77DAO|u_?s{4K-(S`b>f8(sD0>8j|>JO!v%h}y ziE+c_3a`4P!@q%B@v~R_|4~QrBYALF;zsia#iPgO(Ih%%XJN+1(flKx#!QYhS49n#0I_T|o zBGQ<$2)dgT=62Sc8(c6kg|DGoZLKv=M<1)!m$Sd~ei0ZAXX-q3WklHdDtxf1KL@Bk z?$<_W#j+7zb0c(e!7pJkxpO$Q1<$1*cWc2fb%TV3jnbcL2ulmS@EK0!KIxLXetQcR z%m$ctZ=Z=(Rq|y@m0NJ5QrHJ~cAvH3NI9fvShF1|h23LNIYD%>Z@-)d10#fqvd;Ex zhEJ|D$nM=wW4{^GYco3f30f{?@1vhc!w|wyf+Xrs_mM>1eleY-ddcahE9%IfbOed| z6=Wiza+;>;@q=k*`s`OeC2z;qc+S6P#AXoAYrlDYUnHvCL+krCKr?K8|ACqTG(Q)n zCyZ_fEHuD@*7VB43|*0h7XFYkvYogPB6s62M6i>l^*;iQMbmKO3)bJp@mF`8IX>mc zI|D)YXIQ;_V-Q&tHw-dAa!Z9qEb|YDW!|(lidR_b>pR3!uh$0C99)&JF7Ys%3&K#F zFWzE>$3odl@-NR39Go12xkBkl((0p3jDmouWDNvxLdOzEpuzXy?;lxLf&l z+0r_c-!?43Y-R`Xp0c~Q!kSR9+1Kwv!qLBl{^Y`U=!He}cf(TrEe~<$)zAyvjq;%Z z!CV_|M-uqZ&3rw<)iNq41T(6_<)WC>Qke1#Wro;<^wb=NA(18414VBD^1fDbZ;}3; zs}3H5>OfIbLV19z>Dt*D1Y?$<>dx*C92TZRFeYOM O|J)}24J(IBNn3_vgA!a}s z3cmaqX$?^D3^WJA|5iGLvaS4C!)p1d{Vd=D7EB26$fGbLpb9Qy*g*Iy^>8+Kh~2jZ z`l~R#lFoA0RQBhK2#EO|0Rk|uvL=om7qgt@*iv)`zqx^Q1{x)^Z;Mcjm9bC1NYzwN zW4c&Jq}q*f%|(1%pU1d1@pW|0-eKcP&Z6uz-os`|0wKC_7ls(wKaqoR=mlcqTn?xY zxr4anDf9v@e|#|wfC+&dl*QDCUsy~#{z8lC6K9n1rp&aS(o)+*mtMsepP%B=QOUho zYM?wr*@DGN1AeHNLTA%rdev8AG3A++^;CznMnd~*v_uD;v`Vu9s@3_L#Fy@2OY}s# ze~B;PLkO&33}S`u92u6|b=c-?6`NC9Sf$<&=!Mo+iFeZw;!uICaXCS3Z3jg!Cs?qy zkJO+ZCv)2`(cVVB!jIU41;0!Yg<1`t>^#rE z;{)oRFE_BGoTa_mNUY6M^x(-kS2h@|WF*vVtcSa}S=&nya zF1uAej*~q!w@;pH9s$4+8dHw*3^*;Ms@g_qP&p(VslnXj2LGAfDGiR6BRIXAiYEU| zP0A7SeoaRoD^K7|ue)o1*8!YmD*n==;yizpDYZ{=r^N7OU0&V!`DQR29|vW zG&c~cH8u=BijA=mbb?6uM%=Z|6`r=78iS-MLil27K(;G}YHX2G_{^6m+=+AMhHffM zyKbE0_D3g*!njG$)>)H=()|(+Gf$|h`OUxQs(JPYD$cD;rSa3-a&-fwH#k>kkakbE z<4T#6ZxOND+G=&7+LJQ0B|Ng$;ALv@mGR>9^wam~+QJqqX{O$(OP_v#Zs6M`%Jx;~ zFzadRCT1n0s1^G3)OOY~BuIg}n+Vhk=cd&f)9(Arqa zmS@43cCVha^`x2aV)ByAP@$!@5C!?XL7XPhmC&1{mJKb;lAhT~4;3>IxgHPxS_@7= z19*bP?IM3aqI_$tE^1;mF62qzj7!erR620rFd^h_@#rru39BU=A4>P zPUkbXIlQz`N7L~@NQE?z_Jy*FXJF;SD_X^GP)4I917|E`^IJm}#W+cTdJ_!sBwq?Y zfO?ehkaTv*iG}_5=R^;{Q>c<^rIygysnWt3>~|l7P4cnEDWQW-2@kMzP4E3zK5i(#XCZ~_qsrxSC625 z)J-zAd!KI6p>Sxt682di%7Sbc6CHRBQU&z*mZ(K4oC%zyOSu;CNS%aOTognfah6UZ zGsp$Pc^1|Nggr-ZL`|t=-ir%=k&(YBD|NA_qNpA7hj;!f$d2%)f@hcct4`EM$A=+` zTw)>P+QVd1@me}{=b1yrUdc5h#F3Bl1b_;_0B5&`z?&<;LkQ#8Dd))G9T{Q|hI8$bG3;H4u_DH&dWhuEdE{ z^&tB>4f-=gz0zD7tI__I7(m6Lfg-6?0D6PJc90-<;VjM6+%Bbqng<}Ks)JTWLs-FX zc^pHRSQlRf^{u&9t@I*qV*bpKY6LkE@I>y+6c&qVbQ-)kQ+n8;e$f_?^A{UsTco(ey$blUV1EPE5vWSh-P9dgmv;@bTmRk60W4$ zgHkhL?Zy_UohhILF)lA6oBYGhL-dE10v?AQQlX^;JYp=Z!jC|8;X6TBc2Cqj0#1W# z1Uhk5gXHJiGiWdN>fXVEr8{Nos7obH>|WemB(3JD&Q6{41g*F2b!pS>Cr}bD8upIa z_D<9K(|{40+}gdS^>h%$_m7;24tR|?8+z&*t)};PNT34*{J#5Y^faGN7#?w9RBJG* zre=y4dZOEYB61Viphjnznw=47ePyKdfYgu@a-WEd-303{dPc^v8}UrO)_82Mj@)_!REgZ)Ec$0*D$?Z>))a;&@ZSR*}r z$oAS2=M1dxKHP;5*dT5DIvj*UZgj=jltD^qy(Q@&C_Es(vS{*aX!v3N`~K- zEuHdgL@AabIt3|Ao#ib=r7o%b_7bUh5!YKR-$6)>#oDBJPlR7~9e< z@PiCJq7N-J)8tKv1*ISi+-RX0xH2Y1h4O3@`@H0+P|(7`r(|QVpcZr7o)T+cTN<^* zUfP?gIb%0yy`#$6y~s$L#a8UB5)5F0U!I?$qOpHV$e(@5{Gf-rI6g*>&xxzxam^Sm zF2%_dIR(2B0M*+F&(_KdZE0IEOGCvggXJ~Bs@kBV?v_2?$-+X5eLbK13yz*$<@Kh9 z^WiX<-6E$lL@39qPIgT&KFF9BbgWtydeKi~p3w@kal1LV<_Y}5Y88HyBwq?ZPHIpS zjzWVKIN~fy2`)^DrY#Fy8C;TL+hclqcXcMJAdA~z`HIksq&;PKF@=7tt&k~E$27(J zt*pd$RB?Xeo^NFI0_9Wg6$t_=-99OV)5{X8L1W$wa%u~&526joZ(PnNgOyVeVX%@Y zm7*5*+yeB33q@s$NXkXL3byUTR88o`7E}{@@g)CC#}%O$X*(2rk+^T@#R2}8cuwd= zF0_qs>qz2-8W?{JQK01_EwInPM>&BuXcn59^J-1a^J=np2@9*h4r)OIyJwM8%+Fv| zY<@rv8hL*n^7VGE{$K=$2<=CA~tbC@N+D=daLn6fSVFU*n;I zz4SL4AozCn>+@~Af||q9n=Pn7S$n7cTMC%)N}f%hB#r9fZC1gm)RgVfN-jkp+A-(2IKSu7@w5mFq+~pwgNW#{5|2L z!YJsaInF-vDL*OB*rWea=QMKKRsBKCEWvE)D*vmE{iD5y5FB`1O7e%ey{ z_L4|*!Pburq%zq2$}D>;`_nAqBgE$3GH$-nj+2M*C-Le5vfc zX)x4JrfI%-+=;v-YUmM4U*o$k6e+2ilVQNoc1K8Jy%VZf#al~XOy$`gDGG@2ExQ<$+^i@!VX@f zH%6&Hy9ESO1RGB_Z!IVWqC3&!B#M5IOF(c54@7WW3X1y|k0$-Dl4Lu->n!PjoV@lAw!CrT27K zBjTI=ffffZtjYWEO_y}p53;)VN%MYC>K8WVVGC`6u|q9q%RV5ov4E(3qFJg}X|5a_ zhm#MWW{GM>#Eyd@=g!&OFy4EUs2`Qvc6lE{Gi-`zL#bQXu$Pvg`JGIwG*Vctv7J&- zYU2nbl;WJhz6nh|`n7^mjBfqaXc8gl$#|A^NEq`Zj1r1=0tYmYV&I~RpA)n)T!b2( z2ZFFTLHJldqw$bPI__$T1pDzT5}XOzk=;+r{sH{#+^@daF8dMYLemMw?(?$WfJcPP z#E)%IZIXVogFMPBPcYMW&JKzZL4@0m(sFhy+6{3|<*(CY^3xE&6xd9bzhph!iQgFZ zXNi^-`;pO@replGf8tB?KZ+aM&AEZydQJupo$SR0%^Ln}3cQF%C>30C{oxjJVv#sI z91s2`%_ZKvFl$$D%9&kS-_N@~Rh@r-ET^ux8&Og1>r~f|JfFL3S7fXQZIz*@moywkHb~VpQAA~JcI|Sy>}UX4Ll_}2D` z53M0sKgc+XqOcdQ1P>H;*}iNM`5JUUgz^KdVZymt>Mv(!8)LS;RmTXfMFOrwwUN3! zCDreZ*ME~zzn0hk-oAMK;AgDDA8X;lQEK6g%zHjU3yJVwvi?UB>(`dEuU5tCf1cO> z%s16fA_?IqNU^}dT#1ukn^+?Ny!JgD$OwDq+jxxym#7i&j#S@Y+7mDU2>PO&wpP7Z9N8OmkdYAwV_90X)ct6(=!(CJP^z*`xJMfM`l5fdzvdt@h(u#A1P3 z`Z&y;11E4>rgCW&GC>|xGIcANx;49^DgKYN*=PK<-Oy&&cYD7pJt^3Z1RI!ABdOa+ z(zzTh()J9hTWrtlM$+DH#!RDft#;gZdW8~c;L4-e6LUwJYE97I!B!n3z4>vCcP?f1 zl3KG^O)qEKTDbPyD6OR?DLRt#7}h^tT4_vaY98gA#A`642BXM>@@vyH0D3CkO_83Y zZ41dw<*c^@tGXfd2YMqq=QT<20pV4bvX22IijXV7_mH!kU4NXDGTEr}7zx+nH%Uze zwB-gpVLu#AlxHt#zvnZ+l~5H zP9&0A!>azANT;D-$EfwLGYaQF|2GsDLaIR zGS>cHI)c4GiX42B>X!Iln21}#^&F3q>q5w^0y4)t;Jlp=^J?72;{++I|ua+^{opnQ3SJV#5ETm{ko3sjrw52T%Z9!$2b@&}>0tMPoOoP2W1W{qrZ8-MzVQf?9 zCOUjTTVQQ@P^vO*Bo}ex>n^6Us!R%Yd#^a-Ty zj(({-220s96a|62OCUBlDUuwdBtr7;h>^UHkmcYZsyqnO2kKNrCJ~q-*bVIV`cG!N_Gq=nfJvv7XGG2o{c7v zyIi$1OohKeX2aZu77}@cJ_$KH6bppx9a4I9Mpa8#CG8T{B!JaTcDU$meT`G8|2tpD zBzuR-Z6vDfT?{y~ECoxPl^p_G2SjFZNvs51S^ZuTqFXR=S(nnr=>(&)lm|YU(FH4w z(gEPN@tB^J>s4`%rf;a4t@a=%n| zh^&X*6NKfje_@<^lzf@Zm4g3WJCS3nN!G zHI`;3H}OLtu9+Ez2~hs7*v`b&s+>Mb<>FI1y!dM2WwQ|+GMy0dLhQ$ia|9EU*^(i# zDo(+9E1mLqeHSIz2L%soo`;o^F~nEveuodb0lc`{c;dyqqsDCDha|G0&?Ljo6d@Ca z7*_RW0J$(6Es~g>PUtZrJ$T2QM#sQ(0Gwi-Qn4}-ygE@4XS!4RIc2VV z_BO9Nzw#nVg9>^fV@?debY7Gi9mxM8HB_klSEWc`lMoe0yaXcTXy3dPU4m<_B69xu z;Ve;!wkhyMV2O$~`Qr5b%LUrt`>^`gqALZIS;(5S0=%2Yel=J{(Y`~O_iNRZ6pRrM ztVvsayQmR#O-0XeyUPQhZCXmoaewKKV%9{baBg}jRXs0xz#*ccqf`FvAvJQ`cAPrR zv#b76ZvRrWL$M%lCpn+h8XOjlX1|7KsPF0BJM5cbln3O zqS=PnV@3~t>Wk=~?fw0;zWWGJyHFQFx>1HMf?7i=$;qKiv=QEtrLrxs)PSNeBL(8^ z;y%Q(9kAH+z}VqQiF@smw$kM?l0S?7Zq4EE6Z!ij{Iv#64?b-GF<{?5%|6NW;4krh z)cz#z%HS!vtOHidNA26`h+YQEU`6@OVRdzXX3Cf7HJlsB%yt~qWqHSuz~T5~gFDWj zkD`=Ee@T6piLxef!AR5ALc(JkX~Rl>C;feq6n+(Jv<#Gh&7VrR0(>yAZpaoG4_Dkt zkJA%BlXywb#47J7U%jQ`NJ_A%5T(bRbI+8Q6M7%*#p!@s0B?@90LI$)*ZokVg$3~D zh_Wj-3s9=WZ-fQ#X3C)D_d_gz|Jz1VI4J*e#oy9EfGS+9S^ciN4@XNL#DX{*j8Ur5 zLZwQ!C<9GLhay<`s>gwr^9eW(TrOf&dmHB2IIIe|`!25jef*-Q-}`TBZ8YU5S}BoQ zX)oA#nhN-nP+Qo-n|bV`=+4VH|JT%b4BdfKgKUJ1eLvo@P0%)v+M)}WQ|J8!;KaZs z4Pp#IQXS^TINMcSV$vp+xLr7QD56S`ip0|n3{oUs zidfCwHn5fS&139yuE2FYEI@%5(EsHoIvr81;)6VQRex8i+Cf%2T&vi=BSw=KRyzMn z8DB_~mmBH~tHrF@>g-T8K=`gbjJeK^0b0dVW3l2|#RH6XCJ-)Q%??|lu-mzS>M%Fm z8BQ6Qtao^tr~%JL>dM?0t>UsYO&g#@C>C$WSa}lHfI3yxc-MBAQ7NleW7J2b27gQ> z9U}Wdv4*Pp#XmyzSb&BNiHyZ!JM-tX*ruw0QS0}U)Z5MW%qd`Saw8%sKr0#+5-C8- z*_8d*uC}&{6^{L!IG<90RyZ~@%_$raD*w6@Lz6FxZ0f|nK-N3t4%HX`LWho&$a^fX z6lnu|)pRQ7Hz6OMaA=bIwIl(K2fRd4O64Guxab~Z9wZ#Xe43Is;oP5yNjr=LEpVSE9YR8G@*hI`fq3AERPrS?GpI)LCTg zP_~0LRHDh?n1TF+lx+tMg-*hx#`6rGjnGL_7o6W*^ItUi@Ar$o8D?AiC8B6s%hZk4 zC7J2DTYcj7NnDRx>}Tg<_p?lteu4cAgi+NbNYUl3Aue2)C~t)senj31Lq56zR^A3P zgb4c#j!2*s7W$s#H=-ScJ66e0-7Rs~k}#H$w|mMX(Eq0fC!uLAWVDo(gGT9^CYDXITh~rzFs}g$osIujE99 zpaE$GWV@f%D;gk-zsdRS!x3E%&>GAEY!d&$psaiIs68uGRCv^$87i`Y`93#|{yS8+|SjyjDn8KoI*gIkv6o-68c$Mcs#(gU<>jMtulvx zmd@I%tEJ3n}$8TKqu zg%9{r9g~K1x$mN6NkT7!@eU82mAL&8PS+};#$%t;2Vo-XgQsxhkD3iY$^1oMX>yCH1u6R{{rA}*-1ek1p<`cu>-}y{Q+JMd=Qdp3aIixfU=Rx8sw@b z+0Vxn^`M`RLiFPk&-X=+^6Js?a$7n9`-$p`shhH?JqZ`5@Wv8E+EKo^gm!cAZK(i1 z%Im3RKf5Yz-et_fH#2*$Bw+CM4N_YbiHXZ zizqshsyGS|KQ%L0J>Lq;8=_dE(PP?4xu#7Uc)^b(KWzumK}l_xh-k(~E93NoSc#=T zHUK5Q_MU9UDkcXR?+tVrjoTSIU4w4rL=r{~f7JcyKCxUD!YyJiakThF+PKhde{ip8 zDKwqi0~-gGb)V~5f!uhN#n9`*J>}ZKqlSpK&Nubf#OJjB5`83!7HFKTqxv5uu|W7? z?xZ)ism=juk!IMYI*Mq~Jd8KOV`p|#b4q?{E73d%N}~_neB{l4tAMCPpq115^+~y+ zaTy$?Z6#nUH;=1N!b(P9Y-=V3zU@tvw(^YmvOr3pV!?-AW&c_Vfg!EG{IX75VXz%v z5JEo6S5*6ET2x=U5&I?I!)04~e?quXriA>cZ-JX-p3o*1Zmtvl9MTP4w$(|>S1=@r zS-_9-RuIb|mHR(P@8j)5dkR^>(}ps@Q@E+I1#)8_Z!xewUcd)RROfL2?4`=6^CF&d z&VNSh-x}_+sX^^BoXwh)_&q8;L6nh3YP7gBef6n&PXf;NVb%VcH$7P=k|_kIl2os0 ze~09UxLQKR3<72FVQg^XZ~(T9QKw{?oNP%(u5nQ2;L2z)r{Ssur>01+JEz!2S7!jFkHzk;3!w-uG`O>9=GI($@g zWP1Z2L!9!a!60*kI=q=>Ki^>Zu0V5_y6yVwNh_!cE-~^_SqV2ZATMl?VqHu&Bnab_ z5J+v9n^^uA7;>UhRr7}4BNuB*TTh@ilD~(lzi0b76y1T|54JEqsW9v9Es|eBtmwU9 zixlX_SMVbi&;jEvFcnWzEMEeJ9Ys6$bc)sihq0I9jHW+{b&}I#%mxkcJz{f~0<=mB z4FFLKBb~yPs}< zQ1ym{g^%*~9MpIMdzfOHp|GC}F=6DL=D`;r<2Yc1slXsE7!MWl%0V10a>+&acfkkJ zS#n!Pcu2cxrIKExML+UR6W^dIPDP6R*n!Pzq4FDcao(N#=4goA@F?mgZ?z$<+_0U$ z)1DzW?4b8;Gzj6~oxjt7Ir%DPzU*tjUrw6wcwZ9*aza-2t)O?z)?5FEt#BL9*H^2C zP32)cf4=_#P7t?hL7e!C27&~guvZ(3RAc^?{5 z@}ZcLdEx;j@7LVM0xzC+V9BSczG}&*hn9RqGhs#QBST7#tus%omdrJ%KKK}BQFfj zyMIvbM^mu`bBvXi@8Fdy;?P_!Ol7m1`&e-@8zJYv-HY}3S5yt=gu1eY&3=eS-D;pX zXQ})vsx72Ehn@tQ6U6zF&7>6p3z+)S+(b)56Xyr9t+b-3su`jcp)dEHGAK5+VDVl= zkq4Di4+0~>9a2{t7+1k|Rj6ZZe~#2Kez{95?nQmgJelx0xEZ0u35nHhr$zo-uJ31s z1=}C3 zt(xGgH|tFs{{mF7`!?yhn}MpmGa^Yk-KuL48E4u^UZ4>PM`E8OPQ0bz6Pux-DOcY# zmfzAvu%MV%-3ZwDy1P}l4-DH%)-2JzgrD=theb zjVb02D2+lAMH-)&w7_B(%@}p*?wgUb&;r9x|C7M`w*=rKeJxiu-kg!~jc|?&6_~_b zma?(tz2$7G4kr0El{`A~p8C=Tbl)RREU9l<0(11P6hIe8##@LNybn$gJ#kHmyC}=T zYB(*lHSYKHvhfB0vC&5Sh4g!TGW`}Bma)-hdUG0^$B)CyGN~}R$vLJyfzTwHi)W*G z3)a>L>)aT5Z1QmSD=_niE$8%|>ikps2RTG+pzk`AINYiT!%e#qPQayU^>BEgE(_CE zS-cZjMQXEQLQa}{RL&%KQqJ}6ct?07WM)`Ge}C&&5$h^TU5Kr+_|C_9XR?CSmurE0 zAV2dw?xG&+xxDcm_c)j`O)u7a#`s#|e64Y~efh+cmTUha z26MV}D zE2iMY=CUuc%MG5?X8WjYM}jA%x#;ukIf)(^_a$eSCz%>)1z`4WrYo58rM1>rx07st zO@1;lXsG`}=g#A$qO@A0cLrD(7bs_Rw!`2tHkXacE>Ccm#%73{cpK_R>2TB_^hHLK zFPzax$|wP-x<})N4Hl=O^%kpF&K2-z^fP4E+s0c;`^2S>r5 zxS&pZ84FN_N{d}}0{b{+GRHoJaV~>;!HPgQPkHZ74Ov0S@!F#hGgf<47N>6V%UZ2` z@|;#>ZG#c?&&vMwx467QZR!}i+Oybw*j%Gi+1*4s0xCV&llv9KW6j7D1z>glwRi*d z*1tMKy#W3*V2=b?)lpcQx-KirHF&z3h4u!JEEpXXLn^UAYYeGZAYVe7NSHncv!*5HS!&r-mP`lR&N=T95H zWqp3TD7rrmw>g4w0G0l^mQqEg98n+7f*P}utljB-y;e;J=7`qMwNz$ow=JTRg6KD*7e`c` zEW0aF7*Nyqt%jUM?_#YKYk~4yHO&K9 zr}RX|Hvm#xMr+VjO!;zPfHwR!{qfRh`+@~BY!hoJNcG4&iiO3g)Bp?eT4`$Z14Z9j zBiKqtyy?8P3~CP`+>%+&Lu^GV8g9)W)zxpD7%N6;l4kp%@9Mz@n*Pr#^7e?UNW#5 z4psIy!wNu`DI;@uxYEnVn%fd}k@3r^ z*{)nce|xd0XS-4mQxSO={Wz+ev&>Vl@OL)ctn-X_^ z4FEMy%I!NB$So$#HBbVmz`|ZD@^Fu@aGenQiGlmtN^Z-)Iig#imhBI!Vx2HdiIUqMtu>&5QYTn_PvlpO?6DqMvQ{!!(yc4jwR=G@pP~@+#?Tq(@3YU zvl)8=^D_HfSwNsfoYT&9T@k6V_}b%@@hF^#pk&EI#mA8u=LW1b!0X)8Dqi~#5f6kx z9dyw4{9^W3aADa2zhQP-XaP$pl>@Qx2EdOdJ4?WCMNMcsgL{~_;2!2p7CPLR*Sk?gNGT@@9*z@hDoUl~$cDR$yU08GE5E4B9>8^Wyk6 z%>4edZ8CS{)mIZKafYu31fB0ay@R19U?~R(JX;-BR#|~|hhhfy)vTS?&FFy^_`x3O zo8=i(BGxmWlwnrrz38UWr_^rhfh$#OyL%*=Y>&GMh4!*Gwe1Gi-`6|Jv}p^vj?GPd zr5EEOFkAm0jF-!%Uyk$ZQJh~F=%+8F^J{CIZ%-VW=fsr#IKS@q1tV}kZ3?XTPt7Q* z+%IY7w;?<4t&~-_nw9Dw;@jpox@MHGyp8fJGFK?MUX|9mPSsF<9yO{^ZhMkf z7CYUL-D@ovTxswC!qIY;9P5Bf%$l%Rc^%sQkGs+pIM}{NEHCiNw5Xp2ZAJwVtdo6~L z`3ZFh5nV?!8k3g~qoY$*x~@m15(Z&SlP*%yqQ%Su0$rJ-WqurasHY>|gO_EA9)ze{&0U#*A_wPoFVD^M7v?zuf`QX}Lq(O; zUhp8qf&m{Vu;C`4PenAZP-9JeUr=vZyKpgsa`Z=hYLTzkY2^km+4Bn3gMEuffyLFD#(r{6>?RtXeZg;%UK1qR1G&CTwjy_keu1d6&g6L zH^b%U)&Ra`|9HLze$etXEdDFMCjD^u)m|k0>WhS5bCK|CzXpDo_OH{xsr`q>ztn(r zHx^_-6AlC4_;v6X8ebclDAmv*c0u^f7YR=~92b-S_Tk}&>Ho*ct%=q^gNNvF@PD{S z_&;7G{4-wz-@N`=#P$;zV*h}uj4)A;*{{F>82Pu>v2B?BpxqUZiOk;Fk6=(f_Q(DKiilc5)=ey0L_Bm(>|vX{2Ayj5AIN+^9VVt zV2CK|#h#AHZbV{F&!8)#a^Y>682aPs;K>i1u)ZSLz7l2L3S?4$8U*j|DPQ96^};Dk z1^gYWmARrqx$0AazJa69ed&Pz`W2zCF*Bk&uS&xK|Dj#V8E9sw@~3-=AnFpUBrIu1 zK?UI8-$5%L_1qA6{kZ5BUR4sYZE8V)^6F6xo>(G^d-%Zmx|hG8JTb+3V^Y(d)Hr1s z;Xi_3RaK%)1(SO~F4{?tru3rlijSICvM81gU8#y5C;#hQ4LPx!LMk+nFORjGy%>V z!y}ce{{a;+Xd3kJqrZ0kik|=H1}WiLejshJ7I9I7P&YM0-b++_}VRL;0BvUZc*F%pAB9NP$17 zk3rAArHn(Wa^=@35ou6+WvmMC>13X6B%_1Ze>u=900DwVr7IQ45rNZ>mxsr~Fq2AF z3EMI7%nZy<%>oL-_3&X0tv`_39Y>4XqXMZh7)M2llKU@?*BYcLR}5;Ahm|!dyyuA8 ze?I|ExW93$zzO~hJtvA?$%Gc@;(oM_aR3!9m8lK1=;&QB3|g0HXicJaQc>~>l>T)a zp>&U0;$h`qMWPB(l-@&z6^c^@!2OUe1L4D}PL#%?G%FS*Qf|N@ZY*rwfzmz{L2in> z$xShN7{i$CX3<9Ts3p3WbL?1^f;Ky-JV*F>NmkRdMLoKZtcG+R=zK_B@ap%~^Wm)2 zsNChRM&8n9Uaemur8)vkTD^V=%u&AD;JtvWQ0^5alqx61%dp2?mA`zfe}1g*-eruD z_6=rH8l^za860*3mH1mM`EB@V_tOXQJ|g3rXldQO9N%%fn$V{1gkAISgWF~hR8j** ztP?)kbPbX5MnFQk=P$vh(ngi9%6OA`gT(;dXt~*C0Kx<1G$kI=aJa-z)A|^g~G08{W%eMI|ADp z69@+sNg_h%@7 zI=t!Z{v0or?yGAmOR*4SU60gTsdq}~(}Pd?t^dJSk-%=MP4+hW>#lOyfE?Bb+7q>{ z*$7@c?$m;V!gka-%Hj)TWy8D+sUYxiVv*lXBLSuM)Xfq+C60av!LYPOr`l~w}7V0wk`O= zMTrSY&rx`ITbqW_{0xGm-OuB%$-fx-RxSfz122-Cxh*eEuUw0D01LFV8#c+2X)|=6 zPcWMr$&uq5^hy^|D~C6$!(%5v(A~7eU~G+JCgKe%jEwh%4fR?Z-2zp*aTPuq5ON3sN(C%R4ccI z1*m_DU7ctmXhi6hV=IMA(-&zRI3)hl`mw?HIt>F|fd82Mr(lDlHj{e=T4krmHrguc z&XgC}N=-@Pfn{l5*D8FxV3brim1pnjZH zUik7VTB}I70Agz&YS%A({wX>Lyyd{w*9bRsR0n3KkoF`qenc`Se_PS?*!yJ^Uf-2E$4 zBW+Et9r&ZHrg$7c4AP4<8D2<)!)Yv13tcu0eq|A2Db7#b0Rc=BCfto|E1?$%Q2sDV zBAobTBvkJ))#oG58t}i#Z*}Vz@aZ~kN{iDezW+?$kkP;>1|Wx#UxFepfaJ4F)EHh) zxAkYOky{z*=W%Ju!2f}*RCoyph)=8Ft9CbiC7T*Q!gLr2NZTZ}4q4&siGmtT{&4_h zJj&ObWb#i&ZOOQ31z=GpYDU*<=(cu0g;3kJ`)l!r7oD>I=JT)zqlkEuUqJ|l#1A({ z+mYHC$Qu!Xmb{7i2|KtTc3)m&aJqV5hrXk(CQ13?K4HN=o($aCR_Zf&xlmra62pUw z*R`WFd1;PHE=_Lz#r2*RSpolI2YJ>Nmrf|z}A3&7X6bs$#b z2&*fEpJH$?OGT;j+ojRDz|@YRN-K|Ddp>gVZ6qbX%6`fN1!tmQbs(ulkEEh`15ghP zZGg}3jcowl=HyQC)l?13ex;Cin+In;N@PCuLan=XK<3fbtw!cF1S_0Ot>{E;oV{=- z7~layRoQjTd3Ag%&*F8+kWz>KLFH$QFnZPg9qFlLe(J@SyIV}=Z>e!;^3TRvCgzfo zJ6u^}q=z5&^(LGAR{@yuRiJ-?SKVChtbp2>vPt6IKip=&4pDKD5SLU0HqeLwAEaB1C|V7ar$LCYIa$<_rOIPR@X=e7 z2I~BFTpO(Xr%8-ix=UHx*12Gz2B-KP$Di* zg02O5q^pTImA_r~RpjIc2v&Z9+||owNV|jY>2z62`9uaEVzW}=VZrDuQI6nn3lTL5 z2&Y}HjlhahSD(5yX~QIEMIUOC4&M_MitY|oz~#mSxG}aC>b}gU-8ssR?*;ux<3J7n zGA8T;8p0_+m_&{%ErHpILXfU1JfrH^9w1I{@wLQDxYp3zj(VZOMcta-3>sx1c?X$U zN>qQT>f{5bq84tVzvuPq7hT}K2I}n@E^JMKP1K*o(gq^$BJT%&ejK0BE(~lSj*2t# zzViEpiR3W-h6l86)QEQyK25Sgdk`~DZ9ggT4O&FQM)f$dnHhFlN-j0SrOgPs&s?bk z{M^+iA$D1~UY#cH-Hs8Okm1*J_s3VQ88&}8L@v7ods1C*<+^(_*(s+pRVq*zGRn~$j0 z+XF#Ul#oBB1wb|Tm;dxYP>b^CSWu$?rRKl#tZ?z97N4p##RAgq5Uo?mz&z8Hs#s83 z%>hbz?(ZFt=OiU77LZovXrAX6s&kMPicA4dNL*!ou6swh~)Xo znpjYi0HrE^d}cru3zgznKyyFfZWT~YD4Y)(HchTx!fQIa-Qm>ZpHw|>nN&T$soJHC ziOpiH$YPAEkS-GFQ2sVvEOkMp#wh>139-l~cuI0Cmw!&2`gX?(TDj!Mx8pBrjmz}f z94k@lT3oAzQtIT#i9s*D-oXDh;%__WNYddN7n=}ETiNJ~8BgO7!u!d9;;%iTl4eH2 zUR+yJ9SeO)EOhU?Dl~6KZtAUM#zLnEXg>d2g`v9Wu?;PVQ4F!LJ#S^FyXr~->UyBVV!Zx=c2&S`h42m-+QCAQ!{1?DF)SD@Zwr>oHEH4 znS8o?Kqiz!jmY7u*c>bx4y#?86-r*~g04n~^4!qm86x@d^)XerOiTV9S6UAx@Ax+# zH9s4gyc`4QqM6v z8TNbBd<9Ztvww`vx`UOdG%XJ7Ke(|KEcvsPVp9c9&o`-Dk=*ITr99ZcE6!GtP=Q#u zKj6VTMQ|xwYz){>6FvWVIcY5P!aSDL6?l-TB7uY(H;`~cIH{5^Ax_IGk$4dFN5}v= zgzu;n)cr_Ys?1i4Ukp!=y~LtqR17R|;glIxy*y|KPw1X3{1Kj3i@KK=m8=#uFvZG& zMP15^`e>b~>0(?y*Gq?)G%5-;2^1-)6%U#rib9RFVisYNlq*k(w1c2c1B-bcpQXwg zt(cU6y_Xv9&#Pk2YNaFKDb<&?-{0 ztu}hEvF;-XF3`ol7!9$3g6mUtY%xeHgIakrAv&-< zt`~^Lw8vLt3C*n{(?QUYc>{WQI}(>FH3F2ank3M?3TBx(kE$p>i`*)df&nO25sIgi z3B?--#kHA)A`OMZ%H1eo!00%vH1k5;S*ra^v^`5WqTThp_Jh)tYlsR7&l!AHD6ImN zt)0l*}^b6=J^;copuTm`S> zX~EfQ@WVXgNg7g68^X{zto#&t4j4^`l_p-y;Q3L)(|6a2`p1r^!%F!;#LnOo^%tOQ zZ6V#ib9!2V(YtIjGqJ4oz^IgcN1eClOkm|ITU6j`qZ<5A9$c>lC#%7Scxz4h1?X8cCsS|o2 z>BSY|Corp_(G^#cKy!K#?u-rOk@2!!nU?9j8a4nqe^y8P8)Im+FHrL;QykXGbsu7d zNJ7!2O0EbGKxzhYG@O9{ZY_o=0yWLc*bpj|C)Ai3OZyKgF~^hjZFD?= z%pt02Bh)U$`Da7_`N!978UDO0d5+^6#cpsm1CP@Ys8p&_R3-`8d!#%0eB^5PGwA#$ z@R96MJ8>?)>Px26AEanp;ceJmkvS%jTV+*jX4>=?3S-b`>t=?Ba2YjD+$~>Dz;!rA zoNr+)!7g(uPiJth^C~i4cxg2?t5LFt)N>N}J)oB1rnp1#qI9B{`^`8eWpaxYc9``j zEr_j97V`9l8ZHlclhB2Y2j^2O|3)!_HMtN=;6sQjS26$&L8`9>=3$r@$HfOu$&t?-&`B+-g+qZk`Ea;h zV<;|>9GQxhhh$5RETpb6+Hi+cIt7*I*BE9>j%?=0fkW%@TDn4!E=P$+2o$+wG{wFg z$x+DQN_R zC^LMAMmzBW&lBDyI+kc^#0}!)mcI);MAHL*!zl@#S{vslnSmvI6+lpN87@Ft=9H_A zaATmpTI6z^WnizLEdo+o30*krpVw)OAYHYYyRpvJ@k}g~3G$N5jc{g2Sy4U?{F=Wu zleE}OI3~bAE{Cr_kyy=yFhRY5OjoNs1QTOyCK@JcHH>n(S7+mB%ka3bOKS zIpqKvDr8SX9prhfgPM@aut4WM8%-G6OSu{`hEc|WuDz&$c2UDP3N-eYfG$SJFr`qp zMN5gcu~rk5ViBdl(yz8Efzee)d(#X zTc2;Ni`GjtnlTF1KHL0iS&~^-o@BP;X~y#*p1g`A^Y8JP(1~K@kFNi88My3#QC__< zwAiR#kRlftX#zMbp~aSHkOc!74x2-Z&EnFPL~;^`G3`hTE#`M{C#*pbdO8ig8((0v zI`X|sSa~KG{a9P_l@ITu)g}we46Q8&T3ZfL5Y`qW))uTd*;rd%=1^FZa%v2@v?ygO zPxGKc$&teoBHb2*h=pcK%d`TU_YReT#ws#d%Kfy+RAN3x-h=ehBVd!TisKvz=pIm1CYF0O-o@s zd36qQ5XlQ{dWQ{G(xml7>$4uSXw?GDpt^-g(T(A?7`W8W{2C-bQj>TDFEg;T^*zr{c8qhSvV%k!e6}DV_)M zJb?%GDdrQ`#+J=mFO0E3!Zkq69 zE!J+mxqbpWD9)4Z*qd}Zo3GViO|1JIN7p%qj1JtbXh0~r)yZft4y5=>l&60N3>_ep z))EfYMt;69OE-t}tzBO1%By%c2Qb`yG;n^UY@#iMX9@dUh7DCYm6xoDyMJbzt` z#oej=cR4`k0Q5Ayl-o*_FugeDal2FZIHz)ye^)w{ckqG^ukP0^*4DFJkdmZFl^{7Q zF{O($(OOb*R%ij=yb{)=qBZ3Ow4llh(6SbJfkmE~CNJPC7%fE$qr02{L}M?EZfPP_ zt|hcEy77rD&=m`#yIeFst>M&Pn(w&=ZNXo@66-Fwdt8pHceiO02o0Pee@j;3=NVv|QV|rP#37Ev~>dLDf(ok#V8S zVYZ+W42hdHL=x6a^{ut(<)9aLG$}O3OO?q4Yv+!f;f0uA=2S*eSc};rr1tF5QtLgJ zAk>j5+d%uTphExvuR{PliH~-PZyr~B3bd(uGgT6HDV|>!C1XwMsP=^>UJX;vlZpc3W5pI~kkWI2@$^fEWQ zSgEI%ikuR~&7%sP%1`l9q5Qapu2^GhE78K3vM2K}W+op+FE&+u(eB6xI&?YU<)FUe zb8R5y8dlHOxxpa9F_N!CCvk|xNl=;LLjL9XGi!{TZo2`WoM;1!&^+ZG%-Q)<_D?Ic zdA}d^*HW9wavlUZCFMNAYOw_#1{Ef!zMDJKT4)15w9tl73NpYi{ksjdD0URc`F6uY zX~1bFvsGef|FOC13tpQ zq7$gkc&tg->cN|4DZ5IP%}?uedPiY_)tg^;2vsGju)L6;S#j*m_tmZMyB^+JaCO7t zrt;a`4u-t9@|<9I&7a~h929~kr+9 zt$#%hD>lf6_t-sHx6JXMGbif%`C_bF)#U7X0bD*~`AU~<>GFb1BGN#;QHllT#pDHg zfdx5e$CP);qjMny>RrX7}b`HIxl)YR!7sOk~g_YDTMS8ef>x<(XQ|*gXceh6-VE zrEGyQ{+Mj|klkwto>~O5vekD!e$}kV#235KYVrr<$Q+n~mJS}kpDsAm#}9n+tNhh zG?ADW3IYLDL~ZCxt&I_hr%+;>c`-t(kG&Z@6oNl+?0Otia#YN}(Z!VUvHc-!IMEjx zfPZ6P#l)%uOwDnB$i;V2k>3GcA)H!i(C{?81a*k^mw+4zO_WEe2j+N)@*rM86wCe) zEmxF-x>Gr6QQTZ&3aFUU*T8(Wx@vXjwL@E6IM4o4`thaK6ZwV~%O6x@fMJl);KunX zYiW{s8J-Vqa|2+8sp|k3I*ei=KwZD(VH9<}0?6Nj=T1Bi z;L++jpnnI}cV7+F7ffojw;Am{00&-QUKd8zN3Krjd)7&tim(6zGca5rh!47XCeGz-)~+#om&>o{9yi|@Oms+{{OBXM&xbAW5JV#2l(RzUT-9ldr5^-)q~vt9J*@t zDD)kbk5|C?bhyj`47`H0#o*BK9UE-Fp)9{)_f4UOLXGPF5rm*5i?bD`0e z9+O$lOYb@uZ0$O9tW_CtJ7_MGwY5G$D(Zm^%3OoB)yv-#7%oKaA(x1qB)V)hJ@^z( z9@b8HB)P8=;wk{IGO`01rgpo6!Kq_eo5=ajB>-Z1Hi$?}{vHaRmC3xB@h$N!(9*rv z!Y;LrHTmC11nxQpqrumXrOj}k0%~?r^VsQ^n*6U)T(5rRmCeQa=@l?0B0?XI?^h-_ z=f+Jhh%@;gqio5A--WSBrsyL&P%N{n7;w;&fm7y|@V?v<&wfX#uy>;bIvs z_Yud7gf&h%FD9^UkUv!?t z0z`Ow0LuYx13;icA1E+bJL~VT78}-8fi`7(GW5Qj4LY}-wP)yA>IMp6h7Bx}D9%h! zoJ*;zz%#v*m^3z=0EtP1o?1do8uZ5!_4JppCXSs$Yt4i}(Art|IK#bKUEnH>&6vir zHlqbGL{na#5~*1tc`KEXXaq?4;6i8?PU7I8mG#Q=u?0KYu9hoR|#S&aj0SRGeuvK(N2vQW~DImtih&grw%7$|cQqc)(>$-TH z=)3kS8)Bk+5WUg>4PI*-MZ@=Z@}P}sZ4wpGg$ucY!M&l;4g3V&6;8yJ)cb=6G2Y~a zT7<#9$WaKHo2nw#`X1AR-=|qgQ)US}%*Pl@Zo&}~>do(B_TW+?TxcCCPN!j?yJk%A zp!%+Nn~~?+;#hk0_Z76;?L+%QqWLOc1{sYH((ILi2x@YJ_}r0tw}M$Y4E?}UV=l5dy$-* z9$JzfUeFH?Jlb$Pp^678Z>p6kZt1*c?t z7lwswha9y-wzr&IN*LIPn$Q@J^VOQW3sLT>**Ldu;f;8jHxxt}ST0m@g;>%+lNxq} zoD1saH1=~XANA_(T+ayXSGPLDBY`M(gTkd#z9i%N{caDdpz4jL>KUnecM+2c%&LA| zoEk1UgbNs`Tc4!n27iV#P{Ezi7^vXQa0V(kGlIzz%Rr^_!9aajFi_EMaDwK9U$Jl^ z{z%@X(p)Eq8R#@gUY9cAQ!G9e`TFt|5yx`!k6YDy@&VW+vb^OaPT4k@3CI)8KOjtP#xkwY4!iJsz+VIwIro#<`U$DXjE#0vgf{lKvGB3f%`A^IK!KirpV~EGolQ^~XIM5l+YHMl zXPWU{NbiBCg?wznnB=)9w^3#^Xbn^L^HYQm0m$tL4Z8dC5qREE_Y<(ce0#v(?Fr57 z?b@%=F{b8nA|>Jm1^$QvafXhIg`CI(&7qmb*w9N5iXIC%{_ES5X6>lkW_`;P&?689 z22u_T$>n1#RJZ`$Xnm{hpC^N~YF!XT!BI9fzW}Hxidx_!6sV!71+M3T8j4ydDL)TE zu^NGcP}Jb_5EVu7M+A346#4lRBn+81L{ZcNi8_j+s0AM5ff|Zh=$i-~f+7)0gHY7q zw<9Ww;v)p7p}1ghM{_2;-g8eBJvE>#ikccwK-?11xoXJtSd^Z%zEwYI5IQQ{comg@ zg-7Lvjc~~qj`y|3i}49TnYHKy=1S3tnG)X1px6@+>VyQ*`E}zME=!+l)2FNtB(Jww z3uuZ}z$v+n88%}(v*yw)%dG{U9HfoWn!DVz@sAJ|VIeF6iB+r|EhVvsA?jos=g%Mh zPm*~B9_=^uvn2BgJl%Lsn+TJgMvXT-g@>lDvrc&^6Njpq!WxcidKqw&<@ zX~5HjXEPqRJIP#!rxDLaJePQq%va#~9-a(5x8pIaL^*hFTh)JdQqRA|3<73mn;&c` zoqQDgB5ky$2gq^>Q#NG{ZH3qdAazkkqyHmB(YB~cQ%udBJfL4L1M%<+dh{L0h8WSz zd~98AKgnJyW)lN}lN>hj zI~X8?0QhEkmCO&>g8?|}fY>tMmUU19Z12Of1<#9kwEB$&lXY?n?beI1TOVJGMV!^W=Cus-N;6z8RxtznPQ*<;{OC-v5FPY4hAT)R}G}e3) zo3Bi@T)-3OdODxibB?u&hnoK@vH3qstBEF3+^)h$t_AjU@Obe&hlltFV( zf;&SC#fKZ>|3dA6 zPLRihyE(1_9nr8lq0raSXVWP1x_Vk9Y?uh4CE1X%LhTUClF4BhSw zo)XL+y=Slj42w)dHu-pF+>m3`xk;@8l}~|425b}GTah+{QBOij zT9v}_0CP0tjL}IR*q835A(%IR@_vZ&@3j`p_uS3mvkNQTCL&>k5|rMP5gsk$AgF3i zLHLZ*pwO8DQA0G2P>sr66ZdC*|EJzj|b5#ABZo0&U zoZpgFQ(LRB50m22hYi`#-Q~8Z?nyXa0$CFqRj!QN1M9Vwm2j*385`>85~`~W^~pH? zoT4^gF`H~AyOH!qU?N!GcVEM9TWUS*zH~Fo%|w}CIaKC>r0xX~Z$0e!gq74;-}fAa zuTfaN!VLi=~85W1)E9Pjt0GM z*o3@uk@tUCi4&RrA6rys{m}D1xrw)7%SINS-K?tH)w7T&dGb+2lst%d>xbTy%`EwK zRtClMk;-?`CI!~7>OL)3l4b)FQjCSerqfF$vY*#2Pa3>f?CE zRNxLr&Q`rzrrRs~TDL;Qb*5+|TBf&0=g~f@0ECx+9{sAj( zATvccZFro1FJ$l~+V9$VG4|ybm~ZSp7pBBfS*>H; zBWwV(I*fJyn6@^}Cov4kDwqU z`sYhF%Z0Wsr$)Xn4R?N`8RHQ^qJckC;fnDkT+~>aPOR7K@a^+XG{BZ0Z#V|mgHs2i z%kC-f1YX}s?Pvm5U`)fbyWzoTwO;3mm&&?*5UAH*h43qQ(D-J3PHvqcXQ9`ab9-G< z&Kj_x%8b7H9$h`Oh}|ism-d9Vw6(wrS?qF%?U{)owEi3o%erk~@?d?RXnLtbZhVWt z1h*j9V+u`eB9#kB66Hnd;Syq2K<}tF&Gb^I-1uHB=%`SO29`l!Jp!ynZsY}PImUoN z3m*iSBU{z|T{{tsy3opONC9W1qqLz51xQMq5e6?GF|MHbwkyWhqS*136V=Wc5nUT!o8t z<+9`SM+uLT!DpP7G~Bf$v}O{61!-p&&Y2cG2qbg?k+iJT2CEHwxq8t7ZAE@6~(Qr=h8OaK36X*a0 zW>TTs*HXvE!fwGF={#!RCnxm4#96F8(&MzM5W`ijb{(~MfHPaY;)9|?a&;}ANJTBu zJx~!I(C1vibdLkrauZVYH?FQTa)SvgXTkr{j?IQx@DI@ovEZBNg;?=X!!(`yN|w76tujhlQ7PTClm|=qEYV)!)P43tIpxZ^zek`Qh7Ca*Z{;=V zPyxM#Q0l`UB+N@1QQLUIbbd*GnnyYhhP_AiM@B z$@b@&eG)6mk};Atv$9NT0SvGm@;KSvBwN}ZpvKk*_dlmquHnu@wlehk`+Dqz-mCSuC@py1}Yg5!QLRE~ou~6m zhpL(!3^*Y0f(8c}5p@OT%%?GSBL*QBKH1(OIp$#z#Aw9gNBm*rMg2^z#JuyR`c#vD z6KZU&SD@X7VHhN^lNiClR4<4~olKyM- zMfp5tc*Ut~xt!+1s|7R}>Qn2mBzN-MARxfn+|`fJL>(xg`qYfC!Xw$7d6*L6@RSI8 zF<^KFYTZRJjO&987;fungi|ZIK`G!kf^i1{Vrs_TDf_LxrpH@ZD~-SX=!J&*be($@ z$dRe6tPdlFYdE8&l+Bls`JDl`O88&Lf_nj!0*0%s_Ea1ks99Lbda4|@Zl$C@b;E^I z1Gl5z@kC{V|C}wOxeDj zE|tb0Hu)RzP2HBF^H7>2JlIWX{uWKMgr|8_qyhWQsiwI?=e_|`l(uRrC@r{0TmOOD z^D6k-Q6%L$8Bck11rN5_uh3%wgGpHx7P1}m+jNp-^#jq))h~qON-J8fK6Fk~pNxyu*Os3X^m9MGsCuXDcRDK(k{tyDyLbE!DWON^5XZqTV*7qUQSw^|$P7Ezl zH$1ip!!@QN4$A85Dx#)qzLGN1DxwAcm&U$U5iKyx1GS21p&uZ0NJTmb8PR$%715IX z0kP4FkR_*_n;cricXGX{@oyOPF%_w&tiG-yYC8Ws%1En-7MMkWT1B+L@A5#cB3kHJ zgbt}lgpe6h5iQ9_h>cd{2tMT8)X<`IT3qy=tH8%7?qg6MF0sY;u&lpiWZlnlvjX{v z`jez?8B<5Dilf%CYPeCSX6cUBPYswKt(zKf8_D#=NK->*#MbFq>-+UL4y}(0I$o^{ zBuq0QVcJ6yrakclzQP8TIm6F046nzNrAwc6k2e7dO=W$)R$HW~*W`Z$vz}DGFmLUr zZBl8rZ!b7Lr>$+KjXy>d3^;`OPcre(v1_y2vI8@-eP8LlDZV|mfwd7f+4@;MZfNrD zvCZ{-jty@tdsp56uQU2uY`1xEAqN*;Boj}@{9uwfi06MdCz*fx&@k(XEqcO){YMJi zWYdKo_2!0kU~YXT)gO}Vo$dB6bOXq_7cd2&b6*rHcKsj?9dlT@Y!wfIp9WV355b94 zF(6Q~?;}IJZl`OKfc*DLYy{p)M0A)MGckmSj7eVi9oA0h@=IfPOcN&~ct<5BJiKln zVP~xM$@Qt~ck&Cx%J0l7A>m1MSUt&gpIFb=XR!A&-q)9PvE)~&(9=$EWDhCzD3KL) zI>|;rX*qNb9KG|n?(Mu(e*_6lfx|$8pC1g-EGIB3VoUT}6!C%ejQ0*!whx)%NPWVo z+=tAV{YbzYvG&$YVJP@84A=5U>Lq&@OCcQSrh=kFCCXWkiUW{Ra*%AUwf;!`QGPRA zQ73mmaVXC(*;<=0`#yNnfh&ZfPUZ81gg|nkWItkRI)kQy zNe1|#_9JlN&{BK7Mh~XETB3a4q_x*2%Hz1L;5>k0Q4VsM`k zi+ck~oMSz+F2v3NSHwOG1_gFBX$-YgS%)YH zK0=NmGPdOw-lVnMlrN`wlGjSi*qlco%alu{|hF6FOzy6Va{>}yb6+#~itet9C zGyEx3^|dGN10xG=(V8|;lTY94PJnGaDDqQSM`X^hY_?mPHW2G@S^I{E@T1c=Ro9Yp z+?VkFa%diWgtcRQ2uGgN-yMA4`t)~W?pwF+^-N_mo`=U}xIAx3bLkq$4%E#*;L#!t z9$rDm<1LlH8R5;%_VXljxOmqtpqX8jN;j}gg!^a^Bbu{cha0@s(QR<>ra^zN!u2S? z`PI=cUdbILH$6_mVCnHK_#0jzRl8kOBjt&yDo-<+c$$Xbd z!gb1O0SS;x*qpk^dg5J6;5~x_4e>8noEs?X;h4J?!4GL(PZ;zw)Tuvjt*bg|_(Zsz zr0?81oK5tFsl-GP{TZGXkS$ZT!Nf$irNatkKj|2TY}pX0 zViP40M3}n{sBi$`z(F369o!qGxz~;k?g_SP=Z34$l)mmXxpJZfj~S0qPMxUKLOY9d zq;S;ZMnV1TDM9bf>))SrMU;t@;aL*JNRrYgj=o|`FGCOjFs)dK$EF{%?7rIiVXe7)&) zZ^-s%z?-q})U>^1mi0@II}=c?25$nkqv#(SBNsyf*kZ*?vg- zvOs^cKW-%H4^lgZS0Y?<$zf_92h4+eH%|&K@LK3LdNM)90U?1 zc;i2iyFynkf$BFuF0Q!?4qLj@__!I}0uRFSyos>JqHYhT%fVR48OQbl5|&OJj>C{A zk9iZ#Y;GN1)KO9rjzkt!FBW9N`X7p=m-IBCg?JRFE1-qZgL4-2HpC6|iVYF!mM9?K8)b!Wl?w&8Vklo6?1ik+YfTTmyqAlMro2MGXo0-L`)hhyMz zFjh;r_pm@wU$L*74{P=ovB6U(M&4mKVA{LLy&9%BJnP$g={_QWu`XUSy8_aWB?6Y& z^6An|B(S#E8{pYG0h&gn4&Q6t3tf5p>9D~Yhg26Xu~REi&A@22_eArspB9-UTK9@< zx?pW@%O}iKj0hh>oM#h*$Jb>8Q8h$3HINWHI_k^P#Bq4gHGs^ws|Tk#3(;r$3xHoL)*Rn67z!1OBpm>S_#4_8!G|qq;xD zT>DL~rHaW-_y@;9BlPp&so1>(JU8(W4To>Ho`1%?bz*BWYDLE^-D6EJZK8dPmhX-5 z!^}4xN8O<7zzGjt-(u2kKu`Bs^PtTR%Da>tDvV-T%@7&SgiL^WVNUP69zp;T3#_3R z5)0fzFC-Rl@hFG|7UQK{S#Sfjh6NN-#M6q=gtG-@?fo5Ghn0dmI}K3SzQrv{+<8yH$dmvUn7?vuMKqhF>Nb4v`eb`we^^(@lz5P z+=6q-A8A0WT0%m}Ai}~UwDZGG>#C*j6NNCGHoP{kCt%-fyUh+SWiv^E+Bd#oU2Jgir`Ox`RL3A7Vv2{N}lHxWm6=Q z(dM=lx%>eTSv%P}jTEd%)lh#Hm&}re@CEDu;5i{~d1{^X8C7 zdJ=%SesivSWUjB>XgyG;U_=%@6SxhRI-8m{pcTQ$jcOGi>XIpIh*)Y6*|kB z(uG4Rr{R#;Ls9g<5U2q+b9!W$qD9LMU-T#{+JT}wNY5<}hzf%t6Zq8@99tu?J}18c3eQ7td_lkmMms@sbs(dvH6(+@3E;51yBwP^#Z zsy2*KEiVoWX#F#EASHSh=4egLRD^x|H2W^kDv|>qm4opuH8w7>fs946Z&$`$2QdpD z&=}cV#)A0=iU?yJnx?Q$0olunwnXf^GCutOi2EA&sEVua-DHz2u&@gR3>Y9PXb?Xl zp>5(unm{%Q9|@a;Bos(cOLZG-i@Ga_$r9WIbGa#l-^ znc&Kn9diC|u;m@*tcAu6FF@zxs&0V(oF6+t9c&V`eSpa_o2e#O|i_1#cC)fwoRPWT-<7^tJf&DQy6MT9rGs{BOO{ty0&qksR`R%5U zVxvnM-S=tbp|HpDWg<2OiD>3?M)!8gaD3V^*Lhvl?No0wCZ&76euXcq(R+JJ$ct^Y zPaE;R3|0gY357nsNJpeWXGU7wwN1o+%)M35na@?<1VqWlnm$A=+LV4k$K;@==5v^I z!*ZZxYbGUIb2RnT=V{mlH&AReyWrsp3x~?gX-8GJ3~0UO_tBdTBf-#8J5#~bS~Th9 z4aSYnQTH5;uI4mlKaf7NYBCgPKlJ7F$))w)I~rg~riAMtl&D4vW99r^#L-FiW2^R& zF%4j95y@zbY}o*Q@8SHu0$*~?Gpqb^$$r`Lq+HNmB$u`!3J8{iyfk+VB7xSySDDvu4?{3!u6o2ax18Ev-{}W3@a4kE|hW`{TRMxT`EtcItw}vIugJPM+K9etKbAt=o z4}%NZadz@{6!eG%?QJY*6ZnF5ob5+?G#2~aVawTZ_AF8*%WkT;y{OWFE%jz9uT;F9 zH9riL&eYUfUV>2`4Y~7HYV0u`9?SP*$6Yu@kvZrh2WlkQO+Z}xuRXxJxZJ|B6K((q z!%|9`#211$ZXJap074MIX|y;{(gw}}`N7ig6(p|@CSwp5EX7d4h{B4^qZhGxKJ?`( z^Ggs@7Gf__3c1C4e+3h0zP81zPU5LGi$|KgxuMi?a>))zDITjIN~PV9S-OhUS}8n@ zvF=7tPxrsDw9=W7vDP1j@laR!M!w4b{7MiISY9g~2kD`dl`IYEISZXtJf5}X1v$?u zmoy+atGoh>F(MIHA8PtE7OF_5nxtTLYFut`@k~IbOor4jh8xHGtz)$x?Z%>^P+Gs8U2zRV*;U zqi{_1G2qD;q4VR6N;X8vb5eX+*64P@rNFj`QEj7KK}MzUkN+hug|>iE-SF>mDH|~+ z@cw3Srh`_+VJyg*CUDLql{ci9A2M#7OsjBptE(l=JgNFJ*|7usqm-CR5h!dn5N@7S z)dRA9;>s|xMH!WBf4Dn>Y}@6swoP& zMQwH5Yt&%B$#pTt8OQJX^6To(VsR_qjm8uc-=n4!ptp5r*Yu=9>)VhC`XQpR95QbB zh#S|o;=a@*zJKW^;+F59FF6`Xg!}@)N&k%{D=Q7ePSyzAT?y*p)v&=e);@_Gn`+j~ zM^N@`yUDf2s(~A$rPZ~nN@Lf~19mg6P_PTmbpi88#jfYCLG0%5LHb#E5zvote5PPjO4*Sl4%NIY8?ogq)K|{qRkV}!ud(iDAP1uO%qr3JItwM_!vc6e zpYv8=hIcI!=)pup@9JoAtEqlTgF6FDwqtDqn`p=F$ATs0?_`C<5h?P&@1Jp_z*!6nU7NDBfO5>G+&7U zs;M_%i>O*O$bSTTgx*r|1~STz;~ln%2VuQId3>E*WUu47Op0r-QFicLLA{!dUYo*H zZ$IguFgLQmEqE1661!t9H|)>!SrRdOVn|CyYBSQ|WjT_8gOGV-VjyS6TIUSwL5;=< zfgnYjlgLAU!x{?VGOE+GjtQ`%a-}CZ#=wqBTbhW~rwmae)J$?gLNU1;^P}LbOgiGf z7dXdB!AMwr0KL5(2Ye-c{G7iIG>&urC;tMY7sRaOx-(!Jp9P+8gJnxE6u-+4^!XQu z<%s7w;3n?94F)l4jz|FTn}-j?C>1C#k_Fk?065-mva{=jwM-vH34+G)h{9ATbcDK0 zPC%q%#CZ|ihE_}UgYG4AnCX;rl3ZyT*#47;z7~I0JOoaVKXjkm?n5-`sMHg+osQen zD$*R~zzb(v^;lq459BB!UG45r0)eBPXsroo6oje~ z{BJ4pG9aKlD6rFlK&yMPMS@*lL1K}$ST0S1_A`m}ymaVLK-ymDa66iK6e~~__z40^ zS>nM_U@JHJnLr>H)tjr+s-8o2p~|4l#YPC*AUQupTM(pvInej#IMetS<4hU&v*M3* zcHm#G2N1OE7P2Tr!%AkzL#&uKxCZSy{OX};73U3TH|J~dg0C6S3M;z-KHy|$C}&|3 z0!QR^G1ON`&Vz+;oAX){qY>yFAAgeIwM_O=1j-rWoQUeCf2bcPc=GAg&a(81WcwmS zTA+LB3vp=2Oc!GK?)wfL23~+V6w(*A0?ELY+9)`D-=XL2f)N?q_8x47%9wpTUsVt;G+C<9so_F-dt^%<$3(U)(XkV-kO^#L zd^a|BVTCO}kR8!M^iO~(@22&p4Z2F#gP3)9m6i$rmirJ|uS%Ltw`wflVm5E&6e#P< z*zs|^kuK+QjeV-3k8>`U7g(hN@`|C}%zvITJ{!vD@RhPhl~+{ccna%Q=r~T%4?h^% zzpU+{AiS4SF`iDPeE?3Y?FPxYs5xh;w)>_@LALzQ8cYu!!-Op4gfc*~JSkN-NY(o# z=Ry=Q*2Y0whXcdm$#b?5U&-_e91X8N@+nm6zH(D(k$TVvr(%ZzB+b#T80=9UNz zBcU#dP{);&{ZV(4Q1==~U7mtEKBUpgh~5~gJF;yU(b1uAmuOes|82WKcNo=Q6BR;g z_LyhEv3)sNHHt_iIHL5(^>hT|dVsFy5C%OPr$S8e#dzL?%^=Y*eJ6@FA({;UW<10x z7g4k|ow+*C5jO9#ApSc9)pl^I^>pSM@cFvX=ZJ2;ZGL~vFUZm)Gm>zNJL8CS52$N< z29fTEow<57VH&EXAqwW8INFrMpyn7N;|5JJ_joMRKhY>{jv(+*^EZtB*5=9OlF0TAhQ*8I-NL)QY~UU!ymcwrMF8!!6^l z2omm|@mDD}wUhf(Nz0b9>qlWcgacx$aH+Zr0xPC){vo;UBPSR(N!*!JDy20Vm+m3QVdOyE zIS&jc**m*{{9LQAWCM13@X#UC@+R-_lu;&=Z`&^*m(UxJOSNS=y&Vv$B45d7vAlb+ z!|-ieN}q(E-?l0|!(@P*CoDUgEpL-C1nkzsV`$w*{l{<&6JfNxh?6xBgXM zTP&?aIHbVfYA>MHveR3UEY|UxAcB^P=hatXmA?)f0sA#R3~PlrL_Dny7$Wop#cGw6+6*#VFulM3_~^|cBsgnEWb~fCId6@#2}`-v6=-j?Y#u15BXq%8bmVC+zB-6 z5%pW3d8k}82%1m)NkQ{2bb!!Y(ne^m4ns3VMLtSso`miMEMB7ClGg5b3E7>bM**c` z_H-QeeB|3=M;5$t@t0f}**!)e6&NM{)e59~{3GSrI$Q=3N7`dAd?Cxh2fUsm-a48+ zEDQO4bu{~ED0UN%(gsIkj?T5cWzaD`Yu?xqde;_y#Mv@3V3XVlG)WsMJQt;9GgnWuL_bJI1$fr;~9C)mrPeA$2 z-r?Lcn4dUiIU3<;bj;`FE?p3oi0}iP@XIhF)5a^q@q}FNl~bVm^C*O$`2z4-M|dRK z4;vJsdJw-r4#wn7=r`K%$ESdS5LU8#Hc;8!ijk?QejAz{W8DghM`mf0(_}|$i-qix zG*CM0~3&Vf3;u-7FHPwYY=#XY#ql7|%4$xT34nGQDh zQSyOw3s}Kr1kK`|)+8?gAE2mCh(f54ADAu3LuIi78-n=`f(NMQBt8Wjg2uo) z2pjks#D>ajL)aKCzr)Qz9CbhfoPX#D2kQ`-uMz2;KzNLtrP8dW{usoYltv;-h#K$j z*dK}3LJPz&Qcjic1Rp?fM$7gfM}s1azF?uWZ05ws=>{Yzya!Qj_da3@mYoWvc|M9@ z1$PVsK6^PeVnrE1hA{g(tYAKaa&lsnJ<%u3A4Jfi2ef#QXpy^cfIkFj@wWqM(cW>; z^M+TY#XAHoP9<8LjEx9vP7zdyFN>?^ig!=5nqr^q{~=FP&&{${UoGA#kHQPW=(0_v z%dwm;jdk~71Qgm#$76^#r{RTYvttgY%wK*NWnyMlw|R~#$eDzDCoe^oQ!YW4-vBfJ z23ZDIr`4FGNkE_?8Q3{l2G^z4AWIW{4I*P8S&oY&OR+Ni8^n3jAO5F^Q>l74k*G)I zj|=H;49k6YU`E?&2>|V%8#g{;OSPP65;9&!bA#toC`qosj@E9?f$d%>^eS?A z$DL1*POKcRV<#OB(fJQ*bZnI24U%JpDa+$c^*j{l5G$*R;i^EokHT2{7fvUJj?dKt z3Yq4VT)MClmuTFt!;x=w1HVOs&X;O8(WwkKc0o`VV6EgN)teD5&s_vj9D9jm>?QJO zFOjSFkH(n_?IM&gkB}n~f%CNI09z0TdS0+x%C$ocJE*>*&pvj`$8+>3XBhv&(KtTxIytK$04G)_l_hga!1np=ie(>#qNlS(?r8#!_r9R!;KipK>mJX z-EmAvITvAQ8l^Hj#w}S6&~-4`RSB3@ihkFcshuIBljadMWLhzb?|~UZp~3gkWbsr+ zaG;t^cP?~ZCqt`JY?TfB;G%QiLg!c+QN(q-%uV)KE33fPBFB8+!m7^@)pTLiyEH36 zLfS%Q-^AQn5V~U6kiVa-8iqFOv>M!8xUlLhtQveV+o7B#yVAzG6UN%HFji1(zOX6C zZ}2TnOFfH?5+=ps%+vrjQ@+J{*kAb;TVV))0Os(A%twTmCGEPz0;Gij2Zgv39-6DC z(nd=);z$e`KsrWXgMAtTXvfiRQpq7uN-(3=kxJ7`iGH{G>Jadl)#I?SRBR-1aiB-0 zIDs8U{MnG-gXOVjAt_E_=6pv;ZR1QPvl&WqVlmyA^n zkzpN&XktR*fHc&D*-3HSPth+JhL;!yNJ{8MwTM^;@>U&kW$wNYsu7C~|Elhc;3mAC z%Fb5T9{LJtQ3yo%`VjvbUKWz3kgf1_IeiUIKCQ8?m?s1$f^q#UrMwcO%~*F6PYBYx zaeWdJs1FJF1>2?wG6+PaC_Gy%Z5RwX+DAMv7&gO`3c>}>WhivYIKkMB-cl=x8i+W9 zk6=`eLkJE0?ca01{X5QHad{OfEO3s8kxcT=O7I+|IR|UjHZ0v)R)@pNZ_F1dJO0YI43H&g7bsVw* z9uG}NbsYrksmg zSOO65p+Z@NjVe9ls1LK0hq}WLz_z1NZUxV2l;$py3KmK87D)>h35I}~P*eqmQ1>s| zbkY7V^)RMd^S&gY(|J`)aS7vT6}9s?_aa3EuM>6{`O;>K-?$7yEztbOpb zo8z-$68o&oXYF--hS^))0LaIQm>U6+>|3%H;Yh#!4|r?pit*`E-0>+&e9A0bn+8Xv z@v^pN)lvk`mkSrsb*Wj4C}Vbt$-OWIj4G28_G)C!L{=W!(H@ITlEH-nO&=labiI4w zba1CUJ}QnFUsfK5G)jTAGzE88DH*{9#Y90M)hvVq4_20-WLY{rLE$-cBN_F?4wq_l zVbv9Kyeq`eh)IMoL|t>%QXE%-SvlU4TTp14oVTd%@Tv(-AEM7UMIZAmO2baywREHM9m3}5ugN15q2ki- zn~(T~56*`*q%?27`655I{nv+dZ2Qlz`Sd^z$NP}2rTbuv<-sCu+}KF+mRQ)a^v`&x zC8r9g=3JTUCFWV>`D-u8h09@X>jU@A(&5?=Ob>C>EnVAZBPk5bH^F;b62G<&h=pWB z*Y;6F{v>{FA9&Pls#(bD;zD$0K6c2EzH{YL0*zZOsg9x)AEKl#Qh53_#KsVp)?9&l zR&k*>S=||}9`4IcQ5Gw${ej9ncD3jv*^?bY7bGw}kQ@nMxX6LTsYOUh@-2b@m5j?D zd^sr+Zf0AA|I=DBGpX0z6za#9i#n7I>>qg@N&Y9o6E}-QnsZvm!w|4zqJv66~9C|81SSpuynipp~JL~7?UT+T!8Py68} zPIY8uqK?^!U<}Xm;<5GwsLU zA^f%BPty`-ioxGd{Efn2Tx*fG*{b*gD3zXDfZh4zY+nS=lEi$bgE>3}Ej7`{=hq_62}|NGk(8&|n$u zOaNoLMb5>wDW9A}biD;|&hpVi8Jx3D(nD^Bl^x}u;GA^;4`pnBA5EDA#d8X_3HC*F zuf4JMuV@hQcU-t#wp1#Gy)r@|5$yFG0AR1B2ZFtd9tifY_$R==0}o~FRsoyfIsWEf z&f{|S(n=&xf`-`yH=vD&1A$_e#`DoVID!0L$44&be2x8D#lA;&&X>zFrPBGRpX@UX z@6$w&RZ1YI!-svTm3?`Zvj^E}gHN)hH$7MH%;WaNMmcdUYMU+(gvJCPxMnckUz z$*X$|g9sM2Lco}9UZ$_Y9i%LY7h22<(b-*CA)~B&Tof|*E99@Enx-HXLE!=jc#4Y1 zWW<>uITIyklH^R0oHHCha#tYADaR)K9Y+Zuh`h-|oTgET(^a_YB^cr~ns1qUd5BZz zeB4YSIdNgka@K+HcBlZC;!sG>79M@-H|(Ld;VvGvrOUxt6>MkBQInOMDEm-=hJHm30kb zEM63!p%S41JjRysLXRU=eu)nttw5sC2wAsR6k6S{(6zkKa*;|#uH3Iu`fN{7sv{)+ zidQI!3gy3%H5YQ+ELZ51`F!l*uP~o&N`{F6nYWZ?`V15;W(DF)DYJ-&mbmODvh#?D zNI4K+r`Z%p4`(fUtsSw7hbZ8i!`$+NIBL=E#luzQc`yLT z4`aBF&xH-0uQbJ1noNPcaiEMt=7!)B?g3KyX!e>5O$09Szr{>|KOfF(1BXJPl)A z0?J7f(fsJ@7_u4tvAd+Vs1nD*?C{^Hb-k{>7zh6{0qwL8f#Ta(*MRQ3^Lt@Ps$~y7 z25<8U6JFV;ZYToR7<2?F-EVeVdaaewfV3t8QaW`|YmWgW$Gsd%x|U^dD9HrnX#pjs zdOW&WUvV?rgoI_yIc6CqAk!zoSo<7k#GLCICYa~W%6kcgbI#tn41#gT}S zZiyu($~s=gc(=0YylVS!!ppOG#@PDD29dVI!;vO>OF|j|0y9or*L<~Tf1c;w=5JEbpmUiGB z>uC28bk=54Fp~`;KvS=wku`3(65NT8X$&DG0YLlFolUQlGl+5u-amsiTT3hqGBw>- z0fbGX6AM}>_OJaAH@2xur$mcFMe=&6A^)6 zL!8=4OpVNoTVh{8?_vBaRp_M(LyM)0m?an^)&r8w^ypkM=X%lhah1_lTi$K z;m&yxeLkoQ@bR26=y;MufWXhGoqUaJiSyE6F4IZN^sbq4Bn2vqTqB(25PwZ7 zhlv1WX>3~b#Y6(OQ*A~Q8~$6Y1Zgq$LW)hoom20E+{OprSUaDxTiSutNCL<^88I3j z7WeC~Fdf^aTpio+@Hc^Ox=<5?&eq@s;@T}lD6YOZ`y6M?eD_ns}a@|O!QK zuau>u#E;g|WNc*b;diilZddC;6~%6N(%L+K1I9YHQTw7DW2mfC0FF}uBWPPAlK>FJ z(V!5=cn$|4HhT#rzo6P$21rn9IPEq>`^RmUTNG+g_meM$=wZdmGp=Ae&N3dM38L#J z&)Z6!O^@72-HMqE)mS#>RoQ1@^GXY8c!EtsnOfP%DmWI%t(a~I_N8ODKZM!8BDE1o z^%iVObj(UA2(4qlR&Fypv60s#TNCBHg!S`>25QdVh)DnEe{3I8c_YAmfjZIPnzS2X z+U^)3mnFXVbS8CWxZ*hj%Zw``vAz}(P(4rd>H@k z=xpN%uGXhEw!|i*x9E-Q$Jm8kaLfftvb({=5T;otm2>_F;ra%_#Kx;ktWpEUY2r;( zxfoa&?StjYm+fxz)lC6!ogQOz%oVUDeQLd>xOW%(JaEgVz^ zaK38orFkB$ht)%9C3Hk%4HS!EklI!Lm-gNCPWKaaFsz8aB{WT=8_?=Iq}itW*`uV{ z@w6sdE*R?;(?AvIVKTuuxz#T{C=6jXfF9GATHU$PsE0ZFR zVB8*pkU;=mLcGqtlNmX*^Yr|05o1)dVb z6K)lPFWYsvr~~o?Bp`(J1_eS&Kx~aXz3tT1G}Im;&Id?iY`*`i0GDP{{CDEE3R1VF_{&{=dPg2*yYL_9 z1BlAh$8H21E`Jh~CDw>yk>gBE>Pue6R}pf1PtIIJ45IxjdVxn;hnYEsBurl0$&>zHHOlh=czh@523B~$P~?9P7;9ys5fhzF9x{9_1_ zCfgwe+%OJ~kLj%hZ_q_w4*b&}XX?YB7L^ReAH5S>6zl)7(0mzOU#Q9}m@R$iCU$u; zO$j3M1t9X@bPbSLZX$^#+CDyFsjX-gyhv>Igv4_5eOzwy&*Vaj|3^gn)5V_446vN$ zNem*Ir@P_ayGKYkpmM$vLl9i)#{tBk9as7ny(mn+86x70*^=X8-j*ZA1)GEGYE0=ONI-OLHvXNg3+{ z6o&P}Z1W<0)ho-EV1G}ts!kJn3MPduhi!!hbeSHtb?!0XoOLQdiY0 z8w=TEKB^Q~$f6&DUgm`G`JMEV09*>KfI>1lM+rz~+4}=XM%k%H3c1so2a?RWUlGaZ z`8g(0jh-7c5iu&8^(X+3Ae;_j1nnz$G5=!PtXr^I@AO5qAsT}ANFcWLB(dhBtQS$N z6j{1v6;2+mv@$nJ0I;$p3fr>RzQI}RC8#ijaqb)r`2_p*Z;?-ShT(V6!#GzB{@<^C zq(T_bzUMJ^Wz3?sN3XPJGf%!@u=b4mpK1?nW(TzAa$fPvog810h}bR?`;DoB2xUmc zeE5F&pq<`;`v3EngpVH#R)5m}aQ(Dnlbz7a@zWj!@t5=ZUw&_}{)@!_vHsJ_F|hx4 z^ZpN3f717>A0Yrr<@*r3$9{bo*>LKe@lX3$qpQboDS47%GY@8mE`ljRMN!>j?z8`rL?4_+m&(dX zbhc>g4Qf-z^81ly^lR#=JE^G?kt$dBptZ$p6g^v@qhol@1XMbkeh%zvFS3@gGalY6 zF^vao{DUh{!;k^%5$p@TAN|s*IAHub@1pT@h}t9P&#fZyd(h9<^MmVi;lSv+xP0;^ z0SdiVW3sA?F~n%Pm31sUa^zzmZPG^s+OhdfL>l6MQ3&IK;dLY06drK*-|hg5lAa=G&YC04AaM8# zL9aPSua)DljTo>USpRCcfU|=fa7lTXzgci#o(PBh?kHuaq3_y(1h?X zjLm+6PH_)mb78D=@EscMcc#M5t_L?Uv6Fe~X8;O28QH&I^yVG-4OQyd-jf_NC>e)K z3xrK6aOmh|zZG93%c(Hgd<38>5))9h_qv-*xaHw{9!Wb!yYml86WJ5EbsDEYx!vtJ zQQKQJ{oes`89*opv&AYHiNN&IRQ-4En-Gv_3&w`1kbVqEWwu6uqJ7+NULj1Gh{Le^ z!Gx&w<-M%)mrC; zN-5ty5Ze7ZY$0$XDVcOs-A3!oo)f%LL>rvyFxKw;lg!8g>uD#~@6com;BZofI^<*_ zJQ-~>Py-uL_e~%@0T&)xPNEU<#UTn`t`=Bd;zc#}R_MJ>vOnJqWYK+z{R}{Y#}~Bz zVz0y;ePQ+=N@E8z-%!?T=_kSU1kv*cM0-pzG~B3g;sF9s(;{0-~?X3qJSW68ZO|0X7J0SsB#0U#CoZHOW|mGbv1xd zK{v?Rjcbvx75V`p=yiAEeeXhaN}Z$MsfIPyy@(kRegqe(fOj?svY<0K(lwq^js?#` zOFo!>vp~&Hy*L!=MKkhOnwYsgtQVvgA!8|GZ@kN`Yyqr(Y4+e1-K2$I?q!SbqI0-I ztO&o|d(cJOd~}hh%?i9hPd**^`S5rcWSZqHDDMEiMV&n(mb<@plw#T6CBM`#{ zh*@Ya-7iJSm_(mKhli=XAsi?vZ3w8H#vNCuD2T5*1;pQuG+VaPs(ru#S~XL_*9H&{ zKyDTw)5AfM`iDSXf#M1V+#drR9go^}fv%{9cL{LQVHJ4cSXd{?vVjUy@IC5eG@?X? zI!PDM*>)%(l2s6CY;QOSvU4C16bp$!Y(nVS5QupMf;76+ijV)MpgMuw9}a?yAP7Vc zehI|gi2fM@p(hag6;%IDfLNfQn$)etuN)lJSA+xAbD(5)f{D<|rsDdR0aTIwb5O%h zWVEFzc%LQUPXjf){bm@VNI(Xi9Hw6hkC2zJar9VZAXf%>NitG@`=(#QF8rG{m^EO^UV0``dQ8#0rhrbxG3tetoX zu~uO~{e-!G)`#kT=9LhC9F#epmh{F055lN$O!R(y0#^5%Pd4)zQF!0VyXQR-`j(suMN1wU|EZ z<{DpoE4uw$hg2Gm{592xO9Ka2ojk%?$YN2Bmr8&!xibN*r1z*Y#EDYS(Dha<=zQ@l zC(R!e(s9an`A58 z;Np!hz9XM*x)>e(&ieQ$*N1TpHOTRHAFeU0xU&81d*e^kkED39$c&uP*7%C`x}o@3 zuMMTT4RfO3!YA2cGVU2#uQ%?QtF6z|t}lwJw?@HoR4wf`tdGtUxeBA|k+)D=Z`B(2 zj6}BebE50>q9Zbg0i7Eaf;vhC+8d=eO+Y_V@mG((CHPavo_>KJS!;aJVu;R407lm9 z^pOSy^gVwvt~^ZeMa$)h-eGux{b`cd=pEr3iR-Gz2h-z`MxT8#HAWL|%JN3xW6Ub? ztnnpyhp)i1SBs0SRZu2g0Fa_QO}_Eo;Wo;EtdSYwc`gdTa4UVM03860|A%9n70hoC zu&7-zR-}stc!%dv9;H&AC)Cc&5V%oPCI1|3Q%o9%LRU1QHXg!=+Q#b-hsO)>HYUP@ zkw8+%0ZvaXO5~V|N#?M4`Qb^FM}ffeC@o4v_>y2YgaQrE!YfInzL|nqyuH@(BV@CmVp$NA#GfW5=K z#$`*WEuJt)?O~8w1*CE12SGoso)tVQrb*|B+ok0zfIM4QDyNo3pN&3iEtMQzR`Lc; zOmHfxM+8ao=A-)^To3B7#iW&4W88QMOEEl?0hjLhW-c>6^@>mH@rkw`~T`_!_D>`2N9qd_c*tEKJ%6L zA)G6J>w5n~1Ri0pq9O3vI+WzIY**sfu14??c)K$H$PT6^Av!HNz={rbR9`gy&2n+L z#-P!7k6;&o$T)YO>MTFCKL1p`vE%NGKFed`ik29-!g+89+He6U%YjSM0Vk}yhnE+n zOHS37X?gKYYH{TEkMwmafph^4x0a!{FaWqy-Ezuv7!80s47I^>sy-LN=X;xT(X+cR zlH#x9uo*VL4-jv6&}Is|<;>`d+=qkVVkP5-0{T?Vqf*GT!Nnh-NMUn~FSbU{i!b!+ zO+8yCKVUMCzV|LyK_*_|a$03(x3Qrpno83}%aIO*X1=$DhVw z`Lv;KYaklZz=>5i;m+ZD!}O+Z{oDu62?>@ee;|P<5w{5cBwjv~KGr@u)76|{ZmIe* zwK15|HVF1T3^(dCMwPBL6{4u!D<*Zakt~lmq$GBpf7uySA z|Mf>jOVIrhMzu1H-Q5Jaid%xbdJ@?s#|1dA&Ic(k#0N|nsm_aJu7PZ}4=u*_W70y! zm@m$qn`CXy)oY}cI%%arT4|D2#!D*`T5`iE1EMT8&6TtxAz8Mi&I53?H)7{r!^Ct` z8w=fAdhx?=-71oD4bsb0XG`5F{7*E7{S^g_iKji%`g&=$^dN zB-c>`n?BJ+uQZg}UO>ZOt=dMjd<)g!hN#R_aUljz<+Z~N9#$^An>^KZXfjON12CZL zMAhlMZN|FoycvYR^xOo;<#JiPtI<@MK<%JJmZ`QQ9>TM{ARgNxx2{P5E){qYm7=k! zXL0LKw!DS+o&PV3mejQZl-1?|Ss~gCHF5nGN|HHVU3F9-s(xXLE(cJ|XR2BiEdC@u zfT%e)%3IoaW*XoktAB*-=I>kRy$}ivm0Axy%QZJLr{n;q2X(mVUW-EZ)BA`pu$UO@ z7^xYz5IF&9sE+@jK<2;3S9XPSR z3QIXsvG|Dh(c<0JoP;~1mIpZ+`DvB8E0S<&3o$&|nYf@Bj*^1uA>7aWU}e*JWo@qx zR@t8Yk(E6w-T{VeL@DJ;MQHk~9#XUbYU}uZgWsQyWjvNv@oBQYn8g)y^Xp0SO!qPs zbAjAMbeEP2ITldJbmVp72(s=q6f;n<_97}~FU&p#LL|Z}+3(}h0lc&C zB67nbOFMC?)BxiQShz?t-79I__?ncPh$mhVU9(OV$%(JYEm&ZxMr@Mjue%@_*5$!U zvigiP(S2XA*le{Jj;O>*?Xrs0MqWPDUD{u{aF?kChqOEhKBVcE9Fs8UgO_uXcuI!a zy0;}KMZBdI+9ux$E$DHGxZ;4NPv|@<3{pilONSO5#_z^DzGWqQt$zLs_0I^cr?6_m zg{papQocbRQj>nsv8?6+ZNTvpIWxhHym)bPFRn$m9qIJSb3+T(PobyOyLjJ(w4nq`Nr`9mh@)K zl~xvWJ$#*U;|Bu2^~&WYdA5o4F{+j+{}i+D!IF%KF(zwip_I?qDH0AGxRSx-nh7V+ zFEI6l-K8HJTTID;*72jB7M*|B#7mH~y;7dcg-oaov!(rsJ%DiTsUd!wa=$nTmoN~Gl7iAIJ z2_E@$I@#ZZKO+dC1hcq}?=F)e&tP{0g`wD;Lc6bElHH9K$@x8`?PA*NaHKg%$GU$T z#DIY`n((T{mBrCgiFY7UUk&C(IOKV_YR-e0osNRR!2Twf72GabzTY5~oE!-77r|V! zQSC|deFg+|a`d>GQ)rxPlfW(gn1hGdGey)BR>-a;%5YEs0vcFccy^4vYoV>u$iHzq zYq6XH0$~brYQV7>7AJ5$mor=AfZ3pxwiWJDQxS{9f=vQZPIma!xSVTpewjqP@-VqM zi{r2nt{=m^y_c_OU7$|KojBnGgMdE;Oe+Zn31(rZu$tbQhRdI@MT^#BSohDQqf!OW8{=bL{fq z4bRDr3t(=yW|Lvf6XHWlPBx)kTniJD0?OocH0mCuQHx3u4kFl|%<%RUU$Q-aw2JmT z&3{wa+hY8Eo;nqP$h*U^$8G1R3auAKY?55B0C@*@2xya0hh( z!zbsMj4$RG(${SV3ads=Zxk1zv$<#k6)SV+d!e;XWbR!6S`hEvvJ@l>4UpH$K8weE zD2SJbK!Y#~R|EyxaawS_8eg%q_EpM?47uLny)C%@qaA`=v+k5F!S71ANe|# zAdpf1g#ho*vX*r_0EGUfzPU?V!R#-Eq+Z|y_l=eCph>vC!zEFKGp)BoiO-}YM0rbi zn4Dk^owr;BQCEvVO|0H`hW)BDw0^DaA*cKxw);h>zOtBIV;|`nWFQqptf0%Eu?^=TTdYwMQw-{gZKNv&tl;3M>PzhH(udS!2DBmdgFWd3ty13Y3@M#cay9-+W z@%|guUL;RU2ZgfI@U$rWKi)rf?ZpwuF7o;kcp6>bwRRIyE4+FXUi%n99_kxkj#I9n z#7*sOlIuPA20l9+dL%2W$JlYPeiO|Lxn?{5uY_0zW)f%z$x!(+*S8(P^(Y_S9w7$=q{$T^7-;Nr1dJw1!AV>G|W z*>DAlD*W5klNg-NoOy1frc$lVmfmDkI0S?~84dR*$QwC^w^BQSVMy?zvG8-1*CZ6b zLP&t*L>hjI3juws>+8&_TV)Or(3Wy&OW8X)9NJb6jWabGuU=)mg4u`i>_gcfzBM~; z%FORsugyp{*1m-LGo~7ApQqoY#@c7-H|gG)88!FaETGm~7=Rifa27Euz|QXe1bO=R ziz)q@-Q9|>#yyQVB>2+i*~)byC*gW5fMJ*bY;-03$-XRNw`F}50MqW?f+B9qm(V0p zd=#Nf5*184Aam5=nBhyM1{Cn$8SJo%pFNgu#kQqqy}myGlqiN443l9T!qyI2=v1(x zsDPzs0Yq%tFslPHbqR`(tDMTRQ5YND{0m?ZlmDbLi2iuta%#?(uT0NKw*Mews{N*n4EyyNGwo9|me{9cl-sY!c+fr} zquM?$<3{_Kj4Ae!8A~y>OZ4EkY#4+$|0o&)TNm(O#y%6$-1=A!-0@7LNFBz2>?UMI6Z( zvT{h}9XN*?k^NO5ePx34L3sjE1)V=ENL;TQQE%yTTR*7zs(j@S(dFsr@(t+nwdnHI z=<=25@kh)h7U{-F)0}S+nH~!j56zM4SM3U$}kv_1}VIEdt0_oswt4A^xJnTeAvU zig#+N^a0H)(a3@lbQIxeb9LjWfcX!p@)oFIZGcZg_ivB^<=5+?>n*1;QmS+r>DYjy z!i=R3%t70iAP=y)_b_y>mMH(?jHwQ2#%;YK`5==0IT^`3tw*H2NNLmEd0jQccpi z7bh4u%qNs_?gMaR(6&Ad1-lw;6RJck)sdii1_3*mLuEGjC1Ia8#k`JCP&I;mc81?I z8{=vXR2)?Ln!Pb8OX6!+{|)KD^3M%kJ{0fF)stDw_pK<4=(-dnza4=UDNac_x#6yy z1n0{5YxO3Z?g^OC8f@%wZb{pUA65kmFIyv?Y%B0zDNBT*sl1WmumV*)_Q$mh8_DX% zIL89$rrM!tXtgb;l2;0j=mZnej7L`!-!ADh)Yk^0yEN6_4q%H-{@tZCvQywy0G@2Q0tQ21eLimZjUUm&%V>FBnHjg%J-Me*hb^i^;8$I7Kuh-F`7 zwPo@=UH_sHGa-T}qU+;uC5fjVK`NNiG>bmdYBTYqzflrLuIo^u(nSrvus^2_K-UNx zeG%vqT`<=D6V0&oJj$u=6A6)rKZ9pwf+j#&6z|jcp4y3WK3s?P)Gq#D z@;ybAgRlfQ_4az7<+;oIeCsH;_gRACzudQuLn~*# zb>e%k_zs&43UXS~g*c~yF*`(9820<$BjB{9Plak9P_Ah}hxYSyz1s3PrRBe;=C3|ZP)Q;K7*P<889LA1meIpA)+a|C+`Mk}BpEC9%>S87yZP)SlYw-S(7PY%>N$DGF(yV{Gz8pab zzNs(2SYLACHA*wqdO;Qb==HccctlOQ(@}e?P(n{b4}D>!)?5OKVN3h2<(@u?WVgKvT>y^X&P{C$kS5AgRX{xsUCnE&>_ zXez1W!;Ur3<+Mi$?&7#!>CUbHiS886_=;BGaQ=CfjlKx}S#eOwyofSSMrJGx;hbJ) zqc7o{UMY}91bIoK(lCMU9;LS}P z3v((_!pRUGV;?iknm;)qG_0IX8>dt4AN zFGvEPrl9HP=qXKVf1D7PB!Z_WyeiC?HEZL0!{kSq;Sd81^hmC83rmK{?03zB8CRQ_bMrBlm_R_e&Lp)L(>64_9uNdgyeF zwVirW$#qQ0^$X;}F<*Xdq0NhL*hyzEfCmnoTy_-MPKuQ2BE_=VYHJfINl0;3*CGT= zG|F=!G~KcTmssdvBV*ZYtn0x8fU;|_O>^h(kUeYyf7&crTG=T6Sd06m*dKmGQt%G; zTmHM5J&NBl+Y6|rl*v7W3St}R4rp&XQrWs2vFTlqNXkO~i8&~NKOzgKmY*WUo!`LZ z$>-rh00rPA&R-~LCz5bwhkZD0A_cMoH)Ag!!NZ}|Yg4p(u1}5YA0cq{xMo|O zY|vbtEMhG*wFf8{Pg(^dnoTxX8-9iso6B2`Zo0`r!Poc*eBHwvp~lGp&frC1f)_=f zD*%kWhGI>9I#-u*ef{9A$oke+h$d4f9_Ga@gbg&mt%&IdE!d_uUp_cxl15YWKun&i zwJXzcqkx3*QPrER#Ap)weEtKg0%%C?&7P7o(4l3dlhHO{+*or0=YOTssNDD)cnuEV#(gMB zCk_h#wX#nsA%YwKol+1@D=X1<9T!W+Fw)wS2Li;ADYi_pZs}NtlOwJlBDl^@o?t+k zGO{Yw5$R@ddRNGYpYmJT@Eg#Hga{cI$}oeX3=5WZw8AV}v0UU5Jr;s{;YvAJwc)!d zRiFJ`{#ztHFe21)1J<6y&PnpRQ+RblA3ZwVgflIx&Rnds-;1ly4d(MzpULaU?ZVvZ zNP;1&!7A}rQ3m?u&1A`7kT?i=9fv#GZO9bgrTBWeSXY@OEv(nEp<9%}?ZvKDJ^XC6 z|Cpj5owx?5`4Jft9c4dErUBwj(yfsfuGJS+YLNUot)k&a`aJ5o0e%5WceYZIGPqXp z#aYT;uMhC{l&W%!5ALl4=);4WBj}@r5+dm1WlGsfYBx80WkD%L?4jveoL$kbDu?`M zt?a{hW4IZS9>zo0gmY6!0!7wJZKb9VeR!|~xkmHIA4ehZMCAZU-~fC= z|CnkWq`%?7l`Wsb69&or|D8TMc=x^+eSG$tOVh_+r-bNZ?A0_t1L)(o*AI@b?@u4E z!jy0TeKerv2>SQ~B}CB27D|bvkK0pJ`X~T>VDPQ%#XsY67Z{ouKq1%HJeaAmPY55% z2s(M`3FM8WlU*@@_-E5uSs&e>$RjmdE>vo)#cb1`D8QxxUZ7-IK^Od3k$vb7 zwVImO8z8}>lwI$BnACq_A=p7_{Dia+c3b2vl9x!6ldXG0OD25F<#m0?lD_Us{9kp2 zyp=q}$Oazg;S&GEGB{2_ob*ZN8DXx^xfi7n_bM|H?^3>Pq*U@7Ub+^%h2YY%q4*BT z<$c9$=p&J%M28l9Hs5nq@X!K2B;g`c_$sX^(p0Azj0aq1!pCBK*C1mt2CySYR7VLB zWAVk6NP*dJRw9tiPQF06hRGjU%-0f1$qbu(<(`-Ue7_>_-3NTTg7_Aa2tUFpl1og$ zv;h~Q&4SbMSh(Y1=ieXS99aV2g$lmgHY$rn7`}@m@m)M9zNvL^j)8Ck+!kyF z*88-^4V>(8I{_V(rthUC)BYoQEiJOJ(Z`aK4x?$-CBw1|`b-d5pIb*}0vh?(l-G** zId8a$RmS^q2y_8+=TI7I>ys@PI9r(-PF~~!$)`pctv|1z31#eO8#s~r?^jMWe&B~~ zsuJL)+DN6Ytvda(C`xxc@YMk52v&bbv2_KMn z#0UfHCPHFXrb)>0Ei|4Wh5xtm z4~FUc%0C+}y}q587+T+cIEiR$!20$&EqjCT`R~8J`8o$7Z4YXWU>+@$5JB26Q%dCe z_Rv&yeJcmill+4?wzRsD{4-ObD<6EA6uN>C6e21pEJzxvievK2F&cjitNmzzSkxp% zGSZ+}luV5o1P%X<LD42Tb(;%j320lu>AcQaZVVd`a@;peAg;f;w0xM%L9ANyz zAVRXOSnnw9GA>shC%IBz$ZF~%5g43epC*P+vG5J1m|{Yh9CV6_{oVi5`Sm#O#`lsR zpR2$0{JQt@(ENG_&x6dbWwdAxLRjB_epP!0A)QrJTloI-PF_9f48g0oHn4N+G)_XQ3=p&LX)i(aN*3Pmzem^s&BI$5l2DotOx3tf==%Gi=yNxk402i3v`+;_+g zl+>V8T<-RlTK^%g~%t#7kP)aIiX92h%CeC_pV` zFVyiqp#Imy6L)ZpG9ry$ct=a^8zF7L~zXkTu))@$D=3Jq(9V#=K zjN6)UkwT$-98y3pI+-9Jql_~gSO2Isft zb5&C?p@VwMp?27a4Y&6Iz|90eAF!}o^F?PT5~fnZWVD2XSzFCML|>glwbh7EMspr3 zPQX}47VNMf%p#;p`KRoyCb_=_{jrpovYYsgBpUtD>}0bz8%tr$(9_ zF%1jxLjrIKEU6(kwwKwGkYYtlK>&pBEP|jKaEebMPw81t56U(3Fah2;4yOT#$Z-aN z2+w(NEl3xR>7C>YWjS>V+0Z*<8t6yH1qV3xz+@_uEM7Jgz*8F`R06T^VA|J<4{4S& zj+lL?@FY&n&8^}z_wzBB8rR`8cZF8b2dmE8%~tv0?LYulUfNZP6_E5{qS-W5IGu8R zJ=EUo`g*vXnajJ3wH+W89F7=kSK~p{U2>4N!))Gbpk|o~UTXh@7KQPf*8qR)1YyU~D z{fxJEnrN-vELvJ$-XrRJV9@${RO4z(JX)AZ6*N1a=72szYk9X_cudQrQ4K#!UlEUvUCFVp9;0*Hg=x(=M(4Kd%1LtmNl@HLbH2BFGC^!;M%fF|Ap86X8;M=W&Mu5bo#W((xvJoi?PzeN&PHF9V`JR5 zUE-Zuata;vm@RF_4UfR`t@{z=`}V%)2TC|$yC@;w4O?ndO9(l8_1#B@ogcO63OzIZwvXK@H1-P7V?ktgkEgbhvz}5in$d6+_`58Y|=>80ANht4FHG(cb-aWwc z&~+vGHSUu7BKWt@!U}7tCzF#YWO{jdB}Uk4)58Xwm)OKhB+vyXl!*C+8WM1gIS$tOFj;LoPs534bUx z9f#3MykA~-yxLmSRVce@bc>nsI~x4Kazeu;8{EhyiU!A4)%(YGFoazqBS1uGYA1U> znEI7_+-_yUfpJlt4B({2>)@Rx2+eTBzXW1n@yL^@g;R?o%JV+u@2FNNT+FVI{ zY4?^i>ZW`|-r5Ejg=MAW7rCG`3NkQcL6FH~APtHj@PBxF8}O*At6zMQOkjY(3=kj+ z)u>ogL8S&;AZUXy5S7t6ua>{#X@~&arWb<)lET`zQ+Bm?0a8A%4OsA(%r4XhYzQF$ zg*=jpiVuIpkAX06DoVS$WQ3}6FQ5yKe2#Ms*#+weoE9(e!cgUBdMa0zt+JrRdT9!F zTa$74HVNU02&W)C9^qsRAR=0JW*%%uX!FJ!i%PH&-v2(_8S$-r)B?T2m4#S^S3QEZ@HR~>T;i;3 za!vv+>U+e>P0oqL%1zD`;IIUb3`rv>iXP&`fnPw*-Zeh%wMtNF)ddBeR%l!GFZyjA z960m%fK>-6tZYC>h<`6L5fQPp{=QlQU1G_LV!~}#>j?bXk6|cUJ}a4%td7jXIrL=p zu+H9if-Cn-o3lHykys|7GZ{SoCze{=4=Xm*QtRA^>BRDj+Q1{g9;S`>t!6~rhy1$v zRO(d919&&+B>F8#M9*z#(2-cf)g*Eo94k4Qk=-U0skLa;nb|uVot}djeKOnjaf>Xg^HS zJBA;Vu1f-1ntuaQPQHW3rRyB{E*z+OCZjjJK5w!`oipBjIrOl(<4F&j$BdTlkfR8C z*sYgo>B&ZVijh9aNS~sy7)p&a9twI>$nf3CBa-36osCCCI_P?`CFq=V^3chz#Enel zpl5O`YzR)&8u4P;Im!O&!JxAgH&9K^((p~n3&S1g>@Ia|rdeHR)(<1kojr$%D#!|E z!9DaQHW-+PziAijp?JH+(!^idrVgwR!MNHcGNwfAx`|9gX#t8?UCHVjG_XRs1awE+P1S#U9WL{!>%N_d9X)cy*TTLKA>6uIiuu-edPK^6-jP_h;v z9o&$ z4v>T14RZbBT?^A}m#rCHGB18>`fPtkdf_lUx7=fl>eRJK5t z^?TP^R{Tu$u3a;%y2a`{H~7+HzE2x{`}1SpCqVWc!&ZF)E`$=-ZT>^;Gw_2~XykNlrE8q+$k|elM4dL{|HiaI zt5r1j7)=0Z*N@b6T&GsRf{%0UiYrJTWlciyw7JD_V~M}99$R{vb8TTO9kEne)Q{pk z6=ajHo+{iaX}ptYOp=p{4VpKD!dcykUdcp%5*VqjB6UVWC?9mMN4@Hd#83hrnZrC8 zfe5-vgBY`(a(mIcV5u{HQggO>eiHO-#QDTcYVKN?)jPIgtZ(`%%VWcQg{$H$j}1)6 zLjPZ_0}nc%!AZkL)wywks{)&~jWD-&RAqqFE#u<3OI7SZU$WSXRPZG1=(#GUy4Kpu zZbLO{?ndr!`;NjjaaA;nPpGx;n7(F0?c+EqE17lE*4h3G>4kIryNKKIyaakNXI4xq zx9^yHGYHOOK-@JLZ`sl=6n@b0P{2uhw2o zd{r{@0U>JYjP`F-UB|=RrWH{hcwUuBI>%u$QPRK zfUJ=iq0E|rK&o1F>mBMA`;LL!Mg4bAR`>2n=BE5m#md6en2!hk4;rH~i7?1&t>_Yo zo;|RT$d+4K0-6#p!Z5nMw3y~R2#Ab?AGpwcK6~IE>c0QdyX#39zYls#RT^b_Gi#k2 ztRMw-m)i|-R5k{AgWigu)2HTEtci{G$!$iTT*D2S(mGJ*?{4GAZfZ~8;|I-fR-hd| z&+ph8KGpdLU;s9foU>-&|A^#=e4jM>_T|UcmCTKfh}*i8Jjy-ui~*4?el?(r;3^OB zXDU#89!&`Pe#h&Rq-M(-K+7I@0Hg-RWIxO?^}*ZS>0N3MR3HY6yxWPOB+<$YZ1m+?%LGIjSgZh5Zh!7 zm1fVcmQ0Cj7da1^mgmF}Se}1h{mA}m?pph+ykrQ4XqS|QwJ^CQIGg@0jVi3qo?mzSxuXidmHG-T8!_r(T!#%*GG#TB3EcBL4g7WF6DigyD>0L4oI)u& znap&ZPA{wyWdU<*;}lAft!06;P{*Y1oeSjAj6#&}$H z<(qR1F|oz(e=3=lxRno;qE|XRUCg9UT$`lAAQncqdz5b9M#j$JNY7W%GXC0u`AR#^ zk8;FRq6m5iZ_CZi6m+@2Z=K~=Nwk}Dkx$~gD zUug(;Q*qDRBpeb*3LwC4v=G8sZw!4P$bb*>!LIQz;|790wg+_1p;y3^Nl?N-82nZ7 z*GS~9vmcL*`2R7`Uuzq*i`K*4+FVaX>_U&d_?w@aStoo_Vv1d{DfAT!8qw74*-(iG-EXKcJChTHr4o=OSt`EsxcV6d2#n4$X{Yf|U?q5m);= ziA5KLT(z)kQcte~{}%?j&s%GO2?h5EwU}?3UDbF1@B0G=?ND_+#o(>Gg3mm0dIvqA zb-*x!#XS)Z8N>(YP#rAyizUBcmTw)cm&j2vxx{{1!nfB)7}$&6z~3l-IpXUL*d#aL zrlOdZt-<0_N}f3$rz1G62q#C6DFo3lSiF{!?;fwsObXxI{blwvPTxT3cP;gw`Kxm)8ljd^DOomH#yRBQ^%M^Ug2{~NB|uaymdTU=5;!=)!YxR~``S3e42LdLI)4v7 z`U(W^@nm+mLMc*2pilIHEMDTS*$&v~wV`Sm9SvmVk!>uSe=i+P8}HwfAF<&Zrr1c^Cu~jWS(l|i7=zKEW$)B`wX<0z$~P0rp}wNnHWh@ zC+IvP9>wxOj6pNI2y`z+SvjgZf!gn*Y(yZuSL`|@YeaP8zb4?8VSNb+6jzMZxtI$lZ-lFQO%o$9i$NTIkM~iQM{CoXKHO zX#Zb`O$ihh@qucz5>Elu^JrvCDxt-DzQCbn)(QJsi%>nEQ{h-oRv$R#zH`+##DaKU z;oYt}cc*oD;XmO>TH^{hBwu-HS_gKGpl@2i5qpR>rc(>H+skGnP(`B&n@z5pyS-&H zqE+v1)w#V7C?gCggBZ$gdl{`|{ri8W6M`70<=?cHKkD!JndfrFh6wT5YXtI$dKey_ zTy;&j+SyFhZp~-&6rNP2la9*vP%dtELEYJ=U;&(qYy$<(HUx3G!@YtCz2sF&I~7@@ z1ODjji6A*6(enihjs(4Ru+_Le1J!(d>P$`i@+-odGov4)wPH$2?v-$VxE*)hE63?+l`<_IY2GxDc&cl=Hx(^9;n}UNr}5n6JaWRbU5;?W zMaf{C*z#a+h5r(5GY|garN|uKycmB8!5N;ZTJbWDQle4c*NO>bxQudzU-99uB!pH> zFv4Y&we_+lXH~e2Gn%qAIjh5Eoa>bmE=PZfBvbIqH74n2Lmpsd|}7A^~}KsmwZ3W@$VXE~Z*qYQT!Jv>rjWM|_i42XR9r zvqcFrLbi5@x}K-C+`865D$i{YDX2u|lXTM76SY90L|FzC1IaGtf0Qx+pp;${iyt_U z`4?O@eev-rt?^9FtmQW~?mN*!eKl+OL@-W+WQRs!&U2mQ1wG#rvvBRuU>sZHxcE4R zNuC(-D#@gz*1ZISn8L`x`~{(??B;x5yu$f<_sY+E_sU%|3YP$3CNv#bn|lf04y~BF zfDf>XspqJswOo^rYf8~Y{)0$Uom*5_CFThnm{vh2N}nh4B}#h6naF4e&Cs98KadpM zESY8hvuq?0-6bC7n{J4tZ+_-UmLe$=g*7&N1N)dPT(sh90B_BflGHO%3Z)dzSA~>t zAGyP9FQ@e~DUCX}5=_e51#_GJMdLzaF74jM!!_nKqx*i*N|rh~D%wTFNkll=m2-d) zb>(!W2!zD!T8vtir82yn!uqWHD})X5)UtdB1M2caEv6v~_MEJ)uR}<6ZOzVF5`$$6 zJ>zuXwi_NWVCC|tNC6aFeDC}695g=fUck#iESfAH-KdkkXrftM>_(;@4|nZ6zw3+d zKCZAw0U|A;ZSjP~&KWF&Az~j;Y1fgm4)xtuph- z&5ezWq}T%oIl2MF(s9~AmMu~wPG-IBAqu7FJ6rRW{C2DiVXuN4VZOt+DU@=tJN$#P zz{T$HyJ-n<4eL+a|3lI4AZbe&Qz*VV4z37BL{5LwmjOhl#)YTi)VG5;4 zOo=KzmbOfh(Xx6ozLFxcmrO?7$bOqq>u5u*ruTw}{gvjNY}}tY>6RT@B;#2S!Qw8-AK1#G+cReZ4IqW~ zzG47~iF>nx{gX|oF_M6X#U)I4Xg(4JIqHVpH1rBO@aE~9T0#Rmi1$Ti+d@gC^HbBt zEYnR1QK1ysp=`1?4g3v_3zsdN_!ZB-?syDzD@?UxE0@-8U$-wyE^L*0^kBYN{Bk4$ zFEEMZ1qKGO^t5VDn}sxtQvJYiTD=^dQj&?{Z@(HPSavTmVDUgt~>vn^mEsH{)(kqp%HJ2=1pqWODRz=p|Oj}654 zz4akrSnPqX@T1OI%9oxe5zfJ_=iJ9|>v^XehYjfoAy%%-_8%;SENkLBf7Gm769)?v zq5gakC+lx{o}h8bOdD$#+lxp}bKgPeO?(N%P9{%DZHgR&t@4{;d?bOb@gO|`=}ogX z7j~Mj{G+@FK1fF6&Xb&EruQxk+TDy45ZcTtneY|n?pGPT;Jq84WrkXfSa7{9B?!_+c;o5ac3d%embKy z7-;!xCg%W$m45&Zza^Q87Q&!_D0lxai=C}O!k;vT# zR}Yt}N1bdDSLDjc0-E`eyFE6?`bTY!b@&}&`t6c7%zxj-Pv24}==}5r!o*LNj}t#F zGxO7`gLKr=H%qm11GoK=$7BoMn}DKJILcB_Bg$Z@#x2BBoNTaEx|yYRBE6|-Gu74y zSKWb1jM7oAs#b*LDlf{mkRUbJZ}Tiu7cTpM&sAlFBK7ZfZvJbiTW75AzW_8gF=H)@ zGL}KW|F^tV&UN2}y8n-OYjQOe@ZpCDy0q|pos^L^JusrQO8 zKP97P**>L;i+lza#`x-4MCQ*^w`HI}ejZFP)*ySn*v{D#{C|&wMb`_k=&H-D?$)!r z=EL-^`s%s9U-ws^=1b|yALjfs`_DfK`QgeG9lP`%c&jx!79LI;X?vkJRWPUyj0K~q z20aw6PwE}{7UP46d785y;_Q0A8}#@}En!B%TMApG9*Zqm1GX08gMN+0E86-w$| zzT05M2igZTCKN>V`!OH({k{t;J6kE)-?$W{;DFsw*_%nd>(FLO z<9%_|!4D#9o)DXjtp@D(>EJU7{Kud4YCw`gDSb2`nF_oj3z!>lokA&nG{8m$yt06~ z0izX4(HmeUQAZ^T%OtU@V$ptPL|d~vGRX7Cz?Qu;utk_zmU z1-fspD5VdSHd28SS-?!CAqu7Rfl?_I$nTBPP=!+ZKuMtj-|vmm6$+*Ffl>w)_|M1O zXfsNeE0od)O5BDTS-{+JvJHKpG>K|>MiwwrX|zHqeV{as3jDk`N)ClmBuZMGQUbw% zAx9it5Z1un1AKk7%Ee}IF^4Q> zIavHR;2T4AKOk{GSWFwH82uzIIfjfq!bfm_JFO6un#VXtMwZ zcqyoe-b)-73ns=|B4za53lrl^W-$R?+Z*s0g;LD?qGK|P3fwCTnEB-rg;LD?B8d>K zh;D-?$Z}>*md`6>lVN*-u9n}pJl2vUR^G;J5PQ2dZJLal7JZW|8tPHOhYK-^2GQHieU!C0!ydDDVoX(dLt6AP>RtXxL{2& z7E4e)C-zHEV)lbtI*EIzsnY_9=kp4#?A!*dot?8(=VaA2QO%u#gOLoJzfQto1keAs zP0Ad_b&0<7SrPvo$t=gbFQ3k@BW|7w*^U< zY@<`H|9otZ6Uo#64U%0eQ8IVMP=!*=jH9<9Oa;!J=rzC-6-tpDlcUrjoG11_3mUk_ zWhvYE^L6_4rs$@{X}tGg9Jh$GmLMw)=9*-Z0ZTc~&2h76SB$s?9JiQ2>RezvtI9x% z$nTAzL!lIjp*Aoo3SpTSq7axs6wYB$_}AIGC`>ZEj*RPD7~XmXJ%-W+pdnpA;)_d{ z>c(D#>tDJ&B@37*jULj4A7P1sSp8D(cB?u%%4xhgdqe79oFvKuX09-c6GM>5CtuHw zr6*taID1Qnbn1?}hFS969blp>qc{}`6zSN-nsdk$|kRM#Xtr%~%Cybnk7YVLG< zU^v90dSb#!|9My;w&H^xQF&7EzKdBp>`KRLtFS9W8_@7h<`HY_J{ z<)%6@^D?l1gXYZ8rKvs;G-o;s)!9<0&Xz)TwiK!ZXzW`5MEk)l&!9VU6Cq2N#NgQO z;A)Y}B{JRdZ1=P*JnmI)JxWclQ6BGxai z6i98neRJW`CN~M;M1&m(+cMdX+zWVH-&_1<-~lDt_B?rV@ zKMHsGBCSa~(Qr)O#akQqx5i=%eMCmYFSj;UBVvE!JJ>*-Y;HV^%{85M9y<9c-0@M^ zbmF-YOpqmh_cSf+Qlf|Sjc=RqnBI7x>y~$_`Qv)r?RH_9tRrL(JdESj#{Dfz=#=$@ z7KTE>3Ea##0%@rBw9|_)crrqBNr9wnEJ)Nku$<`b>QrYZskt5M^m28MRRfvTMcAKn zm&1jx7E5ef!WjRvJQCT9k{BaNX(#PvDuSVfW7Ih|-}C_%7z~(WRqdKJ&RAkthwx~a z)lLH-Nt^W*{W>&jJAQ1q+Cbd|d>#WY2XK0=bTH0;AgT3QHQ%aP{|%!g@OP&3AIc$j zI_Lg5=T4gKKVVDqwAo9Sq1nEbi5A>fIy<{P?VpbPV&(@dwaicZV--r#g|M>&uV6aT zobC2fdfFE4lP*@}kd$A$%<{{9F?~JUvo!j4C1TNz%Tt*8rYAh`Fl;YcVg&)K=!7WF zUQ~dO)H72bNwODZBdAu$nBL8;dfk<{3J2RUb=G-_hR-r2Wt{L|y4fd)(?Cu6pb6(mx^1PX!7;Z(S6!rmM4_ zWb2S0(VG)iZOIf#ebx~JdhX_}Q(+7uW1PA)DYXgqY70&UT}OhRts?gpUBqsnCqH8jLJkD9Z?#CihKC_6(L; z=uj;jq*FJ%Z7kSADzMan_M!`Dm)~0h1-Hl@C{$~8$_n9^BW9W0Mxx1Ix$XA_eeOh(ZWQsG7Lk@S~ zk}$9cE!3$-5;Zsi%;fLV39;yhj z3Z=+}Ph1mU%I4$Q8YD`0V&8 z#!{U5+A%`hgOxDB{2~~MK(|BBC*|vTvwY3$`#cpq1a*3wY>1EPL=~bIRjPH+ZFW_r)!B>at^?x*A{BnY zz*N;MhS_%{4ol6gbH~qynW2BHC)3)!BlY)`i9?X!iJytf;D5tSYU-iX$i7eUaQF!P z&9}zE@=cm20Tv4(7@YO?&@TX+d@vq#j$0;);z*ML3jKlDQxN&4=K>&z{rl8sIb5B-;A)DC=eVdwK z4<^LI>8m=|ZVk?jwL)s)$EN1CVL>@z?ihrQ--eh3C_h#aFTu7_j5F-t@9@{+@#`2k zG&ycBVkJ`N_NmSq&-Gj*bZCKBrsyDAySBr1-V&I3qmKAH#~cCiAp;4G28J>@Cb%HR z>K+K06Py=gWAkIf35$7SV5KWeai|bPk2q`m`y8O5`gK}^7g;LN0i&ZG@FjJCJw&g` zWYS0^*#qVHrfz6$kuUwwI#@TViMY0@-e$ExsGd_m2eK4 z@2$72SO(sJ-Y~>~`n6inma*^eRxm0SEJ0Ud^t~!meKZpQOt)Kn^COzjh~6-3m(u}# zw_xHe>da192oBD=1@ykfHesYe)d}^%dAGo=l>-9%2hKzfq&1bwQ4?OhVG(j_Z= zj-_vU(iX@Lc6*(+`eJ!X^dwFjtqR`t$~2@sTdofF=ep9JuC zcs>MJGKQ54TZ)R6!JXn{N6QIR5tS8m)N`7~^|3>}#Op9lks<;}xSN~~;GyMiZG!Sm z?v_@YiaBB{%|?V3+LoKxAlnRd{-ML%rr{4;QWsIC{09SWtGH}86ZHYzY(7BDw9L7^0V8C;NnQ}5kb zO7js|RXHrK`$s3NIULCjId^A@3YvUwBCk1EaA8q44XYqLL4h5oIO#Sgf5|}cU*n`5 zrw>)QnaSzsf;nN@W?CQTl|q&{gAqJOPS+$MHdU=Zn@3$YR1X#yy^lI$hdj)QQmD zJyx1X`cUQ}n@*6jn)5g?rSpK!qiC~pH<_9mWf)|laz$J&F@k2+gy50~PIy-JUsHC! zHJui}K~31!nRYu+u>l*Gg9cf`?dPFLbbA09^Fz72vy~!Ax0279ZUc5#3OrZ(ux03`*nv&2 zl7cLm;x-15Kmcw503OePq9hZ51pq*gKMr^<#Z-nuxMbi`BW0%ld#fYj`47)UFo$$1?L39-_EBpb zyRbU4@cThImXge4X-bkIGMu~7xoKeB5bbJu4LW-?BeB3%e&Hqt1EIb~=%-_Js3F%z z;>;Xz>NI)c*&64$9VQe=Aiz^stmkI*3Tx|$#Vg54?^YrH+5xh6W{O2X9;LIIGfQ*pa2V1CIuN}&{edP5(A zJPFEn%6#)DQk2*OdenkN`Y#YxMu~(T+oS5Nagt?i_`Vd;-=O-d`d@x{UXPD zuAq%MBzz4?_!_-CGm$e#Bmx+B=Wa0wa3cP3;8^_-D7rc6v6t!Vlg~5Ve(f7_RnMw)Z=8Kv_q9=sVMzOP&0G5i^@C)->u2N<TM5qB=`qn|D-> z$(P^XjGV}Z>ephLI@W#>s4p9N)!Z!vHMc{YIoK!EmC>B8wIS!mOiW3I7lWD;&09cF z#UCU{l5yQFJzi^MG*FEb*IW$fg%Jh;ybSbC0*ZrNXFCrR+{Z4UUKT%k0z<`%p_0VZ zrrx-UvXTXK>StyCP`r6h*D$$IN8AA32a(VbIC#GjAKW;L_}!OLlJ#jI^I=9O35z1` zEgEWDXs8ixICmGXzTGDorx>j;S{x&8`i`37`>m->IQy>Y=x~Ex=&)QcSFAYvum0Qg zJQulGo0>&aO`mPL&s-C+Mx40kJ8Jr2zcq!$BYoHOjYDsfK|QND`%jwaHpY4G!O9}K z;-IAlRvc||#c`Te94S0jljw91Jyw%KtNGE!YMzJKMguwq134R-i@a=NAK^(jaz~}2 z;eZ_A_4t)6MZ#npV2gjAp!uZKisMk6yDAp0uM*{0Hk67@(IUi!r(Z>pH&KMSkoLGJ zrSslZXp=4k`sBT@?L>x)_0wsQN;UNs47<4)uU`$pu(PfInu7h-)Fuw?@3W?Cqb6#T z4xUJCc-rQPVfd~xJa z>RYT{OT|8v>5})`sc%m<=zUvAeQSOsQ$fkU+sQMbd14!`P>MdWIZNq~uY+XfCB_1h zx%A9Xqs7irke64G0@Df(876nDW&>3-rFS(}g;Ju`sNN1(NN+D3m z7iZnk0o(Q#cPb6=rAWgLe6TEm9r&xXfZ=HaJMiD*gW237l$MLM{3i)Ux5+v2DFDY9 z?M=1rk3VE0 zSUi{eV7{!|HcOep6#4wuQHq?{8>PrZNs*4fadA*289Q=G5t}_Q5+6E6@wHF^)eUa?Z2FStm4JL_fNw|`>kuY*s=Q} zG`!iU%b;PrnD!lY4eqzD8gcu@>U!U1;tiUB28g4d_usC~T;yUj42zSyF4C_1&2`Zf zG*JBfJLbOK-~1~`vkIg>A5`4wr-wb3)RQ10ojR9fuh zi*ioou6^?zHqd&73GzNS4gD43Jnc^O$M!*w!8_-kpRaQsZ^}T7vwNW)g#CW5w!h% zLxy}|ma1BpR2|XWf8TvNpg;bv5myoheF@u_xuA*vW5kP}^gj4k zHAIswXlcH99)!g(C7P5Y7g3`Gzxu4CpYpy~Z&2f6bBwhM!|t(x0F>t#A9wV@hkDF0Zy*DjM03nC*})&+B%AT*aW=7* zHcV^vvx&t_qGf=l&s7geIK(KYVSyivq5=+9*nQ4b>0Ep&7iT`Zp71oU6;hBxq-^80 zg86u0ltL-`@qkfaEUtDORN(BZ=OU5S^gJtiLekA6$SBSxk1u-U<67U!G|QRO`PqS%EZ?&Cf*YjIQaH^+Xs54DM=J_6gK*>8%dDi-&2^B zj1j*c;Yb-B;e=f6V#;4Tq0gh65Wsf|zKOruvlN?zJllYLnbLtR3LwlQXl*i_!O*C= z1nr9*UC{Xb4<3-S9eIp)^(*}u`6R&26K|oQyl(=BL-imD1Z2MlJ6@rC^bvpcK6XCyq4PU;u*cOi}>7Nbj{HQU5$}&B`|G8eapZ zNvfIMVY4nFQ|ES5QIpL&dZ$MSpnkOZ?k@WK+s5~;pMZ%fs=AfVkI~a%1rl$EC-1?N z(Ffmv5u_AEsOu+!mZ4i?g2fb2@3KMpE|?2tDd;3L!seb)k~MrLICLM5lwa zI5$f>Q24iQH>2suB<$4fmzX`;!7w?c=%gwTp_#FnIiZBHSxO>9eODn|67R5=@U(R} zRP;MSK9my|cPN^m}+o*G-&Q`P}bP2-bZwj~X(;25#s0;cK z)LO3yEr^YUS*{N{{HJ0<){M*?WjtqEKnT^0^VeQjhKmf_%phF0Y~0B-f1WBKZHV_i z_y(E_{!dYGxjGWczDnooM9X9@o9J(~-t{+R#I`i1q92-MR$BH& z+19~@-asK9ZMC>NPr)$2oqn3L**-HCgs9S=_3#$9P3DCj93C7Kk zB&%xa?N_iMAxAmIneHPj2(c_4B?D=M)Nsn}Ts#SiFA-OQ=RKqI>_v}Z*gyqJO0Z&u!u5Q5AEB8N6+3peUKM(l5zV!Qc;J_7 zP}XTThBzyphOd(2Itc>;SZ6Ae8Qe1rZoKa;2hg;*uOkyH={DoQiC-l-ptMi4rDBSt zYk@?9>Tuc|>7-f%VdiWuT+KwiBPL53$KWS(MoHAUMkO;r-Je8B%jZ1?eXYpfH+{dj zm)hN%zHE%To5h}3rnNpK`XIK-5+4rv(s8WC;i*it6*s6)jRO_b#OId4f!L8y!O*NrHoK1Gt@HgsER6Q-8CgD@%jtURPRl@qd_ zP@f%-bZoAmn>}28HU%FBVCxBH1la@V4s<~!BXi>CfXh=9b-~OG$VKQ(RYH)ds|i{T zo=p%S^@F%$nzdfmXK;kIk$V)t0q;SEwK}LMi&x z#0bsI044``Al+{C`3&$kx_w%-+l?L{fqMJy@s4{g(&Mk-qOjZSD0^H-p^qLXRP05h zdJkK{uHpJ(3sbkT*czceOFcx+rbcYR5N*J!t4W=il%cCFf*6;=X|Zp5jMqIdBT&8a zKdpo1!^#6MBKyp7nP8UcA7XiD}?7EXRyE0h!-mczhqZK7|y`%->qk1QaXB#^sXPmYe_nq%SR>&c*}t#&{uR$0Eiq=}9Rm6^uvG)#eAr7WXG+iw?wN=LRdkA? zByksh9tSWfGBYaP+>A8$?ZJZfshjEie#IoUn4;9eB%5C3ZifG#p|WOLz#f3EWP4gM z-jOmtdTHay%qW904+dC;|ztLsAZI# zC7zZb2rZ(tjuyS009xxQN-Z2|pjA^r&0g3WEq-fJuNG7OOmVY>sg~IQCQDq8Pjx-H zP0SWoQh0AN!qBeM8+-J5BC}0G_3y1mqQyO{b+Fz!7>_DP!qZf+m{Qe6$@;soR3MZi z-c4XS;XYmlbQqqx+5xln;ryWI@H?=EpE(?$p~L@S{P?{#tJls}X$J9HjclHzR!hq# zYgTOpPF?wLtnLaE0BB4aZ8Bhn-;dZdpua(6_~5{KCWK6|#)KIvwvw zvVt5fRt_Pifg?)-r{*WX5OW!o(+X--jtgOHmRP6eSfw{hO7pKA4<82~Sv)^b=Qz|M z*gfp7qWK9&Bv8QzwCv1$^(Z~vc8>z6#X z_KD(49FtI)&>S1hYss*2) zZYo>W5H!6hpFA0Dijn6Q+*G$dfD8ugJ;)QqNP<2AO_x?DP{KmzsMv@MXlN)O3sIRQv!ZGisfH?oB^t;i=rugD0Iyq#q!tTu;5BEooF z)f&HZ8`upM!3rFmh^Xm^s~Y!?9Zsc`#Ykd*s2{e&`*|_Fm&N9MYB9?si~y2J{%Xe( zoP7j2Rd{Xs2%sdDGpZv)3F%UP1rtk{_e-pL5<|c%vf5dzY$#xfaEaFMqiyP~J@A1S z3Im-vKnKfL;`;y4yR-L1?wdI9YdM7v|4(pv_)F%C@j{MM@^KImO zf70`PjJFs)XkY!aOUx~P7;``KT78(Z!SfmJO0fF@e^I=vPbNN z&w-9VfJf|QJ^YcYe6%{DHdC&9bnjxRP30Il{$=CiJ=giXkm1+(s;bEMjOQ|ROvpMC z8m*_4`%q0*WYlq9C?$lvr&=MO>S;7vN2!upJzO)fVE((iOI9Qw;cjH=*6j9UC>*PXE{Od zHarsxdaK=|eS7IOzQr@tx0hBokTS0lSG#O@t{|Y<1Ygu*B(4Ee_$TfdXb%LDnnS87 zHhj91mkjLEi(Sx)A;=D}A$iGjv}he5zj4f!_EU#FbO#DkvC&2`)@T`za$xvXSurGY#9y~SVtE^&b5MU#cLOp62dH{KMZOJj;rIbJQ}x(hm>qa7SqM{6_4l}Itg zlArO*3_s&c*>_sH^+>ckoTqSuw*=llaxfsBnAH6%mRbuR1|)(px*LpQz|ea~QZ-Uf zRIbWW=Fo8SQcnxQD53S8yF)G^W+R)t@#<;NkjKL;9uCB@WNx7? zSa6!2Cl)l*TO)KI-W?UxOnHu2-(IwU$n3H~LAXul{7cX_9GNob{|KphQb8DxmRMhE zAg;Oe0YJ*ZcQayin@sN?PNzK%mghImVEIwzj5kIzx=u@7^W@(f6?M;PGv~x2EMFXa zfmk^VhNlG`Y;}-IdvgzdzvnNMX}8sLkLs-1nSenId?mY2`=BSgPy4{c?szS6o!@(8 zkY_Z`q*Zzw6bAgx>`d`Jybx#|uGlbec%k*sKe!v{UBv$1QrCsWoBVr3`~|VGqfKpdYy`ak1?^|^*OyazF;kEQ5FD0#CY$FL>4Z(oAz!p%S2>R=E}sy zNHo&2klBH>=xy-`934|W@y*lc^jzJp&=NRRPDm0+x1!Uj=v!Sy&D!pEg;M2#kSxli zFzA7kV7f0ZnF~QlOYB3tVCX?gX!oVQy>#AeaSvCPQX`Iv=Ur$awJ$pH4F~2meNca<1akuF4U@c{szaDUkmaEM)7T}^xi&%$_E zzq$@#GLW*GR$5v?ySAD(TCi)bttLT>Q`!`5HLWBOO4e4>Dg&V;Z8e>hAmq?i(*X=Z zxcfLoAspzop&@x<=UlW;B15*rBF8e7Feg+{f_`zA;D4(e&|^H0=r?%Bi3nv+@zTP4 zp@V1UNYPIZpQXt`o{``CsR0ZCh>AxDfC~Ux@{y_A38a-10i$IqlJp9zDgXCmg=YS+ zDU_nGWmXfC5S;Yl`zTr&dWtXzx)N~G?CtLEpo-wVTh?oBG*F=w9iA&e9WL?z!zU*E z=@IQVDj$;N%>;<&6oUYS&+Ta^depsz%4YR0J4m4vy)3uI?@X}bjkCQp4a-N_8F%Xh zSxn~f5_&hlY#xuz&L=Y`;EI{pz$usqU`qlFune{3iK^RaYml;NTa3FpmlYwr$qhMAWr_>;p2ywXQqV-pt->7~2e#pYjsq3i>JkoYXgt*L znU-6LpHH>ia=29KWe4|iiagmU_l=2K!y>I=Ta4DQ5Tc;YeoIawHq z@yn4&r$3y?Weq8USKQTIQUqN;YZrC>9_&iN61v&CNgVgGj0@jitw)dJ=-+TO-cT#zyE|23X^tdA&cPqzj(BrB&?s|?Z*Wt z)T5pWEXqX7Y!rBkj|6#sdrqT;9Yzjjvv-l!LiVk4SPF?fly@}&kTQtuoP?gGvccR0 z(hun^hvot+;at;8kXjW=(Pcqyf*MN;f}>(AYLCuFQ8^SZ%bRzgQVto{ZG=+qx@S?{ zhk{&p_p+2?F8f__s7L}a3qmP}WCJw1WY(1nJ4o`R0Dhjh1?0kAlp%SNEHt05YO4ml zu*yo1V#hRP-qCH!TzF$Oug;<(NAnw^#V>50cR`0BbJ zLMLkXYN)--&P zum;o(Fx3WAYz@r~pKuey)WjJchZZ2D;N&+3G&zSdXmF=q#s}fufU4^oYPw2k>@$SQ z4r|2GL2M=iiJ$tX^og6)-jCS3g0dk|MsG5z`l*uH)8G( zpglUm;FNJwTtnP2H@v}^%!X-Auf;U(N5so+rtpVLQW)!WQvHl?{Z6J<4)c8*0A>S^ZD(ywr6CuTohY0WX zMtx$AUw##l@iH(ajPFyoqC;Mb!Dv1~)AD2l!mYdUk0$8DawRgu^F1agXzw$o_d(Pa zeIf_T5yLUO_~p}>oRd+(2eN|e0JBMJLZe0-T@=6kr`icn6rJ4OI-i0RxJy<`LO?q~ zEuR8vPS!{E@(BO{7cNFM$xKXg0fUmGCNNFP8{se?m1G%YkhWI(gzVjx&}p8#(Sh8Hcf^7cb%H zDvmDE}o7{@&OBO#NnMy~Iyc1h>fDkCpfJ4&m z`!Kowk-*IRI=TMQ_hCbp@Y3rtGv?974;%^;Cg#F&@zb@5dzz5BJ?)zwb<6dSuUA9j z6-wz^dp$|T=lBm;;f1x7?|+Q*078N{V*cJxF17#?)Yp@^UvIvd7eK@!_M&%D362Ob zhw4jw@N`tPK5>=4w6_@sOLg2_7W zQN-O{h|>v#f%uxHt`E=u2Z(=$4Rb$;S2E&zDHE?2w0YxMM~~-dd`yx{0fbUOIyzIK z(-dC6AkJhdc0L3001#ddEn2uOMqiSWa5Q}yTk1Z}m!7NaF&qWDL8Pwe`@D)0C13Ml zSRe3i>j|%o;C<%p3tpT;iT%Sn_fil1+6mq>7lB8KJ>aoP!AqgN>>i~$VXm2N8{^{G zk(n*F!mDY){$#_vpso*g>7uAU*s8X2FiJlvC%NP3Wr26R+NioY zg04X%xQ>=wLAm&Dz5PPOzlYZ0mJ+Ax{=VOvVu58#Z;=HScU)7{To{v_l%<9!M>bR# z4vRrR(YMREaoVNdIFY9@DrFCxLdfS$!~nh(p(TkTu_fLDi5?8#l7f^zfIpoB<5tp_ zPr4uiFLG7v&Qs4UNj9#BPl@hj0ExCwe$FB4$j9 zM9NJsD+as&?e7@s{-?j=7WWhM(Be^Pt)_%|9HJcRRS%KW8mp>XP@Vcr1^zCH32}O+ zoHypIinJrv^O7u!QuL;z1a*q1pEwsujkJbTv#lF=@j#lCoVWyml3PS6VhqI#KStOm z!|MRZ+l9{L5Gzhe-tbq1NNguMD*^jgQK=E3Bg4buWNI1Y=O zehzp8dYAAiPeeY}|xR+4i%1>iB=7;G)0dh`x2O4DTTx9xpQg-kl%m5Fh+_wdyKTv#jbPP;KyKt30ssvdP}Me_JPp z7*Viwa4r9$ghkTSVDcMY>x__*TYK5Hs3o|E#)p3o;flu=Us85CQrTIee19SV(925^ z^C>QDLeTrB4Qdgazd1VNd_?7_2M$G0e$_sU4`z1Hfi|keJ+Nf%Oy4YGIu| z_|jv(PaA#v^JDAnJN8{|$Dl2lchlC#<1_qq?deE9g zEw}a6PTbiSH*;^be2*e!W_aVBys}QBG!|Of<}x$>t**WG|aRW`f`zT+c+r*jX1MC4nc10ulFB+^j+V zJv2%zoATh(kEWDg-N{V_efm1M#Ke76n=hUi!2zlJ=x{Lds0 z4hEo*7>OaV@Fysk1^3Lur*98MVotq*So#^d5j%VA%zit}`j&8hvdujbvHm?YA|O91 zDI@c_%HjtPh=M$mk@*Z+uh^?$CS`dMT1RYzTL}o3DJ(_kyN47hx$6Sv><5mqXxY!A z91f!hF?!)jc4=-;rPYmM-sojs6<2DJPhSVc*)j>fno ziJZu4VxlV6<7`1bD<*w~I62pt zCzGxA9dRrx`=4v>R{3~KpKGe*Tyq`E$y;E#Uw<*md8E`Vp)A?rWe6*hn$4JXOzso% zVEQmiuEbN%VCyPdXb=31@aA6V-PTKx775Y#vj>)U@}x9Ymlwt)$Y3E#uDYH^GYcqV zYN7$ItCEd+nWm*LHJ+AKp7OH9sW3hDxB6LdS={N}g;gJ7_=ZPV44X|ON-->()k8a; zRn$sQCEWlTVi@ul{)4S1iQ7r~8}Q1U$yy7CgPStW18=-U9%Ns=b1*b5q2s*m8$1Et zwhvE$%h1Pff&bab~){KlULqPZVB9akM+cTR58rcXf2b4v%rz9M9(*Ka)A= zIEU_EyXBa~Ii_$9Tt?|RCP^461jEKTMk5CnkBSvvKqMcC@T;i=#Uf*z9OHnz{12WJ z%Ntl5J^pufroJnF9hO9KSd`P%XcW1;P);EPSVOo`na1vZW`?XhohmO!mOSwsRO7vl$nnNM)1Fx(kjb27Hg zh@;yfNt%{4aJ7;`cs@uW9nX95P?dIB?)gfB#5@Zz+RRLNNFC|G=|pn)9PiXo<^1ry znK-oPdegP|a-{2tz#t=hZ#oiz#)h}3d2a#?ZogZnpq%lT4p6zUnHcwRZh9|^+)6UO zK)rXb#h#{fFNgP0xJ`!Npm2>0zl<^&e0!h?LrO{Yjr}GxjSop!7;>Izr|w>{c=CbT1h7i)`Bf{ z7?)FX9HgU1dj6{*K%)8D8>9K2NKmY@*-rs9`~W~h1WzzT7V@hz9qO21U_OQixR5vHyGfZ7gF*ra#7TVp=FCr}J zTby%@f$Ang3h$%?^YA-$46PuY z%ywF4gE`+w+CF<}KICiKe(#S$32zhCCSd;+hm|Zc&GhfH-QI=GP+5McEI*XWdp^J< zL*N>)1Il)wo3K~P#1GOk(O29T*qEkkoD#mNV}!$@ZK7OSz>12)i_^9A4E%+?f=v{m z1rj(qQ%j#8UYtqM6oCh5d#Kx1%+iX!rQ+eCMDB2Wbv3ldJPq;K{yuJnhHb*l)b}gm ztLP_FX(y%>9&ADIOf4aeVHEDBy}9%l(-ffI3=gdWFmXM=s3DqgLhFc@SXSdNSS>v< zJG73%nAw^F?Fd8{jscY&@5B?v)KxljgL)07P>N>`Zg@FC3uK`fqM0H?U)U+ZQ`&Axm_8kdR zU?PvZ;tN3Lt~g)#=|#HXr4Nm6Af`%`y%1u;g0-j{z*_(RcSFo3YI2XxI1g6rrtiUz zg-0p9?kwbEVg1g6i7xvsW*w0;P4B%zfRnxVI!&%&kPM=Cz<5+LGa(bfAs`>WT|oFt z98#!vK%mqAq}ut)nJ&&aMu|CzQadgH9Z4nOhd9SNDt02F`xNpL$Iyr%RX67Et*=Il-r^hcx6 zm3rCQBxG0WWH^C}jwAt&Bh*Me-f!O$y2F8e2WZCj(3yH_ERm+2%F%-lgpXgTm-Dj( zvZ{yYMC#Kz3g1)cZ3i;EP8kgJK|P88L;vX_;M2&#E1J=&+GEDCZi{J+y|Fq|Hj3&V zk0u$Q=W{_=pNKC1W+P@1)_Mz5-@RJ|jv3^{c>Cf@o8vV8z141W9Kyd}er0n!f`6BO zZFAg;e@8oPj)mvsTWq?czo? z2*r!TG~ob}Dm^#ABI7S4iJqy)U)b-j%gUxl0Lc#$4eIdVgtXj|Q@6~GQ|D2}IX2m1 z7gh2Y8!GpB)6l#GFL+>XrA>tE(QzGgbKq^q5eZ<>JM15+&br?3Pqsv1;Hi^Jc;GBU z;J)HLF%`3{$rzN{4g|&V|LTmi-p6kONM~b-P5g>#Yp2@qwqCw?OFK)op@9Qozt!|e z=Er(vqsX_2ntMv+%jE2QV$8!eJ!-=a@J7rEE37NVs>4+4PP)YaL+~;ED+eH((^gmu zKS#+`KDpT$)jcNhk!O*X^sdu@f zE4RV!@2PX#EWKg~)%LG2hIY#hQ{v?$B6>n+egS{=2Y*?2_{+NBmlOON34i-#7ln`W zz+Wzo-fvYohQqK|06ct=GeO!VtIj084B#UzkGWYn2C`3m<@;jIY+rqf`ReK zPVc5VVXu9>y3&fZ3u4rDgh75DyYEsrpwl{nfk}vJwPEqQheA>BfFkX=&PY`{s)Wm` z;3gZ=f`J77+X0M3+hH6pI+kG4n~X@IML!ei^C&){ABfdDmDps#->I9ZJrZ=i91Yz4#1x1p{rU0~!cL6_mRmh0sez-r%0oNC@tkh;Z5-YLoY274Ow? ztbZxi60vCs2MUL#%{^dUIWRr#U}0OU6@Q1!iThAyF8o`!qLbB+?5}Ndq zAq*p)VWb^&n;?8dM_@$WU|<~ns@yYp7aI(;1B$wiFjE7B8J=H_P(y@uTF8dK_E0&t z1UEJv3@pGGUT1mlPy>`OH9(mP2fiNDLVc`;h@z?lZ?&b>ue{1jX`fLV$4)J@Sv?67 zPE5;T#J?V+hNyFeDZmgyDTtjw&dPK`P|y1rT?bs2ITCi#z>i`4Vwfs; z3@nLO)MPI@K&vI4c*I{*1Ju!;S*paZE@T~q=41y0%zQVXA!>*?O|qZL(a4Y`5-EuR zG5=8>mHSK2pJ)q$dPPle^X=gtabKmkXcM<+!KJnc_l$qPBe=kZqe5zK zgb`!hnG4LUIM@E_g}6P$O{(Ol@nBu`BmKTdClGQJzhP^K!> z-G_p6;!c{O74B=flfd!#mBdlk5z!+d?%;qV9nmN~$SfILN8B0=P*fU^V0$S&lJ@VR zc#L4`a(me}=2K<|&Z@paJgG{e@<`AJKD@S(U8xY!v5L5(oBfS-R4Q#9akb|PmHAlr z<;Gz#;Sf3+lWqbRRqyDC;h`8zxJAwBRAn3H-2!$cOr&!+_$p(+Zk^=RjhVu;A;?%u z99Lbiq{F-0BPxsJko67Vr2BZw5qIE>6}3UqOoDGUbq_bHmy>Pa=4ZmaEWmH834WcJr;vuHOD}H}4iqhr>&xt2cR$N^V*I3>2HV zGt@VT{yJjZ0&QI<`qO{V*2FxetrPfc9nSoSZ8tu&07c>Ap1s1Yqn>DzqXoX(uvioFmqBu=i9WJcXd9OsStiDSN_lR9!0mm?h9RB{z}muGb!ZDnt9=YPg+ zHT&?9-s1GDJ6(p&J4i#H256SZpzu24(`+%F!f#AMm^%qd8%IUPZ|E@l%3A#}dlnLD z4M6_mc)FGStn#GWnuPTZ_AR7Hf@|tYPyxU~_+Tjpd;}gle}dHl4!r0t?&mE4!`^^& zt-37-pj0tkqHjL(wkwpPA9?etT%5V)LgX#X0+y1%d|)&63Ag)7<8uEJT}8oOL?=&X z45gCFCzuMMr1J1VeU6_C0Q&k5s9&On9MnFZ;(3CfhG)o9TScirjmF*G4d(LjoC58jeH-BV#v>-^i@>jWmGR50Ry)Aly-Q5M(!_-?XE7FgK@g9eNe z6)hIDD5!)WhL;Vf1U8A;;7h<(tJ`RM5q1SNiNQ@U55pqu6>4vRQj1sG(w0_HPzedp zEEI&ORD)z~^wRpIOKlXYL06Tg?miVm%6sYQ`3910D>A1kg8ZA%HYp!N`DivNd?B zzrQnh>H_~?ov+Hy__7sOu-hAhjnz<9A}=m))$(p=@D!x2pK_LNm$OuQ?G2uKA~3mr zE!PcTwa->$Bb zCW4Xqe}N%nQwj-4f3n@?>Xl0*AMHfz)g?y%y-tfF^xE_6HPhFzbSL2~1Flyjb({1W zi*73xKVFYY)THko0S44gc1GP83Ja<;WUofk)i1N%=g=V9UD>PMAA5~<=fsE0vuthD zE|2oFYz`L_S{fGAqZ+-OC5I;Uj!nKXrgvfino8-TRe?ljxGBkhD#f&cffn$ z+DFWC2rta2sIuRue>xEq>BEfbrg)(Oh<)t%zH+(Zxk56_ zWA-9tOjV+n4wkD_;?Os^UjBMuSy0-GEQ5V~V~}O-3^ZBd8-tWGy&vilTeuv1ec3YQ z%RzYl@waKat94v=ILul9n_RZK!qo0OTrFZ{jpa z^so|1=lP-Zq|=s8nsig-h*SFFga~CCMH!b7E3=rATd?!ZUjLGPi3i z!@)ndbL?Jxt!gTUU7w`*lc*o^R|8k$+(mx4``e*RMJLwmkfx#tMsJdd*Pb8Yc=MvY zj>k!cL`n^lFF~p3TrSnBnE#tbhh&nzmlDvaNdxr~AK!4Gjx=&^-ie&Yi+Xk4cYJ)K zP$H;knyWJ8+=vbs zi}N>|bM4NM?IKlBTOE6n(Y!ph*U~^*}4Hp zz)o{Laq-pQvrMnOKyrYZ?UFS9@HeJ|7FO55|iw-t18N?3D zXGe=xO%5ER!=1ISaH#j{Y-b%I@$o2AZ-2woMsA_}2?!W=kcN&A;hRSW#dQ!64gC|D zzo_Hj(PlYdhvs!D@1R;irZy{Elg^UD@-Au)N(+mqkF!189n6Tmqb;Qso+!w~yBZ!G zXa!j`$QB0eFOEWN3+fbO_FBlT%Xh z+Z`Z^VC*Dg1}b4I!8bZzU4ALw z7pdOU_k#^g&HJ$OljyL-<;fdb#ayPGDaRW|uOhyp{c3LOA3)V+3zNf=#~_8PeW7Vz zJqGnSUw!TZ)~6aR*1c#|aU_W5%m6jhp8-JvGxLV(Si-Ptracw=8O9Str&;cV=nAl3 zMStB+R(45ZVO}!eCd4dlE|8RyEg-0V;~l+*YKDB1G*o-A-L`codFpmqGI3q)@8cUK zffzG7aG@~`;ieEQ&{_KvPzB`UJp?c?>37oLUR%4M(MLxg!d21w18eU z9?wwJ@^d>jDlz?^7oElNM7^{|*4B8Xtwu}f^=tW4RT{lj&_W0!>!4~&3Nn{U5dZh`kYZRV)`zB7Ow9=r)OpMRaQ(Rj zrVsL(?QGE0&M0C~fT1m?C0T%R{gKMF3-3C~ElkaJVsDX-+fVGq278~rZTd+pQfwK? z%EXh`pcc$Tf$b!M1zv5X<){7wWEIL&fS#g2BwEhe--GUk4=86HwI8OZzPcadhwms` zI-ozBiq5?!o2z>*9(}DUc`=?%00z;#D_@#rL6gW{Hz79G=$GLUE9 zOIVZl@p!8Iv>GyUO-@cpMGk{{eyquoU~)|^>`^lpN3O}Nc?(>h$u)Uok4%P2t3NZ% z|Anz8cd5n!MYo4epaq9tv3GE?c1L177e&{&;)*i(T0olQh{&Is^0y&qhN^DbrV}|ZuA%At z9J~YSKm6|K0I*5dx2fsGi*8LPbL~Pd#ZfsiO7!2O0!)m=Yv0M`oUh8+EO)2SbfS(g z|8hwz_{kYuMZh5$e#2mAzA@_i)Nmq;so_-fQmzuJ`5XSP=5Zj0@(M+WQL??T)NmqI zYB=vQ*iDj#GpU=dodO*D^am^Vf??=6SFZ_@Y`M9pSW zd)p@)?ai?&`Os8{tHA;NrrcGTa;I4~uHUR;`Q)qkN&3yi0#d&@Qx!;@-AVdQlbv<* zDd-S^NyjfZX4cBVnM#J0%!cY#B{Hl!N)B|VT;wO#l$J;^zn^>lM?VI+e6v(3~{9Ts+HKqlfHLuwF%qS zVvld4Z1rlk$^cNVyi=8JNwgcsb}mxwB$C4B;~Rq%<=IME?A6;~KZ(1wWl~e@)!)L; zvM>KqrZmOyf5z7mduobf&=fnnm8Q@XCt@fy#cL*krq?HVD^!|d&=i{D>B*X6uYNBY zFz7cu&I`dnIAf%y_zqT>JY$rm*dQ4#BIBB3XIxW8P4R470$xrU=z-M z2AuwDjNM8_f4N&3rgj+A!g8*v4Tz;<83$Ft{Jd}@kO-gC$2SK1DIa5U(jR?6PCCxyq(S^leOMoeWn+=|TUF~gk3M)#qmi@{3U*pdY;38@C? z_z}5U;=(snW&sschUlY7#BOBqrty+?iKMv1$2SHkByy^_D2Wdm!BST%kNx16@qyEq zd}f6H8IK|*`a2uWF=TQT9Es2;s%et|mt)M{8}P(2_69euxpV!_+MRfa@5*VF6yKFA zZFqtbb_oe67e+4qPxDve;`4CsbX(v-aVxyKLhs61MZGI-R!O}pe!~zMdRMpd}aQp?MP3}BVr=jhaN`qgoR-3zcQ_G}& z3Jh;`UEQVC#&I> zsF27dQ#vn&1S$yZh-)NNP0dnUXR3z(LFUk_E^aGLlZ#vO${{tje;g+(O=MC@8rwuDq{j9+RUi?HB#o`&>;Bs$ z_nK;QuL&miN(SK-vY#oc#zgirz{fWR`{_gV`qTA^s#ob^3i{V2$Li6~CGs)@E^0uq zgSUDwgFYBR1U6~quW0Uit$d{!P+*N#zCn|IXSDJf6<qQgSm5qnqrSiMGnin6 z;Lcyh*MPd__r=g(AUq2rY^=bP8vBJ7IKjIE@c#UH z^&#(3L>ricvUHTnqRBxvcA%^(Rd_U1x~bT!|7J2YdgJ>A9vALS-1Q~<2YNQ1-@R3o zc8Nlu*(sjRA#gF=u+ZGk=PtHaUng85MT`;ow!F^*{qo#fowd8k;M`rJhIywWTv6U) zd*uhh{k#aAKqLSqwQW*CD^rCpf1hgK29?Iukka~OIMp3wuKEy-rH*pR= z&DRcp^U$O0(HG#pE5aRQ1O|x0-MqIk8QXLQCb(t-0~6Him*abx8+kVM3uM+dA|~Lo zgiaLMZ!gx54nBiSoAlGUvJ(o%9_gs&Ha~CzFyi*mUPWUDI5^BKh1mIq-1`wZ*nyM4 zFFY)@Uv?d}v(3*?QErX*VEpz6lye#S_@!nyxc7^~L$cEt0jlp#>mScucqml52DIq9 zRd$-Qu2NEYP4A9-ug+|Ny|zKPH;VsC=(Wz=C)sNkGJZm0$8ChqyuLfm=(T-%-l`{)IrT%;87sc8NK_s6?+&oqp#RtPCL&_2+-R1OLK z*Qa0rBA*e!Pn5e5{oUi+w1%r~%`u96doU6p$3X~Nk5v81Z$SvdJ@Cod>QzO|qd&wn znGw0<{l!w5(u$&nsCej?D0?eXd+dcT@bOKe1ZEUyWPyRIKqA|A`S?c3$C2sxF@6GL zBTS!zubjwqUhL0`Oy|XZgf!2`mpw9lJkNPI;#kD1Z;GB}^k`D#-VtN4QH@NeI|JmU zI|JnPG4CU20{A1ZTGZeXzyQlbuohj{Yl+J8@l796cn%8`r~-++|5_j4^nrNSMidyL z3M3*v(8o7@Yz*31;OMozK)S@oH+_Kg3Jbif3M7K`4Ikh10n$bmcv=-m1ZjYeZ~6df z4GR?Y25EqgZ~6dfISb_U2I&GH-}C{Jj|KiYy4R|HxsPx90BHsbG^hfJtNL&s-}C{J z#37^#BvMJ?V1RT`7x^$Qal3d4=?e&EmrSGiLx?WR@_}w;-weDD;8P- zm_3@R;nrN0YwQtY#v9^bF2ZxJkl`(*?DHU^KLLC&{iTEXZBARpQmY@;Yiy}lYKm$l zSleF28pI=>Gk-U6HY%tGDfH>8f4(iovDf8%C9=C!a9@Wp> z0QTBIK7u_oj}hQIgW&lKBTMagFsJ`g3w~e%=R}rvAv2z`B1=2f6ATA+^<>JiV=l&gXg202I*yR#Db%GkA9f}NTN7A=cgb- zUGJK0KECN=rj2~H3fq0(h9q|2Bu6T3Zq5US{X zUDT1h?;j}h8kHuC&{#wr7I~jEkQpk-_(|~NjMSYlM`-FwEpz^Er+9pRFqXFHso03^ zi^umfF2VD%rw5)+;JtK&7Z2#@Qilh3yY;ERRu?{Vc%W(g^_}zNAI~@E?>;nht~_~j zKNtV61StG3H2wj93Bx25W{b&o=kF%l#9YQ1%5?rN9BfR&2`B|mr8%GZBL+`w#M(8v z{am7agb8Yi0qaOy>UBGcabca)^#{KK0L@F~l$wjERJdQ6Tx2BOxRw=z0=Vqrpo@S3 zzyI1r^xa>($f`)sB9IN=0!=JHYpn7AAinu;`rb9}w|`A~{aMkgZ35{v2F{~K<9{Un zN0b<2p$zDE6JWwGlR+B;n)Ol4p(rOylGN}1?%yi52{)<}d+0n{D{bpp4D>q#T8#sm z4;%k4;G6$F=odUTPP{Fak&7XwS?~#eiri|JX^8HJV08fmtJ|*y5T>q83mgqxl6OEJ zcFbwUl%0DO7|^(ip+5eK4mDoQ+xyKT{oN(D7hH>SXZ7q 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

1&NGK9x%2IG9 zivYdYW6depvQzPql+U*Y4I?Fm_-c*$5tRsS>)|}ecJjG32!u+2?IEA6rLr&xV4B~MqB_pNUE`3pD z)(VYNjl;SdxEhu`FerCTh54Er$U$!xs-$DsGQMJ;J8SEQ;-t-Y16-@K$jc7I+lx9wxCyNaoREh)2!ml|{LLPe3;{dv)P($w4t z(@UkT?UfgQERQd}E*n5^?Vr2>wESkPtawX4C06Kau*BnjUMj1wN+Mvi%LwF?j9rNI zMs7I!779Y35GDfGS%G(0flIBxLMw2E6T*X4id=F()Z5im%L>DATQ}(K5e%G z7g>P?R$#FNig0U_qSubvN~Axm6M8IbAz#Z>%<-j~Tq?}}{c*)W(^7RxGNHWbevwRY z?X4i2!2Ke5R6Sq6VL?@`d9;m$id)kaqo=nokb)8ZUV!n%RsVyp<3h?{1sboo>%3uc zjamIqIE=P{ssDI}ZhXutI8~r4%*OGGk}ArQhM9YXwTU=cnn^i9|J}z~o}P1DpE>XQ zBnRr5^S(pGV|j4V^S-8`JdSJhx2^fr637t|X6tO`n}~hdTiXQ6B3wk3zceXSso(mV zZ=MvW;+0rIa*`7rmf!c}XH*@l!173jmhAKULM>MnHw}D-BGl(`2;PuSuXd$!o8Fu1 zYtcZ?Yg#?!%};`${j<1g7|jcbhinrqGv zT7G`ef=NL(a|?bE)8WdFmrtMMy8K;XZyWiB^MR-js0cQO~+q9rxN zAAF3sN>*nK885XBMjQSBLG?jhz98KY97D{9FHTwiJtSY06&odeK#pv_i8!`-mwZwC zCc&{WIQUnY!dbE<9tyjQAJ3|E?Di-t(GsBO)8k|CAUw@mr=p8(AYSf z8hgcqZ%ANR1u8)5CGZ6a1Q3BvlHV?Ya@KSnFp&gbzOSO61I(@RrA6hQX$3pC627GV zUUo9OIj~E8UY?>q;pwyT^mTcv5IALR`~icvWfCs(cd5lTcF(_aV}537@u3eW%7?wv zH0+Xf%#{(AKS24*1Fzm~HnEbpFOxWZ%oQD)hb&*i4ZDjE`(S^Z-^|^8pw_R%HD`(& zwAwrG7%!kJPYTG0os;!VEonTMg+?pQ@65AyRNi_woxXQM1Z%-~+7^}8t0I5Nj@(7$ z@g2K?tb?G{S&^4oku90P{a{j+`R7)Ujr#Mmi2v1ALOXKXS045HTe-G#J8J* zd42RzgxuJKYijU|9TZ`Ww9=)Kztc(=>!p5dx7J_2)>Cok=(9e5_1dZC^jWIU(jO@N zHy^#4uUX{T!(_oz>5W=Bs;>pH)i}5=wssgbr@l!99SGp@1#{Y2ddW)VO`ia+gV$!@ zvbi7|vQOBMv2X^MOhGn`lvjEq*-Vu89pTcYkQ}?bnqqAx<$Z6I@=6zw>4y<7(^jpp zrN&&xwVLv*4KF3Xv)@*^otW*d;E63R+PgLAL8X4Jd+pSiF#E!Io4G~k?Us7QB!X-| zuW~nBX3t+WUd_w?bE$j7h<*^?!{=CVkpA}Wdj4Ban(x*s3ZKP272b_|2=#D^%8~H~VBc1c5yzh=j zsvocx?V=i_T}p&<1jf)YbKkCEqykXAcp|Darve@j-QJD$cU<_0;gF2S=5cr`%-4!k zEGtr`ntTT%vE^Ag2IlXofy&^;;+XwxrbE08X<%{D4$b_1Q2^q5ocsj%sxy zR)02q#!-J0r^l5ZZH|7qUOiX86%`F)UBj4SzV|(Q9jX>@xGle>uKliVyqk_iOg%cf z@Wh#KxT(MHZo~6251O})>>@wJs8c#LLMf3SaNb^PZUHE1?#VT=a8TM?tT^fSn2}$L+P3(@ zygkda<>c~O=de+KrujtuRg`wZs^aV7fz{RPdZ|?3*E+lTMC2D2E$>HvT6t&nGEbn) zy&>0J%8?*)dq4On{f@Iltd0+Bv$a)j{rb6jRX(kxY*u|p3wx>+|Ll8qW;Ss9UHeMX zAy32<$nJ5A#*w#8Jb;&ZwMA{|GR(0|_)eD^)4r4w4|*!9%r?pv8*MTK^8kjkSH8G!4FB^!c~DN+_W)rujkC5yflcYljfD`v!(EQp*DGUsXhjH9}Kw*Eo* zM?b5i-|YTvy4ut}qQ({YTVqD8=QOHT`;jO;74tNh;;Dp)RF zzTVaC*v&MZ@w?jX%zm@l7cCl3x8oSGaONDVaQk*uI%5^Ntt)e4pc6Z7rG?CrdrFc|ZYC3)~z1G754WVfBvGKSQ6XbNIdC+4%X~*Zl zO(HKc*JnqDm3A=kP3K5gG9O@PK{8L6Yby+<$LN?*Q_kE9glm=>H5J;L)kbdlPknxE z&27e-JG3?H5{I-k^~RbzV~;kHm9b{MnV-pDj+t3TNd6!1LpwC=H*(j1&hM{k8Y)N* z>OF}=#%+c9uiWL~=1;t3++64}<`$Y8UbJ9{e`m8W)Gy3C{6Lqm2gq>PF96NyC8t43 z2x{Kpg`LEpB<<0d#CudE1GGogo}Bm~$cuY(Vs0M?Tk6*vb8;{Klx!kxc&+=v%}k;) zrN$oJ!jtysR(>yJQNywZlsUipFZb`;sDeP8&ggSx8slZ@0RSy z=k)I#=!X1P`*(3B|NmtFZskx>VEBK#e}6VHelM&~k$^<&!_RoZHT}I)kLl$L*xHay z8MfJ%FVI&lvR<|2ix9L!RKZzb_xG@zy^jUD>b)ZK^TJP%&pIUSp)H4>UaBo$YLpfm zw-(!MtWjES+*)pwRv5Qd=ryag>eYJbZQAnNjM6)dTkkMR?=)_`bIGc8_1Eg9^|0aE z@_O@oSFzo3L%n6@Qd_=GuUS`#h|Ggm3jlc>S>3QsyP@7FUC$W5b$#oDjZ`7lB)|Hz z&HS1*Dki8oY@-`gQ(3fo?WomtnlitjZ&JfeS%x^+B!iCC`*-5pV*en&tq;msL99uB zTQ@z-FHIKH)q~>2A2_JS^*_i+jsHZRpDE_&)MN1-GV}@JcOnQ}CWzmOAO<}_{MteI zRhBWV%dc-&n+)jKKge%)lMHM5yO1HUNiz1rKZ!ZQ1u&R-vR&mCdpT;KaDg>F!-lW2 z9&LU=#b0!O{6!P-t;ekRrRyrp<8$e$pzx|%X*~iGe?bqL$tww5RS%D0^&owxm98_p zacyS=5m(5A$da#L-nvWINu4qwYRa!~Y21l-xb($TNNj$-g}_s-8N(MGwb7+fH%R*_L?*h7ukOL z;z!b_zBt|C{-PEfOWZZqS2vQli#?Z-ap8Mf$m)-#2D~cfn%_StLTO&KuJvOd4tqZw z^t|+XYsl|9rkA>}^X9C%va@(%<)oTd@smL;_UUg2N{jIPLw5yAv6|Az113`gYv%hl z)=S`;Mc@1?R-R;qgi)sz4IjZNW2amg)S^@Mn&N7l{QlSs&=`ooUso1rSh3-kw~Mt! z;cz9%J1KV?h*4c3i*xU8({DxfofB2rKa@0Lo^djFeDo+1imrv7`#&YDe?#tWxlLlJ z#@q9cY{Ne`ezD!?D!#uF|Lye)vwCgV2o5KrWAxgX-D@6t&7Zi--&Z$E-_e`pl7s5p z>}b-tlcHsJuGPP+AOA0wm#LVA4mLgn9@s&{MyF`%9(X`h@EUlN`(y^0_Ml4x5A3w# z%RTVG9y`9=&x>EFm$*SV1~t|JcFktg{i%ew*9^nW| zmQ6P)oc4S;IQG)(Aat4gy16-eY2n>dTUX)RFEzEbrs(vE0{`)bqR6G`(!%c2qDBf& zv*?**;?sgx_F#5hrmu9jZoX6suhG{P#%7~u`V1t+;Vdsah_&AG`Ob59ORV(+2(Qvp ziLbGE*!&40On_b(o5KfX@71}!(6_6~sox^suTZR}lt8e)j-tPjBV})#ArwomD2!by zA1dZcsq5!E7Xx#q)w1BN%9$vTbmIl8V@v0{^9mGr2&N?5<@t7frgfuF0JX{^oLV~5 z62U5X-F)YQtb!-ugR1Yv%Ku|3{8x5;$IJgc^_Be^>|N}t@%nZ_ik{ks7SFq$cymr8 zsk`tDxC?q#*y{LDX;H23+0HY)S9U6g(?1nrFz;PV++g#P`ckCtsdru7`n`+K$w5$z zTw1rP&WAYav!P-cjJu`lzY*i;KdU}^>HOAwsSxZL3qmY%be#hgV!5NfT`%FUP#MsO zZ=R_t{I2KPeL4GEdn4{obf!K=&GG8t+jT@gKJVw!HZ_9Dq;c(g+SvudDJtlyU`NtZL(x}qxF$$coXMZf1j!gSR-zQFd;?SVt#yBmh0;XQ}a{(PCPkP9= zi_MfH^IpV-qyYXN8{t6h+XdKkqF$qrIToBIE+!>ev6DpS#$>EoF>_);{G-}2bH-za zMqBHqwT9o9_Soo=vDO+yIk7NyHEc1%d`+Re%#nMIj6u22kL5Vm%y)k99I|1L1~vY( z+p9R|zpKGc&}4@Og`1V*rT@DPcG*SSD?F{W?)>)hS!8jQYOkxE)_soLYlKqLYCO|$ ztL8groYQdU;Mf08+RGw|4TU<*a=FvG&y>5uuZH=~PoL9F+4#kL%67OM$w&Pdw)Jc4 zF%;o$dPcZOal)<1gN}6dQ?I&i$#cEJfd}RXZ}B(1=f0;dUDz@D6OJ}BpKS-Q8Yd!$ z@eJ{kM$*3Yp0>Iu^@;H~l0{+aL*wspZ3&g92|A5@~fy<=xg#_ulk<4+;vO-?XER>Ya~fNvzr&|+ugh`A>9a_-Fo?a*nD@1Rz6=Z&$l1*WuE;E32)%4u&)~KnGT_Rj8;(c z+xRg2s(3A-TfC7xnKWX}KwphLW#q15c{$wlRIAg?zm_de;S!c|%9=hAc-H2o)IIR5 z%}uF$;8Ee2;Iz~|@Tko>sk@quGaExTK++yqdmHi!L??W)8-xf>3HW{S5_y7&dO$%1 z%z@SL?$;6;SiQ*i@HO(V##?_Q#uF`XeC`SJQ)U?Vy!7q?thJ3wpXf8Iua9%`K{c5c z@!f!ZnM$j|Rc3xS6C?*VYglP`^nKWNEXiYg|MWg*Jg*$H+B2CSmh#|r)_c;mA%n$piQ5NH%S1*NkAaZQ^0$Ff-8x9PAbB;bdFXAnw7<@^jPppdk zR!*`a-oVp#=c{|*X}j~)J@AnIP2B?z+27QiVge7@xS{TWhXgl#n`Y9HntE3Fkq39| z)*GeM_3|QFvL&MEEa=4xmIVC`E>;(!P!sV87xJRn@gC=^b@_ohzwhB5lC74cnIDPKGU9`)r$KXDVUXma1V}%CIe+&jyjZSSy{cm*(4#`2dByOMW2e_ib9j z_G+dn(x-pw_2-b|RQsCFJu{Z={zTph?A1#^hF5Bizvo(=rx$zUi?HxJ;u>}xl+!Jd zK?>$oRMo<)nO~A~eU7k1BZOuvka5N$JTyG-;Jp5M`2J!KeN+pLAd-!|O>ju|^BD#B z(Qz-RUkrldsT*U6y1Tts4FIppdLqup`8xhKd*Y=u^+|9Z3x5|&CnF-0%j}64(Fsuk zQgvgA=nP}U{sO&rsqg9FjOh+9SQ8ws)sH0Z8t$tbZVe8{$`&N*4%jo?wyZ6iH>t)jW-znD0@>6YXBVEWZkT%!8rg69X zm-CU?k#Tv1i=K}_pj$)UJGZfOr%j1%I>2Z+_~3K=I?R8^rJZb>1-%)IF&~6V=(7D^;e3yV5D$r zxm+jKx?LwEtIAKkr?S#YIe8h5uPNvY4kxFZ910n^jjLz+%@=Tbo1ES`!|%^*77ZD} zp+wZfwu2{Yv&fsc%iCASt^#a)5be|9GIDC#FS#1GBGu<_cnn>4bmbd39MY#4k3J5l zx?lT_AOQ!B+Jn;F+9tsQZIkr$!3ULblM5YOWDer(LB{;_pJ{))<*jZo7hdv5xBt;U z)c(KnWb*n$?SDM*2ikwt%>Vb>pS=}G|HPJb-p8y*1>C#bm9M z+2*jZ$Kv-&-0Hh59xw78PD_imUHrQ$uVSJ+iC?~2GPB@O@3&37tN3@=@e?Mmy~kf2 zIDH`ZKIRFx(QB-}Q$A8tBVpX4o+t2~opPoJkq%hthT3q-$k}~8V(_x%Y?Y_#(|B=s zkt;BY?<5PG#~ROOIxBZ6c~bjG`%C_r_E+l#tpq6iT4kPAk`EvU9}rI5I-?#>%tNOC zwr8aW1=xRYlVc5P3sH>4T zT)(ggB`m5Ilj9N*jGmBlQijXGL1FRr2j#m6813fnShF94OSTmIrvMv!m2Opt9H#@H zZ0RF8!@ZmFnm}9JUy+*C3*Gr?#9;|PbaQ8grB=>bk-xU+$38=>u~~e z=u98-n&!>ZB}?z%VVT{F^Z1G}w^|Oi0``X=$_KH}in+>r3w5@c&R#FO7ZrJDUKN&s z4X4^FbGjY7DU;1{V5nl3lbqi6T1EVwe2aQoZ*sQIx$b!6lIw2D@iiZ%q9QhhCbg~7 zYB_mdctSt2#NY6djJ}s|S6NOB_M*%gE8#XZR+6Jn!Sk9c;NS2DW$9B~tG)Q8@dV1< z8}<#pN48mI_$eyxFat@Z@0#?d!Q^u?3mT_?KyOYaUP8pF; zOH-|vruv{XcY@|VtD5@+&5aoSFHk3;xl?Nh_w722)!W)i^aJhC^2rc3EViM$T8GO& z559}nzvotc3Mw`0^AR8|XETsag7GWPmjlbcwaQwe)@1B$tNRkQCSx_G?n|DPc~HK| zeaW*j6UsNaGlM(J0}tA4lez~UwAm(gXC(36=9|<#@Sr3lJ9QVb>T$G%dayap^((HY z4|)C7T<46}-HpfKN%n*D<65j^P$)fkv(eG>m+5ph1PGd4ZgiademZRg^J|TcV}xrh zTlqDD1;~MP4DqSEW#D1dDHu~mu*m4>RbLkv9V05)VshiN5iB<9P8#YoO5f8VDdO&1 zIqKn`(VQy<#y95ojtRs*1^pH|MHiu%NU}Qn|l5 zDNC*W2}#khTYjyqjE**WX_27*gNf&RUohG`RfvT5KbJU4$dvx)jZ%N_A$d)f78o5d ziBalF##PkP{Ql?r4PGO!q{>7l6ldfsMoV`7$+{aQ zyyU(bBgs7TijmOcs@r;v7O5wPKM1gRtY#1amM@I3gdXZkkYEc*?iF{AhP zH4C4=)jXMIeaQN*p(+X6}9d_-LX_YgV0`Y>~dw_9)!weIao` z+Kc;h{reM#x$o=U*ZZQ;A(0ZDy$9r3GPC=|>IDU<)khQQtR6xP=W;zRNRc=!Puo;l zg&vm~3M^x0+*_PD;5yX5PmZJ$hv?R=(*@5c`FS1%uSmRg`Y}nnb^4IJAL2n_*4F8Q zSA%cU1ct%v+$;YEpXOB!BWwPu$zBiKc0uE0_W)>ell(=iCuP&Lm z%j^p#4+h_HVn03^+LpX)2gwp|^(HJ9V}rijtyio|US=W4UHak1=sS+&HNMTArOCW^ z;@)>+!FSZR`d=DYL5lRIZZLY+78t$erSGFwuY4M5k+Hx*78d9apBe)h&NY7Q#xCxK zLM0<}KXxTmHpeDJGOz<ZfFpo)GI7KR_aycSIEQ-s zq_`5F($|!cKKN5wRLW*p{niL>>wTm5Q15=D4k)+up6pl8EaycNcN{isn`Fb5erGJ5 zfsEcA|3poJtRw>&jUiQ*2A~0hoit$LX#eYsAptA_r~(Fd>(LkYZSE?iivsKFBRLCC z>&5ic@V`5y8i#scvTD%$V;O6`G*@p^aKLXQ(xGEUudk(2Q6Vw4h0153%eV2DXvPW3 zgiqo@(cx>^C@6$^l+!ota4tMy$ukE|MVIPR4R;P=bViB)l|d9l*4 zPgdg;(sUS+xzLcvo_>X2ZW_W{3!6pxhO5TF!TsBb&RS!+M_=zzyIV&1D1$H2iNBOh zpJX(|%hGIZ`8KYmO?))v#*D$W)qe*>GwWvy&B+dta^! zqC9BSHE zyvo;dSg?xr1zC^EcuJD|5ca9!hXk9BcfIa*y)^Hve&)ZcHNdP@k$=l75dwagZE#2q zJb{udZ@!lo1# zF~OdY>gV1kNF6?znr-x@zGsD%L@qa8k=l+(Yg%J2_r6Pvx*=lqriScM7+GQTF{HyU z16+;USVoCPjuA{INkp;B3XEm>{Vy2HJg_=2PswFPa8POh=iWEnN_sMtPljEYVhgg0 zWm~EDKdcU#4i9mZN~NwK@-N6lkzlZT8P+F>84Y88K&;`^V6 z5DEvT0I+CaWV(@{hIJMIolurntk#$+p(CjrBVmPf}|S6|%#;x=kq>|8M$>$9mok6bMEu=@P+#Xc)p9?{%mf7ryb)<6}lkM?hCGuUYU z=3ktS@-w}+U)i~)*p@m`^rMhH)apmz(dV9dpM#m&YK#eE!%^S=$eC!q7CwmyKC}AN z8EjwhWz1YWf@OhFfnJ%17Ucmi!kb~SB)LiEU@bC~x`aH$#X_*a6*RSALEzwqY4Z;1 z5v(T{lwy%MG8%V_8xMyves!PX5iritFpnK>*^Ui)D=O1r1oW87(Hy~#ED{M`ztj6r zabW+3x6uWO^y)9*xFe4RP2)F}TtQxm_>}9zpGYv73DtrF?875j7(~}JGhYYuU3J*M z%M0{w@XYJQ;MOuA3MMfna-!i9J?-*7U3z`wtnaBFv^Kj2=fph4qI-e{=W}WN*0o#% zeXliVOACa@SWoi+oc4b5T-!cARnMd8=8D(?jb-)abtC3e!)M?JIeaN#!4AU+_+dXL8#e8w8SOf~w_JwSdws7q?$mqB3j+rP=6 z8SqK#FmA_W+zO*7E(tGD=R+fy*AlysXltu+i{-l!t!^&c@pa%e+P2o{#PzhahQg1N zfwK*FS^SbK^fEQnLT087&E-mVW|99)|COg7_+^^1a17vRn6F2{#knrjGQvZ&#WYqu z=HI_7jK0?u&8Oy(K@l*J#xIE2`A!VahX#BtE2Vw#Dq%L# z{TFPX8^H+1W08_sn9Gr3buHbF(;mH>;6e;BQ%Cc2v=8QOGqkDJtRSH20K$#=4zR6$ zcCESUEg^y=9yS-Mk6v6Dk(kjqiZepJB41yZhoeV9JiHP5Hi-dDT|}nTxIdagAp@T_ zCy9ca4i|3Fu<6&cVTN8eFt2AhFIrteik*`IC?$Ou!ox>94JL%L(yTnI8b`ok5fB`W z=LuSiue1vDlh!xdaWytX#b{t%4u-s@E0h?_oS~L`IPVw9=y^Wxl80qoTpxnJ_tw z!k%b>hH1HAp;R9$OEldvAxuevT%k@al%&El@x3z-yQE154wNA;HBC+|>vluLu#Fw` zmuq#Va)?+?257@Cde!S!W9c&R4Z`bk|F}$O6-t2xzNhvxnqpH*2d^aZLVQ_k;rUv4 zCnIk$H5G%AH$H<)qypEw5R7VMR;5Uk_ zd{4ckBT05dfhlv!DlEz6b-g(!R+DqE)Ro43Ug~x+_CyiU`PesZ_)8fab{PVDp6k{4 z()*G%3ibI&607XAAC#Ks{j863oA#;XkW5TYcJJjmMo znrbbyRf`Z3h;d-d9U{hmB6%$O-+ZlH}$_%>Z3V;3Pp%t_V_;lxf0?*Wm6EM$+mf{&}1oq>H! z!TWHbg~3}|%Opa0x1gC@2^v+Xp{;lWN{6@E1Ndme%r*umSpBj!evf9xFH2Z~xnWitZoLq0Vf-eYnA`w7 z#_vGnDREgAGVOtS*7kMV9XuceL<{baIa2Ibvc}fJ1L8CS5@ouAG^`2Hoi?vqk3PY3 znS6Re4qlbA--5rBr*}?E21+o@B!myk=l>y2b?IBW$O?sUW zYj3A#Q9<%dVN6>cAc%g^`E&0?c!`2g9VocrFI{ zZFGTSs%yj1pcdSi329|O1h;}Bv?B$-?22q*$ZC-tfng@I=+1e=IB(g3A$2CR&`v?< zv*Nf-#!fSyi$H0xQRp2wpn+ww#Mmvx{x3noNKdI2Nn&CPeiJ*>u3$_HJ`S@xeIA;z zTY1)lZ88!!>!FQo>uI4q%rS6eKK%t0>!L%a{ss7S@m`A?#fCovS5G(OXXt$$QoM7o za0<3AgTeXD%E+ry8#RDV!@l0wg8RPE9>}!Dqj1GTl3NCy|ItF{8-F3_?AxSlNCuPT z`ipk{v3ytX<&}|9icftF^tww>?nTaZhGVPKqM{ofK}A=^c{3Fv0=!AGSbuQ>|0Mz< zt|X{P$MO#rJJTxmFH=_&`-7S6h{RYRl0UZE&JH4=u>5)qECi1c6~YBlZr7tnAqm$l zb9JwTR5HJXPXZl1x5pX*S74(aRz&}~v^Sh+$Yd|siUd^??kkdnw(XWJv%A!8#|||| zbUX1HDa^!h~%j-feZt^Zhj3yP;FYYQ%xJkwD4A8dc#{}N_hAe zS?Ir~9=-nf$@9^_I$ys5+P_;57r;A&dWH7iFSI{&jGkv%sfD15k7w|4rwh!NRngAO z`fH~i+(rWl@egf-^;M|8M-R0rRNtJzb2YZ`H@XQXm(2HQp&hdH-0lhj>~_fP3`J(k zWI#ld!Ui(rlKQB^-#IpZfxJ#dYC+6X*@(G>Orf)FCdbMw*kHbtZ(f$khg~72bvr#c zt(t|g#-iRjQ*RPMfsm@`=@vd*v+d>f}m9PjaX2PkMaNZ1Ii(er9RTseLgDWE` za)Vn3S($N25_x&>G4ir<{hp#Tz4}2>31E6$;mJ`P_(}4c2lkKKAGQX_IrvZRzf!ql zu%6r@76Xxksp}={h857?tZeZK2V*bU7BD8UH=zbx)=(LTG%K&;3XNAte%LOC_%q8o zwQ#cG;|u^7R@TF`fn=-WlcHfJlWT1y{wIG8A_oU#8Veq?p&DQ|(SrxD*YEnd9_=i4 zy=M8+*S-?TV;_rc|L_Pz@Je>OQG~z^tX$GpC7Y`4itS+0>=?^UaA-aEBmznh zo)6r4GCe{b3SkpVtJ!&ZT-@kH+vo-Jf!EHYW#PBSyvKgpY2L+C(}8^H2gX|cRW;_9 z2iTIJx6Qw?9*&x~@Ib{=tfK@Af=4-ni|jG#V%VkcW7cDZiod~{S0Y9Du6+V%IVTnu zH0dx*BJ6Y$F^wAil=(S2zv;kS)#uRMV?tts1IQIrC=DsIM17~@dUTOj&euXra78jJ zUJf-ff-xP1nwOe7b1;cX132)_0qcermUy^vaz}-?0Iv~jG@pF~Oo==ZIM}cO8fEm# zGAz8aqO!C~i#%c8{rbd$5?>T>h)3}ZXx?fC2Oj!s&ALGPc{+F(%wTQJSC!Uf=F-n9<>pvR}iW9)hJ`?Wo^Ze*gbtS zaB9OoJbO^vXq!E6-%|-`92mkpRd6V_?ETa-1P|qb+i+|d)PeoM`SLp{Y$%pTz$@#%X6Lg>s93yhnF8Jh(?oI|{&hU;Mmgqx;zN=SgStmtTaETJ|(Nn)hYzmG~b$$ZyuPSCqGSj{oP!@ zNys2Fiv^k=AwPwhWayS|N37-M{27p%-Q5wivaoz z$vcGfOpWQ{(e>)QGff8y1O0dZT(Uu|C=pxsyn=IEn?4=aYo)9PG!2(>lz=9XMfRqmD0 z9J?}atIC|EGT5~1Y+#H~=aAVt$F~c_K8!+O91J}DZJi?aIAi}QnByvt38q~L|+}T>pF>WlBM&nD1j4rtw zpqSn03a@^tOexD~-xplhOZ(dG>f0pZz^#{;H1|bjNfRrsIt^FN%loN&8sAITM;Z^O zuYJk43n4kCqEo(Ib<@-o^{VS7{gtlwJn_x58p(w4G+?7FBzcelv*M%(QYR~4w(xwr z)}y=kzD(>R`a50klgn($h4Ty*FGo&Yu3pyE>0+xtFpCpbY5F;=Wkv(kYhw|wgHsg+ z{QV8^hGhEt8sOiFVDoeTkG8|XVr259El|%u-t{!0ciPLa3AMGasMOK9hAtSJx=IDMJLRq4v@!EPvO#(zDD(Y_8Ai z=D-2=hLOa17wlHuh;%=Dc9*!ieNUBmMdwJgXkT=NhFtqxtMc@h*bUU`ShC<^xZ$t0 zI*+WkTI;;cm)E;J{?wG#S5CiFkbnH$$dn&90)6pu^tiA&=#>{`q>BvZvh6thmHEWz}C61{B80K&G(ln!6a~QhAf&DC;BC zTtXVZb0Cnhtp*uLHu|$PWj_(UMiW4dWoh^ICWY9xwG9f8me= zULsTPxw0!@)%y>Case;|Wx-drumqHT3zXaCJkp~kslULWb)=x-qil9UGDG?Cycxcy z{H}`pYL_@Kf?t2liS=$AszeGlymBkTzcQ=Lb?E6HnMU66`hRI(Uph5`!-`+`$ch%Z zlO^|fe4C=wvRaFH?ETp?cdL8KGh!qZ*{3@$zs&!*2PZzEF%;oe%bQ!?Mt@u(&pDAC z{rKa~`rK#AdF3oipGr4?|D&AmS65Xw*}{4bc>0<{*?D}F~amHKriL{IDH%Wb0zF>&ELqR$o+ z){CpmCosp_(utON{rQF!d@VvllcP87*vvXl6iR!GGo478)qt02_*0+HCn!0@63FW z8fSmGH1k10XeTKxJm^TLN)>3!6l_A9(EI))V4Il%&CXB(EqZCb<~B@HlCg7=POt8= zHJt9C**`DMzN)a*iM@i;y;pR`f}^Q&3T{PfeKhrHX*?$D(bfuQEPR&fI#nb=0=OGU=_)V$MPtKahp+k9G)n8L1n&tdbN;U2yKKnJ!!E$t-48T2HlF#7 z>*rnXPSrWZy&BRUOqx@EEc%_{v$5bW70Rv0%WG0hJzOloe>{W9wcsyQ^GkN6grigsa->YNYjeYVEpqe-0ZxN#PW;IW7)5>DOxy z=YS45t@~P!V=C-m@O?1ocU&Wmrn9-ehvvXwqam@8F0jCwqwMI+>jFz?t07Tqn9P0N z;3ijqY2EA*vH{H)Ji+iGvaIb$u#b7-8R_zQ@UL^M46>AOZL1y28OhB#iYSE5Sa3wt z7*vapcj(tLbxz{Z8FeaQC#gI_Fk->qiT(p9_7>Z#%zt9t?g#^q=#yw^Epg) zRge01`4srfoot~FTl7MaKdX+yl82Cbk!1+2mmNptNu{VzEKL2vptZ375=Tg=)Z}k0 zP*hEBQ3-^*@8n<{eF)#K6X<(nTK_2lO?t%DLva)v9kmA4pbK4;;fZmD0g*6Gh~bby zC*0!XnvT$wKx`KxLNKN(=@m*MocxdcSaml?p>|2^=ZzOc6xudqPL$?Ckc4(&yMuMn zt}va=^1+%Qe7kM}3qna3ZHZo3a3nh$)n*sX1~jm;QEh@AO;$F)%g%;2y%;q5D+&;r znJ#8Hmt@VH|BGgTSZx<##XO0|(u^tyEOX`ci_HgiUVN&Ie8rw7Q+Ej2=ETf{*GY&z zxRirQlGI-jOe?+L8vdO%rK#5uYd*|OX4ahbL6z=(NjD;spX4{5$W9$ha-_5mZ)VhXv&RkEP_4>ccRq?LZ|1j4T z?|J?Aay>HU^`GEMa^=x9gWNM`Y@&=+in6ydh*w*`0L3uyAuySZpb*WfX8+t)yaeKz38vGRcYDd02 zuCI0x{a-CKgy8#OD>dVYvpFR}KLfG%^#_m*MtWj6m8(UDMPwLW&y(yK_rqn>`Imeh zR`Pw&18uDWZV30t9z3g^VJ3%G_f*)#kLWIA#R5-;D|9aqdE^_O8%rzSJn)2r9i({N zqr2+AAZ3X9UkKX{vL#$;te8@yhY&t4Mv{3k0@B&rul|@3O7Hm zquf2*ho!8a9$^Qu+tqY%0UYRp)CbRp=Bg}cAX=waEJ?fTFQpP|9p*JyRS4Ygu@=@y z>zywvjVp0Z49`_Hn6yP?^+&4u5BJ9~{c#=EJJnX#kVqy(<-}C=%B-5TPX4cTR*O~3 zBFB4aeOOjbA%9JEZ3T=w1T0+{=#3Ty&PFd2ZOo8|K^hu1H{jYewOI81SzO@=wZi=I z&k6aciYL3;(&oce=14E0tnwyO=DY$kb@2S^apC$GIVk6^v9~K_vbd*^ zd@e7~9uDNB(Z4%zv80XB`Hl!e1WX39XAoT>GO%am={jlx@K&aNGGy&DsRn=ZtBYl7 z)!9lp7XbX{r1aq0<3c@2-SG!7Maalw=zY>H7t)a_x~v8J$#c}-0wF;w*wzKEWbJn~);r`lt{3IP zCF|=OmLPx15NE46@T$1(Hv4}%hSX;-u3>3Ep{%`2)_ZVI*JB>wi)`i&p7KJZ5=>$X z51r1*!L=PDM~gOczPHEHm)+t@{KZ2=5k+-Cu#Rechh~Vaak=3Y6DUcH8ID4)lS{j^($1r(Mfuo*qfc2c$vdqo6;T@L%Scaeqw0iS&4T1qLdz5k3IZO-Q)VD2^C*2I zX_l#yLoPY2mEpHXK;ZJP9;kM_`ShI=ZC zXURgRtH)R|l^xFQ$~RIEDt(dZHNmBan$bqp9QoD=;zAD<-BAdB|W4xz>+2uM!FLQ^@X zcNXs5q_A)fJ3B$aLL>;6vG!cuWIF2aD6d~%?rW)#a0EPujTIinY@mwKC0+0)rAmv> zMwdbR%|6(Fa1HJx}v29JpI>@T2>L80Y zDe870D^rD_zy2{DR4~!(K^y57C%<6#&XnJ1n{0Gp#eM{0hK%CIq`hIC}yF0Dm6|F z7`b;s`!iZibC5^nD7xI8EV(1uW7O?LprmedY5^Kqo&B#HxxI|1{&&9+#YIKnEBnAxcDq>1subCL|3ACaCJX)28X?JI*|EdXAoE0sloOix0i zcoU*y&5GyFNnaz6!A;U$y_pt++Dq==eR?d7P|B>N>Py;Je*q>qE7>A#XOZFt(knKhT_99*U!yuN zgz8RV#|E(uyyozO26{*kS`+37{dU#LM*_;jY^x#H46w{+SDoQG=#Npjq)z~KDi$p<~ zN5X0dOF;z)J;2gu2Qo{97E8?N4yIb98H+0_DkbVX0-K_pRDcf%L7Ipn>+OnbrzA9G z!sNRhc{l{IG!-71f*>h)D$9HzY@V?9etnRFy92sX89F2ZT zQ>rbCe>#QZ);f>!Vsz-=WKpz#k6PhfXdRfnVZnNyrLa1Wt7Vxa%2;yh_Jw~2C8K%W zr5>8k!4B0Yl?_AeBXTfTe-&#^-m5(EQdtgrD=WlsA}?C0KQ&YC4IX(!eie^({?<@Y zYjAFBxFEJ_Zfj_MtkfH;nI2np)zWn2!z@Z{ONo&=c(j_exJ?utBGVR!3nNnwmVVsH zDNpIgIdJ)S^P&0~(D-7%?rvF)VZqb~CGGm94i$(4V0X(`BzP(hjXdC;r!`m@6Tf<% z?$V{h9MW@u)v^YHCucWmSN?wYlCq_~7Lor`JE64xtRXBOyld%t3|enT4mG}ebzk8;mRITOCE$0FG_*#diID>0>87?+AOH5X2oB{ ziHX*dPZXT8?3KIHV$ZmKnQTa~%?P$7qFXRvvL(xa$=1YOTl=j(^(c(g-kmd+C>*^7E z56R1A{$oj;9~g`Ml`7*=T|0?eYmS6(xTi><`!Hj%iT z(8QDS+t)Fs5=#WVMV_@_Ptk-TNn ze#J_wZWCz>R9gA%>+q;&i6GFc*G%FS1k!BkHj#L-N+7>|9U>x`cv2!5Ir?5DFlhr_ zXr)!RiL{?rY2~-C!=aufg5ECATDVc?P%$vdfxwHY9KE}`vIbhY4X#($nrRaAhhKp$ zI;OSHCAPAj6laDHoQjm_!N$PoH?Kl(U^B;chuPuVR#QmEnu>vrPFM?^%A z^Q}sapzRfAv3y)6Yy!m;XN=@3)YFUSV_w7vw*`iyA=yiA)*>7Gf*X_9^dDmLr7mU! zV~M>>frJ)Pw~1QbMrh(m`R(iQt7nOz@0DkA<|rlC*`-VV^np5M9^F085GiA1lmBf(f6=8?+u*tJuC-=s(oVZGD@@b z+0{E%R>IUyZ`r`dWbVMztj`Omw16^R+eLf>6m^>bKAcPeuT&ZW92nBQ)`({PuOY)w4v<_sX*Z_@=6wN+6z*0r+wO7^$Xh@mgn@R6L(}#THS<->Gk79=o+o(IH0!p5~;Z{sW@^ z5cwbi$KBuRB&GbNP_x{ceBeG`xlZsQYk!Qy|E-0Q!%i9C{8DX+4Ht*=A_@NvCOh?Bl-5eiTBS7egT6jNnL$_MuGvm&e0TfQxMv@zCQ) zmNZwvgMJ(O2phhfy^3c22Cu+YjeKC6z8qokCsa9YsPj=2(X*_Q-12uY0>Kzeo3B|M zfi&KaNI1%gF~PCS!Rqu($?hLIQ#qeFWCZgjaR?nUhzvE>647&~=go<`Huu#vGB-o2 zsjXrUEVt=uY>0Y2YV`5o4q4VbGLp{NO0NG>rr105=;MLt6TXLM0U;bJT*WpjzMX~8 z5p(KSY1nAYL2KGy#L&H(HB=Py!8>VGlpcfI(FKWzb_hRm8N<4aEE|5PENx}-SF*j~ zZEB!n4cko7s?D92X?OIQq!?BskqP%Nj;^I&OS8n__Vkp`?(c%-f5Pi6$kk!v?4EK z-{OHGG0ZhiA;yf{Hmpsv1QN0!sxr5rt+bn?#CgtmC6I$gfoa{w+xNhfC|$Y-sU>Sq!&1b^%2qi+on%}Er{m@2yvquh`LxFYmm<| zsHHSt^OqGaiZQL?@=2q?s$6tCZ0hz2mNU3Bao5hiI=JTT=+P=pdC~Y#SnFh9K7q1& zEoi-`3IzQ&hW1)A+mNGHI~JlVx^3QmrE|5F zQP%3dg0*L9EI4GvPlrbXkJ<)F)jja2=rLh%RNVv0imBKy-AQObSut&>v#gn>!a)0s zkSu3t&$z92C+lijXxvu&jUS!vKOQ}at@hx_MiqAeTkXLCaj+2_aLfFLSB{X`6n&Y8 zbnxZn>CjeA%cN`3wVLS+??&o(hp%N0A!5EDG`8@0^RJ$?@DX8x;O196HsVBg(zZJ% zP$syYR_&|XNDDLejgkxY#}3l$$m$QZY^Z}e?a)D2#%_65e>gZLravlc8H=j9ox@gt zJjHT_c_X_&LVJXw-0_cAe{=^Qr}olkmNy1HycOF3*Fbx_Ob#q|KO>t2zFnLY?s8mv z_@$J7_@#kqy-e;>v=_=8H?WH22+GYM}?cNo9VYh1vgl zhu-n0b|}!MgFA{{eJBh!;fZVGr=)K=)a+zZQnlFl8(35(JG3q7>n*;PpYT53*Vq3` zrmw95X379#<%%V|3wG$a3(_5Qr9uLT;YOJQDziG&$#Q}Ytz!Lb_1A*8tPbsGKx=hd zEqjt`d|3mqZA)f+1q6=ZZZ)`I!DM!xV)uV$qN%Mmk8Kh*$k+U@(r=mZ7y7S^Ki{UO z$;%ppgHK4df5b4X|5JMHvkb#OZgj`~lx!L1GP*bVTKZIfkB@G~`y<&szWV=UdfXn} zYS}Ysr$Ia=S{>Ug+YA>N-PWc+HboF~Z&?%}yzvi??hg;iUdeVLxCniL4=`Y2k14br ziyM`!CKXMNerM7U#{hv~#(&TF{+P;Y`#2AQ8SI9Q7DmZxh@o#It`Kr%j#`Md5cGgD zM=ge^g`hL%T*$DrZgG_V8e#Q|Sa%ANRikRiCX-_+Q!jr}VpxPJcwUJN}ex z1)abT2n*Wz7G>Elcr}T#L{vyK^S$NHF1Y>AgCL;tFCQASR4oVpDT8kG?Y66@`=8+Y zp?y8g{cG1u_eZ(DWnVGwfA-<&{)JqN?d#Lr8@O7z9=5M`?%((b>9`)UuQ>PTxe{Fc z_H~kbMix@;tj8W0s)y5_;ICwJIo~KPfTJ?Pdk}OpN{hsOd0ny=8`D~giC!0V($n^;Rw~uTD1#DI7DdTN%`&T7*Nj=L2r@g^3+n% z(UnXRdl&Oda&?NWx3d{u({_I32C z5+s7&BG2WiPfY-rZzWf^2>|^nx%~EZbgJYML2r@g;y{bEGviy`f$-$-Z^9z02z8r) zuuv5tzkMA?RS^iG`? zxxgw#-6nusq)L(BzK$MMibT*`{V&AQkz;2dyTO zn(XFTs96^MQ8n>P<`Xpu9B;T(Kdh;);|L=zUD&i&>g0%A+T9`#%&3l*Ly>7}>0-s` zQy(zc|2DSy``_ua20ga>dCy0PqO11^*Yo`dDkWY^Ue&W_re)~Zr~m6TPR8G5x0;M~ zMOQ2JxDZ*;XE(PBU6E!v{Cp8nn2wEvVafMxq&Yga@|=8ENp^H>=NY5$9heb#B&(D$ieQ@T4di|Yf;d5v+LVE zDP$j6Cbbr+3HkE$d19-|15HKb@1=chFitf|LrLr*sha|}!v0nUhyt9}ax>`#LSVDn zaHiHiB9-4OXCi}5G?W&%0!AgceN2AU;YlhY+1oj`sL}+Y^Gi!ukX)#=E9}hMRpxuM zGSAocs$#9PnYi^uV(D>JLc2h2Q_H%!E3en$f(lyuFuztO3-UJYmF{+Tw+L0`9bXii zKybZXf}8e@b-kIhEbdh%HsdXV0~SN0du036b%;r1k5Pfz!t^A^Kt@G=c5s0e>@g~e zvV#{}!Cs@HI6Ju93ck~5KM9hN_38?INgTzi#4nM%I*A0$DaP#T1^urlpOhH=XN}dX zNkjSTC8^QY2>5#WSF$o~lXGw@-*x2J*2b%RR>|77jeo3{kFq5&@lm*Tm8}BD*dly8 zFeq5`*fI4cvHMQ}+tSIqRm_0(1b`+_dhBI>vtb;zzw5DH^~Go#wPRT6 zJ&Y=WM2{uy?{eNe0crmD_r>y*X}&E5D%N=|xJ3>svLP+9((f_a%B?3LRT8POr7DMs zl9*|2NQ%`LqwNkmh6U-m@$W0_@9VR^=WgTWdZS`MDr13ZtTr=kn(5!3EYMaD`&#}7 zU9Br7+QP}eVP9PNE9&}k=biYZ5uBWNJR2XII`1?p9#<_^zSXGMk=51}yRG1DMnz|K z@G&cRyHU|=1z%49i1nv|dXEgao6g(X>JV%-TI4sew-|6?e${P)DOyTs;z{}K>yUw% zcv2#8WF^0_I@!BkPWIukiOFVs`wBi+arSYE!G8Qt4}cwVEKnu~d$gI1V7*;ti`_K5 zBhVA^2ik=OME=aFriIjufD(d^%yG5|RoE>vTQ(~tmX(U!9R0U!=d-3beogTxC`)ZG zl=85jr-JqJ0WH3mSb_4BzJ~`OEvO7vp48f9E!3x&eLXG<0?YG_5^rzc$$Rpe zzZrH^B3SIGwWiUJ87&IX&lM$C7AH#z#vvVF9A!hBXBDGv6HIe~Dn@=YOtVCo#57w_ z7Fx;GZ34<7m0W)NI)sl;JSh?M7J25tQds1`vB*^Wv1xBF0gn3ly6O>KHAW?JUlUC8 z(?~p9;hI`urC`UBl1RGr6|G$sHGy`?*B3tn_GI-pmTw5Q*M;p*@NBow7mJed?Y!bW z;EM%Gl#`*+=X9mts-3z`)GlAuPJT1%7Kvcxs>csVX_B{aV@8V3Qhcea%N}WZ ze1MOb_-}RlV=eXMSYOz*S2ZH1T&m?Ot0%^2E-drlBQ$T5S^zW8%9%|j!4=ke80uji ziebm8@IKjDs{fMSF2ky-I^ME^JDjWekFX2zxV%*&rPkiaL*V-|U$H_~_rUjMwqiA^ z?pD?paz%`xLzTJqUD0MhiiuY&JN+EO`nj{@J%_-Wbe*8uxdm6+Q5^ z{8rA&_ezsE>KE!iQf-4g_px`;&AOyrop6nBBIoih2Uz(csQU{4M0Pr}0c0ZeWPKc$ zw`v-?BWv>okt*HA*;eMX_Xam7S{$H0js;d<^YhdMLu$=(Kr9>6=Wa`HL&G2u-OV|} z?yPeLYxk1VA5Nm=c!F(dHf*wve%P7zvornw5%)FlQ59F;yV*@xVBszhFlrQ3YO%2u zjP@aH!0@pNDvO(hBv1@!RhmX^t!71S5)wBXa&x(eN+|XLq!URgZ!JI++w&nyPY6c!Zd8ruG#Y5z4_@ zoX6+d26fjqr)zWZ-$N++;Auk)25I9t$}3Tc{>qX%6u%9o=TMfox1sJ(d1C`!L?xiV zLM31ms_RP*V{0XO_H@?)WByyv&r?WS`A0%s4tC7iVH)i*bXnue5!qD>zgH#Xc)hOL z4C)*ussjl+M4r6#fvEZdWQ*fdzl$OYvPEbkl3Kl0C=+W`SI{D$(?W-3e}lpd#Z_k| z1iwiIgJvB_6pOyYXZ%6or_M410;zV;y&{eQlvdJ#?2AP_Llu_vRH)fpo1rjX`8DA( zAB3Or@P7y{r8$U7b~8WOrcchTr*dxRtnEQ4%=Rm7dhk_I|6V7Bd%5I zI9xA8IO}?~b>|I;OPmJX!n4pUyuuU@S6DM*{A}Gq5tJHwmU*g$M7MS4YD%T}`4&!G z#EIz&yF7lT2ns*XgruQ$=R~|C9pA#~CUac6g1zBPDFWCV%9ICiPMj(d^DUfs0w<;` zY}hz4MNoFqb4K#sNO|#knjHc>N3`y&643Y-hPIZW(G^CO42>eds1naYd`be|9)yAS zemxKcu^6GK9N)qaZ$~K1_;7`VB}1eL;79R{b8sL^Yyqg_(V#p66yL(2?qg7Nh0P>` zq6iQdapAi0OogNSxezN5io)?N4Dniowr-&-%w8EHMNs_ooaSso;d%@?C!NM7xM=k@ zOzAA~RI^EF)s7?{Y!}o`Q8W*>rzuS5vVT0j$>sGJK3L)z9EG)6TCDl-=AQEPIFdYzBU5Q2^Wla=@5=7bf7KUo$veC6|{Wi{lA}A_7V~`RcrW^Hp z9ib>0-@-7Cb2_@h)SI6vLPt4D5nTkz2FdvrPEHxNZlSAD1d2c-B$omenH~*>asn8> zg~3psty}17l!788sZ=6E5KJUSf?&FXrWT1q>!j*Qm`O?hWsrnXUtUGQR+Gvd&S^Xs zn67Oss%_oEyz0(PfEpr+36d|dGo;$AP0_h#ds3|Y26hGg>a4362u~y4)C<3sp?4(!VJR`bZ8bVPVzJr%Tt2$Ct*_&J zD1y)`gt}>{e1{MN<>OlzMi-}}Yuoxe_?aRE;*x6kEgFHZaRc`&+c;4+zJ=imfe~HX z)^BFG6oGKZukCYfGuwY0UCDYxLjm~~2Gq=n>Dsn_K0i~0B!2E-DPh!rMmYiq-@-sB z@767JHA+JfXf$D20r1^#w59w2hHqgolymDAx*BDn2!t#ySv0bk+Aw^~we_zkwHSc& zEexF6u5}Atjn}=HjUR~SBBTmix<@m20+~;@4x(l*`7f9 zFlJ$K<^!p`utPSa@`C@8kjkqW&!qC&iaVWUv~1~Lm6~xVhDZ%w#EKW##Hx1Op-4-_ z4Q~m!;bXiAbcjG=Ti~bX_b@sa{4-WU6$`N#0JE9ASJ#zIkIXIXluTO+xiBCf?dZItSjm!1cqQ;@vJ_2Rw%Nd*{MK`=#EBd0FQ5Bcm&=MItq9c zfs}RvCaLa(=1$b(>i%jwU17I#Et0)Lbkt#o>{i#+v3_m^zQ{XQ(UsMDBG$iXRMD&Y zU5chxb-`I(k@vr?s)0H~2Po8ktO#U%qo{AB_D>+<61nDfkddx_+)27UnUT;5Z5V$M zRj%PLHH02r1s5iYMWg~GHRRJj6JLc!jzXRnAyIlbuyv$)93H9l>wNgUD!&5Ck6_)% zunc`4g=C)}`u|eP7p>fTh5Ac!(02wqSRMyisM$h`$Lc#Zm34WM`jY?a`n$h1zdNvgZDoixRn6qez8#~2E=!M>g2Dcozl9=00Ko0YtCgvwED z(RUwLLlW9dkE*7DT3NjpsoT##20=Z^dNQ9#XwU0h*0y0K~WO zCdPTjJ4k=jn-~rwL`_117xh*!IlK)hPxD<*hGsNhleUcRAZc)L-3&`ooix_oY-(dM zk%5MGpr@VGVZ?apwIO8|^IN1H<6yeB?py^xs1M)5^+`i0^uf5cttW1)bqhrZ_zAMY zAq=0Gxd`~g$B}l7Vu`q|J69kb;PWjEe-6W>D|FBNOc7u>f~&s`im)~)!k9tDnH3gw zKA2Yo1}Tu(RA&zYAAvrK30G0qAq~I!bXslU^kK+Mga} z{hBUnPp)66{TD(=i3Zn)uwV9_xGyX`hs>OIh8fy3w?k;p;BaJ}R8a*6aUiW3d=ss1 zE!hmM+2*jUL*Fe0wbN#(FUa3#PC{rbD>U=YN1U*2Xq)bO6)jw1hx>U}nn@MJi74*z zWMvQoD+8q|={+dSuyF^zPspl2MS8ABIShKi`305|+M#m`c4b+x6MG{Lj84KPJ)Gyb z8CoFsHZ`X<%d($*S8mdh{t9=S?v2%hpnl~&0Z^J`Xyi=*r6k3K_kfbW7KfI@!AhNX zO-SYTOBkyV&c>Pavw82wD3^3b&fg}}(VlgN#e-`co&93L>7)5@skjzSPO02m;dp^c z;Zf~lxfq+apcpyZvhswotr03R&4fbG1}g6nY+2v|-0a$R4NCFnP$@c=hDtHOJ_^we z{r6F9Mhmp%Lx;JI-K1_sQ-s<9d!!Qjw?ym9q9ssIsp57tIcchvVfdoj)^Tk&b`CM> zi{nh)n^dcl+XLSF=pYShm29+$u~BExJ%n8#Mp}7-S~WibEhSnAn8C?`#Z0%O8 z1fe>-OTk&3IE|LJ$J#Bl)qIooz~Wc)zpe)V!mJM-e&w(^^rMgnCGY_Q98nAEL z6X%)#9aWK*UJw!XzL^1q3BE!2!xK5R52;p8gdIE-5mH+oBgL7v*a zy8y^Nj{3nTggnT+j56F z>hb%#{@aiH7kR@RU*i{p?CEF2Pgo1+=e<7?Io`oXFb>}M0$y6_l*~Ac8SwZI_i|2; zqiBqN4i0Z*ykF0DB3w`c`uu$K!CBifT}pI63AoC?oAT zZzAXL7{SNUc<<$^>Dr9Lc}nVNOR{@y zC(fwQ_wRxJ;oUeN8^Z-!4NhA)%yA2Tx9Pte+?(@Z@;?Mve?QaWC9^r8del=GGz+Ut!ORaEW3$M;NgET-VYH9&6IIa`Q znhG=sj+w-nrZ}<`$E?A*RWlWOcDxX-DjsapYvB-w|A;V~S3Gv;-O214ZZiS+zp9X;2)g)5Bh5rXVY&W5Gm!(kM6{Ct>01fm|M&XKZ85ol3K zcW}bcg{DQ#0qc(ePQXvs)}6Bu3ClRXg_m)&5o-9Xx`Cf5f^Y^DVg_OSrK+h24@nq2 zcp($w#;q`rSx(C@klAS8G8?9f2cS^Ut%-5A$fac^l{8rIbWv3J>UqwrJBN{LsL{ZokHp>< zJdx)TZL*h2g(6csndxw?cx691{*x*c8#*Qe5|sT->^X@_96Ur^+hABeC_4$6osI~3 z2sbT~WF@MuLm1KOGF8_`+?#RViu+pJ>v0EP3->Q@2ogM!yf^uXF`vPJWL5icXm{G@ z46A*&fpp#!GS`M%Ei|SSG$FwIJp#86s}0P<$5?hm9YqDS`vg?Lw=fmh5Zbzhu7=Yw ziV$G8Q&@Qd9|D#i_k+Q~VAGu502O2-n=x0Rjr6`&NjA%>ID{Ca}vZs{DV)EHkH zsaxl0NjWEyN=_(WBz>8*M|$qxI6XC~5pGA}9U}Qi@=2$=-8+VD-ql>Q?5_uMgessj zzKJ*KQ15KwaLl15rMRX=f$J!F=VYqs&J&2Q6z@Eq)Fb;RP>9&rGt1wSpR-QGY=Eq!qv$4R*& z=2m{kri#1q(E57I-j>%}_q6P5*&_^rVjpvg(m?~rdA&vzYJEfbcVIn%NCoE8B^{Ny83nwQ{B~)b|5n5!X!ilyJ1V7Whop|i zR-D}wSYLz(flcRr>4~w#OZsUn$43hf5Tb!Pujcs{1Q+lVo&tUOy5$|$hyuowy8`u>?PSqJlouw%%;njN&GFJ9~ zOO_O~UzUPuwAQ_xCAThR`7h{w8MW~9{sy_bCVX~|2(;>@4YYMy1FeBJ(T-wdYTw_w zzg^dN9z%3^#omX3OCP65`50hzQ!9I!!#HIJPQsESb}|W@BSP2u*D3uLiV*Nq5!=-s zy84Ow27)+_Kq-E{g-5D*gu;S^A+|F_ih%v&cvf~&9?mAOwe_>k9|ZhFM@qBEC+Q$| zo#MTn1(EH4pxR)|DXv6u^C;%2cT`F*J@?{7fE3ek%lj4{rAWp~#f5vr$Mp4z=mMapV$S91+@6SGQPlZ^LqbMBE=2K%T-gKEj{z7#b1!p>2 zq!k}i;EOz49?)(BTa&AG+{H}OR`T0c-|xI4_W1P{B9LejCvXR^UOdckz5bhmyYLsd zB9cE|g@T`lKZdL1>fHlM&&dmh3){$G%Mw9{Xy~=do{- zoKp*ref!`ylRfDc0L!}5KJc3tv!|v_n#Rktp4; z*Ui$Ny@+&1k03G|={`wwld-H*ZoW3fy{QC;(;5}N<%Se@Z7Bk34X1vv!L>v4sC)G5 z=fiCM-GGa^4A<6@D#+%m!Xqr40Yj{`{yzG4f0g2=2Y8kVtpvbXKY$?Oj>CTa8aAO% z?xIBi)~cJNiV>JV@x;9m$HFO#>`L$9prtfkSbjhfY;~$PSgI3~W%gTG zZ93Utt{z@Atn346yj1qSRQ93NX|MmpTCZ8_JFNAc)~fvHs^2O5&{B5B;w`9d(VrGo zx9U&xs`H=IA8(#=$m^QY;GI6j{hT+Iq!5e|WrUJN$8K$$?Fk;iTIfnVDrymMHQr2z zzm6Yk?XnHBnH7~N&K760P9j%vaOjA0ROX8cJQ7dkl=RZahe}^G5@oJ(|FwF!DE$Xg z=LnRWL-ZK3#>o)7Q1RJdYXhtI}(N2d*(8Ai_J$t5FgqS{LS^X|mB+_^BK zLy*YH@XEMR=v@`^%DB$(%D67ecHVa3pL#<z8oZxF^uY zD{18n>JufHJgI+1Ic97Iy=H4!@T`upMR&|-{S>ZLwP_-3V8@I<#77#Y?U4tLVSJ{O zrd&9}y3iz?0eF-zH1E^Mya#m;q@^jdCyEbPmXo*E-anCvAY9)c)DC3#;h4qdd@jl2 zMg3sw`(to^wVhRZjns9a&$AU>B5mK}DR?A&fc5L}`*=Wmt^&ETG!T1=^*kZyKA=@1 zODHe$ciSS%ZpdIF?_3|aBbgO26p12N9C_`+!5khBd??V-hUDs@pej@V9FK?uzSqZb z_6{7sV#3;`p*;-_kB+s^w&9#7W+Bri1oE*cJl0xb$B1T+7}4Te?~iZG?`pl@-sb9R zzyDbKutT3oaJtkD?;PigeH^g^;IhvHtcQX1wBL_ozyz?CfUD(vu}>hjQqrU3cPZ{f z;@+%kP9HgHRh}YgQWo4klD;wL1b)ZyOAv>C4~J=H9KJ6k(nx^3un%)V(Zxi#vvq`)_CQs-R~ zR2JD#>F%Uf_4ZLgkYry{3krn2`D{OtouxemzXeK*?+hxBYQpwJ_xIhMl?%E2LGAj# zpc_704^)vF4ai~#r>wuGBiiH1>lSKR_8Jph(KIqDc7YQ4pcpdzr_-9Cm)2xFz{IhaD}pau+FruJMI zqH|~*&wq*F$kWH+{CVm&+Wea^v9o`jO!(^IFgYhdo6f=G+T$S0(xb|*MD4d643}Z# zkVg+!S~MSm5ukKA-}IVi!oP{L(Q8gu=YqX~Rf{JJhD<{7q4f?Y|H6I3CWsjoTIWKM zl;llRaP03{VG*23%wvv?njz`kY$yb9vn~>JgCr^x?1#^8B3fIvI~h6~7`l z`?Z4Ar*VKs%mRHB3(+yq2^ph-&JdzVT}p_;Y+_=IeX`T_OI#-SS_`3R(lkO-0xF5R zM`7wB{&Ehc9^l|;G~JG{3q#Y9Vg1oWhUZBzMF5)iEAJ}ac%|`~Q)gs%4)P!Tb!mPbjP~gu*)!g~jdx22Wxu^#Sb%Ptzjd^}PPr zqfOIDVFSw1@$BT)NSpgSjpHE%W1v9)DL)alBIrXJs4&?-ur0YQ4?2Aqf5_~%JtaTE zI|^3o`L%LBpFvYn*D;Y0kKSHJqg=}TSw#{ZP6Jn{?=Z1q(B`)e`P85M8mDGP?l~?qJ7z0oeib2WM|Rg-g zW~8zJTpveyu#M7=LE{N9C_)`Pmc`1jZ`ch5rE?GbNkbJ>2fa2Jc}aZ}h;zOpsrw0m z|ac^F2x3L18%5)*a70xy}mAwv;%8V5nqWDjOf%jPUNA$YGEs$2=vT9q5K+5s}ifGT%u@jJ9l-n~vZ z&KJAa>xyfDgU&gWlWUs=3P1O&;K-2ECt$STy!%}EdD=3-59zePm2E9P7TOM9S4?Pj zR_dM&`0lLKJsa?yPf&8GvtqtQ?N{;5YCqC1tFuP*)BK?k#YqOle)48;V^g@ltrWaY zJX()>wF8TVwX9D;2Rxvu@1U~Wl_=fmlJOWclinut7HB}B-6OVqtp}_ihJpjyVv0>{ zcP*@&u5rExI|Bedn6w}9*CM6^`gCwXS=qF0AMvjf@l)JeORhz1@XPV4brkTBv@hul z4iH9Wsy&7ZWDE%hv@@-ghwfgpI$N-)!go0F5f~r(@s3xcF9gy&N?MVadOQW)5fdlu z#SUmcdkm}X%6iZm$k;^cBJa?}EK3KA3MmBvDA4_q@hMMN6XQA8>U*>ZvVqbW1mh|Q zY?KzA&;8eI;&YE>@fhd(wauiQp+QwUfbeXsS%fc)%z~_=95{w!x~+YBf^qU+z}$YP zgM#i11<@W6vfzgdlU%K#cj3I(*3U$6>|;L&RMw&jf$5%>{p~k<_|Q<^JsbO&6ll?` z3(cnXT@<79MrDleF*rvb2_QFn+ApIs{TOto9ip`99%I@cLX|uE8+5)A2o)%#pf7Qv zm0sJP465zFaSAQ?x`m6OA|eHN6R!Xx;N&_!C84Re4a(J_d z?F9%`(LRuL!#jeTYyT_+J-K9@xbB!~;#s=NvI&CwMnz|c2cjk;E+D<_PtQg3#K#e<%YPE?Unst-52x38 za7+=0z<(Bn2TmgVfYvAzk9Eh~{^$pfgEvEVWMUPcORbF; zab9{hFJ1C>^75G$E#o1hec{)`9Q50T-$DHL;3s-F_ixxUbf$#Pcgf7n(}lI+W5aOh zco)Q@B-wP~ux>+*PREX~c&{6R1L(qoiq}>)%{WiK$f?h-)p5$fiuy;#)%V1Wx7h2Y z?e!k5_M`4#_3*OyZz|hcMoUhd-(>b^Pa!lN5WH_FO;J*3RAY}(T){uqV*m(Al>2D0 zSz=fHv-%)GY~tH$G+=Oa)m9v(?Y7~d`!}7CeeUs$-69THu zB;Pt#cDmj>p*quCair)965rLAohkEPG=A)owPhdNv_#rXQmWXwg zoj&HBD^g^XeNbO^y3Bi1b!H3zOb!9~QS}0I*@u+AZOK}KK){B!2zVa?-aAX=bzZWw z#xancqn50FnT#}+T(x`p61jrI)2ZaFvP-Tv!X^(Yy4=@edFW!mu@BOd>{}wwah*sj zOC4nLegMxHIVU(EOJy@|PjGY#5Q9z?G7q)P;c0jeqh>j>6|91D;u?+`e^dR3aYOc0Ror+SB*ByHvcvYGh^8-{9As#JMIK+zu%dXU*IxW? zqjj3;)Vr6~FTAP=60SXw35IwR5Tzu6QuT{Q(1vCMx2X)>w$S)4poGc?<|a=mHKcg& z2D&lxJ_27y=v&Htcp+rF{C;)K6(+BA_KJvLcC z)DH(n*fP;)d)lbaR_E9Y$DuO=&p~-wRjyUD8!7AvMz0`K89ON~-flNKIr6>_v$t|k zPpGA$tV%|rI@gwo%Fm}BcTlb**NvK)EBBu_i^?_j@6$rI$M~40Ga?6(PoDO3Jq1G{ zyp90md4Sx+AynQlQUpE@MCzOD$UHPUMB8J5Bp{pwa;^}0z&SbyEcvAD?Q=F3fmLeO zIte*lh1V$uj4JjX>kbw|garK<;UPCsW!nmm%38$C0%AI|f}=fD_XFBc#L3nE01Qx* z9_07JpvszmS`yRueghnKNq6Dkk`kne8taH?&Ta4P+;hffZtuAwOSCtth_a*or~!wN z5-jeWZnjqnH#pAzw3)O^j5HA~ZcxdQR*@##qa#92w`wZ&^w(`=B{)3iN|H6sNJQhz zq49(|K&_)_P#sYY8Any&fY-=Y7gDq+X|z$lZ<&9TFMPXDQd6+#eklI#$uOoOgRm2n z;Ywzrw0*8MQ=NMQSe#~!RODF2GR3I5wk(u#HcE-}IxkXI60FSrG}W(Bs3rQz)hP({ zT4*aA+mvn{i)!lUw`xacoeIiv$@M@@O1iaFQieBK8>j)&oaxr$FOyHP^K0a*cq0%;=K&gQa&ucdCr?A1QKI4@s`aDX^DHyY?zQ z4If*}-=`_a(qlO|LusfUQnAKRj!3xk$oJyY$N zGQxWaX4H|@9`lq;vv)YoP{y2tGi{H8BSb`wHLPYF1tMIv^AK{^B`Cy)-|BNVYaYUkIGH1&r`lchCs&=fsPM! zbiAFe?5-YzoIg^q_w2~2Q_3X8dK9EJgmo5Ja6+fOU)Wz%Crr5*SD z(Us-&cr`u2gM8#%dsDgtDZ@#_iXfziU={_GAwhK8^OD<=gGtTm16J8>>x@Nt(3&p$ z1YR`do1c~ci`u>{^_0o`W24B{Q)VrYCW<`k6#RBFzUQ5?R@og2 zZt|RP;UD~1DL!TOb#kBh3jxiy3z+2}5@hF>%Tgh0NrI_hJmo@)UNAb?D?&@Qf)z&n z@_%p=3yFB&>j^j^!#+`-bIO|T{Ic{7y><>0>cjl7O=(R&iSjcryMpqO`M|rtS$g?d z#>hu`T98X0MPW#f?I_d*a`TeeWTxmc=6R&;*2^0{wxtj4DNEgLDoRu4?Y2T=@nPAc zHd9fCuf!fxJRR$prWmZ0P0}ulwCh$g$mn}iyTuL1Z2Dl*ut<&pJv*>?b;v1Lzbuv~ zDYJH4iTV&oax6y^i*ck~Ll=ugi)0K;{GI0uDD4Vf;c6BSkvPb$%3KY(+#VSs z&>V*v|G09!nrZ_(CZX;{qo%{E)Vcq`PhcVn-6;xgeF=*$#f|k__l)ueDlVN^W#n3Z znh{)~H%eHz1)3xLk_euZ`cP11Iy{h_YFDE;q7U)Nt}aRd+gVKAvNZm-Y`HW(IJ%5x zub_91VlYd;f0LM{)HK0pgjh<2gTJQ-r(?z|vXLb4Lu}Cm(7OR9SBn#$QNSqXQk0M8 zm*jeMU|_H>k7{{CIZ#n>L8Y>D_J6#oU#Y%2LWO-o4-S{g9a*Xe^v|w7$Z(;Sv=%=u zP#952;14|L{*Q+{?*1K`9&$*K`?(BJD|o!A0$kLRl|-Jz0)KS~T$ot4x^0EVsx{BQ z7lv-5vb4tGU}5Mq^kEYhBse@;?jL(;B`=iV*Kl3-Oyep7G2^P|+Ny4~DO@W%&d`K1 zYrpp8XH1> zGKCIevMC3GhYIc5G*Lj=Pc9up{Ge2E1E@8(;85;h2k=m`58hgM3kla0JNhntimEEV zNGnl@F8pl(U2yl&5#Z~!*WT);Io4aS`chg0XwQlFdkU(=`?LJ+brw$4>dEY?i{{!F zxtOj#8W&K;NqqY1Js)&K2(w?y=lcQ8-wS&V3nY2jc?BK-|Z7?2)8B^ybWXh#D-ri=G)9siK`%QQqPZLzTHX2Z=pa!By zx`u#aAGJ9`fE^YPt_A@0>dPtM8bzbdkw}X+F9S|>Wf7AXS&qKA64&rXGcv0sBe zqX;lSih)zmImnn6nvqb2(w{jr%Zi=#vtCd zudkn9O8aDl1K`kanSZ@NzRc?^gF27vr24-=&wx&coHwf~6*W{6e4vt;QId{l zfz+2~gCax~R$X9zqJ-WAu>G%66*y07p+Ne)x+8#s(_qf+m5n zS3o$cUb;3^*6)HJsiFj_RX??eM>H9qg+f}EUKkiqi$Df!GQb1pS6Behb6@2&J>+7`^FM+|B_fECuEtDn;xr@ds*qk%M;R(JC{tA2& zj!|dtp}zXFSc70Kj3OQG!KN8Pk&bnyz!2`?ri%f3JX$5hoSh>ItVU0-yn@*_EUY8z zk8*MS!H%K+{@>xfBNxL6r|U?=(@Wi8j6_=-^2Q6qhyNNRAGkebkzYk8@V)9KP2_;Q zzgv55A$l3GJ+IN>KSu4Tga7XBiDAQ}*1S$FWOQ-D9q486>w;cjVHSdqPv4PV+9T=Z zFX9EI1o5!%9BwHm*H7}Z@Aw`#c=91$t=YRwRo_64nQ9Gna;BTC#){a_k={gi1e$_x ztA27c0x`-L2aUzlJyzq|{~ zf|r0t(l%Pi4X?$NyKhUA{Pit@{@`2fGE`xP(;F59U9UY;E6`*JkbjU zU-4@s9=JhXsu$o%(P4lG2}h0x---&U0@FZv{wT-G!1G5T%^&yB{Beg`L!4h$Rl@&r z`p~hCp3d+vePnb+PaheVe#7+f@@t~o{~jpq$ib*DrjMKO6fu2VD4zY-A>F{um?<)< z_yTwionL7T@ZY&T{{kojw`a#L(VmY`dk#E*TtItHA^1Fs_Mbmw9X_m%L-5g%-;rM4 ziKG|OuAu{i>M(z-BjVBL4;r&2KYM|o`J)`4Fn^S&DlI>v=8vj%Jb%y)xTM*Gd8O|> zdr&Iue!?|+_8=2m90s9*W(g-gk&r7w#*Xe7c@nZNktM(t_eXKG~U2I4mf*J7;7m(CJiNh+KfW8KTFe!<RR5a)ks0<-ik{1Nt~F2a2FH!6$sN1bO0|%0Ag%IU5{l z$(Qow>O-M)GmUhM7b$x<9a|zsVaL!v{3|@4`c5PaBvF8+4IYyRIK{;Bb4R`*9k1`%$QQ)cZkIeYZ|(P*~FVo{e*DPIU>29Uwwc5RqYa1(nv9zbp zhp{-MU^@u}`e`>v7Rpi;!B1TJNJ#!LP~{ab3I}q70Ti6q<6g4cMxozQ zk?nvxqM+G;YGpbJcx?HtkNhTxUv)Fvl2G-~>SwB>dtc z5?Z|ZXCbuk9S9cMpx^wj?nwy_?V8AR)SXc*J>9WlPWPXBI+4hJJrZ>%MP!EQFCtN` z!ikndB^pnOUW-fw-XtQ?1Ha`&lcEx(Qljd}MBuO@5>5USC+fpgLdr!^ym^!;Co&P3 zPLYV#M+K%jPSg~Y=srp`E;3P^k%$p^sF4#r6_u!h5_SF{0uc~aMHFxCKRHo;RHAj1 zXnSNLC|^V(%(13|JC1Xr2~ml*QKIt5M1ze)T)f$za-uJ5BN6ciC7KhNXh>8dM;|95 z)4?$Ij#HwGA`=abO4KqWfhb9iO4LV*KDZ_V5%#D=8!zTW^P&>PCnM38$V4=dGVSfd zLqWj=PBd0e^k3H^e#8W~$QZ5Ocq2D&7ZGwu3f{Ma_Cftvfc7;el%Ji2w8ifkN>l2e zZd*CTizd^<_7~8zr`Wv8zy!dY|qdsyp7GX#%s zwpddvyvy8%rgAo4TAvnej|HY;Dxr7jFFPG$TI#JE3YslH3w^Dr=m*Y*qPx}jH0KR9WuK39Kp))Pi8E?c ztHcWL@E7k<)tY9hS5}*<5YJNYI~^=a!rlP9fK9m;%ANYstJV0LwZ}%Wyid7azCmGq zPIK{>AQD81k(0T)Ofq1WxqgO;mdqNr2cgC-cP|9(Ekj+aKsM8W>gY@`F z&Y&I{bfNDcgO>fs;HY;vGI$4%dIpQGua8zCfnHz>3?>4~L%2feg4JqDiCtQ`1tn+y zfzbcj%gcMIKo6o{j-tB*@0)QSTHd?qZIkICOGf8I>XUb%2+1d3VNy9=(hCxqCrG51 zN#r_SAH!^C8c~DYpc9BLlor*R9HJBPMW+)GNF5PMu*fEeM0ZkVfhw;DTBjbT%Py_R zL(YX(=tr+SMRzC?wQf9O(-*i`cx?jFj8HSo;B&Nggp;-%R#qp;9Hj?SZHgBa{u*;0 z*#X#J=~vlF<-K?6m4BG-SCseuxBk9|?jzK4(>xq|@Lx$Fr{%qbw%bVl6+*u(Ha}60C=AQS82ZkBnAV40-l5EM(&Uf#{~B!|H4^s)k-bSa!PY zB^k;rpLHr{n>v+pedtC7C%w6Q)D?~eNJ;w&z1RWNZ8>Q7Yce`HRtCV>t?`h8TqVm(6tAt`H zR-d}UM!9xY33wvYxK0_ba>)tVd)x^UQhx&4gqKF()O`Y_b-H_s?wA^+p+n+&_7+={_I1W9hoCnWea^BE1%jm1O z3+RD|CFR|ud`)?qe9q(_N;*qzbVzx?>YP}TMTDc=Ajw`Rkz?c=B*oh!-w=b36f#kM zz)ArVvAUhaQ7kY9pcod#+kyhko$1r8z79l&w}c+pNc@ptgEWsT@RH_!%jashrlDde zGD?S1ub}!l-z^!-#@5|xYB^|FvHw;Xq+Nw|G>lMVXFA_4{zNJ1%W%F|G!;>u)RZWD zpK7nOys6GwMP#3p)0On4tbVdgAhGT#RKDI((@ zBGMIh5-;S(bMMh$Dv%{l($!y>G?%|` zg60n+NDC&#E2bX$8Uw_jsh{Ka@Tl*d#iR>^bUJ2PpRZ}^7JPtl#dw?v;|RMkAY-Fz zsY}f>8}e*2MX^_k3?94^cgX!8y2SZr$r!Axt>XQ`A-Eqycud-iw2T?fJ;i%bXCG_; zrHUgU1R5&ex{Sv`r5VPQZaaB${~Nxj>1ON$hN!^)g2+O8{Lcs$U?m@|CP7Uq-IaMS zV057DrN`G$oYR!0A!2O>=%Jt(Z^!0(Nk)xWRrrn?j-nVmd`h zAN&pOD9uAph-96zfnMg1$oynMVy^P51pi5I9Bp^Z-=p2Ny&LosRDUTFmp1<6VZA&hBnavi5UwagEC0LtwBK87TSg8;@H0p_X@ z7}s&wkIB=mBs#gj?o^NgnQ1-#jQ~Ed8a2=X*a*P2001PY?i1P-^f>6;s{bjJkgJXj z`2{P6Ivm;+2dV02?>w${|MET*O6BVIX{Mravx}tJ-U%2Ln`YASI*==3kZVdKYiI0Z z?;L?jHBG0rm`Q1+VX}|Pr0y8FOrt~4LuG2g0dtyqfXgKLmlGUnj{fCpvxLyl4H?VQ zJF#u6?*%x^{H)U>PE!voq?hzoO*~9({R|J{Q|Cne{yqMxzg)^OTD8uRVE|fZETUvP z0t*wU9)I6Ha6R0i(uBzBz5}So-~U1NFnO;+9k9$CgAI#ekyL__sM_6Kb?Z2Sf6sZ~ ze;Q-dHiW0Q?uvJ4_k0ulIwMiFtGjCE6$C%_Jn&T`D*A*QmXo*H3Rb^?^2aiKGunHg zHpnwl`AWJ?QqAtF>?pJc-6F*o$KB`uX$29QWh*7=AwdR^hi6zdzvjSNvYV zkNI^E`ZMN%IPAVwrc%EKC)ngV#@#w!BZ?kvbmnY`^jgdUDQ_!tZ8(ql7+e|Tfu1wp z5cG`QsKdKy7r$6rX+Z= z#SQKmi{jxzfuxKCJ>5u9&CsiEnM}}E4GjHaBXX$t#|h<60XlWZUkp_Ik4#3DUBE}| z(wjl`<8};#kfKLkVC$TMDmBfpQCIBz5mpjSX*TYnSK@oqbQ|U@Q|A~FhX#IlG2}QT ztfGQ`{I>zo@PLtNHX^}^kxdMl9s8)DgCzevw$acY0pvxa35$*77Z}Wgi#S|xYql--z({&MU1)i{<9SCmoH%S2Z*2Hjt-!@4{5h^ z&@pWu1)WQofwKlKh%$4IWffEr4s#)aADA*X8wsjiU`Z6OjQzk+8 z*wC?hnh~4M)h&wk^~89m=ma_*FA%oJP&cz-E~U1b$`BG@N5Os28{6j6IxX-ygN6vq zXEZ*ByhEeJs>R!eIev!UL-;MjPr>gu_!$fij}QIkdl(F0t_dka&LOnAuLY85uE+S$ z?kOlEBuzp!sDtI|^K;brP!USwI*tu(ODCa&O?;U;Sf9Kgt@BFQ+cwRB)hc%+n)k^3 zhdL6?dTG?bXg1!k6`0~FIBLN9eh4@f2z*09F%@WI6BP)&$Hjolg^B~^>%0eH^YcEV zBJShFvScg0O|BDSSnSr`ZzXs8JVg%F6fbb%gy2yThnk|BnxY#;4lLkV?eph|`#6!s zCXr8lGq}6X3l4Vb?h}Fe%lVyevF1*LF4r7#{Trm{MbN9QW*jxiQ*8FjuVoIMGGvLNNu|BuT+e_l*MV@JX%V` zgqz?T%9|D{5X4M36Y+u1DDrpTWke9 z8~^lmeHwD2Cn_$j(G@hG17b@Q2zM7+qz(o{)cdaF^q6-7-23Wi-nlq(-uYE@+WMC# z+#Tjpfu%X87*bZfrmf zV4EI4hL9j!eIe}mzfgB?*8YTh=a>Qqu)}84DLr3dTq0E`OmC2Z%aMYi0IQxT9+v>vd`T(k}N z5H4Ds_Sv7Q3YzvW+z~!GL8|ySQ3HDKxwLd=Bve}A&Zfa!%PQcN%bN{)Tm2rzLv@>6 z%>t+Aws=A$AXL0rv^eqgs(|7hN{tOQGq$iYj$K^9b8+~OlMyuq-Jn^UJq3540DaF0 zfual!YES8)PX7sLyL57X&^xA}-9U^VaDZHDvr7me|3oCJBCn*QI_1Ks93whq3zh#L z^*R75#vA3YG*!X*Y4H&c#C0SN*X-U4^lb8Uh8x01N05 z463_UwEPL&(ej}Ym^U`=q7J0*ws>VD?HtJ&#tKJe7ZIQS<#+tWseh?FMqlcSE(*@J zdPgcN`4!qHNLw9+=HL-sA*f7`MayUmr_Y!OUD?A?L}MmV(FfICRP zPbXC6d_i}|^TDc|e=CQo++7W)Z7-29p(^LgDpz-t>nljh%uAYFy(W8=`%K5v1lMKL zh7tCAYKlkrdhT5|;#zN&tBakIcX|;}I(F*4R+DL2f={~-68c_SS;t_2u5x|x5;Jd^ z>MHtPewu`OOD}zCHP#79aHQYRW4kuC%KZfa<|uZfIspUQFaq#m%l?QTKGyk<_=*^-}QqE z&C>QmODu<9dU_J3UFo^QFTFM&4@vvV;E%Y_b3_DJ2eR<$oh3i*~sR=|g~*4%_wiu~dh{ODu=#^g0}>6Lr|l zb#M<|BBxUws3bV}qM^fD-avJrtflSlq5Q%3KGh)<9$kmIRt2&(u0v+{QMqU#Dxuq?34SU}zZmM_bpow`-7zJOM%U&EEXr06!=sy2p;8sq`3t4( zLwvmxif;DxUQ;y7xxXk)%{=Biy?E)MsyVTn(>hb3b3F}*8iUGqPprBtR9PhBC`n7yNLdZAx7bzo#)YBj>U+# zYZf{`8SqrnA|5756X9%}V@NCA2npGbDk~{Mqt%A32&~6)5cW!!qUfl`LID?xFUilO zMTrA|ob79h!82(=58xGB^SFr+3bjXyt6N&}TY%Qb!-!H{L1{BMWtX-ZFDPNrgM_oD z7-#dcR8gO0H#nP1E(zC{3swz=G;zdqXLIrA$PV*E8nSY_x=T0erEIjA^d|#vr3tJL z9spl@M6se3<0rSrIDxCn>OMm8n<{Axe;_l#u|%$<@qMeA(M*ey%U&V{g@}=sO)*wd zjOuzlMvV8em-9j~R?VXrYa(C>tQW?3w7*(U;jn;}H1f-wA-fFF;psENXk`r*C3)`> z**?Q)^iH(%E-G*YejA>ELxVgil=j^K^xd_1(~&IcS$8pkSG<^6x`2@wCQMVIj6^&(^YyDGHfGmvD{ID$RzxD zBvh)>`^lbw%kv-6<(Yt2QA3od=;m-~CPF`Jz%2O@iWbfuSir)(XF3Rt_6|GUU8QeD zmf$z2-{OUm|D&&=DB`B`X!Vv;!Q{Q@#ey`b%6d~?FGvF<$Mkkbc)Y`aO`gqjt-fOt%DL6oD;2>Pz_#iqS7Ca=@LQLLv9@Ts zuh;I4!$4a1Lw=v6xH;-I5%EwhiaLg(4i-^G5WG6Nwow37NOkB>KmR(2GNB0?UB5GT zkn>vAbFBO{@imtVS0Ed`Ot`)Y+&r8z zSg%$z6#zdnHlNqZvtd8JS@ziE+4dku#lWx_nnLoa^p!bcDg*syD&+(Wd)Co6gon{S zWw-7gpuY5)Z1tH32=TNP-xRQDy)kAJ+0)qM)UD~hJr1Y05B^MCePSH$=hBh>93zTIiQ#&Nd= zau2o7s1M|}w9hyc$UW9R<5(b9^S%ec(IRI{$rv^JqJ|HwmbcquX8LwJ0`ppXaD4Q? z7YABcTf&|{Lh}POz(6Q7BsO@9Y?Y^JN1g$L^v0g z;&X9yShJjCQ|F~vco+FprVX2=0o#|I4aJn86m~d{M)pd5F1tbFO(t53+;8}gJn|?; z8fUJxA6Gj+!?3NiLqc^ z+^4;aSe8A`GsUBnhaJwd%j`HB%jRrcW`(QTIJi}4jKS#Sz#)Puix)zxN^_#{3_t!r<493$`7r{E^PF;>>F1cKZ_j_)IktYnRyQ+s&4WE&YeRd}o z>9Jsqo~ychGGiZw#@nOj%_5hER&-N1kAsI|Z(lj7_)Vp;yXdB5*etNlD1jAtUU2kr z-e_kh=Uj`R9u<5Nx1<1=QbnA~8IFm6gC5s>{JH_mnS+ZV!}ArirJF#PRKwG!6`@ z{5h`7nBcV4K>qQRl{9>mUZYmh_@j6n7=k?Uvd2n?8hYUSIZ^i5r~B$1PL5a{2=X1; z;Iat}dm^rY189X1hIK)hf~B9LHLU6mlB782L>{>&#^G!58rhDmVTHp$5Ho9d3CPU( zRZwk-rl$66{mgg5jvoF+V9iEUIJqqlYZ~Q%cd;_OfsBHI4X{Ht$r~zg4aAn+?r_K( zU{PiY@Tsfv1`Of|f8+HZvoCMBmYz2x@bg5z z-oV$%d`;zR8eeJ6k*8MDvLR*B4a=ZA>8?rubB#kU=1#;xF_YnW&_nW`CZdSkiOQl> z8Wkp6N>kO9lq%E10|#RK0WtJJVQ!kUS05A_cu*MB$XS?@nv{b$o3{aLxeY`C#KJdA z>J?b*N1Ue%b;B-#od@?U2ef}+V>s5nm}sPm9$-kRoJzBy^h6GlNfjx$`g&kO`2>Ya@v?6Z7;jwpM+Dt6*nCtpMpt}E2$fF;OrsfK?M$M*hm4ytr-Ee6z~-;M!;4I zI35bnHZdUTLwxLu4xY|e*3ZGEPa?FjlDd;Qu%VuV1TLu$8n`^f&jOb%{JfP*b&Q|u z`I-7+=g*ANQ19wopl5n9GBuDzT}`ubz$!e-mDKCBQ{cemO6q!@SM(9C@ zIcOFo3LZLz+4GmE*S>zc7rrK{}M^ zn~h%w578$GjHiKE4<^3s1BA0AmooHC4vr3I7-)$Kh7BVIrGvOE5;-{jY)SS2qFktpSm6DavG1|s28)y)PK{G zz%lt*3X&>P35wvCF(gf1caVl8=#LRRTjp?m>}i@hn8zNfh*ZG~G7Z)QlC=C|grJYC zZ5ie$9Iocx!2Xz_8j#cP>)4NPg+tWz2S`%>I0aU&Mcq68M&ZNNmDG=@R6pZSguD*I z1d^;FIwnwa_!FV5qXd`0@&|BjtR&>2MU`7{V z@PP}`(_ezVAzUP>7k>hSs(*S<|5RvYcO%+|Ul3CYw}1)Q-aJ^B@IbyYH=zl_O8i02 zI+dANI8msdOvyi#VNR83Vg?wN&bH#Wghuj{jk_%N)u7Y(}q;> zA&@LYXILNNd~6C;8k+bRVQNg2Sz6hJCus)@-%@J2DV<8{Wrp}9APRtVSfy0)6ak!@ z;7ctxsn&AXW;?L|n`>niN*RSA0WIo}vNUlL#K0339S>+CX=S33=+Li` zdM1z~^9h_Ery=-f@Iq5%=b>8^UxDw_2~5mkE)19AZw=ZgyApTz5T z9e2)sl1kIO!VxTog-5%R+ZFh4=NkJ`;Q)4|Vuf$Cm%D%oZVg(aRL(~spbDFvr`FK1 zrSP(TKpTWn)T2Gm5efH7Zh~wbArgr*QIRyrUUhw3HS2WIz9}3gty};s>KHcA%XnKs z)N_()s0{T-iMgf!`!NGC0=qP5MB{V~LRenR@t20;3(9!J0Q4?C`1Ztv2({==6kXvW z`cfYU**p%iv1Mx8KNl-BE{RcdB9>v;^X}1h;PW>Vi>M6|3p(OroKwF@`vtfd>Wjuo zOx)3*(SNmj0QaQJW-c0>tEC_!p}w)du8>k7KGturUx9W7%gp39BaK9(2xqLoxXGC0HTh25ALbms=nd?18LU9h3lE4lzam zSR>_(f<+T*-{=};*3Mvyds)dS!P2*1ggWUgeZ%|KvR5Y&xnKs}fe-5ZAjCA5BRV%^ zJ*xBXt47cqsvyQ}k~=Oh#}j6aSHBwODBDFPf^l>;FB#zyzwCXp2kNgpT1=z?NsU}) zQ|o8f@Ad7X86P4`9-gF?ui{1BP=*3@#syp_AUtjTEU*1iaX!aQ)iXN+GrNY&>}|IU zX|8hH-)(?!ijT9xamT#W{M83oDq5uN*}bjDB37V3L2zf`x4y?}ZZFilp<=Al`8k>YItHc@>(h?}C_KcHVHYcM_S}_-NLh4R| zO5R&wuB0aFxMz&X^vy^~!UFlcNGbixHzA~#5z=v9gxvA|fC#x6!QX(8ALyx~5Hgwr zqYz>NN&doreuQ*n93Fx*-9prGJ@(M)6))RW&Sc{#91Nfr188FvgWc1ijfyI3D_#Xc6jZ%mX`KKJBr!h1 z9pHN$ntgF#=C}wfNgO)!n!wD75%ukTmm#DCW~N2p`9luPzBw?{69M5L2raY$AvcJU z`|9I-d+rFKL|Sos5cBf{6d1?ONGm4dfqc-WS*NyuLP`ds1@6T#(RoSo-l_LhU>sWC zaC7R^6juhHxCmFS+7x#V9C-Gad^U>ti9j z@r^`A?gWTWVXB1FmlXGLoP$Mk<*nGK)=`Vk1}T4^jwl@LI}=-Ey&jIPnQ^7l9P?;? z$%hMx0f~ARmJd2ni%|m7XvzN!>ViRzPyQ4m&*lL7EU(IS+K)i=Dtwc+uY44gE!Mjc z=q}=KYh?bm3fBmv&*yKI&+xa6^es%RsrXhHS7=9*{H`3t)U5p$w~j3s>Dl2K&^)(D?=(U*LeH|Y;LM6s|7hX= zlD0SV>0_I}4d5CExO)(+7tA?*9jhY%ib2g1IjH4_!9@>hv&us4f4)KO`L^>nZwwR`11!2?LYoJ!J+-fpGO6R ze&f$w0z$v>$Bj@Pf3}tK$f+=&LZeUz=5!v28pM8cVyX^Vj!4^A93@1l>-d%jc|s;O ztuv!wZ6pcz=~k-1D8$FdgskX~0emcs?eM^Vv2V}m5LQa(sT&9?jmJX+GY3U<t4+nH6cE^YCm+OT&j&lsTE8TizX@xVJL=r)dxwxX3@? z2#QP*a!{w;7iU_I=oLOn6&}nHd^DUuq?G4txWuxT29$vHfEdHH7Q)EWe+p%BX+Nwq zAp&4a+m)9P$G1C?2kefEq3|gG5Furc#@_1Ne|9F#0sgI^pYwH@QqF1M5%H(|ZQv0x zoxcq{BL3k2F!wg_Q59Do@NRZVE@8uNWPu1#Q;RJ&SR=tUae*d~O`@2vn~(&ABz;mm`%+|7(wOF`~oMZCMec@4Yno{&LEdn_*&BUwo5romioq z;ollaoP@nM0-rbdx1A9KQ+NjQ69JRhEqEW%9k%0JG`$|AaLW3&OwbIWnYwa39?j4b zY=(*EhpFRWUyBsX7HQHLfrA05U@xB`g@@C>{*kvDpCN^5c?+r0`2HR#9N&4s7yWSJ zmPi4%d>%9x521O@%^#p^H+&t82~BO3+?8XdHg%Y=(=yLoH#nL4=|oeTby$z}{{f$I z64@@%tQPxn+DtKY9-Ygt?=O>Vt>(IS@cAa#fzdzneVu_UOnW^mNp8ao7(6|Z*uKeA z$BJz;PaUhye~Hwp>jz3{<{oC@;{w9u=xKY6fH0;JTs(Ezm_{ffxHmwD?%Omm%M&5| zEP~wnA7fz}Np6|B{vB|=_iw?Q+A-l9IfX(Yz595C-W{X!P9dZDH6PczTx%wc(VG9u zGkl*qicrkgj)Cx9o`G?_%N6siF^YK|;Zgv#>Z4xcYnfe!Dxj77ThV5rHoh2YV{Ts; z17QCoSPts?#(RPr9+|jxr%~%h5L@3CJFaza#ELcC zn887G{E#1q%A))8Ke~wlR~c1L^9)xRRe!@X@Rk#!>g+M{@jYY!a;SwQz}Gyh%+Orq zH@5em!m({amBue6n{fAXva{q4?hvg_z`>q0uh5u@7b_) zuH;{L9!42|Ll>0{%hnJN(-d)L>$> zI0ENZ{;h$!sy4udVg5|OBq{~T5g0e&8)x5#8;E^G56~v!B7ol~fcJ6$XX#rQ_?-fH zKmRsF-@*vr!oNLniGQQ58V(a(6sj%}-~0`KpSs6^1oQCygk-#s4v0 za>T@lhI}92IE(LmI4q~U6Gi|(zyWX~{hY`M8UC>V9>KIQ!u9;y&P%*vE~mr1C6O)l9e;vl zC9M1(y@Nf@ckz5S!Eylqu6ZxPazFmjx0~NjuzUl*^i2O=$9u~M;rGAe`E%!Rz8(J# z;oqJAhr4j`?;QSp{HFv!N)DwTMw`Ru++T8zP8OH< z$zlfsSEk|SDV(L%**eThE2_~v{t>|hI`LN+72%0)u2bmAj(zv#CgYR^m4`)}~ukNGPlzx}ko5(n7&{FO;??DJQqU|YstIaiYjezXSsgg@ru zus`OuL83ggyFKQm2GS~$>iNj|V2PnmX;`y23M#Hm%ep6-fNr=8;a z*%Z}Eu74RR=A9V;Og^Z4=#_k2yJUq5GpYXYrUctAI>zChB%5Ne05`^fH=|fqjq)s> zGu-!K+Z(lLCpuSXu781A>`8v~gV$*5s!4@C=C`}n0F12#Uaz}|5`K(OA(X1)unS5L zbe`}nwuc_8j794GCk3VlC_zKEW+vlV*#W{8htNNIfpJ1{JZ3x9S@Vzfn57P-C?zoMk99K*9*|2+PN){_cBXHM+e(nqFHcG;@Ox%c|Lc!bfpaRtHa zYOj9_?|NI=I#>ITnz*Mb7eX9&UQTry|A)LkMgDCwqX8 zvm8Bw8pK@xF#d}CEFaB5ZO!&ji^`_Q7;h#er*@JdJ86*)_`npznJGnvm4yBPx2A!qzy+Qgxr|lyPNx?N* z0EVrf6Ph%srZAJ;U#&`T4S$Q{r*K~noVrsS`X-64k-3!G>JX_Mmsr=J$cdw)Y&+GY z3FG)Geu2e;){gGGP(|NWs}PqK6S<;_fyAmLSB}Q3*{-?biE5>r$#g}oW`I-wfw@1m znj_g+R4mu`fJz*HROlCHnbjHJ#AlxBP=9#>TR4*Il_2_-+Bb&4C!Kqfe>;s@zQHvA znmKd*+t1(m<@dS6UPVpr9w>Q&M@*LMXLgay~A(Evk-v}e}IUoZ4 zgV-^$l)7I5m=C`8lHd!Ng|$rML@&C zLRkC>vv~4e=Op7Sc@ec)kR54i9AUwk<5Yim9PF6w8eS{p(&9R+J%v~#G<%V&?+lJ) z?;yxmV)R1wSN{>FFPk8nU0nj6=&VlHe!Qx=SHo|P`YnQ@;dl8DfMRt$r$O1i7j*h? zM?(Q6C{CU%PdWgh z$hDG+KT5@~#1O0cBPy;*UYaQGdxMI@;fK^L9?(U<0()jheV6vm6#C+HEhPA}uZEsg zT^NC0*Dp|9P5Gj!Itho@Kx&ADYsuX`#5`L2WSr+_Q z{8X-*G+eA;&L9^(zJY4%Th4_hzebv$f;4TfXg4>Vdhjsf}7sz{Pw2eCd z={vyl?aNBDShh~ucL;&w7|%t%WKP%ceQ1(KEHkdyo?5y0(N2l#vhz=ayXh@M)z-;x4=xH{fnXQ8o3BmV0IdfwGQC#eMypx+2Axl!yQ{9kh9vd;sxTWGG|&-j@a<3h0f82~UR5Sn24IGDP)ONARZ+G` zfqfS5h<@Cffe)I>4aM`lV*m6TErKsj*IdGW+SRa6Ru@EI-}(D13#}BPbfJ`hk$J}^ zOct-En8*Sa?5HMJQWV`0x|e5*q&ujXo+Y}u`emb^ncmOef0fosR&!Q{TdU{LXQh`= z{1kd?_e*r#K<6&N6CFR$?Va;wi^Vf(Z!&%m8;u&H!G+qd)J3i8&&%*x>!LOwM{_38 zIr*H#Ul&717L*txXuJY@mMU^xt)e7Vj;QE5Q4uD%jVOuk zgr{}liSha2HqJL%EoWiPBNp+t?KFWj?>PMrGPt;o?c;L>#+ReQ%#w|R+%00w3e$oC zp;T7_sLWUC>fyCl;X{i0my$3aYFas8HTy=yjCIF)U?6N9te)1$K}&~bauCJ_ZcYgI z2x0k<6|C26ET~7m-ILV&G`xNI>4^Lc@VC+%&$GpvlY39 z4q%b%D0zPn;XnJSkEm8BFFzNXC0v^W)$jXo<^LZ$uh?{rl zN$rR*MWjbL%Y59>l0*(zhfG>)b4!GICl%M4bTiJh2s@$3QV*jO_ZGi1Lhow)-9z(P zLznF>?>BJ*+qxW_3_XnbjL#L9U}JE5cQK>P>_wj^~snUBoi7*S*c9yi70JbAh}DYtdU;?RezzU~N`>>vl-xA#3;Z2=&&?`RN^?Ns zakY}(OIMUBg)3oXJFYIrJ4Fhs+e+FZca3mbj@$cfG{nHnu*6=0`{#IK4Osw*+k0(p0YCmAz3ii`W7!+V)haGD>k^6noU z&A7o_#|^Od9$w5L^K$CLzvFT#H_&7>^BICwvk{?7IY9`#c$!!=0svr%dmYK19eYwQ&g5Ed%LekxK${Y$u(AW>3&`j}Rq(i<;HRzZX>`um+j} z);DjR*&e5x-#kuu+R#uBaI${(J5l8AFo#&_I++LQ!u`z0)jvWQ3GID&UiB)!spJLL zPz%;9vSRnZJ#7&WPDok5)G4y=Y>FFSqZY4XATuPN6UD1=H5$swR;wfzcEl8q6d@K3Huuy@$Ewnr~1NDP}P$T z>5_26%X&fT{NII09TbFqm6Z%t>7f;lt2d)Wdh@4|$%BGLlR0#mlX)X2a|S~oZn|l6 z44M2Ikuh}gv@wKguh1{s2C3JCcC--w4EKwaC(*G@><%J`i`cP}`2-O2FGN2G+mHtk zgU|{yEz3%h#g0*;%lZH`p3DvW?e!l(^Xm~GobLmO83Bz&Z!sSE0CONf{EDJbHr&5E zf*;;BxZ!!g;P&ApD*m5WFaEoT;>&sQmxINJ zt5AIP)r(h06t|#wRidl9hbB#Lp}imH+dbp^LTD!ly<#tF(+uH5WE{xHB7sN~TOnaW z^VP(Ns($(!stTHl-|{!8fII0iNI!`;>VhPRNc&n)Y{5f*+ zMo_F#OEoOMk3D@Nn#W73leEq!T!jnw_hV!NG7Il5mhDH@4s`x^nK;mpVoejgxEOx z{`S)_eSBe}(MOvFVfs2Lmqs7YH8FiXpOwDn$-Xf$edSb@ps#gln7&esc~_UuMy@=@ zw%=3&#}`8WZF~rb+80Q~YMLth-Ah~-i;ln^ z`3?hODS-y8Tg0CMI@ z<2rXr7+^gCH?oELv(z%!UYiR`2Xy;o4i9yKY)kgbILbRtwKz;+uR48yE z8?}3Bt;iwI4MXNT1HXlh;R+4@Kp1j&b{OaTG|0T|eEZ0bj=uLc{_=f+kXlpcQTd>H zp`=8$!BbcG>2N-}ewzVf}xV`@!;>LA^v(|9G{?$Z6s%;2x zCE7@I2>ZBDG@$|KJ_4x`t8EAPiCEeU2ow7?$`$O0Si?sV`=23VpCw{rg?;Wb(>o1z zs4LLh`Qs41w`7IsZF%ui=?(7(jG@;}GS?L4X>Ktx~JaaH6W8&?w zcjazkh44${Ga^2%)H-mk=9kFV(14-&B_3DLU}7a^zoH*&eu;&eUm^jD^h@OYAiqTZ zAS@B)y6=D%^KW@n^h{o>_v<2rlE!1k9R$Y&Qu;8=VE1i8RE1ZdQwW8!hj?D(2$~_n zL(z5|02hpcy`O6q5juazK@@PU{9$6m^mEEH!KC0+Pz^kdq?c)u8qz=iM@i2{&09}W z(k4m@6?}t|)&cJ#U0_2uqdfQ1 zFhVtntl${)z9QU~y~hyZJN>YTWnYsYib&;MB%-yOe`~!eoyS;_uH zOWEit&E9|~hS=zbl9Ww)@Jn<4L=1FR(v#W&J2QPhjqNmts~QE>N6D(7Z5lCq5A4{Z zr+kM}uz~U8OxicH{fyeOM>94!QQF%lCS&)HlAW8orp<(F+P!eRWUc<3jJbvm7yv%2 zU)p6_`HPjkj9|O=9zp=&N_>a1)Ej^S9ufcQr(CNIe4+Lr$(GKY0BSVvw4wprdflVB zTXN)0LE}WqryD1@yXD3!;*h5PFhj#(K5$@9>_zQG!UrzHXGF%{Ig z7YoCNHp;9c2f>fY@ZiAi6Rr_E$9+AC$;5yp^?Cw>STE|U0t^vqb=`~QyCAf~-bG+> zt0rzMSFDn)LOs26gF?Z6G75O;om&$`{`=^i zn-WC+5_;!$1d-p4cWtd*J(H{mphUAGhyrscgP0mt$H9gm?u5o25z80I1$|-rjn8{d z7iBys>5K3@oJuyD9=1%_4g_jPo&@BhYWi1bU-1wBO}*`$RuitD=k6{A=WKV!Jy87< zG077MeV>Yz@6{%?`^~hz2HwvS-sE1W2^Qw(4giaenuSXq>smG}mF$q%HXc&U-0a5NDnzi1WkBE?VUi$Ak0l ztMdC9vQdmd{~m&VRBc;8=$HTa3jE#?M-@k-pE3r0zE(HaVJ@uW?tdD~FYSpypXUjp zzh2qZqoE(C#k0FQ``;lO#2EbllfWNU4IKZcpS>dfRC6@`U#F5mk@44=X~nMW;6Eb9 zZqOgb7VXftlCThqcKf*^MabVY0#Zl?sg60(*%*ITcAPY)Ya~v$=BkFaV9|= zyBwbjh-Bf2qV$9B3rd#}M56TlXCf%|p>q+z}c&z)~<78IoVN1a|OOW{v0)157cN_JDVxSYLdBnOJng<;jwN(F5bY#&(XHZE< zQV=g~NzHNROKO4p-@g}uJI@n=`?H@J_Z~6=jKTeJ0)15dXBy%Dx02$Y@+!~O+TLFZ`ZG#4cH<`aKPB_Ey zpXh~f7)8DaUQH-!#>7yEo1sCkA?VcMrbj|o3O+#R#D%RN=-G}qTJZ8Ht@!2M#ArVbRFXz7Sf4Vw@~| zKq>j42s0O~T$L4iy42*j>OvM0<&rMG*t9af*i5k5&xKZ+xXMznly8wLP3+)^{&_cW9MO=kKyu#H84~Xkfj;k-;gpZ}0v=}Qthfel`Td0%GhXIVF5I4AEUV_qGGb+XVG zf1-E3A{K+koAl0?!yz4NuO$p0F>^M$a;|1-S%;L8p#CoDyRR|8_5VtPX&PGiC; zmcW>BB5}FruQXE@t6-XT+D}iS;gBn3-dH>CJrC-xx_rvl(|$?lARksx4&2FiO53}o zCyPrGOD{}ggA0=le!dqQJcQP87U*FW@|pKy`60AFFGLFsQ7GVC9Q{I7SPVg~E%V5g z?3b9up&bTh@J{j|IIk^)cs|Sss}G;m zE#oV*qOzZ)Qe}%!S(tlRA`thg-_X3RsgkL#V-r^LZLyw3W1WT*A#(CnsYzKE^d$j9 zQ`X1fHLbpx}ztO<~V|%U#y-jM*b?f}w8X-!( zsy12LGTFDU4hcHS^oYI^_xeTNs0n+@{|E5HUW2X1Xa_>n2U|^YDNBcAl!I zg2t9g=9iY@qA9!8*Btkh@BKJKYYQ%Ye8~>ipPH$#H2^USA+g*uOwVvSqw8}Kxqgn0 z{LA30iQ9A1*bKHr!=J8L7Wh*dvUjDQ0R>i`gOgl1Vdw`djbli z%%`Ze+zFuf5xtIAo2F~5+^9W?`ai9DRDMlVWKXjh8&iVwkpXYmyf7__%MK*C^o^ODCe&0$aUi?IDKP9b(H^F1rPb4$u;WZ9BH6d z6A>jaJc_U^m*Bjj>>NRb{7Cq(p$dHm_iKF-a#=Y+z;fQc65Gwe)lyHUz$c2M8n9TF zyU3wd0Ze(QTOq&Co2>VInc3&Dg1M|@PL(eDf0!^>F^0lPSbg`Ak1fM*7snwq$dm4J|g*LtCM)=EdiU~zl>ek|@D23IW*5CgV!&jT{tbrsSH^Iet zZVDFI%?oS_76|j_yHw_xbNEc2j7q^-KHpL70WzRG_sKtS*;T-AAL>D%82_YoxP2;k z&vUk?r1x(1^p7Utkhp8|#r?)*~o-kH2Srv+p30t@9Y+(TQ$E%@!d5W7=QiMnwB-40#1E zp)79`l}G199eU3+nmh`PPCW>)tD&;y=-gRR>S{tYWn}@)|9)a+RoX z9ubfq{xwZ@3D$H(gl{awjbj#PV`&T`Y8EL4QX_8PvpOBP3{bW2M$41`xdrXJ^?ImS zy9sn{wR&O^MI>%3?hkIny?zvy-=f=#X4QpXLsos%=Tz(?4)veA__P6ca@<^HQnvv% zCJ|uOZ1i_lV&M{u_@3MWy9|?aa-W5(>DUVbShf-O4FJ|Ead7~r--28I&V=DbFYY!c6$_-qO#}6hfrlB(B+p&W46{3t<%!yU@#1BIjoL6n zD5Ki``~s!m^1%8F?(1u7vUKo~72W8cPh7@*J|h?#G;dwp`AS4$ByyBFi`7No4*5h| z>o0%@B_l=Z8HJNdmt%6Vw_x104jd0QAm=Xyy99qNMG<`dfb%&i@~i}vL4JN2gTy|p zAI1HQ<)&?ybw=D)`p~w~(aWP-x2|Ok65efr>Lj+)geOC%V!Z)DLMB&MysXC{jQjU; zja7(zH9CFhWiwpCaFe7b4Od4xT?9xemEhMbA0!~7(}#HuOQqaEplodw=BJxEjLM4n z;vqk0*uQ2gDTsB^#VHeMKS_sf3uT+UkZV8)WB{@#nMpfH-YVKnVRy<+V z6Yzc!+Zi%{#Wquma(5bT$t=aK=hOSdC5-Yx0yQun_x<1(S3Th`{fB!cucr$!3yru{ zop*vwWurV#CNEL_K6`>!@&rfals9cm?wiH;#+><wARd=lp&H`5li3o84 z^!o;_#aP+JF^3W7L}+q`sKE1RQAkP3GCi(fq8XaGer7PSR9P0oQ%U5^jl-nkMP}Tg ziOjgT!eHJ`H_;eAGB^HF`z$F1=w9@l?&p33sXp(B&NIo^9y3rT$`@*2dz7U(2Lu>T zQQ4WS$jXkIUn&?Ou~u?!GZdpc2{4-|gFUbaDq^d7dlkJEj36e>QOJ>dY7C)A7vd2N zc+8JDLiK+Qsi+@O@NcK@9Q|?C&(exoF_&-@D}{nQLZ8-e+q}b~+KXi?Ur#mMD@g!Hrc zijc9~PtEp`c^4EU7ED9TslI#B z1r*zAw|(TzuD`Hp5vFNaEDwRf)xS|H^tSRr@6BM)5+eb{mM zw>gOZ_&Z&K{j*A6hHGH{N1Q;IT#2nyKAB48{oedq&Q~QdyW;iXy6LJIx^KEhm}c)alMtr*z}%je1A7$x8B!kMB0YI z`ce0C^J_8Y*KSK~_uS?i)jyJDer*YEL3mh?gcRRs%pfNlxLD$$aGkEpNC~|Lg7UTYEK$pg#7Llq(#av z>2#7`%S7^9A@YaXUu3>}F=EcHSe*E;%kR{Q;6a`eNuDg@G^+2Kaqu|awx~;f0{X(W^q~>$KGojm4)+E;;Iq7Rzrw8vjKgY{d ztmdUCpNU4mmOmMN?eNdxaNT{u;d#SJRRM2kb!e(sy*C!1c{!*Z&weDt1rs84F!~Qn z3mn@dS2IS7DxCmpuBX_Y0<~A7(mxCi=OXK(T7dQZ2jw*KX*W%9JWhItNqxkL6R85v z#h49Ppj|k%ysB7@+*$ZCW&QOLm1$qx@qC>Vq8{^9DuM~>m&eq{3mZjU_-q!RamN$4 zePxMPUf36Kd+vxTz25UB@y;>z1w0s@IW@evFK{244k9L)c~wa6F_6AF2GUwUnlvG# z<(`F6IIZx^i^6GzCpikIRp-^N*;Ev^wI=G?s0g-D+Mi zcUWSftohoIk1tYbnQI;bE@8=_VzxmGISB(qo{M^-I{h)o4c0tHtdrc$nW|A|@(^ z9>dWktfeU)0r~zf{p7_*jKIaTTV4*6pS*ClufsH2VDkKqF9wCp0Q?o}xu%&|jbE%g z>7H#qcv65Avz#8osa)3DGz0Ml3FN~@*gUMe>rc8fcszW_a8uG9*Lr5vopE2&v{HY^ zD!pea-T2lLQ+LL54wq|An%(T{L7+;v1ylC=lb&OtwcA>Dmk~^@tXNBPCq77%K{?Ob z5!VBzq6qGLp9KcdXw0vk|EjopJ`MUzv!7^J&kxri*Z8aF{giyjSiw$I_TPpnE%Bda zo7kNayqVCXwj&#EYSJ%^cTZlJ;GUFiJF@;mBwbURnc&gmZ3B+H$MdJz&7W@}s~yU3 z8%%*l3cD~8|2E@7&p{a7+UqalFZ^QZEC6!pu*Iey37`KT^B?f2TM17$4dJPVqJW{i{_WY@sw7~=a5HHCSg5Lb*r=8ei=(-E1!Q8 z#4jrRy33+0SI#`%S-OJjYq!avp7(NA;H*1LyDDMD0{xYcbWyXwrUc6ZZ1=2;#KEN9 zvD%-3%p%*DDcdwxPDD1`B206EJiOzhn}3)?xd2 zL(=XhsrFL5+fsWe0XI_?nhb>!!m+OJx3!{CGSDdS!9=5Epi%HNd-U^xFoT~${cI0s z@DdZd3sZ9TuKYAb_pn{saHF=L+GIduo6*=)*btT4wkDX-E~Q|!VNw{hWQ4;FYIRT*5mzb0BrN2ilOa4|5~-vU_@=O5Y)QMsF7 zY?t{xuM0CgQ)_GVpZ7@0T@o$9r!P!1@2Ett$LoaIW4F^nqjY) z!FtUvy`8J9FmyG=GS0JwY2FV6;Hdg)7vE~Gj|Z^a+QmBdiQCc?2i)TqPIkxUApn49 zXxqm`@y5ALH8G)(?rl^t@NITG3BOW*7HwBkx_J+s z`EPbmahM72?xYxffqTt6RjfyaWFQXr5{DikTIdSruWkWHDYOE=Iv1a^LZ6a@pRj|} zp6k$auk$!9QJZ5zKl=tvrDckJ)R8_48v{iRS4+`N`Ml9we+>Mm=-uYJRy=_=r>Z`} zHKS58ilO#60)EThfcp7h5znS#9_@P_fmD{mdUjl;>z-i~({EhOKb25XB`=0HIxgdr zuOpVyS&!RROv-jU6lYzt`^!*yh&FF187_ufM3MT9&Dc9D$7BYR>`|xTGG%=clVFai zyJ>Z84gB)Kdt;G0eG^BK0k>SE4CA6b(4K42V>(h<5oJRM{g_**uVS$jykEC>6EK>{ z7DFsU=GqzujCSgcSW9ALs))X@H~c+@e@`0o9rafF?Hqq8HspK%)@~os`%C9`#L`s2Uz#RYSpB70az&EA zBEPZ1;jhS0&5)(>TYI#v5DwJAt=?gpfG_q(7%sz=AFsk=f(r{p@ zNQW(=ttzHUhn^QQf4grec69}F-hrI>D9mbBQJ@S7gV1L*^EK9wZkCE&FCzay5^WVm zK}soh?Fd2|s)&SyO;@VBOue+>!YHOJ@N)sa!oY5|b+~61t1o%*C0I2KaJ+z)fVM*2 zat7qB4Pb)&0sYvR!SM@=E!*Usf{7K*4q%HCR{j!Gdkg}ThZFc3bsF2a7=)~$2Lmp8xWgt_TX14oYjU_v;K%uj`g3}%+YlwS><1{|PU`FuTv+wc@!~TiGH3gP-J-EhS zxV(z?nh9_S`_)xF6;yAX%ok!r&v^cF6@Xq4%X@)|Y`TMKNIxl%R^_`oIrbjX0bhlvr2iqj4v?i{ti}AF!9H`kNW}%o|6L|zQ9c$<8Sl> z9kqKcz)kNy2+dV`5Hkwh%^p*u-WQnbd0_q--&RTIO=QcBY^D=#8wC_IOwN;7of;4$+uNt%LxlK z6XKjM^eoq^(});b-I0ju5`uVZ^)B|bO$2kbdyg+r?%usygIs20aYdtDV<4OB9sq9z zL@b>u@e)zhA);*HU{KSDU+}+nQPw?47M%<`rmLQ7&DbvK_{OYM{B6Rf`;6J|p< z6|?Ic_(1FJ!qHP?+OORlm!AOrGn&;Hmxn!7wQ{wv69j7keCMZEFgl&!Le$RKNBR;h z4~K<}=^bjuy6}*KF@{Zc;6`QjBxVPg8cZN>$bzly$qt5y5$bgh^K3zn64ru~V~FvR zpVF8=N6t?kQ(TPq@KMP0)uiZZ(s$R^%%S~QIo53&Elk-Qx0hz`1J_{9HkzBv64?^V zXcyn(z`TQsY&{;`1JFOVEv7O6Vy;*4un3#?x<~8;*j)Dto&i^m9Sa$L6PD4FLC7o? zN^74Ea8f($?qs7ut z)Q@GjVxz*`3Bh6LlY!sSQN-H~!b`szylAvTNWbuFeauOY_D_RN6p|NKXyx$bzSV1@a$F2ms|N90109t4J8z=M5XFuNm{POIqZ z51Z@Y*QP6mR|%g&ORUCfe$AB0DkRp@tukx9))sQf2(`m|dfVF@JoP6x$Fbi2mjJ<5 zGP2=IP1bs9ir?|3=%uEp7fmtyW&M{hKQ@8=`bKNqz;xu~O@d6AgT;3O@FVwC0^ zA^v@Yt<-RP)bREgu&2H)SD+zAHmnT+0SGj$00IvcYKRlzW`HluI51e17@DWFjTQu8 zyuv;K)G@@Oay8OlC+ttXUxCd=xZO&+Xre+SF^wcHQ=F)JjmO}xlLWf7teE<^#$t6k zq=9v!0yyvRvx@7d_|9%ks6}n&*A4@lmSWb0vj7x1vY%YTLMOYM?>9{D#M4FRGrRDH3 zwCATX$1pl209X)zPlvB#MSoKe5^X63AwBl5L$dZ~5ySGSq-adI10iKYykgZ&=|(pk zxMobVqCr`ywU~~1Bz~3(vDHr4vzB6DPXwPKv&N@MLv)enql#DzqY$CMluloVvEK^@ zM2V94b_CxB?p41-1^;!zGNS6cg1~Bofhuz9aeoeub5pqCUUW1pA}+p5t3gZ)wG^zV zGb7$b2P1t*i)}a!AMs4uH+L!gbZMR}JEXO;!y25ngft%YN7`Vwl7E@G?l+*nm?GLF4JFejAYtcqmlARjrq#b8e=f)v_Ep1ODm z2pc+z=;3&gii4I=Ay)V$GJBF!F)hJ}fDFaXB16Yh7i-`GUZk$)5OHqWLR$re@CI2T z+mWAu(EwM1lr0>v6{bIXl3ap4DRcdD6oY1hk9u=HQh~+&&6t_!ZlOO zDV3wF3#W0;*oQG;EC}?pD#qJvZQg`4_CQZt&-=YCJs%8MFkPkhLp|^Hw!ZUz@8Q1A z?A{h%tMtwh6nUq`e@_6R3EFY~%ioi$82zd%rBi#j7Nf1v#y>tUW@IxSL*qI*Br|+I5%N^r*i#0ib+(~g33_SEB?3rJ#4k) z?~%*TnO}OBTtEY|=PcdPeE=QaKPhdp#Ugvu^fSH9naHwpE>|(>Cn4NM^N!yGaYTq_ zcUo8*wF;h$N{)o1Rl_hP)o%U-Uk5!m;5=rZwM8+uAU1+Er!Od95&dcR+#2l{O zglH?tly60gB(h9Z*6V?=aj_reM5c^T9?f?dd1A)ofgfBD5MS<2+Ja1ms7xb1`!IKy zY5u3J%eCga1K|rqYZqef1fR%3#RNJ!6;r3ca58vz-meF8OHjW0j|T6^ zQX81G!%uS;?+BU^IhMibsG{YbQtqBSif#lEED4-?-%xD)q`A>JUr3#Dw+`Jv zGI3E4V2@#np<`gOzqh{xKvO%m+0_v=!Kt%`rA9-0qcsDq$LwcC^TCs=Dk#k|`Lr28yTm9%iXyT4; zdDuzQ-sMK8LwyfM$EGrCW2v<%Z#rh;jk(il+8wEsxXtg$Z-f?Tq|e~Gls)=XYAYFL zx-kXwS~CkQ-i%_J4=L6xw&n@!EvifN~OExX4kuQFl>RCZ#PJ8El3 z1QyI&&^}K}wlzP}WZLM(fPt=JBh;UBSVmmJNqLLB6@ z;C$ndqu5!dc4L}Ep4ey1yGqeD$B8Z<^N4XJg(t3%_f^RH4B9{;2mRH+G3Ag;V+CFwCBc2V*ANHvx>;vyJk@lCRZ- z^{D4D&9y3WzT+EQ!&e}5YUFt8QLO`DdJs|j>w4LqZh8fH$5q-Imt&4wbH`y+(AA-9@J4FSq}fhw&8+?!dy~zA_cCd&@*#*B_%&X_24#!`L!`D!hJEcOh?=k21aRwL zwRugCtW?T|*%l7+-L3Phm$Fp>wwd?U@*a}t@;+1%)QzYCNq6Abth~jDFhDX2dPjVJ zH~Icz0*;wp)B4{G7?1?W=6VXuiF(X+U&J^951o1FCoEAeKgQv9vFic1f;UVBhdYCn z9Ft3dCYIk4>n769a?;Lm(o~27!|*&CT?jv`c`u>|(UI`&EPXjgovU00rM?yuC`B*p ze}kIh?}Q@o@qCNeehKJQ!Wb);H?vBRd5B#r-#{H1iJ%cliL9iVlx@i&8W05Pui%^S zZzhDlnh5(>IQDyieSZOJgw!#dd3&DZuKY!I0~AFiR}=t_B-r<#mtj>Zc~LH-Ch5PH z%GfcqtTR@~5_j?z{e|{RVF0BL1Nn|mM25CSC+6`CxowVr1DJMr6d>4zI>Ql5{rctCjz=rsbVF+*v zMnXL;5P~J!O%qgb$YQgbuq2@sLjZSs6sz}<8Y>@xq129KoI)sbs!mGt1|~$2&RC_G zcU2$@NW?L7yA>8hbmP0R$i%*&rgo93=G_SapwKm_pZ5ElDE52o?}A-WgGG4zDhVKk zVGFiuNWBRA(L!iF`0Ue~q^t_AK!0Az=R#N#q@Tgf33e50id)2d0v^+cYp$g+*3?M) z00YiuhuOOF7o?e^A z%>(M>SY$sGypefy)2)0(As2MAp55uq>4$9vo#tm+8?6S6e_4hjwvL*=E&&8X4{Ei& zRnsTH)1>g7vJfLqvH&Q*k*}|?hP1G^PkoSOc(*;rfaO)LR6PjNX?gWdl_~v9RSGm> zhrK5SlBPZdAXr!lsDxYjFb>!3yaQ?pQ(?LMGzDtG|1B)b2*6mHo;r(-CqoC#$IbPZ zxU}|U+K1pqiAzeF(Lxw1dsO`@A{Aj?2BGu-A0ZoN;bi@&eMnB$ljd5Z=kv)GdWm$A zS3=yP(;fD+fCtZ$|K?y1v4XQS`3JizuwG4k>f&YAmfB*lkH{rm^kRny$R*eZ!s=j_ z&b<;;V$?FU(ZaSRwatJLtmGKIfFt%UOxNwl+Kn0d&}(bxtuy)~UQxF}e7BRIRn329 zV=5~i2+B(fUzeW88t_%gkXj`BCoGDwzS>sOgjuWJ*3Jy8d8^GBz^;&8qE)lp{ zTTLw`ZU@^C3A2y@n=@h%$o69)p)l_xCnW}RRDk$FE6_jVcQ-K-ic^;g6?WpO^w!kY z)J`o37-Ig#z?#yQiX}dMz=Wp@5&vQsrk_|gm|yDQo1}(btP0UuI4f2%h+!S;ewu_v zK?JBc)(IgUFvtamXt!pGaXeTYfTA9P!)9$zy#>Qug0YD7*(DQ}K*KisQ1x@fX|x{D z_4b{5pA~dsfo*zHq|*B_yeEGb3_uAiv&=hY5nHj!<|!kTLh~!1cL;{OVd8~(=Uv2_ zl0mG}cZ0FC{+D3qw6&ObQmCw``%Z)WgFurlichzBzg@G91Z8&^kv>BgvwqiBGVCc7 zhY}&M^3oA`8N6+QOlpf@>WS1&jnNdoD8!Xi;0{)IjM_;{f!$cNLh1v)HeyA9df?+w z4nP-DGD1@F4NfO!2&!VrQj&Keq#KAjRK$*sAHy8QW$ORZ*aJO`eo~VGI+XlTFdWj4 z#)G(fJ3&hAkIoBir^-ydj+J> z-se!iO%Dafoa#5|ftIeT@DNh*p@ZL=iss`R#UuETAHynRi@8A_1F2kL$;>k%w&YPm zv*9Qu6k2R~a0zB{K2_(gN^NuBmD*;mucu~#gKsWOV?1nSp(R`Z=SdAXu3!CE(SeE` z>Oc|t4;H}K89pSD!SZeJWngLO`%!$y7cy-d)rbCyZ(j_3dr2#sOJ$!brV-Uq>>wca zLGYnaN=mKUK)L5NG=${74!z2g%dc_0A=1u zN9}^~+~7@xJ*!z5^Um1SzS{{qf1XA4t8h@I{!cjQ_(Olr)v4oG+8PJhGz!-7BvYEZq8K^TRoWQ%`QiXRRV<@l1)>rDnfk|X#rxA(!hXUa{z4<^H=N$wNn zDY6sLubS1L4MU}gvS!a4mlcLudl;%eJB0w3r`#nw(>^6HVCKu&3vd}-c7~k2P|nVh zaeG*H4uj_mVsX%apdbEYuZ811&}^@?Y=V#vm;yE#8e{!SC4YrfW%Mt#_$w^_rEmvJ z@-I#CSER6IbLGOhtazSWI*%<)lXKJ9vITPC0#=+Om*%jg`PjhCR|l4p{AaM-e3}6G z@x}`CtBf!TEmTk7jc#5_l5?{%YjVsx`RUH$3}$@23gfV({?bW%vzY&4@Ta&C6bm2$ z>r*Fg6^pqEQbxCmm1fAs*V)pAtZ-qIH4a;4@RZfAf^%QXa_6CUHggGGpqT36Er?-T zLR=9box1?`3)EAUp=JeZ8bnd3YpUrc)P%K`6M&1F0QP-&4;?{uPG)+YFc|t=5Q8jW zkf+sY10g%UO3E_&vMO%29~0e0^Jn4vjk!6BDF$q^b=ItcgL+&?Zhi{zpp+KxRFIIH z&kA!ey=tjG&r0)IZbp?6C+(LxI7eCO!YaM92%J_kc}f--OiagBWa<}@f|y3&a+Ix2 zlLzz%*n>2;ABya-rO(N2so9?S{$9TmqCbQ<620EBssKW7I#UjXK*J@V$k}rTU{=-q zg(jJsXpsP)ps?J>GejI53^wol`okCHlY8~?nDYXV?{CJQ!wBV>~m0uzyQn;7;IVaZ`Ixt z#Dgmi?T^qO7sC2u;`zJSqO|96_fnf-b+i54fy+;o(XLrO*{WuHjdMrPAc;&YUk zz0SX_HZpammYSU(j61cJXL>%IY0HED>@0t_BbcFrza|A=m-E+6!PhMRwwDR!w&n;3 zsaj&T6wG#QFk53do0-v6oyNHAAarvR?f3AcEUl=y=`UJRWEJM7(>!$(LQCNxl=q;P z8baUPL~~t#HZl^Pp^STYYKA|=2y+wJ4Z+qhueWUAuj_(D-0R;)eDX&!ddqSxQ6J2; zIGC#zd4q{J0|UBLYt5V_;x}D!1-j{eD_q70aa0GeS=;{+Ov!XRpIVML3@7e5h=v~O z9zS;=Y(B&ImYY%QP150%7gR6H%`xw3X*9*qA&rW82sc)^07hA>Lc@qI38ER8mH z6NZ{U-lg>mDLm9nwPfXdd zcL+~Sp|>C1dd4nks!)%sElaFROqgYO`vtpARN%!7Y=1Zb7m(6 z-w?N`Zxt`1_sr^B#nnmnmTP>Qb95ertD!(O7$Fe$hfk*YYmb(I4x^)lNacZN;0ujG zzD(HE(Q6)hr2=g;n}bq>dMV$6)0^uOso#}blwv(>3FORIDhM#Me*iv_T*{F`*7uPB zT@ge*+uU@3r-X?y*Z&GBeHDTrbN!F`n;^>EM1okB&RhFVOh#%l6(esxOagdQ;xjcR zv@0z@p~WP$4f)cWURQI)frIQk4F1*cb4PAc;U*mc8iF|iDv!3iSz#74?Q^IzME=Ba z`76WGAy8VdGCGO#w*nj|2J<;ZM05boUtHo=40~&V$-hMEOX&KW*#c8urePH80oV}n z`zecpa*26{n87ww6T^mVpO}^Jp(fv-v3_s4WPr#LBQWtFh~xHa?Az1pfkmig;lBTh<93Q_gyW2 zF)YD_xem2N z>)i(UakI&pW^as4r>R$);8%Fbb`&}8p8bhzrX1O_S<^J3eoQ_RKxSi3O%~sOWqBDa z`Q`dEIBa@U{V&9EDpq^#T)(m9*ckn__60gqSFFAvzU10A=2W+@W@Q;g>Wev;zUD1- zs6U~Hkgt9*Zn8&c;w)H?cHnCw&^<)5XXlO*ZCYza{b4F>X7X$i#0F*tA< zOfYE)mSBQKOCaO2NTUQCpwKAf`qTuc|LSkqvaMUck&45s&bh@fn}YI9Fl528ro($9 zTb0a)u+QfHNWwZPvtnec*>-)+b%s*fGP1z5GFZFWsB=%R^u4dIV69pRCGOHK=_=_A zL4|pb2$L~C>cj@=M`#x~q+oy12Fcq&TFkc8YTiY@y{Su0SavD6Hv%VluPIG$rd z#p__JlMTJfMt$Z6V@(S4jMUZu{W!SP{z*SzNB+H{KT^K64;y9n*@iD4XA>Q7|-jKQ1cKJG|z-{A8a`` zN}TF$Ol%PM2p)Zf+{YBQ^b(xy8-(?6R|{~*u9B5Dso`{D>xIEKcP^6#l3}Qx%1Y;^ zwy>PXaMUhsAb~!dO6{hD^o40~$AXW+Ww~4x=PL`b+2@VtW#MlyFdNm`0yW0e7R<27 z(%#BolgQb|_sj|c9l2b3G7Hnp%FRFl_SZC4k<2;`huMmwkeQT%Q+=suOkKl2y#xD^tlVNMQI33 z$q2lE}sD1akb2{N_(xZYr30<54MoiLV)!)i~P;?=;Sz_88p3BT*_m z#m+-jP3495@QmI16A^=iIpH|9HdYJz}r*e6*A|A>aoDHQ7ZS{@K9;mz)ads8Kyke zZHjRll3>=R9n$F5o(-t9|BitazAJplw_SqU4^>u}hW^RATi!RaH5Ma82Al9u8o3K- zN?Hy({Sa!m*#lntZW{l)kkL?@oOxGR910h}%NycwII|Gq&{BUG;&2z)B*7n)C^G3w zp*}be%x^GN)I-cCFptiq-l@A7I7DqAT!sNns z3C(_#nw^Wq<@i&`gXxB^10F+hwxi*hR)>Va`#hje<@qCKT0(O%!C8-gXMz9q~wwZ7`(qS zJr-{%wt& z{hh2s$l=2IhBu-64efFU`4(ZOu|<+M5u!2zDnp1b!3{9RcBJ~5)t;MKtD#xByL||S zGiktBn8r3!QIy)uO9A72ld=W-A+R%JsxtzAp0ZN)B{V#Cu_LzoMgPReL@o(ZYI{on zT}k3xAxl0vZzHy0%Lm3XcpAnS-hq;Z>H_w~Lq3dA*di@Pzf=*lGvC)ABbN+9sew2& zC`I9fCJERIByW|V@zc2y_}93ClRrVOaPJ?4Y=LCJ@p;~7;}8hpOo9&??qzU*LyGVr zqg&Fj+63~plA zMmv?URVI0*i9Aw9gm)PPk(`khvn0T}3|KWr+a?WG2AwjQffVv(tpI1q>+E`RfvVsx zQ1}X1_<(K8faZG!xDcAOe;YLnIzh)bIXl|Tf#`>S7@^xvVc(48lJ-y*R6oLk!_ZJ% z2A?z*D1unF{22;Qu|nrg(VWr9afcQ##CMYc&S-58^#w|6Pi@28pt#hLda;z8(Vn10 z2tDUC;>kMIIxf0d{5YWIj8=+d3YSmEyhAM{+6Rq8GZgYiI|JQTs%*4iu~&B-&&BYk z!8{KRKhVF3E`*(1xYh*HPhtnIe8|=V?qf>~Q=1>PvUarPV#J@lf#nbRRGe71kl%#E z%DvQ*5UfjfB*RzjJw6@=s5!aE8k}OwOao;?A8ewMXx4Y#Ql@tZuD$3&-r-R5&Vn~a z(5cbEeN_P!;?OzpdxBbmZD@)6v{9QlF`qTjOx^>3tObzH)fFOYQ`Gzns@*rZiEB-` z6i}B&e5}Y80;HJ9%4>qr5Lk7D|TQ4K55zkp^j4Dj^57P+obJ+gM9d$K5PX0)`{-msbaTfj?J)+GH89zQ^MY2 z^Wk|;>fSR*>K?|7^(OeOvov`=da1nL$ks_Ryb{-$c6OH;jK16%dm7FGr>+%ukVBg;;5PK%@d+eJ)6dDlR#9RF{ zz?Sf@^FWE)0`3eryHV>xk?bZ7RQ@=8*sdiXHaNqJfhvMCAwJzhs)%`dfCQNB=_6))&Jwdd=fG7g z7Qt0;y?+Xc9!3z+^z{tLf)}=4_hPoe$o?Pp-ao#o;#?mMB*+$m*_5DB)0&o`vB5SP zD8ZlxAt42V8^|vdlyi!jQaD9SYKr^_hA7EqYZN@v7F#TM%<17AQ^m5Zu?7-Nz-WU| zO&}Z78g*?GH3CY6$iB}rvu3SX3HJ1yd%vH1|G449+Ru9DecyTKou6xF)~qp=C3Sen ztSqVb-;1nk^j}+wR@1-&^_n^_!J4@?JOuIJ6KRkcKK5) zQ(zmm^U<=E9$JQgG5);ml?$L^N96+8u&Z(b3r;ARTCfwkH0lm*_n&Qddtsv*Znj1{N)Gk>Gk7r2v6r&*%lkA^Rdsm&mU8H5%+M(jely_{Q`bCpuD~p zRh--XP$tYpJRtBBHqK2j1G{7Yio-Et%fIxluNk-%Rx}!DFJ1fnnvxqtqn@ztKDO~; z?(ch$^-|m4$M>{g(!ec#Xi2eyC;4@zhp~DwD&meg3_IuN`xcn^#*o;<7lWY|-Z{ej z7zxS@?5%%T?$^u16D7CWO~3@ay_usy?$>)K`LN|PZU^KRJ?`wgjw#0$Jt)WB zR?)P%`z|IJ>D|!7nH2HV9-WRC93L5tS^9!y-qi3iZ{@m#rmRFS!p-Zh#f;NM*m!~G z_sX@~reIL{@G`6iEvQ_(6-~hPk@%UJQWI4?68kQKi#G>ndhou*Ow3~fg0Qg$lhR`8 zkF;IE%w6qkyRZV(2hclm2PYu=?e~5)*FyCG|A@n70t(f}Vn3qA-qyaN?bwqmMeIlX0Rj#fX!qJ-+--(6ZzQ)7VvU+uNBfG7V^04$=t(plsd%ex zZaijWFp_#La~mFHx6O~-hogfJ?K^{Y|9R%k+-92j`NVX_F#OquV0ocHtR;R_q?f|Sq$ z4!V$Ua)B{J`@y(jr>um;q-S(F2!5?s6r78IBfOB4DfWv*qbQ?XB=ZJ+037*ZI+6_3H0MhoulGi%rjpUN;=) zbs=vs@ezwzR@}vkyEiM>#aFINGJC}dTi7y{HG!|8SuUKG#bKCGd( zM@K04+&{}&Nks$GF)xW3GVK1mj(I6EYw|l5rU}~Wxme^5UM3C+%JXWP#wty#N7C~d zlE+s1f>OHrKN#$j!NE2vVt=M4V==!b_8Bfq!wj^?T=iY9m_!9)-$0SUNga`I@gJr( zVo=o0{0Us7iDkO=*YAWr4EW}9Pxv%6(AUYWNS*|iJG{%XZ|x{NVYy+V{0LS{nu+U@@r#Tl#M*1b&20zD>(hdT&B2V;wgbV!)?h{(f7^l?o&4=gpBJ?TI|QsL6xyra z=j2-^{?`0MXA1CQW=w!hTXi7nFU-{B{a?&zNQnweY(T2AyumD{>H;k9qbh%qkrFi^ zt$MOzv}CQG+&xN9Qc@sF)`oq7sSPh)gopw+hfhB3KPYCr{H?TO^Uu*6;MF+0;smB1 z7OW?q#@wVm(O~%m=_&7t!w$-#xFuW`?=u^d@WEH!_G8mBvfWj2x5jV9xqR0q#nC1 z5H;VHt<0Ol4OLNT_~85EwB!cFe*CreH`?9|=4HU7;|=Jhfz9^DH{ zx9{(`Go$_0jyuiZ;tUMiV>V`^%fnVdwFhHJ$$FS=vmfYK+$)Or>Se(Wb{*(k`NRD< zNH@3SAl2NGgLHFC4ib)G>5uumv3QP5vHbc?%;urUr2Jr;*>e@f?&b%$B30O53LqAb z#oA*bc2KnAMf=M-GE+DQk{93h#-YCUeI1zzZEvzj{2&o+J z59LI)_#V=Hd{*gg@s<`Qs?T5FW{8$!I$lxZYHLo9`QSJnyufYreN*oBth)(qU+Mul z|28HVH6klGV+5wFyupaJgYZV%!Q=ZnGEl^(^rD?Sey}4uj)Wu-`qJ<9ta-is{#cx) z^?f|-`KQM%EA*LHQvU%y3*rT%oZcz3<6<8Ecb0YWUA4-bE~`3bd;AMxBQYzEm+OZH zRt~|3X95LOvnnc!c(%kC(xa|LHX(MbIH1XnJRMwIC)zi(biu`q?F}$fZCSgdeIJ@Y zb$4O#=xalU1lzl>LGfs7uMcL{SwS&X@4_T7XR6LCrs@q)a2z4ZX=h{FFmuJ2sNVGHaWR`$qS|2i&n5=zwLikr zK>6x8`y)VY>oPgtnTn5>rQaL1{z|U>oW^=zPaw)wMh zt_SAfTiOkUC-d?hZ&Z z$Bu+nK%D70#?v=~ZQT|A9Fe$rA$D28s2ji7=ZV=^i*|rJ6R8OQF}KMdl&ZSm&|qE* za1shb$>^%=paY=8O&@c&+Lta zA?2@uw-ISf>7Gm-d5yG~eL)^IxFqkw2MfTASMh`(I*2+tMj0UMMIAe<11$+OM!OSCwQ*8vrnIf8*U!D?XA`UEtrz77s^T7`p z8*DE?k8|bgM;bs@=ww#OZw$VWo%0$Yx6pF@^#4x#k9jQ7{+Ec2K5w%N;C9W3X663Lr}6dNkNs!5 zZ;%ltL=_Jg5n=Rmb8qnGK0U%IdW2Kha<===Vw{SVmIkRXP=Yiuv0P$_MYC(=v2IF} z|Ew%ht9n^<`U1E1hHeHcO9A}1p=T?qJ<2cl<>D24zCSs*nGoaE*h76*^DrMzstwJ7 ziGRkUhPhX)QhNipr}^KeE$3p&e-Fn8=C)P)f1XhOA(My~6Ff-bwBF#fKL0(1skPNW6e1Ecm)Y;j#-VC}Fxf zmrHJOt3uz@@85^HJZ#Xgsz~7FU8wA;y*$Y_QkLHrNN4lOv?HMpe2bZeT|}ADG3B2! z*P6@swc#!D^Ori2r2|}{o9RyVLxWOXIw+Ce7tGA&1*2)xV|K{-;> z`~vCpYk6jW8qO}mVzf2=uy!MRol{m$*p_U>$@BZu;P#VGhkk+Dju=Z;(58=QQ~Lhm ztL?nwG`ouOgqC-tGi@*rf-2=NcW7CyTS&UU58ELS<5gnul0NXC6oP@5mkTkgVe`XF zlKuke6l2;N(n23|Q38V(esjDM8z!PSMAADD(f^_IK>yvIyx1cfzV67wdjB_M-Rk_n zv}PPj#0@>z1JWxaW1o&)O8!1^3yeK%B_4eBx_Lg@#`CDcqG-N(u$Lx9BQemI{92$e z&Bb|eU+9X+gJO+z$oj_y_1Li2wLZ%8PzE>F;z$&(^cn zlq&QRjt6L`6AKnJMG#g&#TiiZ@_9JQSZ)pB7n;pmna^mD7W0e7dHn_R&1_Pzu8P^U zuQAs;9)%~UxQ|Eqra0$h@bM_hPRB^f=_jN_${`!pd%0{o7RADzhZ&In_=N4XJnv6U z#T5jmzJ`xaM4-^%P&B0S_IeYmqlP-B2FgBzzgFZ9}cSUnYYB!Fn)h)`gvaSYQRP#oDqjH=zMIU@y$0c z6YTr~RqXlL)i#-5d6ARJXPYv6I9hAMJ0nO^)MUuVU@KOy&ukdO_&t>u z;EP^i9K6`EEXn*nBC$Ro(t{VuGy8GmQz~m*9^NkK#ao+inkji_S=q6y(sV%)&RDNp z7gu?Ed{b5eqqz&~=n*);ALnc=tsKQD7Y?diay*3md;P$$Seyk~$;;FKfBk#gqvjSEDMbFXY3UfNgr zj%3e`ugp(i%7=Vi$~pQK<6p7!B2GFDNx&!PZV-lH$thmmKj9UR*nIyI5vq?~Y_Pao zD^~C)_2qb)E+}RZoa-_FeIAXB#v%yj)A3#z&MUwSC|()Pz-Cy0B!;AjkTemJfsm?9 zoJNU{++?GyXDzMFDyqy{Qkk`=GOMsMD_1O%$rprz;$&0jR1ro#cacA9N%Gv><=o|!Su2Ziy2`%FtYUmedf#yD^pAPAxpHo?KdbmfG&SH^Q{I=c{=%lL z(gfUgPrsoIKjI#11`*w<;`HS8!z$;Nc3*V^&xRIZ+^*eyiTKdExCcD~U$w}>5vHg> z{;X0tukSP8y@-B91zifi&hXD&6Z7oc(q(vwDjxyYF3c@Y^h^(_&3l>cW% z7{*@Hg58tImz$O9nGz{2*@3cO@!ELN)*d|g#yK4Vf)My6d(D?FAC4N1hf4J#;d&LO)x$-J%0^O0>)c7 z`9)1LMvFvO&Gl`<7S;S>e@=YNv+qRaFZGvV8!$Jl`l6E01lGjh16`$_*D|nMAu}%d zTwu-ca6$4xq%(oXLjJx1!-kwh4BUoThfF6#l;ca$No6?82zwb7^W~7paPrzd{@`Uy>*)%m7XG}1pUZ}Gva!5yAId?~jN!ro3`Bnv zg_od;FzXzBSLKXLm;=CG;tK>r8xW5F0BZ-3$SWL#JL3`@L=IXPP@EWCB?jV0$2@yG zyx9|Z`%+FzeH;B4MkyHBblaJNj6o`(QWC&!NQ{mv3xxxVQQUfb z<_jYSId5&OV&_7modcqlI2WtEv#VSS4z#nY+#OOYJR;lCTjafJKr`@C7sfxXC4cC{ zbdVghur$rYLEtfe@W6TKZ}@?AQ1nlKk{$L%ec5%cSYj=2fPW=`yw z>z#dkA8TYp#PpbF56k!HkAG=ShciV7kqgO~y)$RTJPVN|A%f*3G+%f-9ekGyKJ+Dp z$7W{4?7aniuiL7@mt*mrL>khwa@P(+e}mJFvKC=)T-GAmLb+!T>$=6e{LuZ1kSU~0 zA!U|OW(kcsHgl@5#|z^Sn`V#yYvQ>DGHJp#q+x2uT&6+UR`50P&^E*a+YnENj7Qlv z?rY+qZAg#wl@*Wj)u$uqgMsNm|C&BGw`7z*4>ln2O zWz^ga74QKT%-T)_hZ#H!3yVbN;~)pK;@VDDKDZIgZ3oP9K}jHGv!vW3C`qJjl9W3@ zX+MC&*B`6~A()lYb^?+=6ntrICn_J@D)}Z0N(L$0C1t#zWFH4*8z?x8aSSMx5AMK? zy>h@8_gVkY4(27Gr-{eWG#X2ZjGHikMnMd(R=4|-@e|b-e^D$hU;{Zk+9J8Qf zRP)F9*j77Q^ZbrcHLS2Du?7D7xn_xv`C^gM?=M62*vegrMWyUQv+P3LWwr+LTIUDK z>6WI#pvpDQ8JdVZ-$UiUFKQo?t7Vl2T^31B~zEpew zy?EFkMJxQ$9x5rmw(KRg4ruu|9K@@n5#8g;)+gdaI$*pMlroH2MwhK0A5lEg3Xd#5 zI20*2F;ZQGxDh2Q^u6p%gc$)eoaOtM&o|R(Sq{E)fVWO=irl}i?}Ar21WLvOaUV&6 z_@S8Y!2x*0I8!OJFFCV!-`NW!yFV`;66N)H*EG7kVd%auFGy}clU2SCF9O2(7|MFk zl{ceT>`Fd7Ke@?Qc_Y6Yy`_%5V+UTW=l8eqJ+pF-3v-pVS3BHFf&g}7Hqx{;= z)VImnA16Au^IXE+ze)DW*0WetqID`Mx^7rPC7V)lllhU-8RSC zL^*!P`@c^4<1&|c4gE`OtjGJ@L$C*?2T#~|y9TrU{zuA$EnO^wS@=p8&by2Y_??3|k>Dw`02vnrJL3G=xyPm@iGYq#wWvTx z_p%oOvpL%4bGl-IM1NcoExvg-%6}0)M7->)GhuwY4(|3G!(;wM=}Gy!d_(&?EwY((xd+(utEN7g^xp+-^p_ni1vZ6Cwc>!eLl`N z_%FhFwP(0Kc0WyuK=UE=Dy1|Mhr!JCv>gy;wH*Y7=Di*q*7in>x8vaP7OcAGCMJn% z0=n=9w5&|c>)gkZ7>DmyU{_0`w*#N9H~`?m8h&Q4Ox||KLNRY6(&v@wi-jqO$bXSC z1${A=j*r20(jMF;NMnTFIEHwHrHoDs(H;6-Cq~RXiv0OZ4q(h%PC8#{O~vqJ;9y9W zZ_Gy|uh!;s63#m#!Sc>BBTD$Ph`m_!Wej#Saf~U`f`Aimkh-0QotfFH{esM=2@5ub}`q-jvd2wE! zHFNeF8Y1&)wB9hOY;9KrzUP{A1odOqnKNied*_+uY}qr9FcSvO5y94$@c5l|^-7S|c#njx-J#WhV_CyHx|xK0q)Bymj?*938m7uPs(jTKj~ zxJHYsM_i-CH9}nbpQCLP#WhV_r;2NaxXuvQY;m0T_mnc#C55-7K!U} zaa}2{#o}5ju4}}#Ok7X>fwA|AYrnWg>?I{iTs`6%Ev{a1jTP58ag7((1aVCi*CcVB zAg(FmIuTbt_qs0*U4?ws+Pv`i9(-9bJiZs#^ij3Q?~8$v{ouHf44oVwAj&26F=B{dbxuv6x&Ao5i@rp#{sLi&j#5g3#WDN&RU5 zYj|JZJ0#PiVjFpN3Fj0U6=svW_&}nxa1dl(Emf7H;FpXJm&Arkyy24QaEZsvKuitr zl?Ul)X(_*9&0>C!?;6o?d&;}|&?%gVnGltDph1O=F7JtIpPE>B7rvc!S9#z1;=_VD zp}eaXYudSqaqq54^uD`1(d!5H&q(CN;MC(G96|A|_V`49bmE53U{cwJeTaE|d20mN zV$0u(z|LeDD(P*Kel(&xmJFx&le@%dM+49Q2O8k?YP#h@BrrT_6U_EcO2g+p!*9dU z`0S}jUU=+MnVr5sW|t36@lTp66!}a}-=M+3 ziPPbduJGGC8BMsP6Nqxk#c(b&&}_g+umLeeF($mN0>9A3{4$UD413$cY}e{jOUt~# zoW!W)X7P#kW`E=O2&@jJH?JDu-?wn9KcX@!vEiBsFHZlA=nnbA`9)=jC_J`^N%HsN z!?ZM0*d)xs2TF&f)@R(_P<~~E`D@rxG_>s^uUU-fo1zk1-vQo&k6~{nual)1=_raZe7Sx_-Z7hsF*BEaWUsJpmGu!Q;#jg%cmtC zL^F0GBAoKgGEj@-i}+)%fK#BLcofQiKC&h}mgKNVdMJcA{;s*=AXZ^+dEe0DQ84y> z>I-pHHn0TpVPRvr2#_U=Z#<(M;T2O)d3O}{Ypj}Eo*q$rGCX!UvU5sgaR~2Np&a%} zLSSqDY9sky^i# zu9=eAU*hpuenC0GK=J_I*;wW+s%1Ue7lj=B(tiRYXau4Ti3zaENv&v9x!&?wiFb-9 z6*$Hyrz^CV-s&wLR(@`1$@iXW#`r6JU$Lhke?hpF^$!hA%*HL?0mXa-3%DTCC1n6_d7zDzaoc_1%PU2t|HdMpP{0 z2lIFpKMvx-N$>bdE)JW=$bLGW85W{O-f5QK*(I82ch&7OJxF~+(=a9$Wh*S}2TOR# zFmoiN*(!+DCbI=?^Fr~m3;Lmg{Jhj~>c?p1Vn^1m9Xm4Yj-(a-uOL z&!O%^?T0=()Hd>1$tm)K3+D(SB<}PFk%v}Yln%c8JgZ{T!TYld=O!cI^4!Gu)MG$i zpxB(B4?lhMbX)n+2>(I}ZidIVh-ZQ;`0-~(?93IvQ9BFq8_4wRT*~{3PAo$= z4co!bX5-=dDF*J$#a+{|Y+Nslil}cIHb;a;M2N@o%8E9KJhg|dNJR%1vB$yO!>x zTmv2l>@{M!zA{gB=FY&WXu=y8q2)xf%q`;SByMrTvJv62v9N3>2T`f@u&+?$*dlRV zf@{%oc0)xF;P@cl*z{to_fNbO;AooH?#0Q6XvKzxTk(O<^mD5|!x9&{s>#L6*K#Qz zh;o5}ppkPc*1V?-DaIK;*he{SYUI8RX}-*eh>}lJ-+FN*y3@O+WXHv9-S6M;8j(nO zuIo;XY>wGG4Tqq_?7ioL)VB)oJ;XRP!k2T4U^WgtTXiA6JQUfUes&cP{YpxIdzG15 z4{y2DwFrI7&K0$xhmm>n6kUvWTkT`sug&a3hS=Um&LB_7RD}Ot^SdZ9cqW5ea`Ci% z4QW0Z;u9f!ioM>Sk%-MGr(1C-nw6n?9?I|Hhz=q3tqq)GsTV0{e_*lm0oBQLDG(mN zQsi5)xR#3R8eB!1FbyJ#cln)tqCZ(s@XFi|GG2NKJw-UZsd!Wcj#@h1x|7Ae`_6n$ zeuT$vM$y1N>eLeS@k@YHd&6TFS-)QK%Wubp$Ic)hKFpcfo7#)Be1`w&7HIDFR-EL( zf7Ixi;(OgAdn+fg^qqdYauQ>?{f^rXa2#JI?A$1>o5Xc9uBv=s1|~c{=>_P|P4onA zj`q*$O>H*!p9qJI9VlO+=YSM1e*-x^;qiO$NEgU4!{fIJ&?Pklayn!X9%kV8n2p?3 zQGRFVIX;I7)&@J0Hbk32Qc`|OF&@x0F0JUdA$rOhJhqioW+3= zMh#VT0|_U%{P+EG&3`QrJqR`_T2IjqiqQ&DdS}@`4y8LF7lt@^$h=s z3Ku?Q`fWxQ>G@>rA>+-8QN#xSsgzM)GTBo81h&f+o7f*U9Cz@CVx`6iveuEc7spLl z%J74gW}o>ENE$y?w}NCuy#oGv7e8_-RP)2noMN^a%sas%-Si8JEumPY6tj(>f9_{! zBSU{8LkFbiHiplrNBC?N9?01pN@IMPyL{&NL3Qk`;&Ussjt5o7emUh!DL+!li~5@p zdjH3o$!yVj@#Pw07g;-jbboY+@`*_uv{fB!t3ss^M)B+o{S*KvuMKEHwn5!fRM$y$ zkGj;g1Ohy)AvEb4lq_)&gwT#{OA+NyFG9#Uf1r>Iv0$)stZIf#*=*>NDVs+9vHPI^ z+5!4=c7@`y>7Qz!8G|s*KTD*RVYZgDI3U8vrj~9Xk!U-AnCZtDzM0{#+u>|8qqX)E zBTmshh-p|2$A!mlVlgH?-oo5Da6dG`CX%y5bAP5~17rL#WAyAtj0itoERjr(H6aj zOnbcw+O5D=1@mC)vHgNFDpP1D)u#OaJSd1Aq0u+f zmnag_bMTY&B?_CQ*d^0;n4z3%m#968VwY!B3Oe2{67vD=!#z@v%?m|Z1?waC4x_R1P|dpbEf3a=?Ue~UM0r?LmK6_Q!Yu!`7_%v{;&4Bf$8E9pdLHwjC#gf3#6=N`jCH5fW-3N8+vk$W-s$` z6Xj+cgj}hR^NW`-edbarB1ebMOrq7*%J-X<)#xu|?bxP_-LAk^1#F7q5uI14@jEA> za+6X3F4^y;h@MTPJdoz#9HlQ1?+^nMHM~Z{(HfQ=sZ_`Tr)5_lNy95OoU7qD4Hs#6 zj)oI7JXOOpH0;%Ip@y?HJVC=g2j`S2gA*NMfSskHFLiK^?PCtzRSu|LJrFC1%rRAD zt2`i~6g==()ee*b8AU7)ML@FsDO#-+8g{x1$aLy;FijUsPH_|MRhrJvNfZeBRy>GHw^oGj*Y70RlNj=p9TfD3fPxgu=HvmA%bC#v-aVfTX>~&%w?y zYmsz;ElOLSpg=VR9-;s%#U{nSd;mYwdN=tS$zLG&Gn=*NuV(T5m6VNVBg{F3t^g1n z%#~ne!;8M=atC@HILv`(-nDr@bzqkR+a36(0~;)e>TL*UPlMq2M4!3K3I_az1?zq0 z!wxLBgu1LWXCnBgPViC(F0?pRvqR2Ii`Jn@Un@ma7sCn(|6K*DVbXt>0HHl<)cn^7 z|22o&ijd!;q5R}HeZj#(BWQfqX}ML)dUlrYfJJs$ZvYwnZ$R_QKvo&#WAL^twk;i@ zYW&)H&&K~(lt_6K5`HCt<2}*-_k>0k$(sx zk-1FdzX6HFSXL-0#4agzj-n5~7tAcOfJF{m?7;aBoaw-+4!l-E)f>{`-v!A?`ZH{~ z42W6e53s$$BYSAoRd7jw4UjSiNv8+>6+CFhXa2QA7KS6Cy?4=jt)WK>DYD!V2^4|T z5)YJVc&Uc@(zS{LeV9mkpxGe?rfRre!-j*|Q%S+LuN-Y#6|njlV;qZ`82f=@mNkoLW_c zQ9NZIUMIAPk{&pvY38w8%40J$%Y7PsyN31dv_sRw1#G299|B}+^kNO`U2CjH&(LtA zh7&d0yc&J0hT}D!G>u-TVcFlxV&l>1Ycw3K=}B?uXcgbeW1&Y4mrceF<=uMCyAh5= zV2_5QG_6S*eY1x9HQc9pceh4ArQrk(_iFshHF}SR<21ZTv!_d=o0^_Z4Qm4+G5x4= zJ6e;?&~)kpsTNI7n#Pl);S>!=X*%OHdV+?#G@V|J9^v4e?aIi0jk#A--lyRu8t&Av zso_N$Zq;y`hI1Wk50RZw;2bR!9gHkUn^i%Y;e-YZVY95VH3G}YCDnxmY8{3gyE8=Z zC2RadP1PFBC-%^Dg2RxrM>(q4;S3}?41tvb(-@o*4eTtCS+^6U2G#9VO@wjiZR*}u zLxbmlj9u(^ehZ`xPq7}TT5{SUJ8UDAzl3#dcj$Zb`HOx?_K(d9{38@NSTydAxJ|M3 z*c4l#O|g0GAR7!0vMI{J%zT-ce?;mZ;i(DDn@4A;V$Ed97_32un97lUvMDyO4g5P4 zJE$}CfzN8|Y`@E>RPoD!o9rM7Y;$ZL8(7R(au7pJxvX1#PR!#`7^577;7oQaV{C3_9rn9<{v> zVLAl|<~#L_c!EV&<9XIrRI-A@s*>dZ8A(S_$!-9$$KO3Ft9n(O9z`(~_&XFk$Oav1 zBL2MMGFj0|bC+_3Esh|`V~c}FmZASs6xA@z;!M(&Z^2D--hrDg1`62CpHrbwccne{ zxZcW>fZddH@Ck1d#_wP`ln-r)~z|QC9Q1t}XGP~5E7)rItpzviRE>>`^9kDif6ZxTYkHvQuvv709uDhZIO2+aGq~Y>&4np6v=W zE3i!gcv|YQbK1^?Iu&H+eYK?6zTtew-~J4b?4U>90@6Ja$ca`O2Kxiam+|%t(8-y~ zhpBVdQRrN!>D2rSAz3~=p(}G(a%{KSIcvMsj@x#t?HAjvmS60I?ST;&(uh79@#UK$ zVg3q^!F*;HZt*x8;9clLu}s<#y5ie319>1vGPW7;ucX-a*rwXR_Jz#@!=#i-Xh3_! zu}4z@A1qLQ++hK|^0{UtQ#9t?in&xH?oq1BRH!~K)t-v%jyIq^@NqMW$Wih242AXQ z744ZRXR~6+bqsI~r}f82eYiVOYx;~|@TF1ZRBNFuBcNuK8Ju<2cJ@^#{H(JsdS6BtWX(CNY5spj| zCIm#Cm32$i^(sCGp}JoN4rB8%RQC};Y{vqw`UU+~lcOTCt1T+B3_?>QssLx{B>4Xb z1+EYl2DIvCx6WE|4FvR70^9ZzOn0e>R2ytJ=XSGdH)c*VS?t*FYT%b@biL(M1yE~I>;fMR85H=d zn^E9>$5G%f1!8%(O9W?*2NZTop;9u7-Sme|(9qHe4etP9{u~puO+cKl?hL(w2Uo6_ z4!ev!BQ&~uv_b2owrbAKZl>zEze4pIsuq7j|J|1UuF!v8`oGXSk$Q`%H__HBwY$=f zs?GGU<#4*3JZg|J6)4w+?2IYYn+lfK=?X|G$mj z*2DGKzE`2AmU>Je`l~jSwvdC`)=&z5ZRXuzejO}wEo2|XxSX^xu`#yp))S%O`jimy-*uEPtewNvYpiQeyw;6S&dxi=5TuT+#*3GK*8j zVVh2-pOEPknQl`|uJG9mFMc24>E{hkW_TOJFF$YiNQM`Ffbg>oRwBi^Wt3G6xrS*K zcR`O2y_^h>k(>~EYvBo58efF0mO7TA=Rh*nQ1Ne-V#|g9jsC@s{#8Ohb~`MT`Z>s$ z>yC=M;%73dW=1ucQQ=SwVp;p@BI$L zN(og2ECD|T$w*`b3xTL#51q}>-3(2ap;r7?;m)!@-hrbfR0^0+BS11tDmb%G*>7s= zqbA#{kmnAJW4W+-#|DfJu+Dt?c4WQMLqF_?w2>Hsp64t*^acOW--McVcDAxx$nGP1 z`2cqCWBv4>S+ZoiCD-LeyBR729d?)l8cA+HnMRG9sqtF3#`Eb(lX|S?*Oibj>!GKf zdO|?3U$y_z4$=OHk|MRe$YM9rev0*o^h+_3c3zacoa)1~2+bGmmk6;Z0edkCi0;=3 z(Vjv?pCdz*)7j)>w1xSvk6`N!PNak5gI2!(z`=ES=RmD8L}_J-JN*i@E~3^CfzEHg z=)cZuzY{VrCK;k)qWuBJw2d*XWlZu<<(KrA)n5uLMSn?TXBd@aQ%MVzqzNTxU$n+c zHDoGUmidb*ftTXFp$mWPrAEX-N9}unjRVSe?`~roGDE#yQ zVQ3KY4@n^u_2BnTJ2i>cl~Q;+g^Pr+oml40Z4B&W;A{~HeXK7L>8LShq;tCliX zhEgD^7Xl|<)Rcpt`jh;ce&)$xFc~RiZy3NX`1`7?{7gAdem+6jrIdY$vee(K_y^`c z({eZYw~@bq{Py!J1z*t%0-3KTa|@ZjOJ-sJkx<%`V%Rdkm>TepAb-SX;6Gz1elz5e z{2}kHTDqAB$G~JvB>U?Ydy7Ai`xWL*Ce?sdzzzF0VV*6tH382|fkKA8_}J%m_EeDq zO^-reJ~=%^ohzyHKI&w$%x8rEEKReZ$(6f#l-NayOiH-_M6ZnJReikTfd;2lU~N-MB*UY=K=|1gRC));f8d*1 zq%1v0z-&}gtWk+M>(?_$q}SnsBj(*tf^3>9g9n@ zwAUi0+ZaWvQ_H-R97ZU=ynXE>~?u3@MDEmN6u_Vu1ee_~k5AeDUJkb0Qn`8g&{Z zBlb%Oeh9?&*y2}AtO0M4DT7S=$)x=a=4TQ4E%}#||FHr5d_HbFB_FF>#jhunhC>9f z5hPSE?y)%1ha*h58Tkqk&>MP21el@Uh+l7LxFzt2xUW$H&7n_zuQ`4f47rqsRMU_P zr6J<6;PmfZZeU621<8m%1IA7uCxN1VHc9>gS&yl;&>N47qGx51VE(1UjHg|zU*eXR z&<-+--QquUc(mt6yY9$}g4H*xqMR13iyS3$+&kG42b|!*Z#(c}2aa@LgabdXmGu!C zK62o@4m{$(0}lL?1E04bssmO6Eki%Gm#kzuA9nc59a!p!CfK6yBEx0NAyc1WhrD8M z$!5s;mIj?xM9+xPPK+0*7|*wI*s7vcC7$)E^I4X7mWNg#y*$YG!DWKQ``~i$fQs5E zk$AdvjVIWaJwkY#!*AW~e#vUuqop3Xrjc3ZDhC9b`vJ z3x7qPX7czI55|aLjvudeI+dz4P{6$cJaUZqBg&nk+yW)1BCz!zfkZm_7JUW2WX0zS zzntNX3?FHSpYQpInI*d*BGC`G2!KvWQs4|%j|9R#OR)X$~vumK}cT^M~G1eRUrKJenSEw|Wp|owpiS#S+r$(`9$1hAW~lBH_65;aai`EMDP&d7sv=ofG8j_^V@k1Nl4i^G)6mm+ zr1`t&D|C&~o)M~8CSiW}e<%n!=(OufHeJRKttJ=IF(}-xfhT-b_lqqySc@DnDmw1MQ8&CtJOGv^KtHXP?|=i zp8zQnup29sb|c3{&7lfB5PsM zJ@Lb$SVZwwiWgEG<1KViGXV$8^JI{(pL{8T&#~WB{;>N390OA!G}yh9-P77#i{1N( z(+}h{Nqy)O%rw9gsx(5;R7;UZ#bWgy+VXf^Xs-}Mci@_MV7F5F?=U-ID9rvj5RA5; zmNb)k)k3wE0!%JHQo#b+HDE+&9?nO2Hf|{Sus%j|)9Xc~!7a!I&BfEvL{nAtXwph>#2E>p)|7 z)*4!>3${S;rxdKA;2lCx#W&!QLGK?i^l+^wSUE4p53X^Q(bY7)85yH;8ODP}CPms9ee_ zkw@WD3TF-wh6aB{*QeOA&*QYX->Lj*-r^9M4*XaPA%TD?ZjPV?sl2Lc0|~mn{5Mr^ zKLs*&GnRLOEN{s9uh}`zCk_L@&&4mQr%OABCD(yRHa|&}^HDBN$enjSbSbt$IAwun zRDAX{#MV&kuRwO{t@=rRJ2lMD{ba8p`*Ut~)Mw*=fy{Ew0U2>Blvz|Z<*ZvCJ5 z*EWO?W>cs*?gA*jS}OKuZpZuYz${N2asHKrs`s>pT0=pV{_5KW1#_st1Y&xzsfXzm zjgHJu?Dt`c#J)8>X=-=!9!N@UzoWKlYWt>whSc0pq5*K z7_o?NrX_@^r@Dn&AvB3XJro*m327FK_=j8ah`;~0F4NhBeGZb5;DP-6K$@%_f0Gnq z((7FERcqwH={c&%rHbDUP$ACgddQOBh4ee{9tYkfAp3>7Nlu4SRV=mVB2<>ISwKc9 zM>z!~GUSUaYiWlFz);g0c7>s)WkytuT85NXAsg zIUUHAL!x|KWl3YwDBgmIf3$>35g$TFfMgg{bY_Q2Km7M0ZkT?+4hfYI#}|JA$>^cb ztCo;vfr$SZOCFJKci^KEDpfn6stP0{X#`ZQ2D0<-ZbxX51LsMohh0+WOznGz5-~FJR-eY3Nijg5-Ov8VbM*`2i2HL6*mLvDrEjkAB7L``Tshn&ds8m-}|+yaP2_GcFG6lFFRkqQAe-lJD`E%N_Vn4$ODpYzJmYsN&+p>J*TSQ;h3sAYmUe;Sx&*Fxr7b z9QehPG67)dc3`Ii|Kh-d4tzyIWdKXG0g{m&4FhU`N)dTo9=5}swijyPNLB1-mC z^4zbS6bOHQ2!iH+4goqrGLkL=cdN$D@{dMxubZDE&|326l7Giw{)Z$#>*pG`Xa__~ zD7uNFce+G{lbK01-=lMozMCvPfD_$%nAB@Pl1>^2WHeIa7$7IJfT*uf=-cBnKmDbw z27p}-Y?n}#MwaroK$5DPfecR!R2x8+Kb~<*Fi|_mH<5hRf=?MC>%ZjZ(0{oFQ4{~k zf%y_D6X@VuKr%|H@FpOoPOKl-+@;kwj*7>TcNclb=)4r-c)L{R=M&b+Ux2@r{O0!{FF=aLhgPgC&vGGO8zC}XCWy-=z#L*qv&#q zW++i%C*^A_`2;;art5MF?V!*|B_z^s@%P@NrK=A7-P^!_g#5>VZ2j35f4$DnO7ar< zJ)^<@A2vTb!6$IX{sC~a12;JEXAb;{1D7~3&w;ZXc#{JsI`C=-e#?Pl9O!Z2S5MfM zop9iX4(zZXa^a9eea(R{Ik47&Pdo6J4t&Ic8zt0B^jeUN3H06yAgk1g_N72FBWquN z;FL4J!Ys|EP|7zTlqQ6PldqEeN;!*K0{Iq^&nx&W^=Q8^NILzm(hJxc)gz%U`aVcT zHAVjdqy&|HFe+)FQO{GNjuJZuO0c~3Z9pPe7}g_HE7wd>-Kka#*{9DPt za}a+8hp;|#BLWpi23AAY%sQYu)l}zn^dk z%lR+zuOxrj0Dg=o?soHw@gMoOlixR(f3lmOS(ZZnX7VQt;Gc;7k8b`}@JEpU6#07} zbEQ9X8_GXONPH&71uYV)O0!bt+kQ}`9}FNP>0;=q0fPTCO11WWKe0GT=8SR#2(>RQhEtLHHQB6{+ z5bf_fpgWl_UU+H+$>^g{y-SGY(=%S{M;XITGACXF<}EH}mQN<~K2SCNwGdfJkzEv7 z;t~=3GbKMor@BR%m6IsiM$z#uQIQ}aKirY;ua;R0Ri8Bk33^ZBDAR8!KAhrX834*c5r%`yH@C|N|w+ms~tX_e}HEb-IHx0QUU zicjSM*O1fpV*_etEY812DD6ZO8)ZQZDZUH{sxo;G6rKRdXrscv0V!Sl<@$BnpPdb@ z!aogUkGdS}&ktY+Klan={8Yb%{3+xwAI#qs+E%Rjf$94Z`4^FY!C?N@&^CRCEVDKh zT(V$ZP03A^yi`hx_wj}UTYl)@ETI?mUqa;vHe4TpWHeIzD3DBlYbXiv@wqX9@$$Ux zy)H>L3H+a+N-K6!O+-A@Y_ZhXF7;RN#5SK-@v8%4lvB0$~YWno%w z*;utiR>bL`%9y4A88aBu)j%S@T0^B&$8#oQT+*{@kV20wdd?5gQi^^8qzn|5?i~=U z@{JD?Dc9|jQ2C2R{f{6Sd#K}gK-3}X#}k$ih60-%xWR!xlTfK*oA^_Zj6SNm14x-8 z+uxaPb{44&vL}8M>?t}s#y@dlycH`&th)tr*hq<5mxSmqo^bQCYFCrLmi%P{_&NS@^X~@#67qMD-#36CPMs|I`Ft?J zC92xj?@-$NEl7XECC&W7`2KXID?J?F_ktvg)CWMuRLZ_h8XJ!$H-8`a>&aC_u38|+ zV{A{`7JT8fYrSCJLguYxE_X50zu>RY`B@fzME-j6FBrtn@3Tw>mE{v~f&;(pz>6hR z<*N=lV?Z*Zu7J+)Ce2{RpQOis3cT-u!$=|TVV#$$6qAK}L6!4|qqsSkhkEQ1BmsXR zp^AJv6g~!$^zJ4gqnN5ofv68X+uc%vaFGS^lsZpBrH7sPERc*^N?#8o?1BB;J{sWf zab)Wt+Zc-5c7@tZ>3;4^(zmohb2k@gj^n;n1dkccJcqo@mxm}b?6LO+_UM2Zh zRO1~u+JP5Hs0?5phaZF@lZrkCvJ*(BWDnI|(xWpDlQ;I;;BC-(Da7*AcUDUr%ll69 zXOMr(Abt+o?gLdz7u&Oy6j@G@B`y)_H->2T(DvzM-bUtB7qey0CY_)5j3Iv``6CDM z(`g@82zv-S9C*lr$ne)BRQ@xd^A(T`&v&5nSs=%M9Hc+tW~T|&WS>a(GM(LdznLpd zKOxg*GTo+_m>GV4TnVFA0@jSVf-!$(X@7o&b{NBdpoaVMaN5 zYRGdhdGtT#cQg4pe!B&sS~;O;HbpxqnkGcazfJJN$SWj&CM=6}i?RhAMbWrKh<;UW z$13`(4{^_QBI`OFc*udTIq)S3Rg#%)dq6U#Qt>VzJvD4Ep!6X}e2swIWw+`iyUk;Uq*;s)p8%Eqti|}pfg2q7GYM4! z*x~;aB&q#QAY&!9=K(pY1rqJzWVbk*s1yoqr%-}RNcexa;+0VQ&p>*T z#W;4RTfA6`-(ZWUQhW);#|;pNeLFD!BjpME%VhciKXc#*5-R($Wdt3dO8eddGIlY7 z{XlN}FyHfx6z9~>b_tcwj=*P+Q@D-76$6BY|Bj&lx8wmAx#ephp9hjOV>XZxeHApN z0|_%^|K}EB7m+~LG_raXtK_%J_e`yP7s=#)yk5og9+0t=!iRzM8+&IcbBtdSBl2eHg<>B-?U49lvXuTjw;SYd}2^4;tG(JZ)Nq)jQ3vTn7 zzjxrjIq+8wtd_7MAWR`&(Uq%`zpTwVCkvH_g^g7i1S|@I9Ebds@RLo1W6|A1|TDw@mvFB znJC)3OWb0t4_-2FCv$|u3>h*0{&0;f$DQc^Bvdi7%l!*TMk_^M1#(;=kQfg?W6AfR z{yXqd2R>*)46yE(P({E-wHPGh6eCy)@3kO`M3F;X?7;aBoaw-+4!qWZ zS32-=2YMYi%z`s0+y zm;i|iO1S^z^RHWobz%ufR{m3bImNR`*64CCSzE2hb zw7WeP+=cI-NT@QKz2Z@jq{>zxqn;}3fs{4WhyM5(m!x`|=t*063x#_qT<#JU_AvcY zo~~Hrz@czKNN$frNE zt3sCp+a36(gi1GigC>xSQtI9dB;q$7a9&0e`v-TDr-eLQ$fN%;{!Qd({69sgRv4M@ zMHG$rK12(JDESK&|4nXwCN+)x6Um=s^Cv0(7&kx7_mIDc{Cy=fjr#jGX!euoJp>tB z$#hsSX$j!-cr*D~nVv-`$A5skENJ5UPZBDpaR~evNJbkKJ^-Yt6Y-Y{eelN}mJ+~R zw-P2bhe~2opyUQe3GGi4@+dr4xP|D?I0|J`$RmWv?@{r6iYE!$kNCPA*e;>6pHaUB zlChau_5-=?H$z65=0A2jJIQX4eairLk={)@KaISX{9WY#(O~|bP$Bu*+1-Rt+KYL@ zi4rO!>!IZuYDu^jTE6LM5s4da$swKn_u5JR%z+;`@R$Qz9r&sP|LDMH9k|PZzmQNj z?Js7%%|igN@kJ9%%AM$XMdSOu07;R0CJ2Jn;3?>_|QB160;C zz!nGMlhlg(2M7L#19w=k%x6CCQ0Z=r~f+`K41!Nqd z;nxE(y?mb96p804<(3@~7-tJyN`WO4A#kA-@Mk7r{pY8y{A`6tFG$8_GIjz{v-}ee z_f3QO>&b7Bzjgrs1kC>q=C3Ay7x~Ku@Z(APZWq5A&nyO2CVK&ey^|n3-6aeY{h2j* z|G_0ci!Ml@$W)4WT_U1|~BH(L1huacx;h`DmUvOod=Ti!)zeq6!oM72_yX#c(B9sU80n; z?k;xSdE}icRZHZ|!?_ho?I|E5~fi$X}5w^fw~JX#%RO8L~gxI%=t75q0cv z=@5g1homsa32P)&>1#y#N+?=Q(K}tDR{9b@W%_1Fxr#aB?gy^BB-h<#uDfX0-MQr= z6%}5grB~hsco!Ns8Pm{Kz&kFs1FpN5X;h{^Gfn~zm8f_zKJ*SkzpkM;zG^c|2` zACc4+33ba}13*4OS-+kol zCufGmDP2N;w4M6WX>T}O9Y^M*DPSJsVy1rZ*XaE8^~oQDKbQRP0ZGNymFIyceZ^A% zT0;r=RqdLZbTuF#%gKLH$tEiKwWdUjA07h1^5+@{-eW;{=`IJ}=0Kl>%3&;KKLp9x z%}AyIF+G9Wv#7m)ISCh3>;d$dmr60~Zlvq(E6jCk&DatMF#Vv)M1+8hgzKTd9Z2}g z43$#3Wra$2JsIbau~sr#{u1Tq36Pjx6s64$+#sP!4xfuwgJf)?mSsRxA@*+-hrb9UV>jbEcej(t!pILnfW0UhH}syflPdPPv?0id zONUgmkP`D1doBK+&?uds^{R&abI89<@^imIVd(Sgu*Hj2V~rFgyhlQ1BfElSl-@+? zTZJ_J(TVdfT%1*lWD|7*nfH+S8jG3!t6(VBKiwjQQe*_EGNU6X+)3fyyBu#YKGuJb zoKhmbs?`dT5j_?B^=^KdzOhW7N(rmf4)UjwzuL`T!3j?6{{q#CqAE^~^p=4t?OQ_m zrIh!%<+(q3vRjz#IHJ6UBGnZ6rXwQTKNr7vKP`gHb!6_n)0O_5p7MQ(pOB$bUUxk3tOVJlxqM`<%5K;csIm~agpPRwLssOk_LgghkRI90K!VOTh%u?mA zX#9%)$(793U6xCoL}l+s{4Hw=O;f0$=FKa z_kna_U^)LMg~a-gggLwEnk!N?*KpMNJA_J$b^sZzR9X$>DD^Y{?vqL^PgK>)%wJ&( z|A@l<6kgyGracu=qI^$80Q_1JE$*&#-Nnf}rGC3qe-WtCwkRMY?MA3S^&{OjS}Arg zy#q2ctp)v`gi05xJ3YT2RH@4VGL}x)-3R2wys#LTe$Vj*eik<^P4gi%Wf}pf25jp=!wCxVp{H!1QyMZRWQWD+suk3_cTzqv&k8x{X^pdc&b4h-s(xj+2Jzokg-==>kSlP%SO zNt^rc6KZoGR%&OITHl4Ksr_OZx7!p1w`SpeS-8r=KyjK7RuME-xSUk)0);Z_FyCr~ z8(eqfhp`ErusQO>p08TB-MKrQ+khKZhRoNGe~UEND$TzG+48F1j^mR(K)z!Z29jvC z_)isURIxrC#0tfse8m6l>5-l!Qm-PNR;2c6ku={Ndm%f|G=o(=ELN9m6m9kjqE*^x zrak`Tf=SDp#Z@Y04WJ0={?gqH3_R0 z=Y%-pj?>B4_x!(W{Fyr~%7Lc|Bvu}wR#BefpI7{;cLrrM`N_Wjk4X7s-=BkJtZ*gF ze?E;_^1(kA3xk6*;U@^R)rghtx$5K-C{rB_iJrbQX(Tm2s@PwliQ2Wkl< z)>6efs#tfpSdwSULY<$aalOh$63ofMe|?9o4MKmKh5wL+r?T)bv+%1~_&>7nXcj(^ zg`dd6fh_FJ!tN~GnT5Y)VbCI^`Yi;FO{j#cL23_U^#AF6+K-Nl@6lE8`G(K72G@SB zozAZ==U23EC9W=gMrzbMUw)Xji+SYy} zXslJq%>+e~LtZS{KXH4g@&Zp=7*v5;^IsD*HloO114ScC9*u8*=;8wpW#PvhL<=9b zFi2Hn&Ibq@JEK%npfFWC|8X&qXYKTuY7#3IbKYuVmPKRQB#gayjf;b?5P*!|19``VR3Fq}H>`mgXejD7^Oy^d68!_$eO#QU>uOod#{0^b6 z{e#9@mBctGo1wd#rM_{zu;2wiS)lWMhk1!_ofIMTck*npltb`ILli@)2)>^$Nt;| z{#j=8yx4yf9OZqGJI4R`RWa8#!w8!P&VKOh5MNh5zH+KMc75#l$`^SvD86&nIWGq# zq(`oQ4Q~#8u|NNaz~*!P1y2gBJfA*_+dPnecGIh4&HeeK_)4B_?k{)>a6I>Ee);&b zcRY6tbPZ@nFF!lI9mnuzG++79GjmGD)63?-zH)Nn2FY*+7G%g+=d2vQk=UPK4b#Bp zGyMg%0xL7=)pFOJ?ayBeXdWood-7EXQBa4kWV|`O5%+lRCVovyxm!SdDms*5EB?$H zC&M@T^G}O^T?k0=VSa{RKb6J)Zj_zb4JxbD*d==kaI0z=&^EvNvDSjE~HbG|yNu zPFUxpoPl-Y{rSBx4Qzgg&wo^)z(a!mf+6tn+{gLlQ`=O=K@gP= zIs}>p+PRlH8NrjO33l<2h97`9< zU3aWMe-VH>$^3<>lQ_PSAAt`SgG#=!KfeU`cy1}bCe7RmP*feu@Mr#*(O>F#w!h$* z_*afJTqk`Tv2r|VouhI%E1w>Rhj(80=RYH``N{r*69OxbrccVvKmSjtBYy&XJokBi zZQDErk~)I6c^ZGFD4X-vS1IQqtBA*KmItkKNY22@p>zX0JiXJO-z2d4V1L05ft82S zt+;fMb7V~iX~%b<;NS4lWgwZg+IJwmLL0TnEB15yzf?NS_fjpKicB2@i`I3k7a|V#-wg6!<{?3#RH{02>R1 zi-qFXD)k&Fs?_uNGxMC7SmLKxMA#=iekT z(BQJB#t=y3vz_} z#G9TiI4I`+{5fD+;Tb!nh~d+e;znGIBfnO*d7!9l^YLeXeqY#;4~f4|`%pu9>LM?K ztZ?T-bl#zXVe$1Bj1XtQaHfw44vMuu|4HFtG4~f71vfuY@p{ZodK6sp4b0C|08Apk z&Lo0lb%J5-82-#VoNNrJ%Wjrzx_6DgLDj3fHY=NM>MT0z(177fR}*DmP>h&F;bF02 z5{3K3oL&p06;3U(CK2C(f*0^$62UTwIG99!O%}OZKv5NL#h>{cHZ8D);_sUv?a2>G z%;!N?xKkn05~#ukhQ)_1B#u3Mn?55rC{}Et@UWP%h2Z8V4z5xwoB|saz5u`g^6Lzs zLxBp<opflx@TGf+GB7B{{``Hy!(#0(*e~2C z=JcaLR9L#nsBj3MnNmLn!0z#DN05UcJE|NC(MX7nIJC2incb6kGQYJU=rZ%?N$gSb zn(`&Kb_}i9V<9@`klnIkqDP#8VR16o6E=@yr3(cI#mtNkOqZFkT_%oCU8WcpvTOSmJUNKc07s6zw8x*zN>7$?qv;Y>dxI4IWs{1d{% zV(u?E32uJkw6Qu9-~;*JHCE?Yu&mB=xEM!%t(d1k(XsC|{>*$Q=6HJRYbE6nN!tEyFga#o)GPIh^ZuqsU&fjN(vVo5$+Rj`boh-F*B6}qf^f|%TavN=~K8^ zOMb0R$3Rh?9>bqm3GUH-T&cwuES_Lj!~2>(8Gg&cLuZ`wPwp_lY;15gZhAfBsqF zg69#szhDZC`Q-QSjj-oH*OJHa-18yw<_BRpA(|bcIU$OLXl{t+g=l_={17b&QDKM{ zg(x1P(hyaIXjzD=LsT20)gf9NqPh@m3elDjZ4FUFh?+vQBSftsY7bFYhc@?g`7wM43cf0B7TgG?>Tkkf(BapLIS&*Sb3Xn|)2gt}4~f5Tiaoapb5rvo$clL` zMCTo1Mh;>|P8??B;Per}L9sF;7akTfGjeeA6UQWGdK7#h|4WiG|0%HQ{ByVrH~iWb zIu@d14w)7@jwkc^l|j`n^Cj#gIgIa>PS1d>u$d5@b%?VAAXZ68R22E_7`M?`^20+3#0{%*%mN`??Ay9Obd7c zObhrK4xK028PAgU(ib& zd)PC*PjFDI{rUTahsE4q@F~#Kq+DYsKsVMaA5UKXY(-Q0M2x z-q%agQ%1^tAS>nm5IyS9z@Ye8`B)%v#L3EsaG!Y7vjqpm%*qFtI)Bn=vml0VH{oJ&`L$Bc14X5rk3aJQ%YAOO>yY^S+DW_47_lyp6|*NqdmS1ueCZLQ3=E2q z9Vg*ov9jqT+$ZMrQ6P2RYi9NNqxkH;697BQugM~p7duazupGmm+3bX4-)W&_Yaqc1 zEvNT2fvjvhLe%Qez@Ydru*4Y{7AFQ)xKF(4)q;a!#=wG6;SsCCI(#F4AQf%|83T)p zf#ugqxdjxRR&B+f`ID+pg{Q^eS4Y~DM$AnhE9RCEZFOkC@TDh-GB7AcEUWOaSh1|a zePT|Z1){=bU(_&=KZVbz@C5+&lwT{{IZ#x%^Y}A2I^md^-YVIumAkg8`f7#vJKncC zL~9*lW(s0vN*rdU!Ug+;4R3nC;Gme9nSxQ_h*jYbzLCRH;bS1F@Z-3cQ+}ZG8pE+WKW7st!?Yh*pPatwYAv*Wt;Gyfx_e$M7<3C4a9-jdp>M z(Rg8q7CAI9D8ByuapDXNi?hGr8R0(hrcVeCin%}kBp8h!vKpVjH}ZSZ__H8s{5f2V zD8IH)r$Dw)r$aOuqB9}Ngy^h8rchIOGJnBZx-%173stHZqD@8j7AsqC!hK>+w*sl4gQigV?f4E9ddwpJmO5}2o8#swH4uEF|)P;Zhqo< zwWD+lY?dQ$1Yl|TwR)Zh(v8CL-1#od%)@*p`4kvUehvrI%deGoEJVi~ zvQu0TX|=F&hoqe(i(a!mqZ2_^TAlH+=&VBn!{TFMg*f)?P`XxdP^>Jh2oH;y zg%xn~6GM_VT?amp|Cof$-w4)NTaSwg=GV67Eg&oR)(|y>s3}A{Le%PzY0d3;GXGu~ zbkWmznF=FGyiZl|JjfR1RETugNnrzn;$yLeI0M7tWU)oKPrT`j;GmdUYys0n>x{W6 zn8K%}tQT-G*!-F-a?gRHrtm!eOv1^A#`nHO!s$Si-IRC+WQEf~CyP!x!~__`1eiEX zfWhf~f`ejZ0xUc%W+uSk<|iIC8Xp25$p47Z_+wx*0mj8*^J~RC2#Sh%2!G}aY_!QB zjHl->l9Zz)?dd>9-_t_;9q$_p(J_aZ(Sn%K5{DVBaKQp$!<#M?927G%S}-bHZB-b@ zXBMe(G1&ZCD^LMaYY6AD_%m;CvZ2Cb;_n+F#VNZ;4YI;L8KR>Ou}BSK<$yRWQiIda z2o8#sl>^~nF|%?2Zhm5|u>up|1NrO0tQ>%)!sl=?!Tee=Pl2Mw>NNh$?j?Z=x85l! zhe&$ruha@32U#f(hUkz(EK-A5IUo*;)WQW#!hPaR?+_dmGb;yRR5<5H+Mg(B$7dF) zaWSO)nk;gAK+)>mi$C*sQuHe%+&+*MZhwd#b%;f35QcQI1QKUDM{rQA z7*gS3F=I%HKYIxDZg&Y3W`?meEgXzCmXjZhs594PJ+JK*XaUT*?K~> z*CD2?p!5h)n6d(~pu)pq#exd=i8*}~hy|Vcw%OMi#b>9i0PH5eR=6>Tg0^=If97ym zP^ruAl57nmIAmmN0$JI1gs9aarmP^Qti)l;DqK)2+$Y}jYQaG~Rx(5lA!>5SEA^w=khG?xr1BNf%OO$~@F|wT}JS@9WlSOU?DB7i$;m^!=vdJonzt1Pd@sCQl z1t2S2VTcwv#Cj?St4JKyQ^DzH1P8^6RTLf;GgcAY{KV6rLALY+*i`Ye08AmjR<={1 zsBEY4XIhJ`-l*HH#gZ*Xf zF1C|jx2p%TC72%~-yyrJhbQwwJZQ0p@G=|yzv=h4NbDCuR_t>jI`7cHp!nF;BhJ9E zIN8+`?h|kNNx?xev#STD#qKkjFBru)@?r4dr$D+}`5Z1*m0xF7LD8Y|82-#!CnfD; zS*gUFB<-Pnvp{D+R?JL@&N{?t2vEA3D4d1>avDN-Sgf3e5bhImdM%LlQEg@$`E~eA z-`@zptnzEJ$lU^pHr}oHGkzr|^wDA{D*>k_vx@ixuUUhQi+CIgqXS^C9xeg0LKiOxvA} zC-c+AVei|EpD8EVdWC8P&j|5%yzfMa^wl?oaZm!3?j;HbC4g8~;bF02S%v$=oPJa= ze=jok=MRA~r~LlC5%wU+?mZp~(MX7ngy_i-9SzZFh@KA7Scr~=XgoyEgy=+wPKIbA zM9+ulRESQ8Xfi}+LX-*7*$_>I=*1A73(@%ydF6o)azZpaL~}wE3(?#V&2vacZ|8%V zilg%CJ5+m5gKX`Mh3J?=1BNf{6J=mfjI5mq4~vzx6X8BFrxyX~8$G7^<;U?ID0mnT zyLw=3X9*6&55KkxSAe2jcp3go|Dw=99ut4xhy?j>67z@|4UV zv2}^W9wM))Vu}4Y$clY1M28&WgNdMY15pMB#mLAcJSRYu@oRP31CrdqWOFb6Oqmmo!*ycs>m|VF*&ZG!5cV(4aQCiGK|uS`HEpr@e2&|}a;&{k+IR1Vz& z-3;YGKRr<3or1=pBhcf}BTyH#6{>|6K{rFQpds?n-;4VSyt#cA|KP*q|Jw!L9B46A z0o6k5p)F7Yv=i!r9)TW(4nl{aqtF+iXP^n_G;{_!3%v-Phi2_B@NR_WL$^YYlV1hy z3TQ2~1=KHAbLi$a zBL5I*N^f((q0c}=&`#)ns1&*p zI(t9mfPN3^fi^)E&}`@j)WrmJ80vxUg&v`O>aT7ueT=sCRp|SW*F`&mRznTY!_Z^U zXP_@Z--TX)^15j=P%YF3eF{1TeIGgx#U3IZ=x%5e)C%o`jzHgprl2{yvANJrXdm<> z^d;yt^mFJ9J=7tz5o(4$3>}4@g=X!cE}$*Y`=QT5&qH5$Ke9mkpq_8G^dgk=A!LIVL(8CzPy^Hf?S~FRN1iMKOK2&y9(p&_0Udx2Ltlb| zzq*HC6>ETcp&;z~i5p^{{q_yU-?gVbsqp1(ZKV&e{Xzg)BF5Q&7QYD)n?MEo5ns>WxX4#Bk4D#lAUe0b@|(p zt*N%{epkxx+L81-+mnqg&E)5|C$=XqkO$eEEPlt_*r)#ThS5kj0jAlJ=We>XPLNgC1 zJ5r1MuH7j=)q&DFl1aZk)za40={FP5m}qTHb$3N8V0LvFSFs|@3BR+eBhj*bN0;B) zLe+ihcl7sXRkaTx|DRgjO3#z{rw)6u@x@+ja;FztpXxI8+}PuJje9(A8}2sT1a1O1 z5F**xm}pNnZA$iZc}SRSLoV9Q_BO;*(~*=djVXD(E-~>J-#s_xr@h$d5-;XI?8QjZ zBS|m5IPMwI!JRu&yK55dYN%c-ZFEJV@j-$OUbVZ4yxNi-HL2EA2eP!LI@NxYZB6xE zds>a$on48JF6T1^CvGqKs?4LysFQv#)@Y-9+moqXN_&Fqb;#x2?Zx6BN46ex^54B! ztjdek4STWLjzqHxK-#<7aof4x{#h^9UhT!Ie+S*+_n#tt{1acyjm7rZ{Kl7~)8F-C zoqKj|OSMXO9Z99r!1WRM>sDC)y6(;$buDe)DB;zgqpi5``u1ev!6XHJm^MtfU*q`k z_de~#hHkO`5&SCOxVzq*Owi*x&})%&Be^@2ZxhOHLh0}6PMBW51DPWIc44=#e%RUF z-k$29IYzIQ)L#DD1BtF=$F4-jgBP$w<8SInw69NgUC1#l|GEw}q>go$NXhZ}h=}!S z>uyDI>Vaa)ZE1VZ>qtJdNv00BzA;;O1}|9uFg_RCFuIDR8vdy9_#CFYk5z1a(!YKSLllBq>ADty1RC$|7=Tinm9)B zgx9D);og(ll&X`F=5H0{mbee_jR{OTyJ$tXIH9Y&#F|%?nDPM zo!@k0rMWiIY50WKJG&rEdJieS@T!&$!<1+=X24tS-0qH6Z$(FHw}fm@bWkpu3ng0F zlWbJn<`%SVN=rLWc6PP2Sy@#cUEk4)c8H5zFg9f)oukEO;jK$O)S?2LcqtdrrJ3mJ zN;K}+Wo%rh>VoiHj=z%%po(c9P21JEYcSFp@-}SSN!##u(+`to{7bojM5nJY*$%CX zOo!0$m<+}Md$EXdmt2Cs1=ZbcS&oN&hG;IW=-$56k6bWAw&E}LG2=7?288zZ)?|mj zE73xsw=)!UcD>1~SySt8>gZ0Y!}x_8XnI1o+i-=&Yofv4rFc!;qUn6mbS1g*+w>yX z3)2&iNvFiQMK#u64n};vf1NE;LD)|H!OZZ}|e*`yCz4DT{U*hUB{qf<||55{W7%V9m(cJ{iKhe>V*t5u8>r64)?T`)68hoYKnCfnAGNVW{BelQ1Bh}q*?iTELhwgx$Rq0t}Bh#mL~`T_I_=n!-mdJ_69H1W5z_wS*bzr&t9!~GXw)yIkZ zV{9A#FW?`Cz7Cy)o`p_9--XUVKZ2&9pF`)NnJ2loBmddBH$Zct*FipXE3}HaUA-Gq z?@5Kz2<9aAPC)Yq%@I@-Gf!|H%^!kr%_{<21K$Sdn9BWznX#M?VRvz5Kb6K>{ENu9 z7^;9gf94hSaW_1*(7jL{bU(Be+6L`_+Mq7zVW=1SF!U(&7<3T&J?MGz>?NOxH_eRs z`;ZH~vZEu_vDEjB>lZF8x*ZRHn>VexvjVp^GM`!5Lxr{_T30dOv9r5r*G*7_fAhvQ z=zvjP>9vyIQ1Q%IynJSC3>+^*Cw+`pl{3wl;p4`kYG{4RU$IiRqU%<3)83J^^DWJ8 zXt>O4+8UE@QkvPnsk`@)R=jp*%oFjK^I^uxcakP!RbcO;<5)CYHZd5-rdC>cqT|@a zs+qA#XbOtmgYLG@H1@U*JOWKY@zpjgu*rejPI!Ecy*7p0aPP%@W{j%4I6O#0Zgd=* zT01k=zK(pMtBqsr>*0fWS|yb~grf^eeA>W~{!In@jT9Xl~w`mSaYwK5wHsyz{N-?lSjW zQI^PMueK$zE5%(+O-o0k?t<%CAn3r=op^V9Nv+bW@7~tMRNkw@mek!3VOZ-na1Vli z6$;u6j~G`%1bOw)ibRud)))LbbEkgL$$SQP;!f7sdg+s&%#DrWjzber-SO*Ui+=mc z*L$vVeFb?x0IdQn$6XKA6R=TX+zGML*0>|t_@HUIojWLNlb^`0kJ1j4iCuo8%dAw| z8#Qkoaou>Xrs=IGKoz|r)zPFSn&lMu8RQy2LYbh^63Tb?H8J|d;^OA_`R^@S+W5Xa z&x`nNc(J=)LjU3V#(TK;r_o=5&&DgY>2(sP`O4`A{MIeA`85(pe;*y#`Retx9BZ)9C4NiV5M?c(lOD_!cRjApGg4e-XK z?%fS!?V&N*?(pU|t?Jlt^$npwe&f2CUN9!F+76jk;iC!djA7yts79k|0h|kWi7| z=$ML>qT%6Uh-?ClgxF;BT6NLoWm61tA-hI@M_xqO=DhggvJ(CIfYPF(CFKIeMP=oc z{>FRAq|D!)SLELd?aX`UySNzloAOHib;_;GFCrgX<}!cD;*zpTzo>k1QE8E1>K7IB z(@1Cqp`{cd;1DwLRIakTqO!&1M6i5CBv4G!M05!$#N=ezvbm}xS>Gc@e4DFY_DotQ=)VUdv(RRXitec|_jgxZ?2}%Bv6~ zuVu^Tnw8hepzi5Ksshem$95CHsNGbc=UsleU$lhwT%T8NyhRm(|0=auK}-|7!o`+X zv8^AS)^wygjgkL#*01Z87i*W^Q?Zz74q8!Dv|6DL#igbta;0>x7nMfXN`uhyY@mhW zHl5ihU9n4jo2Uy7yjQ6=_3CWe^lBxHTt!YU88<6eS%j@D2wjp5w9v^Vi%U~d>{8z* zid+f}yxy9&&ZOBu`7h8OF1FHTz8RFTZ>lvE6}TFoRJFVF-ot1NFcp$9B5`>ru}_09 zFQ@2|7#vDWyJ7y4*v=-H*|7RGZbwpWoYB7cLaTkrHd3srDGJ+2DQYb$UL035SYj_N zF`kmR8SRBw7MJ47szPPj~&tis4H4$;CHEF_Q6R22RTvSzu9S<*40BYjR?Tao0ZeV^HtHC%_R{MT36F6%`Nrws#&nk2o^H&a0tz6Y2+!%mzjDcmwfK&v$vK7=N9S?mm zCM*e|vb5Iym#Nb$`!n&2YRo$9=;}}9$x5gq;w>AJ1 z8KV(uvd|zHA+OtCOu-BKq~UFK+&D z@%>a0l}3v~bM$-au4Hi~4V@NVt}&3&$~5;ePBGF;-?`LHnp6Q(`XB zkU^g*UBUn>gCt*^0So~UuT?X0EE|T5-&S0tDrG3wy%W8N97~K;t+*viv;tDh*sLX| zN;(M3I%P^kf=~duA&zp;WEzQ!?fpUtEu7Mnl$#b9QJq=-xImev4@vV@CQ>QGB%);c zueGZxl#c&T7}<=HH5M?ntS}~`yaXBH zQQDNQxKcxR2@^6{u?^^j;ED=k-!&p2WTi3v(w~#nlVA{)bX@tB?}%l zD-2(xS;6=R-n!c8y4+pTEn8o zA`J_lZ5XCu!8@&CF;C4lEEO7oqd7)zK%DnvG1Zc(%tMRApbdjwMuv!GXD?h|OH(v5ShjD#SyjStWw z42~>!sit(w&=rQ;Q3>S=BPnTb?6H`GlZ*_^lo@ffs%Wf>8GVYeG!@Ju8Q%Dx^;j0l zz&)wEnJy+jrXp;HST@z1Nl4|eOr~NDk5;2BH+Q!tfkxfLn+WW00Bs^J#UK~5llT!~ zue!hC#{G-h8x~%(ch$Ib6ngF=Lvh=_I9V z70gcUz6$sA4EAOot`U!+&v>bs@+HhQnT@g*MJuV4+2N*CqoVbZ8@C2xGJe(tER&0A zeP~Qk;B|f`97@o}M2TeWlC_>ja~lMUyJm%Rv7(Y1`xFgOb16cbDl-PlBx}p#u!?Rj zAd@ao8uc=}!plI*^2icp?M$yN8hNC)N}{!8 zJlWc^*KB>bSV7gfo*GZtZGmw>)_Hn^&zJS-%OD@6NmE%t5Q$o8}3bDpU3c%pv!sOIs6r(5uwG zh^LZta>8#+wLQcpiNEe*9RI#DIQmk@EY zJzH<%T{>y+9f?{_LV39D&9w-$En%M>xyxPH#EA|)7g*$*u6j}1;h>VOpAzfx6oihc zxcd1<6$c_rfUU28sS3(sT9bWFQ4cF@(+QG5S)uE+8stIR5BP1W7tQbjE=>kHr?mk> zruhR-g=X7=wMSnQE83+AJ=))K>r}eN=2m4f`=yFwuuHAc`ACQ9i;cn}OqiO*=4encnQ?&y zU}bM3Lb7c;71~9tsi)b+1J+b4g$tzWRrQ>5RIxD8p z=u~AG{fbg>+D5&Jp0$f^W;Ul-T4R^uC1h>lnqC>k<<>2Xd!>Bozh(hbXY%Ofx*MYX zvsOr5Fx59J+3Yiz9cMLmLfKHFP1^Mox%Sr9AkMVuXd_icPh(k`OWY_b9e&Kl*A;a* z%}Pafy|UX~fgOk%CHie(l<3c)6|p!lN!DjAl8s<35`B^wfki_9p+#b4F-*&1XOWD@ zStJ`FDpl4Z*`(=}?9K;`%&Hrc7pNOOS#|5)N$Qr14PsYV>c$tU8&;!a?oHjnM(v=7 z>SnPuRJU%FrEYIi!VLMpw&Pkbpk2H#qgzfb!mB46%q=jMj5Rfua{=a>Sz&X0Ztp#e zRiDQ>6~(4Un6)^&D2R?34zAuQ*<^8%%?S=VnOVLfwuE3!!3d>(5eP^xo+hBK1p-=Z z4t}Vam{?acY{B9OS!?lw?HiI~jY-bcteJ)+6UVGD>bS1r@a(udp zpu=_@>!f5-q5TU+eC<5bi|y8RxUo}K#!jD^F{maPERFf#7#*GxPNtAgh1Ofc-S#4c z!kTDIW4})O7qlpjF;%b#%20zT)6}Mn6sS?IGWB6gpMjf8+HPj2Px}ydi@nrtmy-u` zphiZi^3n?Ji7CsgH>p5_^uwqI?F7^88%l$eIjceDS)m454xFYzOkq@m6gi?nd{GTX zw#JMGDYoqkR)Z{XXElgkA{x{*E~`O~XGJt9eOe7lwm^dnD^7!K7G^a_b77OzX^=dk z8e}WTX^?Y-tO=MWrgUfX>w7h#q7ZIpNu0~v9Sa8fiD0*gK zUS+E_i!nBv38R*9i`W2G8`T*bbpZ^iOSL{vM+5B{^GcJ})l8#g;(i8+%J~l9?VRamRwmQ9q=Q`% zLtHg$nN1aA{*)L89KzHsr>RtKOGtzIb5hx@4;ge#_ssZVu-5gBB=Xri4K2fSLPJA5 zvE`WakY%#SY+3L?jO8;7HooS#F5#d#nmtkz-JEqSc7whi)X}acdeT;vxUX_oLjmlO zB)M6&nik496{wZnEe0mrx#V(}7EvqQsM8$BGHzC_Mbe*T2CyBYtr~MqMWdie-Ko{2 zfee9K?FOWo{~1-fsh+{yl4sRw@4%c|?PZNJp<0EdR?BBJY2!P!D&Y&%N))G7J1??o zwKt97RLNV{(&+s<&yy~>4^LN-9_(C}xeo^?TA4;Wvu36r)TF%3bL2Gl(r^)IiXy7-Tjg-}$|=W;P#pe78yc46MLF`C z2u?Y69EjfVxDwNqvx<=xt%|UGRwL82fnmyOLL2&~;FQ*CKpUcHPJSgy$&titifXVk zx0K%Vk@eO|)>ZZV=K4DJ3wR#4c1^vvg}uAwa@ku8*8T6}on&=-*PZq-%q9T*#96~b zw#(Tk+X$_=2g2O|vG>*8$Xg!lQ=x>jHmkoNJ)V``<+f6m*Xw;r-DYL4s*C;M{9s&_Otzi0Mho=U1kF<*l-MX5qN1J3@bLd z1=-7Ng{_FLyI-(ZmfbhW^IDNJ)_hIOXHRhiIrThk4bOG2OC~zIc?f({svVu$&BPkY zzjo~g_7Z~K5W6R5L^nUTj0abOy^A2PV6FeXrEXuyJ%_BFH|+&co+}mCyD(id!3i_I zr8+y;(YB;-A6@sO8mAr6-i}e;hJUy9GV-Xcm)VbI_XkD?ZCKEbHU{qq^An~eEcFwe zovFqao?345?aRNS&yMeI;nixq&;?CVhoswkgWX?f_{fF(3vQRfzn^W>7V{u=7J1$g z;*QRBWlxOvDD%FKH@kE{YSS%Yqgwkz%GWd-+?74N2ae6=jqG{z{Qgc| zTZ4!F6w~Bt2YH9Uzgk^P{jM!{@a~nYKFQTL*?J}PTWMG61bZ?(_iWyYSVK?L1-N-H zOKUQ^aT2~Dkn8S6fV@90eA%C8x0MywzSqPzFG1m*0&OjNKwIaBQ1&Wbbf*@{J`k@P zRNv-ZrjC=icWnqgI&+lZWjK;C2;|xQ=9cY&4}I*R_xD`ne<0Bzi{N<+JD2(k+1Xpz z$;()pco~a4{4Jv)@@9wk-7PtIF&r;qp&vg)1+Mq%lkFTMurHKb!ybGO4~Va+snO8@ zvk$d4vCZ+;tb04}GOJ0nZoEe}(Zn}UVJxBSabX5}YrA)~w59Az2i>bLy?QCb<+Zvy zNnJO#cDr)}wJp4jP*35btol@Q*IKv#znVAl1=yssq8sUvL+`_^P4NOd)hlNOcvVcT zo^R*k26pw??l;fMlbv}>;(DIvH!q+g?&_t{QWlceWOrmFwyf7k!Rr+@ck z)r{}Xyzamo|Dvkr;ZHBBng176-+j-4kDPho163Qx-oQzL?^j*FYGL=xPo1b*lzi>? zx4hyw{QGv)z|#djvXKSURBQhH{AKu zfBQ+*4{t4+eRTd8s;>RkYw!Ec>yK4^>?eDx%HI9cs?Yq(v+K6S#;d;e(W2dVzUj+V z{m*`P^7@Z|rD}*i?yt@o{vk1nLx80TuBIMfd5Jv4d`k0{_BAzc1@ntM!Bsn$T1}~%9p*6@Ha}%YoS+99yXzpl9v~}IiOA;@WgsF-v zq|=#XC~tZR30*RcEmxNRl+$1A&#p1=0j$Oy5`E#C*aXxLjtS$|;YxqOJB{>S!q``_ zt4@C^$byg9`e^t+iO2QPzhmqj$A*99nppLB34`}?^>L~6T>GtZ?Yzph`zqHDs_sc( z3(0pyD=*SD6D;11vV(I{4$ICvo5L9%?0|3HVCsG2p)dXZ>+bquRsZsaf1J4GxvFik zEtOB7_3xUoxbZ(n{%5TG6Zj|b4?#X*QGMv0h@*cwT_3k^{peT!SK+ZL zv;S7|T-Em;p8MH*zW4pAV=kY%ufeZ)=&ASQOViVZ`M5ID@TL{fw5!=4we>NDTzXHV z|1Df-e=590U!W_o!{qTbu9ojbnCrSHS1_fV@GQ>R~g zIi_e6Lw|4k-^N6ndGVRb!&q{0VZ4^=TdsTdyE*OSRiAk8>k@zVhhM4U=hdHh?>Fr^ zrs@wkx!&d6c=ZVKHLTBx#V3&q`5Mm9rlCiR!#gCXS$D{cAJBxlM(2w^6`Z(_i_kWCjL_eX-km;{y!GulXPN9q7 zJ!e6Gjr}yDA3HCYrk=vqREifZ`CSQySA;S$U5(G30yApD@OE^WBER&wdc~1cY~%8( zqb40?^o{EE9&+w?^mkcVf*R$mgUKG9GPE~Yi~QY*PTe0+W6VcL(VjulTi`o*;k@1* z*O+2baiwzZXrZ;-e)+uh;^L?VFP|1|jdupi_EDC%^MW6}9cvx0G-LMX)$B=Vh%$c_ zL%IjZi4F1YcfFUj_XXCbe5e9W`|6_xx3B1 z%Gi`G%M&e(UziB`M>N=#@p5r={7)@ldfw3lm}k0qA+Kz6g6PHaywBL&#JX3waiY^5 zF$bRE(pnDy^b?~Tj}LVaE$jyL~fyzlltJI`za zx8G0wUI&4vz;$^!vGd@D2P{wRWAwLo=ESD*>2u)GZPVwdr*(Zh<$N{QTXSO7H|E3^ zeH^}Bgxy5Cos@Mh?aKU(-o2M=)eobS*XG1dfQR-G&(Dd4?KWv|Bs;afREr7X=&^ah zD)FU>D`FN~w`+X&D0_KSEfiW9sv(cTSmpU%x7OI+UC zRI1%?;>Bdix2j0$s0i*4VYO%jOaA zQB@y@K=Nw1E~k$Er#3PD0bpFCl7>XQHPi!TD&XB^Gs9=kg1SOdr28Pu%dV zG%?iPm6;^cU;A6FJAU^in}HYM32?CHrg&jl zg2=(F#xHDXYiAMmO>gyWgQnbiAGEJIu9oL$`F1tex8?Q=+jZF}>2`f#WOs-8a*;Gw zXeKtM&6_m3c3>J_Uc60;63C!!2i?ECj3lh=;R{M%PE@-VO#d`#G`S_imyvgEcZZG3 z*M#)l6D_*F9Hp6ef>L&~CT-R${Y!lx3Z1J8Sg#D%P}j6I^Qz3cmSkhnLiQxO5=~}B zoA#fwYfT6e|x4feZR;@d?>=IbfnoKsC5NuaRs$FoMrnxRm zalP^V)yDNdR4u&;-=RO>rs}`dU#N$XY?w=hz}D8K3%4&?$Sc}AEC_jm94^gd)5&T` zE6whG;DJpyGk{%*cCLiicx9=ED%PB*-whd_ntF7^OL|wk@7%D)Za$X7(_((A+2569 zvuh8WD^lCLJDr!$g0-d^Z6$TKJe*WQHF{BQeYbgkYdvAgA3d87%;@DoyJ%b39rK>o zb%{o=c0;|lnLWETyi0XE{Y>9v;aK$=XEx^W%nck8exb>!ef91idwBTd025 zn#c08I%y}2yY8eieOrqirPu2=1b?%A4>I@TT72+XeXtuGN{9Jox7~>4%SU&%CLc<+ zn%)^2p37*5_k)`o74{5YMl5TaB$F)zpN+#qJGBZVT>GTDzGc;N?;)B%s$(Aw!!m1bccJ{_@5SeObV?>4ogf1T9Z(QE~50 zJd|J`tfifqzUGX&320(M9)3~CE_HbK);qiU67@M&KCGVzhV_AZ+i#xs^+K63dQ(FjmW$doT!~EE;8r~+0ON(F)b8#dM3`gXOnvrQc z&Ysx&3Btl;z+7XTN}2mfhF4l&hpsD_d9{1jzEP#SYVJNYf3e>J3&y^ME&jrmeD*5i zMR8!fjGspQt>`57D!yF=Nncv*YH>FS*?z<+2llbm*vH_PJ||zD8_W3w_t3A;&E7xT zf|%P1tg$PW)d@%6Oq-mF3cFd%hpXPI%CE zQq9d2%hzcM(lpg*4%T#f3!4_pEPMWSdD`k_y>^eW58L#~iMrZVc7{+WbJ5b)EzPn` zusg9w1N1_!-zk?(`d32AH)uEA?aVrp{+ilI7()x26Ido2?CIj&cl+L592ijCWy?G- zKQHX-?Sq<{B3UA!O;hO;((T9?ykbN!BGmWn|CDjc}Fz4t=ld4Cwx{*yBfJ@F#GFjLH@walrDwB~hn1tu&b?WPwL38z7uQoina?VX&7DlH1o&GSd>c&0*RD!iL%gBp9{UY3 z_T5ZB(5?Y(-^?3YctM9&202w}FZ%VaoM9U_pW3BYxpfI&pG@7xLTuL_t%A#BqdH_n z+spYz%zCCP7p+gHURoj>SC}t_n7Ppg``MIiyD;)*KVL@u*&|FUi^ho!tGL^Xw4=-M z!y{JB_XB+@-ZcI3-JS&RLQ$DH6!vn|r_LHQB_%sO-HY^K%cB=ryo^lR{N?)c8|A3E z$Lgx)9(2EFgXV8?d3A*LHA7e7_pcQHIt@{b0pb2ds!`K}K$D3MjEfr&;9FSSw6LT} zzmfk6FP#>mtBv}(L%+IS53Pe%L%X22V;}UPIsX2`{ax$L_O9jMtngnhDA&t@=Sp;q zH^X~nltcViLe~)gm;X1zn@I`&>yoN}j`xe88N6(lvwqi#=6K_fH+zm(4fR6fkaykn zUNzT6y@t4c9oM08$jiGh!`a?%&=N&&aC9fGXtAMdy_;#JQF;yjEY0@rf=?7TG|S7U z9bcZV_2v>kOS8Q<5l?iBqc`J<7C2gnE4tOtwcbs{&(dt~jf9JQL$hdISEe~!zdX(M z%1B34Zs=O?b>x?&+1_o0iwYgxjw`yu(OYmuiws@s6%ap4^YLeCw)a-}M0Y!?!WB`u zhN@@G_C&pga=kpP#4DleiT}&fwchKIKTET{-$YK)GDFvS8DHdnM|fHOsp^&GyF8 zv&fq%q70CC4f@9~sy0OVuDQ z8tH#|$|3b%EnQ3gzd8!Wf6YN&$vh-T=jFP={3UShJSLdS1oN3-P7};)?A&Jh{AT(* zXZn06nDYel9y|BZ{3nmG=t?w)aYvMotI7#&hQ| z23lH%yG++-aedbHjCIrLwfM6{eE7y^F;AOesTWr?ZisY{p?c;V<|LNNamytia*ab? z4%eYxXgnv6zHud*P0Uw9*E9DM&0)Nr!#F+1TLj$+Er#xbDxsy&TOn^o9=38iy&8X( z<}h!Z!}vdk`Oh5YJ#(0c&S5?`hxyzb=7n>-Lg;q#yxGufY{x61YsvjrOV^YCua2(6 zwqA*7d+2){dL8t7Xg>5CNPLr{xwvx)y9IhPv;bNN-3r~twE{zPm=Dik9<0T!C|yhF zuaYjbSF_N@EHj_DmT^UP`NHvG`dE4Od+#gHkY+LmdSx_|_`m!=s)IWudvBq`q0CiD z4I>Aso?Z>jgM8?<(2Wqb8oglfAo?^d!J~C}^iGf7$vZir+o4;a*F$q5>=K&fMisgN zDuCugZ-(xG^q$TNs1hoM-U2OvegnE0(lcwXf!+Wugcd<1kd{$?6Do!7gl>i22;BxP zhRPte7O*}ZR+4o^XbGf6M?K_mh&{E@6LwEj`=a`OdbnrG`QJJy@V7_sR0Idtb+%p4 zd+6M-a2$7ydq#69FaFi%a(z~m*t)3UE6Rnp^B^^w7j1X4Y^XI?Rvzt&C~{pYo+5Q# zA=ahhxV)|u%{zk*CL=ls%7Yl*Ij+UNk=4P=(NFIua$`e|MsP2sALS)|>ATj^d}}{B zyXu{ZmKm1h)OH)7?vk{h0z64v?L1m{(^)Us zk67pK=()-|@hO*?M=aOzkv~!--8~P(FN8Q(v3%XMI$Kk!29Cgp- z#_rnK-DWqRFA_!F*BdAQ_UGvrbUv-NW|OaTu}j0pK;m&f z(X39Uts?EIN-eNms1eU=K)#WmkT=h*O^n(497VRex6F!FFXCBnxyU!9YmxdK{j)2} zr*=I>I;(2eAl){0Ahn4{Pj7D3S&DF8Ao*9-tX^63wu>(YXE*iGr`w3++zc<+FwbF3 zPso*#L-k$Bi%aQwqm(l~MH^+@k?mLi^}(WOJ=75%E$74+v29=EIbXtKiUVD0{Plhp zJ-YN8NK@(e;*Pw8^e^m}(w%v3+T|5jJ0X4lU#$F7F8y)({M1$Y{iWo;RQi1IyQH+N zoG&ycwl%UpO0zM4BPy56cM@F~y&Xp$=;i9|Qu$7)Trb#i)qTb3jkL#KLi+x^+0*Lf zQt4M8LVlNiHF}l&Q>5)PKm5;-e~9$`pIP~X^y^5w_dh9p^$*4c*J1oSIrSB8?mCla z&KXYQ$#QUZq2tcrNkSP&rPsAP#d)G#Y=3hkQYTa5r9JQCxra7Cu$5EIXEo5tGl1qIEnDg9I)sJ zlwl*APAH+yB_U(y1&l$umkT#?wmfVP-@8*cL7YHO!C{;r9Gj?_AK78*IWnsb8(dpn zSI3LI^b(h>E|zn&lusJe)#(jv_L679BJqM#jY_9!4+l+J8g2bp6;iV4b~iFT;n@o6 zp5aw;WQSHgi?Q3BW4Lhm4EZ#4;&~gykelk@*jHO;vpJz{{KR%cvCfy;TT6I@@xjgw zvuFD6t=nRU0C?_QvF4uQdskr`1L84;ZkQCV3{4`Z`P{NX?553kLd^}5%rg)CWOY(QF}UEGq*9|*71 z5@0|Asj(%j6~^1Zf}^LMS?0fSRlThzFJlvB#Wt^XUleebi?^FTwuNzL zm84BHBgK$uv*wisM;FV^PU#aJQ`Qy(kmrj!7k#l4=d{V$d%g0s`jo9%SMQg!sg4sn z>9Mm#+rPKvwC@F53Mn_hN%rbCQ|`Ls;Dm=Ad_ONsrlezUj8iQpUTHCiWeJ^M1<`?&5-fT!iUp4=^Vw`Xk_2Fy%P~u5;1Tt`6tjkFmbs1)3&9#GUKS2# zLu72huB;yuIMV9*MT1KDeFIYp-@VGba$D=td@m{U#HO9^W(M}|;0)43#lRqAK=-d*vp!CS(f?PA?nxYaU)q= zOSV{Nol_jx9#3zM(C?g!4CozhWfr^G**suul(sF2`l?Kdo>uFOW&wLOfZQq$aHEEPM|P1E0~9>gv|h~w1En>DLj@f4}Je>949k_V4N$6za+cHO82$upG#~^}S6#V7N^j1_+3r=`q?T zIyiO;Y45l|c zim*Ew*3*8R`H?dZ$grYudJzZEahJ>+Z_IFreDkX4#N@QV2M76~L;O^H8a#O2(oYci zKl{6X`u}=k=i6_7aQ=V)?!Wx|`)^+R=f3!>=ht@Me1i5e`RAX`47MYzaQfF1%B0k2;TU;h2Lk0_w(Oe9^Uzp#^E3QL-4@wt=~hM z|G@Lo=YI4}{oY4kWFP$eXZ8CjXuibntuG6I+F;1*GvElX{IB5me=4r}ACR`s6n46| z{t@y3+*^MWdH>Hq`;QUt=Y_`U_~q}d{|I#Wx%)@Jbzp1y6yGoZ%jZ@me+O~?x8RB2 zXa61O(65=EPx1W`er!*_^SAl;58nKk*XBL_r8ocaU*ScN_urhA&P=5XeHvrL-~`>8 zUGESb9Rt?@>I#H8Bsm)Ovc|jL!;u?yhfS{4F+TbG!L_#(c-6i?Z{Kg)_w%=StK$&E z_p7s+*pe>OT~z7{?@2X!3!6(s}Uo(F7FY(TV=YH4A$^0cB>Zd-6$DCwI;W*9{;clba&CK0xrMxkZA^UpZvnQ)D?X8e-yT6t(>`^V78tYA}^ z^6v0mXp7^sM%v;{V)JwUZM>(kg>R&@?e`2`k=_$bv zXd3BMIwi|A&fEA%NoN0MJX+AnU(;`w4NmG|?J0pi3%lNEoD1G@-ZMz=33tC&EJ5`1 zh8TX9m~cNpSA|R@tu+Da(GoA^&l%(+bsYEWu3cEe-|*GR;XeImcz1ego};ym^|iIl z@Y|!`D;j~&Swe+wqfO_fXgsYDlyieMk1iyb3zmM^I zIzPtf4Z%|r?hj7uBZV<79v}Crroy%s56(}a++gs33LLJSkqJm3nQj=@ z5)Js`Mg&KwKh8DxM!1_AwEN@9_+-rch@RpOB6#BqNyf}9=8xb#3wO8oIP$TN)pck}5-#)5bfvQHrLtZuYJ^`b>ZhqxpLWMHtImpWeSmtw z{F<=Dbud5WBF57LUf^)TMx!u_;pst***^SARa8%Fn996O5bnaBJM9|=v<0UydDo{O z)e+{S-r+}RzRpDafjg?Z<6*Etm{bNeN}iTfE2yV%N14@qToi*bDEv}+)tIKj&Jmsx zS3k2tJlq^3RS3A$;8U37SzAx*1KexMZ#2#7Dfpb9D*?Vo&M~RsI_G)nT;9&R}^^vB&M+1JPqRxAxR6^v9zUK`?;_lD-XmSvD zQ;~z#+s}Lb(U?cki&#N8$dcX(OyS(o(ha%b__iz&e3CtIFUT9<`t=E<2aCQTshHN} z*9zJ2YoxLjuH>gdh4A~BdGvY>tb|p330Lx&#)|VaR-AjKvOLo`Auz^?^LiH5dtQG< z|7-doT&w@p*tW^}l#SQu`PSPJZ;S`2KL(Ee(TxEYIZoI|;UY0euk!ue^hcU#u<&#uyPg1!(VY#8Ko}{t#i8Kn8g=gm@jI@|D zm`puO@#3x;k)>xA|JnGg#vm@GbT)3bf4Hl&C9c82)0UfoW#A56{TJT|+_%5k@bI!yH~l zxIzz#bdeNjvosc* zCAkl{wBH_PD3}}JQn2bQ>C9?vn1~XW?SD+?WjI0^1UT%&$6FMS>PH} zaEJUos~?}sHSkjuD&|I3%oXq$ji$0La)vz|E}O6nk@ZT@i8}=0S(2ZCtvrhSEI5>N zlAn++A$ZSBK0Xc|v8o;?9q_TqtH-7TJ~kQkIAjp$xqRE@juFmTOZ-i|)9E%3><;q; zTlm07{vMA*>{gGH9K&@wxOJrneshb!H#l_m$Cy21OG=Cp>ejJA;UKLZCz<%zbg0KY z)Stp@c`48MJO@j2z zIS<^?=JE{uaq16`ts5RqS|souL^u%NSnKV=~qp>(?z#IIKEv| z7{S^l`URHhBO~((7pSDkRtlHaI-g7j*mQ+$KV#Y7XKpckvfj<@QnXFyc&Eb`Rv5HB zZ$tBmZuFd+ESZmwFyJ$3jKSTYL7YgZ}$o?fSCnzftZyFauj1U6E@7&J{@%u97t^D?$uEZ|je3S+#Vdz!pn zk`AA`b~UGvW_mo0fU)lzn!eR%@=n zVX_1sTc&k`q)Qv%QcD*4Fk!+Em-CKMT$BFbKv@FEGfz+Pq#)zT=1saoHVAAj05EDE zJ4Mh9OOKPd0&n?&yEjrw^{0Ui;dP`jjc*DY!xZ)cE?|eB!k#VSWBdXPH*|36T&#E#L+px}6W;6s{>`i|{4Srk}e*0MC{Pf`ViCAgyMm zU(GPxiwa;I;*M@8ZtDgDdtu?V)TV+UVHcGI(q+YzZZBkXo`_mtuY2oMzZg4|r!t&WOK`(Aj+pzP?WxwEH<$|T(Q5r6X+1OXyde6Qa`eokmGNmsm%%<4h z7~udnG!W7TPGJcTJSMeA^ND`3TTR@%I75 zysU?&MpiRSKYXF$nbYLd!JbJYIsxK&_EdYRUvq}GJUnTP{WEzWi}0iUQ?0fb{A04@ ziU7Vn>9_b!@r(i8hZx#5ES8_3Q4hMRd(aPVSD%Fz?mko`AuxaJfj-5>=c;38oD+76AlW*8KMZk99jvyoH$ zOr}hGCo*kr;Fw3!qsV08FS$8=nz;TA?&!{>HF{xQ;;|H5c%_%bwU+-cO6 z_;dRq-=uqqzq~cl7d>3uHV6o;CBHDTCO?fcq_?HX1i#Qp`C!K{Zn(z4jQV9mD}~3W z*PDJmYLJ)udA9&7;_#Y}ukM%|FG^U|s=t|lg%>CX|u1SZ^E=4RLt z^cU0wplB-t3@^QrkLpy$#I03~FzU80M>pDbzuArz=S|&dn(bO?8XdrR)eS;PCl<7I zLq)7xK_tv->{ykr(%=$Vn1LLY-QPIN(qA&31J|DPlddtI!|gSLqG6F9_wk-Nr(Tpu zR$Sm8eu7GFh*vR+F2NTL@EkVdEvv8ai`Fc#>k)XDN_4?~(j)LbyClrVZ@L4{qddwC zg2C+!fE%NHxLJ;{8{*b=v7%-DdJ$iu*qc5$6bRJtYWRE}@2Qwct;Q_&5x^?X+6nv%E-?=3}e zkCv^8frsFi#1;EdK+|rP%ugSt#`Y4M;~VTi+ejeXkuc1X7q*z-@kUqus#%(k5_a3} zEX^Bd7#B}!_d#A+v$_W&=7ToO!EdqS39gkx!8KA%zNPccUC4GNk^;B(*BAxjR4;=konx47AHy)&BMVlmoyYE0~8Yo`*b96`naT3R05ps zg4BctFs$5-gFcD?GslrHHaFvkCcx{(#eF>YjV041bF_kE*isHt9Mgb2lXL?ML9M9v84K(N>vthDJDo5ZXUr9HbVsJiEMJ&cI!KIaa zaZfPrO@gEn+=yENTDao@m)`>^bR;_pl!k3C*((l5yXOZDx@ZL8BVz!*QQ6beRJXFc zCo4dF8d#EPhrqgoD6R|XyJ@A9;{;X5`yF}gO_>>Y4XP;s!!u%J%rJ3 z3R#b?ocW(98(19_IFYNK3?9&~XfU_J7>pOs3M}ngX^u}0>d638LZdrQypB_NC_#g# zT*bpU%bYr22>7Kbb^n9J?*Kd29A72@*j6X|1o4G%#cRk?+@N!}6}Y$=eu4$oq_2vb zc$l|pbO>&I1MI38VNQC*V>&*?Q{@ih<*7M}mvh!KwE?~07dpUn$y~1B7-#C6W6W8x zGEhba`eO!$gGJmE33yUZ=O4KOF-@HcM2`p%Uk_{t#KLSu;tib&^pSDEG2R149I%zP=Ay)9 z(IswX*2E0mS&Y%`5=h{tIFmTq`a0>d=%Tx!=u3SA*9wT2VbEf{Wig*|)EiSj8W+kEV7dbS>RdaMnwvecbexGCNIss4;d8L5r;ux$Me(4E|*-w)q%vuxlb+ zc#2eD+`rX_mLo8>$8pXruzkDGC8w|tU^e7iYQ^91fn=C&eaMlZh3jIE=(6yFZ_$R6 zaYU{G7x_W{wl+U4BEOnqZ-v_w6fP3FxNR&HTKJoU6916C$Upr}0t*}x*!;}!pni#G zNMWN(l2~X%5}W^mG&X!ln7Cs@B%|B|{+{u0<^kzR2So6Ffv6G+XKT6zqTGa?0 z+XhP8_pA^esKNvM$)LFhq}XN`{m2J5<3LdxnGrhNjj}VENQ1d09ZO_unrlxnU-h#T zfJ@uy9gT-sX3&j^klP%lbhl*)acvnwoXZfx8<0=@uH)S-`EKf0(z+lL;I|+W1eZku zX}o7(DMV&A_+rJHR&iV(0h2pzN{Ztb{T`d~xIu*m<0%BbII^zeIEKsk*8ExQ)^>)# zZP?R;KDHLAeNJGXe@Bl6PtvsAj4RERafcI^-D4uJZqTGTGj{#dbJ_)=cnYkb!e%@c zJj9?|ywt6p28SozuDlX9byfILGjyYw__;1iy5vbei(0~kUJQJFQqo%EVw}MVFU~T( znf`)<%A9yRGxDunm&SW)$7Q=Y;-R}?Drn$PwC3g)#t8fmQ!OGo?_)cYpkN4+QxVg! z52U^sIEGJsrVJ;*7eRxmU1N0PvRiXxBcE<=#&pX7im#=FG(3xu+JLRIrzvA~`WUZi z;}Fu4EkTQZChHc$Bzfk3B{z7l&f>7(;r3vs)&@`KlR=Hi0KPpebRCVH0Oc<@t{FCP zi}L9-><|o7%KB8-g<~W#cXc|`y)&obgK(bUC55u+r@30hmxP92DwM!^nR=Wh7}C)+ z=+W(!i`Um;qyj8l@Qu)?HiN;dm@&7+VYh{~q=G}f(>(ArjS*N)=4SYz}7CzBYl%(Jhp7<2Jfsetd z#RpEIqbz4S+TqoINx7vt#%Y>ksPV*id6PF(KpN^2W5MY*aQOG;Ls*Cqcf$fC_}1i2 zam~dQ!KF!>xMhNukdjBllP7+GBfm);7SZr&LKt+-!z@B~f}kfpJ7iuFhdC9N$zmfR z2_FFk*T}u`YRwOYt~HTVT$At=t62lw(J;3CcyKr7dvK4B+w!2`D)DE&`(ym9ZzVnX z^bpG!q?b*{crKW}Zv2;(-2-RF?9^GW<1L%It0 zp78=e^dh~tx41zIKga;y4j|Ir-7fpZJ;Xh^j1!mML4)jd^$G^MA91S99c+NRb6jB8 zmIvb2Ep70_()H;;9&naC;67w7XA5j?djeB;J>)>63lPPPIL5dK&E|=d(1D59LwT^; z_MXfqwtCaTi{MhP20ew4g=pX@C<0%`hlwqHNt#pNnhdd!lkQj$H;*Z>+~Ai;&E3`S zK+m?2#=528lbR`RiEnUrn&s65%htS|#O2;3>9Uig%Zj?#(m@s1)ImetTt=I;zo2Wa zi3(ooF;v1zaOPCFx$q#krrM`r^1=hlklJlaXH;A>@B}sx9F>GAU=sVJ;KshGve!!7 zW^n*4G7-DM8-)osOcFYw1srEQ08`Cw&IcX1GQU9M>RZUi$+X8}J2O6cT}L zeF?A#5`10#sV0X>p1|D2+e6T=v6L%Y)d){;IfxLaDXxuqp5 zTw1cai@{diMTb>)nc=Csv|x2hI!NS0HPc0RiAd8*T*y?TA#Kvk$V+uYR%-}B3GX@} zX_|XG;12sI;kJzixQ1+p+fAV46&VuP?$ns*0rbY$3~R_!w*gOZvN2yE;%+1rJWI+a zl&!_|Q%$t7RFgDzPiox4Vo8!47}6*#?$+olxaM$*yPV3+4FYe}id%=AqTl+uewn_s zPX=&5nS|-r^F?KJ5?(^m9pGgU=`bRLeZK;3qke;n$9(#v*ydkw(bohlNOG)J1$1KhbW8oMoc zlIBd4BBGG4GG_t|S;;hUTcGY~;+6_75Mm33%ylM3d;y2XQ3~}~9R<8@j)xq%EC3=F&yabcHjZ)J`lp1DxrxLNZ=QC zwF8%Ws?bL-h2O$zByh13DKO5<$TwCag-6N%sO)41-62e^Vs`?pE0UULNc?hU41QWi zSjdvSFbgY_5{Hcp0!PC~SQl+fSCvo3VaE;6bVB^;c~*0Sru1QG2S?bIBNPhf10RTs z{-?au*ky;|L}NM_j@7)NbcICS&~hX^^L%1$Pg{c@X9Ql(otA!Jtpyk8#31aq;>47o zbdM@cJ<7~eVVpO`3YP2##urzz*CC)%5s{|hXIXQ6kQ$x#8#})`E~2@@>oj^1 z9rKJm$5_Ge5{ptaeA``(xwTZ6D*@{`h8wA(j|*(I3Ru#BZ^SrC^Zrqq_mAe~ayY;Z z1u@)6FJSy&c7tJ51(;j{AR9e>E|~C+BJDoq>a*yaD+R6d~XYkVnR*Bg}}B2Szt3@0Oqqx zVrC!=VJNB+;9+k%#YK<0McUlt0~wFhz(|XtrW@sloo1&XWr#M;c32lW2^0ISfyb&l zrdTr=<20bd+p4`8KIhS_{R&?IW3@(DKFdTufHM(H|Bu-*z{81UD2VmlXn+ zEm!JCw<#y??*lpAsXGY0D?apdWyfGqb_7oRLfK*35T^8Wb%%0e{tF6@;X}y@cTsT2 zHwxYSqu|XQ^#fYX-=Y#jJ~{7RQN%|Z0t}YO3yyK&Zmm+b>XxaQ$!5?j%AmlJM`}Yb z>!8;+4t&b~*<|+;w^UY`dcXtWS{DT7eYXO))fxVyLK2lE5fb-(7M#-);>YZUygsq$$M#z83 z+GWAfza)U$n8dEhaDaWCkk5MM^GLGiuyr2E_aR~xGR_f=OkWu7tJoN&@qLtR6r-9} zWEi^py=oM5+9X#3hjgWRZ4xEGmlo~P2K69wScnuk*px&|b&|&TlQSyG#Wr`6#{84V zEx&mQcrq`9ConH_qQk}ycyNFXpQkYHvl7J3XUD8g#4U@~xKah9SaJhK4%s$Ldlq-V zMIfHwQI)}DDDz{$DelESh0A|qTt7+UI+n^)I8{SV=A|+5nHL*Uk*8(+XZ@ka-=~BTh<@R42(O1z6^we1f{YBAanwQJrvpYVrcNa+dTi8GlX!$8h~9 zEWn}MpBUCxfKQUZJvlkRwJGic9wj=!=$mP8&dC6_S*viu`&b9|9XXhOOS|KD%7p{w zgRrf$2yB8>;Ix<^ZlU)&EB)Xk$;Oi;8)bV)l7r$_V?NX;y|+biEcXy;NE4Q)TC!1a zMj~)a4hpU%^Ay)4q?MPOGHfTnc($0G>Li_`KcP%yzI&$_C{hOq+;Q&^w;E<2rB%F7 zHWlYC@`HL~d#z7;aUB@&ID?-Iyy2GkPT;?o;DRIH!BZHHq9dFnISQSyu|JDC;xauk z@GU)2@l8EZ!Y`rAx*TbI96m|1oUqICRuivIyd9>y1F>f1SDfQbLvQ8CbB)tcHnw>Q zBSKjpJgKpw>T(KuSkBY^8nBFsPJmlf7smU7BY)U1(TStKV}t4Gpm`J=G{7l^!j0qK zbsKCJj_EW;A%4yAUB<@3@&9 z#@F+V3*1bO`Zu#h91d1s*V|d$;wEM3ev*(x+u+>9=QJjniLHRoXoaVxI0gHTuC zSH|KK)BIr;Qkf{`q9SzFT!bXKY!rXBt*BoSj%Hc{|4@A@9v&P%DQ$Gv-e+ry?E=2A zsNBzWV0Syo&?Ho1;=o*Ar3|u!UuhbOsW52jE2CM;s-2SwZfJ(^vs8K zM@wZ9@1+BN9>&Ys-Gu}+!HXFTjZmHQN^zxcu>VX7QH`4{)8Yc=3SfBFvaz9A|__yd=UOK2oKo|aO(gNDmI zH5|lGb2}=+9%2g)w1{+(=FsJ|HSTaTZ%J+w7QQWAd`S7-(!~V_kvmMf_|SCmVUq1U zT|uTpJJXLdfJZ*T!_}F~hMdaqoZB@&l zD&>Xntx$oNJ9|aqwJ0p!;=i14x%2lc!b|g89=?Gc;?{H870&In3A?JlB$B!<%EV-Z z$#}d_6vo@*GF_^U%XFzajyXxuZws!1e;I#l4{>#@oB2eSIK~vq?KsD2ZUI>47Eobf zD4U9$Rdp8omSD2QjfeIs5aJk5u*~UP=!V{7L&B~tnVN7`d*6>^sRIL$Qx$6VvM zHP^s4>3Giur|KJXjbk53&<>B2d^}EaP`C3WMW3{?U8iaEXl?s3nx=lPG)+a`a1Fxa z0WKJ@c@AL+g@$fR99jjoT{{)iw5aD59Q{)nr8&Sc)^>RNkB@tB%bWn;6TIdug$?)^ zr?R~0D{On3b2sy^;Ti>HbS*0n@`W?KrH827 zb&BJDFV59c05&!(vK|*Cn&NVYB#jFqz!8Zj_%937JPj zo>BPY@U`8Wu6118QJmEvU6?#bdwTG!7WVXr230AHv|x2m?o#q$iRp7(Y39`+wE-N? zkvHoGyj>^Vc`|5-gd%F1c<>jx!W<^TkkFYX~fHUZiB;m7Mr!$t?{1ksVu!RzSP%nyf|-3f*5NAXM*xG(kbAgL!ssY8d?R!b;RF&8f_AU!p??dPE)KNnV zLSBjUvLW2ifXu(40XeQoI4Dc2;3=CrN;xE*VBn2Ae8JCgCa=TII@Y4#S1$%X8;`rz zq6?0D0HT=}T3n{)*q|L@a@Y=kf`#w}9JXltWLneV7nAc@#S+KhfGM(GE$MYSJd1)& z-8v=7^df2S=0|LNi?m_ohzpjS#AMuT-t}kbn)va7Pk9=6>`}(n-bFsV8Vgxz&|?2j z5BHQW#Mv9-4l0>VHR*fkwtoL$N~0{pFO)U5u=S>RO!OeKn&Uq0@z4ouEgn(k$EV7d z;BjPS8B>sL1F5i6@k5yXdyrPO&=}JB5NQ{EB_`P*~JbaV4 zY|;Re-7MGs65CI~(SI@Ik_x7fX^6m7R?Knc;Ic@@F8CL+i;8)#XHS$ z=wDvr6j-}d%$Wt((qk0YB$PV`a{FQ>Gkw@8<_C)a!;KqNlHCgzHy#-&{El&taoH09 zx79<+Tj0nWvM?L)RblW|k;uboTYtlOj`#N~v z^-aw7tS6G9s59Ok))Il2fW=ARX&CXKrsf>d6rkr^$ zG#gvZxYy;}bn&-z>4E7+!i2^SayKPVBTOT4DNH0T;Wn9=lUEHbsv9ylMj*KIHz#aEE^>izBn=9O()7$`)@`xnY`gF*h#{ z2hR7EJ5639He6AR3NKqGrX?=;yV=5d;|on4Mjm|?g>-NyZsJC};r0lDZ7wXZ=^6@) z+MzU|oftn!6HQ9Grj7Z3RDjw7K$rA{qogO~%e--8D{U>+(GW%+nCkJKj+ScKim>R% zxC^~g#^}1lPj-SV!Dl{g8n?1!8i(D_S^(0MjHO4o9>woI6u3F;R0*dr^8*`>+p~_6 zesYxblcTh*ZKJ0ko<3$*v_dvN^^Ry)# zG(|oiwbotWZq+(@HrHC<_Mr%V+6s?sUYNF)S6Ja0aiI`jy|k0C&3F%C)|vH1ctP>h z`#TjU%i@*5={!Za%>dzU)xNaaJ&sM2b{Sal(=cxJ zCEhEIFrq*jcwmZOyt`m`W*m;?4r5O5fj^A9I}g(yu)|aUhqyjwfPLzRC#}pZowI>nLyrrwc5^50^~H zHyuE9x5f{iX>+sq!5w~BFm(wkezCCXSc^CC#jQM`W_4+MBcAa7nD?U{a6sEyE^C*1`Q5U2|vJC%7i%!-!RJnZf=u+dqJ)NCs~R*N&67s=jiG%&HWB7 z6^BVyOOHu6go``r>-kpj!x|5s;V9^UIK;hC`)UArG!sYg~;s#IQA;(G#E~B;jfhN{hi*zbD%R|t3%SX65 zPqgwICY7pZ_BgyBQzO?*U^FVjwcQY1&Ar4%CwM3BrFC3|5$_D;RhCQ%yBbEg%EF3v zX|5FVl7i+Imtucu;+?L{8K8_%PtI5FfmhV&5ZlG*Ubw44aaNAF{C<9ly2VwJz3Jlo z37Z!}ABlVV3aSbm^N}_z^wK4am{jPFRV?ku1S1J;b@Oftghyub-d}N(Z|mJ`a|5rv zx5v98yd|4~tAVp?1R6N zkKeTK`-PAOar2w`2AisPGMSy*J(_?yTSW&v#kQTY&m#|7`gCLBuTMFGPvWohB$<{W z%kYzVlWAAsuCO{XK+#VNH~1M&!VSwM{y3D33;kn>3v}X^3Q{;kEB%o88NMk^R_wT@ z07qntni6Vq=@wPsn09PD7#t3I>G}xZMoe< zjI5RR4pUn}gUuT-__(L<`{gy~Y6(TyG^P@k@=bPCSG0J_(K|zq%RHnJ< zXU)LzL27j3t=lFLwp1_R*t%B9QX%W#azX{P`?Mh@;Vz{Nc+m(aVf2l^?o-W37uHFt zs;tBAQ~Z)0uABB^r0!-aSfT^2tMp6M6J*0~ZVtgU7i!(j2z=*W% zj`XYxo}`X_<+u$>Y;@h&NejMd9j|+=XPCOl2G$*5CkpTIwKu}(#>iUf7vXu|ycu@7 z$=-!W%){2~-5za7E-P}kcUE_^cpYxE3A=^MJE+r0S>sl3ys?p1e3&M|@jE=ny#TtA zPAjj81Fbd@e2cO!ut{~m()aLk>hy$#;J8BY*UN30vrnwFtpD$$z*!AEp1auF*vRk?|Vd%EQg;$T;c%S_6 zB(4iX!rrjExj>laTqx#tnMFS@oGu;afnzX@cXjD5@uu>IUDbD^fkw-_8!W3XS}k9y zla+Jr#kfQq@V8F!2JvoSaa5{h5!`U67wRzqFTmNyn6XGCF{(v{OtijYMSG{v( zmJtGnoBo!!%tgO4!C3Tb@oKczC4PlfGY!_6U~$JB5*@|tsl~02D}-C}fQ?<_;X8W; zez;rU-S_V@exccYc%{I5?-lqS0z1vU6!vC;A0(c4FW(CK+t&)sjg*G#iT+xK6a7v~ z!_Hnw+s^KG@ZU*k+qwIlg1=MZy?T2m@K<+k7kGOs!J7%*NHFdQw7hRWC_J|>mvXz3 z(tjoKzr0<_L3qHV9Wqrm^v!j_5=$~zv7Ca2ukgt@E9-yB9r9gpY5u8hwSNfgb`0Ub zw1cPPf{WpkUL|xzWz%q_i@IBG7ZzF@WEI|L&ti*|fu!JcToJdF9Zr8k#6#d?BO_i0 zLa&u~plKQD5FQ)p|9*Vxy_MGfq+)Nf1=t(?MLG{-cJt8}Ko zXPBj|SyrB`D5WzB z@mB`TFs=P*Lf1NJC{OONocfpyjCHw2;J7is{A}e%{LrX;T?c9g65QN9XSk9UrD>$l zXc{SWT!T>3HbD9?Fg@eo9{jL11SjkoyF-}nY^e~pPX)q2!4YiX(3iwn@XIE>2$y+a zf~p&;fnP?Bv=oIBv#wAS1h%;YXJfpN3*Q*L0Y~SOi_^=MNEO{>W{CX3ow}R2lbBIA zG)lglL0V)JUol0fEU~5NgK}+Ab&zy?!V*TO`_%y}%FccX47&%eWfX=!_j0#}Q#;2& zDT)fJH-r>d<|&Mk9+GGmRk(Cdbu-QA9lT)?Z;9iH>I2!wuI7<_JPR?8xk18dxrzxz zXf(gYUCO&08wLNUCV|~>cVH8i1Iy$=sV>k(byYkZHRa+R{`8;}jMy^UbGMo9!k_q} zTka=Y$-dc2ATY(sN!hkA*Hh@DFhIxr-0|%__JXI;(J0{#TOQ#h^bK18Xi^-Zq97dy zHBYqY>eY1cggY#BL2D}cN_M^dH0JMn{iQJ*BTbS5C=>L){bUo_$LhW7hk#4^(|EcM z*vHd-8&CJyeQ7sN+ylq5FV$LMJ78R(O;}ePqH4vT`Kr&ETD*FLs|)MLd@du*ja5C5 zjR}#Mgp;t2g5oH@`T%#XOe=QE);xuybsMK+=wI zs+(3~D5XfmxF>LWGtAr~;U)bvs?H}Aka%fNA2Bf+5t;_mN1iWqH}0er9WUHWgYwev z3`>0+%^VN6N@1fDPkn$)!iihV$0Dl>4wDN!V^Z9(DVh#gaK@7n*qdhJC%HmFSV96< zSSVl8sj0HKJ>5L$_Egj;^+R&H^NH9Hgdg3)UkxIGBTq$@a4f})qdTXU?wnS-$;Z<| zcfk8Yo~Mxbkai0eZVMJ}3mWcyJb(^X=7#(MGvx)ikp{RMX@EPY!x+g$E}{#QClnBy zCkQO@uraWI(33O=_|>=-U9VUbIK2U}?6LyI$*vnz3NJMPAGy>(cj5?)GtYc+4=}gG z;p5t+-vP!5^!PxQ9Ys1ERlN zlt18r3iXLCSmhKKAH2K?ds(*^C5no}@XZAc6l2Ga1D2oNP%M<4ZNa_O73pH9z@)?0NN%wL2)H+41LazY z9(hXR9B_H*n>4rwb;_eKn2CoTk?;#TMDlCs5()3afTW>ZSl1wL`9%=HV>4php+~-+ z1A%#-N&3qoKRMeZeocPmz~M8(i2b!0CQ~xrhVp0S>mlebW$%4*aagO8%2U-#?)XSJ z)g=mzcMXK|RXSaHX`j4+d$GP?)S<=`X@x^if1!LwOFVEzY$%5{R_OUo zSDg-g11>sA+C7uhxhko1gk8k}tgC-GIxx7n_4+0c6AHY!elg(nix;;XzkcBoAH@|K zz!{IF0>f`!*a&#zwbu*0^_syKF1;>~zmm^|ORsG?Jz+kwX!uPJ3;dd=hcFND8O@rf zfp|{?;ly(-@m%A9N)3DI^}_$s>m_ZMUN3Q7dcDMT>2=Thg-e%SFY#V_y~KNIBf;`$ zu*D?@H62bo*AmZ^wu>pQiz%*)ub1>+jIzCO>Edf8{TE*=>A(0|iR7h22UtnA!$i8G~jm zgD;LFVPJMQ3BgKo`3QE`%Z)y))sgG}^KEAb{;$2VmlG@MK zdMc~bezsEk*;-HWrna<|+Rs*!ds|7?Z6$fKnPlB&l69L&)@>$Px0&S6W|BXfN!D$? zUdn1S$)C-T8=FWC%X4_QyC~A->ebTC@k^ zy*&etafEP+H^vd-Q@k;b5TD|WafJ93Z;T_vr+8x=0Y1hN!li!KVjLm9)bCo1BgCit zUQF>`Oyztr#e2c!D#C_5C!FF9c}{$aH{?0-Dc+Fh#HV;eo&z89oNy}V)GyXjzgSEC zVl8Al!=`vcwiBP?4cQKS=x>BmyrE|jpXzrrUh+Y)TE zcTIZ%?7X(cS70+)0JaUZ<{9poF)*yrkiSjrd5`&VF?#}E_Uh+DjbaXwvy6JrfViOCf1|PQp#|FOS6~;-exNeTv)?Xc#kIYR2bHS z+V+s@rhbiH;S#O7H9CbOno8ZJfYq&0Dvb1nHG*(V&j52JcAy*TaCe9mUjt9ubg}Xv z{E{a(6HZtpcf^*iAp(!qTr*7w>og&(t%1~5g1E{f5?@1j)f~SuGSwK7YK%lRMxYua zPmK|$#z<3RgsCyI)adLrMwJ?)NsUpY#^_OF)Tl99)EFgdj1DzMg&G~B#wbuj`qz;9 zHKctFDPKdn*O2NpqWzlCXvZtReYoNW2=7 zu12q|A=zpOgc?$-Kz`8A$0w9;)nWk{vlc*;;pxABxAly2jRHn}dtC%?v1hE2MSrwo^L z8&8=&(QQ0sc%<8SqK&W1(8#ax)O)MowlFf~p}&PepHMZNUkfAZH{F&F>nrQlS5)1m zzwxxbf&p0i0{;0Pe!qy{zk=U~`29=xePHV{ADCVEgJk#pAnlj<0Q=8uBn{zD+_F7s zz-JoWeiJ8L5hvDPe68~XTd)3Lk$nZ+0{agN>*|9tED7vqJcviust?MTsxZdC4`lq) z%qk2zllLE_Jy9PZ{QL9i5XXRq%g%#_U*uHi_XkIdhetsRd?9?_M*a7#%->IC{(e4; z$26qNFyAj%N>}ff;aXwRnuhQ`?&w>nEc#>7qND=uRDOiL{NDG#o<4VzM=Dc+QKs+L zuqoSe?9MalDi6hkPf+_YTV)`FW>T-h6&1waIr)p-#mAbP`Aa6+fz40GX@jeWwkG#5mE%Lz**y)Fxg)3Mt{$ed@ef;T5D)Vus}U0E z%Xrk)Ltcn#^(!zFE$c4kXKhIQQvHeB+K0f(tAv$sS|JaeM-K0~^Ez|4%~u|}JHoal zEYjyRrniUNCPCr$bUHoLn&vfxjih>*%c}GVOC`b;oKLkfPGR(&hn^;p@CYXnSb11v zduMo*2$%FAIn^FsIKUdGIQJ~JJzN`;l2h&BK7Jo6_To}nRv5x_OFwSyz%d#Ro6r7>);C0qBnWy8~fD=z$^pFOshzyoI5H3|6YfVJ1dNt;M1{Bc`EzZQX3)cvT9EcNnk6o?s?u@QV)Vd+3%%ufqPaDd%Cr@+NWy-ZSh1di#*_*OaXmU5~DM)~X>;Kd~G3priHx83BQ%5yjA2nv_T)eU+qmg9W^ zk(ayD))S4j{oQL9V1!ov3oy*U0b>ReFf_D)35ve9du{9W1izNxO9{T1;0p;(@ouGf zw^F>Yt>X__uMihOgEv#WnOp^8J-|hDd=HLw^&s_`2WWC4plv(}H|a`q1YGP}4_bZa0s9X6 z%Y)SZ9~|iUhX>Y|eNV=GO_ahe5UNVuN~kb#CB}QIiy0^Q zS>Eo+#!urZu<@{q1n%+vAC&(+)p@LD!d>0N=}{ji?xk|McY;SFa1ze05~$3LFP>>P ztippCwzK;Xss5*UTq|8xBXlO41TN%C&-7`^O!EkL$*-mSUaM{1Lw((A=~wrXekCv~ z&-5#CTPy;5n`Igp&SDf;aVhKxXi96jCw#2zou@nnWVp=RtJ>?$-MK6r$9Pj8A?yWl zFVmNC##d(flDRPz?X6_;Uo2lrza!FqcHmHJt_(~k<_)++_Qdr&*aEG=mYTe zTt-d|U)&Olm7&+ij=6WL>h2uB)RzeZ7aju_BQNo%kyoN{fr}<)h%osZ(T zL5r{p8$?sl?M4H_iqp5-Zl*EpW_l~_X3FPHZPz6|*3NIX-W$8w8l!F|E7#2=fp4bq zLt&&b=~g#WIo{MusY=|bZQV?Li?HO&Gw$W$?%;s99B1ty+$^g!{@l!AN_`q<&=#D+ zsN6!NBD(Rkw7(m84h^q(;4$mDALpwwj`_V|skxEf zl(~`S;5Sk|-biIf*yYjrVUGl8LNn!YL3#pZbz{tz>l=th_X0Iptb{$h{ zA1|+)th>JCE_C1)gb#2_73AwUE)-0ehSZj?r!prD@9KI>K3-4dFEGmgdMf|x*1oP= z`?`)z=lP!R>%7;-J3IMX8^4}p*)=PtYq<2n$Np=ntggY96ZqEs!q=>vt`(C*buG#M zYblS{m=~mfC*4I0*m6)2U+vhvt~*J_DGdKv-%=_t+#%u!d({!HOijWUJeS;cx|8lY z-SIm&O1G;5hceqqcb)FE?6~Hxc5HsJV>Z>D*8QP7t@}gau6CB(8M>412Hi<@u#@ie z+|jlhx~jRYW|3ZN+dJvb&Yg7s=FVWeaF6CrOIGjD?roL6lVtTy())Lk+*TNIr@I?> zQ1`9`5_b7Qm~DN><&4#8xW%u`jdyarl>T7aHTV=(z9JBH*R>LtM4H6R+aF#H)7J z|EiV7Rht7}^(t_DYF}5A&UZD*#H#~&3Ba^Q@HSJp7qq)w#=85ms#HPG`o3zi;wtZ@ zGmZk&FUgOqNk(3c8UYPpZ$nq@j<&0%{G*-(hkCf0M|*6iIa@wJ*ZnzOFEj$k*j`{`Yd4 zYb#t;mn|Kaxz3kiYmI`pJPC{fxNKv@<&^fzCbKW6vs?m`z9pM4r?WSglXSbB)8cDJX)-O+l;tL)jkob=+Pu2$0Tou|ow%cZ@9%uYC?2LD2ra@D|x z%ucjXg9+!&F7s5Fb!xKUvdCaflENigb!&7AM>LhXOTnmHqf{7a7N=%SVc<+Dx$FZp z>(+$+W!Q7xg8NPUejdL+f#0|B;|3~jG~vc6ZZP3StS$VwnS_hEG_J4W_bPs`;P*0q zKZoBR$L~e_{uq8Q;P*%I`y=@MVf;SCwX3JS4<|4O;JhTx#pA(^?Kf8MpYo;Plg_mP zZmjLR^yIaR-&!@ARSDrPu)w^CcTddDZvei(TUA$Zkr46saGA$PbIg1AI3*ryd4T(x zd6U+iJ*Kqg74@Y=nq1$`0gSQ-dotHjddt>&=8y~N~(f#CGNcWSf z>ikpx%s0q)dpsX@M=#Dg)U%d{eo;ar(;wmilGzmNCpWGkRlU8E91b@(B>;VbHG&7mioFl3sZy!8fWGrmF{c>IuJdEq}UD-oOvKQ7LpcYKd`o{#d7abv%=^ zjVKrS?%un$_wMBfJ0ISmt$d2r%n#qi6&>TV>4)2xQJ|r`gEOwU)c3<{tQc-_s-7LD zb9e610@+cUbHA=d{CyF#Y7k>eb01xBl`gHjZdRIQ`h| z)Xf01PhNbvGsT5+v(8`&7VBtbcVA(lU*6rmc@v%RHg?I5q||}@?DVZe`TIM6 zL%;v<%|HJu{QE0!{?6a#-|xTqoB!g!@jrg&H-C-Y8h>B>4gLNfZ*I!p-!uGYZ${dr zedD$PWQASA^HNJrs_Jk3`Zrgq|K|CXwZHXs`F-};&#Y9x{k0W-^N;>?-~W@(t^DY> zzP9rDfA##z5C3!g{*$k*d;whNPd&HtgNr}Avhr^|zw!gXE8s!*|DV6l5U=q63cjTA zI6p&Le)ucTuYB=uKfm%h@Spve=T^RGeTn1G!{qM|e@XG5{dMpG{Jnp@(6Nk2+d&*_ z_^tf=&#z42{~3Oy{Sx2%`0__O;`qz(eehs8x!;fA-+}v6eAn=O3qPkPE#c}{pO<rkB$ZqWp zbV9U8fzeh+$0OWo@C^Tsh^uRZ*_C&|oL|arei;|O59~{Cy6MfY$nE8Ec5nj#ZrH#( zeDA&gL3Md)x;j|wK<`ASnvKUD*&YSHH*r24m-jv+t=NBte1G|S&#!d8KwbUv*H&cg zsOO`D{wvG}$C2ZXJp%?|Q~d_$fB5C|D{uW>wC7L1w&MIBKDb!|AYR-oGkIfmwK_XH z*07)cz2{fH{qKBpS`gr6!yV0gMLN0Ii7Gi z7$1%Jd^JaOExr!2HScxb@W}7o|Ejo=MRbD>EO9JfL029ebC*< zQ-HnG$Gr#WDP9)$wy*D^4|-Jx9@@bP^&9wiw3IqV$6n9~zr_FLPVWiy6I}Gv;m8X{ zxJCy(etg(D)%yfIe(|=!p%ou)Gh@0mzP<60Tuy;&Ii@|Ywk3k1me$Z2?_=&id!vJH zgGS(k56X16H}fo4cX<~BaJQvD!lpRh8&h~R;%Esf_ zo{Dm!V<7{x3)$e@!8OITZw0U0MOvrdrVnX7(hl}_u4XiBiM9`qnN zMTT^cj_!dfGYSv7c;=ATu^|3V=iA)W^7EZ&(JEzcM!#R4DU?R?s_2(Q@%FOcz_;x z-g!w>QuhSM?VDG3&(W?=ZjKJeolWR(C^6gvr1{cvj{+>^d_&kXJW7*_vR4b@(-L7R zt|O7;Cm8TzY*@#dbvV5_Kr2R5!XqhO3r;3!?sV#nPQ8wOdPcoYQcNiQzUnyxY6Du{R4P2tEi0=Ju!InfvaO4zBaBj|HMpJ()` zpiTu9@vU_jgQPuhh08xFbEc0EXLUZ-eBt(2Tne*2#>hP!Lo%Tmceo}@y5cvL`4VyT z*OHKIt-X3d(%zlPXXaU`NGqw`1UW829~!q^Xd(1tQEZ{Hh?fvgLqnWswI^<L-0Wi*{)i&TF!@1<~vYbWp;p(7v_$Lr_pyMFIoyS znRz>3>wePOSw1ZO}qbZb$lg?1emSMFOYq_nP1lhE2D~!bF ze*RrXqvfyCNb0Fc6n%&#!4zhR%CtJ3(v)td)8mVRFSQhYAp)CKC_FBJN@_XU3}b;< zBXj&v1wwbx^6v4tv~|3@HME9tTBq@^)A~1uaT@f5qyNu?LuQEwp zvK?NuK8R+7_74NCq+9!n_D=LqvXK$aU}o`SX@;$i+WDmDdeXAldK{2AKj{oa$6U3rPe_%HFi9GrqGr7Ax1t-v(1zsQV{cuLMAO+rhwotQ-;pFgDy-ZH{_|z&XvJz zdcSwNjkjTN1PWGY?2tR^-N8kAgD0Jme3)9X5Q4W90P|QP(B84CWg{)Ej0&D> z7O15z72yI8O(Gks5=%aX$}-9vVU$5p9-N#Xrz(~7c>CJzP0+kL_*-S8xXF2neYL^c z+?ZqQ^!Tz2=T6{t;_gCW^&%QGlXnMUcQNl`LryXIl+$s_zGM|`Uc<^9kNw`<+M^lY zu9$^7^7gNo)RXLz92r4j+VO_3X?f6_PkG_4(pk-bldJ=Id?ItC#h`&nmvAQDd$eu7 z^ujCXQLlVU#4*YlGgFF6{3A~|BcB~HYoV2jNHu@pKXwa)WJ01sjY`Aiq8ie=>iyNq z;pPbma4g1+WZHxw3vXiqec0Jp#R$rlnq*?gJjuKD737?$bh!X8Dd(Ffch+lb8!utK zj%hdLk*!V&RGSDv`kSx#9f*w1uDL_qp#xA?>(8S?T{s%vxh!G zImoJ&B3us{d`YAcv;fK}3$2B!#CO&i@w3vu)siwD7+9Z03oWf7k-mx+6Xu_ocCie- z?AOuJwd7PoK&Bh=eR)}Go?2yTZO1HPjrOYf`pM}vZ$CM`=5HB&ROePyXR1M$$>=eg zKgPcd$tk>YXBkSig>> zE=^Rcd(?aWxQ?28FqBeV;zBG&W7-{h2c6!SB%M+33>OKbnIFh^`FR&>4&om5j-V;1 zWQsKZB&<;;Z=ZZ6YjENI>!&!B3ekL2qGLR3)H+xy4+hg7O6|zq%kyL9Ng0)X;1d>S zc}c^IzoVrf{;Up6b8!`PN6MY`$#9T4t1=vtfb}SH$nbM4bf6R!Tj)8&I9gbMhZVSx z61Q00p0#K+q0D$%wlF5heAJjdIX|#DsFxCyrrs&KB$v$AkWKGjoepVzP}D*|L&sA_ zxeQzeac{sX>t8I#9G*6H2o_QQ3955Sna{Df=kq34-kCR_Yx>uUxe{GgYE#p>De0s} zzfS%#(rpmJgi^<(7zLl4V=?p$n+w`JFlI_#eO}h8X)iK_Yqq7YY`jwU%q;u#BQ3;e z4+}A@`Q5F8M$9KEPKmWcu z*GiML5}z3OR}~adRFxJ}NL34TavVlyOkq09v##O1cUp_n2I-TpiZovTu+&?tXL+qk zGp66FWg_a8h==mX>BEUeO4AhM3xW@GT|DFtjXq(`lCi{n_{p zZ!5{n3)-s4EEON-iU#{aXU2U?)_szlWsk$9j}9E>q_mPIK$%1fOv+;!*llD`rNQFx zeJ4$W4*SsFQ;VxH0xv|kx_4%D?_xD*cTyja)dCs#O;g%s$X*~?-GkC6`MZy2Dtf1* z?cD`Vt9!J~co}f#1_rejCziXd`*2b&``Ls94>HK=ZfDw?30BKO=~J1VsKo*A)gIv8 zA9bj42?GUM8sQ5ZG}IcS;fVx~MsZPsSj=L>m_m{1p)3xO2huYip-g!$;E(yc_+dx+ z>9fl;+6nh5lx9+`zRR-KF2yR{=rRd@jMEf6i*c0}UW>;QvC8h+=ykN8N;;>#BOEA~ zUAXcPrOpx9mC!WF=1lBjRxUIt0)N6Sv%>*CUQN19;!0-qJ?sQ~+!Lv7pZ6JkG6 zov@>P*!IaLhH0xapm#<6#15>?M(Y$BW}&-=wzfc}bl}2mD5Nlbf$;?@BIT-lH!RjD z&+nK_#`x4Zn&R*yJM~p;g^&qP*%M)fLiz>^!^=2xPxiR5GkV{Kug)a23?#4Uy;DXa z`%Y(t>=2A5affjj00TA#Yg zVs&S8H1CPrd%}b?x12auCL*$7Q_FNAly6dDKIf>6iY=jJ?)D~Yoawe*N?upM3Fu=; zE1b3yjPSBEP==GxThOME81gh7i0ww)t2me_o2{ZrDBZ`Xb)77%Ub}f!re`+AR|_UP zl^0rWL=6q2v`)lTtV5udRVd|4^T-^&(Al5k8Zq1DZS~_l)K*%pIc~X9 z$9#KQH6FOFjA0Y*;l*^A$+>+b_(?z2No|uGdFhOpeOCp8EWAnSH2SnoDv~y|x_t}H z?xO4;!l%)08Pd3=`p$7llDNdzh{?h%#3jjm<{E%xx-9jeP*x9)zcOtSd!$Uu3FVLl z;!h)TeORczzd5a#*2w;|<)>veObsQ>d$Q=yij+!h@Pp3g9=h_`^5lm)m$#p4(+430#a@GPOKrdbQ)7Qu(oXC1}J- z*E8c>g-0PpBw)`l--FG~eGG zRn{sO`Vw&zwTvvt zMPU$9G4f;N$OqI7ZsKhBQE5HEuZrGJU4SD;Gfv^X5TmhPj8n)x#)wa_8r!PONlcIY z=`vR8{d%f(OG++3%dXVFOyxrL{IMA+aU7*MB>jsqywJIS^%|N422)k!q>Qmg!uDWz z|B;mm^GiAjRaZ@8$Jffg3w`nzL^5{dZ&la7J!<2DvB(zfi-*xKtl}36tyhG-nHKiL z+rsCCe%ZaNd&cRj;qssSv0`w=iBJ zO}%e**`|n(VLNFEFCT|hM??W3(xo9i68K{uQppjfskn7jGMCIC)%Q1`?2s3RtDRBz zK3C{o5}-(^;jS@YqVD|OtGHh7Anr#J}{<%ady2CZrkA8P}N zbZL5_mgvH3;K{2QqQp_C(GIP=P+Ut)31BB}l|-hQi%non_ATkr3e8P@I@4BMd1f`f zXmcdX4U_JMMUn_o`${6?=3n8%_Cu;@V|3((=q$|mN@uif<0EBE536W`(Z1ZFeu$e0 zw_SW!($c}7$a5S&E|VKfmdY3@d5ELTr@J*yQl8`9KinTeQ5ork_IZ_$l|L!dQdMDu2h%#n3H||WJP;s_ zBRVM?7nwP+BS}lu_&HQ;Xqa@qhLX1=Bbm=iUXg?EuS58uK`}oVF(fTCS9kEofkZ0@ z&5Q;oU%N{7Y^ghGg;K(BHX1JYP`q3e+ez@6b}qVJ%50z?O1r0jgkHzyMEI%RGSy22 zOmkU%0>aKE?q1|cCKQMuBJU{j>Q`~5@cwWA%*x7d;oSDW@G~ouKlk;OAN|pvS@{yb z6`nc7?=9f&<2M2P<)8if$`1g4{@Y()>43fhynWwU-H)4;y)!i~X$-t-D@z-4C`KapdduU#4M4Br+DVM02E`-BI zO@e4gW%gL#8q7o{w_OOc0{(C|lH#GJ!#ixP;1e{6Yfkfb{0bSi_83Zw(9yK`I))Y(6$pyG`qoQC5F z^|S*cJgaggyCw6Bg&k0f7fuWoF-x(2>~`2OxIdfCP1t7|w4 zt%dLko3Z4->A!P-_u4BU)Hc@eW7^RsoC7GN*6>B1FlChoELmzi?872tM$(H z;rXPGi!rvxM=y77-`;Q%{AMD z#^vpCvE8vy(eC)692uKx&Z?j^#H!G#pSMKSextf$XX4FbDhJi4up)PQC;PY-8Er+Z z(o@~IOQDKJg4R#|xRVLW1a9g+J(l=*Jof*~ADKKH#i?s#gNLmTdG*FK62w&CUIg0p z`}Nco$uWTAl)}pxO3%5{hn-SfVZtIqPfjeO^09&VMD}TlKTa`Q_U6$4heCw7+2(#u zbB{Poew8*P#e$YGLx;oPZM=dv>u=h3!@ldd|8Ge#rn?BRTU~}KFV!Z!Ek#_C>W$!!yt0>ksyob;U8Wg+c98m;R zdJz$j-kT7L0zyavM1epOIw*GR*t=qHsMxz=?~2$J6veJ6D)#n2@9ejGN(${-?tc?_ z&iC!~nVsF8owl*|p*wU7*KcJ7BNqN+C71=_N9f1lyL0Ug92L#iJ$Q_}fsb*|KEH-j zx7~T@%}OrKUst+Z8x2!EHF?PScYjmd<4z{-c_*b+;HI^&@LtW`tLgYLE*XBI#OBT= zJT0fEJv5n(B>X2T3fm)jL4S!7;y)8fDhKX;9~nFY-9yCBoc}0^b(zihoxHiWRnx)K zr}*_LX@Yy)-fe)>wsoql+GS0hU9Y@ZDVEqI^odxjRF#BKW#<+z@Y>|w_w>%^7;SNu zW?#Y%AE8-^{+qUbJc)fa)|hdRb@wFZ@vbI~3xqUT-;gV)ART@`k{PvcUlV68NWZm& zD^ePELU4pAdT|RR#AmG?r-(h$Zy(*dCww?G!Qw~ zG_$G1s=)M64%!!^W5L<>+Z%R7dS<$J9+$jWzDt9zAC{o^33ONA`g0V7Hq-R-+FWXJ z$bETkbW4|gL#~ZtwlT@U6aA|>irD-HGB#l&!t1*cZc^jyil@=BAzc&VjP>-x;i01z zm`e-#z(H}H0U-{WCO(gKZ*(P<v^NCGQbgu3bJ- z6Y@1BU$rdIgOk8nRCJ?bzFVU6zL~gY;hjHy*fDUDMMT}|{1s}_z>V)=vT(T06hbdfpTDe)gI`&<;_=vlC@Jc#2%(v@q@Sp7SUEepuYq~|t zZEMN73_sE_(r{KIZGts3Hs5t{m{1h%&!4)p>7^Uofi*;=z+9h178jFWSiw9A-E-7d zbMd9P3A&_cue;`)YkH;hCHeli$t4WmhW)25UbE{#4`aM%l{QVXpI|U4DLm&V!6rCI ztY71)%9KJkxs3ZbRVUdbUJrOMmhlwPRic_ApV>LuPFD|Ym*!dib4Z~Z z+%V_{bFDtD_O*F0T{O|ee+Sg$C9R36N&6W?;q1r z+_`ZVey~AgG?e|$IhThpm=>*rsVYIc*sZtdLe~%Nm+~L+@xCPIr~G~@Tf@P|LDv3m zo6y&&G-(Q`o2^J&Qx}8g4sonk6`g%kYi@cw$CaD^9HSAK(29vETD55p9#X6oV1908 zh}noz%r>r6TzpNrig{CVtxb0^2XVRh6OH}0sr6}xt<&&c=Gvw5yrHvE^4_ht686sH zC-p?Cd{fOf@FhEvGZ%Co^e(9Nw^la6<{L=FGkV#4M89J+cS>EBar zeVW{H8^*ux$?hE>^oO0wCbS>{(*54fsloFGBme!d8Sd-X@6GN=)9JKu?*s=;XT?|X zY}X#r-A2)~H(W^aA0mF5RvsX+7MHJ(4#08yL)!O=yaGp`o7#x$9b3Fmo%Eq|TF(pXWO9}CnuZpv@$!dqta)Zf_CVCM5M`WRSM93g%D}zH4s^BI z9XD^%k~-dfN-kG}44j^FCs9Xx>gAl;?n>&aBA{R%g~AN{x-D6(Ri;@X!F^p@kZ7c= z2iwlLVj1IF%r1S_-{&f%Hi@92Z6UbO>}A7+=8qn^mHA<4YI%?G+ugLYT@RYIiCtvz zF2}^4JklGUEVm|A4+>J_oBOT3AQki&hxtfJg-bsPE|~@Q;IVADWch9^Gulhee@6(e zP~2#1h{Q706Q#6r)rJHVxuuo4^)3U!vAgB_MoNR_OYZ-QLItX%Gghy z48moLa_riftee&BDV^pdOFx8^bHVvi8<$)Vm^G`d3@mnBsGcdWzTddHol$pLDwO0j zhUeRlgYZ?m^nGEr$9F*~{Uy^ACfao~N)$cyA2S@em$NP4azb_dNU+`;u&vG}WZF`t6H5u$-zTEaPA3K=w z@;t`7r&Ukd$9!oaQOx z-suC}!fFW87wkK}j(8WJN|*`zn6IWR7M$t34csl?uSQ0*HH3lrqP8_MmhE4o)P2lX z|AL#P)iajfN?LCt54R&>WA-su+BDa|r4h-c5G+OU>jl5|g6QScX9fc2I2NNl4at2< zO}YK6OVDoH)(TM|Wx1oWCq&o0O~01B&nER##luW>t8l|HEA@iJ?2{8bLAf%yE}OS$ zJdTxVuFHeQoWICL`xCqU|C9aNHv<^nyrSh(Kr;y9E*OiTUt;{;@&2~p`5CkU zf@;vrZ8JBdm(EwPWLdi3n?gUnn?A0wF&YbgcZnCg|NCatAGZB*_c60zcPFM4+MOzF zKcvQ-(hRm6T&gcNs@0C2E`fSJDQ*5!hj`sSrhyMX@}_PJ))X_h2WpC$+xs=ej2%gn zO%wSc^->0s8D3>c3ydPzfKx38%{|M_xr`UEs_KMw%~TdMPZ|e1nxSik?N$j@x=bk z!?8r--c-LDe&#;okD1H6jbqc_U(?fDcgN8+y^b~8dn5V4QDXQT9_M6alXU83Ef&+0 zh3AV6z78PVkxdh3i(}Vuv}=kwZLDo8b@$G2J^5h++?@%%1=5DZP`$Ne#f33f4ZSu&D4MV>1EKb&+$-iSk58eH z;*lE@UogbpufIwMe>MTIbC<4~1L*&A<1lonH*T)1;Km=8cW&l3r%RW{b8{qxVg1u{ zTeG8F!eNWVLsuS6-KLH%2nO}BRt+xa?HNzW@y?Ye6V9Eo$~85T9k{F#XHq&Tg!lx6gfV>;U2@q4u?H~+k?CRPvT1{Wy3x5~ZgWo3v*Ki1 zC;hTYEblO1yj~{c#WI*|Xc>$fx=f8^d2#nH*+qG|Dt>x|NxS8RISAxq?tHH!7e=h9 zn-DgLCHD4=bP2PVEJNE}1q5hJgy2 zehRZKA9y1WcTeFEnK*~W?^I~hx^>Iey!SFhqayH^-SdK|&Vt}QGLU+&4Mw7*+bg!N zYNyU8($qNoJZjHJ-DmBF;LdCCC~#HH<9bOgw?y~SOU%h}#YCL#^$_6ih>m&|IO!&A zjpSFYOxKXKnKG{NUvxN~RepMy%VZm>&NVA`4z;w#?c#fXDx%WT(i{uSU6VK%F3}Z( znWl4T>bFZHE2K>FsE9ug&CI!VU59$@O`RVUe7vGQ#Td9;FXVIFoB1&+GKgXws zCZXiU^r=5h)(qaRirkx?CZ+hb!1V_OL&a{!zb2~q6Q1sz(3?Ve#rX2+oO zqyo(b3GXSPN$|a&glN6a{j%L&Xp^V=%8>jC`oxnr7~>&Bm`k1y%t`FQH`f*BP~K`8 z$KgF#zsx0?&PTEN>oIAz*OTEYsv&-g7))@>3BBR@{&x)-x%Ko-)m%n|hD^mFe%fAB zH)HidHQcwzvKX)`8jv@0Fw=3T3>wluYv16jkG!jvu+xVSH)LOmKjKUYtFa%7z9?Ci zT}gArE+463#Zy&!cyasE{g#r34R|;%<@)KgB%3Z|*#YlyxG>KC9KF`c^ z%{&V!f9=gaX7--RPoMI`TaHvXcbHs^l)r^$KM|?;`kDRHUvHfu`te*)|FFJOsc+~> zE;_OX-!J4F$V}eiX;?fnk5Biye%4}rzDt{&Ca?IQ8LS6ZDeDm?Gevrw`a0d04{;NG z!Y&?M*afTaF}knn;>@!xnAE&$riD2@nEa+8*Om!7Td*d&(_!MVTc0zaU`k`c@{{jA zli1Dw?0Pog5oVD;CnOACVaQyr3iGEj4y$}kEuI}K)t}xIloc<nx$J0Pdb2ypU*Ih%ELO|X^mFZvp_Awt?mmmp7&BeQqdaD%9*0)UVV|6z;M5&= zEx_XxUTyK4sJn0y3|u@3$4Xm}4Cy_xC+E}jaerB?#8I`=2lR?!vUOipO!)Ak%-kpc z82977TB}&{$Io^vc5?hBZ|HP1>&t{LB>LaSX&YjR_XbL(k+Q$1dDi{{r;t#i&b2T^(SyhHK|3mV(Ptb*yqn(H@i`gpqgV+|QenC{l= zxwTCUYR){+eWo!fgndg!=&dIG8q15#BK?ptqejM9((e);^SlUDf;6ZMRiG+VgX&NN zYQkdA^J*c-g2z?)=O0M^f_p(}V<>8>Yo3X}R4~}djk#>ue!+F^H6W%ieEY6^H@h*P zd~QLQ`5ZJJn$@Fa3qGG69%HA&@Yvb+iXY3KLko{@a{Ix_ea|lO;DX~IxZpSlE;tT? z^A69R;=W7)_H2dEfPYAPD!C?hq$_c|i zwm`#UOL=%~DW{y>_c{9x9GsAD|J&vXar8|{w(_ClAlbojV%Y@tO13v77>m=}+ITKk zcZ<|FJ~h4(Kt*b}r=#gI-9t;oN z2Df&WFxcQ?@IuM^uC6SN%N4O;KyRTM=JWiOn|hB6tvwc%Ov%kI)UA*HfE;d(d4=4@ zrMuTJne!dw&P*WmG}V_k*SNdx7IT981S8g#73C!!2kSE_JbU2Altq7PwBFGfn`Akr zGNuO((==4*q&sj#Ji6EUJ%Zu0Mq8lgmHG(eiIfl=LxC>1@R`JOH{pY!r6fSRV(7V+ z%fr|_YrUMCE>6h~Z_&tRYz8R~qwt3FwC~1Gf_s1FLyFUG!{g6L(vwxtvv)VkE#TX_kDE~>e@|t^e&f!)^-LSOAuC<*lag}@T3rDeo z`vun&eu7W8qI$t+TBQf^Zcce8`5`QoXUa(l>r0qX+%+0?`lH`B{PI9|*@hQJ_y~-j zY}}IRNqKaXXrF_wgFSD?Q) zX1i~cjh{NfeTy$=MsZC3)p^6!NM~o$93GS!KMIzkGN&#(w_wU7zTN3hJ`XGs{CM%b zGQ&CXu5U-(LnZwC5xLO9>HKb&7waFR;li0N;&WOWgHiujV%L(#$AMkg&3GR695U2> z%YEN#`M&k48W~-C^lKb(|o0LT!%3D%rA7$dtcDH(Jc~73- zAM_2YD^R*WYucRp{dvxND|U;pvP%rdC8VjBv?mrhbL;$r1;t8mV+yQ=Ke3uRS7sK1 zT~;kGVa=k$+?AKVe{S;Gpf5C7<6viieCwtboP%HmNWvM2pKG?*Ne+&F4x>bRp;Bu1vP3KI}kb7mJQnu#h-2CDXUm<_JsJGujh^M z=Q0OI+Nr>1N~uj$csTv{yaBQuir*MiBQMcO_}mQE1qHnayi4|^t;Mu1GfDQw`?=e4 z#*Igsu1z=sb2rASXP+Ctin#%-SVLY5C(*JiVa_GWd!lReG4;BD@5fX6Q>L?h10NtA zPn%_e+iAw_jnSYehgo85T*JMB^S5F5L%MF7`=XpF2_7~k`~eB>2k^12TN_P=~W z`$@x|#XPJ_v%|5zGgO|P`_^wcerdUt5q7X&z8tu}shxIg+%4FI>EdPU3)t!bOkY!UwK&JG#qDCG4!50iidkZ$ozOHUEG2g>1`1Q?r9Uc{yVOx?pOxf z6mmaS`uuYprmb*7O#4r9S7(M~Ftai6q6;bh2F3f1W+J(|pT(9jRO{L99VnMUcJfeE zS`eg1ZpY`%+!&|fH6ErQxX1}-Zrz)EzCWkf>chfLgWXyQy+CTab@xetDRigATz{i) z=8{DF!5g9hS;7eVx!Br;TH>~~rq+jd5ZYI4)nVbGaZyj=ZP5(aQ>iewNo){XA`xp7 zg7fOAn#Xs=NgP^T05CScY$zJiq!h)Gfa`sH^9alJky_dqWbY)o5 z4}Elnv^6w{X`vN)^nn%THJ}Gm?--MLZQ`q7lUXnzg_>$ivHsUJ)97JJPhoEb`$d)U9oUe z4r(cm}>HaL%|euCSm+YyeZH1nLoa1&ZhNr7p`0WAz*U0S}@rg_W@=Up5BxxBmP za8`483m=EsaLAei%yP)WJgX260~$VA_|KvqXZ3+@fZHKm2LR`G5OsJ2WI_L~UU%*f z-DrPw!K@2rT`=p?4e-;ON%j-ixXQ0WoUN<^x*hdF>R4A=%b~q9akng8q-ywf#ZBnA zT{~WqaPN9!bWd|rz0{Yd=NLY6@^eGS{_sG&KhpKtPif%3rc=60e-T1y;ZFAxyOCj+ z<=PLwL|Kd*&m4cA5aZ}UaW}nqG_G){8^L`kEWeHL!&A~+ah|m^V|Y}``4~4ohr%v1 z38lneu6q}^T#$B)+Z~1Hz^x5Q&ZmFA^dwQOcFRFLZUU}5B|ApQ-=`FaUF0BNY+%vKZr*Io==Qp?gRB>-pqIep( zX3WfBl+NwmN_#omJ$s9?XY+zp9LwS9F^;=ZJe|4u6EL@!TjoM8Wm^+{l+A_Dz1J(6=DP1a+6SKdx;GcwET&D5zZE7HJk+2Q9h3yt$* z5o{p*{3&#yl@ffIF1@V;4@N&^`kSuv^S$`cS{X6+@%6~}>fW{7*ycj)#^%?`ScjeF zKP($?ROSssJl;8DrnO9RQV-R+W*Hm+MgN9=_tX2m7FFbbi8l*vHU4ZOF zP&~xVqX;7=eBYgWvnto28kX@lgEE8jCXuG4lQ?grX_2LO!93D5s#Zo&MrED1B7Iic z%DwFKm+NnNxIuS$8n6q)bKQAd(;G^IO}6$|;!-}T$3ePy$!g5U&zRlHSK>-usEJuA zr!FQrb8}GW)g2#P?fFG&*u*yu^%KSWVENa5nb)N|he_dwm#4e2klnDon)JS}Vc;@O zxdu^La+%A#`KI=XztC$LUi|(W-*LRZ#=nQD>W3|+(rT>h`^g`l>aTcaf$z`t^Ofe& z<&8VJpIUV`@M=(s!=PgT_x71Oq)NvA7L4!YCY`fmYdiPK6^w0{dWF$J;Ljfp%c;TK z1~F#3&_hz_-cCwMf?{&#s#SQZxtAo@g_8Cq&t({AP+<}~*tlQv$W<2z=R98v^7X#N zi3}YVC%l@*uN!BYaAPN&#FvyFRyUPC<)gp7z9-zC8@Qbo+0Zs@_i7bz&P2WC2a%WR z&%cJt_ZPHj{bWvwH3&Sp`-N(8DUp%R68?fjZqT152hj?m14|3vf(*ENSV~lWu<>$f z5ER%3$|H0}T8ESz>%O6F+06Zs_q7K8rnjosyhqb(pvPBOyLjgL z7n?t^i^&z-m6OJ#yTzwNl-KRcx};LWvL^xe_H4K2fPT|#c1TuBF*Z4mPB_<|q#J;S z5Ldn1za`eYTTyt!>8r(~MOYu1dPmZU6%(FE~mo|%%Jkkoj;dJt}M5OV5giaVzA8H80v zIfp^#sUaSOnOc6q>%HC#Lmz64C-@0h-0OXS7s)V>b`BfX&vpF+>MzJH%%9`Gi@5TS zDIC9AsxFr1f_qqgI2G3G6+d^D%}w{`HwQP)-w;|cTowsg-}q=t@w?Gc5A>^l{B$G~ z>&6t6o1f*^H2)duYW{|O4)S};HL$#4$DBLuO%@wL4cPc%@#l{DY1^nc`dPivJJBHB z|6ZouYV)u4PPQ?`$M~jZ7ZuIqJLtSeyYT_qpM|rhTG#Y=&9@1zg#;Vj@BDLz{m}SJ zTJTv1K2tp-U;Ej`sO(Ku-)XhbW_R=0SmVbKAY0|ppq`S;p1*{_<4&>m&nwQ(<-ODE z+||Gdhj#b8^C^cxl*7~)X}^Qs!xnJ7;3eMC_*P1% z7kJ-*u`m@Dz#<5q8;Q^P9&SZu!t**IWt3Uc?+MCU{tNhd+kL`GFl?$tFv*3d-8r(L$#xeWu zyvvkB77sq)_1hb*X?pCCm%X*I-)2V*mzIBh!>?WYw>+-<&y|iG)c&&T z4!W|_>1STj_WK9!dVBYbKfhb?Li)B}jybQ}4v$}ZM2FnbI~9#RYT(LxdoNnp zzI$k`cUHG(SF?B3Jx4wI%93$sob%!APoCay@~N*4xPH>G^`Ga>xw-m%v+6h5^_n}L z{B%`uwC{r_pER}Oo#T6@{jgPb>*4nv|Hy8azW>wtb@Jaj?d*#(Px$P`p$*?V7OUGI<0HB$KUcS|9hp&-;D4>-6`g^!#S$ zX}dLg;>3cx$8B}Pjt}kf+pvy(pIk9JcHukA{+RW`A?u^7u4}ol+EdMM-6rGU1`EC% zGxne(uixp;KD8TvH0QFjp1Jtz#j6kKfAikoFMaxp``aHg;3$? zIYW0(cl$C@P1Jp;UcSmu9n^;qgHEsjc7ZnFeMp_p*f!mLS7^EFKJ|PZQrCZJ-xYH8 zUT(NHGFEVnpfBQEL;9|e?jNb&6&jq!*IV?|Wfr(Txir48fUP}(F@kdd9M~ z)iTCr53BR#`|($c>a`vv4h-)R~f!yS4C1m&4QSj{-r{P#|TLD6Z*ow>c5txu~v zCx?&R#CY#i_hqclpb^|6^=iY2@CM!Jqa%G<^bH58bT}vrpBw9UP>UqLE&4jY^8cs( zRipY<<6qVAUlp_}u41ZkmPHy>c+04a|1BX6|C|3W!sYkBj81>L_eW3-Hd)qU`G!dM zmVsA2-RlGkU>SHdYI&Wo7YiKN*T5c@fmbs%!s^~`)I^bKG)9Uhj%wca)Y3$D;5bfo zuPN?CGe=dgHubn{s(E$sAE&yvJN`sHV-KX*)2NRWdpW9k+u}b?buWs!h&ifKyUHgW z`?9I-wI&?V#!=1NnfT&V_Zna>8XAp|VsB#~q}bO{&D#$DiR{X8oa$Z&+=-6Hfk@#$ zpsIVFB0dX{3nEp#n%pI}gj)FDe5&z?Or*MZ0BIHN9W}gVRcf%zsD{@GmQ}6hZG-zn zcH}rtb+0-8L<^%OQnYeZ^LE00oa)|wn2WZ?{z%czQPo?<`7MtHRjYbsQ{7ueISa3n zAP?ZBQT7~*P7d;yR>NBW%fPG5F)V;(;86yhU_q5?v^f%~%>D15Wy|jVTk_#RYxxEv#d4hJ2-}X`a zrzX#$@<`{oBWiQ6ug$%`HuwJ8-0RbM=J>Qlw#DA7f;lXJWmVF72Kux|wwHZX?5oz| zS(nUC9LK>w?v_=hofh#~fE3FdghLuSRZ6E#;?oA%M(HD6%fPFQJuHA_m1}Z+lt*=Z zZV9z$`-ya(*XcZ`)4hG6F*Jduus^he4&X&cd{p0K4rR>9mK_X$RM&9h*osO#ewz?^jhRjjFCaQH|${?#rpqhvaAF zhTpx*Zy{Bp4YXxwn%d_7Q->G#jPRTUy2k7@V4j;l*^N)DN4`&&-aT8y`u884Fm5Mz zGqrPJIj-bpoIZ;C-6VEs608(e{1QW9{}_rv!6T7Kd~5~R}h z6o@CRlz8c4%jmixDuo7YEo>G``F-8tDJ(70 zl)8hba-Dmu4V=`)FUy&9*{Yc1>vlFG;W`(7&5h6ADyvB3HaBlladzX4^}+b1&7*MF zyFVmdZid2hL;CEL;DIDXn%VI=69DD-H9*;ATNizMEa_W-Dg<fI!StfQC>XAElPKK0XSZ@~oIa+k`sY^{?iS2S%acFZuzZ#c zCbP#g%{d51Zio6?`v*q;I+b`y+YBeqThBdH<2mcD^PfY(*lzWknXibM%ol;aDWUtI z!uJ1`9_U}bjXP#fnmRFC&j|N`A`&)qWW=UjJ52%7T)@08JP0*QV@j_4#$xqb=C+H? z>QKLJlNRG+vwc~=$p6P0XgBtbJ1uFG`y4I0%tdNpr{`r}`)RKxe|~-7=_emn=b$Ao zAE>oKv5|QN?ds=_<}>x4j=M5r#YJH;NaS=N-VV)X=@Swx#LQ`6Zo~C9?P4=K)TjRI zhk4c=M%I_NvVPpgt=yNHr5+oZs|*u=Y@~aFx05HA7}B3xm_5_Q(Joe0nA4w`Z;Al5 zvkFf7CP(Wv-Mh)&DoZwpo#YRMQ|`e60?dQD&C+l8v}T)$#@fvqX&-!c)oQexC}ImQ zf@g6!|BMADDT_6nOC~r3HvW2t@F@N~PYlCNZx3fL`jPoz$i~)85Qcxm`tji^Bjt^? zS8BMp^%J&E!Y!e}glT2Enc9jMT3NxDEl2}9c`+4iwl-4=^PMR_jrplp~E;D3>W zqSL%Q%`s)AUcBsBe52cKqFr!ZDWG8KhD(SNpZbxFAa$w(<0j>1Phh>k^vRR5-3(@( zz4%~k8oI{2sz6nszmyg^gBhTA2l@(lYw+8xTOqdwU5s^L8`u`=!gjDd>;OB$POvj* zPe%^@Lg8{TSwi9H-ILHCr960UXc$feaAs-Hd!(kFk2AWBp=8e$o z^;mY`QH4C+Aa$>p4qB|h@-9zn7qs|*GG>97dSfHUDNI2+D^bKyKVA1;6k;Uc&gE`dwo zGFS|k!xeBPTm@IdHE=Cl2iL<5a3d^%o8V@+1(w3Ca2wnXcfbQ_o_81WZny{Th5O)s zcmN)Rhu~pY4e!GTum(PakKkkY1U`k&;B)u_zJ#^#6?_fr;2ZcBzJvAfJ^TP~z)$cq z`~ttiZ}2<(0q?UJCt&w6sqb;(X(HfIG8SUC6kVKzelpyF zF77hU#Oz*laUa|d4;l|4ABJV{2s{dp!IKF*gC6iadcce5V!5#bDPA>RM~XL%w~=C{ zp|X3=cpoX&7#|_UC&p(;@rAJ#DZVz=A-_xDdvu|5p#1%8{E8I68-F52gfbFoMir!} zX4F86T1IW8*xIOr6x$lxA;k{HPDrtfu^UoE4b_9)(f2UwBSizF5mM}9G)9W1MsuWS zX|zJNhBmMtw1xel9UK7dp#yY;1ECXihAz+*x>%!ec3NYHh56dVo5z_D-~91kaeuCo&%xW-OFKNU`c z(_s;u0lL=Cg0tZqI2X=?^Wg%x5H5m?;S#tME`!BzIa~o(!c}lJTm#p_b#Oi005`%C zxCw5CTVN^N3b(=Sa0lE8cfs9o58Mm)!Ts<6JO~fL!>|kC7UVsUWM1-b$A2bgty>rcn4O(DtH&(gVpdpd;n|UL-+_jhEL#A z_zXUWFW^g93tz$4unxX~Z{a&w58uNN@FV;LKf^EZEBpq(!yoV`&<*ZIpc14(WvBvG zp&C?&8c-8zK|0iitzc`&fI6@ZYzuW^JJ=p}fE{5c*co<#U12xKgeb&dcc=$@z@AVa z_JRh`5E{YWun+7DjiCuNg=WwkT0l!^1+AeC><4XOe`p5>Kzrx_9pOOe1f8J^bcJqk zFZ6((&$k0=x)|;8wT|ZihSIPPhy1hI>Hec^}*l55R-)5IhXa;1Re2 z9)ri>33w8of~Vmbcov=mr7JN_OF8~e#ILwFCVu5V$oos^%D3{k0$zbvL1|aM-hj7^ zcaUP0@g7pVZ>&LzkBm={;xpq5q*!ZwjT9VLEHRdD5k;YL-aWotQ$HH-NJe&Xv;Y2vuI29>QH_kwcvy5|);ymL5 zq`1hq1Su{vE=P(hjjNI3TH|`8xY4)?DQ+=tMT*;vJCWjU<6fk=-*^xy9yXRCA2l9F ziYJYyk>Xk7d8ByJcnSHk@d{GBX1swEZyE0(#VX@HqgBdPzSbwZJ{n~ z2iwCAup{gQJHsxpE9?fD5QP}*4)tIU*c0l*UeEv`t2hYO`@FKhf%i(2M z0k6QT@EW`hZ@`=I7Q7Abz)DyJ@4|bq8s3KwU=4f-AHm1)3498l!RPP=d(@X4^lKXnj%GWqa{+bHugh`{fz^VqJwcDQgk-D zB1LzjCsOn_`XWVtV<1u-WDG%yp~i5eIM^786r+tXNRe&iAVr=r0V(p0!;xaLQGgWF zj3T6%Zp=iA*~VO?m~R}36h|4yAjNUU2}p6GaWYbzYMhP~XBcN8#W}{g$n)TQxBxB$ z<@I8?1TKZk;Bw zV|<4c-y1(7#m~ks$Y0?%_#OU$KS6_V5vT-dP#LN~Rj39Vvd_Rx9oPo8g}SgEY!5p? zS@xOOMIi>eLp|67_JsPd7c_tk**C*Yb7%oAp%t`-Hn1OT$i5S9Izt!e3f-VP^njjF zmi++i2ErgX2nNFt$bz9T42Hu9ko_Sb`%xhKLqYaqAsfa)4&*`}jE4!ZA^WMgnFfVW z1jR5NX249C1+!re%!PR{AC7<{VF4TkN5e62EF1^N!wIk<`$f1p1I~oA;A}Vt&V}<} zL-vbtb2(fASHe|rHCzMN!iMaZ;^tPk4Q_`!;7+&;?uNjAD!R%^ZXN{nNj!{P29Lm_ z@EAM}Pr#G#6r`4`+`IrU!b`9mUWOI$3cL!h!Rzn_1opxCSc%yxco*J-)$l%i0Bhhw z_y|6RPav=l&Y$eo!dLJ$tb=dhTlfyv!}st5{0PDM56-*X{0@J>pWyAz^$(RG4Jtzw zs0!7fI@ExgPz%zbHf#l3Lk84=ZD3od3){i=umkJ}JHgJd3+xKJK_)~Y2D?K&*aP;2 z`k+3a2FQlc2sHNo5$ucJ7@9y+Xa>!p1+;`#&>Gsne$W>7hjwrPw1*DR5e|e-&>6Zw zSLg=ap$GJYUeFu*Kwszw{b2wMgh6l+42B_)1w&yN42KbLFdPCSVHAvpLtzX|g=tU- zMNkaWVFt{ESuh*sz+9LI^Wg|M61LR;0>U^7j)r64SU3)jhZA5SoCqhu$#4ps3a7#8 zun5k8GvO>a8_t1q;XF7WE`ST+BDfeXflJ{sSPYlL6>ue71y{p0a4lR1*TW5PBP@ZN z;AXf5mcp%Y8{7_ez@2ax+zt1@y>K7g4-deD@DMx<%is}s6dr@e;R$#Wo`R?08F&_+ zgXiG|coANL;{<-g&6D(^RZg!Wb9}*)R@rAQ$pr zJWPOzkPnB!;V=m%!xSihsW1%+p$Lj$I?RBXFbihG9GDC9U_KlHN5TR)3XXGG>97dSfHUDNI2+D^bKyKVA1;6k;Uc&gE`dwoGFS|k!xeBP zTm@IdHE=Cl2iL<5a3d^%o8V@+1(w3Ca2wnXcfg&n9A1VM@Cv*Nufgl^2D}Mx!Q1c- ztb|qYF1!b;;eGf3*1(7G5qu1vz^CvT{9W;UPF!EWm#`MTg0Ep6d;{OYcd#D5hacca z_z8Z7U*K2x4St6|;7{=Wh#*adcl-5?X95QEOp1-e2v=ng%gC-j2e&FgJ3WWfh-sb z!(cd!fP>)>H~`v12j~a~LMQmU;vY#|qhK@~3S(d_WWzYffn3Og@h|}WLA1Si8O za4MVzr^6yR1I~oA;A}Vt&V}>fe7FEEgp1%}xCAbR%V05F4p+dHa1~q)*TA)K9b6AL zz>Tm3Zi1WP7FY_m!fkLn+yQsOU2r$t1NXvxa6dc%55hz6Ff4;d;8A!C9)~C3Nq7pL zhG*becn+S27vM#B36{glumWCzSK&2y9o~R9;VpO@-hq{{3f_hHU^ToCAHW*;5I%yB z;S=~2K7-HU3-}V&!dLJ$tb=dhTlfyv!}st5{0Kk6&+rTU3ctbc@CW<}n&Kh0&j0_|2~Wvx#raPu zkcz`iE`ikc-zEo8;o-l!1U7X3v&h}Z|LPq6|4n?w{r~^(x&6CxwxR3)?@C8SJbzya zRGjm_udM%*F-;&avAO3@>y8rnPo}LRtiQelW)Owg+&}Hxn#ksc^S|mZ7<2nO^RGDd z7f8)J{{Q@U6q!5PGI(qm8I^q^fwRD$#xO*1pA3!zcmHcLs5}2_`TqZu-{4&@sMG(S z@>Y?TiW2xYmcT!Kt^L#S{~OOqMVkKl64;b_r?yqaUquO2l)%Q7fadyF{8f~|=9Ylg zi|C$P@lLn7C0OCFq68{RprQmSN+7iaf_~Cq&xznTkb${mU{+cfiDrQxxz{sGQfB|l ze{Yj{wF9+}h@R)G4MfG?zq|yhlRi=LS5X2%3G5sw6|$lPDoWsgwgiGUQ6T?k3!)<9 z6(vwn0u?1d3245O2;Kuz-~B#Fa1*%Qm^n^*mn;vg-CuAoPLzSaz|G$+6VoES;_p9H z0(ICXiH(vXUL`NhtIVcJRlKTRHQv{1u(whz?W^RwaKE+LUy0q75=_`v?_7K1I4^wX(0hjVan0bdx()lc)@9$;?Zf?A{Wy2@cJg-ic40@( z^VyLz(~EjBcH*q(?cwd|)%W(oQ3J1`*T~!3+sE72YwR`gn))99lHX>V5o&Y%T;N^k zweT)-zFUT2TpVhxyw=_&99?QAZ9*HqZ-joCx7cgzUG7Zx5ACi9wRYYC-jy6(WhU)I z8@_|$bxb%asUGNcDrxu^n{@WNlo`X--Zi1!wcd51F6Daf2Jc31Ns?LF2RD^%wUIVm zy>8{W?C$kQvb{O!=&w4s#asGU1zm2qw|cjEw|jSlZtnE%3ibbH{z?S=x?Yu%p2~s``s_!t}^*oiw8=H;9zI=AisyOm(4@oD71&o-7=GW z$;z}xOg@Ty%;|@^-^ckKh!Yd_0IW}@!r+H__nL4Z45Lxp;kp z?*0u9-bhJ=KOXf~ifP56O5lVNH9B_#1+%b(j}3VES6iQ0A~$*EruU@C$t8T0>)}6S zsX3>A#os9@xf$=hvw>(%P4Te-=l^BaEC0*VyXDDU<$^fP&gujUQ1hBg{+%9K^#APa zpOKQEiQao9vz=seW{Qs_%l|hAXN9>uJJjNt%XjK(_j^t$_HmbgulD}m8(l?g|8*ts zfw$&AlI9P+kCMXq*!v{u`0qS8%9bm_lvirMoz7z5847>cL@$VXjzMIoB!{d$1XB79bP4csQlIwaL z?cq%J49)9@TKQ?%%eiY1X;`NFMv=Xp-9C|hoopOw5@{M~7P@a9Y2oyikyfEuJkHjU zHl>)x?HPYdux}gLpQCmOCZ2adqQ+dU+}zs{Pu?d?kFhH6PcO;V=RYhC^T^i~@Sl`8^cIxFeb9?=<&&y8B(^e$Q~fXS&}F|BQ`fN5(~RlEMsp z<(g?Q{;nCnn!Q$)Uw_`Oj`*%>nKL8&S;20MzDze;TXb#HrqpYjHWmu%RZGjvtd!O& zQY|f-Svl?9NF?nSsGb=~Yr^r~G#mbiR8D&jvtOYG=1no*2lGGilbcyBEf2=S1ng$P z?9A$Eb6_rJ7iFfWeTDxoVJ)nN_ajx)cFnAswj1Fzt5i9yd8PEUuOoHSzKnKHy8|AF z_0j5S)hb8Q(kfR^dl#9G{2125Pw*vXLy+$wM&CYIiL9>1^00zNe$bw-o0uF&ua43w0 zagYZSARjJ;^{@nPhNW;D+yQsNJ#Zg901v@3cod$1r{Ed*5q^T7;TQO$S%a$esy3*) z2h^|nRE_6qJXPcQ8a-?Et<|$uFX&h6$y(3C^R=E#e>VNe^rztI^d7bQ)b3HcU+uSQ zzYQzl-P+5%N4!ThMn@}suS$=5S7biHmHGlVgG!bBOLlQ4QqaEkO-lc*Y8ZA0dpJC2BRpFtv?XR~vpa;2)4?|aN`!%90wc0}Jf^sAWHN7Pn|WoAa=?VXPk;to=$7ENh?LHnj& z=3W+Vro}7Qdzx88C-=4^KOBuXm9*2+xM@o4DBb4UZK< zv!o+G%uf?i*~0AOt({g0KGdp7{p*lm*2df(fLVK|w@dKdG1NPST4HEjN;UOe{1j0O zz#rA@7P?a_Ao%NHrfMPdaeA-N-3mq%U-e$2eJ~)>q|)o&i`Wc`yc|idw3REh#gXSj zx1T438HAk`+71h~z;44*&wl@JMCf}&B%{(ccE%1dQ?(8Jze7WJY9aV-1J4^9nvM-! zGHUKjcdVp#LwSFPg}#GlzFbdoraM;}hwrIQpAx!z%X^!#h>4NIm=B`0VjpKtVMfR+ zk-C+(cPYOm@?mD7xh{;%Kr0UY1}T|W!X&U^R!BK^3o?(+EVdjSpLuL4E*F+!77shF zos@aXM!F3Ae4Dw7xhy|rwjh_MWzLHHP{RNE5+;FP$=Q)bW*?siqggP)-0|csLvCX{!~6xpji?+Gbh@xvkT;4&A9o>~rtI%xxpvMRtgM;oa|Sc8OG} zRL$~zueohhX{X3e=(~h|m9Fn2dS~4|Ga8A%Q+|>VcaZiyN->jN@SeFxW&;a1YVkVT zAZ{o3;rCoK@vPshQDmP;W5WL0d&b!;kJPAC%R+cAblcR$-Z(t=Q`Nden%0#t@m-eu zZJrXATzr<`yQSI3-%eX6_(*<3^+Qj5L*38ZwkO06PCp>Q_kp3_In)wU*tJws-^I_7 zZiM-bJLn#|(;Hpz*V9b(w%6C`y+e0F-(>&Dz(~_dKX@-;b5LYOq;{pPU724Vc_DQB zMM9WC*h53x;h`4TZCL6(&i^|&^nGxoPNi+_e2+9!z32IVV?uX&+tXW~{<1^U?9ip> z{8;_IMcM zqoKAjVXVkpj{Da`Kd*+`#`vZu$GP9v)yn~!==Yt-8!pGMCFT3Aq@%J^ac$<+#CKyD z=k=jhwx2|s-(5UwHZrzEpPu(f=EKDHL>S{^p_c5f^rKfYU&j3#p`X`6t#tRPrv07! z#;(ihw^On#4>$1j5aFM_w&x? z?dVC)_gm37Ildt&q_)xh%k=HX{ZsUmBp*S(Bzr}BZ@~BG(bJso&!V5=^2U^q4&H$8 zU!td{_=v|oF`8ed@86uw)aVlD`_|~vk}ji>l4dUOyP~%{o5JW#WyYNmty6~YAbrKr zo6Gbai|$^A?=Ihz^LTb%EgqsuvdHYucL(dK3Ne%;JxMW0Xd5u{17Q?zp#z7t)Q zrGFUxXam0IL|<@mzaL$V%jc5PF(`Ua8NNR@^Lf!1lYAtmV{A0L4BtOQzjHP#qKnG( zofuNG{)UD6L3HV6`2HkJ(<=$SUro@HW82Wt^2(mK0ygO{bpJY9Bs|)rFmU_`K38s{ z@7m`6gUUOheYlCfgA{&Mc`JObbH2abj4)ot&nuhgZ&2v|4YY&8qaaNiD<8$@s!jAg z+T5?Hyg%AUo9H`8+1Hg1#pgH9_jj8aMi5qo{CAhY64x@htDN>pdFj8&`MY~Fd{>HA z4)1W~#gJ@QHM+q0N{d$64ByqFM>$_Lqv@OBd#mU|=c`t<_Gb9rI(nk>wN13{X7~@MX83LrJ>U7- zC)#*3d^e3=;C!`+w%QEeZOl*0XzR`Jy`TAM7j3^8zB`(q1EL)^!}o#ar%SZkX7~>3 zdDm$7E%4nZ+HVVd_l@@70^ft9S)1W|nE4qJ9l9C5hnt^~(b1dXdyM%R7tP%a-{Yf; zov)l|-e&lo5WU>_IxISAGki~pUgvxr9-X`yz6+w)J6|tEU)&7eFPWcJ(RVk)_j{Y+ zC%86zIknle&Dqq$0x4=%yam2zG@H2vzGpX^vjx8AHJiT$zLz(9c?*2M((KhO@cnwT zH@3j{Tg~3y0^ci}t#ZB>bN+)aQ0-8nu2mi1U{bG_w}8ym@V^aY&+I$)(D3zYNbRN} zH`U=zkQ?UYxm^cjUJJC>fXr)y_8gF|eRH&)U-4)To>uTn-v%BseLLhbP&hlnBc^Ll z{zpyU75SLyYJ5I!dJL%G*d_Vo}g9y()R+b9|gzGQkUq~?OijrPD1RhZIyfYu~R?*&?|C%q4>HeI_uyl?sd0^;!nm!J>*7RKDSEi3ger@_hq{dto*J1Fj>64J(nLY)% z-t?)+?@cd6{$P4B@@La$Ab&A^7V=lq=OBMGeID|6(~m&@Y5D@BDyq_Rv{ciNMXzM~ z@#tx$FGSZoG`T+sGy_8VDNxn))6lD#z6f1&%;e@w(Ch~3XM^UPNk11f+d=yIpc!b= zF9gkfkbW^}ZkhB;L9-g9F9xk4k$weeZHM%$pswlHpl@gTb?DohegparrY}L)+5-8# z8Fn&#Df-T)--fRF|8jE&>}vX5=$iW=^LrrE^!v~?BVXnZK+N=q&^1Rw=F32H)TBQO znjInianKw!=}&@YM@WAf8k+trdLz@HN8j7@7t!}IeL4ERrmsNP3<-tvDrk0^^w&Z2 zBBZ|wnvpO4ZP1K-=_{e7>F=VqGJQ3AYtuhK*Q_a6CSD;(8R%{-T$fz<3X>8htgO;_F2 zY%`gwK5C|dbk#x4OqZ^Dcd+TIYloPw`ZdyY)u~aYs~%})gZ!%Q9BTSrNX=!Cx$4eX z(^X$I??L9OBjZe0y~r_Lbs^Vu)dkITkb9l~iKgqE=bPRZd6?-s&xf0?b34g&olnhw zP&hh=Q%vuIEHGW?aH{D#Z_`ZIxhgch537fnAF`I70! zBbS@L5c#s{Cm~muehTsx(@#UbYWgDNYoNTGiG1DkbC7SCem?R|(=S54W%{Maw@p`> zy<_@S$d#sFi+tDg8<6jreiL%F=_=RvO}`!a0jS*WLas6W9^}WS--rCf^aqfin*I>- zGt-wLKR5kR!Jc|Jrnw_d3&6#^0E(a{bnHmF0J)tNhlR zt}^@Hbd}RjrmJjzHeF@&3n)D*k6%q!zJCLyNBREUbmjFA)0MwJO;?`eSnidNi0Llv zrYpUg7b!PNV`bBSK(AuD(x_S8a-(!3=-cKr96@YBfj zW=Ofy`PB^BeN1nSlpmS5MQR?2^aGIc&wsBYKTS>Vgj85^(*@bw^zKN7%YUyIKbo~A zy$?T2fm7x6L*8n7f8=eZ4?x~-`at9zrVm2iY5GCPyG$R9yxa64$a_rBLcR(7_lENG zmg&QhZ<~HF@*UGhB3GI|8oA2!G04@RG-o5xP1JiS{Tw{6;^25L!*@?oD-#qlr zrjJMOYWf89Zs-VaB0t4uei-^3Gd~>t2s59Ieyo{S$B*Ww@!y+*xn@NmylMQLfUbCp z&^6yd=EeMI_JZ^o{AezM^jZ99W`XoM{Aiwl^m+Vf)L;4${Af&<|K0+0&0UcBG5lyQ zf%N0}(Kx^K6Zp}nz4R0L(HOk+lljqzy7W`|(MT`mu{<`Ul(bt)N zIr=xIUxEIu=~tq!H~lK~?@hlNU7Z0E*YKlpVCmQKlVr&FC7*MYwB4 zqp{L;eP~=)x~>V0R!Y}(pz%}bs{0x}m3|*TLri}FeYoilp^q|s8M?+l5#FQxXbe;O z6OqmaxA?v`g+r=qJM9Cb@ZQ1uZjM%>FMYvRPp_8gARxOG(8i2mgzC{Wv16df7J9n(O)rrFZ4z={PZ+L zZ*KbD=&elO7kz~3P0$ZEy&3u;rnf*JX?iR4QKq*+A8mSD^g~T=hd##i_UO-<-VyzI z(>tNBG`$P@D${j8e$VtC=q+pd>F1HeiT<|fv(ZMXJ&%0*6 z0KMf_ewat2-){P`=qpV>9(|SR3(?c*_|(1qB=oN6O3%sY{Y^g=eX!|zUJW<>4D?Z^ zpM{=p`Z?%Drk{sC-Si94XPABw`b>0$cL_hU%zQEWY%{+C{U|fP8vQKOuSLJa^y|^D zFkR2dYfaa)af#`A9^PyEt>}AY_<6h?T`M=0-#gJGrr(WT$@F{C(@eh~y|U>KqE|8f zVRWt9R5*{IYrT;4$Iz>zBfKa0squeI-38bc)%*2vy1P@lySoHIy1To(ySux)ySrOJ zq(Mr$MMOYC(RZz%-~aWF*Cl?u*=J_X%$nKzInSIu_y;)>ZQKt>#_oHg;0tn8?EW?y zb{`uZUz20t8*)s1Q;vmi%dzoYIgYIVgt(_a;^K$0@9=f~iT?2Yy7qJZ;cs^Oy}XkB zog(|S>~DM8Z)AVZ$aWu|MAm;oywe{^@t^WHwD~M2!+*)i@mDzo{#QnxhO6o7sJKm;<$ud0+*D(#ox&#aT&Q3E-QbB%gg?5sn=Ce_V<$P%JTQv z#j{+23Ze2g?m)uWP8>5RZ@>(Pp&V z7>|{k;PG-(JW+0jC(F(8RJn!hQb?QSOX4%U$qRxhvi-cf&j7?s&J{1Mii4;{9?jd{FL<56gXIzu%*BUwlID zhfmA>@mbm5ruF+eFZ-KV_62zmz9bLESL7l1nmiQWkcZ)$@^I|&@(AoP@<@D79)%ys zqp`=uWAGDsEPf`B!!P9V_?0{Xzm_NBH}WL>R-TOC$y4wLc`E)WPs5+(>G&^s2L39~ z#Q(~(a7c}FXXDWF92{1ji^I$Fa71}Nc5av8$f_^JQRQVgy1X2FB9#@gk8f;wC5|Vr z!U^ToIElOlCzIFWl=3>9MqZE8%NuYec_Yp$Z^Aj`%{aHb1?Q8u;)3!vTtwcEi_1H3 zNqHwOE$_l*<=wb~yhnB>E6aOvRe2w-F7L-R=g!rkT5xR?Bc?CtL>pTYg*vv{C<4iA=p#6#utc)0u%9w}eI z{u%3{?DsWRzJw>pm+@rz3Z5ok#WUq=c#eD>&zEoDMe@&hseBW!kZ3MzJoW) zckvea7rb4*hj+>M@m~1>J|I8Dhvi53nEY7w{yZr^!KdY?_>BAvpOc@<-aqH%7x<$5 z5?_&D;p_6R_@?|C-;saA_vAPDq5M03BEQAYpOglKp=Fl|$oD`q|g_>bf3Q_MNwO1UVdzEQiO@F@*L5@Y6Omb|T zRrWo%US~8_t{_G?tZTxvjF5FM{J*{5mK-qVw+CyaDw`vcUeHW@d zO3p8PnPX*tFU%e<7o^QZxe%Tr`@U7bcDn4lQ0-Z=?@hJm%D&UoULY5jz0AdO3A|MP zmNv`fl6aNudsF?|wX*LtwKvG$%U;fAxh&o)m!r*gxjfz_`#w{@cCYNaM(qP~71_%< zEc?4j_EEVSZH~*;@hRE&jQX`_WZwa5|0w&uQ2T=HyFBg7vcKQvWnPo(Q@x!C_V?wc_@Ug4Hjm}z_?hfGLH*j7vhVA(U(2oV?{aJWPHrQ6d;XN$;!kor z+I*4QqCyJ?4(JK=D0XB<)PB76B!*piPsHWqNw|_c8CR31 z$Xx|K zFT|tdMR=UN7*CRy$X@<5c`2SLFQd&Ic{yGnufR*>m3W1`3a^n@;|=l}*~{M|uf^Nt zb+p+fugCl34fv3}5g(H`;ZyQvd{*8fD@GxHlDFcE@;2G)y)191eofv%{f4}Y`Ym}k z^*iz&>i6Wm_<_6+Ka%%j|E}!-exmw8{7gP1`*X_|^5H=J2!5&hQQ6!1t9%ThkL31BlZoc=Vfn0DETKGR=$8E$QNlJ zS-ym$%9m*qUA}^cX!+M^6HE0Q)Z@rMQ;#p-#0lkFII(;i_tA3h%HFo5s{6iiJGp!h zr;_i}KCS!!rlX^M%BlQaMC+d~u&$yo! z@C8>@{VzO7^}lg-)xTmf#6P&E|0nr>+a`qwnf(7dRKTGF4ij+LfWrkGKHvxeM+`Vp zz%g-cZBKmJ`=pMXntDAs4fXnRTG{)#p`492`ri<_WWTnF>bYgNX(s2T-crs-y|rAJ zdONu&?jV=Jo#gLv7r89%CfAnTXZ4U9%RavTr%Uqx+(-4UxUbv;_mg|#{&FuoK<;#02{8hD z`$uAL|0wM3AC0~JW3ab>EcW(~!`}Y!*xNq=d;2G1Z~r9h?O!AN_>Pi8r1sXU_A*oI-^|oEtDXgW|7XKnoC9x@bK&iB zZmeGihRB0=sGb+^l=IxJz0bSKeqUeZZnBs8uiTw_sEE48 z(mssblX_UW7wyB#y=8BIB)Jc5qRD;bkgEHhu71>GsNP@px?;%#XyfyAAdaj0AneQx z#_?32CwrUY%kyzUc@s`7pT|k%pKvny0!|@^N%`M$Qppi<8aWD1C#S}~{~|5UBxlB1 zb^oeD!DO^Ayc;t^5GDmmlJe@*~_?eu=xvuW)zySKL#6jeE-(QvH|7zH&y~Pi~U>zq|<&Jo` z+zF4AJLA!E7d%$(ipR^{@I<*go-Fr}o!hB$Pdr2Jg=fpX@jST?UMTm)OXPldx!fPG zk_X_m@<2I^*0n(%gty3p@eX+i-Xjmi2jpS+h&&vhkVoJj#_rFT~O0Mc5BCSd8PyOK^O7DNZCW!%5}k zIJvw6r;=CVwDKyPL0*kB%WH5pc`eQ?dRlk5ezP*T@`%Ac)+FZu&)30Fn=~uDGf!AoQJrDLh%T< zR-4DTjr;_AjPw-yUixR)b>N}S^vvzANd{bD~HJTU%j6k68D!w;Q?}JJWvjU2gzaaU^yHfB8SIALUc5!lhquc4@iw^t-Yyr!JLE!mr(78Cl8fNoa#6fTE{6BY#qmD51l})y zix0>p@j5cy zpOvfPb8l0jK(45F*uYw z7KfI{;V|-e99Euy!^snIczF_zAWy~-#+ zG37ZpmOK~7mgnI(@_ZauUV!7t3vqmT5l$d4#tG#mIFY;*CzhAtB=T~cR9=CQ%d;EM89TuHu$E6bip#y{8ipS-w=>iMz9LIrSD)eGWkas}Dr&+2j| zTvM)sYs=MeUAYFXFW18UE>0cXSgwbg$_;RHxe;zDH^HstX1J}~61SII}uJ@Pw;(@Yfk#c>oJOB@sJwKG|!{r%xq&y#w zmKWf$@*+H5UW_No8}MX#6P_w>!PDh!c&5Aq&z5)Lx$+)7U*3lo$_Mab`4C{M>QTfT|+%D3=-`3^oP zdqyd@KPy1@X^vA$(IVjBm+B@NKy$z9SdIcje;v7r6w!Cx46Y%O&vx zxfFgVe}^B*rSW6A41OYikDtnA@iVy`elC~CFXRgNrCbrek}Kg~<;wWATm}CoSH*AS zYWR1#I({qHz<q?9EpRBgB@QjO!eQjrIIP?Thm+gl@NzpGL2i#D${lbdxg(A& zcfwKR&N!;v1xJ&+;^=ZW97FDoW6C{nEV(C+E%(B40+y}>#`{MX=Kb%1Bj}yuR za3XmiPAm_?N#wyesXPRKBM-&NEv-Z zy*wUgkSE}b@M3@xTG8wmzKlfvT}G_ zL5_ec%Mo!kITEfZN5*yJD7d~H6*rQj$zim)P37pgr5ppdm1E+LaxC0cj*WZDad2Nb zE*>by!$alxc%+;FkChYRiE<)5RZfg&%1Q8CIl1i5_ZG@2@De#CUM{D?tK`&pt(*pL zkkjJLayq8|<+` zTkP>bJM8g5d+hN*2kh}cN9^%HC)xWpowm6%&LDTendGiGi`)%ole^;_au1wK?uqlr zy>LFcH!dLe!G+|$xQN^j7nA$r67mq)`_tF$q1e~$Vc6I0;n>&h5xAsYI}(?YN8#_} z(b%7jkHMu?AB+8S**NUa6~|+LrZ)lmv%ZPgpEpjz{;Xg!_Gi&kus^q)ipywS)A0B5 zbX-=Rfjx)8Ozb%fW?|1^FdJ8t=io~6TwGb6hpWi*u|LyZfUBy$5Lc5I;p*~YTti-h zYsyP;EqNKPEicD)Kxf<71`!%?pycXA&*Wm{8dfZUnfE&phabtNC zZX$2SP30}PnY>kY=9|mga7%eRZY}S?ZRMS~y}S!|ly~FK@*dn(-iy1-`*2TrKkh9b zzf1H4gwh&RcP@Mifj-XcH2Tji&CoBRxKm!IPu@(a9E zeu;O5jZO8Rb8<-8$Km7{ zIFcL_N0VdWSaNLH+Y?WYBl|j&M2<`S8#x|MEytI=Pcq60WN&{qIU#L4-%KK0Ku%1X zVsaAM`>Ldz6qlC2p-ovi8LlWNm%Xm4atd5aPAPji_2pE!k(^ro-`HMGL%pS(7Ppbp z$$pRR<@C6toPjo-<&3zSoJscn=_zNHz5M=i7TMc2Sk6kD5pp)#jFYp=-p)yK4m?%P zN&6XcF51tNb5ma==b`-yIj`(>t(Nmq-yr9gy*-=d0@OXnPeJOt*FkP1Ds86h;zt| za4xwq&LcO$`Q)a!fZR;>HWZSZuB>-UXNeM8}P64M*K$JB>Ot>hrC(#x&Kk# zB6}ZxmAB%5kICL|DdgkS)5s@qI{75dAfKXrCi%4NDd>1#9@8cHo1Kdh}h}+1Ia69=i?jS$Go#dyu zi~J0Clb_=r@(bDTy_ftF_myAa0rIbSu>2YilYhe_@x%?TglE2`!@?Us^{5Re#f5qG6fACJ(57_X}cYEZJc&{7^ z`(BICc%SNF@P0WgJ|Ks~2j%eCzaNhfa727aZ6aZRei9iURy_(nB1gqX6134>xC}+cu#ed1=@ZWNI{8g@i|B)-=f8|OzL`;p z4X0GSJ5D9{z^Ub)IE~y3rpAD5LE z;BxXpTwY#;E69s+MR^IXBrnCz)iPXJUXH8CD{xhLC9Wo~!qw%~xQ4t2*Ob@dTJk#V zb8tPbEpNbeP2`=psk{p}lXv6h@*dnm z-iuqx`*16HKW;4_z-{D%xUGB$x04TJU!#uT_NpJn9pq!UqkJ59l271@I>%4q&Z?in zUF6fatNa7*CZECG<+Held=B@Nf5g4y^SHPC6Ye8lze4dx@v2{t8c%f5p?~*La5f8=fh@!L#Jw z@of1mo+JN(=gRN!Jo!DIFMq%b)j#9K@)x{B{tGXa|HjMYuXwrq z4_+bvi&x5ifSAYstK^V)wHykskwfFPau~c$4vW{z;qV4IJl-frz!A02BjQb}N5WB5 zkBm2~9tCfaqvEY{47^Q_g}2Lb@eVm2-YLh&yW|9Tx112~krUy)a$>wsPJ;K#N$~;s z8|?20B*zC;Pl*r7Y4Bk=Ej}V=#7E^!_?Vm-AD6S>6LNNZQqF--$vN?9IT!vx&W+E= zdGJ{|FFq&d!#~RT@p-ub{z)!~FUW=PMY%BccbbaeOR5*em*rykid-CDl}q4j^0)Z9 zToT`qOW~j8@9<5zG`=O5!MEk_@g2D=zAKl*zsTkBJ-Grdt7~ILd|&lS_<>v*Ka{KB zM{-qMQLn9rAFEy+Kap$Tr*cjFOs<8W%e8SSxek6I*Tpa8dia%GAO9*hz^~V=6)tlho<)-+p+zkIAH^=Yf7Wlo~5`U0e;Xmcp*x7D_Kgw~d!Ck5IKJAX#tGy!IH8;tCz8|Q#BzF^M9zRck5ES0`zNXDnQ$^W zv+RA6Le7Fy%UNlYPR@oi$k}lwIS0-n=fv6MTsViE8|RYq;5>3(oKMb=3&;gxdG3^z14nxzzYIi81N$8N9`8}yae}EeQChU0$v{Q z3fb4*l)BKa40si7`e`|<16~vG+JM*L{%XHI;0*z940sbBp!S;s-XeQ@{4A=i0bilb zK()Ub@U?)i2Ydq$Qv06+zKI8`ekA5b#4h zT=ho*KgJ_ee-iN1fS(2Y9FJ7{7XiPN-9L^}{Z+uf;<2i~4)`}bLG?EQ|BffC{x;x0 z@HExm1^gb*RQ*H1f8sf+e+>9jz@PDawfPe8UjhFuyAN8VHeUn&C*Xer_Jhy89~P^9 zNW4rAg;&az9%Qg59Ad1k(?4gkyGJka%%iSPJ>^`Y4K}09eyLH$8Y5f_??^)e~>fb zk8)=GSL`1WqM?i_^#@aXPsa&LDq>Gs&fK7P$=0 zCV!7}$YpUZxg5?Tm&f_!3b=q=5f_px;UaQnTuiQlOUPAmNx2&SPOgs2$Te_Txh5_z z*TNO$+PJb@2UnHr;_7leTvM)(Ys(FAUAZByFE^5XooOgH#!ciVxVhXEx00LTwsLda zL2iLN%Pny?xfS;Hsx|JZdK>I(S6l3BS3B9)u0Cqh9`}Z5?*MwjR4*+koAdZN%=&HevT=o3Z<{E!ch8R_wlP8(yYu z+m79@?Z8juo!I@_F6@47H+H|a2fJU}i`|#)!|u!WWA|kTu=}!u*nQa{?7r+Uc3*Y` zyDvM6-IpE1?z4{Li1G;>Q$C5^ADzN+<b(GS??(HZQ1=PY)=a}K-T`4Rh^I*(V# zKVkPZ7qI)9i`ae5CG7L-GIpPG1-nnViVMltu+O#Yc$ItuyC3-(yC1oU*T}c9`;Ob# zea9W_zT+mkDKanHhXL4lxLXLucjH2RKay0x}j*j2RG4NYC zCieb}g}pyxWAD#6*!wds_Wq2Ay+7k)@6QC-`!gZ-{!E0uKNDl`&m`FUGb#4|{04h} zCd1yJ$+7om3he!v5_^B9!rq^$vG->h?ERS*dw-_G-k<5Q_h$y|{h1Mae`dnopP8}u zXBO<`XT@HAHtgkR$6kI8>~k$A_W6|y`{$9|*nLVK>^>zgcAt_Dd%gLw*INL4y#=wC zSqQsdD2(5!O_6}B$UYxGs9qI+lB)$=9si|z4g8N>6NikWaWW1g*T&)HIyjPCH{g0W zs_OM|47mZ0EjPsR)>Vt73c?fPQ55+CyVYszC9JiB41UwRVRDD#yqj4A2 z#{@hUcUOHJ?j?`MedP&wfIJZomM7t1@?<0`-H~uRVmn$cOP)`3MdXSL0zET0Vxu$;Yvm ze**j5Jc)g6IfWyt{b?LU{vAh`-{M&EAF}WNk1M~!3FY@Vsr&(_kpIMKK;Pmmj9CVtJ4sr<-ciPNzVSye`BHg`pX=$KS!;mdQMzj&V_5pxdYCF z{oIzkc!Zn}*HrubxRzW1*Om+7I&vXgS1yd}$whE|xhQTR7sH+>zc}_h`6X~ewfPn| zl1t*oaw*(IE-U-zRzCx%T)^c6t`KlV+55-)p;Ew=1FnLbs!dhgOsQe%q8t}A$r{l?LKO^9o zc#7(?0-hc4oPg)zscJtj;Q0YB2zVi$ruK^hUL5d}fS2OwYQHStLf;OznL2zV!+r}n!7-W~9s zfcN70YQHbw{Q(~c_#j@O_J;yK9Pp8VkK%=De=OkR0iOu?BwnQUrvg45@DBl>!Hd=Y zY{2IN{xRV5c!}Eo6!3+BF9v)GFID@?0bdFDYQWdmGDB#BdKf$Zi{%OF^0)8Iw3%pwGUk3at;9moNjn}CC zZvnr-BQ+TLJ>a)^t=j()@VkKD2mC?Sr7*-gwf{5Vk9ec%p921jx2XOl;J;+|HQQDH zJK&K2{&)R!pBoDAltW|pm0_^^var~FSvc%IG(6rVN5Jl{BVza0k+A#O$k_dC6zuUs zRO~)C8us`hI^L~%4D4q##KgMYGDIxw?-Iwxeojyv?B@i<#ePmuJnZKL#m9b5Py+1d z1SQ0NPEaE3aZ6(C@k|ozu|ZPo@xV9O15irCLdsf7KUl*(9BafGOXTk5q{aVxnR_H$CIV?QUQ z25zl3HE|oc7WQ*eYGXeqr4IIUQtDzqC#4>4tJl`Y{~LSb{o1z;vB%+!@Bz6oJ}5WA zhvcUCu-pv$XPM^MKg+bh{+XgBJ|efmo=2!PKB{^fd`xbOkIU`w3AsH!DR;o9$DZ?Q0QQ_$1F`448iYOP)nM#7uZCdHc{LP!&Z}YAb6yR{p7Ux1_MBHEvFE%R zg+1rhXzV$!#$eBRH5Pl$t8v(KUX90|^J)V2oL3XE=e(MPJ?GVA>^ZNdV9$9q6?@LB zY1ngKO~;<|Y6kY4S2MBayqbkQ=hbZNIj`nm&v`W$d(NwQ*mGXZ$DZ?Q0rs3%3-LK+ zV-fyQUX0JnOR(o0UWz>@?xb6l^&p5uBo_8gLHu;;j5i#?y@ zI_x>F*JICdy#afU>y6lRTyMgj<9ajp9M@a0=eXXAJxAp>>^Um8W6yED1AC6^o!E0% z?!unqdN=kwmV2=0vD}M2$Mrt!IW6~N&uMu8drr%P*mGJQ!k*Le81@|3$Fb+QK7l>Q z^-1hGu1{glaeW$lj_b47b6lUpp5yvQ>^ZK_W6yE@6ZRa}7qI8JzKlJ`^%d+nuCHRx zaeWPYj_d2#b6nrRp5yu!{#U+@J;(JO98$iIe^T}y;0y9Yd{KTR>k|LJ&*mOezohyT zd|7^qugK5wRrv+JCcnhjM0WptPi~4I z%FXZ-xjB9=x4^ID{j%TJZ}I{ByL^x~f5?aM2l+7mBp<bw=&py`yM-QWwG;C4m)q}_p96MtpuroFiJ7c4;Gd3DKV`H#0HWvGuHV!*)*^Re@`06TAsursz8J7deR zv$PUBGpn#OvlcrW>#(!29y=Qwu(PoVI}e+&^RNXw4_mSGunjv8+p+Vo13M2pvGcGC zI}f|D^RNdy4|}omupc`E2eFU)A?)LR82h*%!9MQCu#fd|>|=cr`}m&1KE9{1kM9rI z$M+2O@jZ)ue9vJY-ygA$?|JOw`xEx@y?}juFJd3xOW4I3ih$SihVt~hJ8J` zj-7!U*w>Svv9BjLv9oatI~%vLuP1k~uP1l0GxG~}X6|8U=00|29$;Tj9%AR`5q5qa zW9R1yc7C2>=jR!Aex76J=LL3tUSj9x6?T4p#m>)b?EL(OeLZ=Dou%KguP1M@bM*&y zuHIqi>OFR@K44!@{>0AMN9>G!!oHq-#?IRp?7aPjowvWS^Y#@xZ~tKD?O*J?g}C=chDwX3Ak_qdaytDqv@$B6c<^VP~T4lw{-q@MxgPoba*w>tX*!k&?ea#twea#t&-G2?j&e&k=j19rg*ih_@ z4a3gZaO{kYz|Po6?EY&McHTx~=WPsj-o|3*Z5(#q#$)Gg0(RaeV&`oVcHSmq=WPmh z-lk&bZ5nporeo)A26os=40n=0e0RNV&`oUcHS0a z=WPjg-j-tLZ5ejnmSg8_1$N$6V&`oYcHUNF=WPvk-qvF0Z5?*r)??>w19sjvV&`oW zcHTB)=WPpi-nL@rZ5wvpwqxgQ2X@|eV&`oacHVYl=WP#m-u7bWZ69{t_G9Pm0CwIE zV(0A;cHRzS=j{k~-i~7D?HG36j$`NT1a{s|V(0A?cHT~7=j{jVyq&?$+ga?qox{%C zkJx!TkDa%lu=92SJ8u`U^L7b4ZJ8y}w^OhJpZ%MH8mJ~a0-(cr08Ft>1W9KaecHUBA=Pea> z-cn=dEe&?w(qiW=9d_Q*W9KaccHS~#=PeU<-ZEq7Eem$uvSQ~g8+P8ZW9KagcHVMg z=Peg@-g0B-Ef03y@?z&LA9mjIW9O{^cHRnN=dBQS-U?&qtq6ABiel%j7nZxylgRtYz1Uqj_vGcYJJ8#Re^R@>6qu)ob#sA9d@Lp~6dK@C2 z#`8F&yb*_zH{sCoX1qe{+JeKVz7>a+x8ZQ|b{t;bfg{K}aYT6+jwJ8K`?PI)aAei@ zVy|}}j-vX0992Gm7pcub98L8@IJ$fo$B>WUjat`H98>jUIF@`IFVHei;5}-86314X zQ+TD?oW^lf{{hF9&)|h>a~8kQ&8Fw@Qq_ON@znl2)?p0s6OOO?1)M;>h!e_}@DeTO zGF~QM!OP{VIFZ_4!-?hVIEj1%FIM}XaZ=T9;&0?zIGKDKCztPF?}xkC@Ant%_jnI` z`S-E6^8xnuKg7=MBkbHh#?I{%?A$)Z&h0bo+&;(7?F;Puyu{A!E9~6CIPBbp$Ifj8?A%7g&TS;@ z+(yPe?oqJw5EVPO(XewH9Xq!%uyY#|J6Ex=a~m5ww{fs@8y7pb@v!$peC+p|0Q)^A z#9n?P?Cng9z5PkBbDI=9x8GprHW_wqlVj&L1$J&zV&^s$c79T0=Qa&?Zqs7tHXU|u z(_`m019omRV&^s!c5X9c=Qa!Wdb48ZHXC+svt#Er2X=0AV&^s&c5ZWH=Qa;^Zu4U2 zHXn9w^JC|>0CsK*Vjr(U*tso?o!cVV$FV4OZi`_b)8g2B0Xw%9v2$AqJGYgwb6W*_KUBqjztyncV|DE1 z*TCM+n%LW43p=;9v2$AoJGXVQb6XEPxAn1e+W^Q z1UnByv2!~NJGaBJb2|b%w420OQ7v2!~Pdq0fFe!mm2-{VB=v2(isJGTq5 zbGryTw~MiJy97J8OR?9x3_G{Wv2(iuJGU#bbGr&Vx2v&py9PVAYq4{?4m-E&v2(it zJGUFLbGr%qcx}ec?H26ZZpA*1+pu%H9s8K>z|QSX?A-3c&h2jO-0s27?OyETz7IPO z`>}I-06Vt_v2%L}JGY0ib9Dqew@0yadkj0b$FXyJ0((E4#D2f0u;1fp?B)M}y`5*U zxBn;HNBj8#)>i>ST*Ur9+9j-A7~(SSr}kHHfB7mNAYa1+wSv7-e=hBeU81} z7uf54iM`%e*z5fjd%dr**ZUjxdf#BL_jl~|zQtbeAK2@ChrQnS*z5g(z1}~u*ZUEB zy`Qkx`x$$^U$EEv7xsGp#$NAN?DhVGz21MZ*Bjzfs1W)nAw&vgH6%_chr+4k&^Wam z2B(q3Vm}ik98Rlxc$`j-fYZwnaRxaO_A^8x&hi@J-HOFFMo#{$fW}=gBz;;J#HkI z#f{~1xQSdIHw^Y3vZY5X8t>qfHja(DAm22U4 za&6pRu7f+sb#X_z9_}R9$DQQ{xQpBncag=fj5@oaevo+FRNbLDY(o;)7UmnYx_@QnF*?VqW5vFg(To{pEOJ_9e6X9heAFH?OsUM|nUE9AL&r92O>lIP>q@&dd@ zUWnJqi|{&mF7VPJ0ZN+{*@;1CzZMNfm@(#RT-iZ&$yYNAIH$Eiq z!H4C&_=vm@AC>pxWAXufTt0|T$cOMr`7k~uAHk>Pqxc8;7(OE($7kge_?&za|0tir z=jGG*C;12LXN{e~7gRrsFUsfeCHY5uSw4@i$Uot$@&)YYlwHKvRKJ9;%a{MJjeG_F zEMLVp!AGZ{R!f&)CmszWIMv{TBX3zK!q6ckq4rE`A{Yf*;EF@FV#?ek?!0 zPvnQ#&tH3lpQ`>CKa-!}=kinhLVku{%FppD`33$}eu-bpukdg3ulSAp8vic;hTqC> z@E`K;_?`R~znA~OALMuVPx(FmD1X2y^dA4jpH%;dKg*x+7x^>(Oa6lYmjA+E<-hSi z@>l$?{0|NhU*mrqQV#L?zwv)4IVART?n2=na%db{ZNlI%a#$Qz4u`|Z;c<950*)X@ z#1Z94IFcM0N0y`DC~{OBRgQ+E$Tu!cp%gdFqpD$kp zS5Unwt|(W-mE`K!&yK8tE2~};SCMOBKX0@)uBv(+TurWv{Y=SvxVq}~aSgcvt|>Rf zwd6(tH^#M9Z-VQ{O>teh8LlTc$MxkFxPjafHUQTpN=FqJLjO0 zi5+s2+YTk6agc<{p%Ai_&Y?4@)QKX5khxhbglwT%RxE@N+Jq1b*&2<|$e8737Bl-l z9@lxlJMVM(RKB+F&F_DGzfaHC>-B!Up6}bw=lWdNIj05N|6SB6!JbI`mSA6teQU6< z!@e!p*JDo&_9X0|2m1!>+k<^0_Ai2c6ZW)VPsY9@*f(SUGT1-Go*wL?Ax*L4fZdvX9fFR?p^l@5X*C*!N)14feg*zYg{+?B4|Y zKJ3SXeLwcRU~j;`h4MtOA0Yl@uxDdG73>GG=Lh>C?5BhMF!nRSo`d~tuphx*5bQ^> zp9}V5*v|)hF7|JO{cG%n!Tt^Q3&DOI`^8{y%s5{P_B`T?g8c;c%fWsU`**>93j6oL zo{zma*iU2sA=uAgzY^?cv0n}L0_-Kheh&MO!G0e5wP61i`}JTi#9kWg7qH(5_KVnW z2Ky!Kw}QP0ds(nw#{N^Ve~0~cuz!#JPOukaFAw$~u-^^#E7*Sy_N&TS4_p!eS_Fu678SD?RzYO+YvHz9W+2;+<|0{?$3-*WD zCBgm(d!1l^jNLrgpJ1;W?7v~R2==Gg>jnGo*e!#-5_|n%e}=t5us_G%FxdaV-YD2# zV1Fyv|HLj0_LtaY!TuL^d9btAJpTu~8TKZ@F2UY3*y~`o3U+hsZwLE8u93}xy)N<1 zgMA3`*1>K;e2ZYOhy9&kx5VBu*z04r3HIUCZWZhei2JGtXvr_74hVL0>^-qN1iu;n z98iJ3BfJMZ6v$mGcz#aoSQq05(w0#pt0R{LUMH}98{Ysv7`Bf?VC!V-Wb0(}4})!& zZI^AAZSQ3>d?ak1Y@TeM9NW!52H!l{JlQ1c99Z9WeOcf6>dX2M0Dal(z zcV7Ck{)WMqkH%Mjcwl|?^<{na_2u5M`6B}BJAZvy-#UF+-#UG{Pl(qC);F#%9|OBj zo_AT_dFadf&O=|;_j=Wr^_`c#tnWVR%g4cv^Q^$$AN0MRWc`-#39$1#JFxzG!I$+N zukEt)b8WZ8cCBUmmhD^i+KF}AvDRxT+MWZ?ne5r}Jjm|1aarHJmc731-|JuYoXVb4 z*)hqENp?)KV~W1>jcXij^R!(9bE6&OQNwtu6YpKdwH=?f`i?Q$#$Cr4kM@(qjlCV% z>sx)VciHPq_BxZDUpv_O$}h`_b$@_iiGu?5#Mko$n{f%T68hbMkL zw(as$fo<1+8Cd`8z%K@Fw?5yWgUY}i0{gde#{JthdGo++0&f#|+rYa7ZXft5_)#GL z9^3-vfLjCqD)62gpbt6+-YoFuf!hS$CGepea<+iHf8gWbZ9o^WZQyHQ|83$Y!K2`* z!1~6!ZG;YZ47?1UO!#-f|4qW?bmN!v_6LUs);|vX0r+o)%fC%H#$G4Zvy{(#;5l#) zm=10Td<(o`89#pj4}czlj|zNj;1dH^!S{hP!0^EJ@I7E`VB56sO*qCPUqhSyO%8lZ z;M)UF4?F?3zwP<{Zg+4Xm-6# zu)g*BvcC2Dvc5X{vi@$tmwgZOTJ*l-`&hJN-2P)++qiYcVtr-sW!vq?`;9r#?uNY{ zu~P#3Ok!U?lgPf;+U|RstnYmEWqt1v`m(;-`m+A+!IuxicRY20^{sOb#-e=$wz1O! z>)W5ctnWJN%RNGTaNy`WhZxf~ZoaWt=X&YOJz@JjJ+Qv>)|d6|UtjJO;zI)K8`qER z_MdX4-=*xXW+~-5v^{vyFeMU9zv#0E5LqCK1Sy0wjM_<-=-SlOz z3+xxE?-g{+z+w^7cE3)^R=$of6+aAYiEZQ;dSYuo} z#-oP$sm?k{zP9=HYo0pMHtzQw?yK==$9U}5c^p}r#kB-`Pwl*jvs&prOIhC>eOX^!eOX^Uefe(S zXG+=sUBc+A6=Tsh&-ul8w2dE>#I>&m#$|P*uU3pjJL<-`x-ma$siE!XK=Wl|(N`5QX6=Ttk zx-qV9%#T`XX!{x5eA!s^)rzrbN8K1#H|9sJsCxr#=F7&SuU3pjJL<-`x-nnf7}xeY z0`p~cqpwzsMLX)oxVk?B#%1fHZ%vFvJJ!dz^?p}jT(&;?*2Gw}V||RL_0A>6wSNxO zi>yYpt&6s~PXo1OYoc#Wj72++A;#5;`B6&^?K^<^va#r^6=Ttkx-qV9%#T`8_ioy( zyD#tqfgcR~Sl}lDKONXQ`#A(?s~2tKYDV9-)Q{s!+Z|){ZFii}k7L!3_m*hKeq%iL zqo4L0+haemJ@wOmV|yHj>teh8YR7(KJoX#gQa_z)lM`m~>PzSk%HYR7(KJoXdY zQa|lCw#R;Bee5^3N8f#p^VN?1#(3-}wxxdBZ)}hK#`?6MbiQ{a{c6X4V?6c~+fqO6 zH@3%qV}0y5wnzVN`tli2_H%(8wYB3}jC#hbjU3OT8dn1~WHn^lD{vj7*Os>Rvd`$+ z4}jRGc06mbPh&Apw%zwPYuvxcX{zVPjyZ9BHxM&lHeZhV z?o((|(hPd48+`;IxW8Gla2XaqNZmfvhfOB`zXiI?tr~JoQFM0D8^pIim?s#f~IU;^bUX{Vw--kRwXf? z^E-mQKqs&_=nVD&yMRr$<@b`|c3>yq_ZEKk9{~n{_F!OQ8$TG#3~ZY@K0`;_=idp< z8gl&p$?ux{o@hs6e*W+~nqC3c`JKg4z_hv&zDta|Rxz$^o;ljCWwc|y z>uQ^E?HG5RW36_~(RLrq*N*wFO&*RV9$~w7%uDSU>q~otG40q^hn)EX`oJ>=6l$M zfxUNFtDg4_+ja`J@o1|XZMCDVezff)+V&G|`--;xMccmYH}%v0wbL=^r{mC1$D*H( zM?W2temXAwbZq+R_~N=aKHI$S8^IKEe%rxr{FYjA-E1q2TQ@fzQCVnZ$QMXK*0s2kO9u;C?V4 zyaj%LEPq?ZB;V`H_j^oq1MqFIBNzdyPvqx-U?nK)$InMWXD|eu0bV_c->C*4f-k}F z@3S7@eDDeQ5^VSbo(DlS7yw3q$3aULwG%i3wC5mp1eb#A!Bp@dSPK3GR!u&whc%Hw&Kycn= zj2FBJvLA8pxPs>{a10m)ZU!}1a(=~JP4M9qps&~w7|)r4m=C~0zL-KCh>P9;8pNCxb6l%Uw|24+Kv3a1$Z57 ze-nET_5@wP3E)qYxvqZ7nt-2!hr#3EIq(Ab4Ez(6{)}@4D!_i=aBwuZ5=;hDz$TSE z!*R|J0Y_lp0D1;KGVoD>dj~!y@Nt2U4}4m99DG@SaqwmRSA#F>-yZzG23`^P!@wT}{%2t4fP29#a3A;%XaUv(Ex|TmeXs%85Nrg#1xi5~@Za^= z7;FLtfRn*MPy=egDd1Ev2-JZeg44iYa5@+QhJs<>3@{vw0RAoYNYEd&0^bJydq)17 zJqy6|z<-0mzw7sJu-kw?g4e-TU~5nTwgqj$j$kLS3)mIx26hMS!JeQKr~-$A9^gap z5jYO?1;>MN;9@WyTn(-P6TwfxEnt1#t6n{pwSsp88-hK#hi(Bk13Mkhz4f?;=N0$J z_uzKGJ&Lw_6m9n@+U`lT*Gp{oCdS>LXm6C*?p2JtXVG@=qU|0=dy~X=obGFEbF8WF zcvIgor@rG(eaD{qjz9ICL+U$^)ORka?|f3+`sxn>aeh9JMB8VOXuG%3 zc8{YyF^QjkRAYRdBz{^FKPid(d=hnhkB_$R-_iE{JlZ}3MBDeLX!{%!ZJ!yUz4M?( zd;O$#|Iv-{Z}Gqv^GEYs5baxd&y4m-N&c*(8{=;!a~YH9oN{84b7rI6jEBsq`Bb9c zCs~KnlfE`i@_jalwLT+g?+Nw@tSveO?ihHlz?}l`9k_GgePC<6ZOZCJ4LL$RZS~?c z5wC@K4aBn_&wM=V@r=i_9nW+;%kd1yvm4KBJgc5D&z5J(^W(m{pYEgk=f1gL?vwko z7~Y%vPTd4@sTHJhy&K+kW7B_RM{Ii~Vw++#mO)&G8NM zcKtd4*H3o+P0*1%2E4P5~c#*bEkn{0-qy%hVc2pX9wS3eP-}^!Dj`Z6MRPS`M_rbp9_2@ z@ICk$@GMvWo&(PVpAUQ%@crNSeBb9^1FwUn;0^F5SO(q(%fY+g&)_}qKKKjpdBA4@ zp96dbaO`UP%>^xwhQykbMB7U9Ovuvl`(zi=DOSweAmbMn-}{t z-?;wtz^VHE?W>**fFyk@=2QjTrwccAP_TZLx2y<2SB`=g2v^Cb4hXex3uK zJ#E+VUSQ2L!8RVB7d@M5L|uLRasSM79-dv-BKB{;_8;@2?HJSbv2V||`D)m%?O5V` z9Fyl<#I^V>wqu$HwBs2~`$^+5Pi@ESnYWMl4DFe9EcThk!cmW7_Wo$tl4J# zHeg?B>Kk|b&hr<*evDh|I(ZG*pX+0t?eV(ueh}?vu^ofHInFJ%spCANua5oMw=vn6 zeZ>8B9j$Ym#>`PS`nJW~m}ic$IL>I>?)9UlbMd@IUrlY-z`fBnW}ddSj`f{jzYa$Y z+w?tSk=2OvQO9d&dWfaA^RT}78e2xp@!PIW^wsh{>6x(3KI6HwP2WA!j^`q-gB^fZ*6_X}`yL|u{v`YUB>Vm(`~D>R{v`YUB>Vm($M>gb z>qk3k8;kndu@CLok9O=!JNBm?`_zv8s$B`}OWU?+$A0D5uXgO0N06)<3;=$wDjx`r z0VfB4I6M>_3+STo^Oq6d=QAS-yYao&6@41MZ}VBC6#rqKQGB*3`vIRrc?R+M#yWL} z1orQ}^-m8zr3L>8_#n_T@S%bI{@1Y~UHA?&_0{eM)Rxtb%wt2L57|EC?trhh3VyWh zFWUAQZKk%$R@eR=v+R1s@yJZQ(B?SIF<*{w%-W74#$@*+wrR&$%!@JO?qST;jdzNgMZI_*k z?AT<-C7XX7Fkd!bHeWVhHb46Q3|C)v4!Z;MWzW|cU|<60crb8o>UmD|J@?kwZk_AH zaW4E$v$~!W*Q*XVz9SOYzgqPmJvZj*t8G8pqk!W(Ik0-M#yoBNu|LkLPCr+(f&>pK?&@*o2GE=R5pM_*uxMe5XF1f759K-x={k zs4dyZKMdnNc?3Ue8OL{PSM!8*wK83`|{=-svn~H1vSsp$0bMdK7SPYNAv8$L4Juo%V~RuF||OmANcE) zjQx7fB;Q|uelz!U4&?cx`40MA^ee89N!{m0T;(8&cL?lJWl)f8AG$P_&ES; z`Ac*jV*SRG^Ch}}x{x^yfd`Z)Q=T(I&cbil%cs%$8*@B|@!p00{)}%V>#&S*e|9YM zI*IjTZSEmfhyIP{^S+06$7?uqtYvrhv-Kp_`$+2Pdlqw=P5cdNE9Lx z!2F(PoClrBxnW#m7}t2_wLAHjF|KnsQy+r^+0)5G_}+s4j%N)|V=u~?)A#9Tcji2b z{9Vx9g0cOX+?|={<;H z0B=%r0Q>q7{cc6wLdH3Xyw5rRx6yXoWPS$8K3tE^v-G)~YoX#CzSm{F<_>3WtY_cL zv6*ux&e^{7zaF@q^?ivk|Bk$iS%S=ooe`nVF znKSu$4{Q2;Pz!zxj%B?cMQf8wxV||Bo1|?AIXH?<(*SIFh}36I{)BE@s`%V%*j} z9*FVNvHt*Ofseog;7zayyb6|pKZ51p&){9~>%hMWJST~18~1m))RJchw$0z~lKmYo z+28Y$?N5F=@b3aI4(u3he)^9r&HV&PyHVB|AsiaXbhd z?_N|mdfBhtw51s{2fTzJia2QDcZadokjdryCJML(!6>T-6t#-84 zjkfv=5`Wjk-Yu~YNo;G|0dr;ZR^|HEhYWm(7vQm(4#Em@nHd+b-KKY?sxO)sthN=Ie_}FbwPw z*f#wN?yWuGqk`Qtv3n)PLIe#CBd@3%1L)$;NjE#%1HOanHJO z*|==n`Ry2N+l|>S3;Q=N8<%6;F~)V$c0C=3Y`?N+-yHW^-&$G3xoSHe*=tAc2ppTN zUj^bCYR5Iyj%(<3<6g*))9XcF*7thRm-X#SU)HxTeOceLpdZhKXGA}q73ZZN*TA{z z%hr2c>&sE!TJ_bGt(9G4$01uUTPN=bjsQCY>s=RZHT0t$<7!4*jyc-an6GVK)H5!O zAD-Bb-*Lshw5`#PcFc+T#S#M=^J84wa~O5vIW#|>J?(g|wBx=+JNB!uzAWPSwPTJP z?RXw+k87wM&z`pLn=wDGXI%HV4skr@#eD5JUv;dBmSD-*Ti`2SKIZm zzZZeykGA(AZQ(i5cPwgppR&(5?)KPGC(hLz`|J+XFfQAtdd6)xrdF(X|5Drc+2XMn zE7mhstm*owZ(FpZmSb>?3qXvuB5_l|N9l}@Y#AT9vsRUSUkS>=#$W@mA=n6f3zUK~ zuq(OE;C+%iPz5T0_k7>qWc?n2dj{?W$M;3seNT+< zjrzVX#`j14qk=E%A02#o8{nMX%bmc^!1rX=$UXHOx~G-EHS$@&z4H2TZ@hNA&-%>j zK6tP6+}?tIj-Wl@YlDohIx@a0$oS}=@i98P zcqcwbffCmIAK(^n8~8c!=a6@Rw}3y-_GfR81AmU?&!_x3l|Ng(6`k9_3~)R6Ik=+& zO*Y2Ad$}6;cP~EyQ;0naj(ee5Rtx$T@sK1oG_i*z_8Eyi9Q#3_)^=BH(4r6H1D!#4FbLcV zJ_Y4`Z`2zM1mnOBU^=)L{03}x9N)cxqxfE_9(=&}O3&ke0=|WPIlL#l2bcktwJOOz zg8vR$9M5-4U|Vo9n6_C-_Gj=u_Le7b9e~Byr@$X{;(O6eOR|<+6#IkuTa;vv!;8Vk zp!Jp|*}vcx++;d%BfDU4zDLHM3$6sqwkpXE=3;sq9s-{R#(+1#o1lGXzE6YOaj|_5 z)Pa+UU9?R}HU^%Fe=oQnTytAVwntn3o$m`7JAFMx`|LeSvR4L_WGCaF2HqleBm5lr zJ@^b<(4RA~5Bm-l?NE|Eh2MPu_dL)GoDJ>)Q^BvmYrvm_ZV65TcY{TsdLZ}VT}ra= z!C&AHf-eHgz=y=Qt>LU<-vM^##$1Pg9=I8t1qKrvT|nZp5$6FiB% z57)-lU`ucv`G>>jg0sM#V0*63L(bqjIH)9>3{Ph*+70J@0aSyQ-Ab~t@U_McEy-GS zFUigx!TAR>_U9f8&xQ}J=Y0^I0WJp5fGrQ;eQ_l3FFi`K{o!lipKTPcF94^^F-F?9PW2uNBptEx2^P;fl9&I9D%|e-HLL7x3)Sl^XcI6F4iUmSp#VpO7;dG{2Dh*N?gG$hjK(a&R-a_h_ye z89V~s23<}o$+o(PeFS%cw%`%)4(JV@0Smwq&}ncuva(M34XXE-|vqq$xazvl08c7DKOcu~EwAM4 zg7)XLJ}pbKXR+si72q}eMOSgZXU$%`xFq`vm^gv?fsL=`o^feOwk6#88tz?SF&G0r z`w8zi6B*0pCE4ZIa@G&y{SrRw%98AD?8mQTzfNLJuV?Rh&-%_a+-nc#yLPaY*aj0z zvU{+vgEySSGZ?u4+LG)|xWf_nH?X(hH{eU~Jh94~O0rFE;u&u;_h;~Y59SL_y%~E- zNwyMu-CIhs(pyWitHH57sr@OjpYePOHUru1CH#kb%i3RU+qNwv!A)Kbl|5Gb`g-ka z>A!6+`!uOXA+z9Vwr%iN(|fPg#AA3>dtnjyPi**qj#-$7S7CWp{ijF6@+3Z4jovCN z8gU(>|3Axx86;iyh?5F)`|qa4?&3Q9clY=;^uv05jm-asl>bBP5N4MgkFRYW-RtOH z{QuG4{1w+hg;jSyL>pe;NH-QY@G8;7K=$7m^Vc6$oJQn0v1G1o8`RQuXw+Z(;x?UZ z6R%O`KW-)c#3?1Af5(jLfW|kx>Oy54gepquH&OhG8!G}gN&YW5p&wOZ)hZ(fsm^p@%h4VN`|ORsUH-_L`cgw>iob?7aM@>kyuWVpXrkb;zszI8XjcTD695+cqS}by#ioS{ssCd}V51Z$NA9VvPxO zTGalPKv8?1`|qzqP)&zZJg~fu|9fKYMB{)~*=z6Y8?OW1*(tBW8)Diyv9)*n?`Aai zpY~Q5P#XBU-K=dL)~Lx{pNWpImU8v$;GVDPwf6QkRePM)*#7kz!#c*~^ro>& zpKmB)yBtc2D6T_MI4%6YWjC4Ae{N`T9oDGWut=s9tcKh$pe75rMwPENHmUvJb-va* ze02u@w<1lSW(&KO4$u`&{jXe;w8wR5x*I*#@M(@V{=e|h(6p#m!`GKZ!PU5Jdo+Fy zCnfO;PyUUKtF=}m{0+m6>rgoJ{HJl{8kf02+^25KafV%)m!F_P@}w`i>fI%d2ny54QY!>+l~ONRZ0A9sC#8 z^lPN%kGaV@ppu_ObX5-_qu_^d+Po_7KVmd=hE!w!{}J6(q#zyFA#M1IovgxF+LAz6 zpaVe{*Ivk4y~uxSOJPO~V6`6=dhi=F z{2MF>^ZQQK{NCS?GY1T=tEuQ&TT?fn;^^8THFfoMLx*HNyBD_|Jh+bEgFCoSUc92~ zfcgPh&x5;WeeP;izHrI6%Hx^iH-D-c?4@&OZ+-E5N6o(LfO8Mn>-!HKHuZpZ-D^kf zRB=w_zBT7o^sXN;yk36;}4I40I z1jFt&U|{{w;Y#k_@zirGLPGl<1BRSBa=@vzhYlY)a#-jvwsjjid|>TCLkHLN88p0h z#2}2~@~r);9GgZo>h`X$J-zpk0mF{1IMMMH8qe9I(6@g3*xtRN9lz$-<0?)J!#io! z_=tDUiyt(2WNo*RXP@14$XR>iy$98uQs1cLM49n@VYYx)Yn#Y*t4RrNhjByS~p|^*J((7pV7l= zEB1&}%W76F_#Tx_l3kx}T@M^Se8A{1ivikhDC;zQ=xMdb)z#DwYC0499#9uLVU8S} zxQ_=P*0tw=VMm^F%81(f3c8ACq6f#*PZOlS+<7)(?fyjp}-vj?^WQa<8>3)fcf$1ZE% ztGeQxbO7fv$l8G;>uU=mC~QIdUR{$MLU}o?$2qLF`;ab2Va84uhI7G4LcNWxRsp!J@?v+Ifn%-45t0=Lx$Ef3A4_vIA!SY z!rtz=SEq`20dp@OQZrz9%^^bu4y|!Qg;n5OuL@uvGo+~{iIQw7o+FG^G zQUBk%ZknnwKCX(v9JD@z>IR-R#N*|qn7pkxhmcb}bVT88vuR5E!|Hi&9eCi7Q;YA& z%)f3#x4KcaHPvE<$SGbSi{$02-8vK!i2M4eH z!9$0Q9^R1JzSq$DIxa?YDw<9)Yky|#aNjj;EL>*|_q%YPP7V+4do;b(XWMTR?*@D9 zw#V+3`>HXJ1*+V06z>@P%}3rf=OHP+nUdRZ0>W84heOwJxqjueRh-*+-t1!v&l>$7 zX;xl2Z{1a>Xx*}W?%W2%c=W3;Xi+|MPP6g}<64wYcmf7XiB(+KqI_(p(gt9cqq)kD z@qd%&((l?d)13bMl$Os7SPIYGtF*j=G0dIFcmr#TRJUs1hIu$g<14~^`(MlFQ&v{&Fyfh*yP&JDF$m(udeV5{X`$nU#% zhm@Az_*_0Z{;E~^go79Z$PQ*aU@oZaT3S8@jHTWB31BWL??w!)0R0awElF{xygO#NCcUN^OY51f%E$ICEkAg`5N^K}BWi0;tEeAZF~Y~@ilL`e?8rql;@lmx z&|Xu^U`}_igKLMJ%41ZL5_6)4j|ypXY#%zTcE~E-!h-mpsvlQ1i%zp<8UO#4S0Ti& z+?O=`+I_=+Lsm)F#4WB}OEbl5EH9S(Z}ku8U3+@n$$Vh2m!pTCK41v!=ctjV^E>w5 ze(?_)I+VX+;_nzLxl2}d2)t+DJpxx|yRyIiALg2WptO7x_LRA$<-;Cnyf!N4l$LiW zygoi)F6|4SrhOjQ0?3}>-tZJ$cvV+5@Pq@7DZE}jV57TRmG|O0Gv57#*5y+`!EQ+W zM*p38OzZOg=&oP9AC;SxcO2eo)#pFYLRb7AF-Cjr@L#;Nt3X;~Y&-JC#2S3NxdMMu zqGe7zeyLVkr&bU0<`-!##$T3bnUklL)~QubUaJv}^K)F=;a7qQ1;vCJ%(@N{qYPH8^ z+$yya`2GQ}AabTIS?wrFCl6khi2r>wWx{iIzEeT4|kH z6Ul2gvgsT<<97vdz0AqeO6$~`LtcH6);RnLiIzEeT4|kH%gCEsq_q%#Nup&=o>p3? z)<%17R=&}hP3PDKza5BkG$&6hty60Wd3}qt2I1F(2_bGyo|bjZJ2c!wTB4O+_cKDR zeONmOe_n{i8rxP^V_OBu~PcN-iZ!vitsnz!7e7oZJ zDbh19PcN;tzGcVF%EuMyO~jvCq-S29URtYONAeaG=`F+mq)5-aJiWA5z5e8FH@fM3 zJK}c*>3K0PPcN-iZvuJsMSA1#Cl=|Mm#3H3syCm!c}04Q@Rt?onU|-R)~dIXyjEv7 zo$q$|9YH!@^YZl4TJ_rPwOM&hkzPIixFS9C^7PVL_4<-Gt4MDi{-Pp1^YZl4TJ^@0 z*ZiEu`FdYzh2JK`Y&XwWdLKw@Y@0@2k3`45`{55tw9Lu7=cRROEhKMJk=8W)S&5c8 zd0J_mTAz@&tVrt<{V|PmG$&6hty8N_r_IVMsnhmlO%;Ck5VO6|z<<)UN^5MdB5zcp z<64c!pO|QwlQ*WcPOZV@%`4JcgugV=GABqoO6$~` zLteikt-<)C5-oG`w9-1YmXSBDNNYC!yhO{KJgu}&t@6Ej|0~jJeqO8cauDZePM%g; zr&bs8s;JZU<$6`)_e-?Q$ftgRvMibTtr$@o)4%y#p9rDrRxv284QixM5jvp3?)<&JV|BY=r$5!|iAkNX8Jgu}&txEE$i?sUV4^Fhq$f)T$sZ&whNRW3#rBw!FRuhgxf_9fO~0_My%;U#aHQq)y#g_;DPCnY`8)H=ScU{EnbH#Ldam zvTii9~}+2BxX+}bgTp0W5^O8ulH*EODoKMO1jadYzex32kq`Tf5iYVF6`FA_at z@wFZGtuHB>KYm9L?@i|9^`F+MH;K>rHHnrr_4s2#%=XxCx>jk8?X$_7k?6Q4bMWUU zTIS@9DXmj$DPv!mXj#+pQl5c8Y?pndV@hjm`vRS+P^0gds`2|JTIS@9DXp`%O&6Yj zinJ!f)T(A37AIQPEXQ9FVz!&-E1h3jV_RqXs~F$(98}_W0a44GyfLM9 zYSqx^;3BOt_~R2TbMmy(I<>}qwopFSWYjS0`HLp1eVcmh&8i zKVBr)zj=A{Ol#GfO5W@uz4`cyi}cLP(@SgBn?qj9D;wupxc>3mfpmQ4<>{rh>MbU( zZ=z?-ApCj|*Up^0wM*;Ns@i|E@~MfIHM8*NgqZE-`AY9`X^n0D$a_E0aZF#}x4f!R z%bdJ1rFCkJBd;@c+P>84j^8KIGABM_Q>zPkvx>Cl z;V(?I%*oSA>(uH`UN)iW9Lw?BfH+5U^0d-AwZ@Uxqe!bC{-8w5oII_xPOTZ_O)Aow zhCeIOGABj@aq#TbMnTN)~Pjtys1T6v+(C8TIS?wrFCk}B5y^a<$XT;3D5l? z)icjA`%34T*4nm+yeew7eW_QC-@i!Dyga?M*7{G#n~>;PGX;MJh}`k6C3C0xH{u^1^q+ZoIEY-s^~`@U+H>|3AM(py&};w z7GM3SpH`GR8-H$y*&cIK{j|pRG32c%(#fvnJrKn9Je{=0_Q~jU4mJ9Yy*qw&h}mx3 zS2~`w#p3?R#)<7CR*0a#h)KywwvcGUH7!cHou=q zzl$+8wkN;;aDCG`x4~}*qP{tK<4o(+>c=?x7HJK_uTQkh$f)S676Qxh%kF|+XJ zfpl!<<=tb_TJ>g=_eqgn^GW>v14#AE%hOA1)munjS8BC=IleylHAQ;n<>{rh*1u2Q z#3H?^__IK|e&*%rrM2p9)Qk83BE3)Wo8Qn>&%8Xnv{t=#>yXx|)t9{Y6D`N~ z1^!02G}SXNZ)|C;dSl4zLanwh^?KmI9 z)}l}O`A?D7a{NycEpzg;(mJ(Pkhk5fjq@z*KYo`Gv)w#jP0mbGW7|f5=lV}#&TkC< zco4PB$s1Ezr`C4l%`VcKkH09M_Q)^1|&C6#dTHbT!;x7d0vzmGC zi?8&WOlxhMOI~(+(|MNTw*m2k@fllMuHR`Xx*9UzF~GAByg%}HJ-f5MOriPXD3?b|+4PTQAr?22EVXql6zmDXANK6zt{v?kzBPPEL) z(@N{qDnDrR@&!d&OYoP2`2C+bd0J_mTJ6Yd{mZ6vY=_?w#P9#i$f)apuJO`>H@ zJ^q*wv)w#j=^0IHY^xz}Mxx`o&%vLcXql5YrnFA22?uXpzB1ABKH73RzyAQz`>1*L z<0~CoT5H=(^14&2?W;NHi$6F=(_H)Wm1?H-wk;rUa*^f?{JA-r=H_Xp^=d9BZ$+Y6 zsF}gN52W)pFK^yyt$M?%x&KkC?aQ^V#_t%*oSA>(rV{UiPcTIa*VW z-#Ww!?I!X{&u>~|+hTOOCpxZqU;LUx%bdJ1rFCk3Lf*t8t*Q7k6D@P{w9-1Y%BxzJ zFD=qqf&T@F>uye-R$8Z)&noG2iLomFM!affc z<4+AS+s*Tp?nPQ-+Zgf|B|4628UBhy%bdJ1rFCk}Ca>b|#`#%OiQhTIY&XwWI;OP7 zw#$BBnjQa2Y4#}i6m)*IG#dq`fVp5P*my~4b}4uXZ1%^}>?kl6OaV`W6=2KPO0y%t zC@=-g1s{W5UN6ng2G4;_mzHK#;2yBs8>LxaFcB;SE#556DwdUIy}>1*?(NcS^LI+K z?%*o$ELdlGX?78q4&DMMzFV681YGxCX*M6M1c$v}nvDaqz=wY+&ANWTSV8Sy@j<^& zOS4~pR+|0o^V00ef0SlheNme2^Uu=kO0db7rP-Iw%Cf&UFUvZvTbBJ8yaTpvQI?$m z?gOuaw(FH;)!-)ZH?U31vTOjj9Lxj@!3t2mepyxr7J*GRD9erl_ke|9;|;~zUX~pMMuUgI$Dn%Svg{%- z3oHh6HYv-NgDo~K%esScU>0}-Y}~3Ws{)sTC&BvPF3Y-r{@^n3Yp@h#o0Vl1pbA_L zUIZ&Z`Q~NWq2Nbg9#{@qw=T<$1#>~OEy}Xa;4<(Y*zP-J*a7z?I@r6BumS@wM}7EA$i!F!-xMOk(RSPXXCrYyS=%m-U-TbBJ0+zVa> z*R(Cm-Uj<`$GU+TVC(J6vVq_-@Fe)|4$KQY4BiDh@5p|D%fOppvvy@!4Y(651}%4D zUf?uvH+UQDurvDv=7EiOVZXtAu<@>CSvPPscmZs@8$P%MJPBItjuto*ya@)j=Ujk? z!KYyV$};|)B67hxd$3;MG;llk035a_?O*{|3EFii%ld)|;8~D$EXyiEfAG`2I9K2w zpi`%^Y!J8&{0?lhcUg7>7zL(*#h_(p#s&s~pMu4pbf2=U7q}ce4Bi7T(Hsi%Ca8dI`FOi=?9z%W`f0FrvsP=7!PKHA7*yUhiU_6)&HtI@0;4<(6SPAw!1Wj-gcnP%W#&rTl zf$3l|XnAN^)(>0`W`f0_yn9(z4<><^K{>z%=jyxa@fJ zz!#v~35*#$4Bi71TB8ZSp}29dZ#fqFbLcRo(2B^od6~?N6POQHf_6g~ANUEF4c-HL45bF#50-;2!_WnHgI7W68JtBh3QPyDfyaka3$`4= z_`n0;3(&2e@qyog%|_A&E(ec-7H4uF0at+M!O3TFJ%hKwmZNwN0xd_A14e=S!KYxC zv$=odynLppexsZWL z9XmEZqc0@anmI-N8LI;E^+?<7E1Iz__S@>=R;%W`9k%g~aLkj9@z0MHexFYNk?p@a z{=w9&Zrm{*$3G$TGbX2>Lhh1={a7E4`?07f*KzoAZ04*7?>eXHxNY}kj=EJr_XOD3 zK-lv!81}R1$l%MK$@5_AX2Q7HGjJSF&06}({@75vAslNat)-@lT>CjJsaddwnktjS ztNc(tHbn>Sj_=qFt&Sa+=c(e6R)uw_fz6LE*C@WUMR~||-CVOm4zVii_}ZiUSNo1_ zQRfP3;<`-6Ha-K6d9pFbJ~ym~{<5MR^L*J>r4e2W4{y32WAN>-1?^)VZJ2K#IOfZ0 z#Fw_H2)XHe%ZaUSP1h7Jf34w6*hifEjG{5v=Bs!DCZTi+LLYl~mDRVBfmM>k#bi}7P0t7H4{3{IsF=e4}3 zFZ+luZ80(Arh77uSXE4fPwxHIzGHLLDWBVT1}m_QSHdw*Hs;uuhILyW*2TUD73G@m zE7nxJ-nfR7*OFUJuC;Zr^=E|I!gpkAshu8br-T~E*8JB^=b^T*IJRtQ!&z(z$FtaZ z4K-DT>W2T^Rnv5z_2W5QjE4QsEYh;y_|g_*LT--Y}CZ`M5L{>;@e z42JE06l~w)VApv9>{*-~_!qF+{(D~I*P^|UT+*^7u-eOEwLgIi*Vh{j+8e>Xw`>l3 zFWMG%Ue&ODSv!9%eT^lTv}}B+DfF|JnwgQ%nkpt~RJm=1>3?g2ZlV*-Ca@Tq}^!S45H*uA+h#4is#5#Esa&G1I> z?IHfFz_VeWTz?I_kk7yy!!L#QC4v70w<7)l{B8K}A)dX}FuwKS*2Fh~w}7_{@$KL( z@ppsUz?}jg6!-}EyTtpz6>$F$uM0d9ZcF?;csuw~czbvPyaRj#yd!)o?7xjZ9X5X! zyfgd=ybJs!yes@$csF=4ygU2`Z2q6&O88^ge|+`}cu#oUW&E3-a4Fmo-W+zns$kr# zJM132SA(Be^Vt~|YR^e(rmUexUB|L=4LOcwX3%{Y_Ut?kJDvrwXXoX>uLXW5@Cw+o z^BL^fDfv@F{)T~D!JeJ1V9(ADA>KZ4XV|lIAnals9^$|7J#HwB&wdv@-GJv;Y@_+x?R!=9ZN;0pMa5PvK1`>=caH`ufDWr(l$cEi~z zgFQQ~Vb9Jsfp-qv0ru>4fjv8iz@D96uxIBuxINq-cCTEEsZXu>3``)GwCpC>y}J!| zucpK9)hyWi?jwPp4E)=`i(&Wb4cN~ie-81F1AhU#SL?o0TK+A#6tAKdS~e=wUKDB_U-z}t&Q1Cmx0c)$p`U+*8rNaw8fqNB=dtC}YaV~a@`my60z3B( zu-f~<&Y>&p8utu*Y~UXR9t69_XTYxU*|0iO;JAkU*D{8Mp#-J zYz;Y%A$zxBEE~e^<)*OOZD8lNec;^#?;W@b_F6v-c5X+*>WzhQv&&(}pjO9cR=>XH zcr9G9h8%Tg2K`52$N4m@_KUFNd^Pa0z<&w+DeO4^1v}1`e{RU%7GfiSQ=)r^4QoMuhnJu1nx)Q$=@?)TWW z)JzFA(~_E5Yp8K7-$8ey=hi%rg+cdS*m-;y_=~{n{iPwkN#Lylw+q}M@cw}h3*0C0 z4+8%%@QA?Y!Jfa%fid~e`K13w-3#lU|I{5I?!mVeN&FFOV92)n*rVAsU^ z*^;%a$*@p6D%4nC@%-x7QC+WzQESL?Y*Wyar^BxGy|CJIV9(7Hfu9fjd)Vjw*I}>q z<*?WKhk-v2yv|=6uJH|F-_u&bUQ=5I-XU;%*lWEryfJ(r?6rP4?6uw-_W7e9?3}!R z^!V-S=j~j^k}F>ZJEv=4=X6tuPYwL5zz+p}BJje%uLgb_w(l#@n7@|3%U3k)_13U` z?;L#D_3RM%K-kYuhrzDrQGri_UC%n$_pgy5enH?XVb}9I*!8?M#P16H5bSzB3A>)Z z4e?h4FN0mr6|nctFJR9}%MTmI(JF8S?7eeW*lV^k?7g#VXs-_35BA_|dSR1-=jaS>T5uekSZ^ybIyI;j2RY#=y73`x3tg-Vc5>#Giq`hyOcx zfB4N1{{XJS{~SILZt-!$xHf?g#@`z53hx}^o#1Zx`@@IAhr-?A>cIU14}y;%UJv(x z$HL}c0r!F@!AHVVLwqKD6#g9eXm~#C`YZ~(6g~$3efU^-Wr#Qbq_n&*emQ(R+y=I; zUEq#zKm02AB)CV2_l19eUjz4thlTi^-PIc8^7CzW+@im$d8#*!!UD{qI)Td(bao-*fJP z?c?F#&x6;)e=hjHgS`j69{lBjKZLypeGYpMTIcWKJE6eeh8@7xu=k)HL%cHZKCt(o zgMxnq>^-PY@K1t$emf=jviG2&u=k)*!M_0B3jasJ{|W3pXmap>4sU~hSMVQzy$3x8 zZwJp0zU)2d1$amJmEgY>_$E)hAHuHF=fPj+pAGljjbQu#cHpf8?+AO}t%T#c&VOai>)M}OO0qiG zHIQ9{;epSAU4wD3nvTECt83PrNiJ#GgRq*in!ka){-1%>d@1;{XJ85Jy#5q?*=zL! z*meDT@Ut%))^&Z@G4-Ks=32&7NiJ#G{;*>@40cRM!H($!*f9+ZJUH;kz~{m4^`)?5 znh@eQz<$R0s99NoZ^idLc6#6kV6Ta}ume~CdriCyyVq~Po5FvAt@{l2{!#L;hP^I@ zy?<;D`yN{X7oPKB^E<=dKMsUF^F3kjAIHL5!zaVuKZd~OpACDzy$IeG{xR%5{U&%j z_!sc@@V&74j|F}fZil}J-U(hBeAztN_tOCA@3D8l`@xUHzQ?`^e-CcPe)t~S7Crzz2(};d zea<)kAlUqaVe`Af<{tvb{2Kg2@rT0Q;j7@o;5%XUjQgB#yhjpWfZq%MW!UHZ*Mcuw zCm#*J1AB%GpA}wP^L15GQdT&_yTYDf*)zOX;Qe9G@S(w%y{@WZudCyOe=_VDJ}vn5 zuxI$(;9ml}r#}w|0wv^!Jgq; zgFhYK1^bo;}&Kw=KL5ybG+B=d8;cYu2hJm$d9eSgnDuTBpHk z)x)0Sa|2%jtL<5yv=;4|HFkVYL^)YQG%#HQ2oJ7GYkn{uZ!(nLB?ieN}~A z`)U5>n&&(=^m8L@{%wJ01P$j~xfYFuA=mo)wd7{&HO%Wfu=8;ZrmUr=D%5mOY8I`f zW^AarEbuj<*7a-o)|%I3Zm4+$c1~{vem`iqCSBK}QQk7_A6(c!IPTy0HPlw29bXFv zH7lGA`}10GEc2+1XKtfq4Zowas7TK|`-E1#q956Hs19Q@smlL^*Ozt83fF-;?vLDs zzGA=QLccxWP*R}(kGZdpkD|KXzq1fTMTklwD%z-2@jXCPR5Y6q5aew`5Kz%%lWfRJ zvb$_HKzwVeR8eW0Dz#Lxjn-PK)JCNhUmGh`e5sAqT3W45)mp4nqoo#=+TU~TJu`D> zc3wii|NQdd;q1&k=iGD8J@<9q-1&BD`t^K?**uO)N%ddVGNVvGz7oeKpC2Wg$Um>b z7P58BU-NBJ+EqEo#;wdt{Nx8Zw;AW6pyuD1E%|wyX~a*6A}`9H^u+L+&eeJ3AH{j| zudLNs#ugm=mCZbFmT?PoQ(k*K@-X$Awrylyy^QN|E^0D@9-qq3+e|j~jCe)`k>c@lVN{!gHJ z3h|7^IkVnyX#EYq2J^g0DQ@Y{q~>{@8vEtTQFs|KH zIm&MU`bqWYOo#1nm$T0$LwxjS%9etcj%R|>xoS;z#CyTz?_^$*n-Rjhh4bm*d|uY_ z$%V*@5z59xK~10Zgvdmhmu!0{gm*LR?*t|O+gO(Bu<-A}KG^R7W!nJDP<_n~k;$=4 z6qNG22-J+r&JfuimVKUOD8I);WCmE~Q%z^WXhnclJ?qO!-p3c+yVQM-2rv`H50_%s=`~=tF*#FM;Zu z&h_G)S(d?4%A4lUovu7_O8${v(zS!j+z%>k7^V8;eVLACIu?}DzYI!pF;G%;87SpN z_HGHutBYms1tnb%^6_J=gZkB-Av*e5=2MoTd`th~^eaxuKeLP_h04}5K}}m5LuBfh zmvpQT;qBIVw}$Zcvn@L`TLuDT%zB*hL2wxh4^(A59F%N64U~Ly4k*tzJ*eEn;Nu*q+{57G>p`ZjcUB(2ZDW*0WY!_t5}BOU_ppXH_L3)WY&bp^s&qzG?~XkWW;D? z+i*}b-UdQsN?B&ICNu7%U?0@6%ra1_iz``{Vqv)`XJj=3F?$UobeWlAr%>&S8d#C(6g;K}~+5l&ASezX`iw7x_*1 z@jEz2{vqDIp{0kYI++4Wwvf(Fmkp$o{><`cSY{Q|ud*!F!R8Rz$G}VW^{@=(HxMGz z&%BiXxPLqSgH!U)w5@QA-M7X2@32Jlkew5PN=fQRhu zW_vd>?E$6w-2zJf@8#p?K#BilQ0iCy!1OPmbPf1DP?~#w#K)h35`W3zb_|dH%-;`W zFI|6X4B#PsG3W!ExEPdCQ5vFg3?@h6DZmH44PSS&(jk?Puo|*tKb`ZV zAjb*}vA9LaWT<=E-s0gL=2UTPEcT_ZY?+9%(pG5;C7O;f;2I>&>OKJ(9= z&&qxC^Xc4+eD3Cax;*kB|Ij>R#1~E*bX&+jC0|l?Hyf1fyMT|+0;L$63K}(qKubqP z9LLQ#COyRBN6org09zTUSZ^07*}R_V z9m{-^>1w7Otb_D#3(?Wbykz4DBe>kuo|Xjg zP&o$JhEG9BXXI#AuKk&g1|_~2A4i$a0Ht!1Z&ruo@m-ebVi~Hl-Vm8S<|UhU1@Mv$ zRQXjAe_xRO7aya_(F965(|o)Nl*;f`Q1br`eEuM&M}X4!ek>nXfKuOkJ}8av^ZER# zpfoR<%*O(h?%&%N)QrI$AvX4a7b4;jroRLwU%dcIHogiW&L z#1>F0-&Roa!7HqT;(K?94l!Q&X@5}i6Uk2*7VM`ec)=!Svg~}O7qbq^Hxr_xi*D??#b+AqR6lQo$P}KS@;e;Vtkc5b!DWxJ%*C2aRe+4yPB#SbP?@_}@9j+QVfp~m$C(Z= z{haBr6P4}cqpcx%l!BM?I-BWqO%^X~m9Yvnjz5YWDg&)a>jHR4Z-(_=!Sp(&-vTB3 zHt_L}m_Eq#38n)~KW94ZB)e?gA$gaASC)-sWc`Q8WLV}3rq{77*;=@tzdtCS9+r8G z=`)(_>=4-jmie6Nu#@d_t_hJX1+Oe8%TPJDhR9@C<_b+_cZf_6%RHvZoV0(iPX<`# zb4{i$M5go7oAutu^yf^UV)`P}--D7) znzyxt=oF_aTlZ&L%=9RxCow&PX(iJ+Ocyil1f@L5Kbu4H?9q7pLwNgH|NAUMz8rBt zu%AjwRhdp?Iti3)qxKRFkLi*D9`AJd;ReTsEZev3nN3@|U{w;_bLWRhJ^+d_C_8t<+U-cImR zJ*{CGs;82|;Ij9y%ob3xy_aRFo)(124zTQQmZAK*LS#zHxNbqIE=guwBOFu3fUtoJ6Sw=%tx=_aNRGyNsgUo+j#^lhdeF#VM2(9={|M=?E&>6e+7 zGCiB=bf(ozmoQz*G|TiVrfZnq!n79@#TGAsn!YVLDA)%B%u98@AcVK%bfy19P*Z?bCSQ3E>rI z+GQ&&3id-(?aq4k9^e#+Q~BJwIuQ<@zP#q$-9a_TA{8p>1m(+PCeV(t>wQVAb*nW1256nS%z%b z79vwPMadk_^jJ`{&)O3r8)KP^S%&hPJjUNgWLp>WlJ8prc%w)+|F-QKD$7O3q{BQ< zl=KkRLzh|q8}XaYZE@+Rb0pu-c7MXuI7j*5eYTC_ZonmrQ}WN$Q3@G4aW1HtZ`q;2 zW$)yCZqj6$17t`q?Tz;a@KAYrS?>#=WZ!Fi{1>J>nf@D;_C_P;s`8eCQXV9;J0y=f z@PbV&V;QRJafbz$x0_`)YBGyMWcrwwY+4(_dvLk3=}1uWg8(JFhJ#YMs2;b3=!h{d z<+mqZ)XT^VN{j*a8N4qv7l5|x&8@}jj_yPrgfU^ju6>y zmbo94eDw&^XITgNdfeDxpNI-&`*2XQnd+)JL?#MeD(5Uv%6B2lQh7Io$ab>qO-yfP zdME23o41GP=wn{Wd+ZS*<*QV-js!K!HzPzQ#=NBe$`Ia8jdx=RZ;!_NY6x#Xc*)1_ zvkcXXI5N1bB~w)y$?mcc-k8Q)7sA`A@vaTw?a_F*gzy?ws*ImNhRWK{^nKPrvG7ia zj#B8LdZjXt|5C7j>Nvk;pv2$C$5*os%5Oo4jvkhIjOjBhOR>K`M0S9ADeqT9cuS_K zc6lNw`DPNR*{6t-ka99F0A^r|2mq6 zNj^GH@j@@jIYR2@dtiRVozqD_L^3!lo)26~>{L4x_G*4>|;32)Gtamcg zDAO5CYnWcjG|99Jl=7hRb%*593tlSkHkKj3yci-gz%sj;4wabNcL z36--GyhLwedMndASqGIP7NVn{d8r)h0(i+UbS>wd03NcXaF(*^aHhvHoxpSo)AN{S zm|ns3I;P(OHS1#ZvB7@mV_qu5;t<}#*=!%vW0_6>HS=2?q9emHSFjAl{nymjD(O3}=;jb*7Ft_hLt1~1j~eJn#|-x?y*&%9KQ z-66asbM11B8}DCcsw?UXa{)YL=k?GBHgPj39mkkn%(RK=a;8@@?PK~T)19D{C;4bo zNS>wh>@w^K;jIHN*|m&i$e*K+4=y+Lg)IR*q^}$Lz$P{_eF&6vKgsktP~sCY+tzg< zI!nPzGH0_4>FW)UA%BsrV@~id59!OW-Yb}1$MjoFH!$r3C7l#U(Ey#qL;GG`0X&p% zVYMpHkxWlwdKS|eOc#PuIT}H!56trUPNv^rx`FAvO#7L>&-4>eGlm8NY$G0eFJ0`! zV84~jSN5OC^mL{bOlLD)#I&AiE7Mg>uVwlTrr%|HHz@h>r+oY<(_b;|WBNMNcbR^~ zbPv;gYLuM^F+GCmc&3w>p38J5(+txqK*=W*Z>1*%`=p0?DMlJYc>6Wpt`OeB1*$zA z4r$yc1>p7slmPwi-z=;&dYEli(gSsG9GgvbuC%x;z;KTa+U_DRWNu1}_un4SwtJ|df2LUd%9mweO{ z!rQI!z7fLP$F^)|`ZmjwFNRO>FBjQO?`vod;Gq~TyimpJ;Y^QZI)UjFrspx8&-4vUd^-z)QtV!5F7fz3sc1hEJJm+GeoBJB4zj4pk^GGO!W7MY>&)Kb+jOaw@2gc z4B_q9csGUc7G7-oWqSy3OyeCoDY$H%;H5I&#Pn8{r8p@Kk?m!f7nr`L$+m>ZiY0b= zHU{ug9nt(?cK{FBAB8@mvzW%2E@axmbR{VD)z|Rx*FkBHzn+hO!1R8mk1!noHREz| zS+Gw_F0px6hw#QU-kuQNPL21C5Z)e*clc?+_6}&gWg)yJm)d2k3*n7vylX>vJ2l=d zA-p}{rMiBMWvJc16CyLfGG8zqQLD;9aX#ksVE@FJSGNBU-Y)QxZMU-w+142#Lor4D z?e+j3($~j&-(1Rxb*QtDqnT`Xcb;LwIK8NWnrWb;mZEL|9!R7B_neQ>(#Pm_7 z+nBz_^c|)jGZk@V+eoHknV!J34%BR0o5O5jncJD(!}I~xL1S)zh>m{nf=%pT8LFdk zlY@OwSZ~|7ID|I}UaGTMEJONNhsboY%uP&hWm$^PEg`bK%uDgPD};AI<1INexXdLD zc9~~~@W#N4?8Rc1p)#)ukfHjf`0NYdAs==@AJN;H-ox|(rjIjymgxW}+3`8keH+#B zA)u5e#fLa6xZE-3C4W|h@OFZi?7NBStt?CRmI;t0TWH+r3*aHWy{z{Irmr#m3)7uU z|IIX#Q1+C9LcKT}l=7nb7=Ctex$9VFnInZfANA%gXYH$o4TW`FT?auV}Jk zaeD}FRO1~wCAiEPjkh#}w_D>~62jZB@vaHsEnI4sZF2~36ueZgvsi}ewLe6rlVxsV zdMnFP9gRLG*gw7Cr8sQXM0e?BO6FWpQ>L&yxSXBfCA+?%$y5c%nC09Mz(e}_Snumh-(~s{(>+W}mnmDR zKDUPGtkZZ0LU_BtOa5EWG8D^WqrtxD)%3?gcn37zD?@loTkJA!4B@QUBf4&m(tFN!5@Vi~I2o)DQH zP5(O~y!{$);nZM%6sA=j9S%yqJ{FYfi2C3uA+j;>f=w)D+5k#rNwE%!w^boJx>@!< zmLd7aLuC4ym+Tz~;VsN48xLoCEYk^~R8F#SY*nz2>X?`Es|(@nV*R%>y@zGVpKC*8 z`?P$vh46~yc5Lnn;f-p%CDVe-n$dV?hwyf5ysJWZ`!wFW19+)U4uY?C1@KTi7Fq6t zm>$gZNT$a#En`~F^n9ilFs);{jOhxdUuC+M>35jk#q=jkw}6s8{h(w!#mMUE!9Ff* zvw3?$c%$GY{j*qx+Q%CKGGqhIKg!MvwxN^t-o*4)rgt*k#I&F32cV>z@?R37yELb4 zJ{#1O=?am_u*?-quVeZx)|wNS2{D;KN;{+{9Vm5vi?J4dNuuB0legI zS`YUJ@KD}jrLtix(~(S1U^*F;;$u3~1$@2>l=2{-k2ycMoPFR0n|PgNsGLh|XfVkZGK03)7WMuVMOirt6vhfa(2AA7Q$c>2H{d4%`08 zvx5Cos_{04@MbjLt`OdC@KTxYV;S=M;~_HrEc3o5GY}$Ea=BgBakKr)`UPx(3F1Yj zbo|_}M;3kz(0MccRV|&)&wI(4uFF&UL%(sR_qpU#uc-91K<5HxKHD|ES%1K{T7ez& z=!Q&;ncztOmM(+km@( zXTbk+&?kXyKp*ff@EI`l2IvM31I7Upfl6R5umrdaxDvP-*ao}>tcAX7fK@;SXaX(; z769i1Q-F!Ucwj7W5HKA00{+?s{2k~AwgWE#_aJQ(Z22B&C64C<&A=*PJ#YaO0SkdL;2>bTg@D?C)h$CPIPzPKN+zi|cJPEuBd<^W@j+gny0X@LEzyhEJSPk3_JP!;2BUYl%1ug?t0~>)Sfj59pfFoC-4uSbV6L1}{3HU8A ztOIocEC#Ly?gw549=#g20IPw^fb)T4fzPf&KMJe|E(6X5#sP1l{C5MF0lm;W7IfeW z_yf2V$N&q0$-o$3*X76`*aF-PtN<1Q;jJcD#`@pfL`EZVB9t6Lx3)T{(cVnBG3=)0)}1-+kjJmDZqT74#)y) zfc3y8;BnwpUn@|gc|ope@}Rii8I;}`P&{MMdWpLzQeHm6vaPye zT68gNOYK|Uluaa}R>IVZ#j%PDs^YnLbV8J@Zzqm)Yeic_GFjW4Xvri{YO{z_D{N*I zm-1OXq`Z8x#Q>Mo&aJ3iSSoehcVKy2b3B`9tWCAIwAgA&%?vB>tfR_rc089%O|V&N zD&X1)6~@cw!mWm40uxZb$uyjIoWk)Vik>4Dw~Q1lADum zOsEFU&XHES}3HvZ3g)ax(X^BvQ3dk~FZWZtU-KV?}J!C_(*>!mHS5!=+mhfai{*}|H z+-8(h{#8`t-ue(Qj>;D=?pwYzn_lT|Uo-GmI~Th4SavElyX`}j*2h)I#`rIjJ6i3- zNcL5uU1u_hRAU|&8XZdehHNK}h?RZvLc~%_BCd6_uz>``GVWiz3r$XhsaE2Gj!dFa zc2Bn7qOu<@qFAmP#8}=_xwk?de zBvBEzS<-emb+%ngS}&vmXhM_j$2F;<$0u)HOoO$9BwU3e1{aBBh-P zGJ@H=yU}Maif2=DVnN+@Wz|I&i4xe}*jSTjXmgcJj+GRH&E7qx^g~qkB&vs9R7SKW zlV~vOBcpkZdtn;O;jy@uLjFYe!6|d)7-jd_)VyGXXtbQO5d@T(vhN-qsAO<5A22Xw05$kG^xM)2+4{ zWYb?POR!wEJhK;bGA)jsF2{H&PqQW}^G1r-MN=`|QKO*VXfSM?M?4q%)jArB>a+i2 zMh)jYG0#2}!kN7s(diT>pVIx)%?O9EtGfbkMI|)Am@}w0n&BK*4yyb))w*4yw{uc< zgW{PVUn%{o&j6f^;t9&DEqOV-?F@ODS*_)j`H3bOFtR1c{^wUTLb!xzpd~euTh`W( zJ=>ycO7Edw(6yRzB`ve+1}rpX|A8urnzNf-i>nOtO|NPb0y*j+s46j2rTTr(w@*{; z;$6gMrX}f|Q`qwcsw~w=DU4`XuOaSARMU!NI*a*%_wbLb&zCJui6M+-KPL7oZ%)J; z6Kcv@T~Xhb%f=gWGjoYnnr^g2qZg)sE-HGkP-?+u-ZO zs+_wG*U$=u8zz2DlofNu0p*!&0zPbS8x;RP5Os5`Ad$VD)k5EwY0`l+X?DlGp_Set z`$Ws7R2fLKl)*hWXrrxEvgsHgU}T)NeV-8TCH*-}zi zz0XbOl1*4LnS+5eM0Qz64KHs@wxK}ME9 zmbWExGeWv#a=S-AOqFkSDiq&WVOEBoT(viEGZ5KW9H7ZZXK2D|gmD@72WcA(4gLzr zGBQIK#j$4=RNfR~9W_kL63kb9P2{do4Sv)DTa=u?A(a?vU;2CB|CJG8_aq|eJ@8fC7O{#H<4 z-`?1We#Fbh{ILNh%cV&P_5CPcfSg|#yGt#r-tG=4{LPS>Ls`%PM-;{A+cfWt`*v4y z21>JYIa6^3`27^$^{q5%^(rn!A{y->mDr9FoDH- z4{Jk@tY&)DzT9u$L;H-C=`?1NmP^bP>LgfCv)Kl1j_IX*;suUbXeI8&l=}~I%~FH^ zX{S|HEgeOvp%&7*cpf%O%$`O0`Qhm9}d>Y)D|nq)VQf}CdY(Ym&c zvdlTdN?I>h10Rk=ON*<0J}x$gUS?My&h=UmP|s?v_)!y2$ejec8*%R|rx9^kbgia{ z>Jw-cB)ep_0EeDx*LwT{zEou~h|`9amtSb>;fxkm+Z;buQJh_Y*s}8Db=?MveC;&4 zCKN=;QkWc$x^P}SiaFA2T=r4SpNz6k(7I4ng;(hNu(o5kFXT&~5MRPEr3W}w%K6w6 zP_1<$X_}4eUM-U=J@!DfoguR3LG%&rIawmD70YUtdkn9l$6L{s%HY~-4!cGdR$v8I zHIp}V#49ouE8ClzFm6ZDdSry&uSOXJASa;uC;*&UH(I@x$p(4ZZw)bfLoH!%`ok&G z>!vV`3VJ(0wTWNoD^uR}t*Ymd=e4Bk<1Okm@7y1J zIOeA-Xn$R953mSd(h-}FMgP>OWr52KWZjixa78AONad$YKcKw61-lrw(MqIx1g?{2 zn&bL;D(Jt^aweE%=bC-^6e^p%TI6sIHy7~_H5w~_vCQ9H=MV{U6ZR|86YN`82hif> zB~wuR{t$WXoExicqFOhZZJ6p=NPFkTTN91!Tif$;j;k?O9#x(KqwWTK&rr@px=+`x zl~_UHViG<3(3N^K9~$9J!@Fc}F|#$klqW0j@-^;rCbOt2ApbKrH?4gKFO`euJ4Zl`;Ul*mW?G7)`Sv6KsR4y+|>QZX7 zJPUn3?j_L5FS{gp!kp5QPuI9D%k$@r$}UV>2LIe$DFdj;{86ik>v z=zc+GypV%T&clYu-d2^w$w(o>C4|abRnJ7zIfpvp;%8KAL293axn_2D zQZ6Db^)w`j`4cGjMajloGx~pLTBFKI4rHc7$u7+`P_d9liM-)MoHh!*1&xB1uo~^z zYlxW^JqIgur~WCH7R8_@KDLgH$5RO^XrR(efSIy6f}Py)TQHmYf_&(L`@ zIxHqR;}~&dc?)LX3(_9xWT$z51o`qt^p5TI$%atoQd%~jbyPWBiw$Ka%}<<2WyDmx zWqvy~Lk?uyRCG~RBDc|V^`6Dzq`vBrGOWq7>17Fd)>NiCUs)#A8z~uif~-7(<`CCo2E(NKaDtYL4Qxk1>r>3iA9* zXFrp5SA|Q}^O~9vhwgsj3o9KirMsXs1{4*0W9lZm=v*~FsLq|&o>bFmaau&S-W+PQ zzMa%2Cl26u&p*mCl-EJ-Py{J#9{&{y@oBBIUXGQkqY?ijcQXy9u0Wvxsh8j&hN0 zSDU4^Y@?!n&%mycrw%o%SZ1J|X4|lO!{r=rgGHOEd1_U~f(qQzBKO(NdBRzS+z*tp zv?boPG4n1B4b1=vn`$Aw9?OpZGOi&jSaaY|%qRpV|KNY@0`7`|= zldtB=+h6sHla3m1@KsZXX=Hm4e=jJ9> z*5c;cG%}ekIhUJf<>7ids)1ZAxhm08ODoVeJVapf(6RUi9jUo!?RhO7nPzbB2gxEm|f46tpH<8!{b29Ayyuf@a*eI&D=$ zB10V_-fcIeHq)L>VArfgTn^dVg?1K|$YK71st#-qBpYyJa84{Kh}YwKm3W!7()><* zrJ$)Loz9963)<9PnHVw*m78j4&4^cs3;q|_FRYz9vu!yf50$@?$oU}p)sRLcA4tD& zc_xKg>nLbyYQd5Y^M-;Z)|h}D$#i{fQ+uibYp6(Vb4Ml(tz$?olU|8yx&;l1WQ!2L zFJKYTR)B3<@N6w;mG`0JDpdh$mTrs}Hx*D+3bCS~Nk)_qUoB{&suM?3uaGdkqpCyA z!>VPC9blBNY{t&0i$z{nsY2gz3FIfZ6L~+VxuCucx8XK4qkZ}1vanR$={1skm`kHX zg+pu2T}ZJXoy9*P?kh;P;gTCVm8ODtE}azOhd9%iT#6zuA6iRyMVwwH&VVegPMuzs zoCMGQv!JaF7c!fKI6!G%Rtw)B8KKZjrG+@SpmwG5!ZiG{1#c~AlDql19ulct+RD2B zTL43v=s<{D3Yu~W#Lu@1Fn>xc6E~25y_$%tq}nEYF&5&sWRLb+il9>uR!s<2|D}Wj5x+F zD+UP)&0~ht%CpZ6spT_e@Q#eo+6HPVsF@V?Z|db1{v$!EcODeF7&ka(D9t$@=Elp%=o6~9Z)E|iQTI!jhwJBZOh^;F80(&an zLige|ryHYIvYM=-Fybg!CEG_WuBB3+Q&x+PvK3uP4f3TM%~P6I+E{F})S?a}7#esS zowlk%98AZ!z=vfYY)#O(h-Y@r95OqB9jvLihLl7-^5l&yo<;`doxh*7EW*POxS^pc ziOvjV>42VkgT`kbzQr`+ie=Lc6^)HF*&)cj~jb$cS)fdvtb-|c>0b4d|d4O*Bqh)gZ5(iwB3+v?LN4@t&SK6{9#FGqIRN@6ZKi zGa!#kZ})0GF;BiS^n#R%Bg#RINgTA|heI(f@lPHyvrXRKhJcQtI^ljX6z5oTIy)W1 zI0iN|?G~#21<8hG(lr=o=(wh%4YwrGePR`D-Gx+`ABc@n-0K1bN&CJ03Db_x*;*+=VofxziA_pWt^iA_|WNZ?z$!cf$6k zpyNH#7ZF`$5s}%0vXnexK5 z(*cx`_`04356YG)iHN$1sLUY|QHr#Fl*!cHH%0M^Me1B3%1vnkZ?@>U3Gxpbq7T12 zA=8iYP1!O^Y?SPRSsMm&Nv)Mx)tlw~S>h`Q{#8MX*;a znsO#J<;>%~rGikW-i1Jp;M zfi?iz317xY4n7u_qwa2iukgDRZIuv%uKoj&eiGt)8`>~LN`M}48-@^HG z&=*Wm^%$#z-P%MHF`S!?sS?6TKK|WZRjjQ~a)5sCat6jda9bNfbOk#d*YKAI=Ss9Q-06 z+l6#0Lx}<3l%Q-sG(?ojgjHcb^bMSiyn*hsV9Ok|FZ?b=S_#N~So@-$j?5yhq z5Tm5G6R1OZqv*FgVS5a|io)juUq{(sM=5@Hquhn?ZzpIM&@6tHu~_p?_%h~?>dSTL-0i~#)Qq)I`a69U-6LpVs0>_1@3ySeB)Kfob;aTXz zk#8UJZ8@`8jK;ViW+R^Fg6~ks9S$Ai5O)WGo`pKBK@0!_V|xG5_`NT97NCsiPof{8 zjlrMPPGY}*5X&<_;;61?z9A8etx7SLP5^dijHz6rbxjJT&r%mS_ez6ZPrM1F+# zNB~~}wgB${!#3hQBR~we7I+94_TwUPA+QnnD=_+AyvqZ)3U~n64jg=6k*EObf$sw^ z0bc;eZNhswfNOzX;6vcRpWr9QISZYXsN?yaF85gLiBI z3E&ps1>irxDG$Il;H$tBzz$%*ELlE96?kAUZZoxuK&piIDApcVKU@FU;_;6vb`M~lP+U;(fSxDEIj@CRU@ z#~=$V0d4>u1>Ohtc^oza=K;%swZKn+mwHyWia^QO42f)+7pMgC<@sp@? zpc=>kHvyY~Ujy$0!+(Kt0keP>U@dSz@OPl3w@6$J+yZO^_5e$s!aGWUPk^z%ED~|x zo4_N$Ay1>Ofy;pJ0^5Lst$2?Muo~C`{2e&*S4H9y;9B5eU?*_=GtdRx0z3_T2%PjR z>I%38cogUdMr?x(z|Fumpy0V8aVpRZ+zIppUwR(#4EzxI44C%<;uClo82e(8SPI+& z{1rI%*Ej~&0Z##+0h3F0a5?ZS@G(&Q*CKHa&;+an zo&eqf4t=*slmk}*j{^S$PW>C+H3e({-UdeRC=xZmH-Tq?0pP6n;1l2x;9tNoe}~<` zEx?OFx@IP=nunidU543Th0k{_UIWPbm_W}AOU@h<{@D4EQL$o>I z>%eB<_rQOF;|9gLV9IzgE0{9sC(!Ywtd|(am25`(LMdAYBeqcK=>ff*rxC8hzFmE^d4&Z*^kHEp7 zqW=YM0UibVfx|w-yQYB4fQNvcz>%Lr7U%@t0Lu5EUj*I-#{38UG0*`#4GjGP^#G)R zp8{_KNBkGQ1MUF+3Y;j4MH8?A*a4hk6pIBwC-4C90WdyNEaJfRzz4t?LyEaPa=cq6)YMcpP{SIB-Og&y zY-F*x47dw;0~k4~SWE*lz&*f=z;58U!eX%ySPlFX_!BVtK$IUy0A0XSz;57-gNns+ zU?cDru;0POVmh!6_ycg<=wh)9*bMv|IJ>A=d{nbY<^k(~?ZAwpJ=SAjjiamOGw0QzcsL<|u_MS&P5_7TIyzG6SIzZfA7z?H{Q*jhOdR}2pp zqXqgVaflcr4i$%q!^K!}1eQ`?5+&j&akMx_j1ylL$BN^`cyT&fHDZCdKrF<@&th?*xJXB+$?Sp>%_Okt>Qakz4)%UP27%ldWZO)_&)X#?i6>4ABelf z55+y=N4Trx$KqabpV%aRBJLMI6+Pks@t}A}Y!*KgKNk;+E#eXJsCY~~E}jriieHFc z@s#+bcv@@~zY@=gXT>)09Cn;v5HE^fi8%D$!VhlA3jA6z;#&Ba_V?SemV}x;lG13@i6dDH_2N?$& zqm3e?*f_)(V;pK6W*lyeHI6WjG`?h%7)KdL8^;*qj4vC<8pj#qjpL0Ij1!HMjFXL1 zj8lzLV}dc!m}Hb0rx~XkXBd->GmW#1vyCanImWq0xe+xgj7npwQDsasrW@xOGmM$W z`Nk|`wlT+;Ys@oZMzt~Ds4*587Z?kTMaE*|LgOOiVq=MMiE*h>Yt$KWquyvR8jXa} zWGpqBjihmzvCL>OT8)&EHZsO?BWtu7IiuZJVXQP(86C#u#udhu##P4E##f9^us#`VS;;|AkK<0fOR@pa=H#y5>F<6Fkf#x2G=&7>^i_8jl%|8&4Qd8ow}lji-!X8c!Qrjb9nh7|$BpjOUE!jTej;jb9rt8NV_5 zjF*ky8m}0y8ox7MGk$MuH~wJ!(Rkf>!}yc&rtxQ^-}sC1mhranj`3IHUE^=Y4&y!J z@5cMaPU9cO2gZlSfbo&>vGGr1m+>#-6XV~;ZsSwqGvjk(kMSSl3j+(gh!KfIhD3%& z3L?WI`$UFE_KoZp**`KOazJEcWK^Uua$w}3$ib1(k)lX(b=*TgVagi@aj*T1_86P=5azf<9$VritBd0`8jg&?vL?%WiMam+lMNW^L z5t$r0Gjdks?8ubJIT75e7>PzIB9)P;k*dhF$n?m0kr|Phk@F+7BC{iNB6B12BC!ar zEktS}3nCXp7Dg6D7Dq0OTok!DvLtdzL=%!$@hU#Jj?#;uT$-h;Mz%IKUQZl^6xo1$ASJbN_0Vrz^~1Z)=tivoSNSozedfpKGC$vWgA-I< zq^Cn>;(D^JRNX$oH#We}z8q$P!(--2r#>w&L>?~DYtct_fM>P5tBoGfn4V1Fu0gAo zxt&-kuI!n=vOSl>D{Cw%t}|C`AlsOH$OS)0A#OZhn#i%NS+AxEQ=8NDNbmool4ad0 z=~X8Gn?fnc>ds5x{hS^>3P-my*&SuhRNELSoF7|$>~_1HkzCr`0t7Vy4<5H&9xS28 z1}1WNGD@WzBcr8ekBKwO zY^B`l=%j}o8Y8Krp@$xs!9((D;2|-7bB3e&XpD=>6hpod!4A@@(#tSk;r!5+M*1`( z)qxR5C2q&V6YPG)pc2*aQ?Pc~t8o7$-D5;=(Q->V93I3-xohc;e$SPbmx+f>(^e%K z=oWXkJ!Z}6C7ECs<-Bw*f3PrR?6O1JZId2Y=f_vXYZ|i2jH~<80z%K<&VEjMMbh79 zysyl9NCSS=Rp32$@&$3WD!Ta=cMo|~iCG+s3Ud;T$+*p_#j}0V&x0P~@~Ln$KJrbf z!EHd=Q0Hn(b8F{Lm;j}ci=UP^o$9jFVG=}2*>LGT-!_5bl6&oRcZ+f*+p*&_S_JZ| zHYHiOFqYM%4pvJ;B!k(J%*){6XWZqKl5fauaT)-Lc~j48!<)CYw+X2Fj#W+fnD}xw zu1Nt))awE#=nGf}%LZPPfa zN+-y-ERZ>Ytg4#OSn;s7mpWPX2uc@&J9OOXwmqbPGokPg%EOtSmxVLgnFI9&uH}$y zM}FC97P+I79EcE8jg=c#O)lP=@zaI%kt;pV^gwYB)BX6#Ha1;8BI{*RUI3^^H~m)f&2w#97*-!-t!Vs#diW~Q&L?11H$kT5t?Fz$ ztc99wgs#|Uz2?E+tnNXLqlscy;!Ax~r?#lRP?lJ=xXGn@PT1xXIw#Lz@iV{j-(o z)jOMs{=KKg;kWd#82l#eCWBw6XQuo2Ymy_bo|mUi$iu<%8F}@38@c41&u zw}udRuzPSTtLarGcyO-<%Y8UX_EvclBKA^!8rPs&qZH!aQ*ZTNyYhPKJn+JTAXo5q zI+ahdKPb7c!V=sNx?C=+E$LI*E+Pdw?=)| z|0+J5Z@9#J_UY25Ge$fdOue!??T8U6+-AGaQ_^&w^iv1fe*dUmkSfxc(q>aV(XJx2ZkweC_`lN=}jLP{H2_2fn)5K@5XUTsb{@a zP=-dSIdaY54u~HVr!mfFql-KHM)|f2&zi)wQ-@Q`y+kz`3KB#l>$4M2L7&x`%G8m# zSK>13aB<(h58I$9v@6-u0vC^6ofXmsSF)#y@D-*q$B*xSRVoeKE2Vy`Ol6-l)zc~` zi`!YwR8Iwg>sV!m2h0DUN;+q-s%Cm`tSWmPiJq1?INVNgBzlTtisrd%tFm{d?wtbZ zguyA$R{pB-(!|j(H=Wtc!=Kr`!m`T6YcJni@&v8CREoc#wL;KpLv{&ND!cREI|X*` zmU>?W?p+VN%qnH@f>y2}f_Zx>c9xvvn`geLm1&r$bCbr-lFs|)=Mu6q3l#F&^s)1^ zQ+;!_S+ts^%wJ-v8pVcsh;^dS=_~Srg|%I-gcsCkX?pr^KUOzNDmA~h4-{tc|lL}%wK2F z?3bDAY2LzKGj-0`PTC7mm2HTq|E8A8+MVv1znjTjc6YkB62I+JHpH2{mnx($Wb}s& zJHD5%*b3(suD2CVo;+tblRZ`VZ3HVLoT+=KKssU&3j8-Cl@0E6Pb=I^ZdbU|T}tqF zU-jMG!2JPS^s+aP@H$^wB{t&4gnpYJ^0lO{&wD5VXD4MFE4OgmUj$JiLAwOB47EOv zLU*6gN0D08CnV)JRFw)Dl^)xNB<|XxL9u)`JDf{i%FNs|prtmy{&MPqIqscW+WW4+ zYns&;(81UyZ^)z@A_EuYrgrIU?Z#G`*g5AF%6W8N-~Z%guPsUMP7B;6r4l-FmG6v` zcW(M`RMTdadbu1i^1fSRJHNs^Z`WFG9XpbPHo$RLDBfl)p9cuqk#rlL!1rhH#a-Gg zYD?xiqJhCJElTLmt@y1p9pQ>qx1;)$yLCNYTj8pz%-s*S4r~6SnphKUym?#}&)>Bt zPP{WTC98N)=DS@+8ms`xnKhRMdM z?S4`7v~ELmx78;38Aw~3qh9G<>E((eq~}2`PUjCw&cy5T=r@1-&)pMI2ccA=hW&0A z~Uh}E$+SJO6AmgEM#QGhq|Oj;^Bd3OSwbNyfYy5f|J!AZSQf^LWSI+_1rS1 z?t(=1o$V&P+_9s&9q*THN+udr44Zb?+C5@$DW1rgi4Fd&+Prq|%H&CEJKQ8FEeI6B z$309w?$<~zk;5gVzG|6f*4h1NM5`d zUGIr}YG3VHgO}cq>A%Tpu_5@;q0=|0o3jb<4=eiTy*r<|8m#A+38+D62Ss0kgY6aq=e_NsLbNgqdTcxO>@{J zdm6m>6dyAWbBAMELzhM-$@`({4Yl4Cqj&EPcD#_&CqQfw``QuhjNz@TZPA(fW|E^6 z5Hmfiex&ePRQ@Qi`Q-~dRv_n$$rLUrH6$wNYPgJ(pgx$H%qnOJFmrqsjbv3Oe>K0BkeYZ+n| zr)J}PyyAFXs-*)C&wIkw!PbD+yf1BUiQ|b~|GBnVay(&`$l_(i3H_EEn*Yn~7f18N z@KWi1+z5BL&=`|!^VMT*o8{|rZ#3A-k`2w$4^~lqy@HP~;Z?uEL5MdAqvzBIHkU0l zPfNBbwm}5y!ekr1XpmCO>rG zxYFk+5UEx?rtYxrM=o4i&`zewV%~|9WQQ!JQ;lkUM9DrKGD$DMo2wP0 zX(0Hu*;k;r_0ct|HOiHH>Odi7v9nJnS-b^Z-l~B+j?^790a`34u@I_%WjxzBcsa@0 z7Stb^BcSKNXO_37IlXdfT-^W@V6JN6R7mWSWpn>~#VASje?X;@DT+yp2swKRFw%@A zeU<~0msQ}#0oi<9f$f+L6QjOO;jv1vu3zbV>JA5UeMtS6TN5nSu6Q|Yz(Si+(N}SO zEVHI+G_{gO2YAWJY==MY0bPnovK35z+0)cYGT}*WIeR73O)qPg7;)bV`&OlDIa%E9 zONDM3S2-;?_jS6#%H{0^3|7!N$`4k`zB*-B;9zB4qu*e~JO;bLipq_}K^q#nrj~VQ zO(WEtxdoqM@R+;E8-NfWwAElX3i_=rB<0iOBtAWhFI#zUDd=i6QTeID^49pOWNUlt zZ26@gd*I9)*5yJ5>*6?Xu?DSXVwD#-K1xpQkwoQovPWN#zt`1@JEmk*`_2vQm5g#> zD?S@Yu65OxHUGjl2jHHB|73#{5{G?{6hyFD(eOXqC)FyStan`}vqK0*p|g_7>`m2Y zNwEdV9A+>T}chBBMw1wPH{5wY4Q$X>RISST!XeKD8_j<)TkDhiRf){1R3`gjzUl zRgS&|2!Vmk`-C%T0)3FE^>sZ?%IOf(YD5Ai6z|1`Kwe|*}CFH=zO zAe)t=A<|fEzvzPNKJDBWlRGnBrprl;yC+m_Ih{@7%?kXQlKiy;GSwJ3+YIq}x*K6B z7a-2DfF?+Tk&)98Vw@fCz#Zt`S&$3yLl3aPX9sypbFdZPdXu3?l!jkiz_iXB&7xC&-knVi;YfQ>fJq*pL*R5by`$J zb)*L}s@t_;rn-`BB7uW@9w>^=TiqbLkC{rjI-?5>rg$1{S=eNG(S4 zwuALBa8K^loiEa}Uh9015ml4$j%DvGe9C7t+n2c1w?{oE50)d$tGQ9fa(hu(jr?w@ zIm5|s{C>KfpR2B@!28A~nA#lo(wHaz?^#%TaHh(=&L_t^PXRS9Z*`J8ZFXg;ZV=o3 z6#>toLQvnxkU@qQgvtjy=fty1lX#z>*D#BBJ}j0qSm>8e@cBO8$vN-<#Tn2@XFhfr z@bsjTMKmnL=!ZvX?ae&Z=OT0Y^i1e;icmzutNlnL4-%w+o}mlc!si0;O$=rMNGkkE zJ;?G@!pzAjBX7Czm~(QT=KhkEsk|e=r z+~a5oa~0zzM@t;{;$W)B_d!rg)ZT|DAM>NomFM}!LprwZN;Jb$ew0&Q;F*rkHPeLM zo;~4%xUthK98JKXBYKhC%R}YS)!xt>;=HRsUHOq$Q^IbFgMztSe8+AO*7J$rcP~?e z#1P_;vU&g#-&UM{J?^SuMKi;0PI20(<~5!d>8!@iw~kzGAqc5Km5uvB-|I4xq760=(S}j@3^$AiEC|) zwE4M~9OGop&75X(s9-XAM-|4$UifkQwETJ2uTx;eYAQbzpP8@newhT?JSjU??n2oeVsmF~j z{yEUh5zBV$6w14S5!AF1AQ!m4R++sBHWclNax8FvoMXV*&q|RQ3Atb5o0&OQN@hHd zlxy}cjKv!`afA74W@itgwBWQi1ijZX?spHlrG2&nXiELxMjyfO?zuAZ6yS+A3RDSCvjFG z=!B(c`Oj~A1 z{kff?Z}0ArA+AIQ_3Tz?$t#v~25}$RMCUzZm8b-l>@>Pz1@q*9t-DfPiOnCC+S1vC z|NG{qRbm9ATf)+UPp zR~3GXy}dERZ@srSI^-(o|5_OT3nP1DifoksXH)!|=iXT2H`ecs4)^N^?4D^av}oS2 zf95>`ocD%VZk5&>@%&pUttV~Gc#R=$6v`!Wo0)ymHhktFTfgs=Q}&S7j8pcl?e(&& zb-jbl&Xey)#&a{0Pcw^015OC z|KL@zx3kn9oAb6d$!RWUuwI}-IQ$uOztv$)SgL!tPJ!D@gzp=EZTJUhXSLoAArfEpfAIhR`y*;a*p?5#d z=Ne6KIl2C<{XtQ~`A zy*%vI3S{H|FMnH?k1ixzPC=LG)rC`<&g0kByZQX)zHUChdA^%3a6a#{QC-kGoewAaa3A8)LsyBHgyZMnv1 z)Zsf>1zzW{QN9e98=@}TB){x^J@oLsk`z}YsMn*E%r_8P-H}6A zbqzkokf7%=<9YTuV?S7rGya2looyhP*P|We3%LC1Y7e3Oj&|X}t^3v4KJuA$URN8* zXLi}`*H*j)^?Y1y#*5ui=dQLB%I|JNUIMy3?zR-hZ_UE(k{M@;p)5vF(S@)+N6#3M6*Leo^sr*mo?|=6dey!m%rtVYi zt^KPGYmeu@U-4PzF=x%bT>Tfs=Uwngp-(p4I_LGd-?m<--0_E;dg1C;!_P0S`C@jS zB5RHLzkk;jGW?tAUl{9kq&&AUHg{du`byDfOvLAwoinX-%N!qWp6krRobNTQHO9~W zyUn%FV{Vzx7+ylcviWPB&u4z(?)>MDaZZ`fY;tbd|Bl`Id4a!bJpevZul)MNabGHVx@JB~vx@b1N&O_Zna@Nf8qg|YbK3u8kZbmTs{ zFgCz`N1vnD(c|cLbU8X59gcQKo1@jy;%Ig>IT{@ej(SI(qt;RFsC1M&N*%?HB1fU4 zz>)9B3+g+UVO(pCJ^ZgJ9O~x<{QYy@=jW~YL1qO$r|zLAuCuRJKmT$yMNjEIUGH#z zul0G)>q7efdOo=N$;;~De!AW8#A)>x($`++==q>ObNQTlYd^90?77(M(|coeyWwW? zS?4c(-Ro_2`PB~!pEHg%Pb5CO|Dy9MFFC_EWj<>ij$A#2PtV7in{im$Pv@<960*9l zuH;y(&C%qjbyPV@9r=!IN8To}SdJsp5qG3IQXEN+1jn>8la6u6+&yx{>0VI#gP_$v z#;K=u^45H(@`wvodM!Pt)Y@YWZLF;PbfM2|>x3f?U8kva`_$H_3w>r=D~>t&=ye)f zr(EI6Q`auAPX3WAR)o8F`C5h6E#q5)HR}{vcg!cQ3V-?^-e+5@rBBV%n)y6-?LzDH z=hrj+RmnOHeRi4ETVvg^6#JcxbI*ucS$8j5E3cNq_eVY{uuj`i{%5xR|9^+{zsmTk z)YtsRqAzgFDYZ@?ht!<8vT{!Ly613$A+A+mo%%$D^3u~W#EnMQa4!?N8nHD22{O9TkncIH!WMsw4(<)Z^jnw~{ zAD(WkyB49*HIIvH_cFJSHGg8irqpNkF*pD7`X~;+^|^KzYt47`*!-z`kXx>E|0H)H zD;Dc>^fn@y+6}{+Edl5dTsf*8gnrF5;hXW5?Bwqs4!d zAJ+ei;;kIpN9%vF|K3;rOY+0|pCR5+{C!%Hgl{{J5Pv8N>;F~p_ToFmVg0`@K3M#= zIIRD9;@!nhaiQZn$8qA1<%jkEyzBoN`E#%TyVaX}{l6oB?)87Yc<%N8ig@nz|3mfX zUjKiV|G%#PpUa;I4P=U)GJi05Aa zzZTEE{-0ET?)Cp~`Ty(se^&n7>;GQ$=U)Hs$)9`u-zc7Y{l6xjd;R}N{khlwU*!L< z>;FahbFcr0)t`I)|3UuT>;HD~-0S}>@!aeG3H9e*|NoNzzpnqE$)9`uKcN2H>;D7! zbFcqf#B;CzH^g(V|3}rId;R}i{{OoEUzR`j`u~CYR^-3e|DQy2um8KmbFcs3i05Aa zPpdok`u|VDRQ{H+@LVE1FInI@K$zVH9?KWF&K7t~Uf_CJ;IVaqeZRnC6UTOr^&Fc!Hg;_5FmvvC3!-P7Glg5m!Wia0 zKM$`SKY50{c3ybP;`8#3S>=bZSHC5ew^naNbu z+55lQbH`cur7wPF`Uw}EyXLoA|C_sZpDR`zSzLVh%FpdHe?{?sYtHS@*R{O<`~Nds zyUoLnJN&;nYt=c^%j`9yS^cLA(OX`hK1Xvfw?Nl%Yrm%v&BNRxYqjy;bDmRT?l{8R z3gO+1IoWI1KjP4Jv)9TGpY_an>*jOvq7Q=d)}EVB>PBzAt+U@g$&Wt#`P}*^`X9_# zS$&beCs5>5V_5wMb8_aM_qv}^eQv$acwg|pbw3%_XScofu+}QG=6&4grTyx+NLCO1 zGs=ASdlzff|D?!g)c@oyiaEtTquiReA^vxn)o*3YE%F)deDVQCnA0VWgp(G@CQ}&3(cIJfi@S1a+nH=M;9OZ5t;qDx!msANu46p45 zxhDr0UaO}4d0{M@{o=jZ$9>q#ec8htc5^>=aep@WEF`?fo$T|zH9Msr$POOFb{@<& z=CPHBuwigv?0L3`XT5f8;=8Qp`+rYJsO2;AtND(&g5foA8Na_cA)%N|iV3)o%kI5?Ax*e@jTHWViG( z#`Isz@PfL4{o2jttI{+0wtiDNu6z>Pv@?6cqS%=9DfTHp5n}nHyhr{Jht==reC>Ag zdatEBc@^8};VYqqOa7CP(8xq_J)hHFE&nO5qL*0-<@~MoOSqCnd{(&vz9-J*9v;lH zcxb3^T^@}azj(g-X-s5{i;OFgdn!M3{G!+m#xuq6+JA!UD>ure#x=xg^VQE&_0!95 zD%Z^w`e}H6VeBC7bcolN-o_V{Z{c6r$YkmD{F(AKOwxV@KM3c;9^)(u>B{AEBj!eC zg?#1Gd8B?)*lYZ=MXo1K@hJ}RC+y)g+qpvdX1*w{We>~vfb%WoxAjxZzZqvfmpR`Y z9|Pt$Ll`Nlo{;=ESP9ElTXUxaX*(&~>cAA)?UIVvb z9nfnoE z9Q=cFjoecI^^rB~QLmD5^~yM)d1UijrXvTN2%Aqc5w}V#tNoNFJlu+c$4)kVgvJ- zu3R>+m7l>~#VK6EBtFFHqZh>GG!@e5(SIj)})Uax!+FEGA5mM?K0TyB0c zc$zqkryEBKkI-(;wz1f^J)k9tr`1p3joNFleVfHoN7)zRF)kL5@O=((Df`)LT)j*a zcSY%KOc%FAHga?0s^g)?UBzZ`32$INo0!EL)yv>s%B68V=`p4|?UwNH;lzz-obdq53!dIvzvFblP{WwHg@Q*k&m#3JF}e2 zwO1Nq<0xVV^Z8`l^~JdSEH0NG=TfGIbS5*yxD%MSz_<%s7v^<>6=B?b+4&4{hP`Yu zo-Q7#+N}G)?#9tB-cvtq+)dofU&wFdAGK4*O!+l@nN@ts_$v64a;5xGxdN`QelC;D zR~BEEALnkyo5FGV3H+1tQ%5X{Wod6Ba+EJ9H^g5m*T*f`%|%P>YYwX3#!2;?IiP$4 zQrXE6(Rm<|~H>n6FHR_r}w?n{uhV)%?YHvpA6tiKh>D9}n}& zGxa;hTjY;0gG0IBzrVQQj#YU@JR0q5pR7!xr|% z-5;X3j(cjin#IOd!Gpv2xP$sdd|$iyY*9a#$xTdxHEZ({5bz%JgKZS-WZQD-XZtRtE@w!cu4;2 zibb&>$e-dt+L_=_jc<&3;$a@j0p{z!m%Hkxi|44E44##%nBpK2bZ{jw;%i234t9?l%zp?oGQmY8q8C{E)>&OgQik8{R(kMKMWMd^LKLVw+SUHMKHhH>&(Hu0~@*Rx3e z$fLenCB0VsT-^Nf3*riHubomJWnPPTzc?po+}Uh5j!d?RGkBtMsmyXd$?TAx#47b> z52AjjdAxqxC%vB4-=ugC<;Qq_I3J!MzuEOvsJ%gPv3~kl!d`xXUHqB)9Z}rEn00Do ziSgI*ORQ!o%UPto5~eNje3W+^Ujb+Br(ABPekM;*KAn3SR|+pMu4H~g{_KHX|8Sfq z8fSuWChuU};uH=tydlxYRCY6$9n54av)RNn)-%o;`WP&slJi-{l`Q6|EZ`TI%afSJ z(-`OROyw7u%&#zkrJT-P6#FtKcnU}PRSxk>>}N51cqTh}I@?&rW)`u5m8|6%tYQVr zS@ZZ*{m#2jF z=XLDhi2c>VH(AF4`=W-W@++BYy-Hc8og$vfd{!`v$1E_9Jbj6ICgZ#$23{>?mB`wh%?KDFFWdKG)MQ_gyp@&Oj}`^@KZ{pN6EX0Td6={$Rh>xnCw z#FLC;%JbScnIQhBa??4^OFNT%jpKYFZr}1P#fJDb#a_(kp2xsQHR+#w#+ zZX2uBZ)Uyp242NlHn56ytl+-dFXhGBEoO=K!grH@p`Ai;qjGutF0*-{ab)mL^ATr` z`l+l@KgN^QPv)=QF`v}Ka!G4CnyzJvX z+Ue#c`s?K7Z0CPm?=5^e?z&-b$!6WBqn;sd2Znlx_S9llO?lPGPI~t88Hzo4Jxr zJjEOT4IyS7Ph|~HV-?F;!SDe=8CS7{r?ZG>uz+PgNXX+7aV}Shvw1o*c^c!~P<|Sd zn8FR1%x&c-u`F(2T*;X|jE7S^jT0>A7*}zGr*nvBaDWx;}4M}V=td%H=klBZ(}{b zWVvtmUJuW+yATIV(7i<@`QN`9l_SJr=Tv`8@9O6~J zG}F&_bl1!E#oZy@7jZj6%vK)6My}`mr8+*PTxCdSX^7<)@T&{V2dku~@Jc3e8y_ss zW~x8W`1u2mixYe$(YW}icz{1-FMq^N{+MliNxc>hd3uZo@i;&z-87#tMedlS&!hrz&E2KO~>& z%;6GdGQ9puXLucw!Z?!{-hZ9lY0c}QDefp9XLwyX!tlCskUO)F;dx>=!|TQlhR>Z^ zxf`1p-v6&>6Kgo1m0XWyOk^<^uz>3`m*M$j7Lyp~hD_x~OlEkUmcV3AXRLW$Fu{J$ z4@dbNhZxqqpL?)}d$N;zv5nbm=H6`J@vP-Otm3{bXAVoa9}BrZ^LPNWnad0w$TS|r z7!PJ5^Ek7k{m4lk$}z6sF!MRU!`REi*~KH+&H}dZNH+2)*70aovyc@$hNV1~MLdrA zyq-Bcftfs!=`3anPht|kz}e+q-*Ad09Oua#;g>kbFSC!O?B*%#;6k?YR5tMf*0Y*5 zeBQcL@~h%9p2lKc$O2x(T>ebCEDkWvUrA5prs2BcLMCw&Ch)BF?Smcc2Trhzqg=@$ zp2~in#vYclldIUq)7i{3*uV0q)9P?#3?e&UR+8g?q4(d$Nvuv6|Ve;NC3dJ}lzC%x4aBxF0jQ zKhv4V6duAP9?IEe_9v&9&v72c5gyJ#7O;;;vYSV-gGaNKg>2$6tmm<;;c={F5zBZy zi+KVIcm#9V&MaQRIIm(VTbRrrFoBnFda390oZ$5w<>MUUtL*3N?BPT9O(&o7IM>E! z*~}p}@DH{)%aQlQF)Q1iP!0zZm z%cq&eKE^}7$L~}wSYrLTITN@gr?+?gSl0>e%2DpdAttk*=Y;j=li_&)#}dpduk?JN zg$u&-BOdCDbalMf<9{`qJWr_LVvqNw3|~?%=Ii0U74p@~4e88c0n-_-vlPDYvGwPR zoLcPtB93uA;~3(A_Ig>s&Je5L!h4LPk>`8gp@zrlzlvd>RxsRGN||My3V5w{a~SSN z>8ua^GTe_Qw{v}nhZySj^Emao8SZ1P{HFXmhU>YKQ{rO&g*lvNI(Jhq#@#uaW}k%l zXW>%w&mVf;HO5X3bKX4rg@-tgJ}x)!-CSSX7Gm$GH1VgNm)3Ks^Q>WYJs6X%BF81ux_+q%ClGY;;|QSM@VL;SY!^z#Js+|6&Yo%7ioV)I_d1;$%L ze|D2l!PdBa$}gF>LWX^q%m2){?pb7987yELPhb)cCN#uK+-O_$mZ;EHDH^Cy~7-O^jI2`irmjR}T`$D??*&Skb@Qtu3VC*)7(*pE<=`;~ZtC^kH^t zcOb;uuYuC1>zXT#WRasha6)!`xxFY>EJT!*2WDMTURznZ{(rYw~pIb|0>oRUpYrv z!aLL};pVO|-TwZ2>WbrTJI6qiZ~@bkAGqo4^S_aABvM0-oKw&=zc7o;6FLS1C{IJKg3-e zv92B5&HicO`fTLxtmYz?aU~15Gjq5<(|NysQdl?7{fWPKzewP>#50?E-#n}{f5|Z} zO*amXnD+s0?KB^<@_wKLB49~w3`LnR!H*p_Oev-)?_oHW(6N*5&z0uUa0*{ z4vJIw7-u#%PsTgVi}%zYx8eZLWH*n}UpqGxH*q7^Q5}<59nx9OWR|eO{1owP%I9)R zX0U=OEYMyOlf=^-d4Gswe9=A`;Vt&f0B>b4Z((#|JaT~v39-A5d{-lw2 zr`r$QM!Cu;F5^kA>teoQe7PKA1~b)5=jO&4;})DvG9LS2is5~NaehTV!+b}2e@NGU zH&c{vWq6&?z#{$B@?GUBxtaVDJ|Vw=xy~n#;rU)RcVnFEEwc})yM!1wU?RicEzfM= z`rsrt;TRWjn458co3ocG?Bdti&KO%5e$%0mP9vd?wp&6qw`B#lVkr$bp@_-MXBu<3 z9W&`NNJyvcpOC^Wm_(Op!tDCqZ{ZYezl3o*?Sv6p-h@GJ!#-}wZf?&GZp>Cb$R_^I zeY1|gGLKbEw% zTMV#)FI$INzQk&_8%G6S(SIplWif|Yz}K0_a^;5IN6SrYw)lP%l)>(t!QXVAE zm$E0sVSPf(7H+JcMsC77o~@mV$RdXKk@MMQ+H?6m^PI&q%wL>;a$c$Y znK+rdEK`unIi28nmi$Rx<2=SVV;sF~H$UBcK-|F{*%o5sY7Q|Q_(kRFc%^o#n9ow4 zzTCX=ROYcmx7mEdI>i}RF2*D#azy&{e|#TYJkC>%cZ5IXAlG9bJ*+15aM@D(hg-0X z;r-=i-nfJL;=(Xryu~_JGpti3^Ig|vtXHmto4T^ zenLn$kg1QA|JZmrE`OL0ryDPq>$i`;GC$qizUTu_GbSbi@mP=6z?+N3I0Jlqx^#OVK!=ikgLKx@;P?%FVees zuX62tUfjz2#Laxiyw>t0R&n+F=UinPO8Gk$^APJ?$m{HrJf^vhvblx!GP#NLbY5fM z#CVu^X4W{wquf@xVQy#L`nhU>b>_F(!C~js#^39&nNNrH<<_j^&sf4|n9paK$>*5L z@OeWr7dWp3E)4zu%RC7uxt0FMBS-j{dV^eOe0@BET@0U>v~xneR$eE)iDwvRJ-2h- zH7qc{m3%@wW!zqRF>}-_WQFTEj~^OG7Q^>g;=JDTy(FG1efFQR*!j{Yc_l}AD+l?Z zetUT?I~m$*WrltmdA9Ya;f|hvm9Zh+eqw5Pykn02o5jmQzr2pAoY77aFJ7Ymf5c)n z>P@o5^)bc`%-;}KCF-9Y`s-q~c3XqS)5v1yThH)0RSlDt+CMx?KV`gv#k_&}yi|WV zT&}-NeyDsZH@7Y^Ze<=5xCN(XVyi#*8fS{~quk?T^TNZ7qmN$>`-Tg&)5>MW)5P#O zSv@ybua|sW3wWRO91bvp^VLh|*2<;wMCFqCW%H8A+c@)gznjPj zp2A^%g?(&`+t>WD>$Z#EV;c{2dM(_}_#1e-an-U`f0bOh)Vy)I{)(ArJOwP#ZZ5;; zZy5}q^QG}&Ch=SH6L}S9|K@&YUrq6Fj`AVrF~}?B_wWeo*}(_c!n2I0j=L>!ALXR` zN(CP^?s8U`=MsKjT*xWsQNSmyQy#ZvHm_GMJu)eB=C5ng$N6jH80AXmH^ffoJ;0wC zXE%>vdx*pJM#D*HV5xR%m>cGacS|n|+TTSCoJXqOyxYL@Lc^Q@|&FgODuLX z$GMsL8DV(eWsu>0nm&fliMqMJ`t6a8QF<-I=S5YKWelGe6-VYXd_I&LnaSUZ(<77l zbMra#5qB_;44=1-ac}cE!tgohApc-q`&e$g-MrR)u!Uz!Z{%0Za~=O+-m5}7%NXhx zaR>V~k8$_A3|{GcQ@Mq8PvYLzedf>ZXWE}+Bggnx^@mxl{Q+L2UJp-lzw6|D^U}c^ z?1MHQuDxd7>hZmS+bdVg?O4U*SWZ8amQc!U=UL2h{S@%`&L@|r=|79tIL{QGB|RzR zJFl53F((-6jWF)~2Dr@mb+O(!+n8cJP5it1HOzLN6}(vcrM#L&oYqc0FLa(ctX4mR z=g3dxH<-kmIP)j>Z|5<|7p>n2@0LEqeD(TyFMD}Lg7b1++@k+3@n6kT2e%TpF?>$i z%p1*PJ)2p@n^?wMoM$n&VgU=CXD&OL#UI)K89dE6QaQ#X{z3Z8hn`osz9zZ7e#iL% zN4bnc{H1XZa655tl-|iD;`YcE?v-et@dW+W@ex+@_@&0lR{Oe;e_<}~U>8(6b`%QexJk_vS zyOq3|CG2AXZ(=r|VH(Goz`Hs5hgj?$4spq1{qYFiDO{z!M83tT-+SJy+$b*;5Ab64 z@C>$dm3eRG%K6rVzcml_EHd62KCImeKEV?1mu?)~pE>MOE|aOwCyhU6j1|f!^Fi}F z^E>lq-6!~1*eCpv^a1WC?hEPGwL8S@;I7KI@*d-B#v9}OV8(b zoo^P8(M~#-XeT8yk>#$V>516t@Ar(ccZvPVKKAeh?RD}I`=%|jDY7oIDzY>(KQc2i zg+Ee1DRSn6HSt8`NMwIxXJk`k4S%S9Wn^h&VPtM(Mr15<=KVGG#<-((9N|LeJIGxY zn`iFCZoXk3c5(;h+PI_iW-enxNM|j>_c3a?YPtJaWEoeAiL>F7<)(k@zW=fHj~wKE(!1EsW_GZae^9Q1HP*G14@%GHr1WfFW}V_3k)O(Ita~E& zkv=_cf7{O!+)q5h@cohD$N`=%zmGp;7ccX8)5>kxz;js5W6fU$HxU=}0{s>6Rp#<) z{bsOR`{|LXk+I06$l3SSl$(egj_l{n=DUXl+UsPM_S(3M$ERjqroRTh#adpgTm{4T zI?5tTB8wsmB6A}%B2yw0xtsQ<-}U^?ePldxluN|@T*^)!Y98D8J8?6^_b}_Zt+$$Jyw7V*Iso3EWt_Q}4L%+Mko$Rq{B0 z8{U85Cd&8oH2a~4?@RAs%(z;4rt&TPBbyk$r&rH!yWiAsN9$V2UA14%A(nE)xQqFr zdIb#M*UROG#+AkE?1NNZ%|sr|i81#>4)H$raDC_1&UtKNx$)L;U->1xm$}@XaURGd z-oeS=xUX}FH?y1ju!Xy@j(f6#?JVK~{baL6oW>j`@&Qi19gE%1VLr$n?#4Fm%m!w& ziu1Ko$~(pRT)|B4#~2Ud^sil~9ObR-=b`N6UTmhH6Hch(-r@@G!6F{O9A+`jTqZG- zQ=`^{Bium!0bVQa;&<7~W;U>i)of%buVDcv}wLgn#up!ywb8 z_eFMdC*|9@!1dk2@HuKDf5ke6&&8{Fyz!KC_i)|uG1q%BUtm6)n8h>a8y|llPGx)C zI`RVJPvi>oFg@aW;y5du&mjM4-g-lKo@H)nHP4Rs5HqKV%N0`Q;5UbZ8V)ihmpH5Eczn#Bw zUaj0)+{8PKyN+LF6~p(0%Nf4ETEaU&w!e56v)RUUhVN~r@M@2jiM&JqGr#n@fK$9f z|KlO%2ro0<0VbM{eja81dU&7lbny}WxASc0m${+myT;KZzDaxa+{3)pGJIdVinp_r zTgWfwTiP#T_?~M%4=}Er$PB(|+^G!Zl7r%z*X?6Y@EGMsc$4uCatrqHXyff>zy8|! zu(*X^GyZx$ZokxWiWNNHd>6BxxvXIZFJUq_v0rD0U2n!Y5jnuWYPW~=@;e#6Pus?0 z*u=Z^U&|`}mhuj99yir)Hov1@27hN=)44BG_-0(cQ9S#a>)Jd{u|Dh{9wV2zRE((VutTK7w~@bo5KefX9r{azBVP%;T-box}CT8NA*+ zrSTZ&7324q$cvr#)GO9q{wTNP08`k@bHg~eDVtfwI__ayRrIs&38maiT)^;s`dsd9 zzO&e(ojC7j3Xj%b67Lbu47pz#$0Ym2W1MHa!~Bu-0bZ(nHy6wA;5W42!mo#MaE1A- z=Ka>Yk`F0Y#`Bab=F8#&-sw8a<8j)}<`0;`8ttdBN4X?E#o3o*u}artf$xz$Vx6YM z`)g&6*7NTy=RcXxe=yE}F`0kk{V>Ejd5 ztB1Giw}YehWgBx{ug#HlQF=AsR==Dt_`X;XeZ}-z*c>hEGI2*)+ z9F*V9KkcAD-lpF+R+_gKCR^V|9wfbI4l>R9^zs662fxM^p3VlAJC9nPJKsKFqw!br9C0bz)9nLR zn!kMh+_>_1vi*_GK95^j+{?OV@JH4&jRzQKjM>r?`A*meFM3@T)}QaOpQo{tXX~$( zg~r>&A^WJFnf65ucQoEgb~?{8#+JEWnIS!oJ1CdUuNz+`k9A(D+|&Fd^BD2$3)a{9 zP4fZsKFLFsALr&vT)+I#`3&)TpH*fMhr&fwd|o5t&$ZxRn(Zen-p`WAyeK`B=Ue|cZ_rOFS80FtIiKek z&NMG@e;(sz9Ar6rc!+uEWEESOD7~KNimP}Y%lLKUF6M7oz$Wcyb2HaVCe!W1G~O(Z z@eSiitZ;l1qOG1_nCUE)T5Q@?e5MqJHz#TC3! zxnllOdI6W%AGv%$oXHNR2CZ+54~rA|BhEbQ{_6UiW%X!9APg9`51foD7*MH+xR4#d9G%fcQOy|2MyxzK5QNH-LGnxA-#$n z?hoZ5W(m)79))~?dA!p3Wb#P$()kVTr*LEYB#CYMNnnBenP*(DoaC>~|2U6!Jq<sbM({7U$BV{`m5zb>E@AF>#vNr8gDVTW?_hpH!sA@<_6(D!5g)c#tW=( zj2AM2=h^3zecmtR2=_Mrey$XEaUXFjv)RN7)^eWw3Z5Y@;pxoh+4{-hi_GLz#+}Mr z-M12XjPXoA?eSLklRR^M`;31w{$W;_p8?+BbE8h4C%rA?JD;WyvyP!&6?f8488eKh zfZs7cIow*m8Qjn3MX9XSZZeygz<;Pe{gmffOFVz#CF+gvg3v#&Vh?X)2TwNcRt{Rn z7KXpqY~)|`U&lN2SIIl&m-0yKSj1zc=kqq}mBV|TPbQZ!9%AQ{8e%4MC+P`1SG&_s zdf%TD{El{p*(SS>n;TCzw`K>!-wn3%{f~`{;qUWmc$fQW89$I-$XDH$vw1YrI3YcW zuZyR8jh7?5hW$(q*Eu(2XJjkO#7#Vp^*o(tly>(v!3Vaf6(WPe_^e7SL<2Di{=|Ae`lYS@DI%6joQiPwyw7f9%sL$bCL8E zUa#Fm9x0xD%=1d|5MNbpfNz`6K2Gbeo3AW(eeqXpn!6NEM)k*>pUK>oosGm zd>PzaKk4-Fk&wa-qH7ocVmhc@GGK@2q_cDhI!+80ScGGx~^-kg!jVpl<>v#H5 z_b25id7^ekc{~SMf&OAFi(5~A*ZpwfC+@e>#~4pHfBfEsz~V`#65 zPint`OPqf#)5HCn-_TAe7a2z(r?sEMJ=`}lS*`zcHZX;M)?N~OwKw}?*R}qq_?UQ{ z$J@W7JW=`(|6o1`_(%3|^DsUh%~lqhw`M-1{YJjOgZn9WSY{uFbnRC1ah7uXrRI^F z8D~C^VirBTBxLYE`bp!F&NG=0a<<#!lJgs9$RFmE`5NHY6Rjuzs(v?*vECg#jxAhb zVH$Z*x^eNSkBx^5tX~$=@uoo_w1vd`LTuA)j?T-+5MZck^G)SHvZ(cYe9-V+OzOd{X%n z{U>t=^O?X5Oh4j0z3v|6A5G{GC)v+y(p{HKGLNl1*?Bhda_KevgL0MJTY3q1(q17C z)?PMe8>L_ zz!ZMh`6cmT`*HUB_A96Pu>Cg?Vvh28?Tqks4)Id=nI3M#c2=^9$C&3jCW|XsGvB;$ zfBUzXp?*H^bG_zrvGgpaF`em*u}`^VUZ|a!@43HloL|%a5LbzNd4afthq9Fmq&Kiy zT*Z8r@gf%RVrKI~rnB0*q;Qx?4EO(;POqcQ_XKw_-(%c9oHxVY{q*yP(tG$z=hwxJ z&3_wDHUG`L)cn@7NO}!N%yT6#*H0O@Uv4~1V-~~jzNKHU11Jv`s(bh9Mf@7Q5p+ZcW?qnW=nU-f*0 zRUBa%!|x-ML>5NoMP^54M5aZ?A`>I0A6(PUSma=2cVtUsU1Vis3GdQ>J{Q?1IUFs8T zTX>*;8hDHR+Q{Hb<{^_`VVuv{H|e}wekw0xGJnjOd(EHx3BD~J z;rZ^j{anN@p2k+@na2k1&uX5@Qtrnh9>82yGK1@>m%@X^3C!i>J)YljmmOgX0bv1uuOp`vq z*Q9sxDYkJx=?#2UT*b{<#^deJB7W6A$Y-te9A3%{o}m9UUSm8l{*j5iUHVj;WpckC z=lATlQI1-dA@0EemfGLF48PaY#cQm88^`1~vW&GMR<45m;zH&#JH*mc`DgnsnN2(B zpO+Zl>|O5PoaEKUHO4=1n74*~$B(qr$-l6Lzt&C@5ao0sJpI`@j*}@*}H8O{FJjuAL z`D0e_GUG4iKg07@zGHned5HN-i%g2rXYTNRuJf7XAsk~q2f2bh+>4#eVjJ_=%&T12 z_1s%r!z;xV%odmOP!_R4dLH)`XYE?+qD4)s) z3?~-StxF;w`PjH__kJ_S*v&zP-#hA$?2YV>?2K%SY>ceo4eC`yaS1OM7xK&2F^^9$ zn^#!RbRJ+lDGa|imKY4z(QRIThVigf`9a<#y^Bj-pY2@67VfKl6VFh;o`1B@s(HHk zs$i1`vO=CCJ&&g>@jQTkw9nE*I%5p=6S$-Prf&6qnETlXznyMfxRw5UxUa|I4qmGL zR$j{{{$2e#eogz;yjZ<*R>?1BJ@eVZY<|(c$l&*Mo5qXHQ;a_`&P0aaznf`sKho|5 zS9;%Xn5VFpW$fUoZ01*4%NuskKW}6O!{>UXQF;;g(S8Ac;=FT1%q-rh|1@r4pT>Bk z{u3B}Uv2sp<->aL(Xc<*!ydk8pLH_4|J@$h%r5OUM%G2vL{>#sM3zPtL}o|E`Gj$# z@Q?aQV)&h}1Rm+SpT2pGW6Tr}Ms{;oaa&{qpHZ)dp?)R9?}L>^7Bc)!SYBi%zv+63 z^I83-@uy6QOyWo4shg~${W8u+Il@}|q@Pdfrzf(5xl4_Q;dj8ASZ#mRGKCeq!n~L8 zba6f}6=(4<#(5c2xW4>EUM`+))*r{Xk7*g@>DFtAlh(PPtL(QPUgtV!Wu^4ykZ;`^ zLd;r*dX-%6Jj%F(ehPV&burSg5 z@GHij#3wm>gLagkV2o%f$u!zVpcA z0ou)Czxz_0cQBO??%+DU-sjmI<8{)9c_#<>5_|XxJNYWxcrBaxIvd!^THeh{-mkqv zKEO-7oEgkv z3P1AvA&K{zui5K7E{FN%K4HGOcbIR6*Qs6nZnzE@o>w>VGS+ir*6?KOR>{Z0KIG-v zD`fbc#5^uiE}P-^C$bn`_r`fOV+_yJ6FBO;r>}K=aGceW^Syp)cm5;dcI5_nFZ&pm z-V@TzOJ|7L#_&6m&HSnRT|J*=6~AhHfq3sF%fC#BrX@RG!0R9_l(u;P*J$=z7rp7;paAdh(v|_`_tj^A_n%Jl%QM^KR*t z+?d7ON&P%FE0@6*rm|gr0ypKv)t*Okn9plxfZsBI-CVN7;~Y0#?mXFMe(L!kt9YjK zE9Xw`S4F&BoX_i-!%Ni9C7=dDUpd$`pkDd-*ny+yj{7`$U)v`fAn#O zuuno9_DP7tK4JKs(q?u#zk0TKK3K_HSjnf_fzYjzV5a1Ld1PmB3 zLeNl7l9P|N`6`4GAV6qCiBh4ZX-Oe%(-I(H!3qTeR;^gLQF=983RLMU0RvQt5Hvv4 zs!jox@I5VS~uhy@FMfB$o4HJfcY)|Rj9J$asg)>^Y?&z@N`d+)RM%vu9&LLX7+ zQfM}GB{Us6j(T~}|Hru4+{Sk*74qh#EHtAK+=g?aedLGx!fogw&H0V6iJ?ISjY2-Tcq0b@cJZL{u=l4KA%DCu+ z9zlEBp;u>d8uS3hNh|af%9#l*;Qdq!y@cyjL2rPTKwqYu9Oy$_E(7{|o?9u<#awsu zwfz1ieHc28b`3&T(tv*GEUw!NeFVA=`Z=z*8rn=fRzSnhCD8MrjnHE09l9HG6;$iL zOQ8|^Ef=~wG|SO+=uG4ksD2-eUPFH)Z-kaZ`=S3qdN1^5#$%VG9nfa2K(txNq8^zop+=p>5Qw5!(GR+6(Q4)=c!s(7zku8Px9M>jdT(a`}%dmZg`v;%rJ<*b0d zOg)!Dzrna{g?T(o=5$Mp~KW~5IT$F{m{>Ie|JF}NnZ>7829TcN0&n* z^m_|*n07Qm8=*6xDz6551of?i9?pG~3%!c-v!MG^|1k8g(#L112lx93^kS~N5qdpz z0QwT;tcNb5Jw4C|8CPAjL2sb`t|`c*;~Qho{a9O|C~J%RRSKyQbpLg!J>kwKTPBOGgGU*ZMBgmQ1$(q;I~Q_Z#{f zg%*;&33?ytLz1Bzpns#jeb6V+ORv%ypz9o6>*y+Xd<9hTv6ng80v+MJ2Iz~>I_T5r zy9WAM+Eoe7r`=KLYVNZf=!Kl00d0b&K<7h8FJpXCk6~!#k<<^mgnDm)?t}ijp+`ek zL)X&}E1-SQrO+$6ZVUAD$W74Alrt0hQ`%b#Jr!Cdnf6AZS0iUbcZX(3COs88o&MU= z$~-3JjX=LnIRnr)pgquApq)x5Upw?QL8ei}M@3BPm5zX_^!A45>ZQQQEP-&hZHU(QA&Cn|3 zTIl2CtAzGJGoTMcH($)}23@9-iM&==xYG_0OMpm^b62l=)-z0KwD97 z_V@M+moeTuk@x4itD(AF9l%V_78#k|L$BhZyxXApV@ z`TL=#(as*|)g14H{t?;%Z6epfPCHVew^N@K=+jhew3+u1>BG=#p&OvzLf<`(u7RFHyV{|zLYG26N4{q0 z)zEtARnRKv*Ep{P`to7iU(hbvp9B3Hawha*j)$RVQBDfa_v-Yvg|D1oYDDXqQW01AU3(?NG&ITLx{TKU$&JQ*IM<7ABDg(NiXLzVrn>A38Fh z_bYS*^cv3Vg+2}Kg3jmscIf9he>wC9=u+r^GwxfU=R%vHm(%}^(7EVuhGgnp2R)nP zHKN=Pl~C;$5`{kY2ICX@I5Z5M&h=8DeaNFty#F*#p`FMB(EB;Q9(ohyuY<0Hu7QxJUE}Q!q`sKs8pGDDQ6uN}txsGN*zryi!XfgUp zg?<$}dLiwH4nb#7uMN;{Xde{f^+0D*|1Rh=w0{kBE_9V-^srnqbO}`XnxIRw&>K`Y zb`3Pb@9t9Q2*)GPW{zh;YZ));(5>ht1)4_s$OU|dfDS?LXWR@x51_s4p%e;ZKs7?qsHKjCe zI$7UC1nEfm56Fr`-|;1{zKq}TqkQijBtdL4}}*@ zJ`7&Sa&qCJTualj8E4*Ti-|A$oSBk;OL&&dy9w%>x*B;gbPTmSH z*mzQ0IbrVHxV{umPMGIp#X%FAoUHhB!aPn;|0oWf(B$L~zzZ!5VD;}F@Pf&9?}~;% z8VKIM=aQBMbbJ<&u0AKlNw(Np8{1MtGff%11dM32pj<^Mji zkniM&;DzY11+n8j@WN!E^B-|A`3J~?$v=b_OzwpjOnwwzF!?cf;RF(-ug4vZhlW6( z!wF9Q5mYew$MAy5KYgUNqE7EFEtUNHGZ_?N%}pz~jLSm@+Ys9^G6;RTa%3AJ@B2u=n2)hws|^?EUus()&r6?ArSWc){fVffr2vBfMbp zMtH&Gr{M*YpMe+bdHO88aJ;+TFuZVrlb?eZO#TzRVDg{g1(P?y3n#hrpNAJr{tLWd z@(b{S$uGjc1nhbGGQ4oQD{mBDF!`_Wg2{Tf3nuIRE|~l(yl|!~?=^VAcJeRbgHfk7v4fEV-|F$7{ST3)vH zzLme6aKAgQbsj>tlRp42Jm%y*;DstDe-K{yg_Bd^_XIz4ver=vM>$#kL&6hI*7_;o zX(tEm(}OrB*zW=Pa|lnm{D;B|dakH_iccrp?&K_Zq2I|z!7FCXubq4}yyo;Cck(CT zg_oSH^)A8;Cu?1VVD|0_@Is|KUJWli=;WF3v%q(q+yF1!=Hxl>!grjkb!ozSCtnRO z{K(16;Dw(!`C52Ec8uD89lY>kC(FM~c+|;SzY$x*;pE$qg>#(zsPclRoSgMh^aoyc z^5O8pU!8n3yzmDne;i(T(aDPUBMdoN>lTG?Ie9j`aHo?MM^^Zzljp(RDkPu1Rd&p26e3s)kPjXDhXL4g{jhNlf9q2}|=d0F<2p75hT2CUhIQcPn z;bJFi|3ab3$xp!x_I&&Rx)bbur*&%rSz`71FuY**m;45WPq^dOU-rDQ{<8kC{<8kG z{<8iHapR`|R1mw~9`JhtE)kQ{;Ew{iPCgp`6mXK0PlXrk_rmG$Q-Qs2ro#)Dxcv3- z=YUox&w@W6-0I{@;8%kCo!kz8HyCj8J@5|!8^8U0YwZDDK>fEGx)QWGxdqw)>YZE- zEdjYsPJe#Ca3*}%$sTm1k@+p4^43GUL8p^fL6?K2PHu%Zf*DS(g;s*7le3^e=J3f1*;oSfn0)WGr4+41_?@!^0x;AAVWH*mbm$wRLE z4kufE+nj9mZE~`;uRf5!GLSzzkU!JOVW`^YIeFx~xI6?^a=(+ioxIk`tDJ24mpj?= zw+8YzJGsHp8IIOETIFce(OgHf91T0_L1Xn*<$QmmXN?PQ0yQ8yoqfrmy^GRgZ5UrM zd3xQP)|S&RZfU-_rE1P4m6e|VSk1fz7fzntGH>Am4)7N|cH+Xig;Te4PR*=~7tC(l z{yAq(Jt=k|Sk`H?=Fe%^PFbfeyky>-?W9jzc=3Ym<)1odLBl*(<={0=o^{ErX|os4 zYi@aqx-GbH!NN=9jSZfww(6{4`sv4?IIU$?i>es=tGjtzxDw=oY>);#FnAMtr-Dwf z26TXxpcTvn*&qcBtz-{4&<}b+H&_E!0?XS9KLczpudZ3cu?*1v753W$OM$M}4At>^ zXf23>T#x}gFnlBXrh-1u1v}KnX|(TfV^F%V0fN2iAgBz{*zp+K{E2251iOK<{$e0@i?)U?!*sYF{n1 z1Z06tpC3z?PCJoRMhnzmMlJI8%E%$zUq&`^D(JZ(-iK|_CJ=FS^YzpL=o(9*l|a|Y zf)2OQPOt)4-CN+RK`z+d_0vcnypH_9UypUjsz)s`L{^^7pNprd43(#A4w1GVtOFgO4K#r|Py(_+7;ITeT|ozE15Ka~ zl!9!K21Y;2H~>9h6<7vlfKreK)?Z0FXa#j30#d=y6`TXwK?|4xYJl$VQfM|v0sTvO z_JC&40O~+B(0!<9gYsq}Z}|-E1Is}Js0Y$Z6gql2Z3UJtJ*`EydwCVSuDb%d6tsdm zkO@XEqYa=3tQAPJYb(E9dnU)WckM9giLRZ(@hz=f19X9vU@1^P)fzD5d>U@9R;l&&SD?kIVysAUJWUy&b{QNJ{M#GU0dTm1%9}hxi)84Un+9XJjNQR2iagtW4z3DQ2pLm1C?%8Kvl;^Xcfo; zTh1Tr+Y$I7umPw}z0fY8I(0zpx^3`k#}cT&4Nb__h7wc^`Hi*E@9|U1LuJjpmeL7>eWD69k6y(!K)qW2kFh~Uqbp@ z)j!O!!P#TwS{u4ZYX@pW6I5-ehpG*wP_-cf9X^ligB74fxQgpLxz^EAM{^y`fco2^ zG1&Itelgo&WEO1(8$b_O4VHoiPz7>97Dxr7Gsz1!fF7_GtO6@QJy4ncvPwzYURk-M z?`U~ePkq195A%Z9-ybXVea0{AdyZe$_Z+{h?>Cb7J?7s3eBsVNRp|SSU)J{=$#>k+ zo7K5^Q-!|gyh(Q7L%#mPosXST{9J{;&**qp-z5)yTyTX;bISUf3{a@ju!)wbT5T9yTm>mp-HewOehCwGUqH zSWjJ5U)53dQeAX?|20pX+d5_8g1A8yNG~~VUdvk@JN~4qZHHpb8JbSt{*rqMWPqU$ z92DN*OKT);l(b&b*1`9HK9IV{LE$c6TIR19v!tyitr@-*ECmtL{CYD!hcZe)HK+$l zZ-usjcCf~!4MI18Eg+5ZlrI9U1hrtMOIrzD4Z1+DOHMRQbVt)1 zO>uO~Q}N@Yj&5>vqoae4Zh)#T>m8l(0MD=p{e-M|0oof@@wpVcHft;Wc{krOGdYK} z(R)b;D-J^6YdO9r&&xlsrj~I{D7ly~uaPm+<{s$vI!++`kjI;bFd4 za-L$QX#WH4-K%{D6;mvQa@*DuqYJb>wqJPkC;NprgN?`~AJ+BoH$!fp!tb6sWX4}V z$CeA^A9w)&_)YtTS0a0$>1X?eN66oG7T-sa`|)#0D`YGb&$oH1_&vb&d#Q(~>rdl-G?npp2G6?Fcy=F0zfVRk;aznM z{6~1l_C#Ai{qJ#LAJxII0NB^#)Zr?@M zpsS01re5;9lh2?0_j*#OLx^!rnTms;7&`;|GInWG=3gl5cFNv^F@7~=P!H`%uQ<)^ zb?C8ze7|8FFebE@MGAUMSp@$>j^D$%l&2Vgil?Z(pA>ghF<09wxgPCNd{xDc)BeNS zD?;(tIv6{dALibs+(z`Sy&|>0wPK$Sqo-=Hmh1MRrxDtwJ*fwQV#z69oMOc(E}Ztf z*ZvXm%hsBOaBd1*G|e-tfmge*mC-I8MK>gH=!f{vv|tFDf8+UUp%MovgX)# zs^GEXW?l3T>C>8Wy2C}Jg5REfGOZapg3rX zEp{?E1=N63!D--hPz%lgQ^7P)2hIf3!C9ald=e<0=-EJV0Tm-v@$(cHbtaev&I7YS z1DFHmg7ZNmmm*1__#Su=bc64Mhrq+22Rs6P0DcI1!K2_Y@HprLKLS4n zKLP8(Pr=W?&p|)<1^6ZS71#it08fIazySC)_zn0i7zDoqzX$&fhQJ@d|A0S&jo@kU z40sj{gXh4Xz@Nb;@I3encma%n7r{&5WiSf<3SI$!1DnCC;59(k#A1cN4t)clMz8XD z{L?@Qh=6R63DQ9rq=8hB0z9x~6MnT|1Z0lKuMD~o41odA57vV|&J@tSDaw^y|-a`gs@3ncZ^AoXaXJ zx08NS%bfWf2%Ix*&g_d9)3)ka3mTf7leWK1f=MlCn6{71_vNbMML++0 z&waQ5_P*!N{=xJKFOGZhldG?)xbcZ|GVf~t)&~bdXJ5VlL)YYG|L}^mGh06W(Be1l z`Tf#6_ZYmT<;7bbd1T%#8?Qa_p%)(e&qwxKe`{Ly!rbSM`PQMBn=O2A^`Zaq!a8$|RA0EGDe(tFU z>{BwW?xMZsJh;~dmyG|>4?cHI?&2G79e((~PMLein&!`+@`KyXXnOqM;TwOEy~kId zyryv9(PlQ%Iq;Fy2d6d_A69c|^-=eAcSgUHmO0_%Zw;Jw`j1zg{qn-`jeoi7ODzj$ z-`<+u+P5rqx5t;iw)LSEe{QN>{kPj*edwjuMUVM)CBMbv7hl%gGOcCtoLO;Y`;VWn zVD_7i`j1u3dFx{**Vg+=jK5D)q%#l*kvuaMIDgLu-1Uq7Z39W&kd^&{Q*^^=C|*H4X;v!EXV_V-R+ zPx{CL(np93NgrzSl|T)ZtoDCavir08NK`v*>@0;>dnGGf%8{({VtUxJb*tLHr%O}E zOV)9v>3WiN{$60n$o40ONUdfqYDd(Gh4@K^AveFI%TmQCo ztJ<&jsr{;m>0`U&k@{MDl-=LH*f^kmIg#- zovi$7zrTHlAp6_*VdN~Ja|P+6((!7e^s)8tas8 zztn@dKujS%8o&{nGgX)jNTjSiAJ}q6>=G1L^H(6A56_t*pQ(Z?? z`PIfx0LeNoeV^iF=|K9nbCqBE{yb1#l`DPk%>l{Mf%Gk2=PJMSeUy{cZ_@YPPQS`8 zeecZ)%CBon-_oDT*7?%+-c&%cbRvEKf#W(?b(X&W>SUEKeXBo|SF-f&*T3{Vi|C##fTeZJZisWwSOGYIg-`>4>?)=p!Vw=9oIF~{$rf1 zGS&W5ovbqT{`-QHv!I^^_V>@)r+lh|t}CeiDpS`H^r!Y%nvSb((xuWQt3B#h)mMLN zkNQcnj;lRtzqMEGvG(hlYLCXGwO8#q8t59T!yDivCuc!-wEtAL{-i^dZEaMU;uH0! zj``caAF{Pq<6HHYEZu2*tN$dcTTBK4Ru7FIUE5^ohMzyN-!nQ- z`qDA0r}Q<f4_}#@^L4fm_KdK`SY=5D=Xh@ zd&Z6tU~B%YW*vHy{TV;+viavNY^s_!`yXR-#g5m`YH69Xc!6d3hwC0s46(Xd=UMER ze>m>5ko?Euabu<}Y?{|_{KBS%u_$Ef6YCf0*ZRrdk0i&&sPt_8Z+Vqpde@&(@m8+R z5ie@z2W3mQGReBj=c|nO>Mt4F>%EKZr9Mx__u77ZuN9{g&yzS@7T-(b^!*s$D@rV~ zMDe||P`0*u!J7{dCSBFV2EQw=w*nwg4JM{SZrai=6Jqs6WgtyICG=reBTCX zVEII1l@f!kzKHMI#AK@jVUSIHxz2pz_z~Z59qm~04dS?w&+42>e8VkiW3r_;5?^kR zy0y^{sh||TUZAcksAtWov;iy!OR48dV!2gAN1*-GZ!M@J_S`U73YMw9#Gu>n1Loj_2+}yXb%MmE4!2%c(d+9y2Z{TH$pdW+B zYZ=o@>vr|=$HL2@t?L-0!(SmLAnhD@h;dE5l)v#6V*Paxm+-2wGS>`*!>KEX$wy2@ zZL3OvaDmKPon*;#w-d;y{WD7SqyVTq@1tUMj|)nhNR+Ns{a*V)8cz724>2lczj94`i#Z%B)+F&bLxGpIGl>@s<@wew<$KL z;(J=0NyPwFd{4y(RoqO)v(&w(xSo2K`=9BG?WuRV;%F+Sreb&6?;*txRm@Yx)>Pb6 z#oJV@O~uyq$J$g3OvOaiJJ@1wD*mTpq$&ocVvs84rp4b>+*HNORIE(JFID_h#ZgrZ zRmD$L996|oRs2!KQB@38#Zt96s*0hiSgICBRq;0!OI2}H6;D;MR24^6@l+LKR545y zPgQYMEvBksbt+D$VvoL`?|;Qf9pzp5?;P7#_s)A2+xNZhd%M2>{?~6h-TxiDA9erh zd7v22dba5P*E7f;A6I4QS!2&D-SbM*{jcYY#m$Yy-sN09+w?qB9^L<H58|-?jR^ulnhC ztkpC2Th|Ht9WHyyj_LQXe(wf^l7}PxRYewu^qh%dU_eAv;BOOb5U5 zWt+&Jku4*eM)rqn8QCVXRb;cszLK3K8%y?;>@3+>vQK1Z$;OhcWp@C?{W^>6#l1*f`iRtXUp!WNJXbJZu_f#fh!D9QjerfFe zU(Y>ivHcZad>#6GJ7W7s>4UZCdjwt2!2Z^EpV}JYnWMKhx@sf(4Wsk!>v>+BKtJ$2 zUWJXd9$i-wCq9RK1L(4vvDSnAy#%D7@ARAFapbf4t(Quk9LBO@`>&@Q#rDsk{)+9t zmYDKGeCudUr;ql={^xzu$=(A4-{!gf0`?)#qPk!6y#aa26SVth@%Z&U=&Jew?id;5?Gdz=TM4$XFZaSE8oWZjKnu2V<|8x1Rp!oj!&7DPi72jX)fBgp5 zegHGksp9tQ{U71{R?hF|UK~aDYv|t^`Y+>B?p5raPOysi`fBvP30;;_UM2QMm~v|1 zYvG$IXDxCa=P%>B=}N;+X}vnWr-1efsOCK^d!-3ff*R_Z3&J2%bqCr5V3>MkbN*7! zZ{vLJOVGsc-d@lTQW-Z9Pzg4I^l{_|r9gTd6Bq^?!4Mb(1E3$Q2YsLy z^nh-#4s?M|uokQa9bgq`2P?q}upG33Wnd|21udW%G=WCY0A_+2pdQqL8c+?YKqV*@ zPrXaTL!;0L$OSnd3uJ-}kPZfE>j2mQ+Gz88)#t(ME~PH5pcyoRdQc0hK{m(*QBVmy zu$lfJ0mEPj41j*n2YSFduokQatH4UI94rG%KnrLB4WJ&>f@)9+Np^`JALo5k~22OR;!U1%n98Lw~g#_Kza@%kQPylg7twa19@1Aiamwa<*DYY!gd_3hnw?fGN; zdfvgt>)VI%%chQ{_dGI|uHOrmCxd6b@pbHXV|>b@F<#FkOYh+wZ@lb9<7M|5uWv}k zYkwi*^?SznWj`I`wU>~kYfmBLwYQM*tvu6>*M39BYu_Q`^;_0>?L}m~_9QZ1dlMP2 zJ&KIiUPZ=h&m!Yz@a`~P`xqIoJ&BB$jcvU4IWk`R9T~rg-yX(CCyt$~{g5nO`yv^y z{gI4c!tW>JwP%v?`c7xO_NFmjdny^Py_JmD9!tjSJDc&^cgc9|zhu1jd@^49J{hn5 znT*#yO~z}#CgbZ~9LuloIhNjW@))oEn=D=XI2o_~oQ&7LPR470C*!rxlkxgpW4!i$ zGG2Q?8Lw{!#_QXN@!BKGcpO|@+IPx$ z?LTGwFu!|^*Pc|y>l?1|`X*|;_N+2q-$9JmK32x-cb@Uu+sb(DC1m{Q z|CRCD2g`Wvhh@C>D>8lro56T}3o?Gt_iS0=duI0Y?aK13{mU4y@6?vA->$~%Tf6c4 z{$srMCNo~&bd1;UX5(XfO1bB`zMERQ_ER!m-?EL@9!$pTJH7Gx_HTUa5605M^F_vM?#%dR-~FZeB1_-wt7rCZW9iyw&GKlz%6RP+XT0`) zGhXvb+jwkRs!9@b-uQw z`R)bn?Pz&qKO3+4I^zd@&r;2kS-R%UjBn;!gz+nV`Pcd0eHp%Ii@q0G{z3n5Mc;2) z&F5Jj%{3T5uw;zaHz7;czLv&ouF?46pO59&JfNkw`QA19MrP?HzVbB>V(B%$-=dm} zwDeM6Uu(}kOV`|p@tTt{UUMVHYu?y+%}*M?+4noV!}l9WbC8y&-FIJT&5Wh5@#WY2 ztfgz8K;yONq4Am@GG22X#%qsHT)ZB{Y(fo<=+H=x)eMd6B&Nr@B`^I*& z?-`{zUCXb14UJ#wd){iFO-t9DuJN(?Dt8~P_l<{I|DEhuzrE_W zWBvAR(vFRfjienL*F(~djqBl^uw(tUtKX(g+Oct+|2x^Se%tj;HtqY^j`iEFew+5K z>{!3;>bGg%O*__aulns+zkQpuW8?SNdu7M^ZCAfd>$hF~HtqYtj`iEFew+6FV8{Ax zSHDgBez0TxwyWQ!P1>>XdoXFoj@hNj`PlfF_*ds+lXh%eZ%I2gu7~7&Y+Mh?`PjG~ z-s$<+q#YaA^Z$kU*rXjB@9TGZJ~nB`#`XMuo{vr1vGG27yXIq)c5Gaq$@$o^Ip5^E z;`o^Ozp$=2X~)L}D^vcl$GITG@{hIr@{cuM{;|f(KlT<^ zru<_qUGuS4zWigYeEG*(8?~<3@@rkO<&l5vUak)EkF~bTjvaRC@{hIa%0Jf1k$NaT7Io7w)$vYv88KW@nok5`N!I|LdSH%P;@fPFH`eE4Fm4 zE4KWyV{QCsU9shnf2>`r4qpz-FaKEMxwO1 z{;`W)edHf&{UZNZ%On3-M=zx-p3mw)WGJY37_DgRiz@8lnA^^t$9^^yEz z?OgfC+8B_3td%4GSWB0Gtfk98)_D2H8ZZA?)2Hm%*W5j#b;VX^`Nvv)Ftu~tv{$66f}f70^HKi2Zd zKlTrGF@YzK|Vj>GF@Ya^xRtdE_5!b(4Rr<&l4^@$!!~eab)9(lsAz<;XwQ z>a2CerZ@S=TDtsWP0#X=wYtea*2kIkE{>{}{{;_th{9}!mf2{HHkF_$juGpSAT32j&w655A`Nvv4wXWFu zSL=$c9IY!Jbh?rqYv;;8_B59+|5z(W{;}2;`NtYB|5)SYA8X~yKi10Be5}<$>xzxn zx?($5>xxZ3@{hH1`|xYEbos|xU&ue!>L&kK%P;@f6|VmBkG1+p}c=^X#nOawD>GF>) zb9I(~tfk98w!zg${;{UdY<|-|;quEr*7D0g)^sTUSWB0GtkqL?tnpe`Z2Hlx!*z zT32l8T32lOJAA*9#;4|E z?Od%Zwme!_Z0Yiko$vZf{;^i3{9`S@{A2AsEB{!_BmY=yul!@}y7G^;bLAgvZIOSh zU03UhtsJc@w)FQ+Yx&1Y>c2CYk6eHS{-D^TDt65`#m8$ z*2m#>Jsm8~1LI9cy)y9cz3dJJ#~ZjIQzYv%@ST02)Zt<@)yO>6lR*|cUG$fmXXE|E=Zb(2l|>@L`}RyWzS z#>=KPK4{Zgx@=l&`;OYQ_Io{%O>5=LrnUP$Xw#a0f;O!^2NKz|R=#Xn%P*VO_@GT| z`FGW(wdc8PS}QYX(^{VfZCcAOo7Q;Qw5Erhv1#qxcgm)5=ss!eM;3EH$7yI|AW7*1r<+I17zv{nb%v^G}X%BD@)v+>_WDT~H_+waW0to2LM zo{j5M^Ro8*N!qhpQ-tPa?YD8#hK-L6&CA;TlC)vtI@!^AS+l>AHf&sPnwPcv zD`~^Vb@F!EuqrKR$C@pjTwfg5b#i@iTo1|h#c@3(*B8h2kX&CJ*F$oBaa<3{^~G^L zB-agJHOpZTS3y@vMB_S{+h@{P6m1nu*Ew#y&=@K_%0 ziEY=?9^6)@-#1qKsarYPE8EuEY~b72x2;UpPX68TT03m{HIHg_)1L0NE|2yPw>;Xf z{BQ9g?Pex6Rz_03QSBKTK*ZMSvd&hXcuU|TCv3oh_ z(`VPs!EfRgSAYMS$E?+3y6yAbQ!{GE?x|k-(R7lAo~r6Sns>m=-y+xT6fQw&KvM?GkwZe&aNe2xxYAl z%0JHPFaJ2pBY!s=Gj%)@OwYOW(QjRSv`%oy=`h{b*ENiR54$`$K3|(s-!m}m>+2LB zzXqQj8_)iAd|}eyc_dcYi9T3 zTHkuo44?k>o!+jS>eJ_vC&tp3k;m#V%9z>TmEZ2`uaPsx{HlWA>+Ri|$dBq4*B1Y8 zo?twXi7ro~cp!Ey#RExo`SqK}^d>(ld$uSZh>aP=1F<>;;@)@FM@tl)|Q|jm6eky9*F5D zQ9KZ-9X%6#pE@j$HpT6bpW2IGO)Z?+wc2V!~-`cc{M z1jPffYX#$hSbY=^#L7`T5c8c^JP=FQ?@Y_D-;>b2Si0hY*u5Y>Dl0SSM`dkKaV%JR+55(@fpdXc`C-S4R zwglsWSUrRBK(?i0TUp%{55)LH@j&ca`d)A4C?1IMK|d-R=Wk`p1ml6&y`Xp?cCO-q z*n3d%KmqIe+oJ3JT<#C|i# zcgo(CiU(r#Z00w)jWPL7S$fcS%Ic8Fcgo5Mt~;}~$al)hm+zGI#g4`UvHQqxX9azy z>|N})trQO=)7^veo$7FWFdm52GZ+tKTf2PEQu$6deceJcWUr0Y+frz@j$kfgU)SS`E6dmJsXS%Vz!3tSu0cV zK&;M+2V!-Q@067v^qsQtsp{#pXM0TuQH<2BCrO%GF{K2@Emae##)~`Ei$J)7yYiV^>TuU1_iR@TAH<2A{ zIuF{hHXo44j8Fl>tDsSw7U7{`GR)r4EJ0O+OamychruxdqH-r)j@GBjhFwFm6^znwLFPp zfY`PCbBBs+`BrwU>A^o=7_?)pOxdwBcfpRe@w20LtmT&-yTJ8DB0JXfskoN*KH60~ z*51YMlpSkre7EgbyB`$;#KwT)TH3oWXvbO}#Q-s0aV>332JKiIGr_o)mL9ZY%`Vy* zJJ#wDi~(ZT+EF{!?)SITjy2mycC6Ji(YiCUA%kBf_pI5``Pf5U{++R7ZLBD+rQMUVV=Y~4)2u$SW9>JA z=3`CoJ7dS1{WI*lZX!F@?i0<&T0LdQ8n5|S({my_*79qA2Kx>joR78VX>flAYrE!S zt$&02GuUs$U|dV<_n;kX-wS2OTKyFV#PqNESUY!T>{x5B>{#Q2^Rad<*|B!5MDwv$ zj^<;n{<32&JvbkGo4q@kkG1k;#~NSfTa)s3+OgIbiRNRi?ZNq28v}{#SQ`VuxR!RV z>{u&5xIcrXYd$vpU$kTG++Z9KyGIhu$C@n^v}3K0WXD?hifd_OGLao?dQehHN_#eHe+GNT%8s@4M0Tv*Q;FhQKIy(0@2DMXeWCeS zt7jrR*7C@X-R$mt&Bt20f6b=$XRvhnPnoVXA8Y-!qw}$*Pua0nj@EdZ4tK_mHGRsC zwK5e4#CXleTK%;@!?twa9PEzH$C{pZbUxOum1sWJ@+hvQwMBNUr6-z?wfZEoW7~JZ zj=jx|jYRXYR!_~xTK~$9HJu0dXR!QQqiV6R-;Z{z^@ZkRtu5N0!RjBhW9>c)?$2Q7 zCbDDqc6yc_n`Q4}*20@UwLgPhH#i?__h4{;2FsslKGxbOJJ#M!J8H+?;hy=4>{zRV z*3#NNuK8H2vwvM}B0JW`xz^s=nACi%l`sD(D>FDBYi)cx?N}>Q^RZTk4&QGi&BxmN zNOr7^fkgW=Se~FAYwb#8$J)8U`BMGJ|%k)gd?^ zYvpKv2D=}(Hy>;D*wy*isMFof*s=E8J<+;iYmeq*t&fBIGg!LpSgTLaj)rTBw655mN3vt>TEY2PYkSa+wQ(PukF{sL=3}jYWyjjRw=?@Q*gU+}729vq zotcldI`8Ow?6$VBhRWKaeFm+ZMDwwx&)~XZvx{WMTKPf$H5)&{b;Z^fK|9v2rFF$t z<~ubXYvX)p>{vTDXvf+&;CI80wSElF$C?g<>xyl>1?^btqeOPBwRcDN8MHd==zOfT zB{(13?(Y2^wPS5O2j^q$+4z39V{Lul&e*Z0^F(%R*IU@JrvF6qv35TuvSTgJ`^Ap6 zcI}KEYi$YIv8Lys9lI_4eQ3wp^LA(EW4GP=teG@@YCiU>yI{xKGbfQ9Yxi={jw z$68-#U9puZJJ#B$b;VYP9kpY##*Yi%@9xh`_#52sgADi&x%71SZ#X^-|GMMT;J@bh zRQNx+b5nf02Y;>0Q=d03obKKkwLZSu$5;CJ5+5J&@i{&|%g1N<_^^-Pe9TxKM&Z+4 zeMWq|e=mkjq}$%8!#;i^{AupoAs;^o-{#T>eEbIZRqmOa;;YY=$e12Ref%aLztP7J z`uGh#e!Y+H_3_<4z7zgRr}H)NQykv`zsm9L@S7dK0{+LYEp5I$OW|*Jd0OGW;rM3w z4#zk8_!;oFhjAVJJeQ{izSQwm@K%RX_?ujM6yAK6a^Y`t>Dll*1}OK6o34z3{iVb9>;2UH!Y^f93df@XtHG3*PG7 z37_H4T?_w!%hN`m+Vg5D{Gdy3g||77W*^@O|3z2MOnB?xdU$(p)xw+3t9^VW{9;%B zYU+^c%ISc&_$jM=^=yZ?{#ptDLwD{9_+G~^hyRx2+u*e&Q;w=1U+-kwVh@b|d%neaCD zX22idu2m0jF)!-itqwC7t5!}OyglDM>b7m%at|6`Lms=oMttd2q}x3;>`N~t-E=A9ra?}2__dY_N)@$u_?d=2dy=h|ECYhx9>J#Q=F?S3zXw>p==TRo%j zOI$e-AD;_vddPtfyF6=sby)4=SNZsrK7P56U*_YN`1lqd-{j*Pe0=}Ba5&?fgTfc@ zHhb%HTld)dlifZRtI=v-`j<$5ez(J8=`FtW$4Q^Fdpy0`m;N;AXY3v?C;j>TV&(sA zx4f;N+WL{L!@Ik;sLQYP>02|m{(1Kw#L8da7_X1gr;`4ulz93oU;1>)nUNANr&Z~5 z4+_7u`)}W<+4|czj!StdR!*(Uf7fnDZLQyWB-gz&mY(HHFCl&7?(we&qvyx3tNdqe zJ!wW3*a;}P%)As`Hy5OL2boWJDw{Bgu^@-i1vGk$5jBrkVMtCzc46Ouh zpdK^>uOK5l3|$BML4^G!BUvCD&I1MQz2mj{lE)1l+zbhs)`M^1>-%~Rr3r=F&N)p0rXq&N+q z9H$wl#A#MdoaUSwr;*d*wB+KDo z$ex~eYij(OlZlpkE7#SvC-aqd9JmI}odiAxJ_N|_Q9Vx|adrJT#|g1dJ_@(0qep@d z0t$|uL8H9AK^oWx918XW>MSbaQ6cXDK;>drp>f_pfJ%D@gG0cF0S$?rdl>X1APXE0 zXr%X1K!am7)%A}8D8V}#d>niNn*Z>pb|w1zXU7wr`*t7G_z}kbMk>a5aY6abN8b%e^Gk{!B;OpxO^wBSCC;eU zTWfz+`D-Yiq5K#$7cF1MjeiJ-wXRjZA6h4oBOf=`oy+G>>*f>_TmILYo0VURd>Z7> zsQ43`)`!E2hoZSct;y4R4XvM)|DXK+v^H`xkRPAcJ<4xS>m0SdQR^DDf0@=OYHgx? z=oAA+YYkT-Yu%vM3(DV3YXardrZs?C<0l_A`H#q#PjmC~uU4!%%}2^lO>6lSQ$?{< z6hlR^Q{-!wMw}(B!&BT7#XHfuI>k59Iy%KL874-WVw1>!OY7RShAp;UjaVUClcv}p z@}-gwmHZeK+gLG^6?^yH=-(Z=hT=C??=vpE!Qj7ulK5E@=ENvJ5$sbFn%YQFejMnRo5L;XE zPqn5XcwBNdzC!+Vt(!M_BW=;zKE)7GY&1)Mzfj%(IrM)wz80#}2zB28R&f8%oW`6S z_kQ0!3FT;8v`H>EXTh$i*;S~#2sJ9Gp7!OL3Wh5H9TXM@C;ms zu2y^lKS=Uf``&9>OaG^FFJ;g+t4lY2W$ipOD^KTng>IGu9ovF_EYApfFn-`=#tr&3 zxe*_)Qgo8#&a-r_ahD%}eAU9-|GgSJ@JnA}tnj|*=lDv-lI720e96Z_Yl#&bF^BQB zj`20jyQhKiwTb7+z(YJ6$Tx$svKXgrl-EJN49cur$v7cjE#t6+ahOXx!&}4QR>og1 zV{wqaZ{+@OW2~3Lr$XzYOGvAy?lq_4dq$f#K$nA+ClXJeF;NXh;JX!SPJby9)RCi%s44w+<+9Yneos~|93-YJV4$Xlked7>K(i*@4n!B@1MQ%^uE)2 z!9?%6ZST3jJ5KK?y`O^bBfWd{z6ri(bk8UTrD9#npSh25s{2m+B5CcZ?nCX5qqSM` zPt-b3-IKb{6l+I5Hj1aKy+w52Y2CEq4k#X|;xsGPgJQ@k4v6CYYR$3yspSJK|6r|o zRBRRb{&sMmC{Co}1!=vi;w@|K>{{*(#VS*5V);QUma^7Er|`^N&%Gf(SNXroe@yVI_Tpa+|@;Xe%B~Y zl6;F5=ePyP4^O_aT8nG%!+-WJ+|hSm@SV54_nqEx!FQYgou>EMo8DvhzCYe!{&&ar z-Whsd=sjWY2mgB^(eq!={eQb>J>PNaKn3t`FfDUmxGV<)gYHI2IIYBy@;3R<$DEQ)GPFgykf7!o8V3KCV8b^nOE*r zc$MB{?>O&xugW{Yo8q15RrB5CWbYK_K2Bx+<#eysJHwmmP4nu!Grj5FSzf*ON$*qM z+1?EA)80AWx!z1~mUo^v+iURVcyqn;y+&`IcY$}I*W}Ij7I+K2W^a+V*t^JU@h*qH^S5c#n8L@P6p^ zdXIXKd5?R2-jBQ=dq45kdq4Gl=Kb93_kQ91()*RS!F$4c(tFAq@P6(6#`~=|=>5+7 zz4zbVkoO1gf4o0>8|?=TUuK^6p7EabhP~&!KY4%lHhIr`e{sDz+k3$q@m};^@?Q2v zy}x>|cz^RY$9nTs?=|o5-WKn5?+tG&pLar`-9o#EQbHdH?GgH5C^fWaXs?hox_2lo zv`=W?(0-wCXk2Lj&;g)2w%^aEJyZGMJqJHr z<^N9G!uR8I=Pj7KkP}&~68ls6JM>4%y<|Kha%^O3h={#E(y`FG^^5qURmCZyesln<$XVI zFz@xe%>1(a8TnV_-=5!_KbZe|{(%KK1@+PMqqjuwkM>0mD?FyKxbUlmYYKl`cvR7e zMRyc+75}byZb@s&>XOcq`%3;)l0M;(3AatSYeMIQ!zV79c*n&1CjNHfvlELZeQwg- zlYTjA_tFDOGfJ;2eX;D7vdQIB$`_S)l&>ydQ@*ymv%IVPiSp6%Sryk-v{yV?@wDn3 z@pv|>&L4?f9JvFX{~_{5B$8L0H<>y&=Pl3c$m`7eUfxh%YJO&Zeg1;{mi#FNHy7Mq zu&Lnnf}^A3qo+sbN52&!(k5yDv zEUNfI#T^y*Rs62vIbFBX;~DLF6_INqt0Rv_Hb$Pv8_k=Ozbya0{0;f9xk#f!RLt;aiq2VCCCd2{nx^KQ>OGQT~;t7{dxPQW<6E;qGe!?pg z_MAA5aaYH+Zkl-a#AhaMp15SvRg4}>ok41hNIX15-@ASM28L>CyJ(%}*?zzJJ3i`e!|JwYoFlP5=yq-{S zX2E#{mlb@W;Ff~B3f2{Tzu-p&PZzw(U6>g?K3WsKB>EZdyl+Myjy@KBGWu+Ezrymu zQ_$`TM)sjaM;A>jIx!*<)xo3{e0=2r4N?&mOfSb`_jLark5RF zHl9(@RCagS`m*QBUM@>3Kf1iE{N(b>%D+UUF2)_>vP#YD+FFxt12~H{l}_rc9`xkUsHro}SlC z{NBWeCq6atjfqE2Dw)(eY2~DEPU@TV%%she?k|0;^wrWm%05;$rL4Z}i)9a%JyQ0w zvcaJHn?+@NZ@0}MJevfnABD^VlkZ<%N3@8H^;a^dIEup&5f%CjXxJ+%i8&u(n zVxl-woG5-P9;TPP5Q|D8ourMlPZ}mqmw%D($??<~S*fBlQrak;c)KH&Dau;qsPdEY ztMXRKrRG;f&T+CjLOrcMQQz^VOKOtlY0b61+DL7uc3dl@D|%=By#AXWZwxo48mA4z ztYm&={%*c9WoxGOt@W$*vE9fXZ$Ad|D30f}az=2Yk2t1V#jWqQc9Y!+?ha~qgeP%_ zhk8f7xNspV=lQT-VECU<6|pYWOW}+zp;{i3BIJtP#t!l#`Ih{g8(3Z$rc72=C|8vP z`d&M=yE;pKp+;$n)>s>;P0%e8CTJtK!Xjvlh#%xqcKT`Q-(%l^x*#o21>^moQP z3jt(#+<3R3Tf{Bl3a;WBuH%+=%e$4`YHm%pF7K*|+uUu*jqJdu?(X(>`@4hO_3q!E zU+4LtyF;MSP^crft-tV06r`W&b%o_>@)^0LGL>_7P5Dj9s}`U-R#6*q<~pfE)REky zwcMiTYK)dkE2bG*4XuvWj8iy9Td8f;PH4Yq541FTl%8D|^lEwoy$$tal)hg-O$8Zm zOflvfKN~rNlh@AdVvglIo;9zVF;*U`NIk2EHQHKfudz4STkP%jZhJqc@VtG;j&cOY z1NTH}8 z3)O_S^!~BJ3E__LO2{eNVp9;zaB;b~Q#>JNl8RDY8cO}8Z=`SOJ@=%)rA$1L?(#JG zdwIWnQ%X8RSw=ov7|r^J^bzRp=e~ho2gl~jj`tSA{ztO>e6cEY_ z3xsvT1EDtg>>Po$&(b6pGqLfp%<*IbozVb%7DmAmWGD?}Loa4QJtTtAs zs*zeDFiu0Qt+q(pq{Zn8RF`hN$;WzDqoAP}ZK;kcjUSDlj0Z*$(=vCPznS^1Ce}b} zfwj`wZe_G{Q%i>0XQ_~eQ^9HEbaJ}!7Uw(Los-TlPBiDYq+8uh;%$zkTW$goTyh_{ zPu+KJJiVhK?{up7rXbYb{)*l&yx^osPtdvpsyC?ix7o&`v>zc^T2DIOFr zi8sVzk|BL2^#g+*0D->YWXp1Mxf7LSt$afMU4A9MlVg>NV9?II%SFn1Wv}v3iBl8Q z(saUw>Ou9I`cf?kqUoxAseR3Re5=`<=C1lMeY`%Ev;3BBSj*@H@;hMMFzy?snP{Fg zZ<`U;2bRR!EeqONX>GDDT2HMvRuwzRPPTX0m+ibxNvFQ^nKQ-N;N0Ww=I8#_a{GZ6 zR=OMAy;Rj0F9+SzqH}hk!i@8#fD%r7SG*$O3ZR6C{`-Yb`p+Pj;DN_G3nM`_dxXPO z(g-ml_@JrSPaGNa+(_xeDPE=g!2AA7$;UbF zt$wLKR6nK?_0}eVZP#gMwM$w)-PFtJt@L*K6#cCJP=BT8F*IY0amu)4q&3Hz$IMgQ z;fv-~^IFg!i&}zJle0a*8fLAqZd&)PXwE@`{RwB`tew#*;?!{_I5V8h&Q9kyI%8f= z_XKyjyVE`7-UKIQ@p5~G!KF>TzTRZ8@mg<-cZrUYGpvTg;X&b<^vFHoQ07#j7@k2Y z;eEjn$_j&pIl}kCexASw-0BM4>3Jaf@B^pDggXs5M%+6$gU8#?P+aKc6XDabIZ zp&C_b&q;C6Mfl1m(*0sRGIJPW4Rk_xe=? zZk5VELZLiDSZD#d-pJ`W2o;Vijj4>jXYmoPItU6zm%(k znsch3sDsq+s1(1eS+scV04JoN?&=j`4lVRK`uF-?{Rb+=@A^AxMINJ&VSy+c7@rwE zc_NdIImS9;8+iJ>aR)q|#*8sPFbkV=&28onJd5AW*Jf%fvz2JIx5ijsTU)3k_jwL! z?ZS2$`(wK`jB<)S*IvvM`N>Y}_$&yX`WN~bkm)l$wV@kLjSj|qT|Rih8rv*#(Zx>V8DG z=?w#IDrf(e$H>Vn!)Ws==N6OlmFlkfulr zr7hA9>4@|jBCG~mYbCdrN5~84!rSB*V2naaaiy}-Lg@id+oT+zKHr0<<_9m@>KCxI zRqAf_XZ0}^Dqb4`?pmX5fU9lScB7GG26^Sw^XTz=5(_sJbRk8!rNIR?h1N?&o6vVl}_uUeZljbrOns=)N<%adM{mrG5JsIBkp8H>QbCt#P;ar zQ}_%SxNXy&eU9w*bceaKsnqc(It{$9ylLJJZ=a`yEAzY-hgZOFQw3ZuN$4&7BD@gZ z2#KK1lRUNiVnw*@TIrzlR7#S2%Rk81Pp_WuS_)2I)O#z zQD-7~!=>~ldNO?VmYx6>n~DMzY349py3Y|amlban4RA<#-fklAw+(7kUuxOWsJU#}w8xBoK zh4vl_B??I>6Wanb9VaG;9w%V|C;q(nP%IAWXb^DMmhwEh$`SdAEYoSffR(SH&mPi- z&`C$ZuFo6uttVC)JJD{#bKPlooIOS=uP=r)l>M4ytCAlo! zMCK058+hI&=yR=Zn)VJ)_Px2ke-dQ!7j z!EX*zt6tj{j4sI;=gf5$ImewJoeZ$Ll5Tsr+9J2Ax5V4z{p>yD?&S>^pkg-&e-}RL zvzj%j0$nhH3RF$#4v*eQzmE{JaH{t6JtL*glCN^bbE2loOHk>4gFh#rx>iB)xu~R3 zm#U{gdYjC{<`c7u^(n0Dp_SjR!~NQ5FK{{pcj~Nr*)7c{IOzT4NnzWseY;cPv4ldy zgek&LLUxd5B{ax5RNvZEwz23>eFF9Iwp>`Lp)}`RPXJ%;Rx(n#dVm~msjt-i+${f| zkI)b3ci|DX(b^bd+%r;}#d&TM>3carOAV~v)_gi%ZJtMD!7-e=P8@u<9+mr{ zo59n(%3cqzFkL;+7fz+Z=OKe8#G@hg5XOUleii-{GKf$hQ3TW^hw=EImb6&b5 zeH3VltK|%Itb@ui|_!!FSZ{Y{NvU4pJeiNBu z_{;agZQ&7$ZpPplmPU0xh++^a=ag%Jz_Oy34^>7h`=|uIP8m$Kj0PewQ7|{oI9^R)z ze=hG-cB(lwdAALmCZOP!Xv627JC5Ksb_cklU zTlcKgpsI=}Ki}9Z=r0HCvQB&Y%NKl-oUVqZ(!)*f)$x+4Kg-}^nZt7MT>K6b%A7jT z$}_>j#tD;!6~bB&WFb!6bo9yvQUV>mk~|CzzOvd}ouszWrfMrd%31YwdRn6+n75AI z(C%hWwU-6D_?ONm&i4(FiRIO%^RDskfY%b>KApoo!V|#yseSqviV_^&!em^GIR08A zaZvCMYM@@Uk~V`_4|0yO$wjGgpUAy%V1~-$-%^%}6>>FA?R^+Y4dm}R^$qRfV7M{dJ6W;R$( zW2=SL*&1anvW{Cf&`5LGrR>^vcY82e&Up}7eP=9{eKqX=0{B8kQ5@uc?cQ>qyScpL zFupQgD>(1K;NIMU>68ptg6~ZUuM2MppW_q;+J=zYml8u+gbzVqgXpG_;taHnrQ%WX zx|p9z)IwS$9f$K5Q$AA4QroA(?;k7is-~7#yQ<^Wzxh^`(KC)}bzt#{@YpnFR?{?V zphpb{7x{X218b7Ciu&`j<#(k~_8$A5{f8aPnVrX{-;Jt!*jWzhc;Lo)1v#62gIcXo zqeG=r2fpuA5c6+fqQX)KIOkgM^850K-0Ql$g_$6tW3r(np~TOk({EM|1v>mQp1-cn zSJ$Bkrqx`n8XdnaC;4mG)k^Igdi6lM&oQ*5%tmgM+(t$(6qtR+b$ECdQ%8a5Zcf6Z z*voByX=buqtBTbc1U<`IYOS@-bN3_dJa&J(uJaAw!dL9ey0t+sL(m=01bwzFXU@la zJ5cIhcoC>d)x$l*gTphz^QmIHskF}n44gXX12KY&lG{jVjw3i-cu!m*ZV~Il4;sm% z@oQ$u$(*C!YJXVL5LDSQRM*MsbR3*5YI!YD`(CT1&(hcEXK;#SqpVTKXinXl3nw}Z z7kCCkj0bnwaKw6MOEZ}&wgsi(lBrtd&=(q7t*pNG82HZ-+wTqA;DMK%H%_Q|YVrzr zW@4Op7shfy`Wa87Hkw8U)bich5$y^nxe2vB3TG$3Srq1Z244QRm4#rDeOY8=c`W9ru)Q7L;*SxJIYBBit7rqeQ);B8J2W zF-nZVzsN4;6!VDjVnGmR2~iLg(V#nL=Y0Mmy^}KIqg6tc8U(Xk53l?~{t$)xGknqc zC{jOjYGcr&>fl}sMkQXa{-EAe|58h7GJL8GXZRDI;-@%a?O|8GQ#uIrIszqj0*>i) z&djR77kI2qqK9RIWgRh!SPJ*F8t1#Em2C9~m&^iJAG8YDA(Kv#Ps<;8RPMV_y0gr9r}R&Ap$0yXW2@nr1714MQ_+gI#@_&7`9 zjl1baaj5H6c_w?@3smL=6z4{u(I0W|BEvcHnR`*0XNGr$FM>l;2VQv;Ortvt`bW-1 zPCVvBJi8%iQk%r~(j*x4anu*zM@^K+gN<*%`!eCl4dZV7sQwnXJJnEAUTVMV3T&ti z-F%s`6@@9AS%~{E5QhA{dC0tL{$-}I@>^xB=Df}6VEPAE;h?)`16dApra7hDR&=kS z?i75vjqU;WockN~Hm#S{%a1=7=9@Hu_4e^5df$2z>8`u{=ej1Pzb1)2z;~KdfpZ-Mj zb|8=i#uhZ{V%Eo?w#8OfyEy0mbGwh7$%%K$(4V&9-rjd&z(PB4xo)_zpiJMn{K32D zrJ@(jM|->!4xLI(wkZ_q3+G=&O(>2d+dy0>8R*J$d26qrHl*2XWa zK#$*UKLz=ipq(Vt{SRFe{(2CUUC(Rb4e}Oy0{{N*@UQ+mxR#n+lkbUs$_aWSG(fw1 z!n-IdHI^pgEM1jK;w!br<((*Ns0FRnQRq;Q)uwc{9h~ceMnj_?nS@=`QUzAh*c@ao zHQ$;A@EsbWDW${XTV-z}k&rJy*0VY5`^W@X^y5EV#rwo-A8_gGd~YFaghz)Lp`794 zJx)zdikUV+;*Ct<`yaz!$OzL4gF?O(*NLaZvuGYyNhSOWD!eZ~COZ%*_E+Y=Vk7QTtXqs731z`1?CL-7P&Ys@X`=9?y(d#+!hP zA2ok6KSCv30&Db9(*b((8@nNnNkRB(KkCGnFz|1^GpGOt&U!;!)$Zswd}uOtK7Z+t zT2oGJE>1$7OG$6&FdJz;T1nDioy- zr`u0Mj0p6#^Pq*IS_OK+Dpa5=T5*`mVSN)_bg7vZH$sE|e~AY%%l;TGEDLxx492hI zeeR9$W_YK93RW6-Wg6Hr4H>ymNEbT6>&FSRK(kxubmxTM&_y%zyf)**_r!Z&h?np} zDj|DvQh+IIfFY+SixJKJiG|0sx%1ndqcX#ZTt!bx`aG`eYpPKlSV| zr@p8sOUaOwBG1qhJn+PHam5>hneOr2irb%|*Ka{td~277p>zSo7Qt)niDPrqjpU{z zftWUV`%#*wqW5eE1EvZ3Qi7m?kIsV)(xQ#{9^AL$T5d!<38Rgue=h=Wd%3(+`CfUY z)TfU6KE6N&9i;ElFYABkrC}>EWF-}q@GUnE_k`Z#Xztl$ke1Erx*&Uq`x}i7HN1yO#<$iLn;Gnn29I!f&39l zY=6-CITBROI5`_pOWvUi#G_bt;j?X3Q{#D8rB1CU8a$* zW;v2VD{z0}!B^X@^dO3n_EKv9OS>m|g1=E#2fD}kjA5|$2=4>x>rRp|Z_o+T1R1qL zcoM$0+8=(MK-#Mg>cvtqMrwznxfJdASLwO*p)AP_4Rc?t&wTWl`R?dba+W^&QxH1RT`wS{! zHmdK}cyZ}8fv4USPjXX`FVOVLdMEVRu_z1Y^(%a*)JA^PxbJxWVS4UqGq?3QD(7$D z`s_Fdb?pXrli-_uXYUL6LQ|(T9b+DD$yJz8G~Y1EokKq27WnQ1{6`nIKZ0D|0We+~ zvMhc!XHIxMHwg`AUP_((06p2oCm8`ZIDmJQM(l#h^g_%m6_S!r8owjG;-k<~xJT9T z*9L%44$7BgIZ|#0#yYZos>z;+Ndy zlojSibT$TY2Ok@WIPMA71ekVn`=Bj=nr}HZ@xS8fE059ZcL!K5O`w&ug{^3)5|711 zX{xjcr{NE|h*D3<&DquhHNFLBpqywqkX zb0nvua-fk+;)MK+R@=utNvbKGC!l{HCsP>4Etp68ax)Io`S5iVq(8xOX#$NdEgDh{ zp#fd4GwG8r=`>S?d2p#LI8Ci#7)?3ngQanN&REcTUu`&w{C?iMg=+Y7aM!}%_REG~ zRyEt88QwHY!E)!(ZSPw(dC#rwLEN~>+_^O%oa^>;J3ViD1&NCNoaXb+P1J|yP6@XK zI_C{^djt2U2D!Ys_=F4nXLc&3ey$B?tcf5r6O)I=zos)G(T2}s3N!gU4M8WqvRe*6 zZ8T5ef;>je#tB}cZA8&+MZWKez8AOQH~!Wpl2wT)2#euMUAa3I`5V2RBzpQ-cXyzK zwBz1a4LqbfV6%t-@B0bd|4>QI* zNG`&%j*=>q?7is2^PGoHRd=bIH>mr|xS>hm0WdCZMko{?!8}GNRG(+N2n8`xDni{* zxptpoUExl`IlM- zUDrG4L%p z>7X%=_x7M}6%98<#Xab!0h&iJ%MuEe!u6gi9HEXUh}*@x(op%Zp4)KYa>HO!S2#Dj z&FVO4H>`?e1P|HC&K6R=`MeU;juB{PkEweNNK)_iRkGxWU{_cX_}c_(n}>Qb4i7yR z2G?I2iyPKVIji*qA$)KBWyRP6?n_Htgq`+D(rzQ2Sxzn#XCK4x!3ISzixUcMp~qwr zMe&ksDy88kH3By$8eOxN`jxs^Jx_wUSfDODMt4xtQlkaVPA2e&;e;kc1Uo_Is{BLq zBACSqg*pj+K|-U2iNXx@(lm0koCQz6CK_b}xe0lxmSi42CHFTI2l5B?BNVi6sU)fX z?Jk!fjkE5#@I#9XA(Uw!7U6$35B|5cHpvzg0juLA@$YWnUq;29>W=F5nC4 zXw9QZ+dNjYY8pMSm*!{3PwLn4hVB^=qyai{i(-O=YHs_i^9JRjB)PE5sKa@|P2*vu zTRCqxQ!=x=Bbdbrg`&}fbCQm}Ej|*nNd=|+XaX8&Zi`Y*t&0lZ3Po}#(*}#x2jmzP z)HGj@>_I~GN9`vq38x?q%-xJ?)WzyYXPHAXbhWjWd;BN)i!Ang)Ta_u%3;*>skl>1 z@KPRw?%u*$N(KqHp=2~?Qq}jO6Q99f$w4afGm@CS$@Yvu^PUbiegM*l@S?r!C^1FI zllsZ$@lc5!c7VP=%1xi5dkjD1e@^FUJI#0w*@~bzngqYSNh~KX+6%- zZ7H?JkClq)~BRPijcQyO3&Zq9(HTNBzpv!{VTX+ zfpEKjGKdiYKg-6*~-o9=LylwdZ-;0%nPjbo_KkfYUvSP>ffu(X#+kNhhFfUDHaii`8g-TPrLPjp-<*aoR-qc z34S&n{kR-i#!2Kjbrg$fWCd;-Z^2J3&9(5!L=u~;Er+wb7?$!OIrjQYFc|I{^v?Xa zfhYK86~lGH%}{|3`O0g2%HK{!{TM`s`x+ULC20TM$w@`xTK+*DdPjOvMvaKYgUd^D zRMa(8or?M=;HRnjGJQRK`;>7F?+OY3KX;~koQnFIfyC%= z4kiZT;T&=}B%}?v^;}`1_=D)XYc=G?xX#PuUm0jDqS#6;rGv5%&i_WK9%M{vYIQkH zO{gxt>4#nEg%ePg&geDp8~n4s92OL9mM~jWYetY~JP*fwhF7|u9;n(E>}uo{7jc*O zJDuH$Xuqo097WLY9wT`Z=l$+oI&Gl&77}WbYafOevYE=0234s7On)FfZY4eLxm1Iy z`Z;?4Xg=))`g(ob-Q`MJT=7^fKlgKl_DE|B_n)LMV)kS|IooHT-8T$qMi^1_{>-HK zbCM*BN8Kv|->ht`COh_!)S-$GH{LvI*0j2Ea~sf`Ucm>dqgG6%0=;&M;+ju!@1m~e zr_xkJZ_J37>L+@-gK-Y{=c#5|<~T#4vS6tUBndX*r{zQGpF(!FAQ)$hY@zYYQ9e#<$&1%Y5z-cQ(f-bA%h6>1^DRpl0^0Xtqm9`C52!mDK!23XAvoGQnZ-F_ zokABcl%#H8ZiV1`zD4W-r*UF@PzzkkxGA~~#-ze;(P(|>= z3pEW+Qdey?IP<#JLjO`fgkl)uWv2S%B>mIfGs)G@g8QZoYG(lVXMRLsPDjeuC2{Bbz1zJm)XhoZ zwjiY z`Y?sEfj;}19D*W$j9cK};n~X7&K&Tb_LT zbSiW|{JN9qdbK5CbSy)Gfge#W^?jytpKZ<_P_UIZLzIf<*Eqz7^; zuwz=iW^iXmz=x#NMifVq} zS_Svc=yY(FJABCa$Y6)wPO%3HADUQ)J<+!reNQ9e9?lm%^jKm9(0hT|H z*K5h8M=18gr;ig>S@4Qg5)wQt2-#7KP{{%5|72lkUow+|cj%{D<%&a;gSc{0>;I z33tMGNJKK7GYvn*<>83mvYuI@JqDE{C82QDz3vwEN_jQH%}L8m^-)93NM`Osp{}Sc z>+lyUqRn*{6Pa#(K^Mxyl;{rfMekGJ+t9i1!yM({Or{#||j|_aQgSZIO#kM@_b)@fNxlM_% zi&FtAnh!@`OYJ|e^+8)1i88Q-{`?Mqr7p_AOgi)h&{8q87BiTKm^Ap6WOOzXjvrYj zp21mb6KT2(_*gEOY#3_P8}#|Ie5(C!32ym8=FwuqCBiOF&;-!grIZXqa%6z16Y!2Z zkfa&TGr5A_(hgmEJtrzs5~O-TQY;VsuN<9wDx58r#7qnFSg-NhrmH*E8z>BA19f#L z+C~EUP-lGwl`0Ou`#$M{=A8EgOGk_BL5G}XWuyE2VVm5VEjTgxU6F)iUl_pxYSZ6z z=X{9GUwoak_)aC?%D%$N;|WEX-z^U#4QrYJEMmXlu$OMXVUcAy`Y zA{FpdiveGD0z0oE9eRmr#vDco+`w+eC}T6+{}H^Z0*Lb~-jck_eDZBl%OjLA@Vd#& zlteQ>a}H*;7ysY)u4kZ$U!zC+@9DK&olMtCRJW3R!(-k{Fl{9`!(l&HIxjN7cGXen z?~>j;$#km=n*9b9H7~PilDq{+F09r-C*4kW*v~}g9knGY(-;5NmtJW(^_I-(e95fJ zMck%qFo!5K(-vr#`{549K%Hj|4IF>Ox*YgBv9LM+JQqWed*P(TL+FFEKZNPjpr@`$ zx%+4>LX;3Ayc8OVQ}Mt2+;%>tD$d}CFv!v9>yJrEeWG_E*Y$vtdkeoZ5o9z2?zq+3 z;$85c+wRDKcQ3(#$ts>=vRP*e_fz#hiH&(6^kqS!`;hhp$ZriU*%o6vx$Xaa_OWJT zYdQaBA~RzvVcOThbwfBWW0;nj%rk9^-^1TJ6&Yv-L*O`LP#`9QN@sJ%7on7G!qdwm zo|g8@ZODP1rJG%4ilHa?^E*!M0qY0w9My5Ia1<2RfywjkFpd6X z&W7N*jY08_K*<>rXx$#z_984Gj!DNxB-k6PaZHAX!4&EA{8W-5oXasF@TE*g_$b!L zcSyeattPEdlOElG+R_}v)5hxnuI%nRN{=J`(fH6I`b-?k(IEL4 z=j}SR>!tb-$G9R{mf5(Sx4>-&y?i{4bSar2VW;>Rop-)8U7k%+aS@+&6M48S>Jc(4 zvG|m^xw~bVNH}7gG^&zS>1d8LBb_1WTerNU^qzD<{aixkc@x@8B1z#FYJ~Qd9P0;s zmyh6zrGpvUB(%1EOa)Ek>@Q?SZYx=c^K|w$=4xi7s&MC?T4D5qnCzgc_1mj z33v?q?J#?Ig$jkm^J?nvZghllB(6u$Stp%;(7En_mi+p%$W~EO z&f>0KbsjkKzc*Fz_nM~*wC0TTqLhr>OBnfl@jDc$Cejhdu9`yJ|zpYeF_-;;#t{ymrAz$<5iHy4o{O34RzASYiQWv8U~k$+kU>7@&L zc_I~Vr%(XLbRo&hJ5qOM#)~Npm1N}?Jls#!x=dd+f%UhfCv_le*qxMgf4sE+ej1C& zTCGHB`af&$MJ7_Nq1$AKZ{`7k)F2-avLoy$JO97uk7C*H@!!l*4W_S$(CIgE?(g~& zSA)_8`12O(jqi${BqKaUDWLsr|ErdNPom%kc#7{gAzgq!&*Ji2#ixEL)j`eZNyhI2 zXuTklAch{+tB`a~qVH>TyJ=>a8raqzU=PFNisVdIWRi)5M<_HeU7&UM5%Q7Jc`W<$ zT4zWftRQu|hwUKywZqymCYNuq8zHA&LEou2AUQRN%G{ILw2w(e6tinlpPqRG!QfCUffQo+hlToMA1^BriTH+JYkR9-N zdD7qity_1%Z|#n=+@J0+Bv7S%pXpQH`87i#sk7cJN+PYNJ(e4kncV&$IDaIQ^CR3< zsIsP4I`C*#lIge)f^J53pD%bV9Xp^xp^C8UK4kl@l6=Ym0&}QE-Nn72#%SgXTA}#N zM}0d9YfKKD?Hjlng-HAM#@Cq1Jj5mC52YXueIsV+kAYw!wLEAaU!%tJgX|&qraITv3qu||Bm8rrZeBQhhFl;Den%ak6flV z76k*W=H#Rg*XHcq_TLHH(fnM8pO=^_Y(RzFjw-&NG(6K)(i*(EEzJ4umJ2B&dqJ9# zE*OUsunvxQpG>%cchU^+WE?DYKUnB02qL4FoAl}zOf9?sDJKUTY+~8MP~P%;+}F&Y z7UqUeMiIQ=e8Q7$g|6R^4t^HC73F2}ntAit3Q-&le-LxZC;iWuIV!*naeT(kK^IFa zX2R*7O||@lSsi~${ala-%7I?%^P;zM7R5kKSxhG10%c63dQX5Q`5RgM`~AI964xu) zjI)aH6kEbA<}$r^3g_#kn+FY^zZoAD;D*^uLY+h79v@^~wn=ADGcuBIc`8?6pU5Po zvRWNaq%odIJ9@+-Jj`S2Gc@=HXh9>jg`~94!UQ9kC$#Vpdocg(`?e>UYE451JpsjH z7wjyX>CiP3;fTFpJj-z8pO7{DSCuTtr!2u#cx6&mje?Zd5E48`tXg(2dm?)emcp7( z{f{Zmiq!dTq^uS(-7Ap0ZR&R8Ue3WYIY~Yv#AX>AJop8Z@vFV-X!`NNY!u&Fhzk5^ z-*M=Ivo(biodsQ_1D*B?sf?0RGuY>To^(OkK>=B;Y)AiaP(YHYIrEvTj3fIn5}#zR zb_O(f8}+&|nHhfyXgc5XhS31!aTCmGi}edLBZI)-*GS6bg6mh}6nq7S8%^@?1a4?L z(vqRlQGq5HFC^gHbmDw2fdlWN=jZ0F&ti)9S1BFUYYmS53u@FWw6n78*vO>W)b}1* z39>DBU~f9zD$fjU$IU$z-;%-6xJ%sG65oUq-`!e9HT z$C^?8dL?GV%Y&f&3FkDTMWuRy6ImD+rne;FF&~v5D0S2Vq#Tm8M|yfr*k{aT^`!cY zHwK%RQQGeEMk6f#q2^J+>}OL}2*ya+q~d7YiR@|Vhf=&++RJx3$CiO~%$^h?-Rv;^ z*+CvG&!K8mSK6VUsUVx)|1geUseSiB&}q@WCvmc4^$G@JOpRJEksWM_jx>w;)f=SqC3?q?@&ocZWl1}a zV@l>nJn3R;eYGEs%D1qFpU7&bCYjWq+2-e3HIQUownQ{RGwBP%Il=pT!?pyODYpio ztaCm0Tj;#SvZ!x!ncCPSjvs>@uzt^R=egI*x3Yz_tDjTK9ac!Z`X2mLW(5y1|4Ld3Z93(hWnu?P zY34P$;Q20~-v3Dd^7kdE`02WB*! zDUU4yL%w0&wr9Y)3$n4Mle>f){)NvwCqxB0KoSgd3Dx`pdS*6uH*Du*Mf#oriJ&FS z+yA0{9OxXyP%0Z6J;1aJU|MI$He_V3qYWD488a)sKr$1rmq~ka0hi&Bj>QdI>)R+ z8C>Y^&~0)#wV6EJNw2&B2Pwm+|Bl3ZCO&%?Qfs@}5)oPx6?jw`g<_n6O8+>?^Mpgf zJupmRQ6|6Lnf&%xaz)oc&k5u+)^VF-0>$_yXs#rRmPid&huL05jj4kw zRG!@270zBEy@dWX_+uX%`W`YdRSf)BJxFUsv1}Ka@-BcZ}lK+x|vMpEZIxwd3Sp(u)PyZr7S~G8t#>SCC>ssiM@JO+5ZalD_g0_En@U`bK8mRBHOlDWmIj~b&HjULr5gyO3xfASPD(+qJ=}nk9D$3J1fE_lPr|9suPWfz)+Ef9z)^pNOTUlXTOIV*fhnZnOdw5THuXBw zDH>hBI~l0c)N09TgDU(z8OnPp8tmh!U=}tSr=MWwxSL7Eui&=_ zP;+kKW8@*ZIF1{1gr%nW!E*yZpZ~Wx_F-KzdIfWZ_uLHJ;*I1wSP` zQ{2qEc_dW<;Xak4lq~G0YQe;v|7rffg$uDE!qRH-Ox7^(csbCh4fZqc(jPG)u!9Wn zRih#f2)M{f&M9>M3)X0KAa0Q!74wz#UQv*@Ob*5u2d{ zPr%36!^z1-C239_nS}$iRXoCm!W>dDI7AXz*i`cSvPG__FPY_2CW2Cx8PrWm zN6u1TH1&0Aab_9Xf(T}^)#3;863^Ht=AZXUWca4BRpmS?LQ|Ogc(QCN)6Lz@0bqhj zxJmgvHpoypvVGUU|uP!)4A)3}KVGG3sNo*lkSg|Nq9D(EPZ zv_{!rbH_+d|7ARYc=lP;lG?M)bT$c_N5uy#)J{KBW&1LmPgs|4+@~h>_xirC@^B=GA|zFOMzLu0(FR z1)kD;J1eP{QKX4}<{TCwA=;fyd=E*p{pIFni(on4l%MynuIxr$^lr#!_dh zv3X<~y*n=G-0gAkcaXjE(>9K}lHCS3wW6rj8_ykzKj8I9r2XeJ6>^$P$UA&T*{Xn+ zJ{JA!5Si|HZf6I31fOaLn*rWu>*7!*T6f_4WJbLoh|_zVPgacd_FnHQDq{o+To2O7 zi_jNWa*8+LGjAvP7idS#(+BfR8QG;!mf3)w!hF=-`+~td$=6_y%WQ?K$q5)vy@@3m zHaR9s#<>-Yu?88D9aaVVi4%PD zLFogoR0bR~L>Pq=xroX2eBhU5?1cJN7TNVQiSGWAS$coVAs zGE?Bajf=+mAh#56c~sP~AlQ4BzwLI9eUqHC#Dsqvr#k7kfo$D*;Wp>nOrl@j=B<3l zcUb{P4s`Ab=>wneLpbs`@T6aaL-fTv)V3F>LJ@KXIjfwDjTVJT#i}^F59Q3t`%0KM z@m8sY5BEl`4C?Fvo;s#I)C-`UEZ|wc7bJRi(a(Q3-=JN`k#q{#=|F->+^S_Lr?2e- zPFc1Ntl@T@bNagFIn8gGJZ#1!PcSn!FMaUaZ8lRzYpD=T#g0s4+!aG>(g%90pS`<* z-s-1hG<2xrhp74l+da4f&LgRY*aL`L=3uE zlsODtKZ$wKG45d0!WG=)%l>}WYw3gEtBV#sKwYgR-ZGz?MzBZbZ!7dTec-JeWd_)v zl*yyM$E-#PwB@y6{9U+hQOu?nC5Q1P?_vgNcqbE2|V@*hck zsdSP#u#po?OQ*%1{Sp;p2YRB;R<5yl9M`ywh2SBd!f?Bg0Go=pc7jvvZ}E#Exot3A zo7*Z#mZ&SL_%zh1qf8q9g-7}cn;Jg1XOZE*Nsb^VbCYG+U6F`dF%ay1n7SB)>roq2 zxZmvte|Y5;XRGfZe52?fvk{4(F`O=WM(BlaHiHhi4Rq6%O;CmLDLz)Np!+Xm>q6Fl zJivc#b@DaYNRrPra5XpRx2e)YdA9eA4`7pn*_TobU0^i4@dD|&8dU1d+`DvqvZmw{ zcfzhRxEA}aj2=zlj*{?Y~B(lMz8D(Z05 zi7564-vFVeqf38_N;Ha__6HLYrJrf z$3~u_UMb%A0!a3)an}gL7R+{F-Tn5!#=q3RSG`&Nvr+J~=w@xbM@As=~jNv>d0xbWNMJsiDQ@V z51fT~COqCJSwN?rkn%qKd97JeKD99Yth9ELynP0}?|*q2`9Z0xz3fcA=b_sdWFu4w zrWktr?mgZ@bZ8L&v7XwL0W4LN{b#e8d5p&2s48U%l+AriGx&Qpo1h!`$#Iti@;YM} zv+EVOTWy)N97kWDiQAKp)3Z3(H++a3&MmlkduEG9psp_EEFGhAT*M#ow;~NEm$u($ z6lN(Zo)u?iqZO#=IJhx0$RRJ=B%3i2IgVQw zOIGkSQzA8R&A+CrJ@PWc`?`~X39=T~QqFfIlL?d1%@T`i)XK2ob{?MOOYCGd|`qs0C^O>LVdEYv#-8Ojim4hCTA?R@*f@c=d;j*Gs_7cCL z9!2Akwr5VYxm=NX{@+MYR-?+VVqz_~Rt~PP5x+q&Q?n7O9y~WS_%6t_X3h}I2St$Y z@lo9iW)L!xv%1f7uZ>?mmtC;0*-W=u`GP5gUr_$bYM*OeP$E5cI4*%DRisB;W=nJo ziPGm*3)Ge&w(nqH0q-iD`yO~;YpJ0b0xS>%9}#lww%6_zh(>0dF75WkS#1b z=#4VpwHGz~8zyaUb1w|kv*RRVic#wZ&{bBGtnJ9$MSnicaopgHWM=)e>~xU!D7=`} z-Y3CK*mnQDmCg|G;Y&g`kmpczI-e=2az_#gQ%IerL7~}>Vz?i#@fZ&LS*B_gltTv% zp*$MLN+#?#{Etk}F`Ss%^z9jBQBRmisQ{U?QRtX2;6CxF&?!DuNj;L3YDFq#SEdDr z@%{3$k$5l()$Pn@wS*;{Wj-iFppQ5K_S!)PA+K1I9lM>;N}r%W<(5mJ6?cMzpMe{c zCTSs|M76-l+^PzwQM1WZbOQOrvT5x*>n;4Hn%xdXX`da7^Ed{seE@$w4%{`9syq?@;+Gvx|8@ z{&j{xyZJmw6J>#SG{R@;L-yh#8f_ek@h0}jo<}!~(%h1z>C z;@QmkRN(A?gTMI}N9!P4b3dTN_Wvh~FoxOI$#jN$WFB(T)wch$5&Rfa-<6n}9Pi9w zJ4APZ@M zM{I^0LXu{TG=aGZ-|KWq{C)CIhQa^sk_1H!X02+gQn26k4L3$1>D!WDB00=nGYhV~ zn2gK{t0%nl8N06Hm_B>qlth)A!sq&dEt(@pM;2pC%*vGg9`iB;I3*2xyru@*3RSl0 z^+!W?;p@pD-k+qJ+HJ5K>}XAf0Hbx{cPsXg5jo9%is$TLnL#=<3iqmw+69gMfPMkb zcokW(b6}mfMoqRE#KOrgS_|1;kcLlGz^Tgo`5*Y)*+7K7*{SmkwYV@|+IH_O8dExU z>5XFA^s>L{ba#fJmlxwX^@aPK1d;n1W<~T7e}m#JDrgd)rzR|Cma>!HHJLm70~5rl znPo7ka2>T#C?I?20nbS!7RA{fpHGqb0JN&-!cov)7&}09%x%f?PHcI)wydBT~JyX<4 zS1(svYs2;L*=sCW^XxLb|4`&!tWEB72KkUXOlK&hRX-26 zg*{9}!C!})h*}XJ6J$3s;6TjbX(X{tc8s(Ne)JWtnrvtvj`m)fS?$@Jgc#G$ zIvk{yrzN{R49>QVT#aYnwJVS=so_5hAtvZui6pG9{r}oy<vd3XHIHFYtU!z`wO1m#Q%#-2xU$3$InFfT?0@;0La<-Ve=p^kq_0Mgs>^L{f*#uvJ$(~d`S;k> zSsuRD4z~7&{g4JqxC`3I38u{(obJ$?lxOmnu$CPY{vuUj?bE512kK?D{j2p;Ivd&+F+bY$9@~=k?JfV*H;e z3bA2=9Aq^#y4F%Ur5f66UEGt_>OAdV>9&`~D07~*4YeU9m34r5K>mx5V}hFcpDgQs zQ6Y2T(7($oy~v4-A<4Xyx^|ij>ly8i-rk(Tc7?jAuTB0hzpUYByT*`gp3H3LY>@XN za*ru9Cl%>M=Tm$Ze(^1yKkeslmq^>BwQR~c#_ut;Hh)K57-)ww#|CFwBi)z?XPAZi zah%*nI`(f3rL%4(Q*?_t|MtvywgP)s~l1S;xbVvld&gXl~N5sbln724P7CJI% znHt>FS5B{FQo7<-SJVDA4cDA$pEjJ@fBnkVp#Rz(_n##4zf_3J+9|yYNsldNvXAqH z*kDd4QjTHr=YeuZt3G0Dc8bhU!y(HYr=Q`JGVC+hc`PZ_K!^qJ!ZbWAKX+BKfDC{Clov{ zhx}IgV4gOAxC(Pi{N3iU{2pQ`RF90}TlEvIISOnWrigl?N;M}X1a<=*xhbKI4T?y&K%Aj_IGn$PVr_2#Rfeq2g&tp zY}xOEYV{j^vJ)9&AA`38L!47uaj!R$w9BJ0V~o#u(AtJxkeOeBXht_V&&?_wOw;CM z-mzk^cVvBlTVjKLoP&)cVKle{yxE-mg;P>BT)8Q%%c+rwz``;iG z=nQ0%KShtrj8C$e-)YFsR;VvwjHQ`XxlP`xfuo|E@8I_ro->8udxo3nmj3tZ4IWtJ z|JnkzCpN&M?}1k~fv}&lqo5d`aWm3lli~PB#GgS?*>H_3lLi?8=UKx zwl5l%LuynZF;V!^J%FvR|LC#fE%L z)duxD^f)%)yW9Vn`SAJS+5eRxdW=%`3J)NpMQBl4jFwr;uI1#0#cKtbJ}7}At+2Pr zA(>L1UR_P|b5)Po5fQ<*NngdCfJ*)c%rAl8QYuT5>IZa_JYFr(<0SG#E6|yqfB-yF z>H**7S6KZyoIK#tMWCjqLQyLAbk5mo+#-!k)j(MIQz;iozRqyq8`R$h!Eep@N!yGh z4%cx1lVIObdM3`#Xnes}`rpjG_Tzc2;HJjme{E&vEoL}{b;BWRJb2Wq8b`uU$5-*|`b2)oRpJd=2d1N=5qL5$3 zvC7L+sYKRsKhwM|=)b?i! zWt}9~n@B$U8~h$WMcW%Cd_1#5nVB8x2cAy=Yn>sDmJd8TBW1T_e8wO@=BxY#nNeLQ zY!!A19|oI~E{F}KKX8R}q3VZ8t@k8vG93+Sur`)oH@L^lo4bO5k`8B0WzG5q63~IWGe|Zp<#U+ijiV4H_Q4?$?DjTCCjYdh@ z%SPX4{Hjn*l0K7_xp=IvsTu{)sYc+VRl-yCcVQMbFT!Nw?Vs>lRZ zljE_#eJ#jNT%`ABlJYYV;crCtH<>-+^CWRMzvc9i$!F8TYc-l*@k+qAEXS{;b!P^3 z4BN9lp4=}JE58RlkWzddl>1GBs3}koW9M@?-+jF*qt%>ow*h4 zo9G~JL*ZMB_HE+%n;=b|_-M*-S|%CMJjd4lhG znRMqZG?b^H;CIxDXw;K9w3GZ`n^Nc|{_l%aIX6NLN~EPEXMW+v=r z3EXBK%w{M11CGLK&ZG3)V6OZjiM~IXDUW25qzIeqJSH0ar0Z;E0l#IQxgF~KICjGo zCCgYAyxW3a?C+Qjoyr*Mz+U1U9jM`!GWET_{A^oLf@SKInQ~{bDq^g4)c|F!dHC;ucn)rxfj{JH(KoA@vSX>G&SvPI=%zP ztvF2CWB3k5*r&Ri{Q5JLJr6P2{v(<7ndH^u$f}K~y}Wp}_~8#9wg-OEA=eFCK zdie0R4>Lo*k}2xl{BB2~^g9nmJj{0gCH^a5?f-)={&#Hl{vFlz(>uS$j@eI{(wv5Z zVtCiri0k8-)x3{=<2i}V+2f27-@1ivE6jW>$~R7ceF}WzvO3@IW?wX}8pUZ?YByi< zb+{V64ZMDF%Lma6#8EANfo-G|_g?M~YlQ{vj&NUs8)7W%hbB zwOFnoz8r0q+*f%meb8H&n_WaN`C*XDzcL5=GWwcZ;PUTc=lthrRDO>}=vn-#&nF9< zPAxbSE$i#qFMd0@V1nJKJ5b={@x=KRzu^MTKkHE9Y-SR5BbvI6%&uQWFL^#J*t^gQ zUxpvkt>iB^dG2LwQTf5==W-3}T`*(c-*z9G-QU3cKZc685 zM|_kUN1sC{^$qgQ5A9c2jpSu@`5z@Q(M$P75#g>%7i&sGy@bTuk8q z_-;BbQN2BzUBOQ?)4847jQ60!ei&WeUzmKHwEGk^`96N#|4u*9g*y8E+&}sR`6~&& zI8H2m(E+^}x0*MiKzI+Th>v+c4JYyy(C!^<*xyH;pTphThO4)SxsZkzDi;0C->Q$v4-rxZ0(P3ELaW)vH;Cg4^RpzPZ4(8sC=%)N|9G!S&^|Dto$mZ<` zJ#-3|eHwPXNJsYn*y-K?Zha}9{%4}{eIN*H{4PY=b%4s`3sn312>#bMb&f$m%h$M^ZZ7z z!v*Z4Uy8y>uA_V&ueF=W(|4iO68FI0!dN#4%j=#YB)qy2N zElG2y=M!sYVk?6>_M2jq=APUh}|7tO?3T-55M z&dL1o1I&2jvX;09ev?}8Ak*|eGf#K=Pr|wRJTm<|!8w=1ihlU!Klz5xsUpSU3W z6o%np7{b5cJG22k->EyBcb>8H)#z9+grAP$q4_yj&fC~xD6lK?5HkkxvpSB>^&i+8 z38Rd;ggb9T>@{4+jPqt#*(}#4?nl}EC~ny6c0Y%iYm>(CtC-}ynXT#f(E)v&82%z! z_gm0==XTHN<>>Wta)?_fFXdK8H!jt;!N@FnPo^6DBbxUyrjS2m_WVm?^f@p>a<%D| za46@gH)n*Y#)r^d4AaB?8@MFha(m17m?+KA%l?*K(`Rj75Bg~Y{j_P;3X`=irMvsI z=B%69w#mX!OUC*GQ|+T+&rd?9@d{kI184}&V;1mEv>N|{|HQ{ZN_%u0=QcEyKVnPo z|Ik%zJmX|0&|WfF1P#)Mn9O~iZB4m2_e&7VMx3mCcp3E)ou5M|aTC0!I9k1cXzXP| z{s}aKH!{;QbJ2cIEeJ1+*R$iyPOHsrAh<7NC zl^<^X1T*(Zl=9DEB73bm9z70f-h#^Re3VF{CHW)1sXKB1x*aF{LA>~X%r5ma`5Lcb zN_81r*L8f4pTJ{2AGQ32WZdteEIuB_U^nxj*U_0r;8VT^hBg}3>s_XG+$OvNZ@q`{ zSQsbv9)&G9y2aUYQp@RV%zXrv&sW)kdRXh?Dg4bot`dqL_XTvI;&MC69Pu{#@*H~0 zUr-4p2L{!3`Znglf8qYGmkXBv$jnW?MuA!G^Yr$wcmv4yxC-~~#pDtLw zpYC*{ABcg2F9w(0C3wNRUi(x#IQS|&vtK}0cquhu9e?3fWcqceU;i0zw)?s5a~Au^ z6KszhOFw!EXsNVi3lp&`(YHQ?^U67RQs2tI$7xj0J>Z|mx1GlCvWF{8O=!ffMEAIW zH%vcHCU^T!riZx!r_giwcew*rUcHm01 zhZ*fL;Es2p#r@vqlj#w!2T81>PFzZ5cn~+3H!;7t74-I8b}aU=t^W{ue%aN1kS*zd zK_mb8HXk`Q%RI9O^!=wD=fmgA<-v1cLGMNpaW>n}qP2~QyE$6W^*SSdK3w-VnV{ZB z=6MI(g!iF-Xr({B9;MGZc62|>D_=)%d?ouGk8-j39Db=;u*UhgSA2`v(Y1Vy=aJv{ zpmutE`*!pgH{+=BGJX{olc0Ao1^Nl|pPjqjg>U~a@GQDiJcYW~Ymf91+_c3#cV0 zfi*7yhujZ$a}Iy|Z~bR6g}M>0X5Fss;P>l!6OYp)eO#^tG0R0T6$rxEGX%=Eh0GAiMQ6$Y){f5waW_qc)lb~&E@3tLa4cX=*q?JN2E zb}HTLQ0Kmd3opHRQkwPcJF2w*3H;bYcoj+)`3g40ce7pjTKe=iqcXh^J@F@$b-D%K z>8>63pl16yHSS@);^WMtJxmGDfLF(TXt_J$DTu0`DgSidr(q+9VNtf z(ZJ?mUS$vT_q!jHZya53q`+}@batmOhj@ecZR}fKj7!qx^hcj$U*~hUQ{M*r{s%gw z|HgE}!$h-*E=evn{@teYlrc=O$uog6?{=!>U3B}u-t>QQDK}BvYnaWRuXh|TWGnWA z?CoDi&vZAD^c(!$9z)@N^yXtXZ=^oT_FEgOJ842iX057{%S6sEIDvwO4@W--c(y2e{JvQB)k) z5FryVV&6mQ{}=eA6F~8e+&tgR<)E|C{GU%Hct0wN8^CaPprg1Kuc{*Nd47wH8<8iY z`FuT_Iv1e*em@oB3Ye8o@z!o215L3n^CzxTZNP`^1vpNh#s=cqWJu93UxU)`t8B)9 zi|OK>O!o5VdP=Q-ptnB`2j80PzFWWIl`s=QI4r_&4GO^fD^>o9W0eCVFFVQp5CqpGUiX6KdEW zg4AsIxOsNIgwFd6?unm?Zd06>-%72&jB0-~DD8W2@p9+Bg!bW&Ahl=jI)TVNosGM< zqo^7Hsa=Va>@fV`7f_l{qDcD@oyIZDm`|Z=_rh-YcfXS9!|R#jUPK=H67}#l@^TIx z!VDW(50MwqiN)8;xfSxl22`&v121$ldw4%=(}&5~pG23vhs}zch+1*Kcm|$7C$YWw z@=Yx$jCK*Tuhyu&nA6)ihZopoyeakcKJ z|2tjjH^Fw_WrO4abS#gt73+noKbxpM4_@mnu$LDSzcI4(7syTb!6u6C;4!+;qnRgd zm?LJi?9lEDeEP8#s!#pzK=`ISK)baExh#gc*WmL zjr}f;NI#{f{f?f)vF!vLXJ1Xv`c~bpejlvj6*$0rX4@C>3%C^@xj)cttfSjFfnD4e zssEI0NOW`k;A(VIEa5z|G zXV%M&$FqsnH^P2~nJM&Bxf8p;z-5B_@cjI3W&bF-Ue52~8*mH`c`x$%z_`MVTn5hl zG@ZuhHCk`M5B~?SrE_TR8i>}duu!kkD>-l0&92MXNB^HVCVU?x_#lz`h)yaUxI&-G z7Or1AP7t#)iaQ~B2mP@!DMUYh8Qvdj(d-TYG! zVu|i!5iZ&3dp->KR+P17pL`{q@-;;2D9)QHbSqinw1`qe;`DSYPR~WXbw1q51?1X~ z(<|Rbj{On-LM1eil{kILmQ5O`@3rFeQ+VfHhkKLUV7Y@Ggdeko_6Qdun(*SrWB4Ut%`!6L#7D083`ub_~k;6S*4Q$X?qnGH(|IVpNDg~cl!e<{1)I_o=vp6=!%eOy@m*J}O%>c~c;3_^_m5Xs%`oG+=TXAL3_`(4%#rgN4+6n@kJ5Iitk}5A2{uxR@q;nD{l)msXS-er_UF)D@kmHbU&H_JEvw(Wmt> zO0g`n)?7=)yQaua&m8+Q^XM4%p1(H0Dm0;` z^wRyU)!8KmTN;NaOH>`b#_5zRt{+omglT%lJUWvZZv7U?3v*m3ou`9btkL(m;M6=I zvPS%pn&H-bRV7F}oJA0xt&5vW-M$`Qm^>1x-WD6IalIR1`aVki8wbOsd}%UDf<4S6 z+UeC5t_MHCj9Yj8lB?{8h-XiJ#kGY6_FjywwypN9j;+pRm%*Fi=|bLG@XSZP=d)h%fDZ5ZBSCkx)8h7B>UQVNxT#?3eA+t@k z?G9cA+fT={K+j{~18nL`+PUG?RduI}Sq^-OUB>Ksorf)5wq4Q<8D(+_DP=G#4ybpPTscLzG0g_YlGA8ug~y$P4rIBTxsM4*Wq;m0^QUs zQ^yiLRT*Y@fzHa~#qPup7aBJ?N|LTQ7@1qABKmk6u z%mKTSfz7O&M;AWBMZ_Ws(>WN0c^rQhm2Y(#{p4c9@~w4xU9e&vIEF?xWt!Pj^5J~! z#|x&S+3kdx4IL!1o%}0rA^R2KR$ioF6nD*a8MDXe<5uHK##_I&=MZB%Z!McD z1*Rg0DhCk&xedA-ma4fn-&f%Np%cUeqUWI%b4 zTE-*F)aPKBiPQ?0cQfzudvv>uM&63B5{U!sOC8E6T&)r&_S$S&D{@*rOZuvN8twGa zE90~eg~fpGYAtovwdhOGmLMa(QN(3r;iP-umV1}E)5mLgu?*U*X_zn1G@|U#I6KgJlp7^9du6&nr|RZ?=s5grLWgjDi&&jd9M&j} z`&DOdmscD7njIPx4@QP-~Hc@#m9P=>x(=l8pC#VcL-PfE~9$D1B%_xily7?F69rUA0973C#fD4w$&odRA zfpc2K=fPR!d40M$7q+wX;R50~ zADXObO5?wzyB$tdA2+eh;bVhCkV}NvAAl1bwW2=*dooM(%Y@rbFVUp^e1}GV2z}xp z8^|W>IR?|2qi-+D9tb{6ZY%D6bmpDv58tnGA7fHD!L)nIx*Sr#rQTo`;bPV;%w!X; zkwJLK9^EC2u-P2f%SI`6cg=ueXSL6^SuBg29U$RUIxRLO%sx|`9rsbo!6}W7S*8`{ z4y28kZ$_UO(6|qxe(lG>bja$Klj!lYR@@iqk!`x!-=sVJvPsm9BE1id&;T>U7)taB ze+mtHR(ER)>}Uv&>Ee=^M1K?a?1Myqk4AqVn~Wnw{{+!LsoO^-s|&V+a=mQWwZRT{ zfGm17L&uqdC+LgQ+6S8*@ErbLE~38)e;6-)ZczK$KJ91+x#KiW^qVz1N9Vd=_N;N2 zVJH3_8vh`KGTSmMP0sK$SaiOLM|U)DK_T*hc3{-v92XWKQxPTHVeCK*O+ zHc9;FH2!B{TJ20bJj8z!`9H`!+?;3hk^e`m=ucAv3Oc1&v?dfz;=h&2cn5mNPO3n+ zPQ{0;iNiQ=K;{c2oh&TiOV+>~!JHtp>D0ii05Tm&`O+wK&HOJ6pq>2hCHC9SXs<^9 zkjDQA+w@cD9!-p2VzX)W`0s}cOXxgc68^9@{>@Ih$;$pFht)^+A0hi!`2H!f{~RyC zu()Lx+22q04_b4)9$vu^3f-i&$CV=c&)^hRCi5>aF}LvwnsHDKSl+3fx``9z6%6Z@ zuTiqU^l`Je_?O_E=E?pZvcK2bgli-F3)>Xc+Y>{)gmL<^46ne%nMJa{Fi8zMscWVJ zbZGXEpz)WtFo@ULI4pJsEt9;3JnvwU{O`0}E*io49oAb29rVDG3tAbZ1_(2hQBEia z53~RSdrTo-XBh!tgqAZkmm%Ct(T(-DGie?;vT`0UKUu4a(FkRe>;0iZQ+A zBUK=Ui*sJL_7|*aq62k^4{b=Lw+~rw!JH0`S#QCdJVDU#EJNN-(9h zAkV~3-h$KuH+ZO3?=6IQ2R-1SL7l#te0Z8!+o7oiA@0*e@c8Km4<&ShHfe4D<>+@8 z*hq2ci||kl+R#u+FWXB+=m!f)k8Lu+%kj|4iqLOip)|9vX(~d&dJ8TpLIXT)dusLcFIPf_xt21fmqF4N23qYE1gQo6x{oHWU<4L+iaJnW?{=0qU?|tV68#KVRUobL zpQQ?vRXt+Y8IIek0-aSBwU^i*hBF&6Qp%LZzz_@qV1 zCqqhA&RKMM95Jjk@BQQb}e7OQjf5DCIRLwrr&4hGmkE%fp3!}>Q*qgz~%z9r)_X6RcS@Uj8^#!eIkvSpOi80;+ z=(%L8B#BQ^mUmj&kHW{ zhUZNzm!l^K%h|Y@=OVrvnK}4)$!!|vA@nm6>jS*!AyA#X=W(rOQ@m=4{36=1d1{(% zlVg*M7tjd4^XWV2z||zg-c>J^Z2+7%q;Fyrq?ZEgP0{1$wbPknvc9;<2HtZ~EXrnQR@a*c{h2wpCUDHo~mc#{kZT3XtZ8Y8z`_79-xQH*T-ubM1L|2 z5*%e7oPrIQq8rMiH7SAy=h2whlo9ag#@3;HQ&IMc<5b2ZwQ+*Cl>sZ}K#Q|fM^RH6 zumw)|f(8_oUN-vz)W{%JvK!7os$`UFe{qmw66`nucaQ-)=FpGKvWZYeX=!K&B^#Sf zTfOvA0cN6|`oadmki($JQFJ3I>gE)cGY^g|f+XirkJ#Asa?wvU^3tTAYO~&()X+XI zNhPVF6YyOb-d&E3uUVLcGHB93;pL>SlCH|jo=yNa{!TO<18@n#BuGxrf+Y)DN6V|e zMal48@CjkwWIw1fX1&aDvVEReWf4AMUbFtb-{LTuh)MNi$ipc}XEo0&wBeoRLLJ{o zf90d9wt*`<{UP3@pvnP#nZsHi#_6*JSx&<-%upxhR4H$RXK+z58mY6soqo864k~Sk zj;ognG62#X0&9-IIE-u8H3iRCL^ypiehmolV+@1wf#yzu_@DvNSze7A;4CiDHRq;sQQ9F2hBQ zGQgrvCWsA8554db0XnlFy;(OpxCp*`QBY|dT$-dmn}D6jpt8@`yx<1y&4O%9cEe7{ z&g2L>`{i%8M0e(@zS1sOi9xMrX)4()@6mjTO;oa=c458N+Z&^r6}6IisazqgT5^0VyAx{NuUEj!5S ztr`Wrn#Yqhzq_5D+)Ktbb9FyVlx)aO!$U2Qm7BpXl7WY~ewBf7ou%Wp!!mW!RfOTr zW$QJgysG3*`Bi)2k;cFud6=WJg*sgH&VIV*ASh!27*oEotTnz7geuvgm%5(j zw=SA{TgFDi!~2)F-^iQa`?bq8bF*HTy@FH&*8A?~c27+C$5E}x=C3X{OM{A6!)QqR znVZDWyd_{>#$X=PM8fo{bvK|<(PvfK2@Ac>ZFa!6H|SnzFPhA#BGeSfbfE^5HuL(s zEEGCHja)*Z4O$gDtGKf^Rtzd%GjG2O^}(Q3>oSTa=kaOr;KthyOA=b8o)uw59H7Yt zP-G8$NMB6_8-))~5izr%$>q7A8y-Y*!3Zc)@Z$`*zy@v<#Mp*ju@|H$I_xR(K|xVr zW3>_m=}r2T<%@$3llbXPfDxytJ>taTCKk+U(+w&ds@9jB>iz@=dWi-T2ga=$A~|7R zqryqm@K3_!T&`?5^j)x8?_(;b{HWCi07n<798-ktyPRa zU}gU_h=ALh4I2$Nxxl8FzCnA|UTVKo{k`*m9|kI{-e5`k)oGNjCK{K%)z2>)({6PV zWq3hxwqWcge$l;AwfWYI7LIPvUsm4w0BXY__QywHRpqMh47J`)t@q*o7vd^M<#(0~ z;Vbd5pWi!NMZn|QW6mn(b-uftfXR7*}&sg}>eEB|f5g~o~eadtW z!S{@i0mj#aPupQE`rvz#^om)K=sXCa0iNFn>a4^=N?8lxEKFo+=7H!kESc9lU{~~L z=79)3-$*SQ02^*J>{X2D(Jn7oMTNukbs2TfaA+sjgz9Kty;}}Fw4ej&)+#SQPGy*+ zBNKe!QYJ%u2O@CrQShJOgqdZ1Rx{kYpE&7Y{wi27MwEyz#w;^e2P}o${Okb_#!=@_ z(ltqZn0cTX)X)VJVd93#eXtI#)C9=}5h_Czz9FtTVN6+vscIJ_(c%S5m^?!qFSJRf zNYh}61?od19ZnE_VSwHyPDQBA3AP%(s*^YwP`)Zv)4xc3IGBhwD3{O)7d~KRh)Ft@ z5*}ABP(c9Js@LKRq!!G965Z7p2!Rq~Fa#sSz!dnPz=TI?K{L~xfU3UxEPS}sPt5Of zsRM;{5Ft=P67IUvGvwu_abTk{qPS4H1fT9QM!|*S^aF*O2xtTscCP5&(;&klzrC3u zLhPAHj!5y_XZh{tTU_M)fMWkpweL2$tUR5zQDv_xxnKCIadkskpbL;K#2}erfWCQ* zE}*c+9(foRDx+QT{1&&>2M4VP=+i#f%n-sf&2dG=1;Z2o8+6j^4uBt}H^|fD8g#cV zxTOI53W6MmnG;S`>p<;qZgP|{GK18C+MFQ2xra$eGjk9>ERr}K4Z|B{)Rk)^gdWb*D-ObG#mRPg z`d&f8A?$E=@=V=Bhqt`0kCV42Q4CDe^UbiO zIR}oDj^2T4z{87fwz|KLYR&J{E`JF9z=(3rDKh*N8iG8izlch5{wZLf2mdO5Y=Ak_ z5WMsVSaMu>1*sZ&E8l=*87%*v_dod%!s^u9!02|&+$LwDtYcf;Ijd!15*x4sFakQt-jiVT< z#ccIV(5vX7o(YENU`F7z#-B1y{VB}^k$N7O-%syW&ja;5(Dz?7TYF0LK(w9*>UrQm zlk<9KFsKUtzhOKOujhe!9;oL5+hM4+c}np>vYrR(d7z#Lp7wcQqMirpd7z#Lp7wbl zQ_lnSJW$UAPy0NOJHhbQcWP>OYU*=C--->JI!|cwglKchCm5ll_-}PNcB;p5gE}0` z#&fGWz=$4QTztAwK8Yq>++d>WY8*#>Cdzd2Vii{mQK!p3bWZ(YWM84I`;~U~qD7C~ z#P&j~rBLfqy}GGUjNwn5VC!-Wzfjp}$UddJm(fGGCzh%4CoOMru!9*u2^T>FDK~Ft z@D6sO`tq?iEgk}Gx;Vv-&Yt%S`yVZ-Mk-ahT zBJiQP_E#MVg6Ph=o{&qhY;B8oLEhTkHXRJgd%0#h55hUM)`PC%cp&Z{2^4V2ss~6) zz3D0rI29*^*;PHv9qNqZu5sD)q77_SKXdWQ>DY&-`N68`W)8}UQ?||nY{<2%_d+L| zt08qG38P1iR6R*dN1nK~lb*D^_9mG0r`ey&RJ}^_PrAcn0?l0#Mc|lqu|178FtY|n z(EV!$O^=k|3jZ~8Q$C8KaCKj+|5{hki8cGFooI}sVKR;Yf6gL*YJA0h|+43$cp%?#WD8gK>>PqS|JYN9BClPKzE(9{>% zt`RS@d6dE8Xk)_*z@ffQ&A0%B)zM5`Jl5*)JZt$`&a20VQN8Bvv2J;LtxMkG`O%>+ zXkGi*X*L}|YHL;jMgJ^H<&t`&&9mja$j*m&MTzsQYoC6V?W?#!N7#k#XCo-8u2SNu zHoOl<{i!v&M2qK$2laR(n*+`DI%+ut1y*q&Uwa)D|2WZTpJ?>)U*eaO#y2Fx4d1M~ zhveBym{A|K5`F_^b(3238k=mJ?D!5i*;jG1)7XIffcQ&!)mg`nvb)Xl*9o)ZIHuc8 zP8_A$z>slJ-ZZ`%M({*q80;Bg|3P-)N;r8mZb}^Vjf~nC?gD2X>Nh?KA~ikSatFMU z;XdRyGA79);s`g*2AB9E?mwfrQGGX{`xd`Aam{LF_cx&4S>hen$zDPT&yk)dkabL# z#livnbQ4d|Yu-U_kvl-Gd%4Tk=8!{?K_q(|Jo;Xay(cYShP-Y)7uB0VH0O49TAfwj zJHLgh%}uvK%j17oJ-?FpG>mVXP~XQXbv;^%oa=~JT*yY*(HUn~_sB!M;&qukOnmUy zj#!QE6Ag#(0aeU8!se>!D7ac}s8oi%>w;@VO^}MvfkRU#J2xZvTBOx=a0(xqx$1Qn zGxv|J`8v|_%)gZP#mUKZBJ9S6Cd?*eA6r@jY(t6{Oomutfh{-Ux)uX|W=V~YWfW7$24eH+h82d@$-%`M- zOFTU6><2dJ{(hTo=!V%8jN;*vz`(wHFLUsxHrP|y{P)3}cHrvMqaJnQ=#bRi?kRfE8Su1h_=>kf zBYmhJKbTHBQS+i>oQyR_w#w+PZ;@Vf5hnogduY~OZ`t|{<0KhXhnP{^vSjN!j}Oos zJ*EwpEDs)8J~ml9bo)Dk6GV)i)+7#eX&BEOU1LezARKgvjksa?)eEZ|j{xxoiQ^44 zh7(o>SBL`p-(~d*kUijLn9w$L2??w7)eubQDBf4%I43*7;u&BARe4}8WiwbDgL>GK z9Z=7pBq%D)_FfJQC2k=GZ_usYL4NjTJ8`+{#p6o648=1fg@@2Iu5(2^=fp)ww!_6e zqD`-U$)<1=e?#+<BU@e z5D4HZ(S^HA1mCI{u2o5Rvow5J4!c#%=xo1!`*ioZbv z?l-M)%t5$iak_}A)4?ccdQ$x^X4UWDFo@ZBJ!X&en2q4U5mTSGab`19>dsVBA3+Dc zo{e}ki5o-bj&5~n60e>SygZNKm`&DW_Hc^XR-DAdEk=ApqIh8@aFP;!CChAb7AL6% zJdT~}aO_h*+AejIdYZ-TL_KDote7=$bagYA@G{pIXPa*3`2F})h^vwK+ho+&s7SxM zh|`NpU9jpg+jig{EK`r!Co5*fukCNC2b=Mi)Wgl8evV%CdF{l>trxHFLG|$(!#6mi zPF_Wv9T&M#;o@3LvpW3+@iz(M`x;f(*HK&^C-MBt;}SfF_p1$uU=I#&KKwsB@c!$; z-FpCU?_qU&OydQb!`r)L`W@xWm^JVK6h}}mZg1`C^d(M!gZLkb_oFzzO*307GE-Y* z^5tTN*34CkHaOCd`T|CoxhG&sCtymmICjq}Q!4IX9wuo%v{@bM{20M8NqoSP@TTJb zm{TXXdFI{@reTdty8ZB>o#?H4VL=DsKS$s_#SL;A->M?h>qXqFT+FYV)fY0To^XB4 zu!nH87w^DHi{YGOf^B1R?NJxOfVzhEsBdsg9V6?!!Q?^m28T_|&K~J8Tj#o-Dz59W ziP`dz9eF1pug$>C+pQkGeiT2QXnlIQUNFe*f)Oqkr0`~( z=4wF^-}c2VcINC2xHh-q+8)HSH;hIniW^%3Pr?Zls#!eTXK`s<;6{N{J)C{$R6Fo= z?NRsM7+TaMu4`%LF*)WiCH&S5=H70+*1YQZ-HCf+FCK=2xOR>(=T70}JdHlI$Ze3t zZFaPv4S2k^s>5pt_wPP#5DcO39L4v05|_g~K8JJM1hJvp^l%Tvhl6K_I{ZfP@QkUO z<2W}!rnmqygOlgH-m>x>fS)~VV%Bi$Ou=|sVS~T*n0<0)3WrV1I_oiec<0%}XAXAQ z#B4)7W}mv4J+f!FhfU0S-G;-u8FOfPgQ6;|WV@?LugbK+8-#R|D~94g_P5e_h|AVi ziQ6IudU;XNdQlv-!W{%Qw^j92UEK9DHC4T;sw(Imukz*)e($wz@21y#rg~w{sdwd? zz>(eTI7ReAfZRx3Xw7bm`dh7wOMS2@8SW^|rkj}^VW6R zA|Cp4xae0j`!-eWyQ-J&1>m%GH=tq4P)hn+U!eX4>R+J#1?pd*{srn^p#BBwU!eX4>R+J#1?pd*{srn^ zp#BBwU!eX4>R+J#1?pd*{srn^p#BBwU!eX4>R+J#1?pd*{srn^p#BBwU!eX4>R;f< H{sR9Wkp*RD literal 0 HcmV?d00001 diff --git a/dependencies/poppler/bin/poppler-qt4.dll.manifest b/dependencies/poppler/bin/poppler-qt4.dll.manifest new file mode 100644 index 00000000..e693382a --- /dev/null +++ b/dependencies/poppler/bin/poppler-qt4.dll.manifest @@ -0,0 +1,10 @@ + + + + + + + + + + \ No newline at end of file diff --git a/dependencies/poppler/bin/poppler-qt5.dll b/dependencies/poppler/bin/poppler-qt5.dll new file mode 100644 index 0000000000000000000000000000000000000000..675d8ad18635de134a03360570d3750af08bc2f2 GIT binary patch literal 1687552 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

hJ5OXu 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

bg2*9$7cse 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

>RK1&;1Yb)AZ|%0t&!C>p#d(6u9#_mA~LW$WMz@ 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 z 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