From 354c69bb5a76de258bc3a21d840d55d77048c899 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luis=20=C3=81ngel=20San=20Mart=C3=ADn?= Date: Thu, 5 Feb 2015 19:50:50 +0100 Subject: [PATCH] fixed compilation error in GCC 4.9 --- 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.1 | 50 + YACReader.desktop | 13 + YACReader.pro | 3 + YACReader/YACReader.icns | Bin 0 -> 133692 bytes YACReader/YACReader.pri | 150 + YACReader/YACReader.pro | 107 + YACReader/bookmarks_dialog.cpp | 197 ++ YACReader/bookmarks_dialog.h | 45 + YACReader/configuration.cpp | 58 + YACReader/configuration.h | 97 + 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 | 167 + YACReader/main_window_viewer.cpp | 1407 +++++++++ YACReader/main_window_viewer.h | 167 + 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 | 1177 +++++++ YACReader/render.h | 215 ++ YACReader/shortcuts_dialog.cpp | 55 + YACReader/shortcuts_dialog.h | 19 + YACReader/translator.cpp | 429 +++ YACReader/translator.h | 102 + YACReader/viewer.cpp | 985 ++++++ YACReader/viewer.h | 168 + YACReader/width_slider.cpp | 94 + YACReader/width_slider.h | 48 + YACReader/yacreader_de.ts | 791 +++++ 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 | 86 + YACReader/yacreader_images_osx.qrc | 57 + YACReader/yacreader_images_win.qrc | 29 + 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.1 | 48 + YACReaderLibrary.desktop | 13 + YACReaderLibrary/YACReaderLibrary.icns | Bin 0 -> 142716 bytes YACReaderLibrary/YACReaderLibrary.pro | 304 ++ YACReaderLibrary/add_label_dialog.cpp | 84 + YACReaderLibrary/add_label_dialog.h | 31 + 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 | 324 ++ YACReaderLibrary/classic_comics_view.h | 69 + YACReaderLibrary/comic_files_manager.cpp | 108 + YACReaderLibrary/comic_files_manager.h | 37 + YACReaderLibrary/comic_flow.cpp | 264 ++ YACReaderLibrary/comic_flow.h | 78 + YACReaderLibrary/comic_flow_widget.cpp | 355 +++ YACReaderLibrary/comic_flow_widget.h | 131 + .../comic_vine/api_key_dialog.cpp | 68 + YACReaderLibrary/comic_vine/api_key_dialog.h | 31 + YACReaderLibrary/comic_vine/comic_vine.pri | 46 + .../comic_vine/comic_vine_client.cpp | 171 ++ .../comic_vine/comic_vine_client.h | 46 + .../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 | 83 + .../comic_vine/model/response_parser.h | 30 + .../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 | 63 + YACReaderLibrary/comics_remover.h | 47 + YACReaderLibrary/comics_view.cpp | 69 + YACReaderLibrary/comics_view.h | 56 + YACReaderLibrary/comics_view_transition.cpp | 85 + YACReaderLibrary/comics_view_transition.h | 31 + YACReaderLibrary/create_library_dialog.cpp | 206 ++ YACReaderLibrary/create_library_dialog.h | 61 + YACReaderLibrary/db/comic_item.cpp | 47 + YACReaderLibrary/db/comic_item.h | 27 + YACReaderLibrary/db/comic_model.cpp | 1139 +++++++ YACReaderLibrary/db/comic_model.h | 166 + YACReaderLibrary/db/data_base_management.cpp | 790 +++++ YACReaderLibrary/db/data_base_management.h | 62 + YACReaderLibrary/db/folder_item.cpp | 103 + YACReaderLibrary/db/folder_item.h | 77 + YACReaderLibrary/db/folder_model.cpp | 784 +++++ YACReaderLibrary/db/folder_model.h | 151 + YACReaderLibrary/db/reading_list_item.cpp | 239 ++ YACReaderLibrary/db/reading_list_item.h | 103 + YACReaderLibrary/db/reading_list_model.cpp | 778 +++++ YACReaderLibrary/db/reading_list_model.h | 117 + YACReaderLibrary/db_helper.cpp | 1000 ++++++ YACReaderLibrary/db_helper.h | 78 + YACReaderLibrary/empty_container_info.cpp | 47 + YACReaderLibrary/empty_container_info.h | 26 + YACReaderLibrary/empty_folder_widget.cpp | 154 + YACReaderLibrary/empty_folder_widget.h | 36 + YACReaderLibrary/empty_label_widget.cpp | 21 + YACReaderLibrary/empty_label_widget.h | 22 + .../empty_reading_list_widget.cpp | 9 + YACReaderLibrary/empty_reading_list_widget.h | 13 + YACReaderLibrary/empty_special_list.cpp | 7 + YACReaderLibrary/empty_special_list.h | 13 + .../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 | 340 ++ YACReaderLibrary/grid_comics_view.h | 79 + 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 | 99 + YACReaderLibrary/images_osx.qrc | 80 + YACReaderLibrary/images_win.qrc | 47 + .../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 | 386 +++ YACReaderLibrary/import_widget.h | 52 + YACReaderLibrary/library_creator.cpp | 694 +++++ YACReaderLibrary/library_creator.h | 94 + YACReaderLibrary/library_window.cpp | 2733 +++++++++++++++++ YACReaderLibrary/library_window.h | 410 +++ YACReaderLibrary/main.cpp | 253 ++ YACReaderLibrary/no_libraries_widget.cpp | 80 + YACReaderLibrary/no_libraries_widget.h | 19 + YACReaderLibrary/no_search_results_widget.cpp | 51 + YACReaderLibrary/no_search_results_widget.h | 26 + YACReaderLibrary/options_dialog.cpp | 86 + YACReaderLibrary/options_dialog.h | 21 + YACReaderLibrary/package_manager.cpp | 55 + YACReaderLibrary/package_manager.h | 24 + YACReaderLibrary/properties_dialog.cpp | 896 ++++++ YACReaderLibrary/properties_dialog.h | 141 + YACReaderLibrary/qml.qrc | 9 + YACReaderLibrary/qml/GridComicsView.qml | 437 +++ 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/star_menu.png | Bin 0 -> 277 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 | 119 + .../server/controllers/comiccontroller.h | 23 + .../comicdownloadinfocontroller.cpp | 24 + .../controllers/comicdownloadinfocontroller.h | 19 + .../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 | 315 ++ .../server/controllers/foldercontroller.h | 20 + .../controllers/folderinfocontroller.cpp | 48 + .../server/controllers/folderinfocontroller.h | 23 + .../server/controllers/formcontroller.cpp | 64 + .../server/controllers/formcontroller.h | 30 + .../controllers/librariescontroller.cpp | 40 + .../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 | 381 +++ .../server/lib/bfHttpServer/httpsession.h | 193 ++ .../lib/bfHttpServer/httpsessionstore.cpp | 107 + .../lib/bfHttpServer/httpsessionstore.h | 104 + .../lib/bfHttpServer/staticfilecontroller.cpp | 235 ++ .../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 | 171 ++ YACReaderLibrary/server/requestmapper.h | 37 + YACReaderLibrary/server/server.pri | 36 + 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 | 326 ++ YACReaderLibrary/server_config_dialog.h | 44 + YACReaderLibrary/yacreader_folders_view.cpp | 104 + YACReaderLibrary/yacreader_folders_view.h | 36 + .../yacreader_history_controller.cpp | 108 + .../yacreader_history_controller.h | 62 + 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 | 151 + YACReaderLibrary/yacreader_main_toolbar.h | 51 + .../yacreader_navigation_controller.cpp | 304 ++ .../yacreader_navigation_controller.h | 53 + .../yacreader_reading_lists_view.cpp | 72 + .../yacreader_reading_lists_view.h | 32 + YACReaderLibrary/yacreaderlibrary_de.ts | 1520 +++++++++ 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 | 799 +++++ common/comic.h | 182 ++ common/comic_db.cpp | 483 +++ common/comic_db.h | 157 + common/custom_widgets.cpp | 1 + common/custom_widgets.h | 6 + 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 | 1409 +++++++++ common/pictureflow.h | 228 ++ common/qnaturalsorting.cpp | 262 ++ common/qnaturalsorting.h | 15 + common/scroll_management.cpp | 61 + common/scroll_management.h | 25 + common/yacreader_flow_gl.cpp | 1628 ++++++++++ common/yacreader_flow_gl.h | 383 +++ common/yacreader_global.cpp | 141 + common/yacreader_global.h | 145 + compileOSX.sh | 47 + 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 | 34 + .../custom_widgets_yacreaderlibrary.pri | 48 + 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 | 156 + .../yacreader_library_item_widget.h | 45 + .../yacreader_library_list_widget.cpp | 128 + .../yacreader_library_list_widget.h | 37 + custom_widgets/yacreader_macosx_toolbar.h | 87 + custom_widgets/yacreader_macosx_toolbar.mm | 395 +++ custom_widgets/yacreader_options_dialog.cpp | 383 +++ custom_widgets/yacreader_options_dialog.h | 65 + custom_widgets/yacreader_search_line_edit.cpp | 146 + custom_widgets/yacreader_search_line_edit.h | 40 + custom_widgets/yacreader_sidebar.cpp | 195 ++ custom_widgets/yacreader_sidebar.h | 47 + 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 | 487 +++ custom_widgets/yacreader_table_view.h | 132 + custom_widgets/yacreader_titled_toolbar.cpp | 130 + custom_widgets/yacreader_titled_toolbar.h | 46 + custom_widgets/yacreader_tool_bar_stretch.cpp | 0 custom_widgets/yacreader_tool_bar_stretch.h | 18 + custom_widgets/yacreader_treeview.cpp | 154 + custom_widgets/yacreader_treeview.h | 29 + 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 | 102 + files/about_es_ES.html | 101 + 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/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/busy_background.png | Bin 0 -> 327 bytes images/center.png | Bin 0 -> 17966 bytes images/clearSearch.png | Bin 0 -> 1225 bytes images/clearSearchNew.png | Bin 0 -> 235 bytes images/clear_shortcut.png | Bin 0 -> 200 bytes images/close.png | Bin 0 -> 215 bytes images/comicFolder.png | Bin 0 -> 26436 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_current_readings.png | Bin 0 -> 1550 bytes images/empty_favorites.png | Bin 0 -> 2839 bytes images/empty_folder.png | Bin 0 -> 2515 bytes images/empty_folder_osx.png | Bin 0 -> 2446 bytes images/empty_label.png | Bin 0 -> 2046 bytes images/empty_reading_list.png | Bin 0 -> 2471 bytes images/empty_reading_list_osx.png | Bin 0 -> 2346 bytes images/empty_search.png | Bin 0 -> 4498 bytes images/empty_search_osx.png | Bin 0 -> 4212 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_overlayed.png | Bin 0 -> 597 bytes images/f_overlayed_retina.png | Bin 0 -> 1224 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/flow_to_grid_osx.gif | Bin 0 -> 149283 bytes images/folder_finished_macosx.png | Bin 0 -> 158 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 -> 908 bytes images/grid_to_flow.gif | Bin 0 -> 129647 bytes images/grid_to_flow_osx.gif | Bin 0 -> 150853 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/iconSearchNew.png | Bin 0 -> 382 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/lists/default_0.png | Bin 0 -> 233 bytes images/lists/default_0_osx.png | Bin 0 -> 242 bytes images/lists/default_0_osx@2x.png | Bin 0 -> 413 bytes images/lists/default_1.png | Bin 0 -> 383 bytes images/lists/default_1_osx.png | Bin 0 -> 384 bytes images/lists/default_1_osx@2x.png | Bin 0 -> 577 bytes images/lists/label_blue.png | Bin 0 -> 253 bytes images/lists/label_blue_osx.png | Bin 0 -> 253 bytes images/lists/label_blue_osx@2x.png | Bin 0 -> 410 bytes images/lists/label_cyan.png | Bin 0 -> 250 bytes images/lists/label_cyan_osx.png | Bin 0 -> 250 bytes images/lists/label_cyan_osx@2x.png | Bin 0 -> 419 bytes images/lists/label_dark.png | Bin 0 -> 243 bytes images/lists/label_dark_osx.png | Bin 0 -> 243 bytes images/lists/label_dark_osx@2x.png | Bin 0 -> 406 bytes images/lists/label_green.png | Bin 0 -> 254 bytes images/lists/label_green_osx.png | Bin 0 -> 254 bytes images/lists/label_green_osx@2x.png | Bin 0 -> 408 bytes images/lists/label_light.png | Bin 0 -> 244 bytes images/lists/label_light_osx.png | Bin 0 -> 244 bytes images/lists/label_light_osx@2x.png | Bin 0 -> 408 bytes images/lists/label_orange.png | Bin 0 -> 249 bytes images/lists/label_orange_osx.png | Bin 0 -> 249 bytes images/lists/label_orange_osx@2x.png | Bin 0 -> 412 bytes images/lists/label_pink.png | Bin 0 -> 248 bytes images/lists/label_pink_osx.png | Bin 0 -> 248 bytes images/lists/label_pink_osx@2x.png | Bin 0 -> 419 bytes images/lists/label_purple.png | Bin 0 -> 259 bytes images/lists/label_purple_osx.png | Bin 0 -> 259 bytes images/lists/label_purple_osx@2x.png | Bin 0 -> 417 bytes images/lists/label_red.png | Bin 0 -> 243 bytes images/lists/label_red_osx.png | Bin 0 -> 243 bytes images/lists/label_red_osx@2x.png | Bin 0 -> 408 bytes images/lists/label_violet.png | Bin 0 -> 244 bytes images/lists/label_violet_osx.png | Bin 0 -> 244 bytes images/lists/label_violet_osx@2x.png | Bin 0 -> 410 bytes images/lists/label_white.png | Bin 0 -> 207 bytes images/lists/label_white_osx.png | Bin 0 -> 207 bytes images/lists/label_white_osx@2x.png | Bin 0 -> 358 bytes images/lists/label_yellow.png | Bin 0 -> 245 bytes images/lists/label_yellow_osx.png | Bin 0 -> 245 bytes images/lists/label_yellow_osx@2x.png | Bin 0 -> 421 bytes images/lists/list.png | Bin 0 -> 184 bytes images/lists/list_osx.png | Bin 0 -> 192 bytes images/lists/list_osx@2x.png | Bin 0 -> 222 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 -> 350 bytes images/main_toolbar/back_osx.png | Bin 0 -> 349 bytes images/main_toolbar/back_osx@2x.png | Bin 0 -> 1578 bytes images/main_toolbar/divider.png | Bin 0 -> 207 bytes images/main_toolbar/flow.png | Bin 0 -> 184 bytes images/main_toolbar/flow_osx.png | Bin 0 -> 1116 bytes images/main_toolbar/flow_osx@2x.png | Bin 0 -> 1537 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 -> 380 bytes images/main_toolbar/forward_osx.png | Bin 0 -> 345 bytes images/main_toolbar/forward_osx@2x.png | Bin 0 -> 1610 bytes images/main_toolbar/fullscreen.png | Bin 0 -> 259 bytes images/main_toolbar/fullscreen_osx.png | Bin 0 -> 563 bytes images/main_toolbar/grid.png | Bin 0 -> 179 bytes images/main_toolbar/grid_osx.png | Bin 0 -> 1050 bytes images/main_toolbar/grid_osx@2x.png | Bin 0 -> 1480 bytes images/main_toolbar/help.png | Bin 0 -> 384 bytes images/main_toolbar/help_osx.png | Bin 0 -> 536 bytes images/main_toolbar/help_osx@2x.png | Bin 0 -> 1972 bytes images/main_toolbar/server.png | Bin 0 -> 196 bytes images/main_toolbar/server_osx.png | Bin 0 -> 376 bytes images/main_toolbar/server_osx@2x.png | Bin 0 -> 1687 bytes images/main_toolbar/settings.png | Bin 0 -> 369 bytes images/main_toolbar/settings_osx.png | Bin 0 -> 781 bytes images/main_toolbar/settings_osx@2x.png | Bin 0 -> 2519 bytes images/new.png | Bin 0 -> 454 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/options.png | Bin 0 -> 20699 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/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/searching_icon.png | Bin 0 -> 1788 bytes images/selectAll.png | Bin 0 -> 216 bytes images/server.png | Bin 0 -> 14333 bytes images/serverConfigBackground.png | Bin 0 -> 7912 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/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/sidebar/addLabelIcon.png | Bin 0 -> 258 bytes images/sidebar/addLabelIcon_osx.png | Bin 0 -> 242 bytes images/sidebar/addLabelIcon_osx@2x.png | Bin 0 -> 364 bytes images/sidebar/addNew_sidebar.png | Bin 0 -> 218 bytes images/sidebar/addNew_sidebar_osx.png | Bin 0 -> 171 bytes images/sidebar/addNew_sidebar_osx@2x.png | Bin 0 -> 219 bytes images/sidebar/branch-closed.png | Bin 0 -> 156 bytes images/sidebar/branch-open.png | Bin 0 -> 145 bytes images/sidebar/colapse.png | Bin 0 -> 253 bytes images/sidebar/colapse_osx.png | Bin 0 -> 207 bytes images/sidebar/colapse_osx@2x.png | Bin 0 -> 298 bytes images/sidebar/collapsed_branch_osx.png | Bin 0 -> 162 bytes images/sidebar/collapsed_branch_selected.png | Bin 0 -> 139 bytes images/sidebar/delete_sidebar.png | Bin 0 -> 229 bytes images/sidebar/delete_sidebar_osx.png | Bin 0 -> 187 bytes images/sidebar/delete_sidebar_osx@2x.png | Bin 0 -> 202 bytes images/sidebar/expand.png | Bin 0 -> 158 bytes images/sidebar/expand_osx.png | Bin 0 -> 168 bytes images/sidebar/expand_osx@2x.png | Bin 0 -> 208 bytes images/sidebar/expanded_branch_osx.png | Bin 0 -> 169 bytes images/sidebar/expanded_branch_selected.png | Bin 0 -> 133 bytes images/sidebar/folder.png | Bin 0 -> 313 bytes images/sidebar/folder_finished.png | Bin 0 -> 386 bytes images/sidebar/libraryIcon.png | Bin 0 -> 280 bytes images/sidebar/libraryIconSelected.png | Bin 0 -> 152 bytes images/sidebar/libraryIcon_osx.png | Bin 0 -> 258 bytes images/sidebar/libraryOptions.png | Bin 0 -> 206 bytes images/sidebar/newLibraryIcon.png | Bin 0 -> 212 bytes images/sidebar/newLibraryIcon_osx.png | Bin 0 -> 169 bytes images/sidebar/newLibraryIcon_osx@2x.png | Bin 0 -> 219 bytes images/sidebar/openLibraryIcon.png | Bin 0 -> 298 bytes images/sidebar/openLibraryIcon_osx.png | Bin 0 -> 232 bytes images/sidebar/openLibraryIcon_osx@2x.png | Bin 0 -> 335 bytes images/sidebar/renameListIcon.png | Bin 0 -> 315 bytes images/sidebar/renameListIcon_osx.png | Bin 0 -> 254 bytes images/sidebar/renameListIcon_osx@2x.png | Bin 0 -> 364 bytes images/sidebar/setRoot.png | Bin 0 -> 389 bytes images/sidebar/setRoot_osx.png | Bin 0 -> 271 bytes images/sidebar/setRoot_osx@2x.png | Bin 0 -> 421 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/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 -> 348 bytes images/viewer_toolbar/bookmark_osx@2x.png | Bin 0 -> 1710 bytes images/viewer_toolbar/close.png | Bin 0 -> 272 bytes images/viewer_toolbar/close_osx.png | Bin 0 -> 685 bytes images/viewer_toolbar/close_osx@2x.png | Bin 0 -> 2272 bytes images/viewer_toolbar/doubleMangaPage.png | Bin 0 -> 200 bytes images/viewer_toolbar/doubleMangaPage_osx.png | Bin 0 -> 1407 bytes .../viewer_toolbar/doubleMangaPage_osx@2x.png | Bin 0 -> 1954 bytes images/viewer_toolbar/doublePage.png | Bin 0 -> 149 bytes images/viewer_toolbar/doublePage_osx.png | Bin 0 -> 290 bytes images/viewer_toolbar/doublePage_osx@2x.png | Bin 0 -> 1536 bytes images/viewer_toolbar/flow.png | Bin 0 -> 153 bytes images/viewer_toolbar/flow_osx.png | Bin 0 -> 239 bytes images/viewer_toolbar/flow_osx@2x.png | Bin 0 -> 1411 bytes images/viewer_toolbar/full.png | Bin 0 -> 204 bytes images/viewer_toolbar/full_osx.png | Bin 0 -> 646 bytes images/viewer_toolbar/full_osx@2x.png | Bin 0 -> 2128 bytes images/viewer_toolbar/goto.png | Bin 0 -> 1118 bytes images/viewer_toolbar/goto_osx.png | Bin 0 -> 597 bytes images/viewer_toolbar/goto_osx@2x.png | Bin 0 -> 2025 bytes images/viewer_toolbar/help.png | Bin 0 -> 287 bytes images/viewer_toolbar/help_osx.png | Bin 0 -> 560 bytes images/viewer_toolbar/help_osx@2x.png | Bin 0 -> 2036 bytes images/viewer_toolbar/info.png | Bin 0 -> 225 bytes images/viewer_toolbar/info_osx.png | Bin 0 -> 422 bytes images/viewer_toolbar/info_osx@2x.png | Bin 0 -> 1712 bytes images/viewer_toolbar/magnifyingGlass.png | Bin 0 -> 346 bytes images/viewer_toolbar/magnifyingGlass_osx.png | Bin 0 -> 705 bytes .../viewer_toolbar/magnifyingGlass_osx@2x.png | Bin 0 -> 2391 bytes images/viewer_toolbar/next.png | Bin 0 -> 194 bytes images/viewer_toolbar/next_osx.png | Bin 0 -> 358 bytes images/viewer_toolbar/next_osx@2x.png | Bin 0 -> 1614 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 -> 280 bytes images/viewer_toolbar/openFolder_osx@2x.png | Bin 0 -> 1464 bytes images/viewer_toolbar/openNext.png | Bin 0 -> 249 bytes images/viewer_toolbar/openNext_osx.png | Bin 0 -> 608 bytes images/viewer_toolbar/openNext_osx@2x.png | Bin 0 -> 2098 bytes images/viewer_toolbar/openPrevious.png | Bin 0 -> 230 bytes images/viewer_toolbar/openPrevious_osx.png | Bin 0 -> 614 bytes images/viewer_toolbar/openPrevious_osx@2x.png | Bin 0 -> 2065 bytes images/viewer_toolbar/open_osx.png | Bin 0 -> 661 bytes images/viewer_toolbar/open_osx@2x.png | Bin 0 -> 2162 bytes images/viewer_toolbar/options.png | Bin 0 -> 331 bytes images/viewer_toolbar/options_osx.png | Bin 0 -> 864 bytes images/viewer_toolbar/options_osx@2x.png | Bin 0 -> 2688 bytes images/viewer_toolbar/previous.png | Bin 0 -> 186 bytes images/viewer_toolbar/previous_osx.png | Bin 0 -> 352 bytes images/viewer_toolbar/previous_osx@2x.png | Bin 0 -> 1582 bytes images/viewer_toolbar/rotateL.png | Bin 0 -> 340 bytes images/viewer_toolbar/rotateL_osx.png | Bin 0 -> 797 bytes images/viewer_toolbar/rotateL_osx@2x.png | Bin 0 -> 2540 bytes images/viewer_toolbar/rotateR.png | Bin 0 -> 344 bytes images/viewer_toolbar/rotateR_osx.png | Bin 0 -> 827 bytes images/viewer_toolbar/rotateR_osx@2x.png | Bin 0 -> 2568 bytes images/viewer_toolbar/save.png | Bin 0 -> 208 bytes images/viewer_toolbar/save_osx.png | Bin 0 -> 468 bytes images/viewer_toolbar/save_osx@2x.png | Bin 0 -> 1849 bytes images/viewer_toolbar/shortcuts.png | Bin 0 -> 284 bytes images/viewer_toolbar/shortcuts_osx.png | Bin 0 -> 531 bytes images/viewer_toolbar/shortcuts_osx@2x.png | Bin 0 -> 2012 bytes images/viewer_toolbar/showBookmarks.png | Bin 0 -> 181 bytes images/viewer_toolbar/showBookmarks_osx.png | Bin 0 -> 347 bytes .../viewer_toolbar/showBookmarks_osx@2x.png | Bin 0 -> 1615 bytes images/viewer_toolbar/toHeight.png | Bin 0 -> 213 bytes images/viewer_toolbar/toHeight_osx.png | Bin 0 -> 401 bytes images/viewer_toolbar/toHeight_osx@2x.png | Bin 0 -> 1712 bytes images/viewer_toolbar/toWidth.png | Bin 0 -> 218 bytes images/viewer_toolbar/toWidthSlider_osx.png | Bin 0 -> 1321 bytes .../viewer_toolbar/toWidthSlider_osx@2x.png | Bin 0 -> 1793 bytes images/viewer_toolbar/toWidth_osx.png | Bin 0 -> 379 bytes images/viewer_toolbar/toWidth_osx@2x.png | Bin 0 -> 1702 bytes images/viewer_toolbar/translator.png | Bin 0 -> 233 bytes images/viewer_toolbar/translator_osx.png | Bin 0 -> 526 bytes images/viewer_toolbar/translator_osx@2x.png | Bin 0 -> 2024 bytes images/zoom.png | Bin 0 -> 17891 bytes mktarball.sh | 22 + release/languages/yacreader_de.qm | Bin 0 -> 13822 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_de.qm | Bin 0 -> 34624 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 | 126 + shortcuts_management/shortcuts_manager.h | 136 + 942 files changed, 82928 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.1 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_de.ts 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.1 create mode 100644 YACReaderLibrary.desktop create mode 100644 YACReaderLibrary/YACReaderLibrary.icns create mode 100644 YACReaderLibrary/YACReaderLibrary.pro create mode 100644 YACReaderLibrary/add_label_dialog.cpp create mode 100644 YACReaderLibrary/add_label_dialog.h 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_files_manager.cpp create mode 100644 YACReaderLibrary/comic_files_manager.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/api_key_dialog.cpp create mode 100644 YACReaderLibrary/comic_vine/api_key_dialog.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/comic_item.cpp create mode 100644 YACReaderLibrary/db/comic_item.h create mode 100644 YACReaderLibrary/db/comic_model.cpp create mode 100644 YACReaderLibrary/db/comic_model.h create mode 100644 YACReaderLibrary/db/data_base_management.cpp create mode 100644 YACReaderLibrary/db/data_base_management.h create mode 100644 YACReaderLibrary/db/folder_item.cpp create mode 100644 YACReaderLibrary/db/folder_item.h create mode 100644 YACReaderLibrary/db/folder_model.cpp create mode 100644 YACReaderLibrary/db/folder_model.h create mode 100644 YACReaderLibrary/db/reading_list_item.cpp create mode 100644 YACReaderLibrary/db/reading_list_item.h create mode 100644 YACReaderLibrary/db/reading_list_model.cpp create mode 100644 YACReaderLibrary/db/reading_list_model.h create mode 100644 YACReaderLibrary/db_helper.cpp create mode 100644 YACReaderLibrary/db_helper.h create mode 100644 YACReaderLibrary/empty_container_info.cpp create mode 100644 YACReaderLibrary/empty_container_info.h create mode 100644 YACReaderLibrary/empty_folder_widget.cpp create mode 100644 YACReaderLibrary/empty_folder_widget.h create mode 100644 YACReaderLibrary/empty_label_widget.cpp create mode 100644 YACReaderLibrary/empty_label_widget.h create mode 100644 YACReaderLibrary/empty_reading_list_widget.cpp create mode 100644 YACReaderLibrary/empty_reading_list_widget.h create mode 100644 YACReaderLibrary/empty_special_list.cpp create mode 100644 YACReaderLibrary/empty_special_list.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/no_search_results_widget.cpp create mode 100644 YACReaderLibrary/no_search_results_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/star_menu.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/comicdownloadinfocontroller.cpp create mode 100644 YACReaderLibrary/server/controllers/comicdownloadinfocontroller.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_folders_view.cpp create mode 100644 YACReaderLibrary/yacreader_folders_view.h create mode 100644 YACReaderLibrary/yacreader_history_controller.cpp create mode 100644 YACReaderLibrary/yacreader_history_controller.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/yacreader_navigation_controller.cpp create mode 100644 YACReaderLibrary/yacreader_navigation_controller.h create mode 100644 YACReaderLibrary/yacreader_reading_lists_view.cpp create mode 100644 YACReaderLibrary/yacreader_reading_lists_view.h create mode 100644 YACReaderLibrary/yacreaderlibrary_de.ts 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/scroll_management.cpp create mode 100644 common/scroll_management.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 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_macosx_toolbar.h create mode 100644 custom_widgets/yacreader_macosx_toolbar.mm 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/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/busy_background.png create mode 100644 images/center.png create mode 100644 images/clearSearch.png create mode 100644 images/clearSearchNew.png create mode 100644 images/clear_shortcut.png create mode 100644 images/close.png create mode 100644 images/comicFolder.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_current_readings.png create mode 100644 images/empty_favorites.png create mode 100644 images/empty_folder.png create mode 100644 images/empty_folder_osx.png create mode 100644 images/empty_label.png create mode 100644 images/empty_reading_list.png create mode 100644 images/empty_reading_list_osx.png create mode 100644 images/empty_search.png create mode 100644 images/empty_search_osx.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_overlayed.png create mode 100644 images/f_overlayed_retina.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/flow_to_grid_osx.gif create mode 100644 images/folder_finished_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/grid_to_flow_osx.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/iconSearchNew.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/lists/default_0.png create mode 100644 images/lists/default_0_osx.png create mode 100644 images/lists/default_0_osx@2x.png create mode 100644 images/lists/default_1.png create mode 100644 images/lists/default_1_osx.png create mode 100644 images/lists/default_1_osx@2x.png create mode 100644 images/lists/label_blue.png create mode 100644 images/lists/label_blue_osx.png create mode 100644 images/lists/label_blue_osx@2x.png create mode 100644 images/lists/label_cyan.png create mode 100644 images/lists/label_cyan_osx.png create mode 100644 images/lists/label_cyan_osx@2x.png create mode 100644 images/lists/label_dark.png create mode 100644 images/lists/label_dark_osx.png create mode 100644 images/lists/label_dark_osx@2x.png create mode 100644 images/lists/label_green.png create mode 100644 images/lists/label_green_osx.png create mode 100644 images/lists/label_green_osx@2x.png create mode 100644 images/lists/label_light.png create mode 100644 images/lists/label_light_osx.png create mode 100644 images/lists/label_light_osx@2x.png create mode 100644 images/lists/label_orange.png create mode 100644 images/lists/label_orange_osx.png create mode 100644 images/lists/label_orange_osx@2x.png create mode 100644 images/lists/label_pink.png create mode 100644 images/lists/label_pink_osx.png create mode 100644 images/lists/label_pink_osx@2x.png create mode 100644 images/lists/label_purple.png create mode 100644 images/lists/label_purple_osx.png create mode 100644 images/lists/label_purple_osx@2x.png create mode 100644 images/lists/label_red.png create mode 100644 images/lists/label_red_osx.png create mode 100644 images/lists/label_red_osx@2x.png create mode 100644 images/lists/label_violet.png create mode 100644 images/lists/label_violet_osx.png create mode 100644 images/lists/label_violet_osx@2x.png create mode 100644 images/lists/label_white.png create mode 100644 images/lists/label_white_osx.png create mode 100644 images/lists/label_white_osx@2x.png create mode 100644 images/lists/label_yellow.png create mode 100644 images/lists/label_yellow_osx.png create mode 100644 images/lists/label_yellow_osx@2x.png create mode 100644 images/lists/list.png create mode 100644 images/lists/list_osx.png create mode 100644 images/lists/list_osx@2x.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/back_osx@2x.png create mode 100644 images/main_toolbar/divider.png create mode 100644 images/main_toolbar/flow.png create mode 100644 images/main_toolbar/flow_osx.png create mode 100644 images/main_toolbar/flow_osx@2x.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/forward_osx@2x.png create mode 100644 images/main_toolbar/fullscreen.png create mode 100644 images/main_toolbar/fullscreen_osx.png create mode 100644 images/main_toolbar/grid.png create mode 100644 images/main_toolbar/grid_osx.png create mode 100644 images/main_toolbar/grid_osx@2x.png create mode 100644 images/main_toolbar/help.png create mode 100644 images/main_toolbar/help_osx.png create mode 100644 images/main_toolbar/help_osx@2x.png create mode 100644 images/main_toolbar/server.png create mode 100644 images/main_toolbar/server_osx.png create mode 100644 images/main_toolbar/server_osx@2x.png create mode 100644 images/main_toolbar/settings.png create mode 100644 images/main_toolbar/settings_osx.png create mode 100644 images/main_toolbar/settings_osx@2x.png create mode 100644 images/new.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/options.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/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/searching_icon.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/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/sidebar/addLabelIcon.png create mode 100644 images/sidebar/addLabelIcon_osx.png create mode 100644 images/sidebar/addLabelIcon_osx@2x.png create mode 100644 images/sidebar/addNew_sidebar.png create mode 100644 images/sidebar/addNew_sidebar_osx.png create mode 100644 images/sidebar/addNew_sidebar_osx@2x.png create mode 100644 images/sidebar/branch-closed.png create mode 100644 images/sidebar/branch-open.png create mode 100644 images/sidebar/colapse.png create mode 100644 images/sidebar/colapse_osx.png create mode 100644 images/sidebar/colapse_osx@2x.png create mode 100644 images/sidebar/collapsed_branch_osx.png create mode 100644 images/sidebar/collapsed_branch_selected.png create mode 100644 images/sidebar/delete_sidebar.png create mode 100644 images/sidebar/delete_sidebar_osx.png create mode 100644 images/sidebar/delete_sidebar_osx@2x.png create mode 100644 images/sidebar/expand.png create mode 100644 images/sidebar/expand_osx.png create mode 100644 images/sidebar/expand_osx@2x.png create mode 100644 images/sidebar/expanded_branch_osx.png create mode 100644 images/sidebar/expanded_branch_selected.png create mode 100644 images/sidebar/folder.png create mode 100644 images/sidebar/folder_finished.png create mode 100644 images/sidebar/libraryIcon.png create mode 100644 images/sidebar/libraryIconSelected.png create mode 100644 images/sidebar/libraryIcon_osx.png create mode 100644 images/sidebar/libraryOptions.png create mode 100644 images/sidebar/newLibraryIcon.png create mode 100644 images/sidebar/newLibraryIcon_osx.png create mode 100644 images/sidebar/newLibraryIcon_osx@2x.png create mode 100644 images/sidebar/openLibraryIcon.png create mode 100644 images/sidebar/openLibraryIcon_osx.png create mode 100644 images/sidebar/openLibraryIcon_osx@2x.png create mode 100644 images/sidebar/renameListIcon.png create mode 100644 images/sidebar/renameListIcon_osx.png create mode 100644 images/sidebar/renameListIcon_osx@2x.png create mode 100644 images/sidebar/setRoot.png create mode 100644 images/sidebar/setRoot_osx.png create mode 100644 images/sidebar/setRoot_osx@2x.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/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/bookmark_osx@2x.png create mode 100644 images/viewer_toolbar/close.png create mode 100755 images/viewer_toolbar/close_osx.png create mode 100644 images/viewer_toolbar/close_osx@2x.png create mode 100644 images/viewer_toolbar/doubleMangaPage.png create mode 100644 images/viewer_toolbar/doubleMangaPage_osx.png create mode 100644 images/viewer_toolbar/doubleMangaPage_osx@2x.png create mode 100644 images/viewer_toolbar/doublePage.png create mode 100755 images/viewer_toolbar/doublePage_osx.png create mode 100644 images/viewer_toolbar/doublePage_osx@2x.png create mode 100644 images/viewer_toolbar/flow.png create mode 100644 images/viewer_toolbar/flow_osx.png create mode 100644 images/viewer_toolbar/flow_osx@2x.png create mode 100644 images/viewer_toolbar/full.png create mode 100755 images/viewer_toolbar/full_osx.png create mode 100644 images/viewer_toolbar/full_osx@2x.png create mode 100644 images/viewer_toolbar/goto.png create mode 100644 images/viewer_toolbar/goto_osx.png create mode 100644 images/viewer_toolbar/goto_osx@2x.png create mode 100644 images/viewer_toolbar/help.png create mode 100755 images/viewer_toolbar/help_osx.png create mode 100644 images/viewer_toolbar/help_osx@2x.png create mode 100644 images/viewer_toolbar/info.png create mode 100755 images/viewer_toolbar/info_osx.png create mode 100644 images/viewer_toolbar/info_osx@2x.png create mode 100644 images/viewer_toolbar/magnifyingGlass.png create mode 100755 images/viewer_toolbar/magnifyingGlass_osx.png create mode 100644 images/viewer_toolbar/magnifyingGlass_osx@2x.png create mode 100644 images/viewer_toolbar/next.png create mode 100755 images/viewer_toolbar/next_osx.png create mode 100644 images/viewer_toolbar/next_osx@2x.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/openFolder_osx@2x.png create mode 100644 images/viewer_toolbar/openNext.png create mode 100644 images/viewer_toolbar/openNext_osx.png create mode 100644 images/viewer_toolbar/openNext_osx@2x.png create mode 100644 images/viewer_toolbar/openPrevious.png create mode 100644 images/viewer_toolbar/openPrevious_osx.png create mode 100644 images/viewer_toolbar/openPrevious_osx@2x.png create mode 100644 images/viewer_toolbar/open_osx.png create mode 100644 images/viewer_toolbar/open_osx@2x.png create mode 100644 images/viewer_toolbar/options.png create mode 100755 images/viewer_toolbar/options_osx.png create mode 100644 images/viewer_toolbar/options_osx@2x.png create mode 100644 images/viewer_toolbar/previous.png create mode 100755 images/viewer_toolbar/previous_osx.png create mode 100644 images/viewer_toolbar/previous_osx@2x.png create mode 100644 images/viewer_toolbar/rotateL.png create mode 100644 images/viewer_toolbar/rotateL_osx.png create mode 100644 images/viewer_toolbar/rotateL_osx@2x.png create mode 100644 images/viewer_toolbar/rotateR.png create mode 100644 images/viewer_toolbar/rotateR_osx.png create mode 100644 images/viewer_toolbar/rotateR_osx@2x.png create mode 100644 images/viewer_toolbar/save.png create mode 100644 images/viewer_toolbar/save_osx.png create mode 100644 images/viewer_toolbar/save_osx@2x.png create mode 100644 images/viewer_toolbar/shortcuts.png create mode 100755 images/viewer_toolbar/shortcuts_osx.png create mode 100644 images/viewer_toolbar/shortcuts_osx@2x.png create mode 100644 images/viewer_toolbar/showBookmarks.png create mode 100755 images/viewer_toolbar/showBookmarks_osx.png create mode 100644 images/viewer_toolbar/showBookmarks_osx@2x.png create mode 100644 images/viewer_toolbar/toHeight.png create mode 100755 images/viewer_toolbar/toHeight_osx.png create mode 100644 images/viewer_toolbar/toHeight_osx@2x.png create mode 100644 images/viewer_toolbar/toWidth.png create mode 100644 images/viewer_toolbar/toWidthSlider_osx.png create mode 100644 images/viewer_toolbar/toWidthSlider_osx@2x.png create mode 100755 images/viewer_toolbar/toWidth_osx.png create mode 100644 images/viewer_toolbar/toWidth_osx@2x.png create mode 100644 images/viewer_toolbar/translator.png create mode 100755 images/viewer_toolbar/translator_osx.png create mode 100644 images/viewer_toolbar/translator_osx@2x.png create mode 100644 images/zoom.png create mode 100755 mktarball.sh create mode 100644 release/languages/yacreader_de.qm 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_de.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.1 b/YACReader.1 new file mode 100644 index 00000000..9777636f --- /dev/null +++ b/YACReader.1 @@ -0,0 +1,50 @@ +.\" Manpage for YACReader. +.\" Contact yoann.gauthier9@gmail.com to correct errors or typos. +.TH man 1 "28 September 2014" "2.0" "YACReader man page" +.SH NAME +YACReader \- launch YACReader application. +.SH SYNOPSIS +YACReader [\fBFile\fR | \fBDirectory\fR] +.br +YACReader [\fB\-h\fR | \fB\-\-help\fR] +.br +YACReader [\fB\-v\fR | \fB\-\-version\fR] +.SH DESCRIPTION +YACReader is a free cross-platform comic reader with support for multiple comic files and image formats. +.SH OPTIONS +.TP +.BR File +Open comic file. +.TP +.BR Directory +Open comic directory. +.TP +.BR \-h, \-\- help +Display this text and exit. +.TP +.BR \-v, \-\- version +Display version information and exit. +.SH FEATURES +- rar, zip, cbr, cbz, tar, pdf, 7z and cb7 comics support with compatibility for jpeg, gif, png, tiff and bmp images. +.TP +- Fast and easy to use. +.TP +- Use your keyboard or mouse for easy navigation. +.TP +- Fully customizable magnifying glass. +.TP +- Image rotation, double page mode, full size view, fullscreen mode, customizable background color and more. +.TP +- Bookmarks and resume reading. +.TP +- Find any page easily and quickly with "go to flow", a customizable eye candy effect for browsing comic pages. +.SH CONTACTS +To report bug or contact developpers, send a mail to : +.RS 3 +.TP +\fBinfo@yacreader.com\fR : for general information or suggestions about YACReader. +.TP +\fBsupport@yacreader.com\fR : for problems with YACReader or bugs detected. +.RE +.SH AUTHOR +Luis Ãngel San Martín Rodríguez (luisangelsm@gmail.com) \ No newline at end of file diff --git a/YACReader.desktop b/YACReader.desktop new file mode 100644 index 00000000..32d5caf3 --- /dev/null +++ b/YACReader.desktop @@ -0,0 +1,13 @@ +[Desktop Entry] +Name=YACReader +GenericName=Yet Another Comic Reader +Comment=A comic reader with support for .cb*, .pdf and whole directories. +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; +Keywords=comic;viewer;pdf;reader; +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..9943539f2f8b228fe6cafd8c53a743120e27b959 GIT binary patch literal 133692 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%0D7+nVj!omx7)o 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 U=Be*K#sfB$7aEGi&rLr6ALwj+#sB~S literal 0 HcmV?d00001 diff --git a/YACReader/YACReader.pri b/YACReader/YACReader.pri new file mode 100644 index 00000000..b4c9ba4e --- /dev/null +++ b/YACReader/YACReader.pri @@ -0,0 +1,150 @@ +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 +#} +CONFIG += objective_c +QT += macextras gui-private + + +LIBS += -framework Foundation -framework ApplicationServices -framework AppKit + +OBJECTIVE_SOURCES += $$PWD/../common/pdf_comic.mm +HEADERS += $$PWD/../common/pdf_comic.h + + +} + +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 \ + $$PWD/../common/scroll_management.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 \ + $$PWD/../common/scroll_management.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..1b189b16 --- /dev/null +++ b/YACReader/YACReader.pro @@ -0,0 +1,107 @@ +# ##################################################################### +# 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_de.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 manpage + +bin.path = $$BINDIR +isEmpty(DESTDIR) { + bin.files = YACReader +} else { + bin.files = $$DESTDIR/YACReader +} + +docs.path = $$DATADIR/doc/yacreader + +#rename docs for better packageability +docs.extra = cp ../CHANGELOG.txt ../changelog; cp ../README.txt ../README +docs.files = ../README ../changelog + +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_* + +manpage.path = $$DATADIR/man/man1 +manpage.files = ../YACReader.1 + +#remove leftover doc files when 'make clean' is invoked +QMAKE_CLEAN += "../changelog" "../README" +} 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..13e7a9b2 --- /dev/null +++ b/YACReader/configuration.cpp @@ -0,0 +1,58 @@ +#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); +} \ No newline at end of file diff --git a/YACReader/configuration.h b/YACReader/configuration.h new file mode 100644 index 00000000..ba9e1531 --- /dev/null +++ b/YACReader/configuration.h @@ -0,0 +1,97 @@ +#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 doubleMangaPage; + 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 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();} + 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..ed774432 --- /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::open() +{ + pageNumber->setFocus(); + QDialog::open(); +} diff --git a/YACReader/goto_dialog.h b/YACReader/goto_dialog.h new file mode 100644 index 00000000..54253605 --- /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 open(); + 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 zcmeEv2YgpW+BLo|qN1ya)DTDr=?Nu;7D5j#bVNWzL7IqwNbkMZkc5zg^iCiNy>|gY z>Am+RA|i`*eb0B!O)l>xhNA24r@I?|hnai-w>)jiotfvEeEFWqmp|XDujb>QUB37K zk}uzz%GqTfr}F$Jo~u|f=h!h{z8*#L<*Qvg``D#!zI=c6%a?EL*qr0%pUGEwRK9#J zJk%rYh-Zxdc@LkH$KNx?CHe9_%k*WUFj0)KAsh(4F`s?Tv3}Qe<~`O__qY@Fh%Uqk zVi@r*(T)fp^m_@SAYr{%d7ZCDwECa_c{a$gQk5Ty6e)V~#ey%*;JHpjAW?;|B`ASk z+%LLByEEZlt$x$xkr`Qdr&Fh&?aDe+A_hOYS13f}e!p>C#@?xbn}@f9yQfd@Zc{@r zW%&jS9y%1R?v+LrE?iX4FF^c42E5+0l5dUmwHvkkp+$!-2<+GsZ;hRck&9Pj?bb~g zI&w6s*Q|frqr7i_dk2???&rO>-u0T_YCRzgqheDrJuwvvmSiA4D+7yHufXz6>#=^@ zM#L{pN5}Wx2cHpXrQf|m<*1r-(%~~T30@<@5HLL+fivRJZcZHDo*RqKbK~$)XdHUZ zjlzJrk*L+CGx$#F!{E8BZ-e6v)*VIZSy?DGB@GU9m%?Uz5}e1xpyGsBc#d8O?~w~o zZA2()jR-;I{xjj>RXx+Z-_F0))u!7{!7*kF>_b<dai?f88$%9`HJTo5y7n;YZ#oW-#e3kJwHMwi4#1JmDKl>c97#`U z(o<@59PEdLqvEt=l&jMQykGYCj7Ry(vsx#vga6KRXtV1q{8k-A-AyOZ?BF%jTYm_y zNuQuh=xUUlwgkmTC%}GYCLC%s`>9CLlKFK%WnuHid%bT3e0B+a&)!Cllegf%`y%}J zTtuBc7f^5KSyU%I6&7xQ!@QL!F)ff)U1nCX3JQhW7SXk{RJZTv!3=? z@6$8%Y{wipFtsO?>YOX#X@QRQo9;2`5UqOF)4o^N>3Tgw&r_WxcoP9cE21OOo#;h$ zB{~uB5?u)W)|gOzDH+A7Ag^K2|d3bF^ZT)L=oXR$9kuJ*L5uj(O8FQ zNQidPU?M;BPm+ab)w|sZJxkANLj?UL|8t+c_+r5yUM%?X?dM;3@%aDz&;Kmp`C-HW zLiF_{Bo9Gti|E(8ZF7*z#`B~_G)i`Qe=S1q7H=K@@|XN`DmHBObBiAR;O6G0dIrTy zl-mFN^DpZCx~3W7Pk0l0x9q`ux6)ULC-JY}MT6*;eMy$}2+>l@#{L{YQq*1n5fG9*15neA++|||7_q&A~cO#ZG z?EYje-tPD|oLwuhJ#~^C@_N6{$qw~A^Vz>SWqlPI(5V;3Z#jf9=__#K>KRO$I00_% zUMT0}(TDFSY=L-|-hcDE+4dJ^J?ZP}?$yGxLgn6k{}T7|zKA(;2{jfh!?45*eEQiA zyw|;_+F;h<-XmPxDoGc=iN}HkUw)SDMngU;g}BOhLSP=A6|2CdQVlrx)JKcx4QPovA9mR>``>=DzMhuuV3k{mTX~>N{t*1;LTmJd(KH|I4;8|!h zCkbtXlkq`#GKN#OqoWcrDkcTJ&#*-F35!Q>_Tjn)#bHErDrTo)we{d9L#>68i zDi#eo_A=zazJ%)2U+97F?&DUm8d^uKgLCR`*i1}?oW&VLl&UI*l2i7iigYa z2$UZ-A5Eu5qtTRT1Wa0p*0a>7j7O7+5okR%67Nh4N4L4r=r|)1Eysr7-N~V7J30tH z^;?RDUD9)I8Yr8n($$)wl5SPH#yEotBj9Ejq#C`RXBWyg`oP-d1yEx9F4}Z-#cO+YL5(sSEs@7WEow4rA1pO(G0Kn@`;b*AJv$2}r!0cagcOt-6^C-9!EJmT zoJPl>C_R;QO-@6pk+CQ(un$~nWDE*_ z91gn?QK-PabS1u1{p#u?=d=I606s%PQNGzbe811`Jl_9V$=3bw{^#U>8^XZG<;Iw+IXwoQAW=VtbYDgRpQj-cL}LvT;u4VQ#%a82F; zAKHa4`^p{}pQBv#W|W~$O3lbZS*H3H@R*zor-?}@GI$|MQa_G-zsrb7+HV+qrX(2O z8>QR!H_9U4uQuX=4NxXC%GGRv57r(;#WhFJanE^pvu|FBy70{S44zs0(1f;GmB0C{ zI)w5|_rZZQ*hxoomh*dLCCbre9A_*>iE+s&PS}i1K#3udC^H}wm8U18g6y}Y!N2+r zB_4Xer%z+m+x-@8My0jKPxAZ+|&EsmDPs0E6%c!~g0;;aq zhpI_i;6U4Oh~alpFzs|MZG`qwa`GY+9TA5Tv`zczi&0@7%T(iTCjSrh1^CUwZv)@U zZ}l017JP5@PcNV`-|^m|t9WvLiW^n2-t^+DggkS+H%=RrY`D5p_UU)xXo%eTvGT@*8X4 zRgC0!*Wh!v(e?OE1n#{IKYq*kZa#%7+s>dKkL!PS5e@d7LmbN zeT905uA|QWtElz)CHV8*wMat^UhB_q!N$}_(@)Nz$-2X+w`4bJC2oUL_+o8)=|-E#vaY_`1KYFa)&BDLrNhTh#h8Pa;lKYn8lL_N?a$xAhu7|*)0J<~;^JL2 zJ^K|Jow$XD$G#v<*YE-LIMcj!euHRd;Z^_XN`8iH{eo&pkF&FV?#Z6wflJ6bD zd-N^!RPz3{*Qw5Ma`v=keO`&ELU>bFwt1E9$2^-q{r~&ZIE98#I};DaDfIRC%RkA` z+HZNW8Wf>bpwcMfyH6ztOCz`7NbuRA1}6(0tynrAPOfu--;9xmt&Fk7SgvpP1QJbe{Ti+>Ktn@rCZ(f@_)$>HR=oAgopcOhN z*yp7BO?833@hcGeR#%@;P~MYJxiRlC`7rgNzJuPEn~o|1LaBIpfe<|=O?lxHoIkwWT{0l^0^qJ=* z|3~F@uRS5%`V#dlsKb0MqE3mDr4yX|8Xmt-sCrZ@MoNq=5+>YNGjU|wTQm4BI#E$ZE}rD_&bdEteEjmtW?UiYZe3_H$V z!{TjQv1jKx%$+tKl`B_=vrBn6xs<-AfH zLbfU7Xa9YK`g`w(BqM$QNi5i~3CE7?#-8nK@n+jMH5Mz}FKt(@H?KEk|0wzEyyT^4 z=S7?K+@t7x+NNW>RCGwYmQw&VB$uMN?nT6XO7?-jvP$MWE zy`vW4#OXui;WEaJ84b?asZV~7<4u8o$)8_3w8Hwnhn}CqkIv;jZ=TDYn)E(&z53_& zWnCQDFW2`=chWI}`Eb&-mbmO$!53r4Pr;mH*AbAm6_qE)V_3pstlY8zix#J%eWxzw zH^pt(rNNr#?J8t%2o4|v-pkk8EO5F;Fw+Z`1k+iISx+Q z3qEi6#kiek(0%g}_|Hp6hma&=oHjg_^X0+wQLSbJg9q`d`^m>yVi08^+j%TKx#Vx^ z%e?17N*mQTVl~pdi?sj5dFSkRQvY6+{o(6Z50&c#qH4o7sNVc-I0bY>vAXY~aQ*Hm z-g+=9O-)7BuoY-9H3q#x;xIif8Jjq+y9D8rtt_xbe*AMOc4#&ih1(?Axfawe4F)Jz_ zbED%hlj*GJc+87WLgJFeSjRO8pX}a-BZqe3!l}KubMq{|zxM@xy#FQBo49fQ5KbTZ z1VJG|sM4r4+$&b4E~F3NpQ(!^wi#+a^Q0};`CstJd6tf(o!`NgQK@l9v>i4N?MDS+ z*n)UWi062C+7g6iWFTVkVnlNOKx-#bS1iR6&L6C#d^c=diG6#wh~Mkl6Y0_Rtzajszo=M|>KCSoSne}tti zLI`bp!J;%orY|B-X`F8$xCSJe=|Y~1rcPp+#%E11j z%nd=yj$Ki{QZ<7Q(y`mo*;94uQ`ryOpn|mb?XnrS2JO%%el41E&ZKovBHm=X)MI`k zK8)bJNi@sj!UTLYKMvy)mtet)b(o#862myBGL8JqNLhqgiJUu0O+zr}Q$kg4d46I{ zBG zBqk#|EfrIP!_cTpKX5&UQ6?zYh87#K>cX_0XQ`)`sjrg+8uXorS{(Ov;aIfOlvH?3 zh=be6NR<6J1nvXpp(5u2s*Z_3(^&}y>Q9a3d|@2x$rRF`g!c0i@y7HhbeI{9x8}s6 zHS?Y3#-q!;1azJi#kr+;3|c__1w}F+i!Pkkd3)*tyf-}@BZ4C_H6jLG$Igad8?&wO z*-_5Py`V)84`gnhV|l4Yef{KI$sg}0Z-p!8n_L#{;aGl#F`wbcF?-EB+4T#8-QWdq zAGHvklat^!j`JKmw*NR7&I9J57Uz2cIR{mL;zIZ{-(XrS+SC5tn3I4&>aOmnPy~z% zGv=f^Oq!1_vlil=IplMCB${(gRp8KBcw<}$+7OLCnvC*Q>#}b3{h9S)W0Tfgslf|% zr1}yi{oA6?xb3zpS#*-(``!$#p34hLA`3wn0wIM-h z#5t`d6T=Zmz5+%Bq4AhdG#EV})%#9E#WwF7Z6f8n+%g7d%Kt&ic5s^NoI-WFV9@S! zsLZi+jThSmuQcelnUrSCy_MuVm*%{5-cISPOxSSlso2PPl$n$Qo57KA7!}QVFV5{u z;N0NoXwHd6qs%bsi}Qs(V^h$9kI}~6VcX`ABYTtX2j!Tuk8$U?=Sy|FWBC3{sJ`<&>EDBLQ5(6wY&C3U zM>Ccfb9eS!-{rvhD$U_(uBznJbQC9FwwxO&H9i?-sHbwA=k%DE0FN;oBOe=wlAOOP zJ2DCm!y{3SbDy5nQ6>v+SgLewH6HMjWI{Bp_=Gl{FyPw=g4xbwQZXB)4IEIk?Y~idAM>5HyU%2rROX& z=Ko4g<{TjBSG11Kequ6ghex5b=9{>lt_0`c%5W~whI4|2KVE>>2ZqC$JXRVVNgg;4 zNj|(eFX%Ul^K!$<6VH2-5AOj%sL+A)Q?zZ;AD2u1rXHT?geP);Nqkauro~ zoaLPDUSqz{QFDfzi;~@xA+4IT6#b6Tn+*P(IhQAUvf=!c0j|B%JZp)uNt`cBLitIF z@SQ=P#>Ao6uxMjGvGBkMV{X%9WP~w?s5#b(oX?cp{kZnfXF?3ThS9zV_a5Vn{t@Xv znv4Di_QT(nxYN#F?>Y@*_gysRO*}cD=a#k$&I#M$9KRJ#F`H4AYZVpAbYjB{O_Bj$XU=KmZx59txS3>7CQq87ggTsa@A@?K<6B#MoQg(J(m z=P1rc4&^*3d2kyVhH7KNQGHSjyvHnro90VDqzzT7s(#)N`p(E@Lxv9cTbRcEi*y=? zNuQoZ`Q--;Xx_BXj#KdBe5E^i(Y&cs%qH@aJ%8%XHI$kYGUtcNMQt+HKZ^E}(=v?u z%aJ@d&B=o6v^026NrEfuUC~}-2<>WgyivZLNT>T~mXD!KHOI|4c$aZ;aA3V=({pzvdu{^o% zQ`fw;^@K4OT75NtU$&oRhGm|5(p;bBJq(HOe z)_W5}SM5Wc^+!>O>trW${-^TB6UMx;7ilbS(EXV)S1kJ9+GY=(ye(_q1N`(sJZzB*8v9V1kJ%?ig+c4iPr%`1)*W+s5_LH;dd-y8qkdOB`kKb@T=OU>K7tT+6QV$(>pGQ;D zU5E3P^@)zhZlD$GfyUd8BXHGcsF?Bz%Bjv}T~Ls)wbKPj2QA~n_g^8k}9XV|_4sITF{{%7(o+QPcy@Y{3AkZA|b z=eF8+8Dmb~MEkv$(ErdCH00d2=2W!~q5SfLXvw+eZk+4vbm|tmpT32buBQ3-7CsgJ2#k4-%l9Z$WqA7-o{qkSAF z?GMpxJ=S`7)}za%gMNcX;v>>idDjKh;+%QieOEA*YmK?%R={&WEoVbNAKQ zR`_f>`CyGhO|CI$#5FGsIByz2T{ZdqJoz|}+S|{t9yn^WD=y@t?7~g7rS(QVWvp$O zqV)~wMwz#ryMi{i9jz(|*K@nKx3l!_MUMu5Xo^Sh4eSR9Jfg z{-nRo!RzRB;tNc^d>3ObeT_++-~IUPm*{o!3)JC!zcKgEHQ?2EoI_R4_t(@q03O%g zbq=*yKI+lFT2p@=sk*U*?yMwOO!|d;Ch2*Ttgqo-x_iqO1;A@$A@pQEMGI$W3>GA z9GZV}+Ng(pHyr1Bt^+yk_NQ=TJ*IZUiEX0lCzsSuQFvTBUVC>EoU1i3zQH(WKFcEO z2XvY;&z*Ao;#rpGW*qOk_4Yemv3lDsL~+eT^P}_^IDQ*}q*d$wJ6-=9KK$ZayvcPG zZFt<8YbRP=`UWkzo;5(FWbcOZ11$@sDk_r2py|F zE*ZKWMf=W!OZA54cl<$)Uz_c&NxPX^b?}RGv?Wh|)9=u@rR-_hhOOB8t^;EkhVPkhQRd~t#j6bw4i<(Q!fpu7ui}(mY2$`o3)n8i{(J6!4KO% zo5(G&={rY06Z-B_xjju=$lc~z>3m{7$4{pppLL6X56kv2uBA!ca}c}k{eT3n9hrXT z2XxW;5%Qt+BOSPor2UQi2AYwd$vk)P{*P#W<}1U-R6eSzJg~g`vaNH@6m(hqtS)C2t4yOhS26}9=-{`LpR{VwN#!PPQiJ_5jZ5Xot~b;wU{kT zd%4egus`Q-9`(&`(r2}u+IzVs}|@NLP*5ZcvI@nP^ndAZkXit6tVM2Dal z^j^ISV~<@!9P8P|m%l>R!IMb4@Fn6--$3x;OPIXp42EnziY}3v2;{h=mv0R|$B_9^ zmOFn-SJUWLb>8Nt8yw>6v?GdjR z@AB_CPQ($d$~n4gzRP@{(x=T^ug%S)d2C+uiC;XRG*O%nN8 zOK{xX_aM{##4chfX`MtJwxvxvbFBO2oUw&pJl7)AynZw3vZlJuJkBeh*EwDPsPm8e z?NRqU?Rmad{!se+Z~Cv6gVt%)CdLxaLLzm|{Ra=mO$Jcuq6xp~)lt%ni)-h_AyAOeY2L_4Ab z(TV6{!Mn^0Ixlo2be-<0M@TMuuH=zh1}YD-MU^?x_|LEj$=KANbZGLg)Rc*I*_aUB zU5VaApB(gOI*1sO1Lb=WCa>aGdEM7NCpG1v-&zpO2t8jil^m>gP=d!M{uwr*cZu)Z z5dYGR^eTRG!<2#eoj^>>f#e{Wh)3B+8$!J6IkJ^HM6Dc%?}milBj`6BOYT+~So6jB zTYlo1-vQ6^8G2W4-er%XO*~6?qPrt8fDnJ;HI#@UGKrNIEMY#Fm_hU;`Vr#4izT({ zUpA$((3lYKdY9yryDV5`@G^h@z4&+LvvTwA&vT+z^hk$-j)RCuVrl;83f$$|$?v$7 z=k~MD{^jyB&-~9}p5H;NC(;SYMlu*fj3P!5rfu|QUNVtvW&h@TBy-6^?{^@qJX`Zc z_}lNnzn)Qsu=4NDWAXbg5tx(Lv1pw_%pt<*cYWutuN1;7e=USp3l&D;*NbW$=1(ub z{L1A51)l$mF+ktwp9;TTd1)?4yy~_qfqagc~ zT)Hy;%An#UY`^pB)DLO zGXbY=+(z8m4M>U&!w22ofs1ST2kT(99=43V<6f@k70=>P@3HDv{N$zQ+m z%V!DIEv7zXkK);c$-DUQXPsBe&fe)}S@PFr@I+j_^9@2*uEmxuYp^VJA=OD=C|@@*&%M8ve~|l>VOOUwr2U+1Cri%Y@$PK$vAjsdVUUQ_60T z*4R3F)j;sJBlzU>d92vA4Mz{{#PI{$5ISozDpvHRJybNn@z{GDb5UJl<;TpMau#eq*Zp-Gcw##&^K z8!7$4#@6oL92r>kFCPCiyqh{8KV3OCW~^74w(c{e9H0+q^coCJ%D|ChdvWipD_jS8 z9{oQ0=z$Dq4?mT%E!&aL5dXPlVAZvLlfA2c`n~JC-^E^74;CQ**Chk*jvwOIw?ClD z!nLRqoQ{6cY1n;WH}2oLitq2ju@Y1s{O|Jg++GccJs2^&IkF zgk#Tf;@_2PF|OUY3-2YLp<387cu$B$w*|@AbKp~a{lx`bJ$D#=`*Mw+#(;S}ahq$M z)!zJJc_e>@9Ll+Qlh$^mdl<)ugUD|Rk2NNIk@%S~_;GXh#-*#*F!taTv|O_fzOxsh z#^hM8CFh#d<;$^T*%Cy?M4@%Nx2-bJ+TZQ;Q|-;ZLgD}4vS1yhwYUQ~&Ye!a6m#VP zeOxS>UuKyVM*;!PLUWY6rSO~2!s zk~q%4sUP?+vk8sk^PMfIhux3j?ZJ7?Z&KmILQ(i%@S%AMlq$hjghJJ3iIxph=q!Xwj)VnsLsld5=M8)n_zX_MeD)y~d(s z%a2j2kp6iL(q8O476X6j<+M1pj${hLKiJY!tyMvv3v1;vdY}~RI zhq!+J#L?Y2cXBVToI8LU7mna>UtPkF-`>XGzyJCHe)!uLxO43UzPNA@Ti0je-H(RA zsVe<69<~Lw85>Ef+QQ$hOz?Td_}sGw@3d)W@7k!+X()QLyf*DO5&h>xU{q8R#>J)} zXmJ+8v$C*|{sWQ3LdN<@VCk^oh8IAOCg>-`=@|Z@;>L3#a#EF@0^C^co5G=H1}M_yU%8LNYkd za;JXhKP&^;lVoc0?o_=I+H>9XTjLfWU~~xjE#%sJ@;oF1z<^;kgZ1PhX6vc*dg~J%#1&t%qbldQ~4g{-ZMB zoaY3CciNdtm3pYncsRX-(-6RQ-QIs|1ctSQ99<+e<4CX8I+5BG9tOQe|~Z@uS>zKxJ1mLzsz)&rFkqz zF{vp?X8DQF%%Hq7ki{~ziRDM~$ymAsW2VhT+wT47n<1Zo?7D$^J5Ady^jl@Xw%eD_ z`bp2z+E>5fj47J61A!rnjQCA&(C4B3{8V%cO~Cu}>FYqhwowZi^C}_^1Hu#0eO@eP zEnbGORT~hodJ}z9mZD!|GA7YKWlHiQ!;fQH0^?wjzd5OkFpu>_C}lG6f!8V-d#B3dUP~hduC5G3b6w{H({x zCCBdNPiC|^wR?=ffW=&oH7^sb<5r?AeIecnrO(R3Md-@9`<>tfG@l-Wrc)x(ZW?1) z&4{5-P7+3>EX&6(7+LWN1V@Mlw#=2&SXx`!Z-j z9NwWX#RU4*%}C8e^vc!r+hJQsKdNc0Uk8W9;^UBLj9w6fsnL`R%U2xBQVeZq#MC)x z+;hl7^&-DJ)OY?>bphM>HU{tHzg)G37{7cEicZafSL9k$3(Z2+DeG{CK{xX1WyRLa0Jk2rKNnTCWN9BW5vBopPY`fqS1aTlx;55|g zJubWNNj|90JY&e^B@^N-$=a z{6Wf+U&ST1p|6O8e0OFkc2*WVXJ*IKDmgY0c8oz+itESaljJjrb}=p*O@dMlKc0Hz zzcy{9-mD};CiSMpp~~oR$nT{Q{mBBx%!fboULVgv)xmSom}R6fW9Kzzyu-k8^U-8{ z80rt2$+E-iM}?r?Ncxy`9}mxlZ$4-bIHsiW>L=O&dCxBU?CTV4*ay=;(P<%|h5wocb{blU9j=n5q;4&uzj$DUs#uO`~ICvwuetQ7d z&`V~+=<7p&fvSwNR*}9xuJi|T9I+6t^gF3Q*(f$&?I|&+!dP&=W z9QY0jLS4q~69Q=)wZ|?%ouPBtCkZv;_*EYpfy(bsglFfG555`rJsSFy{P&Um<#YO6 zgO4!%z(r%7xYqJHF;V zjBytO`)TwgqOXm7g&gJkG%NyclM>)c-w0RQh0aSxr3TS&Wn2v8x8gR0{x^(O=tbVM zeWHSLVg)ir9Q|>8hs;Aw`W00j$TCmAsLCVeqYB|e-zkqCQ;hh5Ire=0Y3f0im2tY> z$-5T$?}wlxR}6nCt!Gp`zjE};k#CWFaL#kO-6=9A?Dit4MiNq7RSv z=m^6WWDhvR&#v37*Y5KPUt2(^Pws4bV=4#lPyl z=Nf&C`6q8MJ{Z@k)BfcTARi0oxNQ%7mTaYG`Xi}cbd1^v7p)U!DxW4JcDa1W$hX#> z3uWlP#qjhVPdi~7>?0pS`g~Sozr@oRhhRJCUMc-4_>U6*<@^Hi z^83LEKXKjgSM%YTY7e#pE{p-MIG6HUQ#nwadOP{w1T(%o^)BBpLq_D=C2>1_z;?oo zV;pw$ld@r5pcsb+=trUY!Ip6jRX;dPT|~eBWYn0E41fCO*p1JQOQ|^XMOY?Yr|)D* z`slfnclm_54`#bMoPLNy<)6p$Pdo5aJ;1)1e8aq1R?5>~%!R&*iuWd-h>K6+pWi&Q zbgfg>0KC>W5|Kx*z z4)R-ih_2Om(<^-ed`Qxtk>|}QvFCDQ^`SonuOm*C}aqW@W?8H-SDMhe{NPp5M5 z`hW2vDFHly62`9?e8NdG{4`Z-r({iktM7sfSpW!-8wE(s<3hogM0K$Cy@ zNNG;-k^GbQE{3ew{{OYxY%Jn>YDdO{l+R!#`lwann*7%EQE5)!H2HhU$H4FpTtr_9 z%0hmr@|O@=(vPYkV_R0HPgn)sFP}CyURRDjQicy6&Ff5mO}r*Cw{$!t?5vVCD$d#M+F zWNI|E^8eUzbVnx-Pm}-RZ3iNJ=Lu9J-Rl2&(ciEM{X~4|BO<>9!$)B?{ki39k>k5( z*a`FU5A9FCByZkVgZ_!tC_BYNl|PnzN*q*wa{SXSYz=)XSvODyicPDsAfIHR9NUy~ z^mX%?#X6I9M4>*RC_+Cz#WXI?enQ!i(XbydA0=5gl=(Omr3WyMG3$@&`Zk-H250&N z7wj<)j{aus3ByOO(&O}>)BnFO8Q3@Kgf_98F_CdaE3Q6b`0y>i{x$sA7Rdk4d)-m` zm}UD`D(0;RZQPT7z3xjsH~g?V?>vLr^!=0FR9tx&K1>x0SGJ(OwBpa&vz?SLnS2>b zQ}2rVYQ)Bs|1ay(vh)jfU>l?Ugvv_s32Bs7hEa#AZjg_r+J@2tLSaX^vkvu}or>zg z8Sr3?h8Mfcf|LJ4`mgXf{`DLCK#KhhcW*z~PF#!?=WiMQXWrDi+CkIr(VxCfl0hZe z0QFso{-^Tmm;ae7%YkgBKYi2;|3{V^l^yk?YS8y9JMQUGxFqj@XY3|;ETmtQA&1q5 zkF9+;>k`%@4v`y;?=AJK)xRu0J~ca5uKJaXlPbT$B15y|N!!vE?C8TOA57mO`g>MoT7|p^(05WXOl1r5 zH5|p*zRfq%CJf)nY+phT`X2V%e;Ex~XDH5UZSJKaKgxRaIcrFtvH<$RHQaa@O;_wj zgC(E9jWKY`(JmB|w=7fniWxqxGwCx;e?;{iN~k|2zkK$)i_#~t$T0RHWD6=Aib24( z#e?yk9hequH#WylE9aYz^HfjLKj)Wo!vkq=0mQ2jn3>} zH>D3_qg|)bV&g&hWqb-}*0+ulH`(3aU|$ z^5gyJ#7(?;knx_0S!ZvdBlSF-_Rx<0bam)|s`^o}E<9O3R$%?ni1PcGzNa71&$B)K zP+K0qfyVUnY_*>;y>_29d{G;3r>`gPabx?Ta^#$nZAUhm(rio0%wLPwM zz2O9$H`32l<(Ry;rQc$2#@3!jU$n>@_b}`FH<*6y8)N^caTmX$pX?RXrr&IB`uEn} zo9#cVnB*Ey(0TdL){tL3&nYIj{DEuIFSses_hgLjZuE2RNqj(`_3oRFqU-AY=)7Vt z-p<;CcQQYPH~XAM`x&wtCFoBsJ1N7qO@LF~lPMgl47G;p_yLk%%j4R%ma**%vhs03IBY?6AICc{OM{gQFw{=SwXvpqLhlxug9Q)W=XCQhjMhwUZGWjW$!?=S8P4H~PGV z-x>us-&$t958D0{`#q1731wizdBxA%8B4Wqzk#@L=`zw-_chyd8E$*7q2}=~(d-=g zB#*7hZx6+uzjhzv8E0Lw-%WpC(|=hp=?$N3$&S9~^2?T;jJfa?*53M?QEyh+dJ2_U zcUI=NxBAt_w*_UOKBm4q44-CwQ&1*~<6kO#6J8yef?^zxclYLc0hU{VYbC-S$J^8X z#`2>WX7YQ4Zxw$`m^ck5Po2i@ufN9|CvL;*;1{TQ@+-!@zi;?#zjNb0eTDDio$K`B zzJ4EV>CbJ%!B;H&%Q^ns^yxPJxdT~$DlWd{P}lHH=hz$T36%r22m0PpU(@JMvmdQ} zH@w)c>f5|@+zz}xm9c($PJx?+chb9qbpNhv0dv-}-PQa%%LRrG8;uhuPU5TYen9e< z-(nE?Z+ht-nlkM~ziFWZ{lK*kfPBCo_<=L8JpoMHXi498`Ls7+9D)X{PqO839o0TL zXYjAR@4VRGR-aow;Z@jfd@D{aWd# z{D12~BnQ8m0hl>A6su2P#Fsz)j8i}S9kXtKkB?af+J8YgeEB`zrYz(SZ~BkRF3fyO z`pbVvncU&M)7X|s7ES3JuUG}zze4S&+6~1lslxpVoU#tVhP>NyT+F#%>xbt3$a@gm zug8vcSo!&_dBwIN@ACb9P~H^Np?v2)=#;h!$ydI@;qU&AFUV8oH$P&=SKp%>ZJ`x? z=-<3aSx_DwSx>fP+3-Gh7yf7OV&^yXPiOm4pY=jhmI=kCsKquz-%M3$3+}6rqjcI{ zl!)AlN}T7ZP=)~ea{s*ZaIfC zD~`c-$sv?T`5Yx8wxUVoY6QOf0V?qM<~#ZR9kk2Gj_+9M{=M^(gR@HoUw%78lBPQ* zT}CSXu#9+AuLsvoy-{%@$EuRoqs5NX7{a=LCfkkJGdHn(%MN7Fmp}dVbu2%94QozZ zM#kX_i2wXFLU$a)xRh1s_TfNyF}{K2`9H94JLX?u?@uc;)w!gv1?dWBs`V&Vdb917 z@-SrLQv>dP0bCobI9`+yuT@_|(7thIUa7^-6&qkTF^snN53hB3TE2eEb+QZFG7gTU zzYFV&P|9LQo-)ZTqlb^LlGlxt=M45`0*e$a@&A1d=x>pQd5>g58!5u|-4!T{Iz$Je zC(%0xy_j|)0_Ydvtn1Bt|C>H~{Q5V@hPV9p?>|)niW4zb%yUNMnS6H2<;%yt&-3_u zuJv(zWov;QgUmHFI%H+}qG{7A1> zNdAJp4^2HwZ}P8FKk{)fud(`KJuYwcS5jW@5o8a)>-Q<2_j}3f&-@&fU+Gk3R`NFW zW!0%Y&;OpVK07a4$jb&yTTneAKg$31bDnI!l99@-bZmv{RLRlQwR9_;{GK}gHP6z! zt#KnhkZl9ft=dYnPBvxlz;n4_{`N;q&3Zw$klO~#V^jZTD!%^{>jLRq zZK7o6_yqi!Y(RGTBs(y5Z>DOyB#&qQ1EBg`d_PI&dHw!-*nys*XPeJ*X5N+1dMU-D z(EL#)OKRqIjjpxc`?xwNw+%e1PB3lYKcQcsx=!+w98Fo~rRPVTGj(RBkCVTyG2bKI zNw?CaFHwc?Bm9Y4M4ev&U9bCezn-J#>OG>x)OqgI)V-OSZGknF9mobGqXNV;|3-M0 zyogs*A9WJe=zgIU zp=ZiAWMiTuw>?PrCaik5=B+mHGOvGu$WKUqe`b47dB{uW)^nx~^Gc;>omV+F^;?B` zKf<4oZUYEYr_yI9qASre2fdg+3Ffukd94XO!-_6Ew&GnL3wmG29LQE$5TdCXA)3Xj zJz>>%UU}JoiDJx~D9pTUB0urWp9td7s^`b$9eHhT=z5hY$x-F29uY{W+)3u$3F);r z(dRMf$KOo!&&liho_Rpev_krq&Xrovn|bqhs~vRUwXy@zDZ6Y)hc;wdlPo99gZ z=T1#qkR3cr{L$?|HY;9DsQj8R&skGP{%(c(WGdeQgvwJ#q9>7C&(f>)oNPfl9!89^ zUwj=#FBdQSM!#gL{b<)3nYfR{OTOuzz zuAZ9+JN|~Kq&Pi0tsEG z{NrpuwjupnZN#((*+4MR?yIP5;l5CI{kC`p$h`{7awGd2gaCA-S3I zlFW46lNd!zA?6TaL?jVKNZ$#>5+ak3j@RbG`kZ6wekGAagcEv(iCN4KvtS7GgL2TH zX+J`A$~I*mqO}Vld+9`U%s~gHdFfv~3C)NWIr(Z#B`=%bfMUF-0P#n%foDmNBavIr zk2;n-OnFMr^$6+Q3du`)klup`Q*YA!^1u8g|HbDDJb#~YW$!Y7>)B_YJ^#!z&m8A( z2Z_DJXT)w|7qOk#NGv512-%LFBZLuSEEvW72tqbt+CqQkWdpKR*?{!liIBc!$L)w# zL@h$PtwQ(`m2%R?Om&^~?@#y!zi4qF#M6`D&TAyEUPNOeFWt)y zWFyjn^dC%w6Vg{4agchy^ZfHKAP@Y^xYM$U%ZyEal-KScHWM3(^~7po36VrZ5@AGc z{g2|g5jhYILkQWTFo5Vs$R5OlX%8KkmyWGy%VVAIm;>=tjgSpU4z`5NuS5x6{|fO( z)4%w58r^&Hx=ut-;&J-dZ_^0rKa_|hqH@wD&pr42truT>2`{`@ka!8o3tUf#S6+RM z@z{%iamexd>&1U!Z1UUuZ7=O%BkxxoApNHi()}nxIut}xFGA&9`jwsqm46dD)^EB- z*PHs6j&-kW#f0od*Xw*&LiQkg_b2`l{TJf>e>DBSLYiD31MzD@Wx|wQJLVe`0ePSk z&q;#rBX>(qCV*QsymoG8q5*)U@BD?Yzr6SR%Hw1tzj z1=)e@O1cs)CQMsUDxGz);9cge5Pjyc^3t>DRUIMw(7mz?>0R|gAW@6(CY%XdLgnUB z(C_9oW~yudi28q)bU6}_vH?@~`n?Vz8<2cWIh%57#&eRB3E75zQ#q1crxD?Q$)A6z z_8X856mj=OgO4ZTuhjxk$i@zZ3cYUZN1#0kegXYPsD2=Bvc5=XJb#sK(eWr7FzrCP z?nHbN1#*UN!;^(U$k(z_=i88{F&Ik4(hc`I~{d2C)^gy)|5144Ew-tvN} zf2Dqe!-(`;o7{TKKN?Ux;NrVbiAw-F3={n^CN+Gl~^e z`7erBwJ!yAU6|$HtP@l}6e(7cdo`3oDO-D#EM<3(^}=%YA@z<`#)>Q$+!TkE+;<%{w9Xa1lNA5YT1WF-BY2w-02 zUhO6u+%p{qT01 zrYK`q7R7m;ZD|K^?+(H_SITp&J4%!+t-hklvq^^^^VJC1t+iZ>M^l$d&AfEwK)_^dyw{zca-gm|{vfhh~6RR~kxk4mL`tN#6YEa>`%L?HKD z>cyBA)U6}zTq?qqakRR{t;7HR@Bd=sku#W+xdJJSC7d{a8s<$LijHr#=H4{!+?&PC zNVPvrUa)s?zFDBa3o7fX!>v%c7X8wtsc$pQts_&1rk-;jo9A<<=5ws)&3nz?{^l@3>C~9JnWyy-Rs`wC#bzS8o|Mu=3yuOkBJIi`Os)`v%4sUzdetiSseEUr#u@ zc*36dsSl-((tqxIp61y4+`RHd_`W}b-gDEa zI!k?5)ty#bP`fG}Kds(n1AZ1%DO$8-nxTKAUU!3?n-?lDR_xWg-{PmAe@4j09SB~v z7VEdJ$G*?E;n3b~*t>N(GUI~LzD=u~y1*UMyY}mHrVUB==H6B8FRXp`FMqM;JFWV& z>gG}U*EQCA9+$U1?@@WZ=VcvW0>&d5zE&gr3A5arsm|qwAJ6%5 zJh)j|2iLm~%CVC>N(Qt?ucdp9I^f3F-(cqIjad2VHhlK^b{skI3634+J{<=?!GUed zF>AsIczIQUlgt0<+|<31a_=radoSx6wFQsUO)cvi~gcXVdqurC05n z?8CGT*@sX#2j;O-T~pAuOu0$)r8i=?7_kKy>$=#$aO}JKm0<%hI}c*+x{cVjZx@an z{uC#Ue2P=tr{y&Fr8#-%6Ku$gNAtiy)&ZW`<(@jHPyRz--Ef|B4*CXAIkf8MY0AH@ z&&v+1&v{&4b?2X<-}h^E{kvSx{rD8$(1`KR)QBnJ6wnq85_TB+*ZzEgkt;B4)h48D z-iUn%cH#{8=+VAJ7f$WPW$sIK>cDnP95sY(KW!kF?#*Lo*Ya0iEclY@DANW^-J9v7 zbeLQBkGd|e^V0pFsn7Sj=#)JFs^>5+k`d?F(7$Ye^+8eY<*@t8ZNmnpZ#fMA@D=E= zcq7K8FT;w>YjO1G9^ATe0$<-ehkIXM!rhx^khLfdqplGE)`@0e$GnzIWk>4YsDJ<8cK$coJ;(P;Q2*aqZ9sc{wOy3m4>i&m&w~T#TzyIb2V*w6_i@VoDF{_L|CfgUrsk->W3kCH} zYxYT{LzUe}LB9*SUb@!!$baki->7fO?ickVb)RC@zo(}+?0SyH)%!ox#@W*c@a6SW)c+R-R3H3!?TfqyCLp_of{)DI3cWb3=hVQI2cyVuy^4zbh_Ehde7tZ}D zvzGJwa0jklIE*i^Fa|8^gZ;ZVW8JC@yx0BxN7?}2JBROAJN|D`j{Zn|P_8c%HiSDN z9V@n=_Vv=x;JmYIMeO-(FTQ1b+N?`o;rqY; zjETpup~j+5P?LQ@)d61PV^Np;;11%xx1;%8Fd{AuL22n&vStO6sQ-wVXv_->Lr_E{ z{OdF{`h!n+w`h(eKlRu){3ndtKDDh=CaSBeQ9gr+wS?mFKU8P)_<{cZ(Y!nR?<2TZ z?~2QJ(3NqgzhWHgU0>cq`LvzzPuYqp{1&J`BOZ;p2ljj6DHxxcfw76{7|wmLqqukd z;?>JBn|rd(3=Tz5cmyWSobzjGJI^{{2zmaGZ6C`yYE7tw-f&YooC0BF3*_L zf?`X1a4yK9N<+NFeS->(S%kuqvlxr}2I7uiHTEKMhb zxAV+}7`l-AQ%58sG9v@Y+{=A(a2O^Bg<$HuP<%Xo1{{6rX79z5^R8cEld9{YXa{Bf z9qmBvT}~hE_^*)3qsmP7`J7nc-18G2{HE7)%X1&d^1e0UQMnGv`UIdxP#a^v3gzK1PAY1Mz%|VR-q&aJ)K*duz?*enUhl?!RQi{dH`(7pZf9?#(lQ!W7O<18nw6=tUpnq({SVu=!56#_d$VxzC=G1VE(x}y^+86 z2gqNm7Ya4_5Dwj^!Hav|SLB|oK0%rA4bFnkv}Aa5f8DpHFXUeJ@fa3K{j%Mkv@j0o z%QLZa*G8N=vJ=opb?-X4aE z9f!h;@FpsD7=j8N2BSXhq}!+&7&IpwBf{e_nq_$c`**^`_*BeJN=0-=IyP-xgFT;a zz}`I@i48b)Xgkgw{sdR3bIl9>?aTAHfBOQ{bNJh>bNH6|Z?2!fIj$ery*?dVmnUFT zRxDO6j>f{sP_+4A0NiUghie7yXYq7)@DpV<=g(RfkoEzDcz95rf05T-5G)t%J9fs5 zIiYCQ^<$Rh{%F-_H1#ClY6ON{FKAaMQzB9uyIy4%= zan!S+{}jUp<|n5jEHxeBixy*HMkb;bFF`Ez8=sYlhG9OAg%h12_S@yr2SK6?;X&mY7!j`Q8PbOc{qIfh$T zkK^{$leoh%zx!WZrfzTJ@88|Q-+#D^pMU%YKQomL{QLv+KiBWU5Ng{(HO8G0W(r4v#e!^Ua=g})N>5W zajf#omSRCh7Q)jPBa+`@u~}J2T$Y72mhp_WE3s_-Dy&?$0&Cb$Tfcb?Hf~*u&D8z2 z?d!39+ZycrWSz0@OZvYg{d3LF72?|YLx%ouT|S1}*ErsL?KJg&8h^We5kLM-W&c)= z9kBe%7EJsgJD^<2>;H~v2N!VY^R38Sl7d;Gp{UbsD9W|zhZ2ptaxd}* zPh$f=Q?{ejm;NK^+e!=RYnP|)S>`qGTB7z_aOpUT-_Wa&xMnAux&L(C9%I<1hhjMU zS>qDYF*Q94von?@UGW*6q9-`?{X%uFt!%Q||cP=kGH+ z0`|_{rjPiV4&z@Qhl5);S8K-7)#@{5mf8-#SFtzTqc%fsb-(#!G4Dmc!ynhw+J8k3 z;G<&syU$$kxAX6@-)U^nbF}(Q&(^jBN7QffO2yqWSKVf&>$aQQRkTF>oDj@$Ra z6*S}VCmta7->;{zCm-AAjutoA4DsQ^X3P;k&m1Z9LodJhw4ONlh#q?NM{oYT|PBT2i2E*HmcO+6qlCtkm-A&01B9eFk_t%<=Q9 zDyX}|0mFVZ_4hPly)-zmlDJ@1WtB3ks+C<+LtIE4$T1r}WD)zV-B6`0d%SXk^5IEg zO_kFNyIp|guP;|YC2=B;7j3H1I(Spad$W08VO6Ed;Ymp~yh3Yot83hQjT1F!%f>C$ z+Pq`4iyyZVW0>7XoUv!m4(-R+>|vI$dgpdHv{fZ&))xHHqfhMDA@G0d>4){?QxEB} z#~y%F59$DZY1@_>%}GhqxEYHyVeT^Zzkbv|!hu!)Y~1TUAMC3xSoi2vIQ5vu1MgD( z8y4uLw?9E8VD>+@4oZ zrX|JewXC8>OG_)Y7+aXPPD|`|L8+FOQHL$tpiH=6SY^P6H5_xmHrMoE^G4-usC51! z51q(I3kuD57!O$M`NjcoclT?#VZ12fxgz5XT*!cfcK$VR$!@PIT&LCWFP*qF-R_e! zuGzr(_}pSXYaP5RBOY~n!qIM*5}O()9$??eV{qY7H02REaNzL=wf~WQaA&)63JW!9 zZi?#s9F;WHc)fls6~ZkTC@%yv2MNQb6jM& z^O$jBPJWs0&MMJDW_5~o?$WyLJC(I{r)ICK(yaVatpFdV4fu++_=h!^VYRBVTC2e; zqnzifs+4E=<136C=0EcBAGsCucX=+mocseHvJ78jw%m@zWlEtQv?Q-sOZdAM7Oi)G zPX-*xD=T;X!K&?>wE-&!cZ*R9H*Mkw_sCa_7d!W||BQTO|Dz8$PVCyhM?1Fj-1;)j zT(VMk%u3NMGZrf`xzpd{!1sLrHhcU}sT0-K_AadNkk|Wlze)ARtU8>j4$Bucvs=Ln_+)Fta;5G-h>?CKjL%71iKgsWs$_ zshnd))kbG2;J9pkl~RgIwG!;oop#hH2fT7v$9eFQVn^mKJCLJ=ts0*Ymnk z&8*s?dDS~LYr}R;EZ;^9QKMOvJG6MqUMc6rJwi&e|fpJH-b;ux!GOwBG<3ZJ!#$5V?-FyD zipt4P%1nEhF`2nEO}C)|qvo!3y{*XTi{^)aLyLbiKfv0*4D4UK0DsngllDWk;h8s- zy6-vtVM38&)3>S(v$Z|e@6*tAJ9O=;GF_Kjqg!$|XmoakZZF!X+w!@M_K!>})CjQZ zL+`!i^en~CFH+K?A|=ns(Vzv?(9_G+Z&9xE7sKrRYu9UL#Wqc^-0t{uXX#emwSGJL zQK=~$=WV=SbF1&uWMYdE*_FC(d7*C3t75N5weEleP^SvAFMx+(1Pf8JS$rdfI*`XBlX?gW_En_xh)utWFg#*=lc9WM< z$AAON3rmy=&Wov6&n+s~?W+ql8ozbh%6#3iivCi1p5|m1!3T84{1Y6=DJdcEsM0#{ zH@~rl*@9_{QZ;sdnxe-nqNkXA(KQDa2Y)LKVC|2;0DE{D(z3IX=9cQ!4?oo(CKjpj zvTcga+Nm~qJJp_RakN{zS*;fqso&aab=7^TcE_nC2BJ>O9PnA9teM~Pc7C>*~XbFjYBtX&97G9RplB)uYPnvjmGfU zZQy-Z@g_~L*p46Bq;dIGXvumF$tc$le9wr?a@|RcIS#uyHD9BMKSr!9&`5mC2skkU zZE?3(OY@^VJ`I41*h^RuWI(S8j(7 zk%q&GoB8c}(}7#?KO>iO8!n9EwsGR7rP=Du>w2^9 zhou(s{tAslUq)saUy5|c$~=wDUat`;dAbXZEw0$4Mbt6JvDPQ0fooqyJa)9hudG-}#X z4V=A3N!O2K=Ha3YQ4xPRmQxW^S(9Eh|%7t1~QJ zr+%xzn|xy!YkoA?-GR>;gI<`9bYGOO9t-o-hnV8pC3(7jb&0O0?|(h({YLN|ieKnD zKUY1M7CL`54&QOxs$$)^x>VPttN+7))?0GG~$ZsMH@6B8xAeV(4F{= z39IwKzf_Z$C%6?3j0FFi(SczurpeOvi?ehC_+OubZ=0K;QS(=8MtUxB19>Gij@fBh znv{-4(c8ZE-W9rLQJxaJ^uMSM{Ms4-_BLaEz7HbbyL`|Lt=xB1OZPpm2CH^C>_aLK zD!Ae)`4t{elchCkFsoEoPc2lld-K$MPLaaq6)Tiu(9~=-yLpj($FEYe39IEbY`!99 z%? zE0wUINbMJ+--`>?ZEm&(5_b+^J>P&I=*R6YHsiCjNS$Y7s2gkC*}NR}Tv?{B>;dS@ z4A@QR&1hoPk?6``-aBx4v3m2K{v0hXxqd~FhO8`B|JfNDxU9hC9OJXfG%_Vew=l0T zb~%<=tlOC%8b>@ab`ASW@H4lP>kLP41~1CipoLi)v@}-(7iO!^K@EWTWAd6J z-MKi!#l$1w)1b*qG;jvJgBfW**E4zx4fxgeo`C(cKgB;VzLR>c+NNWtK2w8vH43eI zTH#xtRm8UEuxHhrxk?|NZDI{d?iSlpKXEE>rN#JO#1F1MvMmw=6~rGUPiU zUCr)FSL3_0_Y-*Nbn9T{Bk*~<*RnEefR-h&M3R%2f4Q@|U$UN|eQ$Z8p z5%2REmn#3U%at%Qo4RA3I>LcYbF$SFeEX)BsW)+EhXn;{Jv&G3i7`7a$WvDy?+or; zmlS$xlUc+HV0%O6dR>EN3?%pH%laF*vP9h%7pUFLHEMtFYIR0SI?vBhS9%%)sb^V^ zdL751vxwi*@%PJe+}bzoAAt|JiM2l*Eg8;n2zgEa`D@j4A-r3_-#2NgdQDGLZ}`-2 zeim4-Qh#`6b+RE-sSn?sqJDF;6xVO~MLB>5T>ESI0CJ3aV1NEh2W$U|j-yog_$wN{ z{t-1^^MImuJjcEY!`@@TrH`oD>TU8(zfa9pZI}1*&8j!0K;AQ1`?CraJS$&;^auQ> zWU)VYi2`R7$Y<+-pF2OI(CU}A*enfCf(g{4#}d|8#6FDO&s)Er%XFRvS) zAs=FhCUB?GElcHnPqw^oU8>+|IZ7h#Y`eHv?JbTY=IphkNPSn9GviyX ztWY1rA71phH&y+Y6{_#Nwd%zF#U6JrSBvq>6dB*5pq88*M$?(an<>WMV1naKEpPd)xq6cZ} zv@qZ4NfO+Oymf(+W@V|%v{mY}B%k^hH4My-Ly0MkAHBiZa@&49-xYoDJSA1#Yw_=g zU+O-2g?hosKD?*RxW#HQYPOOmtyFm5k!Zk$u^?Er{1x!O_NVv<#dp-dT{}ObhXd4rGIyf=tbH?hU`LAw4cjJgpwa9SHJni_FV=OlIVJL4 zRH1qkvmO54%QiaS;hS-v{4(!Tb2t#SYNMi8Z&qB^4#i|}fAuC_yH1Vfmg#DA#fx(W zrc`@4K&;VZ(i%0JnyX+q=5h}66XYGX&#&{UO7+g%fZs>+!MqFHu(-0-^7U#pzd&t? zCEK70N%ITUn%J@<{flI@B60d!B`x4Fc+-XT-iMlIKbI#Is{5QZYBy`G67eeuvvUo-sceaH%uO)f4gRmj=p~dV@NHWjpFkaf?b6xwu@R3rgir91%sF zlEB{Zcyf|ta*7sP zE&i+J?F*GiFFj%6YK8T>>0)t!UDMpIhjqzvL9n;ISwC_O{QuZtwEknjOuhHXmx`}` zx)%Fm4*R6X-c)$a(+bFcz+u>oHEd>U+F`$VgTvam(0GBz7dZU6?X|c{KE#U6(zePM z&G3Uee&~pqFTTJR4g`R^FI@6Y-Qwa(i!Ck3s85dJwWLY`D>f=Hb)(akCh(x?_;mSB z%XhT~|M{hAc6XM-!8j6s60wAOgz3Zl5`~}@%@+}y@|++0gaU~>f_QzC$>b^I8PVjU z?a+%3VAUQUkq93`?^v!F%U5_V9<7L-N-c{u9}gE2u$Xbon)8^=mdEk9)ifcE;epV5Ui-wMnV#-*qgd2KSWZ1S{BC5)V-m@!M$eB^Ad zZ*<}H2#5o(zr^?-Js1B7eE)xT9;g54Jw+dU_LZWx98rq{Z>Z%X$JOfbx3D)Awt*Tz z;eG`c>~|b+d4R=$`1G^JTJ=uZC@+rY4;s%amlqn~1Fp@;8w_v1HTTPJ?JlPSrU|AY zc5GT}KJcOGicRhurUO@w$M?gtKseyTXSrBna*l$CFD%9gnq4Trdvnwb-w;MEDs;(u ztXzT9^Av!&*HNnqMk{=YBYp5oO^8+Mk6n$Af-_tvAYx8|LZ_@%Bz`2E943nSrdaaL zI4oh&Iz>-dsd#Fc&RF{~cP?{!5Q{%Zn3+c2Kvc{`dlg)2>#RIABivnpLZInQZ< zA8~-`fEUj-Fkdint%AWdwATE=KX_(=B8U&8!9NC_k0OVRASa1}7h(7TuPM1|GQCjV z;NIwNID{TV)6b5yc=aCgQ|b+2)F499;uxz{fqy*tM*Q4-#jw`nj041*(RVIa{0#aU z)V1R#r8$3KJ|PZFBdB+`xM!8)fUW;_@FHQ{66)!BjvE$hw6r`0A7J>~{W#7SF=DPF z@1z#cZ`9BEfJ<`yAmYjY0Q>WL{eCUF*Wy1xC&9l3zTeipVQ;zJ zhtPos<)6LVt!G=`KB=3*uGY}+rm1kkIDqeN2F?MwdmL{9$RYf*c6JMnXKlR7RWz>-td1{iOtp6zR zPfS^_DB^>r^UBn4VwSw=ZFr*tjqb|CUz91F`gI`rYY-f04rc=HNJSG0U41*&`~Wqp zn0wbK0e=vO53u~w;(`RY5IZpqE@Yp>fgFXS3%2eJ`}V{LEhl*Rk_3-h!l71R9*>43 zEdYOF(-vq*)F^6YW61+=Tkvx}py#E~fD3c~MENB3arpm7_lY`<{nsMa8l!Qb4tvAj%m+UZ!1)4JZgx3B{RwNq6zojjS>IN72nIK+Cq*vj z8ZdAn91i%S`^~`Ik9eff)B>jgjtkVKj5p2UPZRji5KZx+F5zRZH$6cg!tpW20jpuh zf_n>chIrO(lxcwT3FHO%hXih$2DsRe`KYAn_<%WCE(T~bEkm*Bgz+feo+HkP=RGYf z_aGifqW%zd>q14{PS0dA9AC+qYVU#r7IuY1-$ z*tb6Twh|tERbj-0R{t>!gTXJ9>+c1l13~a1ARm8Vc;mZ0{`}k{aUc8$f+r!oK9qBW zmZBp?7E3&MR!2NBB|IWV^dg3E9B@2<3wEr%-3c{aYLe2Ylg2qeIw~Kf-~CR6byyPU>0H+mOaK=W*vj8Zn309(P)1he%&n&kGXS&67e0$DP@YEw2Itf zh2qewnBkM%bxzO2iFz&C{fp2*(zd!N-R8dAS-F()ym&AwSJ(saO|{#FFnl4*`UN71jc z{4iz)xd6V~<%#G)449i9IQ-Fs7Q};X$q6jaZ%wTyX)3j&rS#3IKbVfhf_Xe~MH0QR zM9VYjbHq*0MHdz;l51O>hZDH~*tPnl8o)Wu@B7;N?=nJv7`{S(yd@3%zfqTmUsJ0m z-*VX7F?RQhirD;&s{wd0!vBN4^9L>_VC`Bwh)?$h2a5$=Twm+g-TJS?-~2zk zaEBL$eKFYEbES_uv*%0)90yX!2hSROupAS!SkcTlVDW?LO_TYo_u14B@6H9&^{i`r z1HK@JII%6cT!)N}icPIjQ~Z5nt5w38dUqLL);SIsU+PWDQA5u0XEdNbuc^=aZ-j3M zHy=R0XdH-v3+5BT?qJO_*Alysngc%F>d}q^<_Cxul4oZ-4QNZv(P{xLmzB%)Ky4m^ zwI2sh65&z;@kR8Vsc1tk{Gir+kycYJeuxM-Kw$;sd}vjGA-gY^xt}eKG0=;q;|^h(oNm)%@;s#hM=g zPpc0(4Il<+OTDkvlvQd?9Xf&a9tp;Y_z}~9L|z+D9vVje7lA+WXVx-2?xMBeUt9+O z|HPl--?aNkU3Tkg{;h@j?3?e^1-(x+AHdpAqV^NBllnjSN756B*!VPSAMC+07$4w_ zIKah)w#LEIf9?Gq?5);EozRc_0cg5m>~QA&#sypV78BZIhIe??Qwk@~h^YC|EY9>L zUu=rTds$rw&G%x>yQAGU1Ap%oTjg#3hMt7QkfsUFH{gewuiU8S%iw_J1gx*lYc{EO z-Zu4EyNPRsu2(a9O}6G6(|fY`&~)SKDS7aqL|5LK;o^dZ=1-_;hkmBm;7_l0TDiXX_6K$5I*Rec{7Fx| zsg{p=YX63RG<~6H`aq%UiT%LEVgS3{+-`$mDDz^b?dBU?o)6}AH3B=<<$uN>%L778 z^Rea?Pn?7O)8Jochq!&neU@L|UiqLSO}Va%x78i-4TiVni%tU!?^X1qjC1hCFN@D! zyGMc9dla6xN0H1gM$l`Bzz;NMop#LHqF(vinKz=3h!zA`{zs0`i1puq^=Ey^t0(0; zKVmt6#fA+ne&BUhV+v=k#WcXKG1#1I%-HOk;U6(8PvO*ntd63j4`ZE%u1BBG!QWzfyXI0D ze6ZWLuIp+({$OD1-}!(#>`epEcjH0bx(_GEjMxa~Ke2Gr9L^n(N3CJ)ZhBAD4XpRb zEQ0xmrl$K=Kd`=J2G7-Uz&Fby2?wI{b}MrA7Um4{s0V^U`X+VF+N>`0jLav5 z&L~i0>tlkwalqylte<(+xJ=?m%MZvUIr_~nQzSJbTkA3Om`nqLCot2Ae~2O`Xg)FB z;U7fI=sPh(p=f<8(*W>qO%Bj@d6`<#!#3<=7M3Vv@>&HhTCV{55@FP|BFI1DdJMT( z4rn>RFS`CD@3;Ej=jUobaOo0|G z0)OJnh|O?d(>Z2*;CbU)05OIi9PnMUOU=*&AN-3CF^20+a@*p>CQHa0G4lyQ#3Nx~ zACYgiR}uU@;ptmlO~~{hE^Q-yVdezTgO*?!L(Ql$F=3;5Z~#5H`tCgZBJm)Yn15)D z1{e2SivL02hM5-2A`!hDd5qAycvxG^d!pV-~$Cv;tqy2K?KtU`~bH&-xU` z!RF+9q4#Dhl-pNNFJXQXZo+~1fg_m1MiL)Ja*Wt$cs>P3 zp7iLv#eM5MwW1(bKk(!LhG$qcJ|BN?YdqXA-(vW49*$8KbC7pN+I^n0*V%il59OD= z*Tn_C)=MxwV9lE~oWB9z;L!pf{DIF(Gy?M_J`BKL1fdln>C_|XRfevpQ5b#2s1?+OOK6R^$dYHyxP59QDisg;QGyXZ<%`Sgk*CEwg%e5?@|3 z`l7SJtlzGe0RQu5|7<_f73}Z)ukm@RdG2j(Ja$sPMGq72nIB*Vh}l2$2XW+qv3t1A zGRH`Gao_u&s_MiUg~AQf1k;CTV#Fxk7m0=#W)ZCIu*xT#S!~dp+lF-*SehSk@t(sL z{B7NXb<}odPdJ}DC!ZV5=R5q3Bbdd6mUH;!zyUPg;zQ>bYW1M8)r+VRS%1nC3+$$* zu(!7VfEMwdAm$GOi4Fa1EmH#sm`dzU|Jd^SmaA)A{n+^eiv_6_xtaplTaCc5H!Uzf zVRK0~`)2DoaPk^r1J*x!5H_FM!adZZsZINgUoD?0RvV@#4F0XjLz1W;1@d{}#1at; zuqBoBPPcF^yFBu+4EglF`C|HSHGuw?0RL_0>VFhH=c{j+qszu+Yx>T^dhL_1` z@Wb*5Ti=o3?QjM2v5z0q{AW+-jwg@lhDTplul+B(SRN0^_m_#=?m z(a&mz#slMkD?1|DOsjrgD$^9Uhxi2=wh0>-C#Vn2EZJ`*$PK{Er1 zE?7??jF_<<@n*Z#Rcc9{D0*eJqPR}A^%x?ohDl6&IcxB;(Y(ITtv`qVjh6s_yQbHV z@g?iuxBGDY5&W-Nw_jiX@E3)b5y!Ds4gW~+kE16P|6r|gI}uB8MvSn}>?Owu^A|DX z2{EP#rW=Me_}V&;wsnn0*esc8i{%ui@!`C_@~zWK=D8#+kOu zbACs}58oe?TFu;0xx6hNL<6i2@aHkBJ@^wBGyr#-MX=nUF@1a6^I|^0@AehUwHB!9 zG>n-wpSzfYMhCo!5B>1CJn7YFQV{8zALxu^1JeN*{GFj+Gq5S z_?&|k>wm;A#h-JWUjyd+p-EkIIrs-H-lXsT@>ku?`io%ZGwOc)ziB_~-1?Y5T|8Rh;Q$(Lm}3^xMe^H?al{YKPrw1oC5<1JS57%{LU%p$ zw&RDbcQgCkSh&yzt!dABW4UiUFg-FJn5LKpm@n`ruJFr+2jm&fC%}VdwXuQ4h>aIj zqX+bZc+Bzw>p`1Y-@;-_J4X<4WbiWj%+v$I)487H+U@eEp52f-QoSjK4*#pHHi+)m z%_n$z)$}K+6$LM2RtcZ)HF1rrA9%q5!{3jbz9}^VU+|A5zHEyIv}Jy^)l%|+j7=U6 z@LqpnlglUM>kl`jsA=EZeh&X%yf+N|8?*lZcAf?>19TBvswipS#k8#<7WwBvQoAHW0Sfc2sSi#--# zaUs1ZUvdrS6W~HKFm~}FTF{gnu`xX)%L^O_j0e0X*m6(eymS{6Ce;%9R_Qy12mfhx6ZC=fMfbk%h zm@kPMaJ%$swW1E#mb|m&+HJ&~H7+K(e0;9{>-xp=>HF)gf71`Jf8jj*i2<4no2knu z7i;PRFY22g{;JltR#|&-yFHfh&}$lZ^gX4$@sV0`JJImR63~G-_d1IWEk6Kzi{HT9 ze7@xa9>aHA96+9s0FI&5cFY%q;~#9^_|8LbY09(jC>s2YZ+6aDaIZ7VN8&iQ9jzv5 z7EW$xxk{k%fOaA_GldNf;aWZ(Olih>df>Y>o{OKP=9I>d1noK7z*6XQUl@!pIZO#H-&w1 zPd}w>MalNz~AZ*rUi9o=QqwJ zQI|Bc*(CEj)}IKnUIhNY&8HMIYhrZ=n={EFMkHS}e_%C4AGE-@(1hBtPbPcZz`d@f zY4vQc<(`^iQ)W*5@f{W;+U%Oul^f0@FC?xs4q5HdaR47-eF@WmMhk2Pwvsu5T=K^X zH72jHy(|_BHlr30M4abKKOzzybO3j&Idm-EqYnILNf>qV{~EDemtQwuem#c$9R9yJ z{x|&TCG|fKe~U5DfCl4MD`4#d?l)DY9C%ex=x+iVZ#c&ji#e?64YZ&(=;p(D+;M?9 z@j-`}6@n9nS-j(bX*E6sEogy{Ppn((@E`yV*i2zCbt9((_yk+?!JD5|$?=bL%hSXH zrYW2&4&9I0^`bj>o2TDV`)5B;$Df;Ao{TUx% z_q_~%Uf+aS$9m+77IR)XKHK>N%O9G+0o%*uLyy3ZT*C|B(v%)X67fY6d*RyG;va(l z|IeX|^ylm5aNV67e-8g&oc|mC%#>Rl@YH!6Km!80U9bA7_v_@>-)S!~xo_pu&cC-{ zCd6>IS&)|C-|_%!&mFnVbGA?5o)=GQ`^nFhz-^ZkfL#I@n$O?*?k5^fUN`yaHx&mL zEdCF+x<7RQTd$#&Pr5yZ7EdUh&(Cyjx)a98IYXA6gb=ZAWJ-0onK3ktq?=6q3=cY&1YJ;a=6+}HMkb0Ei@ASfS zpL$2rENYcjmn65eSi#%$oqD2gZN1qzV70~uv#9~tEDC*Y*N+7EMz#mS=9B3qHY8uL z{R>t%XpA4R9HBn_M5`rRU%3e}hUKR=!_tzy4@p_~sa45-d7&!~{~@#F-}U;R%l|If zHH5*w>^%I@0Q+5qE19d?_Qol_$1F(0+yku3mz~bKSwLIwU~e?g00Kz&wa`Z?$aO_0lF`(6gLWv11H!vJR(FF7378Ar0SB?YEvbWEuJ?A!^@BHE^hr40l z4cvRY;IY9keW*!j#%(XZr@=?xQSZZVI}UVz>bQEtg8_SwsMq$V)NRWXip1xbpLe)h zyuj_?avn1dnErFWSITL1#@Bwukuzq8s`OFcyT%uP0jRWXexNd zfF1{rI~)A;8|wG)i|V%Puo9_FTkaW3{1HNp%WyYe7?y@P!IV?r|~v4_}IPf9^Sm1IJ#d0rZ5?pEs!kfAH0J@-E?f z+RaQj__jUrzS=$W9j%?&wm3iTp*J+=l~cO!^jEs=_!$j; z^|S`ThudENSiRst4`Pp-c)sWJAE+le>X7H)0N1bTe(+6o!+OBuYoCS-kG-aj`>1K5 zFJU&%g!Y7Pc*1cZlzMlF;{<&x$HB+R5AIik`RrpsqrFq{*~E{QBU+!i;R5!(fu+>~ z&c=b@Y`8Zd?z8ycG~M!nrf|gSP+sVLee}V4R2Gj05O;WSUOWF4lL~bC*tM!Rg6l)^ z+cy{Bg1FpMKZH-xiD6e2}gG zw$HqyLHOPi%#LKg^Qrm~hZ$Eoq7OSi`aIwdxs8{uQ@(m4!1KsI0w0-DRcYfQ~>SkMw2aXHI2QxT; zSzWOqy<}g@A&CzSE5pue%$5UKENB|w=1|V+fQJKDvzOWQ;A+;j>rYw^NgUA#KhPv) zv!`BB_z*KK_j8S{E&9`4dAj_LY&9G)U*T~VUJu|>?*pvk59ht*r}*1%8N{^jr8A#@ zrDLCdE#I1F<+}~N2G@4PVHStmI&Viz&<_0DaJ-rCeJ8rV;>|Of^xA3N`pQRI^wuYO z@RM(p`W9>U==;PEC*aUK%6<1UjfM-RA+51i<|DAy#1Tp8eH=Mw^sblKdkzP7JP$9Z zE!O7!ZJs)zR-B_fF+k^+KT_{kPOA@}-RGr~>ivQz59kW7dOY{O`jJBnJ@S@%gL|hZ zkE=ia;99IR9I*T)o_xmYn^sr0Yn(VuKz~B%S(ycrE0{+62k5?; z#Ren6e8=f8wc#ZG|NYO^)?zmNbQQY2^u(vS>(x{0h_7gkeiRW441h90PWSh{N`L4Dg>R>xL>$sQYoKj=$;u?sbdA;g9w=r<%0r`Ts$UTd>1R_n0ZbR;}5|6{sfd8y^677IB1 zYdPRw+iz+xi}`Z;%~ls`On;$?%_&=52#n3=yZgkCw%$E;XzEXJp~*`2L)y8}7qfaZ zEA%JkFaC5pb)rGzFE;;g`@k>B9>9xn2MuTq*M8CyXrcI)?ey8_pX(j!{!KSNC*S*C zQQYHi%X04SFMZ@<^7h08?ZMviGmFs&;g6sC>{~5==QE|S&IiDWj>K#mKK@#_PB6OF}i?zC-lC<6K|+JW<2P5A@d`&-C^FGWc z`g2aJLH>!pi`D+G8l9@ZE<-LB`)#@yZkO^g@XtS2Z|rt+G$22}$Z0?dezfV1m*uyI z`YZa|g>~Hx{dM&>YI2=wYjJnKaYjACd&Y^+wE4_8S^x(|96O_q@TL6p*Shu1Pc-D! zk2H~(pcnaD5n5r~>4o*fN3?|(=2P0E5ogVO3_NIMYZly1yDc{`+}eV-amnJ)_U0GR zm(IMt^FiYPx)1&xABPKOaH~7db?5b6(T*JAm0|Q85;&KuXIej_jt9gBrUOm`tY$>L zs7?p!dJ2ZE#qwUn0*&YojM>g_M!dr{H2D2d%j=qfZ3AYKt|o`D*x&TP$M7eXXtb0( z5Z_;KRwZk{K!0NXzVYzc`2Gv84a)i)e<`uP*tyV`i(U`5ZToIIefqS%VlE)I@}T^7 zzb5~^)MSVSyS_vo1_zu!KodG(i&^(G-~LRqz`yv^S97R*SEvx zJH4sBTVaAEhPm;U1{Y@vdhlDPu1hPbJw$4^61Oj_|iL?LA|M=&7t$0&8V41 zSZv=A?7iWM#SgAefu?xzTa||YpKa~m!M?*03)JkIJ6QV{%>8|f@BhVnzb=OPPaorR z;#v3S&4-5Zdwx^yo#`~-6@FjYuksnE0mgwA2j5k9a3wQ zgl621e@LMZRe1cg*1dUJWoFbvOJ6&w66#AOFTA65)EIJ~eM^(LPGa}UM-|D8aTD?a zZ)!T0*SYxM91dJP8=pY`DTw-p)i?sFad{I%{&{K{GhaEb=ig|=JcYFFMa*}>wV@yS zDNeuUeYEC+Yl7SFI^Mr)kK@25zK4czz<S3g#lmrkoI zd@xRQdHJ;Z@>m7_|0cA5>bqZP$nj5=`r%hva`H=cefasWUgrc~Do(BId*AUv_hvuDEG|d@SxG@A|0^29MFdCf*l2H=jA~ zr+-Ime6rqs_dQQOz-(6Is>99?1nlSf5RbmC&aC@hto7cM z?a^@W0tdR->uSyM0KZ`zaXg?u+3vApYJ=afdV|IPu8&0=VZOvY23Ksp(PmEEOe6g( zH>XHGX#H!OJF?!T&9b_FAu|WI_t9d50Lv4d84u_^Tb_UpSo|1LK^+Zq*w10scLKAX z?7RBIuw~@@a}}C=(ch0djbHvnGyfOE_aAu-4$M541Ev9khu);`zWYv}e)oe0;TM~b z6Zr2troe}eE9hag_$c)k@bB~Hr?ouz1YAG%_z3Xt_WCDUc9I$%vB6-pU?O##1!zFt z$KR+goao2<`l1beIodfq=iyv$eB!v(!#G8LVK{fjPjs|iu=5ulUFZT&x}X^zg9rE% zQ2_HNP-JV^qk_*0GlMYd-x>6`y~vYu@`(1Mv?7-|^T0Zud7{yj^FX zsoz_lIga$=JQjCY9#ZEgx=<6c`#bH@ks5k0V*!FC$? z!8PxHscYZ+N;6J>tBNn+0Qa5Mal!a-E+^o_05dqz-_DJhUYLLBPOYH_{>Ew(UCBYa z!iBE2n$Q7VSWC`PO3t_G{m=9be|H;rkU$O4W*2S!yLndbC%AcLY5_LGMsL#Q%-kL& zhrefks`abf95b)6nRm}zJAAR;68$5aD{}i8$Rq7_jn?c{J+7&K#ne(=!CpJtUl`o3 z?*;z8?tA8YpZ||y^lQJ4zs2^ApS^CBp5z4nkG!qs zhfXT&$cJk2BEJ7Pb)dJw5!?(jThjwredjuCb%DA2q_X!p_yd&T2oyIMExP^&%JPiFGGF?Tbw#2Q47(sQmB? z*S~6e&~gdSb?#gpni_-E8O&c8Cv2A0_A9u#bGP0PIbKBD+#1-{?Nf0x%lJflt?7cz zDcdzvt=HKkYrn2uexI(GUZyK%+Dr)7Kkhid^Si;d>#7d-+T#oO{a^Jr0RFd}L5o88 z+g>{N?=&7xLK~OefAYxMSzK2Hefg<3TM4>a@Z6ir3Kv z`Vy9NcD0%$F+_WE39Bg>2kNeM=fU2Czs)$BZ*cosSm!pUXfa{kJfh8}o}EV}PjGwy zZ`&hfekBU-cojaX23$|O-a@XIy(Ni?nOS=M*c*-q-~a8uwfXcn8bU14_Jz|5f9@3akrH2KO}_gT9QZ~z zocdPR)#5%J4h%;N?C~KdztK=LZW~vInnu8b!Dzu?rwg_E!2LnkwQ#7;Tnqs>JbXZJ z$TN%weTX-E8VA7N864<_S)SU3n9_0u>t|a{p^gJBhy$%h(Y*&5!zSrnQYR+#fT`b41{T@&8y|(=&xcrvyMWX`vyPrIl z19i8J2g$8FX=zHTj-dyqKR#1O1ARiAaO9i6X!GYk5c_|nWa7g3H@{FfFuWf8N1pjk zqdxggH**~JFHT`}eRv0rtO2fA`nq1peUP_`B_Q#J;V={@kO}19Hab*d&d)^KLzK z@R_qbsJrnse&Wcte^tS!->U<9ko3-1>Vfv(2p2}dft$~K&+&VWtknm{1=9la2gC)Y z0cWk21LiB7-!R{S4WZU(F+{%;UurH|^7?mw)yePw>S_-bBc3&K%HHG=mOI$2iOs#c zwT}iQ(hsnHldCyfZJC;q>p_6MT`M_s`wQ|aKPd0Ar_>ni8>X_aaP&L{br|4g>i*8U ze~|C7;jf9)uYGQdSHQjUTtktG=(zAB+#ig{Qvx ztMWekUa6lDL*oCt!-c`8zteEIFai#kpD-I%%ZD5B7j`Bbn@!hZ01?|5U zOm9FR91q~euv)tbe^F=7Z}7e$ytWm&XWa2G6npFo>JDG)%fEVZR{M-WR(EF2TfW$r zdWAFa?}2}?y3&vHt)A;4Tdru9@DO#+tE@!O!IKzG0i}Gg8 zH{EbpP3eU-v^bwRunwc|RnKc~yy&$%&iO9({a@Gne}qr`8~+0yjNspW{G5OLkJv|_ zKi?LKZ-^v!2yfF>p|?&_VEJCPpkFun{crRPKH&`5|E-P;wMX+8Cy6It{_bx&L=15d zE2D;Z3;urqwSrF60ph{ja1W>V5J@f?`SK?UIsBpgs8RYK`aq$F-ltaezG9D@ROB-s zDE!cSYX0~M1wVXT!SuHLx4$TV@D9LyHy`!jUj3}RHylv|{Ji(--5Oc4S9eUAqvST7 zWPOsK`S;rIyietO`9;_8`bWO|f2G&hzn6dawR|TR%@r>CU72o%$GU4_hg~yVK~t8p zKlDlFOit+z{CE!79{c78{eUR^s1WCq!zAAL@g?>V9C`e!AG94!*+8$U99<|OE-WT4 z%%&GHhrF=Y(Nl_e8ZJEbp6g@yQ_F73yj$aqN7cB7xsK}RRKN1D>aicB<^oy_G`mcoR|MqV&9`GI5^(Sr4>})Oh?|f&Tb!1$y zHN`b?Ba%BS?AnnE=67}ii+3rSeF3ehGxw#wG~|_!HSD!B%KqR>J-~d)JKz3AUs2;X z`^z~o=J_l+|K69p_LCp}s?+2K@4&C)Uw)_8KmS%QGtYSB%-4FDe%IdDKTyphhm=`R zsxIC8DGdMhKl~et`M3Wj^E>}rO#WG~hX)POgfV#urq8D{SK?k^oIep2Y>;oLLUQ-Z+}u#Hb15b)dzHQ*#o+vXt##u-LKy4+wVANq?+TW ztQYz}{JkgnHxBz}eaHV-J{K-{@jXt*9>UJ{n*WFI_Q&tu^uhTH@BkOP9l@+hc!%B! z>oHhiJ%=bfxr-u9KiK1LzqxF+O8ea-%So;GYR7QQ_PvDv#3H!wUS}G_`~OZ%Z-0}` zP(O~%WzAo*y_)}5u>7TeXRR*u=lk1)y^sBf^S|zG^JNdQ)^FkOZ~9BWgWuop+233n zFGTXaj^O*XnBfF=!5#%3`>i?hHJ`bMziSeBcjoW>)%Iol&V?WMy4Ui-i|@QG-~VK+ z5Zi?v$9^qd80U6k+2+gn>`3gg-*Nd7mw)zmQOk>l;L;i!XgUmDQ?Qj-3APVAh*@3o z1?;7B%=W(@!XCw{uuN<&HpZ~FbN{pdeZPPHzYf=a$KrP^e&@jN9QYUJ0Ds!=|NZ_A 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..1a85e808 --- /dev/null +++ b/YACReader/main.cpp @@ -0,0 +1,167 @@ +#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"); + qApp->setAttribute(Qt::AA_UseHighDpiPixmaps); + //simple command line parser + //will be replaced by QCommandLineParser in the future + QStringList optlist; + QStringList arglist; + + if (argc > 1) + { + //extract options and arguments + optlist = QCoreApplication::arguments().filter(QRegExp ("^-{1,2}")); //options starting with "-" + arglist = QCoreApplication::arguments().filter(QRegExp ("^(?!-{1,2})")); //positional arguments + //deal with standard options + if (!optlist.isEmpty()) + { + QTextStream parser(stdout); + if (optlist.contains("--version") || optlist.contains("-v")) + { + parser << app.applicationName() << " " << QString(VERSION) << endl << "Copyright 2014 by Luis Angel San Martin Rodriguez" << endl; + return 0; + } + if (optlist.contains("--help") || optlist.contains("-h")) + { + parser << endl << "Usage: YACReader [File|Directory|Option]" << endl << endl; + parser << "Options:" << endl; + parser << " -h, --help\t\tDisplay this text and exit." << endl; + parser << " -v, --version\t\tDisplay version information and exit." << endl << endl; + parser << "Arguments:" << endl; + parser << " file\t\t\tOpen comic file." < 1) + { + if (!optlist.filter("--comicId=").isEmpty() && !optlist.filter("--libraryId=").isEmpty()) + { + if (arglist.count()>1) + { + mwv->open(arglist.at(1), optlist.filter("--comicId=").at(0).split("=").at(1).toULongLong(), optlist.filter("--libraryId=").at(0).split("=").at(1).toULongLong()); + } + } + else if ((arglist.count()>1)) + { + //open first positional argument, silently ignore all following positional arguments + mwv->openComicFromPath(arglist.at(1)); + } + } +#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..782f4b33 --- /dev/null +++ b/YACReader/main_window_viewer.cpp @@ -0,0 +1,1407 @@ +#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 + +/* TODO remove, no longer used +#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 doubleMangaPageAction; + 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"); + + 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::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())); + + //inversed pictures mode + doubleMangaPageAction = new QAction(tr("Double page manga mode"),this); + doubleMangaPageAction->setToolTip(tr("Reverse reading order in double page mode")); + doubleMangaPageAction->setIcon(QIcon(":/images/viewer_toolbar/doubleMangaPage.png")); + doubleMangaPageAction->setDisabled(true); + doubleMangaPageAction->setCheckable(true); + doubleMangaPageAction->setChecked(Configuration::getConfiguration().getDoubleMangaPage()); + doubleMangaPageAction->setData(DOUBLE_MANGA_PAGE_ACTION_Y); + doubleMangaPageAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(DOUBLE_MANGA_PAGE_ACTION_Y)); + connect(doubleMangaPageAction, SIGNAL(triggered()),viewer,SLOT(doubleMangaPageSwitch())); + + 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())); + connect(showShorcutsAction, SIGNAL(triggered()), editShortcutsDialog, 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() +{ +#ifdef Q_OS_MAC + comicToolBar = new YACReaderMacOSXToolbar(this); +#else + comicToolBar = addToolBar(tr("&File")); +#endif + +#ifdef Q_OS_MAC + //comicToolBar->setIconSize(QSize(16,16)); +#else + comicToolBar->setIconSize(QSize(18,18)); + comicToolBar->setStyleSheet("QToolBar{border:none;}"); +#endif + +#ifdef Q_OS_MAC + comicToolBar->addAction(openAction); + comicToolBar->addAction(openFolderAction); +#else + QToolButton * tb = new QToolButton(); + tb->addAction(openAction); + tb->addAction(openFolderAction); + tb->setPopupMode(QToolButton::MenuButtonPopup); + tb->setDefaultAction(openAction); + + comicToolBar->addWidget(tb); +#endif + comicToolBar->addAction(saveImageAction); + comicToolBar->addAction(openPreviousComicAction); + comicToolBar->addAction(openNextComicAction); + + comicToolBar->addSeparator(); + + comicToolBar->addAction(prevAction); + comicToolBar->addAction(nextAction); + comicToolBar->addAction(goToPageAction); + +//#ifndef Q_OS_MAC +// comicToolBar->addSeparator(); +// comicToolBar->addAction(alwaysOnTopAction); +//#else +// alwaysOnTopAction->setEnabled(false); +//#endif + + + comicToolBar->addSeparator(); + + //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); + + +#ifdef Q_OS_MAC + + sliderAction = new YACReaderSlider(this); + sliderAction->hide(); + + comicToolBar->addAction(adjustWidthAction); + + QAction * action = comicToolBar->addFitToWidthSlider(adjustWidthAction); + + connect(action,SIGNAL(triggered()),this,SLOT(toggleFitToWidthSlider())); + +#else + 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); + + //tb2->addAction(); + tb2->setPopupMode(QToolButton::MenuButtonPopup); + tb2->setDefaultAction(adjustWidthAction); + comicToolBar->addWidget(tb2); +#endif + + connect(sliderAction,SIGNAL(fitToWidthRatioChanged(float)),viewer,SLOT(updateFitToWidthRatio(float))); + connect(optionsDialog,SIGNAL(fitToWidthRatioChanged(float)),sliderAction,SLOT(updateFitToWidthRatio(float))); + + comicToolBar->addAction(adjustHeightAction); + comicToolBar->addAction(adjustToFullSizeAction); + comicToolBar->addAction(leftRotationAction); + comicToolBar->addAction(rightRotationAction); + comicToolBar->addAction(doublePageAction); + comicToolBar->addAction(doubleMangaPageAction); + + comicToolBar->addSeparator(); + + comicToolBar->addAction(showMagnifyingGlassAction); + + + comicToolBar->addSeparator(); + + comicToolBar->addAction(setBookmarkAction); + comicToolBar->addAction(showBookmarksAction); + + comicToolBar->addSeparator(); + + comicToolBar->addAction(showDictionaryAction); + comicToolBar->addAction(showFlowAction); + comicToolBar->addAction(showInfoAction); + +#ifdef Q_OS_MAC + comicToolBar->addStretch(); +#else + comicToolBar->addWidget(new QToolBarStretch()); +#endif + + + comicToolBar->addAction(showShorcutsAction); + comicToolBar->addAction(optionsAction); + comicToolBar->addAction(helpAboutAction); + //comicToolBar->addAction(closeAction); + +#ifndef Q_OS_MAC + comicToolBar->setMovable(false); +#endif + + 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); + viewer->addAction(doublePageAction); + 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); + + //attach toolbar + + comicToolBar->attachToWindow(this->windowHandle()); + +#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::open(QString path, qint64 comicId, qint64 libraryId) +{ + //QString pathFile = QCoreApplication::arguments().at(1); + currentDirectory = path; + //quint64 comicId = QCoreApplication::arguments().at(2).split("=").at(1).toULongLong(); + //libraryId = QCoreApplication::arguments().at(3).split("=").at(1).toULongLong(); + this->libraryId=libraryId; +// this->path=path; + + 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(path+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::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); + doubleMangaPageAction->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); + doubleMangaPageAction->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); +} + +//QTBUG-41883 +void MainWindowViewer::toFullScreen() +{ + _size = size(); + _pos = pos(); + hide(); + fromMaximized = this->isMaximized(); + + hideToolBars(); + viewer->hide(); + viewer->fullscreen = true;//TODO, change by the right use of windowState(); + + setWindowFlags(windowFlags() | Qt::WindowStaysOnTopHint); + setWindowState(windowState() | Qt::WindowFullScreen); + resize(windowHandle()->screen()->size()-QSize(0,1)); + + viewer->show(); + if(viewer->magnifyingGlassIsVisible()) + viewer->showMagnifyingGlass(); + + show(); +} + +//QTBUG-41883 +void MainWindowViewer::toNormal() +{ + hide(); + //show all + viewer->hide(); + viewer->fullscreen = false;//TODO, change by the right use of windowState(); + //viewer->hideMagnifyingGlass(); + setWindowFlags(windowFlags() & ~Qt::WindowStaysOnTopHint); + setWindowState(windowState() & ~Qt::WindowFullScreen); + resize(_size); + move(_pos); + if(fromMaximized) + showMaximized(); + else + showNormal(); + + if(Configuration::getConfiguration().getShowToolbars()) + showToolBars(); + viewer->show(); + if(viewer->magnifyingGlassIsVisible()) + viewer->showMagnifyingGlass(); + + show(); +} +void MainWindowViewer::toggleToolBars() +{ + toolbars?hideToolBars():showToolBars(); + + Configuration::getConfiguration().setShowToolbars(toolbars); +#ifndef Q_OS_MAC + comicToolBar->setMovable(false); +#endif +} +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 + << doubleMangaPageAction + << 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); + +} + +#ifdef Q_OS_MAC +void MainWindowViewer::toggleFitToWidthSlider() +{ + if(sliderAction->isVisible()) + { + sliderAction->hide(); + } + else + { + sliderAction->move(250,0); + sliderAction->show(); + } +} +#endif + +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 available")); + 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..255acadf --- /dev/null +++ b/YACReader/main_window_viewer.h @@ -0,0 +1,167 @@ +#ifndef __MAIN_WINDOW_VIEWER_H +#define __MAIN_WINDOW_VIEWER_H +#include +#include +#include +#include +#include +#include +#include +#include + +#ifdef Q_OS_MAC + #include "yacreader_macosx_toolbar.h" +#endif + +#include "comic_db.h" + +class Comic; +class Viewer; +class OptionsDialog; +class HelpAboutDialog; +class HttpVersionChecker; +class ShortcutsDialog; +class YACReaderSliderAction; +class YACReaderSlider; +class EditShortcutsDialog; + + class MainWindowViewer : public QMainWindow + { + Q_OBJECT + + public slots: + void open(); + void open(QString path, ComicDB & comic, QList & siblings); + void open(QString path, qint64 comicId, qint64 libraryId); + 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(); + +#ifdef Q_OS_MAC + void toggleFitToWidthSlider(); +#endif + /*void viewComic(); + void prev(); + void next(); + void updatePage();*/ + + private: + //!State + bool fullscreen; + bool toolbars; + bool alwaysOnTop; + bool fromMaximized; + + //QTBUG-41883 + QSize _size; + QPoint _pos; + + QString currentDirectory; + QString currentDirectoryImgDest; + //!Widgets + Viewer * viewer; + //GoToDialog * goToDialog; + OptionsDialog * optionsDialog; + HelpAboutDialog * had; + //ShortcutsDialog * shortcutsDialog; + EditShortcutsDialog * editShortcutsDialog; + + //! ToolBars + #ifdef Q_OS_MAC + YACReaderMacOSXToolbar * comicToolBar; +#else + QToolBar * comicToolBar; +#endif + + //! 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 *doubleMangaPageAction; + QAction *showShorcutsAction; + QAction *showDictionaryAction; + QAction *alwaysOnTopAction; + QAction *adjustToFullSizeAction; + QAction *showFlowAction; + + QAction *showEditShortcutsAction; +#ifdef Q_OS_MAC + YACReaderSlider * sliderAction; +#else + YACReaderSliderAction * sliderAction; +#endif + + HttpVersionChecker * versionChecker; + QString previousComicPath; + QString nextComicPath; + //! Método que inicializa el interfaz. + void setupUI(); + 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; + QString startComicPath; + 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..3a6e751f --- /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/find_folder.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..5f311657 --- /dev/null +++ b/YACReader/render.cpp @@ -0,0 +1,1177 @@ +#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); +} + +//----------------------------------------------------------------------------- +// Render +//----------------------------------------------------------------------------- + +Render::Render() +:currentIndex(0),doublePage(false),doubleMangaPage(false),comic(0),loadedComic(false),imageRotation(0),numLeftPages(4),numRightPages(4) +{ + 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(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(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(); + //make prepareAvailablePage the only function that emits currentPageReady() + prepareAvailablePage(currentIndex); + } + fillBuffer(); +} + +QPixmap * Render::getCurrentPage() +{ + QPixmap * page = new QPixmap(); + *page = page->fromImage(*buffer[currentPageBufferedIndex]); + return page; +} + +QPixmap * Render::getCurrentDoublePage() +{ + if (currentPageIsDoublePage()) + { + QPoint leftpage(0,0); + QPoint rightpage(0,0); + QSize leftsize = buffer[currentPageBufferedIndex]->size(); + QSize rightsize = buffer[currentPageBufferedIndex+1]->size(); + int totalWidth,totalHeight; + switch (imageRotation) + { + case 0: + totalHeight = qMax(leftsize.rheight(),rightsize.rheight()); + leftsize.scale(leftsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + rightsize.scale(rightsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + totalWidth = leftsize.rwidth() + rightsize.rwidth(); + rightpage.setX(leftsize.rwidth()); + break; + case 90: + totalWidth = qMax(leftsize.rwidth(), rightsize.rwidth()); + leftsize.scale(totalWidth, leftsize.rheight(), Qt::KeepAspectRatioByExpanding); + rightsize.scale(totalWidth, rightsize.rheight(), Qt::KeepAspectRatioByExpanding); + totalHeight = leftsize.rheight() + rightsize.rheight(); + rightpage.setY(leftsize.rheight()); + break; + case 180: + totalHeight = qMax(leftsize.rheight(),rightsize.rheight()); + leftsize.scale(leftsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + rightsize.scale(rightsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + totalWidth = leftsize.rwidth() + rightsize.rwidth(); + leftpage.setX(rightsize.rwidth()); + break; + case 270: + totalWidth = qMax(leftsize.rwidth(), rightsize.rwidth()); + leftsize.scale(totalWidth, leftsize.rheight(), Qt::KeepAspectRatioByExpanding); + rightsize.scale(totalWidth, rightsize.rheight(), Qt::KeepAspectRatioByExpanding); + totalHeight = leftsize.rheight() + rightsize.rheight(); + leftpage.setY(rightsize.rheight()); + break; + default: + return NULL; + } + QPixmap * page = new QPixmap(totalWidth, totalHeight); + QPainter painter(page); + painter.drawImage(QRect(leftpage,leftsize), *buffer[currentPageBufferedIndex]); + painter.drawImage(QRect(rightpage,rightsize), *buffer[currentPageBufferedIndex+1]); + return page; + } + else + { + return NULL; + } +} + +QPixmap * Render::getCurrentDoubleMangaPage() +{ + if (currentPageIsDoublePage()) + { + QPoint leftpage(0,0); + QPoint rightpage(0,0); + QSize leftsize = buffer[currentPageBufferedIndex+1]->size(); + QSize rightsize = buffer[currentPageBufferedIndex]->size(); + int totalWidth,totalHeight; + switch (imageRotation) + { + case 0: + totalHeight = qMax(leftsize.rheight(),rightsize.rheight()); + leftsize.scale(leftsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + rightsize.scale(rightsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + totalWidth = leftsize.rwidth() + rightsize.rwidth(); + rightpage.setX(leftsize.rwidth()); + break; + case 90: + totalWidth = qMax(leftsize.rwidth(), rightsize.rwidth()); + leftsize.scale(totalWidth, leftsize.rheight(), Qt::KeepAspectRatioByExpanding); + rightsize.scale(totalWidth, rightsize.rheight(), Qt::KeepAspectRatioByExpanding); + totalHeight = leftsize.rheight() + rightsize.rheight(); + rightpage.setY(leftsize.rheight()); + break; + case 180: + totalHeight = qMax(leftsize.rheight(),rightsize.rheight()); + leftsize.scale(leftsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + rightsize.scale(rightsize.rwidth(), totalHeight, Qt::KeepAspectRatioByExpanding); + totalWidth = leftsize.rwidth() + rightsize.rwidth(); + leftpage.setX(rightsize.rwidth()); + break; + case 270: + totalWidth = qMax(leftsize.rwidth(), rightsize.rwidth()); + leftsize.scale(totalWidth, leftsize.rheight(), Qt::KeepAspectRatioByExpanding); + rightsize.scale(totalWidth, rightsize.rheight(), Qt::KeepAspectRatioByExpanding); + totalHeight = leftsize.rheight() + rightsize.rheight(); + leftpage.setY(rightsize.rheight()); + break; + default: + return NULL; + } + QPixmap * page = new QPixmap(totalWidth, totalHeight); + QPainter painter(page); + painter.drawImage(QRect(rightpage, rightsize), *buffer[currentPageBufferedIndex]); + painter.drawImage(QRect(leftpage, leftsize), *buffer[currentPageBufferedIndex+1]); + return page; + } + else + { + return NULL; + } +} + +bool Render::currentPageIsDoublePage() +{ + if (buffer[currentPageBufferedIndex]->isNull() || buffer[currentPageBufferedIndex+1]->isNull()) + { + return false; + } + if (imageRotation == 0 || imageRotation == 180) + { + if (buffer[currentPageBufferedIndex]->height() > buffer[currentPageBufferedIndex]->width() && + buffer[currentPageBufferedIndex+1]->height() > buffer[currentPageBufferedIndex+1]->width()) + { + return true; + } + } + else if (imageRotation == 90 || imageRotation == 270) + { + if (buffer[currentPageBufferedIndex]->width() > buffer[currentPageBufferedIndex]->height() && + buffer[currentPageBufferedIndex+1]->width() > buffer[currentPageBufferedIndex+1]->height()) + { + return true; + } + } + return false; +} + +bool Render::nextPageIsDoublePage() +{ + //this function is not used right now + if (buffer[currentPageBufferedIndex+2]->isNull() || buffer[currentPageBufferedIndex+3]->isNull()) + { + return false; + } + if (imageRotation == 0 || imageRotation == 180) + { + if (buffer[currentPageBufferedIndex+2]->height() > buffer[currentPageBufferedIndex+2]->width() && + buffer[currentPageBufferedIndex+3]->height() > buffer[currentPageBufferedIndex+3]->width()) + { + return true; + } + } + else if (imageRotation == 90 || imageRotation == 270) + { + if (buffer[currentPageBufferedIndex]->width() > buffer[currentPageBufferedIndex]->height() && + buffer[currentPageBufferedIndex+1]->width() > buffer[currentPageBufferedIndex+1]->height()) + { + return true; + } + } + return false; +} + +bool Render::previousPageIsDoublePage() +{ + if (buffer[currentPageBufferedIndex-1]->isNull() || buffer[currentPageBufferedIndex-2]->isNull()) + { + return false; + } + if (imageRotation == 0 || imageRotation == 180) + { + if (buffer[currentPageBufferedIndex-1]->height() > buffer[currentPageBufferedIndex-1]->width() && + buffer[currentPageBufferedIndex-2]->height() > buffer[currentPageBufferedIndex-2]->width()) + { + return true; + } + } + else if (imageRotation == 90 || imageRotation == 270) + { + if (buffer[currentPageBufferedIndex-1]->width() > buffer[currentPageBufferedIndex-1]->height() && + buffer[currentPageBufferedIndex-2]->width() > buffer[currentPageBufferedIndex-2]->height()) + { + return true; + } + } + return false; +} + +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(!doublePage) + { + if (currentIndex == page) + { + emit currentPageReady(); + } + } + else + { + //check for last page in double page mode + if ((currentIndex == page) && (currentIndex + 1) >= (int)comic->numPages()) + { + emit currentPageReady(); + } + else if ((currentIndex == page && !buffer[currentPageBufferedIndex+1]->isNull()) || + (currentIndex+1 == page && !buffer[currentPageBufferedIndex]->isNull())) + { + emit currentPageReady(); + } + } +} + +void Render::update() +{ + render(); +} +//----------------------------------------------------------------------------- +// Comic interface +//----------------------------------------------------------------------------- +void Render::load(const QString & path, int atPage) +{ + createComic(path); + if (comic !=0) + { + 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); + if (comic!=0) + { + 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 + 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(); + } +} +void Render::nextDoublePage() +{ + int nextPage; + if (currentIndex +2 < (int)comic->numPages()) + { + nextPage = currentIndex+2; + } + else + { + nextPage = currentIndex; + } + if(currentIndex != nextPage) + { + comic->setIndex(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 + 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(); + } +} + +void Render::previousDoublePage() +{ + int previousPage; //indica cuál será la próxima página + previousPage = qMax(currentIndex-2,0); + if(currentIndex != previousPage) + { + comic->setIndex(previousPage); + previousIndex = currentIndex; + currentIndex = previousPage; + update(); + emit pageChanged(currentIndex); + } +} + +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-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; + 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(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+i]) //preload next pages + { + pageRenders[currentPageBufferedIndex+i] = new PageRender(this,currentIndex+i,comic->getRawData()->at(currentIndex+i),buffer[currentPageBufferedIndex+i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex+i],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-i]) //preload previous pages + { + pageRenders[currentPageBufferedIndex-i] = new PageRender(this,currentIndex-i,comic->getRawData()->at(currentIndex-i),buffer[currentPageBufferedIndex-i],imageRotation,filters); + connect(pageRenders[currentPageBufferedIndex-i],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())) + { + if (currentPageIsDoublePage()) + { + if (doubleMangaPage) + s = QString::number(currentIndex+2) + "-" + s; + else + 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..9c525c55 --- /dev/null +++ b/YACReader/render.h @@ -0,0 +1,215 @@ + #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(); + QPixmap * getCurrentDoublePage(); + QPixmap * getCurrentDoubleMangaPage(); + bool currentPageIsDoublePage(); + bool nextPageIsDoublePage(); + bool previousPageIsDoublePage(); + void goTo(int index); + void doublePageSwitch(); + void doubleMangaPageSwitch(); + 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 nextDoublePage(); + void previousDoublePage(); + 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 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; + bool doubleMangaPage; + 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; +}; + + +#endif // RENDER_H diff --git a/YACReader/shortcuts_dialog.cpp b/YACReader/shortcuts_dialog.cpp new file mode 100644 index 00000000..e88c91a0 --- /dev/null +++ b/YACReader/shortcuts_dialog.cpp @@ -0,0 +1,55 @@ +#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); + + setLayout(mainLayout); + + 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..44ab17bf --- /dev/null +++ b/YACReader/viewer.cpp @@ -0,0 +1,985 @@ +#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), +doubleMangaPage(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(!settings->contains(USE_OPEN_GL)) + { + settings->setValue(USE_OPEN_GL,2); + } + + if((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(); + + if(Configuration::getConfiguration().getDoubleMangaPage()) + doubleMangaPageSwitch(); + + 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; + if (doublePage && render->currentPageIsDoublePage()) + { + render->nextDoublePage(); + } + else + { + render->nextPage(); + } + updateInformation(); + shouldOpenPrevious = false; +} + +void Viewer::prev() +{ + direction = -1; + if (doublePage && render->previousPageIsDoublePage()) + { + render->previousDoublePage(); + } + else + { + render->previousPage(); + } + updateInformation(); + shouldOpenNext = false; +} +void Viewer::showGoToDialog() +{ + goToDialog->open(); +} +void Viewer::goTo(unsigned int page) +{ + direction = 1; //in "go to" direction is always fordward + render->goTo(page); +} + +void Viewer::updatePage() +{ + QPixmap * previousPage = currentPage; + if (doublePage) + { + if (!doubleMangaPage) + currentPage = render->getCurrentDoublePage(); + else + { + currentPage = render->getCurrentDoubleMangaPage(); + } + if (currentPage == NULL) + { + currentPage = render->getCurrentPage(); + } + } + else + { + 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()); + } + } + + if(devicePixelRatio()>1)//only in retina display + { + QPixmap page = currentPage->scaled(content->width()*devicePixelRatio(), content->height()*devicePixelRatio(), Qt::KeepAspectRatio, Qt::SmoothTransformation); + page.setDevicePixelRatio(devicePixelRatio()); + content->setPixmap(page); + } + + 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) + { + if(getMovement(event) == Forward) + { + next(); + verticalScroller->stop(); + event->accept(); + wheelStop = false; + } + return; + } + else + wheelStop = true; + } + else + { + if((event->delta()>0)&&(verticalScrollBar()->sliderPosition()==verticalScrollBar()->minimum())) + { + if(wheelStop) + { + if(getMovement(event) == Backward) + { + 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::doubleMangaPageSwitch() +{ + doubleMangaPage = !doubleMangaPage; + render->doubleMangaPageSwitch(); + Configuration::getConfiguration().setDoubleMangaPage(doubleMangaPage); +} + +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); + if(!(devicePixelRatio()>1)) + 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..b1eceb64 --- /dev/null +++ b/YACReader/viewer.h @@ -0,0 +1,168 @@ +#ifndef __VIEWER_H +#define __VIEWER_H + +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "scroll_management.h" + +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, public ScrollManagement + { + 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 doubleMangaPageSwitch(); + 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; + bool doubleMangaPage; + 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..845823c6 --- /dev/null +++ b/YACReader/width_slider.cpp @@ -0,0 +1,94 @@ +#include "width_slider.h" + +#include +#include +#include +#include +#include "configuration.h" + +YACReaderSliderAction::YACReaderSliderAction (QWidget * parent) + :QWidgetAction (parent) { + + widget = new YACReaderSlider(); + setDefaultWidget(widget); + + connect(widget,SIGNAL(fitToWidthRatioChanged(float)),this,SIGNAL(fitToWidthRatioChanged(float))); + + +} + +void YACReaderSliderAction::updateText(int value) +{ + widget->updateText(value); +} + +void YACReaderSliderAction::updateFitToWidthRatio(float v) +{ + widget->updateFitToWidthRatio(v); +} + +YACReaderSlider::YACReaderSlider(QWidget *parent) + :QWidget(parent) +{ + 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(this); + 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); + + + setLayout (pLayout); + setAutoFillBackground(false); + + setMinimumSize(276,45); + + 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 YACReaderSlider::updateText(int value) +{ + percentageLabel->setText(QString("%1 %").arg(value)); + Configuration::getConfiguration().setFitToWidthRatio(value/100.0); + emit(fitToWidthRatioChanged(value / 100.0f)); +} + +void YACReaderSlider::updateFitToWidthRatio(float v) +{ + int value = v*100; + slider->setValue(value); + percentageLabel->setText(QString("%1 %").arg(value)); +} diff --git a/YACReader/width_slider.h b/YACReader/width_slider.h new file mode 100644 index 00000000..aecedb95 --- /dev/null +++ b/YACReader/width_slider.h @@ -0,0 +1,48 @@ +#ifndef WIDTH_SLIDER_H +#define WIDTH_SLIDER_H + +#include + +class QLabel; +class QSlider; + +class YACReaderSlider : public QWidget +{ + Q_OBJECT +private: + QLabel * percentageLabel; + QSlider * slider; + +public: + + YACReaderSlider (QWidget * parent = 0); + +public slots: + void updateText(int value); + void updateFitToWidthRatio(float v); + + +signals: + void fitToWidthRatioChanged(float value); +}; + +class YACReaderSliderAction : public QWidgetAction +{ + Q_OBJECT +private: + YACReaderSlider * widget; + +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_de.ts b/YACReader/yacreader_de.ts new file mode 100644 index 00000000..16d8d4b0 --- /dev/null +++ b/YACReader/yacreader_de.ts @@ -0,0 +1,791 @@ + + + + + BookmarksDialog + + + Lastest Page + Vorherige Seite + + + + Close + Schliessen + + + + Click on any image to go to the bookmark + Click auf beliebiges Bild um zum Lesezeichen zu gehen + + + + + Loading... + Laden... + + + + FileComic + + + CRC error on page (%1): some of the pages will not be displayed correctly + CRC Error auf Seite (%1): einige Seiten werden nicht korrekt dargestellt + + + + Unknown error opening the file + Unbekannter Fehler beim öffnen des Files + + + + 7z not found + 7z nicht gefunden + + + + Format not supported + Format wird nicht unterstützt + + + + GoToDialog + + + Page : + Seite : + + + + Go To + Gehe nach + + + + Cancel + Abbrechen + + + + + Total pages : + Seiten total : + + + + Go to... + Gehe nach... + + + + GoToFlowToolBar + + + Page : + Seite : + + + + HelpAboutDialog + + + About + Über + + + + Help + + + + + MainWindowViewer + + + &Open + &Öffnen + + + + O + O + + + + Open a comic + Comic öffnen + + + + Open Folder + Ordner Öffnen + + + + Ctrl+O + Crtl+ O + + + + Open image folder + Bilder Ordner öffnen + + + + Save + Speichern + + + + + Save current page + Diese Seite speichern + + + + Previous Comic + Voheriger Comic + + + + Open previous comic + Vorherigen Comic öffnen + + + + Next Comic + Nächster Comic + + + + Open next comic + Nächsten Comic öffnen + + + + &Previous + &Vorherige + + + + Go to previous page + Zur vorherigen Seite gehen + + + + &Next + &Nächste + + + + Go to next page + Zur nächsten Seite gehen + + + + Fit Width + Breite anpassen + + + + Fit image to height + Bild auf Höhe anpassen + + + + Fit Height + Höhe anpassen + + + + Fit image to width + Bildbreite anpassen + + + + Rotate image to the left + Bild nach links drehen + + + + L + L + + + + Rotate image to the right + Bild nach rechts drehen + + + + R + R + + + + Double page mode + Doppelseiten Modus + + + + Switch to double page mode + Zum Doppelseiten Modus wechseln + + + + D + D + + + + Go To + Gehe zu + + + + G + G + + + + Go to page ... + Gehe nach Seite ... + + + + Options + Optionen + + + + C + C + + + + YACReader options + YACReader Optionen + + + + Help + Hilfe + + + + Help, About YACReader + Hilfe, über YACReader + + + + Magnifying glass + Vergößerungsglas + + + + Switch Magnifying glass + Vergrößerungsglas wechseln + + + + Z + Z + + + + Set bookmark + Lesezeichen setzen + + + + Set a bookmark on the current page + Lesezeichen auf dieser Seite setzen + + + + Show bookmarks + Lesezeichen anzeigen + + + + Show the bookmarks of the current comic + Lesezeichen für diesen Comic anzeigen + + + + M + M + + + + Show keyboard shortcuts + Tastaturkürzel anzeigen + + + + Show Info + Info anzeigen + + + + I + I + + + + Close + Schliessen + + + + Show Dictionary + Wörterbuch anzeigen + + + + Always on top + Immer Oberste Ansicht + + + + Show full size + Vollansicht anzeigen + + + + Show go to flow + "Go to Flow" anzeigen + + + + &File + &File + + + + File + File + + + + Open Comic + Comic öffnen + + + + Comic files + Comic Files + + + + Open folder + Ordner öffnen + + + + Image files (*.jpg) + Bilder Files (*.jpg) + + + + page_%1.jpg + Seite_%1.jpg + + + + There is a new version available + Neue Version verfügbar + + + + Do you want to download the new version? + Möchten Sie die neue Version herunterladen? + + + + Remind me in 14 days + In 14 Tagen erneut erinnern + + + + Not now + Nicht jetzt + + + + OptionsDialog + + + "Go to flow" size + "Go to flow" Größe + + + + My comics path + Pfad zu Meine Comics + + + + Page width stretch + Seitenbreite strecken + + + + Background color + Hintergrund Farbe + + + + Choose + Auswählen + + + + Restart is needed + Neustart erforderlich + + + + Brightness + Helligkeit + + + + Contrast + Kontrast + + + + Gamma + Gamma + + + + Reset + Zurücksetzen + + + + Image options + Bilderoptionen + + + + General + Allgemein + + + + Page Flow + Page Flow + + + + Image adjustment + Bildanpassung + + + + Options + Optionen + + + + Comics directory + Comics Verzeichnis + + + + QObject + + + 7z lib not found + 7z Verzeichnis nicht gefunden + + + + unable to load 7z lib from ./utils + 7z Verzeichnis kann von ./utils nicht geladen werden + + + + ShortcutsDialog + + + YACReader keyboard shortcuts + YACReader Tastaturkürzel + + + + Close + schliessen + + + + Keyboard Shortcuts + Tastaturkürzel + + + + Viewer + + + + Press 'O' to open comic. + 'O' drücken um Comic zu öffnen. + + + + Not found + Nicht gefunden + + + + Comic not found + Comic nicht gefunden + + + + Error opening comic + Fehler beim Öffnen des Comics + + + + CRC Error + CRC Fehler + + + + Loading...please wait! + Ladevorgang... Bitte warten! + + + + Page not available! + Seite nicht verfügbar! + + + + Cover! + Titelseite! + + + + Last page! + Letzte Seite! + + + + YACReaderFieldEdit + + + + Click to overwrite + Zum Überschreiben drücken + + + + Restore to default + Ursprungszustand wiederherstellen + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + zum Überschreiben drücken + + + + Restore to default + Urpsrungszustannd wiederherstellen + + + + YACReaderFlowConfigWidget + + + How to show covers: + Wie zeige ich die Titelseite an: + + + + CoverFlow look + Tielseiten Ansicht + + + + Stripe look + Streifen Ansicht + + + + Overlapped Stripe look + Überlappende Streifen Ansicht + + + + YACReaderGLFlowConfigWidget + + + Presets: + Voreinstellungen: + + + + Classic look + Klassische Ansicht + + + + Stripe look + Streifen Ansicht + + + + Overlapped Stripe look + Überlappende Streifenansicht + + + + Modern look + Moderne Ansicht + + + + Roulette look + Zufalls Ansicht + + + + Show advanced settings + Zeige fortgeschrittene Einstellungen + + + + Custom: + Custom: + + + + View angle + Anzeige Winkel + + + + Position + Position + + + + Cover gap + Titelbild Abstand + + + + Central gap + Mittel Abstand + + + + Zoom + Vergrößern + + + + Y offset + Y Anpassung + + + + Z offset + Z Anpassung + + + + Cover Angle + Titelbild Ansichtswinkel + + + + Visibility + Anzeigeintensität + + + + Light + Licht + + + + Max angle + Max Winkel + + + + Low Performance + Niedrige Leistung + + + + High Performance + Hohe Leistung + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Benutz VSync (verbessert die Bildqualität im Vollanzeigemodus, schlechtere Leistung) + + + + Performance: + Leistung: + + + + YACReaderOptionsDialog + + + Save + Speichern + + + + Cancel + Abbrechen + + + + Use hardware acceleration (restart needed) + Benutze Hardware Beschleunigung (Neustart erforderlich) + + + + YACReaderTranslator + + + YACReader translator + YACReader Übersetzer + + + + + Translation + Übersetzung + + + + clear + löschen + + + + Service not available + Service nicht verfügbar + + + 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 available + 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..6d9646dc --- /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 available + 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..d712c04e --- /dev/null +++ b/YACReader/yacreader_images.qrc @@ -0,0 +1,86 @@ + + + ../images/icon.png + ../images/goto.png + ../images/find_folder.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/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..f11dee29 --- /dev/null +++ b/YACReader/yacreader_images_osx.qrc @@ -0,0 +1,57 @@ + + +../images/viewer_toolbar/bookmark_osx.png +../images/viewer_toolbar/bookmark_osx@2x.png +../images/viewer_toolbar/close_osx.png +../images/viewer_toolbar/close_osx@2x.png +../images/viewer_toolbar/doubleMangaPage_osx.png +../images/viewer_toolbar/doubleMangaPage_osx@2x.png +../images/viewer_toolbar/doublePage_osx.png +../images/viewer_toolbar/doublePage_osx@2x.png +../images/viewer_toolbar/flow_osx.png +../images/viewer_toolbar/flow_osx@2x.png +../images/viewer_toolbar/full_osx.png +../images/viewer_toolbar/full_osx@2x.png +../images/viewer_toolbar/goto_osx.png +../images/viewer_toolbar/goto_osx@2x.png +../images/viewer_toolbar/help_osx.png +../images/viewer_toolbar/help_osx@2x.png +../images/viewer_toolbar/info_osx.png +../images/viewer_toolbar/info_osx@2x.png +../images/viewer_toolbar/magnifyingGlass_osx.png +../images/viewer_toolbar/magnifyingGlass_osx@2x.png +../images/viewer_toolbar/next_osx.png +../images/viewer_toolbar/next_osx@2x.png +../images/viewer_toolbar/open_osx.png +../images/viewer_toolbar/open_osx@2x.png +../images/viewer_toolbar/openFolder_osx.png +../images/viewer_toolbar/openFolder_osx@2x.png +../images/viewer_toolbar/openNext_osx.png +../images/viewer_toolbar/openNext_osx@2x.png +../images/viewer_toolbar/openPrevious_osx.png +../images/viewer_toolbar/openPrevious_osx@2x.png +../images/viewer_toolbar/options_osx.png +../images/viewer_toolbar/options_osx@2x.png +../images/viewer_toolbar/previous_osx.png +../images/viewer_toolbar/previous_osx@2x.png +../images/viewer_toolbar/rotateL_osx.png +../images/viewer_toolbar/rotateL_osx@2x.png +../images/viewer_toolbar/rotateR_osx.png +../images/viewer_toolbar/rotateR_osx@2x.png +../images/viewer_toolbar/save_osx.png +../images/viewer_toolbar/save_osx@2x.png +../images/viewer_toolbar/shortcuts_osx.png +../images/viewer_toolbar/shortcuts_osx@2x.png +../images/viewer_toolbar/showBookmarks_osx.png +../images/viewer_toolbar/showBookmarks_osx@2x.png +../images/viewer_toolbar/toHeight_osx.png +../images/viewer_toolbar/toHeight_osx@2x.png +../images/viewer_toolbar/toWidth_osx.png +../images/viewer_toolbar/toWidth_osx@2x.png +../images/viewer_toolbar/toWidthSlider_osx.png +../images/viewer_toolbar/toWidthSlider_osx@2x.png +../images/viewer_toolbar/translator_osx.png +../images/viewer_toolbar/translator_osx@2x.png + + + diff --git a/YACReader/yacreader_images_win.qrc b/YACReader/yacreader_images_win.qrc new file mode 100644 index 00000000..e7251da4 --- /dev/null +++ b/YACReader/yacreader_images_win.qrc @@ -0,0 +1,29 @@ + + + ../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 + ../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..246fbe31 --- /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 available + 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..d2543af9 --- /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 available + 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..67e44ab4 --- /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 available + ДоÑтупно новое обновление + + + + 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..87f1ba74 --- /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 available + + + + + 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..8e7a0412 --- /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 available + 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.1 b/YACReaderLibrary.1 new file mode 100644 index 00000000..45a6737e --- /dev/null +++ b/YACReaderLibrary.1 @@ -0,0 +1,48 @@ +.\" Manpage for YACReaderLibrary. +.\" Contact yoann.gauthier9@gmail.com to correct errors or typos. +.TH man 1 "28 September 2014" "2.0" "YACReaderLibrary man page" +.SH NAME +YACReaderLibrary \- launch YACReaderLibrary application. +.SH SYNOPSIS +YACReaderLibrary +.br +YACReaderLibrary [\fB\-h\fR | \fB\-\-help\fR] +.br +YACReaderLibrary [\fB\-v\fR | \fB\-\-version\fR] + +.SH DESCRIPTION +YACReaderLibrary an application for browsing and managing your comic collections with various smooth transition effects. +.SH OPTIONS +.TP +Without options +Start YACReaderLibrary. +.TP +.BR \-h, \-\- help +Display help text and exit. +.TP +.BR \-v, \-\- version +Display version information and exit. +.SH FEATURES +- Create, manage and browse your comics collections using beautiful, customizable and smooth "comic flow" transitions. +.TP +- Comic Vine support. +.TP +- Easily organization of your comics in libraries. +.TP +- Find your comics quickly using the built-in search engine. +.TP +- Open your comics with YACReader from YACReaderLibrary. +.TP +- Enjoy your comics covers with the fullscreen mode. +.TP +- Mark your comics as read/unread and track your reading progress. +.SH CONTACTS +To report bug or contact developpers, send a mail to : +.RS 3 +.TP +\fBinfo@yacreader.com\fR : for general information or suggestions about YACReader. +.TP +\fBsupport@yacreader.com\fR : for problems with YACReader or bugs detected. +.RE +.SH AUTHOR +Luis Ãngel San Martín Rodríguez (luisangelsm@gmail.com) \ No newline at end of file diff --git a/YACReaderLibrary.desktop b/YACReaderLibrary.desktop new file mode 100644 index 00000000..75f7a4a3 --- /dev/null +++ b/YACReaderLibrary.desktop @@ -0,0 +1,13 @@ +[Desktop Entry] +Name=YACReader Library +GenericName=Yet Another Comic Reader - Library +Comment=A comic library management application with server mode. +Exec=YACReaderLibrary %f +Icon=/usr/share/yacreader/iconLibrary.png +Terminal=false +Type=Application +StartupNotify=true +Categories=Graphics;Viewer; +MimeType= +Keywords=comic;library;server; +X-Desktop-File-Install-Version=0.22 diff --git a/YACReaderLibrary/YACReaderLibrary.icns b/YACReaderLibrary/YACReaderLibrary.icns new file mode 100644 index 0000000000000000000000000000000000000000..30bb0b0df3403297d4575107b67d080c42063bdf GIT binary patch literal 142716 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 {RCvJ|$ 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%0AU%$(2R9pBveZ52Mfm;q zD$0V2UZduzWHd(FlGx5|uWm|tUKyc%l^xvm{ZDr$J*q{Be4=KnT(M2o0WJJX8tgfp zbKfnp9=dG-mB_z46*B!w~JYU~zzR_8}+kmLilF3q^LAU0q#VvXbvn!3a{< zGM3^-tHV7s0B`1sn}BC{UWSezapt@C9sWkc_YircFkvOL8CdY&Vfyc-e%2p@7f1eA zzx2WZ!Wek)eu`&F`3wU(K(@yiLr1^+v6K5lI!%V@@>fV4EnpNs7Q645wsvRkbK5dj zqei0%;=f;hJLU_=AVxscLXbxpUSgXmxbs#E(D<-9TWzZ_G&Gd;21YAYl_+ZbQgqY( zz^)E|dh>sE2ezBBgK#*YpM8I>KwEq80L>M}p1Gyd-*6DBM7Jg+&nh&Oi zzB@0Z)mYl6;)8Wr!=e7CeETv917^fbr`V)2OI=?I&dvK1KJcu45+>MC`n~ z>h|*9xZ0jWr`ubiyi36B$Bx5f4d{Ljz5NJ{BUKW?Xp;~t$Uz=NSS@xltsvSPX@q!g zjGEZX4Z=8isj%4cBuUcU@#XAo!Ai@b-FV}>!gDa{rm8rrCxW6EJeYc=i-{+~;A8An(GZOB&8wyUcY`Ims~ zcwaM)F0Yp?`eXBRqFC}HwD)7)7ykhf&nfV{LvDv2=cWY7989vRnd|TDu{TT`qx23n zXupO*t>kM_PqFER_&-3_)ZMG{D93i^eY9KPuP&V|jnaRoO+Igq;LtyFi$ed}0pQ$` z0<;Bo`@cafFEK|-t~5uvvKqCu+1e;j5q$#zx_)ki5&t4-1B(IG?zTNM<*NSqYA)!P zYU}C5c0^9oeY!g`kWETOfcYv025)R=A=pMtv{$G+)^WyYV30|6Uy)v?1l zchkrWhYk0kE+TE8w4XDnn!cX`C%W^7@KUUI9XKD1FeP5dKnK0GpC~c_pdG}-%qXQ( z{BmU7g6re)xv^?tx&`1fgs7<;(GgpFPO%<`JG74Q>j$UkaMF*gMh38#ME7SXzpyqU zn{pOK{1jF{P*VEH{O*?{-eAPT9h|lyw`ZVI2-`{iBk1qz3(A|XvsQj=)!$SeNs1Op z%DNpxS;*TycNs{iQ(;NeDE_%yx;Ad6bHkW+OLHr(C+-!slm2S_&bXZgw5~-fJ<~Q* zV14Tet zkGPgT)S*?gkdEutwBCQyd-oHi7_4ah{nNr(1a(Mv8?V$2t{&0=7#eLE+^r1v0L>%9 ztE$$ZF;nmYTSL4y?dlh=AGvWN$N`~k)}igQczsmA zzD3PcA{OW7rqW}U0QR#K?hnot_U=+Q0pEZ=%5qyjH$Y91J~g#TLw1yhX8-DTFI2hk zPkkGL{y_1Q@krj{eV^wo8R{8I8QJ(sS`ouZkT=FI%8O}xAp2-3LTsg3XsW&A zS_+e86ux9Lz=7%MtHw!<^^A=C_R&-dzX-0A>P(ro+#(9x`rh**h>H7D;Y0NJVwSdg zz*vAFhGr6AVZr)X;w$0G!$d8^q2#@vuP;cH*4*p>dXZ+9R{HrYcgGl?!S)X2vtYD4 z=Ycm-D`Wd4bFIfJmixP;emF?87Yy3L-(&`d5WdC^$?B6o!h%k51EmNT`Ke@ebf2dO}EyXWpN-Td21Xe%w_}y-mJ~?p;8wUQqY>3)`vR$V10UtAA3pP zpi3o3t9LHWi01E_E=Ez38_VQlWz5>Olz|Ay5h^~Obo@s+(PbemHqmH3K9!j#MNABU zx#7o;C5N?e!GkxFU)-q%;Q*Jmj~}DWJ=h-QM8e*ZYcK-hM6<1VfOqB?`IKCMlWc7a z1a;o$eF(^cIPm>Me#H+;yv%SwNH}WAk1dTz<|v(E@z(SGTY8dsgX7~}lw@^Pzxhy2 zMoI-Hboc9R;nRi!uL1j^kETMT&m6|73{lvV4k|Y_k?*BEe7?Y}Y;AZRIW$uT=ge;g z9sJcIbRvYWt1_mk9%QEGi?2)S4)nzzEd5YIM|?q^aev;Bb5L9zZ$Ap4-8PHQ5y=*7 zu25<0RE(p<5YU|zHP>x71k9lSq173u)c$`vovsJ4{<=7y3-YU0oOUR!wO&>?1ebof z_*Z-J>mw-Dr#p8jpHqQt58XPk;U0}F-Ox>#j?yk%0+)yNFJ0n;wfjqy+irS7ir|3w z4$-4yFE$)e1Idh==^UT)(?Q8P+hvw#Pl;7!mGKV&4n{_8{B0~qP}>|}1284V-zMJcPz^ayQ^l81FxDYZ_|uT zobT1UgGkJ!KPi)(pH-;2vvB0yiF(QJQ1mvb%nMnbOLC$VO{`N(86P(`+#1>-4?lcS z+(sEqHf)mAantzi|G*4_(4gF95i9H%q0`}!5rV@ZDhW3>6Rh9?Wj~NbzLa-#SFPB~ zsP`;GvRh(rhzK{vr!G9v1p`xjLKdFlg|9 zneVEp&$umUtc0}vacCdvVB3guN7wn4?F#8TAgrn^p#RH3&W5Q9922S~lwJUiSqa!^ zh>nbvY@Y=UI=bzDD8Z9$ z4eXgXa{~vc1#*rFy}xlwMFw?4emAspMUh2x8iEb^J~aqH?Tgdsg|GpX&!3GYnfqYy zELaF!pi-td^=Y4Xyfv7J`TCx}b`Z4q9R!W}!9)Gmd%xH-EQY>{mbF zIH74jC>(#IQqir%!1#ly4GFq0Wvhh$>*X`p+`TvC`0AN}19aCBjFx;lSE0_J^(2?7 zzdFU`-m-`iiM8kcKt+xmO=jQVQL{ za7u42V+*|cH=8#vG`vD{AbEER;Be?hno*tn2Xmo>pVbAn&GkaYB$M74H9}|d_|fAl zOdl>C&`MZXcoLhEpNB+L;%PV#Kfy>BxSr2__h0}kBiy`hb+3nyfPO}3MC zF5ot>vSm*ig{kvF^rbH|+Ml~nhm-*CU+T56cfnLh)o^wUd68Qp*Y6mJAY^YtkCsFK zIG2^yhPRmzhfWW(UEfgUrdBSVHL#Ib-ixz2eI2c6>vKwmGS=^m6NCzK@>C6rthq%- z&E3d=u8uEiG&ja;$s}H*)S0{wdXxDzQgkU}2%S<}goJB4JG*~qaw3^=30Lha?8(m; z;>a~iN{cF7wYLl0;OBe?a3PA{f`Esf5~cE_wrzxrEPP`s$=)?)y4^6)`Wp>>Ru@~- z<5P9uK3Zd4DG(m$G9n(}Gect{U#WIanQP_$ZutJ3RZcxN*W*{y1KloByrdT3slcwq zb*k4=fdPYH55{c;ehh;Y^YW#AvjfQ~I39a*yRqlvwfR{qozNGp^$2(hopGsO#vEro zYzZd4;9J#62Yi1`D8y~O9xoG6!WfwV9JPtXy`_#qto0r;I>J4w~ zS%V|SG9$Dz$MZqA5{N0E*A67_&?YA9PDsjUO`T$Qiy;i~-G{^glgrHInFoQ&NB{RA z6gf{8r}57=+2#30ngex>t|t{ri>$7$^k$ANCU1j+P;%EkpS_lI%YmiP@hF?6n;B7h z;mYK!3Uh)rc5)718{`@pGxjfTVbvwFfiq94q6KyXwa5(-r>&G^_-!}W>JKI`ar)6$ zB!qz7kCBm)=~=G7cNK8l;9BJb-~-ZJJqwYh57{46TtfOyYEoR(^P9sc?j?~#tZ-v~ zBsepn2v!2_fZ%jN(YesWJmF>qrWKLDtXiHm5vMaZ%$|aNk;?FIIHM z*b$cE5$)X=`>Xy4osgNL~QH;8kGz}OFvH;p927h zzh9|B8X%`1yFO7+zS^S|a@Vp3-P+0=yN0R2UIB)`WwwLvKho{}d}$1_fjyqruFIrM z);M7|1dAO|C#jl{Gm>7X_HQosYI-8Fx?}F z^wxzprqiGyyBAFIiv#m^l%nGZ6ACerhCfj}J*s6ine1wdKAdaW=C`_?2fHX0tK{=2 zLB_z0z*4*uI*2@_4ly0RQW13$IVNLQG8!IYXl90Q+g6VNeJGRK@%XTD&v7PofSE|= z*}EB-4PG9hE-tLMW(7Y6%p3~VFedEzK7_IU8xLmYgGGi>6h2O+nQdNAV*6bEMw`Nr zYb-1(^4&kN6^9bb`Qmi=nL+d|^m>tUH<9XA{w8@PoaDq1^>(+C>@F=-+ZlnP#F^vk zvX|bn;{@>T?n;W7;WW>qpL4oCgl55d7K<}8Gd>G0^{9m(FS6dCddO240Zlhl<;YwM z+J_|$`yI2QNBsGJ6rkT9HEV`=oc4^c6*htArD;tQ`@x9nhXwx&JD z0bx>M{rh%B%WZn+2byDGBP`iFuCo~#tK#k=~JeN&##)$5HTEGA`PmdjK;F24rEbBpW6Eh5^NF_o@$-B)#2M{wS;! zRTnRiU%yj{v~!PsP7Zr9zqrn5#Eha405W@F-;q$>W5g??l=;<5wm{iWCwiQ0;r39& z&C0w)aDYWp?5SGxJGD!kN=Y{*m|g`7KZ0mmf=7$gg7-Rd1w&%5mCuJB!e3fIPZ=s9 z*`LaCF@2V*WWtVI9-07-vG^V$JIfa_!CN8pZHGZ5>_nsoxf8;|l~9Zq6m|IF17%ac zjsu|VD0eXum)St6pZGk4Wiie2$6HSaMF>pXMHH@~3JgBq8LN<~bGf~)#i^+KGUuLS zNqkg{%&Rom5T5$;fgV+a^`NtFN4b^miK9?b*B=6XC(qQ!0`$nNF zf-41_MZ$=Hc2AA_1FO4f8(hYD3Ej+V30;uZiE#N@Pdn-G_>9db0Cn3X z7jec`2}05Kvr*qzWMqoWlKc9_=H_N?hkhObF^kd^wWHtF<-muh!nCX6pG(|oSo~mv zPf-s%v|g)*MpPi_XaS2)DU|Y+xGBVmf=dQgzw?P2}-9>3F z{9-fL8jl>48St@}`~%)c^P+KGa?3{1d{b!pBgG8cdh*_l<*~0b zt_4GI*GO{$8j8|GkmI5lxkoSg!F%X?^@m5V9JR=*<6z5#GeTMX+vVmQg7>DiF3LP((jq-% zhbJMvhy4Vg`KG{4M(2W%jcXaDb7r5rOyLZT(G>AhDkZJ`FVgHeobDdxbt_`pfl(qOC)_+PjxFM5_?F_`~IS zG*f54zQz+yYjZNC?t3f6YrVqY=EG#KnVj3*pFhv$o{C|JX?Ny4LME zfDM7K*?LDj8q2HLX))PpjZrbYoVOfLY5~-Dql3aoR#u;BqOO@4@fRh#AFD=lC)+Sc zpVrgNQjiaQ@K)O>H5 z+F|hR{#JE6sdtHxf3nB3Wl9a+zbBCt%$TiapKWX)UmfV<$**6214hUGCfa_9)_CFX zi0FGs4yxDhd;TE$U{B`_aiSe@f2z_$qW(?imN+3mhteF%&uyI}sAr9+MR5IyfE`ml zap_3;4Z}}+5dGv(0^X{*HByoO-mApj&O<@oKlxcvifQ(;F(r6sbC^DKMpA?nA7nZo zWVinAc5`o92E7xTD*(KEoK?{JjGxJ2%w{cA zi$__>gYuqJhwM~sWu-dg!a{J<_E=GryA)cKZomD372JCC`yD*?^&F$`f>lX5G+mq{ zjK|BmdWnRJl4^s-227HnPr@c8KFk5$O2n(;eBq9q!o2Y@d1;}-8s z6tXzvQL>M8?7R|l&UVE>$f==p+eZVol?&43S)68|m9_cc^<4D%b`hXD zC--tz>hb+4`+b&vN;}&>OwLOT3vv20nt)f&fRhSK-Sn2ovy!QIisLg2%ndF)@~jyf zR@Y-%;}I6@sdz`lM{U>iu7sL@hLRk)Wr1mB+|%EA&!P`EejURa_M1{!R!qX4 zjUxJ_P8^2?F@vb*mY=6{Io7xfsfvM_?}=|h6-j4}0*R# z-DYv?2PDmU9RL%lKV@LIASG!Y#}7d9d (<=w->I! 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
gR7_dSEFsb+5^_Homda3vq1oI<$>mt& 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 literal 0 HcmV?d00001 diff --git a/YACReaderLibrary/YACReaderLibrary.pro b/YACReaderLibrary/YACReaderLibrary.pro new file mode 100644 index 00000000..be62fe6d --- /dev/null +++ b/YACReaderLibrary/YACReaderLibrary.pro @@ -0,0 +1,304 @@ +###################################################################### +# 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 -framework AppKit + +OBJECTIVE_SOURCES += $$PWD/../common/pdf_comic.mm +HEADERS += $$PWD/../common/pdf_comic.h +CONFIG += objective_c +QT += macextras gui-private +} + +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/folder_item.h \ + ./db/folder_model.h \ + ./db/comic_model.h \ + ./db/comic_item.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 \ + no_search_results_widget.h \ + comic_files_manager.h \ + db/reading_list_model.h \ + db/reading_list_item.h \ + yacreader_folders_view.h \ + yacreader_reading_lists_view.h \ + add_label_dialog.h \ + yacreader_history_controller.h \ + yacreader_navigation_controller.h \ + empty_label_widget.h \ + empty_container_info.h \ + empty_special_list.h \ + empty_reading_list_widget.h \ + ../common/scroll_management.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/folder_item.cpp \ + ./db/folder_model.cpp \ + ./db/comic_model.cpp \ + ./db/comic_item.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 \ + no_search_results_widget.cpp \ + comic_files_manager.cpp \ + db/reading_list_model.cpp \ + db/reading_list_item.cpp \ + yacreader_folders_view.cpp \ + yacreader_reading_lists_view.cpp \ + add_label_dialog.cpp \ + yacreader_history_controller.cpp \ + yacreader_navigation_controller.cpp \ + empty_label_widget.cpp \ + empty_container_info.cpp \ + empty_special_list.cpp \ + empty_reading_list_widget.cpp \ + ../common/scroll_management.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_de.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 manpage + +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_* + +manpage.path = $$DATADIR/man/man1 +manpage.files = ../YACReaderLibrary.1 +} diff --git a/YACReaderLibrary/add_label_dialog.cpp b/YACReaderLibrary/add_label_dialog.cpp new file mode 100644 index 00000000..6d9d73b7 --- /dev/null +++ b/YACReaderLibrary/add_label_dialog.cpp @@ -0,0 +1,84 @@ +#include "add_label_dialog.h" + +AddLabelDialog::AddLabelDialog(QWidget *parent) : + QDialog(parent) +{ + QVBoxLayout * layout = new QVBoxLayout; + + layout->addWidget(new QLabel(tr("Label name:"))); + layout->addWidget(edit = new QLineEdit()); + + layout->addWidget(new QLabel(tr("Choose a color:"))); + layout->addWidget(list = new QListWidget() ); + + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_red.png"), tr("red"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_orange.png"), tr("orange"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_yellow.png"), tr("yellow"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_green.png"), tr("green"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_cyan.png"), tr("cyan"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_blue.png"), tr("blue"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_violet.png"), tr("violet"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_purple.png"), tr("purple"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_pink.png"), tr("pink"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_white.png"), tr("white"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_light.png"), tr("light"))); + list->addItem(new QListWidgetItem(QIcon(":/images/lists/label_dark.png"), tr("dark"))); + + QColor backgroundColor = this->palette().background().color(); + list->setStyleSheet(QString("QListWidget {border : none; background-color: rgb(%1,%2,%3);}").arg(backgroundColor.red()).arg(backgroundColor.green()).arg(backgroundColor.blue())); + list->setMinimumHeight(225); + + setModal(true); + + setMinimumHeight(340); + + //buttons + acceptButton = new QPushButton(tr("accept"),this); + cancelButton = new QPushButton(tr("cancel"),this); + + QHBoxLayout * buttons = new QHBoxLayout; + buttons->addStretch(); + buttons->addWidget(acceptButton); + buttons->addWidget(cancelButton); + + layout->addStretch(); + layout->addLayout(buttons); + + setLayout(layout); + + //connections + connect(edit,SIGNAL(textChanged(QString)),this,SLOT(validateName(QString))); + connect(cancelButton,SIGNAL(clicked()),this,SLOT(close())); + connect(acceptButton,SIGNAL(clicked()),this,SLOT(accept())); + +} + +YACReader::LabelColors AddLabelDialog::selectedColor() +{ + return YACReader::LabelColors(list->currentRow()+1); +} + +QString AddLabelDialog::name() +{ + return edit->text(); +} + +int AddLabelDialog::exec() +{ + edit->clear(); + list->clearSelection(); + + acceptButton->setDisabled(true); + + list->setCurrentRow(0); + + return QDialog::exec(); +} + +void AddLabelDialog::validateName(const QString &name) +{ + if(name.isEmpty()) + acceptButton->setDisabled(true); + else + acceptButton->setEnabled(true); +} diff --git a/YACReaderLibrary/add_label_dialog.h b/YACReaderLibrary/add_label_dialog.h new file mode 100644 index 00000000..1f5de48b --- /dev/null +++ b/YACReaderLibrary/add_label_dialog.h @@ -0,0 +1,31 @@ +#ifndef ADD_LABEL_DIALOG_H +#define ADD_LABEL_DIALOG_H + +#include + +#include "yacreader_global.h" + +class AddLabelDialog : public QDialog +{ + Q_OBJECT +public: + explicit AddLabelDialog(QWidget *parent = 0); + YACReader::LabelColors selectedColor(); + QString name(); +signals: + +public slots: + int exec(); + +protected slots: + void validateName(const QString & name); + +protected: + QLineEdit * edit; + QListWidget * list; + + QPushButton * acceptButton; + QPushButton * cancelButton; +}; + +#endif // ADD_LABEL_DIALOG_H 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..0677bc73 --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.cpp @@ -0,0 +1,324 @@ +#include "classic_comics_view.h" + +#include "yacreader_table_view.h" + +#include "comic_flow_widget.h" +#include "QsLog.h" + +#include "QStackedWidget" + +ClassicComicsView::ClassicComicsView(QWidget *parent) + :ComicsView(parent),searching(false) +{ + 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((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::CustomContextMenu); + + + //layout----------------------------------------------- + sVertical = new QSplitter(Qt::Vertical); //spliter derecha + + stack = new QStackedWidget; + stack->addWidget(comicFlow); + setupSearchingIcon(); + stack->addWidget(searchingIcon); + + + sVertical->addWidget(stack); + 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); + + tableView->setContextMenuPolicy(Qt::CustomContextMenu); + + //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(tableView, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(selectedComicForOpening(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())); + connect(comicFlow, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(requestedViewContextMenu(QPoint))); + connect(tableView, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(requestedItemContextMenu(QPoint))); + layout->addWidget(sVertical); + setLayout(layout); + + layout->setMargin(0); + +#ifdef Q_OS_MAC + sVertical->setCollapsible(1,false); +#endif + + if(settings->contains(COMICS_VIEW_FLOW_SPLITTER_STATUS)) + sVertical->restoreState(settings->value(COMICS_VIEW_FLOW_SPLITTER_STATUS).toByteArray()); +} + +void ClassicComicsView::setToolBar(QToolBar *toolBar) +{ + static_cast(comics->layout())->insertWidget(0,toolBar); +} + +void ClassicComicsView::setModel(ComicModel *model) +{ + QLOG_DEBUG() << "Setting model"; + + ComicsView::setModel(model); + + if(model == NULL) + { + comicFlow->clear(); + } + else + { + connect(model, SIGNAL(dataChanged(QModelIndex,QModelIndex,QVector)), this, SLOT(applyModelChanges(QModelIndex,QModelIndex,QVector)),Qt::UniqueConnection); + connect(model, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(removeItemsFromFlow(QModelIndex,int,int)),Qt::UniqueConnection); + connect(model, SIGNAL(resortedIndexes(QList)),comicFlow,SLOT(resortCovers(QList)),Qt::UniqueConnection); + connect(model, SIGNAL(newSelectedIndex(QModelIndex)),this,SLOT(setCurrentIndex(QModelIndex)),Qt::UniqueConnection); + + tableView->setModel(model); + if(model->rowCount()>0) + tableView->setCurrentIndex(model->index(0,0)); + + 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(ComicModel::Number); + tableView->horizontalHeader()->showSection(ComicModel::Title); + tableView->horizontalHeader()->showSection(ComicModel::FileName); + tableView->horizontalHeader()->showSection(ComicModel::NumPages); + tableView->horizontalHeader()->showSection(ComicModel::Hash); //Size is part of the Hash...TODO add Columns::Size to Columns + tableView->horizontalHeader()->showSection(ComicModel::ReadColumn); + tableView->horizontalHeader()->showSection(ComicModel::CurrentPage); + tableView->horizontalHeader()->showSection(ComicModel::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) +{ + QLOG_INFO() << "*******************************************************ClassicComicsView::setCurrentIndex"; + tableView->setCurrentIndex(index); + centerComicFlow(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::enableFilterMode(bool enabled) +{ + if(enabled) + { + comicFlow->clear(); + if(previousSplitterStatus.isEmpty()) + previousSplitterStatus = sVertical->saveState(); + sVertical->setSizes(QList () << 100 << 10000000); + showSearchingIcon(); + }else + { + hideSearchingIcon(); + sVertical->restoreState(previousSplitterStatus); + previousSplitterStatus.clear(); + } + + //sVertical->setCollapsible(0,!enabled); + searching = enabled; +} + +void ClassicComicsView::selectIndex(int index) +{ + tableView->selectRow(index); +} + +void ClassicComicsView::selectAll() +{ + tableView->selectAll(); +} + +void ClassicComicsView::selectedComicForOpening(const QModelIndex &mi) +{ + emit selected(mi.row()); +} + +void ClassicComicsView::requestedViewContextMenu(const QPoint &point) +{ + emit customContextMenuViewRequested(comicFlow->mapTo(this, point)); +} + +void ClassicComicsView::requestedItemContextMenu(const QPoint &point) +{ + emit customContextMenuItemRequested(tableView->mapTo(this, point)); +} + +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::saveSplitterStatus() +{ + if(!searching) + settings->setValue(COMICS_VIEW_FLOW_SPLITTER_STATUS, sVertical->saveState()); +} + +void ClassicComicsView::applyModelChanges(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QVector &roles) +{ + Q_UNUSED(topLeft); + Q_UNUSED(bottomRight); + if(roles.contains(ComicModel::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(); + saveSplitterStatus(); + ComicsView::closeEvent(event); +} + +void ClassicComicsView::setupSearchingIcon() +{ + searchingIcon = new QWidget(comicFlow); + + QHBoxLayout * h = new QHBoxLayout; + + QPixmap p(":/images/searching_icon.png"); + QLabel * l = new QLabel(searchingIcon); + l->setPixmap(p); + l->setFixedSize(p.size()); + h->addWidget(l,0,Qt::AlignCenter); + searchingIcon->setLayout(h); + + QPalette pal(searchingIcon->palette()); + pal.setColor(QPalette::Background, Qt::black); + searchingIcon->setAutoFillBackground(true); + searchingIcon->setPalette(pal); + + hideSearchingIcon(); +} + +void ClassicComicsView::showSearchingIcon() +{ + stack->setCurrentWidget(searchingIcon); +} + +void ClassicComicsView::hideSearchingIcon() +{ + stack->setCurrentWidget(comicFlow); +} + diff --git a/YACReaderLibrary/classic_comics_view.h b/YACReaderLibrary/classic_comics_view.h new file mode 100644 index 00000000..d98b3910 --- /dev/null +++ b/YACReaderLibrary/classic_comics_view.h @@ -0,0 +1,69 @@ +#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 ComicModel; +class QStackedWidget; + +class ClassicComicsView : public ComicsView +{ + Q_OBJECT +public: + ClassicComicsView(QWidget *parent = 0); + void setToolBar(QToolBar * toolBar); + void setModel(ComicModel *model); + + QModelIndex currentIndex(); + QItemSelectionModel * selectionModel(); + void scrollTo(const QModelIndex & mi, QAbstractItemView::ScrollHint hint ); + void toFullScreen(); + void toNormal(); + void updateConfig(QSettings * settings); + void enableFilterMode(bool enabled); + void selectIndex(int index); + +public slots: + void setCurrentIndex(const QModelIndex &index); + void centerComicFlow(const QModelIndex & mi); + void updateTableView(int i); + void saveTableHeadersStatus(); + void saveSplitterStatus(); + 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(); + void selectedComicForOpening(const QModelIndex & mi); + +protected slots: + void requestedViewContextMenu(const QPoint & point); + void requestedItemContextMenu(const QPoint & point); + +private: + YACReaderTableView * tableView; + QWidget *comics; + QSplitter * sVertical; + ComicFlowWidget * comicFlow; + QSettings * settings; + void closeEvent ( QCloseEvent * event ); + + QStackedWidget * stack; + + QByteArray previousSplitterStatus; + QWidget * searchingIcon; + bool searching; + void setupSearchingIcon(); + void showSearchingIcon(); + void hideSearchingIcon(); + void updateSearchingIconPosition(); +}; + +#endif // CLASSIC_COMICS_VIEW_H diff --git a/YACReaderLibrary/comic_files_manager.cpp b/YACReaderLibrary/comic_files_manager.cpp new file mode 100644 index 00000000..50f2b81d --- /dev/null +++ b/YACReaderLibrary/comic_files_manager.cpp @@ -0,0 +1,108 @@ +#include "comic_files_manager.h" +#include +#include +#include + +#include + +#include "comic.h" + +ComicFilesManager::ComicFilesManager(QObject *parent) : + QObject(parent), canceled(false) +{ +} + +void ComicFilesManager::copyComicsTo(const QList > &sourceComics, const QString &folderDest, const QModelIndex & dest) +{ + comics = sourceComics; + folder = folderDest; + folderDestinationModelIndex = dest; + move = false; +} + +void ComicFilesManager::moveComicsTo(const QList > &sourceComics, const QString &folderDest, const QModelIndex &dest) +{ + comics = sourceComics; + folder = folderDest; + folderDestinationModelIndex = dest; + move = true; +} + +QList > ComicFilesManager::getDroppedFiles(const QList &urls) +{ + QList > dropedFiles; + + QString currentPath; + foreach(QUrl url, urls) + { + currentPath = url.toLocalFile(); + if(currentPath.endsWith('/')) + currentPath = currentPath.remove(currentPath.length()-1,1); //QTBUG-35896 QUrl.toLocalFile inconsistency. + if(Comic::fileIsComic(currentPath)) + dropedFiles << QPair(currentPath,"/"); + else + { + QLOG_DEBUG() << "XXXXXXXXXXXX :" << currentPath; + QFileInfo info(currentPath); + if(info.isDir()) + { + QLOG_DEBUG() << "origin path prior to absoluteFilePath : " << info.absolutePath(); + foreach(QString comicPath, Comic::findValidComicFilesInFolder(info.absoluteFilePath())) + { + QFileInfo comicInfo(comicPath); + QString path = comicInfo.absolutePath(); + QLOG_DEBUG() << "comic path : " << comicPath; + QLOG_DEBUG() << "full comic path : " << path; + QLOG_DEBUG() << "origin path : " << info.absolutePath(); + dropedFiles << QPair(comicPath, path.remove(info.absolutePath())); + } + } + } + } + + return dropedFiles; +} + +void ComicFilesManager::process() +{ + int i=0; + bool successProcesingFiles = false; + QPair source; + foreach (source, comics) { + + if(canceled) + { + if(successProcesingFiles) + emit success(folderDestinationModelIndex); + emit finished(); + + return; //TODO rollback? + } + + QFileInfo info(source.first); + QString destPath = QDir::cleanPath(folder+'/'+source.second); + QLOG_DEBUG() << "crear : " << destPath; + QDir().mkpath(destPath); + if(QFile::copy(source.first, QDir::cleanPath(destPath+'/'+info.fileName()))) + { + successProcesingFiles = true; + if(move) + { + QFile::remove(source.first); //TODO: remove the whole path.... + } + } + + i++; + emit progress(i); + } + + if(successProcesingFiles) + emit success(folderDestinationModelIndex); + emit finished(); +} + +void ComicFilesManager::cancel() +{ + QLOG_DEBUG() << "Operation canceled"; + canceled = true; +} diff --git a/YACReaderLibrary/comic_files_manager.h b/YACReaderLibrary/comic_files_manager.h new file mode 100644 index 00000000..a6b54060 --- /dev/null +++ b/YACReaderLibrary/comic_files_manager.h @@ -0,0 +1,37 @@ +#ifndef COMIC_FILES_MANAGER_H +#define COMIC_FILES_MANAGER_H + +#include +#include +#include +#include + + +//this class is intended to work in background, just use moveToThread and process to start working +class ComicFilesManager : public QObject +{ + Q_OBJECT +public: + explicit ComicFilesManager(QObject *parent = 0); + void copyComicsTo(const QList > & sourceComics, const QString & folderDest, const QModelIndex &dest); + void moveComicsTo(const QList > & comics, const QString & folderDest, const QModelIndex &dest); + static QList > getDroppedFiles(const QList & urls); +signals: + void currentComic(QString); + void progress(int); + void finished(); + void success(QModelIndex); //at least one comics has been copied or moved +public slots: + void process(); + void cancel(); + +protected: + bool move; + bool canceled; + QList > comics; + QString folder; + QModelIndex folderDestinationModelIndex; + +}; + +#endif // COMIC_FILES_MANAGER_H diff --git a/YACReaderLibrary/comic_flow.cpp b/YACReaderLibrary/comic_flow.cpp new file mode 100644 index 00000000..06e2af58 --- /dev/null +++ b/YACReaderLibrary/comic_flow.cpp @@ -0,0 +1,264 @@ +#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(); +} + +void ComicFlow::resortCovers(QList newOrder) +{ + worker->lock(); + worker->reset(); + + YACReaderFlow::resortCovers(newOrder); + + QStringList imageFilesNew; + QVector imagesLoadedNew; + QVector imagesSettedNew; + foreach(int index, newOrder) + { + imageFilesNew << imageFiles.at(index); + imagesLoadedNew << imagesLoaded.at(index); + imagesSettedNew << imagesSetted.at(index); + } + + + + imageFiles = imageFilesNew; + imagesLoaded = imagesLoadedNew; + imagesSetted = imagesSettedNew; + + worker->unlock(); +} +//----------------------------------------------------------------------------- +//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..c49543b3 --- /dev/null +++ b/YACReaderLibrary/comic_flow.h @@ -0,0 +1,78 @@ +#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); + void resortCovers(QList newOrder); + +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..2fce350c --- /dev/null +++ b/YACReaderLibrary/comic_flow_widget.cpp @@ -0,0 +1,355 @@ +#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); +} + +void ComicFlowWidgetSW::resortCovers(QList newOrder) +{ + flow->resortCovers(newOrder); +} + + +////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////// +///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::resortCovers(QList newOrder) +{ + flow->resortCovers(newOrder); +} + +//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);} diff --git a/YACReaderLibrary/comic_flow_widget.h b/YACReaderLibrary/comic_flow_widget.h new file mode 100644 index 00000000..90eaccba --- /dev/null +++ b/YACReaderLibrary/comic_flow_widget.h @@ -0,0 +1,131 @@ +#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; + virtual void resortCovers(QList newOrder) = 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); + void resortCovers(QList newOrder); +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); + void resortCovers(QList newOrder); +//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 diff --git a/YACReaderLibrary/comic_vine/api_key_dialog.cpp b/YACReaderLibrary/comic_vine/api_key_dialog.cpp new file mode 100644 index 00000000..355ecd0e --- /dev/null +++ b/YACReaderLibrary/comic_vine/api_key_dialog.cpp @@ -0,0 +1,68 @@ +#include "api_key_dialog.h" + +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +ApiKeyDialog::ApiKeyDialog(QWidget *parent) : + QDialog(parent) +{ + QVBoxLayout * layout = new QVBoxLayout; + QHBoxLayout * buttonsLayout = 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("ComicVine"); + + QLabel * info = new QLabel(tr("Before you can connect to Comic Vine, you need your own API key. Please, get one free
here")); + info->setWordWrap(true); + info->setOpenExternalLinks(true); + edit = new QLineEdit(); + edit->setPlaceholderText(tr("Paste here your Comic Vine API key")); + connect(edit,SIGNAL(textChanged(QString)),this,SLOT(enableAccept(QString))); + + acceptButton = new QPushButton(tr("Accept")); + acceptButton->setDisabled(true); + connect(acceptButton,SIGNAL(clicked()),this,SLOT(saveApiKey())); + + cancelButton = new QPushButton(tr("Cancel")); + connect(cancelButton,SIGNAL(clicked()),this,SLOT(reject())); + + layout->addWidget(info); + layout->addWidget(edit); + layout->addStretch(); + + buttonsLayout->addStretch(); + buttonsLayout->addWidget(acceptButton); + buttonsLayout->addWidget(cancelButton); + + layout->addLayout(buttonsLayout); + + setLayout(layout); + + resize(400,150); + + if(settings->contains(COMIC_VINE_API_KEY)) + edit->setText(settings->value(COMIC_VINE_API_KEY).toString()); +} + +ApiKeyDialog::~ApiKeyDialog() +{ + delete settings; +} + +void ApiKeyDialog::enableAccept(const QString &text) +{ + //TODO key validation + acceptButton->setEnabled(!text.isEmpty()); +} + +void ApiKeyDialog::saveApiKey() +{ + settings->setValue(COMIC_VINE_API_KEY,edit->text()); + accept(); +} diff --git a/YACReaderLibrary/comic_vine/api_key_dialog.h b/YACReaderLibrary/comic_vine/api_key_dialog.h new file mode 100644 index 00000000..13ff5750 --- /dev/null +++ b/YACReaderLibrary/comic_vine/api_key_dialog.h @@ -0,0 +1,31 @@ +#ifndef API_KEY_DIALOG_H +#define API_KEY_DIALOG_H + +#include + +class QPushButton; +class QLineEdit; +class QSettings; + +class ApiKeyDialog : public QDialog +{ + Q_OBJECT +public: + explicit ApiKeyDialog(QWidget *parent = 0); + ~ApiKeyDialog(); +signals: + +public slots: + +protected slots: + void enableAccept(const QString & text); + void saveApiKey(); + +protected: + QPushButton * acceptButton; + QPushButton * cancelButton; + QLineEdit * edit; + QSettings * settings; +}; + +#endif // API_KEY_DIALOG_H diff --git a/YACReaderLibrary/comic_vine/comic_vine.pri b/YACReaderLibrary/comic_vine/comic_vine.pri new file mode 100644 index 00000000..c76c0acc --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine.pri @@ -0,0 +1,46 @@ + +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 \ + comic_vine/api_key_dialog.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 \ + comic_vine/api_key_dialog.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..cb36d3ca --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_client.cpp @@ -0,0 +1,171 @@ +#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 = "%CV_API_KEY%"; //get from settings +static const QString CV_API_KEY_DEFAULT = "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) +{ + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("ComicVine"); +} + +ComicVineClient::~ComicVineClient() +{ + delete settings; +} + +//CV_SEARCH +void ComicVineClient::search(const QString & query, int page) +{ + HttpWorker * search = new HttpWorker(QString(CV_SEARCH).replace(CV_API_KEY,settings->value(COMIC_VINE_API_KEY,CV_API_KEY_DEFAULT).toString()).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(QString(CV_SERIES_DETAIL).replace(CV_API_KEY,settings->value(COMIC_VINE_API_KEY,CV_API_KEY_DEFAULT).toString()).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(QString(CV_COMICS_INFO).replace(CV_API_KEY,settings->value(COMIC_VINE_API_KEY,CV_API_KEY_DEFAULT).toString()).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(QString(CV_COMIC_DETAIL).replace(CV_API_KEY,settings->value(COMIC_VINE_API_KEY,CV_API_KEY_DEFAULT).toString()).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(QString(CV_COMIC_DETAIL).replace(CV_API_KEY,settings->value(COMIC_VINE_API_KEY,CV_API_KEY_DEFAULT).toString()).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..f5fbc44c --- /dev/null +++ b/YACReaderLibrary/comic_vine/comic_vine_client.h @@ -0,0 +1,46 @@ +#ifndef COMIC_VINE_CLIENT_H +#define COMIC_VINE_CLIENT_H + +#include "http_worker.h" + +#include +#include + +class ComicVineClient : public QObject +{ + Q_OBJECT +public: + explicit ComicVineClient(QObject *parent = 0); + ~ComicVineClient(); + +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); + +protected: + QSettings * settings; + +}; +#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..fcfea60b --- /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"), p.errorDescription()); + 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..eb212107 --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/response_parser.cpp @@ -0,0 +1,83 @@ +#include "response_parser.h" + +#include +#include + +ResponseParser::ResponseParser(QObject *parent) : + QObject(parent),error(false),numResults(-1),currentPage(-1),totalPages(-1),errorTxt("None") +{ +} + +bool ResponseParser::responseError() +{ + return error; +} + +QString ResponseParser::errorDescription() +{ + return errorTxt; +} + +qint32 ResponseParser::getNumResults() +{ + return numResults; +} + +qint32 ResponseParser::getCurrentPage() +{ + return currentPage; +} + +qint32 ResponseParser::getTotalPages() +{ + return totalPages; +} + +bool ResponseParser::isError(qint32 error) +{ + switch(error) + { + case 100: + return true; + + default: + return false; + } +} + +void ResponseParser::loadJSONResponse(const QString &response) +{ + QScriptEngine engine; + QScriptValue sc; + sc = engine.evaluate("(" + response + ")"); + + errorTxt = "None"; + + if (!sc.property("status_code").isValid() || isError(sc.property("status_code").toInt32())) + { + error = true; + if(sc.property("error").isValid()) + errorTxt = sc.property("error").toString(); + else + errorTxt = "Unknown error"; + } + 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..10014ecb --- /dev/null +++ b/YACReaderLibrary/comic_vine/model/response_parser.h @@ -0,0 +1,30 @@ +#ifndef RESPONSE_PARSER_H +#define RESPONSE_PARSER_H + +#include + +class ResponseParser : public QObject +{ + Q_OBJECT +public: + explicit ResponseParser(QObject *parent = 0); + bool responseError(); + QString errorDescription(); + qint32 getNumResults(); + qint32 getCurrentPage(); + qint32 getTotalPages(); + bool isError(qint32 error); +signals: + +public slots: + void loadJSONResponse(const QString & response); + +protected: + bool error; + QString errorTxt; + 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..0e4f479d --- /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 additional information. At least one field is needed.")); + QLabel * label = new QLabel(tr("Please provide some additional 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..8351f685 --- /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 additional 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..ef3fd009 --- /dev/null +++ b/YACReaderLibrary/comics_remover.cpp @@ -0,0 +1,63 @@ +#include "comics_remover.h" + +#include +#include + +#include "QsLog.h" + +ComicsRemover::ComicsRemover(QModelIndexList & il, QList & ps, QObject *parent) + :QObject(parent),indexList(il), paths(ps) +{ +} + +void ComicsRemover::process() +{ + 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(); +} + + +FoldersRemover::FoldersRemover(QModelIndexList &il, QList &ps, QObject *parent) + :QObject(parent),indexList(il), paths(ps) +{ + +} + +void FoldersRemover::process() +{ + QString currentFolderPath; + QListIterator i(indexList); + QListIterator i2(paths); + i.toBack(); + i2.toBack(); + + QLOG_DEBUG() << "Deleting folders" << paths.at(0); + + while (i.hasPrevious() && i2.hasPrevious()) + { + QModelIndex mi = i.previous(); + currentFolderPath = i2.previous(); + QDir d(currentFolderPath); + if(d.removeRecursively() || !d.exists()) //the folder is in the DB but no in the drive... + emit remove(mi); + else + emit removeError(); + } + + emit finished(); +} diff --git a/YACReaderLibrary/comics_remover.h b/YACReaderLibrary/comics_remover.h new file mode 100644 index 00000000..ff9d0a21 --- /dev/null +++ b/YACReaderLibrary/comics_remover.h @@ -0,0 +1,47 @@ +#ifndef COMICS_REMOVER_H +#define COMICS_REMOVER_H + +#include + +#include +#include + +class ComicsRemover : public QObject +{ + Q_OBJECT +public: + explicit ComicsRemover(QModelIndexList & indexList, QList & paths, QObject *parent = 0); + +signals: + void remove(int); + void removeError(); + void finished(); + +public slots: + void process(); + +private: + QModelIndexList indexList; + QList paths; +}; + +class FoldersRemover : public QObject +{ + Q_OBJECT +public: + explicit FoldersRemover(QModelIndexList & indexList, QList & paths, QObject *parent = 0); + +signals: + void remove(QModelIndex); + void removeError(); + void finished(); + +public slots: + void process(); + +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..2aa52bec --- /dev/null +++ b/YACReaderLibrary/comics_view.cpp @@ -0,0 +1,69 @@ +#include "comics_view.h" +#include "comic.h" +#include "comic_files_manager.h" + +#include "QsLog.h" + +ComicsView::ComicsView(QWidget *parent) : + QWidget(parent),model(NULL) +{ + setAcceptDrops(true); +} + +void ComicsView::setModel(ComicModel *m) +{ + model = m; +} + +void ComicsView::dragEnterEvent(QDragEnterEvent *event) +{ + if(model->canDropMimeData(event->mimeData(),event->proposedAction(),0,0,QModelIndex())) + event->acceptProposedAction(); + else + { + QLOG_INFO() << "dragEnterEvent"; + QList urlList; + + if (event->mimeData()->hasUrls() && event->dropAction() == Qt::CopyAction) + { + urlList = event->mimeData()->urls(); + QString currentPath; + foreach (QUrl url, urlList) + { + //comics or folders are accepted, folders' content is validate in dropEvent (avoid any lag before droping) + currentPath = url.toLocalFile(); + if(Comic::fileIsComic(currentPath) || QFileInfo(currentPath).isDir()) + { + event->acceptProposedAction(); + return; + } + } + } + } +} + +void ComicsView::dropEvent(QDropEvent *event) +{ + QLOG_DEBUG() << "drop" << event->dropAction(); + + bool validAction = event->dropAction() == Qt::CopyAction;// || event->dropAction() & Qt::MoveAction; TODO move + + if(event->mimeData()->hasUrls() && validAction) + { + + QList > droppedFiles = ComicFilesManager::getDroppedFiles(event->mimeData()->urls()); + + if(event->dropAction() == Qt::CopyAction) + { + QLOG_DEBUG() << "copy :" << droppedFiles; + emit copyComicsToCurrentFolder(droppedFiles); + } + else if(event->dropAction() & Qt::MoveAction) + { + QLOG_DEBUG() << "move :" << droppedFiles; + emit moveComicsToCurrentFolder(droppedFiles); + } + + event->acceptProposedAction(); + } +} diff --git a/YACReaderLibrary/comics_view.h b/YACReaderLibrary/comics_view.h new file mode 100644 index 00000000..2d4f1940 --- /dev/null +++ b/YACReaderLibrary/comics_view.h @@ -0,0 +1,56 @@ +#ifndef COMICS_VIEW_H +#define COMICS_VIEW_H + +#include + +#include "comic_model.h" + +class YACReaderTableView; +class QSplitter; +class ComicFlowWidget; +class QToolBar; +class ComicModel; +class ComicsView : public QWidget +{ + Q_OBJECT +public: + explicit ComicsView(QWidget *parent = 0); + virtual void setToolBar(QToolBar * toolBar) = 0; + virtual void setModel(ComicModel *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; + virtual void enableFilterMode(bool enabled) = 0; + virtual void selectIndex(int index) = 0; + +signals: + void selected(unsigned int); + void comicRated(int,QModelIndex); + + //Context menus + void customContextMenuViewRequested(QPoint); + void customContextMenuItemRequested(QPoint); + + //Drops + void copyComicsToCurrentFolder(QList >); + void moveComicsToCurrentFolder(QList >); + +public slots: + virtual void setShowMarks(bool show) = 0; + virtual void selectAll() = 0; +protected: + ComicModel * model; + + //Drop to import + void dragEnterEvent(QDragEnterEvent *event); + void dropEvent(QDropEvent *event); + +private: + +}; + +#endif // COMICS_VIEW_H diff --git a/YACReaderLibrary/comics_view_transition.cpp b/YACReaderLibrary/comics_view_transition.cpp new file mode 100644 index 00000000..6734d479 --- /dev/null +++ b/YACReaderLibrary/comics_view_transition.cpp @@ -0,0 +1,85 @@ +#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); + +#ifdef Q_OS_MAC + textLabel->setStyleSheet("QLabel {color:#888888; font-size:24px;font-family:Arial;font-weight:bold;}"); + setStyleSheet("QWidget {background:#FFFFFF}"); +#else + textLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); + setStyleSheet("QWidget {background:#2A2A2A}"); +#endif + + //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); + + //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); + +#ifdef Q_OS_MAC + painter.fillRect(0,0,width(),height(),QColor("#FFFFFF")); +#else + painter.fillRect(0,0,width(),height(),QColor("#2A2A2A")); +#endif +} 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..1ae28ff1 --- /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::open(const YACReaderLibraries & libs) +{ + libraries = libs; + QDialog::open(); +} + +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::open(); + 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..2736552c --- /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 open(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/comic_item.cpp b/YACReaderLibrary/db/comic_item.cpp new file mode 100644 index 00000000..9382d897 --- /dev/null +++ b/YACReaderLibrary/db/comic_item.cpp @@ -0,0 +1,47 @@ + +#include + +#include "comic_item.h" + +//! [0] +ComicItem::ComicItem(const QList &data) + +{ + itemData = data; +} +//! [0] + +//! [1] +ComicItem::~ComicItem() +{ + +} +//! [1] + + +//! [5] +int ComicItem::columnCount() const +{ + return itemData.count(); +} +//! [5] + +//! [6] +QVariant ComicItem::data(int column) const +{ + return itemData.value(column); +} +//! [6] + +void ComicItem::setData(int column,const QVariant & value) +{ + itemData[column] = value; +} + +//! [8] +int ComicItem::row() const +{ + + return 0; +} +//! [8] diff --git a/YACReaderLibrary/db/comic_item.h b/YACReaderLibrary/db/comic_item.h new file mode 100644 index 00000000..35d6fa54 --- /dev/null +++ b/YACReaderLibrary/db/comic_item.h @@ -0,0 +1,27 @@ +#ifndef TABLEITEM_H +#define TABLEITEM_H + +#include +#include + +//! [0] +class ComicItem : public QObject +{ + Q_OBJECT +public: + ComicItem(const QList &data); + ~ComicItem(); + 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 diff --git a/YACReaderLibrary/db/comic_model.cpp b/YACReaderLibrary/db/comic_model.cpp new file mode 100644 index 00000000..922360b6 --- /dev/null +++ b/YACReaderLibrary/db/comic_model.cpp @@ -0,0 +1,1139 @@ + +#include +#include +#include + +#include "comic_item.h" +#include "comic_model.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 +#include "QsLog.h" + + +ComicModel::ComicModel(QObject *parent) + : QAbstractItemModel(parent) +{ + connect(this,SIGNAL(beforeReset()),this,SIGNAL(modelAboutToBeReset())); + connect(this,SIGNAL(reset()),this,SIGNAL(modelReset())); +} + +ComicModel::ComicModel( QSqlQuery &sqlquery, QObject *parent) + : QAbstractItemModel(parent) +{ + setupModelData(sqlquery); +} + +ComicModel::~ComicModel() +{ + qDeleteAll(_data); +} + +int ComicModel::columnCount(const QModelIndex &parent) const +{ + Q_UNUSED(parent) + if(_data.isEmpty()) + return 0; + return _data.first()->columnCount(); +} + +bool ComicModel::canDropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) const +{ + if(!enableResorting) + return false; + return data->formats().contains(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat); +} + +//TODO: optimize this method +bool ComicModel::dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) +{ + + QAbstractItemModel::dropMimeData(data,action,row,column,parent); + QLOG_INFO() << ">>>>>>>>>>>>>>dropMimeData ComicModel<<<<<<<<<<<<<<<<<"<< parent << row << "," << column; + + if(!data->formats().contains(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat)) + return false; + + QList comicIds = YACReader::mimeDataToComicsIds(data); + QList currentIndexes; + int i; + foreach(qulonglong id, comicIds) + { + i = 0; + foreach (ComicItem *item, _data) { + if(item->data(Id)==id) + { + currentIndexes << i; + break; + } + i++; + } + } + + std::sort(currentIndexes.begin(), currentIndexes.end()); + QList resortedData; + + if(currentIndexes.contains(row))//no resorting + return false; + + ComicItem * destinationItem; + if(row == -1 || row >= _data.length()) + destinationItem = 0; + else + destinationItem = _data.at(row); + + QList newSorting; + + i = 0; + foreach (ComicItem *item, _data) { + if(!currentIndexes.contains(i)) + { + + if(item == destinationItem) { + foreach(int index, currentIndexes) + { + resortedData << _data.at(index); + newSorting << index; + } + } + + resortedData << item; + newSorting << i; + } + + i++; + } + + if(destinationItem == 0) + { + foreach(int index, currentIndexes) + { + resortedData << _data.at(index); + newSorting << index; + } + } + + QLOG_INFO() << newSorting; + + if(!beginMoveRows(parent,currentIndexes.first(),currentIndexes.last(),parent,row)) + return false; + _data = resortedData; + + + //TODO emit signals + //TODO fix selection + QList allComicIds; + foreach (ComicItem *item, _data) { + allComicIds << item->data(Id).toULongLong(); + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + switch (mode) { + case Favorites: + DBHelper::reasignOrderToComicsInFavorites(allComicIds,db); + break; + case Label: + DBHelper::reasignOrderToComicsInLabel(sourceId,allComicIds,db); + break; + case ReadingList: + DBHelper::reasignOrderToComicsInReadingList(sourceId,allComicIds,db); + break; + } + + QSqlDatabase::removeDatabase(_databasePath); + + endMoveRows(); + + emit resortedIndexes(newSorting); + int destSelectedIndex = row<0?_data.length():row; + + if(destSelectedIndex>currentIndexes.at(0)) + emit newSelectedIndex(index(qMax(0,destSelectedIndex-1),0,parent)); + else + emit newSelectedIndex(index(qMax(0,destSelectedIndex),0,parent)); + + return true; +} + +bool ComicModel::canBeResorted() +{ + return enableResorting; +} + +QMimeData *ComicModel::mimeData(const QModelIndexList &indexes) const +{ + //custom model data + //application/yacreader-comics-ids + list of ids in a QByteArray + QList ids; + foreach(QModelIndex index, indexes) + { + QLOG_DEBUG() << "dragging : " << index.data(IdRole).toULongLong(); + ids << index.data(IdRole).toULongLong(); + + } + + QByteArray data; + QDataStream out(&data,QIODevice::WriteOnly); + out << ids; //serialize the list of identifiers + + QMimeData * mimeData = new QMimeData(); + mimeData->setData(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat, data); + + return mimeData; +} + +QStringList ComicModel::mimeTypes() const +{ + QLOG_DEBUG() << "mimeTypes"; + QStringList list; + list << YACReader::YACReaderLibrarComiscSelectionMimeDataFormat; + return list; +} + +QHash ComicModel::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; +} + +QVariant ComicModel::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 ComicModel::Number: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::NumPages: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::Hash: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::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 + + ComicItem *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 == FileNameRole) + return item->data(FileName); + 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(ComicModel::HasBeenOpened); + else if (role == IdRole) + return item->data(Id); + + if (role != Qt::DisplayRole) + return QVariant(); + + if(index.column() == ComicModel::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() == ComicModel::ReadColumn) + return (item->data(ComicModel::CurrentPage).toInt()==item->data(ComicModel::NumPages).toInt() || item->data(ComicModel::ReadColumn).toBool())?QVariant(tr("yes")):QVariant(tr("no")); + if(index.column() == ComicModel::CurrentPage) + return item->data(ComicModel::HasBeenOpened).toBool()?item->data(index.column()):QVariant("-"); + + if (index.column() == ComicModel::Rating) + return QVariant(); + + return item->data(index.column()); +} + +Qt::ItemFlags ComicModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + if(index.column() == ComicModel::Rating) + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable; + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDragEnabled ; +} + +QVariant ComicModel::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 ComicModel::Number: + return QVariant(QString("#")); + case ComicModel::Title: + return QVariant(QString(tr("Title"))); + case ComicModel::FileName: + return QVariant(QString(tr("File Name"))); + case ComicModel::NumPages: + return QVariant(QString(tr("Pages"))); + case ComicModel::Hash: + return QVariant(QString(tr("Size"))); + case ComicModel::ReadColumn: + return QVariant(QString(tr("Read"))); + case ComicModel::CurrentPage: + return QVariant(QString(tr("Current Page"))); + case ComicModel::Rating: + return QVariant(QString(tr("Rating"))); + } + } + + if (orientation == Qt::Horizontal && role == Qt::TextAlignmentRole) + { + switch(section)//TODO obtener esto de la query + { + case ComicModel::Number: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::NumPages: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::Hash: + return QVariant(Qt::AlignRight | Qt::AlignVCenter); + case ComicModel::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(ComicModel::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(); +} + +QModelIndex ComicModel::index(int row, int column, const QModelIndex &parent) + const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + return createIndex(row, column, _data.at(row)); +} + +QModelIndex ComicModel::parent(const QModelIndex &index) const +{ + Q_UNUSED(index) + return QModelIndex(); +} + +int ComicModel::rowCount(const QModelIndex &parent) const +{ + if (parent.column() > 0) + return 0; + + if (!parent.isValid()) + return _data.count(); + + return 0; +} + +QStringList ComicModel::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(ComicModel::Hash).toString(); + paths << source+ hash +".jpg"; + } + + return paths; +} + +void ComicModel::setupFolderModelData(unsigned long long int folderId,const QString & databasePath) +{ + enableResorting = false; + mode = Folder; + sourceId=folderId; + + beginResetModel(); + qDeleteAll(_data); + _data.clear(); + + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + QSqlQuery selectQuery(db); + 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(); + setupModelData(selectQuery); + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); + + /*if(_data.length()==0) + emit isEmpty();*/ +} + +void ComicModel::setupLabelModelData(unsigned long long parentLabel, const QString &databasePath) +{ + enableResorting = true; + mode = Label; + sourceId = parentLabel; + + beginResetModel(); + qDeleteAll(_data); + _data.clear(); + + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + QSqlQuery selectQuery(db); + 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) " + "INNER JOIN comic_label cl ON (c.id == cl.comic_id) " + "WHERE cl.label_id = :parentLabelId " + "ORDER BY cl.ordering"); + selectQuery.bindValue(":parentLabelId", parentLabel); + selectQuery.exec(); + setupModelDataForList(selectQuery); + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); + + /*if(_data.length()==0) + emit isEmpty();*/ +} + +void ComicModel::setupReadingListModelData(unsigned long long parentReadingList, const QString &databasePath) +{ + mode = ReadingList; + sourceId = parentReadingList; + + beginResetModel(); + qDeleteAll(_data); + _data.clear(); + + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + QList ids; + ids << parentReadingList; + + QSqlQuery subfolders(db); + subfolders.prepare("SELECT id " + "FROM reading_list " + "WHERE parentId = :parentId " + "ORDER BY ordering ASC"); + subfolders.bindValue(":parentId", parentReadingList); + subfolders.exec(); + while(subfolders.next()) + ids << subfolders.record().value(0).toULongLong(); + + enableResorting = ids.length()==1;//only resorting if no sublists exist + + + foreach(qulonglong id, ids) + { + QSqlQuery selectQuery(db); + 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) " + "INNER JOIN comic_reading_list crl ON (c.id == crl.comic_id) " + "WHERE crl.reading_list_id = :parentReadingList " + "ORDER BY crl.ordering"); + selectQuery.bindValue(":parentReadingList", id); + selectQuery.exec(); + + //TODO, extra information is needed (resorting) + QList tempData = _data; + _data.clear(); + + setupModelDataForList(selectQuery); + + _data = tempData << _data; + } + + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); +} + +void ComicModel::setupFavoritesModelData(const QString &databasePath) +{ + enableResorting = true; + mode = Favorites; + + beginResetModel(); + qDeleteAll(_data); + _data.clear(); + + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + QSqlQuery selectQuery(db); + 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) " + "INNER JOIN comic_default_reading_list cdrl ON (c.id == cdrl.comic_id) " + "WHERE cdrl.default_reading_list_id = :parentDefaultListId " + "ORDER BY cdrl.ordering"); + selectQuery.bindValue(":parentDefaultListId", 1); + selectQuery.exec(); + setupModelData(selectQuery); + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); + + /*if(_data.length()==0) + emit isEmpty();*/ +} + +void ComicModel::setupReadingModelData(const QString &databasePath) +{ + enableResorting = false; + mode = Reading; + + beginResetModel(); + qDeleteAll(_data); + _data.clear(); + + _databasePath = databasePath; + QSqlDatabase db = DataBaseManagement::loadDatabase(databasePath); + { + QSqlQuery selectQuery(db); + 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 ci.hasBeenOpened = 1 AND ci.read = 0 AND ci.currentPage != ci.numPages AND ci.currentPage != 1"); + selectQuery.exec(); + setupModelData(selectQuery); + } + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + endResetModel(); + + /*if(_data.length()==0) + emit isEmpty();*/ +} + +void ComicModel::setupModelData(const SearchModifiers modifier, const QString &filter, 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); + + switch (modifier) { + case YACReader::NoModifiers: + 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 UPPER(ci.title) LIKE UPPER(:filter) OR UPPER(c.fileName) LIKE UPPER(:filter) LIMIT :limit"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":limit",500); //TODO, load this value from settings + break; + + case YACReader::OnlyRead: + 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 (UPPER(ci.title) LIKE UPPER(:filter) OR UPPER(c.fileName) LIKE UPPER(:filter)) AND ci.read = 1 LIMIT :limit"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":limit",500); //TODO, load this value from settings + break; + + case YACReader::OnlyUnread: + 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 (UPPER(ci.title) LIKE UPPER(:filter) OR UPPER(c.fileName) LIKE UPPER(:filter)) AND ci.read = 0 LIMIT :limit"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":limit",500); //TODO, load this value from settings + break; + + default: + QLOG_ERROR() << "not implemented"; + break; + } + + + selectQuery.exec(); + + QLOG_DEBUG() << selectQuery.lastError() << "--"; + + //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(); + + emit searchNumResults(_data.length()); +} + +QString ComicModel::getComicPath(QModelIndex mi) +{ + if(mi.isValid()) + return _data.at(mi.row())->data(ComicModel::Path).toString(); + return ""; +} + +void ComicModel::setupModelData(QSqlQuery &sqlquery) +{ + ComicItem * currentItem; + while (sqlquery.next()) + { + QList data; + QSqlRecord record = sqlquery.record(); + for(int i=0;idata(ComicModel::FileName).toString(); + QString nameCurrent = currentItem->data(ComicModel::FileName).toString(); + int numberLast,numberCurrent; + int max = (std::numeric_limits::max)(); + numberLast = numberCurrent = max; + + if(!last->data(ComicModel::Number).isNull()) + numberLast = last->data(ComicModel::Number).toInt(); + + if(!currentItem->data(ComicModel::Number).isNull()) + numberCurrent = currentItem->data(ComicModel::Number).toInt(); + + QList::iterator i; + i = _data.end(); + i--; + + if(numberCurrent != max) //sort the current item by issue number + { + while ((lessThan =numberCurrent < numberLast) && i != _data.begin()) + { + i--; + numberLast = max; + + if(!(*i)->data(ComicModel::Number).isNull()) + numberLast = (*i)->data(ComicModel::Number).toInt(); + } + + if(lessThan) + _data.insert(i,currentItem); + else + { + if(numberCurrent == numberLast) + if(currentItem->data(ComicModel::IsBis).toBool()) + { + _data.insert(++i,currentItem); + } + else + _data.insert(i,currentItem); + else + _data.insert(++i,currentItem); + } + continue; + } + + else //sort the current item by title + { + while ((lessThan = naturalSortLessThanCI(nameCurrent,nameLast)) && i != _data.begin() && numberLast == max) + { + i--; + nameLast = (*i)->data(ComicModel::FileName).toString(); + numberLast = max; + + if(!(*i)->data(ComicModel::Number).isNull()) + numberLast = (*i)->data(ComicModel::Number).toInt(); + } + + if(numberLast != max) + _data.insert(++i,currentItem); + else + if(lessThan) + _data.insert(i,currentItem); + else + _data.insert(++i,currentItem); + continue; + + } + } + } +} + +//comics are sorted by "ordering", the sorting is done in the sql query +void ComicModel::setupModelDataForList(QSqlQuery &sqlquery) +{ + while (sqlquery.next()) + { + QList data; + QSqlRecord record = sqlquery.record(); + for(int i=0;idata(ComicModel::Id).toULongLong(),db); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return c; +} + +ComicDB ComicModel::_getComic(const QModelIndex & mi) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + ComicDB c = DBHelper::loadComic(_data.at(mi.row())->data(ComicModel::Id).toULongLong(),db); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return c; +} + + +QVector ComicModel::getReadList() +{ + int numComics = _data.count(); + QVector readList(numComics); + for(int i=0;idata(ComicModel::ReadColumn).toBool()) + readList[i] = YACReader::Read; + else if (_data.value(i)->data(ComicModel::CurrentPage).toInt() == _data.value(i)->data(ComicModel::NumPages).toInt()) + readList[i] = YACReader::Read; + else if (_data.value(i)->data(ComicModel::HasBeenOpened).toBool()) + readList[i] = YACReader::Opened; + else + readList[i] = YACReader::Unread; + } + return readList; +} +//TODO untested, this method is no longer used +QVector ComicModel::setAllComicsRead(YACReaderComicReadStatus read) +{ + return setComicsRead(persistentIndexList(),read); +} + +QList ComicModel::getAllComics() +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + + QList comics; + int numComics = _data.count(); + for(int i=0;idata(ComicModel::Id).toULongLong(),db)); + } + + db.commit(); + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + return comics; +} + +QList ComicModel::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 ComicModel::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(ComicModel::ReadColumn, QVariant(true)); + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(ComicModel::Id).toULongLong(),db); + c.info.read = true; + DBHelper::update(&(c.info),db); + } + if(read == YACReader::Unread) + { + _data.value(mi.row())->setData(ComicModel::ReadColumn, QVariant(false)); + _data.value(mi.row())->setData(ComicModel::CurrentPage, QVariant(1)); + _data.value(mi.row())->setData(ComicModel::HasBeenOpened, QVariant(false)); + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(ComicModel::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(),ComicModel::ReadColumn),index(list.last().row(),ComicModel::HasBeenOpened),QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); + + return getReadList(); +} +qint64 ComicModel::asignNumbers(QList list,int startingNumber) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + qint64 idFirst = _data.value(list[0].row())->data(ComicModel::Id).toULongLong(); + int i = 0; + foreach (QModelIndex mi, list) + { + ComicDB c = DBHelper::loadComic(_data.value(mi.row())->data(ComicModel::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(0,ComicModel::Number),index(_data.count()-1,ComicModel::HasBeenOpened)); + + return idFirst; +} +QModelIndex ComicModel::getIndexFromId(quint64 id) +{ + QList::ConstIterator itr; + int i=0; + for(itr = _data.constBegin();itr != _data.constEnd();itr++) + { + if((*itr)->data(ComicModel::Id).toULongLong() == id) + break; + i++; + } + + return index(i,0); +} + +//TODO completely inefficiently +QList ComicModel::getIndexesFromIds(const QList &comicIds) +{ + QList comicsIndexes; + + foreach(qulonglong id,comicIds) + comicsIndexes << getIndexFromId(id); + + return comicsIndexes; +} + +void ComicModel::startTransaction() +{ + + dbTransaction = DataBaseManagement::loadDatabase(_databasePath); + dbTransaction.transaction(); +} + +void ComicModel::finishTransaction() +{ + dbTransaction.commit(); + dbTransaction.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +void ComicModel::removeInTransaction(int row) +{ + ComicDB c = DBHelper::loadComic(_data.at(row)->data(ComicModel::Id).toULongLong(),dbTransaction); + + DBHelper::removeFromDB(&c,dbTransaction); + beginRemoveRows(QModelIndex(),row,row); + removeRow(row); + delete _data.at(row); + _data.removeAt(row); + + endRemoveRows(); +} + +void ComicModel::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 ComicModel::remove(int row) +{ + removeInTransaction(row); +} + +void ComicModel::reload(const ComicDB & comic) +{ + int row = 0; + bool found = false; + foreach(ComicItem * item,_data) + { + if(item->data(ComicModel::Id).toULongLong() == comic.id) + { + found = true; + item->setData(ComicModel::ReadColumn,comic.info.read); + item->setData(ComicModel::CurrentPage,comic.info.currentPage); + item->setData(ComicModel::HasBeenOpened,true); + break; + + } + row++; + } + if(found) + emit dataChanged(index(row,ReadColumn),index(row,HasBeenOpened), QVector() << ReadColumnRole << CurrentPageRole << HasBeenOpenedRole); +} + +void ComicModel::resetComicRating(const QModelIndex &mi) +{ + ComicDB comic = getComic(mi); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + comic.info.rating = 0; + _data[mi.row()]->setData(ComicModel::Rating,0); + DBHelper::update(&(comic.info),db); + + emit dataChanged(mi,mi); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +void ComicModel::addComicsToFavorites(const QList &comicIds) +{ + addComicsToFavorites(getIndexesFromIds(comicIds)); +} + +void ComicModel::addComicsToFavorites(const QList & comicsList) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::insertComicsInFavorites(comics,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +void ComicModel::addComicsToLabel(const QList &comicIds, qulonglong labelId) +{ + addComicsToLabel(getIndexesFromIds(comicIds),labelId); +} + +void ComicModel::addComicsToLabel(const QList &comicsList, qulonglong labelId) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::insertComicsInLabel(comics,labelId,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +void ComicModel::addComicsToReadingList(const QList &comicIds, qulonglong readingListId) +{ + addComicsToReadingList(getIndexesFromIds(comicIds),readingListId); +} + +void ComicModel::addComicsToReadingList(const QList &comicsList, qulonglong readingListId) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::insertComicsInReadingList(comics,readingListId,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +void ComicModel::deleteComicsFromFavorites(const QList &comicsList) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::deleteComicsFromFavorites(comics,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + deleteComicsFromModel(comicsList); + +} + +void ComicModel::deleteComicsFromLabel(const QList &comicsList, qulonglong labelId) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::deleteComicsFromLabel(comics,labelId,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + deleteComicsFromModel(comicsList); +} + +void ComicModel::deleteComicsFromReadingList(const QList &comicsList, qulonglong readingListId) +{ + QList comics = getComics(comicsList); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + DBHelper::deleteComicsFromReadingList(comics,readingListId,db); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); + + deleteComicsFromModel(comicsList); +} + +void ComicModel::deleteComicsFromModel(const QList &comicsList) +{ + QListIterator it(comicsList); + it.toBack(); + while(it.hasPrevious()) + { + int row = it.previous().row(); + beginRemoveRows(QModelIndex(),row,row); + _data.removeAt(row); + endRemoveRows(); + } + + if(_data.isEmpty()) + emit isEmpty(); +} + + +void ComicModel::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(ComicModel::Rating,rating); + DBHelper::update(&(comic.info),db); + + emit dataChanged(mi,mi); + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} diff --git a/YACReaderLibrary/db/comic_model.h b/YACReaderLibrary/db/comic_model.h new file mode 100644 index 00000000..bd48038b --- /dev/null +++ b/YACReaderLibrary/db/comic_model.h @@ -0,0 +1,166 @@ +#ifndef TABLEMODEL_H +#define TABLEMODEL_H + +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +class ComicDB; + +class ComicItem; + +using namespace YACReader; + +//! [0] +class ComicModel : public QAbstractItemModel +{ + Q_OBJECT + +public: + ComicModel(QObject *parent = 0); + ComicModel( QSqlQuery &sqlquery, QObject *parent = 0); + ~ComicModel(); + + 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; + bool canDropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) const; + bool dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent); + bool canBeResorted(); + QMimeData * mimeData(const QModelIndexList &indexes) const; + QStringList mimeTypes() const; + + void setupFolderModelData(unsigned long long int parentFolder,const QString & databasePath); + void setupLabelModelData(unsigned long long int parentLabel, const QString & databasePath); + void setupReadingListModelData(unsigned long long int parentReadingList, const QString & databasePath); + void setupFavoritesModelData(const QString & databasePath); + void setupReadingModelData(const QString & databasePath); + //configures the model for showing the comics matching the filter criteria. + void setupModelData(const SearchModifiers modifier, const QString & filter, 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); + QList getIndexesFromIds(const QList &comicIds); + //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); + + + void addComicsToFavorites(const QList &comicsList); + void addComicsToLabel(const QList &comicsList, qulonglong labelId); + void addComicsToReadingList(const QList &comicsList, qulonglong readingListId); + + void deleteComicsFromFavorites(const QList &comicsList); + void deleteComicsFromLabel(const QList &comicsList, qulonglong labelId); + void deleteComicsFromReadingList(const QList &comicsList, qulonglong readingListId); + + void deleteComicsFromModel(const QList &comicsList); + + 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 + + }; + + enum Mode { + Folder, + Favorites, + Reading, + Label, + ReadingList + }; + + + +public slots: + void remove(int row); + void startTransaction(); + void finishTransaction(); + void updateRating(int rating, QModelIndex mi); + + void addComicsToFavorites(const QList &comicIds); + void addComicsToLabel(const QList &comicIds, qulonglong labelId); + void addComicsToReadingList(const QList &comicIds, qulonglong readingListId); + +protected: + +private: + void setupModelData( QSqlQuery &sqlquery); + void setupModelDataForList(QSqlQuery &sqlquery); + ComicDB _getComic(const QModelIndex & mi); + QList _data; + + QString _databasePath; + + QSqlDatabase dbTransaction; + + bool enableResorting; + Mode mode; + qulonglong sourceId; + +signals: + void beforeReset(); + void reset(); + void isEmpty(); + void searchNumResults(int); + void resortedIndexes(QList); + void newSelectedIndex(const QModelIndex &); +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db/data_base_management.cpp b/YACReaderLibrary/db/data_base_management.cpp new file mode 100644 index 00000000..e8052aee --- /dev/null +++ b/YACReaderLibrary/db/data_base_management.cpp @@ -0,0 +1,790 @@ +#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(); + + //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(); + + //8.0> tables + success = success && DataBaseManagement::createV8Tables(database); + + } + + return success; +} + +bool DataBaseManagement::createV8Tables(QSqlDatabase &database) +{ + bool success = true; + { + //8.0> tables + //LABEL + QSqlQuery queryLabel(database); + success = success && queryLabel.exec("CREATE TABLE label (id INTEGER PRIMARY KEY, " + "name TEXT NOT NULL, " + "color TEXT NOT NULL, " + "ordering INTEGER NOT NULL); "); //order depends on the color + + QSqlQuery queryIndexLabel(database); + success = success && queryIndexLabel.exec("CREATE INDEX label_ordering_index ON label (ordering)"); + + //COMIC LABEL + QSqlQuery queryComicLabel(database); + success = success && queryComicLabel.exec("CREATE TABLE comic_label (" + "comic_id INTEGER, " + "label_id INTEGER, " + "ordering INTEGER, " //TODO order???? + "FOREIGN KEY(label_id) REFERENCES label(id) ON DELETE CASCADE, " + "FOREIGN KEY(comic_id) REFERENCES comic(id) ON DELETE CASCADE, " + "PRIMARY KEY(label_id, comic_id))"); + + QSqlQuery queryIndexComicLabel(database); + success = success && queryIndexComicLabel.exec("CREATE INDEX comic_label_ordering_index ON label (ordering)"); + + //READING LIST + QSqlQuery queryReadingList(database); + success = success && queryReadingList.exec("CREATE TABLE reading_list (" + "id INTEGER PRIMARY KEY, " + "parentId INTEGER, " + "ordering INTEGER DEFAULT 0, " //only use it if the parentId is NULL + "name TEXT NOT NULL, " + "finished BOOLEAN DEFAULT 0, " + "completed BOOLEAN DEFAULT 1, " + "FOREIGN KEY(parentId) REFERENCES reading_list(id) ON DELETE CASCADE)"); + + QSqlQuery queryIndexReadingList(database); + success = success && queryIndexReadingList.exec("CREATE INDEX reading_list_ordering_index ON label (ordering)"); + + //COMIC READING LIST + QSqlQuery queryComicReadingList(database); + success = success && queryComicReadingList.exec("CREATE TABLE comic_reading_list (" + "reading_list_id INTEGER, " + "comic_id INTEGER, " + "ordering INTEGER, " + "FOREIGN KEY(reading_list_id) REFERENCES reading_list(id) ON DELETE CASCADE, " + "FOREIGN KEY(comic_id) REFERENCES comic(id) ON DELETE CASCADE, " + "PRIMARY KEY(reading_list_id, comic_id))"); + + QSqlQuery queryIndexComicReadingList(database); + success = success && queryIndexComicReadingList.exec("CREATE INDEX comic_reading_list_ordering_index ON label (ordering)"); + + //DEFAULT READING LISTS + QSqlQuery queryDefaultReadingList(database); + success = success && queryDefaultReadingList.exec("CREATE TABLE default_reading_list (" + "id INTEGER PRIMARY KEY, " + "name TEXT NOT NULL" + //TODO icon???? + ")"); + + //COMIC DEFAULT READING LISTS + QSqlQuery queryComicDefaultReadingList(database); + success = success && queryComicDefaultReadingList.exec("CREATE TABLE comic_default_reading_list (" + "comic_id INTEGER, " + "default_reading_list_id INTEGER, " + "ordering INTEGER, " //order???? + "FOREIGN KEY(default_reading_list_id) REFERENCES default_reading_list(id) ON DELETE CASCADE, " + "FOREIGN KEY(comic_id) REFERENCES comic(id) ON DELETE CASCADE," + "PRIMARY KEY(default_reading_list_id, comic_id))"); + + QSqlQuery queryIndexComicDefaultReadingList(database); + success = success && queryIndexComicDefaultReadingList.exec("CREATE INDEX comic_default_reading_list_ordering_index ON label (ordering)"); + + //INSERT DEFAULT READING LISTS + QSqlQuery queryInsertDefaultReadingList(database); + //if(!queryInsertDefaultReadingList.prepare()) + + //1 Favorites + //queryInsertDefaultReadingList.bindValue(":name", "Favorites"); + success = success && queryInsertDefaultReadingList.exec("INSERT INTO default_reading_list (name) VALUES (\"Favorites\")"); + + //Reading doesn't need its onw list + + } + 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; + bool pre8 = false; + + if(compareVersions(DataBaseManagement::checkValidDB(fullPath),"7.0.0")<0) + pre7 = true; + if(compareVersions(DataBaseManagement::checkValidDB(fullPath),"7.0.3")<0) + pre7_1 = true; + if(compareVersions(DataBaseManagement::checkValidDB(fullPath),"8.0.0")<0) + pre8 = 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); + } + } + + if(pre8) + { + returnValue = returnValue && createV8Tables(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..68540339 --- /dev/null +++ b/YACReaderLibrary/db/data_base_management.h @@ -0,0 +1,62 @@ +#ifndef __DATA_BASE_MANAGEMENT_H +#define __DATA_BASE_MANAGEMENT_H + +#include +#include +#include + +#include "folder_model.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 bool createV8Tables(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/folder_item.cpp b/YACReaderLibrary/db/folder_item.cpp new file mode 100644 index 00000000..069147f3 --- /dev/null +++ b/YACReaderLibrary/db/folder_item.cpp @@ -0,0 +1,103 @@ +#include + +#include "folder_item.h" +#include "qnaturalsorting.h" + +FolderItem::FolderItem(const QList &data, FolderItem *parent) +{ + parentItem = parent; + itemData = data; +} + +FolderItem::~FolderItem() +{ + qDeleteAll(childItems); +} + +void FolderItem::appendChild(FolderItem *item) +{ + item->parentItem = this; + + if(childItems.isEmpty()) + childItems.append(item); + else + { + FolderItem * 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); +} + +FolderItem *FolderItem::child(int row) +{ + return childItems.value(row); +} + +int FolderItem::childCount() const +{ + return childItems.count(); +} + +int FolderItem::columnCount() const +{ + return itemData.count(); +} + +QVariant FolderItem::data(int column) const +{ + return itemData.value(column); +} + +void FolderItem::setData(int column, const QVariant & value) +{ + itemData[column] = value; +} + +void FolderItem::removeChild(int childIndex) +{ + childItems.removeAt(childIndex); +} + +void FolderItem::clearChildren() +{ + qDeleteAll(childItems); + childItems.clear(); +} + +QList FolderItem::children() +{ + return childItems; +} + +FolderItem *FolderItem::parent() +{ + return parentItem; +} + +int FolderItem::row() const +{ + if (parentItem) + return parentItem->childItems.indexOf(const_cast(this)); + + return 0; +} + +QList FolderItem::getData() const +{ + return itemData; +} diff --git a/YACReaderLibrary/db/folder_item.h b/YACReaderLibrary/db/folder_item.h new file mode 100644 index 00000000..f0841c0f --- /dev/null +++ b/YACReaderLibrary/db/folder_item.h @@ -0,0 +1,77 @@ +/**************************************************************************** +** +** 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 + +class FolderItem +{ +public: + FolderItem(const QList &data, FolderItem *parent = 0); + ~FolderItem(); + + void appendChild(FolderItem *child); + + FolderItem *child(int row); + int childCount() const; + int columnCount() const; + QVariant data(int column) const; + QList getData() const; + int row() const; + FolderItem *parent(); + FolderItem *parentItem; + unsigned long long int id; + QList comicNames; + FolderItem * originalItem; + void setData(int column, const QVariant &value); + void removeChild(int childIndex); + void clearChildren(); + QList children(); +private: + QList childItems; + QList itemData; +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db/folder_model.cpp b/YACReaderLibrary/db/folder_model.cpp new file mode 100644 index 00000000..10420299 --- /dev/null +++ b/YACReaderLibrary/db/folder_model.cpp @@ -0,0 +1,784 @@ +/**************************************************************************** +** +** 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 "folder_item.h" +#include "folder_model.h" +#include "data_base_management.h" +#include "folder.h" +#include "db_helper.h" +#include "qnaturalsorting.h" +#include "yacreader_global.h" +#include "QsLog.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 + +FolderModel::FolderModel(QObject *parent) + : QAbstractItemModel(parent),rootItem(0) +{ + connect(this,SIGNAL(beforeReset()),this,SIGNAL(modelAboutToBeReset())); + connect(this,SIGNAL(reset()),this,SIGNAL(modelReset())); +} + +//! [0] +FolderModel::FolderModel( QSqlQuery &sqlquery, QObject *parent) + : QAbstractItemModel(parent),rootItem(0) +{ + //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 FolderItem(rootData); + rootItem->id = ROOT; + rootItem->parentItem = 0; + setupModelData(sqlquery, rootItem); + //sqlquery.finish(); +} +//! [0] + +//! [1] +FolderModel::~FolderModel() +{ + if(rootItem != 0) + delete rootItem; +} +//! [1] + +//! [2] +int FolderModel::columnCount(const QModelIndex &parent) const +{ + if (parent.isValid()) + return static_cast(parent.internalPointer())->columnCount(); + else + return rootItem->columnCount(); +} +//! [2] + +//! [3] +QVariant FolderModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + + FolderItem *item = static_cast(index.internalPointer()); + + if (role == Qt::DecorationRole) + +#ifdef Q_OS_MAC + if(item->data(FolderModel::Finished).toBool()){ + if(finishedFolderIcon.isNull()){ + drawMacOSXFinishedFolderIcon(); + } + + return QVariant(finishedFolderIcon); + } + else { + return QVariant(QFileIconProvider().icon(QFileIconProvider::Folder)); + } +#else + if(item->data(FolderModel::Finished).toBool()) + return QVariant(YACReader::noHighlightedIcon(":/images/sidebar/folder_finished.png")); + else + return QVariant(YACReader::noHighlightedIcon(":/images/sidebar/folder.png")); +#endif + + if(role == FolderModel::CompletedRole) + return item->data(FolderModel::Completed); + + if(role == FolderModel::FinishedRole) + return item->data(FolderModel::Finished); + + if (role != Qt::DisplayRole) + return QVariant(); + + + + return item->data(index.column()); +} +//! [3] + +//! [4] +Qt::ItemFlags FolderModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDropEnabled | Qt::ItemIsDragEnabled; +} +//! [4] + +//! [5] +QVariant FolderModel::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 FolderModel::index(int row, int column, const QModelIndex &parent) + const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + FolderItem *parentItem; + + if (!parent.isValid()) + parentItem = rootItem; + else + parentItem = static_cast(parent.internalPointer()); + + FolderItem *childItem = parentItem->child(row); + if (childItem) + return createIndex(row, column, childItem); + else + return QModelIndex(); +} +//! [6] + +//! [7] +QModelIndex FolderModel::parent(const QModelIndex &index) const +{ + if (!index.isValid()) + return QModelIndex(); + + FolderItem *childItem = static_cast(index.internalPointer()); + FolderItem *parentItem = childItem->parent(); + + if (parentItem == rootItem) + return QModelIndex(); + + return createIndex(parentItem->row(), 0, parentItem); +} +//! [7] + +/* +QModelIndex FolderModel::indexFromItem(FolderItem * 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 FolderModel::rowCount(const QModelIndex &parent) const +{ + FolderItem *parentItem; + if (parent.column() > 0) + return 0; + + if (!parent.isValid()) + parentItem = rootItem; + else + parentItem = static_cast(parent.internalPointer()); + + return parentItem->childCount(); +} +//! [8] + +void FolderModel::setupModelData(QString path) +{ + beginResetModel(); + if(rootItem != 0) + delete rootItem; //TODO comprobar que se libera bien la memoria + + rootItem = 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 FolderItem(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 FolderModel::setupModelData(QSqlQuery &sqlquery, FolderItem *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(); + FolderItem * item = new FolderItem(data); + + item->id = record.value("id").toULongLong(); + //la inserci�n de hijos se hace de forma ordenada + FolderItem * 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 FolderModel::updateFolderModelData(QSqlQuery &sqlquery, FolderItem *parent) +{ + while (sqlquery.next()) { + QLOG_DEBUG () << "habia 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(); + FolderItem * item = new FolderItem(data); + + item->id = record.value("id").toULongLong(); + //la inserci�n de hijos se hace de forma ordenada + FolderItem * 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); + } +} + +QString FolderModel::getDatabase() +{ + return _databasePath; +} + +QString FolderModel::getFolderPath(const QModelIndex &folder) +{ + if(!folder.isValid()) //root folder + return "/"; + return static_cast(folder.internalPointer())->data(FolderModel::Path).toString(); +} + +/* +void FolderModel::resetFilter() +{ + beginResetModel(); + filter = ""; + includeComics = false; + //TODO hay que liberar la memoria reservada para el filtrado + //items.clear(); + filteredItems.clear(); + FolderItem * 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 FolderModel::updateFolderCompletedStatus(const QModelIndexList &list, bool status) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + foreach (QModelIndex mi, list) + { + FolderItem * item = static_cast(mi.internalPointer()); + item->setData(FolderModel::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(),FolderModel::Name),index(list.last().row(),FolderModel::Completed)); +} + +void FolderModel::updateFolderFinishedStatus(const QModelIndexList &list, bool status) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + db.transaction(); + foreach (QModelIndex mi, list) + { + FolderItem * item = static_cast(mi.internalPointer()); + item->setData(FolderModel::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(),FolderModel::Name),index(list.last().row(),FolderModel::Completed)); +} + +QStringList FolderModel::getSubfoldersNames(const QModelIndex &mi) +{ + QStringList result; + qulonglong id = 1; + if(mi.isValid()){ + FolderItem * 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; +} + +void FolderModel::fetchMoreFromDB(const QModelIndex &parent) +{ + FolderItem * item; + if(parent.isValid()) + item = static_cast(parent.internalPointer()); + else + item = rootItem; + + //Remove all children + if(item->childCount() > 0) + { + beginRemoveRows(parent, 0, item->childCount()-1); + item->clearChildren(); + endRemoveRows(); + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + QList items; + QList nextLevelItems; + + QSqlQuery selectQuery(db); + selectQuery.prepare("select * from folder where id <> 1 and parentId = :parentId order by parentId,name"); + + items << item; + bool firstLevelUpdated = false; + while(items.size() > 0) + { + nextLevelItems.clear(); + foreach(FolderItem * item, items) + { + QLOG_DEBUG() << "ID " << item->id; + selectQuery.bindValue(":parentId", item->id); + + selectQuery.exec(); + + if(!firstLevelUpdated) + { + //NO size support + int numResults = 0; + while(selectQuery.next()) + numResults++; + + if(!selectQuery.seek(-1)) + selectQuery.exec(); + //END no size support + + beginInsertRows(parent, 0, numResults-1); + } + + updateFolderModelData(selectQuery,item); + + if(!firstLevelUpdated) + { + endInsertRows(); + firstLevelUpdated = true; + } + + nextLevelItems << item->children(); + + } + + items.clear(); + items = nextLevelItems; + } + + QLOG_DEBUG() << "item->childCount()-1" << item->childCount()-1; + + + db.close(); + QSqlDatabase::removeDatabase(_databasePath); +} + +QModelIndex FolderModel::addFolderAtParent(const QString &folderName, const QModelIndex &parent) +{ + FolderItem * parentItem; + + if(parent.isValid()) + parentItem = static_cast(parent.internalPointer()); + else + parentItem = rootItem; + + Folder newFolder; + newFolder.name = folderName; + newFolder.parentId = parentItem->id; + newFolder.path = parentItem->data(1).toString() + "/" + folderName; + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + newFolder.id = DBHelper::insert(&newFolder, db); + QSqlDatabase::removeDatabase(_databasePath); + + int destRow = 0; + + QList data; + data << newFolder.name; + data << newFolder.path; + data << false; //finished + data << true; //completed + + FolderItem * item = new FolderItem(data); + item->id = newFolder.id; + + beginInsertRows(parent,0,0); //TODO calculate the destRow before inserting the new child + + parentItem->appendChild(item); + destRow = parentItem->children().indexOf(item); //TODO optimize this, appendChild should return the index of the new item + items.insert(item->id,item); + + endInsertRows(); + + return index(destRow,0,parent); +} + +void FolderModel::deleteFolder(const QModelIndex &mi) +{ + beginRemoveRows(mi.parent(),mi.row(),mi.row()); + + FolderItem * item = static_cast(mi.internalPointer()); + + FolderItem * parent = item->parent(); + parent->removeChild(mi.row()); + + Folder f; + f.setId(item->id); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + DBHelper::removeFromDB(&f,db); + QSqlDatabase::removeDatabase(_databasePath); + + endRemoveRows(); +} + + +//PROXY + +FolderModelProxy::FolderModelProxy(QObject *parent) + :QSortFilterProxyModel(parent),rootItem(0),filterEnabled(false),filter(""),includeComics(true) +{ + +} + +FolderModelProxy::~FolderModelProxy() +{ + +} + +bool FolderModelProxy::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const +{ + if(!filterEnabled) + return true; + + FolderItem * parent = static_cast(source_parent.internalPointer()); + + if(parent == 0) + parent = static_cast(sourceModel())->rootItem; + + FolderItem * item = parent->children().at(source_row); + + return filteredItems.contains(item->id); +} + +void FolderModelProxy::setFilter(const YACReader::SearchModifiers modifier, QString filter, bool includeComics) +{ + clear(); + this->filter = filter; + this->includeComics = includeComics; + this->modifier = modifier; + filterEnabled = true; + setupFilteredModelData(); +} + +void FolderModelProxy::setupFilteredModelData() +{ + beginResetModel(); + + //TODO hay que liberar memoria de anteriores filtrados + + //inicializar el nodo ra�z + + if(rootItem != 0) + delete rootItem; //TODO comprobar que se libera bien la memoria + + rootItem = 0; + + //inicializar el nodo ra�z + QList rootData; + rootData << "root"; + rootItem = new FolderItem(rootData); + rootItem->id = ROOT; + rootItem->parentItem = 0; + + FolderModel * model = static_cast(sourceModel()); + + //cargar la base de datos + QSqlDatabase db = DataBaseManagement::loadDatabase(model->_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 + { + switch(modifier) + { + case YACReader::NoModifiers: + selectQuery.prepare("SELECT DISTINCT f.id, f.parentId, f.name, f.path, f.finished, f.completed " + "FROM folder f LEFT 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+"%%"); + break; + + case YACReader::OnlyRead: + selectQuery.prepare("SELECT DISTINCT f.id, f.parentId, f.name, f.path, f.finished, f.completed " + "FROM folder f LEFT JOIN (comic c INNER JOIN comic_info ci ON (c.comicInfoId = ci.id)) ON (f.id = c.parentId) " + "WHERE f.id <> 1 AND ((UPPER(c.fileName) like UPPER(:filter)) OR (UPPER(f.name) like UPPER(:filter2))) AND ci.read = 1 ORDER BY f.parentId,f.name;"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":filter2", "%%"+filter+"%%"); + break; + + case YACReader::OnlyUnread: + selectQuery.prepare("SELECT DISTINCT f.id, f.parentId, f.name, f.path, f.finished, f.completed " + "FROM folder f LEFT JOIN (comic c INNER JOIN comic_info ci ON (c.comicInfoId = ci.id)) ON (f.id = c.parentId) " + "WHERE f.id <> 1 AND ((UPPER(c.fileName) like UPPER(:filter)) OR (UPPER(f.name) like UPPER(:filter2))) AND ci.read = 0 ORDER BY f.parentId,f.name;"); + selectQuery.bindValue(":filter", "%%"+filter+"%%"); + selectQuery.bindValue(":filter2", "%%"+filter+"%%"); + break; + + default: + QLOG_ERROR() << "not implemented"; + break; + + } + + + } + selectQuery.exec(); + + setupFilteredModelData(selectQuery,rootItem); + } + //selectQuery.finish(); + db.close(); + QSqlDatabase::removeDatabase(model->_databasePath); + + endResetModel(); +} + +void FolderModelProxy::clear() +{ + filterEnabled = false; + + filteredItems.clear(); + + QSortFilterProxyModel::clear(); +} + +void FolderModelProxy::setupFilteredModelData(QSqlQuery &sqlquery, FolderItem *parent) +{ + FolderModel * model = static_cast(sourceModel()); + + //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(); + + FolderItem * item = new FolderItem(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 = model->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 + FolderItem * parentItem = model->items.value(parentId); + //se debe crear un nuevo nodo (para no compartir los hijos con el nodo original) + FolderItem * newparentItem = new FolderItem(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); + } + } +} diff --git a/YACReaderLibrary/db/folder_model.h b/YACReaderLibrary/db/folder_model.h new file mode 100644 index 00000000..4d0f16bb --- /dev/null +++ b/YACReaderLibrary/db/folder_model.h @@ -0,0 +1,151 @@ +/**************************************************************************** +** +** 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 +#include + +#include "yacreader_global.h" + +class FolderItem; + +class FolderModelProxy : public QSortFilterProxyModel +{ + Q_OBJECT +public: + explicit FolderModelProxy(QObject *parent = 0); + ~FolderModelProxy(); + + void setFilter(const YACReader::SearchModifiers modifier, QString filter, bool includeComics); + void setupFilteredModelData( QSqlQuery &sqlquery, FolderItem *parent); + void setupFilteredModelData(); + void clear(); + + bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const; + +protected: + FolderItem *rootItem; + QMap filteredItems; //relación entre folders + + bool includeComics; + QString filter; + bool filterEnabled; + + YACReader::SearchModifiers modifier; +}; + +class FolderModel : public QAbstractItemModel +{ + + Q_OBJECT + + friend class FolderModelProxy; + +public: + FolderModel(QObject *parent = 0); + FolderModel( QSqlQuery &sqlquery, QObject *parent = 0); + ~FolderModel(); + + //QAbstractItemModel methods + 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; + + //Convenience methods + void setupModelData(QString path); + QString getDatabase(); + QString getFolderPath(const QModelIndex &folder); + //QModelIndex indexFromItem(FolderItem * item, int column); + + + //bool isFilterEnabled(){return filterEnabled;}; + + void updateFolderCompletedStatus(const QModelIndexList & list, bool status); + void updateFolderFinishedStatus(const QModelIndexList & list, bool status); + + QStringList getSubfoldersNames(const QModelIndex & mi); + + void fetchMoreFromDB(const QModelIndex & parent); + + QModelIndex addFolderAtParent(const QString & folderName, const QModelIndex & parent); + + 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 + + enum Roles { + FinishedRole = Qt::UserRole + 1, + CompletedRole + }; + +public slots: + void deleteFolder(const QModelIndex & mi); + +private: + void setupModelData( QSqlQuery &sqlquery, FolderItem *parent); + void updateFolderModelData( QSqlQuery &sqlquery, FolderItem *parent); + + FolderItem *rootItem; //el árbol + QMap items; //relación entre folders + + QString _databasePath; + +signals: + void beforeReset(); + void reset(); +}; +//! [0] + +#endif diff --git a/YACReaderLibrary/db/reading_list_item.cpp b/YACReaderLibrary/db/reading_list_item.cpp new file mode 100644 index 00000000..22df4018 --- /dev/null +++ b/YACReaderLibrary/db/reading_list_item.cpp @@ -0,0 +1,239 @@ +#include "reading_list_item.h" +#include "qnaturalsorting.h" + +#include + +ListItem::ListItem(const QList &data) + :itemData(data) +{ + +} + +int ListItem::columnCount() +{ + return itemData.count(); +} + +QVariant ListItem::data(int column) const +{ + return itemData.at(column); +} + +qulonglong ListItem::getId() const +{ + return 0; +} + +//------------------------------------------------------ + +SpecialListItem::SpecialListItem(const QList &data) + :ListItem(data) +{ + +} + +QIcon SpecialListItem::getIcon() const +{ + if(itemData.count()>Id) + { + QString id = itemData.at(Id).toString(); + return YACReader::noHighlightedIcon(QString(":/images/lists/default_%1.png").arg(id)); + } +} + +ReadingListModel::TypeSpecialList SpecialListItem::getType() const +{ + if(itemData.count()>Id) + { + int id = itemData.at(Id).toInt(); + return (ReadingListModel::TypeSpecialList)id; + } +} + +//------------------------------------------------------ + +LabelItem::LabelItem(const QList &data) + :ListItem(data) +{ + +} + +QIcon LabelItem::getIcon() const +{ + if(itemData.count()>Color) + { + QString color = itemData.at(Color).toString(); + return YACReader::noHighlightedIcon(QString(":/images/lists/label_%1.png").arg(color).toLower()); + } +} + +YACReader::LabelColors LabelItem::colorid() const +{ + if(itemData.count()>Ordering) + { + return YACReader::LabelColors(itemData.at(Ordering).toInt()); + } +} + +QString LabelItem::name() const +{ + if(itemData.count()>Name) + { + return itemData.at(Name).toString(); + } +} + +void LabelItem::setName(const QString &name) +{ + if(itemData.count()>Name) + { + itemData[Name] = name; + } +} + +qulonglong LabelItem::getId() const +{ + if(itemData.count()>Id) + { + return YACReader::LabelColors(itemData.at(Id).toULongLong()); + } +} + +//------------------------------------------------------ + +ReadingListItem::ReadingListItem(const QList &data, ReadingListItem *p) + :ListItem(data), parent(p) +{ + +} + +QIcon ReadingListItem::getIcon() const +{ + if(parent->getId() == 0) + return YACReader::noHighlightedIcon(":/images/lists/list.png"); //top level list + else +#ifdef Q_OS_MAC + return QFileIconProvider().icon(QFileIconProvider::Folder); +#else + return YACReader::noHighlightedIcon(":/images/sidebar/folder.png"); //sublist +#endif +} + +int ReadingListItem::childCount() const +{ + return childItems.count(); +} + +ReadingListItem *ReadingListItem::child(int row) +{ + return childItems.at(row); +} + +//items are sorted by order +void ReadingListItem::appendChild(ReadingListItem *item) +{ + item->parent = this; + + if(childItems.isEmpty()) + childItems.append(item); + else + { + if(item->parent->getId()==0) //sort by name, top level child + { + int i= 0; + while(iname(),item->name())) + i++; + childItems.insert(i,item); + } + else + { + int i= 0; + while(igetOrdering()getOrdering())) + i++; + childItems.insert(i,item); + } + + /*ReadingListItem * 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);*/ + + } + +} + +void ReadingListItem::appendChild(ReadingListItem *item, int pos) +{ + childItems.insert(pos, item); +} + +void ReadingListItem::removeChild(ReadingListItem *item) +{ + childItems.removeOne(item); +} + +qulonglong ReadingListItem::getId() const +{ + if(itemData.count()>Id) + return itemData.at(Id).toULongLong(); +} + +QString ReadingListItem::name() const +{ + if(itemData.count()>Name) + return itemData.at(Name).toString(); +} + +void ReadingListItem::setName(const QString &name) +{ + if(itemData.count()>Name) + itemData[Name] = name; +} + +int ReadingListItem::getOrdering() const +{ + if(itemData.count()>Ordering) + return itemData[Ordering].toInt(); +} + +void ReadingListItem::setOrdering(const int ordering) +{ + if(itemData.count()>Ordering) + itemData[Ordering] = ordering; +} + +QList ReadingListItem::children() +{ + return childItems; +} + +int ReadingListItem::row() const +{ + if (parent) + return parent->childItems.indexOf(const_cast(this)); + + return 0; +} + + +ReadingListSeparatorItem::ReadingListSeparatorItem() + :ListItem(QList()) +{ + +} + +QIcon ReadingListSeparatorItem::getIcon() const +{ + return QIcon(); +} diff --git a/YACReaderLibrary/db/reading_list_item.h b/YACReaderLibrary/db/reading_list_item.h new file mode 100644 index 00000000..e79fea62 --- /dev/null +++ b/YACReaderLibrary/db/reading_list_item.h @@ -0,0 +1,103 @@ +#ifndef READING_LIST_ITEM_H +#define READING_LIST_ITEM_H + +#include +#include + +#include "yacreader_global.h" +#include "reading_list_model.h" +//TODO add propper constructors, using QList is not safe + +class ListItem +{ +public: + ListItem(const QList &data); + int columnCount(); + virtual QIcon getIcon() const = 0; + QVariant data(int column) const; + virtual qulonglong getId() const; + QList itemData; +}; + +//------------------------------------------------------ + +class SpecialListItem : public ListItem +{ +public: + SpecialListItem(const QList &data); + QIcon getIcon() const; + ReadingListModel::TypeSpecialList getType() const; +private: + enum DataIndexes { + Name, + Id + }; + +}; + +//------------------------------------------------------ + +class LabelItem : public ListItem +{ +public: + LabelItem(const QList &data); + QIcon getIcon() const; + YACReader::LabelColors colorid() const; + QString name() const; + void setName(const QString & name); + qulonglong getId() const; + + +private: + enum DataIndexes { + Name, + Color, + Id, + Ordering + }; +}; + +//------------------------------------------------------ + +class ReadingListItem : public ListItem +{ +public: + ReadingListItem(const QList &data, ReadingListItem * parent = 0); + QIcon getIcon() const; + ReadingListItem * parent; + int childCount() const; + int row() const; + ReadingListItem * child(int row); + void appendChild(ReadingListItem *item); + void appendChild(ReadingListItem *item, int pos); + void removeChild(ReadingListItem *item); + qulonglong getId() const; + QString name() const; + void setName(const QString & name); + int getOrdering() const; + void setOrdering(const int ordering); + QList children(); + +private: + QList childItems; + + enum DataIndexes { + Name, + Id, + Finished, + Completed, + Ordering + }; + +}; + +//------------------------------------------------------ + +class ReadingListSeparatorItem : public ListItem +{ +public: + ReadingListSeparatorItem(); + QIcon getIcon() const; +}; + +#endif // READING_LIST_ITEM_H diff --git a/YACReaderLibrary/db/reading_list_model.cpp b/YACReaderLibrary/db/reading_list_model.cpp new file mode 100644 index 00000000..7eb76f6d --- /dev/null +++ b/YACReaderLibrary/db/reading_list_model.cpp @@ -0,0 +1,778 @@ +#include "reading_list_model.h" + +#include "reading_list_item.h" + +#include "data_base_management.h" +#include "qnaturalsorting.h" +#include "db_helper.h" + +#include "QsLog.h" + +#include + +ReadingListModel::ReadingListModel(QObject *parent) : + QAbstractItemModel(parent),rootItem(0) +{ + separator1 = new ReadingListSeparatorItem; + separator2 = new ReadingListSeparatorItem; +} + +int ReadingListModel::rowCount(const QModelIndex &parent) const +{ + if(!parent.isValid()) //TOP + { + int separatorsCount = 2;//labels.isEmpty()?1:2; + return specialLists.count() + labels.count() + rootItem->childCount() + separatorsCount; + } + else + { + ListItem * item = static_cast(parent.internalPointer()); + + if(typeid(*item) == typeid(ReadingListItem)) + { + ReadingListItem * item = static_cast(parent.internalPointer()); + return item->childCount(); + } + } + + return 0; +} + +int ReadingListModel::columnCount(const QModelIndex &parent) const +{ + if(parent.isValid()) + { + ListItem * item = static_cast(parent.internalPointer()); + if(typeid(*item) == typeid(ReadingListSeparatorItem)) + return 0; + } + return 1; + /*if (parent.isValid()) + return static_cast(parent.internalPointer())->columnCount(); + else + return rootItem->columnCount();*/ +} + +QVariant ReadingListModel::data(const QModelIndex &index, int role) const +{ + if(!index.isValid()) + return QVariant(); + + ListItem * item = static_cast(index.internalPointer()); + + if (role == ReadingListModel::TypeListsRole) + { + if(typeid(*item) == typeid(SpecialListItem)) + return QVariant(ReadingListModel::SpecialList); + + if(typeid(*item) == typeid(LabelItem)) + return QVariant(ReadingListModel::Label); + + if(typeid(*item) == typeid(ReadingListItem)) + return QVariant(ReadingListModel::ReadingList); + + if(typeid(*item) == typeid(ReadingListSeparatorItem)) + return QVariant(ReadingListModel::Separator); + } + + if (role == ReadingListModel::LabelColorRole && typeid(*item) == typeid(LabelItem) ) + { + LabelItem * labelItem = static_cast(item); + return QVariant(labelItem->colorid()); + } + + if (role == ReadingListModel::IDRole) + { + QLOG_DEBUG() << "getting role"; + return item->getId(); +} + + if (role == ReadingListModel::SpecialListTypeRole && typeid(*item) == typeid(SpecialListItem)) + { + SpecialListItem * specialListItem = static_cast(item); + return QVariant(specialListItem->getType()); + } + + if(typeid(*item) == typeid(ReadingListSeparatorItem)) + return QVariant(); + + if (role == Qt::DecorationRole) + { + return QVariant(item->getIcon()); + } + + if (role != Qt::DisplayRole) + return QVariant(); + + return item->data(index.column()); +} + +Qt::ItemFlags ReadingListModel::flags(const QModelIndex &index) const +{ + if (!index.isValid()) + return 0; + + ListItem * item = static_cast(index.internalPointer()); + if(typeid(*item) == typeid(ReadingListSeparatorItem)) + return 0; + + if(typeid(*item) == typeid(ReadingListItem) && static_cast(item)->parent->getId()!=0) + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDropEnabled | Qt::ItemIsDragEnabled; //only sublists are dragable + + return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDropEnabled; +} + +QVariant ReadingListModel::headerData(int section, Qt::Orientation orientation, int role) const +{ + if (orientation == Qt::Horizontal && role == Qt::DisplayRole) + return rootItem->data(section); + + return QVariant(); +} + +QModelIndex ReadingListModel::index(int row, int column, const QModelIndex &parent) const +{ + if (!hasIndex(row, column, parent)) + return QModelIndex(); + + if(!parent.isValid()) + { + int separatorsCount = 2;//labels.isEmpty()?1:2; + + if(rowIsSpecialList(row,parent)) + return createIndex(row, column, specialLists.at(row)); + + if(row == specialLists.count()) + return createIndex(row,column,separator1); + + if(rowIsLabel(row,parent)) + return createIndex(row,column,labels.at(row-specialLists.count()-1)); + + if(separatorsCount == 2) + if(row == specialLists.count() + labels.count() + 1) + return createIndex(row,column,separator2); + + if(rowIsReadingList(row,parent)) + return createIndex(row,column,rootItem->child(row - (specialLists.count() + labels.count() + separatorsCount))); + + } else //sublist + { + ReadingListItem *parentItem; + + if (!parent.isValid()) + parentItem = rootItem; //this should be impossible + else + parentItem = static_cast(parent.internalPointer()); + + ReadingListItem *childItem = parentItem->child(row); + return createIndex(row,column,childItem); + } + /*FolderItem *childItem = parentItem->child(row); + if (childItem) + return createIndex(row, column, childItem); + else*/ + return QModelIndex(); + +} + +QModelIndex ReadingListModel::parent(const QModelIndex &index) const +{ + + if(!index.isValid()) + return QModelIndex(); + + ListItem * item = static_cast(index.internalPointer()); + + if(typeid(*item) == typeid(ReadingListItem)) + { + ReadingListItem * childItem = static_cast(index.internalPointer()); + ReadingListItem * parent = childItem->parent; + if(parent->getId() != 0) + return createIndex(parent->row()+specialLists.count()+labels.count()+2, 0, parent); + } + + return QModelIndex(); +} + +bool ReadingListModel::canDropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) const +{ + QLOG_DEBUG() << "trying to drop into row = " << row << "column column = " << column << "parent" << parent; + + if(row == -1) + return false; + + if(!parent.isValid()) //top level items + { + if(row == -1) //no list + return false; + + if(row == 1) //reading is just an smart list + return false; + + if(rowIsSeparator(row,parent)) + return false; + } + + if(data->formats().contains(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat)) + return true; + + if(rowIsReadingList(row,parent))// TODO avoid droping in a different parent + if(!parent.isValid()) + return false; + else + { + QList > sublistsRows; + QByteArray rawData = data->data(YACReader::YACReaderLibrarSubReadingListMimeDataFormat); + QDataStream in(&rawData,QIODevice::ReadOnly); + in >> sublistsRows; //deserialize the list of indentifiers + if(parent.row()!= sublistsRows.at(0).second) + return false; + return data->formats().contains(YACReader::YACReaderLibrarSubReadingListMimeDataFormat); + + } + + return false; +} + +bool ReadingListModel::dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) +{ + QLOG_DEBUG() << "drop mimedata into row = " << row << " column = " << column << "parent" << parent; + if(data->formats().contains(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat)) + return dropComics(data, action, row, column, parent); + + if(data->formats().contains(YACReader::YACReaderLibrarSubReadingListMimeDataFormat)) + return dropSublist(data, action, row, column, parent); +} + +bool ReadingListModel::dropComics(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) +{ + QList comicIds = YACReader::mimeDataToComicsIds(data); + + QLOG_DEBUG() << "dropped : " << comicIds; + + QModelIndex dest; + QModelIndex parentDest; + + if(row == -1) + { + dest = parent; + } + else + dest = index(row,column,parent); + + parentDest = dest.parent(); + + if(rowIsSpecialList(dest.row(),parentDest)) { + if(dest.row() == 0) //add to favorites + { + QLOG_DEBUG() << "-------addComicsToFavorites : " << comicIds << " to " << dest.data(IDRole).toULongLong(); + emit addComicsToFavorites(comicIds); + return true; + } + } + + if(rowIsLabel(dest.row(),parentDest)) { + QLOG_DEBUG() << "+++++++++++addComicsToLabel : " << comicIds << " to " << dest.data(IDRole).toULongLong(); + emit addComicsToLabel(comicIds, dest.data(IDRole).toULongLong()); + return true; + } + + if(rowIsReadingList(dest.row(),parentDest)) { + QLOG_DEBUG() << "///////////addComicsToReadingList : " << comicIds << " to " << dest.data(IDRole).toULongLong(); + emit addComicsToReadingList(comicIds, dest.data(IDRole).toULongLong()); + return true; + } + + return false; +} + +bool ReadingListModel::dropSublist(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) +{ + QList > sublistsRows; + QByteArray rawData = data->data(YACReader::YACReaderLibrarSubReadingListMimeDataFormat); + QDataStream in(&rawData,QIODevice::ReadOnly); + in >> sublistsRows; //deserialize the list of indentifiers + + QLOG_DEBUG() << "dropped : " << sublistsRows; + + int sourceRow = sublistsRows.at(0).first; + int destRow = row; + QModelIndex destParent = parent; + if(row == -1) + { + QLOG_DEBUG() << "droping inside parent"; + destRow = parent.row(); + destParent = parent.parent(); + } + QLOG_DEBUG() << "move " << sourceRow << "-" << destRow; + + if(sourceRow == destRow) + return false; + + //beginMoveRows(destParent,sourceRow,sourceRow,destParent,destRow); + + ReadingListItem * parentItem = static_cast(destParent.internalPointer()); + ReadingListItem * child = parentItem->child(sourceRow); + parentItem->removeChild(child); + parentItem->appendChild(child,destRow); + + reorderingChildren(parentItem->children()); + //endMoveRows(); + + return true; +} + +QMimeData *ReadingListModel::mimeData(const QModelIndexList &indexes) const +{ + QLOG_DEBUG() << "mimeData requested" << indexes; + + if(indexes.length() == 0) + { + QLOG_ERROR() << "mimeData requested: indexes is empty"; + return new QMimeData();//TODO what happens if 0 is returned? + } + + if(indexes.length() > 1) + QLOG_DEBUG() << "mimeData requested for more than one index, this shouldn't be possible"; + + QModelIndex modelIndex = indexes.at(0); + + QList > rows; + rows << QPair(modelIndex.row(),modelIndex.parent().row()); + QLOG_DEBUG() << "mimeData requested for row : " << modelIndex.row(); + + QByteArray data; + QDataStream out(&data,QIODevice::WriteOnly); + out << rows; //serialize the list of identifiers + + QMimeData * mimeData = new QMimeData(); + mimeData->setData(YACReader::YACReaderLibrarSubReadingListMimeDataFormat, data); + + return mimeData; +} + +void ReadingListModel::setupReadingListsData(QString path) +{ + beginResetModel(); + + cleanAll(); + + _databasePath = path; + QSqlDatabase db = DataBaseManagement::loadDatabase(path); + + //setup special lists + specialLists = setupSpecialLists(db); + + //separator-------------------------------------------- + + //setup labels + setupLabels(db); + + //separator-------------------------------------------- + + //setup reading list + setupReadingLists(db); + + endResetModel(); +} + +void ReadingListModel::addNewLabel(const QString &name, YACReader::LabelColors color) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + qulonglong id = DBHelper::insertLabel(name, color, db); + + beginInsertRows(QModelIndex(),0, 0); + + int newPos = addLabelIntoList(new LabelItem(QList() << name << YACReader::colorToName(color) << id << color)); + + //beginInsertRows(QModelIndex(),specialLists.count()+1+newPos+1, specialLists.count()+1+newPos+1); + endInsertRows(); + + QSqlDatabase::removeDatabase(_databasePath); +} + +void ReadingListModel::addReadingList(const QString &name) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + beginInsertRows(QModelIndex(), 0, 0); //TODO calculate the right coordinates before inserting + + qulonglong id = DBHelper::insertReadingList(name,db); + ReadingListItem * newItem; + rootItem->appendChild(newItem = new ReadingListItem(QList() + << name + << id + << false + << true + << 0)); + + items.insert(id, newItem); + + /*int pos = rootItem->children().indexOf(newItem); + + pos += specialLists.count()+1+labels.count()+labels.count()>0?1:0;*/ + + + endInsertRows(); + + QSqlDatabase::removeDatabase(_databasePath); +} + +void ReadingListModel::addReadingListAt(const QString &name, const QModelIndex &mi) +{ + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + beginInsertRows(mi, 0, 0); //TODO calculate the right coordinates before inserting + + ReadingListItem * readingListParent = static_cast(mi.internalPointer()); + qulonglong id = DBHelper::insertReadingSubList(name,mi.data(IDRole).toULongLong(),readingListParent->childCount(),db); + ReadingListItem * newItem; + + readingListParent->appendChild(newItem = new ReadingListItem(QList() + << name + << id + << false + << true + << readingListParent->childCount())); + + items.insert(id, newItem); + + /*int pos = readingListParent->children().indexOf(newItem); + + pos += specialLists.count()+1+labels.count()+labels.count()>0?1:0;*/ + + + endInsertRows(); + + QSqlDatabase::removeDatabase(_databasePath); +} + +bool ReadingListModel::isEditable(const QModelIndex &mi) +{ + if(!mi.isValid()) + return false; + ListItem * item = static_cast(mi.internalPointer()); + return typeid(*item) != typeid(SpecialListItem); +} + +bool ReadingListModel::isReadingList(const QModelIndex &mi) +{ + if(!mi.isValid()) + return false; + ListItem * item = static_cast(mi.internalPointer()); + return typeid(*item) == typeid(ReadingListItem); +} + +bool ReadingListModel::isReadingSubList(const QModelIndex &mi) +{ + if(!mi.isValid()) + return false; + ListItem * item = static_cast(mi.internalPointer()); + if(typeid(*item) == typeid(ReadingListItem)) + { + ReadingListItem * readingListItem = static_cast(item); + if(readingListItem->parent == rootItem) + return false; + else + return true; + } + else + return false; +} + +QString ReadingListModel::name(const QModelIndex &mi) +{ + return data(mi,Qt::DisplayRole).toString(); +} + +void ReadingListModel::rename(const QModelIndex &mi, const QString &name) +{ + if(!isEditable(mi)) + return; + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + ListItem * item = static_cast(mi.internalPointer()); + + if(typeid(*item) == typeid(ReadingListItem)) + { + ReadingListItem * rli = static_cast(item); + rli->setName(name); + DBHelper::renameList(item->getId(), name, db); + + if(rli->parent->getId()!=0) + { + //TODO + //move row depending on the name + }else + emit dataChanged(index(mi.row(), 0), index(mi.row(), 0)); + } + else if(typeid(*item) == typeid(LabelItem)) + { + LabelItem * li = static_cast(item); + li->setName(name); + DBHelper::renameLabel(item->getId(), name, db); + emit dataChanged(index(mi.row(), 0), index(mi.row(), 0)); + } + + QSqlDatabase::removeDatabase(_databasePath); +} + +void ReadingListModel::deleteItem(const QModelIndex &mi) +{ + if(isEditable(mi)) + { + QLOG_DEBUG() << "parent row :" << mi.parent().data() << "-" << mi.row(); + beginRemoveRows(mi.parent(),mi.row(),mi.row()); + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + + ListItem * item = static_cast(mi.internalPointer()); + + if(typeid(*item) == typeid(ReadingListItem)) + { + ReadingListItem * rli = static_cast(item); + QLOG_DEBUG() << "num children : " << rli->parent->childCount(); + rli->parent->removeChild(rli); + QLOG_DEBUG() << "num children : " << rli->parent->childCount(); + DBHelper::removeListFromDB(item->getId(), db); + if(rli->parent->getId()!=0) + { + reorderingChildren(rli->parent->children()); + } + QLOG_DEBUG() << "num children : " << rli->parent->childCount(); + } + else if(typeid(*item) == typeid(LabelItem)) + { + LabelItem * li = static_cast(item); + labels.removeOne(li); + DBHelper::removeLabelFromDB(item->getId(), db); + } + + QSqlDatabase::removeDatabase(_databasePath); + + endRemoveRows(); + } +} + +const QList ReadingListModel::getLabels() +{ + return labels; +} + +void ReadingListModel::cleanAll() +{ + if(rootItem != 0) + { + delete rootItem; + + qDeleteAll(specialLists); + qDeleteAll(labels); + + specialLists.clear(); + labels.clear(); + + items.clear(); + } + + rootItem = 0; +} + +void ReadingListModel::setupReadingListsData(QSqlQuery &sqlquery, ReadingListItem *parent) +{ + items.insert(parent->getId(),parent); + + while (sqlquery.next()) + { + QSqlRecord record = sqlquery.record(); + ReadingListItem * rli = new ReadingListItem(QList() + << record.value("name") + << record.value("id") + << record.value("finished") + << record.value("completed") + << record.value("ordering")); + + ReadingListItem * currentParent; + if(record.value("parentId").isNull()) + currentParent = rootItem; + else + currentParent = items.value(record.value("parentId").toULongLong()); + + currentParent->appendChild(rli); + + items.insert(rli->getId(),rli); + } +} + +QList ReadingListModel::setupSpecialLists(QSqlDatabase & db) +{ + QList list; + + QSqlQuery selectQuery("SELECT * FROM default_reading_list ORDER BY id,name",db); + while(selectQuery.next()) { + QSqlRecord record = selectQuery.record(); + list << new SpecialListItem(QList() + << record.value("name") + << record.value("id")); + } + + //Reading after Favorites, Why? Because I want to :P + list.insert(1,new SpecialListItem(QList() << "Reading" << 0)); + + return list; +} + +void ReadingListModel::setupLabels(QSqlDatabase & db) +{ + QSqlQuery selectQuery("SELECT * FROM label ORDER BY ordering,name",db); //TODO add some kind of + while(selectQuery.next()) { + QSqlRecord record = selectQuery.record(); + addLabelIntoList(new LabelItem(QList() + << record.value("name") + << record.value("color") + << record.value("id") + << record.value("ordering"))); + } + + //TEST + +// INSERT INTO label (name, color, ordering) VALUES ("Oh Oh", "red", 1); +// INSERT INTO label (name, color, ordering) VALUES ("lalala", "orange", 2); +// INSERT INTO label (name, color, ordering) VALUES ("we are not sorry", "yellow", 3); +// INSERT INTO label (name, color, ordering) VALUES ("there we go", "green", 4); +// INSERT INTO label (name, color, ordering) VALUES ("oklabunga", "cyan", 5); +// INSERT INTO label (name, color, ordering) VALUES ("hailer mailer", "blue", 6); +// INSERT INTO label (name, color, ordering) VALUES ("lol", "violet", 7); +// INSERT INTO label (name, color, ordering) VALUES ("problems", "purple", 8); +// INSERT INTO label (name, color, ordering) VALUES ("me gussssta", "pink", 9); +// INSERT INTO label (name, color, ordering) VALUES (":D", "white", 10); +// INSERT INTO label (name, color, ordering) VALUES ("ainsss", "light", 11); +// INSERT INTO label (name, color, ordering) VALUES ("put a smile on my face", "dark", 12); + +} + +void ReadingListModel::setupReadingLists(QSqlDatabase & db) +{ + //setup root item + rootItem = new ReadingListItem(QList() << "ROOT" << 0 << true << false); + + QSqlQuery selectQuery("select * from reading_list order by parentId IS NULL DESC",db); + + //setup reading lists + setupReadingListsData(selectQuery,rootItem); + + //TEST +// ReadingListItem * node1; +// rootItem->appendChild(node1 = new ReadingListItem(QList() /*<< 0*/ << "My reading list" << "atr")); +// rootItem->appendChild(new ReadingListItem(QList() /*<< 0*/ << "X timeline" << "atr")); + +// node1->appendChild(new ReadingListItem(QList() /*<< 0*/ << "sublist" << "atr",node1)); +} + +int ReadingListModel::addLabelIntoList(LabelItem *item) +{ + if(labels.isEmpty()) + labels << item; + else + { + int i = 0; + + while (i < labels.count() && (labels.at(i)->colorid() < item->colorid()) ) + i++; + + if(i < labels.count()) + { + if(labels.at(i)->colorid() == item->colorid()) //sort by name + { + while( i < labels.count() && labels.at(i)->colorid() == item->colorid() && naturalSortLessThanCI(labels.at(i)->name(),item->name())) + i++; + } + } + + + if(i >= labels.count()) + { + QLOG_DEBUG() << "insertando label al final " << item->name(); + labels << item; + } + else + { + QLOG_DEBUG() << "insertando label en " << i << "-" << item->name(); + labels.insert(i,item); + } + + return i; + } + + return 0; +} + +void ReadingListModel::reorderingChildren(QList children) +{ + QList childrenIds; + int i = 0; + foreach (ReadingListItem * item, children) { + item->setOrdering(i++); + childrenIds << item->getId(); + } + + QSqlDatabase db = DataBaseManagement::loadDatabase(_databasePath); + DBHelper::reasignOrderToSublists(childrenIds, db); + QSqlDatabase::removeDatabase(_databasePath); +} + +bool ReadingListModel::rowIsSpecialList(int row, const QModelIndex &parent) const +{ + if(parent.isValid()) + return false; //by now no sublists in special list + + if(row >=0 && row < specialLists.count()) + return true; + + return false; +} + +bool ReadingListModel::rowIsLabel(int row, const QModelIndex &parent) const +{ + if(parent.isValid()) + return false; //by now no sublists in labels + + if(row > specialLists.count() && row <= specialLists.count() + labels.count()) + return true; + + return false; +} + +bool ReadingListModel::rowIsReadingList(int row, const QModelIndex &parent) const +{ + if(parent.isValid()) + return true; //only lists with sublists + + int separatorsCount = labels.isEmpty()?1:2; + + if(row >= specialLists.count() + labels.count() + separatorsCount) + return true; + + return false; +} + +bool ReadingListModel::rowIsSeparator(int row, const QModelIndex &parent) const +{ + if(parent.isValid()) + return false; //only separators at top level + + if(row == specialLists.count()) + return true; + + int separatorsCount = labels.isEmpty()?1:2; + if(separatorsCount == 2 && row == specialLists.count() + labels.count() + 1) + return true; + + return false; +} + +ReadingListModelProxy::ReadingListModelProxy(QObject *parent) + :QSortFilterProxyModel(parent) +{ + +} diff --git a/YACReaderLibrary/db/reading_list_model.h b/YACReaderLibrary/db/reading_list_model.h new file mode 100644 index 00000000..c60eaa3b --- /dev/null +++ b/YACReaderLibrary/db/reading_list_model.h @@ -0,0 +1,117 @@ +#ifndef READING_LIST_MODEL_H +#define READING_LIST_MODEL_H + +#include +#include +#include +#include +#include +#include + +#include "yacreader_global.h" + +class LabelItem; +class SpecialListItem; +class ReadingListItem; +class ReadingListSeparatorItem; + +class ReadingListModelProxy : public QSortFilterProxyModel +{ + Q_OBJECT +public: + explicit ReadingListModelProxy(QObject *parent = 0); +}; + +class ReadingListModel : public QAbstractItemModel +{ + Q_OBJECT +public: + explicit ReadingListModel(QObject *parent = 0); + + //QAbstractItemModel methods + int rowCount(const QModelIndex &parent = QModelIndex()) const; + int columnCount(const QModelIndex &parent = QModelIndex()) 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; + QModelIndex parent(const QModelIndex &index) const; + bool canDropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) const; + bool dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent); + QMimeData *mimeData(const QModelIndexList &indexes) const; + + //Convenience methods + void setupReadingListsData(QString path); + void addNewLabel(const QString & name, YACReader::LabelColors color); + void addReadingList(const QString & name);//top level reading list + void addReadingListAt(const QString & name, const QModelIndex & mi); + bool isEditable(const QModelIndex & mi); + bool isReadingList(const QModelIndex & mi); + bool isReadingSubList(const QModelIndex & mi); + QString name(const QModelIndex & mi); + void rename(const QModelIndex & mi, const QString & name); + void deleteItem(const QModelIndex & mi); + const QList getLabels(); + + enum Roles { + TypeListsRole = Qt::UserRole + 1, + IDRole, + LabelColorRole, + SpecialListTypeRole + }; + + enum TypeList { + SpecialList, + Label, + ReadingList, + Separator + }; + + enum TypeSpecialList { + Reading, + Favorites + }; + +signals: + + void addComicsToFavorites(const QList & comicIds); + void addComicsToLabel(const QList & comicIds, qulonglong labelId); + void addComicsToReadingList(const QList & comicIds, qulonglong readingListId); + +private: + void cleanAll(); + void setupReadingListsData(QSqlQuery &sqlquery, ReadingListItem *parent); + QList setupSpecialLists(QSqlDatabase &db); + void setupLabels(QSqlDatabase &db); + void setupReadingLists(QSqlDatabase &db); + int addLabelIntoList(LabelItem *item); + void reorderingChildren(QList children); + + bool rowIsSpecialList(int row, const QModelIndex & parent = QModelIndex()) const; + bool rowIsLabel(int row, const QModelIndex & parent = QModelIndex()) const; + bool rowIsReadingList(int row, const QModelIndex & parent = QModelIndex()) const; + bool rowIsSeparator(int row, const QModelIndex & parent = QModelIndex()) const; + + bool dropComics(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent); + bool dropSublist(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent); + //Special lists + QList specialLists; + + //Label + QList labels; + + //Reading lists + ReadingListItem * rootItem; // + QMap items; //lists relationship + + //separators + ReadingListSeparatorItem * separator1; + ReadingListSeparatorItem * separator2; + + QString _databasePath; + +}; + +#endif // READING_LIST_MODEL_H diff --git a/YACReaderLibrary/db_helper.cpp b/YACReaderLibrary/db_helper.cpp new file mode 100644 index 00000000..94b038d2 --- /dev/null +++ b/YACReaderLibrary/db_helper.cpp @@ -0,0 +1,1000 @@ +#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::getFolderSubfoldersFromLibrary(qulonglong libraryId, qulonglong folderId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList list = DBHelper::getFoldersFromParent(folderId,db,false); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return list; +} +QList DBHelper::getFolderComicsFromLibrary(qulonglong libraryId, qulonglong folderId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList list = DBHelper::getComicsFromParent(folderId,db,false); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return list; +} +qulonglong DBHelper::getParentFromComicFolderId(qulonglong libraryId, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + Folder f = DBHelper::loadFolder(id,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return f.parentId; +} +ComicDB DBHelper::getComicInfo(qulonglong libraryId, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + ComicDB comic = DBHelper::loadComic(id,db); + + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return comic; +} + +QList DBHelper::getSiblings(qulonglong libraryId, qulonglong parentId) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + QSqlDatabase db = DataBaseManagement::loadDatabase(libraryPath+"/.yacreaderlibrary"); + + QList comics = DBHelper::getSortedComicsFromParent(parentId,db); + db.close(); + QSqlDatabase::removeDatabase(libraryPath); + return comics; +} + +QString DBHelper::getFolderName(qulonglong libraryId, qulonglong id) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + 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(); +} + +void DBHelper::removeLabelFromDB(qulonglong id, QSqlDatabase &db) +{ + QSqlQuery query(db); + query.prepare("DELETE FROM label WHERE id = :id"); + query.bindValue(":id", id); + query.exec(); +} + +void DBHelper::removeListFromDB(qulonglong id, QSqlDatabase &db) +{ + QSqlQuery query(db); + query.prepare("DELETE FROM reading_list WHERE id = :id"); + query.bindValue(":id", id); + query.exec(); +} + +void DBHelper::deleteComicsFromFavorites(const QList &comicsList, QSqlDatabase &db) +{ + db.transaction(); + + QLOG_DEBUG() << "deleteComicsFromFavorites----------------------------------"; + + QSqlQuery query(db); + query.prepare("DELETE FROM comic_default_reading_list WHERE comic_id = :comic_id AND default_reading_list_id = 1"); + foreach(ComicDB comic, comicsList) + { + query.bindValue(":comic_id", comic.id); + query.exec(); + } + + db.commit(); +} + +void DBHelper::deleteComicsFromLabel(const QList &comicsList, qulonglong labelId, QSqlDatabase &db) +{ + db.transaction(); + + QLOG_DEBUG() << "deleteComicsFromLabel----------------------------------"; + + QSqlQuery query(db); + query.prepare("DELETE FROM comic_label WHERE comic_id = :comic_id AND label_id = :label_id"); + foreach(ComicDB comic, comicsList) + { + query.bindValue(":comic_id", comic.id); + query.bindValue(":label_id", labelId); + query.exec(); + + QLOG_DEBUG() << "cid = " << comic.id << "lid = " << labelId; + QLOG_DEBUG() << query.lastError().databaseText() << "-" << query.lastError().driverText(); + } + + db.commit(); +} + +void DBHelper::deleteComicsFromReadingList(const QList &comicsList, qulonglong readingListId, QSqlDatabase &db) +{ + db.transaction(); + + QLOG_DEBUG() << "deleteComicsFromReadingList----------------------------------"; + + QSqlQuery query(db); + query.prepare("DELETE FROM comic_reading_list WHERE comic_id = :comic_id AND reading_list_id = :reading_list_id"); + foreach(ComicDB comic, comicsList) + { + query.bindValue(":comic_id", comic.id); + query.bindValue(":reading_list_id", readingListId); + query.exec(); + } + + db.commit(); +} + +//updates +void DBHelper::update(ComicDB * comic, QSqlDatabase & db) +{ + Q_UNUSED(comic) + Q_UNUSED(db) + //do nothing +} + +void DBHelper::update(qulonglong libraryId, ComicInfo & comicInfo) +{ + QString libraryPath = DBHelper::getLibraries().getPath(libraryId); + 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); +} + +void DBHelper::renameLabel(qulonglong id, const QString &name, QSqlDatabase &db) +{ + QSqlQuery renameLabelQuery(db); + renameLabelQuery.prepare("UPDATE label SET " + "name = :name " + "WHERE id = :id"); + renameLabelQuery.bindValue(":name", name); + renameLabelQuery.bindValue(":id", id); + renameLabelQuery.exec(); + + QLOG_DEBUG() << renameLabelQuery.lastError().databaseText(); +} + +void DBHelper::renameList(qulonglong id, const QString &name, QSqlDatabase &db) +{ + QSqlQuery renameLabelQuery(db); + renameLabelQuery.prepare("UPDATE reading_list SET " + "name = :name " + "WHERE id = :id"); + renameLabelQuery.bindValue(":name", name); + renameLabelQuery.bindValue(":id", id); + renameLabelQuery.exec(); +} + +void DBHelper::reasignOrderToSublists(QList ids, QSqlDatabase &db) +{ + QSqlQuery updateOrdering(db); + updateOrdering.prepare("UPDATE reading_list SET " + "ordering = :ordering " + "WHERE id = :id"); + db.transaction(); + int order = 0; + foreach(qulonglong id, ids) + { + updateOrdering.bindValue(":ordering",order++); + updateOrdering.bindValue(":id", id); + updateOrdering.exec(); + } + + db.commit(); +} + +void DBHelper::reasignOrderToComicsInFavorites(QList comicIds, QSqlDatabase &db) +{ + QSqlQuery updateOrdering(db); + updateOrdering.prepare("UPDATE comic_default_reading_list SET " + "ordering = :ordering " + "WHERE comic_id = :comic_id AND default_reading_list_id = 0"); + db.transaction(); + int order = 0; + foreach(qulonglong id, comicIds) + { + updateOrdering.bindValue(":ordering",order++); + updateOrdering.bindValue(":comic_id", id); + updateOrdering.exec(); + } + + db.commit(); +} + +void DBHelper::reasignOrderToComicsInLabel(qulonglong labelId, QList comicIds, QSqlDatabase &db) +{ + QSqlQuery updateOrdering(db); + updateOrdering.prepare("UPDATE comic_label SET " + "ordering = :ordering " + "WHERE comic_id = :comic_id AND label_id = :label_id"); + db.transaction(); + int order = 0; + foreach(qulonglong id, comicIds) + { + updateOrdering.bindValue(":ordering",order++); + updateOrdering.bindValue(":comic_id", id); + updateOrdering.bindValue(":label_id", labelId); + updateOrdering.exec(); + } + + db.commit(); +} + +void DBHelper::reasignOrderToComicsInReadingList(qulonglong readingListId, QList comicIds, QSqlDatabase &db) +{ + QSqlQuery updateOrdering(db); + updateOrdering.prepare("UPDATE comic_reading_list SET " + "ordering = :ordering " + "WHERE comic_id = :comic_id AND reading_list_id = :reading_list_id"); + db.transaction(); + int order = 0; + foreach(qulonglong id, comicIds) + { + updateOrdering.bindValue(":ordering",order++); + updateOrdering.bindValue(":comic_id", id); + updateOrdering.bindValue(":reading_list_id", readingListId); + updateOrdering.exec(); + QLOG_INFO() << updateOrdering.lastError().databaseText() << "-" << updateOrdering.lastError().driverText(); + } + + db.commit(); +} + +//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(); +} + +qulonglong DBHelper::insertLabel(const QString &name, YACReader::LabelColors color, QSqlDatabase &db) +{ + QSqlQuery query(db); + query.prepare("INSERT INTO label (name, color, ordering) " + "VALUES (:name, :color, :ordering)"); + query.bindValue(":name", name); + query.bindValue(":color", YACReader::colorToName(color)); + query.bindValue(":ordering", color); + query.exec(); + return query.lastInsertId().toULongLong(); +} + +qulonglong DBHelper::insertReadingList(const QString &name, QSqlDatabase &db) +{ + QSqlQuery query(db); + query.prepare("INSERT INTO reading_list (name) " + "VALUES (:name)"); + query.bindValue(":name", name); + query.exec(); + return query.lastInsertId().toULongLong(); +} + +qulonglong DBHelper::insertReadingSubList(const QString &name, qulonglong parentId, int ordering, QSqlDatabase &db) +{ + QSqlQuery query(db); + query.prepare("INSERT INTO reading_list (name, parentId, ordering) " + "VALUES (:name, :parentId, :ordering)"); + query.bindValue(":name", name); + query.bindValue(":parentId", parentId); + query.bindValue(":ordering", ordering); + query.exec(); + return query.lastInsertId().toULongLong(); +} + +void DBHelper::insertComicsInFavorites(const QList &comicsList, QSqlDatabase &db) +{ + db.transaction(); + + QSqlQuery query(db); + query.prepare("INSERT INTO comic_default_reading_list (default_reading_list_id, comic_id) " + "VALUES (1, :comic_id)"); + + foreach(ComicDB comic, comicsList) + { + query.bindValue(":comic_id", comic.id); + //query.bindValue(":order", numComics++); + query.exec(); + } + + db.commit(); +} + +void DBHelper::insertComicsInLabel(const QList &comicsList, qulonglong labelId, QSqlDatabase &db) +{ + db.transaction(); + + QSqlQuery query(db); + query.prepare("INSERT INTO comic_label (label_id, comic_id) " + "VALUES (:label_id, :comic_id)"); + + foreach(ComicDB comic, comicsList) + { + query.bindValue(":label_id", labelId); + query.bindValue(":comic_id", comic.id); + query.exec(); + } + + db.commit(); +} + +void DBHelper::insertComicsInReadingList(const QList &comicsList, qulonglong readingListId, QSqlDatabase &db) +{ + QSqlQuery getNumComicsInFavoritesQuery("SELECT count(*) from comic_reading_list;",db); + getNumComicsInFavoritesQuery.next(); + QSqlRecord record = getNumComicsInFavoritesQuery.record(); + int numComics = record.value(0).toInt(); + + db.transaction(); + + QSqlQuery query(db); + query.prepare("INSERT INTO comic_reading_list (reading_list_id, comic_id, ordering) " + "VALUES (:reading_list_id, :comic_id, :ordering)"); + + foreach(ComicDB comic, comicsList) + { + query.bindValue(":reading_list_id", readingListId); + query.bindValue(":comic_id", comic.id); + query.bindValue(":ordering", numComics++); + query.exec(); + } + + db.commit(); +} +//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(); + folder.knownId = true; + //new 7.1 + folder.setFinished(record.value("finished").toBool()); + folder.setCompleted(record.value("completed").toBool()); + } + + return folder; +} + +Folder DBHelper::loadFolder(const QString &folderName, qulonglong parentId, QSqlDatabase &db) +{ + Folder folder; + + QLOG_DEBUG() << "Looking for folder with name = " << folderName << " and parent " << parentId; + + QSqlQuery query(db); + query.prepare("SELECT * FROM folder WHERE parentId = :parentId AND name = :folderName"); + query.bindValue(":parentId",parentId); + query.bindValue(":folderName", folderName); + query.exec(); + + folder.parentId = parentId; + if(query.next()) + { + QSqlRecord record = query.record(); + folder.id = record.value("id").toULongLong(); + folder.name = record.value("name").toString(); + folder.path = record.value("path").toString(); + folder.knownId = true; + //new 7.1 + folder.setFinished(record.value("finished").toBool()); + folder.setCompleted(record.value("completed").toBool()); + QLOG_DEBUG() << "FOUND!!"; + } + + 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..14ca3598 --- /dev/null +++ b/YACReaderLibrary/db_helper.h @@ -0,0 +1,78 @@ +#ifndef DB_HELPER_H +#define DB_HELPER_H + +class QString; +#include +#include +#include "yacreader_global.h" + +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 getFolderSubfoldersFromLibrary(qulonglong libraryId, qulonglong folderId); + static QList getFolderComicsFromLibrary(qulonglong libraryId, qulonglong folderId); + static qulonglong getParentFromComicFolderId(qulonglong libraryId, qulonglong id); + static ComicDB getComicInfo(qulonglong libraryId, qulonglong id); + static QList getSiblings(qulonglong libraryId, qulonglong parentId); + static QString getFolderName(qulonglong libraryId, 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); + static void removeLabelFromDB(qulonglong id, QSqlDatabase & db); + static void removeListFromDB(qulonglong id, QSqlDatabase & db); + //logic deletes + static void deleteComicsFromFavorites(const QList & comicsList, QSqlDatabase & db); + static void deleteComicsFromLabel(const QList & comicsList, qulonglong labelId, QSqlDatabase & db); + static void deleteComicsFromReadingList(const QList & comicsList, qulonglong readingListId, QSqlDatabase & db); + //inserts + static qulonglong insert(Folder * folder, QSqlDatabase & db); + static qulonglong insert(ComicDB * comic, QSqlDatabase & db); + static qulonglong insertLabel(const QString & name, YACReader::LabelColors color , QSqlDatabase & db); + static qulonglong insertReadingList(const QString & name, QSqlDatabase & db); + static qulonglong insertReadingSubList(const QString & name, qulonglong parentId, int ordering, QSqlDatabase & db); + static void insertComicsInFavorites(const QList & comicsList, QSqlDatabase & db); + static void insertComicsInLabel(const QList & comicsList, qulonglong labelId, QSqlDatabase & db); + static void insertComicsInReadingList(const QList & comicsList, qulonglong readingListId, QSqlDatabase & db); + //updates + static void update(qulonglong libraryId, 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); + static void renameLabel(qulonglong id, const QString & name, QSqlDatabase & db); + static void renameList(qulonglong id, const QString & name, QSqlDatabase & db); + static void reasignOrderToSublists(QList ids, QSqlDatabase & db); + static void reasignOrderToComicsInFavorites(QList comicIds, QSqlDatabase & db); + static void reasignOrderToComicsInLabel(qulonglong labelId, QList comicIds, QSqlDatabase & db); + static void reasignOrderToComicsInReadingList(qulonglong readingListId, QList comicIds, QSqlDatabase & db); + + 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 Folder loadFolder(const QString & folderName, qulonglong parentId, 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_container_info.cpp b/YACReaderLibrary/empty_container_info.cpp new file mode 100644 index 00000000..2c5b74d0 --- /dev/null +++ b/YACReaderLibrary/empty_container_info.cpp @@ -0,0 +1,47 @@ +#include "empty_container_info.h" + +EmptyContainerInfo::EmptyContainerInfo(QWidget *parent) : + QWidget(parent), iconLabel(new QLabel()), titleLabel(new QLabel()) +{ +#ifdef Q_OS_MAC + backgroundColor = "#FFFFFF"; + titleLabel->setStyleSheet("QLabel {color:#888888; font-size:24px;font-family:Arial;font-weight:bold;}"); +#else + backgroundColor = "#2A2A2A"; + titleLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); +#endif + + iconLabel->setAlignment(Qt::AlignCenter); + titleLabel->setAlignment(Qt::AlignCenter); +} + +void EmptyContainerInfo::setPixmap(const QPixmap &pixmap) +{ + iconLabel->setPixmap(pixmap); +} + +void EmptyContainerInfo::setText(const QString &text) +{ + titleLabel->setText(text); +} + +QVBoxLayout * EmptyContainerInfo::setUpDefaultLayout(bool addStretch) +{ + QVBoxLayout * layout = new QVBoxLayout; + + layout->addSpacing(100); + layout->addWidget(iconLabel); + layout->addSpacing(30); + layout->addWidget(titleLabel); + if(addStretch) + layout->addStretch(); + + setLayout(layout); + return layout; +} + +void EmptyContainerInfo::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor(backgroundColor)); +} diff --git a/YACReaderLibrary/empty_container_info.h b/YACReaderLibrary/empty_container_info.h new file mode 100644 index 00000000..8fd81747 --- /dev/null +++ b/YACReaderLibrary/empty_container_info.h @@ -0,0 +1,26 @@ +#ifndef EMPTY_CONTAINER_INFO_H +#define EMPTY_CONTAINER_INFO_H + +#include + +class EmptyContainerInfo : public QWidget +{ + Q_OBJECT +public: + explicit EmptyContainerInfo(QWidget *parent = 0); + void setPixmap(const QPixmap & pixmap); + void setText(const QString & text); + QVBoxLayout *setUpDefaultLayout(bool addStretch); +signals: + +public slots: + +protected: + void paintEvent(QPaintEvent *); + + QLabel * iconLabel; + QLabel * titleLabel; + QString backgroundColor; +}; + +#endif // EMPTY_CONTAINER_INFO_H diff --git a/YACReaderLibrary/empty_folder_widget.cpp b/YACReaderLibrary/empty_folder_widget.cpp new file mode 100644 index 00000000..e9760ab6 --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.cpp @@ -0,0 +1,154 @@ +#include "empty_folder_widget.h" + +#include +#include +#include +#include +#include + +#include "comic.h" +#include "comic_files_manager.h" +#include "QsLog.h" + +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) : + EmptyContainerInfo(parent),subfoldersModel(new QStringListModel()) +{ + QVBoxLayout * layout = setUpDefaultLayout(false); + + iconLabel->setPixmap(QPixmap(":/images/empty_folder.png")); + titleLabel->setText(tr("Subfolders in this folder")); + + foldersView = new QListView(); + foldersView->setMinimumWidth(282); + //foldersView->setWrapping(true); + foldersView->setAttribute(Qt::WA_MacShowFocusRect,false); +#ifdef Q_OS_MAC + foldersView->setStyleSheet("QListView {background-color:transparent; border: none; color:#959595; outline:0; font-size: 18px; show-decoration-selected: 0; margin:0}" + "QListView::item:selected {background-color: #EFEFEF; color:#CCCCCC;}" + "QListView::item:hover {background-color:#F4F4F8; color:#757575; }" + + + "QScrollBar:vertical { border-radius:3px; background: #FFFFFF; width: 14px; margin: 0 10px 0 0; }" + "QScrollBar::handle:vertical { border: 1px solid #999999; background: #999999; width: 14px; min-height: 20px; border-radius: 2px; }" + "QScrollBar::add-line:vertical { border: none; background: #999999; height: 0px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #999999; 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;}" + ); +#else + 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;}" + ); + +#endif + foldersView->setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + testListView(foldersView); + + layout->addSpacing(12); + layout->addWidget(foldersView,1,Qt::AlignHCenter); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet(QString("QWidget {background:%1}").arg(backgroundColor)); + + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + + setAcceptDrops(true); + + 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); + + if(foldersNames.isEmpty()) + { + titleLabel->setText(tr("Empty folder") + QString("

%1

").arg(tr("Drag and drop folders and comics here"))); + } + else + { + titleLabel->setText(tr("Subfolders in this folder")); + } +} + +void EmptyFolderWidget::onItemClicked(const QModelIndex &mi) +{ + emit subfolderSelected(parent,mi.row()); +} + +//TODO remove repeated code in drag & drop support.... +void EmptyFolderWidget::dragEnterEvent(QDragEnterEvent *event) +{ + QList urlList; + + if (event->mimeData()->hasUrls() && event->dropAction() == Qt::CopyAction) + { + urlList = event->mimeData()->urls(); + QString currentPath; + foreach (QUrl url, urlList) + { + //comics or folders are accepted, folders' content is validate in dropEvent (avoid any lag before droping) + currentPath = url.toLocalFile(); + if(Comic::fileIsComic(currentPath) || QFileInfo(currentPath).isDir()) + { + event->acceptProposedAction(); + return; + } + } + } +} + +void EmptyFolderWidget::dropEvent(QDropEvent *event) +{ + QLOG_DEBUG() << "drop in emptyfolder" << event->dropAction(); + + bool validAction = event->dropAction() == Qt::CopyAction; // || event->dropAction() & Qt::MoveAction; TODO move + + if(validAction) + { + + QList > droppedFiles = ComicFilesManager::getDroppedFiles(event->mimeData()->urls()); + + if(event->dropAction() == Qt::CopyAction) + { + QLOG_DEBUG() << "copy in emptyfolder:" << droppedFiles; + emit copyComicsToCurrentFolder(droppedFiles); + } + else if(event->dropAction() & Qt::MoveAction) + { + QLOG_DEBUG() << "move in emptyfolder:" << droppedFiles; + emit moveComicsToCurrentFolder(droppedFiles); + } + + event->acceptProposedAction(); + } +} diff --git a/YACReaderLibrary/empty_folder_widget.h b/YACReaderLibrary/empty_folder_widget.h new file mode 100644 index 00000000..98cb4d01 --- /dev/null +++ b/YACReaderLibrary/empty_folder_widget.h @@ -0,0 +1,36 @@ +#ifndef EMPTY_FOLDER_WIDGET_H +#define EMPTY_FOLDER_WIDGET_H + +#include "empty_container_info.h" +#include + + + +class EmptyFolderWidget : public EmptyContainerInfo +{ + Q_OBJECT +public: + explicit EmptyFolderWidget(QWidget *parent = 0); + void setSubfolders(const QModelIndex & mi, const QStringList & foldersNames); +signals: + void subfolderSelected(QModelIndex, int); + + //Drops + void copyComicsToCurrentFolder(QList >); + void moveComicsToCurrentFolder(QList >); + +public slots: + void onItemClicked(const QModelIndex & mi); + +protected: + QListView * foldersView; + QModelIndex parent; + QStringListModel * subfoldersModel; + QString backgroundColor; + + //Drop to import + void dragEnterEvent(QDragEnterEvent *event); + void dropEvent(QDropEvent *event); +}; + +#endif // EMPTY_FOLDER_WIDGET_H diff --git a/YACReaderLibrary/empty_label_widget.cpp b/YACReaderLibrary/empty_label_widget.cpp new file mode 100644 index 00000000..eac010d7 --- /dev/null +++ b/YACReaderLibrary/empty_label_widget.cpp @@ -0,0 +1,21 @@ +#include "empty_label_widget.h" + +EmptyLabelWidget::EmptyLabelWidget(QWidget *parent) : + EmptyContainerInfo(parent) +{ + setUpDefaultLayout(true); + + iconLabel->setPixmap(QPixmap(":/images/empty_label.png")); + + //titleLabel->setText(tr("This label doesn't contain comics yet") + QString("

%1

").arg(tr("Drag and drop folders and comics here"))); + titleLabel->setText(tr("This label doesn't contain comics yet")); +} + +void EmptyLabelWidget::setColor(YACReader::LabelColors color) +{ + QPixmap p(":/images/empty_label.png"); + QImage img = p.toImage().convertToFormat(QImage::Format_ARGB32); + QColor destColor(YACReader::labelColorToRGBString(color)); + YACReader::colorize(img,destColor); + iconLabel->setPixmap(QPixmap::fromImage(img)); +} diff --git a/YACReaderLibrary/empty_label_widget.h b/YACReaderLibrary/empty_label_widget.h new file mode 100644 index 00000000..0e877da6 --- /dev/null +++ b/YACReaderLibrary/empty_label_widget.h @@ -0,0 +1,22 @@ +#ifndef EMPTY_LABEL_WIDGET_H +#define EMPTY_LABEL_WIDGET_H + +#include +#include "empty_container_info.h" +#include "yacreader_global.h" + +class EmptyLabelWidget : public EmptyContainerInfo +{ + Q_OBJECT +public: + explicit EmptyLabelWidget(QWidget *parent = 0); + void setColor(YACReader::LabelColors color); + +signals: + +public slots: + +protected: +}; + +#endif // EMPTY_LABEL_WIDGET_H diff --git a/YACReaderLibrary/empty_reading_list_widget.cpp b/YACReaderLibrary/empty_reading_list_widget.cpp new file mode 100644 index 00000000..f325dc37 --- /dev/null +++ b/YACReaderLibrary/empty_reading_list_widget.cpp @@ -0,0 +1,9 @@ +#include "empty_reading_list_widget.h" + +EmptyReadingListWidget::EmptyReadingListWidget(QWidget *parent) + :EmptyContainerInfo(parent) +{ + setUpDefaultLayout(true); + setPixmap(QPixmap(":/images/empty_reading_list")); + setText(tr("This reading list doesn't cotain comics yet")); +} diff --git a/YACReaderLibrary/empty_reading_list_widget.h b/YACReaderLibrary/empty_reading_list_widget.h new file mode 100644 index 00000000..566b8cfb --- /dev/null +++ b/YACReaderLibrary/empty_reading_list_widget.h @@ -0,0 +1,13 @@ +#ifndef EMPTY_READING_LIST_WIDGET_H +#define EMPTY_READING_LIST_WIDGET_H + +#include +#include "empty_container_info.h" + +class EmptyReadingListWidget : public EmptyContainerInfo +{ +public: + EmptyReadingListWidget(QWidget * parent = 0); +}; + +#endif // EMPTY_READING_LIST_WIDGET_H diff --git a/YACReaderLibrary/empty_special_list.cpp b/YACReaderLibrary/empty_special_list.cpp new file mode 100644 index 00000000..c4ec384d --- /dev/null +++ b/YACReaderLibrary/empty_special_list.cpp @@ -0,0 +1,7 @@ +#include "empty_special_list.h" + +EmptySpecialListWidget::EmptySpecialListWidget(QWidget *parent) + :EmptyContainerInfo(parent) +{ + setUpDefaultLayout(true); +} diff --git a/YACReaderLibrary/empty_special_list.h b/YACReaderLibrary/empty_special_list.h new file mode 100644 index 00000000..f9d4b117 --- /dev/null +++ b/YACReaderLibrary/empty_special_list.h @@ -0,0 +1,13 @@ +#ifndef EMPTY_SPECIAL_LIST_H +#define EMPTY_SPECIAL_LIST_H + +#include +#include "empty_container_info.h" + +class EmptySpecialListWidget : public EmptyContainerInfo +{ +public: + EmptySpecialListWidget(QWidget * parent = 0); +}; + +#endif // EMPTY_SPECIAL_LIST_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..f4f5f637 --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.cpp @@ -0,0 +1,340 @@ +#include "grid_comics_view.h" + +#include +#include + +#include "yacreader_global.h" +#include "comic.h" +#include "comic_files_manager.h" +#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); + this->toolbar = toolBar; +} + +void GridComicsView::setModel(ComicModel *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("contextMenuHelper",this); + ctxt->setContextProperty("comicsSelectionHelper", this); + ctxt->setContextProperty("comicRatingHelper", this); + ctxt->setContextProperty("dummyValue", true); + ctxt->setContextProperty("dragManager", this); + ctxt->setContextProperty("dropManager", this); + + if(model->rowCount()>0) + setCurrentIndex(model->index(0,0)); + } + +#ifdef Q_OS_MAC + ctxt->setContextProperty("backgroundColor", "#F5F5F5"); + ctxt->setContextProperty("cellColor", "#FFFFFF"); + ctxt->setContextProperty("selectedColor", "#FFFFFF"); + ctxt->setContextProperty("selectedBorderColor", "#007AFF"); + ctxt->setContextProperty("borderColor", "#DBDBDB"); + ctxt->setContextProperty("titleColor", "#121212"); + ctxt->setContextProperty("textColor", "#636363"); + //fonts settings + ctxt->setContextProperty("fontSize", 11); + ctxt->setContextProperty("fontFamily", "none"); + ctxt->setContextProperty("fontSpacing", 0.5); + +#else + ctxt->setContextProperty("backgroundColor", "#2A2A2A"); + ctxt->setContextProperty("cellColor", "#212121"); + ctxt->setContextProperty("selectedColor", "#121212"); + ctxt->setContextProperty("selectedBorderColor", "#121212"); + ctxt->setContextProperty("borderColor", "#121212"); + ctxt->setContextProperty("titleColor", "#E6E6E6"); + ctxt->setContextProperty("textColor", "#E6E6E6"); + ctxt->setContextProperty("dropShadow",false); + //fonts settings + ctxt->setContextProperty("fontSize", "none"); + ctxt->setContextProperty("fontFamily", "none"); + ctxt->setContextProperty("fontSpacing", 0.5); +#endif + + +} + +void GridComicsView::setCurrentIndex(const QModelIndex &index) +{ + QLOG_INFO() << "setCurrentIndex"; + _selectionModel->clear(); + _selectionModel->select(index, QItemSelectionModel::Select | QItemSelectionModel::Rows); + view->rootContext()->setContextProperty("dummyValue", true); +} + +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"; + toolbar->hide(); +} + +void GridComicsView::toNormal() +{ + QLOG_INFO() << "toNormal"; + toolbar->show(); +} + +void GridComicsView::updateConfig(QSettings *settings) +{ + QLOG_INFO() << "updateConfig"; +} + +void GridComicsView::enableFilterMode(bool enabled) +{ + +} + +void GridComicsView::selectAll() +{ + QLOG_INFO() << "selectAll"; + QModelIndex top = model->index(0, 0); + QModelIndex bottom = model->index(model->rowCount()-1, 0); + QItemSelection selection(top, bottom); + _selectionModel->select(selection, QItemSelectionModel::Select | QItemSelectionModel::Rows); + view->rootContext()->setContextProperty("dummyValue", true); +} + +void GridComicsView::rate(int index, int rating) +{ + QLOG_INFO() << "Comic "<< index << "rated" << rating; + model->updateRating(rating,model->index(index,0)); +} + +void GridComicsView::requestedContextMenu(const QPoint &point) +{ + emit customContextMenuViewRequested(point); +} + +QSize GridComicsView::sizeHint() +{ + QLOG_INFO() << "sizeHint"; + return QSize(1280,768); +} + +QByteArray GridComicsView::getMimeDataFromSelection() +{ + QByteArray data; + + QMimeData * mimeData = model->mimeData(_selectionModel->selectedIndexes()); + data = mimeData->data(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat); + + delete mimeData; + + return data; +} + +void GridComicsView::startDrag() +{ + QLOG_DEBUG() << "performDrag"; + QDrag *drag = new QDrag(this); + drag->setMimeData(model->mimeData(_selectionModel->selectedRows())); + drag->setPixmap(QPixmap(":/images/openInYACReader.png")); //TODO add better image + + /*Qt::DropAction dropAction =*/ drag->exec(Qt::CopyAction | Qt::MoveAction, Qt::CopyAction); +} + +bool GridComicsView::canDropUrls(const QList &urls, Qt::DropAction action) +{ + if(action == Qt::CopyAction) + { + QString currentPath; + foreach (QUrl url, urls) + { + //comics or folders are accepted, folders' content is validate in dropEvent (avoid any lag before droping) + currentPath = url.toLocalFile(); + if(Comic::fileIsComic(currentPath) || QFileInfo(currentPath).isDir()) + return true; + } + } + return false; +} + +bool GridComicsView::canDropFormats(const QString &formats) +{ + return (formats.contains(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat) && model->canBeResorted()); +} + +void GridComicsView::droppedFiles(const QList &urls, Qt::DropAction action) +{ + bool validAction = action == Qt::CopyAction; //TODO add move + + if(validAction) + { + QList > droppedFiles = ComicFilesManager::getDroppedFiles(urls); + emit copyComicsToCurrentFolder(droppedFiles); + } +} + +void GridComicsView::droppedComicsForResortingAt(const QString &data, int index) +{ + model->dropMimeData(model->mimeData(_selectionModel->selectedRows()), Qt::MoveAction, index, 0, QModelIndex()); +} + +//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); + view->rootContext()->setContextProperty("dummyValue", true); + } +} + +void GridComicsView::setCurrentIndex(int index) +{ + setCurrentIndex(model->index(index,0)); +} + +void GridComicsView::deselectIndex(int index) +{ + if(_selectionModel != NULL && model!=NULL) + { + _selectionModel->select(model->index(index,0),QItemSelectionModel::Deselect | QItemSelectionModel::Rows); + view->rootContext()->setContextProperty("dummyValue", true); + } +} + +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)); +} + +int GridComicsView::numItemsSelected() +{ + if(_selectionModel != NULL) + { + return _selectionModel->selectedRows().length(); + } + + return 0; +} + +int GridComicsView::lastSelectedIndex() +{ + if(_selectionModel != NULL) + { + QLOG_INFO() << "last selected index " << _selectionModel->selectedRows().last().row(); + return _selectionModel->selectedRows().last().row(); + } + + return -1; +} + +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..77d3c343 --- /dev/null +++ b/YACReaderLibrary/grid_comics_view.h @@ -0,0 +1,79 @@ +#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(ComicModel *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 enableFilterMode(bool enabled); + QSize sizeHint(); + QByteArray getMimeDataFromSelection(); + + +signals: + void comicRated(int,QModelIndex); + void doubleClicked(QModelIndex); + +public slots: + //selection helper + void selectIndex(int index); + void setCurrentIndex(int index); + void deselectIndex(int index); + bool isSelectedIndex(int index); + void clear(); + //double clicked item + void selectedItem(int index); + int numItemsSelected(); + int lastSelectedIndex(); + + //ComicsView + void setShowMarks(bool show); + void selectAll(); + + //rating + void rate(int index, int rating); + + //dragManager + void startDrag(); + //dropManager + bool canDropUrls(const QList & urls, Qt::DropAction action); + bool canDropFormats(const QString &formats); + void droppedFiles(const QList & urls, Qt::DropAction action); + void droppedComicsForResortingAt(const QString & data, int index); + + +protected slots: + void requestedContextMenu(const QPoint & point); + +private: + QToolBar * toolbar; + 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..caffa5a1 --- /dev/null +++ b/YACReaderLibrary/images.qrc @@ -0,0 +1,99 @@ + + + ../images/sidebar/folder.png + ../images/sidebar/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/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/sidebar/collapsed_branch_osx.png + ../images/sidebar/expanded_branch_osx.png + ../images/sidebar/libraryIconSelected.png + ../images/sidebar/libraryOptions.png + ../images/sidebar/branch-open.png + ../images/sidebar/branch-closed.png + ../images/sidebar/expanded_branch_selected.png + ../images/sidebar/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/f_overlayed.png + ../images/f_overlayed_retina.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/searching_icon.png + ../images/empty_label.png + ../images/empty_current_readings.png + ../images/empty_favorites.png + + diff --git a/YACReaderLibrary/images_osx.qrc b/YACReaderLibrary/images_osx.qrc new file mode 100644 index 00000000..5f9106a3 --- /dev/null +++ b/YACReaderLibrary/images_osx.qrc @@ -0,0 +1,80 @@ + + + ../images/folder_finished_macosx.png + ../images/main_toolbar/back_osx.png + ../images/main_toolbar/back_osx@2x.png + ../images/main_toolbar/forward_osx.png + ../images/main_toolbar/forward_osx@2x.png + ../images/main_toolbar/settings_osx.png + ../images/main_toolbar/settings_osx@2x.png + ../images/main_toolbar/server_osx.png + ../images/main_toolbar/server_osx@2x.png + ../images/main_toolbar/help_osx.png + ../images/main_toolbar/help_osx@2x.png + ../images/main_toolbar/flow_osx.png + ../images/main_toolbar/flow_osx@2x.png + ../images/main_toolbar/grid_osx.png + ../images/main_toolbar/grid_osx@2x.png + ../images/flow_to_grid_osx.gif + ../images/grid_to_flow_osx.gif + ../images/empty_folder_osx.png + ../images/empty_search_osx.png + ../images/iconSearch.png + ../images/clearSearch.png + + ../images/lists/default_0_osx.png + ../images/lists/default_1_osx.png + ../images/lists/label_blue_osx.png + ../images/lists/label_cyan_osx.png + ../images/lists/label_dark_osx.png + ../images/lists/label_green_osx.png + ../images/lists/label_light_osx.png + ../images/lists/label_orange_osx.png + ../images/lists/label_pink_osx.png + ../images/lists/label_purple_osx.png + ../images/lists/label_red_osx.png + ../images/lists/label_violet_osx.png + ../images/lists/label_white_osx.png + ../images/lists/label_yellow_osx.png + ../images/lists/list_osx.png + ../images/empty_reading_list_osx.png + + ../images/lists/default_0_osx@2x.png + ../images/lists/default_1_osx@2x.png + ../images/lists/label_blue_osx@2x.png + ../images/lists/label_cyan_osx@2x.png + ../images/lists/label_dark_osx@2x.png + ../images/lists/label_green_osx@2x.png + ../images/lists/label_light_osx@2x.png + ../images/lists/label_orange_osx@2x.png + ../images/lists/label_pink_osx@2x.png + ../images/lists/label_purple_osx@2x.png + ../images/lists/label_red_osx@2x.png + ../images/lists/label_violet_osx@2x.png + ../images/lists/label_white_osx@2x.png + ../images/lists/label_yellow_osx@2x.png + ../images/lists/list_osx@2x.png + + ../images/sidebar/libraryIcon_osx.png + ../images/sidebar/setRoot_osx.png + ../images/sidebar/expand_osx.png + ../images/sidebar/colapse_osx.png + ../images/sidebar/newLibraryIcon_osx.png + ../images/sidebar/openLibraryIcon_osx.png + ../images/sidebar/addNew_sidebar_osx.png + ../images/sidebar/delete_sidebar_osx.png + ../images/sidebar/addLabelIcon_osx.png + ../images/sidebar/renameListIcon_osx.png + + + ../images/sidebar/setRoot_osx@2x.png + ../images/sidebar/expand_osx@2x.png + ../images/sidebar/colapse_osx@2x.png + ../images/sidebar/newLibraryIcon_osx@2x.png + ../images/sidebar/openLibraryIcon_osx@2x.png + ../images/sidebar/addNew_sidebar_osx@2x.png + ../images/sidebar/delete_sidebar_osx@2x.png + ../images/sidebar/addLabelIcon_osx@2x.png + ../images/sidebar/renameListIcon_osx@2x.png + + diff --git a/YACReaderLibrary/images_win.qrc b/YACReaderLibrary/images_win.qrc new file mode 100644 index 00000000..0e042c76 --- /dev/null +++ b/YACReaderLibrary/images_win.qrc @@ -0,0 +1,47 @@ + + + ../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/sidebar/libraryIcon.png + ../images/sidebar/setRoot.png + ../images/sidebar/expand.png + ../images/sidebar/colapse.png + ../images/sidebar/newLibraryIcon.png + ../images/sidebar/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 + ../images/empty_search.png + ../images/sidebar/addNew_sidebar.png + ../images/sidebar/delete_sidebar.png + ../images/iconSearchNew.png + ../images/clearSearchNew.png + ../images/sidebar/addLabelIcon.png + ../images/sidebar/renameListIcon.png + + ../images/lists/default_0.png + ../images/lists/default_1.png + ../images/lists/label_blue.png + ../images/lists/label_cyan.png + ../images/lists/label_dark.png + ../images/lists/label_green.png + ../images/lists/label_light.png + ../images/lists/label_orange.png + ../images/lists/label_pink.png + ../images/lists/label_purple.png + ../images/lists/label_red.png + ../images/lists/label_violet.png + ../images/lists/label_white.png + ../images/lists/label_yellow.png + ../images/lists/list.png + ../images/empty_reading_list.png + + \ No newline at end of file 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..7aadbf0c --- /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::open(const YACReaderLibraries &libs) +{ + libraries = libs; + QDialog::open(); +} + +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..09febeae --- /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 open(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..d9a8d74a --- /dev/null +++ b/YACReaderLibrary/import_widget.cpp @@ -0,0 +1,386 @@ +#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); + + glow->setGraphicsEffect(effect); + + 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); + coversViewContainer->setMaximumHeight(316); + coversViewContainer->setSizePolicy(QSizePolicy::Ignored,QSizePolicy::Maximum); + + 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); +} 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..aa3108fa --- /dev/null +++ b/YACReaderLibrary/library_creator.cpp @@ -0,0 +1,694 @@ +#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), partialUpdate(false) +{ + _nameFilter << Comic::comicExtensions; +} + +void LibraryCreator::createLibrary(const QString &source, const QString &target) +{ + creation = true; + processLibrary(source, target); +} + +void LibraryCreator::updateLibrary(const QString &source, const QString &target) +{ + partialUpdate = false; + processLibrary(source, target); +} + +void LibraryCreator::updateFolder(const QString &source, const QString &target, const QString &sourceFolder, const QModelIndex & dest) +{ + partialUpdate = true; + folderDestinationModelIndex = dest; + + _currentPathFolders.clear(); + _currentPathFolders.append(Folder(1,1,"root","/")); + + QString relativeFolderPath = sourceFolder; + relativeFolderPath = relativeFolderPath.remove(QDir::cleanPath(source)); + + if(relativeFolderPath.startsWith("/")) + relativeFolderPath = relativeFolderPath.remove(0,1);//remove firts '/' + + QStringList folders; + + if(!relativeFolderPath.isEmpty()) //updating root + folders = relativeFolderPath.split('/'); + + QLOG_DEBUG() << "folders found in relative path : " << folders << "-" << relativeFolderPath; + + QSqlDatabase db = DataBaseManagement::loadDatabase(target); + + foreach (QString folderName, folders) { + if(folderName.isEmpty()) + break; + qulonglong parentId = _currentPathFolders.last().id; + _currentPathFolders.append(DBHelper::loadFolder(folderName, parentId, db)); + QLOG_DEBUG() << "Folder appended : " << _currentPathFolders.last().id << " " << _currentPathFolders.last().name << " with parent" << _currentPathFolders.last().parentId; + } + + QSqlDatabase::removeDatabase(_database.connectionName()); + + QLOG_DEBUG() << "Relative path : " << relativeFolderPath; + + _sourceFolder = sourceFolder; + + 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 + QDir d(target); + d.removeRecursively(); + _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 folder" << _sourceFolder << "in library ( " << _source << "," << _target << ")"; + if(!partialUpdate) + { + _currentPathFolders.clear(); + _currentPathFolders.append(Folder(1,1,"root","/")); + QLOG_DEBUG() << "update whole library"; + } + + _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(); + if(partialUpdate) + update(QDir(_sourceFolder)); + else + 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(!partialUpdate) + { + 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...) + if(partialUpdate) + { + emit updatedCurrentFolder(folderDestinationModelIndex); + emit finished(); + } + else + 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 available 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..83c06164 --- /dev/null +++ b/YACReaderLibrary/library_creator.h @@ -0,0 +1,94 @@ +#ifndef __LIBRARY_CREATOR_H +#define __LIBRARY_CREATOR_H + +#include +#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 updateFolder(const QString & source, const QString & target, const QString & folder, const QModelIndex &dest); + 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; + QString _sourceFolder; //used for partial updates + 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; + bool partialUpdate; + QModelIndex folderDestinationModelIndex; + + signals: + void finished(); + void coverExtracted(QString); + void folderUpdated(QString); + void comicAdded(QString,QString); + void updated(); + void created(); + void failedCreatingDB(QString); + void failedOpeningDB(QString); + void updatedCurrentFolder(QModelIndex); + }; + + 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..57853e5b --- /dev/null +++ b/YACReaderLibrary/library_window.cpp @@ -0,0 +1,2733 @@ +#include "library_window.h" +#include "custom_widgets.h" +#include "folder_item.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 "comic_model.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_folders_view.h" + +#include "comic_vine_dialog.h" +#include "api_key_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 "empty_label_widget.h" +#include "empty_special_list.h" +#include "empty_reading_list_widget.h" + +#include "edit_shortcuts_dialog.h" +#include "shortcuts_manager.h" + +#include "no_search_results_widget.h" + +#include "comic_files_manager.h" + +#include "reading_list_model.h" +#include "yacreader_reading_lists_view.h" +#include "add_label_dialog.h" + +#include "yacreader_history_controller.h" +#include "db_helper.h" + +#include "reading_list_item.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),status(LibraryWindow::Normal) +{ + 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"); + + historyController = new YACReaderHistoryController(this); + + createActions(); + doModels(); + + doLayout(); + createToolBars(); + doDialogs(); + createMenus(); + + navigationController = new YACReaderNavigationController(this); + + 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 = new YACReaderMacOSXToolbar(this); +#else + libraryToolBar = new YACReaderMainToolBar(this); +#endif + + + //FLOW----------------------------------------------------------------------- + //--------------------------------------------------------------------------- + if(QGLFormat::hasOpenGL() && !settings->contains(USE_OPEN_GL)) + { + settings->setValue(USE_OPEN_GL,2); + } + + //FOLDERS FILTER------------------------------------------------------------- + //--------------------------------------------------------------------------- +#ifndef Q_OS_MAC + //in MacOSX the searchEdit is created using the toolbar wrapper + searchEdit = new YACReaderSearchLineEdit(); +#endif + + //SIDEBAR-------------------------------------------------------------------- + //--------------------------------------------------------------------------- + sideBar = new YACReaderSideBar; + + foldersView = sideBar->foldersView; + listsView = sideBar->readingListsView; + selectedLibrary = sideBar->selectedLibrary; + + YACReaderTitledToolBar * librariesTitle = sideBar->librariesTitle; + YACReaderTitledToolBar * foldersTitle = sideBar->foldersTitle; + YACReaderTitledToolBar * readingListsTitle = sideBar->readingListsTitle; + + librariesTitle->addAction(createLibraryAction); + librariesTitle->addAction(openLibraryAction); + librariesTitle->addSpacing(3); + + foldersTitle->addAction(addFolderAction); + foldersTitle->addAction(deleteFolderAction); + foldersTitle->addSepartor(); + foldersTitle->addAction(setRootIndexAction); + foldersTitle->addAction(expandAllNodesAction); + foldersTitle->addAction(colapseAllNodesAction); + + readingListsTitle->addAction(addReadingListAction); + //readingListsTitle->addSepartor(); + readingListsTitle->addAction(addLabelAction); + //readingListsTitle->addSepartor(); + readingListsTitle->addAction(renameListAction); + readingListsTitle->addAction(deleteReadingListAction); + readingListsTitle->addSpacing(3); + + //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(emptyLabelWidget = new EmptyLabelWidget()); + comicsViewStack->addWidget(emptySpecialList = new EmptySpecialListWidget()); + comicsViewStack->addWidget(emptyReadingList = new EmptyReadingListWidget()); + comicsViewStack->addWidget(noSearchResultsWidget = new NoSearchResultsWidget()); + + 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 + << saveCoversToAction + << setAsReadAction + << setAsNonReadAction + << openContainingFolderComicAction + << resetComicRatingAction + << selectAllComicsAction + << editSelectedComicsAction + << asignOrderAction + << deleteComicsAction + << getInfoAction); + + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("Folders",QIcon(":/images/shortcuts_group_folders.png"), + tmpList = QList() + << addFolderAction + << deleteFolderAction + << setRootIndexAction + << expandAllNodesAction + << colapseAllNodesAction + << openContainingFolderAction + << setFolderAsNotCompletedAction + << setFolderAsCompletedAction + << setFolderAsReadAction + << setFolderAsUnreadAction + << updateCurrentFolderAction); + allActions << tmpList; + + editShortcutsDialog->addActionsGroup("Lists",QIcon(":/images/shortcuts_group_folders.png"), //TODO change icon + tmpList = QList() + << addReadingListAction + << deleteReadingListAction + << addLabelAction + << renameListAction); + 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 + #ifndef Q_OS_MAC + << toggleFullScreenAction + #endif + << toggleComicsViewAction + << hideComicViewAction); + + allActions << tmpList; + + ShortcutsManager::getShortcutsManager().registerActions(allActions); +} + +void LibraryWindow::doModels() +{ + //folders + foldersModel = new FolderModel(); + foldersModelProxy = new FolderModelProxy(); + //foldersModelProxy->setSourceModel(foldersModel); + //comics + comicsModel = new ComicModel(); + //lists + listsModel = new ReadingListModel(); + listsModelProxy = new ReadingListModelProxy(); + + //setSearchFilter(YACReader::NoModifiers, ""); //clear search filter +} + +void LibraryWindow::disconnectComicsViewConnections(ComicsView * widget) +{ + disconnect(widget, SIGNAL(comicRated(int,QModelIndex)), comicsModel, 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())); + disconnect(comicsView, SIGNAL(copyComicsToCurrentFolder(QList >)), this, SLOT(copyAndImportComicsToCurrentFolder(QList >))); + disconnect(comicsView, SIGNAL(moveComicsToCurrentFolder(QList >)), this, SLOT(moveAndImportComicsToCurrentFolder(QList >))); + disconnect(comicsView,SIGNAL(customContextMenuViewRequested(QPoint)),this,SLOT(showComicsViewContextMenu(QPoint))); + disconnect(comicsView,SIGNAL(customContextMenuItemRequested(QPoint)),this,SLOT(showComicsItemContextMenu(QPoint))); +} + +void LibraryWindow::doComicsViewConnections() +{ + connect(comicsView, SIGNAL(comicRated(int,QModelIndex)), comicsModel, 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())); + + connect(comicsView,SIGNAL(customContextMenuViewRequested(QPoint)),this,SLOT(showComicsViewContextMenu(QPoint))); + connect(comicsView,SIGNAL(customContextMenuItemRequested(QPoint)),this,SLOT(showComicsItemContextMenu(QPoint))); + //Drops + connect(comicsView, SIGNAL(copyComicsToCurrentFolder(QList >)), this, SLOT(copyAndImportComicsToCurrentFolder(QList >))); + connect(comicsView, SIGNAL(moveComicsToCurrentFolder(QList >)), this, SLOT(moveAndImportComicsToCurrentFolder(QList >))); +} + +void LibraryWindow::createActions() +{ + backAction = new QAction(this); + QIcon icoBackButton; + icoBackButton.addFile(":/images/main_toolbar/back.png",QSize(), 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.addFile(":/images/main_toolbar/forward.png", QSize(), 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/sidebar/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/sidebar/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")); + + saveCoversToAction = new QAction(tr("Save selected covers to..."),this); + saveCoversToAction->setToolTip(tr("Save covers of the selected comics as JPG files")); + saveCoversToAction->setData(SAVE_COVERS_TO_ACTION_YL); + saveCoversToAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(SAVE_COVERS_TO_ACTION_YL)); + + 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); +#ifndef Q_OS_MAC + 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); +#endif + 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.addFile(":/images/main_toolbar/help.png",QSize(), QIcon::Normal); + helpAboutAction->setIcon(icoHelpButton); + + addFolderAction = new QAction(tr("Add new folder"), this); + addFolderAction->setData(ADD_FOLDER_ACTION_YL); + addFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADD_FOLDER_ACTION_YL)); + addFolderAction->setToolTip(tr("Add new folder to the current library")); + addFolderAction->setIcon(QIcon(":/images/sidebar/addNew_sidebar.png")); + + deleteFolderAction = new QAction(tr("Delete folder"), this); + deleteFolderAction->setData(REMOVE_FOLDER_ACTION_YL); + deleteFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(REMOVE_FOLDER_ACTION_YL)); + deleteFolderAction->setToolTip(tr("Delete current folder from disk")); + deleteFolderAction->setIcon(QIcon(":/images/sidebar/delete_sidebar.png")); + + 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/sidebar/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/sidebar/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/sidebar/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.addFile(":/images/main_toolbar/settings.png", QSize(), 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.addFile(":/images/main_toolbar/server.png", QSize(), 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.addFile(":/images/main_toolbar/grid.png", QSize(), QIcon::Normal); + else + icoViewsButton.addFile(":/images/main_toolbar/flow.png", QSize(), 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->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->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->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->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); + + updateFolderAction = new QAction(tr("Update folder"), this); + updateFolderAction->setIcon(QIcon(":/images/updateLibraryIcon.png")); + + updateCurrentFolderAction = new QAction(tr("Update current folder"), this); + updateCurrentFolderAction->setData(UPDATE_CURRENT_FOLDER_ACTION_YL); + updateCurrentFolderAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(UPDATE_CURRENT_FOLDER_ACTION_YL)); + updateCurrentFolderAction->setIcon(QIcon(":/images/updateLibraryIcon.png")); + + addReadingListAction = new QAction(tr("Add new reading list"), this); + addReadingListAction->setData(ADD_READING_LIST_ACTION_YL); + addReadingListAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADD_READING_LIST_ACTION_YL)); + addReadingListAction->setToolTip(tr("Add a new reading list to the current library")); + addReadingListAction->setIcon(QIcon(":/images/sidebar/addNew_sidebar.png")); + + deleteReadingListAction = new QAction(tr("Remove reading list"), this); + deleteReadingListAction->setData(REMOVE_READING_LIST_ACTION_YL); + deleteReadingListAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(REMOVE_READING_LIST_ACTION_YL)); + deleteReadingListAction->setToolTip(tr("Remove current reading list from the library")); + deleteReadingListAction->setIcon(QIcon(":/images/sidebar/delete_sidebar.png")); + + addLabelAction = new QAction(tr("Add new label"), this); + addLabelAction->setData(ADD_LABEL_ACTION_YL); + addLabelAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADD_LABEL_ACTION_YL)); + addLabelAction->setToolTip(tr("Add a new label to this library")); + addLabelAction->setIcon(QIcon(":/images/sidebar/addLabelIcon.png")); + + renameListAction = new QAction(tr("Rename selected list"), this); + renameListAction->setData(RENAME_LIST_ACTION_YL); + renameListAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(RENAME_LIST_ACTION_YL)); + renameListAction->setToolTip(tr("Rename any selected labels or lists")); + renameListAction->setIcon(QIcon(":/images/sidebar/renameListIcon.png")); + + //-- + addToMenuAction = new QAction(tr("Add to..."), this); + + addToFavoritesAction = new QAction(tr("Favorites"), this); + addToFavoritesAction->setData(ADD_TO_FAVORITES_ACTION_YL); + addToFavoritesAction->setShortcut(ShortcutsManager::getShortcutsManager().getShortcut(ADD_TO_FAVORITES_ACTION_YL)); + addToFavoritesAction->setToolTip(tr("Add selected comics to favorites list")); + addToFavoritesAction->setIcon(QIcon(":/images/lists/default_1.png")); + + //actions not asigned to any widget + this->addAction(saveCoversToAction); + this->addAction(openContainingFolderAction); + this->addAction(updateCurrentFolderAction); + this->addAction(resetComicRatingAction); + this->addAction(setFolderAsCompletedAction); + this->addAction(setFolderAsNotCompletedAction); + this->addAction(setFolderAsReadAction); + this->addAction(setFolderAsUnreadAction); +#ifndef Q_OS_MAC + this->addAction(toggleFullScreenAction); +#endif + + //disable actions + disableAllActions(); +} +void LibraryWindow::disableComicsActions(bool disabled) +{ + //if there aren't comics, no fullscreen option will be available +#ifndef Q_OS_MAC + toggleFullScreenAction->setDisabled(disabled); +#endif + //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); + + updateCurrentFolderAction->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); + + updateFolderAction->setDisabled(disabled); +} + +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); + + libraryToolBar->addSpace(10); + +#ifdef SERVER_RELEASE + libraryToolBar->addAction(serverConfigAction); +#endif + libraryToolBar->addAction(optionsAction); + libraryToolBar->addAction(helpAboutAction); + + libraryToolBar->addSpace(10); + + libraryToolBar->addAction(toggleComicsViewAction); +#ifndef Q_OS_MAC + libraryToolBar->addAction(toggleFullScreenAction); +#endif + + libraryToolBar->addStretch(); + + //Native toolbar search edit + //libraryToolBar->addWidget(searchEdit); + searchEdit = libraryToolBar->addSearchEdit(); + //connect(libraryToolBar,SIGNAL(searchTextChanged(YACReader::SearchModifiers,QString)),this,SLOT(setSearchFilter(YACReader::SearchModifiers, QString))); + + //libraryToolBar->setMovable(false); + + libraryToolBar->attachToWindow(this->windowHandle()); + + +#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); + libraryToolBar->setSearchWidget(searchEdit); +#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() +{ + foldersView->addAction(addFolderAction); + foldersView->addAction(deleteFolderAction); + YACReader::addSperator(foldersView); + + foldersView->addAction(openContainingFolderAction); + foldersView->addAction(updateFolderAction); + 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->addAction(updateFolderAction); + 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()),historyController,SLOT(backward())); + connect(forwardAction,SIGNAL(triggered()),historyController,SLOT(forward())); + //-- + connect(historyController,SIGNAL(enabledBackward(bool)),backAction,SLOT(setEnabled(bool))); + connect(historyController,SIGNAL(enabledForward(bool)),forwardAction,SLOT(setEnabled(bool))); + //connect(foldersView, SIGNAL(clicked(QModelIndex)), historyController, SLOT(updateHistory(QModelIndex))); + + //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(updatedCurrentFolder()), this, SLOT(showRootWidget())); + connect(libraryCreator,SIGNAL(updatedCurrentFolder(QModelIndex)), this, SLOT(reloadAfterCopyMove(QModelIndex))); + 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) + //TODO connect(foldersView, SIGNAL(pressed(QModelIndex)), this, SLOT(updateFoldersViewConextMenu(QModelIndex))); + //connect(foldersView, SIGNAL(clicked(QModelIndex)), this, SLOT(loadCovers(QModelIndex))); + + //drops in folders view + connect(foldersView, SIGNAL(copyComicsToFolder(QList >,QModelIndex)), this, SLOT(copyAndImportComicsToFolder(QList >,QModelIndex))); + connect(foldersView, SIGNAL(moveComicsToFolder(QList >,QModelIndex)), this, SLOT(moveAndImportComicsToFolder(QList >,QModelIndex))); + connect(foldersView, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(showFoldersContextMenu(QPoint))); + + //actions + connect(createLibraryAction,SIGNAL(triggered()),this,SLOT(createLibrary())); + connect(exportLibraryAction,SIGNAL(triggered()),exportLibraryDialog,SLOT(open())); + 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()),navigationController,SLOT(reselectCurrentSource())); + + //comic vine + connect(comicVineDialog,SIGNAL(accepted()),navigationController,SLOT(reselectCurrentSource())); + + 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(addFolderAction,SIGNAL(triggered()),this,SLOT(addFolderToCurrentIndex())); + connect(deleteFolderAction,SIGNAL(triggered()),this,SLOT(deleteSelectedFolder())); + connect(setRootIndexAction,SIGNAL(triggered()),this,SLOT(setRootIndex())); + connect(expandAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(expandAll())); + connect(colapseAllNodesAction,SIGNAL(triggered()),foldersView,SLOT(collapseAll())); +#ifndef Q_OS_MAC + connect(toggleFullScreenAction,SIGNAL(triggered()),this,SLOT(toggleFullScreen())); +#endif + 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(searchEdit,SIGNAL(filterChanged(YACReader::SearchModifiers, QString)),this,SLOT(setSearchFilter(YACReader::SearchModifiers, 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(comicsModel,SIGNAL(isEmpty()),this,SLOT(showEmptyFolderView())); + //connect(comicsModel,SIGNAL(searchNumResults(int)),this,SLOT(checkSearchNumResults(int))); + //connect(emptyFolderWidget,SIGNAL(subfolderSelected(QModelIndex,int)),this,SLOT(selectSubfolder(QModelIndex,int))); + //Drops + connect(emptyFolderWidget, SIGNAL(copyComicsToCurrentFolder(QList >)), this, SLOT(copyAndImportComicsToCurrentFolder(QList >))); + connect(emptyFolderWidget, SIGNAL(moveComicsToCurrentFolder(QList >)), this, SLOT(moveAndImportComicsToCurrentFolder(QList >))); + + connect(showEditShortcutsAction,SIGNAL(triggered()),editShortcutsDialog,SLOT(show())); + + //update folders (partial updates) + connect(updateCurrentFolderAction,SIGNAL(triggered()), this, SLOT(updateCurrentFolder())); + connect(updateFolderAction,SIGNAL(triggered()), this, SLOT(updateCurrentFolder())); + + //lists + connect(addReadingListAction,SIGNAL(triggered()),this,SLOT(addNewReadingList())); + connect(deleteReadingListAction,SIGNAL(triggered()),this,SLOT(deleteSelectedReadingList())); + connect(addLabelAction,SIGNAL(triggered()),this,SLOT(showAddNewLabelDialog())); + connect(renameListAction,SIGNAL(triggered()),this,SLOT(showRenameCurrentList())); + + connect(listsModel,SIGNAL(addComicsToFavorites(QList)),comicsModel,SLOT(addComicsToFavorites(QList))); + connect(listsModel,SIGNAL(addComicsToLabel(QList,qulonglong)),comicsModel,SLOT(addComicsToLabel(QList,qulonglong))); + connect(listsModel,SIGNAL(addComicsToReadingList(QList,qulonglong)),comicsModel,SLOT(addComicsToReadingList(QList,qulonglong))); + //-- + + connect(addToFavoritesAction,SIGNAL(triggered()),this,SLOT(addSelectedComicsToFavorites())); + + //save covers + connect(saveCoversToAction,SIGNAL(triggered()),this,SLOT(saveSelectedCoversTo())); +} + +void LibraryWindow::loadLibrary(const QString & name) +{ + if(!libraries.isEmpty()) //si hay bibliotecas... + { + historyController->clear(); + + 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); + listsView->setModel(NULL); + disableAllActions();//TODO comprobar que se deben deshabilitar + //será possible 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 + { + foldersModel->setupModelData(path); + foldersModelProxy->setSourceModel(foldersModel); + foldersView->setModel(foldersModelProxy); + foldersView->setCurrentIndex(QModelIndex()); //why is this necesary?? by default it seems that returns an arbitrary index. + + listsModel->setupReadingListsData(path); + listsModelProxy->setSourceModel(listsModel); + listsView->setModel(listsModelProxy); + + if(foldersModel->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); +#ifndef Q_OS_MAC + toggleFullScreenAction->setEnabled(true); +#endif + + importedCovers = true; + } + else //librería normal abierta + { + disableLibrariesActions(false); + importedCovers = false; + } + + setRootIndex(); + + searchEdit->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); + listsView->setModel(NULL); + disableAllActions();//TODO comprobar que se deben deshabilitar + //será possible renombrar y borrar estas bibliotecas + renameLibraryAction->setEnabled(true); + removeLibraryAction->setEnabled(true); + } + } + else + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + listsView->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á possible 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á possible 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"); + d.removeRecursively(); + //d.rmdir(path+"/.yacreaderlibrary"); + createLibraryDialog->setDataAndStart(currentLibrary,path); + //create(path,path+"/.yacreaderlibrary",currentLibrary); + } + //será possible 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::loadCoversFromCurrentModel() +{ + comicsView->setModel(comicsModel); +} + +void LibraryWindow::copyAndImportComicsToCurrentFolder(const QList > &comics) +{ + QLOG_DEBUG() << "-copyAndImportComicsToCurrentFolder-"; + if(comics.size()>0) + { + QString destFolderPath = currentFolderPath(); + + QModelIndex folderDestination = getCurrentFolderIndex(); + + QProgressDialog * progressDialog = newProgressDialog(tr("Copying comics..."),comics.size()); + + ComicFilesManager * comicFilesManager = new ComicFilesManager(); + comicFilesManager->copyComicsTo(comics,destFolderPath,folderDestination); + + processComicFiles(comicFilesManager, progressDialog); + } +} + +void LibraryWindow::moveAndImportComicsToCurrentFolder(const QList > &comics) +{ + QLOG_DEBUG() << "-moveAndImportComicsToCurrentFolder-"; + if(comics.size()>0) + { + QString destFolderPath = currentFolderPath(); + + QModelIndex folderDestination = getCurrentFolderIndex(); + + QProgressDialog * progressDialog = newProgressDialog(tr("Moving comics..."),comics.size()); + + ComicFilesManager * comicFilesManager = new ComicFilesManager(); + comicFilesManager->moveComicsTo(comics,destFolderPath,folderDestination); + + processComicFiles(comicFilesManager, progressDialog); + } +} + +void LibraryWindow::copyAndImportComicsToFolder(const QList > &comics, const QModelIndex &miFolder) +{ + QLOG_DEBUG() << "-copyAndImportComicsToFolder-"; + if(comics.size()>0) + { + QModelIndex folderDestination = foldersModelProxy->mapToSource(miFolder); + + QString destFolderPath = QDir::cleanPath(currentPath()+foldersModel->getFolderPath(folderDestination)); + + QLOG_DEBUG() << "Coping to " << destFolderPath; + + QProgressDialog * progressDialog = newProgressDialog(tr("Copying comics..."),comics.size()); + + ComicFilesManager * comicFilesManager = new ComicFilesManager(); + comicFilesManager->copyComicsTo(comics,destFolderPath,folderDestination); + + processComicFiles(comicFilesManager, progressDialog); + } +} + +void LibraryWindow::moveAndImportComicsToFolder(const QList > &comics, const QModelIndex &miFolder) +{ + QLOG_DEBUG() << "-moveAndImportComicsToFolder-"; + if(comics.size()>0) + { + QModelIndex folderDestination = foldersModelProxy->mapToSource(miFolder); + + QString destFolderPath = QDir::cleanPath(currentPath()+foldersModel->getFolderPath(folderDestination)); + + QLOG_DEBUG() << "Moving to " << destFolderPath; + + QProgressDialog * progressDialog = newProgressDialog(tr("Moving comics..."),comics.size()); + + ComicFilesManager * comicFilesManager = new ComicFilesManager(); + comicFilesManager->moveComicsTo(comics,destFolderPath,folderDestination); + + processComicFiles(comicFilesManager, progressDialog); + } +} + +void LibraryWindow::processComicFiles(ComicFilesManager * comicFilesManager, QProgressDialog * progressDialog) +{ + connect(comicFilesManager,SIGNAL(progress(int)), progressDialog, SLOT(setValue(int))); + + QThread * thread = NULL; + + thread = new QThread(); + + comicFilesManager->moveToThread(thread); + + connect(progressDialog, SIGNAL(canceled()), comicFilesManager, SLOT(cancel()), Qt::DirectConnection); + + connect(thread, SIGNAL(started()), comicFilesManager, SLOT(process())); + connect(comicFilesManager, SIGNAL(success(QModelIndex)), this, SLOT(updateCopyMoveFolderDestination(QModelIndex))); + connect(comicFilesManager, SIGNAL(finished()), thread, SLOT(quit())); + connect(comicFilesManager, SIGNAL(finished()), comicFilesManager, SLOT(deleteLater())); + connect(comicFilesManager, SIGNAL(finished()), progressDialog, SLOT(close())); + connect(comicFilesManager, SIGNAL(finished()), progressDialog, SLOT(deleteLater())); + connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater())); + + if(thread != NULL) + thread->start(); +} + +void LibraryWindow::updateCopyMoveFolderDestination(const QModelIndex & mi) +{ + updateFolder(mi); +} + +void LibraryWindow::updateCurrentFolder() +{ + updateFolder(getCurrentFolderIndex()); +} + +void LibraryWindow::updateFolder(const QModelIndex & miFolder) +{ + QLOG_DEBUG() << "UPDATE FOLDER!!!!"; + + importWidget->setUpdateLook(); + showImportingWidget(); + + QString currentLibrary = selectedLibrary->currentText(); + QString path = libraries.getPath(currentLibrary); + _lastAdded = currentLibrary; + libraryCreator->updateFolder(QDir::cleanPath(path),QDir::cleanPath(path+"/.yacreaderlibrary"),QDir::cleanPath(currentPath()+foldersModel->getFolderPath(miFolder)),miFolder); + libraryCreator->start(); +} + +QProgressDialog *LibraryWindow::newProgressDialog(const QString &label, int maxValue) +{ + QProgressDialog * progressDialog = new QProgressDialog(label,"Cancel",0,maxValue,this); + progressDialog->setWindowModality(Qt::WindowModal); + progressDialog->setMinimumWidth(350); + progressDialog->show(); + return progressDialog; +} + +void LibraryWindow::reloadAfterCopyMove(const QModelIndex & mi) +{ + if(getCurrentFolderIndex() == mi) + { + navigationController->loadFolderInfo(mi); + } + + foldersModel->fetchMoreFromDB(mi); + + enableNeededActions(); +} + +QModelIndex LibraryWindow::getCurrentFolderIndex() +{ + if(foldersView->selectionModel()->selectedRows().length()>0) + return foldersModelProxy->mapToSource(foldersView->currentIndex()); + else + return QModelIndex(); +} + +void LibraryWindow::enableNeededActions() +{ + if(foldersModel->rowCount(QModelIndex())>0) + disableFoldersActions(false); + + if(comicsModel->rowCount()>0) + disableComicsActions(false); + + disableLibrariesActions(false); + +} + +void LibraryWindow::addFolderToCurrentIndex() +{ + QModelIndex currentIndex = getCurrentFolderIndex(); + + bool ok; + QString newFolderName = QInputDialog::getText(this, tr("Add new folder"), + tr("Folder name:"), QLineEdit::Normal, + "", &ok); + + //chars not supported in a folder's name: / \ : * ? " < > | + QRegExp invalidChars("\\/\\:\\*\\?\\\"\\<\\>\\|\\\\");//TODO this regexp is not properly written + bool isValid = !newFolderName.contains(invalidChars); + + if (ok && !newFolderName.isEmpty() && isValid) + { + QString parentPath = QDir::cleanPath(currentPath()+foldersModel->getFolderPath(currentIndex)); + QDir parentDir(parentPath); + QDir newFolder(parentPath+"/"+newFolderName); + if(parentDir.mkdir(newFolderName) || newFolder.exists()) + { + QModelIndex newIndex = foldersModel->addFolderAtParent(newFolderName,currentIndex); + foldersView->setCurrentIndex(foldersModelProxy->mapFromSource(newIndex)); + navigationController->loadFolderInfo(newIndex); + historyController->updateHistory(YACReaderLibrarySourceContainer(newIndex,YACReaderLibrarySourceContainer::Folder)); + //a new folder is always an empty folder + showEmptyFolderView(); + } + } +} + +void LibraryWindow::deleteSelectedFolder() +{ + QModelIndex currentIndex = getCurrentFolderIndex(); + QString relativePath = foldersModel->getFolderPath(currentIndex); + QString folderPath = QDir::cleanPath(currentPath()+relativePath); + + if(!currentIndex.isValid()) + QMessageBox::information(this,tr("No folder selected"), tr("Please, select a folder first")); + else + { + QString libraryPath = QDir::cleanPath(currentPath()); + if((libraryPath == folderPath) || relativePath.isEmpty() || relativePath == "/") + QMessageBox::critical(this,tr("Error in path"),tr("There was an error accessing the folder's path")); + else + { + int ret = QMessageBox::question(this,tr("Delete folder"),tr("The selected folder and all its contents will be deleted from your disk. Are you sure?") + "\n\nFolder : " + folderPath,QMessageBox::Yes,QMessageBox::No); + + if(ret == QMessageBox::Yes) + { + //no folders multiselection by now + QModelIndexList indexList; + indexList << currentIndex; + + QList paths; + paths << folderPath; + + FoldersRemover * remover = new FoldersRemover(indexList,paths); + + QThread * thread = NULL; + + thread = new QThread(this); + + remover->moveToThread(thread); + + connect(thread, SIGNAL(started()), remover, SLOT(process())); + connect(remover, SIGNAL(remove(QModelIndex)), foldersModel, SLOT(deleteFolder(QModelIndex))); + connect(remover, SIGNAL(removeError()),this,SLOT(errorDeletingFolder())); + connect(remover, SIGNAL(finished()),navigationController,SLOT(reselectCurrentFolder())); + connect(remover, SIGNAL(finished()), remover, SLOT(deleteLater())); + connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater())); + + if(thread != NULL) + thread->start(); + } + } + } +} + +void LibraryWindow::errorDeletingFolder() +{ + QMessageBox::critical(this,tr("Unable to delete"),tr("There was an issue trying to delete the selected folders. Please, check for write permissions and be sure that any applications are using these folders or any of the contained files.")); +} + +void LibraryWindow::addNewReadingList() +{ + QModelIndexList selectedLists = listsView->selectionModel()->selectedIndexes(); + QModelIndex sourceMI; + if(!selectedLists.isEmpty()) + sourceMI = listsModelProxy->mapToSource(selectedLists.at(0)); + + if(selectedLists.isEmpty() || !listsModel->isReadingSubList(sourceMI) ) + { + bool ok; + QString newListName = QInputDialog::getText(this, tr("Add new reading lists"), + tr("List name:"), QLineEdit::Normal, + "", &ok); + if (ok) { + if(selectedLists.isEmpty() || !listsModel->isReadingList(sourceMI)) + listsModel->addReadingList(newListName); //top level + else + { + listsModel->addReadingListAt(newListName,sourceMI); //sublist + } + } + } +} + +void LibraryWindow::deleteSelectedReadingList() +{ + QModelIndexList selectedLists = listsView->selectionModel()->selectedIndexes(); + if(!selectedLists.isEmpty()) + { + QModelIndex mi = listsModelProxy->mapToSource(selectedLists.at(0)); + if(listsModel->isEditable(mi)) + { + int ret = QMessageBox::question(this,tr("Delete list/label"),tr("The selected item will be deleted, your comics or folders will NOT be deleted from your disk. Are you sure?"),QMessageBox::Yes,QMessageBox::No); + if(ret == QMessageBox::Yes) + { + listsModel->deleteItem(mi); + navigationController->reselectCurrentList(); + } + } + } +} + +void LibraryWindow::showAddNewLabelDialog() +{ + AddLabelDialog * dialog = new AddLabelDialog(); + int ret = dialog->exec(); + + if (ret == QDialog::Accepted) + { + YACReader::LabelColors color = dialog->selectedColor(); + QString name = dialog->name(); + + listsModel->addNewLabel(name,color); + } +} + +//TODO implement editors in treeview +void LibraryWindow::showRenameCurrentList() +{ + QModelIndexList selectedLists = listsView->selectionModel()->selectedIndexes(); + if(!selectedLists.isEmpty()) + { + QModelIndex mi = listsModelProxy->mapToSource(selectedLists.at(0)); + if(listsModel->isEditable(mi)) + { + bool ok; + QString newListName = QInputDialog::getText(this, tr("Rename list name"), + tr("List name:"), QLineEdit::Normal, + listsModel->name(mi), &ok); + + if(ok) + listsModel->rename(mi,newListName); + } + } + +} + +void LibraryWindow::addSelectedComicsToFavorites() +{ + QModelIndexList indexList = getSelectedComics(); + comicsModel->addComicsToFavorites(indexList); +} + +void LibraryWindow::showComicsViewContextMenu(const QPoint &point) +{ + QMenu menu; + + menu.addAction(openComicAction); + menu.addAction(saveCoversToAction); + menu.addSeparator(); + menu.addAction(openContainingFolderComicAction); + menu.addAction(updateCurrentFolderAction); + menu.addSeparator(); + menu.addAction(resetComicRatingAction); + menu.addSeparator(); + menu.addAction(editSelectedComicsAction); + menu.addAction(getInfoAction); + menu.addAction(asignOrderAction); + menu.addSeparator(); + menu.addAction(selectAllComicsAction); + menu.addSeparator(); + menu.addAction(setAsReadAction); + menu.addAction(setAsNonReadAction); + menu.addSeparator(); + menu.addAction(deleteComicsAction); + menu.addSeparator(); + menu.addAction(addToMenuAction); + QMenu subMenu; + setupAddToSubmenu(subMenu); + +#ifndef Q_OS_MAC + menu.addSeparator(); + menu.addAction(toggleFullScreenAction); +#endif + + menu.exec(comicsView->mapToGlobal(point)); +} + +void LibraryWindow::showComicsItemContextMenu(const QPoint &point) +{ + QMenu menu; + + menu.addAction(openComicAction); + menu.addAction(saveCoversToAction); + menu.addSeparator(); + menu.addAction(openContainingFolderComicAction); + menu.addAction(updateCurrentFolderAction); + menu.addSeparator(); + menu.addAction(resetComicRatingAction); + menu.addSeparator(); + menu.addAction(editSelectedComicsAction); + menu.addAction(getInfoAction); + menu.addAction(asignOrderAction); + menu.addSeparator(); + menu.addAction(setAsReadAction); + menu.addAction(setAsNonReadAction); + menu.addSeparator(); + menu.addAction(deleteComicsAction); + menu.addSeparator(); + menu.addAction(addToMenuAction); + QMenu subMenu; + setupAddToSubmenu(subMenu); + + menu.exec(comicsView->mapToGlobal(point)); +} + +void LibraryWindow::setupAddToSubmenu(QMenu &menu) +{ + menu.addAction(addToFavoritesAction); + addToMenuAction->setMenu(&menu); + + const QList labels = listsModel->getLabels(); + if(labels.count() > 0) + menu.addSeparator(); + foreach(LabelItem * label, labels) + { + QAction * action = new QAction(this); + action->setIcon(label->getIcon()); + action->setText(label->name()); + + action->setData(label->getId()); + + menu.addAction(action); + + connect(action,SIGNAL(triggered()),this,SLOT(onAddComicsToLabel())); + } +} + +void LibraryWindow::onAddComicsToLabel() +{ + QAction * action = static_cast(sender()); + + qulonglong labelId = action->data().toULongLong(); + + QModelIndexList comics = getSelectedComics(); + + comicsModel->addComicsToLabel(comics,labelId); +} + +void LibraryWindow::setToolbarTitle(const QModelIndex &modelIndex) +{ +#ifndef Q_OS_MAC + if(!modelIndex.isValid()) + libraryToolBar->setCurrentFolderName(selectedLibrary->currentText()); + else + libraryToolBar->setCurrentFolderName(modelIndex.data().toString()); +#endif +} + +void LibraryWindow::saveSelectedCoversTo() +{ + QFileDialog saveDialog; + QString folderPath = saveDialog.getExistingDirectory(this,tr("Save covers"),QStandardPaths::writableLocation(QStandardPaths::DesktopLocation)); + if (!folderPath.isEmpty()) + { + QModelIndexList comics = getSelectedComics(); + foreach(QModelIndex comic, comics) + { + QString origin = comic.data(ComicModel::CoverPathRole).toString().remove("file:///"); + QString destination = QDir(folderPath).filePath(comic.data(ComicModel::FileNameRole).toString()+".jpg"); + + QLOG_DEBUG() << "From : " << origin; + QLOG_DEBUG() << "To : " << destination; + + QFile::copy(origin,destination); + } + } +} + +void LibraryWindow::selectSubfolder(const QModelIndex &mi, int child) +{ + QModelIndex dest = foldersModel->index(child,0,mi); + foldersView->setCurrentIndex(dest); + navigationController->selectedFolder(dest); +} + +//this methods is only using after deleting comics +//TODO broken window :) +void LibraryWindow::checkEmptyFolder() +{ + if(comicsModel->rowCount()>0 && !importedCovers) + { + disableComicsActions(false); + } + else + { + disableComicsActions(true); +#ifndef Q_OS_MAC + if(comicsModel->rowCount()>0) + toggleFullScreenAction->setEnabled(true); +#endif + if(comicsModel->rowCount() == 0) + navigationController->reselectCurrentFolder(); + } +} + +void LibraryWindow::openComic() +{ + if(!importedCovers) + { + ComicDB comic = comicsModel->getComic(comicsView->currentIndex()); + QString path = currentPath(); + QList siblings = comicsModel->getAllComics(); + + quint64 comicId = comic.id; + //TODO generate IDS for libraries... + quint64 libraryId = libraries.getId(selectedLibrary->currentText()); + + // %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("--comicId=") + QString("%1").arg(comicId); + QString libraryIdS = QString("--libraryId=") + 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(QString("--comicId=") + QString::number(comicId)).arg(QString("--libraryId=") + QString::number(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("--comicId=") + QString::number(comicId)) << (QString("--libraryId=") + 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) { + comicsModel->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->open(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->open(); +} + +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); + d.removeRecursively(); + if(libraries.isEmpty())//no more libraries available. + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + listsView->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")); + messageBox->setParent(this); + messageBox->setWindowModality(Qt::WindowModal); + int ret = messageBox->exec(); + if(ret == QMessageBox::Yes) + { + libraries.remove(currentLibrary); + selectedLibrary->removeItem(selectedLibrary->currentIndex()); + //selectedLibrary->setCurrentIndex(0); + if(libraries.isEmpty())//no more libraries available. + { + comicsView->setModel(NULL); + foldersView->setModel(NULL); + listsView->setModel(NULL); + + disableAllActions(); + showNoLibrariesWidget(); + } + libraries.save(); + } + else if(ret == QMessageBox::YesToAll) + { + deleteCurrentLibrary(); + } + +} + +void LibraryWindow::renameLibrary() +{ + renameLibraryDialog->open(); +} + +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(!foldersModelProxy->mapToSource(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)) + { + navigationController->selectedFolder(QModelIndex()); + } + else + { + comicsView->setModel(NULL); + } + + foldersView->selectionModel()->clear(); + } +} + + +void LibraryWindow::toggleFullScreen() +{ + fullscreen?toNormal():toFullScreen(); + fullscreen = !fullscreen; +} + +//QTBUG-41883 +void LibraryWindow::toFullScreen() +{ + _size = size(); + _pos = pos(); + hide(); + + fromMaximized = this->isMaximized(); + + sideBar->hide(); + libraryToolBar->hide(); + + comicsView->toFullScreen(); + + setWindowFlags(windowFlags() | Qt::WindowStaysOnTopHint); + setWindowState(windowState() | Qt::WindowFullScreen); + resize(windowHandle()->screen()->size()-QSize(0,1)); + + show(); +} + +//QTBUG-41883 +void LibraryWindow::toNormal() +{ + hide(); + + sideBar->show(); + + comicsView->toNormal(); + + setWindowFlags(windowFlags() & ~Qt::WindowStaysOnTopHint); + setWindowState(windowState() & ~Qt::WindowFullScreen); + resize(_size); + move(_pos); + + 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 + + show(); + +} + +void LibraryWindow::setSearchFilter(const YACReader::SearchModifiers modifier, QString filter) +{ + if(!filter.isEmpty()) + { + status = LibraryWindow::Searching; + foldersModelProxy->setFilter(modifier, filter, true);//includeComicsCheckBox->isChecked()); + comicsModel->setupModelData(modifier, filter, foldersModel->getDatabase()); + comicsView->enableFilterMode(true); + comicsView->setModel(comicsModel); //TODO, columns are messed up after ResetModel some times, this shouldn't be necesary + foldersView->expandAll(); + + if(comicsModel->rowCount() == 0) + showNoSearchResultsView(); + else + showComicsView(); + } + else if(status == LibraryWindow::Searching) + {//if no searching, then ignore this + clearSearchFilter(); + navigationController->loadPreviousStatus(); + } +} + +void LibraryWindow::clearSearchFilter() +{ + foldersModelProxy->clear(); + comicsView->enableFilterMode(false); + foldersView->collapseAll(); + status = LibraryWindow::Normal; +} + + +void LibraryWindow::showProperties() +{ + QModelIndexList indexList = getSelectedComics(); + + QList comics = comicsModel->getComics(indexList); + ComicDB c = comics[0]; + _comicIdEdited = c.id;//static_cast(indexList[0].internalPointer())->data(4).toULongLong(); + + propertiesDialog->databasePath = foldersModel->getDatabase(); + propertiesDialog->basePath = currentPath(); + propertiesDialog->setComics(comics); + + propertiesDialog->show(); +} + +void LibraryWindow::showComicVineScraper() +{ + QSettings s(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + s.beginGroup("ComicVine"); + + if(!s.contains(COMIC_VINE_API_KEY)) + { + ApiKeyDialog d; + d.exec(); + } + + //check if the api key was inserted + if(s.contains(COMIC_VINE_API_KEY)) + { + QModelIndexList indexList = getSelectedComics(); + + QList comics = comicsModel->getComics(indexList); + ComicDB c = comics[0]; + _comicIdEdited = c.id;//static_cast(indexList[0].internalPointer())->data(4).toULongLong(); + + comicVineDialog->databasePath = foldersModel->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(); + + comicsModel->startTransaction(); + for(auto & index:indexList) + { + comicsModel->resetComicRating(index); + } + comicsModel->finishTransaction(); +} + +void LibraryWindow::switchToComicsView(ComicsView * from, ComicsView * to) +{ + //setup views + disconnectComicsViewConnections(from); + from->close(); + + comicsView = to; + doComicsViewConnections(); + + comicsView->setToolBar(editInfoToolBar); + + comicsViewStack->removeWidget(from); + comicsViewStack->addWidget(comicsView); + + delete from; + + //load content into current view + loadCoversFromCurrentModel(); + + if(!searchEdit->text().isEmpty()) + { + comicsView->enableFilterMode(true); + } +} + +void LibraryWindow::showComicsViewTransition() +{ + comicsViewStack->setCurrentWidget(comicsViewTransition); + comicsViewTransition->startMovie(); +} + +void LibraryWindow::toggleComicsView_delayed() +{ + if(comicsViewStatus == Flow){ + QIcon icoViewsButton; + icoViewsButton.addFile(":/images/main_toolbar/flow.png", QSize(), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); +#ifdef Q_OS_MAC + libraryToolBar->updateViewSelectorIcon(icoViewsButton); +#endif + switchToComicsView(classicComicsView, gridComicsView = new GridComicsView()); + comicsViewStatus = Grid; + } + else{ + QIcon icoViewsButton; + icoViewsButton.addFile(":/images/main_toolbar/grid.png", QSize(), QIcon::Normal); + toggleComicsViewAction->setIcon(icoViewsButton); +#ifdef Q_OS_MAC + libraryToolBar->updateViewSelectorIcon(icoViewsButton); +#endif + 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); +} + +void LibraryWindow::showEmptyLabelView() +{ + comicsViewStack->setCurrentWidget(emptyLabelWidget); +} + +void LibraryWindow::showEmptySpecialList() +{ + comicsViewStack->setCurrentWidget(emptySpecialList); +} + +void LibraryWindow::showEmptyReadingListWidget() +{ + comicsViewStack->setCurrentWidget(emptyReadingList); +} + +void LibraryWindow::showNoSearchResultsView() +{ + comicsViewStack->setCurrentWidget(noSearchResultsWidget); +} + +//TODO recover the current comics selection and restore it in the destination +void LibraryWindow::toggleComicsView() +{ + if(comicsViewStack->currentWidget()==comicsView) { + QTimer::singleShot(0,this,SLOT(showComicsViewTransition())); + QTimer::singleShot(32,this,SLOT(toggleComicsView_delayed())); + } else + toggleComicsView_delayed(); +} + +void LibraryWindow::checkSearchNumResults(int numResults) +{ + if(numResults == 0) + showNoSearchResultsView(); + else + showComicsView(); +} + +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; + } + qint64 edited = comicsModel->asignNumbers(indexList,startingNumber); + + //TODO add resorting without reloading + navigationController->loadFolderInfo(foldersModelProxy->mapToSource(foldersView->currentIndex())); + + const QModelIndex & mi = comicsModel->getIndexFromId(edited); + if(mi.isValid()) + { + comicsView->scrollTo(mi,QAbstractItemView::PositionAtCenter); + comicsView->setCurrentIndex(mi); + } +} + +void LibraryWindow::openContainingFolderComic() +{ +QModelIndex modelIndex = comicsView->currentIndex(); +QFileInfo file = QDir::cleanPath(currentPath() + comicsModel->getComicPath(modelIndex)); +#if defined Q_OS_UNIX && !defined Q_OS_MAC + 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 = foldersModelProxy->mapToSource(foldersView->currentIndex()); + QString path; + if(modelIndex.isValid()) + path = QDir::cleanPath(currentPath() + foldersModel->getFolderPath(modelIndex)); + else + path = QDir::cleanPath(currentPath()); + QDesktopServices::openUrl(QUrl("file:///"+path, QUrl::TolerantMode)); +} + +void LibraryWindow::setFolderAsNotCompleted() +{ + //foldersModel->updateFolderCompletedStatus(foldersView->selectionModel()->selectedRows(),false); + foldersModel->updateFolderCompletedStatus(QModelIndexList() << foldersModelProxy->mapToSource(foldersView->currentIndex()),false); +} + +void LibraryWindow::setFolderAsCompleted() +{ + //foldersModel->updateFolderCompletedStatus(foldersView->selectionModel()->selectedRows(),true); + foldersModel->updateFolderCompletedStatus(QModelIndexList() << foldersModelProxy->mapToSource(foldersView->currentIndex()),true); +} + +void LibraryWindow::setFolderAsRead() +{ + //foldersModel->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),true); + foldersModel->updateFolderFinishedStatus(QModelIndexList() << foldersModelProxy->mapToSource(foldersView->currentIndex()),true); +} + +void LibraryWindow::setFolderAsUnread() +{ + //foldersModel->updateFolderFinishedStatus(foldersView->selectionModel()->selectedRows(),false); + foldersModel->updateFolderFinishedStatus(QModelIndexList() << foldersModelProxy->mapToSource(foldersView->currentIndex()),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()); +} + +QString LibraryWindow::currentFolderPath() +{ + QString path; + + if(foldersView->selectionModel()->selectedRows().length()>0) + path = foldersModel->getFolderPath(foldersModelProxy->mapToSource(foldersView->currentIndex())); + else + path = foldersModel->getFolderPath(QModelIndex()); + + QLOG_DEBUG() << "current folder path : " << QDir::cleanPath(currentPath()+path); + + return QDir::cleanPath(currentPath()+path); +} + +//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->open(); +} + +void LibraryWindow::showImportComicsInfo() +{ + importComicsInfoDialog->dest = currentPath() + "/.yacreaderlibrary/library.ydb"; + importComicsInfoDialog->open(); +} +#include "startup.h" +extern Startup * s; +void LibraryWindow::closeEvent ( QCloseEvent * event ) +{ + s->stop(); + settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry()); + + comicsView->close(); + sideBar->close(); + + QApplication::instance()->processEvents(); + event->accept(); + QMainWindow::closeEvent(event); +} + +void LibraryWindow::showNoLibrariesWidget() +{ + disableAllActions(); + searchEdit->setDisabled(true); + mainWidget->setCurrentIndex(1); +} + +void LibraryWindow::showRootWidget() +{ +#ifndef Q_OS_MAC + libraryToolBar->setDisabled(false); +#endif + searchEdit->setEnabled(true); + mainWidget->setCurrentIndex(0); +} + +void LibraryWindow::showImportingWidget() +{ + disableAllActions(); + importWidget->clear(); +#ifndef Q_OS_MAC + libraryToolBar->setDisabled(true); +#endif + searchEdit->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->selectIndex(0); + selection = comicsView->selectionModel()->selectedRows(); + } + return selection; +} + +void LibraryWindow::deleteComics() +{ + //TODO + if(!listsView->selectionModel()->selectedRows().isEmpty()) + { + deleteComicsFromList(); + }else + { + deleteComicsFromDisk(); + } +} + +void LibraryWindow::deleteComicsFromDisk() +{ + 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 = comicsModel->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); + QThread * thread = NULL; + + thread = new QThread(this); + + remover->moveToThread(thread); + + comicsModel->startTransaction(); + + connect(thread, SIGNAL(started()), remover, SLOT(process())); + connect(remover, SIGNAL(remove(int)), comicsModel, SLOT(remove(int))); + connect(remover, SIGNAL(removeError()),this,SLOT(setRemoveError())); + connect(remover, SIGNAL(finished()), comicsModel, SLOT(finishTransaction())); + + connect(remover, SIGNAL(finished()),this,SLOT(checkEmptyFolder())); + connect(remover, SIGNAL(finished()),this,SLOT(checkRemoveError())); + connect(remover, SIGNAL(finished()), remover, SLOT(deleteLater())); + connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater())); + + if(thread != NULL) + thread->start(); + } +} + +void LibraryWindow::deleteComicsFromList() +{ + int ret = QMessageBox::question(this,tr("Remove comics"),tr("Comics will only be deleted from the current label/list. Are you sure?"),QMessageBox::Yes,QMessageBox::No); + + if(ret == QMessageBox::Yes) + { + QModelIndexList indexList = getSelectedComics(); + if(indexList.isEmpty()) + return; + + QModelIndex mi = listsModelProxy->mapToSource(listsView->currentIndex()); + + ReadingListModel::TypeList typeList = (ReadingListModel::TypeList)mi.data(ReadingListModel::TypeListsRole).toInt(); + + qulonglong id = mi.data(ReadingListModel::IDRole).toULongLong(); + switch (typeList) { + case ReadingListModel::SpecialList: + //by now only 'favorites' + comicsModel->deleteComicsFromFavorites(indexList); + break; + case ReadingListModel::Label: + comicsModel->deleteComicsFromLabel(indexList,id); + break; + case ReadingListModel::ReadingList: + comicsModel->deleteComicsFromReadingList(indexList,id); + break; + } + } + +} + +void LibraryWindow::showFoldersContextMenu(const QPoint &point) +{ + QModelIndex sourceMI = foldersModelProxy->mapToSource(foldersView->indexAt(point)); + + bool isCompleted = sourceMI.data(FolderModel::CompletedRole).toBool(); + bool isRead = sourceMI.data(FolderModel::FinishedRole).toBool(); + + QMenu menu; + //QMenu * folderMenu = new QMenu(tr("Folder")); + menu.addAction(openContainingFolderAction); + menu.addAction(updateFolderAction); + menu.addSeparator();//------------------------------- + if(isCompleted) + menu.addAction(setFolderAsNotCompletedAction); + else + menu.addAction(setFolderAsCompletedAction); + menu.addSeparator();//------------------------------- + if(isRead) + menu.addAction(setFolderAsUnreadAction); + else + menu.addAction(setFolderAsReadAction); + + menu.exec(foldersView->mapToGlobal(point)); + +} + +/* +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::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->open(libraries); +} + +void LibraryWindow::updateComicsView(quint64 libraryId, const ComicDB & comic) +{ + if(libraryId == libraries.getId(selectedLibrary->currentText())) { + comicsModel->reload(comic); + } +} diff --git a/YACReaderLibrary/library_window.h b/YACReaderLibrary/library_window.h new file mode 100644 index 00000000..69ae896c --- /dev/null +++ b/YACReaderLibrary/library_window.h @@ -0,0 +1,410 @@ +#ifndef __LIBRARYWINDOW_H +#define __LIBRARYWINDOW_H + +#include +#include +#include +#include +#include "yacreader_global.h" +#include "yacreader_libraries.h" + +#include "yacreader_navigation_controller.h" + +#ifdef Q_OS_MAC + #include "yacreader_macosx_toolbar.h" +#endif + +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 ComicModel; +class QSplitter; +class FolderModel; +class FolderModelProxy; +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 YACReaderFoldersView; +class YACReaderMainToolBar; +class ComicVineDialog; +class ComicsView; +class ClassicComicsView; +class GridComicsView; +class ComicsViewTransition; +class EmptyFolderWidget; +class NoSearchResultsWidget; +class EditShortcutsDialog; +class ComicFilesManager; +class QProgressDialog; +class ReadingListModel; +class ReadingListModelProxy; +class YACReaderReadingListsView; +class YACReaderHistoryController; +class EmptyLabelWidget; +class EmptySpecialListWidget; +class EmptyReadingListWidget; + +#include "comic_db.h" + +using namespace YACReader; + +class LibraryWindow : public QMainWindow +{ + friend class YACReaderNavigationController; + + 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; + + PackageManager * packageManager; + + QSize slideSizeW; + QSize slideSizeF; + //search filter +#ifdef Q_OS_MAC + YACReaderMacOSXSearchLineEdit * searchEdit; +#else + YACReaderSearchLineEdit * searchEdit; +#endif + + QString previousFilter; + QCheckBox * includeComicsCheckBox; + //------------- + + YACReaderNavigationController * navigationController; + + ComicsView * comicsView; + ClassicComicsView * classicComicsView; + GridComicsView * gridComicsView; + QStackedWidget * comicsViewStack; + ComicsViewTransition * comicsViewTransition; + EmptyFolderWidget * emptyFolderWidget; + EmptyLabelWidget * emptyLabelWidget; + EmptySpecialListWidget * emptySpecialList; + EmptyReadingListWidget * emptyReadingList; + NoSearchResultsWidget * noSearchResultsWidget; + + YACReaderFoldersView * foldersView; + YACReaderReadingListsView * listsView; + YACReaderLibraryListWidget * selectedLibrary; + FolderModel * foldersModel; + FolderModelProxy * foldersModelProxy; + ComicModel * comicsModel; + ReadingListModel * listsModel; + ReadingListModelProxy * listsModelProxy; + //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; +#ifndef Q_OS_MAC + QAction * toggleFullScreenAction; +#endif + QAction * optionsAction; + QAction * serverConfigAction; + QAction * toggleComicsViewAction; + //QAction * socialAction; + + //tree actions + QAction * addFolderAction; + QAction * deleteFolderAction; + //-- + QAction * setRootIndexAction; + QAction * expandAllNodesAction; + QAction * colapseAllNodesAction; + + QAction * openContainingFolderAction; + QAction * saveCoversToAction; + //-- + 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; + + QAction * updateFolderAction; + QAction * updateCurrentFolderAction; + + //reading lists actions + QAction * addReadingListAction; + QAction * deleteReadingListAction; + QAction * addLabelAction; + QAction * renameListAction; + //-- + QAction * addToMenuAction; + QAction * addToFavoritesAction; + +#ifdef Q_OS_MAC + YACReaderMacOSXToolbar * 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; + //QModelIndex updateDestination; + + quint64 _comicIdEdited; + + enum NavigationStatus + { + Normal, // + Searching + }; + + NavigationStatus status; + + 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(); + QString currentFolderPath(); + + //settings + QSettings * settings; + + //navigation backward and forward + YACReaderHistoryController * historyController; + + bool removeError; + + ComicsViewStatus comicsViewStatus; + + //QTBUG-41883 + QSize _size; + QPoint _pos; + +protected: + virtual void closeEvent ( QCloseEvent * event ); +public: + LibraryWindow(); + +public slots: + void loadLibrary(const QString & path); + void selectSubfolder(const QModelIndex & mi, int child); + void checkEmptyFolder(); + 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 setSearchFilter(const YACReader::SearchModifiers modifier, QString filter); + void clearSearchFilter(); + 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 deleteComicsFromDisk(); + void deleteComicsFromList(); + //void showSocial(); + void showFoldersContextMenu(const QPoint & point); + 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 showEmptyLabelView(); + void showEmptySpecialList(); + void showEmptyReadingListWidget(); + void showNoSearchResultsView(); + void toggleComicsView(); + void checkSearchNumResults(int numResults); + void loadCoversFromCurrentModel(); + void copyAndImportComicsToCurrentFolder(const QList > & comics); + void moveAndImportComicsToCurrentFolder(const QList > &comics); + void copyAndImportComicsToFolder(const QList > & comics, const QModelIndex & miFolder); + void moveAndImportComicsToFolder(const QList > & comics, const QModelIndex & miFolder); + void processComicFiles(ComicFilesManager * comicFilesManager, QProgressDialog * progressDialog); + void updateCopyMoveFolderDestination(const QModelIndex & mi); //imports new comics from the current folder + void updateCurrentFolder(); + void updateFolder(const QModelIndex & miFolder); + QProgressDialog * newProgressDialog(const QString & label, int maxValue); + void reloadAfterCopyMove(const QModelIndex &mi); + QModelIndex getCurrentFolderIndex(); + void enableNeededActions(); + void addFolderToCurrentIndex(); + void deleteSelectedFolder(); + void errorDeletingFolder(); + void addNewReadingList(); + void deleteSelectedReadingList(); + void showAddNewLabelDialog(); + void showRenameCurrentList(); + void addSelectedComicsToFavorites(); + void showComicsViewContextMenu(const QPoint & point); + void showComicsItemContextMenu(const QPoint & point); + void setupAddToSubmenu(QMenu & menu); + void onAddComicsToLabel(); + void setToolbarTitle(const QModelIndex & modelIndex); + void saveSelectedCoversTo(); + +}; + +#endif + + + diff --git a/YACReaderLibrary/main.cpp b/YACReaderLibrary/main.cpp new file mode 100644 index 00000000..5f12d0f0 --- /dev/null +++ b/YACReaderLibrary/main.cpp @@ -0,0 +1,253 @@ +#include "library_window.h" + +#include +#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"); + qApp->setAttribute(Qt::AA_UseHighDpiPixmaps); +//simple command line parser +//will be replaced by QCommandLineParser in the future +//TODO: --headless, --server=[on|off], support for file and directory arguments + if (argc > 1) + { + QTextStream parser(stdout); + QStringList optlist = QCoreApplication::arguments().filter(QRegExp ("^-{1,2}")); + if (optlist.contains("--version") || optlist.contains("-v")) + { + parser << app.applicationName() << " " << QString(VERSION) << endl << "Copyright 2014 by Luis Angel San Martin Rodriguez" << endl; + return 0; + } + if (optlist.contains("--help") || optlist.contains("-h")) + { + parser << endl << "Usage:" << "\tYACReaderLibrary [Option]" << endl << endl; + parser << "Options:" << endl; + parser << " none\t\t\tStart YACReaderLibrary" << endl; + parser << " -h, --help\t\tDisplay help text and exit." << endl; + parser << " -v, --version\t\tDisplay version information and exit." << endl; + return 0; + } + parser << "Unsupported command line options. See YACReaderLibrary --help for further information." << endl; + return 0; + } + + 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/no_search_results_widget.cpp b/YACReaderLibrary/no_search_results_widget.cpp new file mode 100644 index 00000000..38b181a4 --- /dev/null +++ b/YACReaderLibrary/no_search_results_widget.cpp @@ -0,0 +1,51 @@ +#include "no_search_results_widget.h" + +#include +#include +#include + +NoSearchResultsWidget::NoSearchResultsWidget(QWidget *parent) : + QWidget(parent) +{ +#ifdef Q_OS_MAC + backgroundColor = "#FFFFFF"; +#else + backgroundColor = "#2A2A2A"; +#endif + + QVBoxLayout * layout = new QVBoxLayout; + + iconLabel = new QLabel(); + iconLabel->setPixmap(QPixmap(":/images/empty_search.png")); + iconLabel->setAlignment(Qt::AlignCenter); + + titleLabel = new QLabel("No results"); + titleLabel->setAlignment(Qt::AlignCenter); + +#ifdef Q_OS_MAC + titleLabel->setStyleSheet("QLabel {color:#888888; font-size:24px;font-family:Arial;font-weight:bold;}"); +#else + titleLabel->setStyleSheet("QLabel {color:#CCCCCC; font-size:24px;font-family:Arial;font-weight:bold;}"); +#endif + + layout->addSpacing(100); + layout->addWidget(iconLabel); + layout->addSpacing(30); + layout->addWidget(titleLabel); + layout->addStretch(); + layout->setMargin(0); + layout->setSpacing(0); + + setContentsMargins(0,0,0,0); + + setStyleSheet(QString("QWidget {background:%1}").arg(backgroundColor)); + + setSizePolicy(QSizePolicy ::Expanding , QSizePolicy ::Expanding ); + setLayout(layout); +} + +void NoSearchResultsWidget::paintEvent(QPaintEvent *) +{ + QPainter painter (this); + painter.fillRect(0,0,width(),height(),QColor(backgroundColor)); +} diff --git a/YACReaderLibrary/no_search_results_widget.h b/YACReaderLibrary/no_search_results_widget.h new file mode 100644 index 00000000..0cad18fe --- /dev/null +++ b/YACReaderLibrary/no_search_results_widget.h @@ -0,0 +1,26 @@ +#ifndef NO_SEARCH_RESULTS_WIDGET_H +#define NO_SEARCH_RESULTS_WIDGET_H + +#include + +class QLabel; + +class NoSearchResultsWidget : public QWidget +{ + Q_OBJECT +public: + explicit NoSearchResultsWidget(QWidget *parent = 0); + +signals: + +public slots: + +protected: + QLabel * iconLabel; + QLabel * titleLabel; + void paintEvent(QPaintEvent *); + QString backgroundColor; + +}; + +#endif // NO_SEARCH_RESULTS_WIDGET_H diff --git a/YACReaderLibrary/options_dialog.cpp b/YACReaderLibrary/options_dialog.cpp new file mode 100644 index 00000000..f0165e31 --- /dev/null +++ b/YACReaderLibrary/options_dialog.cpp @@ -0,0 +1,86 @@ +#include "options_dialog.h" + +#include "yacreader_flow_gl.h" +#include "yacreader_flow_config_widget.h" +#include "yacreader_gl_flow_config_widget.h" +#include "api_key_dialog.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(); + + QVBoxLayout * apiKeyLayout = new QVBoxLayout(); + QPushButton * apiKeyButton = new QPushButton(tr("Edit Comic Vine API key")); + apiKeyLayout->addWidget(apiKeyButton); + + QGroupBox * apiKeyBox = new QGroupBox(tr("Comic Vine API key")); + apiKeyBox->setLayout(apiKeyLayout); + + connect(apiKeyButton,SIGNAL(clicked()),this,SLOT(editApiKey())); + + QWidget * comicFlowW = new QWidget; + comicFlowW->setLayout(flowLayout); + + QWidget * generalW = new QWidget; + generalW->setLayout(generalLayout); + generalLayout->addWidget(shortcutsBox); + generalLayout->addWidget(apiKeyBox); + 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); + +} + +void OptionsDialog::editApiKey() +{ + ApiKeyDialog d; + d.exec(); +} + + + diff --git a/YACReaderLibrary/options_dialog.h b/YACReaderLibrary/options_dialog.h new file mode 100644 index 00000000..9a2ca1bc --- /dev/null +++ b/YACReaderLibrary/options_dialog.h @@ -0,0 +1,21 @@ +#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); + + public slots: + void editApiKey(); +}; + + +#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..5477ae08 --- /dev/null +++ b/YACReaderLibrary/qml.qrc @@ -0,0 +1,9 @@ + + + qml/GridComicsView.qml + qml/YACReaderScrollView.qml + qml/tick.png + qml/reading.png + qml/star_menu.png + + diff --git a/YACReaderLibrary/qml/GridComicsView.qml b/YACReaderLibrary/qml/GridComicsView.qml new file mode 100644 index 00000000..2c317116 --- /dev/null +++ b/YACReaderLibrary/qml/GridComicsView.qml @@ -0,0 +1,437 @@ +import QtQuick 2.3 + +import QtQuick.Controls 1.2 +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;i<=to;i++) + { + comicsSelectionHelper.selectIndex(i); + } + } + + Component { + id: appDelegate + Rectangle + { + id: cell + width: grid.cellWidth + height: grid.cellHeight + color: backgroundColor + + + Rectangle { + id: realCell + + property int position : 0 + property bool dragging: false; + Drag.active: mouseArea.drag.active + Drag.hotSpot.x: 32 + Drag.hotSpot.y: 32 + Drag.dragType: Drag.Automatic + //Drag.mimeData: { "x": 1 } + Drag.proposedAction: Qt.CopyAction + Drag.onActiveChanged: { + if(!dragging) + { + dragManager.startDrag(); + dragging = true; + }else + dragging = false; + } + + width: 156; height: 287 + color: ((dummyValue || !dummyValue) && comicsSelectionHelper.isSelectedIndex(index))?selectedColor:cellColor; + border.color: ((dummyValue || !dummyValue) && comicsSelectionHelper.isSelectedIndex(index))?selectedBorderColor:borderColor; + border.width: (Qt.platform.os === "osx")?1:0; + + anchors.horizontalCenter: parent.horizontalCenter + + MouseArea { + id: mouseArea + drag.target: realCell + + drag.minimumX: 0 + drag.maximumX: 0 + drag.minimumY: 0 + drag.maximumY: 0 + + anchors.fill: parent + acceptedButtons: Qt.LeftButton | Qt.RightButton + + onDoubleClicked: { + comicsSelectionHelper.clear(); + + comicsSelectionHelper.selectIndex(index); + grid.currentIndex = index; + comicsSelectionHelper.selectedItem(index); + } + + onPressed: { + + var ci = grid.currentIndex; //save current index + + /*if(mouse.button != Qt.RightButton && !(mouse.modifiers & Qt.ControlModifier || mouse.modifiers & Qt.ShiftModifier)) + { + if(!comicsSelectionHelper.isSelectedIndex(index)) + comicsSelectionHelper.clear(); + }*/ + + if(mouse.modifiers & Qt.ShiftModifier) + if(index < ci) + { + selectAll(index,ci); + grid.currentIndex = index; + } + else if (index > ci) + { + selectAll(ci,index); + grid.currentIndex = index; + } + + mouse.accepted = true; + + if(mouse.button == Qt.RightButton) // context menu is requested + { + + if(!comicsSelectionHelper.isSelectedIndex(index)) //the context menu is requested outside the current selection, the selection will be + { + comicsSelectionHelper.setCurrentIndex(index) + grid.currentIndex = index; + } + + var coordinates = main.mapFromItem(realCell,mouseX,mouseY) + contextMenuHelper.requestedContextMenu(Qt.point(coordinates.x,coordinates.y)); + + } else //left button + { + + if(mouse.modifiers & Qt.ControlModifier) + { + if(comicsSelectionHelper.isSelectedIndex(index)) + { + if(comicsSelectionHelper.numItemsSelected()>1) + { + comicsSelectionHelper.deselectIndex(index); + if(grid.currentIndex === index) + grid.currentIndex = comicsSelectionHelper.lastSelectedIndex(); + } + } + else + { + comicsSelectionHelper.selectIndex(index); + grid.currentIndex = index; + } + } + + if(mouse.button != Qt.RightButton && !(mouse.modifiers & Qt.ControlModifier || mouse.modifiers & Qt.ShiftModifier)) //just left button click + { + if(comicsSelectionHelper.isSelectedIndex(index)) //the context menu is requested outside the current selection, the selection will be + { + + } + else + { + comicsSelectionHelper.setCurrentIndex(index) + } + + grid.currentIndex = index; + } + } + + } + + onReleased: { + /*if(mouse.button != Qt.RightButton && !(mouse.modifiers & Qt.ControlModifier || mouse.modifiers & Qt.ShiftModifier)) + { + comicsSelectionHelper.setCurrentIndex(index) + grid.currentIndex = index; + }*/ + } + + } + + } + + /**/ + + //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 + 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 { + id : titleText + 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: fontSpacing + font.pointSize: fontSize + font.family: fontFamily + } + //number + Text { + anchors {bottom: realCell.bottom; left: realCell.left; margins: 4} + text: number?"#"+number:"" + color: textColor + font.letterSpacing: fontSpacing + font.pointSize: fontSize + font.family: fontFamily + } + //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: fontSpacing + font.pointSize: fontSize + font.family: fontFamily + } + //rating icon + Image { + id: ratingImage + anchors {bottom: realCell.bottom; right: pageImage.left; bottomMargin: 5; rightMargin: Math.floor(pages.width)+12} + source: "star.png" + + MouseArea { + anchors.fill: parent + onClicked: { + console.log("rating"); + comicsSelectionHelper.clear(); + comicsSelectionHelper.selectIndex(index); + grid.currentIndex = index; + ratingConextMenu.popup(); + + } + } + + Menu { + id: ratingConextMenu + MenuItem { text: "1"; enabled: true; iconSource:"star_menu.png"; onTriggered: comicRatingHelper.rate(index,1) } + MenuItem { text: "2"; enabled: true; iconSource:"star_menu.png"; onTriggered: comicRatingHelper.rate(index,2) } + MenuItem { text: "3"; enabled: true; iconSource:"star_menu.png"; onTriggered: comicRatingHelper.rate(index,3) } + MenuItem { text: "4"; enabled: true; iconSource:"star_menu.png"; onTriggered: comicRatingHelper.rate(index,4) } + MenuItem { text: "5"; enabled: true; iconSource:"star_menu.png"; onTriggered: comicRatingHelper.rate(index,5) } + + } + } + + //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 + + //QTBUG-39453 + //Another fu%$·#& bug in Qt + //https://bugreports.qt.io/browse/QTBUG-39453 + //To solve this I am going to accept any input drag, drops will be filtered in "onDropped" + DropArea { + anchors.fill: parent + + /* + onEntered: { + console.log("onEntered"); + if(drag.hasUrls) + { + console.log("HAS URLS -> ", drag.urls); + if(dropManager.canDropUrls(drag.urls, drag.action)) + { + drag.accepted = true; + console.log("canDropUrls"); + }else + drag.accepted = false; + } + else if (dropManager.canDropFormats(drag.formats)) { + drag.accepted = true; + console.log("canDropFormats"); + } else + drag.accepted = false; + }*/ + + + onDropped: { + if(drop.hasUrls && dropManager.canDropUrls(drop.urls, drop.action)) + { + dropManager.droppedFiles(drop.urls, drop.action); + } + else{ + if (dropManager.canDropFormats(drop.formats)) + { + var destItem = grid.itemAt(drop.x,drop.y + grid.contentY); + var destLocalX = grid.mapToItem(destItem,drop.x,drop.y + grid.contentY).x + var realIndex = grid.indexAt(drop.x,drop.y + grid.contentY); + + if(realIndex === -1) + realIndex = grid.count - 1; + + var destIndex = destLocalX < (grid.cellWidth / 2) ? realIndex : realIndex + 1; + dropManager.droppedComicsForResortingAt(drop.getDataAsString(), destIndex); + } + } + } + + } + + 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 + + move: Transition { + NumberAnimation { properties: "x,y"; duration: 250 } + } + + moveDisplaced: Transition { + NumberAnimation { properties: "x,y"; duration: 250 } + } + + remove: Transition { + ParallelAnimation { + NumberAnimation { property: "opacity"; to: 0; duration: 250 } + + } + } + + removeDisplaced: Transition { + NumberAnimation { properties: "x,y"; duration: 250 } + } + + + + displaced: Transition { + NumberAnimation { properties: "x,y"; duration: 250 } + } + + function numCellsPerRow() { + return Math.floor(width / 185); + } + + onWidthChanged: { + var numCells = numCellsPerRow(); + var rest = width % 185; + + 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.setCurrentIndex(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/star_menu.png b/YACReaderLibrary/qml/star_menu.png new file mode 100644 index 0000000000000000000000000000000000000000..4472e5a31f0e0cbcd83edaad25020ec221020a18 GIT binary patch literal 277 zcmeAS@N?(olHy`uVBq!ia0vp@K+Mg-1|*kkVowB8k|nMYCBgY=CFO}lsSJ)O`AMk? zp1FzXsX?iUDV2pMQ*D5XwtKobhE&{2N;trnl9D1N;lb#@%jfV%-X(c*`k{sE*%4FkLg~A%fLcuI0kBv+r zZbP0l+XkKon2l~ 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..a7ec0480 --- /dev/null +++ b/YACReaderLibrary/server/controllers/comiccontroller.cpp @@ -0,0 +1,119 @@ +#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('/'); + qulonglong libraryId = pathElements.at(2).toLongLong(); + QString libraryName = DBHelper::getLibraryName(libraryId); + 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; + // } + //} + + YACReaderLibraries libraries = DBHelper::getLibraries(); + + ComicDB comic = DBHelper::getComicInfo(libraryId, comicId); + + if(!remoteComic) + session.setDownloadedComic(comic.info.hash); + + Comic * comicFile = FactoryComic::newComic(libraries.getPath(libraryId)+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(libraryId)+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(libraryId)); + if(remoteComic) //send previous and next comics id + { + QList siblings = DBHelper::getFolderComicsFromLibrary(libraryId, 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/comicdownloadinfocontroller.cpp b/YACReaderLibrary/server/controllers/comicdownloadinfocontroller.cpp new file mode 100644 index 00000000..0260f57f --- /dev/null +++ b/YACReaderLibrary/server/controllers/comicdownloadinfocontroller.cpp @@ -0,0 +1,24 @@ +#include "comicdownloadinfocontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +#include "comic_db.h" + +ComicDownloadInfoController::ComicDownloadInfoController() {} + + +void ComicDownloadInfoController::service(HttpRequest& request, HttpResponse& response) +{ + QString path = QUrl::fromPercentEncoding(request.getPath()).toLatin1(); + QStringList pathElements = path.split('/'); + + qulonglong libraryId = pathElements.at(2).toLongLong(); + qulonglong comicId = pathElements.at(4).toULongLong(); + + ComicDB comic = DBHelper::getComicInfo(libraryId, comicId); + + //TODO: check if the comic wasn't found; + response.writeText(QString("fileName:%1\r\n").arg(comic.getFileName())); + response.writeText(QString("fileSize:%1\r\n").arg(comic.getFileSize()),true); +} diff --git a/YACReaderLibrary/server/controllers/comicdownloadinfocontroller.h b/YACReaderLibrary/server/controllers/comicdownloadinfocontroller.h new file mode 100644 index 00000000..e98518e8 --- /dev/null +++ b/YACReaderLibrary/server/controllers/comicdownloadinfocontroller.h @@ -0,0 +1,19 @@ +#ifndef COMICDOWNLOADINFOCONTROLLER_H +#define COMICDOWNLOADINFOCONTROLLER_H + +#include "httprequest.h" +#include "httpresponse.h" +#include "httprequesthandler.h" + +class ComicDownloadInfoController : public HttpRequestHandler { + Q_OBJECT + Q_DISABLE_COPY(ComicDownloadInfoController); +public: + /** Constructor **/ + ComicDownloadInfoController(); + + /** Generates the response */ + void service(HttpRequest& request, HttpResponse& response); +}; + +#endif // COMICDOWNLOADINFOCONTROLLER_H diff --git a/YACReaderLibrary/server/controllers/covercontroller.cpp b/YACReaderLibrary/server/controllers/covercontroller.cpp new file mode 100644 index 00000000..f3c043af --- /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_overlayed_retina.png")); + else + p.drawImage(0,0,QImage(":/images/f_overlayed.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..0c164a88 --- /dev/null +++ b/YACReaderLibrary/server/controllers/foldercontroller.cpp @@ -0,0 +1,315 @@ +#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" + +#include "QsLog.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 folderId = pathElements.at(4).toULongLong(); + + folderId = qMax(1,folderId); + + QString folderName = DBHelper::getFolderName(libraryId,folderId); + if(folderName.isEmpty()) + { + ErrorController(300).service(request,response); + return; + } + + if(folderId!=1) + t.setVariable("folder.name",folderName); + else + t.setVariable("folder.name",libraryName); + QList folderContent = DBHelper::getFolderSubfoldersFromLibrary(libraryId,folderId); + QList folderComics = DBHelper::getFolderComicsFromLibrary(libraryId,folderId); + + //response.writeText(libraryName); + + folderContent.append(folderComics); + + qSort(folderContent.begin(),folderContent.end(),LibraryItemSorter()); + folderComics.clear(); + + //qulonglong backId = DBHelper::getParentFromComicFolderId(libraryName,folderId); + + 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(folderId == 1) + { + session.clearNavigationPath(); + session.pushNavigationItem(QPair(folderId,page)); + t.setVariable(QString("upurl"),"/"); + } + else + { + if(fromUp) + session.popNavigationItem(); + else //drill down or direct access + { + QStack > path = session.getNavigationPath(); + bool found=false; + for(QStack >::const_iterator itr = path.begin(); itr!=path.end(); itr++) + if(itr->first == folderId) + { + found = true; + break; + } + + if(found) + { + while(session.topNavigationItem().first != folderId) + session.popNavigationItem(); + + session.updateTopItem(QPair(folderId,page)); + } + else + session.pushNavigationItem(QPair(folderId,page)); + } + + QStack > path = session.getNavigationPath(); + if(path.count()>1) + { + QPair parentItem = path.at(path.count()-2); + qulonglong upParent = parentItem.first; + quint32 upPage = parentItem.second; + t.setVariable(QString("upurl"),"/library/" + QString::number(libraryId) + "/folder/" +QString("%1?page=%2&up=true").arg(upParent).arg(upPage)); + } else + t.setVariable(QString("upurl"),"/"); + } + + 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.getNavigationPath(); + t.setVariable(QString("library.name"),libraryName); + t.setVariable(QString("library.url"),QString("/library/%1/folder/1").arg(libraryId)); + t.loop("path",foldersPath.count()-1); + for(int i = 1; i < foldersPath.count(); i++){ + t.setVariable(QString("path%1.url").arg(i-1),QString("/library/%1/folder/%2").arg(libraryId).arg(foldersPath[i].first)); + t.setVariable(QString("path%1.name").arg(i-1),DBHelper::getFolderName(libraryId,foldersPath[i].first)); + } + + 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(libraryId, 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.cover.browse").arg(i),QString("").arg(QString("/library/%1/folder/%2").arg(libraryId).arg(item->id))); + t.setVariable(QString("element%1.cover.browse.end").arg(i),""); + //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.isComicOnDevice(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),""); + + t.setVariable(QString("element%1.cover.browse").arg(i),""); + t.setVariable(QString("element%1.cover.browse.end").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(folderId).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(folderId).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(folderId).arg(0)); + t.setVariable("page.previous",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(folderId).arg((page==0)?page:page-1)); + t.setVariable("page.next",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(folderId).arg((page==numPages-1)?page:page+1)); + t.setVariable("page.last",QString("/library/%1/folder/%2?page=%3").arg(libraryId).arg(folderId).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..40be45c1 --- /dev/null +++ b/YACReaderLibrary/server/controllers/folderinfocontroller.cpp @@ -0,0 +1,48 @@ +#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(); + + serviceComics(libraryId, parentId, response); + + response.writeText("",true); +} + +void FolderInfoController::serviceComics(const int &library, const qulonglong &folderId, HttpResponse &response) +{ + QList folderContent = DBHelper::getFolderSubfoldersFromLibrary(library,folderId); + QList folderComics = DBHelper::getFolderComicsFromLibrary(library,folderId); + + ComicDB * currentComic; + for(QList::const_iterator itr = folderComics.constBegin();itr!=folderComics.constEnd();itr++) + { + currentComic = (ComicDB *)(*itr); + response.writeText(QString("/library/%1/comic/%2:%3:%4\r\n").arg(library).arg(currentComic->id).arg(currentComic->getFileName()).arg(currentComic->getFileSize())); + delete currentComic; + } + + Folder * currentFolder; + for(QList::const_iterator itr = folderContent.constBegin();itr!=folderContent.constEnd();itr++) + { + currentFolder = (Folder *)(*itr); + serviceComics(library, currentFolder->id, response); + delete currentFolder; + } +} diff --git a/YACReaderLibrary/server/controllers/folderinfocontroller.h b/YACReaderLibrary/server/controllers/folderinfocontroller.h new file mode 100644 index 00000000..87df58ce --- /dev/null +++ b/YACReaderLibrary/server/controllers/folderinfocontroller.h @@ -0,0 +1,23 @@ +#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); + +private: + void serviceComics(const int &library, const qulonglong & folderId, 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..a8eda3de --- /dev/null +++ b/YACReaderLibrary/server/controllers/librariescontroller.cpp @@ -0,0 +1,40 @@ +#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,false); + + response.setHeader("Content-Type", "text/html; charset=ISO-8859-1"); + response.setHeader("Connection","close"); + + session.clearNavigationPath(); + + 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..1bfe2f07 --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.cpp @@ -0,0 +1,381 @@ +/** + @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(); +} + +void HttpSession::clearComics() +{ + if(dataPtr) + { + dataPtr->yacreaderSessionData.comicsOnDevice.clear(); + dataPtr->yacreaderSessionData.downloadedComics.clear(); + } +} +//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(); +} + +QPair HttpSession::popNavigationItem() +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.navigationPath.isEmpty())) + return dataPtr->yacreaderSessionData.navigationPath.pop(); + return QPair(); +} + +QPair HttpSession::topNavigationItem() +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.navigationPath.isEmpty())) + return dataPtr->yacreaderSessionData.navigationPath.top(); + return QPair(); +} + +void HttpSession::pushNavigationItem(const QPair &item) +{ + if(dataPtr) + dataPtr->yacreaderSessionData.navigationPath.push(item); +} + +void HttpSession::updateTopItem(const QPair &item) +{ + if(dataPtr && !(dataPtr->yacreaderSessionData.navigationPath.isEmpty())) + { + dataPtr->yacreaderSessionData.navigationPath.pop(); + dataPtr->yacreaderSessionData.navigationPath.push(item); + } else if(dataPtr) + { + dataPtr->yacreaderSessionData.navigationPath.push(item); + } +} + +QStack > HttpSession::getNavigationPath() +{ + if(dataPtr) + return dataPtr->yacreaderSessionData.navigationPath; + else + return QStack >(); +} + diff --git a/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h new file mode 100644 index 00000000..a95f818f --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/httpsession.h @@ -0,0 +1,193 @@ +/** + @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(); + void clearComics(); + + //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); + + + /*int popPage(); + void pushPage(int page); + int topPage(); + + void clearFoldersPath(); + int popFolder(); + void pushFolder(int page); + int topFolder(); + QStack getFoldersPath();*/ + + void clearNavigationPath(); + QPair popNavigationItem(); + QPair topNavigationItem(); + void pushNavigationItem(const QPair & item); + void updateTopItem(const QPair & item); + + //TODO replace QPair by a custom class for storing folderId, page and folderName(save some DB accesses) + QStack > getNavigationPath(); + + + + +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; + + //folder_id, page_number + QStack > navigationPath; + + 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..b2515e3c --- /dev/null +++ b/YACReaderLibrary/server/lib/bfHttpServer/staticfilecontroller.cpp @@ -0,0 +1,235 @@ +/** + @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 = 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..ea3a2dd8 --- /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..21e3083b --- /dev/null +++ b/YACReaderLibrary/server/requestmapper.cpp @@ -0,0 +1,171 @@ +/** + @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 "controllers/comicdownloadinfocontroller.h" + +#include "db_helper.h" +#include "yacreader_libraries.h" + +#include "QsLog.h" + +RequestMapper::RequestMapper(QObject* parent) + :HttpRequestHandler(parent) {} + +void RequestMapper::loadSession(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.contains("currentPage")) + return; + + 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"); + session.clearComics(); + 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"); + + 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("ipad"); + session.setDisplayType("@2x"); + } + + } +} + +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 comicDownloadInfo("/library/.+/comic/[0-9]+/?"); //get comic info (basic/download info) + QRegExp comicFullInfo("/library/.+/comic/[0-9]+/info/?"); //get comic info (full 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(); + + loadSession(request, response); + + //primera petición, se ha hecho un post, se sirven las bibliotecas si la seguridad mediante login no está habilitada + if(path == "/") //Don't send data to the server using '/' !!!! + { + 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(comicDownloadInfo.exactMatch(path)) + { + ComicDownloadInfoController().service(request, response); + } + else if(comicFullInfo.exactMatch(path) || comicOpen.exactMatch(path))//start download or start remote reading + { + 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..332cee09 --- /dev/null +++ b/YACReaderLibrary/server/requestmapper.h @@ -0,0 +1,37 @@ +/** + @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); + void loadSession(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..5fc65043 --- /dev/null +++ b/YACReaderLibrary/server/server.pri @@ -0,0 +1,36 @@ +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 \ + server/controllers/comicdownloadinfocontroller.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 \ + server/controllers/comicdownloadinfocontroller.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..49e0060e --- /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..73761624 --- /dev/null +++ b/YACReaderLibrary/server_config_dialog.cpp @@ -0,0 +1,326 @@ +#include "server_config_dialog.h" +#include +#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) { + 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(64, 112); + qrCode->setFixedSize(200,200); + qrCode->setScaledContents(true); + + QLabel * title1 = new QLabel(tr("Server connectivity information"),this); + title1->move(332, 61); + title1->setStyleSheet("QLabel {color:#474747; font-size:30px; font-family: Arial;}"); + + QLabel * qrMessage = new QLabel(tr("Scan it!"),this); + qrMessage->move(135,388);//373,627); + qrMessage->setStyleSheet("QLabel {color:#A3A3A3; font-size:18px; font-family: Arial;}"); + qrMessage->setWordWrap(true); + qrMessage->setFixedWidth(200); + + QLabel * propaganda = new QLabel(tr("YACReader is available for iOS devices. Discover it! "),this); + propaganda->move(332,505); + propaganda->setStyleSheet("QLabel {color:#4D4D4D; font-size:13px; font-family: Arial; font-style: italic;}"); + /*propaganda->setWordWrap(true); + propaganda->setFixedWidth(590);*/ + propaganda->setOpenExternalLinks(true); + + //FORM--------------------------------------------------------------------- + + QLabel * ipLabel = new QLabel(tr("Choose an IP address"),this); + ipLabel->move(332,117); + ipLabel->setStyleSheet("QLabel {color:#575757; font-size:18px; font-family: Arial;}"); + + QLabel * portLabel = new QLabel(tr("Port"),this); + portLabel->move(332, 211); + portLabel->setStyleSheet("QLabel {color:#575757; font-size:18px; font-family: Arial;}"); + + ip = new QComboBox(this); + connect(ip,SIGNAL(activated(const QString &)),this,SLOT(regenerateQR(const QString &))); + + ip->setFixedWidth(200); + ip->move(332,153); + + + port = new QLineEdit("8080",this); + port->setReadOnly(false); + //port->setFixedWidth(100); + //port->move(332, 244); + + //port->move(520,110); + QValidator *validator = new QIntValidator(1024, 65535, this); + port->setValidator(validator); + + QWidget * portWidget = new QWidget(this); + QHBoxLayout * portWidgetLayout = new QHBoxLayout; + portWidgetLayout->addWidget(port); + portWidgetLayout->addWidget(accept); + portWidgetLayout->setMargin(0); + portWidget->setLayout(portWidgetLayout); + portWidget->move(332, 244); + //accept->move(514,149); + connect(accept,SIGNAL(pressed()),this,SLOT(updatePort())); + //END FORM----------------------------------------------------------------- + + check = new QCheckBox(this); + check->move(332,314); + check->setText(tr("enable the server")); + check->setStyleSheet("QCheckBox {color:#262626; font-size:13px; font-family: Arial;}"); + + + //accept->move(444, 242); + //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 + { + QPixmap p = *qrCodeImage; + QPixmap pMask( p.size() ); + pMask.fill( QColor(66, 66, 66) ); + pMask.setMask( p.createMaskFromColor( Qt::white ) ); + + *qrCodeImage = pMask; + + 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_folders_view.cpp b/YACReaderLibrary/yacreader_folders_view.cpp new file mode 100644 index 00000000..0f774bc5 --- /dev/null +++ b/YACReaderLibrary/yacreader_folders_view.cpp @@ -0,0 +1,104 @@ +#include "yacreader_folders_view.h" + +#include "folder_item.h" +#include "folder_model.h" + +#include "comic.h" +#include "comic_files_manager.h" + +#include "QsLog.h" + + +YACReaderFoldersView::YACReaderFoldersView(QWidget * parent) + :YACReaderTreeView(parent) +{ + setItemDelegate(new YACReaderFoldersViewItemDeletegate(this)); +} + +void YACReaderFoldersView::dragEnterEvent(QDragEnterEvent *event) +{ + YACReaderTreeView::dragEnterEvent(event); + + QList urlList; + + if (event->mimeData()->hasUrls() && event->dropAction() == Qt::CopyAction) + { + urlList = event->mimeData()->urls(); + QString currentPath; + foreach (QUrl url, urlList) + { + //comics or folders are accepted, folders' content is validate in dropEvent (avoid any lag before droping) + currentPath = url.toLocalFile(); + if(Comic::fileIsComic(currentPath) || QFileInfo(currentPath).isDir()) + { + event->acceptProposedAction(); + return; + } + } + } +} + +void YACReaderFoldersView::dragLeaveEvent(QDragLeaveEvent *event) +{ + YACReaderTreeView::dragLeaveEvent(event); +} + +void YACReaderFoldersView::dragMoveEvent(QDragMoveEvent *event) +{ + YACReaderTreeView::dragMoveEvent(event); + event->acceptProposedAction(); +} + +void YACReaderFoldersView::dropEvent(QDropEvent *event) +{ + YACReaderTreeView::dropEvent(event); + + QLOG_DEBUG() << "drop on tree" << event->dropAction(); + + bool validAction = event->dropAction() == Qt::CopyAction; // || event->dropAction() & Qt::MoveAction; TODO move + + if(validAction) + { + QList > droppedFiles = ComicFilesManager::getDroppedFiles(event->mimeData()->urls()); + QModelIndex destinationIndex = indexAt(event->pos()); + + if(event->dropAction() == Qt::CopyAction) + { + QLOG_DEBUG() << "copy - tree :" << droppedFiles; + emit copyComicsToFolder(droppedFiles, destinationIndex); + } + else if(event->dropAction() & Qt::MoveAction) + { + QLOG_DEBUG() << "move - tree :" << droppedFiles; + emit moveComicsToFolder(droppedFiles, destinationIndex); + } + + event->acceptProposedAction(); + } +} + +//---------------------------------------------------------- + +YACReaderFoldersViewItemDeletegate::YACReaderFoldersViewItemDeletegate(QObject *parent) + :QStyledItemDelegate(parent) +{ + +} + +void YACReaderFoldersViewItemDeletegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const +{ + if(!index.data(FolderModel::CompletedRole).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/YACReaderLibrary/yacreader_folders_view.h b/YACReaderLibrary/yacreader_folders_view.h new file mode 100644 index 00000000..07d8091c --- /dev/null +++ b/YACReaderLibrary/yacreader_folders_view.h @@ -0,0 +1,36 @@ +#ifndef YACREADER_FOLDERS_VIEW_H +#define YACREADER_FOLDERS_VIEW_H + +#include "yacreader_treeview.h" + +#include + +class YACReaderFoldersView : public YACReaderTreeView +{ + Q_OBJECT +public: + explicit YACReaderFoldersView(QWidget * parent = 0); + +signals: + //Drops + void copyComicsToFolder(QList >,QModelIndex); + void moveComicsToFolder(QList >,QModelIndex); + +protected: + //Drop to import + void dragEnterEvent(QDragEnterEvent *event); + void dragLeaveEvent(QDragLeaveEvent *event); + void dragMoveEvent(QDragMoveEvent *event); + void dropEvent(QDropEvent *event); +}; + +class YACReaderFoldersViewItemDeletegate: public QStyledItemDelegate +{ + Q_OBJECT +public: + explicit YACReaderFoldersViewItemDeletegate(QObject *parent = 0); + void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const; +}; + + +#endif // YACREADER_FOLDERS_VIEW_H diff --git a/YACReaderLibrary/yacreader_history_controller.cpp b/YACReaderLibrary/yacreader_history_controller.cpp new file mode 100644 index 00000000..cfd82ca8 --- /dev/null +++ b/YACReaderLibrary/yacreader_history_controller.cpp @@ -0,0 +1,108 @@ +#include "yacreader_history_controller.h" + +YACReaderHistoryController::YACReaderHistoryController(QObject *parent) : + QObject(parent) +{ +} + +void YACReaderHistoryController::clear() +{ + currentFolderNavigation = 0; + history.clear(); + history.append(YACReaderLibrarySourceContainer(QModelIndex(),YACReaderLibrarySourceContainer::Folder)); //root folder is always the first item + + emit(enabledBackward(false)); + emit(enabledForward(false)); +} + +void YACReaderHistoryController::backward() +{ + if(currentFolderNavigation>0) + { + currentFolderNavigation--; + emit(modelIndexSelected(history.at(currentFolderNavigation))); + emit(enabledForward(true)); + } + + if(currentFolderNavigation==0) + emit(enabledBackward(false)); +} + +void YACReaderHistoryController::forward() +{ + if(currentFolderNavigation0) + { + numElementsToRemove--; + history.removeLast(); + } + + if(source!=history.at(currentFolderNavigation)) + { + history.append(source); + + emit(enabledBackward(true)); + currentFolderNavigation++; + } + + emit(enabledForward(false)); +} + +YACReaderLibrarySourceContainer YACReaderHistoryController::lastSourceContainer() +{ + return history.last(); +} + +YACReaderLibrarySourceContainer YACReaderHistoryController::currentSourceContainer() +{ + return history.at(currentFolderNavigation); +} + +//------------------------------------------------------------------------------ + +YACReaderLibrarySourceContainer::YACReaderLibrarySourceContainer() + :sourceModelIndex(QModelIndex()),type(None) +{ + +} + +YACReaderLibrarySourceContainer::YACReaderLibrarySourceContainer(const QModelIndex &sourceModelIndex, YACReaderLibrarySourceContainer::SourceType type) + :sourceModelIndex(sourceModelIndex),type(type) +{} + +QModelIndex YACReaderLibrarySourceContainer::getSourceModelIndex() const +{ + return sourceModelIndex; +} + +YACReaderLibrarySourceContainer::SourceType YACReaderLibrarySourceContainer::getType() const +{ + return type; +} + +bool YACReaderLibrarySourceContainer::operator==(const YACReaderLibrarySourceContainer &other) const +{ + return sourceModelIndex == other.sourceModelIndex && type == other.type; +} + +bool YACReaderLibrarySourceContainer::operator!=(const YACReaderLibrarySourceContainer &other) const +{ + return !(*this == other); +} diff --git a/YACReaderLibrary/yacreader_history_controller.h b/YACReaderLibrary/yacreader_history_controller.h new file mode 100644 index 00000000..25e4b8fd --- /dev/null +++ b/YACReaderLibrary/yacreader_history_controller.h @@ -0,0 +1,62 @@ +#ifndef YACREADER_HISTORY_CONTROLLER_H +#define YACREADER_HISTORY_CONTROLLER_H + +#include + +#include + +class YACReaderHistoryController; + +class YACReaderLibrarySourceContainer +{ +public: + enum SourceType { + None, + Folder, + List + }; + + explicit YACReaderLibrarySourceContainer(); + explicit YACReaderLibrarySourceContainer(const QModelIndex & sourceModelIndex, YACReaderLibrarySourceContainer::SourceType type); + QModelIndex getSourceModelIndex() const; + YACReaderLibrarySourceContainer::SourceType getType() const; + + bool operator==(const YACReaderLibrarySourceContainer& other) const; + bool operator!=(const YACReaderLibrarySourceContainer& other) const; + +protected: + QModelIndex sourceModelIndex; + YACReaderLibrarySourceContainer::SourceType type; + + friend class YACReaderHistoryController; + +}; + +Q_DECLARE_METATYPE(YACReaderLibrarySourceContainer) + +class YACReaderHistoryController : public QObject +{ + Q_OBJECT +public: + explicit YACReaderHistoryController(QObject *parent = 0); + +signals: + void enabledForward(bool enabled); + void enabledBackward(bool enabled); + void modelIndexSelected(YACReaderLibrarySourceContainer); + +public slots: + void clear(); + void backward(); + void forward(); + void updateHistory(const YACReaderLibrarySourceContainer & source); + YACReaderLibrarySourceContainer lastSourceContainer(); + YACReaderLibrarySourceContainer currentSourceContainer(); + +protected: + int currentFolderNavigation; + QList history; + +}; + +#endif // YACREADER_HISTORY_CONTROLLER_H 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..0e55078d --- /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(libraryId, comic.id); + siblings = DBHelper::getSiblings(libraryId, comic.parentId); +} + +void YACReaderClientConnectionWorker::updateComic(quint64 libraryId, ComicDB & comic) +{ + QMutexLocker locker(&dbMutex); + DBHelper::update(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..e4562abd --- /dev/null +++ b/YACReaderLibrary/yacreader_main_toolbar.cpp @@ -0,0 +1,151 @@ +#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,0,Qt::AlignVCenter); + addDivider(); + mainLayout->addWidget(forwardButton,0,Qt::AlignVCenter); + + mainLayout->addSpacing(34); + mainLayout->addWidget(settingsButton,0,Qt::AlignVCenter); + addWideDivider(); + mainLayout->addWidget(serverButton,0,Qt::AlignVCenter); + addWideDivider(); + mainLayout->addWidget(helpButton,0,Qt::AlignVCenter); + + mainLayout->addStretch(); + + mainLayout->addWidget(toggleComicsViewButton,0,Qt::AlignVCenter); + addWideDivider(); + mainLayout->addWidget(fullscreenButton,0,Qt::AlignVCenter); + + setLayout(mainLayout); + + setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Fixed); +} + + +QSize YACReaderMainToolBar::sizeHint() const +{ + return QSize(200,40); +} + +void YACReaderMainToolBar::setSearchWidget(QWidget *w) +{ + addWideDivider(); + mainLayout->addWidget(w,0,Qt::AlignVCenter); +} + +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,0,Qt::AlignVCenter); + 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..b8ed359e --- /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 setSearchWidget(QWidget * w); + 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/yacreader_navigation_controller.cpp b/YACReaderLibrary/yacreader_navigation_controller.cpp new file mode 100644 index 00000000..3996932b --- /dev/null +++ b/YACReaderLibrary/yacreader_navigation_controller.cpp @@ -0,0 +1,304 @@ +#include "yacreader_navigation_controller.h" + +#include + +#include "library_window.h" +#include "yacreader_folders_view.h" +#include "yacreader_reading_lists_view.h" +#include "folder_item.h" +#include "yacreader_history_controller.h" +#include "comic_model.h" +#include "folder_model.h" +#include "reading_list_model.h" +#include "comics_view.h" +#include "empty_folder_widget.h" +#include "yacreader_search_line_edit.h" +#include "yacreader_global.h" +#include "empty_label_widget.h" +#include "empty_special_list.h" + +#include "QsLog.h" + +YACReaderNavigationController::YACReaderNavigationController(LibraryWindow *parent) : + QObject(parent),libraryWindow(parent) +{ + setupConnections(); +} + +void YACReaderNavigationController::selectedFolder(const QModelIndex &mi) +{ + //A proxy is used + QModelIndex modelIndex = libraryWindow->foldersModelProxy->mapToSource(mi); + + //update history + libraryWindow->historyController->updateHistory(YACReaderLibrarySourceContainer(modelIndex, YACReaderLibrarySourceContainer::Folder)); + + if(libraryWindow->status == LibraryWindow::Searching) + { + //when a folder is selected the search mode has to be reset + libraryWindow->searchEdit->clearText(); + libraryWindow->clearSearchFilter(); + libraryWindow->foldersView->scrollTo(mi,QAbstractItemView::PositionAtTop); + libraryWindow->foldersView->setCurrentIndex(mi); + } + + loadFolderInfo(modelIndex); + + libraryWindow->setToolbarTitle(modelIndex); +} + +void YACReaderNavigationController::reselectCurrentFolder() +{ + selectedFolder(libraryWindow->foldersView->currentIndex()); +} + +void YACReaderNavigationController::loadFolderInfo(const QModelIndex &modelIndex) +{ + //Get FolderItem + qulonglong folderId = folderModelIndexToID(modelIndex); + + //check comics in folder with id = folderId + libraryWindow->comicsModel->setupFolderModelData(folderId,libraryWindow->foldersModel->getDatabase()); + libraryWindow->comicsView->setModel(libraryWindow->comicsModel); + + //configure views + if(libraryWindow->comicsModel->rowCount() > 0) + { + //updateView + libraryWindow->showComicsView(); + libraryWindow->disableComicsActions(false); + } + else{ + //showEmptyFolder + loadEmptyFolderInfo(modelIndex); + libraryWindow->showEmptyFolderView(); + libraryWindow->disableComicsActions(true); + } + + //libraryWindow->updateFoldersViewConextMenu(modelIndex); + + //if a folder is selected, listsView selection must be cleared + libraryWindow->listsView->clearSelection(); +} + +void YACReaderNavigationController::loadListInfo(const QModelIndex &modelIndex) +{ + switch(modelIndex.data(ReadingListModel::TypeListsRole).toInt()) + { + case ReadingListModel::SpecialList: + loadSpecialListInfo(modelIndex); + break; + + case ReadingListModel::Label: + loadLabelInfo(modelIndex); + break; + + case ReadingListModel::ReadingList: + loadReadingListInfo(modelIndex); + break; + } + + //if a list is selected, foldersView selection must be cleared + libraryWindow->foldersView->clearSelection(); +} + +void YACReaderNavigationController::loadSpecialListInfo(const QModelIndex &modelIndex) +{ + ReadingListModel::TypeSpecialList type = (ReadingListModel::TypeSpecialList)modelIndex.data(ReadingListModel::SpecialListTypeRole).toInt(); + + switch(type) + { + case ReadingListModel::Favorites: + libraryWindow->comicsModel->setupFavoritesModelData(libraryWindow->foldersModel->getDatabase()); + break; + case ReadingListModel::Reading: + libraryWindow->comicsModel->setupReadingModelData(libraryWindow->foldersModel->getDatabase()); + break; + } + + libraryWindow->comicsView->setModel(libraryWindow->comicsModel); + + if(libraryWindow->comicsModel->rowCount() > 0) + { + libraryWindow->showComicsView(); + libraryWindow->disableComicsActions(false); + } + else + { + //setup empty special list widget + switch(type) + { + case ReadingListModel::Favorites: + libraryWindow->emptySpecialList->setPixmap(QPixmap(":/images/empty_favorites.png")); + libraryWindow->emptySpecialList->setText(tr("No favorites")); + break; + case ReadingListModel::Reading: + libraryWindow->emptySpecialList->setPixmap(QPixmap(":/images/empty_current_readings.png")); + libraryWindow->emptySpecialList->setText(tr("You are not reading anything yet, come on!!")); + break; + } + + libraryWindow->showEmptySpecialList(); + libraryWindow->disableComicsActions(true); + } +} + +void YACReaderNavigationController::loadLabelInfo(const QModelIndex &modelIndex) +{ + qulonglong id = modelIndex.data(ReadingListModel::IDRole).toULongLong(); + //check comics in label with id = id + libraryWindow->comicsModel->setupLabelModelData(id,libraryWindow->foldersModel->getDatabase()); + libraryWindow->comicsView->setModel(libraryWindow->comicsModel); + + //configure views + if(libraryWindow->comicsModel->rowCount() > 0) + { + //updateView + libraryWindow->showComicsView(); + libraryWindow->disableComicsActions(false); + } + else{ + //showEmptyFolder + //loadEmptyLabelInfo(); //there is no info in an empty label by now, TODO design something + libraryWindow->emptyLabelWidget->setColor((YACReader::LabelColors)modelIndex.data(ReadingListModel::LabelColorRole).toInt()); + libraryWindow->showEmptyLabelView(); + libraryWindow->disableComicsActions(true); + } +} + +void YACReaderNavigationController::loadReadingListInfo(const QModelIndex &modelIndex) +{ + qulonglong id = modelIndex.data(ReadingListModel::IDRole).toULongLong(); + //check comics in label with id = id + libraryWindow->comicsModel->setupReadingListModelData(id,libraryWindow->foldersModel->getDatabase()); + libraryWindow->comicsView->setModel(libraryWindow->comicsModel); + + //configure views + if(libraryWindow->comicsModel->rowCount() > 0) + { + //updateView + libraryWindow->showComicsView(); + libraryWindow->disableComicsActions(false); + } + else{ + libraryWindow->showEmptyReadingListWidget(); + libraryWindow->disableComicsActions(true); + } +} + +void YACReaderNavigationController::selectedList(const QModelIndex &mi) +{ + //A proxy is used + QModelIndex modelIndex = libraryWindow->listsModelProxy->mapToSource(mi); + + //update history + libraryWindow->historyController->updateHistory(YACReaderLibrarySourceContainer(modelIndex,YACReaderLibrarySourceContainer::List)); + + if(libraryWindow->status == LibraryWindow::Searching) + { + //when a list is selected the search mode has to be reset + libraryWindow->searchEdit->clearText(); + libraryWindow->clearSearchFilter(); + libraryWindow->listsView->scrollTo(mi,QAbstractItemView::PositionAtTop); + libraryWindow->listsView->setCurrentIndex(mi); + } + + loadListInfo(modelIndex); + + libraryWindow->setToolbarTitle(modelIndex); +} + +void YACReaderNavigationController::reselectCurrentList() +{ + selectedList(libraryWindow->listsView->currentIndex()); +} + +void YACReaderNavigationController::reselectCurrentSource() +{ + if(!libraryWindow->listsView->selectionModel()->selectedRows().isEmpty()) + { + reselectCurrentList(); + }else + { + reselectCurrentFolder(); + } +} + +void YACReaderNavigationController::selectedIndexFromHistory(const YACReaderLibrarySourceContainer &sourceContainer) +{ + //TODO NO searching allowed, just disable backward/forward actions in searching mode + if(libraryWindow->status == LibraryWindow::Searching) + { + //when a folder is selected the search mode has to be reset + libraryWindow->searchEdit->clearText(); + libraryWindow->clearSearchFilter(); + } + + loadIndexFromHistory(sourceContainer); +} + +void YACReaderNavigationController::loadIndexFromHistory(const YACReaderLibrarySourceContainer &sourceContainer) +{ + QModelIndex sourceMI = sourceContainer.getSourceModelIndex(); + switch(sourceContainer.getType()) + { + case YACReaderLibrarySourceContainer::Folder: + { + QModelIndex mi = libraryWindow->foldersModelProxy->mapFromSource(sourceMI); + libraryWindow->foldersView->scrollTo(mi,QAbstractItemView::PositionAtTop); + libraryWindow->foldersView->setCurrentIndex(mi); + loadFolderInfo(sourceMI); + break; + } + case YACReaderLibrarySourceContainer::List: + { + QModelIndex mi = libraryWindow->listsModelProxy->mapFromSource(sourceMI); + libraryWindow->listsView->scrollTo(mi,QAbstractItemView::PositionAtTop); + libraryWindow->listsView->setCurrentIndex(mi); + loadListInfo(sourceMI); + break; + } + } +} + +void YACReaderNavigationController::selectSubfolder(const QModelIndex &sourceMIParent, int child) +{ + QModelIndex dest = libraryWindow->foldersModel->index(child,0,sourceMIParent); + libraryWindow->foldersView->setCurrentIndex(libraryWindow->foldersModelProxy->mapFromSource(dest)); + libraryWindow->historyController->updateHistory(YACReaderLibrarySourceContainer(dest,YACReaderLibrarySourceContainer::Folder)); + loadFolderInfo(dest); +} + +void YACReaderNavigationController::loadEmptyFolderInfo(const QModelIndex &modelIndex) +{ + QStringList subfolders; + subfolders = libraryWindow->foldersModel->getSubfoldersNames(modelIndex); + libraryWindow->emptyFolderWidget->setSubfolders(modelIndex,subfolders); +} + +void YACReaderNavigationController::loadPreviousStatus() +{ + YACReaderLibrarySourceContainer sourceContainer = libraryWindow->historyController->currentSourceContainer(); + loadIndexFromHistory(sourceContainer); +} + +void YACReaderNavigationController::setupConnections() +{ + connect(libraryWindow->foldersView,SIGNAL(clicked(QModelIndex)),this,SLOT(selectedFolder(QModelIndex))); + connect(libraryWindow->listsView,SIGNAL(clicked(QModelIndex)),this,SLOT(selectedList(QModelIndex))); + connect(libraryWindow->historyController,SIGNAL(modelIndexSelected(YACReaderLibrarySourceContainer)),this,SLOT(selectedIndexFromHistory(YACReaderLibrarySourceContainer))); + connect(libraryWindow->emptyFolderWidget,SIGNAL(subfolderSelected(QModelIndex,int)),this,SLOT(selectSubfolder(QModelIndex,int))); + connect(libraryWindow->comicsModel,SIGNAL(isEmpty()),this,SLOT(reselectCurrentSource())); +} + +qulonglong YACReaderNavigationController::folderModelIndexToID(const QModelIndex &mi) +{ + if(!mi.isValid()) + return 1; + + FolderItem * folderItem = static_cast(mi.internalPointer()); + if(folderItem != 0) + return folderItem->id; + + return 1; +} diff --git a/YACReaderLibrary/yacreader_navigation_controller.h b/YACReaderLibrary/yacreader_navigation_controller.h new file mode 100644 index 00000000..2dbeac28 --- /dev/null +++ b/YACReaderLibrary/yacreader_navigation_controller.h @@ -0,0 +1,53 @@ +#ifndef YACREADER_NAVIGATION_CONTROLLER_H +#define YACREADER_NAVIGATION_CONTROLLER_H + +#include +class LibraryWindow; +class YACReaderLibrarySourceContainer; + +class YACReaderNavigationController : public QObject +{ + Q_OBJECT +public: + + explicit YACReaderNavigationController(LibraryWindow * parent); + +signals: + +public slots: + //info origins + //folders view + void selectedFolder(const QModelIndex & mi); + void reselectCurrentFolder(); + //reading lists + void selectedList(const QModelIndex & mi); + void reselectCurrentList(); + + void reselectCurrentSource(); + + //history navigation + void selectedIndexFromHistory(const YACReaderLibrarySourceContainer &sourceContainer); + void loadIndexFromHistory(const YACReaderLibrarySourceContainer &sourceContainer); + //empty subfolder + void selectSubfolder(const QModelIndex &sourceMI, int child); + + void loadEmptyFolderInfo(const QModelIndex & modelIndex); + + void loadFolderInfo(const QModelIndex & modelIndex); + void loadListInfo(const QModelIndex & modelIndex); + void loadSpecialListInfo(const QModelIndex & modelIndex); + void loadLabelInfo(const QModelIndex & modelIndex); + void loadReadingListInfo(const QModelIndex & modelIndex); + + void loadPreviousStatus(); + +private: + + void setupConnections(); + LibraryWindow * libraryWindow; + + //convenience methods + qulonglong folderModelIndexToID(const QModelIndex & mi); +}; + +#endif // YACREADER_NAVIGATION_CONTROLLER_H diff --git a/YACReaderLibrary/yacreader_reading_lists_view.cpp b/YACReaderLibrary/yacreader_reading_lists_view.cpp new file mode 100644 index 00000000..b95a567e --- /dev/null +++ b/YACReaderLibrary/yacreader_reading_lists_view.cpp @@ -0,0 +1,72 @@ +#include "yacreader_reading_lists_view.h" + +#include "reading_list_item.h" +#include "reading_list_model.h" + +YACReaderReadingListsView::YACReaderReadingListsView(QWidget *parent) + :YACReaderTreeView(parent) +{ + setItemDelegate(new YACReaderReadingListsViewItemDeletegate(this)); + setUniformRowHeights(false); + + //enabling internal drag&drop + setDragDropMode(QAbstractItemView::DragDrop); +} + +void YACReaderReadingListsView::dragEnterEvent(QDragEnterEvent *event) +{ + YACReaderTreeView::dragEnterEvent(event); + + /*QModelIndex destinationIndex = indexAt(event->pos()); + if(model()->canDropMimeData(event->mimeData(), event->proposedAction(), destinationIndex.row(), destinationIndex.column(), destinationIndex.parent()))*/ + event->acceptProposedAction(); +} + +void YACReaderReadingListsView::dragMoveEvent(QDragMoveEvent *event) +{ + YACReaderTreeView::dragMoveEvent(event); + QModelIndex destinationIndex = indexAt(event->pos()); + if(model()->canDropMimeData(event->mimeData(), event->proposedAction(), destinationIndex.row(), destinationIndex.column(), destinationIndex.parent())) + event->acceptProposedAction(); +} + +void YACReaderReadingListsView::dropEvent(QDropEvent *event) +{ + YACReaderTreeView::dropEvent(event); + + +} + +//---------------------------------------------------------------------- + +YACReaderReadingListsViewItemDeletegate::YACReaderReadingListsViewItemDeletegate(QObject *parent) + :QStyledItemDelegate(parent) +{ + +} + +void YACReaderReadingListsViewItemDeletegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const +{ + ReadingListModel::TypeList typeList = (ReadingListModel::TypeList)index.data(ReadingListModel::TypeListsRole).toInt(); + + if(typeList == ReadingListModel::Separator) + { + return; + } + + QStyledItemDelegate::paint(painter, option, index); +} + +QSize YACReaderReadingListsViewItemDeletegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const +{ + ReadingListModel::TypeList typeList = (ReadingListModel::TypeList)index.data(ReadingListModel::TypeListsRole).toInt(); + + if(typeList == ReadingListModel::Separator) + { + QSize newSize = QStyledItemDelegate::sizeHint(option, index); + newSize.setHeight(7); + return newSize; + } + + return QStyledItemDelegate::sizeHint(option, index); +} diff --git a/YACReaderLibrary/yacreader_reading_lists_view.h b/YACReaderLibrary/yacreader_reading_lists_view.h new file mode 100644 index 00000000..31cb099b --- /dev/null +++ b/YACReaderLibrary/yacreader_reading_lists_view.h @@ -0,0 +1,32 @@ +#ifndef YACREADER_READING_LISTS_VIEW_H +#define YACREADER_READING_LISTS_VIEW_H + +#include "yacreader_treeview.h" + +#include + +class YACReaderReadingListsView : public YACReaderTreeView +{ + Q_OBJECT +public: + explicit YACReaderReadingListsView(QWidget * parent = 0); + +protected: + //Drop to import & internal Drag&Drop for resorting + void dragEnterEvent(QDragEnterEvent *event); + void dragMoveEvent(QDragMoveEvent *event); + void dropEvent(QDropEvent *event); + +}; + +class YACReaderReadingListsViewItemDeletegate: public QStyledItemDelegate +{ + Q_OBJECT +public: + explicit YACReaderReadingListsViewItemDeletegate(QObject *parent = 0); + void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const; + QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const; +}; + + +#endif // YACREADER_READING_LISTS_VIEW_H diff --git a/YACReaderLibrary/yacreaderlibrary_de.ts b/YACReaderLibrary/yacreaderlibrary_de.ts new file mode 100644 index 00000000..60eb37a0 --- /dev/null +++ b/YACReaderLibrary/yacreaderlibrary_de.ts @@ -0,0 +1,1520 @@ + + + + + AddLibraryDialog + + + Comics folder : + Comics Ordner : + + + + Library Name : + Bibliothek Name : + + + + Add + Hinzufügen + + + + Cancel + Cancel + + + + Add an existing library + Eine existierende Bibliothek hinzufügen + + + + ComicVineDialog + + + skip + überspringen + + + + back + zurück + + + + next + nächste + + + + search + suche + + + + close + schliessen + + + + + + + + Looking for volume... + Suche nach Band.... + + + + + comic %1 of %2 - %3 + Comic %1 von %2 - %3 + + + + %1 comics selected + %1 Comic ausgewählt + + + + Error connecting to ComicVine + Fehler bei Verbindung zu ComicVine + + + + unknown error + unbekannter Fehler + + + + + Retrieving tags for : %1 + Runterladen von Tags für : %1 + + + + Retrieving volume info... + Runterladen von Ausgabe Info... + + + + Looking for comic... + Suche nach Comic... + + + + CreateLibraryDialog + + + Comics folder : + Comics Ordner : + + + + Library Name : + Bibliothek Name : + + + + Create + Neu erzeugen + + + + Cancel + Abbrechen + + + + Create a library could take several minutes. You can stop the process and update the library later for completing the task. + Eine neue Bibliothek erzeugen kann einige Minuten dauern. Sie können den Prozess abbrechen und die Bibliothek später updaten um den Prozess zu vervollständigen. + + + + Create new library + Kreiere eine neue Bibliothek + + + + Path not found + Pfad nicht gefunden + + + + The selected path does not exist or is not a valid path. Be sure that you have write access to this folder + Der gewählte Pfad existiert nicht oder ist kein gültiger Pfad. Stellen Sie sicher, dass Sie Schreibzugriff zu dem Ordner haben + + + + ExportComicsInfoDialog + + + Output file : + Ziel File : + + + + Create + Neu erzeugen + + + + Cancel + Abbrechen + + + + Export comics info + Export Comic Info + + + + Destination database name + Ziel Datenbasis Name + + + + Problem found while writing + Problem gefunden beim Schreiben + + + + 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 + Der gewählte Pfad existiert nicht oder ist kein gültiger Pfad. Stellen Sie sicher, dass Sie Schreibzugriff zu dem Ordner haben + + + + ExportLibraryDialog + + + Output folder : + Ziel Ordner : + + + + Create + Erzeuge + + + + Cancel + Abbrechen + + + + Create covers package + Erzeuge Titelbild Paket + + + + Problem found while writing + Problem gefunden beim Schreiben + + + + 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 + Der gewählte Pfad existiert nicht oder ist kein gültiger Pfad. Stellen Sie sicher, dass Sie Schreibzugriff zu dem Ordner haben + + + + Destination directory + Ziel Verzeichnis + + + + FileComic + + + CRC error on page (%1): some of the pages will not be displayed correctly + CRC Fehler auf Seite (%1): einige Seiten werden nicht korrekt dargestellt werden + + + + Unknown error opening the file + Unbekannter Fehler beim Öffnen des Files + + + + 7z not found + 7z nicht gefunden + + + + Format not supported + Format wird nicht unterstützt + + + + HelpAboutDialog + + + About + Über + + + + Help + Hilfe + + + + ImportComicsInfoDialog + + + Import comics info + Importiere Comic Info + + + + Info database location : + Info Datenbasis Speicherort : + + + + Import + Importiere + + + + Cancel + Abbrechen + + + + Comics info file (*.ydb) + Comics Info File (*.ydb) + + + + ImportLibraryDialog + + + Library Name : + Bibliothek Name : + + + + Package location : + Paket Ort : + + + + Destination folder : + Zielordner : + + + + Unpack + Entpacken + + + + Cancel + Abbrechen + + + + Extract a catalog + Einen Katalog Extrahieren + + + + Compresed library covers (*.clc) + Komprimierte Bibliotheks Bilder (*.clc) + + + + ImportWidget + + + stop + Stop + + + + Some of the comics being added... + Einige der Comics werden hinzugefügt... + + + + Importing comics + Comics werden importiert + + + + <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 kreiert nun eine neue Bibliothek. </p><p>Eine neue Bibliothek erzeugen kann einige Minuten dauern. Sie können den Prozess stoppen und die Bibliothek später aktualisieren um den Prozess zu vervollständigen.</p> + + + + Updating the library + Aktualisierung der Bibliothek + + + + <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>Die gerade benutzte Bibliothek wird aktualisiert. Für eine schnellere Aktualisierung aktualisieren Sie bitte die Bibliothek regelmäßig.</p><p>Sie können den Prozess abbrechen und mit der Aktualisierung später fortfahren.<p> + + + + LibraryWindow + + + YACReader Library + YACReader Bibliothek + + + + + Library + Bibliothek + + + + <font color='white'> press 'F' to close fullscreen mode </font> + <font color='white'> drücke 'F' um Vollbildmodus zu schließen </font> + + + + Create a new library + Neue Bibliothek erzeugen + + + + Open an existing library + Eine existierende Bibliothek öffnen + + + + + Export comics info + Export Comics Info + + + + + Import comics info + Import Comics Info + + + + Pack covers + Titelbild Paket erzeugen + + + + Pack the covers of the selected library + Packe die Titelbilder der ausgewählten Bibliothek in ein Paket + + + + Unpack covers + Titelbilder entpacken + + + + Unpack a catalog + Katalog entpacken + + + + Update library + Bibliothek updaten + + + + Update current library + Aktuelle Bibliothek updaten + + + + Rename library + Bibliothek umbenennen + + + + Rename current library + Aktuelle Bibliothek umbenennen + + + + Remove library + Bibliothek entfernen + + + + Remove current library from your collection + Aktuelle Bibliothek aus der Sammlung entfernen + + + + Open current comic + Aktuellen Comic öffnen + + + + Open current comic on YACReader + Aktuellen Comic mit YACReader öffnen + + + + + Set as read + Als gelesen markieren + + + + Set comic as read + Comic als gelesen markieren + + + + + Set as unread + Als ungelesen markieren + + + + Set comic as unread + Comic als ungelesen markieren + + + + Show/Hide marks + Zeige/Verstecke Markierungen + + + + Show or hide readed marks + Zeige oder verstecke gelesene Markierungen + + + + Library not available + Library ' + Bibliothek nicht verfügbar + + + + Fullscreen mode on/off + Vollbildmodus an/aus + + + + Fullscreen mode on/off (F) + Vollbildmodus an/aus (F) + + + + Help, About YACReader + Hilfe, Über YACReader + + + + Select root node + Root Knoten auswählen + + + + + + + + + + + Expand all nodes + Unterordner anzeigen + + + + - + + - + + + + Colapse all nodes + Unterordner verstecken + + + + Show options dialog + Zeige den Optionen Dialog + + + + Show comics server options dialog + Zeige den Comics Optionen Dialog + + + + Open folder... + Öffne Ordner... + + + + Set as uncompleted + Als nicht gelesen markieren + + + + Set as completed + Als gelesen markieren + + + + Open containing folder... + Öffne aktuellen Ordner... + + + + Reset comic rating + Comic Bewertung zurücksetzen + + + + Select all comics + Alle Comics auswählen + + + + Edit + Editieren + + + + Asign current order to comics + Bestimme die Abfolge der Comics + + + + Update cover + Titelbild updaten + + + + Delete selected comics + Ausgewählte Comics löschen + + + + Hide comic flow + Comic "Flow" verstecken + + + + Download tags from Comic Vine + Tags von Comic Vine herunterladen + + + + Folder + Ordner + + + + Comic + Comic + + + + Update needed + Update benötigt + + + + This library was created with a previous version of YACReaderLibrary. It needs to be updated. Update now? + Diese Bibliothek wurde mit einer vorherigen Version von YACReader erzeugt. Sie muss geupdated werden. Jetzt updaten? + + + + Update failed + Update fehlgeschlagen + + + + The current library can't be udpated. Check for write write permissions on: + Die aktuelle Bibliothek kann nicht geupdated werden. Überprüfen Sie die Schreibrechte auf: + + + + Download new version + Neue Version herunterladen + + + + This library was created with a newer version of YACReaderLibrary. Download the new version now? + Die Bibliothek wurde mit einer neueren Version von YACReader erzeugt. Die neue Version jetzt herunterladen? + + + + Library '%1' is no longer available. Do you want to remove it? + Bibliothek '%1' ist nicht länger verfügbar. Wollen Sie sie entfernen? + + + + Old library + Alte Bibliothek + + + + Library '%1' has been created with an older version of YACReaderLibrary. It must be created again. Do you want to create the library now? + Bibliothek '%1' ist mit einer älteren Version von YACREader erzeugt worden. Sie muss neu erzeugt werden. Wollen Sie die Bibliothek jetzt erzeugen? + + + + YACReader not found + YACReader nicht gefunden + + + + YACReader not found, YACReader should be installed in the same folder as YACReaderLibrary. + YACReader nicht gefunden. YACReader sollte in demselben Ordner installiert werden wie YACReaderLibrary. + + + + Library not found + Bibliothek nicht gefunden + + + + The selected folder doesn't contain any library. + Der ausgewählte Ordner enthält keine Bibliothek. + + + + Are you sure? + Sind Sie sicher? + + + + Do you want remove + Möchten Sie entfernen + + + + library? + die Bibliothek? + + + + Remove and delete metadata + Entferne und lösche Metadaten + + + + Unable to delete + Löschen nicht möglich + + + + There was an issue trying to delete the selected comics. Please, check for write permissions in the selected files or containing folder. + Es gab ein Problem beim löschen der ausgewählten Comics. Überprüfen Sie bitte die Schreibberechtigung für die ausgewählten Files oder Ordner. + + + + Asign comics numbers + Comic Nummer setzen + + + + Asign numbers starting in: + Nummern setzen angefangen mit: + + + + Error creating the library + Fehler beim Erzeugen der Bibliothek + + + + Error updating the library + Fehler beim Updaten der Bibliothek + + + + Error opening the library + Fehler beim Öffnen der Bibliothek + + + + Delete comics + Comics löschen + + + + All the selected comics will be deleted from your disk. Are you sure? + Alle ausgewählten Comics werden von Ihrer Festplatte gelöscht. Sind Sie sicher? + + + + Library name already exists + Bibliothek Name existiert bereits + + + + There is another library with the name '%1'. + Es gibt eine andere Bibliothek mit dem Namen '%1'. + + + + LocalComicListModel + + + file name + File Name + + + + NoLibrariesWidget + + + You don't have any librarires yet + Sie haben im Augenblick keine Bibliothek + + + + <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>Sie können eine Bibliothek in einem bliebigen Ordner erzeugen, YACReaderLibrary wird alle Comics und Unterordner von diesem Ordner importieren. Wenn Sie in der Vergangenheit eine Bibliothek erzeugt haben, können Sie sie öffnen.</p><p>Vergessen Sie nicht, Sie können YACReader als unabhängige Anwendung benutzen, um Comics auf Ihrem Computer zu lesen.</p> + + + + create your first library + Erzeugen Sie Ihre erste Bibliothek + + + + add an existing one + Fügen Sie eine existierende hinzu + + + + OptionsDialog + + + Options + Optionen + + + + PropertiesDialog + + + General info + Generelle Info + + + + Authors + Autoren + + + + Publishing + Publishing + + + + Plot + Inhalt + + + + Cover page + Titelbild + + + + Title: + Titel: + + + + Issue number: + Ausgabe Nummer: + + + + Volume: + Band: + + + + Story arc: + Handlung: + + + + Genere: + Genre: + + + + Size: + Größe: + + + + Writer(s): + Author(en): + + + + Penciller(s): + Zeichner: + + + + Inker(s): + Tinte: + + + + Colorist(s): + Farbe: + + + + Letterer(s): + Schrift: + + + + Cover Artist(s): + Titelbild Künstler: + + + + Day: + Tag: + + + + Month: + Monat: + + + + Year: + + + + + Publisher: + Verlag: + + + + Format: + Format: + + + + Color/BW: + Farbe/BW: + + + + Age rating: + Alterhinweis: + + + + Synopsis: + Übersicht: + + + + Characters: + Charaktere: + + + + Notes: + Notizen: + + + + 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 + Nicht gefunden + + + + Comic not found. You should update your library. + Comic nicht gefunden. Sie sollten Ihre Bibliothek updaten. + + + + Edit selected comics information + Ausgewählte Comic Informationen editieren + + + + Edit comic information + Comic Informationen editieren + + + + QObject + + + 7z lib not found + 7z Bibliothek nicht gefunden + + + + unable to load 7z lib from ./utils + 7z Bibliothek kann von ./utils nicht geladen werden + + + + RenameLibraryDialog + + + New Library Name : + Neuer Bibliotheks Name : + + + + Rename + Namen ändern + + + + Cancel + Abbrechen + + + + Rename current library + Namen der Bibliothek ändern + + + + ScraperResultsPaginator + + + Number of volumes found : %1 + Anzahl der gefundenen Bände: %1 + + + + + page %1 of %2 + Seite %1 von %2 + + + + Number of %1 found : %2 + Anzahl von %1 gefunden : %2 + + + + SearchSingleComic + + + Please provide some aditional information. + Bitte einige zusätzliche Informationen. + + + + Series: + Serie: + + + + SearchVolume + + + Please provide some aditional information. + Serie: + Bitte einige zusätzliche Informationen. + + + + Series: + Serie: + + + + SelectComic + + + Please, select the right comic info. + Bitte wählen Sie die richtige Comic Information. + + + + comics + Comics + + + + loading cover + Cover laden + + + + loading description + Beschreibung laden + + + + description unavailable + Beschreibung nicht verfügbar + + + + SelectVolume + + + Please, select the right series for your comic. + Bitte wählen Sie die richtige Serie für Ihre Comics. + + + + volumes + Bände + + + + loading cover + Titelbilder werden geladen + + + + loading description + Beschreibung lädt + + + + description unavailable + Beschreibung nicht verfügbar + + + + SeriesQuestion + + + You are trying to get information for various comics at once, are they part of the same series? + Sie versuchen Informationen zu mehreren Comics auf einmal zu laden, sind sie Teil einer Serie? + + + + yes + Ja + + + + no + Nein + + + + ServerConfigDialog + + + set port + Port setzen + + + + EASY SERVER CONNECTION + Einfache Server Verbindung + + + + SERVER ADDRESS + SERVER Adresse + + + + just scan the code with your device!! + Einfach den Code scannen!! + + + + 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 ist nun verfügbar für IOS Geräte, die beste Comic Lese Erfahrung für Ihr IPAD, IPhone oder IPod Touch. <a href='http://ios.yacreader.com' style='color:rgb(193, 148, 65)'>Discover it! </a> + + + + IP address + IP Adresse + + + + Port + Port + + + + enable the server + Server aktivieren + + + + QR generator error! + QR Generator Fehler! + + + + SortVolumeComics + + + Please, sort the list of comics on the left until it matches the comics' information. + Sortieren Sie bitte die Comic Informationen links, bis die Informationen für die Comics übereinstimmen. + + + + sort comics to match comic information + Sortieren Sie die Comics um die Informationen zur Übereinstimmung zu bringen + + + + issues + Ausgaben + + + + remove selected comics + Löschen der ausgewählten Comics + + + + restore all removed comics + Wiederherstellung der gelöschten Comics + Wiederherstellen aller gelöschten Comics + + + + restore removed comics + + + + + TableModel + + + yes + Ja + + + + no + Nein + + + + Title + Titel + + + + File Name + File Name + + + + Pages + Seiten + + + + Size + Größe + + + + Read + Lesen + + + + Current Page + Aktuelle Seite + + + + Rating + Bewertung + + + + TitleHeader + + + SEARCH + Suche + + + + UpdateLibraryDialog + + + Updating.... + Aktualisierung... + + + + Cancel + Abbrechen + + + + Update library + Aktualisierung der Bibliothek + + + + VolumeComicsModel + + + title + Titel + + + + VolumesModel + + + year + Jahr + + + + issues + Bände + + + + publisher + Herausgeber + + + + YACReaderDeletingProgress + + + Please wait, deleting in progress... + Bitte warten, Löschvorgang läuft... + + + + cancel + Abbrechen + + + + YACReaderFieldEdit + + + + Click to overwrite + Drücken zum Überschreiben + + + + Restore to default + Ursprungseinstellungen wiederherstellen + + + + YACReaderFieldPlainTextEdit + + + + + + Click to overwrite + Drücken zum Überschreiben + + + + Restore to default + Ursprungseinstellungen wiederherstellen + + + + YACReaderFlowConfigWidget + + + How to show covers: + Wie zeige ich Titelseiten an: + + + + CoverFlow look + CoverFlow Ansicht + + + + Stripe look + Streifen Ansicht + + + + Overlapped Stripe look + Überlappende Streifen Ansicht + + + + YACReaderGLFlowConfigWidget + + + Presets: + Voreinstellungen: + + + + Classic look + Klassische Darstellung + + + + Stripe look + + + + + Overlapped Stripe look + Überlappende Streifen Darstellung + + + + Modern look + Moderne Drstellung + + + + Roulette look + Zufällige Darstellung + + + + Show advanced settings + Zeige Fortgeschrittenen Einstellungen + + + + Custom: + Benutzerdefiniert: + + + + View angle + Zeige Winkel + + + + Position + Position + + + + Cover gap + Titelbild Abstand + + + + Central gap + Zentral Abstand + + + + Zoom + Vergößern + + + + Y offset + Y Abstand + + + + Z offset + Z Abstand + + + + Cover Angle + Titelbild Winkel + + + + Visibility + Sichtbarkeit + + + + Light + Helligkeit + + + + Max angle + Max Winkel + + + + Low Performance + Niedrige Leistung + + + + High Performance + Hohe Leistung + + + + Use VSync (improve the image quality in fullscreen mode, worse performance) + Benutze VSync (verbessert die Darstellung im Vollbild Modus, schlechtere Leistung) + + + + Performance: + Leistung: + + + + YACReaderOptionsDialog + + + Save + Speichern + + + + Cancel + Cancel + + + + Use hardware acceleration (restart needed) + Hardwarebeschleunigung benutzen (Neustart erforderlich) + + + + YACReaderSideBar + + + LIBRARIES + Bibliotheken + + + + FOLDERS + ORDNER + + + + Search folders and comics + Ordner und Comics durchsuchen + + + 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 additional information. + Por favor, proporciona alguna información adicional. + + + + Series: + Series: + + + + SearchVolume + + + Please provide some additional 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..60a00ec2 --- /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 additional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some additional 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..9bd4189b --- /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 additional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some additional 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..6be9cf09 --- /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 additional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some additional 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..fde5a9f6 --- /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 additional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some additional 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..d6a95767 --- /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 additional information. + + + + + Series: + + + + + SearchVolume + + + Please provide some additional 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..06761ae6 --- /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 additional information. + + + + Series: + + + + + SearchVolume + + Please provide some additional 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..6454c980 --- /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 + +#if defined Q_OS_UNIX && !defined Q_OS_MAC + 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..d4c81305 --- /dev/null +++ b/common/comic.cpp @@ -0,0 +1,799 @@ +#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" + +#include "QsLog.h" + +const QStringList Comic::imageExtensions = QStringList() << "*.jpg" << "*.jpeg" << "*.png" << "*.gif" << "*.tiff" << "*.tif" << "*.bmp" << "*.webp"; +const QStringList Comic::literalImageExtensions = QStringList() << "jpg" << "jpeg" << "png" << "gif" << "tiff" << "tif" << "bmp" << "webp"; + +const QStringList Comic::comicExtensions = QStringList() << "*.cbr" << "*.cbz" << "*.rar" << "*.zip" << "*.tar" << "*.pdf" << "*.7z" << "*.cb7" << "*.arj" << "*.cbt"; +const QStringList Comic::literalComicExtensions = QStringList() << "cbr" << "cbz" << "rar" << "zip" << "tar" << "pdf" << "7z" << "cb7" << "arj" << "cbt"; + +//----------------------------------------------------------------------------- +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]; +} + +bool Comic::fileIsComic(const QString &path) +{ + QFileInfo info(path); + return literalComicExtensions.contains(info.suffix()); +} + +QList Comic::findValidComicFiles(const QList &list) +{ + QLOG_DEBUG() << "-findValidComicFiles-"; + QList validComicFiles; + QString currentPath; + foreach (QUrl url, list) { + currentPath = url.toLocalFile(); + if(Comic::fileIsComic(currentPath)) + validComicFiles << currentPath; + else if(QFileInfo(currentPath).isDir()) + { + validComicFiles << findValidComicFilesInFolder(currentPath); + } + } + QLOG_DEBUG() << "-" << validComicFiles << "-"; + return validComicFiles; +} + +QList Comic::findValidComicFilesInFolder(const QString &path) +{ + QLOG_DEBUG() << "-findValidComicFilesInFolder-" << path; + + if(!QFileInfo(path).isDir()) + return QList(); + + QList validComicFiles; + QDir folder(path); + folder.setNameFilters(Comic::comicExtensions); + folder.setFilter(QDir::AllDirs|QDir::Files|QDir::NoDotAndDotDot); + QFileInfoList folderContent = folder.entryInfoList(); + + QString currentPath; + foreach (QFileInfo info, folderContent) { + currentPath = info.absoluteFilePath(); + if(info.isDir()) + validComicFiles << findValidComicFilesInFolder(currentPath); //find comics recursively + else if(Comic::fileIsComic(currentPath)) + { + validComicFiles << currentPath; + } + } + + return validComicFiles; +} + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + +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..2f633429 --- /dev/null +++ b/common/comic.h @@ -0,0 +1,182 @@ +#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; + + public: + + static const QStringList imageExtensions; + static const QStringList literalImageExtensions; + static const QStringList comicExtensions; + static const QStringList literalComicExtensions; + + 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 imageExtensions;} + inline static QStringList getSupportedImageLiteralFormats() { return literalImageExtensions;} + + static bool fileIsComic(const QString &path); + static QList findValidComicFiles(const QList & list); + static QList findValidComicFilesInFolder(const QString &path); + 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..6e92b85a --- /dev/null +++ b/common/comic_db.cpp @@ -0,0 +1,483 @@ +#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]; +} + +qulonglong ComicDB::getFileSize() const +{ + //the size is encoded in the hash after the SHA-1 + return info.hash.right(info.hash.length()-40).toLongLong(); +} + +//----------------------------------------------------------------------------- +//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..417efa4d --- /dev/null +++ b/common/comic_db.h @@ -0,0 +1,157 @@ +#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; + + //return the size of the file in bytes + qulonglong getFileSize() 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..5695e144 --- /dev/null +++ b/common/custom_widgets.cpp @@ -0,0 +1 @@ +#include "custom_widgets.h" diff --git a/common/custom_widgets.h b/common/custom_widgets.h new file mode 100644 index 00000000..15430ce7 --- /dev/null +++ b/common/custom_widgets.h @@ -0,0 +1,6 @@ +#ifndef __CUSTOM_WIDGETS_H +#define __CUSTOM_WIDGETS_H + +#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..d92f4299 --- /dev/null +++ b/common/pictureflow.cpp @@ -0,0 +1,1409 @@ +/* + 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->state->centerIndex = index; + 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; +} + +void PictureFlow::resortCovers(QList newOrder) +{ + QVector slideImagesNew; + + QVector marksNew; + + QVector slidesInfo; + slidesInfo << d->state->leftSlides << d->state->centerSlide << d->state->rightSlides; + QVector slidesInfoNew; + + int order = 0; + foreach(int index, newOrder) + { + slideImagesNew << d->state->slideImages.at(index); + marksNew << d->state->marks.at(index); + slidesInfoNew << slidesInfo.at(index); + slidesInfoNew.last().slideIndex = order++; + } + + d->state->slideImages = slideImagesNew; + d->state->marks = marksNew; + d->state->leftSlides = slidesInfoNew.mid(0,d->state->leftSlides.length()); + d->state->centerSlide = slidesInfoNew.at(d->state->centerIndex); + d->state->leftSlides = slidesInfoNew.mid(d->state->centerIndex+1,d->state->leftSlides.length()); + + setCenterIndex(d->state->centerIndex); +} + diff --git a/common/pictureflow.h b/common/pictureflow.h new file mode 100644 index 00000000..b746ef70 --- /dev/null +++ b/common/pictureflow.h @@ -0,0 +1,228 @@ +/* + 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(); + + void resortCovers(QList newOrder); + +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/scroll_management.cpp b/common/scroll_management.cpp new file mode 100644 index 00000000..8db92273 --- /dev/null +++ b/common/scroll_management.cpp @@ -0,0 +1,61 @@ +#include "scroll_management.h" + +ScrollManagement::ScrollManagement() +{ + wheelTimer = new QTime(); + wheelTimer->start(); + wheelAccumulator = 0; +} + +ScrollManagement::Movement ScrollManagement::getMovement(QWheelEvent *event) +{ + /*QLOG_DEBUG() << "WheelEvent angle delta : " << event->angleDelta(); + QLOG_DEBUG() << "WheelEvent pixel delta : " << event->pixelDelta();*/ + + int tooFast = 1; + int timeThrottle = 16; + int minimumMove = 70; + + //avoid any events overflood + if((wheelTimer->elapsed() < tooFast)){ + event->setAccepted(true); + return None; + } + + // Accumulate the delta + if(event->delta()<0 != wheelAccumulator<0 ) //different sign means change in direction + wheelAccumulator = 0; + + wheelAccumulator += event->delta(); + + //Do not process events too fast + if((wheelTimer->elapsed() < timeThrottle)){ + event->setAccepted(true); + return None; + } + + //small intervals are ignored until with have enough acumulated delta + if((wheelAccumulator < minimumMove) && (wheelAccumulator > -minimumMove)){ + event->setAccepted(true); + return None; + } + + Movement m; + if(wheelAccumulator<0) + m = Forward; + else + m = Backward; + + event->accept(); + //Clean up + wheelAccumulator = 0; + wheelTimer->restart(); + + return m; +} + +ScrollManagement::~ScrollManagement() +{ + +} + diff --git a/common/scroll_management.h b/common/scroll_management.h new file mode 100644 index 00000000..8c169179 --- /dev/null +++ b/common/scroll_management.h @@ -0,0 +1,25 @@ +#ifndef SCROLLMANAGAMENT_H +#define SCROLLMANAGAMENT_H + +#include +#include + +class ScrollManagement +{ +public: + enum Movement{ + None, + Forward, + Backward + }; + + ScrollManagement(); + ScrollManagement::Movement getMovement(QWheelEvent * event); + ~ScrollManagement(); + +private: + QTime * wheelTimer; + int wheelAccumulator; +}; + +#endif // SCROLLMANAGAMENT_H diff --git a/common/yacreader_flow_gl.cpp b/common/yacreader_flow_gl.cpp new file mode 100644 index 00000000..1309d0d9 --- /dev/null +++ b/common/yacreader_flow_gl.cpp @@ -0,0 +1,1628 @@ +#include "yacreader_flow_gl.h" + +#include +#include +//#include + +#ifdef Q_OS_MAC + #include +#else + #include +#endif + +#include +#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) + :QOpenGLWidget(/*QOpenGLWidget migration 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();*/ + + QSurfaceFormat f = format(); + + //TODO add antialiasing + //f.setSamples(4); + f.setVersion(2, 1); + f.setSwapInterval(0); + setFormat(f); + + timerId = startTimer(updateInterval); +} + +void YACReaderFlowGL::timerEvent(QTimerEvent * event) +{ + if(timerId == event->timerId()) + update(); + + //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() +{ + glShadeModel(GL_SMOOTH); + + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + + defaultTexture = new QOpenGLTexture(QImage(":/images/defaultCover.png")); + defaultTexture->setMinMagFilters(QOpenGLTexture::LinearMipMapLinear,QOpenGLTexture::LinearMipMapLinear); +#ifdef YACREADER_LIBRARY + markTexture = new QOpenGLTexture(QImage(":/images/readRibbon.png")); + markTexture->setMinMagFilters(QOpenGLTexture::LinearMipMapLinear,QOpenGLTexture::LinearMipMapLinear); + + readingTexture = new QOpenGLTexture(QImage(":/images/readingRibbon.png")); + readingTexture->setMinMagFilters(QOpenGLTexture::LinearMipMapLinear,QOpenGLTexture::LinearMipMapLinear); +#endif + if(lazyPopulateObjects!=-1) + populate(lazyPopulateObjects); + + hasBeenInitialized = true; +} + +void YACReaderFlowGL::paintGL() +{ + QPainter painter; + painter.begin(this); + + painter.beginNativePainting(); + + glEnable(GL_DEPTH_TEST); + glEnable(GL_CULL_FACE); + glEnable(GL_COLOR_MATERIAL); + glEnable(GL_BLEND); + glEnable(GL_MULTISAMPLE); + + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + + if(numObjects>0) + { + updatePositions(); + udpatePerspective(width(),height()); + draw(); + } + + glDisable(GL_MULTISAMPLE); + glDisable(GL_BLEND); + glDisable(GL_COLOR_MATERIAL); + glDisable(GL_CULL_FACE); + glDisable(GL_DEPTH_TEST); + + painter.endNativePainting(); + + QFont font = painter.font() ; + font.setFamily("Arial"); + font.setPixelSize(fontSize); + painter.setFont(font); + + painter.setPen(QColor(76,76,76)); + painter.drawText(10,fontSize + 10, QString("%1/%2").arg(currentSelected+1).arg(numObjects)); + + painter.end(); +} + +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(YACReader3DImage & image, int pos) +{ + if(pos == 0){ + image.current = centerPos; + }else{ + if(pos > 0){ + image.current.x = (config.centerDistance)+(config.xDistance*pos); + image.current.y = config.yDistance*pos*-1; + image.current.z = config.zDistance*pos*-1; + image.current.rot = config.rotation; + }else{ + image.current.x = (config.centerDistance)*-1+(config.xDistance*pos); + image.current.y = config.yDistance*pos; + image.current.z = config.zDistance*pos; + image.current.rot = config.rotation*-1; + } + } + +} +void YACReaderFlowGL::calcVector(YACReader3DVector & vector, int pos) +{ + calcPos(dummy,pos); + + vector.x = dummy.current.x; + vector.y = dummy.current.y; + vector.z = dummy.current.z; + vector.rot = dummy.current.rot; +} + +bool YACReaderFlowGL::animate(YACReader3DVector & currentVector,YACReader3DVector & toVector) +{ + float rotDiff = toVector.rot-currentVector.rot; + float xDiff = toVector.x-currentVector.x; + float yDiff = toVector.y-currentVector.y; + float zDiff = toVector.z-currentVector.z; + + if(fabs(rotDiff) < 0.01 + && fabs(xDiff) < 0.001 + && fabs(yDiff) < 0.001 + && fabs(zDiff) < 0.001) + return true; + + //calculate and apply positions + currentVector.x = currentVector.x+(xDiff)*config.animationStep; + currentVector.y = currentVector.y+(yDiff)*config.animationStep; + currentVector.z = currentVector.z+(zDiff)*config.animationStep; + + if(fabs(rotDiff) > 0.01){ + currentVector.rot = currentVector.rot+(rotDiff)*(config.animationStep*config.preRotation); + } + else + { + viewRotateActive = 0; + } + + return false; +} +void YACReaderFlowGL::drawCover(const YACReader3DImage & image) +{ + float w = image.width; + float h = image.height; + + //fadeout + float opacity = 1-1/(config.animationFadeOutDist+config.viewRotateLightStrenght*fabs(viewRotate))*fabs(0-image.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( image.current.x, image.current.y, image.current.z ); + + glPushMatrix(); + glRotatef(image.current.rot,0,1,0); + + glEnable(GL_TEXTURE_2D); + image.texture->bind(); + + //calculate shading + float LShading = ((config.rotation != 0 )?((image.current.rot < 0)?1-1/config.rotation*image.current.rot:1):1); + float RShading = ((config.rotation != 0 )?((image.current.rot > 0)?1-1/(config.rotation*-1)*image.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/2,LUP*opacity*reflectionUp/2,LUP*opacity*reflectionUp/2,1); + glTexCoord2f(0.0f, 0.0f); + glVertex3f(w/2.f*-1.f, -0.5f-h, 0.f); + + //esquina inferior derecha + glColor4f(RUP*opacity*reflectionUp/2,RUP*opacity*reflectionUp/2,RUP*opacity*reflectionUp/2,1); + glTexCoord2f(1.0f, 0.0f); + glVertex3f(w/2.f, -0.5f-h, 0.f); + + //esquina superior derecha + glColor4f(RDOWN*opacity/3,RDOWN*opacity/3,RDOWN*opacity/3,1); + glTexCoord2f(1.0f, 1.0f); + glVertex3f(w/2.f, -0.5f, 0.f); + + //esquina superior izquierda + glColor4f(LDOWN*opacity/3,LDOWN*opacity/3,LDOWN*opacity/3,1); + glTexCoord2f(0.0f, 1.0f); + glVertex3f(w/2.f*-1.f, -0.5f, 0.f); + + glEnd(); + glDisable(GL_TEXTURE_2D); + + if(showMarks && loaded[image.index] && marks[image.index] != Unread) + { + glEnable(GL_TEXTURE_2D); + if(marks[image.index] == Read) + markTexture->bind(); + else + readingTexture->bind(); + 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(images[count]); + } + } + + //Draw left Covers + for(count = 0;count < numObjects-1;count++){ + if(count < CS){ + drawCover(images[count]); + } + } + + //Draw Center Cover + drawCover(images[CS]); + + +} + +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) +{ + if(!(pos>=0 && pos < images.length() && images.length()>0)) + return; + if(pos >= images.length() && images.length() > 0) + pos = images.length()-1; + + 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--){ + calcVector(images[count].animEnd,count-currentSelected); + if(!animate(images[count].current,images[count].animEnd)) + stopAnimation = false; + } + + //slowly reset view angle + if(!viewRotateActive){ + viewRotate += (0-viewRotate)*config.viewRotateSub; + } + + if(fabs (images[currentSelected].current.x - images[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, QOpenGLTexture * texture, float x, float y,int item) +{ + startAnimationTimer(); + + Q_UNUSED(name) + //set a new entry + if(item == -1){ + images.push_back(YACReader3DImage()); + + item = numObjects; + numObjects++; + + calcVector(images[item].current,item); + images[item].current.z = images[item].current.z-1; + } + + images[item].texture = texture; + images[item].width = x; + images[item].height = y; + images[item].index = item; + //strcpy(cfImages[item].name,name); + + +} + +void YACReaderFlowGL::remove(int item) +{ + if(item < 0 || item >= images.size()) + return; + + startAnimationTimer(); + + loaded.remove(item); + marks.remove(item); + + //reposition current selection + if(item <= currentSelected && currentSelected != 0){ + currentSelected--; + } + + QOpenGLTexture * texture = images[item].texture; + + int count = item; + while(count <= numObjects-2){ + images[count].index--; + count++; + } + images.removeAt(item); + + if(texture != defaultTexture) + delete(texture); + + numObjects--; +} + +/*Info*/ +YACReader3DImage YACReaderFlowGL::getCurrentSelected() +{ + return images[currentSelected]; +} + +void YACReaderFlowGL::replace(char *name, QOpenGLTexture * texture, float x, float y,int item) +{ + startAnimationTimer(); + + Q_UNUSED(name) + if(images[item].index == item) + { + images[item].texture = texture; + images[item].width = x; + images[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;iwidth(); + 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) + { + QSurfaceFormat f = format(); + f.setVersion(2, 1); + f.setSwapInterval(1); + setFormat(f); + } + else + { + QSurfaceFormat 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) +{ + Movement m = getMovement(event); + switch (m) { + case None: + return; + case Forward: + showNext(); + break; + case Backward: + showPrevious(); + break; + default: + break; + } +} + +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) +{ + makeCurrent(); + 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(winX, 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 + QOpenGLWidget::mousePressEvent(event); + doneCurrent(); +} + +void YACReaderFlowGL::mouseDoubleClickEvent(QMouseEvent* event) +{ + makeCurrent(); + 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(); + } + doneCurrent(); +} + +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(); + QOpenGLTexture * texture = new QOpenGLTexture(img); + + if(performance == high || performance == ultraHigh) + { + texture->setAutoMipMapGenerationEnabled(true); + texture->setMinMagFilters(QOpenGLTexture::LinearMipMapLinear,QOpenGLTexture::LinearMipMapLinear); + } + else + { + texture->setMinMagFilters(QOpenGLTexture::Linear, QOpenGLTexture::Linear); + } + + float y = 1 * (float(img.height())/img.width()); + QString s = "cover"; + replace(s.toLocal8Bit().data(), texture, x, y,idx); + } + } + + // 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); + if(item >= 0 && item < paths.size()) + paths.removeAt(item); + worker->unlock(); +} + +void YACReaderComicFlowGL::resortCovers(QList newOrder) +{ + worker->lock(); + worker->reset();//is this necesary? + startAnimationTimer(); + QList pathsNew; + QVector loadedNew; + QVector marksNew; + QVector imagesNew; + + int index = 0; + foreach (int i, newOrder) { + pathsNew << paths.at(i); + loadedNew << loaded.at(i); + marksNew << marks.at(i); + imagesNew << images.at(i); + imagesNew.last().index = index++; + } + + paths = pathsNew; + loaded = loadedNew; + marks = marksNew; + images = imagesNew; + + 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(); + for(int i = 0;ibusy()) + 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(); + QOpenGLTexture * texture = new QOpenGLTexture(img); + + if(performance == high || performance == ultraHigh) + { + texture->setAutoMipMapGenerationEnabled(true); + texture->setMinMagFilters(QOpenGLTexture::LinearMipMapLinear,QOpenGLTexture::LinearMipMapLinear); + } + else + { + texture->setMinMagFilters(QOpenGLTexture::Linear, QOpenGLTexture::Linear); + } + + float y = 1 * (float(img.height())/img.width()); + QString s = "cover"; + replace(s.toLocal8Bit().data(), texture, x, y,idx); + loaded[idx] = true; + } + } + + // 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()) + { + 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; +} diff --git a/common/yacreader_flow_gl.h b/common/yacreader_flow_gl.h new file mode 100644 index 00000000..2fc37d51 --- /dev/null +++ b/common/yacreader_flow_gl.h @@ -0,0 +1,383 @@ +//OpenGL Coverflow API by J.Roth +#ifndef __YACREADER_FLOW_GL_H +#define __YACREADER_FLOW_GL_H + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "pictureflow.h" //TODO mover los tipos de flow de sitio +#include "scroll_management.h" + +class ImageLoaderGL; +class QGLContext; +class WidgetLoader; +class ImageLoaderByteArrayGL; + +enum Performance +{ + low=0, + medium, + high, + ultraHigh +}; + +//Cover Vector +struct YACReader3DVector{ + float x; + float y; + float z; + float rot; +}; + +//the image/texture info struct +struct YACReader3DImage{ + QOpenGLTexture * texture; + //char name[256]; + + float width; + float height; + + int index; + + YACReader3DVector current; + YACReader3DVector 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 QOpenGLWidget, public ScrollManagement +{ + Q_OBJECT +protected: + int timerId; + /*** System variables ***/ + YACReader3DImage dummy; + int viewRotateActive; + float stepBackup; + + /*functions*/ + void calcPos(YACReader3DImage & image, int pos); + void calcVector(YACReader3DVector & vector, int pos); + //returns true if the animation is finished for Current + bool animate(YACReader3DVector ¤tVector, YACReader3DVector &toVector); + void drawCover(const YACReader3DImage & image); + + void udpatePerspective(int width, int height); + + int updateCount; + WidgetLoader * loader; + int fontSize; + + QOpenGLTexture * defaultTexture; + QOpenGLTexture * markTexture; + QOpenGLTexture * readingTexture; + void initializeGL(); + void paintGL(); + void timerEvent(QTimerEvent *); + + //number of Covers + int numObjects; + int lazyPopulateObjects; + bool showMarks; + QVector loaded; + QVector marks; + + QVector images; + + 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 + YACReader3DVector 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, QOpenGLTexture * texture, 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, QOpenGLTexture * texture, float x, float y, int item); + //create n covers with the default nu + void populate(int n); + /*Info*/ + //retuns the YACReader3DImage Struct of the current selected item + //to read title or textures + YACReader3DImage 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 YACReaderComicFlowGL : public YACReaderFlowGL +{ +public: + YACReaderComicFlowGL(QWidget *parent = 0,struct Preset p = defaultYACReaderFlowConfig); + void setImagePaths(QStringList paths); + void updateImageData(); + void remove(int item); + void resortCovers(QList newOrder); + friend class ImageLoaderGL; +private: + ImageLoaderGL * worker; +protected: + QList paths; + +}; + +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; +}; + +#endif diff --git a/common/yacreader_global.cpp b/common/yacreader_global.cpp new file mode 100644 index 00000000..fbf2540e --- /dev/null +++ b/common/yacreader_global.cpp @@ -0,0 +1,141 @@ +#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; +} + + +QString YACReader::colorToName(LabelColors colors) +{ + switch(colors){ + case YRed: + return "red"; + case YOrange: + return "orange"; + case YYellow: + return "yellow"; + case YGreen: + return "green"; + case YCyan: + return "cyan"; + case YBlue: + return "blue"; + case YViolet: + return "violet"; + case YPurple: + return "purple"; + case YPink: + return "pink"; + case YWhite: + return "white"; + case YLight: + return "light"; + case YDark: + return "dark"; + } +} + + +QIcon YACReader::noHighlightedIcon(const QString &path) +{ + QPixmap p(path); + + QIcon icon;//(path); + icon.addFile(path,p.size(),QIcon::Normal); + icon.addFile(path,p.size(),QIcon::Selected); + return icon; +} + + +void YACReader::colorize(QImage &img, QColor &col) +{ + QRgb *data = (QRgb *)img.bits(); + QRgb *end = data + img.width()*img.height(); + + int rcol = col.red(), gcol = col.green(), bcol = col.blue(); + while(data != end) { + *data = qRgba(rcol,gcol,bcol,qAlpha(*data)); + ++data; + } +} + + +QString YACReader::labelColorToRGBString(LabelColors color) +{ + switch (color) { + case YRed: + return "#FD777C"; + + case YOrange: + return "#FEBF34"; + + case YYellow: + return "#F5E934"; + + case YGreen: + return "#B6E525"; + + case YCyan: + return "#9FFFDD"; + + case YBlue: + return "#82C7FF"; + + case YViolet: + return "#8286FF"; + + case YPurple: + return "#E39FFF"; + + case YPink: + return "#FF9FDD"; + +#ifdef Q_OS_MAC + case YWhite: + return "#E3E3E3"; +#else + case YWhite: + return "#FFFFFF"; +#endif + case YLight: + return "#C8C8C8"; + case YDark: + return "#ABABAB"; + + + } +} + + +QList YACReader::mimeDataToComicsIds(const QMimeData *data) +{ + QList comicIds; + QByteArray rawData = data->data(YACReader::YACReaderLibrarComiscSelectionMimeDataFormat); + QDataStream in(&rawData,QIODevice::ReadOnly); + in >> comicIds; //deserialize the list of indentifiers + return comicIds; +} diff --git a/common/yacreader_global.h b/common/yacreader_global.h new file mode 100644 index 00000000..74ce104e --- /dev/null +++ b/common/yacreader_global.h @@ -0,0 +1,145 @@ +#ifndef __YACREADER_GLOBAL_H +#define __YACREADER_GLOBAL_H + +#if QT_VERSION >= 0x050000 + #include +#else + #include +#endif + +#include +#include + +#define VERSION "8.0.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 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" +#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 COMICS_VIEW_FLOW_SPLITTER_STATUS "COMICS_VIEW_FLOW_SPLITTER_STATUS" +#define SIDEBAR_SPLITTER_STATUS "SIDEBAR_SPLITTER_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" + +#define COMIC_VINE_API_KEY "COMIC_VINE_API_KEY" + +namespace YACReader +{ + +static const QString YACReaderLibrarComiscSelectionMimeDataFormat = "application/yacreaderlibrary-comics-ids"; +static const QString YACReaderLibrarSubReadingListMimeDataFormat = "application/yacreaderlibrary-sublist-rows"; + + 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 + }; + + enum SearchModifiers{ + NoModifiers = 0, + OnlyRead, + OnlyUnread, + ByAuthor + }; + + enum LabelColors{ + YRed = 1, + YOrange, + YYellow, + YGreen, + YCyan, + YBlue, + YViolet, + YPurple, + YPink, + YWhite, + YLight, + YDark + }; + +QString getSettingsPath(); +void addSperator(QWidget * w); +QAction * createSeparator(); +QString colorToName(LabelColors colors); +QIcon noHighlightedIcon(const QString & path); +void colorize(QImage &img, QColor &col); +QString labelColorToRGBString(LabelColors color); +QList mimeDataToComicsIds(const QMimeData * data); +} +#endif + diff --git a/compileOSX.sh b/compileOSX.sh new file mode 100755 index 00000000..9a0ad4c1 --- /dev/null +++ b/compileOSX.sh @@ -0,0 +1,47 @@ +#! /bin/bash +if [ $2 == "clean" ]; then +./cleanOSX.sh +fi + +echo "Compiling YACReader" +cd ./YACReader +/Users/luisangel/Qt/5.3/clang_64/bin/qmake -spec macx-clang "CONFIG+=release" +#qmake -spec macx-g++ "CONFIG+=release" +make +cd .. + +echo "Compiling YACReaderLibrary" +cd ./YACReaderLibrary +/Users/luisangel/Qt/5.3/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/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..8878e64f --- /dev/null +++ b/custom_widgets/custom_widgets_yacreader.pri @@ -0,0 +1,34 @@ +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 + +macx{ +HEADERS += $$PWD/yacreader_macosx_toolbar.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 +macx{ +OBJECTIVE_SOURCES += $$PWD/yacreader_macosx_toolbar.mm +} + diff --git a/custom_widgets/custom_widgets_yacreaderlibrary.pri b/custom_widgets/custom_widgets_yacreaderlibrary.pri new file mode 100644 index 00000000..40308ef6 --- /dev/null +++ b/custom_widgets/custom_widgets_yacreaderlibrary.pri @@ -0,0 +1,48 @@ +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 + +macx{ +HEADERS += $$PWD/yacreader_macosx_toolbar.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 + +macx{ +OBJECTIVE_SOURCES += $$PWD/yacreader_macosx_toolbar.mm +} 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..b5f6b259 --- /dev/null +++ b/custom_widgets/yacreader_library_item_widget.cpp @@ -0,0 +1,156 @@ +#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/sidebar/libraryIcon.png"); + icon = new QLabel(this); + icon->setPixmap(iconPixmap); + + nameLabel = new QLabel(name,this); + + options = new QToolButton(this); + options->setIcon(QIcon(":/images/sidebar/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/sidebar/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;"; + QString styleSheet = "color: white; background-color:#91c4f4; border-bottom:1px solid #91c4f4;"; +#else + QString styleSheet = "color: white; background-color:#2E2E2E; font-weight:bold;"; +#endif + setStyleSheet(styleSheet); + + options->setHidden(false); + + QPixmap iconPixmap(":/images/sidebar/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_macosx_toolbar.h b/custom_widgets/yacreader_macosx_toolbar.h new file mode 100644 index 00000000..a37b144b --- /dev/null +++ b/custom_widgets/yacreader_macosx_toolbar.h @@ -0,0 +1,87 @@ +#ifndef YACREADER_MACOSX_TOOLBAR_H +#define YACREADER_MACOSX_TOOLBAR_H + +#include +#include + +#include "yacreader_global.h" + +//Wrapper for NSTextField +class YACReaderMacOSXSearchLineEdit : public QObject +{ + Q_OBJECT +public: + YACReaderMacOSXSearchLineEdit(); + void * getNSTextField(); + +public slots: + QString text(); + void clear(); + void clearText(); //no signal emited + void setDisabled(bool disabled); + void setEnabled(bool enabled); + +private: + void * nstextfield; + + +signals: + //convenience signal for YACReaderLibrary search edit + void filterChanged(YACReader::SearchModifiers, QString); +}; + +class MacToolBarItemWrapper : public QObject +{ + Q_OBJECT +public: + MacToolBarItemWrapper(QAction * action, QMacToolBarItem * toolbaritem); + +public slots: + void actionToggled(bool toogled); + +private: + QAction * action; + QMacToolBarItem * toolbaritem; + + void updateIcon(bool checked); +}; + + +class YACReaderMacOSXToolbar : public QMacToolBar +{ + Q_OBJECT +public: + explicit YACReaderMacOSXToolbar(QObject *parent = 0); + void addAction(QAction * action); + void addDropDownItem(const QList & actions, const QAction * defaultAction = 0); + void addSpace(int size); //size in points + void addSeparator(); + void addStretch(); + void addWidget(QWidget * widget); + void show(); + void hide(); + QMap actions; + + //hacks everywhere + //convenience method for YACReaderLibrary search edit + YACReaderMacOSXSearchLineEdit *addSearchEdit(); + //convenience method for showing the fit to width slider in MacOSX + QAction * addFitToWidthSlider(QAction * attachToAction); + + + //convenience method for switching the icon of the view selector + void updateViewSelectorIcon(const QIcon & icon); + +signals: + +public slots: + +protected: + NSToolbar * nativeToolBar; + void *delegate; + bool yosemite; + QMacToolBarItem * viewSelector; + +}; + +#endif // YACREADER_MACOSX_TOOLBAR_H diff --git a/custom_widgets/yacreader_macosx_toolbar.mm b/custom_widgets/yacreader_macosx_toolbar.mm new file mode 100644 index 00000000..b5c52d3d --- /dev/null +++ b/custom_widgets/yacreader_macosx_toolbar.mm @@ -0,0 +1,395 @@ +#include "yacreader_macosx_toolbar.h" + +#include +#include +#include +#include +#include + +#import +#import +#import + +#import "shortcuts_manager.h" + +//---------------------------- +//A custom items separator for NSToolbar +@interface CustomSeparator : NSView + +@end + + +@implementation CustomSeparator + +- (void) drawRect:(NSRect)rect { + [[NSColor colorWithDeviceRed:0.5 green:0.5 blue:0.5 alpha:1] setFill]; + NSRectFill(rect); + [super drawRect:rect]; +} + +@end + +//---------------------------- +//Toolbar delegate, needed for allow disabled/enabled items +@interface MyToolbarDelegate : NSObject +{ +@public + YACReaderMacOSXToolbar * mytoolbar; +} + +- (NSToolbarItem *) toolbar:(NSToolbar *)toolbar itemForItemIdentifier:(NSString *) itemIdent willBeInsertedIntoToolbar:(BOOL) willBeInserted; +- (NSArray *)toolbarDefaultItemIdentifiers:(NSToolbar*)toolbar; +- (NSArray *)toolbarAllowedItemIdentifiers:(NSToolbar*)toolbar; +//- (NSArray *)toolbarSelectableItemIdentifiers:(NSToolbar *)toolbar; +- (IBAction)itemClicked:(id)sender; +- (BOOL)validateToolbarItem:(NSToolbarItem *)theItem; +@end + + +@implementation MyToolbarDelegate + +- (NSArray *)toolbarDefaultItemIdentifiers:(NSToolbar*)toolbar +{ + Q_UNUSED(toolbar); + + NSMutableArray *array = [[NSMutableArray alloc] init]; + + QList items = mytoolbar->items(); + foreach (const QMacToolBarItem * item, items) { + [array addObject : item->nativeToolBarItem().itemIdentifier]; + } + return array; +} + +- (NSArray *)toolbarAllowedItemIdentifiers:(NSToolbar*)toolbar +{ + Q_UNUSED(toolbar); + + NSMutableArray *array = [[NSMutableArray alloc] init]; + + QList items = mytoolbar->items(); + foreach (const QMacToolBarItem * item, items) { + [array addObject : item->nativeToolBarItem().itemIdentifier]; + } + return array; +} + + +/* +- (NSArray *)toolbarSelectableItemIdentifiers: (NSToolbar *)toolbar +{ + Q_UNUSED(toolbar); + + NSMutableArray *array = [[NSMutableArray alloc] init]; + + QList items = mytoolbar->items(); + foreach (const QMacToolBarItem * item, items) { + [array addObject : item->nativeToolBarItem().itemIdentifier]; + } + return array; + //NSMutableArray *array = toolbarPrivate->getItemIdentifiers(toolbarPrivate->items, true); + //[array addObjectsFromArray:toolbarPrivate->getItemIdentifiers(toolbarPrivate->allowedItems, true)]; + //return array; +}*/ + +- (IBAction)itemClicked:(id)sender +{ + if([sender respondsToSelector:@selector(itemIdentifier)]) + { + NSToolbarItem *item = reinterpret_cast(sender); + + QString identifier = QString::fromNSString([item itemIdentifier]); + QMacToolBarItem *toolButton = reinterpret_cast(identifier.toULongLong()); + Q_EMIT toolButton->activated(); + } +} + +- (NSToolbarItem *) toolbar: (NSToolbar *)toolbar itemForItemIdentifier: (NSString *) itemIdentifier willBeInsertedIntoToolbar:(BOOL) willBeInserted +{ + Q_UNUSED(toolbar); + Q_UNUSED(willBeInserted); + QList items = mytoolbar->items(); + + foreach (const QMacToolBarItem * item, items) { + NSToolbarItem *toolbarItem = item->nativeToolBarItem(); + if([toolbarItem.itemIdentifier isEqual:itemIdentifier]) + { + + [toolbarItem setTarget:self]; + [toolbarItem setAction:@selector(itemClicked:)]; + + return toolbarItem; + } + } + return nil; +} + +- (BOOL)validateToolbarItem:(NSToolbarItem *)theItem +{ + QString identifier = QString::fromNSString(theItem.itemIdentifier); + + if(mytoolbar->actions.contains(identifier)) + { + return mytoolbar->actions.value(identifier)->isEnabled(); + } + else return NO; +} +@end + +//---------------------------- +//detect changes in native text field +//TODO implement validation and auto completion +@interface MyTextFieldDelegate : NSObject +{ +@public + YACReaderMacOSXSearchLineEdit * mylineedit; +} +@end + +@implementation MyTextFieldDelegate + +- (void)controlTextDidChange:(NSNotification *)notification { + NSTextField *textField = [notification object]; + NSLog(@"%@",[textField stringValue]); + Q_EMIT mylineedit->filterChanged(YACReader::NoModifiers, QString::fromNSString([textField stringValue])); +} + +@end +//---------------------------- + +YACReaderMacOSXToolbar::YACReaderMacOSXToolbar(QObject *parent) + :viewSelector(0) +{ + //setup native toolbar + nativeToolBar= nativeToolbar(); + [nativeToolBar setDisplayMode:NSToolbarDisplayModeIconOnly]; + [nativeToolBar setAllowsUserCustomization:NO]; + + delegate = [[MyToolbarDelegate alloc] init]; + ((MyToolbarDelegate *)delegate)->mytoolbar = this; + [nativeToolBar setDelegate:(MyToolbarDelegate *)delegate]; + +#ifdef YACREADER_LIBRARY + NSWindow *nswindow = (NSWindow*) qApp->platformNativeInterface()->nativeResourceForWindow("nswindow", ((QMainWindow*)parent)->windowHandle()); + if([nswindow respondsToSelector:@selector(setTitleVisibility:)]) + { + yosemite = true; + //TODO yosemite new constants are not found in compilation time + [nswindow setTitleVisibility:1]; //NSWindowTitleHidden + //TODO NSFullSizeContentViewWindowMask produces an offset in the windows' content + //nswindow.styleMask |= 1 << 15; // NSFullSizeContentViewWindowMask; + [nativeToolBar setSizeMode:NSToolbarSizeModeSmall]; //TODO figure out how to load specific images in Yosemite + }else + { + [nativeToolBar setSizeMode:NSToolbarSizeModeSmall]; + yosemite = false; + } +#else + yosemite = false; + [nativeToolBar setAutosavesConfiguration:YES]; //TODO this doesn't work + [nativeToolBar setSizeMode:NSToolbarSizeModeSmall]; +#endif +} + +void YACReaderMacOSXToolbar::addAction(QAction *action) +{ + QMacToolBarItem *toolBarItem = addItem(action->icon(),action->text()); + if(action->data().toString() == TOGGLE_COMICS_VIEW_ACTION_YL) + viewSelector = toolBarItem; + connect(toolBarItem,SIGNAL(activated()),action, SIGNAL(triggered())); + + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + actions.insert(QString::fromNSString(nativeItem.itemIdentifier),action); + + MacToolBarItemWrapper * wrapper = new MacToolBarItemWrapper(action,toolBarItem); + //wrapper->actionToogled(true); +} + +void YACReaderMacOSXToolbar::addDropDownItem(const QList &actions, const QAction *defaultAction) +{ + //TODO +} + +void YACReaderMacOSXToolbar::addSpace(int size) +{ + QMacToolBarItem *toolBarItem = addItem(QIcon(),""); + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + + static const NSRect frameRect = { { 0.0, 0.0 }, { size, 16.0 } }; + NSView *view = [[NSView alloc] initWithFrame:frameRect]; + + [nativeItem setView:view]; +} + +//reimplemented for convenience +void YACReaderMacOSXToolbar::addSeparator() +{ + //QMacToolBar::addSeparator(); + + QMacToolBarItem *toolBarItem = addItem(QIcon(),""); + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + + static const NSRect frameRect = { { 0.0, 0.0 }, { 1, 16.0 } }; + CustomSeparator *view = [[CustomSeparator alloc] initWithFrame:frameRect]; + + [nativeItem setView:view]; +} + +void YACReaderMacOSXToolbar::addStretch() +{ + QMacToolBarItem *toolBarItem = addItem(QIcon(),""); + toolBarItem->setStandardItem(QMacToolBarItem::FlexibleSpace); +} + +void YACReaderMacOSXToolbar::addWidget(QWidget *widget) +{ + //TODO fix it + /* QMacNativeWidget *nativeWidget = new QMacNativeWidget(); + QVBoxLayout *layout = new QVBoxLayout(); + layout->addWidget(widget); + nativeWidget->setLayout(layout); + + + NSView *nativeWidgetView = reinterpret_cast(nativeWidget->winId()); + QMacToolBarItem *toolBarItem = addItem(QIcon(),""); + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + [nativeItem setView:nativeWidgetView];*/ +} + +void YACReaderMacOSXToolbar::show() +{ + [nativeToolBar setVisible:YES]; +} + +void YACReaderMacOSXToolbar::hide() +{ + [nativeToolBar setVisible:NO]; +} + +YACReaderMacOSXSearchLineEdit * YACReaderMacOSXToolbar::addSearchEdit() +{ + QMacToolBarItem *toolBarItem = addItem(QIcon(),""); + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + + YACReaderMacOSXSearchLineEdit * searchEdit = new YACReaderMacOSXSearchLineEdit(); + + + if(yosemite) + [nativeItem setView:(NSTextField *)searchEdit->getNSTextField()]; + else + { + static const NSRect searchEditFrameRect = { { 0.0, 0.0 }, { 165, 26.0 } }; + NSView * view = [[NSView alloc] initWithFrame:searchEditFrameRect]; + [view addSubview:((NSTextField *)searchEdit->getNSTextField())]; + [nativeItem setView:view]; + } + + return searchEdit; +} + +QAction *YACReaderMacOSXToolbar::addFitToWidthSlider(QAction *attachToAction) +{ + QMacToolBarItem *toolBarItem = addItem(QIcon(":/images/viewer_toolbar/toWidthSlider.png"),"fit to width slider"); + + NSToolbarItem * nativeItem = toolBarItem->nativeToolBarItem(); + actions.insert(QString::fromNSString(nativeItem.itemIdentifier),attachToAction); + + QAction * action = new QAction("",attachToAction->parent()); + + connect(toolBarItem,SIGNAL(activated()), action, SIGNAL(triggered())); + + return action; +} + +void YACReaderMacOSXToolbar::updateViewSelectorIcon(const QIcon &icon) +{ + if(viewSelector) + viewSelector->setIcon(icon); +} + + +YACReaderMacOSXSearchLineEdit::YACReaderMacOSXSearchLineEdit() + :QObject() +{ + NSRect searchEditFrameRect = { { 0.0, -3.0 }, { 165, 32.0 } }; + //NSTextField * searchEdit = [[NSTextField alloc] initWithFrame:searchEditFrameRect]; + + NSTextField * searchEdit = [[NSSearchField alloc] initWithFrame:searchEditFrameRect]; + //[searchEdit setBezelStyle:NSTextFieldRoundedBezel]; + + [[searchEdit cell] setPlaceholderString:@"type to search"]; + + MyTextFieldDelegate * delegate = [[MyTextFieldDelegate alloc] init]; + delegate->mylineedit = this; + [searchEdit setDelegate:delegate]; + + nstextfield = searchEdit; +} + +void *YACReaderMacOSXSearchLineEdit::getNSTextField() +{ + return nstextfield; +} + +QString YACReaderMacOSXSearchLineEdit::text() +{ + return QString::fromNSString([((NSTextField *)nstextfield) stringValue]); +} + +void YACReaderMacOSXSearchLineEdit::clear() +{ + [((NSTextField *)nstextfield) setStringValue:@""]; + emit filterChanged(YACReader::NoModifiers, ""); +} + +void YACReaderMacOSXSearchLineEdit::clearText() +{ + //TODO be sure that this will not generate any event.... + [((NSTextField *)nstextfield) setStringValue:@""]; +} + +void YACReaderMacOSXSearchLineEdit::setDisabled(bool disabled) +{ + [((NSTextField *)nstextfield) setEnabled:!disabled]; +} + +void YACReaderMacOSXSearchLineEdit::setEnabled(bool enabled) +{ + [((NSTextField *)nstextfield) setEnabled:enabled]; +} + + +MacToolBarItemWrapper::MacToolBarItemWrapper(QAction *action, QMacToolBarItem *toolbaritem) + :action(action),toolbaritem(toolbaritem) +{ + if(action->isCheckable()) + { + connect(action,SIGNAL(toggled(bool)),this,SLOT(actionToggled(bool))); + connect(toolbaritem,SIGNAL(activated()), action, SLOT(toggle())); + updateIcon(action->isChecked()); + } +} + +void MacToolBarItemWrapper::actionToggled(bool toogled) +{ + updateIcon(toogled); +} + +void MacToolBarItemWrapper::updateIcon(bool enabled) +{ + if(enabled) + { + QIcon icon = action->icon(); + QPixmap tempPixmap = icon.pixmap(QSize(24,24)); + QPainter painter; + painter.begin(&tempPixmap); + painter.fillRect(QRect(3,21,18,1),QColor("#3F3F3F")); + painter.fillRect(QRect(3,22,18,1),QColor("#6E6E6E")); + painter.fillRect(QRect(3,23,18,1),QColor("#EEEEEE")); + painter.end(); + + toolbaritem->setIcon(QIcon(tempPixmap)); + } + else + toolbaritem->setIcon(action->icon()); +} 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..f81d276c --- /dev/null +++ b/custom_widgets/yacreader_search_line_edit.cpp @@ -0,0 +1,146 @@ +#include "yacreader_search_line_edit.h" + +#include +#include +#include + +#include + +#include "QsLog.h" + +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 {color: #ABABAB; border:none; border-radius: 4px; background-color:#404040; padding-left: %1px; padding-right: %2px; padding-bottom: 1px; margin-right: 9px;} ").arg(searchLabel->sizeHint().width() + frameWidth + 6 + 5).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); +#else + setMaximumWidth(173); + setFixedHeight(26); +#endif + + setAttribute(Qt::WA_MacShowFocusRect,false); + setPlaceholderText(tr("type to search")); + + //search modifiers + modifiers << "[read]" << "[unread]";//<< "[author]"; + modifiersCompleter = new QCompleter(modifiers); + + QString regExpString; + foreach(QString modifier, modifiers) + { + regExpString = regExpString + modifier.replace("[","\\[").replace("]","\\]") + ".*|"; + } + + regExpString = regExpString + "[^\\[].*"; + + QLOG_INFO () << regExpString; + + QRegExp regExp(regExpString); + QValidator *validator = new QRegExpValidator(regExp, this); + + setValidator(validator); + setCompleter(modifiersCompleter); + + connect(this,SIGNAL(textChanged(QString)),this,SLOT(processText(QString))); +} + +void YACReaderSearchLineEdit::clearText() +{ + disconnect(this,SIGNAL(textChanged(QString)),this,SLOT(processText(QString))); + clear(); + connect(this,SIGNAL(textChanged(QString)),this,SLOT(processText(QString))); +} + +//modifiers are not returned +const QString YACReaderSearchLineEdit::text() +{ + QString text = QLineEdit::text(); + + QRegExp regExp("\\[.*\\]"); + return text.remove(regExp).trimmed(); +} + +void YACReaderSearchLineEdit::resizeEvent(QResizeEvent *) +{ + #ifdef Q_OS_MAC + 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); + #else + QSize sz = clearButton->sizeHint(); + int frameWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth); + int marginRight = style()->pixelMetric(QStyle::PM_LayoutRightMargin); + clearButton->move(rect().right() - frameWidth - sz.width() - marginRight - 6, + (rect().bottom() + 2 - sz.height())/2); + + QSize szl = searchLabel->sizeHint(); + searchLabel->move(8,(rect().bottom() + 2 - szl.height())/2); + #endif +} + +void YACReaderSearchLineEdit::updateCloseButton(const QString& text) +{ + clearButton->setVisible(!text.isEmpty()); +} + +void YACReaderSearchLineEdit::processText(const QString &text) +{ + + QRegExp regExp("(\\[.*\\])(.*)"); + if(text.startsWith("[")) + { + if(regExp.exactMatch(text)) //avoid search while the modifiers are being written + { + QString modifier = regExp.cap(1); + QString searchText = regExp.cap(2).trimmed(); + + int indexOfModifier = modifiers.indexOf(modifier); + if(indexOfModifier != -1) + { + QLOG_INFO() << "modifier : " << modifier << "text : " << searchText; + emit filterChanged(static_cast(indexOfModifier+1), searchText); //TODO, do not use on indexOF + } + else + { + QLOG_ERROR() << "invalid modifier : " << modifier; + } + } + + QLOG_INFO() << "full text :" << text << " : " << regExp.indexIn(text); + } + else + { + QLOG_INFO() << "NoModifiers : " << text; + emit filterChanged(YACReader::NoModifiers,text); + } +} diff --git a/custom_widgets/yacreader_search_line_edit.h b/custom_widgets/yacreader_search_line_edit.h new file mode 100644 index 00000000..50f6fdbe --- /dev/null +++ b/custom_widgets/yacreader_search_line_edit.h @@ -0,0 +1,40 @@ +#ifndef YACREADER_SEARCH_LINE_EDIT_H +#define YACREADER_SEARCH_LINE_EDIT_H + +#include +#include + +#include "yacreader_global.h" + +class QToolButton; +class QLabel; + +class YACReaderSearchLineEdit : public QLineEdit +{ + Q_OBJECT + +public: + YACReaderSearchLineEdit(QWidget *parent = 0); + void clearText(); //no signal emited; + const QString text(); + +protected: + void resizeEvent(QResizeEvent *); + +signals: + void filterChanged(const YACReader::SearchModifiers, QString); + +private slots: + void updateCloseButton(const QString &text); + void processText(const QString & text); + +private: + QToolButton *clearButton; + QLabel * searchLabel; + QCompleter * modifiersCompleter; + QStringList modifiers; +}; + + + +#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..fa9dd5d8 --- /dev/null +++ b/custom_widgets/yacreader_sidebar.cpp @@ -0,0 +1,195 @@ +#include "yacreader_sidebar.h" + +#include +#include + +#include "yacreader_folders_view.h" +#include "yacreader_reading_lists_view.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); + + settings = new QSettings(YACReader::getSettingsPath()+"/YACReaderLibrary.ini",QSettings::IniFormat); //TODO unificar la creación del fichero de config con el servidor + settings->beginGroup("libraryConfig"); + + //widgets + foldersView = new YACReaderFoldersView; + readingListsView = new YACReaderReadingListsView; + selectedLibrary = new YACReaderLibraryListWidget; + + librariesTitle = new YACReaderTitledToolBar(tr("LIBRARIES")); + foldersTitle = new YACReaderTitledToolBar(tr("FOLDERS")); + readingListsTitle = new YACReaderTitledToolBar(tr("READING LISTS")); + + splitter = new QSplitter(this); + splitter->setOrientation(Qt::Vertical); + +#ifndef Q_OS_MAC + splitter->setStyleSheet("QSplitter::handle { " + " image: none; background-color = black; " + " }" + "QSplitter::handle:vertical { height: 39px;}"); +#else + splitter->setStyleSheet("QSplitter::handle:vertical { height: 26px; background-color: transparent;}"); +#endif + + selectedLibrary->setContextMenuPolicy(Qt::ActionsContextMenu); + selectedLibrary->setAttribute(Qt::WA_MacShowFocusRect,false); + selectedLibrary->setFocusPolicy(Qt::NoFocus); + + //layout + QVBoxLayout * l = new QVBoxLayout; + + l->setContentsMargins(0,0,0,0); + + //LIBRARIES------------------------------------------------------- +#ifndef Q_OS_MAC + l->addSpacing(5); +#endif + + l->addWidget(librariesTitle); + +#ifndef Q_OS_MAC + l->addSpacing(4); + l->addWidget(new YACReaderSideBarSeparator(this)); + l->addSpacing(3); +#endif + + l->addWidget(selectedLibrary); +#ifndef Q_OS_MAC + l->addSpacing(11); +#else + l->addSpacing(6); +#endif + + //END LIBRARIES--------------------------------------------------- + + //FOLDERS--------------------------------------------------------- + QWidget * foldersContainer = new QWidget(this); + QVBoxLayout * foldersLayout = new QVBoxLayout; + foldersLayout->setContentsMargins(0,0,0,0); + foldersLayout->setSpacing(0); + +#ifndef Q_OS_MAC + //foldersLayout->addSpacing(6); + + //foldersLayout->addSpacing(5); + foldersLayout->addWidget(new YACReaderSideBarSeparator(this)); + foldersLayout->addSpacing(4); +#else + //foldersLayout->addSpacing(6); +#endif + + foldersLayout->addWidget(foldersTitle); + +#ifndef Q_OS_MAC + foldersLayout->addSpacing(4); + foldersLayout->addWidget(new YACReaderSideBarSeparator(this)); + foldersLayout->addSpacing(4); +#endif + + foldersLayout->addWidget(foldersView); + foldersLayout->addSpacing(6); + + foldersContainer->setLayout(foldersLayout); + splitter->addWidget(foldersContainer); + //END FOLDERS------------------------------------------------------ + + //READING LISTS---------------------------------------------------- + splitter->addWidget(readingListsView); + + QVBoxLayout * readingListsHeaderLayout = new QVBoxLayout; + readingListsHeaderLayout->setContentsMargins(0,0,0,0); + readingListsHeaderLayout->setSpacing(0); + +#ifndef Q_OS_MAC + //readingListsHeaderLayout->addSpacing(6); + + //readingListsHeaderLayout->addSpacing(5); + readingListsHeaderLayout->addWidget(new YACReaderSideBarSeparator(this)); + readingListsHeaderLayout->addSpacing(4); +#else + //readingListsHeaderLayout->addSpacing(6); +#endif + + readingListsHeaderLayout->addWidget(readingListsTitle); + +#ifndef Q_OS_MAC + readingListsHeaderLayout->addSpacing(4); + readingListsHeaderLayout->addWidget(new YACReaderSideBarSeparator(this)); + readingListsHeaderLayout->addSpacing(4); +#endif + + //readingListsLayout->addWidget(readingListsView); + readingListsHeaderLayout->addStretch(); + QSplitterHandle * handle = splitter->handle(1); + //handle->setCursor(QCursor(Qt::ArrowCursor)); + handle->setLayout(readingListsHeaderLayout); + //END READING LISTS------------------------------------------------ + + l->addWidget(splitter); + l->setSpacing(0); + + setLayout(l); + + if(settings->contains(SIDEBAR_SPLITTER_STATUS)) + splitter->restoreState(settings->value(SIDEBAR_SPLITTER_STATUS).toByteArray()); +} + + +void YACReaderSideBar::paintEvent(QPaintEvent * event) +{ + Q_UNUSED(event) + +#ifdef Q_OS_MAC + QPainter painter(this); + + painter.fillRect(0,0,width(),height(),QColor("#FFFFFF")); +#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); +} + +void YACReaderSideBar::closeEvent(QCloseEvent *event) +{ + settings->setValue(SIDEBAR_SPLITTER_STATUS, splitter->saveState()); +} + +QSize YACReaderSideBar::sizeHint() const +{ + return QSize(275,200); +} + +YACReaderSideBarSeparator::YACReaderSideBarSeparator(QWidget *parent) + :QWidget(parent) +{ + setFixedHeight(1); +} + +void YACReaderSideBarSeparator::paintEvent(QPaintEvent * event) +{ + Q_UNUSED(event) + + QPainter painter(this); + + painter.fillRect(5,0,width()-10,height(),QColor("#575757")); +} diff --git a/custom_widgets/yacreader_sidebar.h b/custom_widgets/yacreader_sidebar.h new file mode 100644 index 00000000..9cd6d377 --- /dev/null +++ b/custom_widgets/yacreader_sidebar.h @@ -0,0 +1,47 @@ +#ifndef YACREADER_SIDEBAR_H +#define YACREADER_SIDEBAR_H + +#include + +class YACReaderFoldersView; +class YACReaderLibraryListWidget; +class YACReaderSearchLineEdit; +class YACReaderTitledToolBar; +class YACReaderTitledToolBar; +class YACReaderReadingListsView; + +class YACReaderSideBarSeparator : public QWidget +{ +public: + explicit YACReaderSideBarSeparator(QWidget * parent = 0); +protected: + void paintEvent(QPaintEvent *event); +}; + +class YACReaderSideBar : public QWidget +{ + Q_OBJECT +public: + explicit YACReaderSideBar(QWidget *parent = 0); + QSize sizeHint() const; + + YACReaderFoldersView * foldersView; + YACReaderReadingListsView * readingListsView; + YACReaderLibraryListWidget * selectedLibrary; + YACReaderTitledToolBar * librariesTitle; + YACReaderTitledToolBar * foldersTitle; + YACReaderTitledToolBar * readingListsTitle; + +signals: + +public slots: + +protected: + void paintEvent(QPaintEvent *); + void closeEvent ( QCloseEvent * event ); + QSettings * settings; + QSplitter * splitter; + +}; + +#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..9ace5aef --- /dev/null +++ b/custom_widgets/yacreader_table_view.cpp @@ -0,0 +1,487 @@ +#include "yacreader_table_view.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "QsLog.h" + +#include "comic_item.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);*/ + + //drag: if the default drag is enabled there is no way for setting a custom image + //TODO report bug/suggestion + //setDragEnabled(true); + //setDragDropMode(QAbstractItemView::DragDrop); + setAcceptDrops(true); +} + +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(); + + //are we in a drag action?? + if(event->buttons() & Qt::LeftButton) { + int distance = (event->pos() - startDragPos).manhattanLength(); + if (distance >= QApplication::startDragDistance()) + performDrag(); + } + + //disabled mouseMoveEvent in the parent class +} +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); + } + return; + } + } + } + + //this could be the origin of a new drag acction + if(event->button() == Qt::LeftButton) + { + startDragPos = event->pos(); + } +} +void YACReaderTableView::leaveEvent(QEvent * event) +{ + closeRatingEditor(); + event->accept(); +} + +void YACReaderTableView::performDrag() +{ + QLOG_DEBUG() << "performDrag"; + QDrag *drag = new QDrag(this); + drag->setMimeData(model()->mimeData(selectionModel()->selectedRows())); + drag->setPixmap(QPixmap(":/images/openInYACReader.png")); //TODO add better image + + /*Qt::DropAction dropAction =*/ drag->exec(Qt::CopyAction | Qt::MoveAction, Qt::CopyAction); +} + +void YACReaderTableView::dragEnterEvent(QDragEnterEvent *event) +{ + QTableView::dragEnterEvent(event); + + if(model()->canDropMimeData(event->mimeData(),event->proposedAction(),0,0,QModelIndex())) + event->acceptProposedAction(); + QLOG_DEBUG() << "drag enter table"; +} + +void YACReaderTableView::dragMoveEvent(QDragMoveEvent *event) +{ + QTableView::dragMoveEvent(event); + + if(model()->canDropMimeData(event->mimeData(),event->proposedAction(),0,0,QModelIndex())) + event->acceptProposedAction(); + QLOG_DEBUG() << "dragMoveEvent table"; +} + +void YACReaderTableView::dropEvent(QDropEvent *event) +{ + QTableView::dropEvent(event); + + if(model()->canDropMimeData(event->mimeData(),event->proposedAction(),0,0,QModelIndex())) + event->acceptProposedAction(); + QLOG_DEBUG() << "drop on table"; + +} + +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 = ((ComicItem *)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 = ((ComicItem *)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 = ((ComicItem *)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..0c7bb607 --- /dev/null +++ b/custom_widgets/yacreader_table_view.h @@ -0,0 +1,132 @@ +#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); + void performDrag(); + void dragEnterEvent(QDragEnterEvent * event); + void dragMoveEvent(QDragMoveEvent * event); + void dropEvent(QDropEvent * event); + + + bool editing; + QModelIndex currentIndexEditing; + QWidget * myeditor; + + //drag from here + QPoint startDragPos; +}; + +//--- + +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..e47139f7 --- /dev/null +++ b/custom_widgets/yacreader_titled_toolbar.cpp @@ -0,0 +1,130 @@ +#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->setCursor(QCursor(Qt::ArrowCursor)); + 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); +} + +void YACReaderTitledToolBar::addSepartor() +{ + QHBoxLayout * mainLayout = dynamic_cast(layout()); + + QWidget * w = new QWidget(this); + w->setFixedSize(1,14); +#ifdef Q_OS_MAC + w->setStyleSheet("QWidget {background-color:#AFAFAF;}"); +#else + w->setStyleSheet("QWidget {background-color:#6F6F6F;}"); +#endif + + mainLayout->addSpacing(10); + mainLayout->addWidget(w); + mainLayout->addSpacing(10); +} diff --git a/custom_widgets/yacreader_titled_toolbar.h b/custom_widgets/yacreader_titled_toolbar.h new file mode 100644 index 00000000..21b9b75c --- /dev/null +++ b/custom_widgets/yacreader_titled_toolbar.h @@ -0,0 +1,46 @@ +#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); + void addSepartor(); +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..b180b64b --- /dev/null +++ b/custom_widgets/yacreader_treeview.cpp @@ -0,0 +1,154 @@ +#include "yacreader_treeview.h" + +YACReaderTreeView::YACReaderTreeView(QWidget *parent) : + QTreeView(parent) +{ + setAcceptDrops(true); + setDragDropMode(QAbstractItemView::DropOnly); + setItemsExpandable(true); + + //setDragEnabled(true); + /*viewport()->setAcceptDrops(true); + setDropIndicatorShown(true);*/ + + setContextMenuPolicy(Qt::CustomContextMenu); + + header()->hide(); + setUniformRowHeights(true); + setSelectionBehavior(QAbstractItemView::SelectRows); + setAttribute(Qt::WA_MacShowFocusRect,false); + +#ifdef Q_OS_MAC + + bool oldStyle = true; + switch (QSysInfo::MacVersion()) + { + case QSysInfo::MV_SNOWLEOPARD: + case QSysInfo::MV_LION: + case QSysInfo::MV_MOUNTAINLION: + case QSysInfo::MV_MAVERICKS: + oldStyle = true; //TODO fix this + break; + default: + oldStyle = false; + break; + } + + if(oldStyle) + { + 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/sidebar/expanded_branch_osx.png');}" + "QTreeView::branch:closed:selected:has-children {image: url(':/images/sidebar/collapsed_branch_osx.png');}" + + "QScrollBar:vertical { border: none; background: #EFEFEF; width: 9px; margin: 0 3px 0 0; }" + "QScrollBar::handle:vertical { background: #DDDDDD; width: 7px; min-height: 20px; margin: 1px; border: 1px solid #D0D0D0; }" + "QScrollBar::add-line:vertical { border: none; background: #EFEFEF; height: 10px; subcontrol-position: bottom; subcontrol-origin: margin; margin: 0 3px 0 0;}" + + "QScrollBar::sub-line:vertical { border: none; background: #EFEFEF; 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; }" + ); + } + else + { + setStyleSheet("QTreeView {background-color:transparent; border: none;}" + "QTreeView::item:selected {background-color:#91c4f4; border-top: 1px solid #91c4f4; border-left:none;border-right:none;border-bottom:1px solid #91c4f4;}" + "QTreeView::branch:selected {background-color:#91c4f4; border-top: 1px solid #91c4f4; border-left:none;border-right:none;border-bottom:1px solid #91c4f4;}" + "QTreeView::branch:open:selected:has-children {image: url(':/images/sidebar/expanded_branch_osx.png');}" + "QTreeView::branch:closed:selected:has-children {image: url(':/images/sidebar/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/sidebar/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/sidebar/collapsed_branch_selected.png');}" + + "QTreeView::branch:open:has-children:!has-siblings,QTreeView::branch:open:has-children:has-siblings {border-image: none;image: url(':/images/sidebar/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/sidebar/expanded_branch_selected.png');}" + ); +#endif + +} + +void YACReaderTreeView::mousePressEvent(QMouseEvent *event) +{ + QTreeView::mousePressEvent(event); + + QModelIndex destinationIndex = indexAt(event->pos()); + + if(!destinationIndex.isValid() && event->button() == Qt::LeftButton) + { + clearSelection(); + } +} + +void YACReaderTreeView::expandCurrent() +{ + QModelIndex index = indexAt(expandPos); + if(index.isValid()) + expand(index); +} + +void YACReaderTreeView::dragEnterEvent(QDragEnterEvent *event) +{ + QTreeView::dragEnterEvent(event); +} + +void YACReaderTreeView::dragLeaveEvent(QDragLeaveEvent *event) +{ + Q_UNUSED(event) +} + +void YACReaderTreeView::dragMoveEvent(QDragMoveEvent *event) +{ + QTreeView::dragMoveEvent(event); + + //fix for drop auto expand + QModelIndex underMouse = indexAt(event->pos()); + if( underMouse.isValid()) { + expandPos = event->pos(); + connect(&expandTimer,SIGNAL(timeout()),this,SLOT(expandCurrent())); + expandTimer.setSingleShot(true); + expandTimer.start(500); + } + //force mouse hover decoration, TODO why the event loop is not working here? + if(!t.isActive()) + { + t.setSingleShot(true); + t.setInterval(50); + t.start(); + repaint(); + } + +} + +void YACReaderTreeView::dropEvent(QDropEvent *event) +{ + t.stop(); + + QTreeView::dropEvent(event); +} + + + diff --git a/custom_widgets/yacreader_treeview.h b/custom_widgets/yacreader_treeview.h new file mode 100644 index 00000000..d4c719b3 --- /dev/null +++ b/custom_widgets/yacreader_treeview.h @@ -0,0 +1,29 @@ +#ifndef YACREADER_TREEVIEW_H +#define YACREADER_TREEVIEW_H + +#include + +class YACReaderTreeView : public QTreeView +{ + Q_OBJECT +public: + explicit YACReaderTreeView(QWidget *parent = 0); + void mousePressEvent(QMouseEvent *event); +protected slots: + //fix for drop auto expand + void expandCurrent(); + +protected: + //Drop to import + void dragEnterEvent(QDragEnterEvent *event); + void dragLeaveEvent(QDragLeaveEvent *event); + void dragMoveEvent(QDragMoveEvent *event); + void dropEvent(QDropEvent *event); + + //fix for drop auto expand + QTimer expandTimer; + QTimer t; + QPoint expandPos; +}; + +#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~ 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 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 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%0Ag%IU5{l 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 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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